@cuby-ui/core 0.0.321 → 0.0.323

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 (84) hide show
  1. package/components/sidebar/sidebar-header/sidebar-header.component.d.ts +1 -1
  2. package/components/sidebar/sidebar-navigation-item/sidebar-navigation-item.component.d.ts +1 -1
  3. package/esm2022/interfaces/image.mjs +2 -0
  4. package/esm2022/interfaces/index.mjs +2 -1
  5. package/esm2022/widgets/criterion-info/criterion-info.component.mjs +137 -0
  6. package/esm2022/widgets/criterion-info/index.mjs +2 -0
  7. package/esm2022/widgets/criterion-info-form/criterion-info-form.component.mjs +147 -0
  8. package/esm2022/widgets/criterion-info-form/criterion-info-form.options.mjs +2 -0
  9. package/esm2022/widgets/criterion-info-form/index.mjs +2 -0
  10. package/esm2022/widgets/criterion-readonly-info/criterion-readonly-info.component.mjs +72 -0
  11. package/esm2022/widgets/criterion-readonly-info/index.mjs +2 -0
  12. package/esm2022/widgets/index.mjs +6 -2
  13. package/esm2022/widgets/instruction-info/index.mjs +2 -0
  14. package/esm2022/widgets/instruction-info/instruction-info.component.mjs +147 -0
  15. package/esm2022/widgets/instruction-info-form/index.mjs +2 -0
  16. package/esm2022/widgets/instruction-info-form/instruction-info-form.component.mjs +153 -0
  17. package/esm2022/widgets/instruction-info-form/instruction-info-form.options.mjs +2 -0
  18. package/esm2022/widgets/instruction-readonly-info/index.mjs +2 -0
  19. package/esm2022/widgets/instruction-readonly-info/instruction-readonly-info.component.mjs +72 -0
  20. package/esm2022/widgets/utility-modal/components/criterion-modal-create/criterion-modal-create.component.mjs +40 -0
  21. package/esm2022/widgets/utility-modal/components/criterion-modal-create/index.mjs +2 -0
  22. package/esm2022/widgets/utility-modal/components/index.mjs +5 -0
  23. package/esm2022/widgets/utility-modal/components/instruction-modal-create/index.mjs +2 -0
  24. package/esm2022/widgets/utility-modal/components/instruction-modal-create/instruction-modal-create.component.mjs +40 -0
  25. package/esm2022/widgets/utility-modal/components/readonly-criterion-modal/index.mjs +2 -0
  26. package/esm2022/widgets/utility-modal/components/readonly-criterion-modal/readonly-criterion-modal.component.mjs +40 -0
  27. package/esm2022/widgets/utility-modal/components/readonly-instruction-modal/index.mjs +2 -0
  28. package/esm2022/widgets/utility-modal/components/readonly-instruction-modal/readonly-instruction-modal.component.mjs +40 -0
  29. package/esm2022/widgets/utility-modal/components/readonly-utility-modal/index.mjs +2 -0
  30. package/esm2022/widgets/utility-modal/components/readonly-utility-modal/readonly-utility-modal.component.mjs +42 -0
  31. package/esm2022/widgets/utility-modal/components/utility-modal-create/index.mjs +2 -0
  32. package/esm2022/widgets/utility-modal/components/utility-modal-create/utility-modal-create.component.mjs +55 -0
  33. package/esm2022/widgets/utility-modal/index.mjs +4 -0
  34. package/esm2022/widgets/utility-modal/utility-modal.component.mjs +22 -0
  35. package/esm2022/widgets/utility-modal/utility-modal.options.mjs +3 -0
  36. package/esm2022/widgets/utility-readonly-thumbnail/index.mjs +2 -0
  37. package/esm2022/widgets/utility-readonly-thumbnail/utility-readonly-thumbnail.component.mjs +15 -0
  38. package/esm2022/widgets/utility-thumbnail/index.mjs +2 -0
  39. package/esm2022/widgets/utility-thumbnail/utility-thumbnail.component.mjs +94 -0
  40. package/fesm2022/cuby-ui-core.mjs +982 -235
  41. package/fesm2022/cuby-ui-core.mjs.map +1 -1
  42. package/interfaces/image.d.ts +5 -0
  43. package/interfaces/index.d.ts +1 -0
  44. package/package.json +4 -4
  45. package/widgets/categories/components/category-item/category-item.component.d.ts +2 -2
  46. package/widgets/criterion-info/criterion-info.component.d.ts +43 -0
  47. package/widgets/criterion-info/index.d.ts +1 -0
  48. package/widgets/criterion-info-form/criterion-info-form.component.d.ts +39 -0
  49. package/widgets/criterion-info-form/criterion-info-form.options.d.ts +7 -0
  50. package/widgets/criterion-info-form/index.d.ts +2 -0
  51. package/widgets/criterion-readonly-info/criterion-readonly-info.component.d.ts +26 -0
  52. package/widgets/criterion-readonly-info/index.d.ts +1 -0
  53. package/widgets/index.d.ts +5 -1
  54. package/widgets/instruction-info/index.d.ts +1 -0
  55. package/widgets/instruction-info/instruction-info.component.d.ts +45 -0
  56. package/widgets/instruction-info-form/index.d.ts +2 -0
  57. package/widgets/{utility-info/utility-info.component.d.ts → instruction-info-form/instruction-info-form.component.d.ts} +12 -23
  58. package/widgets/{utility-info/utility-info.options.d.ts → instruction-info-form/instruction-info-form.options.d.ts} +3 -3
  59. package/widgets/instruction-readonly-info/index.d.ts +1 -0
  60. package/widgets/instruction-readonly-info/instruction-readonly-info.component.d.ts +26 -0
  61. package/widgets/utility-modal/components/criterion-modal-create/criterion-modal-create.component.d.ts +18 -0
  62. package/widgets/utility-modal/components/criterion-modal-create/index.d.ts +1 -0
  63. package/widgets/utility-modal/components/index.d.ts +4 -0
  64. package/widgets/utility-modal/components/instruction-modal-create/index.d.ts +1 -0
  65. package/widgets/utility-modal/components/instruction-modal-create/instruction-modal-create.component.d.ts +18 -0
  66. package/widgets/utility-modal/components/readonly-criterion-modal/index.d.ts +1 -0
  67. package/widgets/utility-modal/components/readonly-criterion-modal/readonly-criterion-modal.component.d.ts +16 -0
  68. package/widgets/utility-modal/components/readonly-instruction-modal/index.d.ts +1 -0
  69. package/widgets/utility-modal/components/readonly-instruction-modal/readonly-instruction-modal.component.d.ts +16 -0
  70. package/widgets/utility-modal/components/readonly-utility-modal/index.d.ts +1 -0
  71. package/widgets/utility-modal/components/readonly-utility-modal/readonly-utility-modal.component.d.ts +18 -0
  72. package/widgets/utility-modal/components/utility-modal-create/index.d.ts +1 -0
  73. package/widgets/utility-modal/components/utility-modal-create/utility-modal-create.component.d.ts +25 -0
  74. package/widgets/utility-modal/index.d.ts +3 -0
  75. package/widgets/utility-modal/utility-modal.component.d.ts +10 -0
  76. package/widgets/utility-modal/utility-modal.options.d.ts +12 -0
  77. package/widgets/utility-readonly-thumbnail/index.d.ts +1 -0
  78. package/widgets/utility-readonly-thumbnail/utility-readonly-thumbnail.component.d.ts +8 -0
  79. package/widgets/utility-thumbnail/index.d.ts +1 -0
  80. package/widgets/utility-thumbnail/utility-thumbnail.component.d.ts +21 -0
  81. package/esm2022/widgets/utility-info/index.mjs +0 -2
  82. package/esm2022/widgets/utility-info/utility-info.component.mjs +0 -249
  83. package/esm2022/widgets/utility-info/utility-info.options.mjs +0 -2
  84. package/widgets/utility-info/index.d.ts +0 -2
@@ -1,7 +1,11 @@
1
1
  export { CuiLoaderComponent } from './loader';
2
2
  export { CuiDeleteModalComponent } from './delete-modal';
3
3
  export { CuiCategoriesComponent, CuiSelectedCategoryService } from './categories';
4
- export { CuiUtilityInfoComponent } from './utility-info';
5
4
  export { CuiChecklistBlockComponent } from './checklist-block';
5
+ export { CuiInstructionReadonlyInfoComponent } from './instruction-readonly-info';
6
+ export { CuiCriterionReadonlyInfoComponent } from './criterion-readonly-info';
7
+ export { CuiCriterionInfoComponent } from './criterion-info';
8
+ export { CuiInstructionInfoComponent } from './instruction-info';
6
9
  export { CuiGeneralControlErrorHintComponent } from './general-control-error-hint';
