From 9dd0a00f0825d7def8048ae1b2ed8b57890c5f44 Mon Sep 17 00:00:00 2001 From: "Adam D. Ruppe" Date: Wed, 19 May 2021 15:27:41 -0400 Subject: [PATCH] more wip gui --- README.md | 29 ++ cgi.d | 16 +- http2.d | 65 ++- minigui.d | 1164 ++++++++++++++++++++++++++++++++++------------- simpledisplay.d | 6 +- 5 files changed, 945 insertions(+), 335 deletions(-) diff --git a/README.md b/README.md index 436523a..55b0b02 100644 --- a/README.md +++ b/README.md @@ -16,12 +16,41 @@ This only lists changes that broke things and got a major version bump. I didn't Please note that I DO consider changes to build process to be a breaking change, but I do NOT consider symbol additions, changes to undocumented members, or the occasional non-fatal deprecation to be breaking changes. Undocumented members may be changed at any time, whereas additions and/or deprecations will be a minor version change. +## 10.0 + +Released: May 2020 + +minigui 2.0 came out with deprecations on some event properties, moved style properties, and various other changes. See http://arsd-official.dpldocs.info/arsd.minigui.html#history for details. + ## 9.0 +Released: December 2020 + simpledisplay's OperatingSystemFont, which is also used by terminalemulator.d (which is used by terminal.d's -version=TerminalDirectToEmulator function) would previously only load X Core Fonts. It now prefers TrueType fonts via Xft. This loads potentially different fonts and the sizes are interpreted differently, so you may need to adjust your preferences there. To restore previous behavior, prefix your font name strings with "core:". http2.d's "connection refused" handler used to throw an exception for any pending connection. Now it instead just sets that connection to `aborted` and carries on with other ones. When you are doing a request, be sure to check `response.code`. It may be < 100 if connection refused and other errors. You should already have been checking the http response code, but now some things that were exceptions are now codes, so it is even more important to check this properly. +## Prehistory: + +8.0 Released: June 2020 + +7.0 and 6.0 Released: March 2020 (these were changes to the terminal.d virtual methods, tag 6.0 was a mistake, i pressed it too early) + +5.0 Released: January 2019 + +4.0 and 3.0 Released: July 2019 and June 2019, respectively. These had to do with dub subpackage configuration changes IIRC. + +2.0 Released (first use of semver tagging, before this I would only push to master): March 2018 + +April 2016: simpledisplay and terminal renamed to arsd.simpledisplay and arsd.terminal + +September 2015: simpledisplay started to depend on color.d instead of being standalone + +Joined dub (tagged 1.0): June 2015 + +Joined github: July 2011 + +Started project on my website: 2008 ## Credits diff --git a/cgi.d b/cgi.d index a0249ee..ac4545b 100644 --- a/cgi.d +++ b/cgi.d @@ -2991,14 +2991,23 @@ struct Uri { if(part == ".") { continue; } else if(part == "..") { - toKeep = toKeep[0 .. $-1]; + //if(toKeep.length > 1) + toKeep = toKeep[0 .. $-1]; + //else + //toKeep = [""]; continue; } else { + //if(toKeep.length && toKeep[$-1].length == 0 && part.length == 0) + //continue; // skip a `//` situation toKeep ~= part; } } - this.path = toKeep.join("/"); + auto path = toKeep.join("/"); + if(path.length && path[0] != '/') + path = "/" ~ path; + + this.path = path; } unittest { @@ -3081,6 +3090,9 @@ struct Uri { assert(Uri("./").basedOn(url) == "/test/", Uri("./").basedOn(url)); assert(Uri("../").basedOn(url) == "/"); + url = Uri("http://example.com/"); + assert(Uri("../foo").basedOn(url) == "http://example.com/foo"); + //auto uriBefore = url; url = Uri("#anchor"); // everything should remain the same except the anchor //uriBefore.anchor = "anchor"); diff --git a/http2.d b/http2.d index 3f51d6d..17cc9ec 100644 --- a/http2.d +++ b/http2.d @@ -12,16 +12,37 @@ It has no dependencies for basic operation, but does require OpenSSL - libraries (or compatible) to be support HTTPS. Compile with - `-version=with_openssl` to enable such support. + libraries (or compatible) to be support HTTPS. This dynamically loaded + on-demand (meaning it won't be loaded if you don't use it, but if you do + use it, the openssl dynamic libraries must be found in the system search path). + + You can compile with `-version=without_openssl` to entirely disable ssl support. http2.d, despite its name, does NOT implement HTTP/2.0, but this shouldn't matter for 99.9% of usage, since all servers will continue to support HTTP/1.1 for a very long time. - +/ module arsd.http2; +/// +unittest { + import arsd.http2; + + void main() { + auto client = new HttpClient(); + + auto request = client.request(Uri("http://dlang.org/")); + auto response = request.waitForCompletion(); + + import std.stdio; + writeln(response.contentText); + writeln(response.code, " ", response.codeText); + writeln(response.contentType); + } + + version(arsd_http2_integration_test) main(); // exclude from docs +} + // FIXME: I think I want to disable sigpipe here too. import std.uri : encodeComponent; @@ -660,16 +681,24 @@ struct Uri { if(part == ".") { continue; } else if(part == "..") { - toKeep = toKeep[0 .. $-1]; + //if(toKeep.length > 1) + toKeep = toKeep[0 .. $-1]; + //else + //toKeep = [""]; continue; } else { + //if(toKeep.length && toKeep[$-1].length == 0 && part.length == 0) + //continue; // skip a `//` situation toKeep ~= part; } } - this.path = toKeep.join("/"); - } + auto path = toKeep.join("/"); + if(path.length && path[0] != '/') + path = "/" ~ path; + this.path = path; + } } /* @@ -1853,26 +1882,12 @@ enum HttpVerb { MERGE } -/** - Usage: +/++ + HttpClient keeps cookies, location, and some other state to reuse connections, when possible, like a web browser. + You can use it as your entry point to make http requests. - --- - auto client = new HttpClient("localhost", 80); - // relative links work based on the current url - HttpRequest request = client.get("foo/bar"); - request = client.get("baz"); // gets foo/baz - - // requests are not sent until you tell them to; - // they are just objects representing potential. - // to realize it and fetch the response, use waitForCompletion: - - HttpResponse response = request.waitForCompletion(); - - // now you can use response.headers, response.contentText, etc - --- -*/ - -/// HttpClient keeps cookies, location, and some other state to reuse connections, when possible, like a web browser. + See the example on [arsd.http2#examples]. ++/ class HttpClient { /* Protocol restrictions, useful to disable when debugging servers */ bool useHttp11 = true; /// diff --git a/minigui.d b/minigui.d index 0cc195e..c7b9bae 100644 --- a/minigui.d +++ b/minigui.d @@ -9,9 +9,11 @@ the virtual functions remain as the default calculated values. then the reads go // FIXME: opt-in file picker widget with image support -// FIXME: slider widget. // FIXME: number widget +// https://www.codeguru.com/cpp/controls/buttonctrl/advancedbuttons/article.php/c5161/Native-Win32-ThemeAware-OwnerDraw-Controls-No-MFC.htm +// https://docs.microsoft.com/en-us/windows/win32/controls/using-visual-styles + // osx style menu search. // would be cool for a scroll bar to have marking capabilities @@ -95,22 +97,22 @@ the virtual functions remain as the default calculated values. then the reads go may be later, and should use native controls) to keep size down. The Linux appearance is similar to Windows 95 and avoids using images to maintain network efficiency on remote X connections, though you can customize that. - - minigui's only required dependencies are [arsd.simpledisplay] and [arsd.color]. + + + minigui's only required dependencies are [arsd.simpledisplay] and [arsd.color], + on which it is built. simpledisplay provides the low-level interfaces and minigui + builds the concept of widgets inside the windows on top of it. Its #1 goal is to be useful without being large and complicated like GTK and Qt. It isn't hugely concerned with appearance - on Windows, it just uses the native controls and native theme, and on Linux, it keeps it simple and I may change that - at any time. - - I love Qt, if you want something full featured, use it! But if you want something - you can just drop into a small project and expect the basics to work without outside - dependencies, hopefully minigui will work for you. + at any time, though after May 2021, you can customize some things with css-inspired + [Widget.Style] classes. (On Windows, if you compile with `-version=custom_widgets`, + you can use the custom implementation there too, but... you shouldn't.) The event model is similar to what you use in the browser with Javascript and the layout engine tries to automatically fit things in, similar to a css flexbox. - FOR BEST RESULTS: be sure to link with the appropriate subsystem command `-L/SUBSYSTEM:WINDOWS:5.0`, for example, because otherwise you'll get a console and other visual bugs. @@ -140,8 +142,10 @@ the virtual functions remain as the default calculated values. then the reads go another thread and posting events back and forth. $(H2 Add ons) + See the `minigui_addons` directory in the arsd repo for some add on widgets + you can import separately too. - $(H2 XML definitions) + $(H3 XML definitions) If you use [arsd.minigui_xml], you can create widget trees from XML at runtime. $(H3 Scriptability) @@ -165,12 +169,68 @@ the virtual functions remain as the default calculated values. then the reads go More to come. History: - The event model changed slightly in May 2021 (dub v10.0). See [Event] for details. + Minigui had mostly additive changes or bug fixes since its inception until May 2021. + + In May 2021 (dub v10.0), minigui got an overhaul. If it was versioned independently, I'd + tag this as version 2.0. + + Among the changes: + $(LIST + * The event model changed to prefer strongly-typed events, though the Javascript string style ones still work, using properties off them is deprecated. It will still compile and function, but you should change the handler to use the classes in its argument list. I adapted my code to use the new model in just a few minutes, so it shouldn't too hard. + + See [Event] for details. + + * A [DoubleClickEvent] was added. Previously, you'd get two rapidly repeated click events. Now, you get one click event followed by a double click event. If you must recreate the old way exactly, you can listen for a DoubleClickEvent, set a flag upon receiving one, then send yourself a synthetic ClickEvent on the next MouseUpEvent, but your program might be better served just working with [MouseDownEvent]s instead. + + See [DoubleClickEvent] for details. + + * Styling hints were added, and the few that existed before have been moved to a new helper class. Deprecated forwarders exist for the (few) old properties to help you transition. Note that most of these only affect a `custom_events` build, which is the default on Linux, but opt in only on Windows. + + See [Widget.Style] for details. + + * A widget must now opt in to receiving keyboard focus, rather than opting out. + + * Most Widget constructors no longer have a default `parent` argument. You must pass the parent to almost all widgets, or in rare cases, an explict `null`, but more often than not, you need the parent so the default argument was not very useful at best and misleading to a crash at worst. + + * [LabeledLineEdit] changed its default layout to vertical instead of horizontal. You can restore the old behavior by passing a `TextAlignment` argument to the constructor. + + * Several conversions of public fields to properties, deprecated, or made private. It is unlikely this will affect you, but the compiler will tell you if it does. + + * Various non-breaking additions. + ) +/ module arsd.minigui; +/++ + This hello world sample will have an oversized button, but that's ok, you see your first window! ++/ +version(Demo) +unittest { + import arsd.minigui; + + void main() { + auto window = new MainWindow(); + + auto hello = new TextLabel("Hello, world!", TextAlignment.Center, window); + auto button = new Button("Close", window); + button.addEventListener((scope ClickEvent ev) { + window.close(); + }); + + window.loop(); + } + + main(); // exclude from docs +} + public import arsd.simpledisplay; -private alias Rectangle = arsd.color.Rectangle; // I specifically want this in here, not the win32 GDI Rectangle() +/++ + Convenience import to override the Windows GDI Rectangle function (you can still use it through fully-qualified imports) + + History: + Was private until May 15, 2021. ++/ +public alias Rectangle = arsd.color.Rectangle; // I specifically want this in here, not the win32 GDI Rectangle() version(Windows) { import core.sys.windows.winnls; @@ -265,20 +325,50 @@ version(Windows) { +/ /++ - The way this module works is it builds on top of a SimpleWindow - from simpledisplay to provide simple controls and such. - - Non-native controls suck, but nevertheless, I'm going to do it that - way to avoid dependencies on stuff like gtk on X... and since I'll - be writing the widgets there, I might as well just use them on Windows - too if you like, using `-version=custom_widgets`. - - So, by extension, this sucks. But gtkd is just too big for me. + The `Widget` is the base class for minigui's functionality, ranging from UI components like checkboxes or text displays to abstract groupings of other widgets like a layout container or a html `
`. You will likely want to use pre-made widgets as well as creating your own. - The goal is to look kinda like Windows 95, perhaps with customizability. - Nothing too fancy, just the basics that work, but of course you can do whatever - you want in the draw handler. + To create your own widget, you must inherit from it and create a constructor that passes a parent to `super`. Everything else after that is optional. + + --- + class MinimalWidget : Widget { + this(Widget parent) { + super(parent); + } + } + --- + + $(SIDEBAR + I'm not entirely happy with leaf, container, and windows all coming from the same base Widget class, but I so far haven't thought of a better solution that's good enough to justify the breakage of a transition. It hasn't been a major problem in practice anyway. + ) + + Broadly, there's two kinds of widgets: leaf widgets, which are intended to be the direct user-interactive components, and container widgets, which organize, lay out, and aggregate other widgets in the object tree. A special case of a container widget is [Window], which represents a separate top-level window on the screen. Both leaf and container widgets inherit from `Widget`, so this distinction is more conventional than formal. + + Among the things you'll most likely want to change in your custom widget: + + $(LIST + * In your constructor, set `tabStop = false;` if the widget is not supposed to receive keyboard focus. (Please note its childen still can, so `tabStop = false;` is appropriate on most container widgets.) + + You may explicitly set `tabStop = true;` to ensure you get it, even against future changes to the library, though that's the default right now. + + Do this $(I after) calling the `super` constructor. + + * Override [paint] if you want full control of the widget's drawing area (except the area obscured by children!), or [paintContent] if you want to participate in the styling engine's system. You'll also possibly want to make a subclass of [Style] and use [OverrideStyle] to change the default hints given to the styling engine for widget. + + Generally, painting is a job for leaf widgets, since child widgets would obscure your drawing area anyway. However, it is your decision. + + * Override default event handlers with your behavior. For example [defaultEventHandler_click] may be overridden to make clicks do something. Again, this is generally a job for leaf widgets rather than containers; most events are dispatched to the lowest leaf on the widget tree, but they also pass through all their parents. See [Event] for more details about the event model. + + * You may also want to override the various layout hints like [minWidth], [maxHeight], etc. In particular [Padding] and [Margin] are often relevant for both container and leaf widgets and the default values of 0 are often not what you want. + ) + + On Microsoft Windows, many widgets are also based on native controls. You can also do this if `static if(UsingWin32Widgets)` passes. You should use the helper function [createWin32Window] to create the window and let minigui do what it needs to do to create its bridge structures. This will populate [Widget.hwnd] which you can access later for communcating with the native window. You may also consider overriding [Widget.handleWmCommand] and [Widget.handleWmNotify] for the widget to translate those messages into appropriate minigui [Event]s. + + It is also possible to embed a [SimpleWindow]-based native window inside a widget. See [OpenGlWidget]'s source code as an example. + + Your own custom-drawn and native system controls can exist side-by-side. + + Later I'll add more complete examples, but for now [TextLabel] and [LabeledPasswordEdit] are both simple widgets you can view implementation to get some ideas. +/ class Widget { @@ -412,9 +502,13 @@ class Widget { selected = (1 << 5), /// the widget represents one option of many and is currently selected, but is not necessarily focused nor checked. disabled = (1 << 6), /// the widget is currently unable to perform its designated task indeterminate = (1 << 7), /// the widget has tri-state and is between checked and not checked + depressed = (1 << 8), /// the widget is being actively pressed or clicked (compare to css `:active`). Can be combined with hover to visually indicate if a mouse up would result in a click event. USER_BEGIN = (1UL << 32), } + + // I want to add the primary and cancel styles to buttons at least at some point somehow. + /// ditto @property ulong dynamicState() { return dynamicState_; } /// ditto @@ -457,6 +551,8 @@ class Widget { It is here so there can be a specificity switch. + See [OverrideStyle] for a helper function to use your own. + History: Added May 11, 2021 +/ @@ -533,18 +629,46 @@ class Widget { } /++ - This goes in order like this: + This mixin overrides the [useStyleProperties] method to direct it toward your own style class. + The basic usage is simple: + --- + static class Style : YourParentClass.Style { /* YourParentClass is frequently Widget, of course, but not always */ + // override style hints as-needed here + } + OverrideStyle!Style; // add the method + --- + + $(TIP + While the class is not forced to be `static`, for best results, it should be. A non-static class + can not be inherited by other objects whereas the static one can. A property on the base class, + called [Widget.Style.widget|widget], is available for you to access its properties. + ) + + This exists just because [useStyleProperties] has a somewhat convoluted signature and its overrides must + repeat them. Moreover, its implementation uses a stack class to optimize GC pressure from small fetches + and that's a little tedious to repeat in your child classes too when you only care about changing the type. + + + It also has a further facility to pick a wholly differnet class based on the [DynamicState] of the Widget. + + --- mixin OverrideStyle!( DynamicState.focus, YourFocusedStyle, DynamicState.hover, YourHoverStyle, YourDefaultStyle ) + --- It checks if `dynamicState` matches the state and if so, returns the object given. - If there is no state mask given, the next one matches everything. The first match - given is used. + If there is no state mask given, the next one matches everything. The first match given is used. + + However, since in most cases you'll want check state inside your individual methods, you probably won't + find much use for this whole-class swap out. + + History: + Added May 16, 2021 +/ static protected mixin template OverrideStyle(S...) { override void useStyleProperties(scope void delegate(scope Widget.Style props) dg) { @@ -554,6 +678,7 @@ class Widget { mask = thing; } else { if(!(idx & 1) || (this.dynamicState & mask) == mask) { + //static assert(!__traits(isNested, thing), thing.stringof ~ " is a nested class. For best results, mark it `static`. You can still access the widget through a `widget` variable inside the Style class."); scope Widget.Style s = new thing(); s.widget = this; dg(s); @@ -574,8 +699,7 @@ class Widget { protected void sendResizeEvent() { - auto event = new Event("resize", this); - event.sendDirectly(); + this.emit!ResizeEvent(); } Menu contextMenu(int x, int y) { return null; } @@ -745,9 +869,13 @@ class Widget { addDirectEventListener just inserts a check `if(e.target !is this) return;` meaning it opts out of participating in handler delegation. + + $(TIP + Use `scope` on your handlers when you can. While it currently does nothing, this will future-proof your code against future optimizations I want to do. Instead of copying whole event objects out if you do need to store them, just copy the properties you need. + ) +/ EventListener addDirectEventListener(string event, void delegate() handler, bool useCapture = false) { - return addEventListener(event, (Widget, Event e) { + return addEventListener(event, (Widget, scope Event e) { if(e.srcElement is this) handler(); }, useCapture); @@ -764,7 +892,7 @@ class Widget { /// ditto @scriptable EventListener addEventListener(string event, void delegate() handler, bool useCapture = false) { - return addEventListener(event, (Widget, Event) { handler(); }, useCapture); + return addEventListener(event, (Widget, scope Event) { handler(); }, useCapture); } /// ditto @@ -865,9 +993,11 @@ class Widget { } version(win32_widgets) + /// Called when a WM_COMMAND is sent to the associated hwnd. void handleWmCommand(ushort cmd, ushort id) {} version(win32_widgets) + /// Called when a WM_NOTIFY is sent to the associated hwnd. int handleWmNotify(NMHDR* hdr, int code) { return 0; } /++ @@ -882,7 +1012,7 @@ class Widget { /++ If true, this widget can be focused via keyboard control with the tab key. - If false, it is assumed the widget itself does not contain the keyboard focus (though its childen are free to). + If false, it is assumed the widget itself does will never receive the keyboard focus (though its childen are free to). +/ bool tabStop = true; /++ @@ -892,6 +1022,7 @@ class Widget { version(win32_widgets) { static Widget[HWND] nativeMapping; + /// The native handle, if there is one. HWND hwnd; WNDPROC originalWindowProcedure; @@ -979,7 +1110,7 @@ class Widget { } /// Creates the widget and adds it to the parent. - this(Widget parent = null) { + this(Widget parent) { if(parent !is null) parent.addChild(this); setupDefaultEventHandlers(); @@ -1037,10 +1168,10 @@ class Widget { if(parentWindow.focusedWidget) { // FIXME: more details here? like from and to - auto evt = new Event("blur", parentWindow.focusedWidget); + auto from = parentWindow.focusedWidget; parentWindow.focusedWidget.setDynamicState(DynamicState.focus, false); parentWindow.focusedWidget = null; - evt.sendDirectly(); + from.emit!BlurEvent(); } @@ -1051,8 +1182,7 @@ class Widget { parentWindow.focusedWidget = this; parentWindow.focusedWidget.setDynamicState(DynamicState.focus, true); - auto evt = new Event("focus", this); - evt.dispatch(); + this.emit!FocusEvent(); } @@ -1123,9 +1253,43 @@ class Widget { /++ Responsible for actually painting the widget to the screen. The clip rectangle and coordinate translation in the [WidgetPainter] are pre-configured so you can draw independently. + This function paints the entire widget, including styled borders, backgrounds, etc. You are also responsible for displaying any important active state to the user, including if you hold the active keyboard focus. If you only want to be responsible for the content while letting the style engine draw the rest, override [paintContent] instead. + + [paint] is not called for system widgets as the OS library draws them instead. + + + The default implementation forwards to [WidgetPainter.drawThemed], passing [paintContent] as the delegate. If you override this, you might use those same functions or you can do your own thing. + You should also look at [WidgetPainter.visualTheme] to be theme aware. + + History: + Prior to May 15, 2021, the default implementation was empty. Now, it is `painter.drawThemed(&paintContent);`. You may wish to override [paintContent] instead of [paint] to take advantage of the new styling engine. +/ - void paint(WidgetPainter painter) {} + void paint(WidgetPainter painter) { + painter.drawThemed(&paintContent); // note this refers to the following overload + } + + /++ + Responsible for drawing the content as the theme engine is responsible for other elements. + + $(WARNING If you override [paint], this method may never be used as it is only called from inside the default implementation of `paint`.) + + Params: + painter = your painter (forwarded from [paint]) for drawing on the widget. The clip rectangle and coordinate translation are prepared for you ahead of time so you can use widget coordinates. It also has the theme foreground preloaded into the painter outline color, the theme font preloaded as the painter's active font, and the theme background preloaded as the painter's fill color. + + bounds = the bounds, inside the widget, where your content should be drawn. This is the rectangle inside the border and padding (if any). The stuff outside is not clipped - it is still part of your widget - but you should respect these bounds for visual consistency and respecting the theme's area. + + If you do want to clip it, you can of course call `auto oldClip = painter.setClipRectangle(bounds); scope(exit) painter.setClipRectangle(oldClip);` to modify it and return to the previous setting when you return. + + Returns: + The rectangle representing your actual content. Typically, this is simply `return bounds;`. The theme engine uses this return value to determine where the outline and overlay should be. + + History: + Added May 15, 2021 + +/ + Rectangle paintContent(WidgetPainter painter, const Rectangle bounds) { + return bounds; + } deprecated("Change ScreenPainter to WidgetPainter") final void paint(ScreenPainter) { assert(0, "Change ScreenPainter to WidgetPainter and recompile your code"); } @@ -1274,7 +1438,7 @@ class Widget { +/ final protected bool emit(EventType, this This, Args...)(Args args) { static assert(classStaticallyEmits!(This, EventType), "The " ~ This.stringof ~ " class is not declared to emit " ~ EventType.stringof); - auto e = new EventType(args, this); + auto e = new EventType(this, args); e.dispatch(); return !e.defaultPrevented; } @@ -1308,6 +1472,10 @@ class Widget { return StyleInformation(this); } + // FIXME: I kinda want to hide events from implementation widgets + // so it just catches them all and stops propagation... + // i guess i can do it with a event listener on star. + mixin Emits!KeyDownEvent; /// mixin Emits!KeyUpEvent; /// mixin Emits!CharEvent; /// @@ -1315,11 +1483,17 @@ class Widget { mixin Emits!MouseDownEvent; /// mixin Emits!MouseUpEvent; /// mixin Emits!ClickEvent; /// + mixin Emits!DoubleClickEvent; /// mixin Emits!MouseMoveEvent; /// mixin Emits!MouseOverEvent; /// mixin Emits!MouseOutEvent; /// mixin Emits!MouseEnterEvent; /// mixin Emits!MouseLeaveEvent; /// + + mixin Emits!ResizeEvent; /// + + mixin Emits!BlurEvent; /// + mixin Emits!FocusEvent; /// } /// @@ -1327,12 +1501,12 @@ abstract class ComboboxBase : Widget { // if the user can enter arbitrary data, we want to use 2 == CBS_DROPDOWN // or to always show the list, we want CBS_SIMPLE == 1 version(win32_widgets) - this(uint style, Widget parent = null) { + this(uint style, Widget parent) { super(parent); createWin32Window(this, "ComboBox"w, null, style); } else version(custom_widgets) - this(Widget parent = null) { + this(Widget parent) { super(parent); addEventListener((KeyDownEvent event) { @@ -1461,7 +1635,7 @@ abstract class ComboboxBase : Widget { given options. Like `