@scania-nl/tegel-angular-extensions 0.0.8 → 0.0.9
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 +122 -24
- package/esm2022/index.mjs +3 -1
- package/esm2022/lib/components/tae-date-time-picker/tae-date-time-picker.component.mjs +366 -0
- package/esm2022/lib/components/tae-date-time-picker/tae-date-time-picker.constants.mjs +24 -0
- package/esm2022/lib/components/tae-date-time-picker/tae-date-time-picker.utils.mjs +148 -0
- package/esm2022/lib/directives/typed-template.directive.mjs +38 -0
- package/esm2022/lib/modal/components/modal/modal.component.mjs +10 -3
- package/esm2022/lib/modal/schema/modal.model.mjs +1 -1
- package/index.d.ts +176 -7
- package/package.json +1 -1
|
@@ -0,0 +1,148 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Parses an ISO date segment (`YYYY-MM-DD`) into local date parts.
|
|
3
|
+
* @param datePart ISO date segment to parse.
|
|
4
|
+
* @returns Parsed date parts or `null` when invalid.
|
|
5
|
+
*/
|
|
6
|
+
export function parseDatePartsFromIso(datePart) {
|
|
7
|
+
if (!datePart)
|
|
8
|
+
return null;
|
|
9
|
+
// Split on the ISO separator and validate expected parts.
|
|
10
|
+
const parts = datePart.split('-');
|
|
11
|
+
if (parts.length !== 3)
|
|
12
|
+
return null;
|
|
13
|
+
// Convert to numbers and normalize month to 0-based.
|
|
14
|
+
const year = parseInt(parts[0], 10);
|
|
15
|
+
const month = parseInt(parts[1], 10) - 1;
|
|
16
|
+
const day = parseInt(parts[2], 10);
|
|
17
|
+
// Reject invalid dates early.
|
|
18
|
+
if (!isValidDateParts(year, month, day))
|
|
19
|
+
return null;
|
|
20
|
+
return { year, month, day };
|
|
21
|
+
}
|
|
22
|
+
/**
|
|
23
|
+
* Validates date parts (0-based month) against calendar limits.
|
|
24
|
+
* @param year Full year value.
|
|
25
|
+
* @param month Zero-based month.
|
|
26
|
+
* @param day Day of month (1-based).
|
|
27
|
+
* @returns `true` when the date exists in the calendar.
|
|
28
|
+
*/
|
|
29
|
+
export function isValidDateParts(year, month, day) {
|
|
30
|
+
if (!Number.isFinite(year) ||
|
|
31
|
+
!Number.isFinite(month) ||
|
|
32
|
+
!Number.isFinite(day)) {
|
|
33
|
+
return false;
|
|
34
|
+
}
|
|
35
|
+
// Basic range checks for month/day.
|
|
36
|
+
if (month < 0 || month > 11 || day < 1)
|
|
37
|
+
return false;
|
|
38
|
+
// Ensure the day exists in the target month/year.
|
|
39
|
+
const lastDay = new Date(year, month + 1, 0).getDate();
|
|
40
|
+
return day <= lastDay;
|
|
41
|
+
}
|
|
42
|
+
/**
|
|
43
|
+
* Formats an ISO date/datetime value into `dd-MM-YYYY` display format.
|
|
44
|
+
* @param value ISO date or datetime string.
|
|
45
|
+
* @param mode Formatting mode (`date` or `datetime`).
|
|
46
|
+
* @returns Display-friendly string.
|
|
47
|
+
*/
|
|
48
|
+
export function formatDisplayDateTime(value, mode) {
|
|
49
|
+
if (!value) {
|
|
50
|
+
return '';
|
|
51
|
+
}
|
|
52
|
+
if (mode === 'time') {
|
|
53
|
+
return value;
|
|
54
|
+
}
|
|
55
|
+
if (mode === 'datetime') {
|
|
56
|
+
const [datePart, timePart] = value.split('T');
|
|
57
|
+
if (!datePart)
|
|
58
|
+
return value;
|
|
59
|
+
// Expect ISO `YYYY-MM-DD` for the date portion.
|
|
60
|
+
const dateParts = datePart.split('-');
|
|
61
|
+
if (dateParts.length !== 3)
|
|
62
|
+
return value;
|
|
63
|
+
const [year, month, day] = dateParts;
|
|
64
|
+
const formattedDate = `${day}-${month}-${year}`;
|
|
65
|
+
// Preserve time if present.
|
|
66
|
+
if (timePart) {
|
|
67
|
+
return `${formattedDate} ${timePart}`;
|
|
68
|
+
}
|
|
69
|
+
return formattedDate;
|
|
70
|
+
}
|
|
71
|
+
// Date-only mode expects ISO `YYYY-MM-DD`.
|
|
72
|
+
const parts = value.split('-');
|
|
73
|
+
if (parts.length !== 3)
|
|
74
|
+
return value;
|
|
75
|
+
const [year, month, day] = parts;
|
|
76
|
+
return `${day}-${month}-${year}`;
|
|
77
|
+
}
|
|
78
|
+
/**
|
|
79
|
+
* Builds a 6x7 calendar grid for the provided month/year.
|
|
80
|
+
* Includes trailing days from previous/next months to fill the grid.
|
|
81
|
+
* @param currentYear Target year.
|
|
82
|
+
* @param currentMonth Target month (0-based).
|
|
83
|
+
* @param selected Selected date parts or `null`.
|
|
84
|
+
* @param today Reference date for the "today" flag.
|
|
85
|
+
* @returns Calendar day entries for rendering.
|
|
86
|
+
*/
|
|
87
|
+
export function buildCalendarDays(currentYear, currentMonth, selected, today) {
|
|
88
|
+
const calendarDays = [];
|
|
89
|
+
const firstDay = new Date(currentYear, currentMonth, 1);
|
|
90
|
+
const lastDay = new Date(currentYear, currentMonth + 1, 0);
|
|
91
|
+
const todayYear = today.getFullYear();
|
|
92
|
+
const todayMonth = today.getMonth();
|
|
93
|
+
const todayDay = today.getDate();
|
|
94
|
+
// Get the day of week for the first day (0 = Sunday, we want Monday = 0).
|
|
95
|
+
let startDay = firstDay.getDay() - 1;
|
|
96
|
+
if (startDay < 0)
|
|
97
|
+
startDay = 6;
|
|
98
|
+
// Helper to flag if a given day is selected.
|
|
99
|
+
const isSelected = (day, month, year) => !!selected &&
|
|
100
|
+
selected.day === day &&
|
|
101
|
+
selected.month === month &&
|
|
102
|
+
selected.year === year;
|
|
103
|
+
// Previous month days
|
|
104
|
+
const prevMonthLastDay = new Date(currentYear, currentMonth, 0).getDate();
|
|
105
|
+
const prevMonth = currentMonth === 0 ? 11 : currentMonth - 1;
|
|
106
|
+
const prevYear = currentMonth === 0 ? currentYear - 1 : currentYear;
|
|
107
|
+
for (let i = startDay - 1; i >= 0; i--) {
|
|
108
|
+
const day = prevMonthLastDay - i;
|
|
109
|
+
calendarDays.push({
|
|
110
|
+
date: day,
|
|
111
|
+
month: prevMonth,
|
|
112
|
+
year: prevYear,
|
|
113
|
+
isCurrentMonth: false,
|
|
114
|
+
isToday: false,
|
|
115
|
+
isSelected: isSelected(day, prevMonth, prevYear),
|
|
116
|
+
});
|
|
117
|
+
}
|
|
118
|
+
// Current month days
|
|
119
|
+
for (let day = 1; day <= lastDay.getDate(); day++) {
|
|
120
|
+
const isToday = day === todayDay &&
|
|
121
|
+
currentMonth === todayMonth &&
|
|
122
|
+
currentYear === todayYear;
|
|
123
|
+
calendarDays.push({
|
|
124
|
+
date: day,
|
|
125
|
+
month: currentMonth,
|
|
126
|
+
year: currentYear,
|
|
127
|
+
isCurrentMonth: true,
|
|
128
|
+
isToday,
|
|
129
|
+
isSelected: isSelected(day, currentMonth, currentYear),
|
|
130
|
+
});
|
|
131
|
+
}
|
|
132
|
+
// Next month days to fill the grid (6 rows x 7 days = 42).
|
|
133
|
+
const remainingDays = 42 - calendarDays.length;
|
|
134
|
+
const nextMonth = currentMonth === 11 ? 0 : currentMonth + 1;
|
|
135
|
+
const nextYear = currentMonth === 11 ? currentYear + 1 : currentYear;
|
|
136
|
+
for (let day = 1; day <= remainingDays; day++) {
|
|
137
|
+
calendarDays.push({
|
|
138
|
+
date: day,
|
|
139
|
+
month: nextMonth,
|
|
140
|
+
year: nextYear,
|
|
141
|
+
isCurrentMonth: false,
|
|
142
|
+
isToday: false,
|
|
143
|
+
isSelected: isSelected(day, nextMonth, nextYear),
|
|
144
|
+
});
|
|
145
|
+
}
|
|
146
|
+
return calendarDays;
|
|
147
|
+
}
|
|
148
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"tae-date-time-picker.utils.js","sourceRoot":"","sources":["../../../../../../../libs/tegel-angular-extensions/src/lib/components/tae-date-time-picker/tae-date-time-picker.utils.ts"],"names":[],"mappings":"AAiBA;;;;GAIG;AACH,MAAM,UAAU,qBAAqB,CAAC,QAAgB;IACpD,IAAI,CAAC,QAAQ;QAAE,OAAO,IAAI,CAAC;IAC3B,0DAA0D;IAC1D,MAAM,KAAK,GAAG,QAAQ,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC;IAClC,IAAI,KAAK,CAAC,MAAM,KAAK,CAAC;QAAE,OAAO,IAAI,CAAC;IAEpC,qDAAqD;IACrD,MAAM,IAAI,GAAG,QAAQ,CAAC,KAAK,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC;IACpC,MAAM,KAAK,GAAG,QAAQ,CAAC,KAAK,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,GAAG,CAAC,CAAC;IACzC,MAAM,GAAG,GAAG,QAAQ,CAAC,KAAK,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC;IAEnC,8BAA8B;IAC9B,IAAI,CAAC,gBAAgB,CAAC,IAAI,EAAE,KAAK,EAAE,GAAG,CAAC;QAAE,OAAO,IAAI,CAAC;IAErD,OAAO,EAAE,IAAI,EAAE,KAAK,EAAE,GAAG,EAAE,CAAC;AAC9B,CAAC;AAED;;;;;;GAMG;AACH,MAAM,UAAU,gBAAgB,CAC9B,IAAY,EACZ,KAAa,EACb,GAAW;IAEX,IACE,CAAC,MAAM,CAAC,QAAQ,CAAC,IAAI,CAAC;QACtB,CAAC,MAAM,CAAC,QAAQ,CAAC,KAAK,CAAC;QACvB,CAAC,MAAM,CAAC,QAAQ,CAAC,GAAG,CAAC,EACrB,CAAC;QACD,OAAO,KAAK,CAAC;IACf,CAAC;IACD,oCAAoC;IACpC,IAAI,KAAK,GAAG,CAAC,IAAI,KAAK,GAAG,EAAE,IAAI,GAAG,GAAG,CAAC;QAAE,OAAO,KAAK,CAAC;IACrD,kDAAkD;IAClD,MAAM,OAAO,GAAG,IAAI,IAAI,CAAC,IAAI,EAAE,KAAK,GAAG,CAAC,EAAE,CAAC,CAAC,CAAC,OAAO,EAAE,CAAC;IACvD,OAAO,GAAG,IAAI,OAAO,CAAC;AACxB,CAAC;AAED;;;;;GAKG;AACH,MAAM,UAAU,qBAAqB,CACnC,KAAa,EACb,IAAkC;IAElC,IAAI,CAAC,KAAK,EAAE,CAAC;QACX,OAAO,EAAE,CAAC;IACZ,CAAC;IAED,IAAI,IAAI,KAAK,MAAM,EAAE,CAAC;QACpB,OAAO,KAAK,CAAC;IACf,CAAC;IAED,IAAI,IAAI,KAAK,UAAU,EAAE,CAAC;QACxB,MAAM,CAAC,QAAQ,EAAE,QAAQ,CAAC,GAAG,KAAK,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC;QAC9C,IAAI,CAAC,QAAQ;YAAE,OAAO,KAAK,CAAC;QAE5B,gDAAgD;QAChD,MAAM,SAAS,GAAG,QAAQ,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC;QACtC,IAAI,SAAS,CAAC,MAAM,KAAK,CAAC;YAAE,OAAO,KAAK,CAAC;QAEzC,MAAM,CAAC,IAAI,EAAE,KAAK,EAAE,GAAG,CAAC,GAAG,SAAS,CAAC;QACrC,MAAM,aAAa,GAAG,GAAG,GAAG,IAAI,KAAK,IAAI,IAAI,EAAE,CAAC;QAEhD,4BAA4B;QAC5B,IAAI,QAAQ,EAAE,CAAC;YACb,OAAO,GAAG,aAAa,IAAI,QAAQ,EAAE,CAAC;QACxC,CAAC;QACD,OAAO,aAAa,CAAC;IACvB,CAAC;IAED,2CAA2C;IAC3C,MAAM,KAAK,GAAG,KAAK,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC;IAC/B,IAAI,KAAK,CAAC,MAAM,KAAK,CAAC;QAAE,OAAO,KAAK,CAAC;IAErC,MAAM,CAAC,IAAI,EAAE,KAAK,EAAE,GAAG,CAAC,GAAG,KAAK,CAAC;IACjC,OAAO,GAAG,GAAG,IAAI,KAAK,IAAI,IAAI,EAAE,CAAC;AACnC,CAAC;AAED;;;;;;;;GAQG;AACH,MAAM,UAAU,iBAAiB,CAC/B,WAAmB,EACnB,YAAoB,EACpB,QAA0B,EAC1B,KAAW;IAEX,MAAM,YAAY,GAAkB,EAAE,CAAC;IAEvC,MAAM,QAAQ,GAAG,IAAI,IAAI,CAAC,WAAW,EAAE,YAAY,EAAE,CAAC,CAAC,CAAC;IACxD,MAAM,OAAO,GAAG,IAAI,IAAI,CAAC,WAAW,EAAE,YAAY,GAAG,CAAC,EAAE,CAAC,CAAC,CAAC;IAC3D,MAAM,SAAS,GAAG,KAAK,CAAC,WAAW,EAAE,CAAC;IACtC,MAAM,UAAU,GAAG,KAAK,CAAC,QAAQ,EAAE,CAAC;IACpC,MAAM,QAAQ,GAAG,KAAK,CAAC,OAAO,EAAE,CAAC;IAEjC,0EAA0E;IAC1E,IAAI,QAAQ,GAAG,QAAQ,CAAC,MAAM,EAAE,GAAG,CAAC,CAAC;IACrC,IAAI,QAAQ,GAAG,CAAC;QAAE,QAAQ,GAAG,CAAC,CAAC;IAE/B,6CAA6C;IAC7C,MAAM,UAAU,GAAG,CAAC,GAAW,EAAE,KAAa,EAAE,IAAY,EAAW,EAAE,CACvE,CAAC,CAAC,QAAQ;QACV,QAAQ,CAAC,GAAG,KAAK,GAAG;QACpB,QAAQ,CAAC,KAAK,KAAK,KAAK;QACxB,QAAQ,CAAC,IAAI,KAAK,IAAI,CAAC;IAEzB,sBAAsB;IACtB,MAAM,gBAAgB,GAAG,IAAI,IAAI,CAAC,WAAW,EAAE,YAAY,EAAE,CAAC,CAAC,CAAC,OAAO,EAAE,CAAC;IAC1E,MAAM,SAAS,GAAG,YAAY,KAAK,CAAC,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,YAAY,GAAG,CAAC,CAAC;IAC7D,MAAM,QAAQ,GAAG,YAAY,KAAK,CAAC,CAAC,CAAC,CAAC,WAAW,GAAG,CAAC,CAAC,CAAC,CAAC,WAAW,CAAC;IACpE,KAAK,IAAI,CAAC,GAAG,QAAQ,GAAG,CAAC,EAAE,CAAC,IAAI,CAAC,EAAE,CAAC,EAAE,EAAE,CAAC;QACvC,MAAM,GAAG,GAAG,gBAAgB,GAAG,CAAC,CAAC;QACjC,YAAY,CAAC,IAAI,CAAC;YAChB,IAAI,EAAE,GAAG;YACT,KAAK,EAAE,SAAS;YAChB,IAAI,EAAE,QAAQ;YACd,cAAc,EAAE,KAAK;YACrB,OAAO,EAAE,KAAK;YACd,UAAU,EAAE,UAAU,CAAC,GAAG,EAAE,SAAS,EAAE,QAAQ,CAAC;SACjD,CAAC,CAAC;IACL,CAAC;IAED,qBAAqB;IACrB,KAAK,IAAI,GAAG,GAAG,CAAC,EAAE,GAAG,IAAI,OAAO,CAAC,OAAO,EAAE,EAAE,GAAG,EAAE,EAAE,CAAC;QAClD,MAAM,OAAO,GACX,GAAG,KAAK,QAAQ;YAChB,YAAY,KAAK,UAAU;YAC3B,WAAW,KAAK,SAAS,CAAC;QAC5B,YAAY,CAAC,IAAI,CAAC;YAChB,IAAI,EAAE,GAAG;YACT,KAAK,EAAE,YAAY;YACnB,IAAI,EAAE,WAAW;YACjB,cAAc,EAAE,IAAI;YACpB,OAAO;YACP,UAAU,EAAE,UAAU,CAAC,GAAG,EAAE,YAAY,EAAE,WAAW,CAAC;SACvD,CAAC,CAAC;IACL,CAAC;IAED,2DAA2D;IAC3D,MAAM,aAAa,GAAG,EAAE,GAAG,YAAY,CAAC,MAAM,CAAC;IAC/C,MAAM,SAAS,GAAG,YAAY,KAAK,EAAE,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,YAAY,GAAG,CAAC,CAAC;IAC7D,MAAM,QAAQ,GAAG,YAAY,KAAK,EAAE,CAAC,CAAC,CAAC,WAAW,GAAG,CAAC,CAAC,CAAC,CAAC,WAAW,CAAC;IACrE,KAAK,IAAI,GAAG,GAAG,CAAC,EAAE,GAAG,IAAI,aAAa,EAAE,GAAG,EAAE,EAAE,CAAC;QAC9C,YAAY,CAAC,IAAI,CAAC;YAChB,IAAI,EAAE,GAAG;YACT,KAAK,EAAE,SAAS;YAChB,IAAI,EAAE,QAAQ;YACd,cAAc,EAAE,KAAK;YACrB,OAAO,EAAE,KAAK;YACd,UAAU,EAAE,UAAU,CAAC,GAAG,EAAE,SAAS,EAAE,QAAQ,CAAC;SACjD,CAAC,CAAC;IACL,CAAC;IAED,OAAO,YAAY,CAAC;AACtB,CAAC","sourcesContent":["/** Represents a normalized date in local time (0-based month). */\nexport type DateParts = {\n  year: number;\n  month: number;\n  day: number;\n};\n\n/** A single day entry used by the calendar grid renderer. */\nexport type CalendarDay = {\n  date: number;\n  month: number;\n  year: number;\n  isCurrentMonth: boolean;\n  isToday: boolean;\n  isSelected: boolean;\n};\n\n/**\n * Parses an ISO date segment (`YYYY-MM-DD`) into local date parts.\n * @param datePart ISO date segment to parse.\n * @returns Parsed date parts or `null` when invalid.\n */\nexport function parseDatePartsFromIso(datePart: string): DateParts | null {\n  if (!datePart) return null;\n  // Split on the ISO separator and validate expected parts.\n  const parts = datePart.split('-');\n  if (parts.length !== 3) return null;\n\n  // Convert to numbers and normalize month to 0-based.\n  const year = parseInt(parts[0], 10);\n  const month = parseInt(parts[1], 10) - 1;\n  const day = parseInt(parts[2], 10);\n\n  // Reject invalid dates early.\n  if (!isValidDateParts(year, month, day)) return null;\n\n  return { year, month, day };\n}\n\n/**\n * Validates date parts (0-based month) against calendar limits.\n * @param year Full year value.\n * @param month Zero-based month.\n * @param day Day of month (1-based).\n * @returns `true` when the date exists in the calendar.\n */\nexport function isValidDateParts(\n  year: number,\n  month: number,\n  day: number,\n): boolean {\n  if (\n    !Number.isFinite(year) ||\n    !Number.isFinite(month) ||\n    !Number.isFinite(day)\n  ) {\n    return false;\n  }\n  // Basic range checks for month/day.\n  if (month < 0 || month > 11 || day < 1) return false;\n  // Ensure the day exists in the target month/year.\n  const lastDay = new Date(year, month + 1, 0).getDate();\n  return day <= lastDay;\n}\n\n/**\n * Formats an ISO date/datetime value into `dd-MM-YYYY` display format.\n * @param value ISO date or datetime string.\n * @param mode Formatting mode (`date` or `datetime`).\n * @returns Display-friendly string.\n */\nexport function formatDisplayDateTime(\n  value: string,\n  mode: 'date' | 'datetime' | 'time',\n): string {\n  if (!value) {\n    return '';\n  }\n\n  if (mode === 'time') {\n    return value;\n  }\n\n  if (mode === 'datetime') {\n    const [datePart, timePart] = value.split('T');\n    if (!datePart) return value;\n\n    // Expect ISO `YYYY-MM-DD` for the date portion.\n    const dateParts = datePart.split('-');\n    if (dateParts.length !== 3) return value;\n\n    const [year, month, day] = dateParts;\n    const formattedDate = `${day}-${month}-${year}`;\n\n    // Preserve time if present.\n    if (timePart) {\n      return `${formattedDate} ${timePart}`;\n    }\n    return formattedDate;\n  }\n\n  // Date-only mode expects ISO `YYYY-MM-DD`.\n  const parts = value.split('-');\n  if (parts.length !== 3) return value;\n\n  const [year, month, day] = parts;\n  return `${day}-${month}-${year}`;\n}\n\n/**\n * Builds a 6x7 calendar grid for the provided month/year.\n * Includes trailing days from previous/next months to fill the grid.\n * @param currentYear Target year.\n * @param currentMonth Target month (0-based).\n * @param selected Selected date parts or `null`.\n * @param today Reference date for the \"today\" flag.\n * @returns Calendar day entries for rendering.\n */\nexport function buildCalendarDays(\n  currentYear: number,\n  currentMonth: number,\n  selected: DateParts | null,\n  today: Date,\n): CalendarDay[] {\n  const calendarDays: CalendarDay[] = [];\n\n  const firstDay = new Date(currentYear, currentMonth, 1);\n  const lastDay = new Date(currentYear, currentMonth + 1, 0);\n  const todayYear = today.getFullYear();\n  const todayMonth = today.getMonth();\n  const todayDay = today.getDate();\n\n  // Get the day of week for the first day (0 = Sunday, we want Monday = 0).\n  let startDay = firstDay.getDay() - 1;\n  if (startDay < 0) startDay = 6;\n\n  // Helper to flag if a given day is selected.\n  const isSelected = (day: number, month: number, year: number): boolean =>\n    !!selected &&\n    selected.day === day &&\n    selected.month === month &&\n    selected.year === year;\n\n  // Previous month days\n  const prevMonthLastDay = new Date(currentYear, currentMonth, 0).getDate();\n  const prevMonth = currentMonth === 0 ? 11 : currentMonth - 1;\n  const prevYear = currentMonth === 0 ? currentYear - 1 : currentYear;\n  for (let i = startDay - 1; i >= 0; i--) {\n    const day = prevMonthLastDay - i;\n    calendarDays.push({\n      date: day,\n      month: prevMonth,\n      year: prevYear,\n      isCurrentMonth: false,\n      isToday: false,\n      isSelected: isSelected(day, prevMonth, prevYear),\n    });\n  }\n\n  // Current month days\n  for (let day = 1; day <= lastDay.getDate(); day++) {\n    const isToday =\n      day === todayDay &&\n      currentMonth === todayMonth &&\n      currentYear === todayYear;\n    calendarDays.push({\n      date: day,\n      month: currentMonth,\n      year: currentYear,\n      isCurrentMonth: true,\n      isToday,\n      isSelected: isSelected(day, currentMonth, currentYear),\n    });\n  }\n\n  // Next month days to fill the grid (6 rows x 7 days = 42).\n  const remainingDays = 42 - calendarDays.length;\n  const nextMonth = currentMonth === 11 ? 0 : currentMonth + 1;\n  const nextYear = currentMonth === 11 ? currentYear + 1 : currentYear;\n  for (let day = 1; day <= remainingDays; day++) {\n    calendarDays.push({\n      date: day,\n      month: nextMonth,\n      year: nextYear,\n      isCurrentMonth: false,\n      isToday: false,\n      isSelected: isSelected(day, nextMonth, nextYear),\n    });\n  }\n\n  return calendarDays;\n}\n"]}
|
|
@@ -0,0 +1,38 @@
|
|
|
1
|
+
/* eslint-disable @angular-eslint/directive-selector */
|
|
2
|
+
import { Directive, input } from '@angular/core';
|
|
3
|
+
import * as i0 from "@angular/core";
|
|
4
|
+
/**
|
|
5
|
+
* Directive that provides typed context for an `ng-template`.
|
|
6
|
+
*
|
|
7
|
+
* This is a compile-time helper: it does not affect runtime behavior,
|
|
8
|
+
* but it allows Angular's template type checker to infer the template context.
|
|
9
|
+
*
|
|
10
|
+
* Usage:
|
|
11
|
+
* ```html
|
|
12
|
+
* <ng-template #tpl let-data [typedContext]="someContext">...</ng-template>
|
|
13
|
+
* ```
|
|
14
|
+
*/
|
|
15
|
+
export class TypedTemplateDirective {
|
|
16
|
+
/**
|
|
17
|
+
* Typed context value used only for type inference.
|
|
18
|
+
* This input is not read at runtime.
|
|
19
|
+
*/
|
|
20
|
+
typedContext = input.required();
|
|
21
|
+
/**
|
|
22
|
+
* Template context guard to inform Angular's template type system.
|
|
23
|
+
* Always returns `true` at runtime; used purely for typing.
|
|
24
|
+
*/
|
|
25
|
+
static ngTemplateContextGuard(dir, ctx) {
|
|
26
|
+
return true;
|
|
27
|
+
}
|
|
28
|
+
static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "19.2.17", ngImport: i0, type: TypedTemplateDirective, deps: [], target: i0.ɵɵFactoryTarget.Directive });
|
|
29
|
+
static ɵdir = i0.ɵɵngDeclareDirective({ minVersion: "17.1.0", version: "19.2.17", type: TypedTemplateDirective, isStandalone: true, selector: "ng-template[typedContext]", inputs: { typedContext: { classPropertyName: "typedContext", publicName: "typedContext", isSignal: true, isRequired: true, transformFunction: null } }, ngImport: i0 });
|
|
30
|
+
}
|
|
31
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "19.2.17", ngImport: i0, type: TypedTemplateDirective, decorators: [{
|
|
32
|
+
type: Directive,
|
|
33
|
+
args: [{
|
|
34
|
+
selector: 'ng-template[typedContext]',
|
|
35
|
+
standalone: true,
|
|
36
|
+
}]
|
|
37
|
+
}] });
|
|
38
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidHlwZWQtdGVtcGxhdGUuZGlyZWN0aXZlLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vLi4vLi4vbGlicy90ZWdlbC1hbmd1bGFyLWV4dGVuc2lvbnMvc3JjL2xpYi9kaXJlY3RpdmVzL3R5cGVkLXRlbXBsYXRlLmRpcmVjdGl2ZS50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSx1REFBdUQ7QUFDdkQsT0FBTyxFQUFFLFNBQVMsRUFBZSxLQUFLLEVBQUUsTUFBTSxlQUFlLENBQUM7O0FBRTlEOzs7Ozs7Ozs7O0dBVUc7QUFLSCxNQUFNLE9BQU8sc0JBQXNCO0lBQ2pDOzs7T0FHRztJQUNNLFlBQVksR0FBRyxLQUFLLENBQUMsUUFBUSxFQUFrQixDQUFDO0lBRXpEOzs7T0FHRztJQUNILE1BQU0sQ0FBQyxzQkFBc0IsQ0FDM0IsR0FBOEIsRUFDOUIsR0FBWTtRQUVaLE9BQU8sSUFBSSxDQUFDO0lBQ2QsQ0FBQzt3R0FoQlUsc0JBQXNCOzRGQUF0QixzQkFBc0I7OzRGQUF0QixzQkFBc0I7a0JBSmxDLFNBQVM7bUJBQUM7b0JBQ1QsUUFBUSxFQUFFLDJCQUEyQjtvQkFDckMsVUFBVSxFQUFFLElBQUk7aUJBQ2pCIiwic291cmNlc0NvbnRlbnQiOlsiLyogZXNsaW50LWRpc2FibGUgQGFuZ3VsYXItZXNsaW50L2RpcmVjdGl2ZS1zZWxlY3RvciAqL1xuaW1wb3J0IHsgRGlyZWN0aXZlLCBUZW1wbGF0ZVJlZiwgaW5wdXQgfSBmcm9tICdAYW5ndWxhci9jb3JlJztcblxuLyoqXG4gKiBEaXJlY3RpdmUgdGhhdCBwcm92aWRlcyB0eXBlZCBjb250ZXh0IGZvciBhbiBgbmctdGVtcGxhdGVgLlxuICpcbiAqIFRoaXMgaXMgYSBjb21waWxlLXRpbWUgaGVscGVyOiBpdCBkb2VzIG5vdCBhZmZlY3QgcnVudGltZSBiZWhhdmlvcixcbiAqIGJ1dCBpdCBhbGxvd3MgQW5ndWxhcidzIHRlbXBsYXRlIHR5cGUgY2hlY2tlciB0byBpbmZlciB0aGUgdGVtcGxhdGUgY29udGV4dC5cbiAqXG4gKiBVc2FnZTpcbiAqIGBgYGh0bWxcbiAqIDxuZy10ZW1wbGF0ZSAjdHBsIGxldC1kYXRhIFt0eXBlZENvbnRleHRdPVwic29tZUNvbnRleHRcIj4uLi48L25nLXRlbXBsYXRlPlxuICogYGBgXG4gKi9cbkBEaXJlY3RpdmUoe1xuICBzZWxlY3RvcjogJ25nLXRlbXBsYXRlW3R5cGVkQ29udGV4dF0nLFxuICBzdGFuZGFsb25lOiB0cnVlLFxufSlcbmV4cG9ydCBjbGFzcyBUeXBlZFRlbXBsYXRlRGlyZWN0aXZlPFQ+IHtcbiAgLyoqXG4gICAqIFR5cGVkIGNvbnRleHQgdmFsdWUgdXNlZCBvbmx5IGZvciB0eXBlIGluZmVyZW5jZS5cbiAgICogVGhpcyBpbnB1dCBpcyBub3QgcmVhZCBhdCBydW50aW1lLlxuICAgKi9cbiAgcmVhZG9ubHkgdHlwZWRDb250ZXh0ID0gaW5wdXQucmVxdWlyZWQ8VGVtcGxhdGVSZWY8VD4+KCk7XG5cbiAgLyoqXG4gICAqIFRlbXBsYXRlIGNvbnRleHQgZ3VhcmQgdG8gaW5mb3JtIEFuZ3VsYXIncyB0ZW1wbGF0ZSB0eXBlIHN5c3RlbS5cbiAgICogQWx3YXlzIHJldHVybnMgYHRydWVgIGF0IHJ1bnRpbWU7IHVzZWQgcHVyZWx5IGZvciB0eXBpbmcuXG4gICAqL1xuICBzdGF0aWMgbmdUZW1wbGF0ZUNvbnRleHRHdWFyZDxUPihcbiAgICBkaXI6IFR5cGVkVGVtcGxhdGVEaXJlY3RpdmU8VD4sXG4gICAgY3R4OiB1bmtub3duLFxuICApOiBjdHggaXMgeyAkaW1wbGljaXQ6IFQgfSB7XG4gICAgcmV0dXJuIHRydWU7XG4gIH1cbn1cbiJdfQ==
|
|
@@ -53,6 +53,13 @@ export class ModalComponent {
|
|
|
53
53
|
isComponentBody(body) {
|
|
54
54
|
return typeof body === 'object' && body !== null && 'component' in body;
|
|
55
55
|
}
|
|
56
|
+
/**
|
|
57
|
+
* Type guard for template-backed modal bodies.
|
|
58
|
+
* @param body Body value to test.
|
|
59
|
+
*/
|
|
60
|
+
isTemplateBody(body) {
|
|
61
|
+
return typeof body === 'object' && body !== null && 'template' in body;
|
|
62
|
+
}
|
|
56
63
|
/**
|
|
57
64
|
* Invokes an action button handler (sync or async).
|
|
58
65
|
* @param button Modal button descriptor.
|
|
@@ -112,10 +119,10 @@ export class ModalComponent {
|
|
|
112
119
|
}
|
|
113
120
|
}
|
|
114
121
|
static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "19.2.17", ngImport: i0, type: ModalComponent, deps: [], target: i0.ɵɵFactoryTarget.Component });
|
|
115
|
-
static ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "19.2.17", type: ModalComponent, isStandalone: true, selector: "tae-modal", inputs: { modal: { classPropertyName: "modal", publicName: "modal", isSignal: true, isRequired: true, transformFunction: null } }, outputs: { closed: "closed" }, viewQueries: [{ propertyName: "_modalRef", first: true, predicate: TdsModal, descendants: true, read: ElementRef, isSignal: true }], ngImport: i0, template: "@let modalObj = modal();\n\n<tds-modal\n [id]=\"modalObj.id\"\n [selector]=\"modalObj.selector\"\n [referenceEl]=\"modalObj.referenceEl\"\n [size]=\"modalObj.size\"\n [actionsPosition]=\"modalObj.actionsPosition\"\n [prevent]=\"modalObj.prevent\"\n [closable]=\"modalObj.closable\"\n [tdsAlertDialog]=\"modalObj.alertDialog\"\n (tdsClose)=\"this.closed.emit(modalObj.id)\"\n>\n @let header = modalObj.header;\n @if (header) {\n @if (isTemplateRef(header)) {\n <span slot=\"header\">\n <ng-container [ngTemplateOutlet]=\"header\" />\n </span>\n } @else {\n <span slot=\"header\">{{ header }}</span>\n }\n }\n\n @if (!modalObj.lazy || modalObj.isOpen) {\n @let body = modalObj.body;\n @if (body) {\n <span slot=\"body\">\n @if (isComponentBody(body)) {\n <ng-container\n taeDynamicComponent\n [component]=\"body.component\"\n [inputs]=\"body.inputs\"\n [outputs]=\"body.outputs\"\n />\n } @else if (
|
|
122
|
+
static ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "19.2.17", type: ModalComponent, isStandalone: true, selector: "tae-modal", inputs: { modal: { classPropertyName: "modal", publicName: "modal", isSignal: true, isRequired: true, transformFunction: null } }, outputs: { closed: "closed" }, viewQueries: [{ propertyName: "_modalRef", first: true, predicate: TdsModal, descendants: true, read: ElementRef, isSignal: true }], ngImport: i0, template: "@let modalObj = modal();\n\n<tds-modal\n [id]=\"modalObj.id\"\n [selector]=\"modalObj.selector\"\n [referenceEl]=\"modalObj.referenceEl\"\n [size]=\"modalObj.size\"\n [actionsPosition]=\"modalObj.actionsPosition\"\n [prevent]=\"modalObj.prevent\"\n [closable]=\"modalObj.closable\"\n [tdsAlertDialog]=\"modalObj.alertDialog\"\n (tdsClose)=\"this.closed.emit(modalObj.id)\"\n>\n @let header = modalObj.header;\n @if (header) {\n @if (isTemplateRef(header)) {\n <span slot=\"header\">\n <ng-container [ngTemplateOutlet]=\"header\" />\n </span>\n } @else {\n <span slot=\"header\">{{ header }}</span>\n }\n }\n\n @if (!modalObj.lazy || modalObj.isOpen) {\n @let body = modalObj.body;\n @if (body) {\n <span slot=\"body\">\n @if (isComponentBody(body)) {\n <ng-container\n taeDynamicComponent\n [component]=\"body.component\"\n [inputs]=\"body.inputs\"\n [outputs]=\"body.outputs\"\n />\n } @else if (isTemplateBody(body)) {\n <ng-container\n [ngTemplateOutlet]=\"body.template\"\n [ngTemplateOutletContext]=\"{ $implicit: body.context }\"\n />\n } @else {\n {{ body }}\n }\n </span>\n }\n }\n\n @let buttons = modalObj.buttons;\n @if (buttons?.length) {\n <span slot=\"actions\" class=\"tds-u-flex tds-u-gap2\">\n @for (button of buttons; track $index) {\n <tds-button\n [text]=\"button.text\"\n [variant]=\"button.variant ?? 'primary'\"\n [size]=\"button.size ?? 'md'\"\n [disabled]=\"button.disabled ?? false\"\n [attr.data-dismiss-modal]=\"button.dismiss ? '' : null\"\n (click)=\"handleButtonClick(button)\"\n ></tds-button>\n }\n </span>\n }\n</tds-modal>\n", styles: [":host{display:contents}\n"], dependencies: [{ kind: "ngmodule", type: CommonModule }, { kind: "directive", type: i1.NgTemplateOutlet, selector: "[ngTemplateOutlet]", inputs: ["ngTemplateOutletContext", "ngTemplateOutlet", "ngTemplateOutletInjector"] }, { kind: "ngmodule", type: TegelModule }, { kind: "component", type: i2.TdsButton, selector: "tds-button", inputs: ["animation", "disabled", "fullbleed", "modeVariant", "name", "size", "tdsAriaLabel", "text", "type", "value", "variant"] }, { kind: "component", type: i2.TdsModal, selector: "tds-modal", inputs: ["actionsPosition", "closable", "header", "prevent", "referenceEl", "selector", "show", "size", "tdsAlertDialog"] }, { kind: "directive", type: DynamicComponentDirective, selector: "[taeDynamicComponent]", inputs: ["component", "inputs", "outputs"] }], changeDetection: i0.ChangeDetectionStrategy.OnPush });
|
|
116
123
|
}
|
|
117
124
|
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "19.2.17", ngImport: i0, type: ModalComponent, decorators: [{
|
|
118
125
|
type: Component,
|
|
119
|
-
args: [{ selector: 'tae-modal', imports: [CommonModule, TegelModule, DynamicComponentDirective], changeDetection: ChangeDetectionStrategy.OnPush, template: "@let modalObj = modal();\n\n<tds-modal\n [id]=\"modalObj.id\"\n [selector]=\"modalObj.selector\"\n [referenceEl]=\"modalObj.referenceEl\"\n [size]=\"modalObj.size\"\n [actionsPosition]=\"modalObj.actionsPosition\"\n [prevent]=\"modalObj.prevent\"\n [closable]=\"modalObj.closable\"\n [tdsAlertDialog]=\"modalObj.alertDialog\"\n (tdsClose)=\"this.closed.emit(modalObj.id)\"\n>\n @let header = modalObj.header;\n @if (header) {\n @if (isTemplateRef(header)) {\n <span slot=\"header\">\n <ng-container [ngTemplateOutlet]=\"header\" />\n </span>\n } @else {\n <span slot=\"header\">{{ header }}</span>\n }\n }\n\n @if (!modalObj.lazy || modalObj.isOpen) {\n @let body = modalObj.body;\n @if (body) {\n <span slot=\"body\">\n @if (isComponentBody(body)) {\n <ng-container\n taeDynamicComponent\n [component]=\"body.component\"\n [inputs]=\"body.inputs\"\n [outputs]=\"body.outputs\"\n />\n } @else if (
|
|
126
|
+
args: [{ selector: 'tae-modal', imports: [CommonModule, TegelModule, DynamicComponentDirective], changeDetection: ChangeDetectionStrategy.OnPush, template: "@let modalObj = modal();\n\n<tds-modal\n [id]=\"modalObj.id\"\n [selector]=\"modalObj.selector\"\n [referenceEl]=\"modalObj.referenceEl\"\n [size]=\"modalObj.size\"\n [actionsPosition]=\"modalObj.actionsPosition\"\n [prevent]=\"modalObj.prevent\"\n [closable]=\"modalObj.closable\"\n [tdsAlertDialog]=\"modalObj.alertDialog\"\n (tdsClose)=\"this.closed.emit(modalObj.id)\"\n>\n @let header = modalObj.header;\n @if (header) {\n @if (isTemplateRef(header)) {\n <span slot=\"header\">\n <ng-container [ngTemplateOutlet]=\"header\" />\n </span>\n } @else {\n <span slot=\"header\">{{ header }}</span>\n }\n }\n\n @if (!modalObj.lazy || modalObj.isOpen) {\n @let body = modalObj.body;\n @if (body) {\n <span slot=\"body\">\n @if (isComponentBody(body)) {\n <ng-container\n taeDynamicComponent\n [component]=\"body.component\"\n [inputs]=\"body.inputs\"\n [outputs]=\"body.outputs\"\n />\n } @else if (isTemplateBody(body)) {\n <ng-container\n [ngTemplateOutlet]=\"body.template\"\n [ngTemplateOutletContext]=\"{ $implicit: body.context }\"\n />\n } @else {\n {{ body }}\n }\n </span>\n }\n }\n\n @let buttons = modalObj.buttons;\n @if (buttons?.length) {\n <span slot=\"actions\" class=\"tds-u-flex tds-u-gap2\">\n @for (button of buttons; track $index) {\n <tds-button\n [text]=\"button.text\"\n [variant]=\"button.variant ?? 'primary'\"\n [size]=\"button.size ?? 'md'\"\n [disabled]=\"button.disabled ?? false\"\n [attr.data-dismiss-modal]=\"button.dismiss ? '' : null\"\n (click)=\"handleButtonClick(button)\"\n ></tds-button>\n }\n </span>\n }\n</tds-modal>\n", styles: [":host{display:contents}\n"] }]
|
|
120
127
|
}], ctorParameters: () => [] });
|
|
121
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"modal.component.js","sourceRoot":"","sources":["../../../../../../../../libs/tegel-angular-extensions/src/lib/modal/components/modal/modal.component.ts","../../../../../../../../libs/tegel-angular-extensions/src/lib/modal/components/modal/modal.component.html"],"names":[],"mappings":"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","sourcesContent":["import { CommonModule } from '@angular/common';\nimport {\n  AfterViewInit,\n  ChangeDetectionStrategy,\n  Component,\n  ElementRef,\n  TemplateRef,\n  computed,\n  effect,\n  input,\n  output,\n  viewChild,\n} from '@angular/core';\nimport { TdsModal, TegelModule } from '@scania/tegel-angular-17';\nimport { DynamicComponentDirective } from '../../directives/dynamic-component.directive';\nimport { Modal, ModalBody, ModalComponentBody } from '../../schema/modal.model';\nimport { ModalButton } from '../../schema/modal.types';\n\n/**\n * Renders a single modal instance backed by the Tegel <tds-modal> web component.\n *\n * Responsibilities:\n * - Initialize the underlying web component once it is ready.\n * - Apply the desired open/close state based on `modal.isOpen`.\n * - Render header/body/action content (including dynamic components).\n */\n@Component({\n  selector: 'tae-modal',\n  templateUrl: './modal.component.html',\n  styleUrls: ['./modal.component.scss'],\n  imports: [CommonModule, TegelModule, DynamicComponentDirective],\n  changeDetection: ChangeDetectionStrategy.OnPush,\n})\nexport class ModalComponent implements AfterViewInit {\n  /**\n   * Modal configuration and runtime state.\n   */\n  readonly modal = input.required<Modal>();\n  /**\n   * Emits the modal id when the modal is closed by the user.\n   */\n  protected readonly closed = output<string>();\n  // Reference to the underlying web component instance.\n  private readonly _modalRef = viewChild(TdsModal, {\n    read: ElementRef<HTMLTdsModalElement>,\n  });\n\n  /**\n   * Resolved DOM element for the underlying <tds-modal>.\n   */\n  protected readonly modalEl = computed(\n    () => this._modalRef()?.nativeElement ?? null,\n  );\n\n  /** Tracks whether the modal has been initialized. */\n  private initialized = false;\n\n  constructor() {\n    effect(() => {\n      // Track input changes and sync desired open state to the web component.\n      this.modal();\n      void this.applyOpenState();\n    });\n  }\n\n  /**\n   * Type guard for TemplateRef values.\n   * @param value Value to test.\n   */\n  protected isTemplateRef(value: unknown): value is TemplateRef<unknown> {\n    return value instanceof TemplateRef;\n  }\n\n  /**\n   * Type guard for component-backed modal bodies.\n   * @param body Body value to test.\n   */\n  protected isComponentBody(\n    body: ModalBody<unknown>,\n  ): body is ModalComponentBody<unknown> {\n    return typeof body === 'object' && body !== null && 'component' in body;\n  }\n\n  /**\n   * Invokes an action button handler (sync or async).\n   * @param button Modal button descriptor.\n   */\n  async handleButtonClick(button: ModalButton): Promise<void> {\n    try {\n      await button.onClick?.();\n    } catch (err) {\n      console.error('[tae-modal] Button onClick failed:', err);\n    }\n  }\n\n  /**\n   * Initializes the underlying <tds-modal> after view init.\n   */\n  ngAfterViewInit(): void {\n    void this.initializeModal();\n  }\n\n  /**\n   * Calls the web component's initializeModal() once.\n   */\n  private async initializeModal(): Promise<void> {\n    const el = this.modalEl();\n    if (!el) {\n      console.warn('[tae-modal] Modal element not found; cannot initialize.');\n      return;\n    }\n    if (this.initialized) return;\n\n    const ready = await el.componentOnReady();\n    if (!ready) {\n      console.warn(\n        '[tae-modal] Modal element not ready; initialization skipped.',\n      );\n      return;\n    }\n\n    await el.initializeModal();\n    this.initialized = true;\n  }\n\n  /**\n   * Applies the desired open/close state to the web component.\n   */\n  private async applyOpenState(): Promise<void> {\n    const modal = this.modal();\n    const el = this.modalEl();\n    if (!el) return;\n    if (!this.initialized) {\n      // Initialize once before issuing show/close calls.\n      await this.initializeModal();\n      if (!this.initialized) return;\n    }\n    if (modal.isOpen) {\n      await el.showModal();\n    } else {\n      await el.closeModal();\n    }\n  }\n}\n","@let modalObj = modal();\n\n<tds-modal\n  [id]=\"modalObj.id\"\n  [selector]=\"modalObj.selector\"\n  [referenceEl]=\"modalObj.referenceEl\"\n  [size]=\"modalObj.size\"\n  [actionsPosition]=\"modalObj.actionsPosition\"\n  [prevent]=\"modalObj.prevent\"\n  [closable]=\"modalObj.closable\"\n  [tdsAlertDialog]=\"modalObj.alertDialog\"\n  (tdsClose)=\"this.closed.emit(modalObj.id)\"\n>\n  @let header = modalObj.header;\n  @if (header) {\n    @if (isTemplateRef(header)) {\n      <span slot=\"header\">\n        <ng-container [ngTemplateOutlet]=\"header\" />\n      </span>\n    } @else {\n      <span slot=\"header\">{{ header }}</span>\n    }\n  }\n\n  @if (!modalObj.lazy || modalObj.isOpen) {\n    @let body = modalObj.body;\n    @if (body) {\n      <span slot=\"body\">\n        @if (isComponentBody(body)) {\n          <ng-container\n            taeDynamicComponent\n            [component]=\"body.component\"\n            [inputs]=\"body.inputs\"\n            [outputs]=\"body.outputs\"\n          />\n        } @else if (isTemplateRef(body)) {\n          <ng-container [ngTemplateOutlet]=\"body\" />\n        } @else {\n          {{ body }}\n        }\n      </span>\n    }\n  }\n\n  @let buttons = modalObj.buttons;\n  @if (buttons?.length) {\n    <span slot=\"actions\" class=\"tds-u-flex tds-u-gap2\">\n      @for (button of buttons; track $index) {\n        <tds-button\n          [text]=\"button.text\"\n          [variant]=\"button.variant ?? 'primary'\"\n          [size]=\"button.size ?? 'md'\"\n          [disabled]=\"button.disabled ?? false\"\n          [attr.data-dismiss-modal]=\"button.dismiss ? '' : null\"\n          (click)=\"handleButtonClick(button)\"\n        ></tds-button>\n      }\n    </span>\n  }\n</tds-modal>\n"]}
|
|
128
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"modal.component.js","sourceRoot":"","sources":["../../../../../../../../libs/tegel-angular-extensions/src/lib/modal/components/modal/modal.component.ts","../../../../../../../../libs/tegel-angular-extensions/src/lib/modal/components/modal/modal.component.html"],"names":[],"mappings":"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","sourcesContent":["import { CommonModule } from '@angular/common';\nimport {\n  AfterViewInit,\n  ChangeDetectionStrategy,\n  Component,\n  ElementRef,\n  TemplateRef,\n  computed,\n  effect,\n  input,\n  output,\n  viewChild,\n} from '@angular/core';\nimport { TdsModal, TegelModule } from '@scania/tegel-angular-17';\nimport { DynamicComponentDirective } from '../../directives/dynamic-component.directive';\nimport {\n  Modal,\n  ModalBody,\n  ModalComponentBody,\n  ModalTemplateBody,\n} from '../../schema/modal.model';\nimport { ModalButton } from '../../schema/modal.types';\n\n/**\n * Renders a single modal instance backed by the Tegel <tds-modal> web component.\n *\n * Responsibilities:\n * - Initialize the underlying web component once it is ready.\n * - Apply the desired open/close state based on `modal.isOpen`.\n * - Render header/body/action content (including dynamic components).\n */\n@Component({\n  selector: 'tae-modal',\n  templateUrl: './modal.component.html',\n  styleUrls: ['./modal.component.scss'],\n  imports: [CommonModule, TegelModule, DynamicComponentDirective],\n  changeDetection: ChangeDetectionStrategy.OnPush,\n})\nexport class ModalComponent implements AfterViewInit {\n  /**\n   * Modal configuration and runtime state.\n   */\n  readonly modal = input.required<Modal>();\n  /**\n   * Emits the modal id when the modal is closed by the user.\n   */\n  protected readonly closed = output<string>();\n  // Reference to the underlying web component instance.\n  private readonly _modalRef = viewChild(TdsModal, {\n    read: ElementRef<HTMLTdsModalElement>,\n  });\n\n  /**\n   * Resolved DOM element for the underlying <tds-modal>.\n   */\n  protected readonly modalEl = computed(\n    () => this._modalRef()?.nativeElement ?? null,\n  );\n\n  /** Tracks whether the modal has been initialized. */\n  private initialized = false;\n\n  constructor() {\n    effect(() => {\n      // Track input changes and sync desired open state to the web component.\n      this.modal();\n      void this.applyOpenState();\n    });\n  }\n\n  /**\n   * Type guard for TemplateRef values.\n   * @param value Value to test.\n   */\n  protected isTemplateRef(value: unknown): value is TemplateRef<unknown> {\n    return value instanceof TemplateRef;\n  }\n\n  /**\n   * Type guard for component-backed modal bodies.\n   * @param body Body value to test.\n   */\n  protected isComponentBody(\n    body: ModalBody<unknown>,\n  ): body is ModalComponentBody<unknown> {\n    return typeof body === 'object' && body !== null && 'component' in body;\n  }\n\n  /**\n   * Type guard for template-backed modal bodies.\n   * @param body Body value to test.\n   */\n  protected isTemplateBody(\n    body: ModalBody<unknown>,\n  ): body is ModalTemplateBody {\n    return typeof body === 'object' && body !== null && 'template' in body;\n  }\n\n  /**\n   * Invokes an action button handler (sync or async).\n   * @param button Modal button descriptor.\n   */\n  async handleButtonClick(button: ModalButton): Promise<void> {\n    try {\n      await button.onClick?.();\n    } catch (err) {\n      console.error('[tae-modal] Button onClick failed:', err);\n    }\n  }\n\n  /**\n   * Initializes the underlying <tds-modal> after view init.\n   */\n  ngAfterViewInit(): void {\n    void this.initializeModal();\n  }\n\n  /**\n   * Calls the web component's initializeModal() once.\n   */\n  private async initializeModal(): Promise<void> {\n    const el = this.modalEl();\n    if (!el) {\n      console.warn('[tae-modal] Modal element not found; cannot initialize.');\n      return;\n    }\n    if (this.initialized) return;\n\n    const ready = await el.componentOnReady();\n    if (!ready) {\n      console.warn(\n        '[tae-modal] Modal element not ready; initialization skipped.',\n      );\n      return;\n    }\n\n    await el.initializeModal();\n    this.initialized = true;\n  }\n\n  /**\n   * Applies the desired open/close state to the web component.\n   */\n  private async applyOpenState(): Promise<void> {\n    const modal = this.modal();\n    const el = this.modalEl();\n    if (!el) return;\n    if (!this.initialized) {\n      // Initialize once before issuing show/close calls.\n      await this.initializeModal();\n      if (!this.initialized) return;\n    }\n    if (modal.isOpen) {\n      await el.showModal();\n    } else {\n      await el.closeModal();\n    }\n  }\n}\n","@let modalObj = modal();\n\n<tds-modal\n  [id]=\"modalObj.id\"\n  [selector]=\"modalObj.selector\"\n  [referenceEl]=\"modalObj.referenceEl\"\n  [size]=\"modalObj.size\"\n  [actionsPosition]=\"modalObj.actionsPosition\"\n  [prevent]=\"modalObj.prevent\"\n  [closable]=\"modalObj.closable\"\n  [tdsAlertDialog]=\"modalObj.alertDialog\"\n  (tdsClose)=\"this.closed.emit(modalObj.id)\"\n>\n  @let header = modalObj.header;\n  @if (header) {\n    @if (isTemplateRef(header)) {\n      <span slot=\"header\">\n        <ng-container [ngTemplateOutlet]=\"header\" />\n      </span>\n    } @else {\n      <span slot=\"header\">{{ header }}</span>\n    }\n  }\n\n  @if (!modalObj.lazy || modalObj.isOpen) {\n    @let body = modalObj.body;\n    @if (body) {\n      <span slot=\"body\">\n        @if (isComponentBody(body)) {\n          <ng-container\n            taeDynamicComponent\n            [component]=\"body.component\"\n            [inputs]=\"body.inputs\"\n            [outputs]=\"body.outputs\"\n          />\n        } @else if (isTemplateBody(body)) {\n          <ng-container\n            [ngTemplateOutlet]=\"body.template\"\n            [ngTemplateOutletContext]=\"{ $implicit: body.context }\"\n          />\n        } @else {\n          {{ body }}\n        }\n      </span>\n    }\n  }\n\n  @let buttons = modalObj.buttons;\n  @if (buttons?.length) {\n    <span slot=\"actions\" class=\"tds-u-flex tds-u-gap2\">\n      @for (button of buttons; track $index) {\n        <tds-button\n          [text]=\"button.text\"\n          [variant]=\"button.variant ?? 'primary'\"\n          [size]=\"button.size ?? 'md'\"\n          [disabled]=\"button.disabled ?? false\"\n          [attr.data-dismiss-modal]=\"button.dismiss ? '' : null\"\n          (click)=\"handleButtonClick(button)\"\n        ></tds-button>\n      }\n    </span>\n  }\n</tds-modal>\n"]}
|
|
@@ -1,2 +1,2 @@
|
|
|
1
1
|
export {};
|
|
2
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
2
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoibW9kYWwubW9kZWwuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi8uLi8uLi9saWJzL3RlZ2VsLWFuZ3VsYXItZXh0ZW5zaW9ucy9zcmMvbGliL21vZGFsL3NjaGVtYS9tb2RhbC5tb2RlbC50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgVGVtcGxhdGVSZWYsIFR5cGUgfSBmcm9tICdAYW5ndWxhci9jb3JlJztcbmltcG9ydCB7XG4gIENvbXBvbmVudElucHV0cyxcbiAgQ29tcG9uZW50T3V0cHV0cyxcbn0gZnJvbSAnLi4vLi4vdXRpbHMvYW5ndWxhci1jb21wb25lbnQtaW8nO1xuaW1wb3J0IHsgTW9kYWxDb25maWcgfSBmcm9tICcuL21vZGFsLmNvbmZpZyc7XG5pbXBvcnQgeyBNb2RhbEJ1dHRvbiB9IGZyb20gJy4vbW9kYWwudHlwZXMnO1xuXG4vKiogQ29tcG9uZW50LWJhc2VkIG1vZGFsIGJvZHkgd2l0aCB0eXBlZCBpbnB1dHMvb3V0cHV0cy4gKi9cbmV4cG9ydCB0eXBlIE1vZGFsQ29tcG9uZW50Qm9keTxUQ29tcG9uZW50ID0gdW5rbm93bj4gPSB7XG4gIGNvbXBvbmVudDogVHlwZTxUQ29tcG9uZW50PjtcbiAgaW5wdXRzOiBDb21wb25lbnRJbnB1dHM8VENvbXBvbmVudD47XG4gIG91dHB1dHM/OiBDb21wb25lbnRPdXRwdXRzPFRDb21wb25lbnQ+O1xufTtcblxuLyoqIFRlbXBsYXRlLWJhc2VkIG1vZGFsIGJvZHkgd2l0aCBhbiBvcHRpb25hbCBjb250ZXh0LiAqL1xuZXhwb3J0IHR5cGUgTW9kYWxUZW1wbGF0ZUJvZHk8VENvbnRleHQgPSB1bmtub3duPiA9IHtcbiAgdGVtcGxhdGU6IFRlbXBsYXRlUmVmPFRDb250ZXh0PjtcbiAgY29udGV4dD86IFRDb250ZXh0O1xufTtcblxuLyoqIFVuaW9uIG9mIHN1cHBvcnRlZCBtb2RhbCBib2R5IHR5cGVzLiAqL1xuZXhwb3J0IHR5cGUgTW9kYWxCb2R5PFQgPSB1bmtub3duPiA9XG4gIHwgc3RyaW5nXG4gIHwgTW9kYWxUZW1wbGF0ZUJvZHk8VD5cbiAgfCBNb2RhbENvbXBvbmVudEJvZHk8VD47XG5cbi8qKiBQdWJsaWMgb3B0aW9ucyB1c2VkIHdoZW4gY3JlYXRpbmcgYSBtb2RhbC4gKi9cbmV4cG9ydCB0eXBlIE1vZGFsT3B0aW9uczxUID0gdW5rbm93bj4gPSBQYXJ0aWFsPE1vZGFsQ29uZmlnPiAmIHtcbiAgc2VsZWN0b3I/OiBzdHJpbmc7XG4gIHJlZmVyZW5jZUVsPzogSFRNTEVsZW1lbnQ7XG4gIGhlYWRlcj86IHN0cmluZyB8IFRlbXBsYXRlUmVmPHVua25vd24+O1xuICBib2R5PzogTW9kYWxCb2R5PFQ+O1xuICBidXR0b25zPzogTW9kYWxCdXR0b25bXTtcbiAgLyoqIENhbGxiYWNrIGZpcmVkIHdoZW4gdGhlIG1vZGFsIGlzIGNsb3NlZCAodXNlciBvciBwcm9ncmFtbWF0aWMpLiAqL1xuICBvbkNsb3NlZD86ICgpID0+IHZvaWQ7XG4gIC8qKiBDYWxsYmFjayBmaXJlZCB3aGVuIHRoZSBtb2RhbCBpcyByZW1vdmVkIGZyb20gdGhlIERPTS4gKi9cbiAgb25SZW1vdmVkPzogKCkgPT4gdm9pZDtcbn07XG5cbi8qKiBSdW50aW1lIG1vZGFsIHN0YXRlIHN0b3JlZCBieSB0aGUgc2VydmljZS4gKi9cbmV4cG9ydCB0eXBlIE1vZGFsID0gUmVxdWlyZWQ8TW9kYWxDb25maWc+ICZcbiAgT21pdDxNb2RhbE9wdGlvbnM8dW5rbm93bj4sIGtleW9mIE1vZGFsQ29uZmlnPiAmIHtcbiAgICBpZDogc3RyaW5nO1xuICAgIGlzT3BlbjogYm9vbGVhbjtcbiAgfTtcbiJdfQ==
|
package/index.d.ts
CHANGED
|
@@ -1,8 +1,9 @@
|
|
|
1
1
|
import * as _angular_core from '@angular/core';
|
|
2
|
-
import { InjectionToken, EnvironmentProviders, InputSignal, OutputEmitterRef, TemplateRef, Type } from '@angular/core';
|
|
2
|
+
import { InjectionToken, EnvironmentProviders, InputSignal, OutputEmitterRef, TemplateRef, Type, ElementRef } from '@angular/core';
|
|
3
3
|
import z$1, { z } from 'zod';
|
|
4
4
|
import { zx } from '@traversable/zod';
|
|
5
5
|
import { Subject } from 'rxjs';
|
|
6
|
+
import { ControlValueAccessor } from '@angular/forms';
|
|
6
7
|
|
|
7
8
|
/**
|
|
8
9
|
* List of available toast types (inherited from Tegel)
|
|
@@ -372,13 +373,18 @@ declare const MODAL_CONFIG: InjectionToken<ModalConfig>;
|
|
|
372
373
|
declare const DEFAULT_MODAL_CONFIG: Required<ModalConfig>;
|
|
373
374
|
|
|
374
375
|
/** Component-based modal body with typed inputs/outputs. */
|
|
375
|
-
type ModalComponentBody<
|
|
376
|
-
component: Type<
|
|
377
|
-
inputs: ComponentInputs<
|
|
378
|
-
outputs?: ComponentOutputs<
|
|
376
|
+
type ModalComponentBody<TComponent = unknown> = {
|
|
377
|
+
component: Type<TComponent>;
|
|
378
|
+
inputs: ComponentInputs<TComponent>;
|
|
379
|
+
outputs?: ComponentOutputs<TComponent>;
|
|
380
|
+
};
|
|
381
|
+
/** Template-based modal body with an optional context. */
|
|
382
|
+
type ModalTemplateBody<TContext = unknown> = {
|
|
383
|
+
template: TemplateRef<TContext>;
|
|
384
|
+
context?: TContext;
|
|
379
385
|
};
|
|
380
386
|
/** Union of supported modal body types. */
|
|
381
|
-
type ModalBody<T = unknown> = string |
|
|
387
|
+
type ModalBody<T = unknown> = string | ModalTemplateBody<T> | ModalComponentBody<T>;
|
|
382
388
|
/** Public options used when creating a modal. */
|
|
383
389
|
type ModalOptions<T = unknown> = Partial<ModalConfig> & {
|
|
384
390
|
selector?: string;
|
|
@@ -795,6 +801,169 @@ declare class HardRefreshDirective {
|
|
|
795
801
|
static ɵdir: _angular_core.ɵɵDirectiveDeclaration<HardRefreshDirective, "[taeHardRefresh]", never, { "clicksRequired": { "alias": "clicksRequired"; "required": false; "isSignal": true; }; "clickWindowMs": { "alias": "clickWindowMs"; "required": false; "isSignal": true; }; }, {}, never, never, true, never>;
|
|
796
802
|
}
|
|
797
803
|
|
|
804
|
+
/**
|
|
805
|
+
* Directive that provides typed context for an `ng-template`.
|
|
806
|
+
*
|
|
807
|
+
* This is a compile-time helper: it does not affect runtime behavior,
|
|
808
|
+
* but it allows Angular's template type checker to infer the template context.
|
|
809
|
+
*
|
|
810
|
+
* Usage:
|
|
811
|
+
* ```html
|
|
812
|
+
* <ng-template #tpl let-data [typedContext]="someContext">...</ng-template>
|
|
813
|
+
* ```
|
|
814
|
+
*/
|
|
815
|
+
declare class TypedTemplateDirective<T> {
|
|
816
|
+
/**
|
|
817
|
+
* Typed context value used only for type inference.
|
|
818
|
+
* This input is not read at runtime.
|
|
819
|
+
*/
|
|
820
|
+
readonly typedContext: _angular_core.InputSignal<TemplateRef<T>>;
|
|
821
|
+
/**
|
|
822
|
+
* Template context guard to inform Angular's template type system.
|
|
823
|
+
* Always returns `true` at runtime; used purely for typing.
|
|
824
|
+
*/
|
|
825
|
+
static ngTemplateContextGuard<T>(dir: TypedTemplateDirective<T>, ctx: unknown): ctx is {
|
|
826
|
+
$implicit: T;
|
|
827
|
+
};
|
|
828
|
+
static ɵfac: _angular_core.ɵɵFactoryDeclaration<TypedTemplateDirective<any>, never>;
|
|
829
|
+
static ɵdir: _angular_core.ɵɵDirectiveDeclaration<TypedTemplateDirective<any>, "ng-template[typedContext]", never, { "typedContext": { "alias": "typedContext"; "required": true; "isSignal": true; }; }, {}, never, never, true, never>;
|
|
830
|
+
}
|
|
831
|
+
|
|
832
|
+
/** A single day entry used by the calendar grid renderer. */
|
|
833
|
+
type CalendarDay = {
|
|
834
|
+
date: number;
|
|
835
|
+
month: number;
|
|
836
|
+
year: number;
|
|
837
|
+
isCurrentMonth: boolean;
|
|
838
|
+
isToday: boolean;
|
|
839
|
+
isSelected: boolean;
|
|
840
|
+
};
|
|
841
|
+
|
|
842
|
+
/**
|
|
843
|
+
* Standalone date/time picker that integrates with Angular forms via CVA.
|
|
844
|
+
*
|
|
845
|
+
* Supports `date`, `datetime`, and `time` modes, exposes a typed API via inputs,
|
|
846
|
+
* and emits form updates through the registered CVA callbacks.
|
|
847
|
+
*
|
|
848
|
+
* Adheres to Tegel design guidelines and can be used as a drop-in replacement.
|
|
849
|
+
*/
|
|
850
|
+
declare class TaeDateTimePickerComponent implements ControlValueAccessor {
|
|
851
|
+
private readonly elementRef;
|
|
852
|
+
readonly label: _angular_core.InputSignal<string>;
|
|
853
|
+
readonly labelPosition: _angular_core.InputSignal<"inside" | "outside" | "no-label">;
|
|
854
|
+
readonly size: _angular_core.InputSignal<"lg" | "md" | "sm">;
|
|
855
|
+
readonly state: _angular_core.InputSignal<"error" | undefined>;
|
|
856
|
+
readonly helper: _angular_core.InputSignal<string>;
|
|
857
|
+
readonly mode: _angular_core.InputSignal<"date" | "datetime" | "time">;
|
|
858
|
+
private readonly internalValue;
|
|
859
|
+
readonly displayValue: _angular_core.Signal<string>;
|
|
860
|
+
private readonly suppressChange;
|
|
861
|
+
readonly disabled: _angular_core.WritableSignal<boolean>;
|
|
862
|
+
readonly labelId: string;
|
|
863
|
+
readonly showPicker: _angular_core.WritableSignal<boolean>;
|
|
864
|
+
readonly currentMonth: _angular_core.WritableSignal<number>;
|
|
865
|
+
readonly currentYear: _angular_core.WritableSignal<number>;
|
|
866
|
+
readonly calendarDays: _angular_core.Signal<CalendarDay[]>;
|
|
867
|
+
readonly selectedDay: _angular_core.WritableSignal<number | null>;
|
|
868
|
+
readonly selectedMonth: _angular_core.WritableSignal<number | null>;
|
|
869
|
+
readonly selectedYear: _angular_core.WritableSignal<number | null>;
|
|
870
|
+
readonly selectedHour: _angular_core.WritableSignal<string | null>;
|
|
871
|
+
readonly selectedMinute: _angular_core.WritableSignal<string | null>;
|
|
872
|
+
readonly selectedSecond: _angular_core.WritableSignal<string | null>;
|
|
873
|
+
readonly weekDays: readonly ["Mo", "Tu", "We", "Th", "Fr", "Sa", "Su"];
|
|
874
|
+
readonly months: readonly ["January", "February", "March", "April", "May", "June", "July", "August", "September", "October", "November", "December"];
|
|
875
|
+
readonly hours: string[];
|
|
876
|
+
readonly minutes: string[];
|
|
877
|
+
readonly seconds: string[];
|
|
878
|
+
readonly placeholder: _angular_core.Signal<"dd-MM-YYYY HH:mm:ss" | "HH:mm:ss" | "dd-MM-YYYY">;
|
|
879
|
+
readonly currentMonthName: _angular_core.Signal<"January" | "February" | "March" | "April" | "May" | "June" | "July" | "August" | "September" | "October" | "November" | "December">;
|
|
880
|
+
private onChange;
|
|
881
|
+
private onTouched;
|
|
882
|
+
private updateScheduled;
|
|
883
|
+
constructor(elementRef: ElementRef);
|
|
884
|
+
/**
|
|
885
|
+
* Writes an ISO value from the outside into the picker state.
|
|
886
|
+
* @param value ISO date/datetime string.
|
|
887
|
+
*/
|
|
888
|
+
writeValue(value: string): void;
|
|
889
|
+
/**
|
|
890
|
+
* Registers the CVA change handler.
|
|
891
|
+
* @param fn Callback invoked on value changes.
|
|
892
|
+
*/
|
|
893
|
+
registerOnChange(fn: (value: string) => void): void;
|
|
894
|
+
/**
|
|
895
|
+
* Registers the CVA touched handler.
|
|
896
|
+
* @param fn Callback invoked on touch.
|
|
897
|
+
*/
|
|
898
|
+
registerOnTouched(fn: () => void): void;
|
|
899
|
+
/**
|
|
900
|
+
* Updates the disabled state for the control.
|
|
901
|
+
* @param isDisabled Whether the control is disabled.
|
|
902
|
+
*/
|
|
903
|
+
setDisabledState(isDisabled: boolean): void;
|
|
904
|
+
/**
|
|
905
|
+
* Closes the picker when clicking outside the component.
|
|
906
|
+
* @param event Document click event.
|
|
907
|
+
*/
|
|
908
|
+
onDocumentClick(event: MouseEvent): void;
|
|
909
|
+
/** Marks the control as touched. */
|
|
910
|
+
onBlur(): void;
|
|
911
|
+
/**
|
|
912
|
+
* Toggles the picker popover.
|
|
913
|
+
* @param event Click event from the control.
|
|
914
|
+
*/
|
|
915
|
+
togglePicker(event: Event): void;
|
|
916
|
+
/** Moves the calendar to the previous month. */
|
|
917
|
+
previousMonth(): void;
|
|
918
|
+
/** Moves the calendar to the next month. */
|
|
919
|
+
nextMonth(): void;
|
|
920
|
+
/**
|
|
921
|
+
* Builds the calendar day grid for the current month.
|
|
922
|
+
* @returns Calendar day entries for rendering.
|
|
923
|
+
*/
|
|
924
|
+
private buildCalendar;
|
|
925
|
+
/**
|
|
926
|
+
* Selects a date from the calendar grid.
|
|
927
|
+
* @param day Calendar day entry.
|
|
928
|
+
*/
|
|
929
|
+
selectDate(day: CalendarDay): void;
|
|
930
|
+
/**
|
|
931
|
+
* Selects an hour in datetime mode.
|
|
932
|
+
* @param hour Two-digit hour string.
|
|
933
|
+
*/
|
|
934
|
+
selectHour(hour: string): void;
|
|
935
|
+
/**
|
|
936
|
+
* Selects a minute in datetime mode.
|
|
937
|
+
* @param minute Two-digit minute string.
|
|
938
|
+
*/
|
|
939
|
+
selectMinute(minute: string): void;
|
|
940
|
+
/**
|
|
941
|
+
* Selects a second in datetime mode.
|
|
942
|
+
* @param second Two-digit second string.
|
|
943
|
+
*/
|
|
944
|
+
selectSecond(second: string): void;
|
|
945
|
+
/** Confirms the selection and closes the picker. */
|
|
946
|
+
confirm(): void;
|
|
947
|
+
/**
|
|
948
|
+
* Parses an ISO date segment into selected date parts.
|
|
949
|
+
* @param datePart ISO date segment.
|
|
950
|
+
*/
|
|
951
|
+
private parseDatePart;
|
|
952
|
+
/** Computes and emits the ISO value from current parts. */
|
|
953
|
+
private updateValue;
|
|
954
|
+
/** Resets all selected date/time parts to defaults. */
|
|
955
|
+
private resetSelection;
|
|
956
|
+
/**
|
|
957
|
+
* Returns current selected date parts when fully defined.
|
|
958
|
+
* @returns Date parts or `null` if incomplete.
|
|
959
|
+
*/
|
|
960
|
+
private getSelectedDateParts;
|
|
961
|
+
/** Returns today's date in local time. */
|
|
962
|
+
private getToday;
|
|
963
|
+
static ɵfac: _angular_core.ɵɵFactoryDeclaration<TaeDateTimePickerComponent, never>;
|
|
964
|
+
static ɵcmp: _angular_core.ɵɵComponentDeclaration<TaeDateTimePickerComponent, "tae-date-time-picker", never, { "label": { "alias": "label"; "required": false; "isSignal": true; }; "labelPosition": { "alias": "labelPosition"; "required": false; "isSignal": true; }; "size": { "alias": "size"; "required": false; "isSignal": true; }; "state": { "alias": "state"; "required": false; "isSignal": true; }; "helper": { "alias": "helper"; "required": false; "isSignal": true; }; "mode": { "alias": "mode"; "required": false; "isSignal": true; }; }, {}, never, never, true, never>;
|
|
965
|
+
}
|
|
966
|
+
|
|
798
967
|
/**
|
|
799
968
|
* An enhanced standalone footer component based on the Tegel `TdsFooterComponent`.
|
|
800
969
|
*
|
|
@@ -835,5 +1004,5 @@ declare class TaeFooterComponent {
|
|
|
835
1004
|
static ɵcmp: _angular_core.ɵɵComponentDeclaration<TaeFooterComponent, "tae-footer", never, { "variant": { "alias": "variant"; "required": false; "isSignal": true; }; "version": { "alias": "version"; "required": false; "isSignal": true; }; }, {}, never, never, true, never>;
|
|
836
1005
|
}
|
|
837
1006
|
|
|
838
|
-
export { BarcodeScannerDirective, DEFAULT_MODAL_CONFIG, DEFAULT_TOAST_CONFIG, HardRefreshDirective, MODAL_CONFIG, ModalService, TOAST_CONFIG, TaeFooterComponent, ToastService, createEnvKit, parseEnvFile, provideModal, provideRuntimeConfig, provideStaticConfig, provideToast };
|
|
1007
|
+
export { BarcodeScannerDirective, DEFAULT_MODAL_CONFIG, DEFAULT_TOAST_CONFIG, HardRefreshDirective, MODAL_CONFIG, ModalService, TOAST_CONFIG, TaeDateTimePickerComponent, TaeFooterComponent, ToastService, TypedTemplateDirective, createEnvKit, parseEnvFile, provideModal, provideRuntimeConfig, provideStaticConfig, provideToast };
|
|
839
1008
|
export type { Modal, ModalActionsPosition, ModalAlertDialogRole, ModalButton, ModalConfig, ModalOptions, ModalRef, ModalSize, Toast, ToastConfig, ToastOptions };
|