@valtimo/task 12.11.0 → 12.12.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 (33) hide show
  1. package/esm2022/lib/components/assign-user-to-task/assign-user-to-task.component.mjs +128 -43
  2. package/esm2022/lib/components/set-task-due-date/set-task-due-date.component.mjs +128 -0
  3. package/esm2022/lib/components/task-detail-content/task-detail-content.component.mjs +24 -12
  4. package/esm2022/lib/components/task-detail-intermediate-save/task-detail-intermediate-save.component.mjs +3 -3
  5. package/esm2022/lib/components/task-detail-modal/task-detail-modal.component.mjs +14 -4
  6. package/esm2022/lib/components/task-list/task-list.component.mjs +4 -3
  7. package/esm2022/lib/models/task.model.mjs +1 -1
  8. package/esm2022/lib/services/task.service.mjs +7 -1
  9. package/esm2022/lib/task-permissions.mjs +7 -2
  10. package/esm2022/lib/task.module.mjs +7 -3
  11. package/esm2022/public_api.mjs +2 -1
  12. package/fesm2022/valtimo-task.mjs +307 -77
  13. package/fesm2022/valtimo-task.mjs.map +1 -1
  14. package/lib/components/assign-user-to-task/assign-user-to-task.component.d.ts +38 -17
  15. package/lib/components/assign-user-to-task/assign-user-to-task.component.d.ts.map +1 -1
  16. package/lib/components/set-task-due-date/set-task-due-date.component.d.ts +36 -0
  17. package/lib/components/set-task-due-date/set-task-due-date.component.d.ts.map +1 -0
  18. package/lib/components/task-detail-content/task-detail-content.component.d.ts +1 -1
  19. package/lib/components/task-detail-content/task-detail-content.component.d.ts.map +1 -1
  20. package/lib/components/task-detail-intermediate-save/task-detail-intermediate-save.component.d.ts.map +1 -1
  21. package/lib/components/task-detail-modal/task-detail-modal.component.d.ts +1 -0
  22. package/lib/components/task-detail-modal/task-detail-modal.component.d.ts.map +1 -1
  23. package/lib/models/task.model.d.ts +4 -1
  24. package/lib/models/task.model.d.ts.map +1 -1
  25. package/lib/services/task.service.d.ts +3 -1
  26. package/lib/services/task.service.d.ts.map +1 -1
  27. package/lib/task-permissions.d.ts +2 -1
  28. package/lib/task-permissions.d.ts.map +1 -1
  29. package/lib/task.module.d.ts +2 -1
  30. package/lib/task.module.d.ts.map +1 -1
  31. package/package.json +1 -1
  32. package/public_api.d.ts +1 -0
  33. package/public_api.d.ts.map +1 -1
@@ -14,43 +14,61 @@
14
14
  * limitations under the License.
15
15
  */
16
16
  import { Component, EventEmitter, Input, Output, } from '@angular/core';
17
- import { SearchableDropdownSelectModule } from '@valtimo/components';
18
- import { BehaviorSubject, combineLatest, Subscription, take, tap } from 'rxjs';
17
+ import { RemoveClassnamesDirective, SearchableDropdownSelectModule, } from '@valtimo/components';
18
+ import { BehaviorSubject, combineLatest, Subject, Subscription, take, tap } from 'rxjs';
19
19
  import { CommonModule } from '@angular/common';
20
20
  import { TranslateModule } from '@ngx-translate/core';
21
+ import { ButtonModule, ComboBoxModule, DatePickerModule, IconModule, LayerModule, ToggletipModule, } from 'carbon-components-angular';
22
+ import { UserFollow16 } from '@carbon/icons';
23
+ import { filter, map } from 'rxjs/operators';
21
24
  import * as i0 from "@angular/core";
22
25
  import * as i1 from "../../services";
