diff --git a/dub.json b/dub.json index 04e27b5..48afcc2 100644 --- a/dub.json +++ b/dub.json @@ -92,6 +92,17 @@ "importPaths": ["."], "sourceFiles": ["minigui_addons/color_dialog.d"] }, + { + "name": "minigui-datetime_picker", + "description": "Date and time picker widget add-on for minigui. New in v10.7.", + "targetType": "library", + "dflags": ["-mv=arsd.minigui_addons.datetime_picker=minigui_addons/datetime_picker.d"], + "dependencies": { + "arsd-official:minigui":"*" + }, + "importPaths": ["."], + "sourceFiles": ["minigui_addons/datetime_picker.d"] + }, { "name": "minigui-webview", "description": "Webview widget add-on for minigui. New to dub in v10.5 but NOT STABLE in that release and it will break at random without notice until I say it is stable.", diff --git a/minigui_addons/datetime_picker.d b/minigui_addons/datetime_picker.d new file mode 100644 index 0000000..3fe39ba --- /dev/null +++ b/minigui_addons/datetime_picker.d @@ -0,0 +1,334 @@ +/++ + Add-on to [arsd.minigui] to provide date and time widgets. + + History: + Added March 22, 2022 (dub v10.7) + + Bugs: + The Linux implementation is currently extremely minimal. The Windows implementation has more actual graphical functionality. ++/ +module arsd.minigui_addons.datetime_picker; + +import arsd.minigui; + +import std.datetime; + +static if(UsingWin32Widgets) { + import core.sys.windows.windows; + import core.sys.windows.commctrl; +} + +/++ + A DatePicker is a single row input for picking a date. It can drop down a calendar to help the user pick the date they want. + + See also: [TimePicker], [CalendarPicker] ++/ +// on Windows these support a min/max range too +class DatePicker : Widget { + /// + this(Widget parent) { + super(parent); + static if(UsingWin32Widgets) { + createWin32Window(this, "SysDateTimePick32"w, null, 0); + } else { + date = new LabeledLineEdit("Date (YYYY-Mon-DD)", TextAlignment.Right, this); + + date.addEventListener((ChangeEvent!string ce) { changed(); }); + + this.tabStop = false; + } + } + + private Date value_; + + /++ + Current value the user selected. Please note this is NOT valid until AFTER a change event is emitted. + +/ + Date value() { + return value_; + } + + /++ + Changes the current value displayed. Will not send a change event. + +/ + void value(Date v) { + static if(UsingWin32Widgets) { + SYSTEMTIME st; + st.wYear = v.year; + st.wMonth = v.month; + st.wDay = v.day; + SendMessage(hwnd, DTM_SETSYSTEMTIME, GDT_VALID, cast(LPARAM) &st); + } else { + date.content = value_.toSimpleString(); + } + } + + static if(UsingCustomWidgets) private { + LabeledLineEdit date; + string lastMsg; + + void changed() { + try { + value_ = Date.fromSimpleString(date.content); + + this.emit!(ChangeEvent!Date)(&value); + } catch(Exception e) { + if(e.msg != lastMsg) { + messageBox(e.msg); + lastMsg = e.msg; + } + } + } + } + + + + static if(UsingWin32Widgets) { + override int minHeight() { return defaultLineHeight + 6; } + override int maxHeight() { return defaultLineHeight + 6; } + + override int handleWmNotify(NMHDR* hdr, int code, out int mustReturn) { + switch(code) { + case DTN_DATETIMECHANGE: + auto lpChange = cast(LPNMDATETIMECHANGE) hdr; + if(true || (lpChange.dwFlags & GDT_VALID)) { // this flag only set if you use SHOWNONE + auto st = lpChange.st; + value_ = Date(st.wYear, st.wMonth, st.wDay); + + this.emit!(ChangeEvent!Date)(&value); + + mustReturn = true; + } + break; + default: + } + return false; + } + } else { + override int minHeight() { return defaultLineHeight + 4; } + override int maxHeight() { return defaultLineHeight + 4; } + } + + override bool encapsulatedChildren() { + return true; + } + + mixin Emits!(ChangeEvent!Date); +} + +/++ + A TimePicker is a single row input for picking a time. It does not work with timezones. + + See also: [DatePicker] ++/ +class TimePicker : Widget { + /// + this(Widget parent) { + super(parent); + static if(UsingWin32Widgets) { + createWin32Window(this, "SysDateTimePick32"w, null, DTS_TIMEFORMAT); + } else { + time = new LabeledLineEdit("Time", TextAlignment.Right, this); + + time.addEventListener((ChangeEvent!string ce) { changed(); }); + + this.tabStop = false; + } + + } + + private TimeOfDay value_; + + static if(UsingCustomWidgets) private { + LabeledLineEdit time; + string lastMsg; + + void changed() { + try { + value_ = TimeOfDay.fromISOExtString(time.content); + + this.emit!(ChangeEvent!TimeOfDay)(&value); + } catch(Exception e) { + if(e.msg != lastMsg) { + messageBox(e.msg); + lastMsg = e.msg; + } + } + } + } + + + /++ + Current value the user selected. Please note this is NOT valid until AFTER a change event is emitted. + +/ + TimeOfDay value() { + return value_; + } + + /++ + Changes the current value displayed. Will not send a change event. + +/ + void value(TimeOfDay v) { + static if(UsingWin32Widgets) { + SYSTEMTIME st; + st.wHour = v.hour; + st.wMinute = v.minute; + st.wSecond = v.second; + SendMessage(hwnd, DTM_SETSYSTEMTIME, GDT_VALID, cast(LPARAM) &st); + } else { + time.content = value_.toISOExtString(); + } + } + + static if(UsingWin32Widgets) { + override int minHeight() { return defaultLineHeight + 6; } + override int maxHeight() { return defaultLineHeight + 6; } + + override int handleWmNotify(NMHDR* hdr, int code, out int mustReturn) { + switch(code) { + case DTN_DATETIMECHANGE: + auto lpChange = cast(LPNMDATETIMECHANGE) hdr; + if(true || (lpChange.dwFlags & GDT_VALID)) { // this flag only set if you use SHOWNONE + auto st = lpChange.st; + value_ = TimeOfDay(st.wHour, st.wMinute, st.wSecond); + + this.emit!(ChangeEvent!TimeOfDay)(&value); + + mustReturn = true; + } + break; + default: + } + return false; + } + + } else { + override int minHeight() { return defaultLineHeight + 4; } + override int maxHeight() { return defaultLineHeight + 4; } + } + + override bool encapsulatedChildren() { + return true; + } + + mixin Emits!(ChangeEvent!TimeOfDay); +} + +/++ + A CalendarPicker is a rectangular input for picking a date or a range of dates on a + calendar viewer. + + The current value is an [Interval] of dates. Please note that the interval is non-inclusive, + that is, the end day is one day $(I after) the final date the user selected. + + If the user only selected one date, start will be the selection and end is the day after. ++/ +/+ + Note the Windows control also supports bolding dates, changing the max selection count, + week numbers, and more. ++/ +class CalendarPicker : Widget { + /// + this(Widget parent) { + super(parent); + static if(UsingWin32Widgets) { + createWin32Window(this, "SysMonthCal32"w, null, MCS_MULTISELECT); + SendMessage(hwnd, MCM_SETMAXSELCOUNT, int.max, 0); + } else { + start = new LabeledLineEdit("Start", this); + end = new LabeledLineEdit("End", this); + + start.addEventListener((ChangeEvent!string ce) { changed(); }); + end.addEventListener((ChangeEvent!string ce) { changed(); }); + + this.tabStop = false; + } + } + + static if(UsingCustomWidgets) private { + LabeledLineEdit start; + LabeledLineEdit end; + string lastMsg; + + void changed() { + try { + value_ = Interval!Date( + Date.fromSimpleString(start.content), + Date.fromSimpleString(end.content) + 1.days + ); + + this.emit!(ChangeEvent!(Interval!Date))(&value); + } catch(Exception e) { + if(e.msg != lastMsg) { + messageBox(e.msg); + lastMsg = e.msg; + } + } + } + } + + private Interval!Date value_; + + /++ + Current value the user selected. Please note this is NOT valid until AFTER a change event is emitted. + +/ + Interval!Date value() { return value_; } + + /++ + Sets a new interval. Remember, the end date of the interval is NOT included. You might want to `end + 1.days` when creating it. + +/ + void value(Interval!Date v) { + value_ = v; + + auto end = v.end - 1.days; + + static if(UsingWin32Widgets) { + SYSTEMTIME[2] arr; + + arr[0].wYear = v.begin.year; + arr[0].wMonth = v.begin.month; + arr[0].wDay = v.begin.day; + + arr[1].wYear = end.year; + arr[1].wMonth = end.month; + arr[1].wDay = end.day; + + SendMessage(hwnd, MCM_SETSELRANGE, 0, cast(LPARAM) arr.ptr); + } else { + this.start.content = v.begin.toString(); + this.end.content = end.toString(); + } + } + + static if(UsingWin32Widgets) { + override int handleWmNotify(NMHDR* hdr, int code, out int mustReturn) { + switch(code) { + case MCN_SELECT: + auto lpChange = cast(LPNMSELCHANGE) hdr; + auto start = lpChange.stSelStart; + auto end = lpChange.stSelEnd; + + auto et = Date(end.wYear, end.wMonth, end.wDay); + et += dur!"days"(1); + + value_ = Interval!Date( + Date(start.wYear, start.wMonth, start.wDay), + Date(end.wYear, end.wMonth, end.wDay) + 1.days // the interval is non-inclusive + ); + + this.emit!(ChangeEvent!(Interval!Date))(&value); + + mustReturn = true; + break; + default: + } + return false; + } + } + + override bool encapsulatedChildren() { + return true; + } + + mixin Emits!(ChangeEvent!(Interval!Date)); +}