x4js 2.0.35 → 2.1.0-manual.2
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/README.md +21 -21
- package/package.json +39 -26
- package/src/components/base.scss +25 -89
- package/src/components/boxes/boxes.module.scss +54 -54
- package/src/components/boxes/boxes.ts +513 -513
- package/src/components/breadcrumb/breadcrumb.scss +56 -56
- package/src/components/breadcrumb/breadcrumb.ts +93 -93
- package/src/components/btngroup/btngroup.module.scss +40 -40
- package/src/components/btngroup/btngroup.ts +152 -152
- package/src/components/button/button.module.scss +172 -172
- package/src/components/button/button.ts +232 -232
- package/src/components/calendar/calendar.module.scss +162 -162
- package/src/components/calendar/calendar.ts +326 -326
- package/src/components/canvas/canvas.module.scss +24 -24
- package/src/components/canvas/canvas.ts +195 -195
- package/src/components/canvas/canvas_ex.ts +275 -275
- package/src/components/checkbox/check.svg +3 -3
- package/src/components/checkbox/checkbox.module.scss +141 -141
- package/src/components/checkbox/checkbox.ts +139 -139
- package/src/components/colorinput/colorinput.module.scss +64 -64
- package/src/components/colorinput/colorinput.ts +90 -90
- package/src/components/colorpicker/colorpicker.module.scss +132 -132
- package/src/components/colorpicker/colorpicker.ts +481 -481
- package/src/components/combobox/combobox.module.scss +145 -145
- package/src/components/combobox/combobox.ts +282 -282
- package/src/components/combobox/updown.svg +3 -3
- package/src/components/components.ts +45 -44
- package/src/components/dialog/dialog.module.scss +103 -105
- package/src/components/dialog/dialog.ts +233 -233
- package/src/components/filedrop/filedrop.module.scss +69 -69
- package/src/components/filedrop/filedrop.ts +130 -130
- package/src/components/form/form.module.scss +38 -38
- package/src/components/form/form.ts +172 -172
- package/src/components/gridview/gridview.module.scss +323 -337
- package/src/components/gridview/gridview.ts +1276 -1316
- package/src/components/header/header.module.scss +40 -40
- package/src/components/header/header.ts +141 -141
- package/src/components/icon/icon.module.scss +32 -32
- package/src/components/icon/icon.ts +165 -165
- package/src/components/image/image.module.scss +27 -27
- package/src/components/image/image.ts +168 -168
- package/src/components/input/input.module.scss +74 -74
- package/src/components/input/input.ts +537 -537
- package/src/components/keyboard/keyboard.module.scss +136 -136
- package/src/components/keyboard/keyboard.ts +549 -549
- package/src/components/label/label.module.scss +90 -91
- package/src/components/label/label.ts +101 -101
- package/src/components/link/link.module.scss +44 -44
- package/src/components/link/link.ts +87 -87
- package/src/components/listbox/listbox.module.scss +179 -179
- package/src/components/listbox/listbox.ts +596 -596
- package/src/components/menu/menu.module.scss +128 -128
- package/src/components/menu/menu.ts +174 -174
- package/src/components/messages/messages.module.scss +92 -146
- package/src/components/messages/messages.ts +237 -303
- package/src/components/normalize.scss +391 -391
- package/src/components/notification/notification.module.scss +83 -83
- package/src/components/notification/notification.ts +107 -107
- package/src/components/panel/panel.module.scss +66 -71
- package/src/components/panel/panel.ts +57 -57
- package/src/components/popup/popup.module.scss +51 -51
- package/src/components/popup/popup.ts +457 -457
- package/src/components/progress/progress.module.scss +56 -56
- package/src/components/progress/progress.ts +43 -43
- package/src/components/propgrid/progrid.module.scss +111 -111
- package/src/components/propgrid/propgrid.ts +300 -300
- package/src/components/propgrid/updown.svg +3 -3
- package/src/components/radio/radio.module.scss +163 -163
- package/src/components/radio/radio.svg +3 -3
- package/src/components/radio/radio.ts +141 -141
- package/src/components/rating/rating.module.scss +22 -22
- package/src/components/rating/rating.ts +131 -131
- package/src/components/select/select.module.scss +8 -8
- package/src/components/select/select.ts +134 -134
- package/src/components/shared.scss +141 -71
- package/src/components/sizers/sizer.module.scss +90 -112
- package/src/components/sizers/sizer.ts +131 -155
- package/src/components/slider/slider.module.scss +117 -117
- package/src/components/slider/slider.ts +197 -197
- package/src/components/spreadsheet/spreadsheet.module.scss +307 -307
- package/src/components/spreadsheet/spreadsheet.ts +1223 -1223
- package/src/components/switch/switch.module.scss +126 -126
- package/src/components/switch/switch.ts +61 -61
- package/src/components/tabs/tabs.module.scss +46 -67
- package/src/components/tabs/tabs.ts +229 -234
- package/src/components/textarea/textarea.module.scss +63 -63
- package/src/components/textarea/textarea.ts +131 -131
- package/src/components/textedit/textedit.module.scss +115 -115
- package/src/components/textedit/textedit.ts +122 -122
- package/src/components/themes.scss +90 -90
- package/src/components/tickline/tickline.module.scss +25 -25
- package/src/components/tickline/tickline.ts +81 -81
- package/src/components/tooltips/tooltips.scss +71 -71
- package/src/components/tooltips/tooltips.ts +120 -120
- package/src/components/treeview/treeview.module.scss +192 -192
- package/src/components/treeview/treeview.ts +484 -484
- package/src/components/viewport/viewport.module.scss +31 -31
- package/src/components/viewport/viewport.ts +41 -41
- package/src/core/component.ts +1299 -1299
- package/src/core/core_application.ts +361 -361
- package/src/core/core_colors.ts +512 -512
- package/src/core/core_data.ts +1297 -1310
- package/src/core/core_dom.ts +481 -481
- package/src/core/core_dragdrop.ts +225 -225
- package/src/core/core_element.ts +221 -221
- package/src/core/core_events.ts +214 -214
- package/src/core/core_i18n.ts +395 -395
- package/src/core/core_pdf.ts +454 -454
- package/src/core/core_react.ts +78 -78
- package/src/core/core_router.ts +296 -296
- package/src/core/core_state.ts +62 -62
- package/src/core/core_styles.ts +213 -213
- package/src/core/core_svg.ts +1042 -1042
- package/src/core/core_tools.ts +996 -996
- package/src/types/scss.d.ts +4 -4
- package/src/types/x4react.d.ts +8 -8
- package/src/x4.scss +19 -19
- package/src/x4.ts +36 -36
- package/src/x4tsx.d.ts +26 -26
- package/.vscode/launch.json +0 -14
- package/.vscode/settings.json +0 -2
- package/demo/assets/house-light.svg +0 -1
- package/demo/assets/radio.svg +0 -4
- package/demo/index.html +0 -12
- package/demo/main.scss +0 -23
- package/demo/main.ts +0 -324
- package/demo/package.json +0 -26
- package/demo/scss.d.ts +0 -4
- package/demo/svg.d.ts +0 -1
- package/demo/tsconfig.json +0 -14
- package/src/components/gridview/folder-open.svg +0 -1
- package/src/components/messages/spinner.svg +0 -1
- package/src/x4.d.ts +0 -10
- package/tsconfig.json +0 -11
|
@@ -1,327 +1,327 @@
|
|
|
1
|
-
/**
|
|
2
|
-
* ___ ___ __
|
|
3
|
-
* \ \/ / / _
|
|
4
|
-
* \ / /_| |_
|
|
5
|
-
* / \____ _|
|
|
6
|
-
* /__/\__\ |_|
|
|
7
|
-
*
|
|
8
|
-
* @file calendar.ts
|
|
9
|
-
* @author Etienne Cochard
|
|
10
|
-
*
|
|
11
|
-
* @copyright (c) 2024 R-libre ingenierie
|
|
12
|
-
*
|
|
13
|
-
* Use of this source code is governed by an MIT-style license
|
|
14
|
-
* that can be found in the LICENSE file or at https://opensource.org/licenses/MIT.
|
|
15
|
-
**/
|
|
16
|
-
|
|
17
|
-
import { Component, ComponentEvents, ComponentProps, EvChange, Flex } from '../../core/component'
|
|
18
|
-
import { class_ns, date_clone, date_hash, formatIntlDate, Point, unsafeHtml } from "../../core/core_tools"
|
|
19
|
-
import { _tr } from '../../core/core_i18n';
|
|
20
|
-
import { EventCallback } from '../../core/core_events';
|
|
21
|
-
|
|
22
|
-
import { Button } from '../button/button';
|
|
23
|
-
import { Label } from '../label/label';
|
|
24
|
-
import { HBox, VBox } from '../boxes/boxes'
|
|
25
|
-
import { Menu, MenuItem } from '../menu/menu';
|
|
26
|
-
|
|
27
|
-
import "./calendar.module.scss"
|
|
28
|
-
|
|
29
|
-
import icon_prev from "./chevron-left-sharp-light.svg";
|
|
30
|
-
import icon_today from "./calendar-check-sharp-light.svg";
|
|
31
|
-
import icon_next from "./chevron-right-sharp-light.svg";
|
|
32
|
-
|
|
33
|
-
interface CalendarEventMap extends ComponentEvents {
|
|
34
|
-
change?: EvChange;
|
|
35
|
-
}
|
|
36
|
-
|
|
37
|
-
|
|
38
|
-
interface CalendarProps extends ComponentProps {
|
|
39
|
-
date?: Date; // initial date to display
|
|
40
|
-
minDate?: Date; // minimal date before the user cannot go
|
|
41
|
-
maxDate?: Date; // maximal date after the user cannot go
|
|
42
|
-
|
|
43
|
-
change?: EventCallback<EvChange>; // shortcut to events: { change: ... }
|
|
44
|
-
}
|
|
45
|
-
|
|
46
|
-
|
|
47
|
-
/**
|
|
48
|
-
* default calendar control
|
|
49
|
-
*
|
|
50
|
-
* fires:
|
|
51
|
-
* EventChange ( value = Date )
|
|
52
|
-
*/
|
|
53
|
-
|
|
54
|
-
@class_ns( "x4" )
|
|
55
|
-
export class Calendar extends VBox<CalendarProps, CalendarEventMap>
|
|
56
|
-
{
|
|
57
|
-
private m_date: Date;
|
|
58
|
-
|
|
59
|
-
constructor(props: CalendarProps) {
|
|
60
|
-
super(props);
|
|
61
|
-
|
|
62
|
-
this.mapPropEvents( props, 'change' );
|
|
63
|
-
this.m_date = props.date ? date_clone( props.date ) : new Date();
|
|
64
|
-
|
|
65
|
-
this._update( );
|
|
66
|
-
}
|
|
67
|
-
|
|
68
|
-
/** @ignore */
|
|
69
|
-
|
|
70
|
-
private _update( ) {
|
|
71
|
-
|
|
72
|
-
let month_start = date_clone(this.m_date);
|
|
73
|
-
month_start.setDate(1);
|
|
74
|
-
|
|
75
|
-
let day = month_start.getDay();
|
|
76
|
-
if (day == 0) {
|
|
77
|
-
day = 7;
|
|
78
|
-
}
|
|
79
|
-
|
|
80
|
-
month_start.setDate(-day + 1 + 1);
|
|
81
|
-
let dte = date_clone(month_start);
|
|
82
|
-
|
|
83
|
-
let selection = date_hash( this.m_date );
|
|
84
|
-
let today = date_hash( new Date() );
|
|
85
|
-
|
|
86
|
-
let month_end = date_clone(this.m_date);
|
|
87
|
-
month_end.setDate(1);
|
|
88
|
-
month_end.setMonth(month_end.getMonth() + 1);
|
|
89
|
-
month_end.setDate(0);
|
|
90
|
-
|
|
91
|
-
let end_of_month = date_hash(month_end);
|
|
92
|
-
|
|
93
|
-
let rows: HBox[] = [];
|
|
94
|
-
|
|
95
|
-
// month selector
|
|
96
|
-
let header = new HBox({
|
|
97
|
-
cls: 'month-sel',
|
|
98
|
-
content: [
|
|
99
|
-
new Label({
|
|
100
|
-
cls: 'month',
|
|
101
|
-
text: formatIntlDate(this.m_date, 'O'),
|
|
102
|
-
dom_events: {
|
|
103
|
-
click: () => this._choose('month')
|
|
104
|
-
}
|
|
105
|
-
}),
|
|
106
|
-
new Label({
|
|
107
|
-
cls: 'year',
|
|
108
|
-
text: formatIntlDate(this.m_date, 'Y'),
|
|
109
|
-
dom_events: {
|
|
110
|
-
click: () => this._choose('year')
|
|
111
|
-
}
|
|
112
|
-
}),
|
|
113
|
-
new Flex( ),
|
|
114
|
-
new Button({ icon: icon_prev, click: () => this._next(false) } ),
|
|
115
|
-
new Button({ icon: icon_today, click: () => this.setDate(new Date()), tooltip: _tr.global.today } ),
|
|
116
|
-
new Button({ icon: icon_next, click: () => this._next(true) } )
|
|
117
|
-
]
|
|
118
|
-
});
|
|
119
|
-
|
|
120
|
-
rows.push(header);
|
|
121
|
-
|
|
122
|
-
// calendar part
|
|
123
|
-
let day_names = [];
|
|
124
|
-
|
|
125
|
-
// day names
|
|
126
|
-
// empty week num
|
|
127
|
-
day_names.push(new HBox({
|
|
128
|
-
cls: 'weeknum cell',
|
|
129
|
-
}));
|
|
130
|
-
|
|
131
|
-
for (let d = 0; d < 7; d++) {
|
|
132
|
-
day_names.push(new Label({
|
|
133
|
-
cls: 'cell',
|
|
134
|
-
text: _tr.global.day_short[(d + 1) % 7]
|
|
135
|
-
}));
|
|
136
|
-
}
|
|
137
|
-
|
|
138
|
-
rows.push(new HBox({
|
|
139
|
-
cls: 'week header',
|
|
140
|
-
content: day_names
|
|
141
|
-
}));
|
|
142
|
-
|
|
143
|
-
let cmonth = this.m_date.getMonth();
|
|
144
|
-
|
|
145
|
-
// weeks
|
|
146
|
-
let first = true;
|
|
147
|
-
while (date_hash(dte) <= end_of_month) {
|
|
148
|
-
|
|
149
|
-
let days: Component[] = [
|
|
150
|
-
new HBox({ cls: 'weeknum cell', content: new Component({ tag: 'span', content: formatIntlDate(dte, 'w') }) })
|
|
151
|
-
];
|
|
152
|
-
|
|
153
|
-
// days
|
|
154
|
-
for (let d = 0; d < 7; d++) {
|
|
155
|
-
|
|
156
|
-
let cls = 'cell day';
|
|
157
|
-
if (date_hash(dte) == selection) {
|
|
158
|
-
cls += ' selection';
|
|
159
|
-
}
|
|
160
|
-
|
|
161
|
-
if (date_hash(dte) == today) {
|
|
162
|
-
cls += ' today';
|
|
163
|
-
}
|
|
164
|
-
|
|
165
|
-
if (dte.getMonth() != cmonth) {
|
|
166
|
-
cls += ' out';
|
|
167
|
-
}
|
|
168
|
-
|
|
169
|
-
const mkItem = ( dte: Date ) => {
|
|
170
|
-
return new HBox({
|
|
171
|
-
cls,
|
|
172
|
-
flex: 1,
|
|
173
|
-
content: new Component({
|
|
174
|
-
cls: "text",
|
|
175
|
-
content: unsafeHtml( `<span>${formatIntlDate(dte, 'd')}</span>` ),
|
|
176
|
-
}),
|
|
177
|
-
dom_events: {
|
|
178
|
-
click: () => this.select(dte)
|
|
179
|
-
}
|
|
180
|
-
})
|
|
181
|
-
}
|
|
182
|
-
|
|
183
|
-
days.push( mkItem( date_clone( dte ) ) );
|
|
184
|
-
|
|
185
|
-
dte.setDate(dte.getDate() + 1);
|
|
186
|
-
first = false;
|
|
187
|
-
}
|
|
188
|
-
|
|
189
|
-
rows.push(new HBox({
|
|
190
|
-
cls: 'week',
|
|
191
|
-
flex: 1,
|
|
192
|
-
content: days
|
|
193
|
-
}));
|
|
194
|
-
}
|
|
195
|
-
|
|
196
|
-
this.setContent(rows);
|
|
197
|
-
}
|
|
198
|
-
|
|
199
|
-
/**
|
|
200
|
-
* select the given date
|
|
201
|
-
* @param date
|
|
202
|
-
*/
|
|
203
|
-
|
|
204
|
-
private select(date: Date) {
|
|
205
|
-
this.m_date = date;
|
|
206
|
-
this.fire('change', {value:date} );
|
|
207
|
-
this._update();
|
|
208
|
-
}
|
|
209
|
-
|
|
210
|
-
/**
|
|
211
|
-
*
|
|
212
|
-
*/
|
|
213
|
-
|
|
214
|
-
private _next(n: boolean) {
|
|
215
|
-
this.m_date.setMonth(this.m_date.getMonth() + (n ? 1 : -1));
|
|
216
|
-
this._update();
|
|
217
|
-
}
|
|
218
|
-
|
|
219
|
-
/**
|
|
220
|
-
*
|
|
221
|
-
*/
|
|
222
|
-
|
|
223
|
-
private _choose(type: 'month' | 'year') {
|
|
224
|
-
|
|
225
|
-
let items: MenuItem[] = [];
|
|
226
|
-
|
|
227
|
-
if (type == 'month') {
|
|
228
|
-
for (let m = 0; m < 12; m++) {
|
|
229
|
-
items.push(({
|
|
230
|
-
text: _tr.global.month_long[m],
|
|
231
|
-
click: () => { this.m_date.setMonth(m); this._update(); }
|
|
232
|
-
}));
|
|
233
|
-
}
|
|
234
|
-
}
|
|
235
|
-
else if (type == 'year') {
|
|
236
|
-
|
|
237
|
-
let min = this.props.minDate?.getFullYear() ?? 1900;
|
|
238
|
-
let max = this.props.maxDate?.getFullYear() ?? 2037;
|
|
239
|
-
|
|
240
|
-
for (let m = max; m >= min; m--) {
|
|
241
|
-
items.push({
|
|
242
|
-
text: '' + m,
|
|
243
|
-
click: () => { this.m_date.setFullYear(m); this._update(); }
|
|
244
|
-
});
|
|
245
|
-
}
|
|
246
|
-
}
|
|
247
|
-
|
|
248
|
-
let menu = new Menu({
|
|
249
|
-
items
|
|
250
|
-
});
|
|
251
|
-
|
|
252
|
-
let rc = this.getBoundingRect();
|
|
253
|
-
menu.displayAt(rc.left, rc.top);
|
|
254
|
-
}
|
|
255
|
-
|
|
256
|
-
getDate() {
|
|
257
|
-
return this.m_date;
|
|
258
|
-
}
|
|
259
|
-
|
|
260
|
-
setDate(date: Date) {
|
|
261
|
-
this.m_date = date;
|
|
262
|
-
this._update();
|
|
263
|
-
}
|
|
264
|
-
}
|
|
265
|
-
|
|
266
|
-
|
|
267
|
-
|
|
268
|
-
|
|
269
|
-
/**
|
|
270
|
-
* default popup calendar
|
|
271
|
-
* /
|
|
272
|
-
|
|
273
|
-
export class PopupCalendar extends Popup {
|
|
274
|
-
|
|
275
|
-
m_cal: Calendar;
|
|
276
|
-
|
|
277
|
-
constructor(props: CalendarProps) {
|
|
278
|
-
super({ tabIndex: 1 });
|
|
279
|
-
|
|
280
|
-
this.enableMask(false);
|
|
281
|
-
|
|
282
|
-
this.m_cal = new Calendar(props);
|
|
283
|
-
this.m_cal.addClass('@fit');
|
|
284
|
-
|
|
285
|
-
this.setContent(this.m_cal);
|
|
286
|
-
}
|
|
287
|
-
|
|
288
|
-
// binded
|
|
289
|
-
private _handleClick = (e: MouseEvent) => {
|
|
290
|
-
if (!this.dom) {
|
|
291
|
-
return;
|
|
292
|
-
}
|
|
293
|
-
|
|
294
|
-
let newfocus = <HTMLElement>e.target;
|
|
295
|
-
|
|
296
|
-
// child of this: ok
|
|
297
|
-
if (this.dom.contains(newfocus)) {
|
|
298
|
-
return;
|
|
299
|
-
}
|
|
300
|
-
|
|
301
|
-
// menu: ok
|
|
302
|
-
let dest = Component.getElement(newfocus, MenuItem);
|
|
303
|
-
if (dest) {
|
|
304
|
-
return;
|
|
305
|
-
}
|
|
306
|
-
|
|
307
|
-
this.close();
|
|
308
|
-
}
|
|
309
|
-
|
|
310
|
-
/ ** @ignore * /
|
|
311
|
-
show(modal?: boolean, at?: Point ) {
|
|
312
|
-
x4document.addEventListener('mousedown', this._handleClick);
|
|
313
|
-
if( at ) {
|
|
314
|
-
super.displayAt( at.x, at.y, 'top left', undefined, modal );
|
|
315
|
-
}
|
|
316
|
-
else {
|
|
317
|
-
super.show(modal);
|
|
318
|
-
}
|
|
319
|
-
}
|
|
320
|
-
|
|
321
|
-
/ ** @ignore * /
|
|
322
|
-
close() {
|
|
323
|
-
x4document.removeEventListener('mousedown', this._handleClick);
|
|
324
|
-
super.close();
|
|
325
|
-
}
|
|
326
|
-
}
|
|
1
|
+
/**
|
|
2
|
+
* ___ ___ __
|
|
3
|
+
* \ \/ / / _
|
|
4
|
+
* \ / /_| |_
|
|
5
|
+
* / \____ _|
|
|
6
|
+
* /__/\__\ |_|
|
|
7
|
+
*
|
|
8
|
+
* @file calendar.ts
|
|
9
|
+
* @author Etienne Cochard
|
|
10
|
+
*
|
|
11
|
+
* @copyright (c) 2024 R-libre ingenierie
|
|
12
|
+
*
|
|
13
|
+
* Use of this source code is governed by an MIT-style license
|
|
14
|
+
* that can be found in the LICENSE file or at https://opensource.org/licenses/MIT.
|
|
15
|
+
**/
|
|
16
|
+
|
|
17
|
+
import { Component, ComponentEvents, ComponentProps, EvChange, Flex } from '../../core/component'
|
|
18
|
+
import { class_ns, date_clone, date_hash, formatIntlDate, Point, unsafeHtml } from "../../core/core_tools"
|
|
19
|
+
import { _tr } from '../../core/core_i18n';
|
|
20
|
+
import { EventCallback } from '../../core/core_events';
|
|
21
|
+
|
|
22
|
+
import { Button } from '../button/button';
|
|
23
|
+
import { Label } from '../label/label';
|
|
24
|
+
import { HBox, VBox } from '../boxes/boxes'
|
|
25
|
+
import { Menu, MenuItem } from '../menu/menu';
|
|
26
|
+
|
|
27
|
+
import "./calendar.module.scss"
|
|
28
|
+
|
|
29
|
+
import icon_prev from "./chevron-left-sharp-light.svg";
|
|
30
|
+
import icon_today from "./calendar-check-sharp-light.svg";
|
|
31
|
+
import icon_next from "./chevron-right-sharp-light.svg";
|
|
32
|
+
|
|
33
|
+
interface CalendarEventMap extends ComponentEvents {
|
|
34
|
+
change?: EvChange;
|
|
35
|
+
}
|
|
36
|
+
|
|
37
|
+
|
|
38
|
+
interface CalendarProps extends ComponentProps {
|
|
39
|
+
date?: Date; // initial date to display
|
|
40
|
+
minDate?: Date; // minimal date before the user cannot go
|
|
41
|
+
maxDate?: Date; // maximal date after the user cannot go
|
|
42
|
+
|
|
43
|
+
change?: EventCallback<EvChange>; // shortcut to events: { change: ... }
|
|
44
|
+
}
|
|
45
|
+
|
|
46
|
+
|
|
47
|
+
/**
|
|
48
|
+
* default calendar control
|
|
49
|
+
*
|
|
50
|
+
* fires:
|
|
51
|
+
* EventChange ( value = Date )
|
|
52
|
+
*/
|
|
53
|
+
|
|
54
|
+
@class_ns( "x4" )
|
|
55
|
+
export class Calendar extends VBox<CalendarProps, CalendarEventMap>
|
|
56
|
+
{
|
|
57
|
+
private m_date: Date;
|
|
58
|
+
|
|
59
|
+
constructor(props: CalendarProps) {
|
|
60
|
+
super(props);
|
|
61
|
+
|
|
62
|
+
this.mapPropEvents( props, 'change' );
|
|
63
|
+
this.m_date = props.date ? date_clone( props.date ) : new Date();
|
|
64
|
+
|
|
65
|
+
this._update( );
|
|
66
|
+
}
|
|
67
|
+
|
|
68
|
+
/** @ignore */
|
|
69
|
+
|
|
70
|
+
private _update( ) {
|
|
71
|
+
|
|
72
|
+
let month_start = date_clone(this.m_date);
|
|
73
|
+
month_start.setDate(1);
|
|
74
|
+
|
|
75
|
+
let day = month_start.getDay();
|
|
76
|
+
if (day == 0) {
|
|
77
|
+
day = 7;
|
|
78
|
+
}
|
|
79
|
+
|
|
80
|
+
month_start.setDate(-day + 1 + 1);
|
|
81
|
+
let dte = date_clone(month_start);
|
|
82
|
+
|
|
83
|
+
let selection = date_hash( this.m_date );
|
|
84
|
+
let today = date_hash( new Date() );
|
|
85
|
+
|
|
86
|
+
let month_end = date_clone(this.m_date);
|
|
87
|
+
month_end.setDate(1);
|
|
88
|
+
month_end.setMonth(month_end.getMonth() + 1);
|
|
89
|
+
month_end.setDate(0);
|
|
90
|
+
|
|
91
|
+
let end_of_month = date_hash(month_end);
|
|
92
|
+
|
|
93
|
+
let rows: HBox[] = [];
|
|
94
|
+
|
|
95
|
+
// month selector
|
|
96
|
+
let header = new HBox({
|
|
97
|
+
cls: 'month-sel',
|
|
98
|
+
content: [
|
|
99
|
+
new Label({
|
|
100
|
+
cls: 'month',
|
|
101
|
+
text: formatIntlDate(this.m_date, 'O'),
|
|
102
|
+
dom_events: {
|
|
103
|
+
click: () => this._choose('month')
|
|
104
|
+
}
|
|
105
|
+
}),
|
|
106
|
+
new Label({
|
|
107
|
+
cls: 'year',
|
|
108
|
+
text: formatIntlDate(this.m_date, 'Y'),
|
|
109
|
+
dom_events: {
|
|
110
|
+
click: () => this._choose('year')
|
|
111
|
+
}
|
|
112
|
+
}),
|
|
113
|
+
new Flex( ),
|
|
114
|
+
new Button({ icon: icon_prev, click: () => this._next(false) } ),
|
|
115
|
+
new Button({ icon: icon_today, click: () => this.setDate(new Date()), tooltip: _tr.global.today } ),
|
|
116
|
+
new Button({ icon: icon_next, click: () => this._next(true) } )
|
|
117
|
+
]
|
|
118
|
+
});
|
|
119
|
+
|
|
120
|
+
rows.push(header);
|
|
121
|
+
|
|
122
|
+
// calendar part
|
|
123
|
+
let day_names = [];
|
|
124
|
+
|
|
125
|
+
// day names
|
|
126
|
+
// empty week num
|
|
127
|
+
day_names.push(new HBox({
|
|
128
|
+
cls: 'weeknum cell',
|
|
129
|
+
}));
|
|
130
|
+
|
|
131
|
+
for (let d = 0; d < 7; d++) {
|
|
132
|
+
day_names.push(new Label({
|
|
133
|
+
cls: 'cell',
|
|
134
|
+
text: _tr.global.day_short[(d + 1) % 7]
|
|
135
|
+
}));
|
|
136
|
+
}
|
|
137
|
+
|
|
138
|
+
rows.push(new HBox({
|
|
139
|
+
cls: 'week header',
|
|
140
|
+
content: day_names
|
|
141
|
+
}));
|
|
142
|
+
|
|
143
|
+
let cmonth = this.m_date.getMonth();
|
|
144
|
+
|
|
145
|
+
// weeks
|
|
146
|
+
let first = true;
|
|
147
|
+
while (date_hash(dte) <= end_of_month) {
|
|
148
|
+
|
|
149
|
+
let days: Component[] = [
|
|
150
|
+
new HBox({ cls: 'weeknum cell', content: new Component({ tag: 'span', content: formatIntlDate(dte, 'w') }) })
|
|
151
|
+
];
|
|
152
|
+
|
|
153
|
+
// days
|
|
154
|
+
for (let d = 0; d < 7; d++) {
|
|
155
|
+
|
|
156
|
+
let cls = 'cell day';
|
|
157
|
+
if (date_hash(dte) == selection) {
|
|
158
|
+
cls += ' selection';
|
|
159
|
+
}
|
|
160
|
+
|
|
161
|
+
if (date_hash(dte) == today) {
|
|
162
|
+
cls += ' today';
|
|
163
|
+
}
|
|
164
|
+
|
|
165
|
+
if (dte.getMonth() != cmonth) {
|
|
166
|
+
cls += ' out';
|
|
167
|
+
}
|
|
168
|
+
|
|
169
|
+
const mkItem = ( dte: Date ) => {
|
|
170
|
+
return new HBox({
|
|
171
|
+
cls,
|
|
172
|
+
flex: 1,
|
|
173
|
+
content: new Component({
|
|
174
|
+
cls: "text",
|
|
175
|
+
content: unsafeHtml( `<span>${formatIntlDate(dte, 'd')}</span>` ),
|
|
176
|
+
}),
|
|
177
|
+
dom_events: {
|
|
178
|
+
click: () => this.select(dte)
|
|
179
|
+
}
|
|
180
|
+
})
|
|
181
|
+
}
|
|
182
|
+
|
|
183
|
+
days.push( mkItem( date_clone( dte ) ) );
|
|
184
|
+
|
|
185
|
+
dte.setDate(dte.getDate() + 1);
|
|
186
|
+
first = false;
|
|
187
|
+
}
|
|
188
|
+
|
|
189
|
+
rows.push(new HBox({
|
|
190
|
+
cls: 'week',
|
|
191
|
+
flex: 1,
|
|
192
|
+
content: days
|
|
193
|
+
}));
|
|
194
|
+
}
|
|
195
|
+
|
|
196
|
+
this.setContent(rows);
|
|
197
|
+
}
|
|
198
|
+
|
|
199
|
+
/**
|
|
200
|
+
* select the given date
|
|
201
|
+
* @param date
|
|
202
|
+
*/
|
|
203
|
+
|
|
204
|
+
private select(date: Date) {
|
|
205
|
+
this.m_date = date;
|
|
206
|
+
this.fire('change', {value:date} );
|
|
207
|
+
this._update();
|
|
208
|
+
}
|
|
209
|
+
|
|
210
|
+
/**
|
|
211
|
+
*
|
|
212
|
+
*/
|
|
213
|
+
|
|
214
|
+
private _next(n: boolean) {
|
|
215
|
+
this.m_date.setMonth(this.m_date.getMonth() + (n ? 1 : -1));
|
|
216
|
+
this._update();
|
|
217
|
+
}
|
|
218
|
+
|
|
219
|
+
/**
|
|
220
|
+
*
|
|
221
|
+
*/
|
|
222
|
+
|
|
223
|
+
private _choose(type: 'month' | 'year') {
|
|
224
|
+
|
|
225
|
+
let items: MenuItem[] = [];
|
|
226
|
+
|
|
227
|
+
if (type == 'month') {
|
|
228
|
+
for (let m = 0; m < 12; m++) {
|
|
229
|
+
items.push(({
|
|
230
|
+
text: _tr.global.month_long[m],
|
|
231
|
+
click: () => { this.m_date.setMonth(m); this._update(); }
|
|
232
|
+
}));
|
|
233
|
+
}
|
|
234
|
+
}
|
|
235
|
+
else if (type == 'year') {
|
|
236
|
+
|
|
237
|
+
let min = this.props.minDate?.getFullYear() ?? 1900;
|
|
238
|
+
let max = this.props.maxDate?.getFullYear() ?? 2037;
|
|
239
|
+
|
|
240
|
+
for (let m = max; m >= min; m--) {
|
|
241
|
+
items.push({
|
|
242
|
+
text: '' + m,
|
|
243
|
+
click: () => { this.m_date.setFullYear(m); this._update(); }
|
|
244
|
+
});
|
|
245
|
+
}
|
|
246
|
+
}
|
|
247
|
+
|
|
248
|
+
let menu = new Menu({
|
|
249
|
+
items
|
|
250
|
+
});
|
|
251
|
+
|
|
252
|
+
let rc = this.getBoundingRect();
|
|
253
|
+
menu.displayAt(rc.left, rc.top);
|
|
254
|
+
}
|
|
255
|
+
|
|
256
|
+
getDate() {
|
|
257
|
+
return this.m_date;
|
|
258
|
+
}
|
|
259
|
+
|
|
260
|
+
setDate(date: Date) {
|
|
261
|
+
this.m_date = date;
|
|
262
|
+
this._update();
|
|
263
|
+
}
|
|
264
|
+
}
|
|
265
|
+
|
|
266
|
+
|
|
267
|
+
|
|
268
|
+
|
|
269
|
+
/**
|
|
270
|
+
* default popup calendar
|
|
271
|
+
* /
|
|
272
|
+
|
|
273
|
+
export class PopupCalendar extends Popup {
|
|
274
|
+
|
|
275
|
+
m_cal: Calendar;
|
|
276
|
+
|
|
277
|
+
constructor(props: CalendarProps) {
|
|
278
|
+
super({ tabIndex: 1 });
|
|
279
|
+
|
|
280
|
+
this.enableMask(false);
|
|
281
|
+
|
|
282
|
+
this.m_cal = new Calendar(props);
|
|
283
|
+
this.m_cal.addClass('@fit');
|
|
284
|
+
|
|
285
|
+
this.setContent(this.m_cal);
|
|
286
|
+
}
|
|
287
|
+
|
|
288
|
+
// binded
|
|
289
|
+
private _handleClick = (e: MouseEvent) => {
|
|
290
|
+
if (!this.dom) {
|
|
291
|
+
return;
|
|
292
|
+
}
|
|
293
|
+
|
|
294
|
+
let newfocus = <HTMLElement>e.target;
|
|
295
|
+
|
|
296
|
+
// child of this: ok
|
|
297
|
+
if (this.dom.contains(newfocus)) {
|
|
298
|
+
return;
|
|
299
|
+
}
|
|
300
|
+
|
|
301
|
+
// menu: ok
|
|
302
|
+
let dest = Component.getElement(newfocus, MenuItem);
|
|
303
|
+
if (dest) {
|
|
304
|
+
return;
|
|
305
|
+
}
|
|
306
|
+
|
|
307
|
+
this.close();
|
|
308
|
+
}
|
|
309
|
+
|
|
310
|
+
/ ** @ignore * /
|
|
311
|
+
show(modal?: boolean, at?: Point ) {
|
|
312
|
+
x4document.addEventListener('mousedown', this._handleClick);
|
|
313
|
+
if( at ) {
|
|
314
|
+
super.displayAt( at.x, at.y, 'top left', undefined, modal );
|
|
315
|
+
}
|
|
316
|
+
else {
|
|
317
|
+
super.show(modal);
|
|
318
|
+
}
|
|
319
|
+
}
|
|
320
|
+
|
|
321
|
+
/ ** @ignore * /
|
|
322
|
+
close() {
|
|
323
|
+
x4document.removeEventListener('mousedown', this._handleClick);
|
|
324
|
+
super.close();
|
|
325
|
+
}
|
|
326
|
+
}
|
|
327
327
|
*/
|