23
- import * as i2 from "@angular/common";
24
- import * as i3 from "@ngx-translate/core";
25
- import * as i4 from "@valtimo/components";
26
+ import * as i2 from "carbon-components-angular";
27
+ import * as i3 from "@valtimo/components";
28
+ import * as i4 from "@angular/common";
29
+ import * as i5 from "@ngx-translate/core";
30
+ import * as i6 from "carbon-components-angular/dropdown";
26
31
  export class AssignUserToTaskComponent {
27
- constructor(taskService) {
32
+ set canAssignUserToTask(value) {
33
+ this.canAssignUserToTaskSet$.next(true);
34
+ this.canAssignUserToTask$.next(value);
35
+ }
36
+ constructor(taskService, iconService, elementRef, renderer2, cdsThemeService) {
28
37
  this.taskService = taskService;
38
+ this.iconService = iconService;
39
+ this.elementRef = elementRef;
40
+ this.renderer2 = renderer2;
41
+ this.cdsThemeService = cdsThemeService;
29
42
  this.assignmentOfTaskChanged = new EventEmitter();
43
+ this.canAssignUserToTaskSet$ = new BehaviorSubject(false);
44
+ this.canAssignUserToTask$ = new BehaviorSubject(false);
30
45
  this.assignedIdOnServer$ = new BehaviorSubject(null);
31
- this.assignedUserFullName$ = new BehaviorSubject(null);
32
- this.candidateUsersForTask$ = new BehaviorSubject(undefined);
46
+ this._assignedUserFullName$ = new BehaviorSubject(null);
47
+ this.assignedUserFullName$ = this._assignedUserFullName$.pipe(map(fullName => `${fullName?.trim()}`));
48
+ this._candidateUsersForTask$ = new BehaviorSubject(undefined);
49
+ this._selectedUserId$ = new BehaviorSubject(null);
50
+ this.selectedUserId$ = this._selectedUserId$.asObservable();
51
+ this.candidateUsersForTask$ = combineLatest([
52
+ this._candidateUsersForTask$,
53
+ this._selectedUserId$,
54
+ ]).pipe(map(([users, selectedUserId]) => this.mapUsersForDropdown(users, selectedUserId)));
55
+ this.mouseIsOverAssignee$ = new BehaviorSubject(false);
56
+ this.open$ = new Subject();
33
57
  this.disabled$ = new BehaviorSubject(true);
34
- this.userIdToAssign = null;
58
+ this.toggletipTheme$ = this.cdsThemeService.toggletipTheme$;
35
59
  this._subscriptions = new Subscription();
60
+ this.iconService.registerAll([UserFollow16]);
36
61
  }
37
62
  ngOnInit() {
38
- this._subscriptions.add(this.taskService.getCandidateUsers(this.taskId).subscribe(candidateUsers => {
39
- this.candidateUsersForTask$.next(candidateUsers);
40
- if (this.assigneeId) {
41
- this.assignedIdOnServer$.next(this.assigneeId);
42
- this.userIdToAssign = this.assigneeId;
43
- this.assignedUserFullName$.next(this.getAssignedUserName(candidateUsers, this.assigneeId));
44
- }
45
- this.enable();
46
- }));
63
+ this.fetchCandidateUsers();
64
+ this.openHideElementSubscription();
47
65
  }
48
66
  ngOnChanges(changes) {
49
- this.candidateUsersForTask$.pipe(take(1)).subscribe(candidateUsers => {
67
+ this._candidateUsersForTask$.pipe(take(1)).subscribe(candidateUsers => {
50
68
  const currentUserId = changes.assigneeId?.currentValue || this.assigneeId;
51
69
  this.assignedIdOnServer$.next(currentUserId || null);
52
- this.userIdToAssign = currentUserId || null;
53
- this.assignedUserFullName$.next(this.getAssignedUserName(candidateUsers ?? [], currentUserId));
70
+ this._selectedUserId$.next(currentUserId || null);
71
+ this._assignedUserFullName$.next(this.getAssignedUserName(candidateUsers ?? [], currentUserId));
54
72
  });
55
73
  }
56
74
  ngOnDestroy() {
@@ -59,26 +77,32 @@ export class AssignUserToTaskComponent {
59
77
  assignTask(userId) {
60
78
  this.disable();
61
79
  combineLatest([
62
- this.candidateUsersForTask$,
80
+ this._candidateUsersForTask$,
63
81
  this.taskService.assignTask(this.taskId, { assignee: userId }),
64
82
  ])
65
- .pipe(take(1), tap(([candidateUsers]) => {
66
- this.userIdToAssign = userId;
67
- this.assignedIdOnServer$.next(userId);
68
- this.assignedUserFullName$.next(this.getAssignedUserName(candidateUsers ?? [], userId));
69
- this.emitChange();
70
- this.enable();
71
- }))
72
- .subscribe();
83
+ .pipe(take(1))
84
+ .subscribe({
85
+ next: ([candidateUsers]) => {
86
+ this._selectedUserId$.next(userId);
87
+ this.assignedIdOnServer$.next(userId);
88
+ this._assignedUserFullName$.next(this.getAssignedUserName(candidateUsers ?? [], userId));
89
+ this.closeToggletip();
90
+ this.emitChange();
91
+ this.enable();
92
+ },
93
+ error: () => {
94
+ this.enable();
95
+ },
96
+ });
73
97
  }
74
98
  unassignTask() {
75
99
  this.disable();
76
100
  this.taskService
77
101
  .unassignTask(this.taskId)
78
102
  .pipe(tap(() => {
79
- this.clear();
80
103
  this.emitChange();
81
104
  this.enable();
105
+ this.clear();
82
106
  }))
83
107
  .subscribe();
84
108
  }
@@ -89,16 +113,30 @@ export class AssignUserToTaskComponent {
89
113
  }
90
114
  return userId || '-';
91
115
  }
92
- mapUsersForDropdown(users) {
93
- return (users &&
94
- users
95
- .map(user => ({ ...user, lastName: user.lastName?.split(' ').splice(-1)[0] || '' }))
96
- .sort((a, b) => a.lastName.localeCompare(b.lastName))
97
- .map(user => ({ text: user.label, id: user.id })));
116
+ onMouseEnterAssignee() {
117
+ this.mouseIsOverAssignee$.next(true);
118
+ }
119
+ onMouseLeaveAssignee() {
120
+ this.mouseIsOverAssignee$.next(false);
121
+ }
122
+ onSubmitButtonClick() {
123
+ this.assignTask(this._selectedUserId$.getValue());
124
+ }
125
+ onUserSelect(event) {
126
+ if (!event?.id)
127
+ return;
128
+ this._selectedUserId$.next(event.id);
98
129
  }
99
130
  clear() {
100
131
  this.assignedIdOnServer$.next(null);
101
- this.userIdToAssign = null;
132
+ this._selectedUserId$.next(null);
133
+ }
134
+ mapUsersForDropdown(users, selectedUserId) {
135
+ return (users
136
+ ?.map(user => ({ ...user, lastName: user.lastName?.split(' ').splice(-1)[0] || '' }))
137
+ .sort((a, b) => a.lastName.localeCompare(b.lastName))
138
+ .map(user => ({ content: user.label, id: user.id, selected: user.id === selectedUserId })) ||
139
+ []);
102
140
  }
103
141
  emitChange() {
104
142
  this.assignmentOfTaskChanged.emit();
@@ -109,17 +147,64 @@ export class AssignUserToTaskComponent {
109
147
  disable() {
110
148
  this.disabled$.next(true);
111
149
  }
112
- static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: AssignUserToTaskComponent, deps: [{ token: i1.TaskService }], target: i0.ɵɵFactoryTarget.Component }); }
113
- static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "17.3.12", type: AssignUserToTaskComponent, isStandalone: true, selector: "valtimo-assign-user-to-task", inputs: { taskId: "taskId", assigneeId: "assigneeId" }, outputs: { assignmentOfTaskChanged: "assignmentOfTaskChanged" }, usesOnChanges: true, ngImport: i0, template: "<!--\n ~ Copyright 2015-2024 Ritense BV, the Netherlands.\n ~\n ~ Licensed under EUPL, Version 1.2 (the \"License\");\n ~ you may not use this file except in compliance with the License.\n ~ You may obtain a copy of the License at\n ~\n ~ https://joinup.ec.europa.eu/collection/eupl/eupl-text-eupl-12\n ~\n ~ Unless required by applicable law or agreed to in writing, software\n ~ distributed under the License is distributed on an \"AS IS\" basis,\n ~ WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.\n ~ See the License for the specific language governing permissions and\n ~ limitations under the License.\n -->\n\n<ng-container\n *ngIf=\"{\n candidateUsers: candidateUsersForTask$ | async,\n disabled: disabled$ | async,\n idOnServer: assignedIdOnServer$ | async,\n } as obs\"\n>\n <div class=\"container-fluid\">\n <div class=\"mt-2 mb-2\">\n <div class=\"col-12 pl-0 d-flex flex-row align-items-center\">\n <valtimo-searchable-dropdown-select\n *ngIf=\"obs.candidateUsers; else loading\"\n [style]=\"'underlinedText'\"\n [items]=\"mapUsersForDropdown(obs.candidateUsers)\"\n [buttonText]=\"'assignTask.header' | translate\"\n [searchText]=\"'interface.typeToSearch' | translate\"\n [noResultsText]=\"'interface.noSearchResults' | translate\"\n [disabled]=\"obs.disabled\"\n [selectedText]=\"'assignTask.assignedTo' | translate\"\n [selectedTextValue]=\"assignedUserFullName$ | async\"\n [clearSelectionButtonTitle]=\"'assignTask.remove' | translate\"\n [hasSelection]=\"userIdToAssign === obs.idOnServer && obs.idOnServer !== null\"\n [width]=\"250\"\n (itemSelected)=\"assignTask($event)\"\n (clearSelection)=\"unassignTask()\"\n >\n </valtimo-searchable-dropdown-select>\n </div>\n </div>\n </div>\n</ng-container>\n\n<ng-template #loading>\n <h5>\n <b>{{ 'assignTask.fetchingUsers' | translate }}</b>\n </h5>\n</ng-template>\n", styles: [".container-fluid{color:#959595;padding:0;margin:0}i{font-size:13px}\n/*!\n * Copyright 2015-2024 Ritense BV, the Netherlands.\n *\n * Licensed under EUPL, Version 1.2 (the \"License\");\n * you may not use this file except in compliance with the License.\n * You may obtain a copy of the License at\n *\n * https://joinup.ec.europa.eu/collection/eupl/eupl-text-eupl-12\n *\n * Unless required by applicable law or agreed to in writing, software\n * distributed under the License is distributed on an \"AS IS\" basis,\n * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.\n * See the License for the specific language governing permissions and\n * limitations under the License.\n */\n"], dependencies: [{ kind: "ngmodule", type: CommonModule }, { kind: "directive", type: i2.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "pipe", type: i2.AsyncPipe, name: "async" }, { kind: "ngmodule", type: TranslateModule }, { kind: "pipe", type: i3.TranslatePipe, name: "translate" }, { kind: "ngmodule", type: SearchableDropdownSelectModule }, { kind: "component", type: i4.SearchableDropdownSelectComponent, selector: "valtimo-searchable-dropdown-select", inputs: ["style", "items", "buttonText", "searchText", "noResultsText", "disabled", "selectedText", "selectedTextValue", "clearSelectionButtonTitle", "hasSelection", "width", "hasPermission", "showClearSelection"], outputs: ["itemSelected", "clearSelection"] }] }); }
150
+ closeToggletip() {
151
+ // needed to reliably trigger toggle tip closure
152
+ this.open$.next(true);
153
+ setTimeout(() => this.open$.next(false));
154
+ }
155
+ fetchCandidateUsers() {
156
+ this.canAssignUserToTask$
157
+ .pipe(filter(allowed => !!allowed), take(1))
158
+ .subscribe(() => {
159
+ this.taskService.getCandidateUsers(this.taskId).subscribe(candidateUsers => {
160
+ this._candidateUsersForTask$.next(candidateUsers);
161
+ if (this.assigneeId) {
162
+ this.assignedIdOnServer$.next(this.assigneeId);
163
+ this._selectedUserId$.next(this.assigneeId);
164
+ this._assignedUserFullName$.next(this.getAssignedUserName(candidateUsers, this.assigneeId));
165
+ }
166
+ this.enable();
167
+ });
168
+ });
169
+ }
170
+ openHideElementSubscription() {
171
+ this._subscriptions.add(combineLatest([
172
+ this.selectedUserId$,
173
+ this.assignedIdOnServer$,
174
+ this.canAssignUserToTask$,
175
+ ]).subscribe(([selectedUserId, idOnServer, canAssignUserToTask]) => {
176
+ if (!canAssignUserToTask && !(selectedUserId === idOnServer && idOnServer !== null)) {
177
+ this.renderer2.setStyle(this.elementRef.nativeElement, 'display', 'none');
178
+ }
179
+ else {
180
+ this.renderer2.removeStyle(this.elementRef.nativeElement, 'display');
181
+ }
182
+ }));
183
+ }
184
+ static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: AssignUserToTaskComponent, deps: [{ token: i1.TaskService }, { token: i2.IconService }, { token: i0.ElementRef }, { token: i0.Renderer2 }, { token: i3.CdsThemeService }], target: i0.ɵɵFactoryTarget.Component }); }
185
+ static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "17.3.12", type: AssignUserToTaskComponent, isStandalone: true, selector: "valtimo-assign-user-to-task", inputs: { taskId: "taskId", assigneeId: "assigneeId", canAssignUserToTask: "canAssignUserToTask" }, outputs: { assignmentOfTaskChanged: "assignmentOfTaskChanged" }, usesOnChanges: true, ngImport: i0, template: "<!--\n ~ Copyright 2015-2024 Ritense BV, the Netherlands.\n ~\n ~ Licensed under EUPL, Version 1.2 (the \"License\");\n ~ you may not use this file except in compliance with the License.\n ~ You may obtain a copy of the License at\n ~\n ~ https://joinup.ec.europa.eu/collection/eupl/eupl-text-eupl-12\n ~\n ~ Unless required by applicable law or agreed to in writing, software\n ~ distributed under the License is distributed on an \"AS IS\" basis,\n ~ WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.\n ~ See the License for the specific language governing permissions and\n ~ limitations under the License.\n -->\n\n<ng-container\n *ngIf=\"\n (canAssignUserToTaskSet$ | async) && {\n candidateUsers: candidateUsersForTask$ | async,\n disabled: disabled$ | async,\n idOnServer: assignedIdOnServer$ | async,\n mouseIsOverAssignee: mouseIsOverAssignee$ | async,\n assignedUserFullName: assignedUserFullName$ | async,\n selectedUserId: selectedUserId$ | async,\n canAssignUserToTask: canAssignUserToTask$ | async,\n } as obs\n \"\n>\n <ng-container\n *ngIf=\"{hasSelection: obs.selectedUserId === obs.idOnServer && obs.idOnServer !== null} as vars\"\n >\n <div\n class=\"assign-user-container\"\n *ngIf=\"obs.canAssignUserToTask || (!obs.canAssignUserToTask && vars.hasSelection)\"\n (mouseenter)=\"onMouseEnterAssignee()\"\n (mouseleave)=\"onMouseLeaveAssignee()\"\n >\n <div\n class=\"assignee-text element\"\n [ngClass]=\"{\n active: obs.canAssignUserToTask\n ? vars.hasSelection && !obs.mouseIsOverAssignee\n : vars.hasSelection,\n }\"\n >\n <span class=\"bold\">{{ 'assignTask.assignedTo' | translate }}</span>\n\n &nbsp;\n\n <span class=\"name\">{{ obs.assignedUserFullName }}</span>\n </div>\n\n <button\n *ngIf=\"obs.canAssignUserToTask\"\n cdsButton=\"ghost\"\n class=\"element remove-button\"\n [ngClass]=\"{active: vars.hasSelection && obs.mouseIsOverAssignee}\"\n [disabled]=\"obs.disabled\"\n size=\"sm\"\n (click)=\"unassignTask()\"\n >\n {{ 'assignTask.remove' | translate }}\n </button>\n\n <cds-toggletip\n align=\"bottom\"\n class=\"element main\"\n [autoAlign]=\"true\"\n [isOpen]=\"open$ | async\"\n [ngClass]=\"{active: !vars.hasSelection}\"\n (onOpen)=\"clear()\"\n >\n <button\n cdsToggletipButton\n [removeClassnames]=\"['cds--toggletip-button']\"\n cdsButton=\"tertiary\"\n size=\"sm\"\n class=\"set-assignee-button\"\n >\n {{ 'assignTask.buttonText' | translate }}\n\n <svg class=\"cds--btn__icon\" cdsIcon=\"user--follow\" size=\"16\"></svg>\n </button>\n\n <div\n cdsToggletipContent\n [attr.data-carbon-theme]=\"toggletipTheme$ | async\"\n class=\"assign-task-popover-content\"\n >\n <cds-combo-box\n *ngIf=\"!vars.hasSelection\"\n [label]=\"'assignTask.comboboxLabel' | translate\"\n [placeholder]=\"'assignTask.placeholder' | translate\"\n [appendInline]=\"true\"\n [dropUp]=\"false\"\n [cdsLayer]=\"1\"\n [items]=\"obs?.candidateUsers || []\"\n [disabled]=\"obs.disabled\"\n (selected)=\"onUserSelect($event)\"\n (clear)=\"clear()\"\n >\n <cds-dropdown-list onclick=\"event.stopPropagation()\"></cds-dropdown-list>\n </cds-combo-box>\n\n <button\n cdsButton\n class=\"submit-task-button\"\n [disabled]=\"!obs.selectedUserId || obs.disabled\"\n (click)=\"onSubmitButtonClick()\"\n >\n {{ 'assignTask.submitButtonText' | translate }}\n\n <svg class=\"cds--btn__icon\" cdsIcon=\"user--follow\" size=\"16\"></svg>\n </button>\n </div>\n </cds-toggletip>\n </div>\n </ng-container>\n</ng-container>\n", styles: [":host ::ng-deep .cds--popover-content{max-inline-size:unset}.assign-user-container{display:flex;align-items:center;justify-content:center;position:relative;font-weight:400;font-size:14px;line-height:18px;letter-spacing:.16px;color:var(--cds-link-01)}.assign-user-container .bold{font-weight:700;flex-shrink:0}.assign-user-container .name{white-space:nowrap;overflow:hidden;text-overflow:ellipsis}.remove-button{width:100%}.assignee-text{display:flex;justify-content:flex-start;align-items:center;width:100%}.element{opacity:0;visibility:hidden;position:absolute;transition:opacity .3s ease-in-out}.element.active{opacity:1;visibility:visible}.element.main{position:relative}.assign-task-popover-content{width:400px;max-inline-size:400px;display:flex;flex-direction:column;gap:16px}.assign-task-popover-content .cds--btn{width:100%;max-width:unset}.set-assignee-button{width:250px}\n/*!\n * Copyright 2015-2024 Ritense BV, the Netherlands.\n *\n * Licensed under EUPL, Version 1.2 (the \"License\");\n * you may not use this file except in compliance with the License.\n * You may obtain a copy of the License at\n *\n * https://joinup.ec.europa.eu/collection/eupl/eupl-text-eupl-12\n *\n * Unless required by applicable law or agreed to in writing, software\n * distributed under the License is distributed on an \"AS IS\" basis,\n * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.\n * See the License for the specific language governing permissions and\n * limitations under the License.\n */\n"], dependencies: [{ kind: "ngmodule", type: CommonModule }, { kind: "directive", type: i4.NgClass, selector: "[ngClass]", inputs: ["class", "ngClass"] }, { kind: "directive", type: i4.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "pipe", type: i4.AsyncPipe, name: "async" }, { kind: "ngmodule", type: TranslateModule }, { kind: "pipe", type: i5.TranslatePipe, name: "translate" }, { kind: "ngmodule", type: SearchableDropdownSelectModule }, { kind: "ngmodule", type: ButtonModule }, { kind: "directive", type: i2.Button, selector: "[cdsButton], [ibmButton]", inputs: ["ibmButton", "cdsButton", "size", "skeleton", "iconOnly", "isExpressive"] }, { kind: "ngmodule", type: ToggletipModule }, { kind: "component", type: i2.Toggletip, selector: "cds-toggletip, ibm-toggletip", inputs: ["id", "isOpen"] }, { kind: "directive", type: i2.ToggletipButton, selector: "[cdsToggletipButton], [ibmToggletipButton]", inputs: ["ariaLabel"] }, { kind: "directive", type: i2.ToggletipContent, selector: "[cdsToggletipContent], [ibmToggletipContent]" }, { kind: "ngmodule", type: IconModule }, { kind: "directive", type: i2.IconDirective, selector: "[cdsIcon], [ibmIcon]", inputs: ["ibmIcon", "cdsIcon", "size", "title", "ariaLabel", "ariaLabelledBy", "ariaHidden", "isFocusable"] }, { kind: "ngmodule", type: LayerModule }, { kind: "directive", type: i2.LayerDirective, selector: "[cdsLayer], [ibmLayer]", inputs: ["ibmLayer", "cdsLayer"], exportAs: ["layer"] }, { kind: "ngmodule", type: DatePickerModule }, { kind: "ngmodule", type: ComboBoxModule }, { kind: "component", type: i2.ComboBox, selector: "cds-combo-box, ibm-combo-box", inputs: ["placeholder", "openMenuAria", "closeMenuAria", "clearSelectionsTitle", "clearSelectionsAria", "clearSelectionTitle", "clearSelectionAria", "id", "labelId", "items", "type", "size", "itemValueKey", "label", "hideLabel", "helperText", "appendInline", "invalid", "invalidText", "warn", "warnText", "maxLength", "theme", "selectionFeedback", "autocomplete", "dropUp", "disabled"], outputs: ["selected", "submit", "close", "search", "clear"] }, { kind: "component", type: i6.DropdownList, selector: "cds-dropdown-list, ibm-dropdown-list", inputs: ["ariaLabel", "items", "listTpl", "type", "showTitles"], outputs: ["select", "scroll", "blurIntent"] }, { kind: "directive", type: RemoveClassnamesDirective, selector: "[removeClassnames]", inputs: ["removeClassnames"] }] }); }
114
186
  }
