@mintplayer/ng-bootstrap 13.1.24 → 13.2.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (141) hide show
  1. package/esm2020/lib/components/accordion/accordion/accordion.component.mjs +3 -3
  2. package/esm2020/lib/components/accordion/accordion-tab/accordion-tab.component.mjs +3 -3
  3. package/esm2020/lib/components/accordion/accordion-tab-header/accordion-tab-header.component.mjs +3 -3
  4. package/esm2020/lib/components/accordion/accordion.module.mjs +4 -4
  5. package/esm2020/lib/components/alert/alert/alert.component.mjs +3 -3
  6. package/esm2020/lib/components/alert/alert-close/alert-close.component.mjs +3 -3
  7. package/esm2020/lib/components/alert/alert.module.mjs +4 -4
  8. package/esm2020/lib/components/calendar/calendar.component.mjs +3 -3
  9. package/esm2020/lib/components/calendar/calendar.module.mjs +4 -4
  10. package/esm2020/lib/components/card/card/card.component.mjs +3 -3
  11. package/esm2020/lib/components/card/card-header/card-header.component.mjs +3 -3
  12. package/esm2020/lib/components/card/card.module.mjs +4 -4
  13. package/esm2020/lib/components/carousel/carousel/carousel.component.mjs +3 -3
  14. package/esm2020/lib/components/carousel/carousel-image/carousel-image.directive.mjs +3 -3
  15. package/esm2020/lib/components/carousel/carousel.module.mjs +4 -4
  16. package/esm2020/lib/components/code-snippet/code-snippet.component.mjs +3 -3
  17. package/esm2020/lib/components/code-snippet/code-snippet.module.mjs +4 -4
  18. package/esm2020/lib/components/context-menu/context-menu.directive.mjs +4 -5
  19. package/esm2020/lib/components/context-menu/context-menu.module.mjs +4 -4
  20. package/esm2020/lib/components/copy/copy.directive.mjs +3 -3
  21. package/esm2020/lib/components/copy/copy.module.mjs +4 -4
  22. package/esm2020/lib/components/datatable/datatable/datatable.component.mjs +3 -3
  23. package/esm2020/lib/components/datatable/datatable-column/datatable-column.directive.mjs +3 -3
  24. package/esm2020/lib/components/datatable/datatable.module.mjs +4 -4
  25. package/esm2020/lib/components/datatable/row-template/row-template.directive.mjs +3 -3
  26. package/esm2020/lib/components/datepicker/datepicker.component.mjs +3 -3
  27. package/esm2020/lib/components/datepicker/datepicker.module.mjs +4 -4
  28. package/esm2020/lib/components/dropdown/dropdown/dropdown.directive.mjs +3 -3
  29. package/esm2020/lib/components/dropdown/dropdown-menu/dropdown-menu.directive.mjs +3 -3
  30. package/esm2020/lib/components/dropdown/dropdown-toggle/dropdown-toggle.directive.mjs +3 -3
  31. package/esm2020/lib/components/dropdown/dropdown.module.mjs +4 -4
  32. package/esm2020/lib/components/file-upload/component/file-upload.component.mjs +4 -5
  33. package/esm2020/lib/components/file-upload/directive/file-upload-template.directive.mjs +3 -3
  34. package/esm2020/lib/components/file-upload/file-upload.module.mjs +4 -4
  35. package/esm2020/lib/components/for/for.directive.mjs +3 -3
  36. package/esm2020/lib/components/for/for.module.mjs +4 -4
  37. package/esm2020/lib/components/index.mjs +2 -1
  38. package/esm2020/lib/components/list-group/list-group/list-group.component.mjs +3 -3
  39. package/esm2020/lib/components/list-group/list-group-item/list-group-item.component.mjs +3 -3
  40. package/esm2020/lib/components/list-group/list-group.module.mjs +4 -4
  41. package/esm2020/lib/components/modal/component/modal/modal.component.mjs +3 -3
  42. package/esm2020/lib/components/modal/component/modal-content/modal-content.component.mjs +3 -3
  43. package/esm2020/lib/components/modal/directives/modal-body/modal-body.directive.mjs +3 -3
  44. package/esm2020/lib/components/modal/directives/modal-footer/modal-footer.directive.mjs +3 -3
  45. package/esm2020/lib/components/modal/directives/modal-header/modal-header.directive.mjs +3 -3
  46. package/esm2020/lib/components/modal/modal.module.mjs +4 -4
  47. package/esm2020/lib/components/modal/service/modal.service.mjs +4 -5
  48. package/esm2020/lib/components/multiselect/component/multiselect.component.mjs +3 -3
  49. package/esm2020/lib/components/multiselect/directives/button-template/button-template.directive.mjs +3 -3
  50. package/esm2020/lib/components/multiselect/directives/footer-template/footer-template.directive.mjs +3 -3
  51. package/esm2020/lib/components/multiselect/directives/header-template/header-template.directive.mjs +3 -3
  52. package/esm2020/lib/components/multiselect/multiselect.module.mjs +4 -4
  53. package/esm2020/lib/components/navbar/dropdown-toggle/dropdown-toggle.directive.mjs +3 -3
  54. package/esm2020/lib/components/navbar/expand-button/expand-button.directive.mjs +3 -3
  55. package/esm2020/lib/components/navbar/nav-link/nav-link.directive.mjs +3 -3
  56. package/esm2020/lib/components/navbar/navbar/navbar.component.mjs +3 -3
  57. package/esm2020/lib/components/navbar/navbar-brand/navbar-brand.component.mjs +3 -3
  58. package/esm2020/lib/components/navbar/navbar-content/navbar-content.directive.mjs +3 -3
  59. package/esm2020/lib/components/navbar/navbar-dropdown/navbar-dropdown.component.mjs +3 -3
  60. package/esm2020/lib/components/navbar/navbar-item/navbar-item.component.mjs +3 -3
  61. package/esm2020/lib/components/navbar/navbar-nav/navbar-nav.component.mjs +3 -3
  62. package/esm2020/lib/components/navbar/navbar-toggler/navbar-toggler.component.mjs +3 -3
  63. package/esm2020/lib/components/navbar/navbar.module.mjs +4 -4
  64. package/esm2020/lib/components/pagination/pagination/pagination.component.mjs +3 -3
  65. package/esm2020/lib/components/pagination/pagination.module.mjs +4 -4
  66. package/esm2020/lib/components/progress-bar/progress/progress.component.mjs +3 -3
  67. package/esm2020/lib/components/progress-bar/progress-bar/progress-bar.component.mjs +3 -3
  68. package/esm2020/lib/components/progress-bar/progress-bar.module.mjs +4 -4
  69. package/esm2020/lib/components/rating/rating.component.mjs +3 -3
  70. package/esm2020/lib/components/rating/rating.module.mjs +4 -4
  71. package/esm2020/lib/components/scheduler/components/scheduler/scheduler.component.mjs +234 -74
  72. package/esm2020/lib/components/scheduler/interfaces/drag-operation.mjs +1 -1
  73. package/esm2020/lib/components/scheduler/interfaces/preview-event.mjs +2 -0
  74. package/esm2020/lib/components/scheduler/interfaces/scheduler-event-part.mjs +1 -1
  75. package/esm2020/lib/components/scheduler/interfaces/timeline-track.mjs +2 -0
  76. package/esm2020/lib/components/scheduler/pipes/bs-seconds-timespan.pipe/bs-seconds-timespan.pipe.mjs +3 -3
  77. package/esm2020/lib/components/scheduler/pipes/bs-seconds-today-offset/bs-seconds-today-offset.pipe.mjs +3 -3
  78. package/esm2020/lib/components/scheduler/pipes/date-offset/date-offset.pipe.mjs +3 -3
  79. package/esm2020/lib/components/scheduler/pipes/day-of-week/day-of-week.pipe.mjs +3 -3
  80. package/esm2020/lib/components/scheduler/scheduler.module.mjs +4 -4
  81. package/esm2020/lib/components/scheduler/services/timeline/timeline.service.mjs +65 -0
  82. package/esm2020/lib/components/scrollspy/component/scrollspy.component.mjs +3 -3
  83. package/esm2020/lib/components/scrollspy/directives/scrollspy.directive.mjs +3 -3
  84. package/esm2020/lib/components/scrollspy/scrollspy.module.mjs +4 -4
  85. package/esm2020/lib/components/select2/component/select2.component.mjs +3 -3
  86. package/esm2020/lib/components/select2/directive/item-template.directive.mjs +3 -3
  87. package/esm2020/lib/components/select2/select2.module.mjs +4 -4
  88. package/esm2020/lib/components/snackbar/component/snackbar.component.mjs +3 -3
  89. package/esm2020/lib/components/snackbar/directives/snackbar-close/snackbar-close.directive.mjs +3 -3
  90. package/esm2020/lib/components/snackbar/service/snackbar.service.mjs +3 -3
  91. package/esm2020/lib/components/snackbar/snackbar.module.mjs +4 -4
  92. package/esm2020/lib/components/tab-control/tab-control/tab-control.component.mjs +3 -3
  93. package/esm2020/lib/components/tab-control/tab-control.module.mjs +4 -4
  94. package/esm2020/lib/components/tab-control/tab-page/tab-page.component.mjs +3 -3
  95. package/esm2020/lib/components/timepicker/index.mjs +3 -0
  96. package/esm2020/lib/components/timepicker/timepicker.component.mjs +108 -0
  97. package/esm2020/lib/components/timepicker/timepicker.module.mjs +38 -0
  98. package/esm2020/lib/components/toggle-button/toggle-button.component.mjs +3 -3
  99. package/esm2020/lib/components/toggle-button/toggle-button.module.mjs +4 -4
  100. package/esm2020/lib/components/tooltip/component/tooltip.component.mjs +3 -3
  101. package/esm2020/lib/components/tooltip/directive/tooltip.directive.mjs +3 -3
  102. package/esm2020/lib/components/tooltip/tooltip.module.mjs +4 -4
  103. package/esm2020/lib/components/typeahead/typeahead.component.mjs +3 -3
  104. package/esm2020/lib/components/typeahead/typeahead.module.mjs +4 -4
  105. package/esm2020/lib/directives/enhanced-paste/enhanced-paste.directive.mjs +58 -0
  106. package/esm2020/lib/directives/enhanced-paste/enhanced-paste.module.mjs +26 -0
  107. package/esm2020/lib/interfaces/index.mjs +2 -1
  108. package/esm2020/lib/interfaces/number-overflow.mjs +2 -0
  109. package/esm2020/lib/pipes/font-color/font-color.module.mjs +4 -4
  110. package/esm2020/lib/pipes/font-color/font-color.pipe.mjs +3 -3
  111. package/esm2020/lib/pipes/format-bytes/format-bytes.module.mjs +4 -4
  112. package/esm2020/lib/pipes/format-bytes/format-bytes.pipe.mjs +3 -3
  113. package/esm2020/lib/pipes/in-list/in-list.module.mjs +4 -4
  114. package/esm2020/lib/pipes/in-list/in-list.pipe.mjs +3 -3
  115. package/esm2020/lib/pipes/month-name/month-name.module.mjs +4 -4
  116. package/esm2020/lib/pipes/month-name/month-name.pipe.mjs +3 -3
  117. package/esm2020/lib/pipes/uc-first/uc-first.module.mjs +4 -4
  118. package/esm2020/lib/pipes/uc-first/uc-first.pipe.mjs +3 -3
  119. package/esm2020/lib/pipes/weekday-name/weekday-name.module.mjs +4 -4
  120. package/esm2020/lib/pipes/weekday-name/weekday-name.pipe.mjs +3 -3
  121. package/esm2020/lib/services/calendar-month/calendar-month.service.mjs +3 -3
  122. package/esm2020/lib/services/scroll-offset/scroll-offset.service.mjs +3 -3
  123. package/fesm2015/mintplayer-ng-bootstrap.mjs +855 -430
  124. package/fesm2015/mintplayer-ng-bootstrap.mjs.map +1 -1
  125. package/fesm2020/mintplayer-ng-bootstrap.mjs +855 -430
  126. package/fesm2020/mintplayer-ng-bootstrap.mjs.map +1 -1
  127. package/lib/components/index.d.ts +1 -0
  128. package/lib/components/scheduler/components/scheduler/scheduler.component.d.ts +28 -5
  129. package/lib/components/scheduler/interfaces/drag-operation.d.ts +1 -0
  130. package/lib/components/scheduler/interfaces/preview-event.d.ts +4 -0
  131. package/lib/components/scheduler/interfaces/scheduler-event-part.d.ts +1 -1
  132. package/lib/components/scheduler/interfaces/timeline-track.d.ts +5 -0
  133. package/lib/components/scheduler/services/timeline/timeline.service.d.ts +14 -0
  134. package/lib/components/timepicker/index.d.ts +2 -0
  135. package/lib/components/timepicker/timepicker.component.d.ts +21 -0
  136. package/lib/components/timepicker/timepicker.module.d.ts +11 -0
  137. package/lib/directives/enhanced-paste/enhanced-paste.directive.d.ts +14 -0
  138. package/lib/directives/enhanced-paste/enhanced-paste.module.d.ts +8 -0
  139. package/lib/interfaces/index.d.ts +1 -0
  140. package/lib/interfaces/number-overflow.d.ts +5 -0
  141. package/package.json +12 -12
