@colijnit/corecomponents_v12 255.1.10 → 255.1.11
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/bundles/colijnit-corecomponents_v12.umd.js +104 -35
- package/bundles/colijnit-corecomponents_v12.umd.js.map +1 -1
- package/colijnit-corecomponents_v12.metadata.json +1 -1
- package/esm2015/lib/components/hour-scheduling/hour-scheduling.component.js +170 -59
- package/fesm2015/colijnit-corecomponents_v12.js +169 -58
- package/fesm2015/colijnit-corecomponents_v12.js.map +1 -1
- package/lib/components/hour-scheduling/hour-scheduling.component.d.ts +19 -7
- package/lib/components/hour-scheduling/style/_layout.scss +30 -13
- package/package.json +1 -1
|
@@ -5,45 +5,57 @@ export class HourSchedulingComponent {
|
|
|
5
5
|
this.cdRef = cdRef;
|
|
6
6
|
this.datePipe = datePipe;
|
|
7
7
|
this.showClass = true;
|
|
8
|
-
this.
|
|
8
|
+
this.hourLabels = [];
|
|
9
9
|
this.scheduledObjects = {};
|
|
10
|
+
this.activeHour = null;
|
|
11
|
+
this.activeHalfHour = null;
|
|
10
12
|
this.draggedObject = null;
|
|
11
13
|
this.customTemplateUsed = false;
|
|
12
14
|
this.timeChangeEvent = new EventEmitter();
|
|
15
|
+
this.newObjectPlanEvent = new EventEmitter();
|
|
16
|
+
document.addEventListener('click', this.onDocumentClick.bind(this));
|
|
13
17
|
}
|
|
14
18
|
ngOnInit() {
|
|
15
|
-
if (!this.childProp) {
|
|
16
|
-
this.startTime = parseInt(this.datePipe.transform(new Date(this.schedule[this.startTimeProp]).toISOString(), 'H:mm'));
|
|
17
|
-
this.endTime = parseInt(this.datePipe.transform(new Date(this.schedule[this.endTimeProp]).toISOString(), 'H:mm'));
|
|
18
|
-
}
|
|
19
|
-
else {
|
|
20
|
-
this.startTime = parseInt(this.datePipe.transform(new Date(this.schedule[this.childProp][this.startTimeProp]).toISOString(), 'H:mm'));
|
|
21
|
-
this.endTime = parseInt(this.datePipe.transform(new Date(this.schedule[this.childProp][this.endTimeProp]).toISOString(), 'H:mm'));
|
|
22
|
-
}
|
|
23
19
|
this.generateTimeBlocks();
|
|
20
|
+
this.generateScheduledObjects();
|
|
21
|
+
}
|
|
22
|
+
ngOnDestroy() {
|
|
23
|
+
document.removeEventListener('click', this.onDocumentClick.bind(this));
|
|
24
24
|
}
|
|
25
25
|
generateTimeBlocks() {
|
|
26
|
+
let startUnix = !this.childProp ? this.dateToUnixEpoch(new Date(this.schedule[this.startTimeProp])) : this.dateToUnixEpoch(new Date(this.schedule[this.childProp][this.startTimeProp]));
|
|
27
|
+
let endUnix = !this.childProp ? this.dateToUnixEpoch(new Date(this.schedule[this.endTimeProp])) : this.dateToUnixEpoch(new Date(this.schedule[this.childProp][this.endTimeProp]));
|
|
28
|
+
let interval = 60 * 60;
|
|
29
|
+
for (let hourCount = startUnix; hourCount <= endUnix; hourCount += interval) {
|
|
30
|
+
let hour = new Date(hourCount * 1000);
|
|
31
|
+
let hourString = `${hour.getHours()}:${hour.getMinutes() === 0 ? '00' : hour.getMinutes()}`;
|
|
32
|
+
this.hourLabels.push(hourString);
|
|
33
|
+
}
|
|
34
|
+
}
|
|
35
|
+
generateScheduledObjects() {
|
|
26
36
|
const objectsList = this.schedule[this.objectsProp];
|
|
27
|
-
|
|
28
|
-
|
|
29
|
-
|
|
37
|
+
let startUnix = !this.childProp ? this.dateToUnixEpoch(new Date(this.schedule[this.startTimeProp])) : this.dateToUnixEpoch(new Date(this.schedule[this.childProp][this.startTimeProp]));
|
|
38
|
+
let endUnix = !this.childProp ? this.dateToUnixEpoch(new Date(this.schedule[this.endTimeProp])) : this.dateToUnixEpoch(new Date(this.schedule[this.childProp][this.endTimeProp]));
|
|
39
|
+
let interval = 30 * 60;
|
|
40
|
+
for (let hourCount = startUnix; hourCount <= endUnix; hourCount += interval) {
|
|
41
|
+
let hour = new Date(hourCount * 1000);
|
|
42
|
+
let hourString = `${hour.getHours()}:${hour.getMinutes() === 0 ? '00' : hour.getMinutes()}`;
|
|
43
|
+
this.scheduledObjects[hourString] = this._getObjectsForHourAndMinutes(hour.getHours(), hour.getMinutes(), objectsList);
|
|
30
44
|
}
|
|
31
45
|
}
|
|
32
|
-
|
|
46
|
+
_getObjectsForHourAndMinutes(hour, minutes, objectsList) {
|
|
33
47
|
const objectsForHour = [];
|
|
34
48
|
objectsList.forEach((obj) => {
|
|
35
|
-
|
|
49
|
+
let split = this.convertToHourNotation(obj[this.startTimeProp]).split(':');
|
|
50
|
+
if (parseInt(split[0]) === hour && parseInt(split[1]) === minutes) {
|
|
36
51
|
objectsForHour.push(Object.assign({}, obj));
|
|
37
52
|
}
|
|
38
53
|
});
|
|
39
54
|
return objectsForHour;
|
|
40
55
|
}
|
|
41
|
-
formatHour(hour) {
|
|
42
|
-
return `${hour}:00`;
|
|
43
|
-
}
|
|
44
56
|
onDragStart(event, obj) {
|
|
45
57
|
const currentTarget = event.currentTarget;
|
|
46
|
-
const currentHourSpan = currentTarget.parentElement.parentElement.querySelector('.hour-label
|
|
58
|
+
const currentHourSpan = currentTarget.parentElement.parentElement.querySelector('.hidden-hour-label');
|
|
47
59
|
const currentHour = currentHourSpan.textContent;
|
|
48
60
|
this.draggedObject = obj;
|
|
49
61
|
event.dataTransfer.setData('text', JSON.stringify({ obj: this.draggedObject, currentHour }));
|
|
@@ -53,30 +65,39 @@ export class HourSchedulingComponent {
|
|
|
53
65
|
}
|
|
54
66
|
onDragOver(event) {
|
|
55
67
|
event.preventDefault();
|
|
68
|
+
const currentTarget = event.currentTarget;
|
|
69
|
+
currentTarget.classList.add('drag-over');
|
|
70
|
+
}
|
|
71
|
+
onDragLeave(event) {
|
|
72
|
+
const currentTarget = event.currentTarget;
|
|
73
|
+
currentTarget.classList.remove('drag-over'); // Remove 'drag-over' class
|
|
56
74
|
}
|
|
57
75
|
onDrop(event, hour) {
|
|
58
76
|
event.preventDefault();
|
|
59
77
|
event.stopPropagation();
|
|
60
|
-
const
|
|
61
|
-
|
|
78
|
+
const currentTarget = event.currentTarget;
|
|
79
|
+
currentTarget.classList.remove('drag-over');
|
|
80
|
+
let parsed = this.tryParseJSONObject(event.dataTransfer.getData("text"));
|
|
81
|
+
if (!parsed) {
|
|
82
|
+
this.newObjectPlanEvent.emit({ currentHour: hour, data: parsed.toString() });
|
|
62
83
|
return;
|
|
63
84
|
}
|
|
64
|
-
const
|
|
65
|
-
const
|
|
66
|
-
const
|
|
85
|
+
const splitHour = hour.split(':');
|
|
86
|
+
const newEndMinutes = splitHour[1] === "00" ? 30 : 0;
|
|
87
|
+
const newEndHour = splitHour[1] === "30" ? (parseInt(splitHour[0]) + 1) : parseInt(splitHour[0]);
|
|
88
|
+
const originalStartHour = this.convertToHourNotation(parsed.obj[this.startTimeProp]);
|
|
67
89
|
// Get the unique identifier value from the object using the `idProp`
|
|
68
|
-
const objId =
|
|
90
|
+
const objId = parsed.obj[this.idProp];
|
|
69
91
|
// Ensure we create a new object to avoid mutation issues
|
|
70
|
-
const updatedObject = Object.assign(Object.assign({},
|
|
92
|
+
const updatedObject = Object.assign(Object.assign({}, parsed.obj), { [this.startTimeProp]: this.createDate(parseInt(splitHour[0]), parseInt(splitHour[1])), [this.endTimeProp]: this.createDate(newEndHour, newEndMinutes) });
|
|
71
93
|
// Remove the object from its old hour block using its unique id
|
|
72
|
-
const originalHourBlock = this.formatHour(originalStartHour);
|
|
94
|
+
const originalHourBlock = this.formatHour(parseInt(originalStartHour.split(':')[0]), parseInt(originalStartHour.split(':')[1]));
|
|
73
95
|
this.scheduledObjects[originalHourBlock] = this.scheduledObjects[originalHourBlock].filter(o => o[this.idProp] !== objId);
|
|
74
96
|
// Add the object to the new hour block
|
|
75
|
-
|
|
76
|
-
|
|
77
|
-
this.scheduledObjects[newHourBlock] = [];
|
|
97
|
+
if (!this.scheduledObjects[hour]) {
|
|
98
|
+
this.scheduledObjects[hour] = [];
|
|
78
99
|
}
|
|
79
|
-
this.scheduledObjects[
|
|
100
|
+
this.scheduledObjects[hour].push(updatedObject);
|
|
80
101
|
// Clear the dragged object
|
|
81
102
|
this.draggedObject = null;
|
|
82
103
|
this.timeChangeEvent.emit(updatedObject);
|
|
@@ -84,50 +105,139 @@ export class HourSchedulingComponent {
|
|
|
84
105
|
this.cdRef.detectChanges();
|
|
85
106
|
}
|
|
86
107
|
convertToHourNotation(date) {
|
|
87
|
-
return
|
|
108
|
+
return this.datePipe.transform(new Date(date).toISOString(), 'HH:mm');
|
|
88
109
|
}
|
|
89
|
-
createDate(hours) {
|
|
110
|
+
createDate(hours, minutes) {
|
|
90
111
|
let date = new Date();
|
|
91
112
|
date.setHours(hours);
|
|
92
|
-
date.setMinutes(
|
|
113
|
+
date.setMinutes(minutes ? minutes : 0);
|
|
93
114
|
return date;
|
|
94
115
|
}
|
|
116
|
+
dateToUnixEpoch(date) {
|
|
117
|
+
return Math.floor(date.getTime()) / 1000;
|
|
118
|
+
}
|
|
119
|
+
formatHour(hour, minutes) {
|
|
120
|
+
return `${hour}:${minutes ? minutes : '00'}`;
|
|
121
|
+
}
|
|
122
|
+
addMinutes(hour) {
|
|
123
|
+
let split = hour.split(":");
|
|
124
|
+
split[1] = "30";
|
|
125
|
+
return split.join(':');
|
|
126
|
+
}
|
|
127
|
+
onDocumentClick(event) {
|
|
128
|
+
const targetElement = event.target;
|
|
129
|
+
// Clear active state when clicking outside
|
|
130
|
+
if (!targetElement.closest('.object-display')) {
|
|
131
|
+
this.activeHalfHour = null;
|
|
132
|
+
this.cdRef.detectChanges();
|
|
133
|
+
}
|
|
134
|
+
}
|
|
135
|
+
onObjectDisplayClick(hour, half) {
|
|
136
|
+
var _a, _b;
|
|
137
|
+
const halfHourIdentifier = `${hour}-${half}`;
|
|
138
|
+
if (this.activeHalfHour === halfHourIdentifier) {
|
|
139
|
+
// Deactivate if clicked again
|
|
140
|
+
this.activeHalfHour = null;
|
|
141
|
+
}
|
|
142
|
+
else if (((_a = this.scheduledObjects[hour]) === null || _a === void 0 ? void 0 : _a.length) > 0 || ((_b = this.scheduledObjects[this.addMinutes(hour)]) === null || _b === void 0 ? void 0 : _b.length) > 0) {
|
|
143
|
+
// Activate the half-hour if there are objects
|
|
144
|
+
this.activeHalfHour = halfHourIdentifier;
|
|
145
|
+
}
|
|
146
|
+
}
|
|
147
|
+
tryParseJSONObject(jsonString) {
|
|
148
|
+
try {
|
|
149
|
+
let o = JSON.parse(jsonString);
|
|
150
|
+
// Handle non-exception-throwing cases:
|
|
151
|
+
// Neither JSON.parse(false) or JSON.parse(1234) throw errors, hence the type-checking,
|
|
152
|
+
// but... JSON.parse(null) returns null, and typeof null === "object",
|
|
153
|
+
// so we must check for that, too. Thankfully, null is falsey, so this suffices:
|
|
154
|
+
if (o && typeof o === "object") {
|
|
155
|
+
return o;
|
|
156
|
+
}
|
|
157
|
+
}
|
|
158
|
+
catch (e) {
|
|
159
|
+
}
|
|
160
|
+
return false;
|
|
161
|
+
}
|
|
162
|
+
;
|
|
95
163
|
}
|
|
96
164
|
HourSchedulingComponent.decorators = [
|
|
97
165
|
{ type: Component, args: [{
|
|
98
166
|
selector: 'co-hour-scheduling',
|
|
99
167
|
template: `
|
|
100
|
-
<div class="time-block" *ngFor="let hour of
|
|
168
|
+
<div class="time-block" *ngFor="let hour of hourLabels">
|
|
101
169
|
<div class="hour-label"><span [textContent]="hour"></span></div>
|
|
102
|
-
<div
|
|
103
|
-
class="object-
|
|
104
|
-
|
|
105
|
-
|
|
106
|
-
|
|
107
|
-
|
|
108
|
-
|
|
109
|
-
|
|
110
|
-
|
|
111
|
-
class="
|
|
112
|
-
|
|
113
|
-
|
|
114
|
-
|
|
115
|
-
|
|
116
|
-
|
|
117
|
-
|
|
170
|
+
<div class="object-display">
|
|
171
|
+
<div class="first-half-hour object-half"
|
|
172
|
+
[ngClass]="{'has-objects': scheduledObjects[hour]?.length > 0, 'active': activeHalfHour === hour + '-firstHalf'}"
|
|
173
|
+
(dragover)="onDragOver($event)"
|
|
174
|
+
(dragleave)="onDragLeave($event)"
|
|
175
|
+
(click)="onObjectDisplayClick(hour, 'firstHalf')"
|
|
176
|
+
(drop)="onDrop($event, hour)">
|
|
177
|
+
<ng-container *ngIf="!customTemplateUsed">
|
|
178
|
+
<ng-container *ngIf="scheduledObjects[hour]">
|
|
179
|
+
<span class="hidden-hour-label" [hidden]="true" [textContent]="hour"></span>
|
|
180
|
+
<div
|
|
181
|
+
*ngFor="let obj of scheduledObjects[hour]"
|
|
182
|
+
class="scheduled-object"
|
|
183
|
+
[draggable]="true"
|
|
184
|
+
(dragstart)="onDragStart($event, obj)"
|
|
185
|
+
>
|
|
186
|
+
<span class="hidden-hour-label" [hidden]="true" [textContent]="addMinutes(hour)"></span>
|
|
187
|
+
<span class="title" [textContent]="obj.title"></span>
|
|
188
|
+
<span class="sub-title" [textContent]="obj.subTitle"></span>
|
|
189
|
+
</div>
|
|
190
|
+
</ng-container>
|
|
118
191
|
</ng-container>
|
|
119
|
-
</ng-container>
|
|
120
192
|
|
|
121
|
-
|
|
122
|
-
|
|
123
|
-
|
|
124
|
-
|
|
193
|
+
<ng-container *ngIf="customTemplateUsed">
|
|
194
|
+
<ng-template
|
|
195
|
+
[ngTemplateOutlet]="customTemplate"
|
|
196
|
+
[ngTemplateOutletContext]="{
|
|
125
197
|
hour: hour,
|
|
198
|
+
hiddenHour: hour,
|
|
126
199
|
objects: scheduledObjects[hour]
|
|
127
200
|
}"
|
|
128
|
-
|
|
129
|
-
|
|
130
|
-
|
|
201
|
+
>
|
|
202
|
+
</ng-template>
|
|
203
|
+
</ng-container>
|
|
204
|
+
</div>
|
|
205
|
+
|
|
206
|
+
<div class="second-half-hour object-half"
|
|
207
|
+
[ngClass]="{'has-objects': scheduledObjects[addMinutes(hour)]?.length > 0, 'active': activeHalfHour === hour + '-secondHalf'}"
|
|
208
|
+
(dragover)="onDragOver($event)"
|
|
209
|
+
(dragleave)="onDragLeave($event)"
|
|
210
|
+
(click)="onObjectDisplayClick(hour, 'secondHalf')"
|
|
211
|
+
(drop)="onDrop($event, addMinutes(hour))">
|
|
212
|
+
<ng-container *ngIf="!customTemplateUsed">
|
|
213
|
+
<ng-container *ngIf="scheduledObjects[addMinutes(hour)]">
|
|
214
|
+
<div
|
|
215
|
+
*ngFor="let obj of scheduledObjects[addMinutes(hour)]"
|
|
216
|
+
class="scheduled-object"
|
|
217
|
+
[draggable]="true"
|
|
218
|
+
(dragstart)="onDragStart($event, obj)"
|
|
219
|
+
>
|
|
220
|
+
<span class="title" [textContent]="obj.title"></span>
|
|
221
|
+
<span class="sub-title" [textContent]="obj.subTitle"></span>
|
|
222
|
+
<span class="hidden-hour-label" [hidden]="true" [textContent]="addMinutes(hour)"></span>
|
|
223
|
+
</div>
|
|
224
|
+
</ng-container>
|
|
225
|
+
</ng-container>
|
|
226
|
+
|
|
227
|
+
<ng-container *ngIf="customTemplateUsed">
|
|
228
|
+
<ng-template
|
|
229
|
+
[ngTemplateOutlet]="customTemplate"
|
|
230
|
+
[ngTemplateOutletContext]="{
|
|
231
|
+
hour: hour,
|
|
232
|
+
hiddenHour: addMinutes(hour),
|
|
233
|
+
objects: scheduledObjects[addMinutes(hour)]
|
|
234
|
+
}"
|
|
235
|
+
>
|
|
236
|
+
</ng-template>
|
|
237
|
+
</ng-container>
|
|
238
|
+
</div>
|
|
239
|
+
|
|
240
|
+
|
|
131
241
|
</div>
|
|
132
242
|
</div>
|
|
133
243
|
`,
|
|
@@ -148,6 +258,7 @@ HourSchedulingComponent.propDecorators = {
|
|
|
148
258
|
customTemplateUsed: [{ type: Input }],
|
|
149
259
|
idProp: [{ type: Input }],
|
|
150
260
|
timeChangeEvent: [{ type: Output }],
|
|
261
|
+
newObjectPlanEvent: [{ type: Output }],
|
|
151
262
|
generateTimeBlocks: [{ type: HostBinding, args: ['class.co-hour-scheduling',] }]
|
|
152
263
|
};
|
|
153
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"hour-scheduling.component.js","sourceRoot":"","sources":["../../../../../../projects/corecomponents/src/lib/components/hour-scheduling/hour-scheduling.component.ts"],"names":[],"mappings":"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","sourcesContent":["import {\r\n  Component,\r\n  Input,\r\n  OnInit,\r\n  HostBinding,\r\n  ViewEncapsulation,\r\n  ChangeDetectorRef,\r\n  TemplateRef, Output, EventEmitter,\r\n} from '@angular/core';\r\nimport {DatePipe} from '@angular/common';\r\nimport {OUTPUT} from \"three/examples/jsm/postprocessing/SAOPass\";\r\n\r\ninterface SchedulingObject {\r\n  start: number;\r\n  end: number;\r\n  title?: string;\r\n  subTitle?: string;\r\n\r\n  [key: string]: any; // Allows dynamic properties including unique identifiers\r\n}\r\n\r\n@Component({\r\n  selector: 'co-hour-scheduling',\r\n  template: `\r\n    <div class=\"time-block\" *ngFor=\"let hour of hours\">\r\n      <div class=\"hour-label\"><span [textContent]=\"hour\"></span></div>\r\n      <div\r\n        class=\"object-display\"\r\n        (dragover)=\"onDragOver($event)\"\r\n        (drop)=\"onDrop($event, hour)\"\r\n      >\r\n        <ng-container *ngIf=\"!customTemplateUsed\">\r\n          <ng-container *ngIf=\"scheduledObjects[hour]\">\r\n            <div\r\n              *ngFor=\"let obj of scheduledObjects[hour]\"\r\n              class=\"scheduled-object\"\r\n              [draggable]=\"true\"\r\n              (dragstart)=\"onDragStart($event, obj)\"\r\n            >\r\n              <span class=\"title\" [textContent]=\"obj.title\"></span>\r\n              <span class=\"sub-title\" [textContent]=\"obj.subTitle\"></span>\r\n            </div>\r\n          </ng-container>\r\n        </ng-container>\r\n\r\n        <ng-container *ngIf=\"customTemplateUsed\">\r\n          <ng-template\r\n            [ngTemplateOutlet]=\"customTemplate\"\r\n            [ngTemplateOutletContext]=\"{\r\n              hour: hour,\r\n              objects: scheduledObjects[hour]\r\n            }\"\r\n          >\r\n          </ng-template>\r\n        </ng-container>\r\n      </div>\r\n    </div>\r\n  `,\r\n  encapsulation: ViewEncapsulation.None,\r\n})\r\nexport class HourSchedulingComponent implements OnInit {\r\n\r\n  public showClass = true;\r\n  public startTime: number;\r\n  public endTime: number;\r\n  public hours: string[] = [];\r\n  public scheduledObjects: { [hour: string]: SchedulingObject[] } = {};\r\n\r\n  private draggedObject: any | null = null;\r\n\r\n  @Input() public schedule: any;\r\n  @Input() public startTimeProp: string;\r\n  @Input() public endTimeProp: string;\r\n  @Input() public objectsProp: string;\r\n  @Input() public childProp: string;\r\n  @Input() public customTemplate: TemplateRef<any>;\r\n  @Input() public customTemplateUsed: boolean = false;\r\n  @Input() public idProp: string;  // Input for unique identifier property\r\n\r\n  @Output() public timeChangeEvent: EventEmitter<Object> = new EventEmitter<Object>();\r\n\r\n  constructor(private cdRef: ChangeDetectorRef, private datePipe: DatePipe) {\r\n  }\r\n\r\n  ngOnInit() {\r\n    if (!this.childProp) {\r\n      this.startTime = parseInt(\r\n        this.datePipe.transform(\r\n          new Date(this.schedule[this.startTimeProp]).toISOString(),\r\n          'H:mm'\r\n        )\r\n      );\r\n      this.endTime = parseInt(\r\n        this.datePipe.transform(\r\n          new Date(this.schedule[this.endTimeProp]).toISOString(),\r\n          'H:mm'\r\n        )\r\n      );\r\n    } else {\r\n      this.startTime = parseInt(\r\n        this.datePipe.transform(\r\n          new Date(this.schedule[this.childProp][this.startTimeProp]).toISOString(),\r\n          'H:mm'\r\n        )\r\n      );\r\n      this.endTime = parseInt(\r\n        this.datePipe.transform(\r\n          new Date(this.schedule[this.childProp][this.endTimeProp]).toISOString(),\r\n          'H:mm'\r\n        )\r\n      );\r\n    }\r\n    this.generateTimeBlocks();\r\n  }\r\n\r\n  @HostBinding('class.co-hour-scheduling')\r\n\r\n  public generateTimeBlocks() {\r\n    const objectsList = this.schedule[this.objectsProp];\r\n\r\n    for (let hour = this.startTime; hour <= this.endTime; hour++) {\r\n      this.hours.push(this.formatHour(hour));\r\n      this.scheduledObjects[this.formatHour(hour)] = this._getObjectsForHour(\r\n        hour,\r\n        objectsList\r\n      );\r\n    }\r\n  }\r\n\r\n  private _getObjectsForHour(\r\n    hour: number,\r\n    objectsList: any[]\r\n  ): SchedulingObject[] {\r\n    const objectsForHour: SchedulingObject[] = [];\r\n\r\n    objectsList.forEach((obj) => {\r\n      if (this.convertToHourNotation(obj[this.startTimeProp]) === hour) {\r\n        objectsForHour.push({...obj});\r\n      }\r\n    });\r\n\r\n    return objectsForHour;\r\n  }\r\n\r\n  public formatHour(hour: number): string {\r\n    return `${hour}:00`;\r\n  }\r\n\r\n  public onDragStart(event: DragEvent, obj: any) {\r\n    const currentTarget = event.currentTarget as HTMLElement;\r\n    const currentHourSpan =\r\n      currentTarget.parentElement!.parentElement!.querySelector(\r\n        '.hour-label span'\r\n      )!;\r\n    const currentHour = currentHourSpan.textContent!;\r\n\r\n    this.draggedObject = obj;\r\n    event.dataTransfer!.setData(\r\n      'text',\r\n      JSON.stringify({obj: this.draggedObject, currentHour})\r\n    );\r\n  }\r\n\r\n  public onCustomDragStart(event: DragEvent, obj: SchedulingObject) {\r\n    this.onDragStart(event, obj);\r\n  }\r\n\r\n  public onDragOver(event: DragEvent) {\r\n    event.preventDefault();\r\n  }\r\n\r\n  public onDrop(event: DragEvent, hour: string) {\r\n    event.preventDefault();\r\n    event.stopPropagation();\r\n\r\n    const data = JSON.parse(event.dataTransfer!.getData('text')) as {\r\n      obj: any;\r\n      currentHour: string;\r\n    };\r\n\r\n    if (!data || !data.obj) {\r\n      return;\r\n    }\r\n\r\n    const newStartHour = parseInt(hour.split(':')[0], 10);\r\n    const newEndHour = newStartHour + 1;\r\n\r\n    const originalStartHour = this.convertToHourNotation(data.obj[this.startTimeProp]);\r\n\r\n    // Get the unique identifier value from the object using the `idProp`\r\n    const objId = data.obj[this.idProp];\r\n\r\n    // Ensure we create a new object to avoid mutation issues\r\n\r\n    const updatedObject = {\r\n      ...data.obj,\r\n      [this.startTimeProp]: this.createDate(newStartHour),\r\n      [this.endTimeProp]: this.createDate(newEndHour)\r\n    };\r\n\r\n\r\n    // Remove the object from its old hour block using its unique id\r\n    const originalHourBlock = this.formatHour(originalStartHour);\r\n    this.scheduledObjects[originalHourBlock] = this.scheduledObjects[originalHourBlock].filter(o => o[this.idProp] !== objId);\r\n\r\n    // Add the object to the new hour block\r\n    const newHourBlock = this.formatHour(newStartHour);\r\n    if (!this.scheduledObjects[newHourBlock]) {\r\n      this.scheduledObjects[newHourBlock] = [];\r\n    }\r\n    this.scheduledObjects[newHourBlock].push(updatedObject);\r\n\r\n    // Clear the dragged object\r\n    this.draggedObject = null;\r\n\r\n    this.timeChangeEvent.emit(updatedObject);\r\n    // Trigger change detection to update the view\r\n    this.cdRef.detectChanges();\r\n  }\r\n\r\n  public convertToHourNotation(date: Date): number {\r\n    return parseInt(this.datePipe.transform(new Date(date).toISOString(), 'H:mm'));\r\n  }\r\n\r\n  public createDate(hours: number) {\r\n    let date = new Date();\r\n    date.setHours(hours);\r\n    date.setMinutes(0, 0, 0);\r\n    return date;\r\n  }\r\n}\r\n"]}
|
|
264
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"hour-scheduling.component.js","sourceRoot":"","sources":["../../../../../../projects/corecomponents/src/lib/components/hour-scheduling/hour-scheduling.component.ts"],"names":[],"mappings":"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","sourcesContent":["import {\r\n  Component,\r\n  Input,\r\n  OnInit,\r\n  HostBinding,\r\n  ViewEncapsulation,\r\n  ChangeDetectorRef,\r\n  TemplateRef, Output, EventEmitter,\r\n} from '@angular/core';\r\nimport {DatePipe} from '@angular/common';\r\nimport {OUTPUT} from \"three/examples/jsm/postprocessing/SAOPass\";\r\nimport {min} from \"three/examples/jsm/nodes/math/MathNode\";\r\nimport {parse} from \"@angular/compiler/src/render3/view/style_parser\";\r\n\r\ninterface SchedulingObject {\r\n  start: number;\r\n  end: number;\r\n  title?: string;\r\n  subTitle?: string;\r\n\r\n  [key: string]: any; // Allows dynamic properties including unique identifiers\r\n}\r\n\r\n@Component({\r\n  selector: 'co-hour-scheduling',\r\n  template: `\r\n    <div class=\"time-block\" *ngFor=\"let hour of hourLabels\">\r\n      <div class=\"hour-label\"><span [textContent]=\"hour\"></span></div>\r\n      <div class=\"object-display\">\r\n        <div class=\"first-half-hour object-half\"\r\n             [ngClass]=\"{'has-objects': scheduledObjects[hour]?.length > 0, 'active': activeHalfHour === hour + '-firstHalf'}\"\r\n             (dragover)=\"onDragOver($event)\"\r\n             (dragleave)=\"onDragLeave($event)\"\r\n             (click)=\"onObjectDisplayClick(hour, 'firstHalf')\"\r\n             (drop)=\"onDrop($event, hour)\">\r\n          <ng-container *ngIf=\"!customTemplateUsed\">\r\n            <ng-container *ngIf=\"scheduledObjects[hour]\">\r\n              <span class=\"hidden-hour-label\" [hidden]=\"true\" [textContent]=\"hour\"></span>\r\n              <div\r\n                *ngFor=\"let obj of scheduledObjects[hour]\"\r\n                class=\"scheduled-object\"\r\n                [draggable]=\"true\"\r\n                (dragstart)=\"onDragStart($event, obj)\"\r\n              >\r\n                <span class=\"hidden-hour-label\" [hidden]=\"true\" [textContent]=\"addMinutes(hour)\"></span>\r\n                <span class=\"title\" [textContent]=\"obj.title\"></span>\r\n                <span class=\"sub-title\" [textContent]=\"obj.subTitle\"></span>\r\n              </div>\r\n            </ng-container>\r\n          </ng-container>\r\n\r\n          <ng-container *ngIf=\"customTemplateUsed\">\r\n            <ng-template\r\n              [ngTemplateOutlet]=\"customTemplate\"\r\n              [ngTemplateOutletContext]=\"{\r\n              hour: hour,\r\n              hiddenHour: hour,\r\n              objects: scheduledObjects[hour]\r\n            }\"\r\n            >\r\n            </ng-template>\r\n          </ng-container>\r\n        </div>\r\n\r\n        <div class=\"second-half-hour object-half\"\r\n             [ngClass]=\"{'has-objects': scheduledObjects[addMinutes(hour)]?.length > 0, 'active': activeHalfHour === hour + '-secondHalf'}\"\r\n             (dragover)=\"onDragOver($event)\"\r\n             (dragleave)=\"onDragLeave($event)\"\r\n             (click)=\"onObjectDisplayClick(hour, 'secondHalf')\"\r\n             (drop)=\"onDrop($event, addMinutes(hour))\">\r\n          <ng-container *ngIf=\"!customTemplateUsed\">\r\n            <ng-container *ngIf=\"scheduledObjects[addMinutes(hour)]\">\r\n              <div\r\n                *ngFor=\"let obj of scheduledObjects[addMinutes(hour)]\"\r\n                class=\"scheduled-object\"\r\n                [draggable]=\"true\"\r\n                (dragstart)=\"onDragStart($event, obj)\"\r\n              >\r\n                <span class=\"title\" [textContent]=\"obj.title\"></span>\r\n                <span class=\"sub-title\" [textContent]=\"obj.subTitle\"></span>\r\n                <span class=\"hidden-hour-label\" [hidden]=\"true\" [textContent]=\"addMinutes(hour)\"></span>\r\n              </div>\r\n            </ng-container>\r\n          </ng-container>\r\n\r\n          <ng-container *ngIf=\"customTemplateUsed\">\r\n            <ng-template\r\n              [ngTemplateOutlet]=\"customTemplate\"\r\n              [ngTemplateOutletContext]=\"{\r\n              hour: hour,\r\n              hiddenHour: addMinutes(hour),\r\n              objects: scheduledObjects[addMinutes(hour)]\r\n            }\"\r\n            >\r\n            </ng-template>\r\n          </ng-container>\r\n        </div>\r\n\r\n\r\n      </div>\r\n    </div>\r\n  `,\r\n  encapsulation: ViewEncapsulation.None,\r\n})\r\nexport class HourSchedulingComponent implements OnInit {\r\n\r\n  public showClass = true;\r\n  public hourLabels: string[] = [];\r\n  public scheduledObjects: { [hour: string]: SchedulingObject[] } = {};\r\n  public activeHour: string | null = null;\r\n  public activeHalfHour: string | null = null;\r\n\r\n  private draggedObject: any | null = null;\r\n\r\n  @Input() public schedule: any;\r\n  @Input() public startTimeProp: string;\r\n  @Input() public endTimeProp: string;\r\n  @Input() public objectsProp: string;\r\n  @Input() public childProp: string;\r\n  @Input() public customTemplate: TemplateRef<any>;\r\n  @Input() public customTemplateUsed: boolean = false;\r\n  @Input() public idProp: string;  // Input for unique identifier property\r\n\r\n  @Output() public timeChangeEvent: EventEmitter<Object> = new EventEmitter<Object>();\r\n  @Output() public newObjectPlanEvent: EventEmitter<{ currentHour: string, data: string }> = new EventEmitter<{\r\n    currentHour: string,\r\n    data: string\r\n  }>();\r\n\r\n\r\n  constructor(private cdRef: ChangeDetectorRef, private datePipe: DatePipe) {\r\n    document.addEventListener('click', this.onDocumentClick.bind(this));\r\n  }\r\n\r\n  ngOnInit() {\r\n    this.generateTimeBlocks();\r\n    this.generateScheduledObjects();\r\n  }\r\n\r\n  ngOnDestroy() {\r\n    document.removeEventListener('click', this.onDocumentClick.bind(this));\r\n  }\r\n\r\n  @HostBinding('class.co-hour-scheduling')\r\n\r\n  public generateTimeBlocks() {\r\n    let startUnix: number = !this.childProp ? this.dateToUnixEpoch(new Date(this.schedule[this.startTimeProp])) : this.dateToUnixEpoch(new Date(this.schedule[this.childProp][this.startTimeProp]));\r\n    let endUnix: number = !this.childProp ? this.dateToUnixEpoch(new Date(this.schedule[this.endTimeProp])) : this.dateToUnixEpoch(new Date(this.schedule[this.childProp][this.endTimeProp]));\r\n\r\n    let interval = 60 * 60;\r\n\r\n    for (let hourCount = startUnix; hourCount <= endUnix; hourCount += interval) {\r\n      let hour = new Date(hourCount * 1000)\r\n      let hourString = `${hour.getHours()}:${hour.getMinutes() === 0 ? '00' : hour.getMinutes()}`;\r\n      this.hourLabels.push(hourString);\r\n    }\r\n  }\r\n\r\n  public generateScheduledObjects() {\r\n    const objectsList = this.schedule[this.objectsProp];\r\n\r\n    let startUnix: number = !this.childProp ? this.dateToUnixEpoch(new Date(this.schedule[this.startTimeProp])) : this.dateToUnixEpoch(new Date(this.schedule[this.childProp][this.startTimeProp]));\r\n    let endUnix: number = !this.childProp ? this.dateToUnixEpoch(new Date(this.schedule[this.endTimeProp])) : this.dateToUnixEpoch(new Date(this.schedule[this.childProp][this.endTimeProp]));\r\n\r\n    let interval = 30 * 60;\r\n\r\n    for (let hourCount = startUnix; hourCount <= endUnix; hourCount += interval) {\r\n      let hour = new Date(hourCount * 1000)\r\n      let hourString = `${hour.getHours()}:${hour.getMinutes() === 0 ? '00' : hour.getMinutes()}`;\r\n      this.scheduledObjects[hourString] = this._getObjectsForHourAndMinutes(\r\n        hour.getHours(),\r\n        hour.getMinutes(),\r\n        objectsList\r\n      );\r\n    }\r\n  }\r\n\r\n  private _getObjectsForHourAndMinutes(\r\n    hour: number,\r\n    minutes: number,\r\n    objectsList: any[]\r\n  ): SchedulingObject[] {\r\n    const objectsForHour: SchedulingObject[] = [];\r\n    objectsList.forEach((obj) => {\r\n      let split = this.convertToHourNotation(obj[this.startTimeProp]).split(':');\r\n      if (parseInt(split[0]) === hour && parseInt(split[1]) === minutes) {\r\n        objectsForHour.push({...obj});\r\n      }\r\n    });\r\n\r\n    return objectsForHour;\r\n  }\r\n\r\n\r\n  public onDragStart(event: DragEvent, obj: any) {\r\n    const currentTarget = event.currentTarget as HTMLElement;\r\n    const currentHourSpan =\r\n      currentTarget.parentElement!.parentElement!.querySelector(\r\n        '.hidden-hour-label'\r\n      )!;\r\n    const currentHour = currentHourSpan.textContent!;\r\n\r\n    this.draggedObject = obj;\r\n    event.dataTransfer!.setData(\r\n      'text',\r\n      JSON.stringify({obj: this.draggedObject, currentHour})\r\n    );\r\n  }\r\n\r\n  public onCustomDragStart(event: DragEvent, obj: SchedulingObject) {\r\n    this.onDragStart(event, obj);\r\n  }\r\n\r\n  public onDragOver(event: DragEvent) {\r\n    event.preventDefault();\r\n    const currentTarget = event.currentTarget as HTMLElement;\r\n    currentTarget.classList.add('drag-over');\r\n  }\r\n\r\n  public onDragLeave(event: DragEvent) {\r\n    const currentTarget = event.currentTarget as HTMLElement;\r\n    currentTarget.classList.remove('drag-over');  // Remove 'drag-over' class\r\n  }\r\n\r\n  public onDrop(event: DragEvent, hour: string) {\r\n    event.preventDefault();\r\n    event.stopPropagation();\r\n\r\n    const currentTarget = event.currentTarget as HTMLElement;\r\n    currentTarget.classList.remove('drag-over');\r\n\r\n    let parsed = this.tryParseJSONObject(event.dataTransfer!.getData(\"text\"));\r\n\r\n    if (!parsed) {\r\n      this.newObjectPlanEvent.emit({currentHour: hour, data: parsed.toString()});\r\n      return;\r\n    }\r\n\r\n\r\n    const splitHour = hour.split(':');\r\n\r\n    const newEndMinutes = splitHour[1] === \"00\" ? 30 : 0;\r\n    const newEndHour = splitHour[1] === \"30\" ? (parseInt(splitHour[0]) + 1) : parseInt(splitHour[0]);\r\n\r\n    const originalStartHour = this.convertToHourNotation(parsed.obj[this.startTimeProp]);\r\n    // Get the unique identifier value from the object using the `idProp`\r\n    const objId = parsed.obj[this.idProp];\r\n\r\n    // Ensure we create a new object to avoid mutation issues\r\n\r\n    const updatedObject = {\r\n      ...parsed.obj,\r\n      [this.startTimeProp]: this.createDate(parseInt(splitHour[0]), parseInt(splitHour[1])),\r\n      [this.endTimeProp]: this.createDate(newEndHour, newEndMinutes)\r\n    };\r\n\r\n\r\n    // Remove the object from its old hour block using its unique id\r\n    const originalHourBlock = this.formatHour(parseInt(originalStartHour.split(':')[0]), parseInt(originalStartHour.split(':')[1]));\r\n    this.scheduledObjects[originalHourBlock] = this.scheduledObjects[originalHourBlock].filter(o => o[this.idProp] !== objId);\r\n\r\n    // Add the object to the new hour block\r\n\r\n    if (!this.scheduledObjects[hour]) {\r\n      this.scheduledObjects[hour] = [];\r\n    }\r\n    this.scheduledObjects[hour].push(updatedObject);\r\n\r\n    // Clear the dragged object\r\n    this.draggedObject = null;\r\n\r\n    this.timeChangeEvent.emit(updatedObject);\r\n    // Trigger change detection to update the view\r\n    this.cdRef.detectChanges();\r\n\r\n  }\r\n\r\n  public convertToHourNotation(date: Date): string {\r\n    return this.datePipe.transform(new Date(date).toISOString(), 'HH:mm');\r\n  }\r\n\r\n  public createDate(hours: number, minutes?: number) {\r\n    let date = new Date();\r\n    date.setHours(hours);\r\n    date.setMinutes(minutes ? minutes : 0);\r\n    return date;\r\n  }\r\n\r\n  public dateToUnixEpoch(date: Date): number {\r\n    return Math.floor(date.getTime()) / 1000;\r\n  }\r\n\r\n  public formatHour(hour: number, minutes?: number): string {\r\n    return `${hour}:${minutes ? minutes : '00'}`;\r\n  }\r\n\r\n  public addMinutes(hour: string) {\r\n    let split = hour.split(\":\");\r\n    split[1] = \"30\";\r\n    return split.join(':');\r\n  }\r\n\r\n  public onDocumentClick(event: Event) {\r\n    const targetElement = event.target as HTMLElement;\r\n\r\n    // Clear active state when clicking outside\r\n    if (!targetElement.closest('.object-display')) {\r\n      this.activeHalfHour = null;\r\n      this.cdRef.detectChanges();\r\n    }\r\n  }\r\n\r\n  public onObjectDisplayClick(hour: string, half: 'firstHalf' | 'secondHalf') {\r\n    const halfHourIdentifier = `${hour}-${half}`;\r\n\r\n    if (this.activeHalfHour === halfHourIdentifier) {\r\n      // Deactivate if clicked again\r\n      this.activeHalfHour = null;\r\n    } else if (this.scheduledObjects[hour]?.length > 0 || this.scheduledObjects[this.addMinutes(hour)]?.length > 0) {\r\n      // Activate the half-hour if there are objects\r\n      this.activeHalfHour = halfHourIdentifier;\r\n    }\r\n  }\r\n\r\n  public tryParseJSONObject(jsonString: string) {\r\n    try {\r\n      let o = JSON.parse(jsonString);\r\n\r\n      // Handle non-exception-throwing cases:\r\n      // Neither JSON.parse(false) or JSON.parse(1234) throw errors, hence the type-checking,\r\n      // but... JSON.parse(null) returns null, and typeof null === \"object\",\r\n      // so we must check for that, too. Thankfully, null is falsey, so this suffices:\r\n      if (o && typeof o === \"object\") {\r\n        return o;\r\n      }\r\n    } catch (e) {\r\n    }\r\n\r\n    return false;\r\n  };\r\n}\r\n"]}
|