7
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi9wcm9qZWN0cy9jb3JlL3dpZGdldHMvaW5kZXgudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsT0FBTyxFQUFFLGtCQUFrQixFQUFFLE1BQU0sVUFBVSxDQUFDO0FBQzlDLE9BQU8sRUFBRSx1QkFBdUIsRUFBRSxNQUFNLGdCQUFnQixDQUFDO0FBQ3pELE9BQU8sRUFBRSxzQkFBc0IsRUFBRSwwQkFBMEIsRUFBRSxNQUFNLGNBQWMsQ0FBQztBQUNsRixPQUFPLEVBQUUsdUJBQXVCLEVBQTZDLE1BQU0sZ0JBQWdCLENBQUM7QUFDcEcsT0FBTyxFQUFFLDBCQUEwQixFQUFFLE1BQU0sbUJBQW1CLENBQUM7QUFDL0QsT0FBTyxFQUFFLG1DQUFtQyxFQUFFLE1BQU0sOEJBQThCLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyJleHBvcnQgeyBDdWlMb2FkZXJDb21wb25lbnQgfSBmcm9tICcuL2xvYWRlcic7XG5leHBvcnQgeyBDdWlEZWxldGVNb2RhbENvbXBvbmVudCB9IGZyb20gJy4vZGVsZXRlLW1vZGFsJztcbmV4cG9ydCB7IEN1aUNhdGVnb3JpZXNDb21wb25lbnQsIEN1aVNlbGVjdGVkQ2F0ZWdvcnlTZXJ2aWNlIH0gZnJvbSAnLi9jYXRlZ29yaWVzJztcbmV4cG9ydCB7IEN1aVV0aWxpdHlJbmZvQ29tcG9uZW50LCBDdWlVdGlsaXR5Rm9ybUdyb3VwLCBDdWlVdGlsaXR5Rm9ybVZhbHVlcyB9IGZyb20gJy4vdXRpbGl0eS1pbmZvJztcbmV4cG9ydCB7IEN1aUNoZWNrbGlzdEJsb2NrQ29tcG9uZW50IH0gZnJvbSAnLi9jaGVja2xpc3QtYmxvY2snO1xuZXhwb3J0IHsgQ3VpR2VuZXJhbENvbnRyb2xFcnJvckhpbnRDb21wb25lbnQgfSBmcm9tICcuL2dlbmVyYWwtY29udHJvbC1lcnJvci1oaW50JztcbiJdfQ==
10
+ export * from './utility-modal';
11
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi9wcm9qZWN0cy9jb3JlL3dpZGdldHMvaW5kZXgudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsT0FBTyxFQUFFLGtCQUFrQixFQUFFLE1BQU0sVUFBVSxDQUFDO0FBQzlDLE9BQU8sRUFBRSx1QkFBdUIsRUFBRSxNQUFNLGdCQUFnQixDQUFDO0FBQ3pELE9BQU8sRUFBRSxzQkFBc0IsRUFBRSwwQkFBMEIsRUFBRSxNQUFNLGNBQWMsQ0FBQztBQUNsRixPQUFPLEVBQUUsMEJBQTBCLEVBQUUsTUFBTSxtQkFBbUIsQ0FBQztBQUMvRCxPQUFPLEVBQUUsbUNBQW1DLEVBQUUsTUFBTSw2QkFBNkIsQ0FBQztBQUNsRixPQUFPLEVBQUUsaUNBQWlDLEVBQUUsTUFBTSwyQkFBMkIsQ0FBQztBQUM5RSxPQUFPLEVBQUUseUJBQXlCLEVBQUUsTUFBTSxrQkFBa0IsQ0FBQztBQUM3RCxPQUFPLEVBQUUsMkJBQTJCLEVBQUUsTUFBTSxvQkFBb0IsQ0FBQztBQUNqRSxPQUFPLEVBQUUsbUNBQW1DLEVBQUUsTUFBTSw4QkFBOEIsQ0FBQztBQUNuRixjQUFjLGlCQUFpQixDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiZXhwb3J0IHsgQ3VpTG9hZGVyQ29tcG9uZW50IH0gZnJvbSAnLi9sb2FkZXInO1xuZXhwb3J0IHsgQ3VpRGVsZXRlTW9kYWxDb21wb25lbnQgfSBmcm9tICcuL2RlbGV0ZS1tb2RhbCc7XG5leHBvcnQgeyBDdWlDYXRlZ29yaWVzQ29tcG9uZW50LCBDdWlTZWxlY3RlZENhdGVnb3J5U2VydmljZSB9IGZyb20gJy4vY2F0ZWdvcmllcyc7XG5leHBvcnQgeyBDdWlDaGVja2xpc3RCbG9ja0NvbXBvbmVudCB9IGZyb20gJy4vY2hlY2tsaXN0LWJsb2NrJztcbmV4cG9ydCB7IEN1aUluc3RydWN0aW9uUmVhZG9ubHlJbmZvQ29tcG9uZW50IH0gZnJvbSAnLi9pbnN0cnVjdGlvbi1yZWFkb25seS1pbmZvJztcbmV4cG9ydCB7IEN1aUNyaXRlcmlvblJlYWRvbmx5SW5mb0NvbXBvbmVudCB9IGZyb20gJy4vY3JpdGVyaW9uLXJlYWRvbmx5LWluZm8nO1xuZXhwb3J0IHsgQ3VpQ3JpdGVyaW9uSW5mb0NvbXBvbmVudCB9IGZyb20gJy4vY3JpdGVyaW9uLWluZm8nO1xuZXhwb3J0IHsgQ3VpSW5zdHJ1Y3Rpb25JbmZvQ29tcG9uZW50IH0gZnJvbSAnLi9pbnN0cnVjdGlvbi1pbmZvJztcbmV4cG9ydCB7IEN1aUdlbmVyYWxDb250cm9sRXJyb3JIaW50Q29tcG9uZW50IH0gZnJvbSAnLi9nZW5lcmFsLWNvbnRyb2wtZXJyb3ItaGludCc7XG5leHBvcnQgKiBmcm9tICcuL3V0aWxpdHktbW9kYWwnO1xuIl19
@@ -0,0 +1,2 @@
1
+ export { CuiInstructionInfoComponent } from './instruction-info.component';
2
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi9wcm9qZWN0cy9jb3JlL3dpZGdldHMvaW5zdHJ1Y3Rpb24taW5mby9pbmRleC50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxPQUFPLEVBQUUsMkJBQTJCLEVBQUUsTUFBTSw4QkFBOEIsQ0FBQyIsInNvdXJjZXNDb250ZW50IjpbImV4cG9ydCB7IEN1aUluc3RydWN0aW9uSW5mb0NvbXBvbmVudCB9IGZyb20gJy4vaW5zdHJ1Y3Rpb24taW5mby5jb21wb25lbnQnO1xyXG4iXX0=
@@ -0,0 +1,147 @@
1
+ import { ChangeDetectionStrategy, Component, computed, DestroyRef, effect, inject, input, model, output, signal, untracked, viewChild } from '@angular/core';
2
+ import { takeUntilDestroyed } from '@angular/core/rxjs-interop';
3
+ import { TranslocoModule, TranslocoService } from '@jsverse/transloco';
4
+ import { finalize, forkJoin } from 'rxjs';
5
+ import { CuiTreeStructNavigatorApiService } from '@cuby-ui/api';
6
+ import { CuiIdService } from '@cuby-ui/cdk';
7
+ import { CuiBadgeModule, CuiButtonModule, CuiDialogModule, CuiFormFieldModule, CuiLabelModule, CuiSvgModule } from '../../components';
8
+ import { CuiCategoriesComponent, CuiSelectedCategoryService } from '../categories';
9
+ import { CuiInstructionInfoFormComponent } from '../instruction-info-form';
10
+ import { UTILITY_MODAL_SERVICE_TOKEN } from '../utility-modal';
11
+ import * as i0 from "@angular/core";
12
+ import * as i1 from "../../components/form-field/form-field.component";
13
+ import * as i2 from "../../components/label/label.component";
14
+ import * as i3 from "../../components/svg/svg.component";
15
+ import * as i4 from "@jsverse/transloco";
16
+ export class CuiInstructionInfoComponent {
17
+ constructor() {
18
+ this.destroyRef = inject(DestroyRef);
19
+ this.translocoService = inject(TranslocoService);
20
+ this.cuiIdService = inject(CuiIdService);
21
+ this.cuiCategoriesModalService = inject(CuiSelectedCategoryService);
22
+ this.treeStructNavigatorApiService = inject(CuiTreeStructNavigatorApiService);
23
+ this.instructionApiService = inject(UTILITY_MODAL_SERVICE_TOKEN);
24
+ this.TYPE = "INSTRUCTION_STORAGE" /* TreeStructType.Instruction */;
25
+ this.skipCategoryChange = false;
26
+ this.categoryId = this.cuiIdService.generate();
27
+ this.category = this.cuiCategoriesModalService.category;
28
+ this.isFormValid = signal(false);
29
+ this.prevCategoryId = signal(null);
30
+ this.editorConfig = signal(null);
31
+ this.labelColor = computed(() => this.category()?.title ? 'var(--cui-base-900)' : 'var(--cui-base-400)');
32
+ this.selectTitle = computed(() => this.category()?.title || this.translocoService.translate('SELECT') + '...');
33
+ this.element = viewChild('scrollContent');
34
+ this.instruction = model();
35
+ this.tools = input.required();
36
+ this.editorResourceUrl = input.required();
37
+ this.validationChanged = output();
38
+ this.instructionCreated = output();
39
+ this.initConfigEffect();
40
+ this.initCategoryEffect();
41
+ }
42
+ ngOnInit() {
43
+ this.initCategory();
44
+ }
45
+ onFormChanged(values, instructionId) {
46
+ const actions = {
47
+ title: (title) => this.changeTitle(title, instructionId),
48
+ description: (description) => this.changeDescription(description, instructionId)
49
+ };
50
+ forkJoin(Object.keys(values).map((key) => actions[key](values[key] || '')))
51
+ .pipe(takeUntilDestroyed(this.destroyRef))
52
+ .subscribe();
53
+ }
54
+ onFormValidationChanged(isValid) {
55
+ this.isFormValid.set(isValid);
56
+ this.validationChanged.emit(isValid);
57
+ }
58
+ initCategoryEffect() {
59
+ effect(() => {
60
+ const category = this.category();
61
+ const storageObjectInstruction = untracked(this.instruction);
62
+ if (!category || this.skipCategoryChange) {
63
+ this.skipCategoryChange = false;
64
+ return;
65
+ }
66
+ if (storageObjectInstruction) {
67
+ this.changeCategory(category.id, storageObjectInstruction.id);
68
+ return;
69
+ }
70
+ this.createInstruction(category.id);
71
+ });
72
+ }
73
+ initConfigEffect() {
74
+ effect(() => {
75
+ const element = this.element()?.nativeElement;
76
+ if (!element) {
77
+ return;
78
+ }
79
+ this.editorConfig.set({ tools: this.tools(), element });
80
+ });
81
+ }
82
+ initCategory() {
83
+ const storageObjectId = this.instruction()?.id;
84
+ if (!storageObjectId) {
85
+ return;
86
+ }
87
+ this.treeStructNavigatorApiService
88
+ .getCategory(storageObjectId, "INSTRUCTION_STORAGE" /* TreeStructType.Instruction */)
89
+ .pipe(takeUntilDestroyed(this.destroyRef))
90
+ .subscribe((category) => {
91
+ this.skipCategoryChange = true;
92
+ this.prevCategoryId.set(category.id);
93
+ this.cuiCategoriesModalService.setCurrentCategory(category);
94
+ });
95
+ }
96
+ changeCategory(categoryId, targetId) {
97
+ const prevCategoryId = untracked(this.prevCategoryId);
98
+ if (!prevCategoryId) {
99
+ return;
100
+ }
101
+ this.subscription?.unsubscribe();
102
+ this.subscription = this.treeStructNavigatorApiService
103
+ .moveElement({
104
+ type: "MOVING_TARGET_TREE_STRUCT" /* MovingType.TARGET_TREE_STRUCT */,
105
+ movableItemId: targetId,
106
+ treeStructTargetItemId: categoryId,
107
+ treeStructMovableItemId: prevCategoryId
108
+ })
109
+ .subscribe(() => this.prevCategoryId.set(categoryId));
110
+ }
111
+ createInstruction(categoryId) {
112
+ this.instructionCreated.emit(categoryId);
113
+ this.prevCategoryId.set(categoryId);
114
+ }
115
+ changeTitle(value, instructionId) {
116
+ return this.instructionApiService.changeTitle(instructionId, value).pipe(finalize(() => this.instruction.update((prev) => ({
117
+ ...prev,
118
+ storageElement: { ...prev.storageElement, title: value }
119
+ }))));
120
+ }
121
+ changeDescription(value, instructionId) {
122
+ if (!this.instructionApiService.changeDescription) {
123
+ throw new Error('Change description not implemented');
124
+ }
125
+ return this.instructionApiService.changeDescription(instructionId, value).pipe(finalize(() => this.instruction.update((prev) => ({
126
+ ...prev,
127
+ storageElement: { ...prev.storageElement, shortDescription: value }
128
+ }))));
129
+ }
130
+ static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: CuiInstructionInfoComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
131
+ static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "18.2.13", type: CuiInstructionInfoComponent, isStandalone: true, selector: "cui-instruction-info", inputs: { instruction: { classPropertyName: "instruction", publicName: "instruction", isSignal: true, isRequired: false, transformFunction: null }, tools: { classPropertyName: "tools", publicName: "tools", isSignal: true, isRequired: true, transformFunction: null }, editorResourceUrl: { classPropertyName: "editorResourceUrl", publicName: "editorResourceUrl", isSignal: true, isRequired: true, transformFunction: null } }, outputs: { instruction: "instructionChange", validationChanged: "validationChanged", instructionCreated: "instructionCreated" }, providers: [CuiSelectedCategoryService], viewQueries: [{ propertyName: "element", first: true, predicate: ["scrollContent"], descendants: true, isSignal: true }], ngImport: i0, template: "<div\r\n class=\"wrapper\"\r\n *transloco=\"let t\"\r\n>\r\n <div\r\n #scrollContent\r\n class=\"content content_scrollable\"\r\n >\r\n <div class=\"category\">\r\n <cui-form-field>\r\n <label\r\n cuiLabel\r\n [for]=\"categoryId\"\r\n [isRequired]=\"true\"\r\n >\r\n {{ t('SELECT_CATEGORY') }}\r\n </label>\r\n <cui-categories\r\n [buttonId]=\"categoryId\"\r\n [treeStructType]=\"TYPE\"\r\n >\r\n <div class=\"select\">\r\n <span [style.color]=\"labelColor()\">\r\n {{ selectTitle() }}\r\n </span>\r\n\r\n <cui-svg icon=\"cuiIconChevronDown\" />\r\n </div>\r\n </cui-categories>\r\n </cui-form-field>\r\n </div>\r\n\r\n @let instructionData = instruction();\r\n @let editorConfigData = editorConfig();\r\n\r\n @if (editorConfigData && instructionData) {\r\n <cui-instruction-info-form\r\n [config]=\"editorConfigData\"\r\n [baseResourceUrl]=\"editorResourceUrl()\"\r\n [title]=\"instructionData.storageElement.title\"\r\n [description]=\"instructionData.storageElement.shortDescription\"\r\n [framerId]=\"instructionData.storageElement.framerInstructionId\"\r\n class=\"utility\"\r\n (isFormValidChanged)=\"onFormValidationChanged($event)\"\r\n (formChanged)=\"onFormChanged($event, instructionData.id)\"\r\n />\r\n }\r\n </div>\r\n</div>\r\n", styles: [".wrapper{display:flex;flex-direction:column;height:100%}.content{height:calc(100vh - 109px)}.content_scrollable{overflow:auto}.category{padding-top:16px;padding-right:23px;padding-left:23px}.select{padding:7px 13px;font-weight:400;font-size:14px;line-height:20px;display:flex;justify-content:space-between;height:36px;border-radius:8px;border:1px solid var(--cui-base-200);color:var(--cui-base-900);background:var(--cui-input)}.utility{--editor-height: none}\n"], dependencies: [{ kind: "ngmodule", type: CuiBadgeModule }, { kind: "ngmodule", type: CuiButtonModule }, { kind: "component", type: CuiCategoriesComponent, selector: "cui-categories", inputs: ["treeStructType", "buttonId", "disabled"] }, { kind: "ngmodule", type: CuiDialogModule }, { kind: "ngmodule", type: CuiFormFieldModule }, { kind: "component", type: i1.CuiFormFieldComponent, selector: "cui-form-field" }, { kind: "component", type: CuiInstructionInfoFormComponent, selector: "cui-instruction-info-form", inputs: ["title", "description", "framerId", "baseResourceUrl", "config"], outputs: ["formChanged", "isFormValidChanged"] }, { kind: "ngmodule", type: CuiLabelModule }, { kind: "component", type: i2.CuiLabelComponent, selector: "label[cuiLabel]", inputs: ["isRequired"] }, { kind: "ngmodule", type: CuiSvgModule }, { kind: "component", type: i3.CuiSvgComponent, selector: "cui-svg[icon]", inputs: ["width", "height", "strokeWidth", "color", "icon"] }, { kind: "ngmodule", type: TranslocoModule }, { kind: "directive", type: i4.TranslocoDirective, selector: "[transloco]", inputs: ["transloco", "translocoParams", "translocoScope", "translocoRead", "translocoPrefix", "translocoLang", "translocoLoadingTpl"] }], changeDetection: i0.ChangeDetectionStrategy.OnPush }); }
132
+ }
133
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: CuiInstructionInfoComponent, decorators: [{
134
+ type: Component,
135
+ args: [{ selector: 'cui-instruction-info', imports: [
136
+ CuiBadgeModule,
137
+ CuiButtonModule,
138
+ CuiCategoriesComponent,
139
+ CuiDialogModule,
140
+ CuiFormFieldModule,
141
+ CuiInstructionInfoFormComponent,
142
+ CuiLabelModule,
143
+ CuiSvgModule,
144
+ TranslocoModule
145
+ ], standalone: true, changeDetection: ChangeDetectionStrategy.OnPush, providers: [CuiSelectedCategoryService], template: "<div\r\n class=\"wrapper\"\r\n *transloco=\"let t\"\r\n>\r\n <div\r\n #scrollContent\r\n class=\"content content_scrollable\"\r\n >\r\n <div class=\"category\">\r\n <cui-form-field>\r\n <label\r\n cuiLabel\r\n [for]=\"categoryId\"\r\n [isRequired]=\"true\"\r\n >\r\n {{ t('SELECT_CATEGORY') }}\r\n </label>\r\n <cui-categories\r\n [buttonId]=\"categoryId\"\r\n [treeStructType]=\"TYPE\"\r\n >\r\n <div class=\"select\">\r\n <span [style.color]=\"labelColor()\">\r\n {{ selectTitle() }}\r\n </span>\r\n\r\n <cui-svg icon=\"cuiIconChevronDown\" />\r\n </div>\r\n </cui-categories>\r\n </cui-form-field>\r\n </div>\r\n\r\n @let instructionData = instruction();\r\n @let editorConfigData = editorConfig();\r\n\r\n @if (editorConfigData && instructionData) {\r\n <cui-instruction-info-form\r\n [config]=\"editorConfigData\"\r\n [baseResourceUrl]=\"editorResourceUrl()\"\r\n [title]=\"instructionData.storageElement.title\"\r\n [description]=\"instructionData.storageElement.shortDescription\"\r\n [framerId]=\"instructionData.storageElement.framerInstructionId\"\r\n class=\"utility\"\r\n (isFormValidChanged)=\"onFormValidationChanged($event)\"\r\n (formChanged)=\"onFormChanged($event, instructionData.id)\"\r\n />\r\n }\r\n </div>\r\n</div>\r\n", styles: [".wrapper{display:flex;flex-direction:column;height:100%}.content{height:calc(100vh - 109px)}.content_scrollable{overflow:auto}.category{padding-top:16px;padding-right:23px;padding-left:23px}.select{padding:7px 13px;font-weight:400;font-size:14px;line-height:20px;display:flex;justify-content:space-between;height:36px;border-radius:8px;border:1px solid var(--cui-base-200);color:var(--cui-base-900);background:var(--cui-input)}.utility{--editor-height: none}\n"] }]
146
+ }], ctorParameters: () => [] });
147
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"instruction-info.component.js","sourceRoot":"","sources":["../../../../../projects/core/widgets/instruction-info/instruction-info.component.ts","../../../../../projects/core/widgets/instruction-info/instruction-info.component.html"],"names":[],"mappings":"AACA,OAAO,EACH,uBAAuB,EACvB,SAAS,EACT,QAAQ,EACR,UAAU,EACV,MAAM,EACN,MAAM,EACN,KAAK,EACL,KAAK,EACL,MAAM,EACN,MAAM,EACN,SAAS,EACT,SAAS,EACZ,MAAM,eAAe,CAAC;AACvB,OAAO,EAAE,kBAAkB,EAAE,MAAM,4BAA4B,CAAC;AAChE,OAAO,EAAE,eAAe,EAAE,gBAAgB,EAAE,MAAM,oBAAoB,CAAC;AAEvE,OAAO,EAAE,QAAQ,EAAE,QAAQ,EAAE,MAAM,MAAM,CAAC;AAE1C,OAAO,EAAE,gCAAgC,EAA8B,MAAM,cAAc,CAAC;AAE5F,OAAO,EAAE,YAAY,EAAE,MAAM,cAAc,CAAC;AAE5C,OAAO,EACH,cAAc,EACd,eAAe,EACf,eAAe,EACf,kBAAkB,EAClB,cAAc,EACd,YAAY,EACf,MAAM,kBAAkB,CAAC;AAC1B,OAAO,EAAE,sBAAsB,EAAE,0BAA0B,EAAE,MAAM,eAAe,CAAC;AAGnF,OAAO,EAAE,+BAA+B,EAAE,MAAM,0BAA0B,CAAC;AAE3E,OAAO,EAAE,2BAA2B,EAAuB,MAAM,kBAAkB,CAAC;;;;;;AAsBpF,MAAM,OAAO,2BAA2B;IAmCpC;QAlCiB,eAAU,GAAG,MAAM,CAAC,UAAU,CAAC,CAAC;QAChC,qBAAgB,GAAG,MAAM,CAAC,gBAAgB,CAAC,CAAC;QAC5C,iBAAY,GAAG,MAAM,CAAC,YAAY,CAAC,CAAC;QACpC,8BAAyB,GAAG,MAAM,CAAC,0BAA0B,CAAC,CAAC;QAC/D,kCAA6B,GAAG,MAAM,CAAC,gCAAgC,CAAC,CAAC;QACzE,0BAAqB,GAAG,MAAM,CAAmC,2BAA2B,CAAC,CAAC;QAE5F,SAAI,0DAA8B;QAG7C,uBAAkB,GAAG,KAAK,CAAC;QAEhB,eAAU,GAAG,IAAI,CAAC,YAAY,CAAC,QAAQ,EAAE,CAAC;QAE1C,aAAQ,GAAG,IAAI,CAAC,yBAAyB,CAAC,QAAQ,CAAC;QACnD,gBAAW,GAAG,MAAM,CAAC,KAAK,CAAC,CAAC;QAC5B,mBAAc,GAAG,MAAM,CAAsB,IAAI,CAAC,CAAC;QACnD,iBAAY,GAAG,MAAM,CAA+B,IAAI,CAAC,CAAC;QAC1D,eAAU,GAAG,QAAQ,CAAC,GAAG,EAAE,CAC1C,IAAI,CAAC,QAAQ,EAAE,EAAE,KAAK,CAAC,CAAC,CAAC,qBAAqB,CAAC,CAAC,CAAC,qBAAqB,CACzE,CAAC;QACiB,gBAAW,GAAG,QAAQ,CACrC,GAAG,EAAE,CAAC,IAAI,CAAC,QAAQ,EAAE,EAAE,KAAK,IAAI,IAAI,CAAC,gBAAgB,CAAC,SAAS,CAAC,QAAQ,CAAC,GAAG,KAAK,CACpF,CAAC;QAEiB,YAAO,GAAG,SAAS,CAA0B,eAAe,CAAC,CAAC;QAEjE,gBAAW,GAAG,KAAK,EAA8B,CAAC;QAClD,UAAK,GAAG,KAAK,CAAC,QAAQ,EAAY,CAAC;QACnC,sBAAiB,GAAG,KAAK,CAAC,QAAQ,EAAU,CAAC;QAE7C,sBAAiB,GAAG,MAAM,EAAW,CAAC;QACtC,uBAAkB,GAAG,MAAM,EAAU,CAAC;QAGlD,IAAI,CAAC,gBAAgB,EAAE,CAAC;QACxB,IAAI,CAAC,kBAAkB,EAAE,CAAC;IAC9B,CAAC;IAEM,QAAQ;QACX,IAAI,CAAC,YAAY,EAAE,CAAC;IACxB,CAAC;IAES,aAAa,CAAC,MAAyC,EAAE,aAAqB;QACpF,MAAM,OAAO,GAAgF;YACzF,KAAK,EAAE,CAAC,KAAa,EAAE,EAAE,CAAC,IAAI,CAAC,WAAW,CAAC,KAAK,EAAE,aAAa,CAAC;YAChE,WAAW,EAAE,CAAC,WAAmB,EAAE,EAAE,CAAC,IAAI,CAAC,iBAAiB,CAAC,WAAW,EAAE,aAAa,CAAC;SAC3F,CAAC;QAEF,QAAQ,CACJ,MAAM,CAAC,IAAI,CAAC,MAAM,CAAC,CAAC,GAAG,CAAC,CAAC,GAAG,EAAE,EAAE,CAC5B,OAAO,CAAC,GAAqC,CAAC,CAAC,MAAM,CAAC,GAAqC,CAAC,IAAI,EAAE,CAAC,CACtG,CACJ;aACI,IAAI,CAAC,kBAAkB,CAAC,IAAI,CAAC,UAAU,CAAC,CAAC;aACzC,SAAS,EAAE,CAAC;IACrB,CAAC;IAES,uBAAuB,CAAC,OAAgB;QAC9C,IAAI,CAAC,WAAW,CAAC,GAAG,CAAC,OAAO,CAAC,CAAC;QAC9B,IAAI,CAAC,iBAAiB,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC;IACzC,CAAC;IAEO,kBAAkB;QACtB,MAAM,CAAC,GAAG,EAAE;YACR,MAAM,QAAQ,GAAG,IAAI,CAAC,QAAQ,EAAE,CAAC;YACjC,MAAM,wBAAwB,GAAG,SAAS,CAAC,IAAI,CAAC,WAAW,CAAC,CAAC;YAE7D,IAAI,CAAC,QAAQ,IAAI,IAAI,CAAC,kBAAkB,EAAE,CAAC;gBACvC,IAAI,CAAC,kBAAkB,GAAG,KAAK,CAAC;gBAEhC,OAAO;YACX,CAAC;YAED,IAAI,wBAAwB,EAAE,CAAC;gBAC3B,IAAI,CAAC,cAAc,CAAC,QAAQ,CAAC,EAAE,EAAE,wBAAwB,CAAC,EAAE,CAAC,CAAC;gBAE9D,OAAO;YACX,CAAC;YAED,IAAI,CAAC,iBAAiB,CAAC,QAAQ,CAAC,EAAE,CAAC,CAAC;QACxC,CAAC,CAAC,CAAC;IACP,CAAC;IAEO,gBAAgB;QACpB,MAAM,CAAC,GAAG,EAAE;YACR,MAAM,OAAO,GAAG,IAAI,CAAC,OAAO,EAAE,EAAE,aAAa,CAAC;YAE9C,IAAI,CAAC,OAAO,EAAE,CAAC;gBACX,OAAO;YACX,CAAC;YAED,IAAI,CAAC,YAAY,CAAC,GAAG,CAAC,EAAE,KAAK,EAAE,IAAI,CAAC,KAAK,EAAE,EAAE,OAAO,EAAE,CAAC,CAAC;QAC5D,CAAC,CAAC,CAAC;IACP,CAAC;IAEO,YAAY;QAChB,MAAM,eAAe,GAAG,IAAI,CAAC,WAAW,EAAE,EAAE,EAAE,CAAC;QAE/C,IAAI,CAAC,eAAe,EAAE,CAAC;YACnB,OAAO;QACX,CAAC;QAED,IAAI,CAAC,6BAA6B;aAC7B,WAAW,CAAC,eAAe,yDAA6B;aACxD,IAAI,CAAC,kBAAkB,CAAC,IAAI,CAAC,UAAU,CAAC,CAAC;aACzC,SAAS,CAAC,CAAC,QAAQ,EAAE,EAAE;YACpB,IAAI,CAAC,kBAAkB,GAAG,IAAI,CAAC;YAC/B,IAAI,CAAC,cAAc,CAAC,GAAG,CAAC,QAAQ,CAAC,EAAE,CAAC,CAAC;YACrC,IAAI,CAAC,yBAAyB,CAAC,kBAAkB,CAAC,QAAsB,CAAC,CAAC;QAC9E,CAAC,CAAC,CAAC;IACX,CAAC;IAEO,cAAc,CAAC,UAAkB,EAAE,QAAgB;QACvD,MAAM,cAAc,GAAG,SAAS,CAAC,IAAI,CAAC,cAAc,CAAC,CAAC;QAEtD,IAAI,CAAC,cAAc,EAAE,CAAC;YAClB,OAAO;QACX,CAAC;QAED,IAAI,CAAC,YAAY,EAAE,WAAW,EAAE,CAAC;QACjC,IAAI,CAAC,YAAY,GAAG,IAAI,CAAC,6BAA6B;aACjD,WAAW,CAAC;YACT,IAAI,iEAA+B;YACnC,aAAa,EAAE,QAAQ;YACvB,sBAAsB,EAAE,UAAU;YAClC,uBAAuB,EAAE,cAAc;SAC1C,CAAC;aACD,SAAS,CAAC,GAAG,EAAE,CAAC,IAAI,CAAC,cAAc,CAAC,GAAG,CAAC,UAAU,CAAC,CAAC,CAAC;IAC9D,CAAC;IAEO,iBAAiB,CAAC,UAAkB;QACxC,IAAI,CAAC,kBAAkB,CAAC,IAAI,CAAC,UAAU,CAAC,CAAC;QACzC,IAAI,CAAC,cAAc,CAAC,GAAG,CAAC,UAAU,CAAC,CAAC;IACxC,CAAC;IAEO,WAAW,CAAC,KAAa,EAAE,aAAqB;QACpD,OAAO,IAAI,CAAC,qBAAqB,CAAC,WAAW,CAAC,aAAa,EAAE,KAAK,CAAC,CAAC,IAAI,CACpE,QAAQ,CAAC,GAAG,EAAE,CACV,IAAI,CAAC,WAAW,CAAC,MAAM,CAAC,CAAC,IAAI,EAAE,EAAE,CAAC,CAAC;YAC/B,GAAG,IAAK;YACR,cAAc,EAAE,EAAE,GAAG,IAAK,CAAC,cAAc,EAAE,KAAK,EAAE,KAAK,EAAE;SAC5D,CAAC,CAAC,CACN,CACJ,CAAC;IACN,CAAC;IAEO,iBAAiB,CAAC,KAAa,EAAE,aAAqB;QAC1D,IAAI,CAAC,IAAI,CAAC,qBAAqB,CAAC,iBAAiB,EAAE,CAAC;YAChD,MAAM,IAAI,KAAK,CAAC,oCAAoC,CAAC,CAAC;QAC1D,CAAC;QAED,OAAO,IAAI,CAAC,qBAAqB,CAAC,iBAAiB,CAAC,aAAa,EAAE,KAAK,CAAC,CAAC,IAAI,CAC1E,QAAQ,CAAC,GAAG,EAAE,CACV,IAAI,CAAC,WAAW,CAAC,MAAM,CAAC,CAAC,IAAI,EAAE,EAAE,CAAC,CAAC;YAC/B,GAAG,IAAK;YACR,cAAc,EAAE,EAAE,GAAG,IAAK,CAAC,cAAc,EAAE,gBAAgB,EAAE,KAAK,EAAE;SACvE,CAAC,CAAC,CACN,CACJ,CAAC;IACN,CAAC;+GAjKQ,2BAA2B;mGAA3B,2BAA2B,4mBAFzB,CAAC,0BAA0B,CAAC,oJCzD3C,yxDAiDA,qgBDNQ,cAAc,8BACd,eAAe,+BACf,sBAAsB,8GACtB,eAAe,8BACf,kBAAkB,kHAClB,+BAA+B,gMAC/B,cAAc,sIACd,YAAY,uKACZ,eAAe;;4FAQV,2BAA2B;kBAnBvC,SAAS;+BACI,sBAAsB,WACvB;wBACL,cAAc;wBACd,eAAe;wBACf,sBAAsB;wBACtB,eAAe;wBACf,kBAAkB;wBAClB,+BAA+B;wBAC/B,cAAc;wBACd,YAAY;wBACZ,eAAe;qBAClB,cACW,IAAI,mBAGC,uBAAuB,CAAC,MAAM,aACpC,CAAC,0BAA0B,CAAC","sourcesContent":["import type { ElementRef, OnInit } from '@angular/core';\r\nimport {\r\n    ChangeDetectionStrategy,\r\n    Component,\r\n    computed,\r\n    DestroyRef,\r\n    effect,\r\n    inject,\r\n    input,\r\n    model,\r\n    output,\r\n    signal,\r\n    untracked,\r\n    viewChild\r\n} from '@angular/core';\r\nimport { takeUntilDestroyed } from '@angular/core/rxjs-interop';\r\nimport { TranslocoModule, TranslocoService } from '@jsverse/transloco';\r\nimport type { Observable, Subscription } from 'rxjs';\r\nimport { finalize, forkJoin } from 'rxjs';\r\nimport type { Instruction, StorageObject, TreeStruct } from '@cuby-ui/api';\r\nimport { CuiTreeStructNavigatorApiService, MovingType, TreeStructType } from '@cuby-ui/api';\r\nimport type { CuiNullable } from '@cuby-ui/cdk';\r\nimport { CuiIdService } from '@cuby-ui/cdk';\r\n\r\nimport {\r\n    CuiBadgeModule,\r\n    CuiButtonModule,\r\n    CuiDialogModule,\r\n    CuiFormFieldModule,\r\n    CuiLabelModule,\r\n    CuiSvgModule\r\n} from '../../components';\r\nimport { CuiCategoriesComponent, CuiSelectedCategoryService } from '../categories';\r\nimport { CuiEditorConfig } from '../../editor';\r\nimport type { CuiTools } from '../../editor';\r\nimport { CuiInstructionInfoFormComponent } from '../instruction-info-form';\r\nimport type { CuiInstructionFormValues } from '../instruction-info-form';\r\nimport { UTILITY_MODAL_SERVICE_TOKEN, UtilityModalService } from '../utility-modal';\r\n\r\n\r\n@Component({\r\n    selector: 'cui-instruction-info',\r\n    imports: [\r\n        CuiBadgeModule,\r\n        CuiButtonModule,\r\n        CuiCategoriesComponent,\r\n        CuiDialogModule,\r\n        CuiFormFieldModule,\r\n        CuiInstructionInfoFormComponent,\r\n        CuiLabelModule,\r\n        CuiSvgModule,\r\n        TranslocoModule\r\n    ],\r\n    standalone: true,\r\n    templateUrl: './instruction-info.component.html',\r\n    styleUrls: ['./instruction-info.component.scss'],\r\n    changeDetection: ChangeDetectionStrategy.OnPush,\r\n    providers: [CuiSelectedCategoryService]\r\n})\r\nexport class CuiInstructionInfoComponent implements OnInit {\r\n    private readonly destroyRef = inject(DestroyRef);\r\n    private readonly translocoService = inject(TranslocoService);\r\n    private readonly cuiIdService = inject(CuiIdService);\r\n    private readonly cuiCategoriesModalService = inject(CuiSelectedCategoryService);\r\n    private readonly treeStructNavigatorApiService = inject(CuiTreeStructNavigatorApiService);\r\n    private readonly instructionApiService = inject<UtilityModalService<Instruction>>(UTILITY_MODAL_SERVICE_TOKEN);\r\n\r\n    protected readonly TYPE = TreeStructType.Instruction;\r\n\r\n    private subscription!: Subscription;\r\n    private skipCategoryChange = false;\r\n\r\n    protected readonly categoryId = this.cuiIdService.generate();\r\n\r\n    protected readonly category = this.cuiCategoriesModalService.category;\r\n    protected readonly isFormValid = signal(false);\r\n    protected readonly prevCategoryId = signal<CuiNullable<string>>(null);\r\n    protected readonly editorConfig = signal<CuiNullable<CuiEditorConfig>>(null);\r\n    protected readonly labelColor = computed(() =>\r\n        this.category()?.title ? 'var(--cui-base-900)' : 'var(--cui-base-400)'\r\n    );\r\n    protected readonly selectTitle = computed(\r\n        () => this.category()?.title || this.translocoService.translate('SELECT') + '...'\r\n    );\r\n\r\n    protected readonly element = viewChild<ElementRef<HTMLElement>>('scrollContent');\r\n\r\n    public readonly instruction = model<StorageObject<Instruction>>();\r\n    public readonly tools = input.required<CuiTools>();\r\n    public readonly editorResourceUrl = input.required<string>();\r\n\r\n    public readonly validationChanged = output<boolean>();\r\n    public readonly instructionCreated = output<string>();\r\n\r\n    constructor() {\r\n        this.initConfigEffect();\r\n        this.initCategoryEffect();\r\n    }\r\n\r\n    public ngOnInit(): void {\r\n        this.initCategory();\r\n    }\r\n\r\n    protected onFormChanged(values: Partial<CuiInstructionFormValues>, instructionId: string): void {\r\n        const actions: Record<keyof CuiInstructionFormValues, (value: string) => Observable<void>> = {\r\n            title: (title: string) => this.changeTitle(title, instructionId),\r\n            description: (description: string) => this.changeDescription(description, instructionId)\r\n        };\r\n\r\n        forkJoin(\r\n            Object.keys(values).map((key) =>\r\n                actions[key as keyof CuiInstructionFormValues](values[key as keyof CuiInstructionFormValues] || '')\r\n            )\r\n        )\r\n            .pipe(takeUntilDestroyed(this.destroyRef))\r\n            .subscribe();\r\n    }\r\n\r\n    protected onFormValidationChanged(isValid: boolean): void {\r\n        this.isFormValid.set(isValid);\r\n        this.validationChanged.emit(isValid);\r\n    }\r\n\r\n    private initCategoryEffect(): void {\r\n        effect(() => {\r\n            const category = this.category();\r\n            const storageObjectInstruction = untracked(this.instruction);\r\n\r\n            if (!category || this.skipCategoryChange) {\r\n                this.skipCategoryChange = false;\r\n\r\n                return;\r\n            }\r\n\r\n            if (storageObjectInstruction) {\r\n                this.changeCategory(category.id, storageObjectInstruction.id);\r\n\r\n                return;\r\n            }\r\n\r\n            this.createInstruction(category.id);\r\n        });\r\n    }\r\n\r\n    private initConfigEffect(): void {\r\n        effect(() => {\r\n            const element = this.element()?.nativeElement;\r\n\r\n            if (!element) {\r\n                return;\r\n            }\r\n\r\n            this.editorConfig.set({ tools: this.tools(), element });\r\n        });\r\n    }\r\n\r\n    private initCategory(): void {\r\n        const storageObjectId = this.instruction()?.id;\r\n\r\n        if (!storageObjectId) {\r\n            return;\r\n        }\r\n\r\n        this.treeStructNavigatorApiService\r\n            .getCategory(storageObjectId, TreeStructType.Instruction)\r\n            .pipe(takeUntilDestroyed(this.destroyRef))\r\n            .subscribe((category) => {\r\n                this.skipCategoryChange = true;\r\n                this.prevCategoryId.set(category.id);\r\n                this.cuiCategoriesModalService.setCurrentCategory(category as TreeStruct);\r\n            });\r\n    }\r\n\r\n    private changeCategory(categoryId: string, targetId: string): void {\r\n        const prevCategoryId = untracked(this.prevCategoryId);\r\n\r\n        if (!prevCategoryId) {\r\n            return;\r\n        }\r\n\r\n        this.subscription?.unsubscribe();\r\n        this.subscription = this.treeStructNavigatorApiService\r\n            .moveElement({\r\n                type: MovingType.TARGET_TREE_STRUCT,\r\n                movableItemId: targetId,\r\n                treeStructTargetItemId: categoryId,\r\n                treeStructMovableItemId: prevCategoryId\r\n            })\r\n            .subscribe(() => this.prevCategoryId.set(categoryId));\r\n    }\r\n\r\n    private createInstruction(categoryId: string): void {\r\n        this.instructionCreated.emit(categoryId);\r\n        this.prevCategoryId.set(categoryId);\r\n    }\r\n\r\n    private changeTitle(value: string, instructionId: string): Observable<void> {\r\n        return this.instructionApiService.changeTitle(instructionId, value).pipe(\r\n            finalize(() =>\r\n                this.instruction.update((prev) => ({\r\n                    ...prev!,\r\n                    storageElement: { ...prev!.storageElement, title: value }\r\n                }))\r\n            )\r\n        );\r\n    }\r\n\r\n    private changeDescription(value: string, instructionId: string): Observable<void> {\r\n        if (!this.instructionApiService.changeDescription) {\r\n            throw new Error('Change description not implemented');\r\n        }\r\n\r\n        return this.instructionApiService.changeDescription(instructionId, value).pipe(\r\n            finalize(() =>\r\n                this.instruction.update((prev) => ({\r\n                    ...prev!,\r\n                    storageElement: { ...prev!.storageElement, shortDescription: value }\r\n                }))\r\n            )\r\n        );\r\n    }\r\n}\r\n","<div\r\n    class=\"wrapper\"\r\n    *transloco=\"let t\"\r\n>\r\n    <div\r\n        #scrollContent\r\n        class=\"content content_scrollable\"\r\n    >\r\n        <div class=\"category\">\r\n            <cui-form-field>\r\n                <label\r\n                    cuiLabel\r\n                    [for]=\"categoryId\"\r\n                    [isRequired]=\"true\"\r\n                >\r\n                    {{ t('SELECT_CATEGORY') }}\r\n                </label>\r\n                <cui-categories\r\n                    [buttonId]=\"categoryId\"\r\n                    [treeStructType]=\"TYPE\"\r\n                >\r\n                    <div class=\"select\">\r\n                        <span [style.color]=\"labelColor()\">\r\n                            {{ selectTitle() }}\r\n                        </span>\r\n\r\n                        <cui-svg icon=\"cuiIconChevronDown\" />\r\n                    </div>\r\n                </cui-categories>\r\n            </cui-form-field>\r\n        </div>\r\n\r\n        @let instructionData = instruction();\r\n        @let editorConfigData = editorConfig();\r\n\r\n        @if (editorConfigData && instructionData) {\r\n            <cui-instruction-info-form\r\n                [config]=\"editorConfigData\"\r\n                [baseResourceUrl]=\"editorResourceUrl()\"\r\n                [title]=\"instructionData.storageElement.title\"\r\n                [description]=\"instructionData.storageElement.shortDescription\"\r\n                [framerId]=\"instructionData.storageElement.framerInstructionId\"\r\n                class=\"utility\"\r\n                (isFormValidChanged)=\"onFormValidationChanged($event)\"\r\n                (formChanged)=\"onFormChanged($event, instructionData.id)\"\r\n            />\r\n        }\r\n    </div>\r\n</div>\r\n"]}
@@ -0,0 +1,2 @@
1
+ export { CuiInstructionInfoFormComponent } from './instruction-info-form.component';
2
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi9wcm9qZWN0cy9jb3JlL3dpZGdldHMvaW5zdHJ1Y3Rpb24taW5mby1mb3JtL2luZGV4LnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLE9BQU8sRUFBRSwrQkFBK0IsRUFBRSxNQUFNLG1DQUFtQyxDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiZXhwb3J0IHsgQ3VpSW5zdHJ1Y3Rpb25JbmZvRm9ybUNvbXBvbmVudCB9IGZyb20gJy4vaW5zdHJ1Y3Rpb24taW5mby1mb3JtLmNvbXBvbmVudCc7XG5leHBvcnQgeyBDdWlJbnN0cnVjdGlvbkZvcm1Hcm91cCwgQ3VpSW5zdHJ1Y3Rpb25Gb3JtVmFsdWVzIH0gZnJvbSAnLi9pbnN0cnVjdGlvbi1pbmZvLWZvcm0ub3B0aW9ucyc7XG4iXX0=
@@ -0,0 +1,153 @@
1
+ import { ChangeDetectionStrategy, Component, DestroyRef, effect, inject, input, output, signal } from '@angular/core';
2
+ import { takeUntilDestroyed } from '@angular/core/rxjs-interop';
3
+ import { NonNullableFormBuilder, ReactiveFormsModule } from '@angular/forms';
4
+ import { TranslocoDirective } from '@jsverse/transloco';
5
+ import { debounceTime, map, pairwise, startWith } from 'rxjs';
6
+ import { CuiFrameApiService } from '@cuby-ui/api';
7
+ import { CuiIdService, takeCaptureFromVideo } from '@cuby-ui/cdk';
8
+ import { CuiAccordionModule, CuiButtonModule, CuiFormFieldModule, CuiHintModule, CuiInputModule, CuiLabelModule } from '../../components';
9
+ import { cuiCreateDefaultValidators } from '../../utils';
10
+ import { CuiGeneralControlErrorHintComponent } from '../general-control-error-hint';
11
+ import { CuiEditorBlockComponent } from '../../editor';
12
+ import { CuiUtilityThumbnailComponent } from '../utility-thumbnail';
13
+ import * as i0 from "@angular/core";
14
+ import * as i1 from "../../components/form-field/form-field.component";
15
+ import * as i2 from "../../components/input-text/input-text.component";
16
+ import * as i3 from "../../directives/text-field-controller/text-field-id.directive";
17
+ import * as i4 from "../../directives/text-field-controller/text-field-placeholder.directive";
18
+ import * as i5 from "../../directives/text-field-controller/text-field-is-error.directive";
19
+ import * as i6 from "../../components/label/label.component";
20
+ import * as i7 from "@angular/forms";
21
+ export class CuiInstructionInfoFormComponent {
22
+ get controls() {
23
+ return this.form.controls;
24
+ }
25
+ get isTitleError() {
26
+ return this.controls.title.invalid && this.controls.title.dirty;
27
+ }
28
+ get isDescriptionError() {
29
+ return this.controls.description.invalid && this.controls.description.dirty;
30
+ }
31
+ constructor() {
32
+ this.destroy = inject(DestroyRef);
33
+ this.formBuilder = inject(NonNullableFormBuilder);
34
+ this.frameApiService = inject(CuiFrameApiService);
35
+ this.cuiIdService = inject(CuiIdService);
36
+ this.CONTROL_DELAY = 500;
37
+ this.titleId = this.cuiIdService.generate();
38
+ this.descriptionId = this.cuiIdService.generate();
39
+ this.form = this.formBuilder.group({
40
+ title: this.formBuilder.control('', cuiCreateDefaultValidators({ asyncDelay: this.CONTROL_DELAY })),
41
+ description: this.formBuilder.control('', cuiCreateDefaultValidators({ asyncDelay: this.CONTROL_DELAY }))
42
+ });
43
+ this.heading = null;
44
+ this.isDescriptionEmpty = signal(false);
45
+ this.editorId = signal(null);
46
+ this.thumbnail = signal(null);
47
+ this.title = input.required();
48
+ this.description = input.required();
49
+ this.framerId = input.required();
50
+ this.baseResourceUrl = input.required();
51
+ this.config = input.required();
52
+ this.formChanged = output();
53
+ this.isFormValidChanged = output();
54
+ this.initUtilityEffect();
55
+ }
56
+ onEditorEmpty(isEmpty) {
57
+ this.isDescriptionEmpty.set(isEmpty);
58
+ this.isFormValidChanged.emit(this.checkFormValidation());
59
+ }
60
+ initUtilityEffect() {
61
+ const effectRef = effect(() => {
62
+ const title = this.title();
63
+ if (!title) {
64
+ return;
65
+ }
66
+ this.initFormSubscriptions();
67
+ this.initFrameFragments();
68
+ effectRef.destroy();
69
+ });
70
+ }
71
+ initFormSubscriptions() {
72
+ this.form.setValue({ title: this.title(), description: this.description() }, { emitEvent: false });
73
+ this.form.statusChanges
74
+ .pipe(takeUntilDestroyed(this.destroy))
75
+ .subscribe(() => this.isFormValidChanged.emit(this.checkFormValidation()));
76
+ this.form.valueChanges
77
+ .pipe(takeUntilDestroyed(this.destroy), startWith(this.form.value), pairwise(), debounceTime(this.CONTROL_DELAY), map(([prev, current]) => this.compareFormValues([prev, current])))
78
+ .subscribe((value) => {
79
+ if (!Object.keys(value).length) {
80
+ return;
81
+ }
82
+ this.formChanged.emit(value);
83
+ });
84
+ }
85
+ initFrameFragments() {
86
+ const frameId = this.framerId();
87
+ if (!frameId) {
88
+ return;
89
+ }
90
+ this.frameApiService.getFrame(frameId).subscribe((frame) => {
91
+ frame.fragments.forEach((fragment) => {
92
+ if (fragment.type === "J_TEXT" /* FragmentType.jText */) {
93
+ this.editorId.set(fragment?.id);
94
+ }
95
+ if (fragment.type === "RESOURCE_FILE" /* FragmentType.resourceFile */) {
96
+ this.setResourceFile(fragment);
97
+ }
98
+ });
99
+ });
100
+ }
101
+ compareFormValues([prevGroup, currentGruop]) {
102
+ const keys = Object.keys(currentGruop);
103
+ const changedFields = {};
104
+ keys.forEach((key) => {
105
+ const prev = prevGroup[key];
106
+ const current = currentGruop[key];
107
+ if (prev === current || this.controls[key].invalid) {
108
+ return;
109
+ }
110
+ changedFields[key] = current;
111
+ });
112
+ return changedFields;
113
+ }
114
+ setResourceFile(resource) {
115
+ const url = this.baseResourceUrl() + resource.fileId;
116
+ this.thumbnail.set(resource);
117
+ if (!url || resource?.name?.split('.').pop() !== 'mp4') {
118
+ return;
119
+ }
120
+ this.getCaptureFromVideo(url);
121
+ }
122
+ getCaptureFromVideo(url, file) {
123
+ takeCaptureFromVideo(url).then((capture) => {
124
+ this.thumbnail.update((prev) => ({
125
+ ...prev,
126
+ name: (file?.name ?? prev?.name),
127
+ resource: capture
128
+ }));
129
+ });
130
+ }
131
+ checkFormValidation() {
132
+ return !this.isDescriptionEmpty() && !this.form.invalid;
133
+ }
134
+ static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: CuiInstructionInfoFormComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
135
+ static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "18.2.13", type: CuiInstructionInfoFormComponent, isStandalone: true, selector: "cui-instruction-info-form", inputs: { title: { classPropertyName: "title", publicName: "title", isSignal: true, isRequired: true, transformFunction: null }, description: { classPropertyName: "description", publicName: "description", isSignal: true, isRequired: true, transformFunction: null }, framerId: { classPropertyName: "framerId", publicName: "framerId", isSignal: true, isRequired: true, transformFunction: null }, baseResourceUrl: { classPropertyName: "baseResourceUrl", publicName: "baseResourceUrl", isSignal: true, isRequired: true, transformFunction: null }, config: { classPropertyName: "config", publicName: "config", isSignal: true, isRequired: true, transformFunction: null } }, outputs: { formChanged: "formChanged", isFormValidChanged: "isFormValidChanged" }, ngImport: i0, template: "<ng-container *transloco=\"let t\">\r\n <section class=\"content\">\r\n <cui-form-field>\r\n <label\r\n cuiLabel\r\n [isRequired]=\"true\"\r\n [for]=\"titleId\"\r\n >\r\n {{ t('TITLE') }}\r\n </label>\r\n\r\n <cui-input-text\r\n [cuiTextFieldId]=\"titleId\"\r\n [cuiTextFieldPlaceholder]=\"t('ADD_A_TITLE')\"\r\n [cuiTextFieldIsError]=\"isTitleError\"\r\n [formControl]=\"form.controls.title\"\r\n />\r\n\r\n @if (isTitleError) {\r\n <cui-general-control-error-hint\r\n [control]=\"form.controls.title\"\r\n [isVisible]=\"true\"\r\n />\r\n }\r\n </cui-form-field>\r\n\r\n <cui-form-field>\r\n <label\r\n cuiLabel\r\n [isRequired]=\"true\"\r\n [for]=\"descriptionId\"\r\n >\r\n {{ t('SHORT_DESCRIPTION') }}\r\n </label>\r\n\r\n <cui-input-text\r\n [cuiTextFieldId]=\"descriptionId\"\r\n cuiTextFieldPlaceholder=\"{{ t('SHORT_DESCRIPTION') }}...\"\r\n [cuiTextFieldIsError]=\"isDescriptionError\"\r\n [formControl]=\"form.controls.description\"\r\n />\r\n\r\n @if (isDescriptionError) {\r\n <cui-general-control-error-hint\r\n [control]=\"form.controls.description\"\r\n [isVisible]=\"true\"\r\n />\r\n }\r\n </cui-form-field>\r\n\r\n <cui-utility-thumbnail\r\n [baseResourceUrl]=\"baseResourceUrl()\"\r\n [framerId]=\"framerId()\"\r\n [(thumbnail)]=\"thumbnail\"\r\n />\r\n\r\n @if (editorId(); as editorId) {\r\n <cui-editor-block\r\n [title]=\"form.controls.title.value || title()\"\r\n [config]=\"config()\"\r\n [editorId]=\"editorId\"\r\n [required]=\"true\"\r\n class=\"editor\"\r\n (editorEmpty)=\"onEditorEmpty($event)\"\r\n />\r\n }\r\n </section>\r\n</ng-container>\r\n", styles: ["::ng-deep cui-dialog:has(.instruction-description-modal){inset:0;margin:auto;transform:none!important}:host{--editor-height: 650px;display:block}:host .editor{--max-editor-height: var(--editor-height)}.content{display:flex;flex-direction:column;gap:16px;padding:16px 24px}\n"], dependencies: [{ kind: "ngmodule", type: CuiAccordionModule }, { kind: "ngmodule", type: CuiButtonModule }, { kind: "component", type: CuiEditorBlockComponent, selector: "cui-editor-block", inputs: ["title", "editorId", "config", "required"], outputs: ["editorEmpty"] }, { kind: "ngmodule", type: CuiFormFieldModule }, { kind: "component", type: i1.CuiFormFieldComponent, selector: "cui-form-field" }, { kind: "component", type: CuiGeneralControlErrorHintComponent, selector: "cui-general-control-error-hint", inputs: ["control", "isVisible"] }, { kind: "ngmodule", type: CuiHintModule }, { kind: "ngmodule", type: CuiInputModule }, { kind: "component", type: i2.CuiInputTextComponent, selector: "cui-input-text" }, { kind: "directive", type: i3.CuiTextFieldIdDirective, selector: "[cuiTextFieldId]", inputs: ["cuiTextFieldId"] }, { kind: "directive", type: i4.CuiTextFieldPlaceholderDirective, selector: "[cuiTextFieldPlaceholder]", inputs: ["cuiTextFieldPlaceholder"] }, { kind: "directive", type: i5.CuiTextFieldIsErrorDirective, selector: "[cuiTextFieldIsError]", inputs: ["cuiTextFieldIsError"] }, { kind: "ngmodule", type: CuiLabelModule }, { kind: "component", type: i6.CuiLabelComponent, selector: "label[cuiLabel]", inputs: ["isRequired"] }, { kind: "component", type: CuiUtilityThumbnailComponent, selector: "cui-utility-thumbnail", inputs: ["thumbnail", "framerId", "baseResourceUrl"], outputs: ["thumbnailChange"] }, { kind: "ngmodule", type: ReactiveFormsModule }, { kind: "directive", type: i7.NgControlStatus, selector: "[formControlName],[ngModel],[formControl]" }, { kind: "directive", type: i7.FormControlDirective, selector: "[formControl]", inputs: ["formControl", "disabled", "ngModel"], outputs: ["ngModelChange"], exportAs: ["ngForm"] }, { kind: "directive", type: TranslocoDirective, selector: "[transloco]", inputs: ["transloco", "translocoParams", "translocoScope", "translocoRead", "translocoPrefix", "translocoLang", "translocoLoadingTpl"] }], changeDetection: i0.ChangeDetectionStrategy.OnPush }); }
136
+ }
137
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: CuiInstructionInfoFormComponent, decorators: [{
138
+ type: Component,
139
+ args: [{ selector: 'cui-instruction-info-form', imports: [
140
+ CuiAccordionModule,
141
+ CuiButtonModule,
142
+ CuiEditorBlockComponent,
143
+ CuiFormFieldModule,
144
+ CuiGeneralControlErrorHintComponent,
145
+ CuiHintModule,
146
+ CuiInputModule,
147
+ CuiLabelModule,
148
+ CuiUtilityThumbnailComponent,
149
+ ReactiveFormsModule,
150
+ TranslocoDirective
151
+ ], standalone: true, changeDetection: ChangeDetectionStrategy.OnPush, template: "<ng-container *transloco=\"let t\">\r\n <section class=\"content\">\r\n <cui-form-field>\r\n <label\r\n cuiLabel\r\n [isRequired]=\"true\"\r\n [for]=\"titleId\"\r\n >\r\n {{ t('TITLE') }}\r\n </label>\r\n\r\n <cui-input-text\r\n [cuiTextFieldId]=\"titleId\"\r\n [cuiTextFieldPlaceholder]=\"t('ADD_A_TITLE')\"\r\n [cuiTextFieldIsError]=\"isTitleError\"\r\n [formControl]=\"form.controls.title\"\r\n />\r\n\r\n @if (isTitleError) {\r\n <cui-general-control-error-hint\r\n [control]=\"form.controls.title\"\r\n [isVisible]=\"true\"\r\n />\r\n }\r\n </cui-form-field>\r\n\r\n <cui-form-field>\r\n <label\r\n cuiLabel\r\n [isRequired]=\"true\"\r\n [for]=\"descriptionId\"\r\n >\r\n {{ t('SHORT_DESCRIPTION') }}\r\n </label>\r\n\r\n <cui-input-text\r\n [cuiTextFieldId]=\"descriptionId\"\r\n cuiTextFieldPlaceholder=\"{{ t('SHORT_DESCRIPTION') }}...\"\r\n [cuiTextFieldIsError]=\"isDescriptionError\"\r\n [formControl]=\"form.controls.description\"\r\n />\r\n\r\n @if (isDescriptionError) {\r\n <cui-general-control-error-hint\r\n [control]=\"form.controls.description\"\r\n [isVisible]=\"true\"\r\n />\r\n }\r\n </cui-form-field>\r\n\r\n <cui-utility-thumbnail\r\n [baseResourceUrl]=\"baseResourceUrl()\"\r\n [framerId]=\"framerId()\"\r\n [(thumbnail)]=\"thumbnail\"\r\n />\r\n\r\n @if (editorId(); as editorId) {\r\n <cui-editor-block\r\n [title]=\"form.controls.title.value || title()\"\r\n [config]=\"config()\"\r\n [editorId]=\"editorId\"\r\n [required]=\"true\"\r\n class=\"editor\"\r\n (editorEmpty)=\"onEditorEmpty($event)\"\r\n />\r\n }\r\n </section>\r\n</ng-container>\r\n", styles: ["::ng-deep cui-dialog:has(.instruction-description-modal){inset:0;margin:auto;transform:none!important}:host{--editor-height: 650px;display:block}:host .editor{--max-editor-height: var(--editor-height)}.content{display:flex;flex-direction:column;gap:16px;padding:16px 24px}\n"] }]
152
+ }], ctorParameters: () => [] });
153
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"instruction-info-form.component.js","sourceRoot":"","sources":["../../../../../projects/core/widgets/instruction-info-form/instruction-info-form.component.ts","../../../../../projects/core/widgets/instruction-info-form/instruction-info-form.component.html"],"names":[],"mappings":"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","sourcesContent":["import { ChangeDetectionStrategy, Component, DestroyRef, effect, inject, input, output, signal } from '@angular/core';\r\nimport { takeUntilDestroyed } from '@angular/core/rxjs-interop';\r\nimport { NonNullableFormBuilder, ReactiveFormsModule } from '@angular/forms';\r\nimport { TranslocoDirective } from '@jsverse/transloco';\r\nimport { debounceTime, map, pairwise, startWith } from 'rxjs';\r\nimport type { ResourceFragment } from '@cuby-ui/api';\r\nimport { CuiFrameApiService, FragmentType } from '@cuby-ui/api';\r\nimport type { CuiNullable } from '@cuby-ui/cdk';\r\nimport { CuiIdService, takeCaptureFromVideo } from '@cuby-ui/cdk';\r\n\r\nimport {\r\n    CuiAccordionModule,\r\n    CuiButtonModule,\r\n    CuiFormFieldModule,\r\n    CuiHintModule,\r\n    CuiInputModule,\r\n    CuiLabelModule\r\n} from '../../components';\r\nimport { cuiCreateDefaultValidators } from '../../utils';\r\nimport { CuiGeneralControlErrorHintComponent } from '../general-control-error-hint';\r\nimport { CuiEditorBlockComponent } from '../../editor';\r\nimport type { CuiEditorConfig } from '../../editor';\r\nimport { CuiUtilityThumbnailComponent } from '../utility-thumbnail';\r\n\r\nimport type { CuiInstructionFormGroup, CuiInstructionFormValues } from './instruction-info-form.options';\r\n\r\n@Component({\r\n    selector: 'cui-instruction-info-form',\r\n    imports: [\r\n        CuiAccordionModule,\r\n        CuiButtonModule,\r\n        CuiEditorBlockComponent,\r\n        CuiFormFieldModule,\r\n        CuiGeneralControlErrorHintComponent,\r\n        CuiHintModule,\r\n        CuiInputModule,\r\n        CuiLabelModule,\r\n        CuiUtilityThumbnailComponent,\r\n        ReactiveFormsModule,\r\n        TranslocoDirective\r\n    ],\r\n    standalone: true,\r\n    templateUrl: './instruction-info-form.component.html',\r\n    styleUrl: './instruction-info-form.component.scss',\r\n    changeDetection: ChangeDetectionStrategy.OnPush\r\n})\r\nexport class CuiInstructionInfoFormComponent {\r\n    private readonly destroy = inject(DestroyRef);\r\n    private readonly formBuilder = inject(NonNullableFormBuilder);\r\n    private readonly frameApiService = inject(CuiFrameApiService);\r\n    private readonly cuiIdService = inject(CuiIdService);\r\n\r\n    protected readonly CONTROL_DELAY = 500;\r\n\r\n    protected readonly titleId = this.cuiIdService.generate();\r\n    protected readonly descriptionId = this.cuiIdService.generate();\r\n    protected readonly form = this.formBuilder.group<CuiInstructionFormGroup>({\r\n        title: this.formBuilder.control('', cuiCreateDefaultValidators({ asyncDelay: this.CONTROL_DELAY })),\r\n        description: this.formBuilder.control('', cuiCreateDefaultValidators({ asyncDelay: this.CONTROL_DELAY }))\r\n    });\r\n\r\n    protected heading: CuiNullable<string> = null;\r\n\r\n    protected readonly isDescriptionEmpty = signal(false);\r\n    protected readonly editorId = signal<CuiNullable<string>>(null);\r\n    protected readonly thumbnail = signal<CuiNullable<ResourceFragment>>(null);\r\n\r\n    public readonly title = input.required<string>();\r\n    public readonly description = input.required<string>();\r\n    public readonly framerId = input.required<string>();\r\n    public readonly baseResourceUrl = input.required<string>();\r\n    public readonly config = input.required<CuiEditorConfig>();\r\n\r\n    public readonly formChanged = output<Partial<CuiInstructionFormValues>>();\r\n    public readonly isFormValidChanged = output<boolean>();\r\n\r\n    protected get controls(): CuiInstructionFormGroup {\r\n        return this.form.controls;\r\n    }\r\n\r\n    protected get isTitleError(): boolean {\r\n        return this.controls.title.invalid && this.controls.title.dirty;\r\n    }\r\n\r\n    protected get isDescriptionError(): boolean {\r\n        return this.controls.description.invalid && this.controls.description.dirty;\r\n    }\r\n\r\n    constructor() {\r\n        this.initUtilityEffect();\r\n    }\r\n\r\n    protected onEditorEmpty(isEmpty: boolean): void {\r\n        this.isDescriptionEmpty.set(isEmpty);\r\n        this.isFormValidChanged.emit(this.checkFormValidation());\r\n    }\r\n\r\n    private initUtilityEffect(): void {\r\n        const effectRef = effect(() => {\r\n            const title = this.title();\r\n\r\n            if (!title) {\r\n                return;\r\n            }\r\n\r\n            this.initFormSubscriptions();\r\n            this.initFrameFragments();\r\n            effectRef.destroy();\r\n        });\r\n    }\r\n\r\n    private initFormSubscriptions(): void {\r\n        this.form.setValue({ title: this.title(), description: this.description() }, { emitEvent: false });\r\n\r\n        this.form.statusChanges\r\n            .pipe(takeUntilDestroyed(this.destroy))\r\n            .subscribe(() => this.isFormValidChanged.emit(this.checkFormValidation()));\r\n\r\n        this.form.valueChanges\r\n            .pipe(\r\n                takeUntilDestroyed(this.destroy),\r\n                startWith(this.form.value),\r\n                pairwise(),\r\n                debounceTime(this.CONTROL_DELAY),\r\n                map(([prev, current]) => this.compareFormValues([prev, current]))\r\n            )\r\n            .subscribe((value) => {\r\n                if (!Object.keys(value).length) {\r\n                    return;\r\n                }\r\n\r\n                this.formChanged.emit(value);\r\n            });\r\n    }\r\n\r\n    private initFrameFragments(): void {\r\n        const frameId = this.framerId();\r\n\r\n        if (!frameId) {\r\n            return;\r\n        }\r\n\r\n        this.frameApiService.getFrame(frameId).subscribe((frame) => {\r\n            frame.fragments.forEach((fragment) => {\r\n                if (fragment.type === FragmentType.jText) {\r\n                    this.editorId.set(fragment?.id);\r\n                }\r\n\r\n                if (fragment.type === FragmentType.resourceFile) {\r\n                    this.setResourceFile(fragment as ResourceFragment);\r\n                }\r\n            });\r\n        });\r\n    }\r\n\r\n    private compareFormValues([\r\n        prevGroup,\r\n        currentGruop\r\n    ]: Partial<CuiInstructionFormValues>[]): Partial<CuiInstructionFormValues> {\r\n        const keys = Object.keys(currentGruop) as (keyof CuiInstructionFormValues)[];\r\n        const changedFields: Record<string, unknown> = {};\r\n\r\n        keys.forEach((key) => {\r\n            const prev = prevGroup[key];\r\n            const current = currentGruop[key];\r\n\r\n            if (prev === current || this.controls[key].invalid) {\r\n                return;\r\n            }\r\n\r\n            changedFields[key] = current;\r\n        });\r\n\r\n        return changedFields;\r\n    }\r\n\r\n    private setResourceFile(resource: ResourceFragment): void {\r\n        const url = this.baseResourceUrl() + resource.fileId!;\r\n\r\n        this.thumbnail.set(resource);\r\n\r\n        if (!url || resource?.name?.split('.').pop() !== 'mp4') {\r\n            return;\r\n        }\r\n\r\n        this.getCaptureFromVideo(url);\r\n    }\r\n\r\n    private getCaptureFromVideo(url: string, file?: File): void {\r\n        takeCaptureFromVideo(url).then((capture) => {\r\n            this.thumbnail.update((prev) => ({\r\n                ...prev!,\r\n                name: (file?.name ?? prev?.name)!,\r\n                resource: capture\r\n            }));\r\n        });\r\n    }\r\n\r\n    private checkFormValidation(): boolean {\r\n        return !this.isDescriptionEmpty() && !this.form.invalid;\r\n    }\r\n}\r\n","<ng-container *transloco=\"let t\">\r\n    <section class=\"content\">\r\n        <cui-form-field>\r\n            <label\r\n                cuiLabel\r\n                [isRequired]=\"true\"\r\n                [for]=\"titleId\"\r\n            >\r\n                {{ t('TITLE') }}\r\n            </label>\r\n\r\n            <cui-input-text\r\n                [cuiTextFieldId]=\"titleId\"\r\n                [cuiTextFieldPlaceholder]=\"t('ADD_A_TITLE')\"\r\n                [cuiTextFieldIsError]=\"isTitleError\"\r\n                [formControl]=\"form.controls.title\"\r\n            />\r\n\r\n            @if (isTitleError) {\r\n                <cui-general-control-error-hint\r\n                    [control]=\"form.controls.title\"\r\n                    [isVisible]=\"true\"\r\n                />\r\n            }\r\n        </cui-form-field>\r\n\r\n        <cui-form-field>\r\n            <label\r\n                cuiLabel\r\n                [isRequired]=\"true\"\r\n                [for]=\"descriptionId\"\r\n            >\r\n                {{ t('SHORT_DESCRIPTION') }}\r\n            </label>\r\n\r\n            <cui-input-text\r\n                [cuiTextFieldId]=\"descriptionId\"\r\n                cuiTextFieldPlaceholder=\"{{ t('SHORT_DESCRIPTION') }}...\"\r\n                [cuiTextFieldIsError]=\"isDescriptionError\"\r\n                [formControl]=\"form.controls.description\"\r\n            />\r\n\r\n            @if (isDescriptionError) {\r\n                <cui-general-control-error-hint\r\n                    [control]=\"form.controls.description\"\r\n                    [isVisible]=\"true\"\r\n                />\r\n            }\r\n        </cui-form-field>\r\n\r\n        <cui-utility-thumbnail\r\n            [baseResourceUrl]=\"baseResourceUrl()\"\r\n            [framerId]=\"framerId()\"\r\n            [(thumbnail)]=\"thumbnail\"\r\n        />\r\n\r\n        @if (editorId(); as editorId) {\r\n            <cui-editor-block\r\n                [title]=\"form.controls.title.value || title()\"\r\n                [config]=\"config()\"\r\n                [editorId]=\"editorId\"\r\n                [required]=\"true\"\r\n                class=\"editor\"\r\n                (editorEmpty)=\"onEditorEmpty($event)\"\r\n            />\r\n        }\r\n    </section>\r\n</ng-container>\r\n"]}
@@ -0,0 +1,2 @@
1
+ export {};
2
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5zdHJ1Y3Rpb24taW5mby1mb3JtLm9wdGlvbnMuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi9wcm9qZWN0cy9jb3JlL3dpZGdldHMvaW5zdHJ1Y3Rpb24taW5mby1mb3JtL2luc3RydWN0aW9uLWluZm8tZm9ybS5vcHRpb25zLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiIiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgdHlwZSB7IEZvcm1Db250cm9sIH0gZnJvbSAnQGFuZ3VsYXIvZm9ybXMnO1xuXG5leHBvcnQgaW50ZXJmYWNlIEN1aUluc3RydWN0aW9uRm9ybUdyb3VwIHtcbiAgICByZWFkb25seSB0aXRsZTogRm9ybUNvbnRyb2w8c3RyaW5nPjtcbiAgICByZWFkb25seSBkZXNjcmlwdGlvbjogRm9ybUNvbnRyb2w8c3RyaW5nPjtcbn1cblxuZXhwb3J0IGludGVyZmFjZSBDdWlJbnN0cnVjdGlvbkZvcm1WYWx1ZXMge1xuICAgIHJlYWRvbmx5IHRpdGxlOiBzdHJpbmc7XG4gICAgcmVhZG9ubHkgZGVzY3JpcHRpb246IHN0cmluZztcbn1cbiJdfQ==
@@ -0,0 +1,2 @@
1
+ export { CuiInstructionReadonlyInfoComponent } from './instruction-readonly-info.component';
2
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi9wcm9qZWN0cy9jb3JlL3dpZGdldHMvaW5zdHJ1Y3Rpb24tcmVhZG9ubHktaW5mby9pbmRleC50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxPQUFPLEVBQUUsbUNBQW1DLEVBQUUsTUFBTSx1Q0FBdUMsQ0FBQyIsInNvdXJjZXNDb250ZW50IjpbImV4cG9ydCB7IEN1aUluc3RydWN0aW9uUmVhZG9ubHlJbmZvQ29tcG9uZW50IH0gZnJvbSAnLi9pbnN0cnVjdGlvbi1yZWFkb25seS1pbmZvLmNvbXBvbmVudCc7XG4iXX0=
@@ -0,0 +1,72 @@
1
+ import { ChangeDetectionStrategy, Component, computed, effect, ElementRef, inject, input, signal } from '@angular/core';
2
+ import { TranslocoDirective } from '@jsverse/transloco';
3
+ import { CuiFrameApiService } from '@cuby-ui/api';
4
+ import { CuiEditorComponent } from '../../editor';
5
+ import { CuiUtilityReadonlyThumbnailComponent } from '../utility-readonly-thumbnail';
6
+ import { CuiEmptyStateComponent } from '../../components';
7
+ import * as i0 from "@angular/core";
8
+ export class CuiInstructionReadonlyInfoComponent {
9
+ constructor() {
10
+ this.hostElement = inject(ElementRef).nativeElement;
11
+ this.frameApiService = inject(CuiFrameApiService);
12
+ this.isVideo = false;
13
+ this.jText = signal(null);
14
+ this.thumbnailUrl = signal(null);
15
+ this.isDescriptionEmpty = computed(() => !this.jText()?.notEmpty);
16
+ this.framerId = input.required();
17
+ this.element = input(this.hostElement);
18
+ this.tools = input.required();
19
+ this.editorResourceUrl = input.required();
20
+ this.initConfigEffect();
21
+ this.initFramerIdEffect();
22
+ }
23
+ initConfigEffect() {
24
+ effect(() => {
25
+ this.editorConfig = { readOnly: true, tools: this.tools(), element: this.element() };
26
+ });
27
+ }
28
+ initFramerIdEffect() {
29
+ effect(() => {
30
+ const framerId = this.framerId();
31
+ if (!framerId) {
32
+ return;
33
+ }
34
+ this.subscription?.unsubscribe();
35
+ this.initFrameFragments();
36
+ });
37
+ }
38
+ initFrameFragments() {
39
+ const frameId = this.framerId();
40
+ this.jText.set(null);
41
+ this.thumbnailUrl.set(null);
42
+ if (!frameId) {
43
+ return;
44
+ }
45
+ const fragmentActions = {
46
+ ["J_TEXT" /* FragmentType.jText */]: (fragment) => {
47
+ this.jText.set(fragment);
48
+ },
49
+ ["RESOURCE_FILE" /* FragmentType.resourceFile */]: (fragment) => {
50
+ this.setResourceFile(fragment);
51
+ }
52
+ };
53
+ this.subscription = this.frameApiService.getFrame(frameId).subscribe((frame) => {
54
+ frame.fragments.forEach((fragment) => fragmentActions[fragment.type](fragment));
55
+ });
56
+ }
57
+ setResourceFile(resource) {
58
+ const url = `${this.editorResourceUrl()}${resource.fileId}`;
59
+ this.isVideo = resource?.name?.split('.').pop() === 'mp4';
60
+ if (!url) {
61
+ return;
62
+ }
63
+ this.thumbnailUrl.set({ src: resource.fileId && url, name: resource.name || resource.type });
64
+ }
65
+ static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: CuiInstructionReadonlyInfoComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
66
+ static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "18.2.13", type: CuiInstructionReadonlyInfoComponent, isStandalone: true, selector: "cui-instruction-readonly-info", inputs: { framerId: { classPropertyName: "framerId", publicName: "framerId", isSignal: true, isRequired: true, transformFunction: null }, element: { classPropertyName: "element", publicName: "element", isSignal: true, isRequired: false, transformFunction: null }, tools: { classPropertyName: "tools", publicName: "tools", isSignal: true, isRequired: true, transformFunction: null }, editorResourceUrl: { classPropertyName: "editorResourceUrl", publicName: "editorResourceUrl", isSignal: true, isRequired: true, transformFunction: null } }, ngImport: i0, template: "<ng-container *transloco=\"let t\">\n @if (jText()) {\n @if (!isDescriptionEmpty()) {\n @if (thumbnailUrl(); as thumbnailUrl) {\n <cui-utility-readonly-thumbnail [thumbnailUrl]=\"thumbnailUrl\" [isVideo]=\"isVideo\" />\n }\n\n @if (jText()?.id; as id) {\n <div class=\"wrapper\">\n <cui-editor\n class=\"editor\"\n [editorApiId]=\"id\"\n [config]=\"editorConfig\"\n />\n </div>\n }\n } @else {\n <cui-empty-state [subtitle]=\"t('NO_DESCRIPTION_WAS_PROVIDED')\" />\n }\n }\n</ng-container>\n", styles: [":host{padding-top:16px;padding-right:16px;padding-left:16px;display:flex;flex-direction:column;gap:12px;height:100%;overflow:auto}.wrapper{padding:12px}.editor{display:block;height:100%;overflow:auto}\n"], dependencies: [{ kind: "component", type: CuiEditorComponent, selector: "cui-editor", inputs: ["editorApiId", "config", "requestSize"], outputs: ["editorEmpty"] }, { kind: "component", type: CuiEmptyStateComponent, selector: "cui-empty-state, [cuiEmptyState]", inputs: ["title", "subtitle"] }, { kind: "directive", type: TranslocoDirective, selector: "[transloco]", inputs: ["transloco", "translocoParams", "translocoScope", "translocoRead", "translocoPrefix", "translocoLang", "translocoLoadingTpl"] }, { kind: "component", type: CuiUtilityReadonlyThumbnailComponent, selector: "cui-utility-readonly-thumbnail", inputs: ["thumbnailUrl", "isVideo"] }], changeDetection: i0.ChangeDetectionStrategy.OnPush }); }
67
+ }
68
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: CuiInstructionReadonlyInfoComponent, decorators: [{
69
+ type: Component,
70
+ args: [{ selector: 'cui-instruction-readonly-info', imports: [CuiEditorComponent, CuiEmptyStateComponent, TranslocoDirective, CuiUtilityReadonlyThumbnailComponent], standalone: true, changeDetection: ChangeDetectionStrategy.OnPush, template: "<ng-container *transloco=\"let t\">\n @if (jText()) {\n @if (!isDescriptionEmpty()) {\n @if (thumbnailUrl(); as thumbnailUrl) {\n <cui-utility-readonly-thumbnail [thumbnailUrl]=\"thumbnailUrl\" [isVideo]=\"isVideo\" />\n }\n\n @if (jText()?.id; as id) {\n <div class=\"wrapper\">\n <cui-editor\n class=\"editor\"\n [editorApiId]=\"id\"\n [config]=\"editorConfig\"\n />\n </div>\n }\n } @else {\n <cui-empty-state [subtitle]=\"t('NO_DESCRIPTION_WAS_PROVIDED')\" />\n }\n }\n</ng-container>\n", styles: [":host{padding-top:16px;padding-right:16px;padding-left:16px;display:flex;flex-direction:column;gap:12px;height:100%;overflow:auto}.wrapper{padding:12px}.editor{display:block;height:100%;overflow:auto}\n"] }]
71
+ }], ctorParameters: () => [] });
72
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"instruction-readonly-info.component.js","sourceRoot":"","sources":["../../../../../projects/core/widgets/instruction-readonly-info/instruction-readonly-info.component.ts","../../../../../projects/core/widgets/instruction-readonly-info/instruction-readonly-info.component.html"],"names":[],"mappings":"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","sourcesContent":["import { ChangeDetectionStrategy, Component, computed, effect, ElementRef, inject, input, signal } from '@angular/core';\nimport { TranslocoDirective } from '@jsverse/transloco';\nimport type { Subscription } from 'rxjs';\nimport type { Fragment, JTextFragment, ResourceFragment } from '@cuby-ui/api';\nimport { CuiFrameApiService, FragmentType } from '@cuby-ui/api';\nimport type { CuiNullable } from '@cuby-ui/cdk';\n\nimport type { CuiImgUrl } from '../../interfaces';\nimport { CuiEditorComponent } from '../../editor';\nimport type { CuiTools, CuiEditorConfig } from '../../editor';\nimport { CuiUtilityReadonlyThumbnailComponent } from '../utility-readonly-thumbnail';\nimport { CuiEmptyStateComponent  } from '../../components';\n\n@Component({\n    selector: 'cui-instruction-readonly-info',\n    imports: [CuiEditorComponent, CuiEmptyStateComponent, TranslocoDirective, CuiUtilityReadonlyThumbnailComponent],\n    standalone: true,\n    templateUrl: './instruction-readonly-info.component.html',\n    styleUrl: './instruction-readonly-info.component.scss',\n    changeDetection: ChangeDetectionStrategy.OnPush\n})\nexport class CuiInstructionReadonlyInfoComponent {\n    private readonly hostElement = inject(ElementRef).nativeElement;\n    private readonly frameApiService = inject(CuiFrameApiService);\n\n    private subscription!: Subscription;\n\n    protected isVideo = false;\n    protected editorConfig!: CuiEditorConfig;\n\n    protected readonly jText = signal<CuiNullable<JTextFragment>>(null);\n    protected readonly thumbnailUrl = signal<CuiNullable<CuiImgUrl>>(null);\n    protected readonly isDescriptionEmpty = computed(() => !this.jText()?.notEmpty);\n\n    public readonly framerId = input.required<string>();\n    public readonly element = input<HTMLElement>(this.hostElement);\n    public readonly tools = input.required<CuiTools>();\n    public readonly editorResourceUrl = input.required<string>();\n\n    constructor() {\n        this.initConfigEffect();\n        this.initFramerIdEffect();\n    }\n\n    private initConfigEffect(): void {\n        effect(() => {\n            this.editorConfig = { readOnly: true, tools: this.tools(), element: this.element() };\n        });\n    }\n\n    private initFramerIdEffect(): void {\n        effect(() => {\n            const framerId = this.framerId();\n\n            if (!framerId) {\n                return;\n            }\n\n            this.subscription?.unsubscribe();\n            this.initFrameFragments();\n        });\n    }\n\n    private initFrameFragments(): void {\n        const frameId = this.framerId();\n\n        this.jText.set(null);\n        this.thumbnailUrl.set(null);\n\n        if (!frameId) {\n            return;\n        }\n\n        const fragmentActions = {\n            [FragmentType.jText]: (fragment: Fragment): void => {\n                this.jText.set(fragment as JTextFragment);\n            },\n            [FragmentType.resourceFile]: (fragment: Fragment): void => {\n                this.setResourceFile(fragment as ResourceFragment);\n            }\n        };\n\n        this.subscription = this.frameApiService.getFrame(frameId).subscribe((frame) => {\n            frame.fragments.forEach((fragment) => fragmentActions[fragment.type](fragment));\n        });\n    }\n\n    private setResourceFile(resource: ResourceFragment): void {\n        const url = `${this.editorResourceUrl()}${resource.fileId}`;\n\n        this.isVideo = resource?.name?.split('.').pop() === 'mp4';\n\n        if (!url) {\n            return;\n        }\n\n        this.thumbnailUrl.set({ src: resource.fileId && url, name: resource.name || resource.type });\n    }\n}\n","<ng-container *transloco=\"let t\">\n    @if (jText()) {\n        @if (!isDescriptionEmpty()) {\n            @if (thumbnailUrl(); as thumbnailUrl) {\n                <cui-utility-readonly-thumbnail [thumbnailUrl]=\"thumbnailUrl\" [isVideo]=\"isVideo\" />\n            }\n\n            @if (jText()?.id; as id) {\n                <div class=\"wrapper\">\n                    <cui-editor\n                        class=\"editor\"\n                        [editorApiId]=\"id\"\n                        [config]=\"editorConfig\"\n                    />\n                </div>\n            }\n        } @else {\n            <cui-empty-state [subtitle]=\"t('NO_DESCRIPTION_WAS_PROVIDED')\" />\n        }\n    }\n</ng-container>\n"]}
@@ -0,0 +1,40 @@
1
+ import { ChangeDetectionStrategy, Component, computed, DestroyRef, inject, input, signal } from '@angular/core';
2
+ import { takeUntilDestroyed } from '@angular/core/rxjs-interop';
3
+ import { CUI_DIALOG_CONTEXT, CuiButtonModule, CuiSvgModule } from '../../../../components';
4
+ import { UTILITY_MODAL_SERVICE_TOKEN } from '../../utility-modal.options';
5
+ import { CuiCriterionInfoComponent } from "../../../criterion-info";
6
+ import { CuiUtilityModalCreateComponent } from "../utility-modal-create";
7
+ import * as i0 from "@angular/core";
8
+ export class CuiCriterionModalCreateComponent {
9
+ constructor() {
10
+ this.destroyRef = inject(DestroyRef);
11
+ this.cuiDialogContext = inject(CUI_DIALOG_CONTEXT);
12
+ this.criterionApiService = inject(UTILITY_MODAL_SERVICE_TOKEN);
13
+ this.isCreated = false;
14
+ this.editorResourceUrl = this.cuiDialogContext.data.editorResourceUrl;
15
+ this.tools = this.cuiDialogContext.data.tools;
16
+ this.header = this.cuiDialogContext.data.header;
17
+ this._criterion = computed(() => signal(this.initialUtility()));
18
+ this.isFormValid = signal(false);
19
+ this.criterion = computed(() => this._criterion()());
20
+ this.initialUtility = input.required();
21
+ }
22
+ onCriterionCreated(storageNodeId) {
23
+ if (this.isCreated || this.initialUtility()) {
24
+ return;
25
+ }
26
+ this.criterionApiService.createUtility(storageNodeId)
27
+ .pipe(takeUntilDestroyed(this.destroyRef))
28
+ .subscribe((criterion) => {
29
+ this.isCreated = true;
30
+ this._criterion().set(criterion);
31
+ });
32
+ }
33
+ static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: CuiCriterionModalCreateComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
34
+ static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.1.0", version: "18.2.13", type: CuiCriterionModalCreateComponent, isStandalone: true, selector: "ng-component", inputs: { initialUtility: { classPropertyName: "initialUtility", publicName: "initialUtility", isSignal: true, isRequired: true, transformFunction: null } }, ngImport: i0, template: "<cui-utility-modal-create [initialUtility]=\"criterion()\">\n <cui-criterion-info\n [criterion]=\"criterion()\"\n [tools]=\"tools\"\n [editorResourceUrl]=\"editorResourceUrl\"\n (validationChanged)=\"isFormValid.set($event)\"\n (criterionCreated)=\"onCriterionCreated($event)\"\n />\n</cui-utility-modal-create>\n", dependencies: [{ kind: "ngmodule", type: CuiButtonModule }, { kind: "ngmodule", type: CuiSvgModule }, { kind: "component", type: CuiCriterionInfoComponent, selector: "cui-criterion-info", inputs: ["criterion", "tools", "editorResourceUrl"], outputs: ["criterionChange", "validationChanged", "criterionCreated"] }, { kind: "component", type: CuiUtilityModalCreateComponent, selector: "cui-utility-modal-create", inputs: ["initialUtility"] }], changeDetection: i0.ChangeDetectionStrategy.OnPush }); }
35
+ }
36
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: CuiCriterionModalCreateComponent, decorators: [{
37
+ type: Component,
38
+ args: [{ imports: [CuiButtonModule, CuiSvgModule, CuiCriterionInfoComponent, CuiUtilityModalCreateComponent], standalone: true, changeDetection: ChangeDetectionStrategy.OnPush, template: "<cui-utility-modal-create [initialUtility]=\"criterion()\">\n <cui-criterion-info\n [criterion]=\"criterion()\"\n [tools]=\"tools\"\n [editorResourceUrl]=\"editorResourceUrl\"\n (validationChanged)=\"isFormValid.set($event)\"\n (criterionCreated)=\"onCriterionCreated($event)\"\n />\n</cui-utility-modal-create>\n" }]
39
+ }] });
40
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiY3JpdGVyaW9uLW1vZGFsLWNyZWF0ZS5jb21wb25lbnQuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi8uLi8uLi9wcm9qZWN0cy9jb3JlL3dpZGdldHMvdXRpbGl0eS1tb2RhbC9jb21wb25lbnRzL2NyaXRlcmlvbi1tb2RhbC1jcmVhdGUvY3JpdGVyaW9uLW1vZGFsLWNyZWF0ZS5jb21wb25lbnQudHMiLCIuLi8uLi8uLi8uLi8uLi8uLi8uLi9wcm9qZWN0cy9jb3JlL3dpZGdldHMvdXRpbGl0eS1tb2RhbC9jb21wb25lbnRzL2NyaXRlcmlvbi1tb2RhbC1jcmVhdGUvY3JpdGVyaW9uLW1vZGFsLWNyZWF0ZS5jb21wb25lbnQuaHRtbCJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxPQUFPLEVBQUUsdUJBQXVCLEVBQUUsU0FBUyxFQUFFLFFBQVEsRUFBRSxVQUFVLEVBQUUsTUFBTSxFQUFFLEtBQUssRUFBRSxNQUFNLEVBQUUsTUFBTSxlQUFlLENBQUM7QUFDaEgsT0FBTyxFQUFFLGtCQUFrQixFQUFFLE1BQU0sNEJBQTRCLENBQUM7QUFHaEUsT0FBTyxFQUFFLGtCQUFrQixFQUFFLGVBQWUsRUFBRSxZQUFZLEVBQUUsTUFBTSx3QkFBd0IsQ0FBQztBQUMzRixPQUFPLEVBQUUsMkJBQTJCLEVBQUUsTUFBTSw2QkFBNkIsQ0FBQztBQUMxRSxPQUFPLEVBQUUseUJBQXlCLEVBQUUsTUFBTSx5QkFBeUIsQ0FBQztBQUNwRSxPQUFPLEVBQUUsOEJBQThCLEVBQUUsTUFBTSx5QkFBeUIsQ0FBQzs7QUFRekUsTUFBTSxPQUFPLGdDQUFnQztJQU43QztRQU9xQixlQUFVLEdBQUcsTUFBTSxDQUFDLFVBQVUsQ0FBQyxDQUFDO1FBQ2hDLHFCQUFnQixHQUFHLE1BQU0sQ0FBQyxrQkFBa0IsQ0FBQyxDQUFDO1FBQzlDLHdCQUFtQixHQUFHLE1BQU0sQ0FBQywyQkFBMkIsQ0FBQyxDQUFDO1FBRW5FLGNBQVMsR0FBRyxLQUFLLENBQUM7UUFFUCxzQkFBaUIsR0FBRyxJQUFJLENBQUMsZ0JBQWdCLENBQUMsSUFBSSxDQUFDLGlCQUFpQixDQUFDO1FBQ2pFLFVBQUssR0FBRyxJQUFJLENBQUMsZ0JBQWdCLENBQUMsSUFBSSxDQUFDLEtBQUssQ0FBQztRQUN6QyxXQUFNLEdBQUcsSUFBSSxDQUFDLGdCQUFnQixDQUFDLElBQUksQ0FBQyxNQUFNLENBQUM7UUFFN0MsZUFBVSxHQUFHLFFBQVEsQ0FBQyxHQUFHLEVBQUUsQ0FBQyxNQUFNLENBQUMsSUFBSSxDQUFDLGNBQWMsRUFBRSxDQUFDLENBQUMsQ0FBQztRQUV6RCxnQkFBVyxHQUFHLE1BQU0sQ0FBQyxLQUFLLENBQUMsQ0FBQztRQUM1QixjQUFTLEdBQUcsUUFBUSxDQUFDLEdBQUcsRUFBRSxDQUFDLElBQUksQ0FBQyxVQUFVLEVBQUUsRUFBRSxDQUFDLENBQUM7UUFFbkQsbUJBQWMsR0FBRyxLQUFLLENBQUMsUUFBUSxFQUE0QixDQUFDO0tBYy9FO0lBWmEsa0JBQWtCLENBQUMsYUFBcUI7UUFDOUMsSUFBSSxJQUFJLENBQUMsU0FBUyxJQUFJLElBQUksQ0FBQyxjQUFjLEVBQUUsRUFBRSxDQUFDO1lBQzFDLE9BQU87UUFDWCxDQUFDO1FBRUQsSUFBSSxDQUFDLG1CQUFtQixDQUFDLGFBQWEsQ0FBQyxhQUFhLENBQUM7YUFDaEQsSUFBSSxDQUFDLGtCQUFrQixDQUFDLElBQUksQ0FBQyxVQUFVLENBQUMsQ0FBQzthQUN6QyxTQUFTLENBQUMsQ0FBQyxTQUFTLEVBQUUsRUFBRTtZQUNyQixJQUFJLENBQUMsU0FBUyxHQUFHLElBQUksQ0FBQztZQUN0QixJQUFJLENBQUMsVUFBVSxFQUFFLENBQUMsR0FBRyxDQUFDLFNBQXFDLENBQUMsQ0FBQztRQUNqRSxDQUFDLENBQUMsQ0FBQztJQUNYLENBQUM7K0dBN0JRLGdDQUFnQzttR0FBaEMsZ0NBQWdDLHNPQ2Y3QyxvV0FTQSwyQ0RDYyxlQUFlLDhCQUFFLFlBQVksK0JBQUUseUJBQXlCLDJMQUFFLDhCQUE4Qjs7NEZBS3pGLGdDQUFnQztrQkFONUMsU0FBUzs4QkFDRyxDQUFDLGVBQWUsRUFBRSxZQUFZLEVBQUUseUJBQXlCLEVBQUUsOEJBQThCLENBQUMsY0FDdkYsSUFBSSxtQkFFQyx1QkFBdUIsQ0FBQyxNQUFNIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgQ2hhbmdlRGV0ZWN0aW9uU3RyYXRlZ3ksIENvbXBvbmVudCwgY29tcHV0ZWQsIERlc3Ryb3lSZWYsIGluamVjdCwgaW5wdXQsIHNpZ25hbCB9IGZyb20gJ0Bhbmd1bGFyL2NvcmUnO1xuaW1wb3J0IHsgdGFrZVVudGlsRGVzdHJveWVkIH0gZnJvbSAnQGFuZ3VsYXIvY29yZS9yeGpzLWludGVyb3AnO1xuaW1wb3J0IHsgQ3JpdGVyaW9uLCBTdG9yYWdlT2JqZWN0IH0gZnJvbSAnQGN1YnktdWkvYXBpJztcblxuaW1wb3J0IHsgQ1VJX0RJQUxPR19DT05URVhULCBDdWlCdXR0b25Nb2R1bGUsIEN1aVN2Z01vZHVsZSB9IGZyb20gJy4uLy4uLy4uLy4uL2NvbXBvbmVudHMnO1xuaW1wb3J0IHsgVVRJTElUWV9NT0RBTF9TRVJWSUNFX1RPS0VOIH0gZnJvbSAnLi4vLi4vdXRpbGl0eS1tb2RhbC5vcHRpb25zJztcbmltcG9ydCB7IEN1aUNyaXRlcmlvbkluZm9Db21wb25lbnQgfSBmcm9tIFwiLi4vLi4vLi4vY3JpdGVyaW9uLWluZm9cIjtcbmltcG9ydCB7IEN1aVV0aWxpdHlNb2RhbENyZWF0ZUNvbXBvbmVudCB9IGZyb20gXCIuLi91dGlsaXR5LW1vZGFsLWNyZWF0ZVwiO1xuXG5AQ29tcG9uZW50KHtcbiAgICBpbXBvcnRzOiBbQ3VpQnV0dG9uTW9kdWxlLCBDdWlTdmdNb2R1bGUsIEN1aUNyaXRlcmlvbkluZm9Db21wb25lbnQsIEN1aVV0aWxpdHlNb2RhbENyZWF0ZUNvbXBvbmVudF0sXG4gICAgc3RhbmRhbG9uZTogdHJ1ZSxcbiAgICB0ZW1wbGF0ZVVybDogJy4vY3JpdGVyaW9uLW1vZGFsLWNyZWF0ZS5jb21wb25lbnQuaHRtbCcsXG4gICAgY2hhbmdlRGV0ZWN0aW9uOiBDaGFuZ2VEZXRlY3Rpb25TdHJhdGVneS5PblB1c2hcbn0pXG5leHBvcnQgY2xhc3MgQ3VpQ3JpdGVyaW9uTW9kYWxDcmVhdGVDb21wb25lbnQge1xuICAgIHByaXZhdGUgcmVhZG9ubHkgZGVzdHJveVJlZiA9IGluamVjdChEZXN0cm95UmVmKTtcbiAgICBwcml2YXRlIHJlYWRvbmx5IGN1aURpYWxvZ0NvbnRleHQgPSBpbmplY3QoQ1VJX0RJQUxPR19DT05URVhUKTtcbiAgICBwcml2YXRlIHJlYWRvbmx5IGNyaXRlcmlvbkFwaVNlcnZpY2UgPSBpbmplY3QoVVRJTElUWV9NT0RBTF9TRVJWSUNFX1RPS0VOKTtcblxuICAgIHByaXZhdGUgaXNDcmVhdGVkID0gZmFsc2U7XG5cbiAgICBwcm90ZWN0ZWQgcmVhZG9ubHkgZWRpdG9yUmVzb3VyY2VVcmwgPSB0aGlzLmN1aURpYWxvZ0NvbnRleHQuZGF0YS5lZGl0b3JSZXNvdXJjZVVybDtcbiAgICBwcm90ZWN0ZWQgcmVhZG9ubHkgdG9vbHMgPSB0aGlzLmN1aURpYWxvZ0NvbnRleHQuZGF0YS50b29scztcbiAgICBwcm90ZWN0ZWQgcmVhZG9ubHkgaGVhZGVyID0gdGhpcy5jdWlEaWFsb2dDb250ZXh0LmRhdGEuaGVhZGVyO1xuXG4gICAgcHJpdmF0ZSByZWFkb25seSBfY3JpdGVyaW9uID0gY29tcHV0ZWQoKCkgPT4gc2lnbmFsKHRoaXMuaW5pdGlhbFV0aWxpdHkoKSkpO1xuXG4gICAgcHJvdGVjdGVkIHJlYWRvbmx5IGlzRm9ybVZhbGlkID0gc2lnbmFsKGZhbHNlKTtcbiAgICBwcm90ZWN0ZWQgcmVhZG9ubHkgY3JpdGVyaW9uID0gY29tcHV0ZWQoKCkgPT4gdGhpcy5fY3JpdGVyaW9uKCkoKSk7XG5cbiAgICBwdWJsaWMgcmVhZG9ubHkgaW5pdGlhbFV0aWxpdHkgPSBpbnB1dC5yZXF1aXJlZDxTdG9yYWdlT2JqZWN0PENyaXRlcmlvbj4+KCk7XG5cbiAgICBwcm90ZWN0ZWQgb25Dcml0ZXJpb25DcmVhdGVkKHN0b3JhZ2VOb2RlSWQ6IHN0cmluZyk6IHZvaWQge1xuICAgICAgICBpZiAodGhpcy5pc0NyZWF0ZWQgfHwgdGhpcy5pbml0aWFsVXRpbGl0eSgpKSB7XG4gICAgICAgICAgICByZXR1cm47XG4gICAgICAgIH1cblxuICAgICAgICB0aGlzLmNyaXRlcmlvbkFwaVNlcnZpY2UuY3JlYXRlVXRpbGl0eShzdG9yYWdlTm9kZUlkKVxuICAgICAgICAgICAgLnBpcGUodGFrZVVudGlsRGVzdHJveWVkKHRoaXMuZGVzdHJveVJlZikpXG4gICAgICAgICAgICAuc3Vic2NyaWJlKChjcml0ZXJpb24pID0+IHtcbiAgICAgICAgICAgICAgICB0aGlzLmlzQ3JlYXRlZCA9IHRydWU7XG4gICAgICAgICAgICAgICAgdGhpcy5fY3JpdGVyaW9uKCkuc2V0KGNyaXRlcmlvbiBhcyBTdG9yYWdlT2JqZWN0PENyaXRlcmlvbj4pO1xuICAgICAgICAgICAgfSk7XG4gICAgfVxufVxuIiwiPGN1aS11dGlsaXR5LW1vZGFsLWNyZWF0ZSBbaW5pdGlhbFV0aWxpdHldPVwiY3JpdGVyaW9uKClcIj5cbiAgICA8Y3VpLWNyaXRlcmlvbi1pbmZvXG4gICAgICAgIFtjcml0ZXJpb25dPVwiY3JpdGVyaW9uKClcIlxuICAgICAgICBbdG9vbHNdPVwidG9vbHNcIlxuICAgICAgICBbZWRpdG9yUmVzb3VyY2VVcmxdPVwiZWRpdG9yUmVzb3VyY2VVcmxcIlxuICAgICAgICAodmFsaWRhdGlvbkNoYW5nZWQpPVwiaXNGb3JtVmFsaWQuc2V0KCRldmVudClcIlxuICAgICAgICAoY3JpdGVyaW9uQ3JlYXRlZCk9XCJvbkNyaXRlcmlvbkNyZWF0ZWQoJGV2ZW50KVwiXG4gICAgLz5cbjwvY3VpLXV0aWxpdHktbW9kYWwtY3JlYXRlPlxuIl19