mirror of https://github.com/adamdruppe/arsd.git
initial version of date/time/calendar pickers
This commit is contained in:
parent
d773fb4221
commit
28f09ae118
11
dub.json
11
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.",
|
||||
|
|
|
@ -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));
|
||||
}
|
Loading…
Reference in New Issue