@smart-solutions-tech/smart-angular-calendar 0.0.2 → 0.0.4
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/calendar/calendar.component.mjs +9 -7
- package/esm2020/lib/components/day/day.component.mjs +92 -0
- package/esm2020/lib/components/duration-event/duration-event.component.mjs +56 -0
- package/esm2020/lib/components/month-selector/month-selector.component.mjs +82 -0
- package/esm2020/lib/components/more-events-modal/more-events-modal.component.mjs +2 -2
- package/esm2020/lib/components/smart-calendar/smart-calendar.component.mjs +180 -6
- package/esm2020/lib/pipes/translate-month.pipe.mjs +22 -0
- package/esm2020/lib/smart-angular-calendar.module.mjs +15 -3
- package/esm2020/lib/translations.mjs +43 -1
- package/esm2020/lib/types.mjs +1 -1
- package/fesm2015/smart-solutions-tech-smart-angular-calendar.mjs +468 -13
- package/fesm2015/smart-solutions-tech-smart-angular-calendar.mjs.map +1 -1
- package/fesm2020/smart-solutions-tech-smart-angular-calendar.mjs +467 -13
- package/fesm2020/smart-solutions-tech-smart-angular-calendar.mjs.map +1 -1
- package/lib/components/calendar/calendar.component.d.ts +4 -13
- package/lib/components/day/day.component.d.ts +28 -0
- package/lib/components/duration-event/duration-event.component.d.ts +20 -0
- package/lib/components/month-selector/month-selector.component.d.ts +30 -0
- package/lib/components/smart-calendar/smart-calendar.component.d.ts +36 -2
- package/lib/pipes/translate-month.pipe.d.ts +10 -0
- package/lib/smart-angular-calendar.module.d.ts +6 -2
- package/lib/translations.d.ts +14 -0
- package/lib/types.d.ts +14 -0
- package/package.json +1 -1
|
@@ -1,4 +1,4 @@
|
|
|
1
|
-
import { Component,
|
|
1
|
+
import { Component, Input, ViewChild } from '@angular/core';
|
|
2
2
|
import * as i0 from "@angular/core";
|
|
3
3
|
import * as i1 from "@angular/common";
|
|
4
4
|
import * as i2 from "../calendar-event/calendar-event.component";
|
|
@@ -12,7 +12,6 @@ export class CalendarComponent {
|
|
|
12
12
|
};
|
|
13
13
|
this.events = [];
|
|
14
14
|
this.startAt = 'sunday';
|
|
15
|
-
this.monthChange = new EventEmitter();
|
|
16
15
|
this.dayCells = [];
|
|
17
16
|
this.weekDays = [
|
|
18
17
|
{ tag: 'sun' },
|
|
@@ -123,23 +122,26 @@ export class CalendarComponent {
|
|
|
123
122
|
this.setMaxEventsInDayCell();
|
|
124
123
|
});
|
|
125
124
|
}
|
|
125
|
+
ngOnChanges(changes) {
|
|
126
|
+
if (changes['currentMonth']) {
|
|
127
|
+
this.initDaysCells();
|
|
128
|
+
}
|
|
129
|
+
}
|
|
126
130
|
onDestroy() {
|
|
127
131
|
window.removeEventListener('resize', this.resizeListener);
|
|
128
132
|
}
|
|
129
133
|
}
|
|
130
134
|
CalendarComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "14.3.0", ngImport: i0, type: CalendarComponent, deps: [], target: i0.ɵɵFactoryTarget.Component });
|
|
131
|
-
CalendarComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "14.3.0", type: CalendarComponent, selector: "lib-calendar", inputs: { currentMonth: "currentMonth", events: "events", startAt: "startAt" },
|
|
135
|
+
CalendarComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "14.3.0", type: CalendarComponent, selector: "lib-calendar", inputs: { currentMonth: "currentMonth", events: "events", startAt: "startAt" }, viewQueries: [{ propertyName: "calendarRef", first: true, predicate: ["calendar"], descendants: true }, { propertyName: "calendarBodyRef", first: true, predicate: ["calendarBody"], descendants: true }], usesOnChanges: true, ngImport: i0, template: "<div class=\"calendar-component\">\r\n <div class=\"body\" #calendarBody>\r\n <div class=\"week-days\">\r\n <div class=\"week-day\" *ngFor=\"let day of weekDays\">\r\n {{ day.tag | translate | slice:0:3 }}\r\n </div>\r\n </div>\r\n\r\n <div class=\"days-grid\">\r\n <div class=\"day-cell\" *ngFor=\"let day of dayCells; let i = index\"\r\n [ngClass]=\"{'outside-month': day.isOutsideMonth, 'is-today': day.isToday}\">\r\n <div class=\"day-number\">\r\n {{ day.date.getDate() }}\r\n </div>\r\n\r\n <div class=\"events-list\">\r\n <lib-calendar-event *ngFor=\"let event of day.events | slice:0:maxEventsInDayCell\"\r\n [event]=\"event\"></lib-calendar-event>\r\n\r\n <div class=\"more-events\" *ngIf=\"day.events.length > (maxEventsInDayCell || 0)\"\r\n (click)=\"openMoreEventsModal(day.date, day.events)\">\r\n {{'nMoreEvents' | translate: {n:(day.events.length - (maxEventsInDayCell || 0))} }}\r\n </div>\r\n </div>\r\n </div>\r\n </div>\r\n </div>\r\n\r\n <div class=\"footer\">\r\n Footer\r\n </div>\r\n\r\n <lib-more-events-modal *ngIf=\"showMoreEventsModal && moreEventsModalDate\" [events]=\"moreEventsModalEvents\"\r\n [date]=\"moreEventsModalDate\" (close)=\"closeMoreEventsModal()\"\r\n (eventClick)=\"closeMoreEventsModal()\"></lib-more-events-modal>\r\n</div>", styles: ["*{box-sizing:border-box;margin:0;padding:0}.calendar-component{display:flex;flex-direction:column;width:100%;height:100%}.body{flex:1 1 0;display:flex;flex-direction:column;width:100%;min-height:0;height:100%}.week-days{flex-shrink:0;display:grid;grid-template-columns:repeat(7,1fr);width:100%;background-color:#f4f4f4}.week-day{display:flex;align-items:center;justify-content:center;padding:4px;background-color:#fff;font-size:14px;color:#00000080}.days-grid{flex:1 1 0;display:grid;grid-template-columns:repeat(7,1fr);gap:1px;grid-auto-rows:1fr;width:100%;padding:1px;background-color:#f4f4f4;min-height:0;height:100%}.day-cell{display:flex;flex-direction:column;gap:4px;padding:4px;background-color:#fff;overflow:hidden}.day-cell .day-number{display:flex;align-items:center;justify-content:center;width:fit-content;aspect-ratio:1;padding:2px;border-radius:50%;font-size:12px;line-height:16px;font-weight:600;-webkit-user-select:none;user-select:none}.day-cell.outside-month{background-color:#f8f8f8}.day-cell.is-today .day-number{background-color:#007bff;color:#fff}.day-cell .events-list{flex:1 1 auto;display:flex;flex-direction:column;gap:2px;overflow:hidden;min-height:0;max-height:100%}.day-cell .events-list .more-events{padding:2px 8px;border-radius:4px;font-size:12px;color:#555;cursor:pointer}.day-cell .events-list .more-events:hover{background-color:#eee}.footer{flex-shrink:0;background-color:#add8e6}\n"], dependencies: [{ kind: "directive", type: i1.NgClass, selector: "[ngClass]", inputs: ["class", "ngClass"] }, { kind: "directive", type: i1.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "directive", type: i1.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "component", type: i2.CalendarEventComponent, selector: "lib-calendar-event", inputs: ["event"] }, { kind: "component", type: i3.MoreEventsModalComponent, selector: "lib-more-events-modal", inputs: ["events", "date"], outputs: ["close", "eventClick"] }, { kind: "pipe", type: i1.SlicePipe, name: "slice" }, { kind: "pipe", type: i4.TranslatePipe, name: "translate" }] });
|
|
132
136
|
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "14.3.0", ngImport: i0, type: CalendarComponent, decorators: [{
|
|
133
137
|
type: Component,
|
|
134
|
-
args: [{ selector: 'lib-calendar', template: "<div class=\"calendar-component\">\r\n <div class=\"
|
|
138
|
+
args: [{ selector: 'lib-calendar', template: "<div class=\"calendar-component\">\r\n <div class=\"body\" #calendarBody>\r\n <div class=\"week-days\">\r\n <div class=\"week-day\" *ngFor=\"let day of weekDays\">\r\n {{ day.tag | translate | slice:0:3 }}\r\n </div>\r\n </div>\r\n\r\n <div class=\"days-grid\">\r\n <div class=\"day-cell\" *ngFor=\"let day of dayCells; let i = index\"\r\n [ngClass]=\"{'outside-month': day.isOutsideMonth, 'is-today': day.isToday}\">\r\n <div class=\"day-number\">\r\n {{ day.date.getDate() }}\r\n </div>\r\n\r\n <div class=\"events-list\">\r\n <lib-calendar-event *ngFor=\"let event of day.events | slice:0:maxEventsInDayCell\"\r\n [event]=\"event\"></lib-calendar-event>\r\n\r\n <div class=\"more-events\" *ngIf=\"day.events.length > (maxEventsInDayCell || 0)\"\r\n (click)=\"openMoreEventsModal(day.date, day.events)\">\r\n {{'nMoreEvents' | translate: {n:(day.events.length - (maxEventsInDayCell || 0))} }}\r\n </div>\r\n </div>\r\n </div>\r\n </div>\r\n </div>\r\n\r\n <div class=\"footer\">\r\n Footer\r\n </div>\r\n\r\n <lib-more-events-modal *ngIf=\"showMoreEventsModal && moreEventsModalDate\" [events]=\"moreEventsModalEvents\"\r\n [date]=\"moreEventsModalDate\" (close)=\"closeMoreEventsModal()\"\r\n (eventClick)=\"closeMoreEventsModal()\"></lib-more-events-modal>\r\n</div>", styles: ["*{box-sizing:border-box;margin:0;padding:0}.calendar-component{display:flex;flex-direction:column;width:100%;height:100%}.body{flex:1 1 0;display:flex;flex-direction:column;width:100%;min-height:0;height:100%}.week-days{flex-shrink:0;display:grid;grid-template-columns:repeat(7,1fr);width:100%;background-color:#f4f4f4}.week-day{display:flex;align-items:center;justify-content:center;padding:4px;background-color:#fff;font-size:14px;color:#00000080}.days-grid{flex:1 1 0;display:grid;grid-template-columns:repeat(7,1fr);gap:1px;grid-auto-rows:1fr;width:100%;padding:1px;background-color:#f4f4f4;min-height:0;height:100%}.day-cell{display:flex;flex-direction:column;gap:4px;padding:4px;background-color:#fff;overflow:hidden}.day-cell .day-number{display:flex;align-items:center;justify-content:center;width:fit-content;aspect-ratio:1;padding:2px;border-radius:50%;font-size:12px;line-height:16px;font-weight:600;-webkit-user-select:none;user-select:none}.day-cell.outside-month{background-color:#f8f8f8}.day-cell.is-today .day-number{background-color:#007bff;color:#fff}.day-cell .events-list{flex:1 1 auto;display:flex;flex-direction:column;gap:2px;overflow:hidden;min-height:0;max-height:100%}.day-cell .events-list .more-events{padding:2px 8px;border-radius:4px;font-size:12px;color:#555;cursor:pointer}.day-cell .events-list .more-events:hover{background-color:#eee}.footer{flex-shrink:0;background-color:#add8e6}\n"] }]
|
|
135
139
|
}], ctorParameters: function () { return []; }, propDecorators: { currentMonth: [{
|
|
136
140
|
type: Input
|
|
137
141
|
}], events: [{
|
|
138
142
|
type: Input
|
|
139
143
|
}], startAt: [{
|
|
140
144
|
type: Input
|
|
141
|
-
}], monthChange: [{
|
|
142
|
-
type: Output
|
|
143
145
|
}], calendarRef: [{
|
|
144
146
|
type: ViewChild,
|
|
145
147
|
args: ['calendar']
|
|
@@ -147,4 +149,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "14.3.0", ngImpor
|
|
|
147
149
|
type: ViewChild,
|
|
148
150
|
args: ['calendarBody']
|
|
149
151
|
}] } });
|
|
150
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"calendar.component.js","sourceRoot":"","sources":["../../../../../../projects/smart-angular-calendar/src/lib/components/calendar/calendar.component.ts","../../../../../../projects/smart-angular-calendar/src/lib/components/calendar/calendar.component.html"],"names":[],"mappings":"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","sourcesContent":["import { AfterViewInit, Component, ElementRef, EventEmitter, Input, OnInit, Output, ViewChild } from '@angular/core';\r\nimport { CalendarEvent, DayCell, WeekDay } from '../../types';\r\n\r\n@Component({\r\n  selector: 'lib-calendar',\r\n  templateUrl: './calendar.component.html',\r\n  styleUrls: ['./calendar.component.scss']\r\n})\r\nexport class CalendarComponent implements OnInit, AfterViewInit {\r\n\r\n  @Input() currentMonth: { year: number, month: number } = {\r\n    year: new Date().getFullYear(),\r\n    month: new Date().getMonth() + 1\r\n  };\r\n\r\n  @Input() events: CalendarEvent[] = [];\r\n\r\n  @Input() startAt: 'monday' | 'sunday' = 'sunday';\r\n\r\n  @Output() monthChange = new EventEmitter<{\r\n    newMonth: { year: number, month: number },\r\n    oldMonth: { year: number, month: number }\r\n  }>();\r\n\r\n  @ViewChild('calendar') calendarRef!: ElementRef;\r\n  @ViewChild('calendarBody') calendarBodyRef!: ElementRef;\r\n\r\n  calendarWidth?: number;\r\n  columnWidth?: number;\r\n  calendarBodyHeight?: number;\r\n  maxEventsInDayCell?: number;\r\n\r\n  dayCells: DayCell[] = [];\r\n\r\n  weekDays: WeekDay[] = [\r\n    { tag: 'sun' },\r\n    { tag: 'mon' },\r\n    { tag: 'tue' },\r\n    { tag: 'wed' },\r\n    { tag: 'thu' },\r\n    { tag: 'fri' },\r\n    { tag: 'sat' },\r\n  ];\r\n\r\n  showMoreEventsModal: boolean = false;\r\n  moreEventsModalDate?: Date;\r\n  moreEventsModalEvents: CalendarEvent[] = [];\r\n\r\n  constructor() { }\r\n\r\n  isToday(date: Date): boolean {\r\n    const today = new Date();\r\n    return date.getDate() === today.getDate() &&\r\n      date.getMonth() === today.getMonth() &&\r\n      date.getFullYear() === today.getFullYear();\r\n  }\r\n\r\n  getEventsForDay(date: Date): CalendarEvent[] {\r\n    return this.events.filter(event => {\r\n      const eventStart = new Date(event.start);\r\n      const eventEnd = event.end ? new Date(event.end) : eventStart;\r\n\r\n      return date >= new Date(eventStart.getFullYear(), eventStart.getMonth(), eventStart.getDate()) &&\r\n        date <= new Date(eventEnd.getFullYear(), eventEnd.getMonth(), eventEnd.getDate());\r\n    });\r\n  }\r\n\r\n  initDaysCells() {\r\n    const daysInMonth = new Date(this.currentMonth.year, this.currentMonth.month, 0).getDate();\r\n    const firstDayOfWeek = new Date(this.currentMonth.year, this.currentMonth.month - 1, 1).getDay();\r\n    const prevDaysCount = this.startAt === 'monday' ? (firstDayOfWeek === 0 ? 6 : firstDayOfWeek - 1) : firstDayOfWeek;\r\n    // const nextDaysCount = (7 - ((prevDaysCount + daysInMonth) % 7)) % 7;\r\n    const nextDaysCount = 42 - (prevDaysCount + daysInMonth); // Always show 6 weeks\r\n\r\n    this.dayCells = [];\r\n\r\n    for (let i = 0; i < prevDaysCount; i++) {\r\n      const date = new Date(this.currentMonth.year, this.currentMonth.month - 1, i - prevDaysCount + 1);\r\n      const _events = this.getEventsForDay(date);\r\n\r\n      this.dayCells.push({\r\n        date,\r\n        events: _events,\r\n        isOutsideMonth: true,\r\n        isToday: this.isToday(date)\r\n      });\r\n    }\r\n\r\n    for (let day = 1; day <= daysInMonth; day++) {\r\n      const date = new Date(this.currentMonth.year, this.currentMonth.month - 1, day);\r\n      const _events = this.getEventsForDay(date);\r\n\r\n      this.dayCells.push({\r\n        date,\r\n        events: _events,\r\n        isToday: this.isToday(date)\r\n      });\r\n    }\r\n\r\n    for (let i = 1; i <= nextDaysCount; i++) {\r\n      const date = new Date(this.currentMonth.year, this.currentMonth.month - 1, daysInMonth + i);\r\n      const _events = this.getEventsForDay(date);\r\n\r\n      this.dayCells.push({\r\n        date,\r\n        events: _events,\r\n        isOutsideMonth: true,\r\n        isToday: this.isToday(date)\r\n      });\r\n    }\r\n  }\r\n\r\n  setCalendarBodyHeight() {\r\n    this.calendarBodyHeight = this.calendarBodyRef.nativeElement.clientHeight;\r\n  }\r\n\r\n  setMaxEventsInDayCell() {\r\n    const element = document.getElementsByClassName('events-list')[0] as HTMLElement;\r\n    console.log(element);\r\n\r\n    if (element) {\r\n      const eventItemHeight = 22;\r\n      this.maxEventsInDayCell = Math.floor(element.clientHeight / eventItemHeight) - 1;\r\n    }\r\n  }\r\n\r\n  openMoreEventsModal(date: Date, events: CalendarEvent[]) {\r\n    this.moreEventsModalDate = date;\r\n    this.moreEventsModalEvents = events;\r\n    this.showMoreEventsModal = true;\r\n  }\r\n\r\n  closeMoreEventsModal() {\r\n    this.showMoreEventsModal = false;\r\n    this.moreEventsModalDate = undefined;\r\n    this.moreEventsModalEvents = [];\r\n  }\r\n\r\n  resizeListener = () => {\r\n    this.setCalendarBodyHeight();\r\n    this.setMaxEventsInDayCell();\r\n  }\r\n\r\n  ngOnInit(): void {\r\n    if (this.startAt === 'monday') {\r\n      this.weekDays = [\r\n        { tag: 'mon' },\r\n        { tag: 'tue' },\r\n        { tag: 'wed' },\r\n        { tag: 'thu' },\r\n        { tag: 'fri' },\r\n        { tag: 'sat' },\r\n        { tag: 'sun' },\r\n      ];\r\n    }\r\n\r\n    this.initDaysCells();\r\n  }\r\n\r\n  ngAfterViewInit(): void {\r\n    window.addEventListener('resize', this.resizeListener);\r\n\r\n    setTimeout(() => {\r\n      this.setCalendarBodyHeight();\r\n      this.setMaxEventsInDayCell();\r\n    });\r\n  }\r\n\r\n  onDestroy(): void {\r\n    window.removeEventListener('resize', this.resizeListener);\r\n  }\r\n}\r\n","<div class=\"calendar-component\">\r\n    <div class=\"header\">\r\n        Header {{ maxEventsInDayCell }}\r\n    </div>\r\n\r\n    <div class=\"body\" #calendarBody>\r\n        <div class=\"week-days\">\r\n            <div class=\"week-day\" *ngFor=\"let day of weekDays\">\r\n                {{ day.tag | translate | slice:0:3 }}\r\n            </div>\r\n        </div>\r\n\r\n        <div class=\"days-grid\">\r\n            <div class=\"day-cell\" *ngFor=\"let day of dayCells; let i = index\"\r\n                [ngClass]=\"{'outside-month': day.isOutsideMonth, 'is-today': day.isToday}\">\r\n                <div class=\"day-number\">\r\n                    {{ day.date.getDate() }}\r\n                </div>\r\n\r\n                <div class=\"events-list\">\r\n                    <lib-calendar-event *ngFor=\"let event of day.events | slice:0:maxEventsInDayCell\"\r\n                        [event]=\"event\"></lib-calendar-event>\r\n\r\n                    <div class=\"more-events\" *ngIf=\"day.events.length > (maxEventsInDayCell || 0)\"\r\n                        (click)=\"openMoreEventsModal(day.date, day.events)\">\r\n                        {{'nMoreEvents' | translate: {n:(day.events.length - (maxEventsInDayCell || 0))} }}\r\n                    </div>\r\n                </div>\r\n            </div>\r\n        </div>\r\n    </div>\r\n\r\n    <div class=\"footer\">\r\n        Footer\r\n    </div>\r\n\r\n    <lib-more-events-modal *ngIf=\"showMoreEventsModal && moreEventsModalDate\" [events]=\"moreEventsModalEvents\"\r\n        [date]=\"moreEventsModalDate\" (close)=\"closeMoreEventsModal()\"\r\n        (eventClick)=\"closeMoreEventsModal()\"></lib-more-events-modal>\r\n</div>"]}
|
|
152
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"calendar.component.js","sourceRoot":"","sources":["../../../../../../projects/smart-angular-calendar/src/lib/components/calendar/calendar.component.ts","../../../../../../projects/smart-angular-calendar/src/lib/components/calendar/calendar.component.html"],"names":[],"mappings":"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","sourcesContent":["import { AfterViewInit, Component, ElementRef, EventEmitter, HostListener, Input, OnChanges, OnInit, Output, SimpleChanges, ViewChild } from '@angular/core';\r\nimport { CalendarEvent, DayCell, WeekDay } from '../../types';\r\n\r\n@Component({\r\n  selector: 'lib-calendar',\r\n  templateUrl: './calendar.component.html',\r\n  styleUrls: ['./calendar.component.scss']\r\n})\r\nexport class CalendarComponent implements OnInit, AfterViewInit, OnChanges {\r\n\r\n  @Input() currentMonth: { year: number, month: number } = {\r\n    year: new Date().getFullYear(),\r\n    month: new Date().getMonth() + 1\r\n  };\r\n\r\n  @Input() events: CalendarEvent[] = [];\r\n\r\n  @Input() startAt: 'monday' | 'sunday' = 'sunday';\r\n\r\n  @ViewChild('calendar') calendarRef!: ElementRef;\r\n  @ViewChild('calendarBody') calendarBodyRef!: ElementRef;\r\n\r\n  calendarWidth?: number;\r\n  columnWidth?: number;\r\n  calendarBodyHeight?: number;\r\n  maxEventsInDayCell?: number;\r\n\r\n  dayCells: DayCell[] = [];\r\n\r\n  weekDays: WeekDay[] = [\r\n    { tag: 'sun' },\r\n    { tag: 'mon' },\r\n    { tag: 'tue' },\r\n    { tag: 'wed' },\r\n    { tag: 'thu' },\r\n    { tag: 'fri' },\r\n    { tag: 'sat' },\r\n  ];\r\n\r\n  showMoreEventsModal: boolean = false;\r\n  moreEventsModalDate?: Date;\r\n  moreEventsModalEvents: CalendarEvent[] = [];\r\n\r\n  constructor() { }\r\n\r\n  isToday(date: Date): boolean {\r\n    const today = new Date();\r\n    return date.getDate() === today.getDate() &&\r\n      date.getMonth() === today.getMonth() &&\r\n      date.getFullYear() === today.getFullYear();\r\n  }\r\n\r\n  getEventsForDay(date: Date): CalendarEvent[] {\r\n    return this.events.filter(event => {\r\n      const eventStart = new Date(event.start);\r\n      const eventEnd = event.end ? new Date(event.end) : eventStart;\r\n\r\n      return date >= new Date(eventStart.getFullYear(), eventStart.getMonth(), eventStart.getDate()) &&\r\n        date <= new Date(eventEnd.getFullYear(), eventEnd.getMonth(), eventEnd.getDate());\r\n    });\r\n  }\r\n\r\n  initDaysCells() {\r\n    const daysInMonth = new Date(this.currentMonth.year, this.currentMonth.month, 0).getDate();\r\n    const firstDayOfWeek = new Date(this.currentMonth.year, this.currentMonth.month - 1, 1).getDay();\r\n    const prevDaysCount = this.startAt === 'monday' ? (firstDayOfWeek === 0 ? 6 : firstDayOfWeek - 1) : firstDayOfWeek;\r\n    // const nextDaysCount = (7 - ((prevDaysCount + daysInMonth) % 7)) % 7;\r\n    const nextDaysCount = 42 - (prevDaysCount + daysInMonth); // Always show 6 weeks\r\n\r\n    this.dayCells = [];\r\n\r\n    for (let i = 0; i < prevDaysCount; i++) {\r\n      const date = new Date(this.currentMonth.year, this.currentMonth.month - 1, i - prevDaysCount + 1);\r\n      const _events = this.getEventsForDay(date);\r\n\r\n      this.dayCells.push({\r\n        date,\r\n        events: _events,\r\n        isOutsideMonth: true,\r\n        isToday: this.isToday(date)\r\n      });\r\n    }\r\n\r\n    for (let day = 1; day <= daysInMonth; day++) {\r\n      const date = new Date(this.currentMonth.year, this.currentMonth.month - 1, day);\r\n      const _events = this.getEventsForDay(date);\r\n\r\n      this.dayCells.push({\r\n        date,\r\n        events: _events,\r\n        isToday: this.isToday(date)\r\n      });\r\n    }\r\n\r\n    for (let i = 1; i <= nextDaysCount; i++) {\r\n      const date = new Date(this.currentMonth.year, this.currentMonth.month - 1, daysInMonth + i);\r\n      const _events = this.getEventsForDay(date);\r\n\r\n      this.dayCells.push({\r\n        date,\r\n        events: _events,\r\n        isOutsideMonth: true,\r\n        isToday: this.isToday(date)\r\n      });\r\n    }\r\n  }\r\n\r\n  setCalendarBodyHeight() {\r\n    this.calendarBodyHeight = this.calendarBodyRef.nativeElement.clientHeight;\r\n  }\r\n\r\n  setMaxEventsInDayCell() {\r\n    const element = document.getElementsByClassName('events-list')[0] as HTMLElement;\r\n    console.log(element);\r\n\r\n    if (element) {\r\n      const eventItemHeight = 22;\r\n      this.maxEventsInDayCell = Math.floor(element.clientHeight / eventItemHeight) - 1;\r\n    }\r\n  }\r\n\r\n  openMoreEventsModal(date: Date, events: CalendarEvent[]) {\r\n    this.moreEventsModalDate = date;\r\n    this.moreEventsModalEvents = events;\r\n    this.showMoreEventsModal = true;\r\n  }\r\n\r\n  closeMoreEventsModal() {\r\n    this.showMoreEventsModal = false;\r\n    this.moreEventsModalDate = undefined;\r\n    this.moreEventsModalEvents = [];\r\n  }\r\n\r\n  resizeListener = () => {\r\n    this.setCalendarBodyHeight();\r\n    this.setMaxEventsInDayCell();\r\n  }\r\n\r\n  ngOnInit(): void {\r\n    if (this.startAt === 'monday') {\r\n      this.weekDays = [\r\n        { tag: 'mon' },\r\n        { tag: 'tue' },\r\n        { tag: 'wed' },\r\n        { tag: 'thu' },\r\n        { tag: 'fri' },\r\n        { tag: 'sat' },\r\n        { tag: 'sun' },\r\n      ];\r\n    }\r\n\r\n    this.initDaysCells();\r\n  }\r\n\r\n  ngAfterViewInit(): void {\r\n    window.addEventListener('resize', this.resizeListener);\r\n\r\n    setTimeout(() => {\r\n      this.setCalendarBodyHeight();\r\n      this.setMaxEventsInDayCell();\r\n    });\r\n  }\r\n\r\n  ngOnChanges(changes: SimpleChanges): void {\r\n    if (changes['currentMonth']) {\r\n      this.initDaysCells();\r\n    }\r\n  }\r\n\r\n  onDestroy(): void {\r\n    window.removeEventListener('resize', this.resizeListener);\r\n  }\r\n}\r\n","<div class=\"calendar-component\">\r\n    <div class=\"body\" #calendarBody>\r\n        <div class=\"week-days\">\r\n            <div class=\"week-day\" *ngFor=\"let day of weekDays\">\r\n                {{ day.tag | translate | slice:0:3 }}\r\n            </div>\r\n        </div>\r\n\r\n        <div class=\"days-grid\">\r\n            <div class=\"day-cell\" *ngFor=\"let day of dayCells; let i = index\"\r\n                [ngClass]=\"{'outside-month': day.isOutsideMonth, 'is-today': day.isToday}\">\r\n                <div class=\"day-number\">\r\n                    {{ day.date.getDate() }}\r\n                </div>\r\n\r\n                <div class=\"events-list\">\r\n                    <lib-calendar-event *ngFor=\"let event of day.events | slice:0:maxEventsInDayCell\"\r\n                        [event]=\"event\"></lib-calendar-event>\r\n\r\n                    <div class=\"more-events\" *ngIf=\"day.events.length > (maxEventsInDayCell || 0)\"\r\n                        (click)=\"openMoreEventsModal(day.date, day.events)\">\r\n                        {{'nMoreEvents' | translate: {n:(day.events.length - (maxEventsInDayCell || 0))} }}\r\n                    </div>\r\n                </div>\r\n            </div>\r\n        </div>\r\n    </div>\r\n\r\n    <div class=\"footer\">\r\n        Footer\r\n    </div>\r\n\r\n    <lib-more-events-modal *ngIf=\"showMoreEventsModal && moreEventsModalDate\" [events]=\"moreEventsModalEvents\"\r\n        [date]=\"moreEventsModalDate\" (close)=\"closeMoreEventsModal()\"\r\n        (eventClick)=\"closeMoreEventsModal()\"></lib-more-events-modal>\r\n</div>"]}
|
|
@@ -0,0 +1,92 @@
|
|
|
1
|
+
import { Component, Input } 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 DayComponent {
|
|
8
|
+
constructor() {
|
|
9
|
+
this.events = [];
|
|
10
|
+
this.hourBlocks = Array.from({ length: 24 }, (_, i) => i);
|
|
11
|
+
this.allDayEvents = [];
|
|
12
|
+
this.allDayHeight = 60;
|
|
13
|
+
this.eventCols = [];
|
|
14
|
+
this.resizing = false;
|
|
15
|
+
this.startY = 0;
|
|
16
|
+
this.startHeight = 0;
|
|
17
|
+
this.onResizeMouseMove = (event) => {
|
|
18
|
+
if (!this.resizing)
|
|
19
|
+
return;
|
|
20
|
+
const delta = event.clientY - this.startY;
|
|
21
|
+
const newHeight = this.startHeight + delta;
|
|
22
|
+
this.allDayHeight = Math.max(60, newHeight); // mínimo 80px
|
|
23
|
+
};
|
|
24
|
+
this.onResizeMouseUp = () => {
|
|
25
|
+
this.resizing = false;
|
|
26
|
+
document.removeEventListener('mousemove', this.onResizeMouseMove);
|
|
27
|
+
document.removeEventListener('mouseup', this.onResizeMouseUp);
|
|
28
|
+
};
|
|
29
|
+
}
|
|
30
|
+
ngOnInit() {
|
|
31
|
+
this.initEventCols();
|
|
32
|
+
}
|
|
33
|
+
findAvalailableColumn(event) {
|
|
34
|
+
for (let colIndex = 0; colIndex < this.eventCols.length; colIndex++) {
|
|
35
|
+
const col = this.eventCols[colIndex];
|
|
36
|
+
// conflict if event overlaps with any existing event in the column
|
|
37
|
+
const hasConflict = col.events.some(existingEvent => {
|
|
38
|
+
const startsDuringExisting = event.start < existingEvent.end && event.start >= existingEvent.start;
|
|
39
|
+
const endsDuringExisting = event.end > existingEvent.start && event.end <= existingEvent.end;
|
|
40
|
+
const spansExisting = event.start <= existingEvent.start && event.end >= existingEvent.end;
|
|
41
|
+
return startsDuringExisting || endsDuringExisting || spansExisting;
|
|
42
|
+
});
|
|
43
|
+
if (!hasConflict) {
|
|
44
|
+
return colIndex;
|
|
45
|
+
}
|
|
46
|
+
}
|
|
47
|
+
return this.eventCols.length;
|
|
48
|
+
}
|
|
49
|
+
addEventToEventCols(event) {
|
|
50
|
+
const colIndex = this.findAvalailableColumn(event);
|
|
51
|
+
if (colIndex === this.eventCols.length) {
|
|
52
|
+
this.eventCols.push({ events: [event] });
|
|
53
|
+
}
|
|
54
|
+
else {
|
|
55
|
+
this.eventCols[colIndex].events.push(event);
|
|
56
|
+
}
|
|
57
|
+
}
|
|
58
|
+
addEvent(event) {
|
|
59
|
+
if (event.allDay)
|
|
60
|
+
this.allDayEvents.push(event);
|
|
61
|
+
else
|
|
62
|
+
this.addEventToEventCols(event);
|
|
63
|
+
}
|
|
64
|
+
initEventCols() {
|
|
65
|
+
this.eventCols = [];
|
|
66
|
+
this.allDayEvents = [];
|
|
67
|
+
this.events.forEach(event => this.addEvent(event));
|
|
68
|
+
}
|
|
69
|
+
ngOnChanges(changes) {
|
|
70
|
+
if (changes['events']) {
|
|
71
|
+
this.initEventCols();
|
|
72
|
+
}
|
|
73
|
+
}
|
|
74
|
+
onResizeMouseDown(event) {
|
|
75
|
+
this.resizing = true;
|
|
76
|
+
this.startY = event.clientY;
|
|
77
|
+
this.startHeight = this.allDayHeight;
|
|
78
|
+
document.addEventListener('mousemove', this.onResizeMouseMove);
|
|
79
|
+
document.addEventListener('mouseup', this.onResizeMouseUp);
|
|
80
|
+
}
|
|
81
|
+
}
|
|
82
|
+
DayComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "14.3.0", ngImport: i0, type: DayComponent, deps: [], target: i0.ɵɵFactoryTarget.Component });
|
|
83
|
+
DayComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "14.3.0", type: DayComponent, selector: "lib-day", inputs: { events: "events", date: "date" }, usesOnChanges: true, ngImport: i0, template: "<div class=\"day-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=\"events-container\">\n <lib-calendar-event *ngFor=\"let event of allDayEvents\" [event]=\"event\"></lib-calendar-event>\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=\"time-block\" *ngFor=\"let hour of hourBlocks\">\n <div class=\"time-block-middle\"></div>\n </div>\n </div>\n\n <div class=\"events-container\" [style.gridTemplateColumns]=\"'repeat(' + eventCols.length + ', 1fr)'\">\n <div class=\"event-column\" *ngFor=\"let col of eventCols\">\n\n <lib-duration-event *ngFor=\"let event of col.events\" [event]=\"event\"\n [date]=\"date\"></lib-duration-event>\n\n </div>\n </div>\n </div>\n </section>\n</div>", styles: ["*{box-sizing:border-box;margin:0;padding:0}.day-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:auto}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 .events-container{display:grid;grid-template-columns:repeat(auto-fill,minmax(200px,1fr));gap:4px;padding:4px}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:flex;flex-direction:column}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 .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"] }, { kind: "component", type: i3.DurationEventComponent, selector: "lib-duration-event", inputs: ["event", "date"] }, { kind: "pipe", type: i4.TranslatePipe, name: "translate" }] });
|
|
84
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "14.3.0", ngImport: i0, type: DayComponent, decorators: [{
|
|
85
|
+
type: Component,
|
|
86
|
+
args: [{ selector: 'lib-day', template: "<div class=\"day-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=\"events-container\">\n <lib-calendar-event *ngFor=\"let event of allDayEvents\" [event]=\"event\"></lib-calendar-event>\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=\"time-block\" *ngFor=\"let hour of hourBlocks\">\n <div class=\"time-block-middle\"></div>\n </div>\n </div>\n\n <div class=\"events-container\" [style.gridTemplateColumns]=\"'repeat(' + eventCols.length + ', 1fr)'\">\n <div class=\"event-column\" *ngFor=\"let col of eventCols\">\n\n <lib-duration-event *ngFor=\"let event of col.events\" [event]=\"event\"\n [date]=\"date\"></lib-duration-event>\n\n </div>\n </div>\n </div>\n </section>\n</div>", styles: ["*{box-sizing:border-box;margin:0;padding:0}.day-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:auto}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 .events-container{display:grid;grid-template-columns:repeat(auto-fill,minmax(200px,1fr));gap:4px;padding:4px}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:flex;flex-direction:column}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 .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"] }]
|
|
87
|
+
}], ctorParameters: function () { return []; }, propDecorators: { events: [{
|
|
88
|
+
type: Input
|
|
89
|
+
}], date: [{
|
|
90
|
+
type: Input
|
|
91
|
+
}] } });
|
|
92
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"day.component.js","sourceRoot":"","sources":["../../../../../../projects/smart-angular-calendar/src/lib/components/day/day.component.ts","../../../../../../projects/smart-angular-calendar/src/lib/components/day/day.component.html"],"names":[],"mappings":"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","sourcesContent":["import { Component, Input, OnChanges, OnInit, SimpleChanges } from '@angular/core';\nimport { CalendarEvent } from '../../types';\n\n@Component({\n  selector: 'lib-day',\n  templateUrl: './day.component.html',\n  styleUrls: ['./day.component.scss']\n})\nexport class DayComponent implements OnInit, OnChanges {\n\n  @Input() events: CalendarEvent[] = [];\n  @Input() date!: Date;\n\n  hourBlocks: number[] = Array.from({ length: 24 }, (_, i) => i);\n\n  allDayEvents: CalendarEvent[] = [];\n  allDayHeight = 60;\n\n  eventCols: {\n    events: CalendarEvent[];\n  }[] = [];\n\n  constructor() { }\n\n  ngOnInit(): void {\n    this.initEventCols();\n  }\n\n  findAvalailableColumn(event: CalendarEvent): number {\n    for (let colIndex = 0; colIndex < this.eventCols.length; colIndex++) {\n      const col = this.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 this.eventCols.length;\n  }\n\n  addEventToEventCols(event: CalendarEvent): void {\n    const colIndex = this.findAvalailableColumn(event);\n    if (colIndex === this.eventCols.length) {\n      this.eventCols.push({ events: [event] });\n    } else {\n      this.eventCols[colIndex].events.push(event);\n    }\n  }\n\n  addEvent(event: CalendarEvent): void {\n    if (event.allDay) this.allDayEvents.push(event);\n    else this.addEventToEventCols(event);\n  }\n\n  initEventCols(): void {\n    this.eventCols = [];\n    this.allDayEvents = [];\n    this.events.forEach(event => this.addEvent(event));\n  }\n\n  ngOnChanges(changes: SimpleChanges): void {\n    if (changes['events']) {\n      this.initEventCols();\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","<div class=\"day-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=\"events-container\">\n            <lib-calendar-event *ngFor=\"let event of allDayEvents\" [event]=\"event\"></lib-calendar-event>\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=\"time-block\" *ngFor=\"let hour of hourBlocks\">\n                    <div class=\"time-block-middle\"></div>\n                </div>\n            </div>\n\n            <div class=\"events-container\" [style.gridTemplateColumns]=\"'repeat(' + eventCols.length + ', 1fr)'\">\n                <div class=\"event-column\" *ngFor=\"let col of eventCols\">\n\n                    <lib-duration-event *ngFor=\"let event of col.events\" [event]=\"event\"\n                        [date]=\"date\"></lib-duration-event>\n\n                </div>\n            </div>\n        </div>\n    </section>\n</div>"]}
|
|
@@ -0,0 +1,56 @@
|
|
|
1
|
+
import { Component, Input } from '@angular/core';
|
|
2
|
+
import { isDarkColor } from '../../utils/colors';
|
|
3
|
+
import * as i0 from "@angular/core";
|
|
4
|
+
import * as i1 from "@angular/common";
|
|
5
|
+
export class DurationEventComponent {
|
|
6
|
+
constructor() {
|
|
7
|
+
this.isDarkColor = false;
|
|
8
|
+
this.topPosition = 0;
|
|
9
|
+
this.height = 0;
|
|
10
|
+
this.isEventStartInThisDay = false;
|
|
11
|
+
this.isEventEndInThisDay = false;
|
|
12
|
+
}
|
|
13
|
+
ngOnInit() {
|
|
14
|
+
this.isDarkColor = isDarkColor(this.event.color || '#000000');
|
|
15
|
+
this.isEventStartInThisDay = this.getIsEventStartInThisDay();
|
|
16
|
+
this.isEventEndInThisDay = this.getIsEventEndInThisDay();
|
|
17
|
+
this.topPosition = this.getTopPosition();
|
|
18
|
+
this.height = this.getEventHeight();
|
|
19
|
+
}
|
|
20
|
+
getIsEventStartInThisDay() {
|
|
21
|
+
return this.event.start.getDate() === this.date.getDate() && this.event.start.getMonth() === this.date.getMonth() && this.event.start.getFullYear() === this.date.getFullYear();
|
|
22
|
+
}
|
|
23
|
+
getIsEventEndInThisDay() {
|
|
24
|
+
return this.event.end.getDate() === this.date.getDate() && this.event.end.getMonth() === this.date.getMonth() && this.event.end.getFullYear() === this.date.getFullYear();
|
|
25
|
+
}
|
|
26
|
+
getTopPosition() {
|
|
27
|
+
if (!this.isEventStartInThisDay)
|
|
28
|
+
return 0;
|
|
29
|
+
return this.event.start.getHours() * 60 + this.event.start.getMinutes();
|
|
30
|
+
}
|
|
31
|
+
getEventHeight() {
|
|
32
|
+
const max_minutes_in_day = 24 * 60;
|
|
33
|
+
if (!this.isEventStartInThisDay && !this.isEventEndInThisDay)
|
|
34
|
+
return max_minutes_in_day;
|
|
35
|
+
const result = (this.event.end.getTime() - this.event.start.getTime()) / (1000 * 60);
|
|
36
|
+
if (result < 20)
|
|
37
|
+
return 20;
|
|
38
|
+
if (!this.isEventStartInThisDay)
|
|
39
|
+
return (this.event.end.getTime() - new Date(this.date.getFullYear(), this.date.getMonth(), this.date.getDate(), 0, 0, 0).getTime()) / (1000 * 60);
|
|
40
|
+
const startPosition = this.getTopPosition();
|
|
41
|
+
if (startPosition + result > max_minutes_in_day)
|
|
42
|
+
return max_minutes_in_day - startPosition;
|
|
43
|
+
return result;
|
|
44
|
+
}
|
|
45
|
+
}
|
|
46
|
+
DurationEventComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "14.3.0", ngImport: i0, type: DurationEventComponent, deps: [], target: i0.ɵɵFactoryTarget.Component });
|
|
47
|
+
DurationEventComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "14.3.0", type: DurationEventComponent, selector: "lib-duration-event", inputs: { event: "event", date: "date" }, ngImport: i0, template: "<div class=\"duration-event-component\" style=\"--color: {{ event.color }};\" [style.top.px]=\"topPosition\"\n [style.height.px]=\"height\" [ngClass]=\"{'dark-color': isDarkColor}\" [title]=\"event.title\">\n <div class=\"event-title\">{{ event.title }}</div>\n</div>", styles: [".duration-event-component{position:absolute;left:0;z-index:1;width:100%;background-color:var(--color, rgba(0, 0, 0, .05));border-radius:4px;padding:2px 8px;cursor:pointer;-webkit-user-select:none;user-select:none;color:#212121}.duration-event-component.dark-color{color:#fafafa}.event-title{font-size:12px;font-weight:500;white-space:nowrap;overflow:hidden;text-overflow:ellipsis}\n"], dependencies: [{ kind: "directive", type: i1.NgClass, selector: "[ngClass]", inputs: ["class", "ngClass"] }] });
|
|
48
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "14.3.0", ngImport: i0, type: DurationEventComponent, decorators: [{
|
|
49
|
+
type: Component,
|
|
50
|
+
args: [{ selector: 'lib-duration-event', template: "<div class=\"duration-event-component\" style=\"--color: {{ event.color }};\" [style.top.px]=\"topPosition\"\n [style.height.px]=\"height\" [ngClass]=\"{'dark-color': isDarkColor}\" [title]=\"event.title\">\n <div class=\"event-title\">{{ event.title }}</div>\n</div>", styles: [".duration-event-component{position:absolute;left:0;z-index:1;width:100%;background-color:var(--color, rgba(0, 0, 0, .05));border-radius:4px;padding:2px 8px;cursor:pointer;-webkit-user-select:none;user-select:none;color:#212121}.duration-event-component.dark-color{color:#fafafa}.event-title{font-size:12px;font-weight:500;white-space:nowrap;overflow:hidden;text-overflow:ellipsis}\n"] }]
|
|
51
|
+
}], ctorParameters: function () { return []; }, propDecorators: { event: [{
|
|
52
|
+
type: Input
|
|
53
|
+
}], date: [{
|
|
54
|
+
type: Input
|
|
55
|
+
}] } });
|
|
56
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiZHVyYXRpb24tZXZlbnQuY29tcG9uZW50LmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vLi4vLi4vcHJvamVjdHMvc21hcnQtYW5ndWxhci1jYWxlbmRhci9zcmMvbGliL2NvbXBvbmVudHMvZHVyYXRpb24tZXZlbnQvZHVyYXRpb24tZXZlbnQuY29tcG9uZW50LnRzIiwiLi4vLi4vLi4vLi4vLi4vLi4vcHJvamVjdHMvc21hcnQtYW5ndWxhci1jYWxlbmRhci9zcmMvbGliL2NvbXBvbmVudHMvZHVyYXRpb24tZXZlbnQvZHVyYXRpb24tZXZlbnQuY29tcG9uZW50Lmh0bWwiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsT0FBTyxFQUFFLFNBQVMsRUFBRSxLQUFLLEVBQVUsTUFBTSxlQUFlLENBQUM7QUFFekQsT0FBTyxFQUFFLFdBQVcsRUFBRSxNQUFNLG9CQUFvQixDQUFDOzs7QUFPakQsTUFBTSxPQUFPLHNCQUFzQjtJQWFqQztRQVJBLGdCQUFXLEdBQVksS0FBSyxDQUFDO1FBRTdCLGdCQUFXLEdBQVcsQ0FBQyxDQUFDO1FBQ3hCLFdBQU0sR0FBVyxDQUFDLENBQUM7UUFFbkIsMEJBQXFCLEdBQVksS0FBSyxDQUFDO1FBQ3ZDLHdCQUFtQixHQUFZLEtBQUssQ0FBQztJQUVyQixDQUFDO0lBRWpCLFFBQVE7UUFDTixJQUFJLENBQUMsV0FBVyxHQUFHLFdBQVcsQ0FBQyxJQUFJLENBQUMsS0FBSyxDQUFDLEtBQUssSUFBSSxTQUFTLENBQUMsQ0FBQztRQUM5RCxJQUFJLENBQUMscUJBQXFCLEdBQUcsSUFBSSxDQUFDLHdCQUF3QixFQUFFLENBQUM7UUFDN0QsSUFBSSxDQUFDLG1CQUFtQixHQUFHLElBQUksQ0FBQyxzQkFBc0IsRUFBRSxDQUFDO1FBQ3pELElBQUksQ0FBQyxXQUFXLEdBQUcsSUFBSSxDQUFDLGNBQWMsRUFBRSxDQUFDO1FBQ3pDLElBQUksQ0FBQyxNQUFNLEdBQUcsSUFBSSxDQUFDLGNBQWMsRUFBRSxDQUFDO0lBQ3RDLENBQUM7SUFFRCx3QkFBd0I7UUFDdEIsT0FBTyxJQUFJLENBQUMsS0FBSyxDQUFDLEtBQUssQ0FBQyxPQUFPLEVBQUUsS0FBSyxJQUFJLENBQUMsSUFBSSxDQUFDLE9BQU8sRUFBRSxJQUFJLElBQUksQ0FBQyxLQUFLLENBQUMsS0FBSyxDQUFDLFFBQVEsRUFBRSxLQUFLLElBQUksQ0FBQyxJQUFJLENBQUMsUUFBUSxFQUFFLElBQUksSUFBSSxDQUFDLEtBQUssQ0FBQyxLQUFLLENBQUMsV0FBVyxFQUFFLEtBQUssSUFBSSxDQUFDLElBQUksQ0FBQyxXQUFXLEVBQUUsQ0FBQztJQUNsTCxDQUFDO0lBRUQsc0JBQXNCO1FBQ3BCLE9BQU8sSUFBSSxDQUFDLEtBQUssQ0FBQyxHQUFJLENBQUMsT0FBTyxFQUFFLEtBQUssSUFBSSxDQUFDLElBQUksQ0FBQyxPQUFPLEVBQUUsSUFBSSxJQUFJLENBQUMsS0FBSyxDQUFDLEdBQUksQ0FBQyxRQUFRLEVBQUUsS0FBSyxJQUFJLENBQUMsSUFBSSxDQUFDLFFBQVEsRUFBRSxJQUFJLElBQUksQ0FBQyxLQUFLLENBQUMsR0FBSSxDQUFDLFdBQVcsRUFBRSxLQUFLLElBQUksQ0FBQyxJQUFJLENBQUMsV0FBVyxFQUFFLENBQUM7SUFDL0ssQ0FBQztJQUVELGNBQWM7UUFDWixJQUFJLENBQUMsSUFBSSxDQUFDLHFCQUFxQjtZQUFFLE9BQU8sQ0FBQyxDQUFDO1FBRTFDLE9BQU8sSUFBSSxDQUFDLEtBQUssQ0FBQyxLQUFLLENBQUMsUUFBUSxFQUFFLEdBQUcsRUFBRSxHQUFHLElBQUksQ0FBQyxLQUFLLENBQUMsS0FBSyxDQUFDLFVBQVUsRUFBRSxDQUFDO0lBQzFFLENBQUM7SUFFRCxjQUFjO1FBQ1osTUFBTSxrQkFBa0IsR0FBRyxFQUFFLEdBQUcsRUFBRSxDQUFDO1FBRW5DLElBQUksQ0FBQyxJQUFJLENBQUMscUJBQXFCLElBQUksQ0FBQyxJQUFJLENBQUMsbUJBQW1CO1lBQUUsT0FBTyxrQkFBa0IsQ0FBQztRQUV4RixNQUFNLE1BQU0sR0FBRyxDQUFDLElBQUksQ0FBQyxLQUFLLENBQUMsR0FBSSxDQUFDLE9BQU8sRUFBRSxHQUFHLElBQUksQ0FBQyxLQUFLLENBQUMsS0FBSyxDQUFDLE9BQU8sRUFBRSxDQUFDLEdBQUcsQ0FBQyxJQUFJLEdBQUcsRUFBRSxDQUFDLENBQUM7UUFFdEYsSUFBSSxNQUFNLEdBQUcsRUFBRTtZQUFFLE9BQU8sRUFBRSxDQUFDO1FBRTNCLElBQUksQ0FBQyxJQUFJLENBQUMscUJBQXFCO1lBQUUsT0FBTyxDQUFDLElBQUksQ0FBQyxLQUFLLENBQUMsR0FBSSxDQUFDLE9BQU8sRUFBRSxHQUFHLElBQUksSUFBSSxDQUFDLElBQUksQ0FBQyxJQUFJLENBQUMsV0FBVyxFQUFFLEVBQUUsSUFBSSxDQUFDLElBQUksQ0FBQyxRQUFRLEVBQUUsRUFBRSxJQUFJLENBQUMsSUFBSSxDQUFDLE9BQU8sRUFBRSxFQUFFLENBQUMsRUFBRSxDQUFDLEVBQUUsQ0FBQyxDQUFDLENBQUMsT0FBTyxFQUFFLENBQUMsR0FBRyxDQUFDLElBQUksR0FBRyxFQUFFLENBQUMsQ0FBQztRQUVwTCxNQUFNLGFBQWEsR0FBRyxJQUFJLENBQUMsY0FBYyxFQUFFLENBQUM7UUFFNUMsSUFBSSxhQUFhLEdBQUcsTUFBTSxHQUFHLGtCQUFrQjtZQUFFLE9BQU8sa0JBQWtCLEdBQUcsYUFBYSxDQUFDO1FBRTNGLE9BQU8sTUFBTSxDQUFDO0lBQ2hCLENBQUM7O21IQXJEVSxzQkFBc0I7dUdBQXRCLHNCQUFzQixvR0NUbkMsbVJBR007MkZETU8sc0JBQXNCO2tCQUxsQyxTQUFTOytCQUNFLG9CQUFvQjswRUFNckIsS0FBSztzQkFBYixLQUFLO2dCQUNHLElBQUk7c0JBQVosS0FBSyIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IENvbXBvbmVudCwgSW5wdXQsIE9uSW5pdCB9IGZyb20gJ0Bhbmd1bGFyL2NvcmUnO1xuaW1wb3J0IHsgQ2FsZW5kYXJFdmVudCB9IGZyb20gJy4uLy4uL3R5cGVzJztcbmltcG9ydCB7IGlzRGFya0NvbG9yIH0gZnJvbSAnLi4vLi4vdXRpbHMvY29sb3JzJztcblxuQENvbXBvbmVudCh7XG4gIHNlbGVjdG9yOiAnbGliLWR1cmF0aW9uLWV2ZW50JyxcbiAgdGVtcGxhdGVVcmw6ICcuL2R1cmF0aW9uLWV2ZW50LmNvbXBvbmVudC5odG1sJyxcbiAgc3R5bGVVcmxzOiBbJy4vZHVyYXRpb24tZXZlbnQuY29tcG9uZW50LnNjc3MnXVxufSlcbmV4cG9ydCBjbGFzcyBEdXJhdGlvbkV2ZW50Q29tcG9uZW50IGltcGxlbWVudHMgT25Jbml0IHtcblxuICBASW5wdXQoKSBldmVudCE6IENhbGVuZGFyRXZlbnQ7XG4gIEBJbnB1dCgpIGRhdGUhOiBEYXRlO1xuXG4gIGlzRGFya0NvbG9yOiBib29sZWFuID0gZmFsc2U7XG5cbiAgdG9wUG9zaXRpb246IG51bWJlciA9IDA7XG4gIGhlaWdodDogbnVtYmVyID0gMDtcblxuICBpc0V2ZW50U3RhcnRJblRoaXNEYXk6IGJvb2xlYW4gPSBmYWxzZTtcbiAgaXNFdmVudEVuZEluVGhpc0RheTogYm9vbGVhbiA9IGZhbHNlO1xuXG4gIGNvbnN0cnVjdG9yKCkgeyB9XG5cbiAgbmdPbkluaXQoKTogdm9pZCB7XG4gICAgdGhpcy5pc0RhcmtDb2xvciA9IGlzRGFya0NvbG9yKHRoaXMuZXZlbnQuY29sb3IgfHwgJyMwMDAwMDAnKTtcbiAgICB0aGlzLmlzRXZlbnRTdGFydEluVGhpc0RheSA9IHRoaXMuZ2V0SXNFdmVudFN0YXJ0SW5UaGlzRGF5KCk7XG4gICAgdGhpcy5pc0V2ZW50RW5kSW5UaGlzRGF5ID0gdGhpcy5nZXRJc0V2ZW50RW5kSW5UaGlzRGF5KCk7XG4gICAgdGhpcy50b3BQb3NpdGlvbiA9IHRoaXMuZ2V0VG9wUG9zaXRpb24oKTtcbiAgICB0aGlzLmhlaWdodCA9IHRoaXMuZ2V0RXZlbnRIZWlnaHQoKTtcbiAgfVxuXG4gIGdldElzRXZlbnRTdGFydEluVGhpc0RheSgpOiBib29sZWFuIHtcbiAgICByZXR1cm4gdGhpcy5ldmVudC5zdGFydC5nZXREYXRlKCkgPT09IHRoaXMuZGF0ZS5nZXREYXRlKCkgJiYgdGhpcy5ldmVudC5zdGFydC5nZXRNb250aCgpID09PSB0aGlzLmRhdGUuZ2V0TW9udGgoKSAmJiB0aGlzLmV2ZW50LnN0YXJ0LmdldEZ1bGxZZWFyKCkgPT09IHRoaXMuZGF0ZS5nZXRGdWxsWWVhcigpO1xuICB9XG5cbiAgZ2V0SXNFdmVudEVuZEluVGhpc0RheSgpOiBib29sZWFuIHtcbiAgICByZXR1cm4gdGhpcy5ldmVudC5lbmQhLmdldERhdGUoKSA9PT0gdGhpcy5kYXRlLmdldERhdGUoKSAmJiB0aGlzLmV2ZW50LmVuZCEuZ2V0TW9udGgoKSA9PT0gdGhpcy5kYXRlLmdldE1vbnRoKCkgJiYgdGhpcy5ldmVudC5lbmQhLmdldEZ1bGxZZWFyKCkgPT09IHRoaXMuZGF0ZS5nZXRGdWxsWWVhcigpO1xuICB9XG5cbiAgZ2V0VG9wUG9zaXRpb24oKTogbnVtYmVyIHtcbiAgICBpZiAoIXRoaXMuaXNFdmVudFN0YXJ0SW5UaGlzRGF5KSByZXR1cm4gMDtcblxuICAgIHJldHVybiB0aGlzLmV2ZW50LnN0YXJ0LmdldEhvdXJzKCkgKiA2MCArIHRoaXMuZXZlbnQuc3RhcnQuZ2V0TWludXRlcygpO1xuICB9XG5cbiAgZ2V0RXZlbnRIZWlnaHQoKTogbnVtYmVyIHtcbiAgICBjb25zdCBtYXhfbWludXRlc19pbl9kYXkgPSAyNCAqIDYwO1xuXG4gICAgaWYgKCF0aGlzLmlzRXZlbnRTdGFydEluVGhpc0RheSAmJiAhdGhpcy5pc0V2ZW50RW5kSW5UaGlzRGF5KSByZXR1cm4gbWF4X21pbnV0ZXNfaW5fZGF5O1xuXG4gICAgY29uc3QgcmVzdWx0ID0gKHRoaXMuZXZlbnQuZW5kIS5nZXRUaW1lKCkgLSB0aGlzLmV2ZW50LnN0YXJ0LmdldFRpbWUoKSkgLyAoMTAwMCAqIDYwKTtcblxuICAgIGlmIChyZXN1bHQgPCAyMCkgcmV0dXJuIDIwO1xuXG4gICAgaWYgKCF0aGlzLmlzRXZlbnRTdGFydEluVGhpc0RheSkgcmV0dXJuICh0aGlzLmV2ZW50LmVuZCEuZ2V0VGltZSgpIC0gbmV3IERhdGUodGhpcy5kYXRlLmdldEZ1bGxZZWFyKCksIHRoaXMuZGF0ZS5nZXRNb250aCgpLCB0aGlzLmRhdGUuZ2V0RGF0ZSgpLCAwLCAwLCAwKS5nZXRUaW1lKCkpIC8gKDEwMDAgKiA2MCk7XG5cbiAgICBjb25zdCBzdGFydFBvc2l0aW9uID0gdGhpcy5nZXRUb3BQb3NpdGlvbigpO1xuXG4gICAgaWYgKHN0YXJ0UG9zaXRpb24gKyByZXN1bHQgPiBtYXhfbWludXRlc19pbl9kYXkpIHJldHVybiBtYXhfbWludXRlc19pbl9kYXkgLSBzdGFydFBvc2l0aW9uO1xuXG4gICAgcmV0dXJuIHJlc3VsdDtcbiAgfVxuXG59XG4iLCI8ZGl2IGNsYXNzPVwiZHVyYXRpb24tZXZlbnQtY29tcG9uZW50XCIgc3R5bGU9XCItLWNvbG9yOiB7eyBldmVudC5jb2xvciB9fTtcIiBbc3R5bGUudG9wLnB4XT1cInRvcFBvc2l0aW9uXCJcbiAgICBbc3R5bGUuaGVpZ2h0LnB4XT1cImhlaWdodFwiIFtuZ0NsYXNzXT1cInsnZGFyay1jb2xvcic6IGlzRGFya0NvbG9yfVwiIFt0aXRsZV09XCJldmVudC50aXRsZVwiPlxuICAgIDxkaXYgY2xhc3M9XCJldmVudC10aXRsZVwiPnt7IGV2ZW50LnRpdGxlIH19PC9kaXY+XG48L2Rpdj4iXX0=
|
|
@@ -0,0 +1,82 @@
|
|
|
1
|
+
import { Component, EventEmitter, Input, Output } from '@angular/core';
|
|
2
|
+
import { MONTHS } from '../../translations';
|
|
3
|
+
import * as i0 from "@angular/core";
|
|
4
|
+
import * as i1 from "@angular/common";
|
|
5
|
+
import * as i2 from "../../pipes/translate-month.pipe";
|
|
6
|
+
export class MonthSelectorComponent {
|
|
7
|
+
constructor() {
|
|
8
|
+
this.monthSelected = new EventEmitter();
|
|
9
|
+
this.mode = 'decade';
|
|
10
|
+
this.currentYear = new Date().getFullYear();
|
|
11
|
+
this.currentMonth = new Date().getMonth() + 1;
|
|
12
|
+
this.selectedMonth = { year: new Date().getFullYear(), month: new Date().getMonth() + 1 };
|
|
13
|
+
this.selectedYear = new Date().getFullYear();
|
|
14
|
+
this.currentDecadeStart = Math.floor(this.selectedYear / 10) * 10;
|
|
15
|
+
this.years = [];
|
|
16
|
+
this.months = Object.keys(MONTHS).map(key => Number(key));
|
|
17
|
+
}
|
|
18
|
+
ngOnInit() {
|
|
19
|
+
setTimeout(() => {
|
|
20
|
+
this.selectedYear = this.selectedMonth.year;
|
|
21
|
+
this.selectDecadeByYear(this.selectedYear);
|
|
22
|
+
this.setYears();
|
|
23
|
+
});
|
|
24
|
+
}
|
|
25
|
+
toggleMode(event) {
|
|
26
|
+
event.stopPropagation();
|
|
27
|
+
this.mode = this.mode === 'year' ? 'decade' : 'year';
|
|
28
|
+
}
|
|
29
|
+
setYears() {
|
|
30
|
+
this.years = [];
|
|
31
|
+
for (let i = this.currentDecadeStart; i < this.currentDecadeStart + 10; i++) {
|
|
32
|
+
this.years.push(i);
|
|
33
|
+
}
|
|
34
|
+
}
|
|
35
|
+
;
|
|
36
|
+
next(event) {
|
|
37
|
+
event.stopPropagation();
|
|
38
|
+
if (this.mode === 'year') {
|
|
39
|
+
this.selectedYear++;
|
|
40
|
+
}
|
|
41
|
+
else {
|
|
42
|
+
this.currentDecadeStart += 10;
|
|
43
|
+
this.setYears();
|
|
44
|
+
this.selectedYear = this.currentDecadeStart;
|
|
45
|
+
}
|
|
46
|
+
}
|
|
47
|
+
previous(event) {
|
|
48
|
+
event.stopPropagation();
|
|
49
|
+
if (this.mode === 'year') {
|
|
50
|
+
this.selectedYear--;
|
|
51
|
+
}
|
|
52
|
+
else {
|
|
53
|
+
this.currentDecadeStart -= 10;
|
|
54
|
+
this.setYears();
|
|
55
|
+
this.selectedYear = this.currentDecadeStart;
|
|
56
|
+
}
|
|
57
|
+
}
|
|
58
|
+
selectYear(event, year) {
|
|
59
|
+
event.stopPropagation();
|
|
60
|
+
this.selectedYear = year;
|
|
61
|
+
this.mode = 'year';
|
|
62
|
+
}
|
|
63
|
+
selectDecadeByYear(year) {
|
|
64
|
+
this.currentDecadeStart = Math.floor(year / 10) * 10;
|
|
65
|
+
this.setYears();
|
|
66
|
+
this.mode = 'decade';
|
|
67
|
+
}
|
|
68
|
+
selectMonth(month) {
|
|
69
|
+
this.monthSelected.emit({ year: this.selectedYear, month });
|
|
70
|
+
}
|
|
71
|
+
}
|
|
72
|
+
MonthSelectorComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "14.3.0", ngImport: i0, type: MonthSelectorComponent, deps: [], target: i0.ɵɵFactoryTarget.Component });
|
|
73
|
+
MonthSelectorComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "14.3.0", type: MonthSelectorComponent, selector: "lib-month-selector", inputs: { selectedMonth: "selectedMonth" }, outputs: { monthSelected: "monthSelected" }, ngImport: i0, template: "<div class=\"month-selector-component\">\n <div class=\"header\">\n <button class=\"btn-range\" *ngIf=\"mode === 'decade'\" (click)=\"toggleMode($event)\">\n <span>{{currentDecadeStart}} - {{currentDecadeStart + 9}}</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 <button class=\"btn-range\" *ngIf=\"mode === 'year'\" (click)=\"toggleMode($event)\">\n <span>{{selectedYear}}</span>\n\n <svg xmlns=\"http://www.w3.org/2000/svg\" style=\"transform: rotate(180deg);\" height=\"24px\"\n viewBox=\"0 -960 960 960\" width=\"24px\" fill=\"currentColor\">\n <path d=\"M480-360 280-560h400L480-360Z\" />\n </svg>\n </button>\n\n <div class=\"nav-controls\">\n <button class=\"btn-prev\" (click)=\"previous($event)\">\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-next\" (click)=\"next($event)\">\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 </div>\n\n <div class=\"years-grid\" *ngIf=\"mode === 'decade'\">\n <button class=\"btn-year\" *ngFor=\"let year of years\"\n [ngClass]=\"{selected: year === selectedYear, current: year === currentYear }\"\n (click)=\"selectYear($event, year)\">\n {{ year }}\n </button>\n </div>\n\n <div class=\"months-grid\" *ngIf=\"mode === 'year'\">\n <button class=\"btn-month\" *ngFor=\"let month of months\"\n [ngClass]=\"{selected: month === selectedMonth.month && selectedYear === selectedMonth.year, current: currentYear === selectedYear && month === currentMonth}\"\n (click)=\"selectMonth(month)\">\n {{ month | translateMonth | slice:0:3 }}\n </button>\n </div>\n</div>", styles: [".month-selector-component{display:flex;flex-direction:column;width:280px;-webkit-user-select:none;user-select:none}.header{display:flex;align-items:center;justify-content:space-between;padding:8px 16px}.btn-range{display:flex;align-items:center;height:32px;padding:0 4px 0 12px;background:none;border:none;border-radius:50px;cursor:pointer;font-weight:600}.btn-range:hover{background-color:#0000000a}.btn-range span{white-space:nowrap}.nav-controls{display:flex;align-items:center}.btn-prev,.btn-next{display:flex;align-items:center;justify-content:center;width:32px;height:32px;background:none;border:none;border-radius:50%;cursor:pointer}.btn-prev:hover,.btn-next:hover{background-color:#0000000a}.years-grid,.months-grid{display:grid;grid-template-columns:repeat(3,1fr);gap:8px;padding:0 16px 16px}.btn-month,.btn-year{display:flex;align-items:center;justify-content:center;padding:6px 12px;border:none;background:none;border-radius:50px;font-size:14px;cursor:pointer}.btn-month:hover,.btn-year:hover{background-color:#0000000a}.btn-month.selected,.btn-year.selected{border:1px solid rgba(0,0,0,.12)}.btn-month.current,.btn-year.current{background-color:#0000001f}\n"], dependencies: [{ kind: "directive", type: i1.NgClass, selector: "[ngClass]", inputs: ["class", "ngClass"] }, { kind: "directive", type: i1.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "directive", type: i1.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "pipe", type: i1.SlicePipe, name: "slice" }, { kind: "pipe", type: i2.TranslateMonthPipe, name: "translateMonth" }] });
|
|
74
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "14.3.0", ngImport: i0, type: MonthSelectorComponent, decorators: [{
|
|
75
|
+
type: Component,
|
|
76
|
+
args: [{ selector: 'lib-month-selector', template: "<div class=\"month-selector-component\">\n <div class=\"header\">\n <button class=\"btn-range\" *ngIf=\"mode === 'decade'\" (click)=\"toggleMode($event)\">\n <span>{{currentDecadeStart}} - {{currentDecadeStart + 9}}</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 <button class=\"btn-range\" *ngIf=\"mode === 'year'\" (click)=\"toggleMode($event)\">\n <span>{{selectedYear}}</span>\n\n <svg xmlns=\"http://www.w3.org/2000/svg\" style=\"transform: rotate(180deg);\" height=\"24px\"\n viewBox=\"0 -960 960 960\" width=\"24px\" fill=\"currentColor\">\n <path d=\"M480-360 280-560h400L480-360Z\" />\n </svg>\n </button>\n\n <div class=\"nav-controls\">\n <button class=\"btn-prev\" (click)=\"previous($event)\">\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-next\" (click)=\"next($event)\">\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 </div>\n\n <div class=\"years-grid\" *ngIf=\"mode === 'decade'\">\n <button class=\"btn-year\" *ngFor=\"let year of years\"\n [ngClass]=\"{selected: year === selectedYear, current: year === currentYear }\"\n (click)=\"selectYear($event, year)\">\n {{ year }}\n </button>\n </div>\n\n <div class=\"months-grid\" *ngIf=\"mode === 'year'\">\n <button class=\"btn-month\" *ngFor=\"let month of months\"\n [ngClass]=\"{selected: month === selectedMonth.month && selectedYear === selectedMonth.year, current: currentYear === selectedYear && month === currentMonth}\"\n (click)=\"selectMonth(month)\">\n {{ month | translateMonth | slice:0:3 }}\n </button>\n </div>\n</div>", styles: [".month-selector-component{display:flex;flex-direction:column;width:280px;-webkit-user-select:none;user-select:none}.header{display:flex;align-items:center;justify-content:space-between;padding:8px 16px}.btn-range{display:flex;align-items:center;height:32px;padding:0 4px 0 12px;background:none;border:none;border-radius:50px;cursor:pointer;font-weight:600}.btn-range:hover{background-color:#0000000a}.btn-range span{white-space:nowrap}.nav-controls{display:flex;align-items:center}.btn-prev,.btn-next{display:flex;align-items:center;justify-content:center;width:32px;height:32px;background:none;border:none;border-radius:50%;cursor:pointer}.btn-prev:hover,.btn-next:hover{background-color:#0000000a}.years-grid,.months-grid{display:grid;grid-template-columns:repeat(3,1fr);gap:8px;padding:0 16px 16px}.btn-month,.btn-year{display:flex;align-items:center;justify-content:center;padding:6px 12px;border:none;background:none;border-radius:50px;font-size:14px;cursor:pointer}.btn-month:hover,.btn-year:hover{background-color:#0000000a}.btn-month.selected,.btn-year.selected{border:1px solid rgba(0,0,0,.12)}.btn-month.current,.btn-year.current{background-color:#0000001f}\n"] }]
|
|
77
|
+
}], ctorParameters: function () { return []; }, propDecorators: { monthSelected: [{
|
|
78
|
+
type: Output
|
|
79
|
+
}], selectedMonth: [{
|
|
80
|
+
type: Input
|
|
81
|
+
}] } });
|
|
82
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"month-selector.component.js","sourceRoot":"","sources":["../../../../../../projects/smart-angular-calendar/src/lib/components/month-selector/month-selector.component.ts","../../../../../../projects/smart-angular-calendar/src/lib/components/month-selector/month-selector.component.html"],"names":[],"mappings":"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","sourcesContent":["import { Component, EventEmitter, Input, OnInit, Output } from '@angular/core';\nimport { MONTHS } from '../../translations';\n\n@Component({\n  selector: 'lib-month-selector',\n  templateUrl: './month-selector.component.html',\n  styleUrls: ['./month-selector.component.scss']\n})\nexport class MonthSelectorComponent implements OnInit {\n\n  @Output() monthSelected = new EventEmitter<{ year: number, month: number }>();\n\n  mode: 'year' | 'decade' = 'decade';\n  currentYear: number = new Date().getFullYear();\n  currentMonth: number = new Date().getMonth() + 1;\n  @Input() selectedMonth: { year: number, month: number } = { year: new Date().getFullYear(), month: new Date().getMonth() + 1 };\n\n  selectedYear: number = new Date().getFullYear();\n  currentDecadeStart: number = Math.floor(this.selectedYear / 10) * 10;\n\n  years: number[] = [];\n  months = Object.keys(MONTHS).map(key => Number(key));\n\n  constructor() { }\n\n  ngOnInit(): void {\n    setTimeout(() => {\n      this.selectedYear = this.selectedMonth.year;\n\n      this.selectDecadeByYear(this.selectedYear);\n      this.setYears();\n    });\n  }\n\n  toggleMode(event: MouseEvent): void {\n    event.stopPropagation();\n    this.mode = this.mode === 'year' ? 'decade' : 'year';\n  }\n\n  setYears() {\n    this.years = [];\n    for (let i = this.currentDecadeStart; i < this.currentDecadeStart + 10; i++) {\n      this.years.push(i);\n    }\n  };\n\n  next(event: MouseEvent) {\n    event.stopPropagation();\n\n    if (this.mode === 'year') {\n      this.selectedYear++;\n    } else {\n      this.currentDecadeStart += 10;\n      this.setYears();\n      this.selectedYear = this.currentDecadeStart;\n    }\n  }\n\n  previous(event: MouseEvent) {\n    event.stopPropagation();\n\n    if (this.mode === 'year') {\n      this.selectedYear--;\n    } else {\n      this.currentDecadeStart -= 10;\n      this.setYears();\n      this.selectedYear = this.currentDecadeStart;\n    }\n  }\n\n  selectYear(event: MouseEvent, year: number): void {\n    event.stopPropagation();\n\n    this.selectedYear = year;\n    this.mode = 'year';\n  }\n\n  selectDecadeByYear(year: number): void {\n    this.currentDecadeStart = Math.floor(year / 10) * 10;\n    this.setYears();\n    this.mode = 'decade';\n  }\n\n  selectMonth(month: number): void {\n    this.monthSelected.emit({ year: this.selectedYear, month });\n  }\n\n}\n","<div class=\"month-selector-component\">\n    <div class=\"header\">\n        <button class=\"btn-range\" *ngIf=\"mode === 'decade'\" (click)=\"toggleMode($event)\">\n            <span>{{currentDecadeStart}} - {{currentDecadeStart + 9}}</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        <button class=\"btn-range\" *ngIf=\"mode === 'year'\" (click)=\"toggleMode($event)\">\n            <span>{{selectedYear}}</span>\n\n            <svg xmlns=\"http://www.w3.org/2000/svg\" style=\"transform: rotate(180deg);\" height=\"24px\"\n                viewBox=\"0 -960 960 960\" width=\"24px\" fill=\"currentColor\">\n                <path d=\"M480-360 280-560h400L480-360Z\" />\n            </svg>\n        </button>\n\n        <div class=\"nav-controls\">\n            <button class=\"btn-prev\" (click)=\"previous($event)\">\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-next\" (click)=\"next($event)\">\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    </div>\n\n    <div class=\"years-grid\" *ngIf=\"mode === 'decade'\">\n        <button class=\"btn-year\" *ngFor=\"let year of years\"\n            [ngClass]=\"{selected: year === selectedYear, current: year === currentYear }\"\n            (click)=\"selectYear($event, year)\">\n            {{ year }}\n        </button>\n    </div>\n\n    <div class=\"months-grid\" *ngIf=\"mode === 'year'\">\n        <button class=\"btn-month\" *ngFor=\"let month of months\"\n            [ngClass]=\"{selected: month === selectedMonth.month && selectedYear === selectedMonth.year, current: currentYear === selectedYear && month === currentMonth}\"\n            (click)=\"selectMonth(month)\">\n            {{ month | translateMonth | slice:0:3 }}\n        </button>\n    </div>\n</div>"]}
|
|
@@ -22,10 +22,10 @@ export class MoreEventsModalComponent {
|
|
|
22
22
|
}
|
|
23
23
|
}
|
|
24
24
|
MoreEventsModalComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "14.3.0", ngImport: i0, type: MoreEventsModalComponent, deps: [], target: i0.ɵɵFactoryTarget.Component });
|
|
25
|
-
MoreEventsModalComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "14.3.0", type: MoreEventsModalComponent, selector: "lib-more-events-modal", inputs: { events: "events", date: "date" }, outputs: { close: "close", eventClick: "eventClick" }, ngImport: i0, template: "<div class=\"more-events-modal-component\">\n <div class=\"backdrop\" (click)=\"onClose()\"></div>\n\n <div class=\"modal\">\n <div class=\"header\">\n <div class=\"title\">{{'moreEvents' | translate}}</div>\n <div class=\"subtitle\">{{date.toLocaleDateString()}}</div>\n\n <button class=\"btn-close\" (click)=\"onClose()\">\n <svg xmlns=\"http://www.w3.org/2000/svg\" width=\"24\" height=\"24\" viewBox=\"0 0 24 24\" fill=\"none\"\n stroke=\"currentColor\" stroke-width=\"2\" stroke-linecap=\"round\" stroke-linejoin=\"round\"\n class=\"feather feather-x\" fill=\"yellow\">\n <line x1=\"18\" y1=\"6\" x2=\"6\" y2=\"18\"></line>\n <line x1=\"6\" y1=\"6\" x2=\"18\" y2=\"18\"></line>\n </svg>\n </button>\n </div>\n\n <div class=\"events-list\">\n <div class=\"event-item\" *ngFor=\"let event of events\" style=\"--color: {{event.color}};\"\n [ngClass]=\"{'is-dark':isDarkColor(event.color)}\" (click)=\"onEventClick(event)\">\n <div class=\"event-title\">{{ event.title }}</div>\n </div>\n </div>\n </div>\n</div>", styles: [".more-events-modal-component{position:absolute;top:0;left:0;z-index:1;display:flex;align-items:center;justify-content:center;width:
|
|
25
|
+
MoreEventsModalComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "14.3.0", type: MoreEventsModalComponent, selector: "lib-more-events-modal", inputs: { events: "events", date: "date" }, outputs: { close: "close", eventClick: "eventClick" }, ngImport: i0, template: "<div class=\"more-events-modal-component\">\n <div class=\"backdrop\" (click)=\"onClose()\"></div>\n\n <div class=\"modal\">\n <div class=\"header\">\n <div class=\"title\">{{'moreEvents' | translate}}</div>\n <div class=\"subtitle\">{{date.toLocaleDateString()}}</div>\n\n <button class=\"btn-close\" (click)=\"onClose()\">\n <svg xmlns=\"http://www.w3.org/2000/svg\" width=\"24\" height=\"24\" viewBox=\"0 0 24 24\" fill=\"none\"\n stroke=\"currentColor\" stroke-width=\"2\" stroke-linecap=\"round\" stroke-linejoin=\"round\"\n class=\"feather feather-x\" fill=\"yellow\">\n <line x1=\"18\" y1=\"6\" x2=\"6\" y2=\"18\"></line>\n <line x1=\"6\" y1=\"6\" x2=\"18\" y2=\"18\"></line>\n </svg>\n </button>\n </div>\n\n <div class=\"events-list\">\n <div class=\"event-item\" *ngFor=\"let event of events\" style=\"--color: {{event.color}};\"\n [ngClass]=\"{'is-dark':isDarkColor(event.color)}\" (click)=\"onEventClick(event)\">\n <div class=\"event-title\">{{ event.title }}</div>\n </div>\n </div>\n </div>\n</div>", styles: [".more-events-modal-component{position:absolute;top:0;left:0;z-index:1;display:flex;align-items:center;justify-content:center;width:100%;height:100%;box-shadow:0 2px 8px #00000026}.more-events-modal-component .backdrop{position:absolute;top:0;left:0;z-index:1;width:100%;height:100%;background-color:#00000080;cursor:pointer}.modal{z-index:2;display:flex;flex-direction:column;max-width:400px;width:100%;max-height:80%;margin:24px;background-color:#fff;border-radius:8px}.header{position:relative;display:flex;flex-direction:column;padding:16px 64px 16px 16px}.title{font-size:18px;font-weight:700}.subtitle{font-size:13px;color:#666}.btn-close{position:absolute;top:8px;right:8px;display:flex;align-items:center;justify-content:center;width:36px;height:36px;background:none;border:none;border-radius:50%;cursor:pointer}.btn-close:hover{background-color:#f0f0f0}.events-list{flex:1 1 auto;display:flex;flex-direction:column;gap:4px;padding:0 16px 16px;overflow-y:auto}.events-list .event-item{display:flex;padding:8px;background-color:var(--color, #e2e2e2);border-radius:4px;color:#212121;cursor:pointer;-webkit-user-select:none;user-select:none}.events-list .event-item.is-dark{color:#fafafa}\n"], dependencies: [{ kind: "directive", type: i1.NgClass, selector: "[ngClass]", inputs: ["class", "ngClass"] }, { kind: "directive", type: i1.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "pipe", type: i2.TranslatePipe, name: "translate" }] });
|
|
26
26
|
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "14.3.0", ngImport: i0, type: MoreEventsModalComponent, decorators: [{
|
|
27
27
|
type: Component,
|
|
28
|
-
args: [{ selector: 'lib-more-events-modal', template: "<div class=\"more-events-modal-component\">\n <div class=\"backdrop\" (click)=\"onClose()\"></div>\n\n <div class=\"modal\">\n <div class=\"header\">\n <div class=\"title\">{{'moreEvents' | translate}}</div>\n <div class=\"subtitle\">{{date.toLocaleDateString()}}</div>\n\n <button class=\"btn-close\" (click)=\"onClose()\">\n <svg xmlns=\"http://www.w3.org/2000/svg\" width=\"24\" height=\"24\" viewBox=\"0 0 24 24\" fill=\"none\"\n stroke=\"currentColor\" stroke-width=\"2\" stroke-linecap=\"round\" stroke-linejoin=\"round\"\n class=\"feather feather-x\" fill=\"yellow\">\n <line x1=\"18\" y1=\"6\" x2=\"6\" y2=\"18\"></line>\n <line x1=\"6\" y1=\"6\" x2=\"18\" y2=\"18\"></line>\n </svg>\n </button>\n </div>\n\n <div class=\"events-list\">\n <div class=\"event-item\" *ngFor=\"let event of events\" style=\"--color: {{event.color}};\"\n [ngClass]=\"{'is-dark':isDarkColor(event.color)}\" (click)=\"onEventClick(event)\">\n <div class=\"event-title\">{{ event.title }}</div>\n </div>\n </div>\n </div>\n</div>", styles: [".more-events-modal-component{position:absolute;top:0;left:0;z-index:1;display:flex;align-items:center;justify-content:center;width:
|
|
28
|
+
args: [{ selector: 'lib-more-events-modal', template: "<div class=\"more-events-modal-component\">\n <div class=\"backdrop\" (click)=\"onClose()\"></div>\n\n <div class=\"modal\">\n <div class=\"header\">\n <div class=\"title\">{{'moreEvents' | translate}}</div>\n <div class=\"subtitle\">{{date.toLocaleDateString()}}</div>\n\n <button class=\"btn-close\" (click)=\"onClose()\">\n <svg xmlns=\"http://www.w3.org/2000/svg\" width=\"24\" height=\"24\" viewBox=\"0 0 24 24\" fill=\"none\"\n stroke=\"currentColor\" stroke-width=\"2\" stroke-linecap=\"round\" stroke-linejoin=\"round\"\n class=\"feather feather-x\" fill=\"yellow\">\n <line x1=\"18\" y1=\"6\" x2=\"6\" y2=\"18\"></line>\n <line x1=\"6\" y1=\"6\" x2=\"18\" y2=\"18\"></line>\n </svg>\n </button>\n </div>\n\n <div class=\"events-list\">\n <div class=\"event-item\" *ngFor=\"let event of events\" style=\"--color: {{event.color}};\"\n [ngClass]=\"{'is-dark':isDarkColor(event.color)}\" (click)=\"onEventClick(event)\">\n <div class=\"event-title\">{{ event.title }}</div>\n </div>\n </div>\n </div>\n</div>", styles: [".more-events-modal-component{position:absolute;top:0;left:0;z-index:1;display:flex;align-items:center;justify-content:center;width:100%;height:100%;box-shadow:0 2px 8px #00000026}.more-events-modal-component .backdrop{position:absolute;top:0;left:0;z-index:1;width:100%;height:100%;background-color:#00000080;cursor:pointer}.modal{z-index:2;display:flex;flex-direction:column;max-width:400px;width:100%;max-height:80%;margin:24px;background-color:#fff;border-radius:8px}.header{position:relative;display:flex;flex-direction:column;padding:16px 64px 16px 16px}.title{font-size:18px;font-weight:700}.subtitle{font-size:13px;color:#666}.btn-close{position:absolute;top:8px;right:8px;display:flex;align-items:center;justify-content:center;width:36px;height:36px;background:none;border:none;border-radius:50%;cursor:pointer}.btn-close:hover{background-color:#f0f0f0}.events-list{flex:1 1 auto;display:flex;flex-direction:column;gap:4px;padding:0 16px 16px;overflow-y:auto}.events-list .event-item{display:flex;padding:8px;background-color:var(--color, #e2e2e2);border-radius:4px;color:#212121;cursor:pointer;-webkit-user-select:none;user-select:none}.events-list .event-item.is-dark{color:#fafafa}\n"] }]
|
|
29
29
|
}], ctorParameters: function () { return []; }, propDecorators: { events: [{
|
|
30
30
|
type: Input
|
|
31
31
|
}], date: [{
|