123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735 |
- /*
- ===========================================================================
- Doom 3 GPL Source Code
- Copyright (C) 1999-2011 id Software LLC, a ZeniMax Media company.
- This file is part of the Doom 3 GPL Source Code (?Doom 3 Source Code?).
- Doom 3 Source Code is free software: you can redistribute it and/or modify
- it under the terms of the GNU General Public License as published by
- the Free Software Foundation, either version 3 of the License, or
- (at your option) any later version.
- Doom 3 Source Code is distributed in the hope that it will be useful,
- but WITHOUT ANY WARRANTY; without even the implied warranty of
- MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
- GNU General Public License for more details.
- You should have received a copy of the GNU General Public License
- along with Doom 3 Source Code. If not, see <http://www.gnu.org/licenses/>.
- In addition, the Doom 3 Source Code is also subject to certain additional terms. You should have received a copy of these additional terms immediately following the terms and conditions of the GNU General Public License which accompanied the Doom 3 Source Code. If not, please request a copy in writing from id Software at the address below.
- If you have questions concerning this license or the applicable additional terms, you may contact in writing id Software LLC, c/o ZeniMax Media Inc., Suite 120, Rockville, Maryland 20850 USA.
- ===========================================================================
- */
- #include "../idlib/precompiled.h"
- #pragma hdrstop
- #include "../framework/Session_local.h"
- #include "DeviceContext.h"
- #include "Window.h"
- #include "UserInterfaceLocal.h"
- #include "SliderWindow.h"
- #include "ListWindow.h"
- #define DECK_MAX_LINES 16
- // Number of pixels above the text that the rect starts
- static const int pixelOffset = 3;
- // number of pixels between columns
- static const int tabBorder = 4;
- // Time in milliseconds between clicks to register as a double-click
- static const int doubleClickSpeed = 300;
- void idListWindow::CommonInit() {
- typed = "";
- typedTime = 0;
- clickTime = 0;
- currentSel.Clear();
- top = 0;
- sizeBias = 0;
- horizontal = false;
- scroller = new idSliderWindow(dc, gui);
- multipleSel = false;
- //bc
- readonly = false;
- forcescroll = false;
- noscrollbar = false;
- listgap = 0;
- }
- idListWindow::idListWindow(idDeviceContext *d, idUserInterfaceLocal *g) : idWindow(d, g) {
- dc = d;
- gui = g;
- CommonInit();
- }
- idListWindow::idListWindow(idUserInterfaceLocal *g) : idWindow(g) {
- gui = g;
- CommonInit();
- }
- void idListWindow::SetCurrentSel( int sel ) {
- currentSel.Clear();
- currentSel.Append( sel );
- }
- void idListWindow::ClearSelection( int sel ) {
- int cur = currentSel.FindIndex( sel );
- if ( cur >= 0 ) {
- currentSel.RemoveIndex( cur );
- }
- }
- void idListWindow::AddCurrentSel( int sel ) {
- currentSel.Append( sel );
- }
- int idListWindow::GetCurrentSel() {
- return ( currentSel.Num() ) ? currentSel[0] : 0;
- }
- bool idListWindow::IsSelected( int index ) {
- return ( currentSel.FindIndex( index ) >= 0 );
- }
- const char *idListWindow::HandleEvent(const sysEvent_t *event, bool *updateVisuals) {
- // need to call this to allow proper focus and capturing on embedded children
- int key = event->evValue;
-
- //BC ignore input if the list is empty.
- if ( event->evType == SE_KEY )
- {
- if ( key == K_MOUSE1 && listItems.Num() <= 0)
- {
- return cmd;
- }
- }
-
- const char *ret = idWindow::HandleEvent(event, updateVisuals);
- float vert = GetMaxCharHeight() + listgap;
- int numVisibleLines = textRect.h / vert;
-
- if ( event->evType == SE_KEY ) {
- if ( !event->evValue2 ) {
- // We only care about key down, not up
- return ret;
- }
- if ( key == K_MOUSE1 || key == K_MOUSE2 ) {
- // If the user clicked in the scroller, then ignore it
- if ( scroller->Contains(gui->CursorX(), gui->CursorY()) ) {
- return ret;
- }
- }
- if ( ( key == K_ENTER || key == K_KP_ENTER ) ) {
- RunScript( ON_ENTER );
- return cmd;
- }
- if ( key == K_MWHEELUP ) {
- key = K_UPARROW;
- } else if ( key == K_MWHEELDOWN ) {
- key = K_DOWNARROW;
- }
- if (key == K_PGUP || key == K_PGDN)
- {
- if ( idKeyInput::IsDown( K_CTRL ) )
- {
- if (key == K_PGUP)
- scroller->SetValue(0);
- else
- scroller->SetValue(listItems.Num() - DECK_MAX_LINES);
- }
- else
- {
- float curScroll = scroller->GetValue();
- float desiredScroll;
- if (key == K_PGUP)
- desiredScroll = curScroll - 8;
- else
- desiredScroll = curScroll + 8;
- if (desiredScroll < 0)
- desiredScroll = 0;
- else if (desiredScroll > listItems.Num() - DECK_MAX_LINES)
- desiredScroll = listItems.Num() - DECK_MAX_LINES;
-
- scroller->SetValue( desiredScroll );
- }
- UpdateList( false );
- return cmd;
- }
- if ( (key == K_MOUSE1) && !readonly) {
- if (Contains(gui->CursorX(), gui->CursorY())) {
- int cur = ( int )( ( gui->CursorY() - actualY - pixelOffset ) / vert ) + top;
- if ( cur >= 0 && cur < listItems.Num() ) {
- if ( multipleSel && idKeyInput::IsDown( K_CTRL ) ) {
- if ( IsSelected( cur ) ) {
- ClearSelection( cur );
- } else {
- AddCurrentSel( cur );
- }
- } else {
- if ( IsSelected( cur ) && ( gui->GetTime() < clickTime + doubleClickSpeed ) ) {
- // Double-click causes ON_ENTER to get run
- RunScript(ON_ENTER);
- return cmd;
- }
- SetCurrentSel( cur );
- clickTime = gui->GetTime();
- }
- } else {
- SetCurrentSel( listItems.Num() - 1 );
- }
- }
- } else if ( key == K_UPARROW || key == K_PGUP || key == K_DOWNARROW || key == K_PGDN ) {
- int numLines = 1;
- if ( key == K_PGUP || key == K_PGDN ) {
- numLines = numVisibleLines / 2;
- }
- if ( key == K_UPARROW || key == K_PGUP ) {
- numLines = -numLines;
- }
- if ( idKeyInput::IsDown( K_CTRL ) ) {
- top += numLines;
- } else {
- SetCurrentSel( GetCurrentSel() + numLines );
- }
- } else {
- return ret;
- }
- } else if ( event->evType == SE_CHAR ) {
- if ( !idStr::CharIsPrintable(key) ) {
- return ret;
- }
- if ( gui->GetTime() > typedTime + 1000 ) {
- typed = "";
- }
- typedTime = gui->GetTime();
- typed.Append( key );
- for ( int i=0; i<listItems.Num(); i++ ) {
- if ( idStr::Icmpn( typed, listItems[i], typed.Length() ) == 0 ) {
- SetCurrentSel( i );
- break;
- }
- }
- } else {
- return ret;
- }
- if ( GetCurrentSel() < 0 ) {
- SetCurrentSel( 0 );
- }
- if ( GetCurrentSel() >= listItems.Num() ) {
- SetCurrentSel( listItems.Num() - 1 );
- }
- if ( scroller->GetHigh() > 0.0f ) {
- if ( !idKeyInput::IsDown( K_CTRL ) ) {
- if ( top > GetCurrentSel() - 1 ) {
- top = GetCurrentSel() - 1;
- }
- if ( top < GetCurrentSel() - numVisibleLines + 2 ) {
- top = GetCurrentSel() - numVisibleLines + 2;
- }
- }
- if ( top > listItems.Num() - 2 ) {
- top = listItems.Num() - 2;
- }
- if ( top < 0 ) {
- top = 0;
- }
- scroller->SetValue(top);
- } else {
- top = 0;
- scroller->SetValue(0.0f);
- }
-
- if ( key != K_MOUSE1 ) {
- // Send a fake mouse click event so onAction gets run in our parents
- const sysEvent_t ev = sys->GenerateMouseButtonEvent( 1, true );
- idWindow::HandleEvent(&ev, updateVisuals);
- }
- if ( currentSel.Num() > 0 ) {
- for ( int i = 0; i < currentSel.Num(); i++ ) {
- gui->SetStateInt( va( "%s_sel_%i", listName.c_str(), i ), currentSel[i] );
- }
- } else {
- gui->SetStateInt( va( "%s_sel_0", listName.c_str() ), 0 );
- }
- gui->SetStateInt( va( "%s_numsel", listName.c_str() ), currentSel.Num() );
- if (listItems.Num() > 0)
- gui->SetStateString( va( "%s_selected", name.c_str()), listItems[GetCurrentSel()].c_str() );
-
- return ret;
- }
- bool idListWindow::ParseInternalVar(const char *_name, idParser *src) {
-
- //bc
- if ( idStr::Icmp( _name, "readonly" ) == 0) {
- readonly = src->ParseBool();
- return true;
- }
- if ( idStr::Icmp( _name, "forceScroll" ) == 0) {
- forcescroll = src->ParseBool();
- return true;
- }
- if ( idStr::Icmp( _name, "noscrollbar" ) == 0) {
- noscrollbar = src->ParseBool();
- return true;
- }
- if ( idStr::Icmp( _name, "listgap" ) == 0) {
- listgap = src->ParseInt();
- return true;
- }
-
- if (idStr::Icmp(_name, "horizontal") == 0) {
- horizontal = src->ParseBool();
- return true;
- }
- if (idStr::Icmp(_name, "listname") == 0) {
- ParseString(src, listName);
- return true;
- }
- if (idStr::Icmp(_name, "tabstops") == 0) {
- ParseString(src, tabStopStr);
- return true;
- }
- if (idStr::Icmp(_name, "tabaligns") == 0) {
- ParseString(src, tabAlignStr);
- return true;
- }
- if (idStr::Icmp(_name, "multipleSel") == 0) {
- multipleSel = src->ParseBool();
- return true;
- }
- if(idStr::Icmp(_name, "tabvaligns") == 0) {
- ParseString(src, tabVAlignStr);
- return true;
- }
- if(idStr::Icmp(_name, "tabTypes") == 0) {
- ParseString(src, tabTypeStr);
- return true;
- }
- if(idStr::Icmp(_name, "tabIconSizes") == 0) {
- ParseString(src, tabIconSizeStr);
- return true;
- }
- if(idStr::Icmp(_name, "tabIconVOffset") == 0) {
- ParseString(src, tabIconVOffsetStr);
- return true;
- }
-
- idStr strName = _name;
- if(idStr::Icmp(strName.Left(4), "mtr_") == 0) {
- idStr matName;
- const idMaterial* mat;
- ParseString(src, matName);
- mat = declManager->FindMaterial(matName);
- mat->SetImageClassifications( 1 ); // just for resource tracking
- if ( mat && !mat->TestMaterialFlag( MF_DEFAULTED ) ) {
- mat->SetSort(SS_GUI );
- }
- iconMaterials.Set(_name, mat);
- return true;
- }
- return idWindow::ParseInternalVar(_name, src);
- }
- idWinVar *idListWindow::GetWinVarByName(const char *_name, bool fixup, drawWin_t** owner) {
- return idWindow::GetWinVarByName(_name, fixup, owner);
- }
- void idListWindow::PostParse() {
- idWindow::PostParse();
- if (!noscrollbar)
- InitScroller(horizontal);
- idList<int> tabStops;
- idList<int> tabAligns;
- if (tabStopStr.Length()) {
- idParser src(tabStopStr, tabStopStr.Length(), "tabstops", LEXFL_NOFATALERRORS | LEXFL_NOSTRINGCONCAT | LEXFL_NOSTRINGESCAPECHARS);
- idToken tok;
- while (src.ReadToken(&tok)) {
- if (tok == ",") {
- continue;
- }
- tabStops.Append(atoi(tok));
- }
- }
- if (tabAlignStr.Length()) {
- idParser src(tabAlignStr, tabAlignStr.Length(), "tabaligns", LEXFL_NOFATALERRORS | LEXFL_NOSTRINGCONCAT | LEXFL_NOSTRINGESCAPECHARS);
- idToken tok;
- while (src.ReadToken(&tok)) {
- if (tok == ",") {
- continue;
- }
- tabAligns.Append(atoi(tok));
- }
- }
- idList<int> tabVAligns;
- if (tabVAlignStr.Length()) {
- idParser src(tabVAlignStr, tabVAlignStr.Length(), "tabvaligns", LEXFL_NOFATALERRORS | LEXFL_NOSTRINGCONCAT | LEXFL_NOSTRINGESCAPECHARS);
- idToken tok;
- while (src.ReadToken(&tok)) {
- if (tok == ",") {
- continue;
- }
- tabVAligns.Append(atoi(tok));
- }
- }
- idList<int> tabTypes;
- if (tabTypeStr.Length()) {
- idParser src(tabTypeStr, tabTypeStr.Length(), "tabtypes", LEXFL_NOFATALERRORS | LEXFL_NOSTRINGCONCAT | LEXFL_NOSTRINGESCAPECHARS);
- idToken tok;
- while (src.ReadToken(&tok)) {
- if (tok == ",") {
- continue;
- }
- tabTypes.Append(atoi(tok));
- }
- }
- idList<idVec2> tabSizes;
- if (tabIconSizeStr.Length()) {
- idParser src(tabIconSizeStr, tabIconSizeStr.Length(), "tabiconsizes", LEXFL_NOFATALERRORS | LEXFL_NOSTRINGCONCAT | LEXFL_NOSTRINGESCAPECHARS);
- idToken tok;
- while (src.ReadToken(&tok)) {
- if (tok == ",") {
- continue;
- }
- idVec2 size;
- size.x = atoi(tok);
-
- src.ReadToken(&tok); //","
- src.ReadToken(&tok);
- size.y = atoi(tok);
- tabSizes.Append(size);
- }
- }
- idList<float> tabIconVOffsets;
- if (tabIconVOffsetStr.Length()) {
- idParser src(tabIconVOffsetStr, tabIconVOffsetStr.Length(), "tabiconvoffsets", LEXFL_NOFATALERRORS | LEXFL_NOSTRINGCONCAT | LEXFL_NOSTRINGESCAPECHARS);
- idToken tok;
- while (src.ReadToken(&tok)) {
- if (tok == ",") {
- continue;
- }
- tabIconVOffsets.Append(atof(tok));
- }
- }
- int c = tabStops.Num();
- bool doAligns = (tabAligns.Num() == tabStops.Num());
- for (int i = 0; i < c; i++) {
- idTabRect r;
- r.x = tabStops[i];
- r.w = (i < c - 1) ? tabStops[i+1] - r.x - tabBorder : -1;
- r.align = (doAligns) ? tabAligns[i] : 0;
- if(tabVAligns.Num() > 0) {
- r.valign = tabVAligns[i];
- } else {
- r.valign = 0;
- }
- if(tabTypes.Num() > 0) {
- r.type = tabTypes[i];
- } else {
- r.type = TAB_TYPE_TEXT;
- }
- if(tabSizes.Num() > 0) {
- r.iconSize = tabSizes[i];
- } else {
- r.iconSize.Zero();
- }
- if(tabIconVOffsets.Num() > 0 ) {
- r.iconVOffset = tabIconVOffsets[i];
- } else {
- r.iconVOffset = 0;
- }
- tabInfo.Append(r);
- }
- if (!readonly)
- flags |= WIN_CANFOCUS;
- }
- /*
- ================
- idListWindow::InitScroller
- This is the same as in idEditWindow
- ================
- */
- void idListWindow::InitScroller( bool horizontal )
- {
- const char *thumbImage = "guis/assets/scrollbar_thumb.tga";
- const char *barImage = "guis/assets/scrollbarv.tga";
- const char *scrollerName = "_scrollerWinV";
- if (horizontal) {
- barImage = "guis/assets/scrollbarh.tga";
- scrollerName = "_scrollerWinH";
- }
- const idMaterial *mat = declManager->FindMaterial( barImage );
- mat->SetSort( SS_GUI );
- sizeBias = mat->GetImageWidth();
- idRectangle scrollRect;
- if (horizontal) {
- sizeBias = mat->GetImageHeight();
- scrollRect.x = 0;
- scrollRect.y = (clientRect.h - sizeBias);
- scrollRect.w = clientRect.w;
- scrollRect.h = sizeBias;
- } else {
- scrollRect.x = (clientRect.w - sizeBias);
- scrollRect.y = 0;
- scrollRect.w = sizeBias;
- scrollRect.h = clientRect.h;
- }
- scroller->InitWithDefaults(scrollerName, scrollRect, foreColor, matColor, mat->GetName(), thumbImage, !horizontal, true);
- InsertChild(scroller, NULL);
- scroller->SetBuddy(this);
- }
- void idListWindow::Draw(int time, float x, float y) {
- idVec4 color;
- idStr work;
- int count = listItems.Num();
- idRectangle rect = textRect;
- float scale = textScale;
- float lineHeight = GetMaxCharHeight() + listgap;
- float bottom = textRect.Bottom();
- float width = textRect.w;
- if ( scroller->GetHigh() > 0.0f ) {
- if ( horizontal ) {
- bottom -= sizeBias;
- } else {
- width -= sizeBias;
- rect.w = width;
- }
- }
- if ( noEvents || !Contains(gui->CursorX(), gui->CursorY()) ) {
- hover = false;
- }
- for (int i = top; i < count; i++) {
- if ( IsSelected( i ) ) {
- rect.h = lineHeight - listgap;
- dc->DrawFilledRect(rect.x, rect.y + pixelOffset - 2, rect.w, rect.h, borderColor);
- /*
- if ( flags & WIN_FOCUS ) {
- idVec4 color = borderColor;
- color.w = 1.0f;
- dc->DrawRect(rect.x, rect.y + pixelOffset, rect.w, rect.h, 1.0f, color );
- }*/
- }
- rect.y ++;
- rect.h = lineHeight - 1;
- if ( hover && !noEvents && Contains(rect, gui->CursorX(), gui->CursorY()) ) {
- color = hoverColor;
- } else {
- color = foreColor;
- }
- rect.h = lineHeight + pixelOffset;
- rect.y --;
- if ( tabInfo.Num() > 0 ) {
- int start = 0;
- int tab = 0;
- int stop = listItems[i].Find('\t', 0);
- while ( start < listItems[i].Length() ) {
- if ( tab >= tabInfo.Num() ) {
- common->Warning( "idListWindow::Draw: gui '%s' window '%s' tabInfo.Num() exceeded", gui->GetSourceFile(), name.c_str() );
- break;
- }
- listItems[i].Mid(start, stop - start, work);
- rect.x = textRect.x + tabInfo[tab].x;
- rect.w = (tabInfo[tab].w == -1) ? width - tabInfo[tab].x : tabInfo[tab].w;
- dc->PushClipRect( rect );
- if ( tabInfo[tab].type == TAB_TYPE_TEXT ) {
- dc->DrawText(work, scale, tabInfo[tab].align, color, rect, false, -1);
- } else if (tabInfo[tab].type == TAB_TYPE_ICON) {
-
- const idMaterial **hashMat;
- const idMaterial *iconMat;
- // leaving the icon name empty doesn't draw anything
- if ( work[0] != '\0' ) {
- if ( iconMaterials.Get(work, &hashMat) == false ) {
- iconMat = declManager->FindMaterial("_default");
- } else {
- iconMat = *hashMat;
- }
- idRectangle iconRect;
- iconRect.w = tabInfo[tab].iconSize.x;
- iconRect.h = tabInfo[tab].iconSize.y;
- if(tabInfo[tab].align == idDeviceContext::ALIGN_LEFT) {
- iconRect.x = rect.x;
- } else if (tabInfo[tab].align == idDeviceContext::ALIGN_CENTER) {
- iconRect.x = rect.x + rect.w/2.0f - iconRect.w/2.0f;
- } else if (tabInfo[tab].align == idDeviceContext::ALIGN_RIGHT) {
- iconRect.x = rect.x + rect.w - iconRect.w;
- }
- if(tabInfo[tab].valign == 0) { //Top
- iconRect.y = rect.y + tabInfo[tab].iconVOffset;
- } else if(tabInfo[tab].valign == 1) { //Center
- iconRect.y = rect.y + rect.h/2.0f - iconRect.h/2.0f + tabInfo[tab].iconVOffset;
- } else if(tabInfo[tab].valign == 2) { //Bottom
- iconRect.y = rect.y + rect.h - iconRect.h + tabInfo[tab].iconVOffset;
- }
- dc->DrawMaterial(iconRect.x, iconRect.y, iconRect.w, iconRect.h, iconMat, idVec4(1.0f,1.0f,1.0f,1.0f), 1.0f, 1.0f);
- }
- }
- dc->PopClipRect();
- start = stop + 1;
- stop = listItems[i].Find('\t', start);
- if ( stop < 0 ) {
- stop = listItems[i].Length();
- }
- tab++;
- }
- rect.x = textRect.x;
- rect.w = width;
- } else {
- dc->DrawText(listItems[i], scale, 0, color, rect, false, -1);
- }
- rect.y += lineHeight;
- if ( rect.y > bottom ) {
- break;
- }
- }
- }
- void idListWindow::Activate(bool activate, idStr &act) {
- idWindow::Activate(activate, act);
- if ( activate ) {
- UpdateList( forcescroll );
- }
- }
- void idListWindow::HandleBuddyUpdate(idWindow *buddy) {
- top = scroller->GetValue();
- }
- void idListWindow::UpdateList( bool doForceScroll ) {
- idStr str, strName;
- listItems.Clear();
- for (int i = 0; i < MAX_LIST_ITEMS; i++) {
- if (gui->State().GetString( va("%s_item_%i", listName.c_str(), i), "", str) ) {
- if ( str.Length() ) {
- listItems.Append(str);
- }
- } else {
- break;
- }
- }
- float vert = GetMaxCharHeight() + listgap;
- int fit = textRect.h / vert;
- if ( listItems.Num() < fit ) {
- scroller->SetRange(0.0f, 0.0f, 1.0f);
- } else {
- scroller->SetRange(0.0f, (listItems.Num() - fit) + 1.0f, 1.0f);
- }
- if (!readonly)
- SetCurrentSel( gui->State().GetInt( va( "%s_sel_0", listName.c_str() ) ) );
- float value = scroller->GetValue();
- if ( value > listItems.Num() - 1 ) {
- value = listItems.Num() - 1;
- }
- if ( value < 0.0f ) {
- value = 0.0f;
- }
- //bc
- if (doForceScroll && listItems.Num() > DECK_MAX_LINES)
- {
- value = listItems.Num() - DECK_MAX_LINES;
- }
- scroller->SetValue(value);
- top = value;
- typedTime = 0;
- clickTime = 0;
- typed = "";
- if (listItems.Num() > 0 && GetCurrentSel() >= 0)
- gui->SetStateString( va( "%s_selected", name.c_str()), listItems[GetCurrentSel()].c_str() );
- }
- void idListWindow::StateChanged( bool redraw ) {
- UpdateList( forcescroll );
- }
|