@@ -1,24 +1,33 @@
1
1
  import { Component, EventEmitter, HostListener, Input, Output, QueryList, ViewChildren } from '@angular/core';
2
- import { BehaviorSubject, combineLatest, map, Subject, take, takeUntil } from 'rxjs';
2
+ import { BehaviorSubject, combineLatest, filter, map, Subject, take, takeUntil } from 'rxjs';
3
3
  import { BsCalendarMonthService } from '../../../../services/calendar-month/calendar-month.service';
4
4
  import { EDragOperation } from '../../enums/drag-operation';
5
+ import { BsTimelineService } from '../../services/timeline/timeline.service';
5
6
  import * as i0 from "@angular/core";
6
7
  import * as i1 from "../../../../services/calendar-month/calendar-month.service";
7
- import * as i2 from "@angular/common";
8
- import * as i3 from "../../pipes/bs-seconds-today-offset/bs-seconds-today-offset.pipe";
9
- import * as i4 from "../../pipes/bs-seconds-timespan.pipe/bs-seconds-timespan.pipe";
10
- import * as i5 from "../../pipes/day-of-week/day-of-week.pipe";
8
+ import * as i2 from "../../services/timeline/timeline.service";
9
+ import * as i3 from "@angular/common";
10
+ import * as i4 from "../../pipes/bs-seconds-today-offset/bs-seconds-today-offset.pipe";
11
+ import * as i5 from "../../pipes/bs-seconds-timespan.pipe/bs-seconds-timespan.pipe";
12
+ import * as i6 from "../../pipes/day-of-week/day-of-week.pipe";
11
13
  export class BsSchedulerComponent {
12
- constructor(calendarMonthService) {
14
+ constructor(calendarMonthService, timelineService) {
13
15
  this.calendarMonthService = calendarMonthService;
14
- this.eventPartsForThisWeek$ = new BehaviorSubject([]);
16
+ this.timelineService = timelineService;
17
+ this.events$ = new BehaviorSubject([]);
18
+ this.previewEvent$ = new BehaviorSubject(null);
15
19
  this.timeSlotDuration$ = new BehaviorSubject(1800);
20
+ this.timeSlots$ = new BehaviorSubject([]);
16
21
  this.mouseState$ = new BehaviorSubject(false);
17
22
  this.hoveredTimeSlot$ = new BehaviorSubject(null);
23
+ this.hoveredEvent$ = new BehaviorSubject(null);
18
24
  this.destroyed$ = new Subject();
19
25
  //#region UnitHeight
20
26
  this.unitHeight$ = new BehaviorSubject(40);
21
27
  this.unitHeightChange = new EventEmitter();
28
+ //#endregion
29
+ //#region maxInnerHeight
30
+ this.maxInnerHeight = null;
22
31
  this.operation = null;
23
32
  this.dragStartTimeslot = null;
24
33
  const monday = this.calendarMonthService.getMondayBefore(new Date());
@@ -27,66 +36,97 @@ export class BsSchedulerComponent {
27
36
  weekMonday.setHours(0);
28
37
  weekMonday.setMinutes(0);
29
38
  weekMonday.setSeconds(0);
39
+ weekMonday.setMilliseconds(0);
30
40
  return Array.from(Array(7).keys()).map((x) => this.addDays(weekMonday, x));
31
41
  }));
32
- this.events$ = new BehaviorSubject([]);
33
- this.eventParts$ = this.events$.pipe(map((events) => {
34
- return events.map((ev) => {
35
- let startTime = ev.start;
36
- const result = [];
37
- while (!this.dateEquals(startTime, ev.end)) {
38
- const end = new Date(startTime.getFullYear(), startTime.getMonth(), startTime.getDate() + 1, 0, 0, 0);
39
- result.push({ start: startTime, end: end, event: ev });
40
- startTime = end;
41
- }
42
- if (startTime != ev.end) {
43
- result.push({ start: startTime, end: ev.end, event: ev });
44
- }
45
- return { event: ev, parts: result };
46
- });
42
+ this.daysOfWeekWithTimestamps$ = this.daysOfWeek$
43
+ .pipe(map((daysOfWeek) => {
44
+ return { start: daysOfWeek[0].getTime(), end: daysOfWeek[daysOfWeek.length - 1].getTime() + 24 * 60 * 60 * 1000 };
47
45
  }));
48
- combineLatest([
49
- this.daysOfWeek$
50
- .pipe(map((daysOfWeek) => {
51
- return { start: daysOfWeek[0].getTime(), end: daysOfWeek[daysOfWeek.length - 1].getTime() + 24 * 60 * 60 * 1000 };
52
- })),
46
+ this.eventParts$ = this.events$.pipe(map((events) => events.map((ev) => this.timelineService.splitInParts(ev))));
47
+ this.eventPartsForThisWeek$ = combineLatest([
48
+ this.daysOfWeekWithTimestamps$,
53
49
  this.eventParts$
54
50
  .pipe(map(eventParts => eventParts.map(evp => evp.parts)))
55
51
  .pipe(map(jaggedParts => {
56
52
  return jaggedParts.reduce((flat, toFlatten) => flat.concat(toFlatten), []);
57
- })),
58
- this.eventParts$
53
+ }))
59
54
  ])
60
- .pipe(map(([startAndEnd, eventParts, originalEventParts]) => {
55
+ .pipe(map(([startAndEnd, eventParts]) => {
61
56
  return eventParts.filter(eventPart => {
62
- return !((eventPart.end.getTime() < startAndEnd.start) || (eventPart.start.getTime() > startAndEnd.end));
57
+ return !((eventPart.end.getTime() <= startAndEnd.start) || (eventPart.start.getTime() >= startAndEnd.end));
63
58
  });
64
- }))
65
- .pipe(takeUntil(this.destroyed$))
66
- .subscribe((eventPartsForThisWeek) => {
67
- this.eventPartsForThisWeek$.next(eventPartsForThisWeek);
68
- });
69
- this.timeSlots$ = combineLatest([this.daysOfWeek$, this.timeSlotDuration$])
59
+ }));
60
+ this.previewEventParts$ = this.previewEvent$.pipe(map((event) => {
61
+ if (event) {
62
+ return this.timelineService.splitInParts(event);
63
+ }
64
+ else {
65
+ return null;
66
+ }
67
+ }));
68
+ this.previewEventPartsForThisWeek$ = combineLatest([this.daysOfWeekWithTimestamps$, this.previewEventParts$])
69
+ .pipe(map(([startAndEnd, previewEventParts]) => {
70
+ if (previewEventParts) {
71
+ return previewEventParts.parts.filter(eventPart => {
72
+ return !((eventPart.end.getTime() <= startAndEnd.start) || (eventPart.start.getTime() >= startAndEnd.end));
73
+ });
74
+ }
75
+ else {
76
+ return [];
77
+ }
78
+ }));
79
+ this.timelinedEventPartsForThisWeek$ = this.eventPartsForThisWeek$
80
+ .pipe(map(eventParts => {
81
+ // We'll only use the events for this week
82
+ const events = eventParts.map(ep => ep.event)
83
+ .filter((e, i, list) => list.indexOf(e) === i)
84
+ .filter((e) => !!e)
85
+ .map((e) => e);
86
+ const timeline = this.timelineService.getTimeline(events);
87
+ const result = timeline.map(track => {
88
+ return track.events.map(ev => {
89
+ return { event: ev, index: track.index };
90
+ });
91
+ })
92
+ .reduce((flat, toFlatten) => flat.concat(toFlatten), [])
93
+ .map((evi) => eventParts.filter(p => p.event === evi.event).map(p => {
94
+ return { part: p, index: evi.index };
95
+ }))
96
+ .reduce((flat, toFlatten) => flat.concat(toFlatten), []);
97
+ return {
98
+ total: timeline.length,
99
+ parts: result
100
+ };
101
+ }));
102
+ combineLatest([this.daysOfWeek$, this.timeSlotDuration$])
70
103
  .pipe(map(([daysOfWeek, duration]) => {
71
104
  const timeSlotsPerDay = Math.floor((60 * 60 * 24) / duration);
72
105
  return Array.from(Array(timeSlotsPerDay).keys()).map((index) => {
73
106
  const timeSlotStart = new Date(daysOfWeek[0]);
74
- timeSlotStart.setSeconds(timeSlotStart.getSeconds() + index * duration);
107
+ timeSlotStart.setTime(+timeSlotStart.getTime() + index * duration * 1000);
75
108
  const timeSlotEnd = new Date(timeSlotStart);
76
- timeSlotEnd.setSeconds(timeSlotEnd.getSeconds() + duration);
109
+ timeSlotEnd.setTime(+timeSlotEnd.getTime() + duration * 1000);
77
110
  return daysOfWeek.map((day) => {
78
111
  const start = new Date(day);
79
112
  start.setHours(timeSlotStart.getHours());
80
113
  start.setMinutes(timeSlotStart.getMinutes());
81
114
  start.setSeconds(timeSlotStart.getSeconds());
115
+ start.setMilliseconds(timeSlotStart.getMilliseconds());
82
116
  const end = new Date(day);
83
117
  end.setHours(timeSlotEnd.getHours());
84
118
  end.setMinutes(timeSlotEnd.getMinutes());
85
119
  end.setSeconds(timeSlotEnd.getSeconds());
120
+ end.setMilliseconds(timeSlotEnd.getMilliseconds());
121
+ end.setDate(end.getDate() + timeSlotEnd.getDate() - timeSlotStart.getDate());
86
122
  return { start, end };
87
123
  });
88
124
  });
89
- }));
125
+ }))
126
+ .subscribe((timeslots) => {
127
+ // For performance reasons, we're not using an observable here, but persist the timeslots in a BehaviorSubject.
128
+ this.timeSlots$.next(timeslots);
129
+ });
90
130
  this.unitHeight$
91
131
  .pipe(takeUntil(this.destroyed$))
