@smart-solutions-tech/smart-angular-calendar 0.0.9 → 0.0.11
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/esm2020/lib/components/smart-calendar/smart-calendar.component.mjs +110 -25
- package/esm2020/lib/components/week/week.component.mjs +133 -0
- package/esm2020/lib/smart-angular-calendar.module.mjs +6 -3
- package/fesm2015/smart-solutions-tech-smart-angular-calendar.mjs +237 -24
- package/fesm2015/smart-solutions-tech-smart-angular-calendar.mjs.map +1 -1
- package/fesm2020/smart-solutions-tech-smart-angular-calendar.mjs +237 -24
- package/fesm2020/smart-solutions-tech-smart-angular-calendar.mjs.map +1 -1
- package/lib/components/smart-calendar/smart-calendar.component.d.ts +11 -2
- package/lib/components/week/week.component.d.ts +34 -0
- package/lib/smart-angular-calendar.module.d.ts +3 -2
- package/package.json +1 -1
|
@@ -4,11 +4,13 @@ import * as i1 from "@angular/common";
|
|
|
4
4
|
import * as i2 from "../calendar/calendar.component";
|
|
5
5
|
import * as i3 from "../month-selector/month-selector.component";
|
|
6
6
|
import * as i4 from "../day/day.component";
|
|
7
|
-
import * as i5 from "
|
|
8
|
-
import * as i6 from "../../pipes/translate
|
|
7
|
+
import * as i5 from "../week/week.component";
|
|
8
|
+
import * as i6 from "../../pipes/translate.pipe";
|
|
9
|
+
import * as i7 from "../../pipes/translate-month.pipe";
|
|
9
10
|
export class SmartCalendarComponent {
|
|
10
11
|
constructor() {
|
|
11
12
|
this.mode = 'month';
|
|
13
|
+
this.startAt = 'sunday';
|
|
12
14
|
this.currentMonth = {
|
|
13
15
|
year: new Date().getFullYear(),
|
|
14
16
|
month: new Date().getMonth() + 1
|
|
@@ -20,6 +22,9 @@ export class SmartCalendarComponent {
|
|
|
20
22
|
this.showDateSelector = false;
|
|
21
23
|
this.selectedDate = new Date();
|
|
22
24
|
this.selectedDateEvents = [];
|
|
25
|
+
this.weekStartDate = new Date();
|
|
26
|
+
this.weekEndDate = new Date(this.weekStartDate.getTime() + 6 * 24 * 60 * 60 * 1000);
|
|
27
|
+
this.weekEvents = [];
|
|
23
28
|
}
|
|
24
29
|
onDocumentClick(event) {
|
|
25
30
|
if (!this.showDateSelector)
|
|
@@ -89,24 +94,25 @@ export class SmartCalendarComponent {
|
|
|
89
94
|
oldMonth
|
|
90
95
|
});
|
|
91
96
|
}
|
|
92
|
-
|
|
93
|
-
const today = new Date();
|
|
97
|
+
nextDay() {
|
|
94
98
|
const oldMonth = { ...this.currentMonth };
|
|
99
|
+
const newDate = new Date(this.selectedDate);
|
|
100
|
+
newDate.setDate(newDate.getDate() + 1);
|
|
101
|
+
this.selectDate(newDate);
|
|
95
102
|
this.currentMonth = {
|
|
96
|
-
year:
|
|
97
|
-
month:
|
|
103
|
+
year: newDate.getFullYear(),
|
|
104
|
+
month: newDate.getMonth() + 1
|
|
98
105
|
};
|
|
99
|
-
this.selectDate(today);
|
|
100
106
|
if (this.currentMonth.month !== oldMonth.month || this.currentMonth.year !== oldMonth.year)
|
|
101
107
|
this.monthChange.emit({
|
|
102
108
|
newMonth: this.currentMonth,
|
|
103
109
|
oldMonth
|
|
104
110
|
});
|
|
105
111
|
}
|
|
106
|
-
|
|
112
|
+
prevDay() {
|
|
107
113
|
const oldMonth = { ...this.currentMonth };
|
|
108
114
|
const newDate = new Date(this.selectedDate);
|
|
109
|
-
newDate.setDate(newDate.getDate()
|
|
115
|
+
newDate.setDate(newDate.getDate() - 1);
|
|
110
116
|
this.selectDate(newDate);
|
|
111
117
|
this.currentMonth = {
|
|
112
118
|
year: newDate.getFullYear(),
|
|
@@ -118,11 +124,26 @@ export class SmartCalendarComponent {
|
|
|
118
124
|
oldMonth
|
|
119
125
|
});
|
|
120
126
|
}
|
|
121
|
-
|
|
127
|
+
nextWeek() {
|
|
122
128
|
const oldMonth = { ...this.currentMonth };
|
|
123
|
-
const newDate = new Date(this.
|
|
124
|
-
newDate.setDate(newDate.getDate()
|
|
125
|
-
this.
|
|
129
|
+
const newDate = new Date(this.weekStartDate);
|
|
130
|
+
newDate.setDate(newDate.getDate() + 7);
|
|
131
|
+
this.selectWeekByDate(newDate);
|
|
132
|
+
this.currentMonth = {
|
|
133
|
+
year: newDate.getFullYear(),
|
|
134
|
+
month: newDate.getMonth() + 1
|
|
135
|
+
};
|
|
136
|
+
if (this.currentMonth.month !== oldMonth.month || this.currentMonth.year !== oldMonth.year)
|
|
137
|
+
this.monthChange.emit({
|
|
138
|
+
newMonth: this.currentMonth,
|
|
139
|
+
oldMonth
|
|
140
|
+
});
|
|
141
|
+
}
|
|
142
|
+
prevWeek() {
|
|
143
|
+
const oldMonth = { ...this.currentMonth };
|
|
144
|
+
const newDate = new Date(this.weekStartDate);
|
|
145
|
+
newDate.setDate(newDate.getDate() - 7);
|
|
146
|
+
this.selectWeekByDate(newDate);
|
|
126
147
|
this.currentMonth = {
|
|
127
148
|
year: newDate.getFullYear(),
|
|
128
149
|
month: newDate.getMonth() + 1
|
|
@@ -136,29 +157,51 @@ export class SmartCalendarComponent {
|
|
|
136
157
|
onNext() {
|
|
137
158
|
if (this.mode === 'month')
|
|
138
159
|
this.nextMonth();
|
|
160
|
+
else if (this.mode === 'week')
|
|
161
|
+
this.nextWeek();
|
|
139
162
|
else if (this.mode === 'day')
|
|
140
163
|
this.nextDay();
|
|
141
164
|
}
|
|
142
165
|
onPrev() {
|
|
143
166
|
if (this.mode === 'month')
|
|
144
167
|
this.prevMonth();
|
|
168
|
+
else if (this.mode === 'week')
|
|
169
|
+
this.prevWeek();
|
|
145
170
|
else if (this.mode === 'day')
|
|
146
171
|
this.prevDay();
|
|
147
172
|
}
|
|
148
|
-
|
|
149
|
-
|
|
173
|
+
goToToday() {
|
|
174
|
+
const today = new Date();
|
|
175
|
+
const oldMonth = { ...this.currentMonth };
|
|
176
|
+
this.currentMonth = {
|
|
177
|
+
year: today.getFullYear(),
|
|
178
|
+
month: today.getMonth() + 1
|
|
179
|
+
};
|
|
180
|
+
this.selectDate(today);
|
|
181
|
+
this.selectWeekByDate(today);
|
|
182
|
+
if (this.currentMonth.month !== oldMonth.month || this.currentMonth.year !== oldMonth.year)
|
|
183
|
+
this.monthChange.emit({
|
|
184
|
+
newMonth: this.currentMonth,
|
|
185
|
+
oldMonth
|
|
186
|
+
});
|
|
187
|
+
}
|
|
188
|
+
setSelectedDateEvents() {
|
|
150
189
|
this.selectedDateEvents = this.events.filter(event => {
|
|
151
190
|
const eventDate = new Date(event.start);
|
|
152
191
|
// return events that starts, ends or spans the selected date
|
|
153
|
-
return (eventDate.getDate() ===
|
|
154
|
-
eventDate.getMonth() ===
|
|
155
|
-
eventDate.getFullYear() ===
|
|
192
|
+
return (eventDate.getDate() === this.selectedDate.getDate() &&
|
|
193
|
+
eventDate.getMonth() === this.selectedDate.getMonth() &&
|
|
194
|
+
eventDate.getFullYear() === this.selectedDate.getFullYear()) ||
|
|
156
195
|
(event.end &&
|
|
157
|
-
event.end.getDate() ===
|
|
158
|
-
event.end.getMonth() ===
|
|
159
|
-
event.end.getFullYear() ===
|
|
160
|
-
(event.start <
|
|
196
|
+
event.end.getDate() === this.selectedDate.getDate() &&
|
|
197
|
+
event.end.getMonth() === this.selectedDate.getMonth() &&
|
|
198
|
+
event.end.getFullYear() === this.selectedDate.getFullYear()) ||
|
|
199
|
+
(event.start < this.selectedDate && event.end && event.end > this.selectedDate);
|
|
161
200
|
});
|
|
201
|
+
}
|
|
202
|
+
selectDate(date) {
|
|
203
|
+
this.selectedDate = date;
|
|
204
|
+
this.setSelectedDateEvents();
|
|
162
205
|
const oldMonth = this.currentMonth;
|
|
163
206
|
this.currentMonth = {
|
|
164
207
|
year: date.getFullYear(),
|
|
@@ -171,14 +214,56 @@ export class SmartCalendarComponent {
|
|
|
171
214
|
});
|
|
172
215
|
this.dateChanged.emit(this.selectedDate);
|
|
173
216
|
}
|
|
217
|
+
selectWeekByDate(date) {
|
|
218
|
+
// Calculate the start date of the week based on startAt
|
|
219
|
+
const dayOfWeek = date.getDay(); // 0 (Sun) to 6 (Sat)
|
|
220
|
+
let diffToStart;
|
|
221
|
+
if (this.startAt === 'monday') {
|
|
222
|
+
diffToStart = (dayOfWeek === 0) ? -6 : 1 - dayOfWeek; // If Sunday, go back 6 days
|
|
223
|
+
}
|
|
224
|
+
else {
|
|
225
|
+
diffToStart = -dayOfWeek; // Sunday start
|
|
226
|
+
}
|
|
227
|
+
this.weekStartDate = new Date(date);
|
|
228
|
+
this.weekStartDate.setDate(date.getDate() + diffToStart);
|
|
229
|
+
this.weekEndDate = new Date(this.weekStartDate);
|
|
230
|
+
this.weekEndDate.setDate(this.weekStartDate.getDate() + 6);
|
|
231
|
+
// Filter events for the week
|
|
232
|
+
this.weekEvents = this.events.filter(event => {
|
|
233
|
+
const eventStart = new Date(event.start);
|
|
234
|
+
const eventEnd = event.end ? new Date(event.end) : eventStart;
|
|
235
|
+
return (eventStart <= this.weekEndDate) && (eventEnd >= this.weekStartDate);
|
|
236
|
+
});
|
|
237
|
+
const oldMonth = this.currentMonth;
|
|
238
|
+
this.currentMonth = {
|
|
239
|
+
year: this.weekStartDate.getFullYear(),
|
|
240
|
+
month: this.weekStartDate.getMonth() + 1
|
|
241
|
+
};
|
|
242
|
+
if (this.currentMonth.month !== oldMonth.month || this.currentMonth.year !== oldMonth.year)
|
|
243
|
+
this.monthChange.emit({
|
|
244
|
+
newMonth: this.currentMonth,
|
|
245
|
+
oldMonth
|
|
246
|
+
});
|
|
247
|
+
}
|
|
248
|
+
setMode(newMode) {
|
|
249
|
+
this.mode = newMode;
|
|
250
|
+
if (newMode === 'day') {
|
|
251
|
+
this.setSelectedDateEvents();
|
|
252
|
+
}
|
|
253
|
+
else if (newMode === 'week') {
|
|
254
|
+
this.selectWeekByDate(this.selectedDate);
|
|
255
|
+
}
|
|
256
|
+
}
|
|
174
257
|
}
|
|
175
258
|
SmartCalendarComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "14.3.0", ngImport: i0, type: SmartCalendarComponent, deps: [], target: i0.ɵɵFactoryTarget.Component });
|
|
176
|
-
SmartCalendarComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "14.3.0", type: SmartCalendarComponent, selector: "smart-calendar", inputs: { mode: "mode", currentMonth: "currentMonth", events: "events" }, outputs: { monthChange: "monthChange", dateChanged: "dateChanged", eventClick: "eventClick" }, host: { listeners: { "document:click": "onDocumentClick($event)" } }, viewQueries: [{ propertyName: "dateSelectorDropdown", first: true, predicate: ["dateSelectorDropdown"], descendants: true }], ngImport: i0, template: "<div class=\"smart-calendar-component\">\n <div class=\"header\">\n <div class=\"nav-controls\">\n <button class=\"btn-prev\" (click)=\"onPrev()\">\n <svg xmlns=\"http://www.w3.org/2000/svg\" height=\"24px\" viewBox=\"0 -960 960 960\" width=\"24px\"\n fill=\"currentColor\">\n <path d=\"M560-240 320-480l240-240 56 56-184 184 184 184-56 56Z\" />\n </svg>\n </button>\n\n <button class=\"btn-today\" (click)=\"goToToday()\">{{'today' | translate}}</button>\n\n <button class=\"btn-next\" (click)=\"onNext()\">\n <svg xmlns=\"http://www.w3.org/2000/svg\" height=\"24px\" viewBox=\"0 -960 960 960\" width=\"24px\"\n fill=\"currentColor\">\n <path d=\"M504-480 320-664l56-56 240 240-240 240-56-56 184-184Z\" />\n </svg>\n </button>\n </div>\n\n <div class=\"spacer\"></div>\n\n <div class=\"date-selector-container\">\n <button class=\"btn-date-selector\" (click)=\"toggleDateSelector()\">\n <span *ngIf=\"mode === 'month'\">{{ currentMonth.month | translateMonth}} - {{ currentMonth.year }}</span>\n <span *ngIf=\"mode === 'day'\">{{ selectedDate.toDateString() }}</span>\n\n <svg xmlns=\"http://www.w3.org/2000/svg\" height=\"24px\" viewBox=\"0 -960 960 960\" width=\"24px\"\n fill=\"currentColor\">\n <path d=\"M480-360 280-560h400L480-360Z\" />\n </svg>\n </button>\n\n <div class=\"date-selector-dropdown\" *ngIf=\"showDateSelector\" #dateSelectorDropdown>\n <lib-month-selector [selectedMonth]=\"currentMonth\"\n (monthSelected)=\"onMonthSelected($event)\"></lib-month-selector>\n </div>\n </div>\n\n <div class=\"spacer\"></div>\n\n <div class=\"view-selector\">\n <button class=\"btn-view\" [ngClass]=\"{active: mode === 'month'}\" (click)=\"
|
|
259
|
+
SmartCalendarComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "14.3.0", type: SmartCalendarComponent, selector: "smart-calendar", inputs: { mode: "mode", startAt: "startAt", currentMonth: "currentMonth", events: "events" }, outputs: { monthChange: "monthChange", dateChanged: "dateChanged", eventClick: "eventClick" }, host: { listeners: { "document:click": "onDocumentClick($event)" } }, viewQueries: [{ propertyName: "dateSelectorDropdown", first: true, predicate: ["dateSelectorDropdown"], descendants: true }], ngImport: i0, template: "<div class=\"smart-calendar-component\">\n <div class=\"header\">\n <div class=\"nav-controls\">\n <button class=\"btn-prev\" (click)=\"onPrev()\">\n <svg xmlns=\"http://www.w3.org/2000/svg\" height=\"24px\" viewBox=\"0 -960 960 960\" width=\"24px\"\n fill=\"currentColor\">\n <path d=\"M560-240 320-480l240-240 56 56-184 184 184 184-56 56Z\" />\n </svg>\n </button>\n\n <button class=\"btn-today\" (click)=\"goToToday()\">{{'today' | translate}}</button>\n\n <button class=\"btn-next\" (click)=\"onNext()\">\n <svg xmlns=\"http://www.w3.org/2000/svg\" height=\"24px\" viewBox=\"0 -960 960 960\" width=\"24px\"\n fill=\"currentColor\">\n <path d=\"M504-480 320-664l56-56 240 240-240 240-56-56 184-184Z\" />\n </svg>\n </button>\n </div>\n\n <div class=\"spacer\"></div>\n\n <div class=\"date-selector-container\">\n <button class=\"btn-date-selector\" (click)=\"toggleDateSelector()\">\n <span *ngIf=\"mode === 'month'\">{{ currentMonth.month | translateMonth}} - {{ currentMonth.year }}</span>\n <span *ngIf=\"mode === 'week'\">\n {{ weekStartDate.toDateString() }} - {{ weekEndDate.toDateString() }}\n </span>\n <span *ngIf=\"mode === 'day'\">{{ selectedDate.toDateString() }}</span>\n\n <svg xmlns=\"http://www.w3.org/2000/svg\" height=\"24px\" viewBox=\"0 -960 960 960\" width=\"24px\"\n fill=\"currentColor\">\n <path d=\"M480-360 280-560h400L480-360Z\" />\n </svg>\n </button>\n\n <div class=\"date-selector-dropdown\" *ngIf=\"showDateSelector\" #dateSelectorDropdown>\n <lib-month-selector [selectedMonth]=\"currentMonth\"\n (monthSelected)=\"onMonthSelected($event)\"></lib-month-selector>\n </div>\n </div>\n\n <div class=\"spacer\"></div>\n\n <div class=\"view-selector\">\n <button class=\"btn-view\" [ngClass]=\"{active: mode === 'month'}\" (click)=\"setMode('month')\">\n <!-- month view -->\n <svg xmlns=\"http://www.w3.org/2000/svg\" height=\"24px\" viewBox=\"0 -960 960 960\" width=\"24px\"\n fill=\"currentColor\">\n <path\n d=\"M200-80q-33 0-56.5-23.5T120-160v-560q0-33 23.5-56.5T200-800h40v-80h80v80h320v-80h80v80h40q33 0 56.5 23.5T840-720v560q0 33-23.5 56.5T760-80H200Zm0-80h560v-400H200v400Zm0-480h560v-80H200v80Zm0 0v-80 80Zm280 240q-17 0-28.5-11.5T440-440q0-17 11.5-28.5T480-480q17 0 28.5 11.5T520-440q0 17-11.5 28.5T480-400Zm-160 0q-17 0-28.5-11.5T280-440q0-17 11.5-28.5T320-480q17 0 28.5 11.5T360-440q0 17-11.5 28.5T320-400Zm320 0q-17 0-28.5-11.5T600-440q0-17 11.5-28.5T640-480q17 0 28.5 11.5T680-440q0 17-11.5 28.5T640-400ZM480-240q-17 0-28.5-11.5T440-280q0-17 11.5-28.5T480-320q17 0 28.5 11.5T520-280q0 17-11.5 28.5T480-240Zm-160 0q-17 0-28.5-11.5T280-280q0-17 11.5-28.5T320-320q17 0 28.5 11.5T360-280q0 17-11.5 28.5T320-240Zm320 0q-17 0-28.5-11.5T600-280q0-17 11.5-28.5T640-320q17 0 28.5 11.5T680-280q0 17-11.5 28.5T640-240Z\" />\n </svg>\n </button>\n\n <button class=\"btn-view\" [ngClass]=\"{active: mode === 'week'}\" (click)=\"setMode('week')\">\n <!-- week view -->\n <svg xmlns=\"http://www.w3.org/2000/svg\" height=\"24px\" viewBox=\"0 -960 960 960\" width=\"24px\"\n fill=\"currentColor\">\n <path\n d=\"M160-160q-33 0-56.5-23.5T80-240v-480q0-33 23.5-56.5T160-800h640q33 0 56.5 23.5T880-720v480q0 33-23.5 56.5T800-160H160Zm360-80h100v-480H520v480Zm-180 0h100v-480H340v480Zm-180 0h100v-480H160v480Zm540 0h100v-480H700v480Z\" />\n </svg>\n </button>\n\n <button class=\"btn-view\" [ngClass]=\"{active: mode === 'day'}\" (click)=\"setMode('day')\">\n <!-- day view -->\n <svg xmlns=\"http://www.w3.org/2000/svg\" height=\"24px\" viewBox=\"0 -960 960 960\" width=\"24px\"\n fill=\"currentColor\">\n <path\n d=\"M120-160v-80h720v80H120Zm0-560v-80h720v80H120Zm80 400q-33 0-56.5-23.5T120-400v-160q0-33 23.5-56.5T200-640h560q33 0 56.5 23.5T840-560v160q0 33-23.5 56.5T760-320H200Zm0-80h560v-160H200v160Zm0-160v160-160Z\" />\n </svg>\n </button>\n </div>\n </div>\n\n <div class=\"view-container\">\n <lib-calendar [events]=\"events\" [currentMonth]=\"currentMonth\" *ngIf=\"mode === 'month'\" [startAt]=\"startAt\"\n (eventClick)=\"onEventClick($event)\"></lib-calendar>\n\n <lib-week *ngIf=\"mode === 'week'\" [startAt]=\"startAt\" [startDate]=\"weekStartDate\"\n [events]=\"weekEvents\"></lib-week>\n\n <lib-day *ngIf=\"mode === 'day'\" [date]=\"selectedDate\" [events]=\"selectedDateEvents\"\n (eventClick)=\"onEventClick($event)\"></lib-day>\n </div>\n\n</div>", styles: ["*{box-sizing:border-box;margin:0;padding:0}.smart-calendar-component{display:flex;flex-direction:column;width:100%;height:100%;background-color:#fff}.header{flex-shrink:0;display:flex;align-items:center;gap:16px;padding:2px 16px}.header .spacer{flex:1 1 auto}.header .nav-controls{display:flex;align-items:center}.header .nav-controls .btn-prev,.header .nav-controls .btn-next{display:flex;align-items:center;justify-content:center;width:30px;height:30px;border:none;background:none;border-radius:50%;cursor:pointer;-webkit-user-select:none;user-select:none}.header .nav-controls .btn-prev:hover,.header .nav-controls .btn-next:hover{background-color:#eee}.header .nav-controls .btn-today{display:flex;align-items:center;height:30px;padding:0 12px;border:none;background:none;border-radius:50px;cursor:pointer;font-weight:600;-webkit-user-select:none;user-select:none}.header .nav-controls .btn-today:hover{background-color:#eee}.header .view-selector{display:flex;align-items:center}.header .view-selector .btn-view{display:flex;align-items:center;justify-content:center;width:30px;height:30px;border:1px solid #cccccc;background:none;cursor:pointer;-webkit-user-select:none;user-select:none}.header .view-selector .btn-view:first-child{border-top-left-radius:4px;border-bottom-left-radius:4px;border-right:none}.header .view-selector .btn-view:last-child{border-top-right-radius:4px;border-bottom-right-radius:4px;border-left:none}.header .view-selector .btn-view:hover{background-color:#eee}.header .view-selector .btn-view svg{width:22px;height:22px}.header .view-selector .btn-view.active{background-color:#0000001a}.header .date-selector-container{position:relative}.header .date-selector-container .date-selector-dropdown{position:absolute;top:100%;left:50%;transform:translate(-50%);background-color:#fff;border-radius:4px;box-shadow:0 2px 24px #0000004d;z-index:3}.header .date-selector-container .date-selector-dropdown:before{content:\"\";position:absolute;top:-6px;left:50%;transform:translate(-50%) rotate(45deg);width:12px;height:12px;background-color:#fff;border-radius:2px}.header .btn-date-selector{display:flex;align-items:center;gap:8px;height:30px;padding:0 8px 0 12px;border:none;background:none;border-radius:50px;cursor:pointer;font-weight:600;-webkit-user-select:none;user-select:none}.header .btn-date-selector:hover{background-color:#eee}.view-container{flex:1 1 auto;overflow:hidden}\n"], dependencies: [{ kind: "directive", type: i1.NgClass, selector: "[ngClass]", inputs: ["class", "ngClass"] }, { kind: "directive", type: i1.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "component", type: i2.CalendarComponent, selector: "lib-calendar", inputs: ["currentMonth", "events", "startAt"], outputs: ["eventClick"] }, { kind: "component", type: i3.MonthSelectorComponent, selector: "lib-month-selector", inputs: ["selectedMonth"], outputs: ["monthSelected"] }, { kind: "component", type: i4.DayComponent, selector: "lib-day", inputs: ["events", "date"], outputs: ["eventClick"] }, { kind: "component", type: i5.WeekComponent, selector: "lib-week", inputs: ["startAt", "events", "startDate"], outputs: ["eventClick"] }, { kind: "pipe", type: i6.TranslatePipe, name: "translate" }, { kind: "pipe", type: i7.TranslateMonthPipe, name: "translateMonth" }] });
|
|
177
260
|
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "14.3.0", ngImport: i0, type: SmartCalendarComponent, decorators: [{
|
|
178
261
|
type: Component,
|
|
179
|
-
args: [{ selector: 'smart-calendar', template: "<div class=\"smart-calendar-component\">\n <div class=\"header\">\n <div class=\"nav-controls\">\n <button class=\"btn-prev\" (click)=\"onPrev()\">\n <svg xmlns=\"http://www.w3.org/2000/svg\" height=\"24px\" viewBox=\"0 -960 960 960\" width=\"24px\"\n fill=\"currentColor\">\n <path d=\"M560-240 320-480l240-240 56 56-184 184 184 184-56 56Z\" />\n </svg>\n </button>\n\n <button class=\"btn-today\" (click)=\"goToToday()\">{{'today' | translate}}</button>\n\n <button class=\"btn-next\" (click)=\"onNext()\">\n <svg xmlns=\"http://www.w3.org/2000/svg\" height=\"24px\" viewBox=\"0 -960 960 960\" width=\"24px\"\n fill=\"currentColor\">\n <path d=\"M504-480 320-664l56-56 240 240-240 240-56-56 184-184Z\" />\n </svg>\n </button>\n </div>\n\n <div class=\"spacer\"></div>\n\n <div class=\"date-selector-container\">\n <button class=\"btn-date-selector\" (click)=\"toggleDateSelector()\">\n <span *ngIf=\"mode === 'month'\">{{ currentMonth.month | translateMonth}} - {{ currentMonth.year }}</span>\n <span *ngIf=\"mode === 'day'\">{{ selectedDate.toDateString() }}</span>\n\n <svg xmlns=\"http://www.w3.org/2000/svg\" height=\"24px\" viewBox=\"0 -960 960 960\" width=\"24px\"\n fill=\"currentColor\">\n <path d=\"M480-360 280-560h400L480-360Z\" />\n </svg>\n </button>\n\n <div class=\"date-selector-dropdown\" *ngIf=\"showDateSelector\" #dateSelectorDropdown>\n <lib-month-selector [selectedMonth]=\"currentMonth\"\n (monthSelected)=\"onMonthSelected($event)\"></lib-month-selector>\n </div>\n </div>\n\n <div class=\"spacer\"></div>\n\n <div class=\"view-selector\">\n <button class=\"btn-view\" [ngClass]=\"{active: mode === 'month'}\" (click)=\"
|
|
262
|
+
args: [{ selector: 'smart-calendar', template: "<div class=\"smart-calendar-component\">\n <div class=\"header\">\n <div class=\"nav-controls\">\n <button class=\"btn-prev\" (click)=\"onPrev()\">\n <svg xmlns=\"http://www.w3.org/2000/svg\" height=\"24px\" viewBox=\"0 -960 960 960\" width=\"24px\"\n fill=\"currentColor\">\n <path d=\"M560-240 320-480l240-240 56 56-184 184 184 184-56 56Z\" />\n </svg>\n </button>\n\n <button class=\"btn-today\" (click)=\"goToToday()\">{{'today' | translate}}</button>\n\n <button class=\"btn-next\" (click)=\"onNext()\">\n <svg xmlns=\"http://www.w3.org/2000/svg\" height=\"24px\" viewBox=\"0 -960 960 960\" width=\"24px\"\n fill=\"currentColor\">\n <path d=\"M504-480 320-664l56-56 240 240-240 240-56-56 184-184Z\" />\n </svg>\n </button>\n </div>\n\n <div class=\"spacer\"></div>\n\n <div class=\"date-selector-container\">\n <button class=\"btn-date-selector\" (click)=\"toggleDateSelector()\">\n <span *ngIf=\"mode === 'month'\">{{ currentMonth.month | translateMonth}} - {{ currentMonth.year }}</span>\n <span *ngIf=\"mode === 'week'\">\n {{ weekStartDate.toDateString() }} - {{ weekEndDate.toDateString() }}\n </span>\n <span *ngIf=\"mode === 'day'\">{{ selectedDate.toDateString() }}</span>\n\n <svg xmlns=\"http://www.w3.org/2000/svg\" height=\"24px\" viewBox=\"0 -960 960 960\" width=\"24px\"\n fill=\"currentColor\">\n <path d=\"M480-360 280-560h400L480-360Z\" />\n </svg>\n </button>\n\n <div class=\"date-selector-dropdown\" *ngIf=\"showDateSelector\" #dateSelectorDropdown>\n <lib-month-selector [selectedMonth]=\"currentMonth\"\n (monthSelected)=\"onMonthSelected($event)\"></lib-month-selector>\n </div>\n </div>\n\n <div class=\"spacer\"></div>\n\n <div class=\"view-selector\">\n <button class=\"btn-view\" [ngClass]=\"{active: mode === 'month'}\" (click)=\"setMode('month')\">\n <!-- month view -->\n <svg xmlns=\"http://www.w3.org/2000/svg\" height=\"24px\" viewBox=\"0 -960 960 960\" width=\"24px\"\n fill=\"currentColor\">\n <path\n d=\"M200-80q-33 0-56.5-23.5T120-160v-560q0-33 23.5-56.5T200-800h40v-80h80v80h320v-80h80v80h40q33 0 56.5 23.5T840-720v560q0 33-23.5 56.5T760-80H200Zm0-80h560v-400H200v400Zm0-480h560v-80H200v80Zm0 0v-80 80Zm280 240q-17 0-28.5-11.5T440-440q0-17 11.5-28.5T480-480q17 0 28.5 11.5T520-440q0 17-11.5 28.5T480-400Zm-160 0q-17 0-28.5-11.5T280-440q0-17 11.5-28.5T320-480q17 0 28.5 11.5T360-440q0 17-11.5 28.5T320-400Zm320 0q-17 0-28.5-11.5T600-440q0-17 11.5-28.5T640-480q17 0 28.5 11.5T680-440q0 17-11.5 28.5T640-400ZM480-240q-17 0-28.5-11.5T440-280q0-17 11.5-28.5T480-320q17 0 28.5 11.5T520-280q0 17-11.5 28.5T480-240Zm-160 0q-17 0-28.5-11.5T280-280q0-17 11.5-28.5T320-320q17 0 28.5 11.5T360-280q0 17-11.5 28.5T320-240Zm320 0q-17 0-28.5-11.5T600-280q0-17 11.5-28.5T640-320q17 0 28.5 11.5T680-280q0 17-11.5 28.5T640-240Z\" />\n </svg>\n </button>\n\n <button class=\"btn-view\" [ngClass]=\"{active: mode === 'week'}\" (click)=\"setMode('week')\">\n <!-- week view -->\n <svg xmlns=\"http://www.w3.org/2000/svg\" height=\"24px\" viewBox=\"0 -960 960 960\" width=\"24px\"\n fill=\"currentColor\">\n <path\n d=\"M160-160q-33 0-56.5-23.5T80-240v-480q0-33 23.5-56.5T160-800h640q33 0 56.5 23.5T880-720v480q0 33-23.5 56.5T800-160H160Zm360-80h100v-480H520v480Zm-180 0h100v-480H340v480Zm-180 0h100v-480H160v480Zm540 0h100v-480H700v480Z\" />\n </svg>\n </button>\n\n <button class=\"btn-view\" [ngClass]=\"{active: mode === 'day'}\" (click)=\"setMode('day')\">\n <!-- day view -->\n <svg xmlns=\"http://www.w3.org/2000/svg\" height=\"24px\" viewBox=\"0 -960 960 960\" width=\"24px\"\n fill=\"currentColor\">\n <path\n d=\"M120-160v-80h720v80H120Zm0-560v-80h720v80H120Zm80 400q-33 0-56.5-23.5T120-400v-160q0-33 23.5-56.5T200-640h560q33 0 56.5 23.5T840-560v160q0 33-23.5 56.5T760-320H200Zm0-80h560v-160H200v160Zm0-160v160-160Z\" />\n </svg>\n </button>\n </div>\n </div>\n\n <div class=\"view-container\">\n <lib-calendar [events]=\"events\" [currentMonth]=\"currentMonth\" *ngIf=\"mode === 'month'\" [startAt]=\"startAt\"\n (eventClick)=\"onEventClick($event)\"></lib-calendar>\n\n <lib-week *ngIf=\"mode === 'week'\" [startAt]=\"startAt\" [startDate]=\"weekStartDate\"\n [events]=\"weekEvents\"></lib-week>\n\n <lib-day *ngIf=\"mode === 'day'\" [date]=\"selectedDate\" [events]=\"selectedDateEvents\"\n (eventClick)=\"onEventClick($event)\"></lib-day>\n </div>\n\n</div>", styles: ["*{box-sizing:border-box;margin:0;padding:0}.smart-calendar-component{display:flex;flex-direction:column;width:100%;height:100%;background-color:#fff}.header{flex-shrink:0;display:flex;align-items:center;gap:16px;padding:2px 16px}.header .spacer{flex:1 1 auto}.header .nav-controls{display:flex;align-items:center}.header .nav-controls .btn-prev,.header .nav-controls .btn-next{display:flex;align-items:center;justify-content:center;width:30px;height:30px;border:none;background:none;border-radius:50%;cursor:pointer;-webkit-user-select:none;user-select:none}.header .nav-controls .btn-prev:hover,.header .nav-controls .btn-next:hover{background-color:#eee}.header .nav-controls .btn-today{display:flex;align-items:center;height:30px;padding:0 12px;border:none;background:none;border-radius:50px;cursor:pointer;font-weight:600;-webkit-user-select:none;user-select:none}.header .nav-controls .btn-today:hover{background-color:#eee}.header .view-selector{display:flex;align-items:center}.header .view-selector .btn-view{display:flex;align-items:center;justify-content:center;width:30px;height:30px;border:1px solid #cccccc;background:none;cursor:pointer;-webkit-user-select:none;user-select:none}.header .view-selector .btn-view:first-child{border-top-left-radius:4px;border-bottom-left-radius:4px;border-right:none}.header .view-selector .btn-view:last-child{border-top-right-radius:4px;border-bottom-right-radius:4px;border-left:none}.header .view-selector .btn-view:hover{background-color:#eee}.header .view-selector .btn-view svg{width:22px;height:22px}.header .view-selector .btn-view.active{background-color:#0000001a}.header .date-selector-container{position:relative}.header .date-selector-container .date-selector-dropdown{position:absolute;top:100%;left:50%;transform:translate(-50%);background-color:#fff;border-radius:4px;box-shadow:0 2px 24px #0000004d;z-index:3}.header .date-selector-container .date-selector-dropdown:before{content:\"\";position:absolute;top:-6px;left:50%;transform:translate(-50%) rotate(45deg);width:12px;height:12px;background-color:#fff;border-radius:2px}.header .btn-date-selector{display:flex;align-items:center;gap:8px;height:30px;padding:0 8px 0 12px;border:none;background:none;border-radius:50px;cursor:pointer;font-weight:600;-webkit-user-select:none;user-select:none}.header .btn-date-selector:hover{background-color:#eee}.view-container{flex:1 1 auto;overflow:hidden}\n"] }]
|
|
180
263
|
}], ctorParameters: function () { return []; }, propDecorators: { mode: [{
|
|
181
264
|
type: Input
|
|
265
|
+
}], startAt: [{
|
|
266
|
+
type: Input
|
|
182
267
|
}], currentMonth: [{
|
|
183
268
|
type: Input
|
|
184
269
|
}], events: [{
|
|
@@ -196,4 +281,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "14.3.0", ngImpor
|
|
|
196
281
|
type: HostListener,
|
|
197
282
|
args: ['document:click', ['$event']]
|
|
198
283
|
}] } });
|
|
199
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"smart-calendar.component.js","sourceRoot":"","sources":["../../../../../../projects/smart-angular-calendar/src/lib/components/smart-calendar/smart-calendar.component.ts","../../../../../../projects/smart-angular-calendar/src/lib/components/smart-calendar/smart-calendar.component.html"],"names":[],"mappings":"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","sourcesContent":["import { Component, ElementRef, EventEmitter, HostListener, Input, OnInit, Output, SimpleChanges, ViewChild } from '@angular/core';\nimport { CalendarEvent } from '../../types';\n\n@Component({\n  selector: 'smart-calendar',\n  templateUrl: './smart-calendar.component.html',\n  styleUrls: ['./smart-calendar.component.scss']\n})\nexport class SmartCalendarComponent implements OnInit {\n\n  @Input() mode: 'month' | 'week' | 'day' = 'month';\n\n  @Input() currentMonth: { year: number, month: number } = {\n    year: new Date().getFullYear(),\n    month: new Date().getMonth() + 1\n  };\n\n  @Input() events: CalendarEvent[] = [];\n\n  @Output() monthChange = new EventEmitter<{\n    newMonth: { year: number, month: number },\n    oldMonth: { year: number, month: number }\n  }>();\n\n  @Output() dateChanged = new EventEmitter<Date>();\n\n  @Output() eventClick = new EventEmitter<CalendarEvent>();\n\n  @ViewChild('dateSelectorDropdown', { static: false }) dateSelectorDropdown!: ElementRef;\n\n  @HostListener('document:click', ['$event'])\n  onDocumentClick(event: MouseEvent) {\n    if (!this.showDateSelector) return;\n    const dropdown = this.dateSelectorDropdown?.nativeElement;\n    const button = document.querySelector('.btn-date-selector');\n    if (dropdown && !dropdown.contains(event.target) && button && !button.contains(event.target as Node)) {\n      this.showDateSelector = false;\n    }\n  }\n\n  showDateSelector: boolean = false;\n\n  selectedDate: Date = new Date();\n  selectedDateEvents: CalendarEvent[] = [];\n\n  constructor() { }\n\n  ngOnInit(): void {\n    this.selectDate(new Date());\n  }\n\n  toggleDateSelector() {\n    this.showDateSelector = !this.showDateSelector;\n  }\n\n  onMonthSelected(newMonth: { year: number, month: number }) {\n    const oldMonth = { ...this.currentMonth };\n    this.currentMonth = newMonth;\n\n    this.selectDate(new Date(this.currentMonth.year, this.currentMonth.month - 1, 1));\n\n    this.monthChange.emit({\n      newMonth: this.currentMonth,\n      oldMonth\n    });\n\n    this.showDateSelector = false;\n  }\n\n  onEventClick(event: CalendarEvent) {\n    this.eventClick.emit(event);\n  }\n\n  nextMonth() {\n    const oldMonth = { ...this.currentMonth };\n\n    if (this.currentMonth.month === 12) {\n      this.currentMonth = {\n        year: this.currentMonth.year + 1,\n        month: 1\n      };\n    } else {\n      this.currentMonth = {\n        year: this.currentMonth.year,\n        month: this.currentMonth.month + 1\n      };\n    }\n\n    this.selectDate(new Date(this.currentMonth.year, this.currentMonth.month - 1, 1));\n\n    this.monthChange.emit({\n      newMonth: this.currentMonth,\n      oldMonth\n    });\n  }\n\n  prevMonth() {\n    const oldMonth = { ...this.currentMonth };\n    if (this.currentMonth.month === 1) {\n      this.currentMonth = {\n        year: this.currentMonth.year - 1,\n        month: 12\n      };\n    } else {\n      this.currentMonth = {\n        year: this.currentMonth.year,\n        month: this.currentMonth.month - 1\n      };\n    }\n\n    this.selectDate(new Date(this.currentMonth.year, this.currentMonth.month - 1, 1));\n\n    this.monthChange.emit({\n      newMonth: this.currentMonth,\n      oldMonth\n    });\n  }\n\n  goToToday() {\n    const today = new Date();\n    const oldMonth = { ...this.currentMonth };\n\n    this.currentMonth = {\n      year: today.getFullYear(),\n      month: today.getMonth() + 1\n    };\n\n    this.selectDate(today);\n\n    if (this.currentMonth.month !== oldMonth.month || this.currentMonth.year !== oldMonth.year)\n      this.monthChange.emit({\n        newMonth: this.currentMonth,\n        oldMonth\n      });\n  }\n\n  nextDay() {\n    const oldMonth = { ...this.currentMonth };\n\n    const newDate = new Date(this.selectedDate);\n    newDate.setDate(newDate.getDate() + 1);\n\n    this.selectDate(newDate);\n\n    this.currentMonth = {\n      year: newDate.getFullYear(),\n      month: newDate.getMonth() + 1\n    };\n\n    if (this.currentMonth.month !== oldMonth.month || this.currentMonth.year !== oldMonth.year)\n      this.monthChange.emit({\n        newMonth: this.currentMonth,\n        oldMonth\n      });\n  }\n\n  prevDay() {\n    const oldMonth = { ...this.currentMonth };\n\n    const newDate = new Date(this.selectedDate);\n    newDate.setDate(newDate.getDate() - 1);\n\n    this.selectDate(newDate);\n\n    this.currentMonth = {\n      year: newDate.getFullYear(),\n      month: newDate.getMonth() + 1\n    };\n\n    if (this.currentMonth.month !== oldMonth.month || this.currentMonth.year !== oldMonth.year)\n      this.monthChange.emit({\n        newMonth: this.currentMonth,\n        oldMonth\n      });\n  }\n\n  onNext() {\n    if (this.mode === 'month')\n      this.nextMonth();\n    else if (this.mode === 'day')\n      this.nextDay();\n  }\n\n  onPrev() {\n    if (this.mode === 'month')\n      this.prevMonth();\n    else if (this.mode === 'day')\n      this.prevDay();\n  }\n\n  selectDate(date: Date) {\n    this.selectedDate = date;\n\n    this.selectedDateEvents = this.events.filter(event => {\n      const eventDate = new Date(event.start);\n\n      // return events that starts, ends or spans the selected date\n      return (eventDate.getDate() === date.getDate() &&\n        eventDate.getMonth() === date.getMonth() &&\n        eventDate.getFullYear() === date.getFullYear()) ||\n        (event.end &&\n          event.end.getDate() === date.getDate() &&\n          event.end.getMonth() === date.getMonth() &&\n          event.end.getFullYear() === date.getFullYear()) ||\n        (event.start < date && event.end && event.end > date);\n    });\n\n    const oldMonth = this.currentMonth;\n\n    this.currentMonth = {\n      year: date.getFullYear(),\n      month: date.getMonth() + 1\n    };\n\n    if (this.currentMonth.month !== oldMonth.month || this.currentMonth.year !== oldMonth.year)\n      this.monthChange.emit({\n        newMonth: this.currentMonth,\n        oldMonth\n      });\n\n    this.dateChanged.emit(this.selectedDate);\n  }\n\n}\n","<div class=\"smart-calendar-component\">\n    <div class=\"header\">\n        <div class=\"nav-controls\">\n            <button class=\"btn-prev\" (click)=\"onPrev()\">\n                <svg xmlns=\"http://www.w3.org/2000/svg\" height=\"24px\" viewBox=\"0 -960 960 960\" width=\"24px\"\n                    fill=\"currentColor\">\n                    <path d=\"M560-240 320-480l240-240 56 56-184 184 184 184-56 56Z\" />\n                </svg>\n            </button>\n\n            <button class=\"btn-today\" (click)=\"goToToday()\">{{'today' | translate}}</button>\n\n            <button class=\"btn-next\" (click)=\"onNext()\">\n                <svg xmlns=\"http://www.w3.org/2000/svg\" height=\"24px\" viewBox=\"0 -960 960 960\" width=\"24px\"\n                    fill=\"currentColor\">\n                    <path d=\"M504-480 320-664l56-56 240 240-240 240-56-56 184-184Z\" />\n                </svg>\n            </button>\n        </div>\n\n        <div class=\"spacer\"></div>\n\n        <div class=\"date-selector-container\">\n            <button class=\"btn-date-selector\" (click)=\"toggleDateSelector()\">\n                <span *ngIf=\"mode === 'month'\">{{ currentMonth.month | translateMonth}} - {{ currentMonth.year }}</span>\n                <span *ngIf=\"mode === 'day'\">{{ selectedDate.toDateString() }}</span>\n\n                <svg xmlns=\"http://www.w3.org/2000/svg\" height=\"24px\" viewBox=\"0 -960 960 960\" width=\"24px\"\n                    fill=\"currentColor\">\n                    <path d=\"M480-360 280-560h400L480-360Z\" />\n                </svg>\n            </button>\n\n            <div class=\"date-selector-dropdown\" *ngIf=\"showDateSelector\" #dateSelectorDropdown>\n                <lib-month-selector [selectedMonth]=\"currentMonth\"\n                    (monthSelected)=\"onMonthSelected($event)\"></lib-month-selector>\n            </div>\n        </div>\n\n        <div class=\"spacer\"></div>\n\n        <div class=\"view-selector\">\n            <button class=\"btn-view\" [ngClass]=\"{active: mode === 'month'}\" (click)=\"mode = 'month'\">\n                <!-- month view -->\n                <svg xmlns=\"http://www.w3.org/2000/svg\" height=\"24px\" viewBox=\"0 -960 960 960\" width=\"24px\"\n                    fill=\"currentColor\">\n                    <path\n                        d=\"M200-80q-33 0-56.5-23.5T120-160v-560q0-33 23.5-56.5T200-800h40v-80h80v80h320v-80h80v80h40q33 0 56.5 23.5T840-720v560q0 33-23.5 56.5T760-80H200Zm0-80h560v-400H200v400Zm0-480h560v-80H200v80Zm0 0v-80 80Zm280 240q-17 0-28.5-11.5T440-440q0-17 11.5-28.5T480-480q17 0 28.5 11.5T520-440q0 17-11.5 28.5T480-400Zm-160 0q-17 0-28.5-11.5T280-440q0-17 11.5-28.5T320-480q17 0 28.5 11.5T360-440q0 17-11.5 28.5T320-400Zm320 0q-17 0-28.5-11.5T600-440q0-17 11.5-28.5T640-480q17 0 28.5 11.5T680-440q0 17-11.5 28.5T640-400ZM480-240q-17 0-28.5-11.5T440-280q0-17 11.5-28.5T480-320q17 0 28.5 11.5T520-280q0 17-11.5 28.5T480-240Zm-160 0q-17 0-28.5-11.5T280-280q0-17 11.5-28.5T320-320q17 0 28.5 11.5T360-280q0 17-11.5 28.5T320-240Zm320 0q-17 0-28.5-11.5T600-280q0-17 11.5-28.5T640-320q17 0 28.5 11.5T680-280q0 17-11.5 28.5T640-240Z\" />\n                </svg>\n            </button>\n\n            <button class=\"btn-view\" [ngClass]=\"{active: mode === 'week'}\" (click)=\"mode = 'week'\">\n                <!-- week view -->\n                <svg xmlns=\"http://www.w3.org/2000/svg\" height=\"24px\" viewBox=\"0 -960 960 960\" width=\"24px\"\n                    fill=\"currentColor\">\n                    <path\n                        d=\"M160-160q-33 0-56.5-23.5T80-240v-480q0-33 23.5-56.5T160-800h640q33 0 56.5 23.5T880-720v480q0 33-23.5 56.5T800-160H160Zm360-80h100v-480H520v480Zm-180 0h100v-480H340v480Zm-180 0h100v-480H160v480Zm540 0h100v-480H700v480Z\" />\n                </svg>\n            </button>\n\n            <button class=\"btn-view\" [ngClass]=\"{active: mode === 'day'}\" (click)=\"mode = 'day'\">\n                <!-- day view -->\n                <svg xmlns=\"http://www.w3.org/2000/svg\" height=\"24px\" viewBox=\"0 -960 960 960\" width=\"24px\"\n                    fill=\"currentColor\">\n                    <path\n                        d=\"M120-160v-80h720v80H120Zm0-560v-80h720v80H120Zm80 400q-33 0-56.5-23.5T120-400v-160q0-33 23.5-56.5T200-640h560q33 0 56.5 23.5T840-560v160q0 33-23.5 56.5T760-320H200Zm0-80h560v-160H200v160Zm0-160v160-160Z\" />\n                </svg>\n            </button>\n        </div>\n    </div>\n\n    <div class=\"view-container\">\n        <lib-calendar [events]=\"events\" [currentMonth]=\"currentMonth\" *ngIf=\"mode === 'month'\"\n            (eventClick)=\"onEventClick($event)\"></lib-calendar>\n\n        <lib-day *ngIf=\"mode === 'day'\" [date]=\"selectedDate\" [events]=\"selectedDateEvents\"\n            (eventClick)=\"onEventClick($event)\"></lib-day>\n    </div>\n\n</div>"]}
|
|
284
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"smart-calendar.component.js","sourceRoot":"","sources":["../../../../../../projects/smart-angular-calendar/src/lib/components/smart-calendar/smart-calendar.component.ts","../../../../../../projects/smart-angular-calendar/src/lib/components/smart-calendar/smart-calendar.component.html"],"names":[],"mappings":"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","sourcesContent":["import { Component, ElementRef, EventEmitter, HostListener, Input, OnInit, Output, SimpleChanges, ViewChild } from '@angular/core';\nimport { CalendarEvent } from '../../types';\n\n@Component({\n  selector: 'smart-calendar',\n  templateUrl: './smart-calendar.component.html',\n  styleUrls: ['./smart-calendar.component.scss']\n})\nexport class SmartCalendarComponent implements OnInit {\n\n  @Input() mode: 'month' | 'week' | 'day' = 'month';\n\n  @Input() startAt: 'monday' | 'sunday' = 'sunday';\n\n  @Input() currentMonth: { year: number, month: number } = {\n    year: new Date().getFullYear(),\n    month: new Date().getMonth() + 1\n  };\n\n  @Input() events: CalendarEvent[] = [];\n\n  @Output() monthChange = new EventEmitter<{\n    newMonth: { year: number, month: number },\n    oldMonth: { year: number, month: number }\n  }>();\n\n  @Output() dateChanged = new EventEmitter<Date>();\n\n  @Output() eventClick = new EventEmitter<CalendarEvent>();\n\n  @ViewChild('dateSelectorDropdown', { static: false }) dateSelectorDropdown!: ElementRef;\n\n  @HostListener('document:click', ['$event'])\n  onDocumentClick(event: MouseEvent) {\n    if (!this.showDateSelector) return;\n    const dropdown = this.dateSelectorDropdown?.nativeElement;\n    const button = document.querySelector('.btn-date-selector');\n    if (dropdown && !dropdown.contains(event.target) && button && !button.contains(event.target as Node)) {\n      this.showDateSelector = false;\n    }\n  }\n\n  showDateSelector: boolean = false;\n\n  selectedDate: Date = new Date();\n  selectedDateEvents: CalendarEvent[] = [];\n\n  weekStartDate: Date = new Date();\n  weekEndDate: Date = new Date(this.weekStartDate.getTime() + 6 * 24 * 60 * 60 * 1000);\n  weekEvents: CalendarEvent[] = [];\n\n  constructor() { }\n\n  ngOnInit(): void {\n    this.selectDate(new Date());\n  }\n\n  toggleDateSelector() {\n    this.showDateSelector = !this.showDateSelector;\n  }\n\n  onMonthSelected(newMonth: { year: number, month: number }) {\n    const oldMonth = { ...this.currentMonth };\n    this.currentMonth = newMonth;\n\n    this.selectDate(new Date(this.currentMonth.year, this.currentMonth.month - 1, 1));\n\n    this.monthChange.emit({\n      newMonth: this.currentMonth,\n      oldMonth\n    });\n\n    this.showDateSelector = false;\n  }\n\n  onEventClick(event: CalendarEvent) {\n    this.eventClick.emit(event);\n  }\n\n  nextMonth() {\n    const oldMonth = { ...this.currentMonth };\n\n    if (this.currentMonth.month === 12) {\n      this.currentMonth = {\n        year: this.currentMonth.year + 1,\n        month: 1\n      };\n    } else {\n      this.currentMonth = {\n        year: this.currentMonth.year,\n        month: this.currentMonth.month + 1\n      };\n    }\n\n    this.selectDate(new Date(this.currentMonth.year, this.currentMonth.month - 1, 1));\n\n    this.monthChange.emit({\n      newMonth: this.currentMonth,\n      oldMonth\n    });\n  }\n\n  prevMonth() {\n    const oldMonth = { ...this.currentMonth };\n    if (this.currentMonth.month === 1) {\n      this.currentMonth = {\n        year: this.currentMonth.year - 1,\n        month: 12\n      };\n    } else {\n      this.currentMonth = {\n        year: this.currentMonth.year,\n        month: this.currentMonth.month - 1\n      };\n    }\n\n    this.selectDate(new Date(this.currentMonth.year, this.currentMonth.month - 1, 1));\n\n    this.monthChange.emit({\n      newMonth: this.currentMonth,\n      oldMonth\n    });\n  }\n\n  nextDay() {\n    const oldMonth = { ...this.currentMonth };\n\n    const newDate = new Date(this.selectedDate);\n    newDate.setDate(newDate.getDate() + 1);\n\n    this.selectDate(newDate);\n\n    this.currentMonth = {\n      year: newDate.getFullYear(),\n      month: newDate.getMonth() + 1\n    };\n\n    if (this.currentMonth.month !== oldMonth.month || this.currentMonth.year !== oldMonth.year)\n      this.monthChange.emit({\n        newMonth: this.currentMonth,\n        oldMonth\n      });\n  }\n\n  prevDay() {\n    const oldMonth = { ...this.currentMonth };\n\n    const newDate = new Date(this.selectedDate);\n    newDate.setDate(newDate.getDate() - 1);\n\n    this.selectDate(newDate);\n\n    this.currentMonth = {\n      year: newDate.getFullYear(),\n      month: newDate.getMonth() + 1\n    };\n\n    if (this.currentMonth.month !== oldMonth.month || this.currentMonth.year !== oldMonth.year)\n      this.monthChange.emit({\n        newMonth: this.currentMonth,\n        oldMonth\n      });\n  }\n\n  nextWeek() {\n    const oldMonth = { ...this.currentMonth };\n\n    const newDate = new Date(this.weekStartDate);\n    newDate.setDate(newDate.getDate() + 7);\n    this.selectWeekByDate(newDate);\n\n    this.currentMonth = {\n      year: newDate.getFullYear(),\n      month: newDate.getMonth() + 1\n    };\n\n    if (this.currentMonth.month !== oldMonth.month || this.currentMonth.year !== oldMonth.year)\n      this.monthChange.emit({\n        newMonth: this.currentMonth,\n        oldMonth\n      });\n  }\n\n  prevWeek() {\n    const oldMonth = { ...this.currentMonth };\n\n    const newDate = new Date(this.weekStartDate);\n    newDate.setDate(newDate.getDate() - 7);\n    this.selectWeekByDate(newDate);\n\n    this.currentMonth = {\n      year: newDate.getFullYear(),\n      month: newDate.getMonth() + 1\n    };\n\n    if (this.currentMonth.month !== oldMonth.month || this.currentMonth.year !== oldMonth.year)\n      this.monthChange.emit({\n        newMonth: this.currentMonth,\n        oldMonth\n      });\n  }\n\n  onNext() {\n    if (this.mode === 'month')\n      this.nextMonth();\n    else if (this.mode === 'week')\n      this.nextWeek();\n    else if (this.mode === 'day')\n      this.nextDay();\n  }\n\n  onPrev() {\n    if (this.mode === 'month')\n      this.prevMonth();\n    else if (this.mode === 'week')\n      this.prevWeek();\n    else if (this.mode === 'day')\n      this.prevDay();\n  }\n\n  goToToday() {\n    const today = new Date();\n    const oldMonth = { ...this.currentMonth };\n\n    this.currentMonth = {\n      year: today.getFullYear(),\n      month: today.getMonth() + 1\n    };\n\n    this.selectDate(today);\n    this.selectWeekByDate(today);\n\n    if (this.currentMonth.month !== oldMonth.month || this.currentMonth.year !== oldMonth.year)\n      this.monthChange.emit({\n        newMonth: this.currentMonth,\n        oldMonth\n      });\n  }\n\n  setSelectedDateEvents() {\n    this.selectedDateEvents = this.events.filter(event => {\n      const eventDate = new Date(event.start);\n\n      // return events that starts, ends or spans the selected date\n      return (eventDate.getDate() === this.selectedDate.getDate() &&\n        eventDate.getMonth() === this.selectedDate.getMonth() &&\n        eventDate.getFullYear() === this.selectedDate.getFullYear()) ||\n        (event.end &&\n          event.end.getDate() === this.selectedDate.getDate() &&\n          event.end.getMonth() === this.selectedDate.getMonth() &&\n          event.end.getFullYear() === this.selectedDate.getFullYear()) ||\n        (event.start < this.selectedDate && event.end && event.end > this.selectedDate);\n    });\n  }\n\n  selectDate(date: Date) {\n    this.selectedDate = date;\n\n    this.setSelectedDateEvents();\n\n    const oldMonth = this.currentMonth;\n\n    this.currentMonth = {\n      year: date.getFullYear(),\n      month: date.getMonth() + 1\n    };\n\n    if (this.currentMonth.month !== oldMonth.month || this.currentMonth.year !== oldMonth.year)\n      this.monthChange.emit({\n        newMonth: this.currentMonth,\n        oldMonth\n      });\n\n    this.dateChanged.emit(this.selectedDate);\n  }\n\n  selectWeekByDate(date: Date) {\n    // Calculate the start date of the week based on startAt\n    const dayOfWeek = date.getDay(); // 0 (Sun) to 6 (Sat)\n    let diffToStart: number;\n\n    if (this.startAt === 'monday') {\n      diffToStart = (dayOfWeek === 0) ? -6 : 1 - dayOfWeek; // If Sunday, go back 6 days\n    } else {\n      diffToStart = -dayOfWeek; // Sunday start\n    }\n\n    this.weekStartDate = new Date(date);\n    this.weekStartDate.setDate(date.getDate() + diffToStart);\n\n    this.weekEndDate = new Date(this.weekStartDate);\n    this.weekEndDate.setDate(this.weekStartDate.getDate() + 6);\n\n    // Filter events for the week\n    this.weekEvents = this.events.filter(event => {\n      const eventStart = new Date(event.start);\n      const eventEnd = event.end ? new Date(event.end) : eventStart;\n      return (eventStart <= this.weekEndDate) && (eventEnd >= this.weekStartDate);\n    });\n\n    const oldMonth = this.currentMonth;\n\n    this.currentMonth = {\n      year: this.weekStartDate.getFullYear(),\n      month: this.weekStartDate.getMonth() + 1\n    };\n\n    if (this.currentMonth.month !== oldMonth.month || this.currentMonth.year !== oldMonth.year)\n      this.monthChange.emit({\n        newMonth: this.currentMonth,\n        oldMonth\n      });\n  }\n\n  setMode(newMode: typeof this.mode) {\n    this.mode = newMode;\n    if (newMode === 'day') {\n      this.setSelectedDateEvents();\n    } else if (newMode === 'week') {\n      this.selectWeekByDate(this.selectedDate);\n    }\n  }\n\n}\n","<div class=\"smart-calendar-component\">\n    <div class=\"header\">\n        <div class=\"nav-controls\">\n            <button class=\"btn-prev\" (click)=\"onPrev()\">\n                <svg xmlns=\"http://www.w3.org/2000/svg\" height=\"24px\" viewBox=\"0 -960 960 960\" width=\"24px\"\n                    fill=\"currentColor\">\n                    <path d=\"M560-240 320-480l240-240 56 56-184 184 184 184-56 56Z\" />\n                </svg>\n            </button>\n\n            <button class=\"btn-today\" (click)=\"goToToday()\">{{'today' | translate}}</button>\n\n            <button class=\"btn-next\" (click)=\"onNext()\">\n                <svg xmlns=\"http://www.w3.org/2000/svg\" height=\"24px\" viewBox=\"0 -960 960 960\" width=\"24px\"\n                    fill=\"currentColor\">\n                    <path d=\"M504-480 320-664l56-56 240 240-240 240-56-56 184-184Z\" />\n                </svg>\n            </button>\n        </div>\n\n        <div class=\"spacer\"></div>\n\n        <div class=\"date-selector-container\">\n            <button class=\"btn-date-selector\" (click)=\"toggleDateSelector()\">\n                <span *ngIf=\"mode === 'month'\">{{ currentMonth.month | translateMonth}} - {{ currentMonth.year }}</span>\n                <span *ngIf=\"mode === 'week'\">\n                    {{ weekStartDate.toDateString() }} - {{ weekEndDate.toDateString() }}\n                </span>\n                <span *ngIf=\"mode === 'day'\">{{ selectedDate.toDateString() }}</span>\n\n                <svg xmlns=\"http://www.w3.org/2000/svg\" height=\"24px\" viewBox=\"0 -960 960 960\" width=\"24px\"\n                    fill=\"currentColor\">\n                    <path d=\"M480-360 280-560h400L480-360Z\" />\n                </svg>\n            </button>\n\n            <div class=\"date-selector-dropdown\" *ngIf=\"showDateSelector\" #dateSelectorDropdown>\n                <lib-month-selector [selectedMonth]=\"currentMonth\"\n                    (monthSelected)=\"onMonthSelected($event)\"></lib-month-selector>\n            </div>\n        </div>\n\n        <div class=\"spacer\"></div>\n\n        <div class=\"view-selector\">\n            <button class=\"btn-view\" [ngClass]=\"{active: mode === 'month'}\" (click)=\"setMode('month')\">\n                <!-- month view -->\n                <svg xmlns=\"http://www.w3.org/2000/svg\" height=\"24px\" viewBox=\"0 -960 960 960\" width=\"24px\"\n                    fill=\"currentColor\">\n                    <path\n                        d=\"M200-80q-33 0-56.5-23.5T120-160v-560q0-33 23.5-56.5T200-800h40v-80h80v80h320v-80h80v80h40q33 0 56.5 23.5T840-720v560q0 33-23.5 56.5T760-80H200Zm0-80h560v-400H200v400Zm0-480h560v-80H200v80Zm0 0v-80 80Zm280 240q-17 0-28.5-11.5T440-440q0-17 11.5-28.5T480-480q17 0 28.5 11.5T520-440q0 17-11.5 28.5T480-400Zm-160 0q-17 0-28.5-11.5T280-440q0-17 11.5-28.5T320-480q17 0 28.5 11.5T360-440q0 17-11.5 28.5T320-400Zm320 0q-17 0-28.5-11.5T600-440q0-17 11.5-28.5T640-480q17 0 28.5 11.5T680-440q0 17-11.5 28.5T640-400ZM480-240q-17 0-28.5-11.5T440-280q0-17 11.5-28.5T480-320q17 0 28.5 11.5T520-280q0 17-11.5 28.5T480-240Zm-160 0q-17 0-28.5-11.5T280-280q0-17 11.5-28.5T320-320q17 0 28.5 11.5T360-280q0 17-11.5 28.5T320-240Zm320 0q-17 0-28.5-11.5T600-280q0-17 11.5-28.5T640-320q17 0 28.5 11.5T680-280q0 17-11.5 28.5T640-240Z\" />\n                </svg>\n            </button>\n\n            <button class=\"btn-view\" [ngClass]=\"{active: mode === 'week'}\" (click)=\"setMode('week')\">\n                <!-- week view -->\n                <svg xmlns=\"http://www.w3.org/2000/svg\" height=\"24px\" viewBox=\"0 -960 960 960\" width=\"24px\"\n                    fill=\"currentColor\">\n                    <path\n                        d=\"M160-160q-33 0-56.5-23.5T80-240v-480q0-33 23.5-56.5T160-800h640q33 0 56.5 23.5T880-720v480q0 33-23.5 56.5T800-160H160Zm360-80h100v-480H520v480Zm-180 0h100v-480H340v480Zm-180 0h100v-480H160v480Zm540 0h100v-480H700v480Z\" />\n                </svg>\n            </button>\n\n            <button class=\"btn-view\" [ngClass]=\"{active: mode === 'day'}\" (click)=\"setMode('day')\">\n                <!-- day view -->\n                <svg xmlns=\"http://www.w3.org/2000/svg\" height=\"24px\" viewBox=\"0 -960 960 960\" width=\"24px\"\n                    fill=\"currentColor\">\n                    <path\n                        d=\"M120-160v-80h720v80H120Zm0-560v-80h720v80H120Zm80 400q-33 0-56.5-23.5T120-400v-160q0-33 23.5-56.5T200-640h560q33 0 56.5 23.5T840-560v160q0 33-23.5 56.5T760-320H200Zm0-80h560v-160H200v160Zm0-160v160-160Z\" />\n                </svg>\n            </button>\n        </div>\n    </div>\n\n    <div class=\"view-container\">\n        <lib-calendar [events]=\"events\" [currentMonth]=\"currentMonth\" *ngIf=\"mode === 'month'\" [startAt]=\"startAt\"\n            (eventClick)=\"onEventClick($event)\"></lib-calendar>\n\n        <lib-week *ngIf=\"mode === 'week'\" [startAt]=\"startAt\" [startDate]=\"weekStartDate\"\n            [events]=\"weekEvents\"></lib-week>\n\n        <lib-day *ngIf=\"mode === 'day'\" [date]=\"selectedDate\" [events]=\"selectedDateEvents\"\n            (eventClick)=\"onEventClick($event)\"></lib-day>\n    </div>\n\n</div>"]}
|
|
@@ -0,0 +1,133 @@
|
|
|
1
|
+
import { Component, EventEmitter, Input, Output } from '@angular/core';
|
|
2
|
+
import * as i0 from "@angular/core";
|
|
3
|
+
import * as i1 from "@angular/common";
|
|
4
|
+
import * as i2 from "../calendar-event/calendar-event.component";
|
|
5
|
+
import * as i3 from "../duration-event/duration-event.component";
|
|
6
|
+
import * as i4 from "../../pipes/translate.pipe";
|
|
7
|
+
export class WeekComponent {
|
|
8
|
+
constructor() {
|
|
9
|
+
this.startAt = 'sunday';
|
|
10
|
+
this.events = [];
|
|
11
|
+
this.eventClick = new EventEmitter();
|
|
12
|
+
this.weekDays = [
|
|
13
|
+
{ tag: 'sun' },
|
|
14
|
+
{ tag: 'mon' },
|
|
15
|
+
{ tag: 'tue' },
|
|
16
|
+
{ tag: 'wed' },
|
|
17
|
+
{ tag: 'thu' },
|
|
18
|
+
{ tag: 'fri' },
|
|
19
|
+
{ tag: 'sat' },
|
|
20
|
+
];
|
|
21
|
+
this.hourBlocks = Array.from({ length: 24 }, (_, i) => i);
|
|
22
|
+
this.allDayHeight = 60;
|
|
23
|
+
this.weekDayCols = [];
|
|
24
|
+
this.resizing = false;
|
|
25
|
+
this.startY = 0;
|
|
26
|
+
this.startHeight = 0;
|
|
27
|
+
this.onResizeMouseMove = (event) => {
|
|
28
|
+
if (!this.resizing)
|
|
29
|
+
return;
|
|
30
|
+
const delta = event.clientY - this.startY;
|
|
31
|
+
const newHeight = this.startHeight + delta;
|
|
32
|
+
this.allDayHeight = Math.max(60, newHeight); // mínimo 80px
|
|
33
|
+
};
|
|
34
|
+
this.onResizeMouseUp = () => {
|
|
35
|
+
this.resizing = false;
|
|
36
|
+
document.removeEventListener('mousemove', this.onResizeMouseMove);
|
|
37
|
+
document.removeEventListener('mouseup', this.onResizeMouseUp);
|
|
38
|
+
};
|
|
39
|
+
}
|
|
40
|
+
ngOnInit() {
|
|
41
|
+
if (this.startAt === 'monday') {
|
|
42
|
+
this.weekDays = [
|
|
43
|
+
{ tag: 'mon' },
|
|
44
|
+
{ tag: 'tue' },
|
|
45
|
+
{ tag: 'wed' },
|
|
46
|
+
{ tag: 'thu' },
|
|
47
|
+
{ tag: 'fri' },
|
|
48
|
+
{ tag: 'sat' },
|
|
49
|
+
{ tag: 'sun' },
|
|
50
|
+
];
|
|
51
|
+
}
|
|
52
|
+
this.initWeekDayCols();
|
|
53
|
+
}
|
|
54
|
+
initWeekDayCols() {
|
|
55
|
+
this.weekDayCols = [];
|
|
56
|
+
for (let i = 0; i < 7; i++) {
|
|
57
|
+
const currentDate = new Date(this.startDate.getTime() + i * 24 * 60 * 60 * 1000);
|
|
58
|
+
this.weekDayCols.push({
|
|
59
|
+
date: currentDate,
|
|
60
|
+
eventCols: [],
|
|
61
|
+
allDayEvents: []
|
|
62
|
+
});
|
|
63
|
+
}
|
|
64
|
+
this.events.forEach(event => this.addEvent(event));
|
|
65
|
+
}
|
|
66
|
+
addEvent(event) {
|
|
67
|
+
const index = this.weekDayCols.findIndex(col => col.date.getFullYear() === event.start.getFullYear() &&
|
|
68
|
+
col.date.getMonth() === event.start.getMonth() &&
|
|
69
|
+
col.date.getDate() === event.start.getDate());
|
|
70
|
+
if (index === -1)
|
|
71
|
+
return;
|
|
72
|
+
const eventCols = this.weekDayCols[index].eventCols;
|
|
73
|
+
if (event.allDay) {
|
|
74
|
+
this.weekDayCols[index].allDayEvents.push(event);
|
|
75
|
+
}
|
|
76
|
+
else {
|
|
77
|
+
const colIndex = this.findAvalailableColumn(event, index);
|
|
78
|
+
if (colIndex === eventCols.length) {
|
|
79
|
+
eventCols.push({ events: [event] });
|
|
80
|
+
}
|
|
81
|
+
else {
|
|
82
|
+
eventCols[colIndex].events.push(event);
|
|
83
|
+
}
|
|
84
|
+
}
|
|
85
|
+
}
|
|
86
|
+
findAvalailableColumn(event, dayIndex) {
|
|
87
|
+
const eventCols = this.weekDayCols[dayIndex].eventCols;
|
|
88
|
+
for (let colIndex = 0; colIndex < eventCols.length; colIndex++) {
|
|
89
|
+
const col = eventCols[colIndex];
|
|
90
|
+
// conflict if event overlaps with any existing event in the column
|
|
91
|
+
const hasConflict = col.events.some(existingEvent => {
|
|
92
|
+
const startsDuringExisting = event.start < existingEvent.end && event.start >= existingEvent.start;
|
|
93
|
+
const endsDuringExisting = event.end > existingEvent.start && event.end <= existingEvent.end;
|
|
94
|
+
const spansExisting = event.start <= existingEvent.start && event.end >= existingEvent.end;
|
|
95
|
+
return startsDuringExisting || endsDuringExisting || spansExisting;
|
|
96
|
+
});
|
|
97
|
+
if (!hasConflict) {
|
|
98
|
+
return colIndex;
|
|
99
|
+
}
|
|
100
|
+
}
|
|
101
|
+
return eventCols.length;
|
|
102
|
+
}
|
|
103
|
+
ngOnChanges(changes) {
|
|
104
|
+
if (changes['events'] || changes['startDate']) {
|
|
105
|
+
this.initWeekDayCols();
|
|
106
|
+
}
|
|
107
|
+
}
|
|
108
|
+
onResizeMouseDown(event) {
|
|
109
|
+
this.resizing = true;
|
|
110
|
+
this.startY = event.clientY;
|
|
111
|
+
this.startHeight = this.allDayHeight;
|
|
112
|
+
document.addEventListener('mousemove', this.onResizeMouseMove);
|
|
113
|
+
document.addEventListener('mouseup', this.onResizeMouseUp);
|
|
114
|
+
}
|
|
115
|
+
onEventClick(event) {
|
|
116
|
+
this.eventClick.emit(event);
|
|
117
|
+
}
|
|
118
|
+
}
|
|
119
|
+
WeekComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "14.3.0", ngImport: i0, type: WeekComponent, deps: [], target: i0.ɵɵFactoryTarget.Component });
|
|
120
|
+
WeekComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "14.3.0", type: WeekComponent, selector: "lib-week", inputs: { startAt: "startAt", events: "events", startDate: "startDate" }, outputs: { eventClick: "eventClick" }, usesOnChanges: true, ngImport: i0, template: "<div class=\"week-view-component\">\n <section class=\"all-day-events\" [style.height.px]=\"allDayHeight\">\n <div class=\"header\">\n {{'allDayEvents' | translate}}\n </div>\n\n <div class=\"days-container\">\n <div class=\"events-container\" *ngFor=\"let day of weekDayCols; let i = index\">\n <div class=\"week-day\">\n {{ weekDays[i].tag | translate | slice:0:3 }}\n </div>\n\n <lib-calendar-event *ngFor=\"let event of day.allDayEvents\" [event]=\"event\"\n (eventClick)=\"onEventClick($event)\"></lib-calendar-event>\n </div>\n </div>\n </section>\n\n <div class=\"resize-divider\" (mousedown)=\"onResizeMouseDown($event)\"></div>\n\n <section class=\"duration-events\">\n <div class=\"hour-blocks\">\n <div class=\"hour-block\" *ngFor=\"let hour of hourBlocks\">\n <span>{{ hour }}:00</span>\n </div>\n </div>\n\n <div class=\"events-wrapper\">\n <div class=\"time-blocks\">\n <div class=\"day-block\" *ngFor=\"let day of weekDayCols\">\n <div class=\"time-block\" *ngFor=\"let hour of hourBlocks\">\n <div class=\"time-block-middle\"></div>\n </div>\n </div>\n </div>\n\n <div class=\"days-container\">\n <div class=\"events-container\" *ngFor=\"let day of weekDayCols\"\n [style.gridTemplateColumns]=\"'repeat(' + day.eventCols.length + ', 1fr)'\">\n <div class=\"event-column\" *ngFor=\"let col of day.eventCols\">\n\n <lib-duration-event *ngFor=\"let event of col.events\" [event]=\"event\" [date]=\"day.date\"\n (eventClick)=\"onEventClick($event)\"></lib-duration-event>\n\n </div>\n </div>\n </div>\n </div>\n </section>\n</div>", styles: ["*{box-sizing:border-box;margin:0;padding:0}.week-view-component{display:flex;flex-direction:column;width:100%;height:100%;max-height:100%;overflow:hidden}section.all-day-events{position:relative;flex-shrink:0;display:flex;flex-direction:column;width:100%;min-height:60px;border-top:1px solid rgba(0,0,0,.12);overflow-y:scroll}section.all-day-events .header{position:sticky;top:0;left:0;z-index:2;width:100%;flex-shrink:0;padding:2px 8px;background-color:#fff;font-size:12px;color:#000000b3;-webkit-user-select:none;user-select:none}section.all-day-events .days-container{display:grid;grid-template-columns:repeat(7,1fr);height:100%;padding-left:45px}section.all-day-events .events-container{display:flex;flex-direction:column;gap:1px;border-right:1px solid rgba(0,0,0,.12)}section.all-day-events .week-day{position:sticky;top:0;left:0;z-index:2;width:100%;font-size:12px;text-align:center;-webkit-user-select:none;user-select:none}section.duration-events{flex-grow:1;display:flex;width:100%;overflow:auto}section.duration-events .hour-blocks{display:flex;flex-direction:column;width:45px;-webkit-user-select:none;user-select:none}section.duration-events .hour-blocks .hour-block{flex-shrink:0;height:60px;padding:4px;border:1px solid rgba(0,0,0,.12);border-top:none;font-size:12px;color:#00000080}section.duration-events .events-wrapper{position:relative;flex-grow:1;width:100%}section.duration-events .events-wrapper .time-blocks{position:absolute;top:0;left:0;width:100%;height:100%;z-index:0;display:grid;grid-template-columns:repeat(7,1fr)}section.duration-events .events-wrapper .time-blocks .day-block{display:flex;flex-direction:column;border-right:1px solid rgba(0,0,0,.12)}section.duration-events .events-wrapper .time-blocks .time-block{flex-shrink:0;height:60px;padding:4px;border-bottom:1px solid rgba(0,0,0,.12);border-top:none;font-size:12px;color:#00000080}section.duration-events .events-wrapper .time-blocks .time-block .time-block-middle{width:100%;height:50%;border-bottom:1px dashed rgba(0,0,0,.12)}section.duration-events .events-wrapper .days-container{display:grid;grid-template-columns:repeat(7,1fr);column-gap:1px}section.duration-events .events-wrapper .events-container{display:grid;column-gap:1px}section.duration-events .events-wrapper .events-container .event-column{position:relative}.resize-divider{flex-shrink:0;width:100%;height:4px;background-color:#0000001f;cursor:row-resize;transition:transform .2s ease}.resize-divider:hover{z-index:5;background:repeating-linear-gradient(45deg,rgba(0,0,0,.12),rgba(0,0,0,.12) 4px,#ffffff 4px,#ffffff 8px);background-color:#fff;background-size:22px 8px;animation:stripe-move 1s linear infinite;transform:scaleY(2)}@keyframes stripe-move{0%{background-position:0 0}to{background-position:22px 0}}\n"], dependencies: [{ kind: "directive", type: i1.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "component", type: i2.CalendarEventComponent, selector: "lib-calendar-event", inputs: ["event"], outputs: ["eventClick"] }, { kind: "component", type: i3.DurationEventComponent, selector: "lib-duration-event", inputs: ["event", "date"], outputs: ["eventClick"] }, { kind: "pipe", type: i1.SlicePipe, name: "slice" }, { kind: "pipe", type: i4.TranslatePipe, name: "translate" }] });
|
|
121
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "14.3.0", ngImport: i0, type: WeekComponent, decorators: [{
|
|
122
|
+
type: Component,
|
|
123
|
+
args: [{ selector: 'lib-week', template: "<div class=\"week-view-component\">\n <section class=\"all-day-events\" [style.height.px]=\"allDayHeight\">\n <div class=\"header\">\n {{'allDayEvents' | translate}}\n </div>\n\n <div class=\"days-container\">\n <div class=\"events-container\" *ngFor=\"let day of weekDayCols; let i = index\">\n <div class=\"week-day\">\n {{ weekDays[i].tag | translate | slice:0:3 }}\n </div>\n\n <lib-calendar-event *ngFor=\"let event of day.allDayEvents\" [event]=\"event\"\n (eventClick)=\"onEventClick($event)\"></lib-calendar-event>\n </div>\n </div>\n </section>\n\n <div class=\"resize-divider\" (mousedown)=\"onResizeMouseDown($event)\"></div>\n\n <section class=\"duration-events\">\n <div class=\"hour-blocks\">\n <div class=\"hour-block\" *ngFor=\"let hour of hourBlocks\">\n <span>{{ hour }}:00</span>\n </div>\n </div>\n\n <div class=\"events-wrapper\">\n <div class=\"time-blocks\">\n <div class=\"day-block\" *ngFor=\"let day of weekDayCols\">\n <div class=\"time-block\" *ngFor=\"let hour of hourBlocks\">\n <div class=\"time-block-middle\"></div>\n </div>\n </div>\n </div>\n\n <div class=\"days-container\">\n <div class=\"events-container\" *ngFor=\"let day of weekDayCols\"\n [style.gridTemplateColumns]=\"'repeat(' + day.eventCols.length + ', 1fr)'\">\n <div class=\"event-column\" *ngFor=\"let col of day.eventCols\">\n\n <lib-duration-event *ngFor=\"let event of col.events\" [event]=\"event\" [date]=\"day.date\"\n (eventClick)=\"onEventClick($event)\"></lib-duration-event>\n\n </div>\n </div>\n </div>\n </div>\n </section>\n</div>", styles: ["*{box-sizing:border-box;margin:0;padding:0}.week-view-component{display:flex;flex-direction:column;width:100%;height:100%;max-height:100%;overflow:hidden}section.all-day-events{position:relative;flex-shrink:0;display:flex;flex-direction:column;width:100%;min-height:60px;border-top:1px solid rgba(0,0,0,.12);overflow-y:scroll}section.all-day-events .header{position:sticky;top:0;left:0;z-index:2;width:100%;flex-shrink:0;padding:2px 8px;background-color:#fff;font-size:12px;color:#000000b3;-webkit-user-select:none;user-select:none}section.all-day-events .days-container{display:grid;grid-template-columns:repeat(7,1fr);height:100%;padding-left:45px}section.all-day-events .events-container{display:flex;flex-direction:column;gap:1px;border-right:1px solid rgba(0,0,0,.12)}section.all-day-events .week-day{position:sticky;top:0;left:0;z-index:2;width:100%;font-size:12px;text-align:center;-webkit-user-select:none;user-select:none}section.duration-events{flex-grow:1;display:flex;width:100%;overflow:auto}section.duration-events .hour-blocks{display:flex;flex-direction:column;width:45px;-webkit-user-select:none;user-select:none}section.duration-events .hour-blocks .hour-block{flex-shrink:0;height:60px;padding:4px;border:1px solid rgba(0,0,0,.12);border-top:none;font-size:12px;color:#00000080}section.duration-events .events-wrapper{position:relative;flex-grow:1;width:100%}section.duration-events .events-wrapper .time-blocks{position:absolute;top:0;left:0;width:100%;height:100%;z-index:0;display:grid;grid-template-columns:repeat(7,1fr)}section.duration-events .events-wrapper .time-blocks .day-block{display:flex;flex-direction:column;border-right:1px solid rgba(0,0,0,.12)}section.duration-events .events-wrapper .time-blocks .time-block{flex-shrink:0;height:60px;padding:4px;border-bottom:1px solid rgba(0,0,0,.12);border-top:none;font-size:12px;color:#00000080}section.duration-events .events-wrapper .time-blocks .time-block .time-block-middle{width:100%;height:50%;border-bottom:1px dashed rgba(0,0,0,.12)}section.duration-events .events-wrapper .days-container{display:grid;grid-template-columns:repeat(7,1fr);column-gap:1px}section.duration-events .events-wrapper .events-container{display:grid;column-gap:1px}section.duration-events .events-wrapper .events-container .event-column{position:relative}.resize-divider{flex-shrink:0;width:100%;height:4px;background-color:#0000001f;cursor:row-resize;transition:transform .2s ease}.resize-divider:hover{z-index:5;background:repeating-linear-gradient(45deg,rgba(0,0,0,.12),rgba(0,0,0,.12) 4px,#ffffff 4px,#ffffff 8px);background-color:#fff;background-size:22px 8px;animation:stripe-move 1s linear infinite;transform:scaleY(2)}@keyframes stripe-move{0%{background-position:0 0}to{background-position:22px 0}}\n"] }]
|
|
124
|
+
}], ctorParameters: function () { return []; }, propDecorators: { startAt: [{
|
|
125
|
+
type: Input
|
|
126
|
+
}], events: [{
|
|
127
|
+
type: Input
|
|
128
|
+
}], startDate: [{
|
|
129
|
+
type: Input
|
|
130
|
+
}], eventClick: [{
|
|
131
|
+
type: Output
|
|
132
|
+
}] } });
|
|
133
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"week.component.js","sourceRoot":"","sources":["../../../../../../projects/smart-angular-calendar/src/lib/components/week/week.component.ts","../../../../../../projects/smart-angular-calendar/src/lib/components/week/week.component.html"],"names":[],"mappings":"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","sourcesContent":["import { Component, EventEmitter, Input, OnChanges, OnInit, Output, SimpleChanges } from '@angular/core';\nimport { CalendarEvent, WeekDay } from '../../types';\n\n@Component({\n  selector: 'lib-week',\n  templateUrl: './week.component.html',\n  styleUrls: ['./week.component.scss']\n})\nexport class WeekComponent implements OnInit, OnChanges {\n\n  @Input() startAt: 'monday' | 'sunday' = 'sunday';\n\n  @Input() events: CalendarEvent[] = [];\n  @Input() startDate!: Date;\n\n  @Output() eventClick = new EventEmitter<CalendarEvent>();\n\n  weekDays: WeekDay[] = [\n    { tag: 'sun' },\n    { tag: 'mon' },\n    { tag: 'tue' },\n    { tag: 'wed' },\n    { tag: 'thu' },\n    { tag: 'fri' },\n    { tag: 'sat' },\n  ];\n\n  hourBlocks: number[] = Array.from({ length: 24 }, (_, i) => i);\n\n  allDayHeight = 60;\n\n  weekDayCols: {\n    date: Date;\n    eventCols: {\n      events: CalendarEvent[];\n    }[];\n    allDayEvents: CalendarEvent[];\n  }[] = [];\n\n  constructor() { }\n\n  ngOnInit(): void {\n    if (this.startAt === 'monday') {\n      this.weekDays = [\n        { tag: 'mon' },\n        { tag: 'tue' },\n        { tag: 'wed' },\n        { tag: 'thu' },\n        { tag: 'fri' },\n        { tag: 'sat' },\n        { tag: 'sun' },\n      ];\n    }\n\n    this.initWeekDayCols();\n  }\n\n  initWeekDayCols() {\n    this.weekDayCols = [];\n    for (let i = 0; i < 7; i++) {\n      const currentDate = new Date(this.startDate.getTime() + i * 24 * 60 * 60 * 1000);\n      this.weekDayCols.push({\n        date: currentDate,\n        eventCols: [],\n        allDayEvents: []\n      });\n    }\n\n    this.events.forEach(event => this.addEvent(event));\n  }\n\n  addEvent(event: CalendarEvent) {\n    const index = this.weekDayCols.findIndex(col =>\n      col.date.getFullYear() === event.start.getFullYear() &&\n      col.date.getMonth() === event.start.getMonth() &&\n      col.date.getDate() === event.start.getDate()\n    );\n\n    if (index === -1) return;\n\n    const eventCols = this.weekDayCols[index].eventCols;\n\n    if (event.allDay) {\n      this.weekDayCols[index].allDayEvents.push(event);\n    } else {\n      const colIndex = this.findAvalailableColumn(event, index);\n\n      if (colIndex === eventCols.length) {\n        eventCols.push({ events: [event] });\n      } else {\n        eventCols[colIndex].events.push(event);\n      }\n    }\n  }\n\n  findAvalailableColumn(event: CalendarEvent, dayIndex: number): number {\n    const eventCols = this.weekDayCols[dayIndex].eventCols;\n\n    for (let colIndex = 0; colIndex < eventCols.length; colIndex++) {\n      const col = eventCols[colIndex];\n\n      // conflict if event overlaps with any existing event in the column\n      const hasConflict = col.events.some(existingEvent => {\n        const startsDuringExisting = event.start < existingEvent.end! && event.start >= existingEvent.start;\n        const endsDuringExisting = event.end! > existingEvent.start && event.end! <= existingEvent.end!;\n        const spansExisting = event.start <= existingEvent.start && event.end! >= existingEvent.end!;\n        return startsDuringExisting || endsDuringExisting || spansExisting;\n      });\n\n      if (!hasConflict) {\n        return colIndex;\n      }\n    }\n    return eventCols.length;\n  }\n\n  ngOnChanges(changes: SimpleChanges): void {\n    if (changes['events'] || changes['startDate']) {\n      this.initWeekDayCols();\n    }\n  }\n\n  private resizing = false;\n  private startY = 0;\n  private startHeight = 0;\n\n  onResizeMouseDown(event: MouseEvent): void {\n    this.resizing = true;\n    this.startY = event.clientY;\n    this.startHeight = this.allDayHeight;\n    document.addEventListener('mousemove', this.onResizeMouseMove);\n    document.addEventListener('mouseup', this.onResizeMouseUp);\n  }\n\n  onResizeMouseMove = (event: MouseEvent): void => {\n    if (!this.resizing) return;\n    const delta = event.clientY - this.startY;\n    const newHeight = this.startHeight + delta;\n    this.allDayHeight = Math.max(60, newHeight); // mínimo 80px\n  };\n\n  onResizeMouseUp = (): void => {\n    this.resizing = false;\n    document.removeEventListener('mousemove', this.onResizeMouseMove);\n    document.removeEventListener('mouseup', this.onResizeMouseUp);\n  };\n\n  onEventClick(event: CalendarEvent): void {\n    this.eventClick.emit(event);\n  }\n\n}\n","<div class=\"week-view-component\">\n    <section class=\"all-day-events\" [style.height.px]=\"allDayHeight\">\n        <div class=\"header\">\n            {{'allDayEvents' | translate}}\n        </div>\n\n        <div class=\"days-container\">\n            <div class=\"events-container\" *ngFor=\"let day of weekDayCols; let i = index\">\n                <div class=\"week-day\">\n                    {{ weekDays[i].tag | translate | slice:0:3 }}\n                </div>\n\n                <lib-calendar-event *ngFor=\"let event of day.allDayEvents\" [event]=\"event\"\n                    (eventClick)=\"onEventClick($event)\"></lib-calendar-event>\n            </div>\n        </div>\n    </section>\n\n    <div class=\"resize-divider\" (mousedown)=\"onResizeMouseDown($event)\"></div>\n\n    <section class=\"duration-events\">\n        <div class=\"hour-blocks\">\n            <div class=\"hour-block\" *ngFor=\"let hour of hourBlocks\">\n                <span>{{ hour }}:00</span>\n            </div>\n        </div>\n\n        <div class=\"events-wrapper\">\n            <div class=\"time-blocks\">\n                <div class=\"day-block\" *ngFor=\"let day of weekDayCols\">\n                    <div class=\"time-block\" *ngFor=\"let hour of hourBlocks\">\n                        <div class=\"time-block-middle\"></div>\n                    </div>\n                </div>\n            </div>\n\n            <div class=\"days-container\">\n                <div class=\"events-container\" *ngFor=\"let day of weekDayCols\"\n                    [style.gridTemplateColumns]=\"'repeat(' + day.eventCols.length + ', 1fr)'\">\n                    <div class=\"event-column\" *ngFor=\"let col of day.eventCols\">\n\n                        <lib-duration-event *ngFor=\"let event of col.events\" [event]=\"event\" [date]=\"day.date\"\n                            (eventClick)=\"onEventClick($event)\"></lib-duration-event>\n\n                    </div>\n                </div>\n            </div>\n        </div>\n    </section>\n</div>"]}
|