nmce-func 1.4.0 → 1.5.0

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.
@@ -1,308 +0,0 @@
1
- import { DateTime, Duration, Settings } from 'luxon';
2
- export class DateFunc {
3
- /**
4
- * At runtime, there's no simple guarantee that the input is Date. Sometimes you codes expect date, but the response from the Web service may give you string or number.
5
- * This function give you safe parse of date data from I/O, not of your control.
6
- * If the data is invalid, throws RangeError or TypeError.
7
- * @param dt
8
- * @returns
9
- */
10
- static dateDataToDate(dt) {
11
- if (dt instanceof Date) {
12
- return dt;
13
- }
14
- if (typeof dt === 'string') {
15
- const r = Date.parse(dt);
16
- if (isNaN(r)) {
17
- throw new RangeError('Invalid string for Date');
18
- }
19
- return new Date(r);
20
- }
21
- if (typeof dt === 'number') {
22
- const rd = new Date(dt);
23
- if (Number.isNaN(rd.valueOf())) {
24
- throw new RangeError('Invalid number for Date');
25
- }
26
- return rd;
27
- }
28
- throw new TypeError('Expect Date, string or number');
29
- }
30
- /**
31
- * Similar to dateDataToDate, but allow null and defined semantically.
32
- * @param dt
33
- * @returns
34
- */
35
- static dateDataToDateOrNull(dt) {
36
- if (dt instanceof Date) {
37
- return dt;
38
- }
39
- if (typeof dt === 'string') {
40
- const r = Date.parse(dt);
41
- if (isNaN(r)) {
42
- throw new RangeError('Invalid string for Date');
43
- }
44
- return new Date(r);
45
- }
46
- if (typeof dt === 'number') {
47
- const rd = new Date(dt);
48
- if (Number.isNaN(rd.valueOf())) {
49
- throw new RangeError('Invalid number for Date');
50
- }
51
- return rd;
52
- }
53
- if (dt == null) {
54
- return dt;
55
- }
56
- throw new TypeError('Expect Date, string or number');
57
- }
58
- /**
59
- * Transform UTC DateTime to local date without H, M and S. For example, the month day of 2018-01-23T22:00:00Z is 24 in Australia.
60
- * @param dtUtc
61
- */
62
- static dateTimeUtcToLocalDateNumber(dtUtc) {
63
- if (dtUtc == null) {
64
- return dtUtc;
65
- }
66
- const localDt = this.dateDataToDate(dtUtc);
67
- const localDNum = localDt.setHours(0, 0, 0, 0);
68
- return localDNum;
69
- }
70
- /**
71
- * Date only. However, the date may still be in UTC.
72
- * @param dtUtc
73
- * @returns new Date object.
74
- */
75
- static dateTimeUtcToLocalDate(dtUtc) {
76
- if (dtUtc == null) {
77
- return dtUtc;
78
- }
79
- const localDt = this.dateDataToDate(dtUtc);
80
- const localNum = localDt.setHours(0, 0, 0, 0);
81
- return new Date(localNum);
82
- }
83
- /**
84
- * '2018-01-23T22:00:00Z' will become '2018-01-24' in Australia.
85
- * @param dtUtc
86
- * @returns new Date object.
87
- */
88
- static localISODateString(dtUtc) {
89
- const dt = this.dateTimeUtcToLocalDate(dtUtc);
90
- if (dt == null) {
91
- return dt;
92
- }
93
- const d = DateTime.fromJSDate(dt);
94
- return d.toISODate();
95
- }
96
- /**
97
- * local date ONLY (no time) to UTC date.
98
- * The input could be a string of yyyy-MM-dd, or a Date Object.
99
- * https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Date
100
- * While the time value at the heart of a Date object is UTC, the basic methods to fetch the date and time
101
- * or its components all work in the local (i.e. host system) time zone and offset.
102
- * @param dt if dt contain time info, it will become dt.setHours(0, 0, 0, 0)
103
- */
104
- static localDateToUtc(d) {
105
- if (d == null) {
106
- return d;
107
- }
108
- const dt = this.dateDataToDate(d);
109
- const n = dt.setHours(0, 0, 0, 0);
110
- const offset = dt.getTimezoneOffset() * 60000;
111
- return new Date(n + offset);
112
- }
113
- static getEndOfWeek(dt) {
114
- const dateTime = DateTime.fromJSDate(this.dateDataToDate(dt));
115
- return dateTime.endOf('week').toJSDate();
116
- }
117
- static getStartOfWeek(dt) {
118
- const dateTime = DateTime.fromJSDate(this.dateDataToDate(dt));
119
- return dateTime.startOf('week').toJSDate();
120
- }
121
- static getEndOfMonth(dt) {
122
- const dateTime = DateTime.fromJSDate(this.dateDataToDate(dt));
123
- return dateTime.endOf('month').toJSDate();
124
- }
125
- static getStartOfMonth(dt) {
126
- const dateTime = DateTime.fromJSDate(this.dateDataToDate(dt));
127
- return dateTime.startOf('month').toJSDate();
128
- }
129
- static getEndOfDate(dt) {
130
- const dateTime = DateTime.fromJSDate(this.dateDataToDate(dt));
131
- return dateTime.endOf('day').toJSDate();
132
- }
133
- static getStartOfDate(dt) {
134
- const dateTime = DateTime.fromJSDate(this.dateDataToDate(dt));
135
- return dateTime.startOf('day').toJSDate();
136
- }
137
- static getEndOfToday() {
138
- return this.getEndOfDate(Date.now());
139
- }
140
- static getStartOfToday() {
141
- return this.getStartOfDate(Date.now());
142
- }
143
- static getDaysBetween(d1, d2) {
144
- const dm1 = DateTime.fromJSDate(this.dateDataToDate(d1));
145
- const dm2 = DateTime.fromJSDate(this.dateDataToDate(d1));
146
- return dm2.diff(dm1, 'days').days;
147
- }
148
- //inspired https://stackoverflow.com/questions/563406/add-days-to-javascript-date
149
- static addDays(dt, days) {
150
- const dat = DateTime.fromJSDate(this.dateDataToDate(dt));
151
- const r = dat.plus({ days: days });
152
- return r.toJSDate();
153
- }
154
- /**
155
- * Start of today
156
- */
157
- static get today() {
158
- return this.getStartOfToday();
159
- }
160
- static get now() {
161
- return new Date(Date.now());
162
- }
163
- /**
164
- * From now, next 5 minute mark. For example 2:23:44 will be 2:25:00;
165
- * @returns
166
- */
167
- static getNext5MinuteMark() {
168
- const m = DateTime.now().set({ second: 0, millisecond: 0 });
169
- const minute = m.minute;
170
- const mod = minute % 5;
171
- if (mod) {
172
- const delta = 5 - mod;
173
- return m.plus({ minutes: delta }).toJSDate();
174
- }
175
- return m.toJSDate();
176
- }
177
- static getYMD(d) {
178
- const dt = DateTime.fromJSDate(this.dateDataToDate(d));
179
- return dt.toFormat('yyyyMMdd');
180
- }
181
- static getDMYWithSlash(d) {
182
- const dt = DateTime.fromJSDate(this.dateDataToDate(d));
183
- return dt.toFormat('dd/MM/yyyy');
184
- }
185
- static getDMYHmWithSlash(d) {
186
- const dt = DateTime.fromJSDate(this.dateDataToDate(d));
187
- return dt.toFormat('dd/MM/yyyy HH:mm');
188
- }
189
- /**
190
- *
191
- * @param dtUtc In 24 hour format, and the date separator depending on the system or Luxon default locale
192
- * @returns
193
- */
194
- static getDateTime24Simple(dtUtc) {
195
- if (dtUtc == null) {
196
- return dtUtc;
197
- }
198
- const d = this.dateDataToDate(dtUtc);
199
- const dt = DateTime.fromJSDate(d);
200
- return dt.toLocaleString(DateTime.DATE_SHORT) + ' ' + dt.toLocaleString(DateTime.TIME_24_SIMPLE);
201
- }
202
- static setDefaultLocale(locale) {
203
- Settings.defaultLocale = locale;
204
- }
205
- static getDefaultLocale() {
206
- return Settings.defaultLocale;
207
- }
208
- static setDefaultZone(zone) {
209
- Settings.defaultZone = zone;
210
- }
211
- static getDefaultZone() {
212
- return Settings.defaultZone.name;
213
- }
214
- static isZoneValid() {
215
- return Settings.defaultZone.isValid;
216
- }
217
- /**
218
- * For example, in AEST, it is -600.
219
- * @returns
220
- */
221
- static getLocalTimezoneOffset() {
222
- const dt = new Date(Date.now());
223
- return dt.getTimezoneOffset();
224
- }
225
- /**
226
- * Get hour of the date. If Date is not defined, the hour will be current hour.
227
- * @param dtUtc
228
- */
229
- static getHour(dtUtc) {
230
- const dt = DateTime.fromJSDate(this.dateDataToDate(dtUtc));
231
- return dt.hour;
232
- }
233
- static getMinute(dtUtc) {
234
- const dt = DateTime.fromJSDate(this.dateDataToDate(dtUtc));
235
- return dt.minute;
236
- }
237
- static GetHM(dtUtc) {
238
- const dt = DateTime.fromJSDate(this.dateDataToDate(dtUtc));
239
- return { h: dt.hour, m: dt.minute };
240
- }
241
- static composeDateTime(dt, h = 0, minute = 0, second = 0) {
242
- return new Date(this.dateDataToDate(dt).setHours(h, minute, second, 0));
243
- }
244
- static olderThan24Hours(d) {
245
- const m = DateTime.fromJSDate(this.dateDataToDate(d));
246
- return DateTime.now().diff(m, 'hours') >= Duration.fromISO('PT24H');
247
- }
248
- static olderThanHours(d, hours) {
249
- const m = DateTime.fromJSDate(this.dateDataToDate(d));
250
- const diff = DateTime.now().diff(m, 'hours');
251
- const duration = Duration.fromISO(`PT${hours}H`);
252
- console.debug('diff: ' + diff.hours + ' duration: ' + duration.hours);
253
- return diff.hours >= duration.hours;
254
- }
255
- static olderThanMinutes(d, minutes) {
256
- const m = DateTime.fromJSDate(this.dateDataToDate(d));
257
- return DateTime.now().diff(m, 'minutes') >= Duration.fromMillis(minutes * 60 * 1000);
258
- }
259
- /**
260
- * It could be 11PM yesterday, and 1 AM today. Actually based on local today.
261
- */
262
- static olderThan1Day(dtUtc) {
263
- return DateFunc.getDayAge(dtUtc) > 0;
264
- }
265
- static getHourAge(d) {
266
- const m = DateTime.fromJSDate(this.dateDataToDate(d));
267
- return DateTime.now().diff(m, 'hours').hours;
268
- }
269
- /**
270
- * Compare date with now.
271
- * @param dtUtc
272
- */
273
- static getDayAge(d) {
274
- const m = DateTime.fromJSDate(this.dateDataToDate(d));
275
- return DateTime.now().diff(m, 'days').days;
276
- }
277
- /**
278
- * How many years from now.
279
- * @param d
280
- * @returns
281
- */
282
- static getAge(d) {
283
- const m = DateTime.fromJSDate(this.dateDataToDate(d));
284
- return DateTime.now().diff(m, 'years').years;
285
- }
286
- /**
287
- * Convert minutes from midnight to HH:mm text
288
- * @param minutes
289
- */
290
- static getHMFromMins(minutes) {
291
- if (minutes >= 24 * 60 || minutes < 0) {
292
- throw new RangeError('Valid input should be greater than or equal to 0 and less than 1440.');
293
- }
294
- const h = minutes / 60 | 0, m = minutes % 60 | 0;
295
- return DateTime.utc().set({ hour: h, minute: m }).toFormat('HH:mm');
296
- }
297
- static getMinutesSinceMidnight(d) {
298
- const dt = DateTime.fromJSDate(this.dateDataToDate(d));
299
- const midnight = dt.startOf('day');
300
- return dt.diff(midnight, 'minutes').minutes;
301
- }
302
- static getMinutesBetween(start, end) {
303
- const m = DateTime.fromJSDate(this.dateDataToDate(start));
304
- const m2 = DateTime.fromJSDate(this.dateDataToDate(end));
305
- return m2.diff(m, 'minutes').minutes;
306
- }
307
- }
308
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"dateFunc.js","sourceRoot":"","sources":["../../../../projects/nmce-func/src/_func/dateFunc.ts"],"names":[],"mappings":"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","sourcesContent":["import { DateTime, Duration, Settings } from 'luxon';\r\n\r\nexport class DateFunc {\r\n\t/**\r\n\t * At runtime, there's no simple guarantee that the input is Date. Sometimes you codes expect date, but the response from the Web service may give you string or number.\r\n\t * This function give you safe parse of date data from I/O, not of your control.\r\n\t * If the data is invalid, throws RangeError or TypeError.\r\n\t * @param dt \r\n\t * @returns \r\n\t */\r\n\tstatic dateDataToDate(dt: Date | string | number) : Date {\r\n\t\tif (dt instanceof Date){\r\n\t\t\treturn dt;\r\n\t\t}\r\n\r\n\t\tif (typeof dt === 'string'){\r\n\t\t\tconst r = Date.parse(dt);\r\n\t\t\tif (isNaN(r)){\r\n\t\t\t\tthrow new RangeError('Invalid string for Date');\r\n\t\t\t}\r\n\r\n\t\t\treturn new Date(r);\r\n\t\t}\r\n\r\n\t\tif (typeof dt === 'number'){\r\n\t\t\tconst rd = new Date(dt)\r\n\t\t\tif (Number.isNaN(rd.valueOf())){\r\n\t\t\t\tthrow new RangeError('Invalid number for Date');\r\n\t\t\t}\r\n\r\n\t\t\treturn rd;\r\n\t\t}\r\n\r\n\t\tthrow new TypeError('Expect Date, string or number');\r\n\t}\r\n\r\n\t/**\r\n\t * Similar to dateDataToDate, but allow null and defined semantically.\r\n\t * @param dt \r\n\t * @returns \r\n\t */\r\n\tstatic dateDataToDateOrNull(dt: Date | string | number | null | undefined) : Date | null | undefined {\r\n\t\tif (dt instanceof Date){\r\n\t\t\treturn dt;\r\n\t\t}\r\n\r\n\t\tif (typeof dt === 'string'){\r\n\t\t\tconst r = Date.parse(dt);\r\n\t\t\tif (isNaN(r)){\r\n\t\t\t\tthrow new RangeError('Invalid string for Date');\r\n\t\t\t}\r\n\r\n\t\t\treturn new Date(r);\r\n\t\t}\r\n\r\n\t\tif (typeof dt === 'number'){\r\n\t\t\tconst rd = new Date(dt)\r\n\t\t\tif (Number.isNaN(rd.valueOf())){\r\n\t\t\t\tthrow new RangeError('Invalid number for Date');\r\n\t\t\t}\r\n\r\n\t\t\treturn rd;\r\n\t\t}\r\n\r\n\t\tif (dt == null){ \r\n\t\t\treturn dt;\r\n\t\t}\r\n\r\n\t\tthrow new TypeError('Expect Date, string or number');\r\n\t}\r\n\r\n\t/**\r\n\t * Transform UTC DateTime to local date without H, M and S. For example, the month day of 2018-01-23T22:00:00Z is 24 in Australia.\r\n\t * @param dtUtc\r\n\t */\r\n\tstatic dateTimeUtcToLocalDateNumber(dtUtc: Date | string | number | null | undefined): number | null | undefined {\r\n\t\tif (dtUtc == null) {\r\n\t\t\treturn dtUtc;\r\n\t\t}\r\n\r\n\t\tconst localDt = this.dateDataToDate(dtUtc);\r\n\t\tconst localDNum = localDt.setHours(0, 0, 0, 0);\r\n\r\n\t\treturn localDNum;\r\n\t}\r\n\r\n\t/**\r\n\t * Date only. However, the date may still be in UTC.\r\n\t * @param dtUtc\r\n\t * @returns new Date object.\r\n\t */\r\n\tstatic dateTimeUtcToLocalDate(dtUtc: Date | string | number | null | undefined): Date | null | undefined {\r\n\t\tif (dtUtc == null) {\r\n\t\t\treturn dtUtc;\r\n\t\t}\r\n\r\n\t\tconst localDt = this.dateDataToDate(dtUtc);\r\n\t\tconst localNum = localDt.setHours(0, 0, 0, 0);\r\n\t\treturn new Date(localNum);\r\n\t}\r\n\r\n\t/**\r\n\t * '2018-01-23T22:00:00Z' will become '2018-01-24' in Australia.\r\n\t * @param dtUtc \r\n\t * @returns  new Date object.\r\n\t */\r\n\tstatic localISODateString(dtUtc: Date | string | number | null | undefined): string | null | undefined {\t\t\r\n\t    const dt = this.dateTimeUtcToLocalDate(dtUtc);\r\n\t\tif (dt == null) {\r\n\t\t\treturn dt;\r\n\t\t}\r\n\r\n\t\tconst d= DateTime.fromJSDate(dt);\r\n\t\treturn d.toISODate();\r\n\t}\r\n\r\n\t/**\r\n\t * local date ONLY (no time) to UTC date.\r\n\t * The input could be a string of yyyy-MM-dd, or a Date Object.\r\n\t * https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Date\r\n\t * While the time value at the heart of a Date object is UTC, the basic methods to fetch the date and time \r\n\t * or its components all work in the local (i.e. host system) time zone and offset.\r\n\t * @param dt if dt contain time info, it will become dt.setHours(0, 0, 0, 0)\r\n\t */\r\n\tstatic localDateToUtc(d: Date | string | number | null | undefined | string): Date | null | undefined {\r\n\t\tif (d == null) {\r\n\t\t\treturn d;\r\n\t\t}\r\n\r\n\t\tconst dt = this.dateDataToDate(d);\r\n\t\tconst n = dt.setHours(0, 0, 0, 0);\r\n\t\tconst offset = dt.getTimezoneOffset() * 60000;\r\n\t\treturn new Date(n + offset);\r\n\t}\r\n\r\n\tstatic getEndOfWeek(dt: Date | string | number) : Date {\r\n\t\tconst dateTime = DateTime.fromJSDate(this.dateDataToDate(dt));\r\n\t\treturn dateTime.endOf('week').toJSDate();\r\n\t}\r\n\r\n\tstatic getStartOfWeek(dt: Date | string | number) : Date {\r\n\t\tconst dateTime = DateTime.fromJSDate(this.dateDataToDate(dt));\r\n\t\treturn dateTime.startOf('week').toJSDate();\r\n\t}\r\n\r\n\tstatic getEndOfMonth(dt: Date | string | number) : Date {\r\n\t\tconst dateTime = DateTime.fromJSDate(this.dateDataToDate(dt));\r\n\t\treturn dateTime.endOf('month').toJSDate();\r\n\t}\r\n\r\n\tstatic getStartOfMonth(dt: Date | string | number) : Date {\r\n\t\tconst dateTime = DateTime.fromJSDate(this.dateDataToDate(dt));\r\n\t\treturn dateTime.startOf('month').toJSDate();\r\n\t}\r\n\r\n\tstatic getEndOfDate(dt: Date | string | number): Date {\r\n\t\tconst dateTime = DateTime.fromJSDate(this.dateDataToDate(dt));\r\n\t\treturn dateTime.endOf('day').toJSDate();\r\n\t}\r\n\r\n\tstatic getStartOfDate(dt: Date | string | number): Date {\r\n\t\tconst dateTime = DateTime.fromJSDate(this.dateDataToDate(dt));\r\n\t\treturn dateTime.startOf('day').toJSDate();\r\n\t}\r\n\r\n\tstatic getEndOfToday(): Date | null | undefined {\r\n\t\treturn this.getEndOfDate(Date.now());\r\n\r\n\t}\r\n\r\n\tstatic getStartOfToday(): Date {\r\n\t\treturn this.getStartOfDate(Date.now());\r\n\t}\r\n\r\n\tstatic getDaysBetween(d1: Date | string | number, d2: Date | string | number): number {\r\n\t\tconst dm1 = DateTime.fromJSDate(this.dateDataToDate(d1));\r\n\t\tconst dm2 = DateTime.fromJSDate(this.dateDataToDate(d1));\r\n\t\treturn dm2.diff(dm1, 'days').days;\r\n\t}\r\n\r\n\t//inspired https://stackoverflow.com/questions/563406/add-days-to-javascript-date\r\n\tstatic addDays(dt: Date | string | number, days: number) : Date {\r\n\t\tconst dat = DateTime.fromJSDate(this.dateDataToDate(dt));\r\n\t\tconst r = dat.plus({days: days});\r\n\t\treturn r.toJSDate();\r\n\t}\r\n\r\n\t/**\r\n\t * Start of today\r\n\t */\r\n\tstatic get today(): Date {\r\n\t\treturn this.getStartOfToday();\r\n\t}\r\n\r\n\tstatic get now(): Date {\r\n\t\treturn new Date(Date.now());\r\n\t}\r\n\r\n\t/**\r\n\t * From now, next 5 minute mark. For example 2:23:44 will be 2:25:00;\r\n\t * @returns \r\n\t */\r\n\tstatic getNext5MinuteMark(): Date {\r\n\t\tconst m = DateTime.now().set({second: 0, millisecond:0});\r\n\t\tconst minute = m.minute;\r\n\t\tconst mod = minute % 5;\r\n\t\tif (mod) {\r\n\t\t\tconst delta = 5 - mod;\r\n\t\t\treturn m.plus({minutes: delta}).toJSDate();\r\n\t\t}\r\n\r\n\t\treturn m.toJSDate();\r\n\t}\r\n\r\n\tstatic getYMD(d: Date | string | number) {\r\n\t\tconst dt = DateTime.fromJSDate(this.dateDataToDate(d));\r\n\t\treturn dt.toFormat('yyyyMMdd');\r\n\t}\r\n\r\n\tstatic getDMYWithSlash(d: Date | string | number) {\r\n\t\tconst dt = DateTime.fromJSDate(this.dateDataToDate(d));\r\n\t\treturn dt.toFormat('dd/MM/yyyy');\r\n\t}\r\n\r\n\tstatic getDMYHmWithSlash(d: Date | string | number) {\r\n\t\tconst dt = DateTime.fromJSDate(this.dateDataToDate(d));\r\n\t\treturn dt.toFormat('dd/MM/yyyy HH:mm');\r\n\t}\r\n\r\n\t/**\r\n\t * \r\n\t * @param dtUtc In 24 hour format, and the date separator depending on the system or Luxon default locale\r\n\t * @returns \r\n\t */\r\n\tstatic getDateTime24Simple(dtUtc: Date | string | number | null | undefined) {\r\n\t\tif (dtUtc == null) {\r\n\t\t\treturn dtUtc;\r\n\t\t}\r\n\t\t\r\n\t\tconst d = this.dateDataToDate(dtUtc);\r\n\t\tconst dt = DateTime.fromJSDate(d);\r\n\t\treturn dt.toLocaleString(DateTime.DATE_SHORT) + ' ' + dt.toLocaleString(DateTime.TIME_24_SIMPLE);\r\n\t}\r\n\r\n\tstatic setDefaultLocale(locale: string){\r\n\t\tSettings.defaultLocale=locale;\r\n\t}\r\n\r\n\tstatic getDefaultLocale(){\r\n\t\treturn Settings.defaultLocale;\r\n\t}\r\n\r\n\tstatic setDefaultZone(zone: string){\r\n\t\tSettings.defaultZone=zone;\r\n\t}\r\n\r\n\tstatic getDefaultZone() : string {\r\n\t\treturn Settings.defaultZone.name;\r\n\t}\r\n\r\n\tstatic isZoneValid(): boolean{\r\n\t\treturn Settings.defaultZone.isValid;\r\n\t}\r\n\r\n\t/**\r\n\t * For example, in AEST, it is -600.\r\n\t * @returns \r\n\t */\r\n\tstatic getLocalTimezoneOffset(): number {\r\n\t\tconst dt = new Date(Date.now());\r\n\t\treturn dt.getTimezoneOffset();\r\n\t}\r\n\r\n\t/**\r\n\t * Get hour of the date. If Date is not defined, the hour will be current hour.\r\n\t * @param dtUtc\r\n\t */\r\n\tstatic getHour(dtUtc: Date | string | number): number {\r\n\t\tconst dt = DateTime.fromJSDate(this.dateDataToDate(dtUtc));\r\n\t\treturn dt.hour;\r\n\t}\r\n\r\n\tstatic getMinute(dtUtc: Date | string | number): number {\r\n\t\tconst dt = DateTime.fromJSDate(this.dateDataToDate(dtUtc));\r\n\t\treturn dt.minute;\r\n\t}\r\n\r\n\tstatic GetHM(dtUtc: Date | string | number): {h: number, m: number}{\r\n\t\tconst dt = DateTime.fromJSDate(this.dateDataToDate(dtUtc));\r\n\t\treturn {h: dt.hour, m: dt.minute};\r\n\t}\r\n\r\n\tstatic composeDateTime(dt: Date | string | number, h: number = 0, minute: number = 0, second: number = 0): Date {\r\n\t\treturn new Date( this.dateDataToDate(dt).setHours(h, minute, second, 0));\r\n\t}\r\n\r\n\tstatic olderThan24Hours(d: Date | string | number): boolean {\r\n\t\tconst m = DateTime.fromJSDate(this.dateDataToDate(d));\r\n\t\treturn DateTime.now().diff(m, 'hours') >= Duration.fromISO('PT24H');\r\n\t}\r\n\r\n\tstatic olderThanHours(d: Date | string | number, hours: number) : boolean {\r\n\t\tconst m = DateTime.fromJSDate(this.dateDataToDate(d));\r\n\t\tconst diff = DateTime.now().diff(m, 'hours'); \r\n\t\tconst duration = Duration.fromISO(`PT${hours}H`);\r\n\t\tconsole.debug('diff: ' + diff.hours + '   duration: ' + duration.hours);\r\n\t\treturn diff.hours >= duration.hours;\r\n\t}\r\n\r\n\tstatic olderThanMinutes(d: Date | string | number, minutes: number) {\r\n\t\tconst m = DateTime.fromJSDate(this.dateDataToDate(d));\r\n\t\treturn DateTime.now().diff(m, 'minutes') >= Duration.fromMillis(minutes*60*1000);\r\n\t}\r\n\r\n\t/**\r\n\t * It could be 11PM yesterday, and 1 AM today. Actually based on local today.\r\n\t */\r\n\tstatic olderThan1Day(dtUtc: Date | string | number): boolean {\r\n\t\treturn DateFunc.getDayAge(dtUtc) > 0;\r\n\t}\r\n\r\n\tstatic getHourAge(d: Date | string | number) {\r\n\t\tconst m = DateTime.fromJSDate(this.dateDataToDate(d));\r\n\t\treturn DateTime.now().diff(m, 'hours').hours;\r\n\t}\r\n\r\n\t/**\r\n\t * Compare date with now.\r\n\t * @param dtUtc\r\n\t */\r\n\tstatic getDayAge(d: Date | string | number) {\r\n\t\tconst m = DateTime.fromJSDate(this.dateDataToDate(d));\r\n\t\treturn DateTime.now().diff(m, 'days').days;\r\n\t}\r\n\r\n\t/**\r\n\t * How many years from now.\r\n\t * @param d\r\n\t * @returns\r\n\t */\r\n\tstatic getAge(d: Date | string | number) {\r\n\t\tconst m = DateTime.fromJSDate(this.dateDataToDate(d));\r\n\t\treturn DateTime.now().diff(m, 'years').years;\r\n\t}\r\n\r\n\t/**\r\n\t * Convert minutes from midnight to HH:mm text\r\n\t * @param minutes\r\n\t */\r\n\tstatic getHMFromMins(minutes: number): string {\r\n\t\tif (minutes >= 24 * 60 || minutes < 0) {\r\n\t\t\tthrow new RangeError('Valid input should be greater than or equal to 0 and less than 1440.');\r\n\t\t}\r\n\t\tconst h = minutes / 60 | 0,\r\n\t\t\tm = minutes % 60 | 0;\r\n\t\treturn DateTime.utc().set({hour: h, minute: m}).toFormat('HH:mm');\r\n\t}\r\n\r\n\tstatic getMinutesSinceMidnight(d: Date | string | number) {\r\n\t\tconst dt = DateTime.fromJSDate(this.dateDataToDate(d));\r\n\t\tconst midnight = dt.startOf('day');\r\n\t\treturn dt.diff(midnight, 'minutes').minutes;\r\n\t}\r\n\r\n\tstatic getMinutesBetween(start: Date | string | number, end: Date | string | number) {\r\n\t\tconst m = DateTime.fromJSDate(this.dateDataToDate(start));\r\n\t\tconst m2 = DateTime.fromJSDate(this.dateDataToDate(end));\r\n\t\treturn m2.diff(m, 'minutes').minutes;\r\n\t}\r\n}\r\n\r\n"]}
@@ -1,45 +0,0 @@
1
- export class HtmlPrintFunc {
2
- /**
3
- * Print with CSS for internal reports
4
- * @param htmlTags
5
- * @param cssUrl
6
- */
7
- static printWithCSS(htmlTags, cssUrl) {
8
- if (window) {
9
- const htmlToPrint = `<html><head>
10
- <meta name="viewport" content="width=device-width, initial-scale=1, shrink-to-fit=no"/>
11
- <link rel="stylesheet" type="text/css" href="${cssUrl}" media="screen,print"/>
12
- </head><body onload="window.print()">${htmlTags}</body></html>`;
13
- const popup = window.open('', '_blank', 'width=1024,height=768');
14
- popup?.document.open();
15
- popup?.document.write(htmlToPrint);
16
- popup?.document.close();
17
- }
18
- return true;
19
- }
20
- /**
21
- * Print for external documents.
22
- * @param htmlTags
23
- */
24
- static print(htmlTags) {
25
- if (window) {
26
- const htmlToPrint = `<html><head>
27
- <meta name="viewport" content="width=device-width, initial-scale=1, shrink-to-fit=no"/>
28
- </head><body onload="window.print()">${htmlTags}</body></html>`;
29
- const popup = window.open('', '_blank', 'width=1024,height=768');
30
- popup?.document.open();
31
- popup?.document.write(htmlToPrint);
32
- popup?.document.close();
33
- }
34
- return true;
35
- }
36
- /**
37
- * Print image url through html img.
38
- * @param url
39
- */
40
- static printImage(url) {
41
- const imageTags = `<img src="${url}" alt="Image from URL"/>`;
42
- HtmlPrintFunc.print(imageTags);
43
- }
44
- }
45
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaHRtbFByaW50RnVuYy5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uL3Byb2plY3RzL25tY2UtZnVuYy9zcmMvX2Z1bmMvaHRtbFByaW50RnVuYy50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxNQUFNLE9BQU8sYUFBYTtJQUN6Qjs7OztPQUlHO0lBQ0gsTUFBTSxDQUFDLFlBQVksQ0FBQyxRQUFnQixFQUFFLE1BQWM7UUFDbkQsSUFBSSxNQUFNLEVBQUUsQ0FBQztZQUNaLE1BQU0sV0FBVyxHQUFHOzsrQ0FFd0IsTUFBTTt1Q0FDZCxRQUFRLGdCQUFnQixDQUFDO1lBRTdELE1BQU0sS0FBSyxHQUFHLE1BQU0sQ0FBQyxJQUFJLENBQUMsRUFBRSxFQUFFLFFBQVEsRUFBRSx1QkFBdUIsQ0FBQyxDQUFDO1lBQ2pFLEtBQUssRUFBRSxRQUFRLENBQUMsSUFBSSxFQUFFLENBQUM7WUFDdkIsS0FBSyxFQUFFLFFBQVEsQ0FBQyxLQUFLLENBQUMsV0FBVyxDQUFDLENBQUM7WUFFbkMsS0FBSyxFQUFFLFFBQVEsQ0FBQyxLQUFLLEVBQUUsQ0FBQztRQUN6QixDQUFDO1FBQ0QsT0FBTyxJQUFJLENBQUM7SUFDYixDQUFDO0lBRUQ7OztPQUdHO0lBQ0gsTUFBTSxDQUFDLEtBQUssQ0FBQyxRQUFnQjtRQUM1QixJQUFJLE1BQU0sRUFBRSxDQUFDO1lBQ1osTUFBTSxXQUFXLEdBQUc7O3VDQUVnQixRQUFRLGdCQUFnQixDQUFDO1lBRTdELE1BQU0sS0FBSyxHQUFHLE1BQU0sQ0FBQyxJQUFJLENBQUMsRUFBRSxFQUFFLFFBQVEsRUFBRSx1QkFBdUIsQ0FBQyxDQUFDO1lBQ2pFLEtBQUssRUFBRSxRQUFRLENBQUMsSUFBSSxFQUFFLENBQUM7WUFDdkIsS0FBSyxFQUFFLFFBQVEsQ0FBQyxLQUFLLENBQUMsV0FBVyxDQUFDLENBQUM7WUFFbkMsS0FBSyxFQUFFLFFBQVEsQ0FBQyxLQUFLLEVBQUUsQ0FBQztRQUN6QixDQUFDO1FBQ0QsT0FBTyxJQUFJLENBQUM7SUFDYixDQUFDO0lBRUQ7OztPQUdHO0lBQ0gsTUFBTSxDQUFDLFVBQVUsQ0FBQyxHQUFXO1FBQzVCLE1BQU0sU0FBUyxHQUFHLGFBQWEsR0FBRywwQkFBMEIsQ0FBQztRQUM3RCxhQUFhLENBQUMsS0FBSyxDQUFDLFNBQVMsQ0FBQyxDQUFDO0lBQ2hDLENBQUM7Q0FDRCIsInNvdXJjZXNDb250ZW50IjpbImV4cG9ydCBjbGFzcyBIdG1sUHJpbnRGdW5jIHtcclxuXHQvKipcclxuXHQgKiBQcmludCB3aXRoIENTUyBmb3IgaW50ZXJuYWwgcmVwb3J0c1xyXG5cdCAqIEBwYXJhbSBodG1sVGFnc1xyXG5cdCAqIEBwYXJhbSBjc3NVcmxcclxuXHQgKi9cclxuXHRzdGF0aWMgcHJpbnRXaXRoQ1NTKGh0bWxUYWdzOiBzdHJpbmcsIGNzc1VybDogc3RyaW5nKSB7XHJcblx0XHRpZiAod2luZG93KSB7XHJcblx0XHRcdGNvbnN0IGh0bWxUb1ByaW50ID0gYDxodG1sPjxoZWFkPlxyXG48bWV0YSBuYW1lPVwidmlld3BvcnRcIiBjb250ZW50PVwid2lkdGg9ZGV2aWNlLXdpZHRoLCBpbml0aWFsLXNjYWxlPTEsIHNocmluay10by1maXQ9bm9cIi8+XHJcbjxsaW5rIHJlbD1cInN0eWxlc2hlZXRcIiB0eXBlPVwidGV4dC9jc3NcIiBocmVmPVwiJHtjc3NVcmx9XCIgbWVkaWE9XCJzY3JlZW4scHJpbnRcIi8+XHJcbjwvaGVhZD48Ym9keSBvbmxvYWQ9XCJ3aW5kb3cucHJpbnQoKVwiPiR7aHRtbFRhZ3N9PC9ib2R5PjwvaHRtbD5gO1xyXG5cclxuXHRcdFx0Y29uc3QgcG9wdXAgPSB3aW5kb3cub3BlbignJywgJ19ibGFuaycsICd3aWR0aD0xMDI0LGhlaWdodD03NjgnKTtcclxuXHRcdFx0cG9wdXA/LmRvY3VtZW50Lm9wZW4oKTtcclxuXHRcdFx0cG9wdXA/LmRvY3VtZW50LndyaXRlKGh0bWxUb1ByaW50KTtcclxuXHJcblx0XHRcdHBvcHVwPy5kb2N1bWVudC5jbG9zZSgpO1xyXG5cdFx0fVxyXG5cdFx0cmV0dXJuIHRydWU7XHJcblx0fVxyXG5cclxuXHQvKipcclxuXHQgKiBQcmludCBmb3IgZXh0ZXJuYWwgZG9jdW1lbnRzLlxyXG5cdCAqIEBwYXJhbSBodG1sVGFnc1xyXG5cdCAqL1xyXG5cdHN0YXRpYyBwcmludChodG1sVGFnczogc3RyaW5nKSB7XHJcblx0XHRpZiAod2luZG93KSB7XHJcblx0XHRcdGNvbnN0IGh0bWxUb1ByaW50ID0gYDxodG1sPjxoZWFkPlxyXG48bWV0YSBuYW1lPVwidmlld3BvcnRcIiBjb250ZW50PVwid2lkdGg9ZGV2aWNlLXdpZHRoLCBpbml0aWFsLXNjYWxlPTEsIHNocmluay10by1maXQ9bm9cIi8+XHJcbjwvaGVhZD48Ym9keSBvbmxvYWQ9XCJ3aW5kb3cucHJpbnQoKVwiPiR7aHRtbFRhZ3N9PC9ib2R5PjwvaHRtbD5gO1xyXG5cclxuXHRcdFx0Y29uc3QgcG9wdXAgPSB3aW5kb3cub3BlbignJywgJ19ibGFuaycsICd3aWR0aD0xMDI0LGhlaWdodD03NjgnKTtcclxuXHRcdFx0cG9wdXA/LmRvY3VtZW50Lm9wZW4oKTtcclxuXHRcdFx0cG9wdXA/LmRvY3VtZW50LndyaXRlKGh0bWxUb1ByaW50KTtcclxuXHJcblx0XHRcdHBvcHVwPy5kb2N1bWVudC5jbG9zZSgpO1xyXG5cdFx0fVxyXG5cdFx0cmV0dXJuIHRydWU7XHJcblx0fVxyXG5cclxuXHQvKipcclxuXHQgKiBQcmludCBpbWFnZSB1cmwgdGhyb3VnaCBodG1sIGltZy5cclxuXHQgKiBAcGFyYW0gdXJsIFxyXG5cdCAqL1xyXG5cdHN0YXRpYyBwcmludEltYWdlKHVybDogc3RyaW5nKSB7XHJcblx0XHRjb25zdCBpbWFnZVRhZ3MgPSBgPGltZyBzcmM9XCIke3VybH1cIiBhbHQ9XCJJbWFnZSBmcm9tIFVSTFwiLz5gO1xyXG5cdFx0SHRtbFByaW50RnVuYy5wcmludChpbWFnZVRhZ3MpO1xyXG5cdH1cclxufVxyXG4iXX0=
@@ -1,10 +0,0 @@
1
- export * from './addressFunc';
2
- export * from './currencyFunc';
3
- export * from './dateFunc';
4
- export * from './htmlPrintFunc';
5
- export * from './javascriptFunc';
6
- export * from './jsonFunc';
7
- export * from './stringAusFunc';
8
- export * from './stringFunc';
9
- export * from './uuidFunc';
10
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi9wcm9qZWN0cy9ubWNlLWZ1bmMvc3JjL19mdW5jL2luZGV4LnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLGNBQWMsZUFBZSxDQUFDO0FBQzlCLGNBQWMsZ0JBQWdCLENBQUM7QUFDL0IsY0FBYyxZQUFZLENBQUM7QUFDM0IsY0FBYyxpQkFBaUIsQ0FBQztBQUNoQyxjQUFjLGtCQUFrQixDQUFDO0FBQ2pDLGNBQWMsWUFBWSxDQUFDO0FBQzNCLGNBQWMsaUJBQWlCLENBQUM7QUFDaEMsY0FBYyxjQUFjLENBQUM7QUFDN0IsY0FBYyxZQUFZLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyJleHBvcnQgKiBmcm9tICcuL2FkZHJlc3NGdW5jJztcclxuZXhwb3J0ICogZnJvbSAnLi9jdXJyZW5jeUZ1bmMnO1xyXG5leHBvcnQgKiBmcm9tICcuL2RhdGVGdW5jJztcclxuZXhwb3J0ICogZnJvbSAnLi9odG1sUHJpbnRGdW5jJztcclxuZXhwb3J0ICogZnJvbSAnLi9qYXZhc2NyaXB0RnVuYyc7XHJcbmV4cG9ydCAqIGZyb20gJy4vanNvbkZ1bmMnO1xyXG5leHBvcnQgKiBmcm9tICcuL3N0cmluZ0F1c0Z1bmMnO1xyXG5leHBvcnQgKiBmcm9tICcuL3N0cmluZ0Z1bmMnO1xyXG5leHBvcnQgKiBmcm9tICcuL3V1aWRGdW5jJztcclxuXHJcblxyXG4iXX0=
@@ -1,21 +0,0 @@
1
- export class JavaScriptFunc {
2
- /**
3
- * Some business functions depend on external JavaScript libraries. Lazy loading of respective business modules is good,
4
- * and this function supports lazy loading of JS libraries.
5
- * @param scriptUrl
6
- * @param type things like module.
7
- * @returns Promise for subsequent JS function calls.
8
- */
9
- static loadExternalScript(scriptUrl, type) {
10
- return new Promise((resolve, reject) => {
11
- const scriptElement = document.createElement('script');
12
- if (type) {
13
- scriptElement.type = type;
14
- }
15
- scriptElement.src = scriptUrl;
16
- scriptElement.onload = resolve;
17
- document.body.appendChild(scriptElement);
18
- });
19
- }
20
- }
21
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiamF2YXNjcmlwdEZ1bmMuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi9wcm9qZWN0cy9ubWNlLWZ1bmMvc3JjL19mdW5jL2phdmFzY3JpcHRGdW5jLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLE1BQU0sT0FBTyxjQUFjO0lBQzFCOzs7Ozs7T0FNRztJQUNILE1BQU0sQ0FBQyxrQkFBa0IsQ0FBQyxTQUFpQixFQUFFLElBQWE7UUFDekQsT0FBTyxJQUFJLE9BQU8sQ0FBQyxDQUFDLE9BQU8sRUFBRSxNQUFNLEVBQUUsRUFBRTtZQUN0QyxNQUFNLGFBQWEsR0FBRyxRQUFRLENBQUMsYUFBYSxDQUFDLFFBQVEsQ0FBQyxDQUFDO1lBQ3ZELElBQUksSUFBSSxFQUFDLENBQUM7Z0JBQ1QsYUFBYSxDQUFDLElBQUksR0FBQyxJQUFJLENBQUM7WUFDekIsQ0FBQztZQUVELGFBQWEsQ0FBQyxHQUFHLEdBQUcsU0FBUyxDQUFDO1lBQzlCLGFBQWEsQ0FBQyxNQUFNLEdBQUcsT0FBTyxDQUFDO1lBQy9CLFFBQVEsQ0FBQyxJQUFJLENBQUMsV0FBVyxDQUFDLGFBQWEsQ0FBQyxDQUFDO1FBQzFDLENBQUMsQ0FBQyxDQUFDO0lBQ0osQ0FBQztDQUNEIiwic291cmNlc0NvbnRlbnQiOlsiZXhwb3J0IGNsYXNzIEphdmFTY3JpcHRGdW5jIHtcclxuXHQvKipcclxuXHQgKiBTb21lIGJ1c2luZXNzIGZ1bmN0aW9ucyBkZXBlbmQgb24gZXh0ZXJuYWwgSmF2YVNjcmlwdCBsaWJyYXJpZXMuIExhenkgbG9hZGluZyBvZiByZXNwZWN0aXZlIGJ1c2luZXNzIG1vZHVsZXMgaXMgZ29vZCwgXHJcblx0ICogYW5kIHRoaXMgZnVuY3Rpb24gc3VwcG9ydHMgbGF6eSBsb2FkaW5nIG9mIEpTIGxpYnJhcmllcy5cclxuXHQgKiBAcGFyYW0gc2NyaXB0VXJsIFxyXG5cdCAqIEBwYXJhbSB0eXBlIHRoaW5ncyBsaWtlIG1vZHVsZS5cclxuXHQgKiBAcmV0dXJucyBQcm9taXNlIGZvciBzdWJzZXF1ZW50IEpTIGZ1bmN0aW9uIGNhbGxzLlxyXG5cdCAqL1xyXG5cdHN0YXRpYyBsb2FkRXh0ZXJuYWxTY3JpcHQoc2NyaXB0VXJsOiBzdHJpbmcsIHR5cGU/OiBzdHJpbmcpIHtcclxuXHRcdHJldHVybiBuZXcgUHJvbWlzZSgocmVzb2x2ZSwgcmVqZWN0KSA9PiB7XHJcblx0XHRcdGNvbnN0IHNjcmlwdEVsZW1lbnQgPSBkb2N1bWVudC5jcmVhdGVFbGVtZW50KCdzY3JpcHQnKTtcclxuXHRcdFx0aWYgKHR5cGUpe1xyXG5cdFx0XHRcdHNjcmlwdEVsZW1lbnQudHlwZT10eXBlO1xyXG5cdFx0XHR9XHJcblxyXG5cdFx0XHRzY3JpcHRFbGVtZW50LnNyYyA9IHNjcmlwdFVybDtcclxuXHRcdFx0c2NyaXB0RWxlbWVudC5vbmxvYWQgPSByZXNvbHZlO1xyXG5cdFx0XHRkb2N1bWVudC5ib2R5LmFwcGVuZENoaWxkKHNjcmlwdEVsZW1lbnQpO1xyXG5cdFx0fSk7XHJcblx0fVxyXG59XHJcbiJdfQ==
@@ -1,69 +0,0 @@
1
- import { DateFunc } from './dateFunc';
2
- /**
3
- * Basic JSON functions
4
- */
5
- export class JsonFunc {
6
- /**
7
- *
8
- * @param array Group by a property of array element.
9
- * @param propertyName
10
- * @returns
11
- */
12
- static groupBy(array, propertyName) {
13
- return array.reduce(function (acc, obj) {
14
- const key = obj[propertyName];
15
- if (!acc[key]) {
16
- acc[key] = [];
17
- }
18
- acc[key].push(obj);
19
- return acc;
20
- }, {});
21
- }
22
- /**
23
- * Group by a date property. The key is always of string type and representing milliseconds.
24
- * The client should convert the string to number.
25
- * Angular date pipe could actually consume such string without explicitly converting to number.
26
- * @param array
27
- * @param propertyName
28
- */
29
- static groupByDate(array, propertyName) {
30
- return array.reduce(function (acc, obj) {
31
- const key = DateFunc.dateTimeUtcToLocalDateNumber(obj[propertyName]);
32
- if (key != null) {
33
- if (!acc[key]) {
34
- acc[key] = [];
35
- }
36
- acc[key].push(obj);
37
- }
38
- return acc;
39
- }, {});
40
- }
41
- /**
42
- * Remove null or empty fields including those in nested objects.
43
- * This is useful for reducing payload of AJAX serialization.
44
- * @param obj
45
- */
46
- static removeNullOrEmptyFields(obj) {
47
- for (const f in obj) {
48
- let p = obj[f];
49
- if (p === null || p === '') {
50
- delete obj[f];
51
- }
52
- else if (typeof p === 'object' && p !== null) {
53
- this.removeNullOrEmptyFields(p);
54
- }
55
- }
56
- }
57
- /**
58
- *
59
- * @param obj Remove null fields of object at only the 1st level.
60
- */
61
- static removeNullFields(obj) {
62
- for (const f in obj) {
63
- if (obj[f] === null) {
64
- delete obj[f];
65
- }
66
- }
67
- }
68
- }
69
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoianNvbkZ1bmMuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi9wcm9qZWN0cy9ubWNlLWZ1bmMvc3JjL19mdW5jL2pzb25GdW5jLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLE9BQU8sRUFBRSxRQUFRLEVBQUUsTUFBTSxZQUFZLENBQUM7QUFFdEM7O0dBRUc7QUFDSCxNQUFNLE9BQU8sUUFBUTtJQUNwQjs7Ozs7T0FLRztJQUNILE1BQU0sQ0FBQyxPQUFPLENBQUksS0FBZSxFQUFFLFlBQW9CO1FBQ3RELE9BQU8sS0FBSyxDQUFDLE1BQU0sQ0FBQyxVQUFVLEdBQVEsRUFBRSxHQUFRO1lBQy9DLE1BQU0sR0FBRyxHQUFHLEdBQUcsQ0FBQyxZQUFZLENBQUMsQ0FBQztZQUM5QixJQUFJLENBQUMsR0FBRyxDQUFDLEdBQUcsQ0FBQyxFQUFFLENBQUM7Z0JBQ2YsR0FBRyxDQUFDLEdBQUcsQ0FBQyxHQUFHLEVBQUUsQ0FBQztZQUNmLENBQUM7WUFDRCxHQUFHLENBQUMsR0FBRyxDQUFDLENBQUMsSUFBSSxDQUFDLEdBQUcsQ0FBQyxDQUFDO1lBQ25CLE9BQU8sR0FBRyxDQUFDO1FBQ1osQ0FBQyxFQUFFLEVBQUUsQ0FBQyxDQUFDO0lBQ1IsQ0FBQztJQUVEOzs7Ozs7T0FNRztJQUNILE1BQU0sQ0FBQyxXQUFXLENBQUksS0FBZSxFQUFFLFlBQW9CO1FBQzFELE9BQU8sS0FBSyxDQUFDLE1BQU0sQ0FBQyxVQUFVLEdBQVEsRUFBRSxHQUFRO1lBQy9DLE1BQU0sR0FBRyxHQUFHLFFBQVEsQ0FBQyw0QkFBNEIsQ0FBQyxHQUFHLENBQUMsWUFBWSxDQUFDLENBQUMsQ0FBQztZQUNyRSxJQUFJLEdBQUcsSUFBSSxJQUFJLEVBQUUsQ0FBQztnQkFDakIsSUFBSSxDQUFDLEdBQUcsQ0FBQyxHQUFHLENBQUMsRUFBRSxDQUFDO29CQUNmLEdBQUcsQ0FBQyxHQUFHLENBQUMsR0FBRyxFQUFFLENBQUM7Z0JBQ2YsQ0FBQztnQkFDRCxHQUFHLENBQUMsR0FBRyxDQUFDLENBQUMsSUFBSSxDQUFDLEdBQUcsQ0FBQyxDQUFDO1lBQ3BCLENBQUM7WUFDRCxPQUFPLEdBQUcsQ0FBQztRQUNaLENBQUMsRUFBRSxFQUFFLENBQUMsQ0FBQztJQUNSLENBQUM7SUFFRDs7OztPQUlHO0lBQ0gsTUFBTSxDQUFDLHVCQUF1QixDQUFDLEdBQVE7UUFDdEMsS0FBSyxNQUFNLENBQUMsSUFBSSxHQUFHLEVBQUUsQ0FBQztZQUNyQixJQUFJLENBQUMsR0FBRyxHQUFHLENBQUMsQ0FBQyxDQUFDLENBQUM7WUFDZixJQUFJLENBQUMsS0FBSyxJQUFJLElBQUksQ0FBQyxLQUFLLEVBQUUsRUFBRSxDQUFDO2dCQUM1QixPQUFPLEdBQUcsQ0FBQyxDQUFDLENBQUMsQ0FBQztZQUNmLENBQUM7aUJBQU0sSUFBSSxPQUFPLENBQUMsS0FBSyxRQUFRLElBQUksQ0FBQyxLQUFLLElBQUksRUFBRSxDQUFDO2dCQUNoRCxJQUFJLENBQUMsdUJBQXVCLENBQUMsQ0FBQyxDQUFDLENBQUM7WUFDakMsQ0FBQztRQUNGLENBQUM7SUFDRixDQUFDO0lBRUQ7OztPQUdHO0lBQ0gsTUFBTSxDQUFDLGdCQUFnQixDQUFDLEdBQVE7UUFDL0IsS0FBSyxNQUFNLENBQUMsSUFBSSxHQUFHLEVBQUUsQ0FBQztZQUNyQixJQUFJLEdBQUcsQ0FBQyxDQUFDLENBQUMsS0FBSyxJQUFJLEVBQUUsQ0FBQztnQkFDckIsT0FBTyxHQUFHLENBQUMsQ0FBQyxDQUFDLENBQUM7WUFDZixDQUFDO1FBQ0YsQ0FBQztJQUNGLENBQUM7Q0FFRCIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IERhdGVGdW5jIH0gZnJvbSAnLi9kYXRlRnVuYyc7XHJcblxyXG4vKipcclxuICogQmFzaWMgSlNPTiBmdW5jdGlvbnNcclxuICovXHJcbmV4cG9ydCBjbGFzcyBKc29uRnVuYyB7XHJcblx0LyoqXHJcblx0ICogXHJcblx0ICogQHBhcmFtIGFycmF5IEdyb3VwIGJ5IGEgcHJvcGVydHkgb2YgYXJyYXkgZWxlbWVudC5cclxuXHQgKiBAcGFyYW0gcHJvcGVydHlOYW1lIFxyXG5cdCAqIEByZXR1cm5zIFxyXG5cdCAqL1xyXG5cdHN0YXRpYyBncm91cEJ5PFQ+KGFycmF5OiBBcnJheTxUPiwgcHJvcGVydHlOYW1lOiBzdHJpbmcpIHtcclxuXHRcdHJldHVybiBhcnJheS5yZWR1Y2UoZnVuY3Rpb24gKGFjYzogYW55LCBvYmo6IGFueSkge1xyXG5cdFx0XHRjb25zdCBrZXkgPSBvYmpbcHJvcGVydHlOYW1lXTtcclxuXHRcdFx0aWYgKCFhY2Nba2V5XSkge1xyXG5cdFx0XHRcdGFjY1trZXldID0gW107XHJcblx0XHRcdH1cclxuXHRcdFx0YWNjW2tleV0ucHVzaChvYmopO1xyXG5cdFx0XHRyZXR1cm4gYWNjO1xyXG5cdFx0fSwge30pO1xyXG5cdH1cclxuXHJcblx0LyoqXHJcblx0ICogR3JvdXAgYnkgYSBkYXRlIHByb3BlcnR5LiBUaGUga2V5IGlzIGFsd2F5cyBvZiBzdHJpbmcgdHlwZSBhbmQgcmVwcmVzZW50aW5nIG1pbGxpc2Vjb25kcy4gXHJcblx0ICogVGhlIGNsaWVudCBzaG91bGQgY29udmVydCB0aGUgc3RyaW5nIHRvIG51bWJlci5cclxuXHQgKiBBbmd1bGFyIGRhdGUgcGlwZSBjb3VsZCBhY3R1YWxseSBjb25zdW1lIHN1Y2ggc3RyaW5nIHdpdGhvdXQgZXhwbGljaXRseSBjb252ZXJ0aW5nIHRvIG51bWJlci5cclxuXHQgKiBAcGFyYW0gYXJyYXlcclxuXHQgKiBAcGFyYW0gcHJvcGVydHlOYW1lXHJcblx0ICovXHJcblx0c3RhdGljIGdyb3VwQnlEYXRlPFQ+KGFycmF5OiBBcnJheTxUPiwgcHJvcGVydHlOYW1lOiBzdHJpbmcpIHtcclxuXHRcdHJldHVybiBhcnJheS5yZWR1Y2UoZnVuY3Rpb24gKGFjYzogYW55LCBvYmo6IGFueSkge1xyXG5cdFx0XHRjb25zdCBrZXkgPSBEYXRlRnVuYy5kYXRlVGltZVV0Y1RvTG9jYWxEYXRlTnVtYmVyKG9ialtwcm9wZXJ0eU5hbWVdKTtcclxuXHRcdFx0aWYgKGtleSAhPSBudWxsKSB7XHJcblx0XHRcdFx0aWYgKCFhY2Nba2V5XSkge1xyXG5cdFx0XHRcdFx0YWNjW2tleV0gPSBbXTtcclxuXHRcdFx0XHR9XHJcblx0XHRcdFx0YWNjW2tleV0ucHVzaChvYmopO1xyXG5cdFx0XHR9XHJcblx0XHRcdHJldHVybiBhY2M7XHJcblx0XHR9LCB7fSk7XHJcblx0fVxyXG5cclxuXHQvKipcclxuXHQgKiBSZW1vdmUgbnVsbCBvciBlbXB0eSBmaWVsZHMgaW5jbHVkaW5nIHRob3NlIGluIG5lc3RlZCBvYmplY3RzLlxyXG5cdCAqIFRoaXMgaXMgdXNlZnVsIGZvciByZWR1Y2luZyBwYXlsb2FkIG9mIEFKQVggc2VyaWFsaXphdGlvbi5cclxuXHQgKiBAcGFyYW0gb2JqIFxyXG5cdCAqL1xyXG5cdHN0YXRpYyByZW1vdmVOdWxsT3JFbXB0eUZpZWxkcyhvYmo6IGFueSkge1xyXG5cdFx0Zm9yIChjb25zdCBmIGluIG9iaikge1xyXG5cdFx0XHRsZXQgcCA9IG9ialtmXTtcclxuXHRcdFx0aWYgKHAgPT09IG51bGwgfHwgcCA9PT0gJycpIHtcclxuXHRcdFx0XHRkZWxldGUgb2JqW2ZdO1xyXG5cdFx0XHR9IGVsc2UgaWYgKHR5cGVvZiBwID09PSAnb2JqZWN0JyAmJiBwICE9PSBudWxsKSB7XHJcblx0XHRcdFx0dGhpcy5yZW1vdmVOdWxsT3JFbXB0eUZpZWxkcyhwKTtcclxuXHRcdFx0fVxyXG5cdFx0fVxyXG5cdH1cclxuXHJcblx0LyoqXHJcblx0ICogXHJcblx0ICogQHBhcmFtIG9iaiBSZW1vdmUgbnVsbCBmaWVsZHMgb2Ygb2JqZWN0IGF0IG9ubHkgdGhlIDFzdCBsZXZlbC5cclxuXHQgKi9cclxuXHRzdGF0aWMgcmVtb3ZlTnVsbEZpZWxkcyhvYmo6IGFueSkge1xyXG5cdFx0Zm9yIChjb25zdCBmIGluIG9iaikge1xyXG5cdFx0XHRpZiAob2JqW2ZdID09PSBudWxsKSB7XHJcblx0XHRcdFx0ZGVsZXRlIG9ialtmXTtcclxuXHRcdFx0fVxyXG5cdFx0fVxyXG5cdH1cclxuXHJcbn1cclxuIl19