mirror of
https://github.com/scummvm/scummvm.git
synced 2026-05-21 05:40:43 +00:00
02b6e11e1c
Turns out my initial fix for bug #16698 was only enough to keep the search result in view. Clicking on it didn't work. Now it should.
609 lines
18 KiB
C++
609 lines
18 KiB
C++
/* 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.
|
|
*
|
|
* 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 3 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, see <http://www.gnu.org/licenses/>.
|
|
*
|
|
*/
|
|
|
|
#include "common/system.h"
|
|
#include "common/frac.h"
|
|
#include "common/tokenizer.h"
|
|
#include "common/translation.h"
|
|
#include "common/config-manager.h"
|
|
|
|
#include "gui/widgets/groupedlist.h"
|
|
#include "gui/widgets/scrollbar.h"
|
|
#include "gui/dialog.h"
|
|
#include "gui/gui-manager.h"
|
|
#include "gui/animation/FluidScroll.h"
|
|
|
|
#include "gui/ThemeEval.h"
|
|
|
|
#define kGroupTag -2
|
|
#define isGroupHeader(x) (_groupsVisible && ((x) <= kGroupTag))
|
|
#define indexToGroupID(x) (kGroupTag - x)
|
|
|
|
namespace GUI {
|
|
|
|
GroupedListWidget::GroupedListWidget(Dialog *boss, const Common::String &name, const Common::U32String &tooltip, uint32 cmd)
|
|
: ListWidget(boss, name, tooltip, cmd) {
|
|
_groupsVisible = true;
|
|
}
|
|
|
|
void GroupedListWidget::setList(const Common::U32StringArray &list) {
|
|
ListWidget::setList(list);
|
|
|
|
_attributeValues.clear(); // Regenerate attributes for the new list
|
|
groupByAttribute();
|
|
scrollBarRecalc();
|
|
}
|
|
|
|
void GroupedListWidget::setAttributeValues(const Common::U32StringArray &attrValues) {
|
|
_attributeValues = attrValues;
|
|
// Make sure we always have the attribute values for all the entries of the _dataList.
|
|
// This is not foolproof, but can prevent accidentally passing attributes for the wrong
|
|
// list of entries, such as after adding / removing games.
|
|
if (!attrValues.empty())
|
|
assert(_attributeValues.size() == _dataList.size());
|
|
}
|
|
|
|
void GroupedListWidget::setMetadataNames(const Common::StringMap &metadata) {
|
|
_metadataNames = metadata;
|
|
}
|
|
|
|
void GroupedListWidget::setGroupHeaderFormat(const Common::U32String &prefix, const Common::U32String &suffix) {
|
|
_groupHeaderPrefix = prefix;
|
|
_groupHeaderSuffix = suffix;
|
|
}
|
|
|
|
void GroupedListWidget::groupByAttribute() {
|
|
_groupExpanded.clear();
|
|
_groupHeaders.clear();
|
|
_groupValueIndex.clear();
|
|
_itemsInGroup.clear();
|
|
|
|
if (_attributeValues.empty()) {
|
|
_groupExpanded.push_back(true);
|
|
// I18N: Group for All items
|
|
_groupHeaders.push_back(_("All"));
|
|
_groupValueIndex.setVal(Common::String("All"), 0);
|
|
for (uint i = 0; i < _dataList.size(); ++i) {
|
|
_itemsInGroup[0].push_back(i);
|
|
}
|
|
sortGroups();
|
|
return;
|
|
}
|
|
|
|
for (uint i = 0; i < _dataList.size(); ++i) {
|
|
Common::U32StringArray::iterator attrVal = _attributeValues.begin() + i;
|
|
if (!_groupValueIndex.contains(*attrVal)) {
|
|
int newGroupID = _groupValueIndex.size();
|
|
_groupValueIndex.setVal(*attrVal, newGroupID);
|
|
_groupHeaders.push_back(*attrVal);
|
|
_groupExpanded.push_back(true);
|
|
}
|
|
int groupID = _groupValueIndex.getVal(*attrVal);
|
|
|
|
_itemsInGroup[groupID].push_back(i);
|
|
}
|
|
|
|
sortGroups();
|
|
}
|
|
|
|
void GroupedListWidget::sortGroups() {
|
|
uint oldListSize = _list.size();
|
|
_list.clear();
|
|
_listIndex.clear();
|
|
|
|
Common::sort(_groupHeaders.begin(), _groupHeaders.end(),
|
|
[](const Common::String &first, const Common::String &second) {
|
|
return first.empty() ? 0 : second.empty() ? 1 : first < second;
|
|
}
|
|
|
|
);
|
|
|
|
for (uint i = 0; i != _groupHeaders.size(); ++i) {
|
|
Common::U32String header = _groupHeaders[i];
|
|
Common::U32String displayedHeader;
|
|
if (_metadataNames.contains(header)) {
|
|
displayedHeader = _metadataNames[header];
|
|
} else {
|
|
displayedHeader = header;
|
|
}
|
|
uint groupID = _groupValueIndex[header];
|
|
|
|
if (_groupsVisible) {
|
|
_listIndex.push_back(kGroupTag - groupID);
|
|
|
|
displayedHeader.toUppercase();
|
|
|
|
_list.push_back(_groupHeaderPrefix + displayedHeader + _groupHeaderSuffix);
|
|
}
|
|
|
|
if (_groupExpanded[groupID]) {
|
|
for (int *k = _itemsInGroup[groupID].begin(); k != _itemsInGroup[groupID].end(); ++k) {
|
|
_list.push_back(_dataList[*k].orig);
|
|
_listIndex.push_back(*k);
|
|
}
|
|
}
|
|
}
|
|
|
|
_selectedItem = -1;
|
|
checkBounds();
|
|
scrollBarRecalc();
|
|
|
|
_scrollBar->_currentPos = _currentPos;
|
|
_scrollBar->recalc();
|
|
// FIXME: Temporary solution to clear/display the background ofthe scrollbar when list
|
|
// grows too small or large during group toggle. We shouldn't have to redraw the top dialog,
|
|
// but not doing so the background of scrollbar isn't cleared.
|
|
if ((((uint)_scrollBar->_entriesPerPage < oldListSize) && ((uint)_scrollBar->_entriesPerPage > _list.size())) ||
|
|
(((uint)_scrollBar->_entriesPerPage > oldListSize) && ((uint)_scrollBar->_entriesPerPage < _list.size()))) {
|
|
g_gui.scheduleTopDialogRedraw();
|
|
} else {
|
|
markAsDirty();
|
|
}
|
|
}
|
|
|
|
void GroupedListWidget::loadClosedGroups(const Common::U32String &groupName) {
|
|
// Recalls what groups were closed from the config
|
|
if (ConfMan.hasKey("group_" + groupName, ConfMan.kApplicationDomain)) {
|
|
const Common::String &val = ConfMan.get("group_" + groupName, ConfMan.kApplicationDomain);
|
|
Common::StringTokenizer hiddenGroups(val);
|
|
|
|
for (Common::String tok = hiddenGroups.nextToken(); tok.size(); tok = hiddenGroups.nextToken()) {
|
|
// See if the hidden group is in our group headers still, if so, hide it
|
|
for (Common::U32StringArray::size_type i = 0; i < _groupHeaders.size(); ++i) {
|
|
if (_groupHeaders[i] == tok || (tok == "unnamed" && _groupHeaders[i].size() == 0)) {
|
|
_groupExpanded[i] = false;
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
sortGroups();
|
|
}
|
|
}
|
|
|
|
void GroupedListWidget::saveClosedGroups(const Common::U32String &groupName) {
|
|
// Save the hidden groups to the config
|
|
Common::String hiddenGroups;
|
|
for (Common::U32StringArray::size_type i = 0; i < _groupHeaders.size(); ++i) {
|
|
if (!_groupExpanded[i]) {
|
|
if (_groupHeaders[i].size()) {
|
|
hiddenGroups += _groupHeaders[i];
|
|
} else {
|
|
hiddenGroups += "unnamed";
|
|
}
|
|
hiddenGroups += ' ';
|
|
}
|
|
}
|
|
ConfMan.set("group_" + groupName, hiddenGroups, ConfMan.kApplicationDomain);
|
|
ConfMan.flushToDisk();
|
|
}
|
|
|
|
Common::Array<bool> GroupedListWidget::saveSelection() const {
|
|
return _selectedItems;
|
|
}
|
|
|
|
void GroupedListWidget::loadSelection(const Common::Array<bool> &savedSelection) {
|
|
_selectedItems.clear();
|
|
_selectedItems.resize(_dataList.size(), false);
|
|
|
|
int count = MIN((int)savedSelection.size(), (int)_selectedItems.size());
|
|
for (int i = 0; i < count; ++i) {
|
|
_selectedItems[i] = savedSelection[i];
|
|
}
|
|
|
|
_selectedItem = -1;
|
|
_lastSelectionStartItem = -1;
|
|
|
|
int topMostSel = -1;
|
|
int bottomMostSel = -1;
|
|
|
|
for (int visualRow = 0; visualRow < (int)_listIndex.size(); ++visualRow) {
|
|
int dataIndex = _listIndex[visualRow];
|
|
|
|
if (dataIndex >= 0 &&
|
|
dataIndex < (int)_selectedItems.size() &&
|
|
_selectedItems[dataIndex]) {
|
|
|
|
if (topMostSel == -1) {
|
|
topMostSel = visualRow;
|
|
_selectedItem = visualRow;
|
|
_lastSelectionStartItem = visualRow;
|
|
}
|
|
|
|
bottomMostSel = visualRow;
|
|
}
|
|
}
|
|
|
|
if (topMostSel != -1 && _entriesPerPage > 0) {
|
|
int span = bottomMostSel - topMostSel + 1;
|
|
|
|
if (topMostSel == bottomMostSel) {
|
|
_currentPos = topMostSel - _entriesPerPage / 2;
|
|
} else if (span <= _entriesPerPage) {
|
|
int spanCenter = (topMostSel + bottomMostSel) / 2;
|
|
_currentPos = spanCenter - _entriesPerPage / 2;
|
|
} else {
|
|
_currentPos = topMostSel;
|
|
}
|
|
} else {
|
|
_currentPos = 0;
|
|
}
|
|
|
|
checkBounds();
|
|
scrollBarRecalc();
|
|
|
|
_scrollBar->_currentPos = _currentPos;
|
|
_scrollBar->recalc();
|
|
|
|
markAsDirty();
|
|
}
|
|
|
|
void GroupedListWidget::handleMouseDown(int x, int y, int button, int clickCount) {
|
|
if (!isEnabled())
|
|
return;
|
|
|
|
_isMouseDown = true;
|
|
_isDragging = false;
|
|
_dragLastY = 0;
|
|
|
|
if (button == 1) {
|
|
_dragStartY = y;
|
|
_dragLastY = y;
|
|
_fluidScroller->stopAnimation();
|
|
}
|
|
|
|
// TODO: Determine where inside the string the user clicked and place the
|
|
// caret accordingly.
|
|
// See _editScrollOffset and EditTextWidget::handleMouseDown.
|
|
if (_editMode)
|
|
abortEditMode();
|
|
}
|
|
|
|
void GroupedListWidget::handleMouseUp(int x, int y, int button, int clickCount) {
|
|
if (button == 1 || button == 2) {
|
|
if (_isMouseDown && button == 1 && _isDragging)
|
|
_fluidScroller->startFling();
|
|
|
|
if (_isMouseDown && !_isDragging) {
|
|
int newSelectedItem = findItem(x, y);
|
|
if (newSelectedItem != -1) {
|
|
if (isGroupHeader(_listIndex[newSelectedItem])) {
|
|
int groupID = indexToGroupID(_listIndex[newSelectedItem]);
|
|
int oldSelection = getSelected();
|
|
_selectedItem = -1;
|
|
toggleGroup(groupID);
|
|
if (oldSelection != -1) {
|
|
_selectedItem = findDataIndex(oldSelection);
|
|
sendCommand(kListSelectionChangedCmd, _selectedItem);
|
|
}
|
|
applyScrollPos();
|
|
} else {
|
|
int dataIndex = _listIndex[newSelectedItem];
|
|
if (dataIndex >= 0) {
|
|
// Get modifier keys
|
|
int modifiers = g_system->getEventManager()->getModifierState();
|
|
bool ctrlClick = (modifiers & Common::KBD_CTRL) != 0;
|
|
bool shiftClick = (modifiers & Common::KBD_SHIFT) != 0;
|
|
|
|
// Only handle multi-select if it's enabled
|
|
if (_multiSelectEnabled && (shiftClick || ctrlClick)) {
|
|
if (shiftClick && _lastSelectionStartItem != -1) {
|
|
// Shift+Click: Select range in terms of underlying data indices
|
|
int startListIndex = _lastSelectionStartItem;
|
|
int endListIndex = newSelectedItem;
|
|
selectItemRange(startListIndex, endListIndex);
|
|
_selectedItem = newSelectedItem;
|
|
_lastSelectionStartItem = newSelectedItem;
|
|
sendCommand(kListSelectionChangedCmd, _selectedItem);
|
|
} else if (ctrlClick) {
|
|
// Ctrl+Click: toggle selection for the underlying data index
|
|
if (isItemSelected(newSelectedItem)) {
|
|
markSelectedItem(newSelectedItem, false);
|
|
} else {
|
|
markSelectedItem(newSelectedItem, true);
|
|
_selectedItem = newSelectedItem;
|
|
_lastSelectionStartItem = newSelectedItem;
|
|
}
|
|
sendCommand(kListSelectionChangedCmd, _selectedItem);
|
|
}
|
|
} else {
|
|
// Regular click: clear selection and select only this underlying item
|
|
clearSelection();
|
|
_selectedItem = newSelectedItem;
|
|
markSelectedItem(newSelectedItem, true);
|
|
sendCommand(kListSelectionChangedCmd, _selectedItem);
|
|
}
|
|
|
|
// Notify clients if an item was clicked
|
|
if (newSelectedItem >= 0)
|
|
sendCommand(kListItemSingleClickedCmd, _selectedItem);
|
|
|
|
applyScrollPos();
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
_isMouseDown = false;
|
|
_isDragging = false;
|
|
}
|
|
|
|
// If this was a double click and the mouse is still over
|
|
// the selected item, send the double click command
|
|
if (clickCount == 2 && (_selectedItem == findItem(x, y))) {
|
|
int selectID = getSelected();
|
|
if (selectID >= 0) {
|
|
sendCommand(kListItemDoubleClickedCmd, _selectedItem);
|
|
}
|
|
}
|
|
}
|
|
|
|
void GroupedListWidget::handleMouseWheel(int x, int y, int direction) {
|
|
_fluidScroller->handleMouseWheel(direction);
|
|
}
|
|
|
|
void GroupedListWidget::handleCommand(CommandSender *sender, uint32 cmd, uint32 data) {
|
|
switch (cmd) {
|
|
case kSetPositionCmd:
|
|
if (_currentPos != (int)data) {
|
|
_scrollPos = (float)data * (kLineHeight + _itemSpacing);
|
|
_fluidScroller->stopAnimation();
|
|
_scrollPos = _fluidScroller->setPosition(_scrollPos, false);
|
|
applyScrollPos();
|
|
|
|
// Scrollbar actions cause list focus (which triggers a redraw)
|
|
// NOTE: ListWidget's boss is always GUI::Dialog
|
|
((GUI::Dialog *)_boss)->setFocusWidget(this);
|
|
}
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
}
|
|
|
|
int GroupedListWidget::getItemPos(int item) {
|
|
for (uint i = 0; i < _listIndex.size(); i++) {
|
|
if (_listIndex[i] == item) {
|
|
return i;
|
|
}
|
|
}
|
|
|
|
return -1;
|
|
}
|
|
|
|
int GroupedListWidget::getNewSel(int index) {
|
|
if (_listIndex.empty()) {
|
|
return -1;
|
|
}
|
|
|
|
if (index < 0) {
|
|
return -1;
|
|
}
|
|
if (index >= (int)_listIndex.size()) {
|
|
index = _listIndex.size() - 1;
|
|
}
|
|
|
|
// First try the next selectable visible item
|
|
int selectable = findSelectableItem(index, +1);
|
|
|
|
// If there is none below, try the previous selectable one
|
|
if (selectable == -1) {
|
|
selectable = findSelectableItem(index - 1, -1);
|
|
}
|
|
|
|
if (selectable == -1) {
|
|
return -1;
|
|
}
|
|
|
|
return _listIndex[selectable];
|
|
}
|
|
|
|
void GroupedListWidget::toggleGroup(int groupID) {
|
|
_groupExpanded[groupID] = !_groupExpanded[groupID];
|
|
sortGroups();
|
|
}
|
|
|
|
void GroupedListWidget::drawWidget() {
|
|
int i, pos, len = _list.size();
|
|
Common::U32String buffer;
|
|
|
|
// Draw a thin frame around the list.
|
|
g_gui.theme()->drawWidgetBackground(Common::Rect(_x, _y, _x + _w, _y + _h),
|
|
ThemeEngine::kWidgetBackgroundBorder);
|
|
|
|
// Draw the list items
|
|
const int lineHeight = kLineHeight + _itemSpacing;
|
|
const int firstItem = MAX(0, (int)(_scrollPos / lineHeight));
|
|
const int offset = _scrollPos < 0 ? (int)_scrollPos : (int)_scrollPos % lineHeight;
|
|
const int indentSpacing = g_gui.getFontHeight();
|
|
|
|
Common::Rect innerRect(_x, _y + _topPadding, _x + _w - _scrollBarWidth, _y + _h - _bottomPadding);
|
|
Common::Rect oldClip = g_gui.theme()->swapClipRect(innerRect.findIntersectingRect(g_gui.theme()->getClipRect()));
|
|
|
|
for (i = 0, pos = firstItem; i <= _entriesPerPage && pos < len; i++, pos++) {
|
|
const int y = _y + _topPadding + lineHeight * i - offset;
|
|
ThemeEngine::TextInversionState inverted = ThemeEngine::kTextInversionNone;
|
|
#if 0
|
|
ThemeEngine::FontStyle bold = ThemeEngine::kFontStyleBold;
|
|
#endif
|
|
|
|
// For grouped lists, only real items (non-headers) may be highlighted.
|
|
int mapped = _listIndex[pos];
|
|
bool isRealItem = (mapped >= 0);
|
|
if (isRealItem) {
|
|
if (isItemSelected(pos))
|
|
inverted = _inversion;
|
|
}
|
|
|
|
ThemeEngine::WidgetStateInfo itemState = getItemState(pos);
|
|
|
|
Common::Rect r(getEditRect());
|
|
int pad = _leftPadding;
|
|
int rtlPad = (_x + r.left + _leftPadding) - (_x + _hlLeftPadding);
|
|
|
|
// Group header / grouped list indentation logic
|
|
if (isGroupHeader(mapped)) {
|
|
int groupID = indexToGroupID(mapped);
|
|
#if 0
|
|
bold = ThemeEngine::kFontStyleBold;
|
|
#endif
|
|
r.left += indentSpacing + _leftPadding;
|
|
g_gui.theme()->drawFoldIndicator(Common::Rect(_x + _hlLeftPadding + _leftPadding, y, _x + indentSpacing + _leftPadding, y + lineHeight), _groupExpanded[groupID]);
|
|
pad = 0;
|
|
} else if (_groupsVisible) {
|
|
r.left += indentSpacing + _leftPadding;
|
|
r.right -= _rightPadding;
|
|
pad = 0;
|
|
}
|
|
|
|
// If in numbering mode & not in RTL based GUI, we first print a number prefix
|
|
if (_numberingMode != kListNumberingOff && g_gui.useRTL() == false) {
|
|
buffer = Common::String::format("%2d. ", (pos + _numberingMode));
|
|
g_gui.theme()->drawText(Common::Rect(_x + _hlLeftPadding, y, _x + r.left + _leftPadding, y + lineHeight),
|
|
buffer, itemState, _drawAlign, inverted, _leftPadding, true);
|
|
pad = 0;
|
|
}
|
|
|
|
Common::Rect r1(_x + r.left, y, _x + r.right, y + lineHeight);
|
|
|
|
if (g_gui.useRTL()) {
|
|
if (_scrollBar->isVisible()) {
|
|
r1.translate(_scrollBarWidth, 0);
|
|
}
|
|
|
|
if (_numberingMode != kListNumberingOff) {
|
|
r1.translate(-rtlPad, 0);
|
|
}
|
|
}
|
|
|
|
ThemeEngine::FontColor color = ThemeEngine::kFontColorFormatting;
|
|
|
|
if (_selectedItem == pos && _editMode) {
|
|
buffer = _editString;
|
|
color = _editColor;
|
|
adjustOffset();
|
|
} else {
|
|
buffer = _list[pos];
|
|
}
|
|
|
|
drawFormattedText(r1, buffer, itemState, _drawAlign, inverted, pad, true, color);
|
|
|
|
if (_numberingMode != kListNumberingOff && g_gui.useRTL()) {
|
|
buffer = Common::String::format(" .%2d", (pos + _numberingMode));
|
|
|
|
Common::Rect r2 = r1;
|
|
|
|
r2.left = r1.right;
|
|
r2.right = r1.right + rtlPad;
|
|
|
|
g_gui.theme()->drawText(r2, buffer, itemState, _drawAlign, inverted, _leftPadding, true);
|
|
}
|
|
}
|
|
|
|
g_gui.theme()->swapClipRect(oldClip);
|
|
|
|
if (_editMode)
|
|
EditableWidget::drawWidget();
|
|
}
|
|
|
|
void GroupedListWidget::setFilter(const Common::U32String &filter, bool redraw) {
|
|
// FIXME: This method does not deal correctly with edit mode!
|
|
// Until we fix that, let's make sure it isn't called while editing takes place
|
|
assert(!_editMode);
|
|
|
|
Common::U32String filt = filter;
|
|
filt.toLowercase();
|
|
|
|
if (_filter == filt) // Filter was not changed
|
|
return;
|
|
|
|
int selectedItem = getSelected();
|
|
|
|
_filter = filt;
|
|
|
|
if (_filter.empty()) {
|
|
// No filter -> display everything
|
|
sortGroups();
|
|
} else {
|
|
// Restrict the list to everything which contains all words in _filter
|
|
// as substrings, ignoring case.
|
|
|
|
Common::U32StringTokenizer tok(_filter);
|
|
Common::U32String tmp;
|
|
int n = 0;
|
|
|
|
_list.clear();
|
|
_listIndex.clear();
|
|
|
|
for (auto i = _dataList.begin(); i != _dataList.end(); ++i, ++n) {
|
|
tmp = i->clean;
|
|
tmp.toLowercase();
|
|
bool matches = true;
|
|
tok.reset();
|
|
while (!tok.empty()) {
|
|
if (!_filterMatcher(_filterMatcherArg, n, tmp, tok.nextToken())) {
|
|
matches = false;
|
|
break;
|
|
}
|
|
}
|
|
|
|
if (matches) {
|
|
_list.push_back(i->orig);
|
|
_listIndex.push_back(n);
|
|
}
|
|
}
|
|
}
|
|
|
|
_currentPos = 0;
|
|
_scrollPos = 0.0f;
|
|
_fluidScroller->setPosition(_scrollPos);
|
|
_selectedItem = -1;
|
|
// Try to preserve the previous selection
|
|
if (selectedItem != -1)
|
|
setSelected(selectedItem);
|
|
|
|
if (redraw) {
|
|
scrollBarRecalc();
|
|
// Redraw the whole dialog. This is annoying, as this might be rather
|
|
// expensive when really only the list widget and its scroll bar area
|
|
// to be redrawn. However, since the scrollbar might change its
|
|
// visibility status, and the list its width, we cannot just redraw
|
|
// the two.
|
|
// TODO: A more efficient (and elegant?) way to handle this would be to
|
|
// introduce a kind of "BoxWidget" or "GroupWidget" which defines a
|
|
// rectangular region and subwidgets can be placed within it.
|
|
// Such a widget could also (optionally) draw a border (or even different
|
|
// kinds of borders) around the objects it groups; and also a 'title'
|
|
// (I am borrowing these "ideas" from the NSBox class in Cocoa :).
|
|
g_gui.scheduleTopDialogRedraw();
|
|
}
|
|
}
|
|
ThemeEngine::WidgetStateInfo GroupedListWidget::getItemState(int item) const {
|
|
return _state;
|
|
}
|
|
|
|
bool GroupedListWidget::isItemSelectable(int item) const {
|
|
if (item < 0 || item >= (int)_listIndex.size())
|
|
return false;
|
|
return !isGroupHeader(_listIndex[item]);
|
|
}
|
|
|
|
} // End of namespace GUI
|