92
132
  .subscribe((unitHeight) => {
@@ -100,11 +140,6 @@ export class BsSchedulerComponent {
100
140
  this.unitHeight$.next(value);
101
141
  }
102
142
  //#endregion
103
- dateEquals(date1, date2) {
104
- return (date1.getFullYear() === date2.getFullYear() &&
105
- date1.getMonth() === date2.getMonth() &&
106
- date1.getDate() === date2.getDate());
107
- }
108
143
  addDays(date, days) {
109
144
  const result = new Date(date);
110
145
  result.setDate(result.getDate() + days);
@@ -120,6 +155,13 @@ export class BsSchedulerComponent {
120
155
  .pipe(map((w) => this.addDays(w, 7)), take(1))
121
156
  .subscribe((w) => this.currentWeek$.next(w));
122
157
  }
158
+ onHoverEvent(ev) {
159
+ console.log('hovered', ev);
160
+ this.hoveredEvent$.next(ev);
161
+ }
162
+ onLeaveEvent(ev) {
163
+ this.hoveredEvent$.next(null);
164
+ }
123
165
  onCreateEvent(ev, slot) {
124
166
  ev.preventDefault();
125
167
  this.mouseState$.next(true);
@@ -131,20 +173,61 @@ export class BsSchedulerComponent {
131
173
  end: slot.end,
132
174
  color: '#F00',
133
175
  description: 'Test event',
134
- }
176
+ },
177
+ meta: null,
135
178
  };
136
- this.events$.next([...this.events$.value, this.operation.event]);
179
+ this.previewEvent$.next({ start: slot.start, end: slot.end });
180
+ }
181
+ randomColor() {
182
+ const brightness = 128;
183
+ return '#' + this.randomChannel(brightness) + this.randomChannel(brightness) + this.randomChannel(brightness);
184
+ }
185
+ randomChannel(brightness) {
186
+ const r = 255 - brightness;
187
+ const n = 0 | ((Math.random() * r) + brightness);
188
+ const s = n.toString(16);
189
+ return (s.length == 1) ? '0' + s : s;
137
190
  }
138
191
  onStartDragEvent(eventPart, ev) {
139
192
  ev.preventDefault();
140
193
  this.hoveredTimeSlot$.pipe(take(1)).subscribe((hoveredTimeSlot) => {
141
- this.dragStartTimeslot = hoveredTimeSlot;
142
- this.operation = {
143
- operation: EDragOperation.moveEvent,
144
- event: eventPart.event
145
- };
194
+ if (eventPart.event) {
195
+ this.dragStartTimeslot = hoveredTimeSlot;
196
+ this.operation = {
197
+ operation: EDragOperation.moveEvent,
198
+ event: eventPart.event,
199
+ meta: null,
200
+ };
201
+ this.previewEvent$.next({ start: eventPart.event.start, end: eventPart.event.end });
202
+ }
146
203
  });
147
204
  }
205
+ onStartResizeEvent(event, position) {
206
+ if (event) {
207
+ switch (position) {
208
+ case 'top':
209
+ {
210
+ this.operation = {
211
+ operation: EDragOperation.resizeEvent,
212
+ event: event,
213
+ meta: { position },
214
+ };
215
+ this.previewEvent$.next({ start: event.start, end: event.end });
216
+ }
217
+ break;
218
+ case 'bottom':
219
+ {
220
+ this.operation = {
221
+ operation: EDragOperation.resizeEvent,
222
+ event: event,
223
+ meta: { position },
224
+ };
225
+ this.previewEvent$.next({ start: event.start, end: event.end });
226
+ }
227
+ break;
228
+ }
229
+ }
230
+ }
148
231
  //#region hoveredTimeslot$
149
232
  getHoveredTimeslot(ev, timeSlots) {
150
233
  const hoveredSlots = this.timeSlotElements.filter((el) => {
@@ -187,15 +270,31 @@ export class BsSchedulerComponent {
187
270
  }
188
271
  else if (this.dragStartTimeslot.start.getTime() < hovered.start.getTime()) {
189
272
  // Drag down
190
- this.operation.event.start = this.dragStartTimeslot.start;
191
- this.operation.event.end = hovered.end;
192
- this.events$.next(this.events$.value);
273
+ this.previewEvent$
274
+ .pipe(filter((ev) => !!ev && !!this.dragStartTimeslot))
275
+ .pipe(map((ev) => {
276
+ if (ev && this.dragStartTimeslot) {
277
+ ev.start = this.dragStartTimeslot.start;
278
+ ev.end = hovered.end;
279
+ }
280
+ return ev;
281
+ }))
282
+ .pipe(take(1))
283
+ .subscribe((ev) => this.previewEvent$.next(ev));
193
284
  }
194
285
  else if (this.dragStartTimeslot.start.getTime() > hovered.start.getTime()) {
195
286
  // Drag up
196
- this.operation.event.start = hovered.start;
197
- this.operation.event.end = this.dragStartTimeslot.end;
198
- this.events$.next(this.events$.value);
287
+ this.previewEvent$
288
+ .pipe(filter((ev) => !!ev && !!this.dragStartTimeslot))
289
+ .pipe(map((ev) => {
290
+ if (ev && this.dragStartTimeslot) {
291
+ ev.start = hovered.start;
292
+ ev.end = this.dragStartTimeslot.end;
293
+ }
294
+ return ev;
295
+ }))
296
+ .pipe(take(1))
297
+ .subscribe((ev) => this.previewEvent$.next(ev));
199
298
  }
200
299
  }
201
300
  }
@@ -203,10 +302,46 @@ export class BsSchedulerComponent {
203
302
  case EDragOperation.moveEvent:
204
303
  {
205
304
  if (hovered && this.dragStartTimeslot) {
206
- this.operation.event.start.setTime(this.operation.event.start.getTime() + hovered.start.getTime() - this.dragStartTimeslot.start.getTime());
207
- this.operation.event.end.setTime(this.operation.event.end.getTime() + hovered.start.getTime() - this.dragStartTimeslot.start.getTime());
208
- this.dragStartTimeslot = hovered;
209
- this.events$.next(this.events$.value);
305
+ this.previewEvent$
306
+ .pipe(filter((ev) => !!ev && !!this.dragStartTimeslot))
307
+ .pipe(map((ev) => {
308
+ if (ev && this.dragStartTimeslot) {
309
+ const result = {
310
+ start: new Date(ev.start.getTime() + hovered.start.getTime() - this.dragStartTimeslot.start.getTime()),
311
+ end: new Date(ev.end.getTime() + hovered.start.getTime() - this.dragStartTimeslot.start.getTime())
312
+ };
313
+ this.dragStartTimeslot = hovered;
314
+ return result;
315
+ }
316
+ else {
317
+ return ev;
318
+ }
319
+ }))
320
+ .pipe(take(1))
321
+ .subscribe(ev => this.previewEvent$.next(ev));
322
+ }
323
+ }
324
+ break;
325
+ case EDragOperation.resizeEvent:
326
+ {
327
+ if (hovered) {
328
+ this.previewEvent$
329
+ .pipe(filter((ev) => !!ev))
330
+ .pipe(map((ev) => {
331
+ if (ev && this.operation && this.operation.event) {
332
+ if (this.operation.meta.position === 'top') {
333
+ ev.start = hovered.start;
334
+ ev.end = this.operation.event.end;
335
+ }
336
+ else if (this.operation.meta.position === 'bottom') {
337
+ ev.start = this.operation.event.start;
338
+ ev.end = hovered.end;
339
+ }
340
+ }
341
+ return ev;
342
+ }))
343
+ .pipe(take(1))
344
+ .subscribe((ev) => this.previewEvent$.next(ev));
210
345
  }
211
346
  }
212
347
  break;
@@ -220,16 +355,39 @@ export class BsSchedulerComponent {
220
355
  switch (this.operation.operation) {
221
356
  case EDragOperation.createEvent:
222
357
  {
223
- if (this.operation.event) {
224
- this.operation = null;
225
- this.dragStartTimeslot = null;
226
- }
358
+ combineLatest([this.previewEvent$])
359
+ .pipe(take(1))
360
+ .subscribe(([previewEvent]) => {
361
+ if (previewEvent) {
362
+ this.operation = null;
363
+ this.dragStartTimeslot = null;
364
+ this.events$.next([...this.events$.value, {
365
+ start: previewEvent.start,
366
+ end: previewEvent.end,
367
+ color: this.randomColor(),
368
+ description: 'New event'
369
+ }]);
370
+ this.previewEvent$.next(null);
371
+ }
372
+ });
227
373
  }
228
374
  break;
229
375
  case EDragOperation.moveEvent:
376
+ case EDragOperation.resizeEvent:
230
377
  {
231
- this.operation = null;
232
- this.dragStartTimeslot = null;
378
+ this.previewEvent$
379
+ .pipe(filter((ev) => !!ev))
380
+ .pipe(take(1))
381
+ .subscribe((previewEvent) => {
382
+ if (this.operation && this.operation.event && previewEvent) {
383
+ this.operation.event.start = previewEvent.start;
384
+ this.operation.event.end = previewEvent.end;
385
+ this.operation = null;
386
+ this.dragStartTimeslot = null;
387
+ this.events$.next(this.events$.value);
388
+ this.previewEvent$.next(null);
389
+ }
390
+ });
233
391
  }
234
392
  break;
235
393
  }
@@ -239,18 +397,20 @@ export class BsSchedulerComponent {
239
397
  this.destroyed$.next(true);
240
398
  }
241
399
  }
242
- BsSchedulerComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "13.1.1", ngImport: i0, type: BsSchedulerComponent, deps: [{ token: i1.BsCalendarMonthService }], target: i0.ɵɵFactoryTarget.Component });
243
- BsSchedulerComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "12.0.0", version: "13.1.1", type: BsSchedulerComponent, selector: "bs-scheduler", inputs: { unitHeight: "unitHeight" }, outputs: { unitHeightChange: "unitHeightChange" }, host: { listeners: { "document:mousemove": "onMousemove($event)", "document:mouseup": "onMouseUp($event)" } }, viewQueries: [{ propertyName: "timeSlotElements", predicate: ["slot"], descendants: true }], ngImport: i0, template: "{{ unitHeight$ | async }}\n<div class=\"table d-flex w-100 overflow-y-auto\" [style.max-height.px]=\"400\">\n <div class=\"calendar-head\">\n <div class=\"w-100 d-flex flex-row\">\n <div class=\"d-flex calendar-left justify-content-between\">\n <button class=\"btn btn-default flex-start\" (click)=\"onPreviousWeek()\">&lt;</button>\n <button class=\"btn btn-default flex-end\" (click)=\"onNextWeek()\">&gt;</button>\n </div>\n <div class=\"flex-grow-1\" *ngFor=\"let day of (daysOfWeek$ | async)\">\n <span class=\"d-block col-form-label text-center\">\n {{ day | date: 'dd-MM-yyyy' }}\n </span>\n </div>\n </div>\n </div>\n <div class=\"calendar-body\">\n <div class=\"position-relative\">\n <!-- Timeslots -->\n <div *ngFor=\"let timeslots of (timeSlots$ | async); let i = index\" class=\"d-flex flex-row p-0 timeslot\" [style.height.px]=\"unitHeight$ | async\">\n <div class=\"calendar-cell calendar-left align-top py-0\">{{ timeslots[0].start | date: 'HH:mm:ss' }}</div>\n <div class=\"calendar-cell flex-grow-1\" *ngFor=\"let slot of timeslots; let j = index\" #slot (mousedown)=\"onCreateEvent($event, slot)\" [attr.data-row]=\"i\" [attr.data-column]=\"j\"></div>\n </div>\n \n <!-- Events -->\n <ng-container *ngIf=\"(timeSlotDuration$ | async) as timeSlotDuration\">\n\n <div *ngFor=\"let eventPart of (eventPartsForThisWeek$ | async)\" class=\"position-absolute event\"\n [style.top.px]=\"(eventPart | bsSecondsTodayOffset) / timeSlotDuration * (unitHeight$ | async)!\"\n [style.height.px]=\"(eventPart | bsSecondsTimespan) / timeSlotDuration * (unitHeight$ | async)!\"\n [style.left]=\"'calc(90px + ((100% - 90px) / 7 * ' + ((eventPart | dayOfWeek) - 1) + '))'\">\n <div class=\"event-inner\" [style.background-color]=\"eventPart.event.color\" (mousedown)=\"onStartDragEvent(eventPart, $event)\">\n {{ eventPart.event.description }}\n <br>\n {{ eventPart.start | date: 'HH:mm:ss' }}\n <br>\n {{ eventPart.end | date: 'HH:mm:ss' }}\n </div>\n </div>\n \n <!-- <ng-container *ngFor=\"let eventWithParts of (eventParts$ | async)\">\n <div *ngFor=\"let eventPart of eventWithParts.parts\" class=\"position-absolute event\"\n [style.top.px]=\"(eventPart | bsSecondsTodayOffset) / timeSlotDuration * 40\"\n [style.height.px]=\"(eventPart | bsSecondsTimespan) / timeSlotDuration * 40\"\n [style.left]=\"'calc(90px + ((100% - 90px) / 7 * ' + ((eventPart | dayOfWeek) - 1) + '))'\">\n <div class=\"event-inner\">\n {{ eventPart.start | date: 'HH:mm:ss' }} - {{ eventPart.end | date: 'HH:mm:ss' }}\n </div>\n </div>\n </ng-container> -->\n \n </ng-container>\n </div>\n </div>\n</div>", styles: [":host{display:block}.table{flex-flow:column}.table .calendar-head{flex:0 0 auto}.table .calendar-head>div{padding-right:18px}.table .calendar-body{flex:1 1 auto;display:block;overflow-y:visible;overflow-x:hidden}.table .calendar-body .calendar-cell{border-right:1px solid #DEE2E6;border-bottom:1px solid #DEE2E6;cursor:default}.table .calendar-body .calendar-cell.hover{border-top-width:3px}.table .calendar-left{width:90px}.event{z-index:5;width:calc((100% - 90px) / 7);height:100px;overflow:hidden;position:relative;-webkit-user-select:none;user-select:none}.event .event-inner{position:absolute;left:4px;right:5px;top:4px;bottom:5px;cursor:move}\n"], directives: [{ type: i2.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { type: i2.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }], pipes: { "async": i2.AsyncPipe, "date": i2.DatePipe, "bsSecondsTodayOffset": i3.BsSecondsTodayOffsetPipe, "bsSecondsTimespan": i4.BsSecondsTimespanPipe, "dayOfWeek": i5.DayOfWeekPipe } });
244
- i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "13.1.1", ngImport: i0, type: BsSchedulerComponent, decorators: [{
400
+ BsSchedulerComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "13.2.0", ngImport: i0, type: BsSchedulerComponent, deps: [{ token: i1.BsCalendarMonthService }, { token: i2.BsTimelineService }], target: i0.ɵɵFactoryTarget.Component });
401
+ BsSchedulerComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "12.0.0", version: "13.2.0", type: BsSchedulerComponent, selector: "bs-scheduler", inputs: { unitHeight: "unitHeight", maxInnerHeight: "maxInnerHeight" }, outputs: { unitHeightChange: "unitHeightChange" }, host: { listeners: { "document:mousemove": "onMousemove($event)", "document:mouseup": "onMouseUp($event)" } }, viewQueries: [{ propertyName: "timeSlotElements", predicate: ["slot"], descendants: true }], ngImport: i0, template: "<div class=\"table d-flex w-100 overflow-y-auto\" [style.max-height.px]=\"null\">\n <div class=\"calendar-head\">\n <div class=\"w-100 d-flex flex-row\">\n <div class=\"d-flex calendar-left justify-content-between\">\n <button class=\"btn btn-default flex-start\" (click)=\"onPreviousWeek()\">&lt;</button>\n <button class=\"btn btn-default flex-end\" (click)=\"onNextWeek()\">&gt;</button>\n </div>\n <div class=\"flex-grow-1 calendar-head-cell\" *ngFor=\"let day of (daysOfWeek$ | async)\">\n <span class=\"d-block col-form-label text-center\">\n <span>{{ day | date: 'dd-MM' }}</span><span class=\"d-none d-lg-inline-block\">-{{ day | date: 'yyyy' }}</span>\n </span>\n </div>\n </div>\n </div>\n <div class=\"calendar-body\" [style.height.px]=\"maxInnerHeight\">\n <div class=\"position-relative\">\n <!-- Timeslots -->\n <div *ngFor=\"let timeslots of (timeSlots$ | async); let i = index\" class=\"d-flex flex-row p-0 timeslot\" [style.height.px]=\"unitHeight$ | async\">\n <div class=\"calendar-cell calendar-left align-top px-2 py-0\">{{ timeslots[0].start | date: 'HH:mm:ss' }}</div>\n <div class=\"calendar-cell flex-grow-1\" *ngFor=\"let slot of timeslots; let j = index\" #slot (mousedown)=\"onCreateEvent($event, slot)\" [attr.data-row]=\"i\" [attr.data-column]=\"j\"></div>\n </div>\n \n <!-- Events -->\n <ng-container *ngIf=\"(timeSlotDuration$ | async) as timeSlotDuration\">\n <ng-container *ngIf=\"(unitHeight$ | async) as unitHeight\">\n <ng-container *ngIf=\"(timelinedEventPartsForThisWeek$ | async) as partData\">\n <div *ngFor=\"let eventPart of partData.parts\" class=\"event p-0\"\n [style.top.px]=\"(eventPart.part | bsSecondsTodayOffset) / timeSlotDuration * unitHeight\"\n [style.height.px]=\"(eventPart.part | bsSecondsTimespan) / timeSlotDuration * unitHeight\"\n [style.left]=\"'calc(90px + ((100% - 90px) / 7 * ' + ((eventPart.part | dayOfWeek) - 1) + '))'\">\n <div class=\"event-inner\"\n [class.hover]=\"(hoveredEvent$ | async) === eventPart.part.event\"\n [style.width]=\"'calc(100% / ' + partData.total + ')'\"\n [style.height.px]=\"(eventPart.part | bsSecondsTimespan) / timeSlotDuration * unitHeight\"\n [style.margin-left]=\"'calc(100% / ' + partData.total + ' * ' + eventPart.index + ')'\">\n\n <div class=\"event-resize top\" *ngIf=\"eventPart.part.start === eventPart.part.event?.start\" (mousedown)=\"onStartResizeEvent(eventPart.part.event, 'top')\"></div>\n <div class=\"event-border\"></div>\n <div class=\"event-wrapper\" *ngIf=\"eventPart.part.event\"\n (mousedown)=\"onStartDragEvent(eventPart.part, $event)\"\n (mouseenter)=\"onHoverEvent(eventPart.part.event)\"\n (mouseleave)=\"onLeaveEvent(eventPart.part.event)\">\n <div class=\"event-bg\" [style.background-color]=\"eventPart.part.event.color\"></div>\n <div class=\"event-label\">{{ eventPart.part.event.description }}</div>\n </div>\n <div class=\"event-resize bottom\" *ngIf=\"eventPart.part.end === eventPart.part.event?.end\" (mousedown)=\"onStartResizeEvent(eventPart.part.event, 'bottom')\"></div>\n </div>\n </div>\n </ng-container>\n <ng-container *ngIf=\"(previewEventPartsForThisWeek$ | async) as previewPartData\">\n <div *ngFor=\"let eventPart of previewPartData\" class=\"event preview p-0\"\n [style.top.px]=\"(eventPart | bsSecondsTodayOffset) / timeSlotDuration * unitHeight\"\n [style.height.px]=\"(eventPart | bsSecondsTimespan) / timeSlotDuration * unitHeight\"\n [style.left]=\"'calc(90px + ((100% - 90px) / 7 * ' + ((eventPart | dayOfWeek) - 1) + '))'\">\n <div class=\"event-inner w-100\" [style.height.px]=\"(eventPart | bsSecondsTimespan) / timeSlotDuration * unitHeight\"></div>\n </div>\n </ng-container>\n </ng-container>\n </ng-container>\n </div>\n </div>\n</div>", styles: [":host{display:block}.table{flex-flow:column}.table .calendar-head{flex:0 0 auto}.table .calendar-head>div{padding-right:18px}.table .calendar-body{flex:1 1 auto;display:block;overflow-y:scroll;overflow-x:hidden}.table .calendar-body .calendar-cell{border-right:1px solid #DEE2E6;border-bottom:1px solid #DEE2E6;cursor:default}.table .calendar-body .calendar-cell.hover{border-top-width:3px}.table .calendar-left{width:90px}.event{z-index:5;width:calc((100% - 90px) / 7);height:100px;overflow:hidden;position:absolute;-webkit-user-select:none;user-select:none;pointer-events:none}.event.preview{background:#666;opacity:.6}.event .event-border{background:black;top:0;left:0;bottom:3px;width:3px;position:absolute;z-index:10;opacity:.3}.event .event-inner{position:relative;left:0px;right:5px;top:0px;bottom:5px;cursor:move;pointer-events:all;overflow:hidden}.event .event-inner .event-wrapper{width:calc(100% - 2px);margin:1px auto 1px 0;height:calc(100% - 3px)}.event .event-inner .event-wrapper .event-bg{opacity:.5;height:100%;transition:opacity .15s ease-in-out}.event .event-inner .event-wrapper .event-label{position:absolute;top:0;font-size:12px;font-weight:600;padding:4px}.event .event-inner .event-resize{position:absolute;cursor:ns-resize;height:4px;left:0;right:0;z-index:400}.event .event-inner .event-resize.top{top:0}.event .event-inner .event-resize.bottom{bottom:0}.event .event-inner .event-resize:hover{background:#DDD}.event .event-inner.hover .event-bg{opacity:.7!important}@media (max-width: 767px){.table .calendar-head .calendar-head-cell{text-orientation:sideways;writing-mode:vertical-rl}}\n"], directives: [{ type: i3.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { type: i3.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }], pipes: { "async": i3.AsyncPipe, "date": i3.DatePipe, "bsSecondsTodayOffset": i4.BsSecondsTodayOffsetPipe, "bsSecondsTimespan": i5.BsSecondsTimespanPipe, "dayOfWeek": i6.DayOfWeekPipe } });
402
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "13.2.0", ngImport: i0, type: BsSchedulerComponent, decorators: [{
245
403
  type: Component,
246
- args: [{ selector: 'bs-scheduler', template: "{{ unitHeight$ | async }}\n<div class=\"table d-flex w-100 overflow-y-auto\" [style.max-height.px]=\"400\">\n <div class=\"calendar-head\">\n <div class=\"w-100 d-flex flex-row\">\n <div class=\"d-flex calendar-left justify-content-between\">\n <button class=\"btn btn-default flex-start\" (click)=\"onPreviousWeek()\">&lt;</button>\n <button class=\"btn btn-default flex-end\" (click)=\"onNextWeek()\">&gt;</button>\n </div>\n <div class=\"flex-grow-1\" *ngFor=\"let day of (daysOfWeek$ | async)\">\n <span class=\"d-block col-form-label text-center\">\n {{ day | date: 'dd-MM-yyyy' }}\n </span>\n </div>\n </div>\n </div>\n <div class=\"calendar-body\">\n <div class=\"position-relative\">\n <!-- Timeslots -->\n <div *ngFor=\"let timeslots of (timeSlots$ | async); let i = index\" class=\"d-flex flex-row p-0 timeslot\" [style.height.px]=\"unitHeight$ | async\">\n <div class=\"calendar-cell calendar-left align-top py-0\">{{ timeslots[0].start | date: 'HH:mm:ss' }}</div>\n <div class=\"calendar-cell flex-grow-1\" *ngFor=\"let slot of timeslots; let j = index\" #slot (mousedown)=\"onCreateEvent($event, slot)\" [attr.data-row]=\"i\" [attr.data-column]=\"j\"></div>\n </div>\n \n <!-- Events -->\n <ng-container *ngIf=\"(timeSlotDuration$ | async) as timeSlotDuration\">\n\n <div *ngFor=\"let eventPart of (eventPartsForThisWeek$ | async)\" class=\"position-absolute event\"\n [style.top.px]=\"(eventPart | bsSecondsTodayOffset) / timeSlotDuration * (unitHeight$ | async)!\"\n [style.height.px]=\"(eventPart | bsSecondsTimespan) / timeSlotDuration * (unitHeight$ | async)!\"\n [style.left]=\"'calc(90px + ((100% - 90px) / 7 * ' + ((eventPart | dayOfWeek) - 1) + '))'\">\n <div class=\"event-inner\" [style.background-color]=\"eventPart.event.color\" (mousedown)=\"onStartDragEvent(eventPart, $event)\">\n {{ eventPart.event.description }}\n <br>\n {{ eventPart.start | date: 'HH:mm:ss' }}\n <br>\n {{ eventPart.end | date: 'HH:mm:ss' }}\n </div>\n </div>\n \n <!-- <ng-container *ngFor=\"let eventWithParts of (eventParts$ | async)\">\n <div *ngFor=\"let eventPart of eventWithParts.parts\" class=\"position-absolute event\"\n [style.top.px]=\"(eventPart | bsSecondsTodayOffset) / timeSlotDuration * 40\"\n [style.height.px]=\"(eventPart | bsSecondsTimespan) / timeSlotDuration * 40\"\n [style.left]=\"'calc(90px + ((100% - 90px) / 7 * ' + ((eventPart | dayOfWeek) - 1) + '))'\">\n <div class=\"event-inner\">\n {{ eventPart.start | date: 'HH:mm:ss' }} - {{ eventPart.end | date: 'HH:mm:ss' }}\n </div>\n </div>\n </ng-container> -->\n \n </ng-container>\n </div>\n </div>\n</div>", styles: [":host{display:block}.table{flex-flow:column}.table .calendar-head{flex:0 0 auto}.table .calendar-head>div{padding-right:18px}.table .calendar-body{flex:1 1 auto;display:block;overflow-y:visible;overflow-x:hidden}.table .calendar-body .calendar-cell{border-right:1px solid #DEE2E6;border-bottom:1px solid #DEE2E6;cursor:default}.table .calendar-body .calendar-cell.hover{border-top-width:3px}.table .calendar-left{width:90px}.event{z-index:5;width:calc((100% - 90px) / 7);height:100px;overflow:hidden;position:relative;-webkit-user-select:none;user-select:none}.event .event-inner{position:absolute;left:4px;right:5px;top:4px;bottom:5px;cursor:move}\n"] }]
247
- }], ctorParameters: function () { return [{ type: i1.BsCalendarMonthService }]; }, propDecorators: { timeSlotElements: [{
404
+ args: [{ selector: 'bs-scheduler', template: "<div class=\"table d-flex w-100 overflow-y-auto\" [style.max-height.px]=\"null\">\n <div class=\"calendar-head\">\n <div class=\"w-100 d-flex flex-row\">\n <div class=\"d-flex calendar-left justify-content-between\">\n <button class=\"btn btn-default flex-start\" (click)=\"onPreviousWeek()\">&lt;</button>\n <button class=\"btn btn-default flex-end\" (click)=\"onNextWeek()\">&gt;</button>\n </div>\n <div class=\"flex-grow-1 calendar-head-cell\" *ngFor=\"let day of (daysOfWeek$ | async)\">\n <span class=\"d-block col-form-label text-center\">\n <span>{{ day | date: 'dd-MM' }}</span><span class=\"d-none d-lg-inline-block\">-{{ day | date: 'yyyy' }}</span>\n </span>\n </div>\n </div>\n </div>\n <div class=\"calendar-body\" [style.height.px]=\"maxInnerHeight\">\n <div class=\"position-relative\">\n <!-- Timeslots -->\n <div *ngFor=\"let timeslots of (timeSlots$ | async); let i = index\" class=\"d-flex flex-row p-0 timeslot\" [style.height.px]=\"unitHeight$ | async\">\n <div class=\"calendar-cell calendar-left align-top px-2 py-0\">{{ timeslots[0].start | date: 'HH:mm:ss' }}</div>\n <div class=\"calendar-cell flex-grow-1\" *ngFor=\"let slot of timeslots; let j = index\" #slot (mousedown)=\"onCreateEvent($event, slot)\" [attr.data-row]=\"i\" [attr.data-column]=\"j\"></div>\n </div>\n \n <!-- Events -->\n <ng-container *ngIf=\"(timeSlotDuration$ | async) as timeSlotDuration\">\n <ng-container *ngIf=\"(unitHeight$ | async) as unitHeight\">\n <ng-container *ngIf=\"(timelinedEventPartsForThisWeek$ | async) as partData\">\n <div *ngFor=\"let eventPart of partData.parts\" class=\"event p-0\"\n [style.top.px]=\"(eventPart.part | bsSecondsTodayOffset) / timeSlotDuration * unitHeight\"\n [style.height.px]=\"(eventPart.part | bsSecondsTimespan) / timeSlotDuration * unitHeight\"\n [style.left]=\"'calc(90px + ((100% - 90px) / 7 * ' + ((eventPart.part | dayOfWeek) - 1) + '))'\">\n <div class=\"event-inner\"\n [class.hover]=\"(hoveredEvent$ | async) === eventPart.part.event\"\n [style.width]=\"'calc(100% / ' + partData.total + ')'\"\n [style.height.px]=\"(eventPart.part | bsSecondsTimespan) / timeSlotDuration * unitHeight\"\n [style.margin-left]=\"'calc(100% / ' + partData.total + ' * ' + eventPart.index + ')'\">\n\n <div class=\"event-resize top\" *ngIf=\"eventPart.part.start === eventPart.part.event?.start\" (mousedown)=\"onStartResizeEvent(eventPart.part.event, 'top')\"></div>\n <div class=\"event-border\"></div>\n <div class=\"event-wrapper\" *ngIf=\"eventPart.part.event\"\n (mousedown)=\"onStartDragEvent(eventPart.part, $event)\"\n (mouseenter)=\"onHoverEvent(eventPart.part.event)\"\n (mouseleave)=\"onLeaveEvent(eventPart.part.event)\">\n <div class=\"event-bg\" [style.background-color]=\"eventPart.part.event.color\"></div>\n <div class=\"event-label\">{{ eventPart.part.event.description }}</div>\n </div>\n <div class=\"event-resize bottom\" *ngIf=\"eventPart.part.end === eventPart.part.event?.end\" (mousedown)=\"onStartResizeEvent(eventPart.part.event, 'bottom')\"></div>\n </div>\n </div>\n </ng-container>\n <ng-container *ngIf=\"(previewEventPartsForThisWeek$ | async) as previewPartData\">\n <div *ngFor=\"let eventPart of previewPartData\" class=\"event preview p-0\"\n [style.top.px]=\"(eventPart | bsSecondsTodayOffset) / timeSlotDuration * unitHeight\"\n [style.height.px]=\"(eventPart | bsSecondsTimespan) / timeSlotDuration * unitHeight\"\n [style.left]=\"'calc(90px + ((100% - 90px) / 7 * ' + ((eventPart | dayOfWeek) - 1) + '))'\">\n <div class=\"event-inner w-100\" [style.height.px]=\"(eventPart | bsSecondsTimespan) / timeSlotDuration * unitHeight\"></div>\n </div>\n </ng-container>\n </ng-container>\n </ng-container>\n </div>\n </div>\n</div>", styles: [":host{display:block}.table{flex-flow:column}.table .calendar-head{flex:0 0 auto}.table .calendar-head>div{padding-right:18px}.table .calendar-body{flex:1 1 auto;display:block;overflow-y:scroll;overflow-x:hidden}.table .calendar-body .calendar-cell{border-right:1px solid #DEE2E6;border-bottom:1px solid #DEE2E6;cursor:default}.table .calendar-body .calendar-cell.hover{border-top-width:3px}.table .calendar-left{width:90px}.event{z-index:5;width:calc((100% - 90px) / 7);height:100px;overflow:hidden;position:absolute;-webkit-user-select:none;user-select:none;pointer-events:none}.event.preview{background:#666;opacity:.6}.event .event-border{background:black;top:0;left:0;bottom:3px;width:3px;position:absolute;z-index:10;opacity:.3}.event .event-inner{position:relative;left:0px;right:5px;top:0px;bottom:5px;cursor:move;pointer-events:all;overflow:hidden}.event .event-inner .event-wrapper{width:calc(100% - 2px);margin:1px auto 1px 0;height:calc(100% - 3px)}.event .event-inner .event-wrapper .event-bg{opacity:.5;height:100%;transition:opacity .15s ease-in-out}.event .event-inner .event-wrapper .event-label{position:absolute;top:0;font-size:12px;font-weight:600;padding:4px}.event .event-inner .event-resize{position:absolute;cursor:ns-resize;height:4px;left:0;right:0;z-index:400}.event .event-inner .event-resize.top{top:0}.event .event-inner .event-resize.bottom{bottom:0}.event .event-inner .event-resize:hover{background:#DDD}.event .event-inner.hover .event-bg{opacity:.7!important}@media (max-width: 767px){.table .calendar-head .calendar-head-cell{text-orientation:sideways;writing-mode:vertical-rl}}\n"] }]
405
+ }], ctorParameters: function () { return [{ type: i1.BsCalendarMonthService }, { type: i2.BsTimelineService }]; }, propDecorators: { timeSlotElements: [{
248
406
  type: ViewChildren,
249
407
  args: ['slot']
250
408
  }], unitHeightChange: [{
251
409
  type: Output
252
410
  }], unitHeight: [{
253
411
  type: Input
412
+ }], maxInnerHeight: [{
413
+ type: Input
254
414
  }], onMousemove: [{
255
415
  type: HostListener,
256
416
  args: ['document:mousemove', ['$event']]
@@ -258,4 +418,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "13.1.1", ngImpor
258
418
  type: HostListener,
259
419
  args: ['document:mouseup', ['$event']]
260
420
  }] } });
