2007-05-30 21:56:52 +00:00
|
|
|
/* ScummVM - Graphic Adventure Engine
|
|
|
|
*
|
|
|
|
* ScummVM is the legal property of its developers, whose names
|
|
|
|
* are too numerous to list here. Please refer to the COPYRIGHT
|
|
|
|
* file distributed with this source distribution.
|
2002-07-10 22:49:41 +00:00
|
|
|
*
|
|
|
|
* This program 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 2
|
|
|
|
* of the License, or (at your option) any later version.
|
|
|
|
*
|
|
|
|
* This program 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 this program; if not, write to the Free Software
|
2005-10-18 01:30:26 +00:00
|
|
|
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
|
2002-07-10 22:49:41 +00:00
|
|
|
*
|
2006-02-11 10:08:56 +00:00
|
|
|
* $URL$
|
|
|
|
* $Id$
|
2002-07-10 22:49:41 +00:00
|
|
|
*/
|
|
|
|
|
2005-10-15 15:46:12 +00:00
|
|
|
#include "common/system.h"
|
2007-06-22 07:49:02 +00:00
|
|
|
#include "common/events.h"
|
2003-11-10 23:40:48 +00:00
|
|
|
#include "gui/ListWidget.h"
|
|
|
|
#include "gui/ScrollBarWidget.h"
|
|
|
|
#include "gui/dialog.h"
|
2009-01-02 03:21:40 +00:00
|
|
|
#include "gui/GuiManager.h"
|
2002-07-10 22:49:41 +00:00
|
|
|
|
2008-08-06 13:58:07 +00:00
|
|
|
#include "gui/ThemeEval.h"
|
|
|
|
|
2003-11-10 23:40:48 +00:00
|
|
|
namespace GUI {
|
|
|
|
|
2009-06-06 17:54:28 +00:00
|
|
|
ListWidget::ListWidget(GuiObject *boss, const String &name, uint32 cmd)
|
|
|
|
: EditableWidget(boss, name), _cmd(cmd) {
|
2006-03-07 05:39:52 +00:00
|
|
|
|
2006-08-12 17:24:49 +00:00
|
|
|
_scrollBar = NULL;
|
|
|
|
_textWidth = NULL;
|
|
|
|
|
|
|
|
// This ensures that _entriesPerPage is properly initialised.
|
|
|
|
reflowLayout();
|
2006-03-07 05:39:52 +00:00
|
|
|
|
2007-12-24 03:01:21 +00:00
|
|
|
_scrollBar = new ScrollBarWidget(this, _w - _scrollBarWidth + 1, 0, _scrollBarWidth, _h);
|
2008-01-30 22:51:21 +00:00
|
|
|
_scrollBar->setTarget(this);
|
|
|
|
|
|
|
|
setFlags(WIDGET_ENABLED | WIDGET_CLEARBG | WIDGET_RETAIN_FOCUS | WIDGET_WANT_TICKLE);
|
|
|
|
_type = kListWidget;
|
|
|
|
_editMode = false;
|
|
|
|
_numberingMode = kListNumberingOne;
|
|
|
|
_currentPos = 0;
|
|
|
|
_selectedItem = -1;
|
|
|
|
_currentKeyDown = 0;
|
|
|
|
|
|
|
|
_quickSelectTime = 0;
|
|
|
|
|
|
|
|
// The item is selected, thus _bgcolor is used to draw the caret and _textcolorhi to erase it
|
|
|
|
_caretInverse = true;
|
|
|
|
|
|
|
|
// FIXME: This flag should come from widget definition
|
|
|
|
_editable = true;
|
2009-06-06 17:54:28 +00:00
|
|
|
|
|
|
|
_quickSelect = true;
|
2008-01-30 22:51:21 +00:00
|
|
|
}
|
|
|
|
|
2009-06-06 17:54:28 +00:00
|
|
|
ListWidget::ListWidget(GuiObject *boss, int x, int y, int w, int h, uint32 cmd)
|
|
|
|
: EditableWidget(boss, x, y, w, h), _cmd(cmd) {
|
2008-01-30 22:51:21 +00:00
|
|
|
|
|
|
|
_scrollBar = NULL;
|
|
|
|
_textWidth = NULL;
|
|
|
|
|
|
|
|
// This ensures that _entriesPerPage is properly initialised.
|
|
|
|
reflowLayout();
|
|
|
|
|
|
|
|
_scrollBar = new ScrollBarWidget(this, _w - _scrollBarWidth + 1, 0, _scrollBarWidth, _h);
|
2006-04-19 01:05:28 +00:00
|
|
|
_scrollBar->setTarget(this);
|
2005-07-30 21:11:48 +00:00
|
|
|
|
2007-11-04 03:38:30 +00:00
|
|
|
setFlags(WIDGET_ENABLED | WIDGET_CLEARBG | WIDGET_RETAIN_FOCUS | WIDGET_WANT_TICKLE);
|
2002-07-10 22:49:41 +00:00
|
|
|
_type = kListWidget;
|
2005-04-09 17:51:40 +00:00
|
|
|
_editMode = false;
|
2002-07-12 16:24:11 +00:00
|
|
|
_numberingMode = kListNumberingOne;
|
2002-07-18 20:26:35 +00:00
|
|
|
_currentPos = 0;
|
2002-07-13 09:19:23 +00:00
|
|
|
_selectedItem = -1;
|
2002-07-18 14:47:25 +00:00
|
|
|
_currentKeyDown = 0;
|
2005-07-30 21:11:48 +00:00
|
|
|
|
2004-07-21 14:28:57 +00:00
|
|
|
_quickSelectTime = 0;
|
2003-11-08 23:22:16 +00:00
|
|
|
|
2005-01-29 16:30:51 +00:00
|
|
|
// The item is selected, thus _bgcolor is used to draw the caret and _textcolorhi to erase it
|
|
|
|
_caretInverse = true;
|
|
|
|
|
2002-07-16 10:52:48 +00:00
|
|
|
// FIXME: This flag should come from widget definition
|
|
|
|
_editable = true;
|
2002-07-18 20:26:35 +00:00
|
|
|
}
|
2002-07-16 10:52:48 +00:00
|
|
|
|
2003-03-06 19:52:54 +00:00
|
|
|
ListWidget::~ListWidget() {
|
2006-01-28 18:21:46 +00:00
|
|
|
delete[] _textWidth;
|
2002-07-18 20:26:35 +00:00
|
|
|
}
|
2002-07-12 16:24:11 +00:00
|
|
|
|
2006-05-27 06:38:45 +00:00
|
|
|
Widget *ListWidget::findWidget(int x, int y) {
|
|
|
|
if (x >= _w - _scrollBarWidth)
|
|
|
|
return _scrollBar;
|
|
|
|
|
|
|
|
return this;
|
|
|
|
}
|
|
|
|
|
2004-10-01 21:12:18 +00:00
|
|
|
void ListWidget::setSelected(int item) {
|
|
|
|
assert(item >= -1 && item < (int)_list.size());
|
|
|
|
|
|
|
|
if (isEnabled() && _selectedItem != item) {
|
2005-01-29 16:30:51 +00:00
|
|
|
if (_editMode)
|
|
|
|
abortEditMode();
|
2004-10-01 21:12:18 +00:00
|
|
|
|
2005-01-29 16:30:51 +00:00
|
|
|
_selectedItem = item;
|
2004-10-01 21:12:18 +00:00
|
|
|
sendCommand(kListSelectionChangedCmd, _selectedItem);
|
|
|
|
|
|
|
|
_currentPos = _selectedItem - _entriesPerPage / 2;
|
|
|
|
scrollToCurrent();
|
|
|
|
draw();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2003-03-06 19:52:54 +00:00
|
|
|
void ListWidget::setList(const StringList &list) {
|
2002-10-23 14:00:47 +00:00
|
|
|
if (_editMode && _caretVisible)
|
|
|
|
drawCaret(true);
|
2002-10-19 01:22:41 +00:00
|
|
|
int size = list.size();
|
2009-06-06 17:54:08 +00:00
|
|
|
_dataList = list;
|
|
|
|
|
|
|
|
// Copy everything
|
2002-10-19 01:22:41 +00:00
|
|
|
_list = list;
|
2009-06-06 17:54:08 +00:00
|
|
|
_filter = "";
|
|
|
|
|
2002-10-19 01:22:41 +00:00
|
|
|
if (_currentPos >= size)
|
|
|
|
_currentPos = size - 1;
|
2002-11-15 17:38:50 +00:00
|
|
|
if (_currentPos < 0)
|
|
|
|
_currentPos = 0;
|
2002-10-19 01:22:41 +00:00
|
|
|
_selectedItem = -1;
|
|
|
|
_editMode = false;
|
2005-10-15 15:46:12 +00:00
|
|
|
g_system->setFeatureState(OSystem::kFeatureVirtualKeyboard, false);
|
2002-10-19 01:22:41 +00:00
|
|
|
scrollBarRecalc();
|
|
|
|
}
|
|
|
|
|
2009-06-06 17:49:59 +00:00
|
|
|
void ListWidget::append(const String &s) {
|
2009-06-06 17:54:08 +00:00
|
|
|
_dataList.push_back(s);
|
2009-06-06 17:49:59 +00:00
|
|
|
_list.push_back(s);
|
|
|
|
scrollBarRecalc();
|
|
|
|
}
|
|
|
|
|
2003-03-06 19:52:54 +00:00
|
|
|
void ListWidget::scrollTo(int item) {
|
2002-11-14 14:42:39 +00:00
|
|
|
int size = _list.size();
|
|
|
|
if (item >= size)
|
|
|
|
item = size - 1;
|
2002-11-15 17:38:50 +00:00
|
|
|
if (item < 0)
|
2002-11-14 14:42:39 +00:00
|
|
|
item = 0;
|
|
|
|
|
|
|
|
if (_currentPos != item) {
|
|
|
|
_currentPos = item;
|
|
|
|
scrollBarRecalc();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2003-03-06 19:52:54 +00:00
|
|
|
void ListWidget::scrollBarRecalc() {
|
2002-07-12 16:24:11 +00:00
|
|
|
_scrollBar->_numEntries = _list.size();
|
|
|
|
_scrollBar->_entriesPerPage = _entriesPerPage;
|
|
|
|
_scrollBar->_currentPos = _currentPos;
|
|
|
|
_scrollBar->recalc();
|
2002-07-10 22:49:41 +00:00
|
|
|
}
|
|
|
|
|
2003-03-06 19:52:54 +00:00
|
|
|
void ListWidget::handleTickle() {
|
2005-01-29 16:30:51 +00:00
|
|
|
if (_editMode)
|
|
|
|
EditableWidget::handleTickle();
|
2002-10-23 14:00:47 +00:00
|
|
|
}
|
|
|
|
|
2003-03-06 19:52:54 +00:00
|
|
|
void ListWidget::handleMouseDown(int x, int y, int button, int clickCount) {
|
2005-01-29 16:30:51 +00:00
|
|
|
if (!isEnabled())
|
|
|
|
return;
|
|
|
|
|
|
|
|
// First check whether the selection changed
|
2006-06-24 13:21:46 +00:00
|
|
|
int newSelectedItem = findItem(x, y);
|
2005-01-29 16:30:51 +00:00
|
|
|
|
2006-06-24 13:21:46 +00:00
|
|
|
if (_selectedItem != newSelectedItem && newSelectedItem != -1) {
|
2005-01-29 16:30:51 +00:00
|
|
|
if (_editMode)
|
|
|
|
abortEditMode();
|
|
|
|
_selectedItem = newSelectedItem;
|
|
|
|
sendCommand(kListSelectionChangedCmd, _selectedItem);
|
2002-07-10 22:49:41 +00:00
|
|
|
}
|
2005-07-30 21:11:48 +00:00
|
|
|
|
2005-01-29 16:30:51 +00:00
|
|
|
// TODO: Determine where inside the string the user clicked and place the
|
2006-06-24 13:21:46 +00:00
|
|
|
// caret accordingly.
|
|
|
|
// See _editScrollOffset and EditTextWidget::handleMouseDown.
|
2005-01-29 16:30:51 +00:00
|
|
|
draw();
|
|
|
|
|
2002-07-10 22:49:41 +00:00
|
|
|
}
|
|
|
|
|
2003-03-06 19:52:54 +00:00
|
|
|
void ListWidget::handleMouseUp(int x, int y, int button, int clickCount) {
|
2007-09-19 08:40:12 +00:00
|
|
|
// If this was a double click and the mouse is still over
|
2006-06-24 13:21:46 +00:00
|
|
|
// the selected item, send the double click command
|
|
|
|
if (clickCount == 2 && (_selectedItem == findItem(x, y)) &&
|
|
|
|
_selectedItem >= 0) {
|
2002-07-27 14:16:14 +00:00
|
|
|
sendCommand(kListItemDoubleClickedCmd, _selectedItem);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2003-03-06 19:52:54 +00:00
|
|
|
void ListWidget::handleMouseWheel(int x, int y, int direction) {
|
2002-10-16 20:32:12 +00:00
|
|
|
_scrollBar->handleMouseWheel(x, y, direction);
|
|
|
|
}
|
|
|
|
|
2004-07-21 14:28:57 +00:00
|
|
|
|
2005-03-12 15:29:16 +00:00
|
|
|
int ListWidget::findItem(int x, int y) const {
|
2006-06-24 13:21:46 +00:00
|
|
|
if (y < _topPadding) return -1;
|
|
|
|
int item = (y - _topPadding) / kLineHeight + _currentPos;
|
2007-09-19 08:40:12 +00:00
|
|
|
if (item >= _currentPos && item < _currentPos + _entriesPerPage &&
|
2006-06-24 13:21:46 +00:00
|
|
|
item < (int)_list.size())
|
|
|
|
return item;
|
|
|
|
else
|
|
|
|
return -1;
|
2005-03-12 15:29:16 +00:00
|
|
|
}
|
|
|
|
|
2004-07-21 14:28:57 +00:00
|
|
|
static int matchingCharsIgnoringCase(const char *x, const char *y, bool &stop) {
|
|
|
|
int match = 0;
|
2009-05-11 20:53:28 +00:00
|
|
|
while (*x && *y && tolower(*x) == tolower(*y)) {
|
2004-07-21 14:28:57 +00:00
|
|
|
++x;
|
|
|
|
++y;
|
|
|
|
++match;
|
|
|
|
}
|
2009-05-11 20:53:28 +00:00
|
|
|
stop = !*y || (*x && (tolower(*x) >= tolower(*y)));
|
2004-07-21 14:28:57 +00:00
|
|
|
return match;
|
|
|
|
}
|
|
|
|
|
2007-06-30 12:26:59 +00:00
|
|
|
bool ListWidget::handleKeyDown(Common::KeyState state) {
|
2002-07-27 14:16:14 +00:00
|
|
|
bool handled = true;
|
2002-07-16 10:52:48 +00:00
|
|
|
bool dirty = false;
|
|
|
|
int oldSelectedItem = _selectedItem;
|
|
|
|
|
2007-06-30 12:26:59 +00:00
|
|
|
if (!_editMode && isprint((char)state.ascii)) {
|
2004-07-21 14:28:57 +00:00
|
|
|
// Quick selection mode: Go to first list item starting with this key
|
|
|
|
// (or a substring accumulated from the last couple key presses).
|
|
|
|
// Only works in a useful fashion if the list entries are sorted.
|
2005-01-10 22:06:49 +00:00
|
|
|
uint32 time = getMillis();
|
2004-07-21 14:28:57 +00:00
|
|
|
if (_quickSelectTime < time) {
|
2007-06-30 12:26:59 +00:00
|
|
|
_quickSelectStr = (char)state.ascii;
|
2004-07-21 14:28:57 +00:00
|
|
|
} else {
|
2007-06-30 12:26:59 +00:00
|
|
|
_quickSelectStr += (char)state.ascii;
|
2004-07-21 14:28:57 +00:00
|
|
|
}
|
|
|
|
_quickSelectTime = time + 300; // TODO: Turn this into a proper constant (kQuickSelectDelay ?)
|
2005-07-30 21:11:48 +00:00
|
|
|
|
2009-06-06 17:54:28 +00:00
|
|
|
if (_quickSelect) {
|
|
|
|
// FIXME: This is bad slow code (it scans the list linearly each time a
|
|
|
|
// key is pressed); it could be much faster. Only of importance if we have
|
|
|
|
// quite big lists to deal with -- so for now we can live with this lazy
|
|
|
|
// implementation :-)
|
|
|
|
int newSelectedItem = 0;
|
|
|
|
int bestMatch = 0;
|
|
|
|
bool stop;
|
|
|
|
for (StringList::const_iterator i = _list.begin(); i != _list.end(); ++i) {
|
|
|
|
const int match = matchingCharsIgnoringCase(i->c_str(), _quickSelectStr.c_str(), stop);
|
|
|
|
if (match > bestMatch || stop) {
|
|
|
|
_selectedItem = newSelectedItem;
|
|
|
|
bestMatch = match;
|
|
|
|
if (stop)
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
newSelectedItem++;
|
2004-07-21 14:28:57 +00:00
|
|
|
}
|
|
|
|
|
2009-06-06 17:54:28 +00:00
|
|
|
scrollToCurrent();
|
|
|
|
} else {
|
|
|
|
sendCommand(_cmd, 0);
|
|
|
|
}
|
2004-07-21 14:28:57 +00:00
|
|
|
} else if (_editMode) {
|
2005-01-29 18:04:34 +00:00
|
|
|
// Class EditableWidget handles all text editing related key presses for us
|
2007-06-30 12:26:59 +00:00
|
|
|
handled = EditableWidget::handleKeyDown(state);
|
2002-07-16 10:52:48 +00:00
|
|
|
} else {
|
|
|
|
// not editmode
|
|
|
|
|
2007-06-30 12:26:59 +00:00
|
|
|
switch (state.keycode) {
|
2007-06-22 07:49:02 +00:00
|
|
|
case Common::KEYCODE_RETURN:
|
|
|
|
case Common::KEYCODE_KP_ENTER:
|
2002-07-16 10:52:48 +00:00
|
|
|
if (_selectedItem >= 0) {
|
2002-09-30 12:56:59 +00:00
|
|
|
// override continuous enter keydown
|
2007-06-22 07:49:02 +00:00
|
|
|
if (_editable && (_currentKeyDown != Common::KEYCODE_RETURN && _currentKeyDown != Common::KEYCODE_KP_ENTER)) {
|
2002-07-18 14:47:25 +00:00
|
|
|
dirty = true;
|
2005-01-29 16:30:51 +00:00
|
|
|
startEditMode();
|
2003-11-18 23:44:28 +00:00
|
|
|
} else
|
|
|
|
sendCommand(kListItemActivatedCmd, _selectedItem);
|
2002-07-16 10:52:48 +00:00
|
|
|
}
|
|
|
|
break;
|
2007-07-17 21:59:44 +00:00
|
|
|
case Common::KEYCODE_BACKSPACE:
|
|
|
|
case Common::KEYCODE_KP_PERIOD:
|
|
|
|
case Common::KEYCODE_DELETE:
|
|
|
|
if (_selectedItem >= 0) {
|
|
|
|
if (_editable) {
|
|
|
|
// Ignore delete and backspace when the list item is editable
|
|
|
|
} else {
|
|
|
|
sendCommand(kListItemRemovalRequestCmd, _selectedItem);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
break;
|
2007-06-22 07:49:02 +00:00
|
|
|
case Common::KEYCODE_UP:
|
2002-07-16 10:52:48 +00:00
|
|
|
if (_selectedItem > 0)
|
|
|
|
_selectedItem--;
|
|
|
|
break;
|
2007-06-22 07:49:02 +00:00
|
|
|
case Common::KEYCODE_DOWN:
|
2004-02-05 00:19:57 +00:00
|
|
|
if (_selectedItem < (int)_list.size() - 1)
|
2002-07-16 10:52:48 +00:00
|
|
|
_selectedItem++;
|
|
|
|
break;
|
2007-06-22 07:49:02 +00:00
|
|
|
case Common::KEYCODE_PAGEUP:
|
2002-07-16 10:52:48 +00:00
|
|
|
_selectedItem -= _entriesPerPage - 1;
|
|
|
|
if (_selectedItem < 0)
|
|
|
|
_selectedItem = 0;
|
|
|
|
break;
|
2007-06-22 07:49:02 +00:00
|
|
|
case Common::KEYCODE_PAGEDOWN:
|
2002-07-16 10:52:48 +00:00
|
|
|
_selectedItem += _entriesPerPage - 1;
|
2004-02-05 00:19:57 +00:00
|
|
|
if (_selectedItem >= (int)_list.size() )
|
2002-07-16 10:52:48 +00:00
|
|
|
_selectedItem = _list.size() - 1;
|
|
|
|
break;
|
2007-06-22 07:49:02 +00:00
|
|
|
case Common::KEYCODE_HOME:
|
2002-07-16 10:52:48 +00:00
|
|
|
_selectedItem = 0;
|
|
|
|
break;
|
2007-06-22 07:49:02 +00:00
|
|
|
case Common::KEYCODE_END:
|
2002-07-16 10:52:48 +00:00
|
|
|
_selectedItem = _list.size() - 1;
|
|
|
|
break;
|
2002-07-27 14:16:14 +00:00
|
|
|
default:
|
|
|
|
handled = false;
|
2002-07-16 10:52:48 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
scrollToCurrent();
|
|
|
|
}
|
|
|
|
|
|
|
|
if (dirty || _selectedItem != oldSelectedItem)
|
|
|
|
draw();
|
|
|
|
|
|
|
|
if (_selectedItem != oldSelectedItem) {
|
2002-10-01 23:11:19 +00:00
|
|
|
sendCommand(kListSelectionChangedCmd, _selectedItem);
|
2002-07-16 10:52:48 +00:00
|
|
|
// also draw scrollbar
|
|
|
|
_scrollBar->draw();
|
|
|
|
}
|
|
|
|
|
2005-09-03 16:24:58 +00:00
|
|
|
#if !defined(PALMOS_MODE)
|
2004-01-26 07:35:31 +00:00
|
|
|
// not done on PalmOS because keyboard is emulated and keyup is not generated
|
2007-06-30 12:26:59 +00:00
|
|
|
_currentKeyDown = state.keycode;
|
2002-10-22 22:35:07 +00:00
|
|
|
#endif
|
2003-03-06 19:52:54 +00:00
|
|
|
|
2002-07-27 14:16:14 +00:00
|
|
|
return handled;
|
2002-07-18 14:47:25 +00:00
|
|
|
}
|
|
|
|
|
2007-06-30 12:26:59 +00:00
|
|
|
bool ListWidget::handleKeyUp(Common::KeyState state) {
|
|
|
|
if (state.keycode == _currentKeyDown)
|
2002-07-18 14:47:25 +00:00
|
|
|
_currentKeyDown = 0;
|
2002-08-04 16:30:59 +00:00
|
|
|
return true;
|
2002-07-16 10:52:48 +00:00
|
|
|
}
|
|
|
|
|
2003-03-06 19:52:54 +00:00
|
|
|
void ListWidget::lostFocusWidget() {
|
2005-01-29 16:30:51 +00:00
|
|
|
// If we loose focus, we simply forget the user changes
|
2002-10-23 14:00:47 +00:00
|
|
|
_editMode = false;
|
2005-10-15 15:46:12 +00:00
|
|
|
g_system->setFeatureState(OSystem::kFeatureVirtualKeyboard, false);
|
2002-10-23 14:00:47 +00:00
|
|
|
drawCaret(true);
|
2002-07-16 10:52:48 +00:00
|
|
|
draw();
|
2002-07-10 22:49:41 +00:00
|
|
|
}
|
|
|
|
|
2003-03-06 19:52:54 +00:00
|
|
|
void ListWidget::handleCommand(CommandSender *sender, uint32 cmd, uint32 data) {
|
2002-07-12 16:24:11 +00:00
|
|
|
switch (cmd) {
|
|
|
|
case kSetPositionCmd:
|
2002-08-24 21:57:00 +00:00
|
|
|
if (_currentPos != (int)data) {
|
2002-07-12 16:24:11 +00:00
|
|
|
_currentPos = data;
|
|
|
|
draw();
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
2002-07-10 22:49:41 +00:00
|
|
|
}
|
|
|
|
|
2007-11-04 03:38:30 +00:00
|
|
|
void ListWidget::drawWidget() {
|
2003-03-06 19:52:54 +00:00
|
|
|
int i, pos, len = _list.size();
|
2003-10-02 17:43:02 +00:00
|
|
|
Common::String buffer;
|
2002-07-10 22:49:41 +00:00
|
|
|
|
2002-09-08 16:00:13 +00:00
|
|
|
// Draw a thin frame around the list.
|
2008-11-10 11:24:55 +00:00
|
|
|
g_gui.theme()->drawWidgetBackground(Common::Rect(_x, _y, _x + _w, _y + _h), 0, ThemeEngine::kWidgetBackgroundBorder);
|
2008-08-10 22:03:00 +00:00
|
|
|
const int scrollbarW = (_scrollBar && _scrollBar->isVisible()) ? _scrollBarWidth : 0;
|
2002-09-08 16:00:13 +00:00
|
|
|
|
2002-07-12 16:24:11 +00:00
|
|
|
// Draw the list items
|
2002-07-18 20:26:35 +00:00
|
|
|
for (i = 0, pos = _currentPos; i < _entriesPerPage && pos < len; i++, pos++) {
|
2006-04-16 10:23:36 +00:00
|
|
|
const int y = _y + _topPadding + kLineHeight * i;
|
2006-01-27 15:43:23 +00:00
|
|
|
const int fontHeight = kLineHeight;
|
|
|
|
bool inverted = false;
|
2005-01-29 18:04:34 +00:00
|
|
|
|
|
|
|
// Draw the selected item inverted, on a highlighted background.
|
|
|
|
if (_selectedItem == pos) {
|
|
|
|
if (_hasFocus)
|
2006-01-27 15:43:23 +00:00
|
|
|
inverted = true;
|
2005-01-29 18:04:34 +00:00
|
|
|
else
|
2008-12-22 11:22:15 +00:00
|
|
|
g_gui.theme()->drawWidgetBackground(Common::Rect(_x, y - 1, _x + _w - 1, y + fontHeight - 1),
|
2008-11-10 11:24:55 +00:00
|
|
|
0, ThemeEngine::kWidgetBackgroundBorderSmall);
|
2005-01-29 18:04:34 +00:00
|
|
|
}
|
|
|
|
|
2006-01-28 18:21:46 +00:00
|
|
|
Common::Rect r(getEditRect());
|
2006-04-16 10:23:36 +00:00
|
|
|
int pad = _leftPadding;
|
2006-01-28 18:21:46 +00:00
|
|
|
|
2005-01-29 18:04:34 +00:00
|
|
|
// If in numbering mode, we first print a number prefix
|
2005-01-29 16:30:51 +00:00
|
|
|
if (_numberingMode != kListNumberingOff) {
|
2002-07-12 16:24:11 +00:00
|
|
|
char temp[10];
|
|
|
|
sprintf(temp, "%2d. ", (pos + _numberingMode));
|
|
|
|
buffer = temp;
|
2008-12-22 11:22:15 +00:00
|
|
|
g_gui.theme()->drawText(Common::Rect(_x, y, _x + r.left + _leftPadding, y + fontHeight - 2),
|
2009-03-11 20:39:31 +00:00
|
|
|
buffer, _state, Graphics::kTextAlignLeft, inverted, _leftPadding, true);
|
2006-04-16 10:23:36 +00:00
|
|
|
pad = 0;
|
2005-01-29 16:30:51 +00:00
|
|
|
}
|
2002-07-18 20:26:35 +00:00
|
|
|
|
2006-01-28 18:21:46 +00:00
|
|
|
int width;
|
2005-01-29 18:04:34 +00:00
|
|
|
|
2006-01-28 18:21:46 +00:00
|
|
|
if (_selectedItem == pos && _editMode) {
|
2005-01-29 18:04:34 +00:00
|
|
|
buffer = _editString;
|
|
|
|
adjustOffset();
|
2008-08-10 22:03:00 +00:00
|
|
|
width = _w - r.left - _hlRightPadding - _leftPadding - scrollbarW;
|
2009-03-11 20:39:31 +00:00
|
|
|
g_gui.theme()->drawText(Common::Rect(_x + r.left, y, _x + r.left + width, y + fontHeight - 2),
|
|
|
|
buffer, _state, Graphics::kTextAlignLeft, inverted, pad, true);
|
2005-01-29 18:04:34 +00:00
|
|
|
} else {
|
|
|
|
buffer = _list[pos];
|
2009-03-11 20:39:31 +00:00
|
|
|
width = _w - r.left - scrollbarW;
|
|
|
|
g_gui.theme()->drawText(Common::Rect(_x + r.left, y, _x + r.left + width, y + fontHeight - 2),
|
2009-05-24 15:17:42 +00:00
|
|
|
buffer, _state, Graphics::kTextAlignLeft, inverted, pad, true);
|
2002-09-08 16:00:13 +00:00
|
|
|
}
|
2006-01-28 18:21:46 +00:00
|
|
|
|
|
|
|
_textWidth[i] = width;
|
2002-07-16 10:52:48 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2005-01-29 16:30:51 +00:00
|
|
|
Common::Rect ListWidget::getEditRect() const {
|
2006-04-20 08:14:55 +00:00
|
|
|
Common::Rect r(_hlLeftPadding, 0, _w - _hlLeftPadding - _hlRightPadding, kLineHeight - 1);
|
|
|
|
const int offset = (_selectedItem - _currentPos) * kLineHeight + _topPadding;
|
2005-01-29 16:30:51 +00:00
|
|
|
r.top += offset;
|
|
|
|
r.bottom += offset;
|
2003-11-18 23:44:28 +00:00
|
|
|
|
2005-01-29 16:30:51 +00:00
|
|
|
if (_numberingMode != kListNumberingOff) {
|
2003-11-18 23:44:28 +00:00
|
|
|
char temp[10];
|
2005-05-16 13:28:34 +00:00
|
|
|
// FIXME: Assumes that all digits have the same width.
|
|
|
|
sprintf(temp, "%2d. ", (_list.size() - 1 + _numberingMode));
|
2006-04-16 10:23:36 +00:00
|
|
|
r.left += g_gui.getStringWidth(temp) + _leftPadding;
|
2003-11-18 23:44:28 +00:00
|
|
|
}
|
2005-07-30 21:11:48 +00:00
|
|
|
|
2005-01-29 16:30:51 +00:00
|
|
|
return r;
|
2003-11-18 23:44:28 +00:00
|
|
|
}
|
|
|
|
|
2002-07-16 10:52:48 +00:00
|
|
|
void ListWidget::scrollToCurrent() {
|
|
|
|
// Only do something if the current item is not in our view port
|
|
|
|
if (_selectedItem < _currentPos) {
|
|
|
|
// it's above our view
|
|
|
|
_currentPos = _selectedItem;
|
|
|
|
} else if (_selectedItem >= _currentPos + _entriesPerPage ) {
|
|
|
|
// it's below our view
|
|
|
|
_currentPos = _selectedItem - _entriesPerPage + 1;
|
2002-07-12 16:24:11 +00:00
|
|
|
}
|
2002-07-16 10:52:48 +00:00
|
|
|
|
2004-10-06 22:19:49 +00:00
|
|
|
if (_currentPos < 0 || _entriesPerPage > (int)_list.size())
|
2002-09-09 20:28:48 +00:00
|
|
|
_currentPos = 0;
|
2004-10-01 21:12:18 +00:00
|
|
|
else if (_currentPos + _entriesPerPage > (int)_list.size())
|
|
|
|
_currentPos = _list.size() - _entriesPerPage;
|
2002-09-09 20:28:48 +00:00
|
|
|
|
2002-07-16 10:52:48 +00:00
|
|
|
_scrollBar->_currentPos = _currentPos;
|
|
|
|
_scrollBar->recalc();
|
2002-07-10 22:49:41 +00:00
|
|
|
}
|
2002-10-19 01:22:41 +00:00
|
|
|
|
2009-06-06 17:49:59 +00:00
|
|
|
void ListWidget::scrollToEnd() {
|
|
|
|
if (_currentPos + _entriesPerPage < (int)_list.size()) {
|
|
|
|
_currentPos = _list.size() - _entriesPerPage;
|
|
|
|
} else {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
_scrollBar->_currentPos = _currentPos;
|
|
|
|
_scrollBar->recalc();
|
|
|
|
_scrollBar->draw();
|
|
|
|
}
|
|
|
|
|
2003-03-06 19:52:54 +00:00
|
|
|
void ListWidget::startEditMode() {
|
2002-10-19 01:22:41 +00:00
|
|
|
if (_editable && !_editMode && _selectedItem >= 0) {
|
|
|
|
_editMode = true;
|
2005-01-29 18:04:34 +00:00
|
|
|
setEditString(_list[_selectedItem]);
|
2002-10-19 01:22:41 +00:00
|
|
|
draw();
|
2005-10-15 15:46:12 +00:00
|
|
|
g_system->setFeatureState(OSystem::kFeatureVirtualKeyboard, true);
|
2002-10-19 01:22:41 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2005-01-29 16:30:51 +00:00
|
|
|
void ListWidget::endEditMode() {
|
2005-05-08 21:49:52 +00:00
|
|
|
if (!_editMode)
|
2005-03-27 11:27:07 +00:00
|
|
|
return;
|
2005-01-29 16:30:51 +00:00
|
|
|
// send a message that editing finished with a return/enter key press
|
|
|
|
_editMode = false;
|
|
|
|
_list[_selectedItem] = _editString;
|
2005-10-15 15:46:12 +00:00
|
|
|
g_system->setFeatureState(OSystem::kFeatureVirtualKeyboard, false);
|
2005-01-29 16:30:51 +00:00
|
|
|
sendCommand(kListItemActivatedCmd, _selectedItem);
|
|
|
|
}
|
|
|
|
|
2003-03-06 19:52:54 +00:00
|
|
|
void ListWidget::abortEditMode() {
|
2005-01-29 16:30:51 +00:00
|
|
|
// undo any changes made
|
|
|
|
assert(_selectedItem >= 0);
|
|
|
|
_editMode = false;
|
|
|
|
//drawCaret(true);
|
|
|
|
//draw();
|
2005-10-15 15:46:12 +00:00
|
|
|
g_system->setFeatureState(OSystem::kFeatureVirtualKeyboard, false);
|
2002-10-19 01:22:41 +00:00
|
|
|
}
|
2003-11-10 23:40:48 +00:00
|
|
|
|
2006-08-04 13:55:53 +00:00
|
|
|
void ListWidget::reflowLayout() {
|
|
|
|
Widget::reflowLayout();
|
2006-04-19 01:05:28 +00:00
|
|
|
|
2008-08-06 13:58:07 +00:00
|
|
|
_leftPadding = g_gui.xmlEval()->getVar("Globals.ListWidget.Padding.Left", 0);
|
|
|
|
_rightPadding = g_gui.xmlEval()->getVar("Globals.ListWidget.Padding.Right", 0);
|
|
|
|
_topPadding = g_gui.xmlEval()->getVar("Globals.ListWidget.Padding.Top", 0);
|
|
|
|
_bottomPadding = g_gui.xmlEval()->getVar("Globals.ListWidget.Padding.Bottom", 0);
|
|
|
|
_hlLeftPadding = g_gui.xmlEval()->getVar("Globals.ListWidget.hlLeftPadding", 0);
|
|
|
|
_hlRightPadding = g_gui.xmlEval()->getVar("Globals.ListWidget.hlRightPadding", 0);
|
2006-04-19 01:05:28 +00:00
|
|
|
|
2008-11-12 13:07:41 +00:00
|
|
|
_scrollBarWidth = g_gui.xmlEval()->getVar("Globals.Scrollbar.Width", 0);
|
2006-04-19 01:05:28 +00:00
|
|
|
|
2008-08-08 15:06:28 +00:00
|
|
|
// HACK: Once we take padding into account, there are times where
|
|
|
|
// integer rounding leaves a big chunk of white space in the bottom
|
|
|
|
// of the list.
|
|
|
|
// We do a rough rounding on the decimal places of Entries Per Page,
|
|
|
|
// to add another entry even if it goes a tad over the padding.
|
|
|
|
_entriesPerPage = ((_h - _topPadding - _bottomPadding) << 16) / kLineHeight;
|
2008-12-22 11:22:15 +00:00
|
|
|
|
2008-08-08 15:06:28 +00:00
|
|
|
if ((uint)(_entriesPerPage & 0xFFFF) >= 0xF000)
|
|
|
|
_entriesPerPage += (1 << 16);
|
|
|
|
|
|
|
|
_entriesPerPage >>= 16;
|
2008-12-22 11:22:15 +00:00
|
|
|
|
2008-08-09 22:40:05 +00:00
|
|
|
assert(_entriesPerPage > 0);
|
2006-04-19 01:05:28 +00:00
|
|
|
|
2008-05-07 14:31:45 +00:00
|
|
|
delete[] _textWidth;
|
2006-04-19 01:05:28 +00:00
|
|
|
_textWidth = new int[_entriesPerPage];
|
|
|
|
|
|
|
|
for (int i = 0; i < _entriesPerPage; i++)
|
|
|
|
_textWidth[i] = 0;
|
|
|
|
|
2006-08-12 17:24:49 +00:00
|
|
|
if (_scrollBar) {
|
2007-12-24 03:01:21 +00:00
|
|
|
_scrollBar->resize(_w - _scrollBarWidth + 1, 0, _scrollBarWidth, _h);
|
2006-08-12 17:24:49 +00:00
|
|
|
scrollBarRecalc();
|
|
|
|
scrollToCurrent();
|
|
|
|
}
|
2006-04-19 01:05:28 +00:00
|
|
|
}
|
|
|
|
|
2009-06-06 17:54:08 +00:00
|
|
|
void ListWidget::setFilter(const String &filter) {
|
|
|
|
String filt;
|
|
|
|
|
|
|
|
filt = filter;
|
|
|
|
|
|
|
|
filt.toLowercase();
|
|
|
|
_filter = filt;
|
|
|
|
|
|
|
|
if (_filter == "") {
|
|
|
|
_list = _dataList;
|
|
|
|
} else {
|
|
|
|
String tmp;
|
|
|
|
|
|
|
|
_list.clear();
|
|
|
|
|
|
|
|
for (StringList::iterator i = _dataList.begin(); i != _dataList.end(); ++i) {
|
|
|
|
tmp = *i;
|
|
|
|
tmp.toLowercase();
|
|
|
|
if (tmp.contains(_filter.c_str())) {
|
|
|
|
_list.push_back(*i);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
_currentPos = 0;
|
|
|
|
_selectedItem = -1;
|
|
|
|
scrollBarRecalc();
|
|
|
|
|
|
|
|
draw();
|
|
|
|
}
|
|
|
|
|
2003-11-10 23:40:48 +00:00
|
|
|
} // End of namespace GUI
|