Add ComboBox widget
|
@ -48,6 +48,7 @@ loadWidget("ScrollArea");
|
||||||
loadWidget("Rule");
|
loadWidget("Rule");
|
||||||
loadWidget("Slider");
|
loadWidget("Slider");
|
||||||
loadWidget("TabWidget");
|
loadWidget("TabWidget");
|
||||||
|
loadWidget("ComboBox");
|
||||||
|
|
||||||
# standard dialogs
|
# standard dialogs
|
||||||
loadDialog("InputDialog");
|
loadDialog("InputDialog");
|
||||||
|
|
|
@ -52,6 +52,10 @@ gui.MenuItem = {
|
||||||
return m;
|
return m;
|
||||||
},
|
},
|
||||||
|
|
||||||
|
text: func {
|
||||||
|
return me._text;
|
||||||
|
},
|
||||||
|
|
||||||
setMenu: func(menu) {
|
setMenu: func(menu) {
|
||||||
menu._parent_item = me;
|
menu._parent_item = me;
|
||||||
menu._canvas_item = me._parent_menu._canvas_item;
|
menu._canvas_item = me._parent_menu._canvas_item;
|
||||||
|
|
|
@ -159,7 +159,7 @@ gui.Widget = {
|
||||||
# add listener when first binding is added
|
# add listener when first binding is added
|
||||||
# if this happens before the view is set, we instead do it in _setView below
|
# if this happens before the view is set, we instead do it in _setView below
|
||||||
if ((size(me._bindings) == 1) and me._view) {
|
if ((size(me._bindings) == 1) and me._view) {
|
||||||
me.addEventListener("keydown", func(e) me._onKeyPressed(e));
|
me._view._root.addEventListener("keydown", func(e) me._onKeyPressed(e));
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
|
||||||
|
|
|
@ -159,7 +159,15 @@ var WidgetsFactoryDialog = {
|
||||||
"tick-count" : 10})
|
"tick-count" : 10})
|
||||||
.setValue(42);
|
.setValue(42);
|
||||||
m.numericControlsTab.addItem(m.slider);
|
m.numericControlsTab.addItem(m.slider);
|
||||||
|
|
||||||
|
|
||||||
|
m.combo1 = gui.widgets.ComboBox.new(m.tabsContent, style, {});
|
||||||
|
m.combo1.addMenuItem("Apples", 0);
|
||||||
|
m.combo1.addMenuItem("Pears", 1);
|
||||||
|
m.combo1.addMenuItem("Lemons", 2);
|
||||||
|
m.combo1.addMenuItem("Oranges", 3);
|
||||||
|
|
||||||
|
m.numericControlsTab.addItem(m.combo1);
|
||||||
return m;
|
return m;
|
||||||
},
|
},
|
||||||
|
|
||||||
|
|
|
@ -973,7 +973,7 @@ DefaultStyle.widgets["menu-bar"] = {
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
# A button
|
# A combo-box
|
||||||
DefaultStyle.widgets["combo-box"] = {
|
DefaultStyle.widgets["combo-box"] = {
|
||||||
new: func(parent, cfg)
|
new: func(parent, cfg)
|
||||||
{
|
{
|
||||||
|
@ -1129,4 +1129,5 @@ DefaultStyle.widgets.list = {
|
||||||
|
|
||||||
return me;
|
return me;
|
||||||
}
|
}
|
||||||
}
|
};
|
||||||
|
|
||||||
|
|
109
Nasal/canvas/gui/widgets/ComboBox.nas
Normal file
|
@ -0,0 +1,109 @@
|
||||||
|
# SPDX-FileCopyrightText: (C) 2022 James Turner <james@flightgear.org>
|
||||||
|
# SPDX-License-Identifier: GPL-2.0-or-later
|
||||||
|
|
||||||
|
gui.widgets.ComboBox = {
|
||||||
|
new: func(parent, style, cfg)
|
||||||
|
{
|
||||||
|
var cfg = Config.new(cfg);
|
||||||
|
var m = gui.Widget.new(gui.widgets.ComboBox);
|
||||||
|
m._focus_policy = m.StrongFocus;
|
||||||
|
# m._flat = cfg.get("flat", 0);
|
||||||
|
m._menu = gui.Menu.new();
|
||||||
|
m._setView( style.createWidget(parent, cfg.get("type", "combo-box"), cfg) );
|
||||||
|
m._items = [];
|
||||||
|
m._currentIndex = nil;
|
||||||
|
m._style = style; # cache reference to style for creating items
|
||||||
|
m._down = 0;
|
||||||
|
|
||||||
|
return m;
|
||||||
|
},
|
||||||
|
|
||||||
|
setText: func(text)
|
||||||
|
{
|
||||||
|
if( me._view != nil )
|
||||||
|
me._view.setText(me, text);
|
||||||
|
return me;
|
||||||
|
},
|
||||||
|
|
||||||
|
show: func()
|
||||||
|
{
|
||||||
|
# check if enabled
|
||||||
|
},
|
||||||
|
|
||||||
|
menu: func()
|
||||||
|
{
|
||||||
|
return me._menu;
|
||||||
|
},
|
||||||
|
|
||||||
|
# convenience helper to add simple items
|
||||||
|
addMenuItem: func(text, value) {
|
||||||
|
var index = size(me._items);
|
||||||
|
var item = me.menu().createItem(text, func { me._itemCallback(index);}, {});
|
||||||
|
item.menuValue = value;
|
||||||
|
append(me._items, item);
|
||||||
|
if (me._currentIndex == nil) {
|
||||||
|
# select first item added, if we were previously empty
|
||||||
|
me.setCurrentByIndex(0);
|
||||||
|
}
|
||||||
|
},
|
||||||
|
|
||||||
|
# helper to set the current item by passing in
|
||||||
|
# a value of an item
|
||||||
|
setCurrentByValue: func(value) {
|
||||||
|
var index = 0;
|
||||||
|
foreach(var i; me._items) {
|
||||||
|
if (i.menuValue == value) {
|
||||||
|
setCurrentByIndex(index);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
index+=1;
|
||||||
|
}
|
||||||
|
|
||||||
|
logprint(DEV_WARN, "Canvas.Gui ComboBox: no such value in menu:" ~ value);
|
||||||
|
},
|
||||||
|
|
||||||
|
setCurrentByIndex: func(index) {
|
||||||
|
if (me._currentIndex == index)
|
||||||
|
return;
|
||||||
|
|
||||||
|
if (index >= size(me._items)) {
|
||||||
|
logprint(DEV_WARN, "Canvas.Gui ComboBox: invalid index passed to setCUrrentByIndex" ~ index);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
me._currentIndex = index;
|
||||||
|
me._view.setText(me, me._items[index].text());
|
||||||
|
},
|
||||||
|
|
||||||
|
setDown: func(down = 1)
|
||||||
|
{
|
||||||
|
if (me._down == down )
|
||||||
|
return me;
|
||||||
|
|
||||||
|
me._down = down;
|
||||||
|
me._onStateChange();
|
||||||
|
return me;
|
||||||
|
},
|
||||||
|
|
||||||
|
# protected:
|
||||||
|
_itemCallback: func(index)
|
||||||
|
{
|
||||||
|
me.setCurrentByIndex(index);
|
||||||
|
},
|
||||||
|
|
||||||
|
_setView: func(view)
|
||||||
|
{
|
||||||
|
call(gui.Widget._setView, [view], me);
|
||||||
|
|
||||||
|
var el = view._root;
|
||||||
|
el.addEventListener("mousedown", func if( me._enabled ) { me.setDown(1); me._openMenu(); });
|
||||||
|
el.addEventListener("mouseup", func if( me._enabled ) me.setDown(0));
|
||||||
|
el.addEventListener("mouseleave",func me.setDown(0));
|
||||||
|
},
|
||||||
|
|
||||||
|
_openMenu: func
|
||||||
|
{
|
||||||
|
me.menu().show();
|
||||||
|
}
|
||||||
|
};
|
|
@ -1,61 +0,0 @@
|
||||||
# SPDX-FileCopyrightText: (C) 2022 James Turner <james@flightgear.org>
|
|
||||||
# SPDX-License-Identifier: GPL-2.0-or-later
|
|
||||||
|
|
||||||
gui.widgets.PopupMenu = {
|
|
||||||
new: func(parent, style, cfg)
|
|
||||||
{
|
|
||||||
var cfg = Config.new(cfg);
|
|
||||||
var m = gui.Widget.new(gui.widgets.Button);
|
|
||||||
m._focus_policy = m.StrongFocus;
|
|
||||||
m._flat = cfg.get("flat", 0);
|
|
||||||
m._menu = nil;
|
|
||||||
m._setView( style.createWidget(parent, cfg.get("type", "button"), cfg) );
|
|
||||||
|
|
||||||
return m;
|
|
||||||
},
|
|
||||||
setText: func(text)
|
|
||||||
{
|
|
||||||
if( me._view != nil )
|
|
||||||
me._view.setText(me, text);
|
|
||||||
return me;
|
|
||||||
},
|
|
||||||
|
|
||||||
show: func()
|
|
||||||
{
|
|
||||||
# check if enabled
|
|
||||||
},
|
|
||||||
|
|
||||||
menu: func()
|
|
||||||
{
|
|
||||||
if (!me._menu) {
|
|
||||||
# FIXME pass style
|
|
||||||
me.menu = gui.widgets.Menu.new(cfg);
|
|
||||||
}
|
|
||||||
|
|
||||||
return me._menu;
|
|
||||||
},
|
|
||||||
|
|
||||||
# convenience helper to add simple items
|
|
||||||
addMenuItem: func(text, value) {
|
|
||||||
# FIXME pass style
|
|
||||||
me.menu().append(gui.widgets.MenuItem.new());
|
|
||||||
},
|
|
||||||
|
|
||||||
# helper to set the current item by passing in
|
|
||||||
# a value of an item
|
|
||||||
setCurrentByValue: func(value) {
|
|
||||||
|
|
||||||
|
|
||||||
},
|
|
||||||
|
|
||||||
# protected:
|
|
||||||
_setView: func(view)
|
|
||||||
{
|
|
||||||
call(gui.Widget._setView, [view], me);
|
|
||||||
|
|
||||||
var el = view._root;
|
|
||||||
el.addEventListener("mousedown", func if( me._enabled ) me.setDown(1));
|
|
||||||
el.addEventListener("mouseup", func if( me._enabled ) me.setDown(0));
|
|
||||||
el.addEventListener("mouseleave",func me.setDown(0));
|
|
||||||
}
|
|
||||||
};
|
|
|
@ -514,7 +514,7 @@ var XMLVRule =
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
||||||
var XMLPopupMenu =
|
var XMLComboBox =
|
||||||
{
|
{
|
||||||
init: func(objectProps)
|
init: func(objectProps)
|
||||||
{
|
{
|
||||||
|
@ -524,7 +524,7 @@ var XMLPopupMenu =
|
||||||
{
|
{
|
||||||
# do we support a label or is that a seperate widget?
|
# do we support a label or is that a seperate widget?
|
||||||
|
|
||||||
me._view = cwidgets.PopupMenu.new(viewParent, canvas.style, {});
|
me._view = cwidgets.ComboBox.new(viewParent, canvas.style, {});
|
||||||
me._layout = me._view;
|
me._layout = me._view;
|
||||||
me._applyLayoutConfig();
|
me._applyLayoutConfig();
|
||||||
me.update();
|
me.update();
|
||||||
|
@ -585,10 +585,8 @@ var _createCompatObject = func(type)
|
||||||
widget = XMLVRule;
|
widget = XMLVRule;
|
||||||
}
|
}
|
||||||
|
|
||||||
# these are called combos in XML, but are really
|
|
||||||
# popup menus, no free value entry is possible
|
|
||||||
if (type == "combo") {
|
if (type == "combo") {
|
||||||
widget = XMLPopupMenu;
|
widget = XMLComboBox;
|
||||||
}
|
}
|
||||||
|
|
||||||
return gui.xml.Object.new({
|
return gui.xml.Object.new({
|
||||||
|
|
BIN
gui/styles/AmbianceClassic/widgets/combobox-arrow-disabled.png
Normal file
After Width: | Height: | Size: 241 B |
BIN
gui/styles/AmbianceClassic/widgets/combobox-arrow.png
Normal file
After Width: | Height: | Size: 241 B |
BIN
gui/styles/AmbianceClassic/widgets/combobox-button-disabled.png
Normal file
After Width: | Height: | Size: 375 B |
BIN
gui/styles/AmbianceClassic/widgets/combobox-button-focused.png
Normal file
After Width: | Height: | Size: 624 B |
After Width: | Height: | Size: 588 B |
BIN
gui/styles/AmbianceClassic/widgets/combobox-button.png
Normal file
After Width: | Height: | Size: 544 B |
BIN
gui/styles/AmbianceClassic/widgets/combobox-entry-disabled.png
Normal file
After Width: | Height: | Size: 372 B |
BIN
gui/styles/AmbianceClassic/widgets/combobox-entry-focused.png
Normal file
After Width: | Height: | Size: 580 B |
BIN
gui/styles/AmbianceClassic/widgets/combobox-entry.png
Normal file
After Width: | Height: | Size: 494 B |