261
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"scheduler.component.js","sourceRoot":"","sources":["../../../../../../../../../libs/mintplayer-ng-bootstrap/src/lib/components/scheduler/components/scheduler/scheduler.component.ts","../../../../../../../../../libs/mintplayer-ng-bootstrap/src/lib/components/scheduler/components/scheduler/scheduler.component.html"],"names":[],"mappings":"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","sourcesContent":["import { Component, ElementRef, EventEmitter, HostListener, Input, OnDestroy, Output, QueryList, ViewChildren } from '@angular/core';\nimport { BehaviorSubject, combineLatest, map, Observable, Subject, take, takeUntil } from 'rxjs';\nimport { BsCalendarMonthService } from '../../../../services/calendar-month/calendar-month.service';\nimport { EDragOperation } from '../../enums/drag-operation';\nimport { DragOperation } from '../../interfaces/drag-operation';\nimport { SchedulerEvent } from '../../interfaces/scheduler-event';\nimport { SchedulerEventPart } from '../../interfaces/scheduler-event-part';\nimport { SchedulerEventWithParts } from '../../interfaces/scheduler-event-with-parts';\nimport { TimeSlot } from '../../interfaces/time-slot';\n\n@Component({\n  selector: 'bs-scheduler',\n  templateUrl: './scheduler.component.html',\n  styleUrls: ['./scheduler.component.scss'],\n})\nexport class BsSchedulerComponent implements OnDestroy {\n  constructor(private calendarMonthService: BsCalendarMonthService) {\n    const monday = this.calendarMonthService.getMondayBefore(new Date());\n    this.currentWeek$ = new BehaviorSubject<Date>(monday);\n\n    this.daysOfWeek$ = this.currentWeek$.pipe(\n      map((weekMonday) => {\n        weekMonday.setHours(0);\n        weekMonday.setMinutes(0);\n        weekMonday.setSeconds(0);\n        return Array.from(Array(7).keys()).map((x) => this.addDays(weekMonday, x));\n      })\n    );\n\n    this.events$ = new BehaviorSubject<SchedulerEvent[]>([]);\n    this.eventParts$ = this.events$.pipe(\n      map((events) => {\n        return events.map((ev) => {\n          let startTime = ev.start;\n          const result: SchedulerEventPart[] = [];\n          while (!this.dateEquals(startTime, ev.end)) {\n            const end = new Date(startTime.getFullYear(), startTime.getMonth(), startTime.getDate() + 1, 0, 0, 0);\n            result.push({ start: startTime, end: end, event: ev });\n            startTime = end;\n          }\n          if (startTime != ev.end) {\n            result.push({ start: startTime, end: ev.end, event: ev });\n          }\n\n          return <SchedulerEventWithParts>{ event: ev, parts: result };\n        });\n      })\n    );\n\n    combineLatest([\n        this.daysOfWeek$\n          .pipe(map((daysOfWeek) => {\n            return { start: daysOfWeek[0].getTime(), end: daysOfWeek[daysOfWeek.length - 1].getTime() + 24 * 60 * 60 * 1000 };\n          })),\n        this.eventParts$\n          .pipe(map(eventParts => eventParts.map(evp => evp.parts)))\n          .pipe(map(jaggedParts => {\n            return jaggedParts.reduce((flat, toFlatten) => flat.concat(toFlatten), []);\n          })),\n        this.eventParts$\n      ])\n      .pipe(map(([startAndEnd, eventParts, originalEventParts]) => {\n        return eventParts.filter(eventPart => {\n          return !((eventPart.end.getTime() < startAndEnd.start) || (eventPart.start.getTime() > startAndEnd.end));\n        });\n      }))\n      .pipe(takeUntil(this.destroyed$))\n      .subscribe((eventPartsForThisWeek) => {\n        this.eventPartsForThisWeek$.next(eventPartsForThisWeek);\n      });\n\n    this.timeSlots$ = combineLatest([this.daysOfWeek$, this.timeSlotDuration$])\n      .pipe(map(([daysOfWeek, duration]) => {\n        const timeSlotsPerDay = Math.floor((60 * 60 * 24) / duration);\n\n        return Array.from(Array(timeSlotsPerDay).keys()).map((index) => {\n          const timeSlotStart = new Date(daysOfWeek[0]);\n          timeSlotStart.setSeconds(timeSlotStart.getSeconds() + index * duration);\n          const timeSlotEnd = new Date(timeSlotStart);\n          timeSlotEnd.setSeconds(timeSlotEnd.getSeconds() + duration);\n\n          return daysOfWeek.map((day) => {\n            const start = new Date(day);\n            start.setHours(timeSlotStart.getHours());\n            start.setMinutes(timeSlotStart.getMinutes());\n            start.setSeconds(timeSlotStart.getSeconds());\n\n            const end = new Date(day);\n            end.setHours(timeSlotEnd.getHours());\n            end.setMinutes(timeSlotEnd.getMinutes());\n            end.setSeconds(timeSlotEnd.getSeconds());\n\n            return <TimeSlot>{ start, end };\n          });\n        });\n\n      }));\n    \n    this.unitHeight$\n      .pipe(takeUntil(this.destroyed$))\n      .subscribe((unitHeight) => {\n        this.unitHeightChange.emit(unitHeight);\n      })\n  }\n\n  events$: BehaviorSubject<SchedulerEvent[]>;\n  eventParts$: Observable<SchedulerEventWithParts[]>;\n  eventPartsForThisWeek$ = new BehaviorSubject<SchedulerEventPart[]>([]);\n  currentWeek$: BehaviorSubject<Date>;\n  daysOfWeek$: Observable<Date[]>;\n  timeSlotDuration$ = new BehaviorSubject<number>(1800);\n  timeSlots$: Observable<TimeSlot[][]>;\n  mouseState$ = new BehaviorSubject<boolean>(false);\n  hoveredTimeSlot$ = new BehaviorSubject<TimeSlot | null>(null);\n  destroyed$ = new Subject();\n\n  @ViewChildren('slot') timeSlotElements!: QueryList<ElementRef<HTMLDivElement>>;\n\n  //#region UnitHeight\n  unitHeight$ = new BehaviorSubject<number>(40);\n  @Output() public unitHeightChange = new EventEmitter<number>();\n  public get unitHeight() {\n    return this.unitHeight$.value;\n  }\n  @Input() public set unitHeight(value: number) {\n    this.unitHeight$.next(value);\n  }\n  //#endregion\n\n  private dateEquals(date1: Date, date2: Date) {\n    return (\n      date1.getFullYear() === date2.getFullYear() &&\n      date1.getMonth() === date2.getMonth() &&\n      date1.getDate() === date2.getDate()\n    );\n  }\n\n  private addDays(date: Date, days: number) {\n    const result = new Date(date);\n    result.setDate(result.getDate() + days);\n    return result;\n  }\n\n  onPreviousWeek() {\n    this.currentWeek$\n      .pipe(map((w) => this.addDays(w, -7)), take(1))\n      .subscribe((w) => this.currentWeek$.next(w));\n  }\n\n  onNextWeek() {\n    this.currentWeek$\n      .pipe(map((w) => this.addDays(w, 7)), take(1))\n      .subscribe((w) => this.currentWeek$.next(w));\n  }\n\n  operation: DragOperation | null = null;\n  dragStartTimeslot: TimeSlot | null = null;\n  onCreateEvent(ev: MouseEvent, slot: TimeSlot) {\n    ev.preventDefault();\n    this.mouseState$.next(true);\n    this.dragStartTimeslot = slot;\n    this.operation = {\n      operation: EDragOperation.createEvent,\n      event: {\n        start: slot.start,\n        end: slot.end,\n        color: '#F00',\n        description: 'Test event',\n      }\n    };\n    this.events$.next([...this.events$.value, this.operation.event]);\n  }\n\n  onStartDragEvent(eventPart: SchedulerEventPart, ev: MouseEvent) {\n    ev.preventDefault();\n    this.hoveredTimeSlot$.pipe(take(1)).subscribe((hoveredTimeSlot) => {\n      this.dragStartTimeslot = hoveredTimeSlot;\n      this.operation = {\n        operation: EDragOperation.moveEvent,\n        event: eventPart.event\n      }\n    });\n  }\n\n  //#region hoveredTimeslot$\n  private getHoveredTimeslot(ev: MouseEvent, timeSlots: TimeSlot[][]) {\n    const hoveredSlots = this.timeSlotElements.filter((el) => {\n      const rct = el.nativeElement.getBoundingClientRect();\n      if (rct.left <= ev.x && ev.x <= rct.right && rct.top <= ev.y && ev.y <= rct.bottom) {\n        return true;\n      } else {\n        return false;\n      }\n    });\n\n    if (!hoveredSlots || hoveredSlots.length === 0) {\n      return null;\n    }\n\n    const slotElement = hoveredSlots[0].nativeElement;\n\n    const strRow = slotElement.getAttribute('data-row');\n    if (!strRow) {\n      return null;\n    }\n    const row = parseInt(strRow);\n\n    const strColumn = slotElement.getAttribute('data-column');\n    if (!strColumn) {\n      return null;\n    }\n    const column = parseInt(strColumn);\n\n    const slot = timeSlots[row][column];\n    return slot;\n  }\n\n  @HostListener('document:mousemove', ['$event'])\n  onMousemove(ev: MouseEvent) {\n    this.timeSlots$.pipe(take(1)).subscribe((timeSlots) => {\n      const hovered = this.getHoveredTimeslot(ev, timeSlots);\n      this.hoveredTimeSlot$.next(hovered);\n\n      if (this.operation) {\n        switch (this.operation.operation) {\n          case EDragOperation.createEvent: {\n            if (this.operation.event && this.dragStartTimeslot && hovered && (this.operation.event.end.getTime() != hovered.end.getTime())) {\n              if (this.dragStartTimeslot.start.getTime() === hovered.start.getTime()) {\n                // 1 slot\n              } else if (this.dragStartTimeslot.start.getTime() < hovered.start.getTime()) {\n                // Drag down\n                this.operation.event.start = this.dragStartTimeslot.start;\n                this.operation.event.end = hovered.end;\n                this.events$.next(this.events$.value);\n              } else if (this.dragStartTimeslot.start.getTime() > hovered.start.getTime()) {\n                // Drag up\n                this.operation.event.start = hovered.start;\n                this.operation.event.end = this.dragStartTimeslot.end;\n                this.events$.next(this.events$.value);\n              }\n              \n            }\n          } break;\n          case EDragOperation.moveEvent: {\n            if (hovered && this.dragStartTimeslot) {\n              this.operation.event.start.setTime(this.operation.event.start.getTime() + hovered.start.getTime() - this.dragStartTimeslot.start.getTime());\n              this.operation.event.end.setTime(this.operation.event.end.getTime() + hovered.start.getTime() - this.dragStartTimeslot.start.getTime());\n              this.dragStartTimeslot = hovered;\n              this.events$.next(this.events$.value);\n            }\n          } break;\n        }\n      }\n\n    });\n  }\n  //#endregion\n\n  @HostListener('document:mouseup', ['$event'])\n  onMouseUp(ev: MouseEvent) {\n    if (this.operation) {\n      switch (this.operation.operation) {\n        case EDragOperation.createEvent: {\n          if (this.operation.event) {\n            this.operation = null;\n            this.dragStartTimeslot = null;\n          }\n        } break;\n        case EDragOperation.moveEvent: {\n          this.operation = null;\n          this.dragStartTimeslot = null;\n        } break;\n      }\n    }\n  }\n\n  ngOnDestroy() {\n    this.destroyed$.next(true);\n  }\n}\n","{{ unitHeight$ | async }}\n<div class=\"table d-flex w-100 overflow-y-auto\" [style.max-height.px]=\"400\">\n    <div class=\"calendar-head\">\n        <div class=\"w-100 d-flex flex-row\">\n            <div class=\"d-flex calendar-left justify-content-between\">\n                <button class=\"btn btn-default flex-start\" (click)=\"onPreviousWeek()\">&lt;</button>\n                <button class=\"btn btn-default flex-end\" (click)=\"onNextWeek()\">&gt;</button>\n            </div>\n            <div class=\"flex-grow-1\" *ngFor=\"let day of (daysOfWeek$ | async)\">\n                <span class=\"d-block col-form-label text-center\">\n                    {{ day | date: 'dd-MM-yyyy' }}\n                </span>\n            </div>\n        </div>\n    </div>\n    <div class=\"calendar-body\">\n        <div class=\"position-relative\">\n            <!-- Timeslots -->\n            <div *ngFor=\"let timeslots of (timeSlots$ | async); let i = index\" class=\"d-flex flex-row p-0 timeslot\" [style.height.px]=\"unitHeight$ | async\">\n                <div class=\"calendar-cell calendar-left align-top py-0\">{{ timeslots[0].start | date: 'HH:mm:ss' }}</div>\n                <div class=\"calendar-cell flex-grow-1\" *ngFor=\"let slot of timeslots; let j = index\" #slot (mousedown)=\"onCreateEvent($event, slot)\" [attr.data-row]=\"i\" [attr.data-column]=\"j\"></div>\n            </div>\n            \n            <!-- Events -->\n            <ng-container *ngIf=\"(timeSlotDuration$ | async) as timeSlotDuration\">\n\n                <div *ngFor=\"let eventPart of (eventPartsForThisWeek$ | async)\" class=\"position-absolute event\"\n                    [style.top.px]=\"(eventPart | bsSecondsTodayOffset) / timeSlotDuration * (unitHeight$ | async)!\"\n                    [style.height.px]=\"(eventPart | bsSecondsTimespan) / timeSlotDuration * (unitHeight$ | async)!\"\n                    [style.left]=\"'calc(90px + ((100% - 90px) / 7 * ' + ((eventPart | dayOfWeek) - 1) + '))'\">\n                    <div class=\"event-inner\" [style.background-color]=\"eventPart.event.color\" (mousedown)=\"onStartDragEvent(eventPart, $event)\">\n                        {{ eventPart.event.description }}\n                        <br>\n                        {{ eventPart.start | date: 'HH:mm:ss' }}\n                        <br>\n                        {{ eventPart.end | date: 'HH:mm:ss' }}\n                    </div>\n                </div>\n                \n                <!-- <ng-container *ngFor=\"let eventWithParts of (eventParts$ | async)\">\n                    <div *ngFor=\"let eventPart of eventWithParts.parts\" class=\"position-absolute event\"\n                        [style.top.px]=\"(eventPart | bsSecondsTodayOffset) / timeSlotDuration * 40\"\n                        [style.height.px]=\"(eventPart | bsSecondsTimespan) / timeSlotDuration * 40\"\n                        [style.left]=\"'calc(90px + ((100% - 90px) / 7 * ' + ((eventPart | dayOfWeek) - 1) + '))'\">\n                        <div class=\"event-inner\">\n                            {{ eventPart.start | date: 'HH:mm:ss' }} - {{ eventPart.end | date: 'HH:mm:ss' }}\n                        </div>\n                    </div>\n                </ng-container> -->\n                \n            </ng-container>\n        </div>\n    </div>\n</div>"]}
421
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"scheduler.component.js","sourceRoot":"","sources":["../../../../../../../../../libs/mintplayer-ng-bootstrap/src/lib/components/scheduler/components/scheduler/scheduler.component.ts","../../../../../../../../../libs/mintplayer-ng-bootstrap/src/lib/components/scheduler/components/scheduler/scheduler.component.html"],"names":[],"mappings":"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","sourcesContent":["import { Component, ElementRef, EventEmitter, HostListener, Input, OnDestroy, Output, QueryList, ViewChildren } from '@angular/core';\nimport { BehaviorSubject, combineLatest, filter, map, Observable, Subject, take, takeUntil, tap } from 'rxjs';\nimport { BsCalendarMonthService } from '../../../../services/calendar-month/calendar-month.service';\nimport { EDragOperation } from '../../enums/drag-operation';\nimport { DragOperation } from '../../interfaces/drag-operation';\nimport { PreviewEvent } from '../../interfaces/preview-event';\nimport { SchedulerEvent } from '../../interfaces/scheduler-event';\nimport { SchedulerEventPart } from '../../interfaces/scheduler-event-part';\nimport { SchedulerEventWithParts } from '../../interfaces/scheduler-event-with-parts';\nimport { TimeSlot } from '../../interfaces/time-slot';\nimport { BsTimelineService } from '../../services/timeline/timeline.service';\n\n@Component({\n  selector: 'bs-scheduler',\n  templateUrl: './scheduler.component.html',\n  styleUrls: ['./scheduler.component.scss'],\n})\nexport class BsSchedulerComponent implements OnDestroy {\n  constructor(private calendarMonthService: BsCalendarMonthService, private timelineService: BsTimelineService) {\n    const monday = this.calendarMonthService.getMondayBefore(new Date());\n    this.currentWeek$ = new BehaviorSubject<Date>(monday);\n\n    this.daysOfWeek$ = this.currentWeek$.pipe(\n      map((weekMonday) => {\n        weekMonday.setHours(0);\n        weekMonday.setMinutes(0);\n        weekMonday.setSeconds(0);\n        weekMonday.setMilliseconds(0);\n        return Array.from(Array(7).keys()).map((x) => this.addDays(weekMonday, x));\n      })\n    );\n\n    this.daysOfWeekWithTimestamps$ = this.daysOfWeek$\n      .pipe(map((daysOfWeek) => {\n        return { start: daysOfWeek[0].getTime(), end: daysOfWeek[daysOfWeek.length - 1].getTime() + 24 * 60 * 60 * 1000 };\n      }));\n\n    this.eventParts$ = this.events$.pipe(\n      map((events) =>  events.map((ev) => this.timelineService.splitInParts(ev)))\n    );\n\n    this.eventPartsForThisWeek$ = combineLatest([\n      this.daysOfWeekWithTimestamps$,\n      this.eventParts$\n        .pipe(map(eventParts => eventParts.map(evp => evp.parts)))\n        .pipe(map(jaggedParts => {\n          return jaggedParts.reduce((flat, toFlatten) => flat.concat(toFlatten), []);\n        }))\n      ])\n      .pipe(map(([startAndEnd, eventParts]) => {\n        return eventParts.filter(eventPart => {\n          return !((eventPart.end.getTime() <= startAndEnd.start) || (eventPart.start.getTime() >= startAndEnd.end));\n        });\n      }));\n\n    this.previewEventParts$ = this.previewEvent$.pipe(\n      map((event) => {\n        if (event) {\n          return this.timelineService.splitInParts(event)\n        } else {\n          return null;\n        }\n      })\n    );\n    this.previewEventPartsForThisWeek$ = combineLatest([this.daysOfWeekWithTimestamps$, this.previewEventParts$])\n      .pipe(map(([startAndEnd, previewEventParts]) => {\n        if (previewEventParts) {\n          return previewEventParts.parts.filter(eventPart => {\n            return !((eventPart.end.getTime() <= startAndEnd.start) || (eventPart.start.getTime() >= startAndEnd.end));\n          });\n        } else {\n          return [];\n        }\n      }));\n\n    this.timelinedEventPartsForThisWeek$ = this.eventPartsForThisWeek$\n      .pipe(map(eventParts => {\n        // We'll only use the events for this week\n        const events = eventParts.map(ep => ep.event)\n          .filter((e, i, list) => list.indexOf(e) === i)\n          .filter((e) => !!e)\n          .map((e) => <SchedulerEvent>e);\n        const timeline = this.timelineService.getTimeline(events);\n\n        const result = timeline.map(track => {\n          return track.events.map(ev => {\n            return { event: ev, index: track.index };\n          });\n        })\n        .reduce((flat, toFlatten) => flat.concat(toFlatten), [])\n        .map((evi) => eventParts.filter(p => p.event === evi.event).map(p => {\n            return { part: p, index: evi.index };\n          })\n        )\n        .reduce((flat, toFlatten) => flat.concat(toFlatten), []);\n\n        return {\n          total: timeline.length,\n          parts: result\n        };\n      }))\n\n    combineLatest([this.daysOfWeek$, this.timeSlotDuration$])\n      .pipe(map(([daysOfWeek, duration]) => {\n        const timeSlotsPerDay = Math.floor((60 * 60 * 24) / duration);\n        return Array.from(Array(timeSlotsPerDay).keys()).map((index) => {\n          const timeSlotStart = new Date(daysOfWeek[0]);\n          timeSlotStart.setTime(+timeSlotStart.getTime() + index * duration * 1000);\n          const timeSlotEnd = new Date(timeSlotStart);\n          timeSlotEnd.setTime(+timeSlotEnd.getTime() + duration * 1000);\n\n          return daysOfWeek.map((day) => {\n            const start = new Date(day);\n            start.setHours(timeSlotStart.getHours());\n            start.setMinutes(timeSlotStart.getMinutes());\n            start.setSeconds(timeSlotStart.getSeconds());\n            start.setMilliseconds(timeSlotStart.getMilliseconds());\n\n            const end = new Date(day);\n            end.setHours(timeSlotEnd.getHours());\n            end.setMinutes(timeSlotEnd.getMinutes());\n            end.setSeconds(timeSlotEnd.getSeconds());\n            end.setMilliseconds(timeSlotEnd.getMilliseconds());\n            end.setDate(end.getDate() + timeSlotEnd.getDate() - timeSlotStart.getDate());\n\n            return <TimeSlot>{ start, end };\n          });\n        });\n      }))\n      .subscribe((timeslots) => {\n        // For performance reasons, we're not using an observable here, but persist the timeslots in a BehaviorSubject.\n        this.timeSlots$.next(timeslots);\n      });\n    \n    this.unitHeight$\n      .pipe(takeUntil(this.destroyed$))\n      .subscribe((unitHeight) => {\n        this.unitHeightChange.emit(unitHeight);\n      })\n  }\n\n  events$ = new BehaviorSubject<SchedulerEvent[]>([]);\n  eventParts$: Observable<SchedulerEventWithParts[]>;\n  eventPartsForThisWeek$: Observable<SchedulerEventPart[]>;\n  timelinedEventPartsForThisWeek$: Observable<{ total: number, parts: { part: SchedulerEventPart, index: number}[] }>;\n  \n  previewEvent$ = new BehaviorSubject<PreviewEvent | null>(null);\n  previewEventParts$: Observable<SchedulerEventWithParts | null>;\n  previewEventPartsForThisWeek$: Observable<SchedulerEventPart[]>;\n  \n  currentWeek$: BehaviorSubject<Date>;\n  daysOfWeek$: Observable<Date[]>;\n  daysOfWeekWithTimestamps$: Observable<{start: number, end: number}>;\n  timeSlotDuration$ = new BehaviorSubject<number>(1800);\n  timeSlots$ = new BehaviorSubject<TimeSlot[][]>([]);\n  mouseState$ = new BehaviorSubject<boolean>(false);\n  hoveredTimeSlot$ = new BehaviorSubject<TimeSlot | null>(null);\n  hoveredEvent$ = new BehaviorSubject<SchedulerEvent | null>(null);\n  destroyed$ = new Subject();\n\n  @ViewChildren('slot') timeSlotElements!: QueryList<ElementRef<HTMLDivElement>>;\n\n  //#region UnitHeight\n  unitHeight$ = new BehaviorSubject<number>(40);\n  @Output() public unitHeightChange = new EventEmitter<number>();\n  public get unitHeight() {\n    return this.unitHeight$.value;\n  }\n  @Input() public set unitHeight(value: number) {\n    this.unitHeight$.next(value);\n  }\n  //#endregion\n  //#region maxInnerHeight\n  @Input() public maxInnerHeight: number | null = null;\n  //#endregion\n\n  private addDays(date: Date, days: number) {\n    const result = new Date(date);\n    result.setDate(result.getDate() + days);\n    return result;\n  }\n\n  onPreviousWeek() {\n    this.currentWeek$\n      .pipe(map((w) => this.addDays(w, -7)), take(1))\n      .subscribe((w) => this.currentWeek$.next(w));\n  }\n\n  onNextWeek() {\n    this.currentWeek$\n      .pipe(map((w) => this.addDays(w, 7)), take(1))\n      .subscribe((w) => this.currentWeek$.next(w));\n  }\n\n  onHoverEvent(ev: SchedulerEvent | null) {\n    console.log('hovered', ev);\n    this.hoveredEvent$.next(ev);\n  }\n\n  onLeaveEvent(ev: SchedulerEvent | null) {\n    this.hoveredEvent$.next(null);\n  }\n\n  operation: DragOperation | null = null;\n  dragStartTimeslot: TimeSlot | null = null;\n  onCreateEvent(ev: MouseEvent, slot: TimeSlot) {\n    ev.preventDefault();\n    this.mouseState$.next(true);\n    this.dragStartTimeslot = slot;\n    this.operation = {\n      operation: EDragOperation.createEvent,\n      event: {\n        start: slot.start,\n        end: slot.end,\n        color: '#F00',\n        description: 'Test event',\n      },\n      meta: null,\n    };\n    this.previewEvent$.next({ start: slot.start, end: slot.end });\n  }\n\n  randomColor() {\n    const brightness = 128;\n    return '#' + this.randomChannel(brightness) + this.randomChannel(brightness) + this.randomChannel(brightness);\n  }\n  randomChannel(brightness: number){\n    const r = 255-brightness;\n    const n = 0|((Math.random() * r) + brightness);\n    const s = n.toString(16);\n    return (s.length==1) ? '0'+s : s;\n  }\n\n  onStartDragEvent(eventPart: SchedulerEventPart, ev: MouseEvent) {\n    ev.preventDefault();\n    this.hoveredTimeSlot$.pipe(take(1)).subscribe((hoveredTimeSlot) => {\n      if (eventPart.event) {\n        this.dragStartTimeslot = hoveredTimeSlot;\n        this.operation = {\n          operation: EDragOperation.moveEvent,\n          event: eventPart.event,\n          meta: null,\n        };\n        this.previewEvent$.next({ start: eventPart.event.start, end: eventPart.event.end });\n      }\n    });\n  }\n\n  onStartResizeEvent(event: SchedulerEvent | null, position: 'top' | 'bottom') {\n    if (event) {\n      switch (position) {\n        case 'top': {\n          this.operation = {\n            operation: EDragOperation.resizeEvent,\n            event: event,\n            meta: { position },\n          }\n          this.previewEvent$.next({ start: event.start, end: event.end });\n        } break;\n        case 'bottom': {\n          this.operation = {\n            operation: EDragOperation.resizeEvent,\n            event: event,\n            meta: { position },\n          }\n          this.previewEvent$.next({ start: event.start, end: event.end });\n        } break;\n      }\n    }\n  }\n\n  //#region hoveredTimeslot$\n  private getHoveredTimeslot(ev: MouseEvent, timeSlots: TimeSlot[][]) {\n    const hoveredSlots = this.timeSlotElements.filter((el) => {\n      const rct = el.nativeElement.getBoundingClientRect();\n      if (rct.left <= ev.x && ev.x <= rct.right && rct.top <= ev.y && ev.y <= rct.bottom) {\n        return true;\n      } else {\n        return false;\n      }\n    });\n\n    if (!hoveredSlots || hoveredSlots.length === 0) {\n      return null;\n    }\n\n    const slotElement = hoveredSlots[0].nativeElement;\n\n    const strRow = slotElement.getAttribute('data-row');\n    if (!strRow) {\n      return null;\n    }\n    const row = parseInt(strRow);\n\n    const strColumn = slotElement.getAttribute('data-column');\n    if (!strColumn) {\n      return null;\n    }\n    const column = parseInt(strColumn);\n\n    const slot = timeSlots[row][column];\n    return slot;\n  }\n\n  @HostListener('document:mousemove', ['$event'])\n  onMousemove(ev: MouseEvent) {\n    this.timeSlots$.pipe(take(1)).subscribe((timeSlots) => {\n      const hovered = this.getHoveredTimeslot(ev, timeSlots);\n      this.hoveredTimeSlot$.next(hovered);\n\n      if (this.operation) {\n        switch (this.operation.operation) {\n          case EDragOperation.createEvent: {\n            if (this.operation.event && this.dragStartTimeslot && hovered && (this.operation.event.end.getTime() != hovered.end.getTime())) {\n              if (this.dragStartTimeslot.start.getTime() === hovered.start.getTime()) {\n                // 1 slot\n              } else if (this.dragStartTimeslot.start.getTime() < hovered.start.getTime()) {\n                // Drag down\n                this.previewEvent$\n                  .pipe(filter((ev) => !!ev && !!this.dragStartTimeslot))\n                  .pipe(map((ev) => {\n                    if (ev && this.dragStartTimeslot) {\n                      ev.start = this.dragStartTimeslot.start;\n                      ev.end = hovered.end;\n                    }\n                    return ev;\n                  }))\n                  .pipe(take(1))\n                  .subscribe((ev) => this.previewEvent$.next(ev));\n              } else if (this.dragStartTimeslot.start.getTime() > hovered.start.getTime()) {\n                // Drag up\n                this.previewEvent$\n                  .pipe(filter((ev) => !!ev && !!this.dragStartTimeslot))\n                  .pipe(map((ev) => {\n                    if (ev && this.dragStartTimeslot) {\n                      ev.start = hovered.start;\n                      ev.end = this.dragStartTimeslot.end;\n                    }\n                    return ev;\n                  }))\n                  .pipe(take(1))\n                  .subscribe((ev) => this.previewEvent$.next(ev));\n              }\n            }\n          } break;\n          case EDragOperation.moveEvent: {\n            if (hovered && this.dragStartTimeslot) {\n              this.previewEvent$\n                .pipe(filter((ev) => !!ev && !!this.dragStartTimeslot))\n                .pipe(map((ev) => {\n                  if (ev && this.dragStartTimeslot) {\n                    const result =  <PreviewEvent>{\n                      start: new Date(ev.start.getTime() + hovered.start.getTime() - this.dragStartTimeslot.start.getTime()),\n                      end: new Date(ev.end.getTime() + hovered.start.getTime() - this.dragStartTimeslot.start.getTime())\n                    };\n                    \n                    this.dragStartTimeslot = hovered;\n\n                    return result;\n                  } else {\n                    return ev;\n                  }\n                }))\n                .pipe(take(1))\n                .subscribe(ev => this.previewEvent$.next(ev));\n            }\n          } break;\n          case EDragOperation.resizeEvent: {\n            if (hovered) {\n              this.previewEvent$\n                .pipe(filter((ev) => !!ev))\n                .pipe(map((ev) => {\n                  if (ev && this.operation && this.operation.event) {\n                    if (this.operation.meta.position === 'top') {\n                      ev.start = hovered.start;\n                      ev.end = this.operation.event.end;\n                    } else if (this.operation.meta.position === 'bottom') {\n                      ev.start = this.operation.event.start;\n                      ev.end = hovered.end;\n                    }\n                  }\n                  return ev;\n                }))\n                .pipe(take(1))\n                .subscribe((ev) => this.previewEvent$.next(ev));\n            }\n          } break;\n        }\n      }\n\n    });\n  }\n  //#endregion\n\n  @HostListener('document:mouseup', ['$event'])\n  onMouseUp(ev: MouseEvent) {\n    if (this.operation) {\n      switch (this.operation.operation) {\n        case EDragOperation.createEvent: {\n          combineLatest([this.previewEvent$])\n            .pipe(take(1))\n            .subscribe(([previewEvent]) => {\n              if (previewEvent) {\n                this.operation = null;\n                this.dragStartTimeslot = null;\n                this.events$.next([...this.events$.value, {\n                  start: previewEvent.start,\n                  end: previewEvent.end,\n                  color: this.randomColor(),\n                  description: 'New event'\n                }]);\n                this.previewEvent$.next(null);\n              }\n            });\n        } break;\n        case EDragOperation.moveEvent:\n        case EDragOperation.resizeEvent: {\n          this.previewEvent$\n            .pipe(filter((ev) => !!ev))\n            .pipe(take(1))\n            .subscribe((previewEvent) => {\n              if (this.operation && this.operation.event && previewEvent) {\n                this.operation.event.start = previewEvent.start;\n                this.operation.event.end = previewEvent.end;\n                this.operation = null;\n                this.dragStartTimeslot = null;\n                this.events$.next(this.events$.value);\n                this.previewEvent$.next(null);\n              }\n            });\n        } break;\n      }\n    }\n  }\n\n  ngOnDestroy() {\n    this.destroyed$.next(true);\n  }\n}\n","<div class=\"table d-flex w-100 overflow-y-auto\" [style.max-height.px]=\"null\">\n    <div class=\"calendar-head\">\n        <div class=\"w-100 d-flex flex-row\">\n            <div class=\"d-flex calendar-left justify-content-between\">\n                <button class=\"btn btn-default flex-start\" (click)=\"onPreviousWeek()\">&lt;</button>\n                <button class=\"btn btn-default flex-end\" (click)=\"onNextWeek()\">&gt;</button>\n            </div>\n            <div class=\"flex-grow-1 calendar-head-cell\" *ngFor=\"let day of (daysOfWeek$ | async)\">\n                <span class=\"d-block col-form-label text-center\">\n                    <span>{{ day | date: 'dd-MM' }}</span><span class=\"d-none d-lg-inline-block\">-{{ day | date: 'yyyy' }}</span>\n                </span>\n            </div>\n        </div>\n    </div>\n    <div class=\"calendar-body\" [style.height.px]=\"maxInnerHeight\">\n        <div class=\"position-relative\">\n            <!-- Timeslots -->\n            <div *ngFor=\"let timeslots of (timeSlots$ | async); let i = index\" class=\"d-flex flex-row p-0 timeslot\" [style.height.px]=\"unitHeight$ | async\">\n                <div class=\"calendar-cell calendar-left align-top px-2 py-0\">{{ timeslots[0].start | date: 'HH:mm:ss' }}</div>\n                <div class=\"calendar-cell flex-grow-1\" *ngFor=\"let slot of timeslots; let j = index\" #slot (mousedown)=\"onCreateEvent($event, slot)\" [attr.data-row]=\"i\" [attr.data-column]=\"j\"></div>\n            </div>\n            \n            <!-- Events -->\n            <ng-container *ngIf=\"(timeSlotDuration$ | async) as timeSlotDuration\">\n                <ng-container *ngIf=\"(unitHeight$ | async) as unitHeight\">\n                    <ng-container *ngIf=\"(timelinedEventPartsForThisWeek$ | async) as partData\">\n                        <div *ngFor=\"let eventPart of partData.parts\" class=\"event p-0\"\n                            [style.top.px]=\"(eventPart.part | bsSecondsTodayOffset) / timeSlotDuration * unitHeight\"\n                            [style.height.px]=\"(eventPart.part | bsSecondsTimespan) / timeSlotDuration * unitHeight\"\n                            [style.left]=\"'calc(90px + ((100% - 90px) / 7 * ' + ((eventPart.part | dayOfWeek) - 1) + '))'\">\n                            <div class=\"event-inner\"\n                                [class.hover]=\"(hoveredEvent$ | async) === eventPart.part.event\"\n                                [style.width]=\"'calc(100% / ' + partData.total + ')'\"\n                                [style.height.px]=\"(eventPart.part | bsSecondsTimespan) / timeSlotDuration * unitHeight\"\n                                [style.margin-left]=\"'calc(100% / ' + partData.total + ' * ' + eventPart.index + ')'\">\n\n                                <div class=\"event-resize top\" *ngIf=\"eventPart.part.start === eventPart.part.event?.start\" (mousedown)=\"onStartResizeEvent(eventPart.part.event, 'top')\"></div>\n                                <div class=\"event-border\"></div>\n                                <div class=\"event-wrapper\" *ngIf=\"eventPart.part.event\"\n                                    (mousedown)=\"onStartDragEvent(eventPart.part, $event)\"\n                                    (mouseenter)=\"onHoverEvent(eventPart.part.event)\"\n                                    (mouseleave)=\"onLeaveEvent(eventPart.part.event)\">\n                                    <div class=\"event-bg\" [style.background-color]=\"eventPart.part.event.color\"></div>\n                                    <div class=\"event-label\">{{ eventPart.part.event.description }}</div>\n                                </div>\n                                <div class=\"event-resize bottom\" *ngIf=\"eventPart.part.end === eventPart.part.event?.end\" (mousedown)=\"onStartResizeEvent(eventPart.part.event, 'bottom')\"></div>\n                            </div>\n                        </div>\n                    </ng-container>\n                    <ng-container *ngIf=\"(previewEventPartsForThisWeek$ | async) as previewPartData\">\n                        <div *ngFor=\"let eventPart of previewPartData\" class=\"event preview p-0\"\n                            [style.top.px]=\"(eventPart | bsSecondsTodayOffset) / timeSlotDuration * unitHeight\"\n                            [style.height.px]=\"(eventPart | bsSecondsTimespan) / timeSlotDuration * unitHeight\"\n                            [style.left]=\"'calc(90px + ((100% - 90px) / 7 * ' + ((eventPart | dayOfWeek) - 1) + '))'\">\n                            <div class=\"event-inner w-100\" [style.height.px]=\"(eventPart | bsSecondsTimespan) / timeSlotDuration * unitHeight\"></div>\n                        </div>\n                    </ng-container>\n                </ng-container>\n            </ng-container>\n        </div>\n    </div>\n</div>"]}