diff --git a/minigui.d b/minigui.d index 2bd2618..8794ff1 100644 --- a/minigui.d +++ b/minigui.d @@ -189,6 +189,17 @@ the virtual functions remain as the default calculated values. then the reads go More to come. + My_UI_Guidelines: + In a perfect world, you'd achieve all the following goals: + + $(LIST + * All operations are present in the menu + * The operations the user wants at the moment are right where they want them + * All operations can be scripted + * The UI does not move any elements without explicit user action + * All numbers can be seen and typed in if wanted, even if the ui usually hides them + ) + History: Minigui had mostly additive changes or bug fixes since its inception until May 2021. @@ -2355,6 +2366,24 @@ abstract class ComboboxBase : Widget { return cast(string[]) options_; } + /++ + Replaces the list of options in the box. Note that calling this will also reset the selection. + + History: + Added December, 29 2024 + +/ + final @property void options(string[] options) { + version(win32_widgets) + SendMessageW(hwnd, 331 /*CB_RESETCONTENT*/, 0, 0); + selection_ = -1; + options_ = null; + foreach(opt; options) + addOption(opt); + + version(custom_widgets) + redraw(); + } + private string[] options_; private int selection_ = -1; @@ -2459,69 +2488,125 @@ abstract class ComboboxBase : Widget { override int maxHeight() { return defaultLineHeight + 4; } } - version(custom_widgets) { + version(custom_widgets) + void popup() { + CustomComboBoxPopup popup = new CustomComboBoxPopup(this); + } + +} + +private class CustomComboBoxPopup : Window { + private ComboboxBase associatedWidget; + private ListWidget lw; + + this(ComboboxBase associatedWidget) { + this.associatedWidget = associatedWidget; // FIXME: this should scroll if there's too many elements to reasonably fit on screen - SimpleWindow dropDown; - void popup() { - auto w = width; - // FIXME: suggestedDropdownHeight see below - auto h = cast(int) this.options.length * defaultLineHeight + 8; + auto w = associatedWidget.width; + // FIXME: suggestedDropdownHeight see below + auto h = cast(int) associatedWidget.options.length * defaultLineHeight + 8; - auto coord = this.globalCoordinates(); - auto dropDown = new SimpleWindow( - w, h, - null, OpenGlOptions.no, Resizability.fixedSize, WindowTypes.dropdownMenu, WindowFlags.dontAutoShow, parentWindow ? parentWindow.win : null); + if(h > associatedWidget.parentWindow.height) + h = associatedWidget.parentWindow.height; - dropDown.move(coord.x, coord.y + this.height); + auto coord = associatedWidget.globalCoordinates(); + auto dropDown = new SimpleWindow( + w, h, + null, OpenGlOptions.no, Resizability.fixedSize, WindowTypes.dropdownMenu, WindowFlags.dontAutoShow, associatedWidget.parentWindow ? associatedWidget.parentWindow.win : null); - { - auto cs = getComputedStyle(); - auto painter = dropDown.draw(); - draw3dFrame(0, 0, w, h, painter, FrameStyle.risen, getComputedStyle().background.color); - auto p = Point(4, 4); - painter.outlineColor = cs.foregroundColor; - foreach(option; options) { - painter.drawText(p, option); - p.y += defaultLineHeight; - } + super(dropDown); + + dropDown.move(coord.x, coord.y + associatedWidget.height); + + this.lw = new ListWidget(this); + version(custom_widgets) + lw.multiSelect = false; + foreach(option; associatedWidget.options) + lw.addOption(option); + + lw.setSelection(associatedWidget.getSelection); + lw.scrollSelectionIntoView(); + + /+ + { + auto cs = getComputedStyle(); + auto painter = dropDown.draw(); + draw3dFrame(0, 0, w, h, painter, FrameStyle.risen, getComputedStyle().background.color); + auto p = Point(4, 4); + painter.outlineColor = cs.foregroundColor; + foreach(option; associatedWidget.options) { + painter.drawText(p, option); + p.y += defaultLineHeight; } - - dropDown.setEventHandlers( - (MouseEvent event) { - if(event.type == MouseEventType.buttonReleased) { - dropDown.close(); - auto element = (event.y - 4) / defaultLineHeight; - if(element >= 0 && element <= options.length) { - selection_ = element; - - fireChangeEvent(); - } - } - } - ); - - dropDown.visibilityChanged = (bool visible) { - if(visible) { - this.redraw(); - dropDown.grabInput(); - } else { - dropDown.releaseInputGrab(); - } - }; - - dropDown.show(); } + dropDown.setEventHandlers( + (MouseEvent event) { + if(event.type == MouseEventType.buttonReleased) { + dropDown.close(); + auto element = (event.y - 4) / defaultLineHeight; + if(element >= 0 && element <= associatedWidget.options.length) { + associatedWidget.selection_ = element; + + associatedWidget.fireChangeEvent(); + } + } + } + ); + +/ + + Widget previouslyFocusedWidget; + + dropDown.visibilityChanged = (bool visible) { + if(visible) { + this.redraw(); + captureMouse(this); + //dropDown.grabInput(); + + if(previouslyFocusedWidget is null) + previouslyFocusedWidget = associatedWidget.parentWindow.focusedWidget; + associatedWidget.parentWindow.focusedWidget = lw; + } else { + //dropDown.releaseInputGrab(); + releaseMouseCapture(); + + associatedWidget.setSelection(lw.getSelection); + + associatedWidget.parentWindow.focusedWidget = previouslyFocusedWidget; + } + }; + + dropDown.show(); + } + + override void defaultEventHandler_click(ClickEvent ce) { + if(ce.button == MouseButton.left && (ce.target is this || ce.target is lw)) { + this.win.close(); + } + } + + override void defaultEventHandler_char(CharEvent ce) { + if(ce.character == '\n') + this.win.close(); } } /++ A drop-down list where the user must select one of the given options. Like `