2008-07-15 10:35:54 +02:00
/** @file
Utility functions for User Interface functions .
2012-01-31 08:17:42 +01:00
Copyright ( c ) 2004 - 2012 , Intel Corporation . All rights reserved . < BR >
2010-04-24 11:33:45 +02:00
This program and the accompanying materials
2008-07-15 10:35:54 +02:00
are licensed and made available under the terms and conditions of the BSD License
which accompanies this distribution . The full text of the license may be found at
http : //opensource.org/licenses/bsd-license.php
THE PROGRAM IS DISTRIBUTED UNDER THE BSD LICENSE ON AN " AS IS " BASIS ,
WITHOUT WARRANTIES OR REPRESENTATIONS OF ANY KIND , EITHER EXPRESS OR IMPLIED .
* */
# include "Setup.h"
2009-10-26 04:06:06 +01:00
LIST_ENTRY gMenuOption ;
2009-10-26 04:03:12 +01:00
LIST_ENTRY gMenuList = INITIALIZE_LIST_HEAD_VARIABLE ( gMenuList ) ;
2011-06-08 10:09:47 +02:00
MENU_REFRESH_ENTRY * gMenuRefreshHead ; // Menu list used for refresh timer opcode.
MENU_REFRESH_ENTRY * gMenuEventGuidRefreshHead ; // Menu list used for refresh event guid opcode.
2008-07-15 10:35:54 +02:00
//
// Search table for UiDisplayMenu()
//
SCAN_CODE_TO_SCREEN_OPERATION gScanCodeToOperation [ ] = {
{
SCAN_UP ,
UiUp ,
} ,
{
SCAN_DOWN ,
UiDown ,
} ,
{
SCAN_PAGE_UP ,
UiPageUp ,
} ,
{
SCAN_PAGE_DOWN ,
UiPageDown ,
} ,
{
SCAN_ESC ,
UiReset ,
} ,
{
SCAN_LEFT ,
UiLeft ,
} ,
{
SCAN_RIGHT ,
UiRight ,
}
} ;
2011-09-13 11:32:54 +02:00
UINTN mScanCodeNumber = sizeof ( gScanCodeToOperation ) / sizeof ( gScanCodeToOperation [ 0 ] ) ;
2008-07-15 10:35:54 +02:00
SCREEN_OPERATION_T0_CONTROL_FLAG gScreenOperationToControlFlag [ ] = {
{
UiNoOperation ,
CfUiNoOperation ,
} ,
{
UiSelect ,
CfUiSelect ,
} ,
{
UiUp ,
CfUiUp ,
} ,
{
UiDown ,
CfUiDown ,
} ,
{
UiLeft ,
CfUiLeft ,
} ,
{
UiRight ,
CfUiRight ,
} ,
{
UiReset ,
CfUiReset ,
} ,
{
UiPageUp ,
CfUiPageUp ,
} ,
{
UiPageDown ,
CfUiPageDown
2011-09-13 11:32:54 +02:00
} ,
{
UiHotKey ,
CfUiHotKey
2008-07-15 10:35:54 +02:00
}
} ;
2008-10-31 03:04:40 +01:00
BOOLEAN mInputError ;
2008-12-03 09:52:39 +01:00
BOOLEAN GetLineByWidthFinished = FALSE ;
2008-10-31 03:04:40 +01:00
2008-07-15 10:35:54 +02:00
/**
Set Buffer to Value for Size bytes .
@ param Buffer Memory to set .
@ param Size Number of bytes to set
@ param Value Value of the set operation .
* */
VOID
SetUnicodeMem (
IN VOID * Buffer ,
IN UINTN Size ,
IN CHAR16 Value
)
{
CHAR16 * Ptr ;
Ptr = Buffer ;
while ( ( Size - - ) ! = 0 ) {
* ( Ptr + + ) = Value ;
}
}
/**
Initialize Menu option list .
* */
VOID
UiInitMenu (
VOID
)
{
2009-10-26 04:06:06 +01:00
InitializeListHead ( & gMenuOption ) ;
2008-07-15 10:35:54 +02:00
}
/**
2009-10-26 04:03:12 +01:00
Free Menu option linked list .
2008-07-15 10:35:54 +02:00
* */
VOID
2009-10-26 04:03:12 +01:00
UiFreeMenu (
2008-07-15 10:35:54 +02:00
VOID
)
{
2009-10-26 04:03:12 +01:00
UI_MENU_OPTION * MenuOption ;
2009-10-26 04:06:06 +01:00
while ( ! IsListEmpty ( & gMenuOption ) ) {
MenuOption = MENU_OPTION_FROM_LINK ( gMenuOption . ForwardLink ) ;
2009-10-26 04:03:12 +01:00
RemoveEntryList ( & MenuOption - > Link ) ;
//
// We allocated space for this description when we did a GetToken, free it here
//
if ( MenuOption - > Skip ! = 0 ) {
//
// For date/time, MenuOption->Description is shared by three Menu Options
// Data format : [01/02/2004] [11:22:33]
// Line number : 0 0 1 0 0 1
//
FreePool ( MenuOption - > Description ) ;
}
FreePool ( MenuOption ) ;
}
2008-07-15 10:35:54 +02:00
}
/**
2009-10-26 04:03:12 +01:00
Create a menu with specified formset GUID and form ID , and add it as a child
of the given parent menu .
2008-07-15 10:35:54 +02:00
2009-10-26 04:03:12 +01:00
@ param Parent The parent of menu to be added .
2011-10-31 04:33:55 +01:00
@ param HiiHandle Hii handle related to this formset .
2009-10-26 04:03:12 +01:00
@ param FormSetGuid The Formset Guid of menu to be added .
@ param FormId The Form ID of menu to be added .
@ return A pointer to the newly added menu or NULL if memory is insufficient .
2008-07-15 10:35:54 +02:00
* */
2009-10-26 04:03:12 +01:00
UI_MENU_LIST *
UiAddMenuList (
IN OUT UI_MENU_LIST * Parent ,
2011-10-31 04:33:55 +01:00
IN EFI_HII_HANDLE HiiHandle ,
2009-10-26 04:03:12 +01:00
IN EFI_GUID * FormSetGuid ,
IN UINT16 FormId
2008-07-15 10:35:54 +02:00
)
{
2009-10-26 04:03:12 +01:00
UI_MENU_LIST * MenuList ;
2008-07-15 10:35:54 +02:00
2009-10-26 04:03:12 +01:00
MenuList = AllocateZeroPool ( sizeof ( UI_MENU_LIST ) ) ;
if ( MenuList = = NULL ) {
return NULL ;
2008-07-15 10:35:54 +02:00
}
2009-10-26 04:03:12 +01:00
MenuList - > Signature = UI_MENU_LIST_SIGNATURE ;
InitializeListHead ( & MenuList - > ChildListHead ) ;
2008-07-15 10:35:54 +02:00
2011-10-31 04:33:55 +01:00
MenuList - > HiiHandle = HiiHandle ;
2009-10-26 04:03:12 +01:00
CopyMem ( & MenuList - > FormSetGuid , FormSetGuid , sizeof ( EFI_GUID ) ) ;
MenuList - > FormId = FormId ;
MenuList - > Parent = Parent ;
2008-07-15 10:35:54 +02:00
2009-10-26 04:03:12 +01:00
if ( Parent = = NULL ) {
//
// If parent is not specified, it is the root Form of a Formset
//
InsertTailList ( & gMenuList , & MenuList - > Link ) ;
} else {
InsertTailList ( & Parent - > ChildListHead , & MenuList - > Link ) ;
2008-07-15 10:35:54 +02:00
}
2009-10-26 04:03:12 +01:00
return MenuList ;
2008-07-15 10:35:54 +02:00
}
/**
2011-10-31 04:33:55 +01:00
Search Menu with given FormId and FormSetGuid in all cached menu list .
2008-07-15 10:35:54 +02:00
2009-10-26 04:03:12 +01:00
@ param Parent The parent of menu to search .
2011-10-31 04:33:55 +01:00
@ param FormSetGuid The Formset GUID of the menu to search .
2009-10-26 04:03:12 +01:00
@ param FormId The Form ID of menu to search .
@ return A pointer to menu found or NULL if not found .
2008-07-15 10:35:54 +02:00
* */
2009-10-26 04:03:12 +01:00
UI_MENU_LIST *
UiFindChildMenuList (
IN UI_MENU_LIST * Parent ,
2011-10-31 04:33:55 +01:00
IN EFI_GUID * FormSetGuid ,
2009-10-26 04:03:12 +01:00
IN UINT16 FormId
2008-07-15 10:35:54 +02:00
)
{
2009-10-26 04:03:12 +01:00
LIST_ENTRY * Link ;
UI_MENU_LIST * Child ;
UI_MENU_LIST * MenuList ;
2011-10-31 04:33:55 +01:00
ASSERT ( Parent ! = NULL ) ;
if ( Parent - > FormId = = FormId & & CompareGuid ( FormSetGuid , & Parent - > FormSetGuid ) ) {
2009-10-26 04:03:12 +01:00
return Parent ;
}
2008-07-15 10:35:54 +02:00
2009-10-26 04:03:12 +01:00
Link = GetFirstNode ( & Parent - > ChildListHead ) ;
while ( ! IsNull ( & Parent - > ChildListHead , Link ) ) {
Child = UI_MENU_LIST_FROM_LINK ( Link ) ;
2008-07-15 10:35:54 +02:00
2011-10-31 04:33:55 +01:00
MenuList = UiFindChildMenuList ( Child , FormSetGuid , FormId ) ;
2009-10-26 04:03:12 +01:00
if ( MenuList ! = NULL ) {
return MenuList ;
}
2008-07-15 10:35:54 +02:00
2009-10-26 04:03:12 +01:00
Link = GetNextNode ( & Parent - > ChildListHead , Link ) ;
}
return NULL ;
2008-07-15 10:35:54 +02:00
}
/**
2009-10-26 04:03:12 +01:00
Search Menu with given FormSetGuid and FormId in all cached menu list .
@ param FormSetGuid The Formset GUID of the menu to search .
@ param FormId The Form ID of menu to search .
@ return A pointer to menu found or NULL if not found .
2008-07-15 10:35:54 +02:00
* */
2009-10-26 04:03:12 +01:00
UI_MENU_LIST *
UiFindMenuList (
IN EFI_GUID * FormSetGuid ,
IN UINT16 FormId
2008-07-15 10:35:54 +02:00
)
{
2009-10-26 04:03:12 +01:00
LIST_ENTRY * Link ;
UI_MENU_LIST * MenuList ;
UI_MENU_LIST * Child ;
2008-07-15 10:35:54 +02:00
2009-10-26 04:03:12 +01:00
Link = GetFirstNode ( & gMenuList ) ;
while ( ! IsNull ( & gMenuList , Link ) ) {
MenuList = UI_MENU_LIST_FROM_LINK ( Link ) ;
2008-07-15 10:35:54 +02:00
2011-10-31 04:33:55 +01:00
Child = UiFindChildMenuList ( MenuList , FormSetGuid , FormId ) ;
if ( Child ! = NULL ) {
return Child ;
2008-07-15 10:35:54 +02:00
}
2009-10-26 04:03:12 +01:00
Link = GetNextNode ( & gMenuList , Link ) ;
2008-07-15 10:35:54 +02:00
}
2009-10-26 04:03:12 +01:00
return NULL ;
2008-07-15 10:35:54 +02:00
}
/**
Free Menu option linked list .
* */
VOID
UiFreeRefreshList (
VOID
)
{
MENU_REFRESH_ENTRY * OldMenuRefreshEntry ;
while ( gMenuRefreshHead ! = NULL ) {
OldMenuRefreshEntry = gMenuRefreshHead - > Next ;
2008-11-21 07:59:58 +01:00
FreePool ( gMenuRefreshHead ) ;
2008-07-15 10:35:54 +02:00
gMenuRefreshHead = OldMenuRefreshEntry ;
}
2011-06-08 10:09:47 +02:00
while ( gMenuEventGuidRefreshHead ! = NULL ) {
OldMenuRefreshEntry = gMenuEventGuidRefreshHead - > Next ;
if ( gMenuEventGuidRefreshHead ! = NULL ) {
gBS - > CloseEvent ( gMenuEventGuidRefreshHead - > Event ) ;
}
FreePool ( gMenuEventGuidRefreshHead ) ;
gMenuEventGuidRefreshHead = OldMenuRefreshEntry ;
}
2008-07-15 10:35:54 +02:00
}
2011-11-16 08:16:03 +01:00
/**
Process option string for date / time opcode .
@ param MenuOption Menu option point to date / time .
@ param OptionString Option string input for process .
@ param AddOptCol Whether need to update MenuOption - > OptCol .
* */
VOID
ProcessStringForDateTime (
UI_MENU_OPTION * MenuOption ,
CHAR16 * OptionString ,
BOOLEAN AddOptCol
)
{
UINTN Index ;
UINTN Count ;
FORM_BROWSER_STATEMENT * Statement ;
ASSERT ( MenuOption ! = NULL & & OptionString ! = NULL ) ;
Statement = MenuOption - > ThisTag ;
//
// If leading spaces on OptionString - remove the spaces
//
for ( Index = 0 ; OptionString [ Index ] = = L ' ' ; Index + + ) {
//
// Base on the blockspace to get the option column info.
//
if ( AddOptCol ) {
MenuOption - > OptCol + + ;
}
}
for ( Count = 0 ; OptionString [ Index ] ! = CHAR_NULL ; Index + + ) {
OptionString [ Count ] = OptionString [ Index ] ;
Count + + ;
}
OptionString [ Count ] = CHAR_NULL ;
//
// Enable to suppress field in the opcode base on the flag.
//
if ( Statement - > Operand = = EFI_IFR_DATE_OP ) {
//
// OptionString format is: <**: **: ****>
// |month|day|year|
// 4 3 5
//
if ( ( Statement - > Flags & EFI_QF_DATE_MONTH_SUPPRESS ) & & ( MenuOption - > Sequence = = 0 ) ) {
//
// At this point, only "<**:" in the optionstring.
// Clean the day's ** field, after clean, the format is "< :"
//
SetUnicodeMem ( & OptionString [ 1 ] , 2 , L ' ' ) ;
} else if ( ( Statement - > Flags & EFI_QF_DATE_DAY_SUPPRESS ) & & ( MenuOption - > Sequence = = 1 ) ) {
//
// At this point, only "**:" in the optionstring.
// Clean the month's "**" field, after clean, the format is " :"
//
SetUnicodeMem ( & OptionString [ 0 ] , 2 , L ' ' ) ;
} else if ( ( Statement - > Flags & EFI_QF_DATE_YEAR_SUPPRESS ) & & ( MenuOption - > Sequence = = 2 ) ) {
//
// At this point, only "****>" in the optionstring.
// Clean the year's "****" field, after clean, the format is " >"
//
SetUnicodeMem ( & OptionString [ 0 ] , 4 , L ' ' ) ;
}
} else if ( Statement - > Operand = = EFI_IFR_TIME_OP ) {
//
// OptionString format is: <**: **: **>
// |hour|minute|second|
// 4 3 3
//
if ( ( Statement - > Flags & QF_TIME_HOUR_SUPPRESS ) & & ( MenuOption - > Sequence = = 0 ) ) {
//
// At this point, only "<**:" in the optionstring.
// Clean the hour's ** field, after clean, the format is "< :"
//
SetUnicodeMem ( & OptionString [ 1 ] , 2 , L ' ' ) ;
} else if ( ( Statement - > Flags & QF_TIME_MINUTE_SUPPRESS ) & & ( MenuOption - > Sequence = = 1 ) ) {
//
// At this point, only "**:" in the optionstring.
// Clean the minute's "**" field, after clean, the format is " :"
//
SetUnicodeMem ( & OptionString [ 0 ] , 2 , L ' ' ) ;
} else if ( ( Statement - > Flags & QF_TIME_SECOND_SUPPRESS ) & & ( MenuOption - > Sequence = = 2 ) ) {
//
// At this point, only "**>" in the optionstring.
// Clean the second's "**" field, after clean, the format is " >"
//
SetUnicodeMem ( & OptionString [ 0 ] , 2 , L ' ' ) ;
}
}
}
2008-07-15 10:35:54 +02:00
/**
2011-06-08 10:09:47 +02:00
Refresh question .
2008-07-15 10:35:54 +02:00
2011-06-08 10:09:47 +02:00
@ param MenuRefreshEntry Menu refresh structure which has info about the refresh question .
2008-07-15 10:35:54 +02:00
* */
2011-06-08 10:09:47 +02:00
EFI_STATUS
RefreshQuestion (
IN MENU_REFRESH_ENTRY * MenuRefreshEntry
2008-07-15 10:35:54 +02:00
)
{
2008-11-04 15:28:08 +01:00
CHAR16 * OptionString ;
EFI_STATUS Status ;
UI_MENU_SELECTION * Selection ;
FORM_BROWSER_STATEMENT * Question ;
2008-07-15 10:35:54 +02:00
2011-06-08 10:09:47 +02:00
Selection = MenuRefreshEntry - > Selection ;
Question = MenuRefreshEntry - > MenuOption - > ThisTag ;
2008-07-15 10:35:54 +02:00
2011-06-08 10:09:47 +02:00
Status = GetQuestionValue ( Selection - > FormSet , Selection - > Form , Question , FALSE ) ;
if ( EFI_ERROR ( Status ) ) {
return Status ;
}
2008-07-15 10:35:54 +02:00
2011-06-08 10:09:47 +02:00
OptionString = NULL ;
ProcessOptions ( Selection , MenuRefreshEntry - > MenuOption , FALSE , & OptionString ) ;
if ( OptionString ! = NULL ) {
//
// If old Text is longer than new string, need to clean the old string before paint the newer.
// This option is no need for time/date opcode, because time/data opcode has fixed string length.
//
if ( ( MenuRefreshEntry - > MenuOption - > ThisTag - > Operand ! = EFI_IFR_DATE_OP ) & &
( MenuRefreshEntry - > MenuOption - > ThisTag - > Operand ! = EFI_IFR_TIME_OP ) ) {
ClearLines (
MenuRefreshEntry - > CurrentColumn ,
MenuRefreshEntry - > CurrentColumn + gOptionBlockWidth - 1 ,
MenuRefreshEntry - > CurrentRow ,
MenuRefreshEntry - > CurrentRow ,
PcdGet8 ( PcdBrowserFieldTextColor ) | FIELD_BACKGROUND
) ;
}
2008-07-15 10:35:54 +02:00
2011-06-08 10:09:47 +02:00
gST - > ConOut - > SetAttribute ( gST - > ConOut , MenuRefreshEntry - > CurrentAttribute ) ;
2011-11-16 08:16:03 +01:00
ProcessStringForDateTime ( MenuRefreshEntry - > MenuOption , OptionString , FALSE ) ;
PrintStringAt ( MenuRefreshEntry - > CurrentColumn , MenuRefreshEntry - > CurrentRow , OptionString ) ;
2011-06-08 10:09:47 +02:00
FreePool ( OptionString ) ;
}
2008-11-04 15:28:08 +01:00
2011-06-08 10:09:47 +02:00
//
// Question value may be changed, need invoke its Callback()
//
Status = ProcessCallBackFunction ( Selection , Question , EFI_BROWSER_ACTION_CHANGING , FALSE ) ;
2008-11-04 15:28:08 +01:00
2011-06-08 10:09:47 +02:00
return Status ;
}
2008-11-04 15:28:08 +01:00
2011-06-08 10:09:47 +02:00
/**
Refresh the question which has refresh guid event attribute .
@ param Event The event which has this function related .
@ param Context The input context info related to this event or the status code return to the caller .
* */
VOID
2011-06-13 03:55:28 +02:00
EFIAPI
2011-06-08 10:09:47 +02:00
RefreshQuestionNotify (
IN EFI_EVENT Event ,
IN VOID * Context
)
{
MENU_REFRESH_ENTRY * MenuRefreshEntry ;
UI_MENU_SELECTION * Selection ;
2011-03-10 03:28:15 +01:00
2011-06-08 10:09:47 +02:00
//
// Reset FormPackage update flag
//
mHiiPackageListUpdated = FALSE ;
2008-11-04 15:28:08 +01:00
2011-06-08 10:09:47 +02:00
MenuRefreshEntry = ( MENU_REFRESH_ENTRY * ) Context ;
ASSERT ( MenuRefreshEntry ! = NULL ) ;
Selection = MenuRefreshEntry - > Selection ;
RefreshQuestion ( MenuRefreshEntry ) ;
if ( mHiiPackageListUpdated ) {
//
// Package list is updated, force to reparse IFR binary of target Formset
//
mHiiPackageListUpdated = FALSE ;
Selection - > Action = UI_ACTION_REFRESH_FORMSET ;
}
}
/**
Refresh screen .
* */
EFI_STATUS
RefreshForm (
VOID
)
{
MENU_REFRESH_ENTRY * MenuRefreshEntry ;
EFI_STATUS Status ;
UI_MENU_SELECTION * Selection ;
if ( gMenuRefreshHead ! = NULL ) {
//
// call from refresh interval process.
//
MenuRefreshEntry = gMenuRefreshHead ;
Selection = MenuRefreshEntry - > Selection ;
//
// Reset FormPackage update flag
//
mHiiPackageListUpdated = FALSE ;
do {
Status = RefreshQuestion ( MenuRefreshEntry ) ;
2011-05-31 02:59:15 +02:00
if ( EFI_ERROR ( Status ) ) {
return Status ;
2008-07-15 10:35:54 +02:00
}
MenuRefreshEntry = MenuRefreshEntry - > Next ;
} while ( MenuRefreshEntry ! = NULL ) ;
2008-11-04 15:28:08 +01:00
if ( mHiiPackageListUpdated ) {
//
// Package list is updated, force to reparse IFR binary of target Formset
//
mHiiPackageListUpdated = FALSE ;
Selection - > Action = UI_ACTION_REFRESH_FORMSET ;
return EFI_SUCCESS ;
}
2008-07-15 10:35:54 +02:00
}
2008-11-04 15:28:08 +01:00
return EFI_TIMEOUT ;
2008-07-15 10:35:54 +02:00
}
/**
Wait for a given event to fire , or for an optional timeout to expire .
@ param Event The event to wait for
@ param Timeout An optional timeout value in 100 ns units .
@ param RefreshInterval Menu refresh interval ( in seconds ) .
@ retval EFI_SUCCESS Event fired before Timeout expired .
@ retval EFI_TIME_OUT Timout expired before Event fired .
* */
EFI_STATUS
UiWaitForSingleEvent (
IN EFI_EVENT Event ,
IN UINT64 Timeout , OPTIONAL
IN UINT8 RefreshInterval OPTIONAL
)
{
EFI_STATUS Status ;
UINTN Index ;
EFI_EVENT TimerEvent ;
EFI_EVENT WaitList [ 2 ] ;
if ( Timeout ! = 0 ) {
//
// Create a timer event
//
Status = gBS - > CreateEvent ( EVT_TIMER , 0 , NULL , NULL , & TimerEvent ) ;
if ( ! EFI_ERROR ( Status ) ) {
//
// Set the timer event
//
gBS - > SetTimer (
TimerEvent ,
TimerRelative ,
Timeout
) ;
//
// Wait for the original event or the timer
//
WaitList [ 0 ] = Event ;
WaitList [ 1 ] = TimerEvent ;
Status = gBS - > WaitForEvent ( 2 , WaitList , & Index ) ;
gBS - > CloseEvent ( TimerEvent ) ;
//
// If the timer expired, change the return to timed out
//
if ( ! EFI_ERROR ( Status ) & & Index = = 1 ) {
Status = EFI_TIMEOUT ;
}
}
} else {
//
// Update screen every second
//
if ( RefreshInterval = = 0 ) {
Timeout = ONE_SECOND ;
} else {
Timeout = RefreshInterval * ONE_SECOND ;
}
do {
Status = gBS - > CreateEvent ( EVT_TIMER , 0 , NULL , NULL , & TimerEvent ) ;
//
// Set the timer event
//
gBS - > SetTimer (
TimerEvent ,
TimerRelative ,
Timeout
) ;
//
// Wait for the original event or the timer
//
WaitList [ 0 ] = Event ;
WaitList [ 1 ] = TimerEvent ;
Status = gBS - > WaitForEvent ( 2 , WaitList , & Index ) ;
//
// If the timer expired, update anything that needs a refresh and keep waiting
//
if ( ! EFI_ERROR ( Status ) & & Index = = 1 ) {
Status = EFI_TIMEOUT ;
if ( RefreshInterval ! = 0 ) {
2008-11-04 15:28:08 +01:00
Status = RefreshForm ( ) ;
2008-07-15 10:35:54 +02:00
}
}
gBS - > CloseEvent ( TimerEvent ) ;
} while ( Status = = EFI_TIMEOUT ) ;
}
return Status ;
}
/**
Add one menu option by specified description and context .
@ param String String description for this option .
@ param Handle Hii handle for the package list .
2011-11-16 06:23:38 +01:00
@ param Form The form this statement belong to .
2008-07-15 10:35:54 +02:00
@ param Statement Statement of this Menu Option .
@ param NumberOfLines Display lines for this Menu Option .
@ param MenuItemCount The index for this Option in the Menu .
2009-10-26 04:05:16 +01:00
@ retval Pointer Pointer to the added Menu Option .
2008-07-15 10:35:54 +02:00
* */
2009-10-26 04:05:16 +01:00
UI_MENU_OPTION *
2008-07-15 10:35:54 +02:00
UiAddMenuOption (
IN CHAR16 * String ,
IN EFI_HII_HANDLE Handle ,
2011-11-16 06:23:38 +01:00
IN FORM_BROWSER_FORM * Form ,
2008-07-15 10:35:54 +02:00
IN FORM_BROWSER_STATEMENT * Statement ,
IN UINT16 NumberOfLines ,
IN UINT16 MenuItemCount
)
{
UI_MENU_OPTION * MenuOption ;
UINTN Index ;
UINTN Count ;
Count = 1 ;
2009-10-26 04:05:16 +01:00
MenuOption = NULL ;
2008-07-15 10:35:54 +02:00
if ( Statement - > Operand = = EFI_IFR_DATE_OP | | Statement - > Operand = = EFI_IFR_TIME_OP ) {
//
// Add three MenuOptions for Date/Time
// Data format : [01/02/2004] [11:22:33]
// Line number : 0 0 1 0 0 1
//
NumberOfLines = 0 ;
Count = 3 ;
if ( Statement - > Storage = = NULL ) {
//
// For RTC type of date/time, set default refresh interval to be 1 second
//
if ( Statement - > RefreshInterval = = 0 ) {
Statement - > RefreshInterval = 1 ;
}
}
}
for ( Index = 0 ; Index < Count ; Index + + ) {
MenuOption = AllocateZeroPool ( sizeof ( UI_MENU_OPTION ) ) ;
ASSERT ( MenuOption ) ;
MenuOption - > Signature = UI_MENU_OPTION_SIGNATURE ;
MenuOption - > Description = String ;
MenuOption - > Handle = Handle ;
MenuOption - > ThisTag = Statement ;
MenuOption - > EntryNumber = MenuItemCount ;
if ( Index = = 2 ) {
//
// Override LineNumber for the MenuOption in Date/Time sequence
//
MenuOption - > Skip = 1 ;
} else {
MenuOption - > Skip = NumberOfLines ;
}
MenuOption - > Sequence = Index ;
2012-01-31 08:17:42 +01:00
if ( EvaluateExpressionList ( Statement - > Expression , FALSE , NULL , NULL ) = = ExpressGrayOut ) {
MenuOption - > GrayOut = TRUE ;
} else {
MenuOption - > GrayOut = FALSE ;
2008-07-15 10:35:54 +02:00
}
2011-11-16 06:23:38 +01:00
//
// If the form or the question has the lock attribute, deal same as grayout.
//
if ( Form - > Locked | | Statement - > Locked ) {
MenuOption - > GrayOut = TRUE ;
}
2009-10-26 04:05:16 +01:00
switch ( Statement - > Operand ) {
case EFI_IFR_ORDERED_LIST_OP :
case EFI_IFR_ONE_OF_OP :
case EFI_IFR_NUMERIC_OP :
case EFI_IFR_TIME_OP :
case EFI_IFR_DATE_OP :
case EFI_IFR_CHECKBOX_OP :
case EFI_IFR_PASSWORD_OP :
case EFI_IFR_STRING_OP :
//
// User could change the value of these items
//
MenuOption - > IsQuestion = TRUE ;
break ;
2011-03-08 07:55:15 +01:00
case EFI_IFR_TEXT_OP :
if ( FeaturePcdGet ( PcdBrowserGrayOutTextStatement ) ) {
//
// Initializing GrayOut option as TRUE for Text setup options
// so that those options will be Gray in colour and un selectable.
//
MenuOption - > GrayOut = TRUE ;
}
2009-10-26 04:05:16 +01:00
default :
MenuOption - > IsQuestion = FALSE ;
break ;
}
2008-07-15 10:35:54 +02:00
if ( ( Statement - > ValueExpression ! = NULL ) | |
( ( Statement - > QuestionFlags & EFI_IFR_FLAG_READ_ONLY ) ! = 0 ) ) {
MenuOption - > ReadOnly = TRUE ;
}
2009-10-26 04:06:06 +01:00
InsertTailList ( & gMenuOption , & MenuOption - > Link ) ;
2008-07-15 10:35:54 +02:00
}
2009-10-26 04:05:16 +01:00
return MenuOption ;
2008-07-15 10:35:54 +02:00
}
/**
Routine used to abstract a generic dialog interface and return the selected key or string
@ param NumberOfLines The number of lines for the dialog box
@ param HotKey Defines whether a single character is parsed
( TRUE ) and returned in KeyValue or a string is
returned in StringBuffer . Two special characters
are considered when entering a string , a SCAN_ESC
and an CHAR_CARRIAGE_RETURN . SCAN_ESC terminates
string input and returns
@ param MaximumStringSize The maximum size in bytes of a typed in string
( each character is a CHAR16 ) and the minimum
string returned is two bytes
@ param StringBuffer The passed in pointer to the buffer which will
hold the typed in string if HotKey is FALSE
@ param KeyValue The EFI_KEY value returned if HotKey is TRUE . .
@ param . . . A series of ( quantity = = NumberOfLines ) text
strings which will be used to construct the dialog
box
@ retval EFI_SUCCESS Displayed dialog and received user interaction
@ retval EFI_INVALID_PARAMETER One of the parameters was invalid ( e . g .
( StringBuffer = = NULL ) & & ( HotKey = = FALSE ) )
@ retval EFI_DEVICE_ERROR User typed in an ESC character to exit the routine
* */
EFI_STATUS
2010-04-29 03:07:07 +02:00
EFIAPI
2008-07-15 10:35:54 +02:00
CreateDialog (
IN UINTN NumberOfLines ,
IN BOOLEAN HotKey ,
IN UINTN MaximumStringSize ,
OUT CHAR16 * StringBuffer ,
OUT EFI_INPUT_KEY * KeyValue ,
. . .
)
{
VA_LIST Marker ;
UINTN Count ;
EFI_INPUT_KEY Key ;
UINTN LargestString ;
CHAR16 * TempString ;
CHAR16 * BufferedString ;
CHAR16 * StackString ;
CHAR16 KeyPad [ 2 ] ;
UINTN Start ;
UINTN Top ;
UINTN Index ;
EFI_STATUS Status ;
BOOLEAN SelectionComplete ;
UINTN InputOffset ;
UINTN CurrentAttribute ;
UINTN DimensionsWidth ;
UINTN DimensionsHeight ;
DimensionsWidth = gScreenDimensions . RightColumn - gScreenDimensions . LeftColumn ;
DimensionsHeight = gScreenDimensions . BottomRow - gScreenDimensions . TopRow ;
SelectionComplete = FALSE ;
InputOffset = 0 ;
TempString = AllocateZeroPool ( MaximumStringSize * 2 ) ;
BufferedString = AllocateZeroPool ( MaximumStringSize * 2 ) ;
CurrentAttribute = gST - > ConOut - > Mode - > Attribute ;
ASSERT ( TempString ) ;
ASSERT ( BufferedString ) ;
//
// Zero the outgoing buffer
//
ZeroMem ( StringBuffer , MaximumStringSize ) ;
if ( HotKey ) {
if ( KeyValue = = NULL ) {
return EFI_INVALID_PARAMETER ;
}
} else {
if ( StringBuffer = = NULL ) {
return EFI_INVALID_PARAMETER ;
}
}
//
// Disable cursor
//
gST - > ConOut - > EnableCursor ( gST - > ConOut , FALSE ) ;
2008-07-16 05:04:53 +02:00
LargestString = 0 ;
2008-07-15 10:35:54 +02:00
2012-02-01 07:06:08 +01:00
VA_START ( Marker , KeyValue ) ;
2008-07-15 10:35:54 +02:00
//
// Determine the largest string in the dialog box
// Notice we are starting with 1 since String is the first string
//
2008-07-16 05:04:53 +02:00
for ( Count = 0 ; Count < NumberOfLines ; Count + + ) {
2008-07-15 10:35:54 +02:00
StackString = VA_ARG ( Marker , CHAR16 * ) ;
if ( StackString [ 0 ] = = L ' ' ) {
InputOffset = Count + 1 ;
}
if ( ( GetStringWidth ( StackString ) / 2 ) > LargestString ) {
//
// Size of the string visually and subtract the width by one for the null-terminator
//
LargestString = ( GetStringWidth ( StackString ) / 2 ) ;
}
}
2009-07-23 07:32:33 +02:00
VA_END ( Marker ) ;
2008-07-15 10:35:54 +02:00
Start = ( DimensionsWidth - LargestString - 2 ) / 2 + gScreenDimensions . LeftColumn + 1 ;
Top = ( ( DimensionsHeight - NumberOfLines - 2 ) / 2 ) + gScreenDimensions . TopRow - 1 ;
Count = 0 ;
//
// Display the Popup
//
2009-07-23 07:32:33 +02:00
VA_START ( Marker , KeyValue ) ;
CreateSharedPopUp ( LargestString , NumberOfLines , Marker ) ;
VA_END ( Marker ) ;
2008-07-15 10:35:54 +02:00
//
// Take the first key typed and report it back?
//
if ( HotKey ) {
Status = WaitForKeyStroke ( & Key ) ;
ASSERT_EFI_ERROR ( Status ) ;
CopyMem ( KeyValue , & Key , sizeof ( EFI_INPUT_KEY ) ) ;
} else {
do {
Status = WaitForKeyStroke ( & Key ) ;
switch ( Key . UnicodeChar ) {
case CHAR_NULL :
switch ( Key . ScanCode ) {
case SCAN_ESC :
2008-11-21 07:59:58 +01:00
FreePool ( TempString ) ;
FreePool ( BufferedString ) ;
2008-07-15 10:35:54 +02:00
gST - > ConOut - > SetAttribute ( gST - > ConOut , CurrentAttribute ) ;
gST - > ConOut - > EnableCursor ( gST - > ConOut , TRUE ) ;
return EFI_DEVICE_ERROR ;
default :
break ;
}
break ;
case CHAR_CARRIAGE_RETURN :
SelectionComplete = TRUE ;
2008-11-21 07:59:58 +01:00
FreePool ( TempString ) ;
FreePool ( BufferedString ) ;
2008-07-15 10:35:54 +02:00
gST - > ConOut - > SetAttribute ( gST - > ConOut , CurrentAttribute ) ;
gST - > ConOut - > EnableCursor ( gST - > ConOut , TRUE ) ;
return EFI_SUCCESS ;
break ;
case CHAR_BACKSPACE :
if ( StringBuffer [ 0 ] ! = CHAR_NULL ) {
for ( Index = 0 ; StringBuffer [ Index ] ! = CHAR_NULL ; Index + + ) {
TempString [ Index ] = StringBuffer [ Index ] ;
}
//
// Effectively truncate string by 1 character
//
TempString [ Index - 1 ] = CHAR_NULL ;
StrCpy ( StringBuffer , TempString ) ;
}
default :
//
// If it is the beginning of the string, don't worry about checking maximum limits
//
if ( ( StringBuffer [ 0 ] = = CHAR_NULL ) & & ( Key . UnicodeChar ! = CHAR_BACKSPACE ) ) {
StrnCpy ( StringBuffer , & Key . UnicodeChar , 1 ) ;
StrnCpy ( TempString , & Key . UnicodeChar , 1 ) ;
} else if ( ( GetStringWidth ( StringBuffer ) < MaximumStringSize ) & & ( Key . UnicodeChar ! = CHAR_BACKSPACE ) ) {
KeyPad [ 0 ] = Key . UnicodeChar ;
KeyPad [ 1 ] = CHAR_NULL ;
StrCat ( StringBuffer , KeyPad ) ;
StrCat ( TempString , KeyPad ) ;
}
//
// If the width of the input string is now larger than the screen, we nee to
// adjust the index to start printing portions of the string
//
SetUnicodeMem ( BufferedString , LargestString , L ' ' ) ;
PrintStringAt ( Start + 1 , Top + InputOffset , BufferedString ) ;
if ( ( GetStringWidth ( StringBuffer ) / 2 ) > ( DimensionsWidth - 2 ) ) {
Index = ( GetStringWidth ( StringBuffer ) / 2 ) - DimensionsWidth + 2 ;
} else {
Index = 0 ;
}
for ( Count = 0 ; Index + 1 < GetStringWidth ( StringBuffer ) / 2 ; Index + + , Count + + ) {
BufferedString [ Count ] = StringBuffer [ Index ] ;
}
PrintStringAt ( Start + 1 , Top + InputOffset , BufferedString ) ;
break ;
}
} while ( ! SelectionComplete ) ;
}
gST - > ConOut - > SetAttribute ( gST - > ConOut , CurrentAttribute ) ;
gST - > ConOut - > EnableCursor ( gST - > ConOut , TRUE ) ;
return EFI_SUCCESS ;
}
/**
Draw a pop up windows based on the dimension , number of lines and
strings specified .
@ param RequestedWidth The width of the pop - up .
@ param NumberOfLines The number of lines .
2008-07-16 05:04:53 +02:00
@ param Marker The variable argument list for the list of string to be printed .
2008-07-15 10:35:54 +02:00
* */
VOID
CreateSharedPopUp (
IN UINTN RequestedWidth ,
IN UINTN NumberOfLines ,
2008-07-16 05:04:53 +02:00
IN VA_LIST Marker
2008-07-15 10:35:54 +02:00
)
{
UINTN Index ;
UINTN Count ;
CHAR16 Character ;
UINTN Start ;
UINTN End ;
UINTN Top ;
UINTN Bottom ;
CHAR16 * String ;
UINTN DimensionsWidth ;
UINTN DimensionsHeight ;
DimensionsWidth = gScreenDimensions . RightColumn - gScreenDimensions . LeftColumn ;
DimensionsHeight = gScreenDimensions . BottomRow - gScreenDimensions . TopRow ;
gST - > ConOut - > SetAttribute ( gST - > ConOut , POPUP_TEXT | POPUP_BACKGROUND ) ;
if ( ( RequestedWidth + 2 ) > DimensionsWidth ) {
RequestedWidth = DimensionsWidth - 2 ;
}
//
// Subtract the PopUp width from total Columns, allow for one space extra on
// each end plus a border.
//
Start = ( DimensionsWidth - RequestedWidth - 2 ) / 2 + gScreenDimensions . LeftColumn + 1 ;
End = Start + RequestedWidth + 1 ;
Top = ( ( DimensionsHeight - NumberOfLines - 2 ) / 2 ) + gScreenDimensions . TopRow - 1 ;
Bottom = Top + NumberOfLines + 2 ;
Character = BOXDRAW_DOWN_RIGHT ;
PrintCharAt ( Start , Top , Character ) ;
Character = BOXDRAW_HORIZONTAL ;
for ( Index = Start ; Index + 2 < End ; Index + + ) {
PrintChar ( Character ) ;
}
Character = BOXDRAW_DOWN_LEFT ;
PrintChar ( Character ) ;
Character = BOXDRAW_VERTICAL ;
2008-07-16 05:04:53 +02:00
Count = 0 ;
for ( Index = Top ; Index + 2 < Bottom ; Index + + , Count + + ) {
String = VA_ARG ( Marker , CHAR16 * ) ;
2008-07-15 10:35:54 +02:00
//
// This will clear the background of the line - we never know who might have been
// here before us. This differs from the next clear in that it used the non-reverse
// video for normal printing.
//
if ( GetStringWidth ( String ) / 2 > 1 ) {
ClearLines ( Start , End , Index + 1 , Index + 1 , POPUP_TEXT | POPUP_BACKGROUND ) ;
}
//
// Passing in a space results in the assumption that this is where typing will occur
//
if ( String [ 0 ] = = L ' ' ) {
ClearLines ( Start + 1 , End - 1 , Index + 1 , Index + 1 , POPUP_INVERSE_TEXT | POPUP_INVERSE_BACKGROUND ) ;
}
//
// Passing in a NULL results in a blank space
//
if ( String [ 0 ] = = CHAR_NULL ) {
ClearLines ( Start , End , Index + 1 , Index + 1 , POPUP_TEXT | POPUP_BACKGROUND ) ;
}
PrintStringAt (
( ( DimensionsWidth - GetStringWidth ( String ) / 2 ) / 2 ) + gScreenDimensions . LeftColumn + 1 ,
Index + 1 ,
String
) ;
gST - > ConOut - > SetAttribute ( gST - > ConOut , POPUP_TEXT | POPUP_BACKGROUND ) ;
PrintCharAt ( Start , Index + 1 , Character ) ;
PrintCharAt ( End - 1 , Index + 1 , Character ) ;
}
Character = BOXDRAW_UP_RIGHT ;
PrintCharAt ( Start , Bottom - 1 , Character ) ;
Character = BOXDRAW_HORIZONTAL ;
for ( Index = Start ; Index + 2 < End ; Index + + ) {
PrintChar ( Character ) ;
}
Character = BOXDRAW_UP_LEFT ;
PrintChar ( Character ) ;
}
/**
Draw a pop up windows based on the dimension , number of lines and
strings specified .
@ param RequestedWidth The width of the pop - up .
@ param NumberOfLines The number of lines .
@ param . . . A series of text strings that displayed in the pop - up .
* */
VOID
2010-04-29 03:07:07 +02:00
EFIAPI
2009-07-07 08:28:42 +02:00
CreateMultiStringPopUp (
2008-07-15 10:35:54 +02:00
IN UINTN RequestedWidth ,
IN UINTN NumberOfLines ,
. . .
)
{
2008-07-16 05:04:53 +02:00
VA_LIST Marker ;
VA_START ( Marker , NumberOfLines ) ;
2009-10-26 04:03:12 +01:00
2008-07-16 05:04:53 +02:00
CreateSharedPopUp ( RequestedWidth , NumberOfLines , Marker ) ;
VA_END ( Marker ) ;
2008-07-15 10:35:54 +02:00
}
/**
Update status bar on the bottom of menu .
2011-05-31 02:59:15 +02:00
@ param Selection Current Selction info .
2008-07-15 10:35:54 +02:00
@ param MessageType The type of message to be shown .
@ param Flags The flags in Question header .
@ param State Set or clear .
* */
VOID
UpdateStatusBar (
2011-05-31 02:59:15 +02:00
IN UI_MENU_SELECTION * Selection ,
2008-07-15 10:35:54 +02:00
IN UINTN MessageType ,
IN UINT8 Flags ,
IN BOOLEAN State
)
{
UINTN Index ;
CHAR16 * NvUpdateMessage ;
CHAR16 * InputErrorMessage ;
2011-09-13 11:32:54 +02:00
LIST_ENTRY * Link ;
FORM_BROWSER_FORMSET * LocalFormSet ;
FORM_BROWSER_STATEMENT * Question ;
2008-07-15 10:35:54 +02:00
NvUpdateMessage = GetToken ( STRING_TOKEN ( NV_UPDATE_MESSAGE ) , gHiiHandle ) ;
InputErrorMessage = GetToken ( STRING_TOKEN ( INPUT_ERROR_MESSAGE ) , gHiiHandle ) ;
switch ( MessageType ) {
case INPUT_ERROR :
if ( State ) {
gST - > ConOut - > SetAttribute ( gST - > ConOut , ERROR_TEXT ) ;
PrintStringAt (
gScreenDimensions . LeftColumn + gPromptBlockWidth ,
gScreenDimensions . BottomRow - 1 ,
InputErrorMessage
) ;
2008-10-31 03:04:40 +01:00
mInputError = TRUE ;
2008-07-15 10:35:54 +02:00
} else {
2011-02-12 02:51:25 +01:00
gST - > ConOut - > SetAttribute ( gST - > ConOut , PcdGet8 ( PcdBrowserFieldTextHighlightColor ) ) ;
2008-07-15 10:35:54 +02:00
for ( Index = 0 ; Index < ( GetStringWidth ( InputErrorMessage ) - 2 ) / 2 ; Index + + ) {
PrintAt ( gScreenDimensions . LeftColumn + gPromptBlockWidth + Index , gScreenDimensions . BottomRow - 1 , L " " ) ;
}
2008-10-31 03:04:40 +01:00
mInputError = FALSE ;
2008-07-15 10:35:54 +02:00
}
break ;
case NV_UPDATE_REQUIRED :
2011-09-13 11:32:54 +02:00
//
// Global setting support. Show configuration change on every form.
//
if ( State ) {
gResetRequired = ( BOOLEAN ) ( gResetRequired | ( ( Flags & EFI_IFR_FLAG_RESET_REQUIRED ) = = EFI_IFR_FLAG_RESET_REQUIRED ) ) ;
if ( Selection ! = NULL & & Selection - > Statement ! = NULL ) {
Question = Selection - > Statement ;
if ( Question - > Storage ! = NULL | | Question - > Operand = = EFI_IFR_DATE_OP | | Question - > Operand = = EFI_IFR_TIME_OP ) {
//
// Update only for Question value that need to be saved into Storage.
//
Selection - > Form - > NvUpdateRequired = TRUE ;
}
}
if ( Selection = = NULL | | IsNvUpdateRequired ( Selection - > FormSet ) ) {
2008-07-15 10:35:54 +02:00
gST - > ConOut - > SetAttribute ( gST - > ConOut , INFO_TEXT ) ;
PrintStringAt (
gScreenDimensions . LeftColumn + gPromptBlockWidth + gOptionBlockWidth ,
gScreenDimensions . BottomRow - 1 ,
NvUpdateMessage
) ;
2011-09-13 11:32:54 +02:00
}
} else {
gST - > ConOut - > SetAttribute ( gST - > ConOut , PcdGet8 ( PcdBrowserFieldTextHighlightColor ) ) ;
for ( Index = 0 ; Index < ( GetStringWidth ( NvUpdateMessage ) - 2 ) / 2 ; Index + + ) {
PrintAt (
( gScreenDimensions . LeftColumn + gPromptBlockWidth + gOptionBlockWidth + Index ) ,
gScreenDimensions . BottomRow - 1 ,
L " "
) ;
2008-07-15 10:35:54 +02:00
}
}
break ;
case REFRESH_STATUS_BAR :
2008-10-31 03:04:40 +01:00
if ( mInputError ) {
2011-05-31 02:59:15 +02:00
UpdateStatusBar ( Selection , INPUT_ERROR , Flags , TRUE ) ;
2008-07-15 10:35:54 +02:00
}
2011-09-13 11:32:54 +02:00
switch ( gBrowserSettingScope ) {
case SystemLevel :
//
// Check the maintain list to see whether there is any change.
//
Link = GetFirstNode ( & gBrowserFormSetList ) ;
while ( ! IsNull ( & gBrowserFormSetList , Link ) ) {
LocalFormSet = FORM_BROWSER_FORMSET_FROM_LINK ( Link ) ;
if ( IsNvUpdateRequired ( LocalFormSet ) ) {
UpdateStatusBar ( NULL , NV_UPDATE_REQUIRED , Flags , TRUE ) ;
break ;
}
Link = GetNextNode ( & gBrowserFormSetList , Link ) ;
}
break ;
case FormSetLevel :
case FormLevel :
UpdateStatusBar ( Selection , NV_UPDATE_REQUIRED , Flags , TRUE ) ;
default :
break ;
2008-07-15 10:35:54 +02:00
}
2011-09-13 11:32:54 +02:00
2008-07-15 10:35:54 +02:00
break ;
default :
break ;
}
2008-11-21 07:59:58 +01:00
FreePool ( InputErrorMessage ) ;
FreePool ( NvUpdateMessage ) ;
2008-07-15 10:35:54 +02:00
return ;
}
/**
Get the supported width for a particular op - code
@ param Statement The FORM_BROWSER_STATEMENT structure passed in .
@ param Handle The handle in the HII database being used
@ return Returns the number of CHAR16 characters that is support .
* */
UINT16
GetWidth (
IN FORM_BROWSER_STATEMENT * Statement ,
IN EFI_HII_HANDLE Handle
)
{
CHAR16 * String ;
UINTN Size ;
UINT16 Width ;
Size = 0 ;
//
// See if the second text parameter is really NULL
//
if ( ( Statement - > Operand = = EFI_IFR_TEXT_OP ) & & ( Statement - > TextTwo ! = 0 ) ) {
String = GetToken ( Statement - > TextTwo , Handle ) ;
Size = StrLen ( String ) ;
2008-11-21 07:59:58 +01:00
FreePool ( String ) ;
2008-07-15 10:35:54 +02:00
}
if ( ( Statement - > Operand = = EFI_IFR_SUBTITLE_OP ) | |
( Statement - > Operand = = EFI_IFR_REF_OP ) | |
( Statement - > Operand = = EFI_IFR_PASSWORD_OP ) | |
( Statement - > Operand = = EFI_IFR_ACTION_OP ) | |
( Statement - > Operand = = EFI_IFR_RESET_BUTTON_OP ) | |
//
// Allow a wide display if text op-code and no secondary text op-code
//
( ( Statement - > Operand = = EFI_IFR_TEXT_OP ) & & ( Size = = 0 ) )
) {
Width = ( UINT16 ) ( gPromptBlockWidth + gOptionBlockWidth ) ;
} else {
Width = ( UINT16 ) gPromptBlockWidth ;
}
if ( Statement - > InSubtitle ) {
Width - = SUBTITLE_INDENT ;
}
2010-10-26 05:28:44 +02:00
return ( UINT16 ) ( Width - LEFT_SKIPPED_COLUMNS ) ;
2008-07-15 10:35:54 +02:00
}
/**
Will copy LineWidth amount of a string in the OutputString buffer and return the
number of CHAR16 characters that were copied into the OutputString buffer .
2012-05-29 13:07:05 +02:00
The output string format is :
Glyph Info + String info + ' \0 ' .
2012-04-23 12:26:04 +02:00
In the code , it deals \ r , \ n , \ r \ n same as \ n \ r , also it not process the \ r or \ g .
2008-07-15 10:35:54 +02:00
@ param InputString String description for this option .
@ param LineWidth Width of the desired string to extract in CHAR16
characters
2012-05-29 13:07:05 +02:00
@ param GlyphWidth The glyph width of the begin of the char in the string .
2008-07-15 10:35:54 +02:00
@ param Index Where in InputString to start the copy process
@ param OutputString Buffer to copy the string into
2012-05-29 13:07:05 +02:00
@ return Returns the number of CHAR16 characters that were copied into the OutputString
buffer , include extra glyph info and ' \0 ' info .
2008-07-15 10:35:54 +02:00
* */
UINT16
GetLineByWidth (
IN CHAR16 * InputString ,
IN UINT16 LineWidth ,
2012-05-29 13:07:05 +02:00
IN OUT UINT16 * GlyphWidth ,
2008-07-15 10:35:54 +02:00
IN OUT UINTN * Index ,
OUT CHAR16 * * OutputString
)
{
2012-05-29 13:07:05 +02:00
UINT16 StrOffset ;
UINT16 GlyphOffset ;
UINT16 OriginalGlyphWidth ;
BOOLEAN ReturnFlag ;
UINT16 LastSpaceOffset ;
UINT16 LastGlyphWidth ;
if ( InputString = = NULL | | Index = = NULL | | OutputString = = NULL ) {
return 0 ;
}
2008-07-15 10:35:54 +02:00
2012-05-29 13:07:05 +02:00
if ( LineWidth = = 0 | | * GlyphWidth = = 0 ) {
return 0 ;
2008-07-15 10:35:54 +02:00
}
2012-05-29 13:07:05 +02:00
//
// Save original glyph width.
//
OriginalGlyphWidth = * GlyphWidth ;
LastGlyphWidth = OriginalGlyphWidth ;
ReturnFlag = FALSE ;
LastSpaceOffset = 0 ;
2008-07-15 10:35:54 +02:00
2012-05-29 13:07:05 +02:00
//
// NARROW_CHAR can not be printed in screen, so if a line only contain the two CHARs: 'NARROW_CHAR + CHAR_CARRIAGE_RETURN' , it is a empty line in Screen.
// To avoid displaying this empty line in screen, just skip the two CHARs here.
//
if ( ( InputString [ * Index ] = = NARROW_CHAR ) & & ( InputString [ * Index + 1 ] = = CHAR_CARRIAGE_RETURN ) ) {
* Index = * Index + 2 ;
}
2008-07-15 10:35:54 +02:00
//
2012-05-29 13:07:05 +02:00
// Fast-forward the string and see if there is a carriage-return in the string
2008-07-15 10:35:54 +02:00
//
2012-05-29 13:07:05 +02:00
for ( StrOffset = 0 , GlyphOffset = 0 ; GlyphOffset < = LineWidth ; StrOffset + + ) {
switch ( InputString [ * Index + StrOffset ] ) {
case NARROW_CHAR :
* GlyphWidth = 1 ;
break ;
2008-07-15 10:35:54 +02:00
2012-05-29 13:07:05 +02:00
case WIDE_CHAR :
* GlyphWidth = 2 ;
break ;
2008-07-15 10:35:54 +02:00
2012-05-29 13:07:05 +02:00
case CHAR_CARRIAGE_RETURN :
case CHAR_LINEFEED :
case CHAR_NULL :
ReturnFlag = TRUE ;
break ;
default :
GlyphOffset = GlyphOffset + * GlyphWidth ;
2008-07-15 10:35:54 +02:00
2012-05-29 13:07:05 +02:00
//
// Record the last space info in this line. Will be used in rewind.
//
if ( ( InputString [ * Index + StrOffset ] = = CHAR_SPACE ) & & ( GlyphOffset < = LineWidth ) ) {
LastSpaceOffset = StrOffset ;
LastGlyphWidth = * GlyphWidth ;
}
break ;
}
if ( ReturnFlag ) {
break ;
}
}
//
// Rewind the string from the maximum size until we see a space to break the line
//
if ( GlyphOffset > LineWidth ) {
2008-07-15 10:35:54 +02:00
//
2012-05-29 13:07:05 +02:00
// Rewind the string to last space char in this line.
2008-07-15 10:35:54 +02:00
//
2012-05-29 13:07:05 +02:00
if ( LastSpaceOffset ! = 0 ) {
StrOffset = LastSpaceOffset ;
* GlyphWidth = LastGlyphWidth ;
} else {
2008-07-15 10:35:54 +02:00
//
2012-05-29 13:07:05 +02:00
// Roll back to last char in the line width.
2008-07-15 10:35:54 +02:00
//
2012-05-29 13:07:05 +02:00
StrOffset - - ;
2008-07-15 10:35:54 +02:00
}
2012-05-29 13:07:05 +02:00
}
//
// The CHAR_NULL has process last time, this time just return 0 to stand for the end.
//
if ( StrOffset = = 0 & & ( InputString [ * Index + StrOffset ] = = CHAR_NULL ) ) {
return 0 ;
}
//
// Need extra glyph info and '\0' info, so +2.
//
* OutputString = AllocateZeroPool ( ( ( UINTN ) ( StrOffset + 2 ) * sizeof ( CHAR16 ) ) ) ;
if ( * OutputString = = NULL ) {
return 0 ;
}
//
// Save the glyph info at the begin of the string, will used by Print function.
//
if ( OriginalGlyphWidth = = 1 ) {
* ( * OutputString ) = NARROW_CHAR ;
} else {
* ( * OutputString ) = WIDE_CHAR ;
}
2008-07-15 10:35:54 +02:00
2012-05-29 13:07:05 +02:00
CopyMem ( ( * OutputString ) + 1 , & InputString [ * Index ] , StrOffset * sizeof ( CHAR16 ) ) ;
2008-07-15 10:35:54 +02:00
2012-05-29 13:07:05 +02:00
if ( InputString [ * Index + StrOffset ] = = CHAR_SPACE ) {
2008-07-15 10:35:54 +02:00
//
2012-05-29 13:07:05 +02:00
// Skip the space info at the begin of next line.
//
* Index = ( UINT16 ) ( * Index + StrOffset + 1 ) ;
} else if ( ( InputString [ * Index + StrOffset ] = = CHAR_LINEFEED ) ) {
2008-07-15 10:35:54 +02:00
//
2012-05-29 13:07:05 +02:00
// Skip the /n or /n/r info.
//
if ( InputString [ * Index + StrOffset + 1 ] = = CHAR_CARRIAGE_RETURN ) {
* Index = ( UINT16 ) ( * Index + StrOffset + 2 ) ;
} else {
* Index = ( UINT16 ) ( * Index + StrOffset + 1 ) ;
}
} else if ( ( InputString [ * Index + StrOffset ] = = CHAR_CARRIAGE_RETURN ) ) {
//
// Skip the /r or /r/n info.
//
if ( InputString [ * Index + StrOffset + 1 ] = = CHAR_LINEFEED ) {
* Index = ( UINT16 ) ( * Index + StrOffset + 2 ) ;
} else {
* Index = ( UINT16 ) ( * Index + StrOffset + 1 ) ;
}
2008-07-15 10:35:54 +02:00
} else {
2012-05-29 13:07:05 +02:00
* Index = ( UINT16 ) ( * Index + StrOffset ) ;
2008-07-15 10:35:54 +02:00
}
2012-05-29 13:07:05 +02:00
//
// Include extra glyph info and '\0' info, so +2.
//
return StrOffset + 2 ;
2008-07-15 10:35:54 +02:00
}
/**
Update display lines for a Menu Option .
@ param Selection The user ' s selection .
@ param MenuOption The MenuOption to be checked .
* */
VOID
UpdateOptionSkipLines (
IN UI_MENU_SELECTION * Selection ,
2011-10-25 11:00:23 +02:00
IN UI_MENU_OPTION * MenuOption
2008-07-15 10:35:54 +02:00
)
{
UINTN Index ;
UINT16 Width ;
UINTN Row ;
UINTN OriginalRow ;
CHAR16 * OutputString ;
CHAR16 * OptionString ;
2012-05-29 13:07:05 +02:00
UINT16 GlyphWidth ;
2008-07-15 10:35:54 +02:00
Row = 0 ;
2011-10-25 11:00:23 +02:00
OptionString = NULL ;
2008-07-15 10:35:54 +02:00
ProcessOptions ( Selection , MenuOption , FALSE , & OptionString ) ;
if ( OptionString ! = NULL ) {
Width = ( UINT16 ) gOptionBlockWidth ;
OriginalRow = Row ;
2012-05-29 13:07:05 +02:00
GlyphWidth = 1 ;
2008-07-15 10:35:54 +02:00
2012-05-29 13:07:05 +02:00
for ( Index = 0 ; GetLineByWidth ( OptionString , Width , & GlyphWidth , & Index , & OutputString ) ! = 0x0000 ; ) {
2008-07-15 10:35:54 +02:00
//
// If there is more string to process print on the next row and increment the Skip value
//
2009-02-09 06:29:17 +01:00
if ( StrLen ( & OptionString [ Index ] ) ! = 0 ) {
2011-10-25 11:00:23 +02:00
Row + + ;
//
// Since the Number of lines for this menu entry may or may not be reflected accurately
// since the prompt might be 1 lines and option might be many, and vice versa, we need to do
// some testing to ensure we are keeping this in-sync.
//
// If the difference in rows is greater than or equal to the skip value, increase the skip value
//
if ( ( Row - OriginalRow ) > = MenuOption - > Skip ) {
MenuOption - > Skip + + ;
2008-07-15 10:35:54 +02:00
}
}
2008-11-21 07:59:58 +01:00
FreePool ( OutputString ) ;
2008-07-15 10:35:54 +02:00
}
Row = OriginalRow ;
}
2011-10-25 11:00:23 +02:00
if ( OptionString ! = NULL ) {
FreePool ( OptionString ) ;
}
2008-07-15 10:35:54 +02:00
}
/**
Check whether this Menu Option could be highlighted .
This is an internal function .
@ param MenuOption The MenuOption to be checked .
@ retval TRUE This Menu Option is selectable .
@ retval FALSE This Menu Option could not be selected .
* */
BOOLEAN
IsSelectable (
UI_MENU_OPTION * MenuOption
)
{
if ( ( MenuOption - > ThisTag - > Operand = = EFI_IFR_SUBTITLE_OP ) | |
MenuOption - > GrayOut | | MenuOption - > ReadOnly ) {
return FALSE ;
} else {
return TRUE ;
}
}
/**
Determine if the menu is the last menu that can be selected .
This is an internal function .
2009-10-26 04:03:12 +01:00
2008-07-15 10:35:54 +02:00
@ param Direction The scroll direction . False is down . True is up .
@ param CurrentPos The current focus .
@ return FALSE - - the menu isn ' t the last menu that can be selected .
@ return TRUE - - the menu is the last menu that can be selected .
* */
BOOLEAN
ValueIsScroll (
IN BOOLEAN Direction ,
IN LIST_ENTRY * CurrentPos
)
{
LIST_ENTRY * Temp ;
Temp = Direction ? CurrentPos - > BackLink : CurrentPos - > ForwardLink ;
2009-10-26 04:06:06 +01:00
if ( Temp = = & gMenuOption ) {
2008-07-15 10:35:54 +02:00
return TRUE ;
}
2011-03-08 07:55:15 +01:00
return FALSE ;
2008-07-15 10:35:54 +02:00
}
/**
Move to next selectable statement .
2009-10-26 04:03:12 +01:00
2008-07-15 10:35:54 +02:00
This is an internal function .
2009-10-26 04:03:12 +01:00
2011-10-25 11:00:23 +02:00
@ param Selection Menu selection .
2008-07-15 10:35:54 +02:00
@ param GoUp The navigation direction . TRUE : up , FALSE : down .
@ param CurrentPosition Current position .
2011-03-08 07:55:15 +01:00
@ param GapToTop Gap position to top or bottom .
2008-07-15 10:35:54 +02:00
@ return The row distance from current MenuOption to next selectable MenuOption .
* */
INTN
MoveToNextStatement (
2011-10-25 11:00:23 +02:00
IN UI_MENU_SELECTION * Selection ,
2008-07-15 10:35:54 +02:00
IN BOOLEAN GoUp ,
2011-03-08 07:55:15 +01:00
IN OUT LIST_ENTRY * * CurrentPosition ,
IN UINTN GapToTop
2008-07-15 10:35:54 +02:00
)
{
INTN Distance ;
LIST_ENTRY * Pos ;
UI_MENU_OPTION * NextMenuOption ;
2011-03-08 07:55:15 +01:00
UI_MENU_OPTION * PreMenuOption ;
2008-07-15 10:35:54 +02:00
2011-03-08 07:55:15 +01:00
Distance = 0 ;
Pos = * CurrentPosition ;
PreMenuOption = MENU_OPTION_FROM_LINK ( Pos ) ;
2008-07-15 10:35:54 +02:00
while ( TRUE ) {
NextMenuOption = MENU_OPTION_FROM_LINK ( Pos ) ;
2011-10-25 11:00:23 +02:00
if ( NextMenuOption - > Row = = 0 ) {
UpdateOptionSkipLines ( Selection , NextMenuOption ) ;
}
2011-03-08 07:55:15 +01:00
if ( GoUp & & ( PreMenuOption ! = NextMenuOption ) ) {
//
// Current Position doesn't need to be caculated when go up.
// Caculate distanct at first when go up
//
if ( ( UINTN ) Distance + NextMenuOption - > Skip > GapToTop ) {
NextMenuOption = PreMenuOption ;
break ;
}
Distance + = NextMenuOption - > Skip ;
}
2008-07-15 10:35:54 +02:00
if ( IsSelectable ( NextMenuOption ) ) {
break ;
}
2009-10-26 04:06:06 +01:00
if ( ( GoUp ? Pos - > BackLink : Pos - > ForwardLink ) = = & gMenuOption ) {
2011-03-08 07:55:15 +01:00
//
// Arrive at top.
//
Distance = - 1 ;
2008-07-15 10:35:54 +02:00
break ;
}
2011-03-08 07:55:15 +01:00
if ( ! GoUp ) {
//
// Caculate distanct at later when go down
//
if ( ( UINTN ) Distance + NextMenuOption - > Skip > GapToTop ) {
NextMenuOption = PreMenuOption ;
2008-07-15 10:35:54 +02:00
break ;
}
2011-03-08 07:55:15 +01:00
Distance + = NextMenuOption - > Skip ;
2008-07-15 10:35:54 +02:00
}
2011-03-08 07:55:15 +01:00
PreMenuOption = NextMenuOption ;
Pos = ( GoUp ? Pos - > BackLink : Pos - > ForwardLink ) ;
2008-07-15 10:35:54 +02:00
}
* CurrentPosition = & NextMenuOption - > Link ;
return Distance ;
}
/**
Adjust Data and Time position accordingly .
Data format : [ 01 / 02 / 2004 ] [ 11 : 22 : 33 ]
Line number : 0 0 1 0 0 1
2009-10-26 04:03:12 +01:00
2008-07-15 10:35:54 +02:00
This is an internal function .
@ param DirectionUp the up or down direction . False is down . True is
up .
@ param CurrentPosition Current position . On return : Point to the last
Option ( Year or Second ) if up ; Point to the first
Option ( Month or Hour ) if down .
@ return Return line number to pad . It is possible that we stand on a zero - advance
@ return data or time opcode , so pad one line when we judge if we are going to scroll outside .
* */
UINTN
AdjustDateAndTimePosition (
IN BOOLEAN DirectionUp ,
IN OUT LIST_ENTRY * * CurrentPosition
)
{
UINTN Count ;
LIST_ENTRY * NewPosition ;
UI_MENU_OPTION * MenuOption ;
UINTN PadLineNumber ;
PadLineNumber = 0 ;
NewPosition = * CurrentPosition ;
MenuOption = MENU_OPTION_FROM_LINK ( NewPosition ) ;
if ( ( MenuOption - > ThisTag - > Operand = = EFI_IFR_DATE_OP ) | |
( MenuOption - > ThisTag - > Operand = = EFI_IFR_TIME_OP ) ) {
//
// Calculate the distance from current position to the last Date/Time MenuOption
//
Count = 0 ;
while ( MenuOption - > Skip = = 0 ) {
Count + + ;
NewPosition = NewPosition - > ForwardLink ;
MenuOption = MENU_OPTION_FROM_LINK ( NewPosition ) ;
PadLineNumber = 1 ;
}
NewPosition = * CurrentPosition ;
if ( DirectionUp ) {
//
// Since the behavior of hitting the up arrow on a Date/Time MenuOption is intended
// to be one that back to the previous set of MenuOptions, we need to advance to the first
// Date/Time MenuOption and leave the remaining logic in CfUiUp intact so the appropriate
// checking can be done.
//
while ( Count + + < 2 ) {
NewPosition = NewPosition - > BackLink ;
}
} else {
//
// Since the behavior of hitting the down arrow on a Date/Time MenuOption is intended
// to be one that progresses to the next set of MenuOptions, we need to advance to the last
// Date/Time MenuOption and leave the remaining logic in CfUiDown intact so the appropriate
// checking can be done.
//
while ( Count - - > 0 ) {
NewPosition = NewPosition - > ForwardLink ;
}
}
* CurrentPosition = NewPosition ;
}
return PadLineNumber ;
}
2009-04-14 12:47:19 +02:00
/**
Find HII Handle in the HII database associated with given Device Path .
If DevicePath is NULL , then ASSERT .
@ param DevicePath Device Path associated with the HII package list
handle .
@ retval Handle HII package list Handle associated with the Device
Path .
@ retval NULL Hii Package list handle is not found .
* */
EFI_HII_HANDLE
EFIAPI
DevicePathToHiiHandle (
IN EFI_DEVICE_PATH_PROTOCOL * DevicePath
)
{
EFI_STATUS Status ;
EFI_DEVICE_PATH_PROTOCOL * TmpDevicePath ;
UINTN BufferSize ;
UINTN HandleCount ;
UINTN Index ;
EFI_HANDLE Handle ;
EFI_HANDLE DriverHandle ;
EFI_HII_HANDLE * HiiHandles ;
EFI_HII_HANDLE HiiHandle ;
ASSERT ( DevicePath ! = NULL ) ;
TmpDevicePath = DevicePath ;
//
// Locate Device Path Protocol handle buffer
//
Status = gBS - > LocateDevicePath (
& gEfiDevicePathProtocolGuid ,
& TmpDevicePath ,
& DriverHandle
) ;
if ( EFI_ERROR ( Status ) | | ! IsDevicePathEnd ( TmpDevicePath ) ) {
return NULL ;
}
//
// Retrieve all HII Handles from HII database
//
BufferSize = 0x1000 ;
HiiHandles = AllocatePool ( BufferSize ) ;
ASSERT ( HiiHandles ! = NULL ) ;
Status = mHiiDatabase - > ListPackageLists (
mHiiDatabase ,
EFI_HII_PACKAGE_TYPE_ALL ,
NULL ,
& BufferSize ,
HiiHandles
) ;
if ( Status = = EFI_BUFFER_TOO_SMALL ) {
FreePool ( HiiHandles ) ;
HiiHandles = AllocatePool ( BufferSize ) ;
ASSERT ( HiiHandles ! = NULL ) ;
Status = mHiiDatabase - > ListPackageLists (
mHiiDatabase ,
EFI_HII_PACKAGE_TYPE_ALL ,
NULL ,
& BufferSize ,
HiiHandles
) ;
}
if ( EFI_ERROR ( Status ) ) {
FreePool ( HiiHandles ) ;
return NULL ;
}
//
// Search Hii Handle by Driver Handle
//
HiiHandle = NULL ;
HandleCount = BufferSize / sizeof ( EFI_HII_HANDLE ) ;
for ( Index = 0 ; Index < HandleCount ; Index + + ) {
Status = mHiiDatabase - > GetPackageListHandle (
mHiiDatabase ,
HiiHandles [ Index ] ,
& Handle
) ;
if ( ! EFI_ERROR ( Status ) & & ( Handle = = DriverHandle ) ) {
HiiHandle = HiiHandles [ Index ] ;
break ;
}
}
FreePool ( HiiHandles ) ;
return HiiHandle ;
}
2008-07-15 10:35:54 +02:00
2011-10-31 04:33:55 +01:00
/**
Find HII Handle in the HII database associated with given form set guid .
If FormSetGuid is NULL , then ASSERT .
@ param ComparingGuid FormSet Guid associated with the HII package list
handle .
@ retval Handle HII package list Handle associated with the Device
Path .
@ retval NULL Hii Package list handle is not found .
* */
EFI_HII_HANDLE
FormSetGuidToHiiHandle (
EFI_GUID * ComparingGuid
)
{
EFI_HII_HANDLE * HiiHandles ;
UINTN Index ;
EFI_HII_PACKAGE_LIST_HEADER * HiiPackageList ;
UINTN BufferSize ;
UINT32 Offset ;
UINT32 Offset2 ;
UINT32 PackageListLength ;
EFI_HII_PACKAGE_HEADER PackageHeader ;
UINT8 * Package ;
UINT8 * OpCodeData ;
EFI_STATUS Status ;
EFI_HII_HANDLE HiiHandle ;
ASSERT ( ComparingGuid ! = NULL ) ;
HiiHandle = NULL ;
//
// Get all the Hii handles
//
HiiHandles = HiiGetHiiHandles ( NULL ) ;
ASSERT ( HiiHandles ! = NULL ) ;
//
// Search for formset of each class type
//
for ( Index = 0 ; HiiHandles [ Index ] ! = NULL ; Index + + ) {
BufferSize = 0 ;
HiiPackageList = NULL ;
Status = mHiiDatabase - > ExportPackageLists ( mHiiDatabase , HiiHandles [ Index ] , & BufferSize , HiiPackageList ) ;
if ( Status = = EFI_BUFFER_TOO_SMALL ) {
HiiPackageList = AllocatePool ( BufferSize ) ;
ASSERT ( HiiPackageList ! = NULL ) ;
Status = mHiiDatabase - > ExportPackageLists ( mHiiDatabase , HiiHandles [ Index ] , & BufferSize , HiiPackageList ) ;
}
if ( EFI_ERROR ( Status ) | | HiiPackageList = = NULL ) {
return NULL ;
}
//
// Get Form package from this HII package List
//
Offset = sizeof ( EFI_HII_PACKAGE_LIST_HEADER ) ;
Offset2 = 0 ;
CopyMem ( & PackageListLength , & HiiPackageList - > PackageLength , sizeof ( UINT32 ) ) ;
while ( Offset < PackageListLength ) {
Package = ( ( UINT8 * ) HiiPackageList ) + Offset ;
CopyMem ( & PackageHeader , Package , sizeof ( EFI_HII_PACKAGE_HEADER ) ) ;
if ( PackageHeader . Type = = EFI_HII_PACKAGE_FORMS ) {
//
// Search FormSet in this Form Package
//
Offset2 = sizeof ( EFI_HII_PACKAGE_HEADER ) ;
while ( Offset2 < PackageHeader . Length ) {
OpCodeData = Package + Offset2 ;
if ( ( ( EFI_IFR_OP_HEADER * ) OpCodeData ) - > OpCode = = EFI_IFR_FORM_SET_OP ) {
//
// Try to compare against formset GUID
//
if ( CompareGuid ( ComparingGuid , ( EFI_GUID * ) ( OpCodeData + sizeof ( EFI_IFR_OP_HEADER ) ) ) ) {
HiiHandle = HiiHandles [ Index ] ;
break ;
}
}
Offset2 + = ( ( EFI_IFR_OP_HEADER * ) OpCodeData ) - > Length ;
}
}
if ( HiiHandle ! = NULL ) {
break ;
}
Offset + = PackageHeader . Length ;
}
FreePool ( HiiPackageList ) ;
if ( HiiHandle ! = NULL ) {
break ;
}
}
FreePool ( HiiHandles ) ;
return HiiHandle ;
}
2011-06-28 08:41:28 +02:00
/**
Process the goto op code , update the info in the selection structure .
@ param Statement The statement belong to goto op code .
@ param Selection The selection info .
@ param Repaint Whether need to repaint the menu .
@ param NewLine Whether need to create new line .
@ retval EFI_SUCCESS The menu process successfully .
@ return Other value if the process failed .
* */
EFI_STATUS
ProcessGotoOpCode (
IN OUT FORM_BROWSER_STATEMENT * Statement ,
IN OUT UI_MENU_SELECTION * Selection ,
OUT BOOLEAN * Repaint ,
OUT BOOLEAN * NewLine
)
{
CHAR16 * StringPtr ;
2011-11-24 09:47:57 +01:00
UINTN StringLen ;
2011-06-28 08:41:28 +02:00
UINTN BufferSize ;
EFI_DEVICE_PATH_PROTOCOL * DevicePath ;
CHAR16 TemStr [ 2 ] ;
UINT8 * DevicePathBuffer ;
UINTN Index ;
UINT8 DigitUint8 ;
FORM_BROWSER_FORM * RefForm ;
EFI_INPUT_KEY Key ;
EFI_STATUS Status ;
UI_MENU_LIST * MenuList ;
2011-10-31 04:33:55 +01:00
BOOLEAN UpdateFormInfo ;
2011-06-28 08:41:28 +02:00
Status = EFI_SUCCESS ;
2011-10-31 04:33:55 +01:00
UpdateFormInfo = TRUE ;
2011-11-24 09:47:57 +01:00
StringPtr = NULL ;
StringLen = 0 ;
2011-06-28 08:41:28 +02:00
2011-11-24 09:47:57 +01:00
//
// Prepare the device path check, get the device path info first.
//
2011-06-28 08:41:28 +02:00
if ( Statement - > HiiValue . Value . ref . DevicePath ! = 0 ) {
2011-11-24 09:47:57 +01:00
StringPtr = GetToken ( Statement - > HiiValue . Value . ref . DevicePath , Selection - > FormSet - > HiiHandle ) ;
if ( StringPtr ! = NULL ) {
StringLen = StrLen ( StringPtr ) ;
}
}
//
// Check whether the device path string is a valid string.
//
if ( Statement - > HiiValue . Value . ref . DevicePath ! = 0 & & StringPtr ! = NULL & & StringLen ! = 0 ) {
2011-06-28 08:41:28 +02:00
if ( Selection - > Form - > ModalForm ) {
return Status ;
}
//
// Goto another Hii Package list
//
Selection - > Action = UI_ACTION_REFRESH_FORMSET ;
BufferSize = StrLen ( StringPtr ) / 2 ;
DevicePath = AllocatePool ( BufferSize ) ;
ASSERT ( DevicePath ! = NULL ) ;
//
// Convert from Device Path String to DevicePath Buffer in the reverse order.
//
DevicePathBuffer = ( UINT8 * ) DevicePath ;
for ( Index = 0 ; StringPtr [ Index ] ! = L ' \0 ' ; Index + + ) {
TemStr [ 0 ] = StringPtr [ Index ] ;
DigitUint8 = ( UINT8 ) StrHexToUint64 ( TemStr ) ;
if ( DigitUint8 = = 0 & & TemStr [ 0 ] ! = L ' 0 ' ) {
//
// Invalid Hex Char as the tail.
//
break ;
}
if ( ( Index & 1 ) = = 0 ) {
DevicePathBuffer [ Index / 2 ] = DigitUint8 ;
} else {
DevicePathBuffer [ Index / 2 ] = ( UINT8 ) ( ( DevicePathBuffer [ Index / 2 ] < < 4 ) + DigitUint8 ) ;
}
}
2011-11-24 09:47:57 +01:00
FreePool ( StringPtr ) ;
2011-06-28 08:41:28 +02:00
Selection - > Handle = DevicePathToHiiHandle ( DevicePath ) ;
2011-11-24 09:47:57 +01:00
FreePool ( DevicePath ) ;
2011-06-28 08:41:28 +02:00
if ( Selection - > Handle = = NULL ) {
//
// If target Hii Handle not found, exit
//
Selection - > Action = UI_ACTION_EXIT ;
Selection - > Statement = NULL ;
return Status ;
}
CopyMem ( & Selection - > FormSetGuid , & Statement - > HiiValue . Value . ref . FormSetGuid , sizeof ( EFI_GUID ) ) ;
Selection - > FormId = Statement - > HiiValue . Value . ref . FormId ;
Selection - > QuestionId = Statement - > HiiValue . Value . ref . QuestionId ;
} else if ( ! CompareGuid ( & Statement - > HiiValue . Value . ref . FormSetGuid , & gZeroGuid ) ) {
if ( Selection - > Form - > ModalForm ) {
return Status ;
}
//
// Goto another Formset, check for uncommitted data
//
Selection - > Action = UI_ACTION_REFRESH_FORMSET ;
2011-10-31 04:33:55 +01:00
Selection - > Handle = FormSetGuidToHiiHandle ( & Statement - > HiiValue . Value . ref . FormSetGuid ) ;
if ( Selection - > Handle = = NULL ) {
//
// If target Hii Handle not found, exit
//
Selection - > Action = UI_ACTION_EXIT ;
Selection - > Statement = NULL ;
return Status ;
}
2011-06-28 08:41:28 +02:00
CopyMem ( & Selection - > FormSetGuid , & Statement - > HiiValue . Value . ref . FormSetGuid , sizeof ( EFI_GUID ) ) ;
Selection - > FormId = Statement - > HiiValue . Value . ref . FormId ;
Selection - > QuestionId = Statement - > HiiValue . Value . ref . QuestionId ;
} else if ( Statement - > HiiValue . Value . ref . FormId ! = 0 ) {
//
// Check whether target From is suppressed.
//
RefForm = IdToForm ( Selection - > FormSet , Statement - > HiiValue . Value . ref . FormId ) ;
if ( ( RefForm ! = NULL ) & & ( RefForm - > SuppressExpression ! = NULL ) ) {
2012-01-31 08:17:42 +01:00
if ( EvaluateExpressionList ( RefForm - > SuppressExpression , TRUE , Selection - > FormSet , RefForm ) ! = ExpressFalse ) {
2011-06-28 08:41:28 +02:00
//
// Form is suppressed.
//
do {
CreateDialog ( 4 , TRUE , 0 , NULL , & Key , gEmptyString , gFormSuppress , gPressEnter , gEmptyString ) ;
} while ( Key . UnicodeChar ! = CHAR_CARRIAGE_RETURN ) ;
if ( Repaint ! = NULL ) {
* Repaint = TRUE ;
}
return Status ;
}
}
//
// Goto another form inside this formset,
//
Selection - > Action = UI_ACTION_REFRESH_FORM ;
Selection - > FormId = Statement - > HiiValue . Value . ref . FormId ;
Selection - > QuestionId = Statement - > HiiValue . Value . ref . QuestionId ;
} else if ( Statement - > HiiValue . Value . ref . QuestionId ! = 0 ) {
//
// Goto another Question
//
Selection - > QuestionId = Statement - > HiiValue . Value . ref . QuestionId ;
if ( ( Statement - > QuestionFlags & EFI_IFR_FLAG_CALLBACK ) ! = 0 ) {
Selection - > Action = UI_ACTION_REFRESH_FORM ;
} else {
if ( Repaint ! = NULL ) {
* Repaint = TRUE ;
}
if ( NewLine ! = NULL ) {
* NewLine = TRUE ;
}
}
2011-10-31 04:33:55 +01:00
UpdateFormInfo = FALSE ;
2011-06-28 08:41:28 +02:00
} else {
if ( ( Statement - > QuestionFlags & EFI_IFR_FLAG_CALLBACK ) ! = 0 ) {
Selection - > Action = UI_ACTION_REFRESH_FORM ;
}
2011-10-31 04:33:55 +01:00
UpdateFormInfo = FALSE ;
}
if ( UpdateFormInfo ) {
//
// Link current form so that we can always go back when someone hits the ESC
//
MenuList = UiFindMenuList ( & Selection - > FormSetGuid , Selection - > FormId ) ;
if ( MenuList = = NULL & & Selection - > CurrentMenu ! = NULL ) {
MenuList = UiAddMenuList ( Selection - > CurrentMenu , Selection - > Handle , & Selection - > FormSetGuid , Selection - > FormId ) ;
}
2011-06-28 08:41:28 +02:00
}
return Status ;
}
2008-07-15 10:35:54 +02:00
/**
Display menu and wait for user to select one menu option , then return it .
If AutoBoot is enabled , then if user doesn ' t select any option ,
after period of time , it will automatically return the first menu option .
@ param Selection Menu selection .
@ retval EFI_SUCESSS This function always return successfully for now .
* */
EFI_STATUS
UiDisplayMenu (
IN OUT UI_MENU_SELECTION * Selection
)
{
INTN SkipValue ;
INTN Difference ;
INTN OldSkipValue ;
UINTN DistanceValue ;
UINTN Row ;
UINTN Col ;
UINTN Temp ;
UINTN Temp2 ;
UINTN TopRow ;
UINTN BottomRow ;
UINTN OriginalRow ;
UINTN Index ;
UINT16 Width ;
CHAR16 * StringPtr ;
CHAR16 * OptionString ;
CHAR16 * OutputString ;
2012-04-11 07:44:40 +02:00
CHAR16 * HelpString ;
CHAR16 * HelpHeaderString ;
CHAR16 * HelpBottomString ;
2008-07-15 10:35:54 +02:00
BOOLEAN NewLine ;
BOOLEAN Repaint ;
BOOLEAN SavedValue ;
2009-10-26 04:06:06 +01:00
BOOLEAN UpArrow ;
BOOLEAN DownArrow ;
2011-03-08 07:55:15 +01:00
BOOLEAN InitializedFlag ;
2008-07-15 10:35:54 +02:00
EFI_STATUS Status ;
EFI_INPUT_KEY Key ;
LIST_ENTRY * Link ;
LIST_ENTRY * NewPos ;
LIST_ENTRY * TopOfScreen ;
LIST_ENTRY * SavedListEntry ;
UI_MENU_OPTION * MenuOption ;
UI_MENU_OPTION * NextMenuOption ;
UI_MENU_OPTION * SavedMenuOption ;
UI_MENU_OPTION * PreviousMenuOption ;
UI_CONTROL_FLAG ControlFlag ;
EFI_SCREEN_DESCRIPTOR LocalScreen ;
MENU_REFRESH_ENTRY * MenuRefreshEntry ;
2011-06-08 10:09:47 +02:00
MENU_REFRESH_ENTRY * MenuUpdateEntry ;
2008-07-15 10:35:54 +02:00
UI_SCREEN_OPERATION ScreenOperation ;
UINT8 MinRefreshInterval ;
UINT16 DefaultId ;
FORM_BROWSER_STATEMENT * Statement ;
2009-10-26 04:03:12 +01:00
UI_MENU_LIST * CurrentMenu ;
2011-06-08 09:27:39 +02:00
UINTN ModalSkipColumn ;
2011-09-13 11:32:54 +02:00
BROWSER_HOT_KEY * HotKey ;
2012-04-11 07:44:40 +02:00
UINTN HelpPageIndex ;
UINTN HelpPageCount ;
UINTN RowCount ;
UINTN HelpLine ;
UINTN HelpHeaderLine ;
UINTN HelpBottomLine ;
BOOLEAN MultiHelpPage ;
2012-05-29 13:07:05 +02:00
UINT16 GlyphWidth ;
UINT16 EachLineWidth ;
UINT16 HeaderLineWidth ;
UINT16 BottomLineWidth ;
2008-07-15 10:35:54 +02:00
CopyMem ( & LocalScreen , & gScreenDimensions , sizeof ( EFI_SCREEN_DESCRIPTOR ) ) ;
Status = EFI_SUCCESS ;
2012-04-11 07:44:40 +02:00
HelpString = NULL ;
HelpHeaderString = NULL ;
HelpBottomString = NULL ;
2008-07-15 10:35:54 +02:00
OptionString = NULL ;
ScreenOperation = UiNoOperation ;
NewLine = TRUE ;
MinRefreshInterval = 0 ;
DefaultId = 0 ;
2012-04-11 07:44:40 +02:00
HelpPageCount = 0 ;
HelpLine = 0 ;
RowCount = 0 ;
HelpBottomLine = 0 ;
HelpHeaderLine = 0 ;
HelpPageIndex = 0 ;
MultiHelpPage = FALSE ;
2012-05-29 13:07:05 +02:00
EachLineWidth = 0 ;
HeaderLineWidth = 0 ;
BottomLineWidth = 0 ;
2008-07-15 10:35:54 +02:00
OutputString = NULL ;
2009-10-26 04:06:06 +01:00
UpArrow = FALSE ;
DownArrow = FALSE ;
2008-07-15 10:35:54 +02:00
SkipValue = 0 ;
OldSkipValue = 0 ;
MenuRefreshEntry = gMenuRefreshHead ;
NextMenuOption = NULL ;
PreviousMenuOption = NULL ;
SavedMenuOption = NULL ;
2011-09-13 11:32:54 +02:00
HotKey = NULL ;
2011-06-08 09:27:39 +02:00
ModalSkipColumn = ( LocalScreen . RightColumn - LocalScreen . LeftColumn ) / 6 ;
2008-07-15 10:35:54 +02:00
ZeroMem ( & Key , sizeof ( EFI_INPUT_KEY ) ) ;
2010-01-06 07:41:09 +01:00
if ( ( gClassOfVfr & FORMSET_CLASS_FRONT_PAGE ) = = FORMSET_CLASS_FRONT_PAGE ) {
2008-07-15 10:35:54 +02:00
TopRow = LocalScreen . TopRow + FRONT_PAGE_HEADER_HEIGHT + SCROLL_ARROW_HEIGHT ;
Row = LocalScreen . TopRow + FRONT_PAGE_HEADER_HEIGHT + SCROLL_ARROW_HEIGHT ;
} else {
TopRow = LocalScreen . TopRow + NONE_FRONT_PAGE_HEADER_HEIGHT + SCROLL_ARROW_HEIGHT ;
Row = LocalScreen . TopRow + NONE_FRONT_PAGE_HEADER_HEIGHT + SCROLL_ARROW_HEIGHT ;
}
2011-06-08 09:27:39 +02:00
if ( Selection - > Form - > ModalForm ) {
Col = LocalScreen . LeftColumn + LEFT_SKIPPED_COLUMNS + ModalSkipColumn ;
} else {
Col = LocalScreen . LeftColumn + LEFT_SKIPPED_COLUMNS ;
}
2011-09-13 11:32:54 +02:00
BottomRow = LocalScreen . BottomRow - STATUS_BAR_HEIGHT - gFooterHeight - SCROLL_ARROW_HEIGHT - 1 ;
2008-07-15 10:35:54 +02:00
Selection - > TopRow = TopRow ;
Selection - > BottomRow = BottomRow ;
Selection - > PromptCol = Col ;
Selection - > OptionCol = gPromptBlockWidth + 1 + LocalScreen . LeftColumn ;
Selection - > Statement = NULL ;
2009-10-26 04:06:06 +01:00
TopOfScreen = gMenuOption . ForwardLink ;
2008-07-15 10:35:54 +02:00
Repaint = TRUE ;
MenuOption = NULL ;
2009-10-26 04:03:12 +01:00
//
// Find current Menu
//
CurrentMenu = UiFindMenuList ( & Selection - > FormSetGuid , Selection - > FormId ) ;
if ( CurrentMenu = = NULL ) {
//
// Current menu not found, add it to the menu tree
//
2011-10-31 04:33:55 +01:00
CurrentMenu = UiAddMenuList ( NULL , Selection - > Handle , & Selection - > FormSetGuid , Selection - > FormId ) ;
2009-10-26 04:03:12 +01:00
}
ASSERT ( CurrentMenu ! = NULL ) ;
2011-05-31 02:59:15 +02:00
Selection - > CurrentMenu = CurrentMenu ;
2009-10-26 04:03:12 +01:00
if ( Selection - > QuestionId = = 0 ) {
//
// Highlight not specified, fetch it from cached menu
//
Selection - > QuestionId = CurrentMenu - > QuestionId ;
2011-07-05 10:20:43 +02:00
Selection - > Sequence = CurrentMenu - > Sequence ;
2009-10-26 04:03:12 +01:00
}
2008-07-15 10:35:54 +02:00
//
2011-03-08 07:55:15 +01:00
// Init option as the current user's selection
2008-07-15 10:35:54 +02:00
//
2011-03-08 07:55:15 +01:00
InitializedFlag = TRUE ;
2009-10-26 04:06:06 +01:00
NewPos = gMenuOption . ForwardLink ;
2008-07-15 10:35:54 +02:00
gST - > ConOut - > EnableCursor ( gST - > ConOut , FALSE ) ;
2011-05-31 02:59:15 +02:00
UpdateStatusBar ( Selection , REFRESH_STATUS_BAR , ( UINT8 ) 0 , TRUE ) ;
2008-07-15 10:35:54 +02:00
ControlFlag = CfInitialization ;
Selection - > Action = UI_ACTION_NONE ;
while ( TRUE ) {
switch ( ControlFlag ) {
case CfInitialization :
2009-10-26 04:06:06 +01:00
if ( IsListEmpty ( & gMenuOption ) ) {
2008-07-15 10:35:54 +02:00
ControlFlag = CfReadKey ;
} else {
ControlFlag = CfCheckSelection ;
}
break ;
case CfCheckSelection :
if ( Selection - > Action ! = UI_ACTION_NONE ) {
ControlFlag = CfExit ;
} else {
ControlFlag = CfRepaint ;
}
break ;
case CfRepaint :
ControlFlag = CfRefreshHighLight ;
if ( Repaint ) {
//
// Display menu
//
2009-10-26 04:06:06 +01:00
DownArrow = FALSE ;
UpArrow = FALSE ;
2008-07-15 10:35:54 +02:00
Row = TopRow ;
2010-09-16 06:51:25 +02:00
Temp = ( UINTN ) SkipValue ;
Temp2 = ( UINTN ) SkipValue ;
2008-07-15 10:35:54 +02:00
2011-06-08 09:27:39 +02:00
if ( Selection - > Form - > ModalForm ) {
ClearLines (
LocalScreen . LeftColumn + ModalSkipColumn ,
LocalScreen . LeftColumn + ModalSkipColumn + gPromptBlockWidth + gOptionBlockWidth ,
TopRow - SCROLL_ARROW_HEIGHT ,
BottomRow + SCROLL_ARROW_HEIGHT ,
PcdGet8 ( PcdBrowserFieldTextColor ) | FIELD_BACKGROUND
) ;
} else {
ClearLines (
LocalScreen . LeftColumn ,
LocalScreen . RightColumn ,
TopRow - SCROLL_ARROW_HEIGHT ,
BottomRow + SCROLL_ARROW_HEIGHT ,
PcdGet8 ( PcdBrowserFieldTextColor ) | FIELD_BACKGROUND
) ;
}
2008-07-15 10:35:54 +02:00
UiFreeRefreshList ( ) ;
MinRefreshInterval = 0 ;
2009-10-26 04:06:06 +01:00
for ( Link = TopOfScreen ; Link ! = & gMenuOption ; Link = Link - > ForwardLink ) {
2008-07-15 10:35:54 +02:00
MenuOption = MENU_OPTION_FROM_LINK ( Link ) ;
MenuOption - > Row = Row ;
MenuOption - > Col = Col ;
2011-06-08 09:27:39 +02:00
if ( Selection - > Form - > ModalForm ) {
MenuOption - > OptCol = gPromptBlockWidth + 1 + LocalScreen . LeftColumn + ModalSkipColumn ;
} else {
MenuOption - > OptCol = gPromptBlockWidth + 1 + LocalScreen . LeftColumn ;
}
2008-07-15 10:35:54 +02:00
Statement = MenuOption - > ThisTag ;
if ( Statement - > InSubtitle ) {
MenuOption - > Col + = SUBTITLE_INDENT ;
}
if ( MenuOption - > GrayOut ) {
gST - > ConOut - > SetAttribute ( gST - > ConOut , FIELD_TEXT_GRAYED | FIELD_BACKGROUND ) ;
} else {
if ( Statement - > Operand = = EFI_IFR_SUBTITLE_OP ) {
2011-02-12 02:51:25 +01:00
gST - > ConOut - > SetAttribute ( gST - > ConOut , PcdGet8 ( PcdBrowserSubtitleTextColor ) | FIELD_BACKGROUND ) ;
2008-07-15 10:35:54 +02:00
}
}
Width = GetWidth ( Statement , MenuOption - > Handle ) ;
OriginalRow = Row ;
2012-05-29 13:07:05 +02:00
GlyphWidth = 1 ;
2008-07-15 10:35:54 +02:00
2011-03-08 07:55:15 +01:00
if ( Statement - > Operand = = EFI_IFR_REF_OP & & MenuOption - > Col > = 2 ) {
2010-11-11 10:44:51 +01:00
//
// Print Arrow for Goto button.
//
PrintAt (
2011-02-21 11:46:14 +01:00
MenuOption - > Col - 2 ,
2010-11-11 10:44:51 +01:00
Row ,
L " %c " ,
GEOMETRICSHAPE_RIGHT_TRIANGLE
) ;
}
2012-05-29 13:07:05 +02:00
for ( Index = 0 ; GetLineByWidth ( MenuOption - > Description , Width , & GlyphWidth , & Index , & OutputString ) ! = 0x0000 ; ) {
2008-07-15 10:35:54 +02:00
if ( ( Temp = = 0 ) & & ( Row < = BottomRow ) ) {
PrintStringAt ( MenuOption - > Col , Row , OutputString ) ;
}
//
// If there is more string to process print on the next row and increment the Skip value
//
2009-02-09 06:29:17 +01:00
if ( StrLen ( & MenuOption - > Description [ Index ] ) ! = 0 ) {
2008-07-15 10:35:54 +02:00
if ( Temp = = 0 ) {
Row + + ;
}
}
2008-11-21 07:59:58 +01:00
FreePool ( OutputString ) ;
2008-07-15 10:35:54 +02:00
if ( Temp ! = 0 ) {
Temp - - ;
}
}
Temp = 0 ;
Row = OriginalRow ;
2008-11-04 15:28:08 +01:00
Status = ProcessOptions ( Selection , MenuOption , FALSE , & OptionString ) ;
if ( EFI_ERROR ( Status ) ) {
//
// Repaint to clear possible error prompt pop-up
//
Repaint = TRUE ;
NewLine = TRUE ;
ControlFlag = CfRepaint ;
break ;
}
2008-07-15 10:35:54 +02:00
if ( OptionString ! = NULL ) {
if ( Statement - > Operand = = EFI_IFR_DATE_OP | | Statement - > Operand = = EFI_IFR_TIME_OP ) {
2011-11-16 08:16:03 +01:00
ProcessStringForDateTime ( MenuOption , OptionString , TRUE ) ;
2008-07-15 10:35:54 +02:00
}
Width = ( UINT16 ) gOptionBlockWidth ;
OriginalRow = Row ;
2012-05-29 13:07:05 +02:00
GlyphWidth = 1 ;
2008-07-15 10:35:54 +02:00
2012-05-29 13:07:05 +02:00
for ( Index = 0 ; GetLineByWidth ( OptionString , Width , & GlyphWidth , & Index , & OutputString ) ! = 0x0000 ; ) {
2008-07-15 10:35:54 +02:00
if ( ( Temp2 = = 0 ) & & ( Row < = BottomRow ) ) {
PrintStringAt ( MenuOption - > OptCol , Row , OutputString ) ;
}
//
// If there is more string to process print on the next row and increment the Skip value
//
2009-02-09 06:29:17 +01:00
if ( StrLen ( & OptionString [ Index ] ) ! = 0 ) {
2008-07-15 10:35:54 +02:00
if ( Temp2 = = 0 ) {
Row + + ;
//
// Since the Number of lines for this menu entry may or may not be reflected accurately
// since the prompt might be 1 lines and option might be many, and vice versa, we need to do
// some testing to ensure we are keeping this in-sync.
//
// If the difference in rows is greater than or equal to the skip value, increase the skip value
//
if ( ( Row - OriginalRow ) > = MenuOption - > Skip ) {
MenuOption - > Skip + + ;
}
}
}
2008-11-21 07:59:58 +01:00
FreePool ( OutputString ) ;
2008-07-15 10:35:54 +02:00
if ( Temp2 ! = 0 ) {
Temp2 - - ;
}
}
Temp2 = 0 ;
Row = OriginalRow ;
2008-11-21 07:59:58 +01:00
FreePool ( OptionString ) ;
2008-07-15 10:35:54 +02:00
}
2011-06-27 11:07:23 +02:00
//
// If Question has refresh guid, register the op-code.
//
if ( ! CompareGuid ( & Statement - > RefreshGuid , & gZeroGuid ) ) {
if ( gMenuEventGuidRefreshHead = = NULL ) {
MenuUpdateEntry = AllocateZeroPool ( sizeof ( MENU_REFRESH_ENTRY ) ) ;
gMenuEventGuidRefreshHead = MenuUpdateEntry ;
} else {
MenuUpdateEntry = gMenuEventGuidRefreshHead ;
while ( MenuUpdateEntry - > Next ! = NULL ) {
MenuUpdateEntry = MenuUpdateEntry - > Next ;
}
MenuUpdateEntry - > Next = AllocateZeroPool ( sizeof ( MENU_REFRESH_ENTRY ) ) ;
MenuUpdateEntry = MenuUpdateEntry - > Next ;
}
ASSERT ( MenuUpdateEntry ! = NULL ) ;
Status = gBS - > CreateEventEx ( EVT_NOTIFY_SIGNAL , TPL_NOTIFY , RefreshQuestionNotify , MenuUpdateEntry , & Statement - > RefreshGuid , & MenuUpdateEntry - > Event ) ;
ASSERT ( ! EFI_ERROR ( Status ) ) ;
MenuUpdateEntry - > MenuOption = MenuOption ;
MenuUpdateEntry - > Selection = Selection ;
MenuUpdateEntry - > CurrentColumn = MenuOption - > OptCol ;
MenuUpdateEntry - > CurrentRow = MenuOption - > Row ;
if ( MenuOption - > GrayOut ) {
MenuUpdateEntry - > CurrentAttribute = FIELD_TEXT_GRAYED | FIELD_BACKGROUND ;
} else {
MenuUpdateEntry - > CurrentAttribute = PcdGet8 ( PcdBrowserFieldTextColor ) | FIELD_BACKGROUND ;
}
}
//
// If Question request refresh, register the op-code
//
if ( Statement - > RefreshInterval ! = 0 ) {
//
// Menu will be refreshed at minimal interval of all Questions
// which have refresh request
//
if ( MinRefreshInterval = = 0 | | Statement - > RefreshInterval < MinRefreshInterval ) {
MinRefreshInterval = Statement - > RefreshInterval ;
}
if ( gMenuRefreshHead = = NULL ) {
MenuRefreshEntry = AllocateZeroPool ( sizeof ( MENU_REFRESH_ENTRY ) ) ;
gMenuRefreshHead = MenuRefreshEntry ;
} else {
MenuRefreshEntry = gMenuRefreshHead ;
while ( MenuRefreshEntry - > Next ! = NULL ) {
MenuRefreshEntry = MenuRefreshEntry - > Next ;
}
MenuRefreshEntry - > Next = AllocateZeroPool ( sizeof ( MENU_REFRESH_ENTRY ) ) ;
MenuRefreshEntry = MenuRefreshEntry - > Next ;
}
ASSERT ( MenuRefreshEntry ! = NULL ) ;
MenuRefreshEntry - > MenuOption = MenuOption ;
MenuRefreshEntry - > Selection = Selection ;
MenuRefreshEntry - > CurrentColumn = MenuOption - > OptCol ;
MenuRefreshEntry - > CurrentRow = MenuOption - > Row ;
if ( MenuOption - > GrayOut ) {
MenuRefreshEntry - > CurrentAttribute = FIELD_TEXT_GRAYED | FIELD_BACKGROUND ;
} else {
MenuRefreshEntry - > CurrentAttribute = PcdGet8 ( PcdBrowserFieldTextColor ) | FIELD_BACKGROUND ;
}
}
2008-07-15 10:35:54 +02:00
//
// If this is a text op with secondary text information
//
if ( ( Statement - > Operand = = EFI_IFR_TEXT_OP ) & & ( Statement - > TextTwo ! = 0 ) ) {
StringPtr = GetToken ( Statement - > TextTwo , MenuOption - > Handle ) ;
Width = ( UINT16 ) gOptionBlockWidth ;
OriginalRow = Row ;
2012-05-29 13:07:05 +02:00
GlyphWidth = 1 ;
2008-07-15 10:35:54 +02:00
2012-05-29 13:07:05 +02:00
for ( Index = 0 ; GetLineByWidth ( StringPtr , Width , & GlyphWidth , & Index , & OutputString ) ! = 0x0000 ; ) {
2008-07-15 10:35:54 +02:00
if ( ( Temp = = 0 ) & & ( Row < = BottomRow ) ) {
PrintStringAt ( MenuOption - > OptCol , Row , OutputString ) ;
}
//
// If there is more string to process print on the next row and increment the Skip value
//
2009-02-09 06:29:17 +01:00
if ( StrLen ( & StringPtr [ Index ] ) ! = 0 ) {
2008-07-15 10:35:54 +02:00
if ( Temp2 = = 0 ) {
Row + + ;
//
// Since the Number of lines for this menu entry may or may not be reflected accurately
// since the prompt might be 1 lines and option might be many, and vice versa, we need to do
// some testing to ensure we are keeping this in-sync.
//
// If the difference in rows is greater than or equal to the skip value, increase the skip value
//
if ( ( Row - OriginalRow ) > = MenuOption - > Skip ) {
MenuOption - > Skip + + ;
}
}
}
2008-11-21 07:59:58 +01:00
FreePool ( OutputString ) ;
2008-07-15 10:35:54 +02:00
if ( Temp2 ! = 0 ) {
Temp2 - - ;
}
}
Row = OriginalRow ;
2008-11-21 07:59:58 +01:00
FreePool ( StringPtr ) ;
2008-07-15 10:35:54 +02:00
}
2011-02-12 02:51:25 +01:00
gST - > ConOut - > SetAttribute ( gST - > ConOut , PcdGet8 ( PcdBrowserFieldTextColor ) | FIELD_BACKGROUND ) ;
2008-07-15 10:35:54 +02:00
//
// Need to handle the bottom of the display
//
if ( MenuOption - > Skip > 1 ) {
Row + = MenuOption - > Skip - SkipValue ;
SkipValue = 0 ;
} else {
Row + = MenuOption - > Skip ;
}
if ( Row > BottomRow ) {
if ( ! ValueIsScroll ( FALSE , Link ) ) {
2009-10-26 04:06:06 +01:00
DownArrow = TRUE ;
2008-07-15 10:35:54 +02:00
}
Row = BottomRow + 1 ;
break ;
}
}
if ( ! ValueIsScroll ( TRUE , TopOfScreen ) ) {
2009-10-26 04:06:06 +01:00
UpArrow = TRUE ;
2008-07-15 10:35:54 +02:00
}
2009-10-26 04:06:06 +01:00
if ( UpArrow ) {
2008-07-15 10:35:54 +02:00
gST - > ConOut - > SetAttribute ( gST - > ConOut , ARROW_TEXT | ARROW_BACKGROUND ) ;
PrintAt (
LocalScreen . LeftColumn + gPromptBlockWidth + gOptionBlockWidth + 1 ,
TopRow - SCROLL_ARROW_HEIGHT ,
L " %c " ,
ARROW_UP
) ;
2011-02-12 02:51:25 +01:00
gST - > ConOut - > SetAttribute ( gST - > ConOut , PcdGet8 ( PcdBrowserFieldTextColor ) | FIELD_BACKGROUND ) ;
2008-07-15 10:35:54 +02:00
}
2009-10-26 04:06:06 +01:00
if ( DownArrow ) {
2008-07-15 10:35:54 +02:00
gST - > ConOut - > SetAttribute ( gST - > ConOut , ARROW_TEXT | ARROW_BACKGROUND ) ;
PrintAt (
LocalScreen . LeftColumn + gPromptBlockWidth + gOptionBlockWidth + 1 ,
BottomRow + SCROLL_ARROW_HEIGHT ,
L " %c " ,
ARROW_DOWN
) ;
2011-02-12 02:51:25 +01:00
gST - > ConOut - > SetAttribute ( gST - > ConOut , PcdGet8 ( PcdBrowserFieldTextColor ) | FIELD_BACKGROUND ) ;
2008-07-15 10:35:54 +02:00
}
MenuOption = NULL ;
}
break ;
case CfRefreshHighLight :
//
// MenuOption: Last menu option that need to remove hilight
// MenuOption is set to NULL in Repaint
// NewPos: Current menu option that need to hilight
//
ControlFlag = CfUpdateHelpString ;
2011-03-08 07:55:15 +01:00
if ( InitializedFlag ) {
InitializedFlag = FALSE ;
2011-10-25 11:00:23 +02:00
MoveToNextStatement ( Selection , FALSE , & NewPos , BottomRow - TopRow ) ;
2011-03-08 07:55:15 +01:00
}
2008-07-15 10:35:54 +02:00
//
// Repaint flag is normally reset when finish processing CfUpdateHelpString. Temporarily
// reset Repaint flag because we may break halfway and skip CfUpdateHelpString processing.
//
SavedValue = Repaint ;
Repaint = FALSE ;
if ( Selection - > QuestionId ! = 0 ) {
2009-10-26 04:06:06 +01:00
NewPos = gMenuOption . ForwardLink ;
2008-07-15 10:35:54 +02:00
SavedMenuOption = MENU_OPTION_FROM_LINK ( NewPos ) ;
2011-07-05 10:20:43 +02:00
while ( ( SavedMenuOption - > ThisTag - > QuestionId ! = Selection - > QuestionId | |
SavedMenuOption - > Sequence ! = Selection - > Sequence ) & &
NewPos - > ForwardLink ! = & gMenuOption ) {
2008-07-15 10:35:54 +02:00
NewPos = NewPos - > ForwardLink ;
SavedMenuOption = MENU_OPTION_FROM_LINK ( NewPos ) ;
}
if ( SavedMenuOption - > ThisTag - > QuestionId = = Selection - > QuestionId ) {
//
// Target Question found, find its MenuOption
//
Link = TopOfScreen ;
for ( Index = TopRow ; Index < = BottomRow & & Link ! = NewPos ; ) {
SavedMenuOption = MENU_OPTION_FROM_LINK ( Link ) ;
Index + = SavedMenuOption - > Skip ;
2011-10-25 11:00:23 +02:00
if ( Link = = TopOfScreen ) {
Index - = OldSkipValue ;
}
2008-07-15 10:35:54 +02:00
Link = Link - > ForwardLink ;
}
2011-10-25 11:00:23 +02:00
if ( NewPos = = Link ) {
SavedMenuOption = MENU_OPTION_FROM_LINK ( Link ) ;
}
2008-07-15 10:35:54 +02:00
2011-12-05 14:20:03 +01:00
//
// Not find the selected menu in current show page.
// Have two case to enter this if:
// 1. Not find the menu at current page.
// 2. Find the menu in current page, but the menu shows at the bottom and not all info shows.
// For case 2, has an exception: The menu can show more than one pages and now only this menu shows.
//
// Base on the selected menu will show at the bottom of the page,
// select the menu which will show at the top of the page.
//
if ( Link ! = NewPos | | Index > BottomRow | |
( Link = = NewPos & & ( SavedMenuOption - > Row + SavedMenuOption - > Skip - 1 > BottomRow ) & & ( Link ! = TopOfScreen ) ) ) {
2011-11-16 06:09:36 +01:00
//
// Find the MenuOption which has the skip value for Date/Time opcode.
//
AdjustDateAndTimePosition ( FALSE , & NewPos ) ;
2008-07-15 10:35:54 +02:00
//
// NewPos is not in the current page, simply scroll page so that NewPos is in the end of the page
//
2011-10-25 11:00:23 +02:00
SavedMenuOption = MENU_OPTION_FROM_LINK ( NewPos ) ;
//
// SavedMenuOption->Row == 0 means the menu not show yet.
//
if ( SavedMenuOption - > Row = = 0 ) {
UpdateOptionSkipLines ( Selection , SavedMenuOption ) ;
}
2011-12-05 14:20:03 +01:00
//
// Base on the selected menu will show at the bottome of next page,
// select the menu show at the top of the next page.
//
2008-07-15 10:35:54 +02:00
Link = NewPos ;
2011-10-25 11:00:23 +02:00
for ( Index = TopRow + SavedMenuOption - > Skip ; Index < = BottomRow + 1 ; ) {
2008-07-15 10:35:54 +02:00
Link = Link - > BackLink ;
SavedMenuOption = MENU_OPTION_FROM_LINK ( Link ) ;
2011-10-25 11:00:23 +02:00
if ( SavedMenuOption - > Row = = 0 ) {
UpdateOptionSkipLines ( Selection , SavedMenuOption ) ;
}
2011-12-05 14:20:03 +01:00
Index + = SavedMenuOption - > Skip ;
2008-07-15 10:35:54 +02:00
}
2011-12-05 14:20:03 +01:00
//
// Found the menu which will show at the top of the page.
//
if ( Link = = NewPos ) {
//
// The menu can show more than one pages, just show the menu at the top of the page.
//
SkipValue = 0 ;
TopOfScreen = Link ;
OldSkipValue = SkipValue ;
2011-10-25 11:00:23 +02:00
} else {
2011-12-05 14:20:03 +01:00
//
// Check whether need to skip some line for menu shows at the top of the page.
//
SkipValue = Index - BottomRow - 1 ;
if ( SkipValue > 0 & & SkipValue < ( INTN ) SavedMenuOption - > Skip ) {
TopOfScreen = Link ;
OldSkipValue = SkipValue ;
} else {
SkipValue = 0 ;
TopOfScreen = Link - > ForwardLink ;
}
2011-10-25 11:00:23 +02:00
}
2008-07-15 10:35:54 +02:00
Repaint = TRUE ;
NewLine = TRUE ;
ControlFlag = CfRepaint ;
break ;
}
} else {
//
// Target Question not found, highlight the default menu option
//
NewPos = TopOfScreen ;
}
Selection - > QuestionId = 0 ;
}
if ( NewPos ! = NULL & & ( MenuOption = = NULL | | NewPos ! = & MenuOption - > Link ) ) {
if ( MenuOption ! = NULL ) {
//
// Remove highlight on last Menu Option
//
gST - > ConOut - > SetCursorPosition ( gST - > ConOut , MenuOption - > Col , MenuOption - > Row ) ;
ProcessOptions ( Selection , MenuOption , FALSE , & OptionString ) ;
2011-02-12 02:51:25 +01:00
gST - > ConOut - > SetAttribute ( gST - > ConOut , PcdGet8 ( PcdBrowserFieldTextColor ) | FIELD_BACKGROUND ) ;
2008-07-15 10:35:54 +02:00
if ( OptionString ! = NULL ) {
if ( ( MenuOption - > ThisTag - > Operand = = EFI_IFR_DATE_OP ) | |
( MenuOption - > ThisTag - > Operand = = EFI_IFR_TIME_OP )
) {
2011-11-16 08:16:03 +01:00
ProcessStringForDateTime ( MenuOption , OptionString , FALSE ) ;
2008-07-15 10:35:54 +02:00
}
Width = ( UINT16 ) gOptionBlockWidth ;
OriginalRow = MenuOption - > Row ;
2012-05-29 13:07:05 +02:00
GlyphWidth = 1 ;
2008-07-15 10:35:54 +02:00
2012-05-29 13:07:05 +02:00
for ( Index = 0 ; GetLineByWidth ( OptionString , Width , & GlyphWidth , & Index , & OutputString ) ! = 0x0000 ; ) {
2008-07-15 10:35:54 +02:00
if ( MenuOption - > Row > = TopRow & & MenuOption - > Row < = BottomRow ) {
PrintStringAt ( MenuOption - > OptCol , MenuOption - > Row , OutputString ) ;
}
//
// If there is more string to process print on the next row and increment the Skip value
//
2009-02-09 06:29:17 +01:00
if ( StrLen ( & OptionString [ Index ] ) ! = 0 ) {
2008-07-15 10:35:54 +02:00
MenuOption - > Row + + ;
}
2008-11-21 07:59:58 +01:00
FreePool ( OutputString ) ;
2008-07-15 10:35:54 +02:00
}
MenuOption - > Row = OriginalRow ;
2008-11-21 07:59:58 +01:00
FreePool ( OptionString ) ;
2008-07-15 10:35:54 +02:00
} else {
if ( NewLine ) {
if ( MenuOption - > GrayOut ) {
gST - > ConOut - > SetAttribute ( gST - > ConOut , FIELD_TEXT_GRAYED | FIELD_BACKGROUND ) ;
} else if ( MenuOption - > ThisTag - > Operand = = EFI_IFR_SUBTITLE_OP ) {
2011-02-12 02:51:25 +01:00
gST - > ConOut - > SetAttribute ( gST - > ConOut , PcdGet8 ( PcdBrowserSubtitleTextColor ) | FIELD_BACKGROUND ) ;
2008-07-15 10:35:54 +02:00
}
OriginalRow = MenuOption - > Row ;
Width = GetWidth ( MenuOption - > ThisTag , MenuOption - > Handle ) ;
2012-05-29 13:07:05 +02:00
GlyphWidth = 1 ;
2008-07-15 10:35:54 +02:00
2012-05-29 13:07:05 +02:00
for ( Index = 0 ; GetLineByWidth ( MenuOption - > Description , Width , & GlyphWidth , & Index , & OutputString ) ! = 0x0000 ; ) {
2008-07-15 10:35:54 +02:00
if ( MenuOption - > Row > = TopRow & & MenuOption - > Row < = BottomRow ) {
PrintStringAt ( MenuOption - > Col , MenuOption - > Row , OutputString ) ;
}
//
// If there is more string to process print on the next row and increment the Skip value
//
2009-02-09 06:29:17 +01:00
if ( StrLen ( & MenuOption - > Description [ Index ] ) ! = 0 ) {
2008-07-15 10:35:54 +02:00
MenuOption - > Row + + ;
}
2008-11-21 07:59:58 +01:00
FreePool ( OutputString ) ;
2008-07-15 10:35:54 +02:00
}
MenuOption - > Row = OriginalRow ;
2011-02-12 02:51:25 +01:00
gST - > ConOut - > SetAttribute ( gST - > ConOut , PcdGet8 ( PcdBrowserFieldTextColor ) | FIELD_BACKGROUND ) ;
2008-07-15 10:35:54 +02:00
}
}
}
//
2011-03-08 07:55:15 +01:00
// This is the current selected statement
2008-07-15 10:35:54 +02:00
//
MenuOption = MENU_OPTION_FROM_LINK ( NewPos ) ;
2011-03-08 07:55:15 +01:00
Statement = MenuOption - > ThisTag ;
Selection - > Statement = Statement ;
2008-07-15 10:35:54 +02:00
if ( ! IsSelectable ( MenuOption ) ) {
2011-03-08 07:55:15 +01:00
Repaint = SavedValue ;
UpdateKeyHelp ( Selection , MenuOption , FALSE ) ;
2008-07-15 10:35:54 +02:00
break ;
}
2009-10-26 04:03:12 +01:00
//
// Record highlight for current menu
//
CurrentMenu - > QuestionId = Statement - > QuestionId ;
2011-07-05 10:20:43 +02:00
CurrentMenu - > Sequence = MenuOption - > Sequence ;
2008-07-15 10:35:54 +02:00
//
// Set reverse attribute
//
2011-02-12 02:51:25 +01:00
gST - > ConOut - > SetAttribute ( gST - > ConOut , PcdGet8 ( PcdBrowserFieldTextHighlightColor ) | PcdGet8 ( PcdBrowserFieldBackgroundHighlightColor ) ) ;
2008-07-15 10:35:54 +02:00
gST - > ConOut - > SetCursorPosition ( gST - > ConOut , MenuOption - > Col , MenuOption - > Row ) ;
//
// Assuming that we have a refresh linked-list created, lets annotate the
// appropriate entry that we are highlighting with its new attribute. Just prior to this
// lets reset all of the entries' attribute so we do not get multiple highlights in he refresh
//
if ( gMenuRefreshHead ! = NULL ) {
for ( MenuRefreshEntry = gMenuRefreshHead ; MenuRefreshEntry ! = NULL ; MenuRefreshEntry = MenuRefreshEntry - > Next ) {
2011-03-08 07:01:23 +01:00
if ( MenuRefreshEntry - > MenuOption - > GrayOut ) {
2011-02-22 09:05:56 +01:00
MenuRefreshEntry - > CurrentAttribute = FIELD_TEXT_GRAYED | FIELD_BACKGROUND ;
} else {
MenuRefreshEntry - > CurrentAttribute = PcdGet8 ( PcdBrowserFieldTextColor ) | FIELD_BACKGROUND ;
}
2008-07-15 10:35:54 +02:00
if ( MenuRefreshEntry - > MenuOption = = MenuOption ) {
2011-02-12 02:51:25 +01:00
MenuRefreshEntry - > CurrentAttribute = PcdGet8 ( PcdBrowserFieldTextHighlightColor ) | PcdGet8 ( PcdBrowserFieldBackgroundHighlightColor ) ;
2008-07-15 10:35:54 +02:00
}
}
}
ProcessOptions ( Selection , MenuOption , FALSE , & OptionString ) ;
if ( OptionString ! = NULL ) {
if ( Statement - > Operand = = EFI_IFR_DATE_OP | | Statement - > Operand = = EFI_IFR_TIME_OP ) {
2011-11-16 08:16:03 +01:00
ProcessStringForDateTime ( MenuOption , OptionString , FALSE ) ;
2008-07-15 10:35:54 +02:00
}
Width = ( UINT16 ) gOptionBlockWidth ;
OriginalRow = MenuOption - > Row ;
2012-05-29 13:07:05 +02:00
GlyphWidth = 1 ;
2008-07-15 10:35:54 +02:00
2012-05-29 13:07:05 +02:00
for ( Index = 0 ; GetLineByWidth ( OptionString , Width , & GlyphWidth , & Index , & OutputString ) ! = 0x0000 ; ) {
2008-07-15 10:35:54 +02:00
if ( MenuOption - > Row > = TopRow & & MenuOption - > Row < = BottomRow ) {
PrintStringAt ( MenuOption - > OptCol , MenuOption - > Row , OutputString ) ;
}
//
// If there is more string to process print on the next row and increment the Skip value
//
2009-02-09 06:29:17 +01:00
if ( StrLen ( & OptionString [ Index ] ) ! = 0 ) {
2008-07-15 10:35:54 +02:00
MenuOption - > Row + + ;
}
2008-11-21 07:59:58 +01:00
FreePool ( OutputString ) ;
2008-07-15 10:35:54 +02:00
}
MenuOption - > Row = OriginalRow ;
2008-11-21 07:59:58 +01:00
FreePool ( OptionString ) ;
2008-07-15 10:35:54 +02:00
} else {
if ( NewLine ) {
OriginalRow = MenuOption - > Row ;
Width = GetWidth ( Statement , MenuOption - > Handle ) ;
2012-05-29 13:07:05 +02:00
GlyphWidth = 1 ;
2008-07-15 10:35:54 +02:00
2012-05-29 13:07:05 +02:00
for ( Index = 0 ; GetLineByWidth ( MenuOption - > Description , Width , & GlyphWidth , & Index , & OutputString ) ! = 0x0000 ; ) {
2008-07-15 10:35:54 +02:00
if ( MenuOption - > Row > = TopRow & & MenuOption - > Row < = BottomRow ) {
PrintStringAt ( MenuOption - > Col , MenuOption - > Row , OutputString ) ;
}
//
// If there is more string to process print on the next row and increment the Skip value
//
2009-02-09 06:29:17 +01:00
if ( StrLen ( & MenuOption - > Description [ Index ] ) ! = 0 ) {
2008-07-15 10:35:54 +02:00
MenuOption - > Row + + ;
}
2008-11-21 07:59:58 +01:00
FreePool ( OutputString ) ;
2008-07-15 10:35:54 +02:00
}
MenuOption - > Row = OriginalRow ;
}
}
2009-10-26 04:05:16 +01:00
UpdateKeyHelp ( Selection , MenuOption , FALSE ) ;
2008-11-04 15:28:08 +01:00
2008-07-15 10:35:54 +02:00
//
// Clear reverse attribute
//
2011-02-12 02:51:25 +01:00
gST - > ConOut - > SetAttribute ( gST - > ConOut , PcdGet8 ( PcdBrowserFieldTextColor ) | FIELD_BACKGROUND ) ;
2008-07-15 10:35:54 +02:00
}
//
// Repaint flag will be used when process CfUpdateHelpString, so restore its value
// if we didn't break halfway when process CfRefreshHighLight.
//
Repaint = SavedValue ;
break ;
case CfUpdateHelpString :
ControlFlag = CfPrepareToReadKey ;
2011-06-08 09:27:39 +02:00
if ( Selection - > Form - > ModalForm ) {
break ;
}
2008-07-15 10:35:54 +02:00
2011-03-08 07:55:15 +01:00
if ( Repaint | | NewLine ) {
2008-07-15 10:35:54 +02:00
//
// Don't print anything if it is a NULL help token
//
2010-09-15 09:48:11 +02:00
ASSERT ( MenuOption ! = NULL ) ;
2011-03-08 07:55:15 +01:00
if ( MenuOption - > ThisTag - > Help = = 0 | | ! IsSelectable ( MenuOption ) ) {
2008-07-15 10:35:54 +02:00
StringPtr = L " \0 " ;
} else {
StringPtr = GetToken ( MenuOption - > ThisTag - > Help , MenuOption - > Handle ) ;
}
2012-04-11 07:44:40 +02:00
RowCount = BottomRow - TopRow ;
HelpPageIndex = 0 ;
//
// 1.Calculate how many line the help string need to print.
//
2012-05-29 13:07:05 +02:00
if ( HelpString ! = NULL ) {
FreePool ( HelpString ) ;
}
HelpLine = ProcessHelpString ( StringPtr , & HelpString , & EachLineWidth , RowCount ) ;
2012-04-11 07:44:40 +02:00
if ( HelpLine > RowCount ) {
MultiHelpPage = TRUE ;
StringPtr = GetToken ( STRING_TOKEN ( ADJUST_HELP_PAGE_UP ) , gHiiHandle ) ;
2012-05-29 13:07:05 +02:00
if ( HelpHeaderString ! = NULL ) {
FreePool ( HelpHeaderString ) ;
}
HelpHeaderLine = ProcessHelpString ( StringPtr , & HelpHeaderString , & HeaderLineWidth , RowCount ) ;
2012-04-11 07:44:40 +02:00
StringPtr = GetToken ( STRING_TOKEN ( ADJUST_HELP_PAGE_DOWN ) , gHiiHandle ) ;
2012-05-29 13:07:05 +02:00
if ( HelpBottomString ! = NULL ) {
FreePool ( HelpBottomString ) ;
}
HelpBottomLine = ProcessHelpString ( StringPtr , & HelpBottomString , & BottomLineWidth , RowCount ) ;
2008-07-15 10:35:54 +02:00
//
2012-04-11 07:44:40 +02:00
// Calculate the help page count.
2008-07-15 10:35:54 +02:00
//
2012-04-11 07:44:40 +02:00
if ( HelpLine > 2 * RowCount - 2 ) {
HelpPageCount = ( HelpLine - RowCount + 1 ) / ( RowCount - 2 ) + 1 ;
if ( ( HelpLine - RowCount + 1 ) % ( RowCount - 2 ) > 1 ) {
HelpPageCount + = 1 ;
}
} else {
HelpPageCount = 2 ;
2008-07-15 10:35:54 +02:00
}
2012-04-11 07:44:40 +02:00
} else {
MultiHelpPage = FALSE ;
}
}
//
// Clean the help field first.
//
ClearLines (
LocalScreen . RightColumn - gHelpBlockWidth ,
LocalScreen . RightColumn ,
TopRow ,
BottomRow ,
PcdGet8 ( PcdBrowserFieldTextColor ) | FIELD_BACKGROUND
) ;
2008-07-15 10:35:54 +02:00
2012-04-11 07:44:40 +02:00
//
// Check whether need to show the 'More(U/u)' at the begin.
// Base on current direct info, here shows aligned to the right side of the column.
// If the direction is multi line and aligned to right side may have problem, so
// add ASSERT code here.
//
if ( HelpPageIndex > 0 ) {
2012-05-29 13:07:05 +02:00
gST - > ConOut - > SetAttribute ( gST - > ConOut , INFO_TEXT | FIELD_BACKGROUND ) ;
2012-04-11 07:44:40 +02:00
for ( Index = 0 ; Index < HelpHeaderLine ; Index + + ) {
ASSERT ( HelpHeaderLine = = 1 ) ;
ASSERT ( GetStringWidth ( HelpHeaderString ) / 2 < ( UINTN ) ( gHelpBlockWidth - 1 ) ) ;
PrintStringAt (
LocalScreen . RightColumn - GetStringWidth ( HelpHeaderString ) / 2 - 1 ,
Index + TopRow ,
2012-05-29 13:07:05 +02:00
& HelpHeaderString [ Index * HeaderLineWidth ]
2012-04-11 07:44:40 +02:00
) ;
}
}
gST - > ConOut - > SetAttribute ( gST - > ConOut , HELP_TEXT | FIELD_BACKGROUND ) ;
//
// Print the help string info.
//
if ( ! MultiHelpPage ) {
2012-04-23 12:26:04 +02:00
for ( Index = 0 ; Index < HelpLine ; Index + + ) {
2008-07-15 10:35:54 +02:00
PrintStringAt (
LocalScreen . RightColumn - gHelpBlockWidth ,
Index + TopRow ,
2012-05-29 13:07:05 +02:00
& HelpString [ Index * EachLineWidth ]
2012-04-11 07:44:40 +02:00
) ;
}
2012-05-29 13:07:05 +02:00
gST - > ConOut - > SetCursorPosition ( gST - > ConOut , LocalScreen . RightColumn - 1 , BottomRow ) ;
2012-04-11 07:44:40 +02:00
} else {
if ( HelpPageIndex = = 0 ) {
for ( Index = 0 ; Index < RowCount - HelpBottomLine ; Index + + ) {
PrintStringAt (
LocalScreen . RightColumn - gHelpBlockWidth ,
Index + TopRow ,
2012-05-29 13:07:05 +02:00
& HelpString [ Index * EachLineWidth ]
2012-04-11 07:44:40 +02:00
) ;
}
} else {
for ( Index = 0 ; ( Index < RowCount - HelpBottomLine - HelpHeaderLine ) & &
( Index + HelpPageIndex * ( RowCount - 2 ) + 1 < HelpLine ) ; Index + + ) {
PrintStringAt (
LocalScreen . RightColumn - gHelpBlockWidth ,
Index + TopRow + HelpHeaderLine ,
2012-05-29 13:07:05 +02:00
& HelpString [ ( Index + HelpPageIndex * ( RowCount - 2 ) + 1 ) * EachLineWidth ]
2012-04-11 07:44:40 +02:00
) ;
}
2012-05-29 13:07:05 +02:00
if ( HelpPageIndex = = HelpPageCount - 1 ) {
gST - > ConOut - > SetCursorPosition ( gST - > ConOut , LocalScreen . RightColumn - 1 , BottomRow ) ;
}
2012-04-11 07:44:40 +02:00
}
}
//
// Check whether need to print the 'More(D/d)' at the bottom.
// Base on current direct info, here shows aligned to the right side of the column.
// If the direction is multi line and aligned to right side may have problem, so
// add ASSERT code here.
//
if ( HelpPageIndex < HelpPageCount - 1 & & MultiHelpPage ) {
2012-05-29 13:07:05 +02:00
gST - > ConOut - > SetAttribute ( gST - > ConOut , INFO_TEXT | FIELD_BACKGROUND ) ;
2012-04-11 07:44:40 +02:00
for ( Index = 0 ; Index < HelpBottomLine ; Index + + ) {
ASSERT ( HelpBottomLine = = 1 ) ;
ASSERT ( GetStringWidth ( HelpBottomString ) / 2 < ( UINTN ) ( gHelpBlockWidth - 1 ) ) ;
PrintStringAt (
LocalScreen . RightColumn - GetStringWidth ( HelpBottomString ) / 2 - 1 ,
Index + BottomRow - HelpBottomLine ,
2012-05-29 13:07:05 +02:00
& HelpBottomString [ Index * BottomLineWidth ]
2008-07-15 10:35:54 +02:00
) ;
}
}
//
// Reset this flag every time we finish using it.
//
Repaint = FALSE ;
NewLine = FALSE ;
break ;
case CfPrepareToReadKey :
ControlFlag = CfReadKey ;
ScreenOperation = UiNoOperation ;
break ;
case CfReadKey :
ControlFlag = CfScreenOperation ;
//
// Wait for user's selection
//
do {
Status = UiWaitForSingleEvent ( gST - > ConIn - > WaitForKey , 0 , MinRefreshInterval ) ;
} while ( Status = = EFI_TIMEOUT ) ;
2008-11-04 15:28:08 +01:00
if ( Selection - > Action = = UI_ACTION_REFRESH_FORMSET ) {
2008-07-15 10:35:54 +02:00
//
2008-11-04 15:28:08 +01:00
// IFR is updated in Callback of refresh opcode, re-parse it
2008-07-15 10:35:54 +02:00
//
2011-09-13 11:32:54 +02:00
ControlFlag = CfCheckSelection ;
2008-11-04 15:28:08 +01:00
Selection - > Statement = NULL ;
2011-09-13 11:32:54 +02:00
break ;
2008-11-04 15:28:08 +01:00
}
Status = gST - > ConIn - > ReadKeyStroke ( gST - > ConIn , & Key ) ;
//
2009-10-26 04:03:12 +01:00
// If we encounter error, continue to read another key in.
2008-11-04 15:28:08 +01:00
//
if ( EFI_ERROR ( Status ) ) {
ControlFlag = CfReadKey ;
break ;
2008-07-15 10:35:54 +02:00
}
switch ( Key . UnicodeChar ) {
case CHAR_CARRIAGE_RETURN :
2011-12-05 13:45:45 +01:00
if ( MenuOption - > GrayOut | | MenuOption - > ReadOnly ) {
ControlFlag = CfReadKey ;
break ;
}
2008-07-15 10:35:54 +02:00
ScreenOperation = UiSelect ;
gDirection = 0 ;
break ;
//
// We will push the adjustment of these numeric values directly to the input handler
// NOTE: we won't handle manual input numeric
//
case ' + ' :
case ' - ' :
2010-09-15 09:48:11 +02:00
//
// If the screen has no menu items, and the user didn't select UiReset
// ignore the selection and go back to reading keys.
//
2011-12-05 13:45:45 +01:00
if ( IsListEmpty ( & gMenuOption ) | | MenuOption - > GrayOut | | MenuOption - > ReadOnly ) {
2010-09-15 09:48:11 +02:00
ControlFlag = CfReadKey ;
break ;
}
ASSERT ( MenuOption ! = NULL ) ;
2008-07-15 10:35:54 +02:00
Statement = MenuOption - > ThisTag ;
if ( ( Statement - > Operand = = EFI_IFR_DATE_OP )
| | ( Statement - > Operand = = EFI_IFR_TIME_OP )
| | ( ( Statement - > Operand = = EFI_IFR_NUMERIC_OP ) & & ( Statement - > Step ! = 0 ) )
) {
if ( Key . UnicodeChar = = ' + ' ) {
gDirection = SCAN_RIGHT ;
} else {
gDirection = SCAN_LEFT ;
}
Status = ProcessOptions ( Selection , MenuOption , TRUE , & OptionString ) ;
2008-11-04 15:28:08 +01:00
if ( EFI_ERROR ( Status ) ) {
//
// Repaint to clear possible error prompt pop-up
//
Repaint = TRUE ;
NewLine = TRUE ;
2010-12-23 06:58:30 +01:00
} else {
Selection - > Action = UI_ACTION_REFRESH_FORM ;
2008-11-04 15:28:08 +01:00
}
2008-10-30 08:32:46 +01:00
if ( OptionString ! = NULL ) {
FreePool ( OptionString ) ;
}
2008-07-15 10:35:54 +02:00
}
break ;
case ' ^ ' :
ScreenOperation = UiUp ;
break ;
case ' V ' :
case ' v ' :
ScreenOperation = UiDown ;
break ;
case ' ' :
2010-01-06 07:41:09 +01:00
if ( ( gClassOfVfr & FORMSET_CLASS_FRONT_PAGE ) ! = FORMSET_CLASS_FRONT_PAGE ) {
2010-09-15 09:48:11 +02:00
//
// If the screen has no menu items, and the user didn't select UiReset
// ignore the selection and go back to reading keys.
//
if ( IsListEmpty ( & gMenuOption ) ) {
ControlFlag = CfReadKey ;
break ;
}
ASSERT ( MenuOption ! = NULL ) ;
2011-12-05 13:45:45 +01:00
if ( MenuOption - > ThisTag - > Operand = = EFI_IFR_CHECKBOX_OP & & ! MenuOption - > GrayOut & & ! MenuOption - > ReadOnly ) {
2008-07-15 10:35:54 +02:00
ScreenOperation = UiSelect ;
}
}
break ;
2012-04-11 07:44:40 +02:00
case ' D ' :
case ' d ' :
if ( ! MultiHelpPage ) {
ControlFlag = CfReadKey ;
break ;
}
ControlFlag = CfUpdateHelpString ;
HelpPageIndex = HelpPageIndex < HelpPageCount - 1 ? HelpPageIndex + 1 : HelpPageCount - 1 ;
break ;
case ' U ' :
case ' u ' :
if ( ! MultiHelpPage ) {
ControlFlag = CfReadKey ;
break ;
}
ControlFlag = CfUpdateHelpString ;
HelpPageIndex = HelpPageIndex > 0 ? HelpPageIndex - 1 : 0 ;
break ;
2008-07-15 10:35:54 +02:00
case CHAR_NULL :
2011-09-13 11:32:54 +02:00
for ( Index = 0 ; Index < mScanCodeNumber ; Index + + ) {
if ( Key . ScanCode = = gScanCodeToOperation [ Index ] . ScanCode ) {
ScreenOperation = gScanCodeToOperation [ Index ] . ScreenOperation ;
break ;
}
}
if ( Selection - > Form - > ModalForm & & ( Key . ScanCode = = SCAN_ESC | | Index = = mScanCodeNumber ) ) {
2008-07-15 10:35:54 +02:00
//
2011-09-13 11:32:54 +02:00
// ModalForm has no ESC key and Hot Key.
2008-07-15 10:35:54 +02:00
//
2011-09-13 11:32:54 +02:00
ControlFlag = CfReadKey ;
} else if ( Index = = mScanCodeNumber ) {
//
// Check whether Key matches the registered hot key.
//
HotKey = NULL ;
if ( ( gBrowserSettingScope = = SystemLevel ) | | ( gFunctionKeySetting ! = NONE_FUNCTION_KEY_SETTING ) ) {
HotKey = GetHotKeyFromRegisterList ( & Key ) ;
}
if ( HotKey ! = NULL ) {
ScreenOperation = UiHotKey ;
2008-07-15 10:35:54 +02:00
}
}
break ;
}
break ;
case CfScreenOperation :
2009-10-26 04:03:12 +01:00
if ( ScreenOperation ! = UiReset ) {
2008-07-15 10:35:54 +02:00
//
2009-10-26 04:03:12 +01:00
// If the screen has no menu items, and the user didn't select UiReset
2008-07-15 10:35:54 +02:00
// ignore the selection and go back to reading keys.
//
2009-10-26 04:06:06 +01:00
if ( IsListEmpty ( & gMenuOption ) ) {
2008-07-15 10:35:54 +02:00
ControlFlag = CfReadKey ;
break ;
}
}
for ( Index = 0 ;
Index < sizeof ( gScreenOperationToControlFlag ) / sizeof ( gScreenOperationToControlFlag [ 0 ] ) ;
Index + +
) {
if ( ScreenOperation = = gScreenOperationToControlFlag [ Index ] . ScreenOperation ) {
ControlFlag = gScreenOperationToControlFlag [ Index ] . ControlFlag ;
break ;
}
}
break ;
case CfUiSelect :
ControlFlag = CfCheckSelection ;
2010-09-15 09:48:11 +02:00
ASSERT ( MenuOption ! = NULL ) ;
2008-07-15 10:35:54 +02:00
Statement = MenuOption - > ThisTag ;
2010-12-30 08:26:56 +01:00
if ( Statement - > Operand = = EFI_IFR_TEXT_OP ) {
2008-07-15 10:35:54 +02:00
break ;
}
//
// Keep highlight on current MenuOption
//
Selection - > QuestionId = Statement - > QuestionId ;
switch ( Statement - > Operand ) {
case EFI_IFR_REF_OP :
2011-06-28 08:41:28 +02:00
ProcessGotoOpCode ( Statement , Selection , & Repaint , & NewLine ) ;
2008-07-15 10:35:54 +02:00
break ;
case EFI_IFR_ACTION_OP :
//
// Process the Config string <ConfigResp>
//
Status = ProcessQuestionConfig ( Selection , Statement ) ;
if ( EFI_ERROR ( Status ) ) {
break ;
}
//
// The action button may change some Question value, so refresh the form
//
Selection - > Action = UI_ACTION_REFRESH_FORM ;
break ;
case EFI_IFR_RESET_BUTTON_OP :
//
// Reset Question to default value specified by DefaultId
//
ControlFlag = CfUiDefault ;
DefaultId = Statement - > DefaultId ;
break ;
default :
//
// Editable Questions: oneof, ordered list, checkbox, numeric, string, password
//
2009-10-26 04:05:16 +01:00
UpdateKeyHelp ( Selection , MenuOption , TRUE ) ;
2008-07-15 10:35:54 +02:00
Status = ProcessOptions ( Selection , MenuOption , TRUE , & OptionString ) ;
if ( EFI_ERROR ( Status ) ) {
Repaint = TRUE ;
NewLine = TRUE ;
2009-10-26 04:05:16 +01:00
UpdateKeyHelp ( Selection , MenuOption , FALSE ) ;
} else {
Selection - > Action = UI_ACTION_REFRESH_FORM ;
}
2008-07-15 10:35:54 +02:00
2009-11-12 02:16:12 +01:00
if ( OptionString ! = NULL ) {
FreePool ( OptionString ) ;
}
2008-07-15 10:35:54 +02:00
break ;
}
break ;
case CfUiReset :
//
2009-10-26 04:03:12 +01:00
// We come here when someone press ESC
2008-07-15 10:35:54 +02:00
//
ControlFlag = CfCheckSelection ;
2011-09-13 11:32:54 +02:00
FindNextMenu ( Selection , & Repaint , & NewLine ) ;
2011-05-31 02:59:15 +02:00
break ;
2008-07-15 10:35:54 +02:00
case CfUiLeft :
ControlFlag = CfCheckSelection ;
2010-09-15 09:48:11 +02:00
ASSERT ( MenuOption ! = NULL ) ;
2008-07-15 10:35:54 +02:00
if ( ( MenuOption - > ThisTag - > Operand = = EFI_IFR_DATE_OP ) | | ( MenuOption - > ThisTag - > Operand = = EFI_IFR_TIME_OP ) ) {
if ( MenuOption - > Sequence ! = 0 ) {
//
// In the middle or tail of the Date/Time op-code set, go left.
//
2010-09-15 09:48:11 +02:00
ASSERT ( NewPos ! = NULL ) ;
2008-07-15 10:35:54 +02:00
NewPos = NewPos - > BackLink ;
}
}
break ;
case CfUiRight :
ControlFlag = CfCheckSelection ;
2010-09-15 09:48:11 +02:00
ASSERT ( MenuOption ! = NULL ) ;
2008-07-15 10:35:54 +02:00
if ( ( MenuOption - > ThisTag - > Operand = = EFI_IFR_DATE_OP ) | | ( MenuOption - > ThisTag - > Operand = = EFI_IFR_TIME_OP ) ) {
if ( MenuOption - > Sequence ! = 2 ) {
//
// In the middle or tail of the Date/Time op-code set, go left.
//
2010-09-15 09:48:11 +02:00
ASSERT ( NewPos ! = NULL ) ;
2008-07-15 10:35:54 +02:00
NewPos = NewPos - > ForwardLink ;
}
}
break ;
case CfUiUp :
ControlFlag = CfCheckSelection ;
2011-03-08 07:55:15 +01:00
SavedListEntry = NewPos ;
2008-07-15 10:35:54 +02:00
2010-09-15 09:48:11 +02:00
ASSERT ( NewPos ! = NULL ) ;
2011-03-08 07:55:15 +01:00
//
// Adjust Date/Time position before we advance forward.
//
AdjustDateAndTimePosition ( TRUE , & NewPos ) ;
2009-10-26 04:06:06 +01:00
if ( NewPos - > BackLink ! = & gMenuOption ) {
2011-03-08 07:55:15 +01:00
MenuOption = MENU_OPTION_FROM_LINK ( NewPos ) ;
2011-07-05 10:24:22 +02:00
ASSERT ( MenuOption ! = NULL ) ;
2011-03-08 07:55:15 +01:00
NewLine = TRUE ;
NewPos = NewPos - > BackLink ;
2008-07-15 10:35:54 +02:00
PreviousMenuOption = MENU_OPTION_FROM_LINK ( NewPos ) ;
2011-10-25 11:00:23 +02:00
if ( PreviousMenuOption - > Row = = 0 ) {
UpdateOptionSkipLines ( Selection , PreviousMenuOption ) ;
}
2008-07-15 10:35:54 +02:00
DistanceValue = PreviousMenuOption - > Skip ;
2011-03-08 07:55:15 +01:00
Difference = 0 ;
if ( MenuOption - > Row > = DistanceValue + TopRow ) {
2011-10-25 11:00:23 +02:00
Difference = MoveToNextStatement ( Selection , TRUE , & NewPos , MenuOption - > Row - TopRow - DistanceValue ) ;
2011-03-08 07:55:15 +01:00
}
NextMenuOption = MENU_OPTION_FROM_LINK ( NewPos ) ;
2008-07-15 10:35:54 +02:00
if ( Difference < 0 ) {
//
2011-03-08 07:55:15 +01:00
// We hit the begining MenuOption that can be focused
// so we simply scroll to the top.
2008-07-15 10:35:54 +02:00
//
2011-03-08 07:55:15 +01:00
if ( TopOfScreen ! = gMenuOption . ForwardLink ) {
2009-10-26 04:06:06 +01:00
TopOfScreen = gMenuOption . ForwardLink ;
2008-07-15 10:35:54 +02:00
Repaint = TRUE ;
2011-03-08 07:55:15 +01:00
} else {
//
// Scroll up to the last page when we have arrived at top page.
//
NewPos = & gMenuOption ;
TopOfScreen = & gMenuOption ;
MenuOption = MENU_OPTION_FROM_LINK ( SavedListEntry ) ;
ScreenOperation = UiPageUp ;
ControlFlag = CfScreenOperation ;
break ;
2008-07-15 10:35:54 +02:00
}
2011-03-08 07:55:15 +01:00
} else if ( MenuOption - > Row < TopRow + DistanceValue + Difference ) {
2010-10-25 07:32:17 +02:00
//
// Previous focus MenuOption is above the TopOfScreen, so we need to scroll
//
TopOfScreen = NewPos ;
Repaint = TRUE ;
SkipValue = 0 ;
OldSkipValue = 0 ;
2011-03-08 07:55:15 +01:00
} else if ( ! IsSelectable ( NextMenuOption ) ) {
//
// Continue to go up until scroll to next page or the selectable option is found.
//
ScreenOperation = UiUp ;
ControlFlag = CfScreenOperation ;
2008-07-15 10:35:54 +02:00
}
//
// If we encounter a Date/Time op-code set, rewind to the first op-code of the set.
//
AdjustDateAndTimePosition ( TRUE , & TopOfScreen ) ;
2011-03-08 07:55:15 +01:00
AdjustDateAndTimePosition ( TRUE , & NewPos ) ;
MenuOption = MENU_OPTION_FROM_LINK ( SavedListEntry ) ;
2011-05-31 02:59:15 +02:00
UpdateStatusBar ( Selection , INPUT_ERROR , MenuOption - > ThisTag - > QuestionFlags , FALSE ) ;
2008-07-15 10:35:54 +02:00
} else {
2011-03-08 07:55:15 +01:00
//
// Scroll up to the last page.
//
NewPos = & gMenuOption ;
TopOfScreen = & gMenuOption ;
MenuOption = MENU_OPTION_FROM_LINK ( SavedListEntry ) ;
ScreenOperation = UiPageUp ;
ControlFlag = CfScreenOperation ;
2008-07-15 10:35:54 +02:00
}
break ;
case CfUiPageUp :
ControlFlag = CfCheckSelection ;
2010-09-15 09:48:11 +02:00
ASSERT ( NewPos ! = NULL ) ;
2009-10-26 04:06:06 +01:00
if ( NewPos - > BackLink = = & gMenuOption ) {
2008-07-15 10:35:54 +02:00
NewLine = FALSE ;
Repaint = FALSE ;
break ;
}
NewLine = TRUE ;
Repaint = TRUE ;
Link = TopOfScreen ;
2011-03-08 07:55:15 +01:00
Index = BottomRow ;
2009-10-26 04:06:06 +01:00
while ( ( Index > = TopRow ) & & ( Link - > BackLink ! = & gMenuOption ) ) {
2008-07-15 10:35:54 +02:00
Link = Link - > BackLink ;
PreviousMenuOption = MENU_OPTION_FROM_LINK ( Link ) ;
2011-10-25 11:00:23 +02:00
if ( PreviousMenuOption - > Row = = 0 ) {
UpdateOptionSkipLines ( Selection , PreviousMenuOption ) ;
}
2011-03-08 07:55:15 +01:00
if ( Index < PreviousMenuOption - > Skip ) {
Index = 0 ;
break ;
}
Index = Index - PreviousMenuOption - > Skip ;
2008-07-15 10:35:54 +02:00
}
2011-03-08 07:55:15 +01:00
if ( ( Link - > BackLink = = & gMenuOption ) & & ( Index > = TopRow ) ) {
if ( TopOfScreen = = & gMenuOption ) {
TopOfScreen = gMenuOption . ForwardLink ;
NewPos = gMenuOption . BackLink ;
2011-10-25 11:00:23 +02:00
MoveToNextStatement ( Selection , TRUE , & NewPos , BottomRow - TopRow ) ;
2011-03-08 07:55:15 +01:00
Repaint = FALSE ;
} else if ( TopOfScreen ! = Link ) {
TopOfScreen = Link ;
NewPos = Link ;
2011-10-25 11:00:23 +02:00
MoveToNextStatement ( Selection , FALSE , & NewPos , BottomRow - TopRow ) ;
2011-03-08 07:55:15 +01:00
} else {
//
// Finally we know that NewPos is the last MenuOption can be focused.
//
Repaint = FALSE ;
NewPos = Link ;
2011-10-25 11:00:23 +02:00
MoveToNextStatement ( Selection , FALSE , & NewPos , BottomRow - TopRow ) ;
2011-03-08 07:55:15 +01:00
}
} else {
if ( Index + 1 < TopRow ) {
//
// Back up the previous option.
//
Link = Link - > ForwardLink ;
}
2008-07-15 10:35:54 +02:00
//
2011-03-08 07:55:15 +01:00
// Move to the option in Next page.
2008-07-15 10:35:54 +02:00
//
2011-03-08 07:55:15 +01:00
if ( TopOfScreen = = & gMenuOption ) {
NewPos = gMenuOption . BackLink ;
2011-10-25 11:00:23 +02:00
MoveToNextStatement ( Selection , TRUE , & NewPos , BottomRow - TopRow ) ;
2011-03-08 07:55:15 +01:00
} else {
NewPos = Link ;
2011-10-25 11:00:23 +02:00
MoveToNextStatement ( Selection , FALSE , & NewPos , BottomRow - TopRow ) ;
2011-03-08 07:55:15 +01:00
}
2008-07-15 10:35:54 +02:00
//
2011-03-08 07:55:15 +01:00
// There are more MenuOption needing scrolling up.
2008-07-15 10:35:54 +02:00
//
2011-03-08 07:55:15 +01:00
TopOfScreen = Link ;
MenuOption = NULL ;
2008-07-15 10:35:54 +02:00
}
//
// If we encounter a Date/Time op-code set, rewind to the first op-code of the set.
// Don't do this when we are already in the first page.
//
AdjustDateAndTimePosition ( TRUE , & TopOfScreen ) ;
AdjustDateAndTimePosition ( TRUE , & NewPos ) ;
break ;
case CfUiPageDown :
ControlFlag = CfCheckSelection ;
2010-09-16 06:51:25 +02:00
ASSERT ( NewPos ! = NULL ) ;
2009-10-26 04:06:06 +01:00
if ( NewPos - > ForwardLink = = & gMenuOption ) {
2008-07-15 10:35:54 +02:00
NewLine = FALSE ;
Repaint = FALSE ;
break ;
}
NewLine = TRUE ;
Repaint = TRUE ;
Link = TopOfScreen ;
NextMenuOption = MENU_OPTION_FROM_LINK ( Link ) ;
Index = TopRow ;
2009-10-26 04:06:06 +01:00
while ( ( Index < = BottomRow ) & & ( Link - > ForwardLink ! = & gMenuOption ) ) {
2008-07-15 10:35:54 +02:00
Index = Index + NextMenuOption - > Skip ;
Link = Link - > ForwardLink ;
NextMenuOption = MENU_OPTION_FROM_LINK ( Link ) ;
}
2011-03-08 07:55:15 +01:00
if ( ( Link - > ForwardLink = = & gMenuOption ) & & ( Index < = BottomRow ) ) {
//
// Finally we know that NewPos is the last MenuOption can be focused.
//
Repaint = FALSE ;
2011-10-25 11:00:23 +02:00
MoveToNextStatement ( Selection , TRUE , & Link , Index - TopRow ) ;
2011-03-08 07:55:15 +01:00
} else {
if ( Index - 1 > BottomRow ) {
//
// Back up the previous option.
//
Link = Link - > BackLink ;
}
2008-07-15 10:35:54 +02:00
//
2011-03-08 07:55:15 +01:00
// There are more MenuOption needing scrolling down.
2008-07-15 10:35:54 +02:00
//
TopOfScreen = Link ;
MenuOption = NULL ;
//
2011-03-08 07:55:15 +01:00
// Move to the option in Next page.
2008-07-15 10:35:54 +02:00
//
2011-10-25 11:00:23 +02:00
MoveToNextStatement ( Selection , FALSE , & Link , BottomRow - TopRow ) ;
2008-07-15 10:35:54 +02:00
}
//
// If we encounter a Date/Time op-code set, rewind to the first op-code of the set.
// Don't do this when we are already in the last page.
//
2011-03-08 07:55:15 +01:00
NewPos = Link ;
2008-07-15 10:35:54 +02:00
AdjustDateAndTimePosition ( TRUE , & TopOfScreen ) ;
AdjustDateAndTimePosition ( TRUE , & NewPos ) ;
break ;
case CfUiDown :
ControlFlag = CfCheckSelection ;
//
// Since the behavior of hitting the down arrow on a Date/Time op-code is intended
// to be one that progresses to the next set of op-codes, we need to advance to the last
// Date/Time op-code and leave the remaining logic in UiDown intact so the appropriate
// checking can be done. The only other logic we need to introduce is that if a Date/Time
// op-code is the last entry in the menu, we need to rewind back to the first op-code of
// the Date/Time op-code.
//
SavedListEntry = NewPos ;
2011-03-08 07:55:15 +01:00
AdjustDateAndTimePosition ( FALSE , & NewPos ) ;
2008-07-15 10:35:54 +02:00
2009-10-26 04:06:06 +01:00
if ( NewPos - > ForwardLink ! = & gMenuOption ) {
2008-07-15 10:35:54 +02:00
MenuOption = MENU_OPTION_FROM_LINK ( NewPos ) ;
NewLine = TRUE ;
NewPos = NewPos - > ForwardLink ;
2011-03-08 07:55:15 +01:00
Difference = 0 ;
if ( BottomRow > = MenuOption - > Row + MenuOption - > Skip ) {
2011-10-25 11:00:23 +02:00
Difference = MoveToNextStatement ( Selection , FALSE , & NewPos , BottomRow - MenuOption - > Row - MenuOption - > Skip ) ;
2011-03-08 07:55:15 +01:00
//
// We hit the end of MenuOption that can be focused
// so we simply scroll to the first page.
//
if ( Difference < 0 ) {
//
// Scroll to the first page.
//
if ( TopOfScreen ! = gMenuOption . ForwardLink ) {
TopOfScreen = gMenuOption . ForwardLink ;
Repaint = TRUE ;
MenuOption = NULL ;
} else {
MenuOption = MENU_OPTION_FROM_LINK ( SavedListEntry ) ;
}
NewPos = gMenuOption . ForwardLink ;
2011-10-25 11:00:23 +02:00
MoveToNextStatement ( Selection , FALSE , & NewPos , BottomRow - TopRow ) ;
2011-03-08 07:55:15 +01:00
//
// If we are at the end of the list and sitting on a Date/Time op, rewind to the head.
//
AdjustDateAndTimePosition ( TRUE , & TopOfScreen ) ;
AdjustDateAndTimePosition ( TRUE , & NewPos ) ;
break ;
}
}
2010-10-25 07:32:17 +02:00
NextMenuOption = MENU_OPTION_FROM_LINK ( NewPos ) ;
2008-07-15 10:35:54 +02:00
//
// An option might be multi-line, so we need to reflect that data in the overall skip value
//
2011-10-25 11:00:23 +02:00
UpdateOptionSkipLines ( Selection , NextMenuOption ) ;
2011-03-08 07:55:15 +01:00
DistanceValue = Difference + NextMenuOption - > Skip ;
2008-07-15 10:35:54 +02:00
Temp = MenuOption - > Row + MenuOption - > Skip + DistanceValue - 1 ;
if ( ( MenuOption - > Row + MenuOption - > Skip = = BottomRow + 1 ) & &
( NextMenuOption - > ThisTag - > Operand = = EFI_IFR_DATE_OP | |
NextMenuOption - > ThisTag - > Operand = = EFI_IFR_TIME_OP )
) {
Temp + + ;
}
//
// If we are going to scroll, update TopOfScreen
//
if ( Temp > BottomRow ) {
do {
//
// Is the current top of screen a zero-advance op-code?
// If so, keep moving forward till we hit a >0 advance op-code
//
SavedMenuOption = MENU_OPTION_FROM_LINK ( TopOfScreen ) ;
//
// If bottom op-code is more than one line or top op-code is more than one line
//
if ( ( DistanceValue > 1 ) | | ( MenuOption - > Skip > 1 ) ) {
//
// Is the bottom op-code greater than or equal in size to the top op-code?
//
if ( ( Temp - BottomRow ) > = ( SavedMenuOption - > Skip - OldSkipValue ) ) {
//
// Skip the top op-code
//
TopOfScreen = TopOfScreen - > ForwardLink ;
Difference = ( Temp - BottomRow ) - ( SavedMenuOption - > Skip - OldSkipValue ) ;
OldSkipValue = Difference ;
SavedMenuOption = MENU_OPTION_FROM_LINK ( TopOfScreen ) ;
//
// If we have a remainder, skip that many more op-codes until we drain the remainder
//
2010-10-25 07:32:17 +02:00
while ( Difference > = ( INTN ) SavedMenuOption - > Skip ) {
2008-07-15 10:35:54 +02:00
//
// Since the Difference is greater than or equal to this op-code's skip value, skip it
//
2010-10-25 07:32:17 +02:00
Difference = Difference - ( INTN ) SavedMenuOption - > Skip ;
2008-07-15 10:35:54 +02:00
TopOfScreen = TopOfScreen - > ForwardLink ;
SavedMenuOption = MENU_OPTION_FROM_LINK ( TopOfScreen ) ;
}
//
// Since we will act on this op-code in the next routine, and increment the
// SkipValue, set the skips to one less than what is required.
//
SkipValue = Difference - 1 ;
} else {
//
// Since we will act on this op-code in the next routine, and increment the
// SkipValue, set the skips to one less than what is required.
//
SkipValue = OldSkipValue + ( Temp - BottomRow ) - 1 ;
}
} else {
if ( ( OldSkipValue + 1 ) = = ( INTN ) SavedMenuOption - > Skip ) {
TopOfScreen = TopOfScreen - > ForwardLink ;
break ;
} else {
SkipValue = OldSkipValue ;
}
}
//
// If the op-code at the top of the screen is more than one line, let's not skip it yet
// Let's set a skip flag to smoothly scroll the top of the screen.
//
if ( SavedMenuOption - > Skip > 1 ) {
if ( SavedMenuOption = = NextMenuOption ) {
SkipValue = 0 ;
} else {
SkipValue + + ;
}
2010-10-25 07:32:17 +02:00
} else if ( SavedMenuOption - > Skip = = 1 ) {
SkipValue = 0 ;
2008-07-15 10:35:54 +02:00
} else {
SkipValue = 0 ;
TopOfScreen = TopOfScreen - > ForwardLink ;
}
} while ( SavedMenuOption - > Skip = = 0 ) ;
Repaint = TRUE ;
OldSkipValue = SkipValue ;
2011-03-08 07:55:15 +01:00
} else if ( ! IsSelectable ( NextMenuOption ) ) {
//
// Continue to go down until scroll to next page or the selectable option is found.
//
ScreenOperation = UiDown ;
ControlFlag = CfScreenOperation ;
2008-07-15 10:35:54 +02:00
}
MenuOption = MENU_OPTION_FROM_LINK ( SavedListEntry ) ;
2011-05-31 02:59:15 +02:00
UpdateStatusBar ( Selection , INPUT_ERROR , MenuOption - > ThisTag - > QuestionFlags , FALSE ) ;
2008-07-15 10:35:54 +02:00
} else {
//
2011-03-08 07:55:15 +01:00
// Scroll to the first page.
2008-07-15 10:35:54 +02:00
//
2011-03-08 07:55:15 +01:00
if ( TopOfScreen ! = gMenuOption . ForwardLink ) {
TopOfScreen = gMenuOption . ForwardLink ;
Repaint = TRUE ;
MenuOption = NULL ;
} else {
MenuOption = MENU_OPTION_FROM_LINK ( SavedListEntry ) ;
}
NewLine = TRUE ;
NewPos = gMenuOption . ForwardLink ;
2011-10-25 11:00:23 +02:00
MoveToNextStatement ( Selection , FALSE , & NewPos , BottomRow - TopRow ) ;
2008-07-15 10:35:54 +02:00
}
2011-03-08 07:55:15 +01:00
//
// If we are at the end of the list and sitting on a Date/Time op, rewind to the head.
//
AdjustDateAndTimePosition ( TRUE , & TopOfScreen ) ;
AdjustDateAndTimePosition ( TRUE , & NewPos ) ;
2008-07-15 10:35:54 +02:00
break ;
2011-09-13 11:32:54 +02:00
case CfUiHotKey :
2008-07-15 10:35:54 +02:00
ControlFlag = CfCheckSelection ;
2011-09-13 11:32:54 +02:00
Status = EFI_SUCCESS ;
2008-07-15 10:35:54 +02:00
//
2011-09-13 11:32:54 +02:00
// Discard changes. After it, no NV flag is showed.
2008-07-15 10:35:54 +02:00
//
2011-09-13 11:32:54 +02:00
if ( ( HotKey - > Action & BROWSER_ACTION_DISCARD ) = = BROWSER_ACTION_DISCARD ) {
Status = DiscardForm ( Selection - > FormSet , Selection - > Form , gBrowserSettingScope ) ;
if ( ! EFI_ERROR ( Status ) ) {
Selection - > Action = UI_ACTION_REFRESH_FORM ;
Selection - > Statement = NULL ;
gResetRequired = FALSE ;
} else {
do {
CreateDialog ( 4 , TRUE , 0 , NULL , & Key , HotKey - > HelpString , gDiscardFailed , gPressEnter , gEmptyString ) ;
} while ( Key . UnicodeChar ! = CHAR_CARRIAGE_RETURN ) ;
//
// Still show current page.
//
Selection - > Action = UI_ACTION_NONE ;
Repaint = TRUE ;
NewLine = TRUE ;
break ;
}
}
2008-07-15 10:35:54 +02:00
2011-09-13 11:32:54 +02:00
//
// Reterieve default setting. After it. NV flag will be showed.
//
if ( ( HotKey - > Action & BROWSER_ACTION_DEFAULT ) = = BROWSER_ACTION_DEFAULT ) {
2012-06-12 05:27:45 +02:00
Status = ExtractDefault ( Selection - > FormSet , Selection - > Form , HotKey - > DefaultId , gBrowserSettingScope , NULL ) ;
2011-09-13 11:32:54 +02:00
if ( ! EFI_ERROR ( Status ) ) {
Selection - > Action = UI_ACTION_REFRESH_FORM ;
Selection - > Statement = NULL ;
gResetRequired = TRUE ;
} else {
do {
CreateDialog ( 4 , TRUE , 0 , NULL , & Key , HotKey - > HelpString , gDefaultFailed , gPressEnter , gEmptyString ) ;
} while ( Key . UnicodeChar ! = CHAR_CARRIAGE_RETURN ) ;
//
// Still show current page.
//
Selection - > Action = UI_ACTION_NONE ;
Repaint = TRUE ;
NewLine = TRUE ;
break ;
}
}
2008-07-15 10:35:54 +02:00
2011-09-13 11:32:54 +02:00
//
// Save changes. After it, no NV flag is showed.
//
if ( ( HotKey - > Action & BROWSER_ACTION_SUBMIT ) = = BROWSER_ACTION_SUBMIT ) {
Status = SubmitForm ( Selection - > FormSet , Selection - > Form , gBrowserSettingScope ) ;
if ( ! EFI_ERROR ( Status ) ) {
ASSERT ( MenuOption ! = NULL ) ;
UpdateStatusBar ( Selection , INPUT_ERROR , MenuOption - > ThisTag - > QuestionFlags , FALSE ) ;
UpdateStatusBar ( Selection , NV_UPDATE_REQUIRED , MenuOption - > ThisTag - > QuestionFlags , FALSE ) ;
} else {
do {
CreateDialog ( 4 , TRUE , 0 , NULL , & Key , HotKey - > HelpString , gSaveFailed , gPressEnter , gEmptyString ) ;
} while ( Key . UnicodeChar ! = CHAR_CARRIAGE_RETURN ) ;
//
// Still show current page.
//
Selection - > Action = UI_ACTION_NONE ;
Repaint = TRUE ;
NewLine = TRUE ;
break ;
}
}
//
// Set Reset required Flag
//
if ( ( HotKey - > Action & BROWSER_ACTION_RESET ) = = BROWSER_ACTION_RESET ) {
gResetRequired = TRUE ;
}
//
// Exit Action
//
if ( ( HotKey - > Action & BROWSER_ACTION_EXIT ) = = BROWSER_ACTION_EXIT ) {
//
// Form Exit without saving, Similar to ESC Key.
// FormSet Exit without saving, Exit SendForm.
// System Exit without saving, CallExitHandler and Exit SendForm.
//
DiscardForm ( Selection - > FormSet , Selection - > Form , gBrowserSettingScope ) ;
if ( gBrowserSettingScope = = FormLevel ) {
ControlFlag = CfUiReset ;
} else if ( gBrowserSettingScope = = FormSetLevel ) {
Selection - > Action = UI_ACTION_EXIT ;
} else if ( gBrowserSettingScope = = SystemLevel ) {
if ( ExitHandlerFunction ! = NULL ) {
ExitHandlerFunction ( ) ;
}
Selection - > Action = UI_ACTION_EXIT ;
}
Selection - > Statement = NULL ;
2008-07-15 10:35:54 +02:00
}
break ;
case CfUiDefault :
ControlFlag = CfCheckSelection ;
2011-06-28 07:01:49 +02:00
//
2011-09-13 11:32:54 +02:00
// Reset to default value for all forms in the whole system.
2011-06-28 07:01:49 +02:00
//
2012-06-12 05:27:45 +02:00
Status = ExtractDefault ( Selection - > FormSet , NULL , DefaultId , FormSetLevel , NULL ) ;
2008-07-15 10:35:54 +02:00
if ( ! EFI_ERROR ( Status ) ) {
Selection - > Action = UI_ACTION_REFRESH_FORM ;
2008-09-02 03:23:52 +02:00
Selection - > Statement = NULL ;
2011-02-28 10:31:09 +01:00
gResetRequired = TRUE ;
2008-07-15 10:35:54 +02:00
}
break ;
case CfUiNoOperation :
ControlFlag = CfCheckSelection ;
break ;
case CfExit :
UiFreeRefreshList ( ) ;
gST - > ConOut - > SetAttribute ( gST - > ConOut , EFI_TEXT_ATTR ( EFI_LIGHTGRAY , EFI_BLACK ) ) ;
gST - > ConOut - > SetCursorPosition ( gST - > ConOut , 0 , Row + 4 ) ;
gST - > ConOut - > EnableCursor ( gST - > ConOut , TRUE ) ;
gST - > ConOut - > OutputString ( gST - > ConOut , L " \n " ) ;
2012-05-29 13:07:05 +02:00
if ( HelpString ! = NULL ) {
FreePool ( HelpString ) ;
}
if ( HelpHeaderString ! = NULL ) {
FreePool ( HelpHeaderString ) ;
}
if ( HelpBottomString ! = NULL ) {
FreePool ( HelpBottomString ) ;
}
2008-07-15 10:35:54 +02:00
return EFI_SUCCESS ;
default :
break ;
}
}
}