@datarailsshared/datarailsshared 1.6.265 → 1.6.267
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/datarailsshared-datarailsshared-1.6.267.tgz +0 -0
- package/esm2022/lib/dr-inputs/date-pickers/dr-date-picker/dr-date-picker.component.mjs +5 -5
- package/esm2022/lib/dr-inputs/date-pickers/dr-date-picker-with-timeframe/dr-date-picker-with-timeframe.component.mjs +2 -2
- package/esm2022/lib/dr-inputs/date-pickers/dr-date-picker_custom-header/dr-date-picker_custom-header.component.mjs +27 -17
- package/esm2022/lib/dr-inputs/date-pickers/dr-fiscal-month-calendar-picker/dr-fiscal-month-calendar-header.component.mjs +3 -3
- package/esm2022/lib/dr-inputs/date-pickers/dr-fiscal-month-calendar-picker/dr-fiscal-month-calendar-picker.component.mjs +5 -5
- package/esm2022/lib/dr-inputs/date-pickers/week-selector/week-selector.component.mjs +68 -34
- package/fesm2022/datarailsshared-datarailsshared.mjs +134 -91
- package/fesm2022/datarailsshared-datarailsshared.mjs.map +1 -1
- package/lib/dr-inputs/date-pickers/dr-date-picker_custom-header/dr-date-picker_custom-header.component.d.ts +7 -5
- package/lib/dr-inputs/date-pickers/week-selector/week-selector.component.d.ts +28 -4
- package/package.json +1 -1
- package/datarailsshared-datarailsshared-1.6.265.tgz +0 -0
|
@@ -1,43 +1,74 @@
|
|
|
1
|
-
import { ChangeDetectionStrategy,
|
|
1
|
+
import { ChangeDetectionStrategy, Component, DestroyRef, inject, signal } from '@angular/core';
|
|
2
|
+
import { takeUntilDestroyed } from '@angular/core/rxjs-interop';
|
|
2
3
|
import { CommonModule } from '@angular/common';
|
|
3
4
|
import { MatCalendar } from '@angular/material/datepicker';
|
|
4
|
-
import {
|
|
5
|
+
import { delay, startWith } from 'rxjs';
|
|
5
6
|
import { DrDatePickerService } from '../services/dr-date-picker.service';
|
|
6
7
|
import moment from 'moment';
|
|
7
8
|
import * as i0 from "@angular/core";
|
|
8
9
|
import * as i1 from "@angular/common";
|
|
9
|
-
function
|
|
10
|
-
const
|
|
11
|
-
i0.ɵɵelementStart(0, "
|
|
12
|
-
i0.ɵɵlistener("click", function
|
|
13
|
-
i0.ɵɵtext(
|
|
14
|
-
i0.ɵɵelementEnd();
|
|
10
|
+
function WeekSelectorComponent_li_4_Template(rf, ctx) { if (rf & 1) {
|
|
11
|
+
const _r4 = i0.ɵɵgetCurrentView();
|
|
12
|
+
i0.ɵɵelementStart(0, "li")(1, "button", 5);
|
|
13
|
+
i0.ɵɵlistener("focus", function WeekSelectorComponent_li_4_Template_button_focus_1_listener($event) { const restoredCtx = i0.ɵɵrestoreView(_r4); const i_r2 = restoredCtx.index; const ctx_r3 = i0.ɵɵnextContext(); return i0.ɵɵresetView(ctx_r3.toggleWeekPreview($event, i_r2, true)); })("blur", function WeekSelectorComponent_li_4_Template_button_blur_1_listener($event) { const restoredCtx = i0.ɵɵrestoreView(_r4); const i_r2 = restoredCtx.index; const ctx_r5 = i0.ɵɵnextContext(); return i0.ɵɵresetView(ctx_r5.toggleWeekPreview($event, i_r2, false)); })("mouseover", function WeekSelectorComponent_li_4_Template_button_mouseover_1_listener($event) { const restoredCtx = i0.ɵɵrestoreView(_r4); const i_r2 = restoredCtx.index; const ctx_r6 = i0.ɵɵnextContext(); return i0.ɵɵresetView(ctx_r6.toggleWeekPreview($event, i_r2, true)); })("mouseleave", function WeekSelectorComponent_li_4_Template_button_mouseleave_1_listener($event) { const restoredCtx = i0.ɵɵrestoreView(_r4); const i_r2 = restoredCtx.index; const ctx_r7 = i0.ɵɵnextContext(); return i0.ɵɵresetView(ctx_r7.toggleWeekPreview($event, i_r2, false)); })("click", function WeekSelectorComponent_li_4_Template_button_click_1_listener() { const restoredCtx = i0.ɵɵrestoreView(_r4); const week_r1 = restoredCtx.$implicit; const ctx_r8 = i0.ɵɵnextContext(); ctx_r8.service.datePickerInstance.close(); return i0.ɵɵresetView(ctx_r8.service.weekSelectionChange$.next(week_r1)); });
|
|
14
|
+
i0.ɵɵtext(2);
|
|
15
|
+
i0.ɵɵelementEnd()();
|
|
15
16
|
} if (rf & 2) {
|
|
16
17
|
const week_r1 = ctx.$implicit;
|
|
17
|
-
|
|
18
|
+
const i_r2 = ctx.index;
|
|
19
|
+
const ctx_r0 = i0.ɵɵnextContext();
|
|
20
|
+
i0.ɵɵadvance(1);
|
|
21
|
+
i0.ɵɵclassProp("previewed", i_r2 === ctx_r0.previewedWeek())("selected", week_r1.selected);
|
|
22
|
+
i0.ɵɵattribute("aria-current", week_r1.selected)("aria-label", "Week " + week_r1.yearWeekNumber);
|
|
18
23
|
i0.ɵɵadvance(1);
|
|
19
24
|
i0.ɵɵtextInterpolate1(" W", week_r1.yearWeekNumber, " ");
|
|
20
25
|
} }
|
|
26
|
+
/**
|
|
27
|
+
* Provides week-based interaction for Angular Material `MatCalendar`.
|
|
28
|
+
* The component renders a list of weeks for the currently active month
|
|
29
|
+
* and keeps it synchronized with the calendar hover (preview) and selection state.
|
|
30
|
+
*
|
|
31
|
+
* Key responsibilities:
|
|
32
|
+
* - Derive logical weeks from the calendar's active month.
|
|
33
|
+
* - Detect which calendar week is currently previewed/selected and reflect it in the UI.
|
|
34
|
+
*
|
|
35
|
+
* ⚠ Stability note:
|
|
36
|
+
* - `_matCalendarBody` and `previewChange` are **private Angular Material APIs**.
|
|
37
|
+
* - This component may require adjustments after Angular Material upgrades.
|
|
38
|
+
* - Related upgrade considerations are documented in `UPGRADE_CHECKLIST.md`.
|
|
39
|
+
*
|
|
40
|
+
* If private API is going to change, and we don't have access to with properties,
|
|
41
|
+
* check the following commit with class and mutation observable implementation:
|
|
42
|
+
* https://github.com/Datarails/datarailsshared/pull/126/commits/590152cff0448c5f48bff32f9cb165bbf464112b
|
|
43
|
+
*/
|
|
21
44
|
export class WeekSelectorComponent {
|
|
22
45
|
constructor() {
|
|
23
46
|
this.service = inject(DrDatePickerService);
|
|
24
47
|
this.calendar = inject(MatCalendar);
|
|
25
|
-
this.
|
|
26
|
-
this.
|
|
27
|
-
this.
|
|
28
|
-
|
|
29
|
-
|
|
30
|
-
this.
|
|
48
|
+
this.destroyRef = inject(DestroyRef);
|
|
49
|
+
this.calendarBody = null;
|
|
50
|
+
this.previewedWeek = signal(-1);
|
|
51
|
+
this.currentMonthWeeks = signal([]);
|
|
52
|
+
this.calendar.stateChanges.pipe(startWith(null), delay(0), takeUntilDestroyed()).subscribe(() => {
|
|
53
|
+
const activeDate = moment(this.calendar.activeDate);
|
|
54
|
+
const selectedRange = this.calendar.selected;
|
|
55
|
+
this.currentMonthWeeks.set(this.getWeeksInMonth(activeDate, selectedRange));
|
|
31
56
|
});
|
|
32
|
-
|
|
33
|
-
|
|
34
|
-
|
|
35
|
-
|
|
36
|
-
|
|
37
|
-
|
|
38
|
-
|
|
57
|
+
}
|
|
58
|
+
ngAfterViewInit() {
|
|
59
|
+
// monthView and _matCalendarBody are ViewChild and can be accessed only in AfterViewInit
|
|
60
|
+
this.calendarBody = this.calendar.monthView._matCalendarBody;
|
|
61
|
+
this.calendarBody.previewChange.pipe(takeUntilDestroyed(this.destroyRef)).subscribe((event) => {
|
|
62
|
+
const value = event.value ? event.value.rawValue : null;
|
|
63
|
+
this.previewedWeek.set(this.currentMonthWeeks().findIndex((week) => this.isInRange(week.start, value)));
|
|
39
64
|
});
|
|
40
65
|
}
|
|
66
|
+
toggleWeekPreview(event, weekIndex, value) {
|
|
67
|
+
if (!this.calendarBody)
|
|
68
|
+
return;
|
|
69
|
+
const weekFestDayCell = this.calendarBody.rows[weekIndex][0];
|
|
70
|
+
this.calendarBody.previewChange.emit({ value: weekFestDayCell?.enabled && value ? weekFestDayCell : null, event });
|
|
71
|
+
}
|
|
41
72
|
getWeeksInMonth(date, selectedRange) {
|
|
42
73
|
const weeks = [];
|
|
43
74
|
const firstDay = moment(date).startOf('month');
|
|
@@ -48,12 +79,9 @@ export class WeekSelectorComponent {
|
|
|
48
79
|
let weekNumber = 1;
|
|
49
80
|
while (currentWeekStart.isSameOrBefore(lastWeekEnd)) {
|
|
50
81
|
const weekEnd = moment(currentWeekStart).endOf('isoWeek');
|
|
51
|
-
const days =
|
|
52
|
-
for (let i = 0; i < 7; i++) {
|
|
53
|
-
days.push(moment(currentWeekStart).add(i, 'days'));
|
|
54
|
-
}
|
|
82
|
+
const days = this.getWeekDays(currentWeekStart);
|
|
55
83
|
const isPartial = days.some((day) => !day.isSame(date, 'month'));
|
|
56
|
-
const selected =
|
|
84
|
+
const selected = this.isInRange(currentWeekStart, selectedRange.start, days);
|
|
57
85
|
// isoWeek() returns the ISO week number (1-53)
|
|
58
86
|
const yearWeekNumber = moment(currentWeekStart).isoWeek();
|
|
59
87
|
weeks.push({
|
|
@@ -70,22 +98,28 @@ export class WeekSelectorComponent {
|
|
|
70
98
|
}
|
|
71
99
|
return weeks;
|
|
72
100
|
}
|
|
101
|
+
isInRange(weekStart, value, days = this.getWeekDays(weekStart)) {
|
|
102
|
+
return days.some((day) => day.isSame(value, 'day'));
|
|
103
|
+
}
|
|
104
|
+
getWeekDays(weekStart) {
|
|
105
|
+
return Array.from(Array(7).keys()).map((i) => moment(weekStart).add(i, 'days'));
|
|
106
|
+
}
|
|
73
107
|
/** @nocollapse */ static { this.ɵfac = function WeekSelectorComponent_Factory(t) { return new (t || WeekSelectorComponent)(); }; }
|
|
74
|
-
/** @nocollapse */ static { this.ɵcmp = /** @pureOrBreakMyCode */ i0.ɵɵdefineComponent({ type: WeekSelectorComponent, selectors: [["dr-week-selector"]], standalone: true, features: [i0.ɵɵStandaloneFeature], decls: 6, vars: 1, consts: [[1, "weeks"], [1, "header"], [
|
|
108
|
+
/** @nocollapse */ static { this.ɵcmp = /** @pureOrBreakMyCode */ i0.ɵɵdefineComponent({ type: WeekSelectorComponent, selectors: [["dr-week-selector"]], standalone: true, features: [i0.ɵɵStandaloneFeature], decls: 6, vars: 1, consts: [[1, "weeks"], [1, "header"], ["role", "list", "aria-label", "Weeks of the month", 1, "weeks__container"], [4, "ngFor", "ngForOf"], [1, "weeks-border"], [1, "btn", 3, "focus", "blur", "mouseover", "mouseleave", "click"]], template: function WeekSelectorComponent_Template(rf, ctx) { if (rf & 1) {
|
|
75
109
|
i0.ɵɵelementStart(0, "div", 0)(1, "div", 1);
|
|
76
110
|
i0.ɵɵtext(2, "Week");
|
|
77
111
|
i0.ɵɵelementEnd();
|
|
78
|
-
i0.ɵɵelementStart(3, "
|
|
79
|
-
i0.ɵɵtemplate(4,
|
|
112
|
+
i0.ɵɵelementStart(3, "ul", 2);
|
|
113
|
+
i0.ɵɵtemplate(4, WeekSelectorComponent_li_4_Template, 3, 7, "li", 3);
|
|
80
114
|
i0.ɵɵelementEnd()();
|
|
81
115
|
i0.ɵɵelement(5, "div", 4);
|
|
82
116
|
} if (rf & 2) {
|
|
83
117
|
i0.ɵɵadvance(4);
|
|
84
|
-
i0.ɵɵproperty("ngForOf", ctx.currentMonthWeeks);
|
|
85
|
-
} }, dependencies: [CommonModule, i1.NgForOf], styles: ["[_nghost-%COMP%]{font-size:14px;line-height:24px;font-weight:400;text-align:center;color:#333;display:flex}[_nghost-%COMP%] .weeks[_ngcontent-%COMP%]{height:100%;flex-direction:column;display:flex}[_nghost-%COMP%] .weeks-border[_ngcontent-%COMP%]{height:100%;width:1px;background-color:#dfe0e3;margin:0 4px}[_nghost-%COMP%] .weeks__container[_ngcontent-%COMP%]{display:flex;flex-direction:column;
|
|
118
|
+
i0.ɵɵproperty("ngForOf", ctx.currentMonthWeeks());
|
|
119
|
+
} }, dependencies: [CommonModule, i1.NgForOf], styles: ["[_nghost-%COMP%]{font-size:14px;line-height:24px;font-weight:400;text-align:center;color:#333;display:flex}[_nghost-%COMP%] .weeks[_ngcontent-%COMP%]{height:100%;flex-direction:column;display:flex}[_nghost-%COMP%] .weeks-border[_ngcontent-%COMP%]{height:100%;width:1px;background-color:#dfe0e3;margin:0 4px}[_nghost-%COMP%] .weeks__container[_ngcontent-%COMP%]{display:flex;flex-direction:column;gap:5px;height:100%;margin:0;padding:0;list-style:none}[_nghost-%COMP%] .weeks__container[_ngcontent-%COMP%] li[_ngcontent-%COMP%]{flex:1}[_nghost-%COMP%] .weeks[_ngcontent-%COMP%] .header[_ngcontent-%COMP%]{font-weight:600;color:#19181a;padding-bottom:8px;line-height:16px}[_nghost-%COMP%] .weeks[_ngcontent-%COMP%] .btn[_ngcontent-%COMP%]{padding:9px 16px;border-radius:40px;cursor:pointer;background-color:transparent;border:none;height:100%}[_nghost-%COMP%] .weeks[_ngcontent-%COMP%] .btn.previewed[_ngcontent-%COMP%], [_nghost-%COMP%] .weeks[_ngcontent-%COMP%] .btn[_ngcontent-%COMP%]:hover{background-color:#f5f5f5;color:#19181a}[_nghost-%COMP%] .weeks[_ngcontent-%COMP%] .btn.selected[_ngcontent-%COMP%]{font-weight:600;background-color:#4646ce;color:#fff}"], changeDetection: 0 }); }
|
|
86
120
|
}
|
|
87
121
|
(function () { (typeof ngDevMode === "undefined" || ngDevMode) && i0.ɵsetClassMetadata(WeekSelectorComponent, [{
|
|
88
122
|
type: Component,
|
|
89
|
-
args: [{ selector: 'dr-week-selector', standalone: true, imports: [CommonModule], changeDetection: ChangeDetectionStrategy.OnPush, template: "<div class=\"weeks\">\n <div class=\"header\">Week</div>\n\n <
|
|
123
|
+
args: [{ selector: 'dr-week-selector', standalone: true, imports: [CommonModule], changeDetection: ChangeDetectionStrategy.OnPush, template: "<div class=\"weeks\">\n <div class=\"header\">Week</div>\n\n <ul role=\"list\" class=\"weeks__container\" aria-label=\"Weeks of the month\">\n <li *ngFor=\"let week of currentMonthWeeks(); let i = index\">\n <button\n class=\"btn\"\n [class.previewed]=\"i === previewedWeek()\"\n [class.selected]=\"week.selected\"\n [attr.aria-current]=\"week.selected\"\n [attr.aria-label]=\"'Week ' + week.yearWeekNumber\"\n (focus)=\"toggleWeekPreview($event, i, true)\"\n (blur)=\"toggleWeekPreview($event, i, false)\"\n (mouseover)=\"toggleWeekPreview($event, i, true)\"\n (mouseleave)=\"toggleWeekPreview($event, i, false)\"\n (click)=\"service.datePickerInstance.close(); service.weekSelectionChange$.next(week)\">\n W{{ week.yearWeekNumber }}\n </button>\n </li>\n </ul>\n</div>\n<div class=\"weeks-border\"></div>\n", styles: [":host{font-size:14px;line-height:24px;font-weight:400;text-align:center;color:#333;display:flex}:host .weeks{height:100%;flex-direction:column;display:flex}:host .weeks-border{height:100%;width:1px;background-color:#dfe0e3;margin:0 4px}:host .weeks__container{display:flex;flex-direction:column;gap:5px;height:100%;margin:0;padding:0;list-style:none}:host .weeks__container li{flex:1}:host .weeks .header{font-weight:600;color:#19181a;padding-bottom:8px;line-height:16px}:host .weeks .btn{padding:9px 16px;border-radius:40px;cursor:pointer;background-color:transparent;border:none;height:100%}:host .weeks .btn.previewed,:host .weeks .btn:hover{background-color:#f5f5f5;color:#19181a}:host .weeks .btn.selected{font-weight:600;background-color:#4646ce;color:#fff}\n"] }]
|
|
90
124
|
}], function () { return []; }, null); })();
|
|
91
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"week-selector.component.js","sourceRoot":"","sources":["../../../../../../../projects/datarailsshared/src/lib/dr-inputs/date-pickers/week-selector/week-selector.component.ts","../../../../../../../projects/datarailsshared/src/lib/dr-inputs/date-pickers/week-selector/week-selector.component.html"],"names":[],"mappings":"AAAA,OAAO,EAAE,uBAAuB,EAAE,iBAAiB,EAAE,SAAS,EAAE,UAAU,EAAE,MAAM,EAAE,MAAM,eAAe,CAAC;AAC1G,OAAO,EAAE,YAAY,EAAE,MAAM,iBAAiB,CAAC;AAC/C,OAAO,EAAa,WAAW,EAAE,MAAM,8BAA8B,CAAC;AACtE,OAAO,EAAE,SAAS,EAAE,OAAO,EAAE,SAAS,EAAE,MAAM,MAAM,CAAC;AACrD,OAAO,EAAE,mBAAmB,EAAE,MAAM,oCAAoC,CAAC;AACzE,OAAO,MAAkB,MAAM,QAAQ,CAAC;;;;;ICDhC,8BAI2C;IAHvC,mNAAS,yCAAkC,SAAE,eAAA,iDAAuC,CAAA,IAAC;IAIrF,YACJ;IAAA,iBAAM;;;IAHF,4CAAgC;IAEhC,eACJ;IADI,wDACJ;;ADeR,MAAM,OAAO,qBAAqB;IAQ9B;QAPS,YAAO,GAAG,MAAM,CAAC,mBAAmB,CAAC,CAAC;QAC9B,aAAQ,GAAG,MAAM,CAAC,WAAW,CAAC,CAAC;QAC/B,QAAG,GAAG,MAAM,CAAC,iBAAiB,CAAC,CAAC;QAChC,aAAQ,GAAG,IAAI,OAAO,EAAQ,CAAC;QAEhD,sBAAiB,GAAe,EAAE,CAAC;QAG/B,MAAM,CAAC,UAAU,EAAE,EAAE,QAAQ,EAAE,IAAI,EAAE,CAAC,CAAC,SAAS,CAAC,GAAG,EAAE;YAClD,IAAI,CAAC,QAAQ,CAAC,IAAI,EAAE,CAAC;YACrB,IAAI,CAAC,QAAQ,CAAC,QAAQ,EAAE,CAAC;QAC7B,CAAC,CAAC,CAAC;QAEH,IAAI,CAAC,QAAQ,CAAC,YAAY,CAAC,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,EAAE,SAAS,CAAC,IAAI,CAAC,QAAQ,CAAC,CAAC,CAAC,SAAS,CAAC,GAAG,EAAE;YACtF,UAAU,CAAC,GAAG,EAAE;gBACZ,MAAM,UAAU,GAAG,MAAM,CAAC,IAAI,CAAC,QAAQ,CAAC,UAAU,CAAC,CAAC;gBACpD,MAAM,aAAa,GAAsB,IAAI,CAAC,QAAQ,CAAC,QAAQ,CAAC;gBAChE,IAAI,CAAC,iBAAiB,GAAG,IAAI,CAAC,eAAe,CAAC,UAAU,EAAE,aAAa,CAAC,CAAC;gBACzE,IAAI,CAAC,GAAG,CAAC,aAAa,EAAE,CAAC;YAC7B,CAAC,CAAC,CAAC;QACP,CAAC,CAAC,CAAC;IACP,CAAC;IAEO,eAAe,CAAC,IAAY,EAAE,aAAgC;QAClE,MAAM,KAAK,GAAe,EAAE,CAAC;QAC7B,MAAM,QAAQ,GAAG,MAAM,CAAC,IAAI,CAAC,CAAC,OAAO,CAAC,OAAO,CAAC,CAAC;QAC/C,MAAM,OAAO,GAAG,MAAM,CAAC,IAAI,CAAC,CAAC,KAAK,CAAC,OAAO,CAAC,CAAC;QAE5C,MAAM,cAAc,GAAG,MAAM,CAAC,QAAQ,CAAC,CAAC,OAAO,CAAC,SAAS,CAAC,CAAC;QAC3D,MAAM,WAAW,GAAG,MAAM,CAAC,OAAO,CAAC,CAAC,KAAK,CAAC,SAAS,CAAC,CAAC;QAErD,IAAI,gBAAgB,GAAG,MAAM,CAAC,cAAc,CAAC,CAAC;QAC9C,IAAI,UAAU,GAAG,CAAC,CAAC;QAEnB,OAAO,gBAAgB,CAAC,cAAc,CAAC,WAAW,CAAC,EAAE;YACjD,MAAM,OAAO,GAAG,MAAM,CAAC,gBAAgB,CAAC,CAAC,KAAK,CAAC,SAAS,CAAC,CAAC;YAC1D,MAAM,IAAI,GAAa,EAAE,CAAC;YAE1B,KAAK,IAAI,CAAC,GAAG,CAAC,EAAE,CAAC,GAAG,CAAC,EAAE,CAAC,EAAE,EAAE;gBACxB,IAAI,CAAC,IAAI,CAAC,MAAM,CAAC,gBAAgB,CAAC,CAAC,GAAG,CAAC,CAAC,EAAE,MAAM,CAAC,CAAC,CAAC;aACtD;YAED,MAAM,SAAS,GAAG,IAAI,CAAC,IAAI,CAAC,CAAC,GAAG,EAAE,EAAE,CAAC,CAAC,GAAG,CAAC,MAAM,CAAC,IAAI,EAAE,OAAO,CAAC,CAAC,CAAC;YACjE,MAAM,QAAQ,GAAG,IAAI,CAAC,IAAI,CAAC,CAAC,GAAG,EAAE,EAAE,CAAC,GAAG,CAAC,MAAM,CAAC,aAAa,CAAC,KAAK,EAAE,KAAK,CAAC,CAAC,CAAC;YAE5E,+CAA+C;YAC/C,MAAM,cAAc,GAAG,MAAM,CAAC,gBAAgB,CAAC,CAAC,OAAO,EAAE,CAAC;YAE1D,KAAK,CAAC,IAAI,CAAC;gBACP,UAAU;gBACV,cAAc;gBACd,KAAK,EAAE,MAAM,CAAC,gBAAgB,CAAC;gBAC/B,GAAG,EAAE,MAAM,CAAC,OAAO,CAAC;gBACpB,IAAI;gBACJ,SAAS;gBACT,QAAQ;aACX,CAAC,CAAC;YAEH,gBAAgB,CAAC,GAAG,CAAC,CAAC,EAAE,MAAM,CAAC,CAAC;YAChC,UAAU,EAAE,CAAC;SAChB;QAED,OAAO,KAAK,CAAC;IACjB,CAAC;yGAhEQ,qBAAqB;mGAArB,qBAAqB;YCzBlC,8BAAmB,aAAA;YACK,oBAAI;YAAA,iBAAM;YAE9B,8BAA8B;YAC1B,sEAMM;YACV,iBAAM,EAAA;YAEV,yBAAgC;;YALH,eAAoB;YAApB,+CAAoB;4BDYnC,YAAY;;uFAKb,qBAAqB;cARjC,SAAS;2BACI,kBAAkB,cAChB,IAAI,WACP,CAAC,YAAY,CAAC,mBAGN,uBAAuB,CAAC,MAAM","sourcesContent":["import { ChangeDetectionStrategy, ChangeDetectorRef, Component, DestroyRef, inject } from '@angular/core';\nimport { CommonModule } from '@angular/common';\nimport { DateRange, MatCalendar } from '@angular/material/datepicker';\nimport { startWith, Subject, takeUntil } from 'rxjs';\nimport { DrDatePickerService } from '../services/dr-date-picker.service';\nimport moment, { Moment } from 'moment';\n\nexport interface WeekInfo {\n    weekNumber: number;\n    yearWeekNumber: number;\n    start: Moment;\n    end: Moment;\n    days: Moment[];\n    isPartial: boolean;\n    selected: boolean;\n}\n\n@Component({\n    selector: 'dr-week-selector',\n    standalone: true,\n    imports: [CommonModule],\n    templateUrl: './week-selector.component.html',\n    styleUrls: ['./week-selector.component.scss'],\n    changeDetection: ChangeDetectionStrategy.OnPush,\n})\nexport class WeekSelectorComponent {\n    readonly service = inject(DrDatePickerService);\n    private readonly calendar = inject(MatCalendar);\n    private readonly cdr = inject(ChangeDetectorRef);\n    private readonly destroy$ = new Subject<void>();\n\n    currentMonthWeeks: WeekInfo[] = [];\n\n    constructor() {\n        inject(DestroyRef, { skipSelf: true }).onDestroy(() => {\n            this.destroy$.next();\n            this.destroy$.complete();\n        });\n\n        this.calendar.stateChanges.pipe(startWith(null), takeUntil(this.destroy$)).subscribe(() => {\n            setTimeout(() => {\n                const activeDate = moment(this.calendar.activeDate);\n                const selectedRange: DateRange<Moment> = this.calendar.selected;\n                this.currentMonthWeeks = this.getWeeksInMonth(activeDate, selectedRange);\n                this.cdr.detectChanges();\n            });\n        });\n    }\n\n    private getWeeksInMonth(date: Moment, selectedRange: DateRange<Moment>): WeekInfo[] {\n        const weeks: WeekInfo[] = [];\n        const firstDay = moment(date).startOf('month');\n        const lastDay = moment(date).endOf('month');\n\n        const firstWeekStart = moment(firstDay).startOf('isoWeek');\n        const lastWeekEnd = moment(lastDay).endOf('isoWeek');\n\n        let currentWeekStart = moment(firstWeekStart);\n        let weekNumber = 1;\n\n        while (currentWeekStart.isSameOrBefore(lastWeekEnd)) {\n            const weekEnd = moment(currentWeekStart).endOf('isoWeek');\n            const days: Moment[] = [];\n\n            for (let i = 0; i < 7; i++) {\n                days.push(moment(currentWeekStart).add(i, 'days'));\n            }\n\n            const isPartial = days.some((day) => !day.isSame(date, 'month'));\n            const selected = days.some((day) => day.isSame(selectedRange.start, 'day'));\n\n            // isoWeek() returns the ISO week number (1-53)\n            const yearWeekNumber = moment(currentWeekStart).isoWeek();\n\n            weeks.push({\n                weekNumber,\n                yearWeekNumber,\n                start: moment(currentWeekStart),\n                end: moment(weekEnd),\n                days,\n                isPartial,\n                selected,\n            });\n\n            currentWeekStart.add(1, 'week');\n            weekNumber++;\n        }\n\n        return weeks;\n    }\n}\n","<div class=\"weeks\">\n    <div class=\"header\">Week</div>\n\n    <div class=\"weeks__container\">\n        <div\n            (click)=\"service.datePickerInstance.close(); service.weekSelectionChange$.next(week)\"\n            class=\"btn\"\n            [class.selected]=\"week.selected\"\n            *ngFor=\"let week of currentMonthWeeks\">\n            W{{ week.yearWeekNumber }}\n        </div>\n    </div>\n</div>\n<div class=\"weeks-border\"></div>\n"]}
|
|
125
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"week-selector.component.js","sourceRoot":"","sources":["../../../../../../../projects/datarailsshared/src/lib/dr-inputs/date-pickers/week-selector/week-selector.component.ts","../../../../../../../projects/datarailsshared/src/lib/dr-inputs/date-pickers/week-selector/week-selector.component.html"],"names":[],"mappings":"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","sourcesContent":["import { AfterViewInit, ChangeDetectionStrategy, Component, DestroyRef, inject, signal } from '@angular/core';\nimport { takeUntilDestroyed } from '@angular/core/rxjs-interop';\nimport { CommonModule } from '@angular/common';\nimport { DateRange, MatCalendar, MatCalendarBody } from '@angular/material/datepicker';\nimport { delay, startWith } from 'rxjs';\nimport { DrDatePickerService } from '../services/dr-date-picker.service';\nimport moment, { Moment } from 'moment';\n\nexport interface WeekInfo {\n    weekNumber: number;\n    yearWeekNumber: number;\n    start: Moment;\n    end: Moment;\n    days: Moment[];\n    isPartial: boolean;\n    selected: boolean;\n}\n\n/**\n * Provides week-based interaction for Angular Material `MatCalendar`.\n * The component renders a list of weeks for the currently active month\n * and keeps it synchronized with the calendar hover (preview) and selection state.\n *\n * Key responsibilities:\n * - Derive logical weeks from the calendar's active month.\n * - Detect which calendar week is currently previewed/selected and reflect it in the UI.\n *\n * ⚠ Stability note:\n * - `_matCalendarBody` and `previewChange` are **private Angular Material APIs**.\n * - This component may require adjustments after Angular Material upgrades.\n * - Related upgrade considerations are documented in `UPGRADE_CHECKLIST.md`.\n *\n * If private API is going to change, and we don't have access to with properties,\n * check the following commit with class and mutation observable implementation:\n * https://github.com/Datarails/datarailsshared/pull/126/commits/590152cff0448c5f48bff32f9cb165bbf464112b\n */\n@Component({\n    selector: 'dr-week-selector',\n    standalone: true,\n    imports: [CommonModule],\n    templateUrl: './week-selector.component.html',\n    styleUrls: ['./week-selector.component.scss'],\n    changeDetection: ChangeDetectionStrategy.OnPush,\n})\nexport class WeekSelectorComponent implements AfterViewInit {\n    readonly service = inject(DrDatePickerService);\n\n    private readonly calendar = inject(MatCalendar);\n    private readonly destroyRef = inject(DestroyRef);\n\n    private calendarBody: MatCalendarBody<Moment> | null = null;\n\n    readonly previewedWeek = signal(-1);\n    readonly currentMonthWeeks = signal<WeekInfo[]>([]);\n\n    constructor() {\n        this.calendar.stateChanges.pipe(startWith(null), delay(0), takeUntilDestroyed()).subscribe(() => {\n            const activeDate = moment(this.calendar.activeDate);\n            const selectedRange: DateRange<Moment> = this.calendar.selected;\n\n            this.currentMonthWeeks.set(this.getWeeksInMonth(activeDate, selectedRange));\n        });\n    }\n\n    public ngAfterViewInit(): void {\n        // monthView and _matCalendarBody are ViewChild and can be accessed only in AfterViewInit\n        this.calendarBody = this.calendar.monthView._matCalendarBody;\n\n        this.calendarBody.previewChange.pipe(takeUntilDestroyed(this.destroyRef)).subscribe((event) => {\n            const value = event.value ? event.value.rawValue! : null;\n\n            this.previewedWeek.set(this.currentMonthWeeks().findIndex((week) => this.isInRange(week.start, value)));\n        });\n    }\n\n    public toggleWeekPreview(event: Event, weekIndex: number, value: boolean): void {\n        if (!this.calendarBody) return;\n\n        const weekFestDayCell = this.calendarBody.rows[weekIndex][0];\n\n        this.calendarBody.previewChange.emit({ value: weekFestDayCell?.enabled && value ? weekFestDayCell : null, event });\n    }\n\n    private getWeeksInMonth(date: Moment, selectedRange: DateRange<Moment>): WeekInfo[] {\n        const weeks: WeekInfo[] = [];\n        const firstDay = moment(date).startOf('month');\n        const lastDay = moment(date).endOf('month');\n\n        const firstWeekStart = moment(firstDay).startOf('isoWeek');\n        const lastWeekEnd = moment(lastDay).endOf('isoWeek');\n\n        let currentWeekStart = moment(firstWeekStart);\n        let weekNumber = 1;\n\n        while (currentWeekStart.isSameOrBefore(lastWeekEnd)) {\n            const weekEnd = moment(currentWeekStart).endOf('isoWeek');\n            const days = this.getWeekDays(currentWeekStart);\n\n            const isPartial = days.some((day) => !day.isSame(date, 'month'));\n            const selected = this.isInRange(currentWeekStart, selectedRange.start, days);\n\n            // isoWeek() returns the ISO week number (1-53)\n            const yearWeekNumber = moment(currentWeekStart).isoWeek();\n\n            weeks.push({\n                weekNumber,\n                yearWeekNumber,\n                start: moment(currentWeekStart),\n                end: moment(weekEnd),\n                days,\n                isPartial,\n                selected,\n            });\n\n            currentWeekStart.add(1, 'week');\n            weekNumber++;\n        }\n\n        return weeks;\n    }\n\n    private isInRange(weekStart: Moment, value: Moment, days = this.getWeekDays(weekStart)): boolean {\n        return days.some((day) => day.isSame(value, 'day'));\n    }\n\n    private getWeekDays(weekStart: Moment): Moment[] {\n        return Array.from(Array(7).keys()).map((i) => moment(weekStart).add(i, 'days'));\n    }\n}\n","<div class=\"weeks\">\n    <div class=\"header\">Week</div>\n\n    <ul role=\"list\" class=\"weeks__container\" aria-label=\"Weeks of the month\">\n        <li *ngFor=\"let week of currentMonthWeeks(); let i = index\">\n            <button\n                class=\"btn\"\n                [class.previewed]=\"i === previewedWeek()\"\n                [class.selected]=\"week.selected\"\n                [attr.aria-current]=\"week.selected\"\n                [attr.aria-label]=\"'Week ' + week.yearWeekNumber\"\n                (focus)=\"toggleWeekPreview($event, i, true)\"\n                (blur)=\"toggleWeekPreview($event, i, false)\"\n                (mouseover)=\"toggleWeekPreview($event, i, true)\"\n                (mouseleave)=\"toggleWeekPreview($event, i, false)\"\n                (click)=\"service.datePickerInstance.close(); service.weekSelectionChange$.next(week)\">\n                W{{ week.yearWeekNumber }}\n            </button>\n        </li>\n    </ul>\n</div>\n<div class=\"weeks-border\"></div>\n"]}
|