Files
mars-flaim/flaim/util/viewmenu.cpp
ahodgkinson 73980c2f6d Updates for Ubuntu Hardy.
git-svn-id: https://svn.code.sf.net/p/flaim/code/trunk@1054 0109f412-320b-0410-ab79-c3e0c5ffbbe6
2008-05-01 18:54:50 +00:00

1231 lines
32 KiB
C++

//-------------------------------------------------------------------------
// Desc: Menuing system for database viewer utility.
// Tabs: 3
//
// Copyright (c) 1992-1995, 1998-2000, 2003-2007 Novell, Inc.
// All Rights Reserved.
//
// This library is free software; you can redistribute it and/or
// modify it under the terms of the GNU Lesser General Public
// License as published by the Free Software Foundation; version 2.1
// of the License.
//
// This library is distributed in the hope that it will be useful,
// but WITHOUT ANY WARRANTY; without even the implied warranty of
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
// Library Lesser General Public License for more details.
//
// You should have received a copy of the GNU Lesser General Public
// License along with this library; if not, contact Novell, Inc.
//
// To contact Novell about this file by physical or electronic mail,
// you may find current contact information at www.novell.com.
//
// $Id$
//------------------------------------------------------------------------------
#include "view.h"
FSTATIC const char * Months[] = {
"Jan",
"Feb",
"Mar",
"Apr",
"May",
"Jun",
"Jul",
"Aug",
"Sep",
"Oct",
"Nov",
"Dec"
};
FSTATIC void ViewDispMenuItem(
VIEW_MENU_ITEM_p ViewMenuPtr
);
FSTATIC void ViewRefreshMenu(
VIEW_MENU_ITEM_p PrevItem
);
FSTATIC void UpdateHorizCursor(
FLMUINT OnFlag
);
FSTATIC void DoUpArrow(
void
);
FSTATIC void DoDownArrow(
void
);
FSTATIC void DoPageDown(
void
);
FSTATIC void DoPageUp(
void
);
FSTATIC void DoHome(
void
);
FSTATIC void DoEnd(
void
);
FSTATIC void DoRightArrow(
void
);
FSTATIC void DoLeftArrow(
void
);
FSTATIC void ByteToHex(
FLMBYTE c,
FLMBYTE * DestBuf,
FLMUINT UpperCaseFlag
);
FSTATIC void ViewHelpScreen( void);
extern FLMUINT gv_uiTopLine;
extern FLMUINT gv_uiBottomLine;
/***************************************************************************
Name: ViewFreeMenuMemory
Desc: This routine frees the memory used to hold menu items.
*****************************************************************************/
void ViewFreeMenuMemory(
void
)
{
if (gv_pViewMenuFirstItem != NULL)
{
gv_ViewPool.poolFree();
gv_pViewMenuFirstItem = NULL;
gv_pViewMenuLastItem = NULL;
gv_pViewMenuCurrItem = NULL;
}
}
/***************************************************************************
Name: ViewMenuInit
Desc: This routine initializes variables to start a new menu.
*****************************************************************************/
FLMINT ViewMenuInit(
const char * Title)
{
FLMUINT Col;
/* Clear the screen and display the menu title */
f_conSetBackFore( FLM_BLACK, FLM_WHITE);
f_conClearScreen( 0, 0);
/* Display the title in the middle of the top line of the screen */
Col = (80 - f_strlen( Title)) / 2;
f_conStrOutXY( Title, Col, 0);
ViewUpdateDate( TRUE, &gv_ViewLastTime);
/* Deallocate any old memory, if any */
ViewFreeMenuMemory();
return( 1);
}
/***************************************************************************
Name: ByteToHex
Desc: This routine converts a FLMBYTE value to two ASCII Hex characters.
*****************************************************************************/
FSTATIC void ByteToHex(
FLMBYTE c,
FLMBYTE * DestBuf,
FLMUINT UpperCaseFlag
)
{
FLMBYTE TempC;
/* Convert the upper four bits */
if ((TempC = ((c & 0xF0) >> 4)) <= 9)
*DestBuf = '0' + TempC;
else if (UpperCaseFlag)
*DestBuf = 'A' + TempC - 10;
else
*DestBuf = 'a' + TempC - 10;
DestBuf++;
/* Convert the lower four bits */
if ((TempC = (c & 0x0F)) <= 9)
*DestBuf = '0' + TempC;
else if (UpperCaseFlag)
*DestBuf = 'A' + TempC - 10;
else
*DestBuf = 'a' + TempC - 10;
DestBuf++;
/* Terminate with a NULL character */
*DestBuf = 0;
}
/***************************************************************************
Name: ViewDispMenuItem
Desc: This routine displays a menu item on the screen.
*****************************************************************************/
FSTATIC void ViewDispMenuItem(
VIEW_MENU_ITEM_p ViewMenuPtr
)
{
FLMUINT Row;
FLMUINT Col = ViewMenuPtr->Col;
FLMUINT uiLoop;
char TempBuf[ 80];
if (!gv_bViewEnabled)
return;
/* Calculate row and column where the item is to be displayed */
Row = gv_uiTopLine + (ViewMenuPtr->Row - gv_uiViewTopRow);
/* If it is a HEX display, output the address first */
if ((ViewMenuPtr->ValueType & 0x0F) == VAL_IS_BINARY_HEX)
{
f_sprintf( (char *)TempBuf, "%03u:%08X",
(unsigned)ViewMenuPtr->ModFileNumber, (unsigned)ViewMenuPtr->ModFileOffset);
f_conSetBackFore( FLM_WHITE, FLM_GREEN);
f_conStrOutXY( TempBuf, 0, Row);
}
/* If the item is the current item, display it using the */
/* select colors. Otherwise, display using the unselect colors */
if (ViewMenuPtr->ItemNum == gv_uiViewMenuCurrItemNum)
{
if (gv_pViewMenuCurrItem == NULL)
gv_pViewMenuCurrItem = ViewMenuPtr;
if (ViewMenuPtr->Option)
f_conSetBackFore( ViewMenuPtr->SelectBackColor,
ViewMenuPtr->SelectForeColor);
else
f_conSetBackFore( ViewMenuPtr->UnselectForeColor,
ViewMenuPtr->UnselectBackColor);
}
else
f_conSetBackFore( ViewMenuPtr->UnselectBackColor,
ViewMenuPtr->UnselectForeColor);
if (ViewMenuPtr->iLabelIndex < 0)
{
Col += (ViewMenuPtr->LabelWidth + 1);
TempBuf[ 0] = 0;
}
else if (!ViewMenuPtr->LabelWidth)
{
f_strcpy( TempBuf, Labels[ ViewMenuPtr->iLabelIndex]);
f_strcpy( &TempBuf[ f_strlen( TempBuf)], " ");
}
else
{
for( uiLoop = 0; uiLoop < ViewMenuPtr->LabelWidth; uiLoop++)
{
TempBuf[ uiLoop] = '.';
}
TempBuf[ ViewMenuPtr->LabelWidth] = ' ';
TempBuf[ ViewMenuPtr->LabelWidth + 1] = 0;
f_memcpy( TempBuf, Labels[ ViewMenuPtr->iLabelIndex],
f_strlen( Labels[ ViewMenuPtr->iLabelIndex]));
}
if (ViewMenuPtr->Option)
{
if (ViewMenuPtr->ItemNum == gv_uiViewMenuCurrItemNum)
f_conStrOutXY( "*>", (Col - 2), Row);
else
f_conStrOutXY( "* ", (Col - 2), Row);
}
else
{
if (ViewMenuPtr->ItemNum == gv_uiViewMenuCurrItemNum)
f_conStrOutXY( " >", (Col - 2), Row);
else
f_conStrOutXY( " ", (Col - 2), Row);
}
if (TempBuf[ 0])
f_conStrOutXY( TempBuf, Col, Row);
/* Now output the value */
Col += f_strlen( TempBuf);
switch( ViewMenuPtr->ValueType & 0x0F)
{
case VAL_IS_LABEL_INDEX:
f_conStrOutXY( (Labels[ ViewMenuPtr->Value]), Col, Row);
break;
case VAL_IS_ERR_INDEX:
{
eCorruptionType eCorruption = (eCorruptionType)ViewMenuPtr->Value;
f_conStrOutXY( FlmVerifyErrToStr( eCorruption), Col, Row);
break;
}
case VAL_IS_TEXT_PTR:
f_conStrOutXY( (const char *)ViewMenuPtr->Value, Col, Row);
break;
case VAL_IS_BINARY_HEX:
case VAL_IS_BINARY_PTR:
{
FLMUINT BytesPerLine = MAX_HORIZ_SIZE( Col);
FLMUINT BytesProcessed = 0;
FLMUINT i;
FLMUINT j;
FLMUINT k;
FLMUINT NumBytes;
FLMBYTE * ValPtr = (FLMBYTE *)ViewMenuPtr->Value;
FLMUINT ValLen = ViewMenuPtr->ValueLen;
/* Process each character in the value */
i = 0;
j = 0;
k = BytesPerLine * 3 + 5;
NumBytes = 0;
/* Fill up a single line with whatever will fit on the line in */
/* hex format. */
f_memset( TempBuf, ' ', 80);
TempBuf[ k - 3] = '|';
while( (BytesProcessed < ValLen) && (i < BytesPerLine))
{
ByteToHex( ValPtr[ BytesProcessed], (FLMBYTE *)&TempBuf[ j], TRUE);
if ((ValPtr[ BytesProcessed] > ' ') &&
(ValPtr[ BytesProcessed] <= 127))
TempBuf[ k] = ValPtr[ BytesProcessed];
k++;
NumBytes++;
BytesProcessed++;
i++;
j += 2;
TempBuf[ j] = ' ';
j++;
}
TempBuf[ k] = 0;
/* Output the line */
f_conStrOutXY( TempBuf, Col, Row);
if (ViewMenuPtr->ItemNum == gv_uiViewMenuCurrItemNum)
UpdateHorizCursor( TRUE);
break;
}
case VAL_IS_NUMBER:
switch( ViewMenuPtr->ValueType & 0xF0)
{
case DISP_DECIMAL:
f_sprintf( (char *)TempBuf, "%u", (unsigned)ViewMenuPtr->Value);
break;
case DISP_HEX:
if (ViewMenuPtr->Value == 0xFFFFFFFF)
f_strcpy( TempBuf, "None");
else if (ViewMenuPtr->Value == 0)
f_strcpy( TempBuf, "0");
else
f_sprintf( (char *)TempBuf, "0x%X", (unsigned)ViewMenuPtr->Value);
break;
case DISP_DECIMAL_HEX:
f_sprintf( (char *)TempBuf, "%u (0x%X)",
(unsigned)ViewMenuPtr->Value, (unsigned)ViewMenuPtr->Value);
break;
case DISP_HEX_DECIMAL:
default:
if (ViewMenuPtr->Value == 0xFFFFFFFF)
f_strcpy( TempBuf, "None");
else if (ViewMenuPtr->Value == 0)
f_strcpy( TempBuf, "0");
else
f_sprintf( (char *)TempBuf, "0x%X (%u)",
(unsigned)ViewMenuPtr->Value, (unsigned)ViewMenuPtr->Value);
break;
}
f_conStrOutXY( TempBuf, Col, Row);
break;
case VAL_IS_EMPTY:
default:
break;
}
}
/***************************************************************************
Name: ViewAddMenuItem
Desc: This routine adds a menu item to the item list.
*****************************************************************************/
FLMINT ViewAddMenuItem(
FLMINT LabelIndex,
FLMUINT LabelWidth,
FLMUINT ValueType,
FLMUINT Value,
FLMUINT ValueLen,
FLMUINT ModFileNumber,
FLMUINT ModFileOffset,
FLMUINT ModBufLen,
FLMUINT ModType,
FLMUINT Col,
FLMUINT Row,
FLMUINT Option,
eColorType UnselectBackColor,
eColorType UnselectForeColor,
eColorType SelectBackColor,
eColorType SelectForeColor)
{
VIEW_MENU_ITEM_p ViewMenuPtr;
FLMUINT Size = sizeof( VIEW_MENU_ITEM);
/* Allocate the memory for the item */
if ((ValueType & 0x0F) == VAL_IS_TEXT_PTR)
{
Size += (f_strlen( (const char *)Value) + 1);
}
else if ((ValueType & 0x0F) == VAL_IS_BINARY)
{
Size += ValueLen;
}
if( RC_BAD( gv_ViewPool.poolAlloc( Size, (void **)&ViewMenuPtr)))
{
ViewShowError( "Could not allocate memory for menu value");
return( 0);
}
/* Link the item into the array and set some values in the item */
ViewMenuPtr->NextItem = NULL;
ViewMenuPtr->PrevItem = gv_pViewMenuLastItem;
if (gv_pViewMenuLastItem != NULL)
{
gv_pViewMenuLastItem->NextItem = ViewMenuPtr;
ViewMenuPtr->ItemNum = gv_pViewMenuLastItem->ItemNum + 1;
}
else
{
gv_pViewMenuFirstItem = ViewMenuPtr;
ViewMenuPtr->ItemNum = 0;
}
gv_pViewMenuLastItem = ViewMenuPtr;
ViewMenuPtr->iLabelIndex = LabelIndex;
ViewMenuPtr->LabelWidth = LabelWidth;
ViewMenuPtr->ValueType = ValueType;
if ((ValueType & 0x0F) == VAL_IS_TEXT_PTR)
{
ViewMenuPtr->Value = (FLMUINT)((FLMBYTE *)&ViewMenuPtr[ 1]);
f_strcpy( (char *)ViewMenuPtr->Value, (const char *)Value);
}
else if ((ValueType & 0x0F) == VAL_IS_BINARY)
{
ViewMenuPtr->ValueType = VAL_IS_BINARY_PTR;
ViewMenuPtr->Value = (FLMUINT)((FLMBYTE *)&ViewMenuPtr[ 1]);
f_memcpy( (void *)ViewMenuPtr->Value, (void *)Value, ValueLen);
}
else
ViewMenuPtr->Value = Value;
ViewMenuPtr->ValueLen = ValueLen;
ViewMenuPtr->ModFileOffset = ModFileOffset;
ViewMenuPtr->ModFileNumber = ModFileNumber;
ViewMenuPtr->ModBufLen = ModBufLen;
ViewMenuPtr->ModType = ModType;
ViewMenuPtr->Col = Col;
ViewMenuPtr->Row = Row;
ViewMenuPtr->Option = Option;
ViewMenuPtr->UnselectBackColor = UnselectBackColor;
ViewMenuPtr->UnselectForeColor = UnselectForeColor;
ViewMenuPtr->SelectBackColor = SelectBackColor;
ViewMenuPtr->SelectForeColor = SelectForeColor;
ViewMenuPtr->HorizCurPos = 0;
if ((ViewMenuPtr->Row >= gv_uiViewTopRow) &&
(ViewMenuPtr->Row <= gv_uiViewBottomRow))
ViewDispMenuItem( ViewMenuPtr);
return( 1);
}
/***************************************************************************
Name: ViewEscPrompt
Desc: This routine displays the prompt to press ESCAPE. This prompt
appears at the bottom of every screen.
*****************************************************************************/
void ViewEscPrompt(
void
)
{
FLMUINT uiNumCols;
FLMUINT uiNumRows;
f_conGetScreenSize( &uiNumCols, &uiNumRows);
f_conSetBackFore( FLM_BLACK, FLM_WHITE);
f_conClearScreen( 0, uiNumRows - 1);
f_conSetBackFore( FLM_RED, FLM_WHITE);
f_conStrOutXY( "ESC=Exit, ?=Help", 0, uiNumRows - 1);
f_conSetBackFore( FLM_BLACK, FLM_WHITE);
f_conStrOutXY( "File: ", 20, uiNumRows - 1);
f_conStrOutXY( gv_szViewFileName, 26, uiNumRows - 1);
gv_uiViewLastFileOffset = VIEW_INVALID_FILE_OFFSET;
}
/***************************************************************************
Name: ViewRefreshMenu
Desc: This routine refreshes the menu display. If NULL is passed in
the entire screen is refreshed. Otherwise, the item passed as
well as the current item are refreshed.
*****************************************************************************/
FSTATIC void ViewRefreshMenu(
VIEW_MENU_ITEM_p PrevItem
)
{
VIEW_MENU_ITEM_p ViewMenuPtr;
gv_uiViewMenuCurrItemNum = gv_pViewMenuCurrItem->ItemNum;
if (PrevItem != NULL)
{
ViewDispMenuItem( PrevItem);
ViewDispMenuItem( gv_pViewMenuCurrItem);
}
else
{
/* Refresh the entire screen */
f_conSetBackFore( FLM_BLACK, FLM_WHITE);
f_conClearScreen( 0, 1);
ViewMenuPtr = gv_pViewMenuFirstItem;
while( ViewMenuPtr)
{
if ((ViewMenuPtr->Row >= gv_uiViewTopRow) &&
(ViewMenuPtr->Row <= gv_uiViewBottomRow))
ViewDispMenuItem( ViewMenuPtr);
ViewMenuPtr = ViewMenuPtr->NextItem;
}
ViewEscPrompt();
}
}
/********************************************************************
Desc: ?
*********************************************************************/
FSTATIC void UpdateHorizCursor(
FLMUINT OnFlag
)
{
char TempBuf[ 4];
FLMUINT i;
FLMUINT Row;
FLMUINT Col = gv_pViewMenuCurrItem->Col + 1;
FLMBYTE * ValPtr = (FLMBYTE *)gv_pViewMenuCurrItem->Value;
if (gv_pViewMenuCurrItem->HorizCurPos > HORIZ_SIZE( gv_pViewMenuCurrItem) - 1)
gv_pViewMenuCurrItem->HorizCurPos = HORIZ_SIZE( gv_pViewMenuCurrItem) - 1;
i = gv_pViewMenuCurrItem->HorizCurPos;
ByteToHex( *(ValPtr + i), (FLMBYTE *)TempBuf, TRUE);
TempBuf[ 2] = 0;
if (OnFlag)
f_conSetBackFore( FLM_RED, FLM_WHITE);
else
f_conSetBackFore( gv_pViewMenuCurrItem->UnselectForeColor,
gv_pViewMenuCurrItem->UnselectBackColor);
/* Calculate row and column where the item is to be displayed */
Row = gv_uiTopLine + (gv_pViewMenuCurrItem->Row - gv_uiViewTopRow);
f_conStrOutXY( TempBuf, (Col + i * 3), Row);
if (((TempBuf[ 0] = ValPtr[ i]) < ' ') ||
((FLMBYTE)TempBuf[ 0] > 127))
TempBuf[ 0] = ' ';
#if defined( FLM_WIN)
if (OnFlag)
TempBuf [0] = 128;
#endif
TempBuf[ 1] = 0;
f_conStrOutXY( TempBuf,
(Col + MAX_HORIZ_SIZE( Col) * 3 + 5 + i), Row);
f_conSetBackFore( gv_pViewMenuCurrItem->UnselectForeColor,
gv_pViewMenuCurrItem->UnselectBackColor);
if (OnFlag)
f_conStrOutXY( ">", (Col + i * 3 - 1), Row);
else
f_conStrOutXY( " ", (Col + i * 3 - 1), Row);
}
/********************************************************************
Desc: ?
*********************************************************************/
FSTATIC void DoUpArrow(
void
)
{
VIEW_MENU_ITEM_p ViewMenuPrevItem;
if ((ViewMenuPrevItem = gv_pViewMenuCurrItem->PrevItem) != NULL)
{
if (ViewMenuPrevItem->HorizCurPos != gv_pViewMenuCurrItem->HorizCurPos)
ViewMenuPrevItem->HorizCurPos = gv_pViewMenuCurrItem->HorizCurPos;
gv_pViewMenuCurrItem = ViewMenuPrevItem;
if (gv_pViewMenuCurrItem->Row < gv_uiViewTopRow)
{
gv_uiViewTopRow--;
gv_uiViewBottomRow--;
if (gv_pViewMenuCurrItem->Row < gv_uiViewTopRow)
gv_pViewMenuCurrItem = gv_pViewMenuCurrItem->NextItem;
ViewRefreshMenu( NULL);
}
else
ViewRefreshMenu( gv_pViewMenuCurrItem->NextItem);
}
}
/********************************************************************
Desc: ?
*********************************************************************/
FSTATIC void DoDownArrow(
void
)
{
VIEW_MENU_ITEM_p ViewMenuNextItem;
if ((ViewMenuNextItem = gv_pViewMenuCurrItem->NextItem) != NULL)
{
if (ViewMenuNextItem->HorizCurPos != gv_pViewMenuCurrItem->HorizCurPos)
ViewMenuNextItem->HorizCurPos = gv_pViewMenuCurrItem->HorizCurPos;
gv_pViewMenuCurrItem = ViewMenuNextItem;
if (gv_pViewMenuCurrItem->Row > gv_uiViewBottomRow)
{
gv_uiViewTopRow++;
gv_uiViewBottomRow++;
if (gv_pViewMenuCurrItem->Row > gv_uiViewBottomRow)
gv_pViewMenuCurrItem = gv_pViewMenuCurrItem->PrevItem;
ViewRefreshMenu( NULL);
}
else
ViewRefreshMenu( gv_pViewMenuCurrItem->PrevItem);
}
}
/********************************************************************
Desc: ?
*********************************************************************/
FSTATIC void DoPageDown(
void
)
{
FLMUINT TargRow;
VIEW_MENU_ITEM_p ViewSaveItem;
if (gv_uiViewBottomRow < gv_pViewMenuLastItem->Row)
{
gv_uiViewBottomRow += LINES_PER_PAGE;
if (gv_uiViewBottomRow > gv_pViewMenuLastItem->Row)
gv_uiViewBottomRow = gv_pViewMenuLastItem->Row;
gv_uiViewTopRow = gv_uiViewBottomRow - LINES_PER_PAGE + 1;
TargRow = gv_pViewMenuCurrItem->Row + LINES_PER_PAGE;
if (TargRow > gv_uiViewBottomRow)
TargRow = gv_uiViewBottomRow;
while( (gv_pViewMenuCurrItem->NextItem != NULL) &&
(gv_pViewMenuCurrItem->Row < TargRow))
gv_pViewMenuCurrItem = gv_pViewMenuCurrItem->NextItem;
if (gv_pViewMenuCurrItem->Row > gv_uiViewBottomRow)
gv_pViewMenuCurrItem = gv_pViewMenuCurrItem->PrevItem;
ViewRefreshMenu( NULL);
}
else if (gv_pViewMenuCurrItem->NextItem != NULL)
{
ViewSaveItem = gv_pViewMenuCurrItem;
gv_pViewMenuCurrItem = gv_pViewMenuLastItem;
ViewRefreshMenu( ViewSaveItem);
}
}
/********************************************************************
Desc: ?
*********************************************************************/
FSTATIC void DoPageUp(
void
)
{
FLMUINT TargRow;
VIEW_MENU_ITEM_p ViewSaveItem;
if (gv_uiViewTopRow > 0)
{
if (gv_uiViewTopRow < LINES_PER_PAGE)
gv_uiViewTopRow = 0;
else
gv_uiViewTopRow -= LINES_PER_PAGE;
gv_uiViewBottomRow = gv_uiViewTopRow + LINES_PER_PAGE - 1;
TargRow = gv_pViewMenuCurrItem->Row - LINES_PER_PAGE;
if (TargRow < gv_uiViewTopRow)
TargRow = gv_uiViewTopRow;
while( (gv_pViewMenuCurrItem->PrevItem != NULL) &&
(gv_pViewMenuCurrItem->Row > TargRow))
gv_pViewMenuCurrItem = gv_pViewMenuCurrItem->PrevItem;
if (gv_pViewMenuCurrItem->Row < gv_uiViewTopRow)
gv_pViewMenuCurrItem = gv_pViewMenuCurrItem->NextItem;
ViewRefreshMenu( NULL);
}
else if (gv_pViewMenuCurrItem->PrevItem != NULL)
{
ViewSaveItem = gv_pViewMenuCurrItem;
gv_pViewMenuCurrItem = gv_pViewMenuFirstItem;
ViewRefreshMenu( ViewSaveItem);
}
}
/********************************************************************
Desc: ?
*********************************************************************/
FSTATIC void DoHome(
void
)
{
VIEW_MENU_ITEM_p ViewSaveItem;
if (gv_uiViewTopRow != 0)
{
gv_uiViewTopRow = 0;
gv_uiViewBottomRow = gv_uiViewTopRow + LINES_PER_PAGE - 1;
gv_pViewMenuCurrItem = gv_pViewMenuFirstItem;
ViewRefreshMenu( NULL);
}
else if (gv_pViewMenuCurrItem->PrevItem != NULL)
{
ViewSaveItem = gv_pViewMenuCurrItem;
gv_pViewMenuCurrItem = gv_pViewMenuFirstItem;
ViewRefreshMenu( ViewSaveItem);
}
}
/********************************************************************
Desc: ?
*********************************************************************/
FSTATIC void DoEnd(
void
)
{
VIEW_MENU_ITEM_p ViewSaveItem;
if (gv_uiViewBottomRow < gv_pViewMenuLastItem->Row)
{
gv_uiViewBottomRow = gv_pViewMenuLastItem->Row;
gv_uiViewTopRow = gv_uiViewBottomRow - LINES_PER_PAGE + 1;
gv_pViewMenuCurrItem = gv_pViewMenuLastItem;
ViewRefreshMenu( NULL);
}
else if (gv_pViewMenuCurrItem->NextItem != NULL)
{
ViewSaveItem = gv_pViewMenuCurrItem;
gv_pViewMenuCurrItem = gv_pViewMenuLastItem;
ViewRefreshMenu( ViewSaveItem);
}
}
/********************************************************************
Desc: ?
*********************************************************************/
FSTATIC void DoRightArrow(
void
)
{
if ((!HAVE_HORIZ_CUR( gv_pViewMenuCurrItem)) ||
(gv_pViewMenuCurrItem->HorizCurPos == HORIZ_SIZE( gv_pViewMenuCurrItem) - 1))
{
if (gv_pViewMenuCurrItem->ItemNum != gv_pViewMenuLastItem->ItemNum)
{
gv_pViewMenuCurrItem->HorizCurPos = 0;
DoDownArrow();
}
}
else if (gv_pViewMenuCurrItem->HorizCurPos <
HORIZ_SIZE( gv_pViewMenuCurrItem) - 1)
{
UpdateHorizCursor( FALSE);
gv_pViewMenuCurrItem->HorizCurPos++;
UpdateHorizCursor( TRUE);
}
}
/********************************************************************
Desc: ?
*********************************************************************/
FSTATIC void DoLeftArrow(
void
)
{
if ((!HAVE_HORIZ_CUR( gv_pViewMenuCurrItem)) ||
(gv_pViewMenuCurrItem->HorizCurPos == 0))
{
if (gv_pViewMenuCurrItem->PrevItem != NULL)
gv_pViewMenuCurrItem->HorizCurPos =
HORIZ_SIZE( gv_pViewMenuCurrItem->PrevItem) - 1;
DoUpArrow();
}
else if (gv_pViewMenuCurrItem->HorizCurPos > 0)
{
UpdateHorizCursor( FALSE);
gv_pViewMenuCurrItem->HorizCurPos--;
UpdateHorizCursor( TRUE);
}
}
/***************************************************************************
Name: ViewHelpScreen
Desc: This routine displays a help screen showing available commands.
*****************************************************************************/
FSTATIC void ViewHelpScreen(
void
)
{
FLMUINT wChar;
/* Clear the screen and display the menu title */
f_conSetBackFore( FLM_BLACK, FLM_WHITE);
f_conClearScreen( 0, 1);
f_conSetCursorPos( 0, 3);
f_conStrOut( " RECOGNIZED KEYBOARD CHARACTERS\n");
f_conStrOut( "\n");
f_conStrOut( " ESCAPE - Exit Screen\n");
f_conStrOut( " U,u,8 - Up Arrow\n");
f_conStrOut( " D,d,2 - Down Arrow\n");
f_conStrOut( " +,3 - Page Down\n");
f_conStrOut( " R,r,6 - Right Arrow\n");
f_conStrOut( " L,l,5 - Left Arrow\n");
f_conStrOut( " -,9 - Page Up\n");
f_conStrOut( " H,h,7 - Home\n");
f_conStrOut( " Z,z,1 - End\n");
f_conStrOut( " E,e - Edit Data\n");
f_conStrOut( " A,a - Edit Data in RAW Mode (no checksum)\n");
f_conStrOut( " G,g - Goto Block\n");
f_conStrOut( " X,x - Display Hex\n");
f_conStrOut( " Y,y - Display Decrypted\n");
f_conStrOut( " S,s - Search\n");
f_conStrOut( " ? - Show this help screen\n");
f_conStrOut( "\n");
f_conStrOut( " PRESS ANY CHARACTER TO EXIT HELP SCREEN\n");
for (;;)
{
/* Update date and time */
ViewUpdateDate( FALSE, &gv_ViewLastTime);
/* See what character was pressed */
wChar = (!f_conHaveKey()) ? 0 : (f_conGetKey());
if (gv_bShutdown)
return;
if (wChar)
break;
viewGiveUpCPU();
}
ViewRefreshMenu( NULL);
}
/***************************************************************************
Name: ViewGetMenuOption
Desc: This routine allows the user to press keys while in a menu.
Keys for navigating through the menu are handled inside this
routine. Other keys are passed to the calling routine or are
ignored altogether.
*****************************************************************************/
FLMINT ViewGetMenuOption(
void
)
{
FLMUINT c;
/* Make sure we have a pointer to the current item */
if (gv_pViewMenuCurrItem == NULL)
{
gv_pViewMenuCurrItem = gv_pViewMenuFirstItem;
while( (gv_pViewMenuCurrItem->NextItem != NULL) &&
(gv_pViewMenuCurrItem->ItemNum < gv_uiViewMenuCurrItemNum))
gv_pViewMenuCurrItem = gv_pViewMenuCurrItem->NextItem;
if (gv_pViewMenuCurrItem->ItemNum != gv_uiViewMenuCurrItemNum)
{
gv_uiViewMenuCurrItemNum = gv_pViewMenuCurrItem->ItemNum;
ViewDispMenuItem( gv_pViewMenuCurrItem);
}
}
/* Loop getting user input */
ViewEscPrompt();
for( ;;)
{
/* Set the file position of the current object */
if (gv_pViewMenuCurrItem->ModFileOffset != VIEW_INVALID_FILE_OFFSET)
{
gv_uiViewCurrFileOffset = gv_pViewMenuCurrItem->ModFileOffset;
gv_uiViewCurrFileNumber = gv_pViewMenuCurrItem->ModFileNumber;
if (HAVE_HORIZ_CUR( gv_pViewMenuCurrItem))
gv_uiViewCurrFileOffset += gv_pViewMenuCurrItem->HorizCurPos;
}
/* Update date and time */
ViewUpdateDate( FALSE, &gv_ViewLastTime);
/* See what character was pressed */
viewGiveUpCPU();
c = (!f_conHaveKey()) ? 0 : (f_conGetKey());
if (gv_bShutdown)
return( ESCAPE_OPTION);
switch( c)
{
case FKB_ESCAPE:
return( ESCAPE_OPTION);
case FKB_UP:
case 'U':
case 'u':
case '8':
DoUpArrow();
break;
case FKB_DOWN:
case 'D':
case 'd':
case '2':
DoDownArrow();
break;
case FKB_PGDN:
case '+':
case '3':
DoPageDown();
break;
case FKB_PGUP:
case '-':
case '9':
DoPageUp();
break;
case FKB_HOME:
case 'H':
case 'h':
case '7':
DoHome();
break;
case FKB_END:
case 'Z':
case 'z':
case '1':
DoEnd();
break;
case '\n':
case '\r':
case FKB_ENTER:
if (gv_pViewMenuCurrItem->Option)
return( gv_pViewMenuCurrItem->Option);
break;
case 'G':
case 'g':
case 7: /* Control-G */
return( GOTO_BLOCK_OPTION);
case FKB_RIGHT:
case 'R':
case 'r':
case '6':
DoRightArrow();
break;
case FKB_LEFT:
case 'L':
case 'l':
case '4':
DoLeftArrow();
break;
case 'E':
case 'e':
return( EDIT_OPTION);
case 'A':
case 'a':
return( EDIT_RAW_OPTION);
case 'x':
case 'X':
return( HEX_OPTION);
case 'Y':
case 'y':
return( DECRYPT_OPTION);
case 'S':
case 's':
return( SEARCH_OPTION);
case '?':
ViewHelpScreen();
break;
default:
break;
}
}
}
/***************************************************************************
Name: ViewUpdateDate
Desc: This routine updates the date and time on the screen.
*****************************************************************************/
void ViewUpdateDate(
FLMUINT UpdateFlag,
F_TMSTAMP * LastTime
)
{
F_TMSTAMP CurrTime;
char TempBuf[ 40];
FLMUINT Hour;
FLMBYTE AmPm[ 4];
FLMUINT uiNumCols;
FLMUINT uiNumRows;
f_conGetScreenSize( &uiNumCols, &uiNumRows);
f_timeGetTimeStamp( &CurrTime);
/* Update the date, if it has changed or the UpdateFlag is set */
if ((UpdateFlag) ||
(LastTime->year != CurrTime.year) ||
(LastTime->month != CurrTime.month) ||
(LastTime->day != CurrTime.day))
{
f_sprintf( (char *)TempBuf, "%s %u, %u",
Months[ CurrTime.month],
(unsigned)CurrTime.day,
(unsigned)CurrTime.year);
f_conSetBackFore( FLM_BLACK, FLM_WHITE);
f_conStrOutXY( TempBuf, 0, 0);
}
/* Update the time, if it has changed or the UpdateFlag is set */
if ((UpdateFlag) ||
(LastTime->hour != CurrTime.hour) ||
(LastTime->minute != CurrTime.minute) ||
(LastTime->second != CurrTime.second))
{
if (CurrTime.hour == 0)
Hour = 12;
else if (CurrTime.hour > 12)
Hour = (FLMUINT)CurrTime.hour - 12;
else
Hour = (FLMUINT)CurrTime.hour;
if (CurrTime.hour >= 12)
f_strcpy( (char *)AmPm, "pm");
else
f_strcpy( (char *)AmPm, "am");
f_sprintf( (char *)TempBuf, "%2u:%02u:%02u %s",
(unsigned)Hour,
(unsigned)CurrTime.minute,
(unsigned)CurrTime.second, AmPm);
f_conSetBackFore( FLM_BLACK, FLM_WHITE);
f_conStrOutXY( TempBuf, 66, 0);
}
if ((UpdateFlag) ||
(gv_uiViewLastFileOffset != gv_uiViewCurrFileOffset) ||
((gv_pViewMenuCurrItem->ModFileOffset == VIEW_INVALID_FILE_OFFSET) &&
(gv_uiViewLastFileOffset != VIEW_INVALID_FILE_OFFSET)))
{
if (gv_pViewMenuCurrItem == NULL)
{
gv_uiViewLastFileOffset = VIEW_INVALID_FILE_OFFSET;
}
else if (gv_pViewMenuCurrItem->ModFileOffset == VIEW_INVALID_FILE_OFFSET)
{
gv_uiViewLastFileNumber = gv_pViewMenuCurrItem->ModFileNumber;
gv_uiViewLastFileOffset = gv_pViewMenuCurrItem->ModFileOffset;
}
else
{
gv_uiViewLastFileNumber = gv_uiViewCurrFileNumber;
gv_uiViewLastFileOffset = gv_uiViewCurrFileOffset;
}
if (gv_uiViewLastFileOffset == VIEW_INVALID_FILE_OFFSET)
f_strcpy( TempBuf, "File: N/A File Pos: N/A ");
else
f_sprintf( (char *)TempBuf, "File: %03u File Pos: 0x%08X",
(unsigned)gv_uiViewLastFileNumber, (unsigned)gv_uiViewLastFileOffset);
f_conSetBackFore( FLM_BLACK, FLM_WHITE);
f_conStrOutXY( TempBuf, 47, uiNumRows - 1);
}
/* Save the date and time */
f_memcpy( LastTime, &CurrTime, sizeof( F_TMSTAMP));
}
/***************************************************************************
Name: ViewReset
Desc: This routine resets the view parameters for a menu and saves
the parameters for the current menu. This is done whenever a new
menu is being entered. It allows the previous menu to be restored
to its original state upon returning.
*****************************************************************************/
void ViewReset(
VIEW_INFO_p SaveView
)
{
SaveView->CurrItem = gv_uiViewMenuCurrItemNum;
SaveView->TopRow = gv_uiViewTopRow;
SaveView->BottomRow = gv_uiViewBottomRow;
SaveView->CurrFileOffset = gv_uiViewCurrFileOffset;
SaveView->CurrFileNumber = gv_uiViewCurrFileNumber;
gv_pViewMenuCurrItem = NULL;
gv_uiViewMenuCurrItemNum = 0;
gv_uiViewTopRow = 0;
gv_uiViewBottomRow = gv_uiViewTopRow + LINES_PER_PAGE - 1;
gv_uiViewCurrFileOffset = 0;
gv_uiViewCurrFileNumber = 0;
}
/********************************************************************
Desc: ?
*********************************************************************/
void ViewDisable(
void
)
{
gv_bViewEnabled = FALSE;
}
/********************************************************************
Desc: ?
*********************************************************************/
void ViewEnable(
void
)
{
VIEW_MENU_ITEM_p vp;
FLMUINT Distance = 0xFFFFFFFF;
VIEW_MENU_ITEM_p Closest = NULL;
FLMUINT StartOffset;
FLMUINT EndOffset;
if (!gv_bViewEnabled)
{
if (gv_uiViewCurrFileOffset != VIEW_INVALID_FILE_OFFSET)
{
vp = gv_pViewMenuFirstItem;
while( vp != NULL)
{
if (vp->ModFileOffset != VIEW_INVALID_FILE_OFFSET)
{
StartOffset = vp->ModFileOffset;
switch( vp->ModType & 0x0F)
{
case MOD_FLMUINT:
case MOD_KEY_LEN:
EndOffset = StartOffset + 3;
break;
case MOD_FLMUINT16:
EndOffset = StartOffset + 1;
break;
case MOD_BINARY:
case MOD_TEXT:
EndOffset = StartOffset + vp->ModBufLen - 1;
break;
case MOD_CHILD_BLK:
EndOffset = StartOffset + 2;
break;
default:
EndOffset = StartOffset;
break;
}
if ((gv_uiViewCurrFileOffset >= StartOffset) &&
(gv_uiViewCurrFileOffset <= EndOffset))
{
if ((vp->ModType & 0xF0) == MOD_DISABLED)
{
Closest = vp;
Distance = 0;
}
else
{
Closest = vp;
break;
}
}
else if (gv_uiViewCurrFileOffset < StartOffset)
{
if (StartOffset - gv_uiViewCurrFileOffset < Distance)
{
Closest = vp;
Distance = StartOffset - gv_uiViewCurrFileOffset;
}
}
else
{
if (gv_uiViewCurrFileOffset - StartOffset < Distance)
{
Closest = vp;
Distance = gv_uiViewCurrFileOffset - StartOffset;
}
}
}
vp = vp->NextItem;
}
}
if (Closest != NULL)
{
gv_pViewMenuCurrItem = vp = Closest;
gv_uiViewMenuCurrItemNum = vp->ItemNum;
if (vp->Row < LINES_PER_PAGE)
gv_uiViewTopRow = 0;
else
gv_uiViewTopRow = vp->Row - LINES_PER_PAGE / 2 + 1;
gv_uiViewBottomRow = gv_uiViewTopRow + LINES_PER_PAGE - 1;
if (gv_uiViewBottomRow > gv_pViewMenuLastItem->Row)
gv_uiViewBottomRow = gv_pViewMenuLastItem->Row;
if (gv_uiViewBottomRow - gv_uiViewTopRow + 1 < LINES_PER_PAGE)
{
if (gv_uiViewBottomRow < LINES_PER_PAGE + 1)
gv_uiViewTopRow = 0;
else
gv_uiViewTopRow = gv_uiViewBottomRow + 1 - LINES_PER_PAGE;
}
if ((HAVE_HORIZ_CUR( vp)) &&
(gv_uiViewCurrFileOffset - vp->ModFileOffset <=
(FLMUINT)vp->ModBufLen))
vp->HorizCurPos =
(FLMUINT)(gv_uiViewCurrFileOffset - vp->ModFileOffset);
}
gv_bViewEnabled = TRUE;
ViewRefreshMenu( NULL);
}
}
/***************************************************************************
Name: ViewRestore
Desc: This routine restores the view parameters for a menu which were
previously saved by the ViewReset routine.
*****************************************************************************/
void ViewRestore(
VIEW_INFO_p SaveView
)
{
gv_uiViewMenuCurrItemNum = SaveView->CurrItem;
gv_pViewMenuCurrItem = NULL;
gv_uiViewTopRow = SaveView->TopRow;
gv_uiViewBottomRow = SaveView->BottomRow;
gv_uiViewCurrFileOffset = SaveView->CurrFileOffset;
gv_uiViewCurrFileNumber = SaveView->CurrFileNumber;
}