115
187
  i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: AssignUserToTaskComponent, decorators: [{
116
188
  type: Component,
117
- args: [{ selector: 'valtimo-assign-user-to-task', standalone: true, imports: [CommonModule, TranslateModule, SearchableDropdownSelectModule], template: "<!--\n ~ Copyright 2015-2024 Ritense BV, the Netherlands.\n ~\n ~ Licensed under EUPL, Version 1.2 (the \"License\");\n ~ you may not use this file except in compliance with the License.\n ~ You may obtain a copy of the License at\n ~\n ~ https://joinup.ec.europa.eu/collection/eupl/eupl-text-eupl-12\n ~\n ~ Unless required by applicable law or agreed to in writing, software\n ~ distributed under the License is distributed on an \"AS IS\" basis,\n ~ WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.\n ~ See the License for the specific language governing permissions and\n ~ limitations under the License.\n -->\n\n<ng-container\n *ngIf=\"{\n candidateUsers: candidateUsersForTask$ | async,\n disabled: disabled$ | async,\n idOnServer: assignedIdOnServer$ | async,\n } as obs\"\n>\n <div class=\"container-fluid\">\n <div class=\"mt-2 mb-2\">\n <div class=\"col-12 pl-0 d-flex flex-row align-items-center\">\n <valtimo-searchable-dropdown-select\n *ngIf=\"obs.candidateUsers; else loading\"\n [style]=\"'underlinedText'\"\n [items]=\"mapUsersForDropdown(obs.candidateUsers)\"\n [buttonText]=\"'assignTask.header' | translate\"\n [searchText]=\"'interface.typeToSearch' | translate\"\n [noResultsText]=\"'interface.noSearchResults' | translate\"\n [disabled]=\"obs.disabled\"\n [selectedText]=\"'assignTask.assignedTo' | translate\"\n [selectedTextValue]=\"assignedUserFullName$ | async\"\n [clearSelectionButtonTitle]=\"'assignTask.remove' | translate\"\n [hasSelection]=\"userIdToAssign === obs.idOnServer && obs.idOnServer !== null\"\n [width]=\"250\"\n (itemSelected)=\"assignTask($event)\"\n (clearSelection)=\"unassignTask()\"\n >\n </valtimo-searchable-dropdown-select>\n </div>\n </div>\n </div>\n</ng-container>\n\n<ng-template #loading>\n <h5>\n <b>{{ 'assignTask.fetchingUsers' | translate }}</b>\n </h5>\n</ng-template>\n", styles: [".container-fluid{color:#959595;padding:0;margin:0}i{font-size:13px}\n/*!\n * Copyright 2015-2024 Ritense BV, the Netherlands.\n *\n * Licensed under EUPL, Version 1.2 (the \"License\");\n * you may not use this file except in compliance with the License.\n * You may obtain a copy of the License at\n *\n * https://joinup.ec.europa.eu/collection/eupl/eupl-text-eupl-12\n *\n * Unless required by applicable law or agreed to in writing, software\n * distributed under the License is distributed on an \"AS IS\" basis,\n * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.\n * See the License for the specific language governing permissions and\n * limitations under the License.\n */\n"] }]
118
- }], ctorParameters: () => [{ type: i1.TaskService }], propDecorators: { taskId: [{
189
+ args: [{ selector: 'valtimo-assign-user-to-task', standalone: true, imports: [
190
+ CommonModule,
191
+ TranslateModule,
192
+ SearchableDropdownSelectModule,
193
+ ButtonModule,
194
+ ToggletipModule,
195
+ IconModule,
196
+ LayerModule,
197
+ DatePickerModule,
198
+ ComboBoxModule,
199
+ RemoveClassnamesDirective,
200
+ ], template: "<!--\n ~ Copyright 2015-2024 Ritense BV, the Netherlands.\n ~\n ~ Licensed under EUPL, Version 1.2 (the \"License\");\n ~ you may not use this file except in compliance with the License.\n ~ You may obtain a copy of the License at\n ~\n ~ https://joinup.ec.europa.eu/collection/eupl/eupl-text-eupl-12\n ~\n ~ Unless required by applicable law or agreed to in writing, software\n ~ distributed under the License is distributed on an \"AS IS\" basis,\n ~ WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.\n ~ See the License for the specific language governing permissions and\n ~ limitations under the License.\n -->\n\n<ng-container\n *ngIf=\"\n (canAssignUserToTaskSet$ | async) && {\n candidateUsers: candidateUsersForTask$ | async,\n disabled: disabled$ | async,\n idOnServer: assignedIdOnServer$ | async,\n mouseIsOverAssignee: mouseIsOverAssignee$ | async,\n assignedUserFullName: assignedUserFullName$ | async,\n selectedUserId: selectedUserId$ | async,\n canAssignUserToTask: canAssignUserToTask$ | async,\n } as obs\n \"\n>\n <ng-container\n *ngIf=\"{hasSelection: obs.selectedUserId === obs.idOnServer && obs.idOnServer !== null} as vars\"\n >\n <div\n class=\"assign-user-container\"\n *ngIf=\"obs.canAssignUserToTask || (!obs.canAssignUserToTask && vars.hasSelection)\"\n (mouseenter)=\"onMouseEnterAssignee()\"\n (mouseleave)=\"onMouseLeaveAssignee()\"\n >\n <div\n class=\"assignee-text element\"\n [ngClass]=\"{\n active: obs.canAssignUserToTask\n ? vars.hasSelection && !obs.mouseIsOverAssignee\n : vars.hasSelection,\n }\"\n >\n <span class=\"bold\">{{ 'assignTask.assignedTo' | translate }}</span>\n\n &nbsp;\n\n <span class=\"name\">{{ obs.assignedUserFullName }}</span>\n </div>\n\n <button\n *ngIf=\"obs.canAssignUserToTask\"\n cdsButton=\"ghost\"\n class=\"element remove-button\"\n [ngClass]=\"{active: vars.hasSelection && obs.mouseIsOverAssignee}\"\n [disabled]=\"obs.disabled\"\n size=\"sm\"\n (click)=\"unassignTask()\"\n >\n {{ 'assignTask.remove' | translate }}\n </button>\n\n <cds-toggletip\n align=\"bottom\"\n class=\"element main\"\n [autoAlign]=\"true\"\n [isOpen]=\"open$ | async\"\n [ngClass]=\"{active: !vars.hasSelection}\"\n (onOpen)=\"clear()\"\n >\n <button\n cdsToggletipButton\n [removeClassnames]=\"['cds--toggletip-button']\"\n cdsButton=\"tertiary\"\n size=\"sm\"\n class=\"set-assignee-button\"\n >\n {{ 'assignTask.buttonText' | translate }}\n\n <svg class=\"cds--btn__icon\" cdsIcon=\"user--follow\" size=\"16\"></svg>\n </button>\n\n <div\n cdsToggletipContent\n [attr.data-carbon-theme]=\"toggletipTheme$ | async\"\n class=\"assign-task-popover-content\"\n >\n <cds-combo-box\n *ngIf=\"!vars.hasSelection\"\n [label]=\"'assignTask.comboboxLabel' | translate\"\n [placeholder]=\"'assignTask.placeholder' | translate\"\n [appendInline]=\"true\"\n [dropUp]=\"false\"\n [cdsLayer]=\"1\"\n [items]=\"obs?.candidateUsers || []\"\n [disabled]=\"obs.disabled\"\n (selected)=\"onUserSelect($event)\"\n (clear)=\"clear()\"\n >\n <cds-dropdown-list onclick=\"event.stopPropagation()\"></cds-dropdown-list>\n </cds-combo-box>\n\n <button\n cdsButton\n class=\"submit-task-button\"\n [disabled]=\"!obs.selectedUserId || obs.disabled\"\n (click)=\"onSubmitButtonClick()\"\n >\n {{ 'assignTask.submitButtonText' | translate }}\n\n <svg class=\"cds--btn__icon\" cdsIcon=\"user--follow\" size=\"16\"></svg>\n </button>\n </div>\n </cds-toggletip>\n </div>\n </ng-container>\n</ng-container>\n", styles: [":host ::ng-deep .cds--popover-content{max-inline-size:unset}.assign-user-container{display:flex;align-items:center;justify-content:center;position:relative;font-weight:400;font-size:14px;line-height:18px;letter-spacing:.16px;color:var(--cds-link-01)}.assign-user-container .bold{font-weight:700;flex-shrink:0}.assign-user-container .name{white-space:nowrap;overflow:hidden;text-overflow:ellipsis}.remove-button{width:100%}.assignee-text{display:flex;justify-content:flex-start;align-items:center;width:100%}.element{opacity:0;visibility:hidden;position:absolute;transition:opacity .3s ease-in-out}.element.active{opacity:1;visibility:visible}.element.main{position:relative}.assign-task-popover-content{width:400px;max-inline-size:400px;display:flex;flex-direction:column;gap:16px}.assign-task-popover-content .cds--btn{width:100%;max-width:unset}.set-assignee-button{width:250px}\n/*!\n * Copyright 2015-2024 Ritense BV, the Netherlands.\n *\n * Licensed under EUPL, Version 1.2 (the \"License\");\n * you may not use this file except in compliance with the License.\n * You may obtain a copy of the License at\n *\n * https://joinup.ec.europa.eu/collection/eupl/eupl-text-eupl-12\n *\n * Unless required by applicable law or agreed to in writing, software\n * distributed under the License is distributed on an \"AS IS\" basis,\n * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.\n * See the License for the specific language governing permissions and\n * limitations under the License.\n */\n"] }]
201
+ }], ctorParameters: () => [{ type: i1.TaskService }, { type: i2.IconService }, { type: i0.ElementRef }, { type: i0.Renderer2 }, { type: i3.CdsThemeService }], propDecorators: { taskId: [{
119
202
  type: Input
120
203
  }], assigneeId: [{
121
204
  type: Input
122
205
  }], assignmentOfTaskChanged: [{
123
206
  type: Output
207
+ }], canAssignUserToTask: [{
208
+ type: Input
124
209
  }] } });
125
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"assign-user-to-task.component.js","sourceRoot":"","sources":["../../../../../../../projects/valtimo/task/src/lib/components/assign-user-to-task/assign-user-to-task.component.ts","../../../../../../../projects/valtimo/task/src/lib/components/assign-user-to-task/assign-user-to-task.component.html"],"names":[],"mappings":"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","sourcesContent":["/*\n * Copyright 2015-2024 Ritense BV, the Netherlands.\n *\n * Licensed under EUPL, Version 1.2 (the \"License\");\n * you may not use this file except in compliance with the License.\n * You may obtain a copy of the License at\n *\n * https://joinup.ec.europa.eu/collection/eupl/eupl-text-eupl-12\n *\n * Unless required by applicable law or agreed to in writing, software\n * distributed under the License is distributed on an \"AS IS\" basis,\n * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.\n * See the License for the specific language governing permissions and\n * limitations under the License.\n */\nimport {\n  Component,\n  EventEmitter,\n  Input,\n  OnChanges,\n  OnDestroy,\n  OnInit,\n  Output,\n  SimpleChanges,\n} from '@angular/core';\nimport {DropdownItem, SearchableDropdownSelectModule} from '@valtimo/components';\nimport {BehaviorSubject, combineLatest, Subscription, take, tap} from 'rxjs';\nimport {TaskService} from '../../services';\nimport {NamedUser} from '@valtimo/config';\nimport {CommonModule} from '@angular/common';\nimport {TranslateModule} from '@ngx-translate/core';\n\n@Component({\n  selector: 'valtimo-assign-user-to-task',\n  templateUrl: './assign-user-to-task.component.html',\n  styleUrls: ['./assign-user-to-task.component.scss'],\n  standalone: true,\n  imports: [CommonModule, TranslateModule, SearchableDropdownSelectModule],\n})\nexport class AssignUserToTaskComponent implements OnInit, OnChanges, OnDestroy {\n  @Input() taskId: string;\n  @Input() assigneeId: string;\n  @Output() assignmentOfTaskChanged = new EventEmitter();\n\n  public assignedIdOnServer$ = new BehaviorSubject<string | null>(null);\n  public assignedUserFullName$ = new BehaviorSubject<string | null>(null);\n  public candidateUsersForTask$ = new BehaviorSubject<NamedUser[] | undefined>(undefined);\n  public disabled$ = new BehaviorSubject<boolean>(true);\n  public userIdToAssign: string | null = null;\n  private _subscriptions = new Subscription();\n\n  constructor(private taskService: TaskService) {}\n\n  public ngOnInit(): void {\n    this._subscriptions.add(\n      this.taskService.getCandidateUsers(this.taskId).subscribe(candidateUsers => {\n        this.candidateUsersForTask$.next(candidateUsers);\n        if (this.assigneeId) {\n          this.assignedIdOnServer$.next(this.assigneeId);\n          this.userIdToAssign = this.assigneeId;\n          this.assignedUserFullName$.next(\n            this.getAssignedUserName(candidateUsers, this.assigneeId)\n          );\n        }\n        this.enable();\n      })\n    );\n  }\n\n  public ngOnChanges(changes: SimpleChanges): void {\n    this.candidateUsersForTask$.pipe(take(1)).subscribe(candidateUsers => {\n      const currentUserId = changes.assigneeId?.currentValue || this.assigneeId;\n      this.assignedIdOnServer$.next(currentUserId || null);\n      this.userIdToAssign = currentUserId || null;\n      this.assignedUserFullName$.next(\n        this.getAssignedUserName(candidateUsers ?? [], currentUserId)\n      );\n    });\n  }\n\n  public ngOnDestroy(): void {\n    this._subscriptions.unsubscribe();\n  }\n\n  public assignTask(userId: string): void {\n    this.disable();\n    combineLatest([\n      this.candidateUsersForTask$,\n      this.taskService.assignTask(this.taskId, {assignee: userId}),\n    ])\n      .pipe(\n        take(1),\n        tap(([candidateUsers]) => {\n          this.userIdToAssign = userId;\n          this.assignedIdOnServer$.next(userId);\n          this.assignedUserFullName$.next(this.getAssignedUserName(candidateUsers ?? [], userId));\n          this.emitChange();\n          this.enable();\n        })\n      )\n      .subscribe();\n  }\n\n  public unassignTask(): void {\n    this.disable();\n    this.taskService\n      .unassignTask(this.taskId)\n      .pipe(\n        tap(() => {\n          this.clear();\n          this.emitChange();\n          this.enable();\n        })\n      )\n      .subscribe();\n  }\n\n  public getAssignedUserName(users: NamedUser[], userId: string): string {\n    if (users && userId) {\n      const findUser =\n        users.find(user => user.id === userId) || users.find(user => user.userName === userId);\n      return findUser ? findUser.label : userId;\n    }\n    return userId || '-';\n  }\n\n  public mapUsersForDropdown(users: NamedUser[]): DropdownItem[] {\n    return (\n      users &&\n      users\n        .map(user => ({...user, lastName: user.lastName?.split(' ').splice(-1)[0] || ''}))\n        .sort((a, b) => a.lastName.localeCompare(b.lastName))\n        .map(user => ({text: user.label, id: user.id}))\n    );\n  }\n\n  private clear(): void {\n    this.assignedIdOnServer$.next(null);\n    this.userIdToAssign = null;\n  }\n\n  private emitChange(): void {\n    this.assignmentOfTaskChanged.emit();\n  }\n\n  private enable(): void {\n    this.disabled$.next(false);\n  }\n\n  private disable(): void {\n    this.disabled$.next(true);\n  }\n}\n","<!--\n  ~ Copyright 2015-2024 Ritense BV, the Netherlands.\n  ~\n  ~ Licensed under EUPL, Version 1.2 (the \"License\");\n  ~ you may not use this file except in compliance with the License.\n  ~ You may obtain a copy of the License at\n  ~\n  ~ https://joinup.ec.europa.eu/collection/eupl/eupl-text-eupl-12\n  ~\n  ~ Unless required by applicable law or agreed to in writing, software\n  ~ distributed under the License is distributed on an \"AS IS\" basis,\n  ~ WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.\n  ~ See the License for the specific language governing permissions and\n  ~ limitations under the License.\n  -->\n\n<ng-container\n  *ngIf=\"{\n    candidateUsers: candidateUsersForTask$ | async,\n    disabled: disabled$ | async,\n    idOnServer: assignedIdOnServer$ | async,\n  } as obs\"\n>\n  <div class=\"container-fluid\">\n    <div class=\"mt-2 mb-2\">\n      <div class=\"col-12 pl-0 d-flex flex-row align-items-center\">\n        <valtimo-searchable-dropdown-select\n          *ngIf=\"obs.candidateUsers; else loading\"\n          [style]=\"'underlinedText'\"\n          [items]=\"mapUsersForDropdown(obs.candidateUsers)\"\n          [buttonText]=\"'assignTask.header' | translate\"\n          [searchText]=\"'interface.typeToSearch' | translate\"\n          [noResultsText]=\"'interface.noSearchResults' | translate\"\n          [disabled]=\"obs.disabled\"\n          [selectedText]=\"'assignTask.assignedTo' | translate\"\n          [selectedTextValue]=\"assignedUserFullName$ | async\"\n          [clearSelectionButtonTitle]=\"'assignTask.remove' | translate\"\n          [hasSelection]=\"userIdToAssign === obs.idOnServer && obs.idOnServer !== null\"\n          [width]=\"250\"\n          (itemSelected)=\"assignTask($event)\"\n          (clearSelection)=\"unassignTask()\"\n        >\n        </valtimo-searchable-dropdown-select>\n      </div>\n    </div>\n  </div>\n</ng-container>\n\n<ng-template #loading>\n  <h5>\n    <b>{{ 'assignTask.fetchingUsers' | translate }}</b>\n  </h5>\n</ng-template>\n"]}
210
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"assign-user-to-task.component.js","sourceRoot":"","sources":["../../../../../../../projects/valtimo/task/src/lib/components/assign-user-to-task/assign-user-to-task.component.ts","../../../../../../../projects/valtimo/task/src/lib/components/assign-user-to-task/assign-user-to-task.component.html"],"names":[],"mappings":"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","sourcesContent":["/*\n * Copyright 2015-2024 Ritense BV, the Netherlands.\n *\n * Licensed under EUPL, Version 1.2 (the \"License\");\n * you may not use this file except in compliance with the License.\n * You may obtain a copy of the License at\n *\n * https://joinup.ec.europa.eu/collection/eupl/eupl-text-eupl-12\n *\n * Unless required by applicable law or agreed to in writing, software\n * distributed under the License is distributed on an \"AS IS\" basis,\n * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.\n * See the License for the specific language governing permissions and\n * limitations under the License.\n */\nimport {\n  Component,\n  ElementRef,\n  EventEmitter,\n  Input,\n  OnChanges,\n  OnDestroy,\n  OnInit,\n  Output,\n  Renderer2,\n  SimpleChanges,\n} from '@angular/core';\nimport {\n  CdsThemeService,\n  RemoveClassnamesDirective,\n  SearchableDropdownSelectModule,\n} from '@valtimo/components';\nimport {BehaviorSubject, combineLatest, Observable, Subject, Subscription, take, tap} from 'rxjs';\nimport {TaskService} from '../../services';\nimport {NamedUser} from '@valtimo/config';\nimport {CommonModule} from '@angular/common';\nimport {TranslateModule} from '@ngx-translate/core';\nimport {\n  ButtonModule,\n  ComboBoxModule,\n  DatePickerModule,\n  IconModule,\n  IconService,\n  LayerModule,\n  ListItem,\n  ToggletipModule,\n} from 'carbon-components-angular';\nimport {UserFollow16} from '@carbon/icons';\nimport {filter, map} from 'rxjs/operators';\n\n@Component({\n  selector: 'valtimo-assign-user-to-task',\n  templateUrl: './assign-user-to-task.component.html',\n  styleUrls: ['./assign-user-to-task.component.scss'],\n  standalone: true,\n  imports: [\n    CommonModule,\n    TranslateModule,\n    SearchableDropdownSelectModule,\n    ButtonModule,\n    ToggletipModule,\n    IconModule,\n    LayerModule,\n    DatePickerModule,\n    ComboBoxModule,\n    RemoveClassnamesDirective,\n  ],\n})\nexport class AssignUserToTaskComponent implements OnInit, OnChanges, OnDestroy {\n  @Input() public readonly taskId: string;\n  @Input() public readonly assigneeId: string;\n  @Output() public readonly assignmentOfTaskChanged = new EventEmitter();\n\n  public readonly canAssignUserToTaskSet$ = new BehaviorSubject<boolean>(false);\n  public readonly canAssignUserToTask$ = new BehaviorSubject<boolean>(false);\n\n  @Input() public set canAssignUserToTask(value: boolean) {\n    this.canAssignUserToTaskSet$.next(true);\n    this.canAssignUserToTask$.next(value);\n  }\n\n  public readonly assignedIdOnServer$ = new BehaviorSubject<string | null>(null);\n  private readonly _assignedUserFullName$ = new BehaviorSubject<string | null>(null);\n  public readonly assignedUserFullName$: Observable<string> = this._assignedUserFullName$.pipe(\n    map(fullName => `${fullName?.trim()}`)\n  );\n\n  private readonly _candidateUsersForTask$ = new BehaviorSubject<NamedUser[] | undefined>(\n    undefined\n  );\n\n  private readonly _selectedUserId$ = new BehaviorSubject<string | null>(null);\n  public readonly selectedUserId$ = this._selectedUserId$.asObservable();\n\n  public readonly candidateUsersForTask$ = combineLatest([\n    this._candidateUsersForTask$,\n    this._selectedUserId$,\n  ]).pipe(map(([users, selectedUserId]) => this.mapUsersForDropdown(users, selectedUserId)));\n\n  public readonly mouseIsOverAssignee$ = new BehaviorSubject<boolean>(false);\n  public readonly open$ = new Subject<boolean>();\n  public readonly disabled$ = new BehaviorSubject<boolean>(true);\n\n  public readonly toggletipTheme$ = this.cdsThemeService.toggletipTheme$;\n\n  private readonly _subscriptions = new Subscription();\n\n  constructor(\n    private readonly taskService: TaskService,\n    private readonly iconService: IconService,\n    private readonly elementRef: ElementRef,\n    private readonly renderer2: Renderer2,\n    private readonly cdsThemeService: CdsThemeService\n  ) {\n    this.iconService.registerAll([UserFollow16]);\n  }\n\n  public ngOnInit(): void {\n    this.fetchCandidateUsers();\n    this.openHideElementSubscription();\n  }\n\n  public ngOnChanges(changes: SimpleChanges): void {\n    this._candidateUsersForTask$.pipe(take(1)).subscribe(candidateUsers => {\n      const currentUserId = changes.assigneeId?.currentValue || this.assigneeId;\n      this.assignedIdOnServer$.next(currentUserId || null);\n      this._selectedUserId$.next(currentUserId || null);\n      this._assignedUserFullName$.next(\n        this.getAssignedUserName(candidateUsers ?? [], currentUserId)\n      );\n    });\n  }\n\n  public ngOnDestroy(): void {\n    this._subscriptions.unsubscribe();\n  }\n\n  public assignTask(userId: string): void {\n    this.disable();\n\n    combineLatest([\n      this._candidateUsersForTask$,\n      this.taskService.assignTask(this.taskId, {assignee: userId}),\n    ])\n      .pipe(take(1))\n      .subscribe({\n        next: ([candidateUsers]) => {\n          this._selectedUserId$.next(userId);\n          this.assignedIdOnServer$.next(userId);\n          this._assignedUserFullName$.next(this.getAssignedUserName(candidateUsers ?? [], userId));\n          this.closeToggletip();\n          this.emitChange();\n          this.enable();\n        },\n        error: () => {\n          this.enable();\n        },\n      });\n  }\n\n  public unassignTask(): void {\n    this.disable();\n    this.taskService\n      .unassignTask(this.taskId)\n      .pipe(\n        tap(() => {\n          this.emitChange();\n          this.enable();\n          this.clear();\n        })\n      )\n      .subscribe();\n  }\n\n  public getAssignedUserName(users: NamedUser[], userId: string): string {\n    if (users && userId) {\n      const findUser =\n        users.find(user => user.id === userId) || users.find(user => user.userName === userId);\n      return findUser ? findUser.label : userId;\n    }\n    return userId || '-';\n  }\n\n  public onMouseEnterAssignee(): void {\n    this.mouseIsOverAssignee$.next(true);\n  }\n\n  public onMouseLeaveAssignee(): void {\n    this.mouseIsOverAssignee$.next(false);\n  }\n\n  public onSubmitButtonClick(): void {\n    this.assignTask(this._selectedUserId$.getValue());\n  }\n  public onUserSelect(event: ListItem): void {\n    if (!event?.id) return;\n    this._selectedUserId$.next(event.id);\n  }\n\n  public clear(): void {\n    this.assignedIdOnServer$.next(null);\n    this._selectedUserId$.next(null);\n  }\n\n  private mapUsersForDropdown(users: NamedUser[], selectedUserId: string): ListItem[] {\n    return (\n      users\n        ?.map(user => ({...user, lastName: user.lastName?.split(' ').splice(-1)[0] || ''}))\n        .sort((a, b) => a.lastName.localeCompare(b.lastName))\n        .map(user => ({content: user.label, id: user.id, selected: user.id === selectedUserId})) ||\n      []\n    );\n  }\n\n  private emitChange(): void {\n    this.assignmentOfTaskChanged.emit();\n  }\n\n  private enable(): void {\n    this.disabled$.next(false);\n  }\n\n  private disable(): void {\n    this.disabled$.next(true);\n  }\n\n  private closeToggletip(): void {\n    // needed to reliably trigger toggle tip closure\n    this.open$.next(true);\n    setTimeout(() => this.open$.next(false));\n  }\n\n  private fetchCandidateUsers(): void {\n    this.canAssignUserToTask$\n      .pipe(\n        filter(allowed => !!allowed),\n        take(1)\n      )\n      .subscribe(() => {\n        this.taskService.getCandidateUsers(this.taskId).subscribe(candidateUsers => {\n          this._candidateUsersForTask$.next(candidateUsers);\n          if (this.assigneeId) {\n            this.assignedIdOnServer$.next(this.assigneeId);\n            this._selectedUserId$.next(this.assigneeId);\n            this._assignedUserFullName$.next(\n              this.getAssignedUserName(candidateUsers, this.assigneeId)\n            );\n          }\n          this.enable();\n        });\n      });\n  }\n\n  private openHideElementSubscription(): void {\n    this._subscriptions.add(\n      combineLatest([\n        this.selectedUserId$,\n        this.assignedIdOnServer$,\n        this.canAssignUserToTask$,\n      ]).subscribe(([selectedUserId, idOnServer, canAssignUserToTask]) => {\n        if (!canAssignUserToTask && !(selectedUserId === idOnServer && idOnServer !== null)) {\n          this.renderer2.setStyle(this.elementRef.nativeElement, 'display', 'none');\n        } else {\n          this.renderer2.removeStyle(this.elementRef.nativeElement, 'display');\n        }\n      })\n    );\n  }\n}\n","<!--\n  ~ Copyright 2015-2024 Ritense BV, the Netherlands.\n  ~\n  ~ Licensed under EUPL, Version 1.2 (the \"License\");\n  ~ you may not use this file except in compliance with the License.\n  ~ You may obtain a copy of the License at\n  ~\n  ~ https://joinup.ec.europa.eu/collection/eupl/eupl-text-eupl-12\n  ~\n  ~ Unless required by applicable law or agreed to in writing, software\n  ~ distributed under the License is distributed on an \"AS IS\" basis,\n  ~ WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.\n  ~ See the License for the specific language governing permissions and\n  ~ limitations under the License.\n  -->\n\n<ng-container\n  *ngIf=\"\n    (canAssignUserToTaskSet$ | async) && {\n      candidateUsers: candidateUsersForTask$ | async,\n      disabled: disabled$ | async,\n      idOnServer: assignedIdOnServer$ | async,\n      mouseIsOverAssignee: mouseIsOverAssignee$ | async,\n      assignedUserFullName: assignedUserFullName$ | async,\n      selectedUserId: selectedUserId$ | async,\n      canAssignUserToTask: canAssignUserToTask$ | async,\n    } as obs\n  \"\n>\n  <ng-container\n    *ngIf=\"{hasSelection: obs.selectedUserId === obs.idOnServer && obs.idOnServer !== null} as vars\"\n  >\n    <div\n      class=\"assign-user-container\"\n      *ngIf=\"obs.canAssignUserToTask || (!obs.canAssignUserToTask && vars.hasSelection)\"\n      (mouseenter)=\"onMouseEnterAssignee()\"\n      (mouseleave)=\"onMouseLeaveAssignee()\"\n    >\n      <div\n        class=\"assignee-text element\"\n        [ngClass]=\"{\n          active: obs.canAssignUserToTask\n            ? vars.hasSelection && !obs.mouseIsOverAssignee\n            : vars.hasSelection,\n        }\"\n      >\n        <span class=\"bold\">{{ 'assignTask.assignedTo' | translate }}</span>\n\n        &nbsp;\n\n        <span class=\"name\">{{ obs.assignedUserFullName }}</span>\n      </div>\n\n      <button\n        *ngIf=\"obs.canAssignUserToTask\"\n        cdsButton=\"ghost\"\n        class=\"element remove-button\"\n        [ngClass]=\"{active: vars.hasSelection && obs.mouseIsOverAssignee}\"\n        [disabled]=\"obs.disabled\"\n        size=\"sm\"\n        (click)=\"unassignTask()\"\n      >\n        {{ 'assignTask.remove' | translate }}\n      </button>\n\n      <cds-toggletip\n        align=\"bottom\"\n        class=\"element main\"\n        [autoAlign]=\"true\"\n        [isOpen]=\"open$ | async\"\n        [ngClass]=\"{active: !vars.hasSelection}\"\n        (onOpen)=\"clear()\"\n      >\n        <button\n          cdsToggletipButton\n          [removeClassnames]=\"['cds--toggletip-button']\"\n          cdsButton=\"tertiary\"\n          size=\"sm\"\n          class=\"set-assignee-button\"\n        >\n          {{ 'assignTask.buttonText' | translate }}\n\n          <svg class=\"cds--btn__icon\" cdsIcon=\"user--follow\" size=\"16\"></svg>\n        </button>\n\n        <div\n          cdsToggletipContent\n          [attr.data-carbon-theme]=\"toggletipTheme$ | async\"\n          class=\"assign-task-popover-content\"\n        >\n          <cds-combo-box\n            *ngIf=\"!vars.hasSelection\"\n            [label]=\"'assignTask.comboboxLabel' | translate\"\n            [placeholder]=\"'assignTask.placeholder' | translate\"\n            [appendInline]=\"true\"\n            [dropUp]=\"false\"\n            [cdsLayer]=\"1\"\n            [items]=\"obs?.candidateUsers || []\"\n            [disabled]=\"obs.disabled\"\n            (selected)=\"onUserSelect($event)\"\n            (clear)=\"clear()\"\n          >\n            <cds-dropdown-list onclick=\"event.stopPropagation()\"></cds-dropdown-list>\n          </cds-combo-box>\n\n          <button\n            cdsButton\n            class=\"submit-task-button\"\n            [disabled]=\"!obs.selectedUserId || obs.disabled\"\n            (click)=\"onSubmitButtonClick()\"\n          >\n            {{ 'assignTask.submitButtonText' | translate }}\n\n            <svg class=\"cds--btn__icon\" cdsIcon=\"user--follow\" size=\"16\"></svg>\n          </button>\n        </div>\n      </cds-toggletip>\n    </div>\n  </ng-container>\n</ng-container>\n"]}
@@ -0,0 +1,128 @@
1
+ /*
2
+ * Copyright 2015-2024 Ritense BV, the Netherlands.
3
+ *
4
+ * Licensed under EUPL, Version 1.2 (the "License");
5
+ * you may not use this file except in compliance with the License.
6
+ * You may obtain a copy of the License at
7
+ *
8
+ * https://joinup.ec.europa.eu/collection/eupl/eupl-text-eupl-12
9
+ *
10
+ * Unless required by applicable law or agreed to in writing, software
11
+ * distributed under the License is distributed on an "AS IS" basis,
12
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13
+ * See the License for the specific language governing permissions and
14
+ * limitations under the License.
15
+ */
16
+ import { Component, Input } from '@angular/core';
17
+ import { CommonModule } from '@angular/common';
18
+ import { TranslateModule } from '@ngx-translate/core';
19
+ import { BehaviorSubject, Subject } from 'rxjs';
20
+ import { filter, map } from 'rxjs/operators';
21
+ import { ButtonModule, DatePickerModule, IconModule, LayerModule, ToggletipModule, } from 'carbon-components-angular';
22
+ import { CalendarAdd16 } from '@carbon/icons';
23
+ import { RemoveClassnamesDirective } from '@valtimo/components';
24
+ import * as i0 from "@angular/core";
25
+ import * as i1 from "carbon-components-angular";
26
+ import * as i2 from "../../services";
27
+ import * as i3 from "@valtimo/components";
28
+ import * as i4 from "@angular/common";
29
+ import * as i5 from "@ngx-translate/core";
30
+ export class SetTaskDueDateComponent {
31
+ set canModifyTask(value) {
32
+ this.canModifyTaskSet$.next(true);
33
+ this.canModifyTask$.next(value);
34
+ }
35
+ get _task() {
36
+ return this._task$.getValue();
37
+ }
38
+ get _selectedDateString() {
39
+ return this.selectedDateString$.getValue();
40
+ }
41
+ set task(value) {
42
+ if (!value)
43
+ return;
44
+ this.hasDueDate$.next(!!value.due);
45
+ this._task$.next(value);
46
+ }
47
+ constructor(iconService, taskService, cdsThemeService) {
48
+ this.iconService = iconService;
49
+ this.taskService = taskService;
50
+ this.cdsThemeService = cdsThemeService;
51
+ this.canModifyTaskSet$ = new BehaviorSubject(false);
52
+ this.canModifyTask$ = new BehaviorSubject(false);
53
+ this._task$ = new BehaviorSubject(null);
54
+ this.hasDueDate$ = new BehaviorSubject(false);
55
+ this.selectedDateString$ = new BehaviorSubject('');
56
+ this.taskDueDate$ = this._task$.pipe(filter(task => !!task), map(task => new Date(task.due)));
57
+ this.disabled$ = new BehaviorSubject(false);
58
+ this.open$ = new Subject();
59
+ this.mouseIsOverDueDate$ = new BehaviorSubject(false);
60
+ this.toggletipTheme$ = this.cdsThemeService.toggletipTheme$;
61
+ this.iconService.registerAll([CalendarAdd16]);
62
+ }
63
+ onDateValueChange(value) {
64
+ const date = Array.isArray(value) && value[0];
65
+ if (!date)
66
+ return;
67
+ this.selectedDateString$.next(date.toISOString());
68
+ }
69
+ onSubmitButtonClick() {
70
+ this.disabled$.next(true);
71
+ this.taskService.setTaskDueDate(this._task.id, { dueDate: this._selectedDateString }).subscribe({
72
+ next: () => {
73
+ this.disabled$.next(false);
74
+ this.hasDueDate$.next(true);
75
+ this._task$.next({ ...this._task, due: this._selectedDateString });
76
+ this.selectedDateString$.next('');
77
+ this.closeToggletip();
78
+ },
79
+ error: () => {
80
+ this.disabled$.next(false);
81
+ },
82
+ });
83
+ }
84
+ onRemoveButtonClick() {
85
+ this.disabled$.next(true);
86
+ this.taskService.removeTaskDueDate(this._task.id).subscribe({
87
+ next: () => {
88
+ this.disabled$.next(false);
89
+ this.hasDueDate$.next(false);
90
+ this._task$.next({ ...this._task, due: null });
91
+ },
92
+ error: () => {
93
+ this.disabled$.next(false);
94
+ },
95
+ });
96
+ }
97
+ closeToggletip() {
98
+ // needed to reliably trigger toggle tip closure
99
+ this.open$.next(true);
100
+ setTimeout(() => this.open$.next(false));
101
+ }
102
+ onMouseEnterDueDate() {
103
+ this.mouseIsOverDueDate$.next(true);
104
+ }
105
+ onMouseLeaveDueDate() {
106
+ this.mouseIsOverDueDate$.next(false);
107
+ }
108
+ static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: SetTaskDueDateComponent, deps: [{ token: i1.IconService }, { token: i2.TaskService }, { token: i3.CdsThemeService }], target: i0.ɵɵFactoryTarget.Component }); }
109
+ static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "17.3.12", type: SetTaskDueDateComponent, isStandalone: true, selector: "valtimo-set-task-due-date", inputs: { canModifyTask: "canModifyTask", task: "task" }, ngImport: i0, template: "<!--\n ~ Copyright 2015-2024 Ritense BV, the Netherlands.\n ~\n ~ Licensed under EUPL, Version 1.2 (the \"License\");\n ~ you may not use this file except in compliance with the License.\n ~ You may obtain a copy of the License at\n ~\n ~ https://joinup.ec.europa.eu/collection/eupl/eupl-text-eupl-12\n ~\n ~ Unless required by applicable law or agreed to in writing, software\n ~ distributed under the License is distributed on an \"AS IS\" basis,\n ~ WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.\n ~ See the License for the specific language governing permissions and\n ~ limitations under the License.\n -->\n<ng-container\n *ngIf=\"\n (canModifyTaskSet$ | async) && {\n hasDueDate: hasDueDate$ | async,\n taskDueDate: taskDueDate$ | async,\n selectedDateString: selectedDateString$ | async,\n disabled: disabled$ | async,\n mouseIsOverDueDate: mouseIsOverDueDate$ | async,\n canModifyTask: canModifyTask$ | async,\n } as obs\n \"\n>\n <div\n *ngIf=\"obs.canModifyTask || (!obs.canModifyTask && obs.hasDueDate)\"\n class=\"due-date-container\"\n (mouseenter)=\"onMouseEnterDueDate()\"\n (mouseleave)=\"onMouseLeaveDueDate()\"\n >\n <div\n class=\"due-date-text element\"\n [ngClass]=\"{\n active: obs.canModifyTask ? obs.hasDueDate && !obs.mouseIsOverDueDate : obs.hasDueDate,\n }\"\n >\n <span class=\"bold\">{{ 'setTaskDueDate.dueDateText' | translate }}</span>\n\n &nbsp;\n\n {{ obs.taskDueDate | date: 'dd-MM-yyyy' }}\n </div>\n\n <button\n *ngIf=\"obs.canModifyTask\"\n cdsButton=\"ghost\"\n class=\"element remove-button\"\n [ngClass]=\"{active: obs.hasDueDate && obs.mouseIsOverDueDate}\"\n [disabled]=\"obs.disabled\"\n size=\"sm\"\n (click)=\"onRemoveButtonClick()\"\n >\n {{ 'setTaskDueDate.removeButtonText' | translate }}\n </button>\n\n <cds-toggletip\n align=\"bottom\"\n class=\"element main\"\n [autoAlign]=\"true\"\n [isOpen]=\"open$ | async\"\n [ngClass]=\"{active: !obs.hasDueDate}\"\n >\n <button\n cdsToggletipButton\n [removeClassnames]=\"['cds--toggletip-button']\"\n cdsButton=\"tertiary\"\n size=\"sm\"\n class=\"set-task-due-date-button\"\n >\n {{ 'setTaskDueDate.buttonText' | translate }}\n\n <svg class=\"cds--btn__icon\" cdsIcon=\"calendar--add\" size=\"16\"></svg>\n </button>\n\n <div\n cdsToggletipContent\n [attr.data-carbon-theme]=\"toggletipTheme$ | async\"\n class=\"assign-due-date-popover-content\"\n >\n <cds-date-picker\n *ngIf=\"!obs.hasDueDate\"\n [cdsLayer]=\"1\"\n [label]=\"'Select date'\"\n [placeholder]=\"'dd-mm-jjjj'\"\n dateFormat=\"d/m/Y\"\n [disabled]=\"obs.disabled\"\n (valueChange)=\"onDateValueChange($event)\"\n ></cds-date-picker>\n\n <button\n cdsButton\n class=\"submit-due-date-button\"\n [disabled]=\"!obs.selectedDateString || obs.disabled\"\n (click)=\"onSubmitButtonClick()\"\n >\n {{ 'setTaskDueDate.submitButtonText' | translate }}\n\n <svg class=\"cds--btn__icon\" cdsIcon=\"calendar--add\" size=\"16\"></svg>\n </button>\n </div>\n </cds-toggletip>\n </div>\n</ng-container>\n", styles: [":host ::ng-deep .cds--popover-content{max-inline-size:unset}.due-date-container{display:flex;align-items:center;justify-content:center;position:relative;font-weight:400;font-size:14px;line-height:18px;letter-spacing:.16px;color:var(--cds-link-01)}.due-date-container .bold{font-weight:700}.remove-button{width:100%}.due-date-text{display:flex;justify-content:flex-start;align-items:center;width:100%}.element{opacity:0;visibility:hidden;position:absolute;transition:opacity .3s ease-in-out}.element.active{opacity:1;visibility:visible}.element.main{position:relative}.assign-due-date-popover-content{width:400px;max-inline-size:400px;display:flex;flex-direction:column;gap:16px}.assign-due-date-popover-content .cds--btn{width:100%;max-width:unset}.assign-due-date-popover-content ::ng-deep .cds--date-picker-input__wrapper,.assign-due-date-popover-content ::ng-deep .cds--date-picker-input__wrapper>span,.assign-due-date-popover-content ::ng-deep .cds--date-picker-container,.assign-due-date-popover-content ::ng-deep .cds--date-picker,.assign-due-date-popover-content ::ng-deep .cds--date-picker__input{width:100%}.set-task-due-date-button{width:250px}\n"], dependencies: [{ kind: "ngmodule", type: CommonModule }, { kind: "directive", type: i4.NgClass, selector: "[ngClass]", inputs: ["class", "ngClass"] }, { kind: "directive", type: i4.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "pipe", type: i4.AsyncPipe, name: "async" }, { kind: "pipe", type: i4.DatePipe, name: "date" }, { kind: "ngmodule", type: TranslateModule }, { kind: "pipe", type: i5.TranslatePipe, name: "translate" }, { kind: "ngmodule", type: ButtonModule }, { kind: "directive", type: i1.Button, selector: "[cdsButton], [ibmButton]", inputs: ["ibmButton", "cdsButton", "size", "skeleton", "iconOnly", "isExpressive"] }, { kind: "ngmodule", type: IconModule }, { kind: "directive", type: i1.IconDirective, selector: "[cdsIcon], [ibmIcon]", inputs: ["ibmIcon", "cdsIcon", "size", "title", "ariaLabel", "ariaLabelledBy", "ariaHidden", "isFocusable"] }, { kind: "ngmodule", type: ToggletipModule }, { kind: "component", type: i1.Toggletip, selector: "cds-toggletip, ibm-toggletip", inputs: ["id", "isOpen"] }, { kind: "directive", type: i1.ToggletipButton, selector: "[cdsToggletipButton], [ibmToggletipButton]", inputs: ["ariaLabel"] }, { kind: "directive", type: i1.ToggletipContent, selector: "[cdsToggletipContent], [ibmToggletipContent]" }, { kind: "ngmodule", type: DatePickerModule }, { kind: "component", type: i1.DatePicker, selector: "cds-date-picker, ibm-date-picker", inputs: ["range", "dateFormat", "language", "label", "helperText", "rangeHelperText", "rangeLabel", "placeholder", "ariaLabel", "inputPattern", "id", "value", "theme", "disabled", "invalid", "invalidText", "warn", "warnText", "size", "rangeInvalid", "rangeInvalidText", "rangeWarn", "rangeWarnText", "skeleton", "plugins", "flatpickrOptions"], outputs: ["valueChange", "onClose"] }, { kind: "ngmodule", type: LayerModule }, { kind: "directive", type: i1.LayerDirective, selector: "[cdsLayer], [ibmLayer]", inputs: ["ibmLayer", "cdsLayer"], exportAs: ["layer"] }, { kind: "directive", type: RemoveClassnamesDirective, selector: "[removeClassnames]", inputs: ["removeClassnames"] }] }); }
110
+ }
111
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: SetTaskDueDateComponent, decorators: [{
112
+ type: Component,
113
+ args: [{ selector: 'valtimo-set-task-due-date', standalone: true, imports: [
114
+ CommonModule,
115
+ TranslateModule,
116
+ ButtonModule,
117
+ IconModule,
118
+ ToggletipModule,
119
+ DatePickerModule,
120
+ LayerModule,
121
+ RemoveClassnamesDirective,
122
+ ], template: "<!--\n ~ Copyright 2015-2024 Ritense BV, the Netherlands.\n ~\n ~ Licensed under EUPL, Version 1.2 (the \"License\");\n ~ you may not use this file except in compliance with the License.\n ~ You may obtain a copy of the License at\n ~\n ~ https://joinup.ec.europa.eu/collection/eupl/eupl-text-eupl-12\n ~\n ~ Unless required by applicable law or agreed to in writing, software\n ~ distributed under the License is distributed on an \"AS IS\" basis,\n ~ WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.\n ~ See the License for the specific language governing permissions and\n ~ limitations under the License.\n -->\n<ng-container\n *ngIf=\"\n (canModifyTaskSet$ | async) && {\n hasDueDate: hasDueDate$ | async,\n taskDueDate: taskDueDate$ | async,\n selectedDateString: selectedDateString$ | async,\n disabled: disabled$ | async,\n mouseIsOverDueDate: mouseIsOverDueDate$ | async,\n canModifyTask: canModifyTask$ | async,\n } as obs\n \"\n>\n <div\n *ngIf=\"obs.canModifyTask || (!obs.canModifyTask && obs.hasDueDate)\"\n class=\"due-date-container\"\n (mouseenter)=\"onMouseEnterDueDate()\"\n (mouseleave)=\"onMouseLeaveDueDate()\"\n >\n <div\n class=\"due-date-text element\"\n [ngClass]=\"{\n active: obs.canModifyTask ? obs.hasDueDate && !obs.mouseIsOverDueDate : obs.hasDueDate,\n }\"\n >\n <span class=\"bold\">{{ 'setTaskDueDate.dueDateText' | translate }}</span>\n\n &nbsp;\n\n {{ obs.taskDueDate | date: 'dd-MM-yyyy' }}\n </div>\n\n <button\n *ngIf=\"obs.canModifyTask\"\n cdsButton=\"ghost\"\n class=\"element remove-button\"\n [ngClass]=\"{active: obs.hasDueDate && obs.mouseIsOverDueDate}\"\n [disabled]=\"obs.disabled\"\n size=\"sm\"\n (click)=\"onRemoveButtonClick()\"\n >\n {{ 'setTaskDueDate.removeButtonText' | translate }}\n </button>\n\n <cds-toggletip\n align=\"bottom\"\n class=\"element main\"\n [autoAlign]=\"true\"\n [isOpen]=\"open$ | async\"\n [ngClass]=\"{active: !obs.hasDueDate}\"\n >\n <button\n cdsToggletipButton\n [removeClassnames]=\"['cds--toggletip-button']\"\n cdsButton=\"tertiary\"\n size=\"sm\"\n class=\"set-task-due-date-button\"\n >\n {{ 'setTaskDueDate.buttonText' | translate }}\n\n <svg class=\"cds--btn__icon\" cdsIcon=\"calendar--add\" size=\"16\"></svg>\n </button>\n\n <div\n cdsToggletipContent\n [attr.data-carbon-theme]=\"toggletipTheme$ | async\"\n class=\"assign-due-date-popover-content\"\n >\n <cds-date-picker\n *ngIf=\"!obs.hasDueDate\"\n [cdsLayer]=\"1\"\n [label]=\"'Select date'\"\n [placeholder]=\"'dd-mm-jjjj'\"\n dateFormat=\"d/m/Y\"\n [disabled]=\"obs.disabled\"\n (valueChange)=\"onDateValueChange($event)\"\n ></cds-date-picker>\n\n <button\n cdsButton\n class=\"submit-due-date-button\"\n [disabled]=\"!obs.selectedDateString || obs.disabled\"\n (click)=\"onSubmitButtonClick()\"\n >\n {{ 'setTaskDueDate.submitButtonText' | translate }}\n\n <svg class=\"cds--btn__icon\" cdsIcon=\"calendar--add\" size=\"16\"></svg>\n </button>\n </div>\n </cds-toggletip>\n </div>\n</ng-container>\n", styles: [":host ::ng-deep .cds--popover-content{max-inline-size:unset}.due-date-container{display:flex;align-items:center;justify-content:center;position:relative;font-weight:400;font-size:14px;line-height:18px;letter-spacing:.16px;color:var(--cds-link-01)}.due-date-container .bold{font-weight:700}.remove-button{width:100%}.due-date-text{display:flex;justify-content:flex-start;align-items:center;width:100%}.element{opacity:0;visibility:hidden;position:absolute;transition:opacity .3s ease-in-out}.element.active{opacity:1;visibility:visible}.element.main{position:relative}.assign-due-date-popover-content{width:400px;max-inline-size:400px;display:flex;flex-direction:column;gap:16px}.assign-due-date-popover-content .cds--btn{width:100%;max-width:unset}.assign-due-date-popover-content ::ng-deep .cds--date-picker-input__wrapper,.assign-due-date-popover-content ::ng-deep .cds--date-picker-input__wrapper>span,.assign-due-date-popover-content ::ng-deep .cds--date-picker-container,.assign-due-date-popover-content ::ng-deep .cds--date-picker,.assign-due-date-popover-content ::ng-deep .cds--date-picker__input{width:100%}.set-task-due-date-button{width:250px}\n"] }]
123
+ }], ctorParameters: () => [{ type: i1.IconService }, { type: i2.TaskService }, { type: i3.CdsThemeService }], propDecorators: { canModifyTask: [{
124
+ type: Input
125
+ }], task: [{
126
+ type: Input
127
+ }] } });
128
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"set-task-due-date.component.js","sourceRoot":"","sources":["../../../../../../../projects/valtimo/task/src/lib/components/set-task-due-date/set-task-due-date.component.ts","../../../../../../../projects/valtimo/task/src/lib/components/set-task-due-date/set-task-due-date.component.html"],"names":[],"mappings":"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","sourcesContent":["/*\n * Copyright 2015-2024 Ritense BV, the Netherlands.\n *\n * Licensed under EUPL, Version 1.2 (the \"License\");\n * you may not use this file except in compliance with the License.\n * You may obtain a copy of the License at\n *\n * https://joinup.ec.europa.eu/collection/eupl/eupl-text-eupl-12\n *\n * Unless required by applicable law or agreed to in writing, software\n * distributed under the License is distributed on an \"AS IS\" basis,\n * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.\n * See the License for the specific language governing permissions and\n * limitations under the License.\n */\n\nimport {Component, Input} from '@angular/core';\nimport {CommonModule} from '@angular/common';\nimport {TranslateModule} from '@ngx-translate/core';\nimport {ProcessInstanceTask} from '@valtimo/process';\nimport {BehaviorSubject, Subject} from 'rxjs';\nimport {filter, map} from 'rxjs/operators';\nimport {\n  ButtonModule,\n  DatePickerModule,\n  IconModule,\n  IconService,\n  LayerModule,\n  ToggletipModule,\n} from 'carbon-components-angular';\nimport {CalendarAdd16} from '@carbon/icons';\nimport {TaskService} from '../../services';\nimport {Task} from '../../models';\nimport {CdsThemeService, RemoveClassnamesDirective} from '@valtimo/components';\n\n@Component({\n  selector: 'valtimo-set-task-due-date',\n  templateUrl: './set-task-due-date.component.html',\n  styleUrls: ['./set-task-due-date.component.scss'],\n  standalone: true,\n  imports: [\n    CommonModule,\n    TranslateModule,\n    ButtonModule,\n    IconModule,\n    ToggletipModule,\n    DatePickerModule,\n    LayerModule,\n    RemoveClassnamesDirective,\n  ],\n})\nexport class SetTaskDueDateComponent {\n  public readonly canModifyTaskSet$ = new BehaviorSubject<boolean>(false);\n  public readonly canModifyTask$ = new BehaviorSubject<boolean>(false);\n\n  @Input() public set canModifyTask(value: boolean) {\n    this.canModifyTaskSet$.next(true);\n    this.canModifyTask$.next(value);\n  }\n\n  private readonly _task$ = new BehaviorSubject<Partial<ProcessInstanceTask> | null>(null);\n\n  private get _task(): Partial<ProcessInstanceTask> | null {\n    return this._task$.getValue();\n  }\n\n  public readonly hasDueDate$ = new BehaviorSubject<boolean>(false);\n\n  public readonly selectedDateString$ = new BehaviorSubject<string>('');\n\n  private get _selectedDateString(): string {\n    return this.selectedDateString$.getValue();\n  }\n\n  @Input() public set task(value: ProcessInstanceTask | Task) {\n    if (!value) return;\n    this.hasDueDate$.next(!!value.due);\n    this._task$.next(value);\n  }\n\n  public readonly taskDueDate$ = this._task$.pipe(\n    filter(task => !!task),\n    map(task => new Date(task.due))\n  );\n\n  public readonly disabled$ = new BehaviorSubject<boolean>(false);\n\n  public readonly open$ = new Subject<boolean>();\n\n  public readonly mouseIsOverDueDate$ = new BehaviorSubject<boolean>(false);\n\n  public readonly toggletipTheme$ = this.cdsThemeService.toggletipTheme$;\n\n  constructor(\n    private readonly iconService: IconService,\n    private readonly taskService: TaskService,\n    private readonly cdsThemeService: CdsThemeService\n  ) {\n    this.iconService.registerAll([CalendarAdd16]);\n  }\n\n  public onDateValueChange(value: Date[]): void {\n    const date = Array.isArray(value) && value[0];\n    if (!date) return;\n    this.selectedDateString$.next(date.toISOString());\n  }\n\n  public onSubmitButtonClick(): void {\n    this.disabled$.next(true);\n\n    this.taskService.setTaskDueDate(this._task.id, {dueDate: this._selectedDateString}).subscribe({\n      next: () => {\n        this.disabled$.next(false);\n        this.hasDueDate$.next(true);\n        this._task$.next({...this._task, due: this._selectedDateString});\n        this.selectedDateString$.next('');\n        this.closeToggletip();\n      },\n      error: () => {\n        this.disabled$.next(false);\n      },\n    });\n  }\n\n  public onRemoveButtonClick(): void {\n    this.disabled$.next(true);\n\n    this.taskService.removeTaskDueDate(this._task.id).subscribe({\n      next: () => {\n        this.disabled$.next(false);\n        this.hasDueDate$.next(false);\n        this._task$.next({...this._task, due: null});\n      },\n      error: () => {\n        this.disabled$.next(false);\n      },\n    });\n  }\n\n  private closeToggletip(): void {\n    // needed to reliably trigger toggle tip closure\n    this.open$.next(true);\n    setTimeout(() => this.open$.next(false));\n  }\n\n  public onMouseEnterDueDate(): void {\n    this.mouseIsOverDueDate$.next(true);\n  }\n\n  public onMouseLeaveDueDate(): void {\n    this.mouseIsOverDueDate$.next(false);\n  }\n}\n","<!--\n  ~ Copyright 2015-2024 Ritense BV, the Netherlands.\n  ~\n  ~ Licensed under EUPL, Version 1.2 (the \"License\");\n  ~ you may not use this file except in compliance with the License.\n  ~ You may obtain a copy of the License at\n  ~\n  ~ https://joinup.ec.europa.eu/collection/eupl/eupl-text-eupl-12\n  ~\n  ~ Unless required by applicable law or agreed to in writing, software\n  ~ distributed under the License is distributed on an \"AS IS\" basis,\n  ~ WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.\n  ~ See the License for the specific language governing permissions and\n  ~ limitations under the License.\n  -->\n<ng-container\n  *ngIf=\"\n    (canModifyTaskSet$ | async) && {\n      hasDueDate: hasDueDate$ | async,\n      taskDueDate: taskDueDate$ | async,\n      selectedDateString: selectedDateString$ | async,\n      disabled: disabled$ | async,\n      mouseIsOverDueDate: mouseIsOverDueDate$ | async,\n      canModifyTask: canModifyTask$ | async,\n    } as obs\n  \"\n>\n  <div\n    *ngIf=\"obs.canModifyTask || (!obs.canModifyTask && obs.hasDueDate)\"\n    class=\"due-date-container\"\n    (mouseenter)=\"onMouseEnterDueDate()\"\n    (mouseleave)=\"onMouseLeaveDueDate()\"\n  >\n    <div\n      class=\"due-date-text element\"\n      [ngClass]=\"{\n        active: obs.canModifyTask ? obs.hasDueDate && !obs.mouseIsOverDueDate : obs.hasDueDate,\n      }\"\n    >\n      <span class=\"bold\">{{ 'setTaskDueDate.dueDateText' | translate }}</span>\n\n      &nbsp;\n\n      {{ obs.taskDueDate | date: 'dd-MM-yyyy' }}\n    </div>\n\n    <button\n      *ngIf=\"obs.canModifyTask\"\n      cdsButton=\"ghost\"\n      class=\"element remove-button\"\n      [ngClass]=\"{active: obs.hasDueDate && obs.mouseIsOverDueDate}\"\n      [disabled]=\"obs.disabled\"\n      size=\"sm\"\n      (click)=\"onRemoveButtonClick()\"\n    >\n      {{ 'setTaskDueDate.removeButtonText' | translate }}\n    </button>\n\n    <cds-toggletip\n      align=\"bottom\"\n      class=\"element main\"\n      [autoAlign]=\"true\"\n      [isOpen]=\"open$ | async\"\n      [ngClass]=\"{active: !obs.hasDueDate}\"\n    >\n      <button\n        cdsToggletipButton\n        [removeClassnames]=\"['cds--toggletip-button']\"\n        cdsButton=\"tertiary\"\n        size=\"sm\"\n        class=\"set-task-due-date-button\"\n      >\n        {{ 'setTaskDueDate.buttonText' | translate }}\n\n        <svg class=\"cds--btn__icon\" cdsIcon=\"calendar--add\" size=\"16\"></svg>\n      </button>\n\n      <div\n        cdsToggletipContent\n        [attr.data-carbon-theme]=\"toggletipTheme$ | async\"\n        class=\"assign-due-date-popover-content\"\n      >\n        <cds-date-picker\n          *ngIf=\"!obs.hasDueDate\"\n          [cdsLayer]=\"1\"\n          [label]=\"'Select date'\"\n          [placeholder]=\"'dd-mm-jjjj'\"\n          dateFormat=\"d/m/Y\"\n          [disabled]=\"obs.disabled\"\n          (valueChange)=\"onDateValueChange($event)\"\n        ></cds-date-picker>\n\n        <button\n          cdsButton\n          class=\"submit-due-date-button\"\n          [disabled]=\"!obs.selectedDateString || obs.disabled\"\n          (click)=\"onSubmitButtonClick()\"\n        >\n          {{ 'setTaskDueDate.submitButtonText' | translate }}\n\n          <svg class=\"cds--btn__icon\" cdsIcon=\"calendar--add\" size=\"16\"></svg>\n        </button>\n      </div>\n    </cds-toggletip>\n  </div>\n</ng-container>\n"]}