@sd-angular/core 19.0.0-beta.92 → 19.0.0-beta.94
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/components/document-builder/src/components/header-footer-builder/header-footer-builder.component.d.ts +4 -1
- package/components/document-builder/src/document-builder.model.d.ts +2 -0
- package/components/editor/src/models/editor.model.d.ts +2 -0
- package/components/editor/src/plugins/image-upload/utils/validate.utils.d.ts +2 -1
- package/components/form-generic/index.d.ts +4 -0
- package/components/{workflow → form-generic}/src/components/form-builder/components/attribute-table/attribute-table.component.d.ts +3 -3
- package/components/{workflow → form-generic}/src/components/form-builder/components/attribute-template/attribute-template.component.d.ts +3 -3
- package/components/{workflow → form-generic}/src/components/form-builder/components/configure-validation/configure-validation.component.d.ts +3 -3
- package/components/{workflow → form-generic}/src/components/form-builder/components/expression-builder/expression-builder.component.d.ts +1 -1
- package/components/{workflow → form-generic}/src/components/form-render/form-render.component.d.ts +3 -3
- package/components/{workflow → form-generic}/src/components/sd-feel-expression/sd-feel-expression.component.d.ts +1 -1
- package/components/form-generic/src/configurations/form-generic.configuration.d.ts +6 -0
- package/components/form-generic/src/configurations/index.d.ts +2 -0
- package/components/{workflow → form-generic}/src/services/form-generic.service.d.ts +3 -3
- package/components/index.d.ts +1 -0
- package/components/splitter/index.d.ts +3 -0
- package/components/splitter/src/splitter-handle/splitter-handle.component.d.ts +24 -0
- package/components/splitter/src/splitter-panel/splitter-panel.component.d.ts +16 -0
- package/components/splitter/src/splitter-state.service.d.ts +26 -0
- package/components/splitter/src/splitter.component.d.ts +28 -0
- package/components/splitter/src/splitter.models.d.ts +23 -0
- package/components/table/src/components/selector-action/selector-action.component.d.ts +1 -0
- package/components/table/src/models/table-option-config.model.d.ts +1 -0
- package/components/table/src/services/column-width.util.d.ts +7 -0
- package/components/table/src/table.component.d.ts +1 -0
- package/components/workflow/index.d.ts +2 -4
- package/components/workflow/src/configurations/workflow.configuration.d.ts +9 -5
- package/configurations/src/sd-core.configuration.d.ts +1 -0
- package/fesm2022/sd-angular-core-components-anchor.mjs +4 -4
- package/fesm2022/sd-angular-core-components-anchor.mjs.map +1 -1
- package/fesm2022/sd-angular-core-components-code-editor.mjs +3 -2
- package/fesm2022/sd-angular-core-components-code-editor.mjs.map +1 -1
- package/fesm2022/sd-angular-core-components-document-builder.mjs +29 -8
- package/fesm2022/sd-angular-core-components-document-builder.mjs.map +1 -1
- package/fesm2022/sd-angular-core-components-editor.mjs +31 -16
- package/fesm2022/sd-angular-core-components-editor.mjs.map +1 -1
- package/fesm2022/sd-angular-core-components-form-generic.mjs +6404 -0
- package/fesm2022/sd-angular-core-components-form-generic.mjs.map +1 -0
- package/fesm2022/sd-angular-core-components-history.mjs +3 -2
- package/fesm2022/sd-angular-core-components-history.mjs.map +1 -1
- package/fesm2022/sd-angular-core-components-import-excel.mjs +25 -23
- package/fesm2022/sd-angular-core-components-import-excel.mjs.map +1 -1
- package/fesm2022/sd-angular-core-components-preview.mjs +6 -4
- package/fesm2022/sd-angular-core-components-preview.mjs.map +1 -1
- package/fesm2022/sd-angular-core-components-splitter.mjs +646 -0
- package/fesm2022/sd-angular-core-components-splitter.mjs.map +1 -0
- package/fesm2022/sd-angular-core-components-tab-router.mjs +7 -5
- package/fesm2022/sd-angular-core-components-tab-router.mjs.map +1 -1
- package/fesm2022/sd-angular-core-components-table.mjs +52 -23
- package/fesm2022/sd-angular-core-components-table.mjs.map +1 -1
- package/fesm2022/sd-angular-core-components-upload-file.mjs +24 -21
- package/fesm2022/sd-angular-core-components-upload-file.mjs.map +1 -1
- package/fesm2022/sd-angular-core-components-workflow.mjs +7 -6387
- package/fesm2022/sd-angular-core-components-workflow.mjs.map +1 -1
- package/fesm2022/sd-angular-core-components.mjs +1 -0
- package/fesm2022/sd-angular-core-components.mjs.map +1 -1
- package/fesm2022/sd-angular-core-configurations.mjs.map +1 -1
- package/fesm2022/sd-angular-core-directives.mjs +6 -2
- package/fesm2022/sd-angular-core-directives.mjs.map +1 -1
- package/fesm2022/sd-angular-core-forms-autocomplete.mjs +3 -1
- package/fesm2022/sd-angular-core-forms-autocomplete.mjs.map +1 -1
- package/fesm2022/sd-angular-core-forms-chip-calendar.mjs +5 -3
- package/fesm2022/sd-angular-core-forms-chip-calendar.mjs.map +1 -1
- package/fesm2022/sd-angular-core-forms-chip.mjs +5 -3
- package/fesm2022/sd-angular-core-forms-chip.mjs.map +1 -1
- package/fesm2022/sd-angular-core-forms-date-range.mjs +8 -5
- package/fesm2022/sd-angular-core-forms-date-range.mjs.map +1 -1
- package/fesm2022/sd-angular-core-forms-date.mjs +7 -5
- package/fesm2022/sd-angular-core-forms-date.mjs.map +1 -1
- package/fesm2022/sd-angular-core-forms-datetime.mjs +10 -8
- package/fesm2022/sd-angular-core-forms-datetime.mjs.map +1 -1
- package/fesm2022/sd-angular-core-forms-input-number.mjs +5 -3
- package/fesm2022/sd-angular-core-forms-input-number.mjs.map +1 -1
- package/fesm2022/sd-angular-core-forms-input.mjs +13 -6
- package/fesm2022/sd-angular-core-forms-input.mjs.map +1 -1
- package/fesm2022/sd-angular-core-forms-radio.mjs +3 -2
- package/fesm2022/sd-angular-core-forms-radio.mjs.map +1 -1
- package/fesm2022/sd-angular-core-forms-select.mjs +5 -3
- package/fesm2022/sd-angular-core-forms-select.mjs.map +1 -1
- package/fesm2022/sd-angular-core-forms-textarea.mjs +8 -5
- package/fesm2022/sd-angular-core-forms-textarea.mjs.map +1 -1
- package/fesm2022/sd-angular-core-handlers.mjs +7 -6
- package/fesm2022/sd-angular-core-handlers.mjs.map +1 -1
- package/fesm2022/sd-angular-core-i18n.mjs +790 -0
- package/fesm2022/sd-angular-core-i18n.mjs.map +1 -0
- package/fesm2022/sd-angular-core-interceptors.mjs +10 -6
- package/fesm2022/sd-angular-core-interceptors.mjs.map +1 -1
- package/fesm2022/sd-angular-core-modules-authom.mjs +1 -0
- package/fesm2022/sd-angular-core-modules-authom.mjs.map +1 -1
- package/fesm2022/sd-angular-core-modules-keycloak.mjs +1 -0
- package/fesm2022/sd-angular-core-modules-keycloak.mjs.map +1 -1
- package/fesm2022/sd-angular-core-modules-layout.mjs +47 -46
- package/fesm2022/sd-angular-core-modules-layout.mjs.map +1 -1
- package/fesm2022/sd-angular-core-services-confirm.mjs +15 -13
- package/fesm2022/sd-angular-core-services-confirm.mjs.map +1 -1
- package/fesm2022/sd-angular-core-services-docx.mjs +7 -7
- package/fesm2022/sd-angular-core-services-docx.mjs.map +1 -1
- package/fesm2022/sd-angular-core-services-excel.mjs +5 -3
- package/fesm2022/sd-angular-core-services-excel.mjs.map +1 -1
- package/fesm2022/sd-angular-core-services-storage.mjs.map +1 -1
- package/fesm2022/sd-angular-core-utilities-extensions.mjs +21 -10
- package/fesm2022/sd-angular-core-utilities-extensions.mjs.map +1 -1
- package/fesm2022/sd-angular-core-utilities-models.mjs +38 -24
- package/fesm2022/sd-angular-core-utilities-models.mjs.map +1 -1
- package/fesm2022/sd-angular-core.mjs +1 -0
- package/fesm2022/sd-angular-core.mjs.map +1 -1
- package/i18n/index.d.ts +5 -0
- package/i18n/src/en.d.ts +2 -0
- package/i18n/src/sd-i18n.messages.d.ts +2 -0
- package/i18n/src/sd-i18n.pipe.d.ts +9 -0
- package/i18n/src/sd-i18n.service.d.ts +12 -0
- package/i18n/src/sd-i18n.token.d.ts +1 -0
- package/i18n/src/sd-i18n.types.d.ts +5 -0
- package/i18n/src/vi.d.ts +312 -0
- package/package.json +52 -40
- package/public-api.d.ts +1 -0
- package/sd-angular-core-19.0.0-beta.94.tgz +0 -0
- package/services/confirm/src/lib/confirm.service.d.ts +1 -0
- package/utilities/extensions/src/string.extension.d.ts +2 -0
- package/utilities/models/src/pattern.model.d.ts +1 -1
- package/sd-angular-core-19.0.0-beta.92.tgz +0 -0
- /package/components/{workflow → form-generic}/src/components/form-builder/components/attribute-expression/attribute-expression.component.d.ts +0 -0
- /package/components/{workflow → form-generic}/src/components/form-builder/components/attribute-input/attribute-input.component.d.ts +0 -0
- /package/components/{workflow → form-generic}/src/components/form-builder/components/attribute-input-number/attribute-input-number.component.d.ts +0 -0
- /package/components/{workflow → form-generic}/src/components/form-builder/components/attribute-parameter/attribute-parameter.component.d.ts +0 -0
- /package/components/{workflow → form-generic}/src/components/form-builder/components/attribute-select/attribute-select.component.d.ts +0 -0
- /package/components/{workflow → form-generic}/src/components/form-builder/components/attribute-selection/attribute-selection.component.d.ts +0 -0
- /package/components/{workflow → form-generic}/src/components/form-builder/components/attribute-selection/components/build-queries/build-queries.component.d.ts +0 -0
- /package/components/{workflow → form-generic}/src/components/form-builder/components/attribute-selection/components/build-variables/build-variables.component.d.ts +0 -0
- /package/components/{workflow → form-generic}/src/components/form-builder/components/attribute-switch/attribute-switch.component.d.ts +0 -0
- /package/components/{workflow → form-generic}/src/components/form-builder/components/attribute-textarea/attribute-textarea.component.d.ts +0 -0
- /package/components/{workflow → form-generic}/src/components/form-builder/components/checkbox/attribute/checkbox-attribute.component.d.ts +0 -0
- /package/components/{workflow → form-generic}/src/components/form-builder/components/checkbox/control/checkbox-control.component.d.ts +0 -0
- /package/components/{workflow → form-generic}/src/components/form-builder/components/checkbox/index.d.ts +0 -0
- /package/components/{workflow → form-generic}/src/components/form-builder/components/chip-calendar/attribute/chip-calendar-attribute.component.d.ts +0 -0
- /package/components/{workflow → form-generic}/src/components/form-builder/components/chip-calendar/control/chip-calendar-control.component.d.ts +0 -0
- /package/components/{workflow → form-generic}/src/components/form-builder/components/chip-calendar/index.d.ts +0 -0
- /package/components/{workflow → form-generic}/src/components/form-builder/components/chip-string/attribute/chip-string-attribute.component.d.ts +0 -0
- /package/components/{workflow → form-generic}/src/components/form-builder/components/chip-string/control/chip-string-control.component.d.ts +0 -0
- /package/components/{workflow → form-generic}/src/components/form-builder/components/chip-string/index.d.ts +0 -0
- /package/components/{workflow → form-generic}/src/components/form-builder/components/datetime/attribute/datetime-attribute.component.d.ts +0 -0
- /package/components/{workflow → form-generic}/src/components/form-builder/components/datetime/control/datetime-control.component.d.ts +0 -0
- /package/components/{workflow → form-generic}/src/components/form-builder/components/datetime/index.d.ts +0 -0
- /package/components/{workflow → form-generic}/src/components/form-builder/components/html/attribute/components/build-queries/build-queries.component.d.ts +0 -0
- /package/components/{workflow → form-generic}/src/components/form-builder/components/html/attribute/html-attribute.component.d.ts +0 -0
- /package/components/{workflow → form-generic}/src/components/form-builder/components/html/control/html-control.component.d.ts +0 -0
- /package/components/{workflow → form-generic}/src/components/form-builder/components/html/index.d.ts +0 -0
- /package/components/{workflow → form-generic}/src/components/form-builder/components/index.d.ts +0 -0
- /package/components/{workflow → form-generic}/src/components/form-builder/components/number/attribute/number-attribute.component.d.ts +0 -0
- /package/components/{workflow → form-generic}/src/components/form-builder/components/number/control/number-control.component.d.ts +0 -0
- /package/components/{workflow → form-generic}/src/components/form-builder/components/number/index.d.ts +0 -0
- /package/components/{workflow → form-generic}/src/components/form-builder/components/radio/attribute/radio-attribute.component.d.ts +0 -0
- /package/components/{workflow → form-generic}/src/components/form-builder/components/radio/control/radio-control.component.d.ts +0 -0
- /package/components/{workflow → form-generic}/src/components/form-builder/components/radio/index.d.ts +0 -0
- /package/components/{workflow → form-generic}/src/components/form-builder/components/select/attribute/select-attribute.component.d.ts +0 -0
- /package/components/{workflow → form-generic}/src/components/form-builder/components/select/control/select-control.component.d.ts +0 -0
- /package/components/{workflow → form-generic}/src/components/form-builder/components/select/index.d.ts +0 -0
- /package/components/{workflow → form-generic}/src/components/form-builder/components/table/attribute/table-attribute.component.d.ts +0 -0
- /package/components/{workflow → form-generic}/src/components/form-builder/components/table/control/table-control.component.d.ts +0 -0
- /package/components/{workflow → form-generic}/src/components/form-builder/components/table/index.d.ts +0 -0
- /package/components/{workflow → form-generic}/src/components/form-builder/components/textarea/attribute/textarea-attribute.component.d.ts +0 -0
- /package/components/{workflow → form-generic}/src/components/form-builder/components/textarea/control/textarea-control.component.d.ts +0 -0
- /package/components/{workflow → form-generic}/src/components/form-builder/components/textarea/index.d.ts +0 -0
- /package/components/{workflow → form-generic}/src/components/form-builder/components/textfield/attribute/textfield-attribute.component.d.ts +0 -0
- /package/components/{workflow → form-generic}/src/components/form-builder/components/textfield/control/textfield-control.component.d.ts +0 -0
- /package/components/{workflow → form-generic}/src/components/form-builder/components/textfield/index.d.ts +0 -0
- /package/components/{workflow → form-generic}/src/components/form-builder/components/upload/attribute/upload-attribute.component.d.ts +0 -0
- /package/components/{workflow → form-generic}/src/components/form-builder/components/upload/control/upload-control.component.d.ts +0 -0
- /package/components/{workflow → form-generic}/src/components/form-builder/components/upload/index.d.ts +0 -0
- /package/components/{workflow → form-generic}/src/components/form-builder/form-builder.component.d.ts +0 -0
- /package/components/{workflow → form-generic}/src/components/form-builder/services/builder.service.d.ts +0 -0
- /package/components/{workflow → form-generic}/src/components/form-builder/services/index.d.ts +0 -0
- /package/components/{workflow → form-generic}/src/components/form-render/components/index.d.ts +0 -0
- /package/components/{workflow → form-generic}/src/components/form-render/components/item/components/chip-calendar/chip-calendar.component.d.ts +0 -0
- /package/components/{workflow → form-generic}/src/components/form-render/components/item/components/chip-string/chip-string.component.d.ts +0 -0
- /package/components/{workflow → form-generic}/src/components/form-render/components/item/components/datetime/datetime.component.d.ts +0 -0
- /package/components/{workflow → form-generic}/src/components/form-render/components/item/components/html/html.component.d.ts +0 -0
- /package/components/{workflow → form-generic}/src/components/form-render/components/item/components/index.d.ts +0 -0
- /package/components/{workflow → form-generic}/src/components/form-render/components/item/components/number/number.component.d.ts +0 -0
- /package/components/{workflow → form-generic}/src/components/form-render/components/item/components/radio/radio.component.d.ts +0 -0
- /package/components/{workflow → form-generic}/src/components/form-render/components/item/components/select/select.component.d.ts +0 -0
- /package/components/{workflow → form-generic}/src/components/form-render/components/item/components/table/table.component.d.ts +0 -0
- /package/components/{workflow → form-generic}/src/components/form-render/components/item/components/textarea/textarea.component.d.ts +0 -0
- /package/components/{workflow → form-generic}/src/components/form-render/components/item/components/textfield/textfield.component.d.ts +0 -0
- /package/components/{workflow → form-generic}/src/components/form-render/components/item/components/upload/upload.component.d.ts +0 -0
- /package/components/{workflow → form-generic}/src/components/form-render/components/item/item.component.d.ts +0 -0
- /package/components/{workflow → form-generic}/src/components/form-render/components/variable/variable.component.d.ts +0 -0
- /package/components/{workflow → form-generic}/src/components/index.d.ts +0 -0
- /package/components/{workflow → form-generic}/src/configurations/form.configuration.d.ts +0 -0
- /package/components/{workflow → form-generic}/src/models/form-generic-component.model.d.ts +0 -0
- /package/components/{workflow → form-generic}/src/models/form-generic-definition-html.model.d.ts +0 -0
- /package/components/{workflow → form-generic}/src/models/form-generic-definition-selection.model.d.ts +0 -0
- /package/components/{workflow → form-generic}/src/models/form-generic-definition-table.model.d.ts +0 -0
- /package/components/{workflow → form-generic}/src/models/form-generic-expression.model.d.ts +0 -0
- /package/components/{workflow → form-generic}/src/models/form-generic-template.model.d.ts +0 -0
- /package/components/{workflow → form-generic}/src/models/form-generic-validation.model.d.ts +0 -0
- /package/components/{workflow → form-generic}/src/models/form-generic.model.d.ts +0 -0
- /package/components/{workflow → form-generic}/src/models/form-render/form-render-args.model.d.ts +0 -0
- /package/components/{workflow → form-generic}/src/models/form-render/form-render-entity.model.d.ts +0 -0
- /package/components/{workflow → form-generic}/src/models/form-render/index.d.ts +0 -0
- /package/components/{workflow → form-generic}/src/models/index.d.ts +0 -0
- /package/components/{workflow → form-generic}/src/pipes/component-viewed.pipe.d.ts +0 -0
- /package/components/{workflow → form-generic}/src/pipes/expression-feel.pipe.d.ts +0 -0
- /package/components/{workflow → form-generic}/src/pipes/expression-query.pipe.d.ts +0 -0
- /package/components/{workflow → form-generic}/src/pipes/expression-view.pipe.d.ts +0 -0
- /package/components/{workflow → form-generic}/src/pipes/html.pipe.d.ts +0 -0
- /package/components/{workflow → form-generic}/src/pipes/hyperlink.pipe.d.ts +0 -0
- /package/components/{workflow → form-generic}/src/pipes/index.d.ts +0 -0
- /package/components/{workflow → form-generic}/src/pipes/when-expression.pipe.d.ts +0 -0
- /package/components/{workflow → form-generic}/src/services/form-render.service.d.ts +0 -0
- /package/components/{workflow → form-generic}/src/services/index.d.ts +0 -0
|
@@ -0,0 +1,646 @@
|
|
|
1
|
+
import * as i0 from '@angular/core';
|
|
2
|
+
import { inject, ElementRef, input, booleanAttribute, numberAttribute, output, HostListener, Component, model, signal, Injectable, EnvironmentInjector, Injector, DestroyRef, computed, contentChildren, effect, afterNextRender, createComponent } from '@angular/core';
|
|
3
|
+
import { SdStorageService } from '@sd-angular/core/services/storage';
|
|
4
|
+
|
|
5
|
+
class SdSplitterHandleComponent {
|
|
6
|
+
elementRef = inject(ElementRef);
|
|
7
|
+
orientation = input('horizontal');
|
|
8
|
+
disabled = input(false, { transform: booleanAttribute });
|
|
9
|
+
keyboardStep = input(10, { transform: numberAttribute });
|
|
10
|
+
ariaValueMin = input(undefined);
|
|
11
|
+
ariaValueMax = input(undefined);
|
|
12
|
+
ariaValueNow = input(undefined);
|
|
13
|
+
dragStart = output();
|
|
14
|
+
dragMove = output();
|
|
15
|
+
dragEnd = output();
|
|
16
|
+
toggleRequest = output();
|
|
17
|
+
#pointerId = null;
|
|
18
|
+
#startCoord = 0;
|
|
19
|
+
#rafPending = null;
|
|
20
|
+
#pendingDelta = 0;
|
|
21
|
+
onDblClick() {
|
|
22
|
+
if (this.disabled())
|
|
23
|
+
return;
|
|
24
|
+
this.toggleRequest.emit();
|
|
25
|
+
}
|
|
26
|
+
onKeyDown(ev) {
|
|
27
|
+
if (this.disabled())
|
|
28
|
+
return;
|
|
29
|
+
const isH = this.orientation() === 'horizontal';
|
|
30
|
+
const step = this.keyboardStep();
|
|
31
|
+
let delta = null;
|
|
32
|
+
switch (ev.key) {
|
|
33
|
+
case 'ArrowRight':
|
|
34
|
+
if (isH)
|
|
35
|
+
delta = step;
|
|
36
|
+
break;
|
|
37
|
+
case 'ArrowLeft':
|
|
38
|
+
if (isH)
|
|
39
|
+
delta = -step;
|
|
40
|
+
break;
|
|
41
|
+
case 'ArrowDown':
|
|
42
|
+
if (!isH)
|
|
43
|
+
delta = step;
|
|
44
|
+
break;
|
|
45
|
+
case 'ArrowUp':
|
|
46
|
+
if (!isH)
|
|
47
|
+
delta = -step;
|
|
48
|
+
break;
|
|
49
|
+
case 'Enter':
|
|
50
|
+
case ' ':
|
|
51
|
+
ev.preventDefault();
|
|
52
|
+
this.toggleRequest.emit();
|
|
53
|
+
return;
|
|
54
|
+
}
|
|
55
|
+
if (delta == null)
|
|
56
|
+
return;
|
|
57
|
+
ev.preventDefault();
|
|
58
|
+
// Keyboard step là 1 lần commit (không live drag) — emit start+move+end liền
|
|
59
|
+
this.dragStart.emit();
|
|
60
|
+
this.dragMove.emit(delta);
|
|
61
|
+
this.dragEnd.emit();
|
|
62
|
+
}
|
|
63
|
+
onPointerDown(ev) {
|
|
64
|
+
if (this.disabled())
|
|
65
|
+
return;
|
|
66
|
+
// Chỉ xử lý nút trái chuột cho pointerType=mouse; touch/pen không có button constraint
|
|
67
|
+
if (ev.button !== 0 && ev.pointerType === 'mouse')
|
|
68
|
+
return;
|
|
69
|
+
this.#pointerId = ev.pointerId;
|
|
70
|
+
this.#startCoord = this.orientation() === 'horizontal' ? ev.clientX : ev.clientY;
|
|
71
|
+
this.elementRef.nativeElement.setPointerCapture(ev.pointerId);
|
|
72
|
+
ev.preventDefault();
|
|
73
|
+
this.dragStart.emit();
|
|
74
|
+
}
|
|
75
|
+
onPointerMove(ev) {
|
|
76
|
+
// Bỏ qua nếu chưa bắt đầu drag hoặc sai pointer
|
|
77
|
+
if (this.#pointerId == null || ev.pointerId !== this.#pointerId)
|
|
78
|
+
return;
|
|
79
|
+
const coord = this.orientation() === 'horizontal' ? ev.clientX : ev.clientY;
|
|
80
|
+
this.#pendingDelta = coord - this.#startCoord;
|
|
81
|
+
// Batch qua rAF để tránh trigger Angular CD quá 60fps
|
|
82
|
+
if (this.#rafPending != null)
|
|
83
|
+
return;
|
|
84
|
+
this.#rafPending = requestAnimationFrame(() => {
|
|
85
|
+
this.#rafPending = null;
|
|
86
|
+
this.dragMove.emit(this.#pendingDelta);
|
|
87
|
+
});
|
|
88
|
+
}
|
|
89
|
+
onPointerUp(ev) {
|
|
90
|
+
if (this.#pointerId == null || ev.pointerId !== this.#pointerId)
|
|
91
|
+
return;
|
|
92
|
+
this.elementRef.nativeElement.releasePointerCapture(ev.pointerId);
|
|
93
|
+
this.#pointerId = null;
|
|
94
|
+
// Hủy rAF đang chờ để tránh emit dragMove sau khi drag kết thúc
|
|
95
|
+
if (this.#rafPending != null) {
|
|
96
|
+
cancelAnimationFrame(this.#rafPending);
|
|
97
|
+
this.#rafPending = null;
|
|
98
|
+
}
|
|
99
|
+
this.dragEnd.emit();
|
|
100
|
+
}
|
|
101
|
+
static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "19.2.21", ngImport: i0, type: SdSplitterHandleComponent, deps: [], target: i0.ɵɵFactoryTarget.Component });
|
|
102
|
+
static ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.1.0", version: "19.2.21", type: SdSplitterHandleComponent, isStandalone: true, selector: "sd-splitter-handle", inputs: { orientation: { classPropertyName: "orientation", publicName: "orientation", isSignal: true, isRequired: false, transformFunction: null }, disabled: { classPropertyName: "disabled", publicName: "disabled", isSignal: true, isRequired: false, transformFunction: null }, keyboardStep: { classPropertyName: "keyboardStep", publicName: "keyboardStep", isSignal: true, isRequired: false, transformFunction: null }, ariaValueMin: { classPropertyName: "ariaValueMin", publicName: "ariaValueMin", isSignal: true, isRequired: false, transformFunction: null }, ariaValueMax: { classPropertyName: "ariaValueMax", publicName: "ariaValueMax", isSignal: true, isRequired: false, transformFunction: null }, ariaValueNow: { classPropertyName: "ariaValueNow", publicName: "ariaValueNow", isSignal: true, isRequired: false, transformFunction: null } }, outputs: { dragStart: "dragStart", dragMove: "dragMove", dragEnd: "dragEnd", toggleRequest: "toggleRequest" }, host: { listeners: { "dblclick": "onDblClick()", "keydown": "onKeyDown($event)", "pointerdown": "onPointerDown($event)", "pointermove": "onPointerMove($event)", "pointerup": "onPointerUp($event)", "pointercancel": "onPointerUp($event)" }, properties: { "class.sd-splitter__handle--horizontal": "orientation() === \"horizontal\"", "class.sd-splitter__handle--vertical": "orientation() === \"vertical\"", "class.sd-splitter__handle--disabled": "disabled()", "attr.tabindex": "disabled() ? -1 : 0", "attr.role": "\"separator\"", "attr.aria-orientation": "orientation() === \"horizontal\" ? \"vertical\" : \"horizontal\"", "attr.aria-disabled": "disabled() ? \"true\" : null", "attr.aria-valuemin": "ariaValueMin() ?? null", "attr.aria-valuemax": "ariaValueMax() ?? null", "attr.aria-valuenow": "ariaValueNow() ?? null" }, classAttribute: "sd-splitter__handle" }, ngImport: i0, template: "<span class=\"sd-splitter__handle-bar\"></span>\n", styles: [":host{--sd-splitter-handle-size: 4px;--sd-splitter-handle-color: var(--sd-color-primary-light, #b0bec5);--sd-splitter-handle-hover-color: var(--sd-color-primary, #1976d2);--sd-splitter-handle-active-color: var(--sd-color-primary, #1976d2);--sd-splitter-handle-hit-area: 8px;--sd-splitter-handle-radius: 0;--sd-splitter-disabled-opacity: .5;display:flex;align-items:center;justify-content:center;flex:0 0 var(--sd-splitter-handle-hit-area);-webkit-user-select:none;user-select:none;outline:none}:host.sd-splitter__handle--horizontal{cursor:col-resize}:host.sd-splitter__handle--horizontal .sd-splitter__handle-bar{width:var(--sd-splitter-handle-size);height:100%}:host.sd-splitter__handle--vertical{cursor:row-resize;flex-direction:column}:host.sd-splitter__handle--vertical .sd-splitter__handle-bar{width:100%;height:var(--sd-splitter-handle-size)}:host .sd-splitter__handle-bar{background:var(--sd-splitter-handle-color);border-radius:var(--sd-splitter-handle-radius);transition:background-color .12s ease}:host:hover .sd-splitter__handle-bar{background:var(--sd-splitter-handle-hover-color)}:host:focus-visible{outline:2px solid var(--sd-splitter-handle-active-color);outline-offset:1px}:host.sd-splitter__handle--disabled{cursor:default;opacity:var(--sd-splitter-disabled-opacity);pointer-events:none}\n"] });
|
|
103
|
+
}
|
|
104
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "19.2.21", ngImport: i0, type: SdSplitterHandleComponent, decorators: [{
|
|
105
|
+
type: Component,
|
|
106
|
+
args: [{ selector: 'sd-splitter-handle', standalone: true, host: {
|
|
107
|
+
'class': 'sd-splitter__handle',
|
|
108
|
+
'[class.sd-splitter__handle--horizontal]': 'orientation() === "horizontal"',
|
|
109
|
+
'[class.sd-splitter__handle--vertical]': 'orientation() === "vertical"',
|
|
110
|
+
'[class.sd-splitter__handle--disabled]': 'disabled()',
|
|
111
|
+
'[attr.tabindex]': 'disabled() ? -1 : 0',
|
|
112
|
+
'[attr.role]': '"separator"',
|
|
113
|
+
'[attr.aria-orientation]': 'orientation() === "horizontal" ? "vertical" : "horizontal"',
|
|
114
|
+
'[attr.aria-disabled]': 'disabled() ? "true" : null',
|
|
115
|
+
'[attr.aria-valuemin]': 'ariaValueMin() ?? null',
|
|
116
|
+
'[attr.aria-valuemax]': 'ariaValueMax() ?? null',
|
|
117
|
+
'[attr.aria-valuenow]': 'ariaValueNow() ?? null',
|
|
118
|
+
}, template: "<span class=\"sd-splitter__handle-bar\"></span>\n", styles: [":host{--sd-splitter-handle-size: 4px;--sd-splitter-handle-color: var(--sd-color-primary-light, #b0bec5);--sd-splitter-handle-hover-color: var(--sd-color-primary, #1976d2);--sd-splitter-handle-active-color: var(--sd-color-primary, #1976d2);--sd-splitter-handle-hit-area: 8px;--sd-splitter-handle-radius: 0;--sd-splitter-disabled-opacity: .5;display:flex;align-items:center;justify-content:center;flex:0 0 var(--sd-splitter-handle-hit-area);-webkit-user-select:none;user-select:none;outline:none}:host.sd-splitter__handle--horizontal{cursor:col-resize}:host.sd-splitter__handle--horizontal .sd-splitter__handle-bar{width:var(--sd-splitter-handle-size);height:100%}:host.sd-splitter__handle--vertical{cursor:row-resize;flex-direction:column}:host.sd-splitter__handle--vertical .sd-splitter__handle-bar{width:100%;height:var(--sd-splitter-handle-size)}:host .sd-splitter__handle-bar{background:var(--sd-splitter-handle-color);border-radius:var(--sd-splitter-handle-radius);transition:background-color .12s ease}:host:hover .sd-splitter__handle-bar{background:var(--sd-splitter-handle-hover-color)}:host:focus-visible{outline:2px solid var(--sd-splitter-handle-active-color);outline-offset:1px}:host.sd-splitter__handle--disabled{cursor:default;opacity:var(--sd-splitter-disabled-opacity);pointer-events:none}\n"] }]
|
|
119
|
+
}], propDecorators: { onDblClick: [{
|
|
120
|
+
type: HostListener,
|
|
121
|
+
args: ['dblclick']
|
|
122
|
+
}], onKeyDown: [{
|
|
123
|
+
type: HostListener,
|
|
124
|
+
args: ['keydown', ['$event']]
|
|
125
|
+
}], onPointerDown: [{
|
|
126
|
+
type: HostListener,
|
|
127
|
+
args: ['pointerdown', ['$event']]
|
|
128
|
+
}], onPointerMove: [{
|
|
129
|
+
type: HostListener,
|
|
130
|
+
args: ['pointermove', ['$event']]
|
|
131
|
+
}], onPointerUp: [{
|
|
132
|
+
type: HostListener,
|
|
133
|
+
args: ['pointerup', ['$event']]
|
|
134
|
+
}, {
|
|
135
|
+
type: HostListener,
|
|
136
|
+
args: ['pointercancel', ['$event']]
|
|
137
|
+
}] } });
|
|
138
|
+
|
|
139
|
+
class SdSplitterPanelComponent {
|
|
140
|
+
elementRef = inject(ElementRef);
|
|
141
|
+
panelId = input(undefined);
|
|
142
|
+
size = input(1, { transform: numberAttribute });
|
|
143
|
+
unit = input('flex');
|
|
144
|
+
minSize = input(0, { transform: numberAttribute });
|
|
145
|
+
maxSize = input(undefined, {
|
|
146
|
+
transform: (v) => v == null || v === '' ? undefined : Number(v),
|
|
147
|
+
});
|
|
148
|
+
collapsible = input(false, { transform: booleanAttribute });
|
|
149
|
+
collapsed = model(false);
|
|
150
|
+
resizable = input(true, { transform: booleanAttribute });
|
|
151
|
+
static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "19.2.21", ngImport: i0, type: SdSplitterPanelComponent, deps: [], target: i0.ɵɵFactoryTarget.Component });
|
|
152
|
+
static ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.1.0", version: "19.2.21", type: SdSplitterPanelComponent, isStandalone: true, selector: "sd-splitter-panel", inputs: { panelId: { classPropertyName: "panelId", publicName: "panelId", isSignal: true, isRequired: false, transformFunction: null }, size: { classPropertyName: "size", publicName: "size", isSignal: true, isRequired: false, transformFunction: null }, unit: { classPropertyName: "unit", publicName: "unit", isSignal: true, isRequired: false, transformFunction: null }, minSize: { classPropertyName: "minSize", publicName: "minSize", isSignal: true, isRequired: false, transformFunction: null }, maxSize: { classPropertyName: "maxSize", publicName: "maxSize", isSignal: true, isRequired: false, transformFunction: null }, collapsible: { classPropertyName: "collapsible", publicName: "collapsible", isSignal: true, isRequired: false, transformFunction: null }, collapsed: { classPropertyName: "collapsed", publicName: "collapsed", isSignal: true, isRequired: false, transformFunction: null }, resizable: { classPropertyName: "resizable", publicName: "resizable", isSignal: true, isRequired: false, transformFunction: null } }, outputs: { collapsed: "collapsedChange" }, host: { properties: { "class.sd-splitter__panel--flex": "unit() === \"flex\"", "class.sd-splitter__panel--px": "unit() === \"px\"", "class.sd-splitter__panel--collapsed": "collapsed()" }, classAttribute: "sd-splitter__panel" }, ngImport: i0, template: "<ng-content></ng-content>\n", styles: [":host{display:block;overflow:hidden;box-sizing:border-box;min-width:0;min-height:0;transition:flex var(--sd-splitter-transition-duration, .2s) ease}:host.sd-splitter__panel--collapsed{flex:0 0 0!important}:host-context(.sd-splitter--dragging){transition:none!important}\n"] });
|
|
153
|
+
}
|
|
154
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "19.2.21", ngImport: i0, type: SdSplitterPanelComponent, decorators: [{
|
|
155
|
+
type: Component,
|
|
156
|
+
args: [{ selector: 'sd-splitter-panel', standalone: true, host: {
|
|
157
|
+
'class': 'sd-splitter__panel',
|
|
158
|
+
'[class.sd-splitter__panel--flex]': 'unit() === "flex"',
|
|
159
|
+
'[class.sd-splitter__panel--px]': 'unit() === "px"',
|
|
160
|
+
'[class.sd-splitter__panel--collapsed]': 'collapsed()',
|
|
161
|
+
}, template: "<ng-content></ng-content>\n", styles: [":host{display:block;overflow:hidden;box-sizing:border-box;min-width:0;min-height:0;transition:flex var(--sd-splitter-transition-duration, .2s) ease}:host.sd-splitter__panel--collapsed{flex:0 0 0!important}:host-context(.sd-splitter--dragging){transition:none!important}\n"] }]
|
|
162
|
+
}] });
|
|
163
|
+
|
|
164
|
+
// Bảo vệ phép chia khi totalFlexWeight = 0 (tất cả flex panel collapsed)
|
|
165
|
+
// hoặc flexBudgetPx = 0 (px panels chiếm hết container)
|
|
166
|
+
const NEAR_ZERO = 1e-9;
|
|
167
|
+
class SplitterStateService {
|
|
168
|
+
liveSizes = signal(new Map());
|
|
169
|
+
collapsedMap = signal(new Map());
|
|
170
|
+
committedLayout = signal({ v: 1, panels: [] });
|
|
171
|
+
#metas = [];
|
|
172
|
+
setPanelMeta(metas) {
|
|
173
|
+
this.#metas = metas;
|
|
174
|
+
}
|
|
175
|
+
getPanelMetas() {
|
|
176
|
+
return this.#metas;
|
|
177
|
+
}
|
|
178
|
+
setLiveSize(id, size) {
|
|
179
|
+
const next = new Map(this.liveSizes());
|
|
180
|
+
next.set(id, size);
|
|
181
|
+
this.liveSizes.set(next);
|
|
182
|
+
}
|
|
183
|
+
setCollapsed(id, collapsed) {
|
|
184
|
+
const next = new Map(this.collapsedMap());
|
|
185
|
+
next.set(id, collapsed);
|
|
186
|
+
this.collapsedMap.set(next);
|
|
187
|
+
}
|
|
188
|
+
reconcile(metas, stored) {
|
|
189
|
+
this.setPanelMeta(metas);
|
|
190
|
+
const liveNext = new Map();
|
|
191
|
+
const collapsedNext = new Map();
|
|
192
|
+
for (const meta of metas) {
|
|
193
|
+
let restoredSize;
|
|
194
|
+
let restoredCollapsed = false;
|
|
195
|
+
if (stored?.panels?.length) {
|
|
196
|
+
// Try match by id, ưu tiên trùng id tuyệt đối
|
|
197
|
+
const byId = stored.panels.find(p => p.id === meta.id);
|
|
198
|
+
// Fallback by index chỉ khi panel không có panelId string. Theo convention
|
|
199
|
+
// ResolvedPanelMeta.id = panelId nếu có (string), else fallback về index (number).
|
|
200
|
+
// → id là số ⇔ template không khai báo panelId → index match là valid.
|
|
201
|
+
const match = byId ?? (typeof meta.id === 'number' ? stored.panels[meta.index] : undefined);
|
|
202
|
+
// Chỉ accept nếu unit trùng
|
|
203
|
+
if (match && match.unit === meta.unit) {
|
|
204
|
+
restoredSize = match.size;
|
|
205
|
+
restoredCollapsed = match.collapsed;
|
|
206
|
+
}
|
|
207
|
+
}
|
|
208
|
+
liveNext.set(meta.id, restoredSize ?? meta.declaredSize);
|
|
209
|
+
collapsedNext.set(meta.id, restoredCollapsed);
|
|
210
|
+
}
|
|
211
|
+
this.liveSizes.set(liveNext);
|
|
212
|
+
this.collapsedMap.set(collapsedNext);
|
|
213
|
+
}
|
|
214
|
+
/**
|
|
215
|
+
* Áp delta px lên 2 panel kề handleIndex (prev = handleIndex, next = handleIndex + 1).
|
|
216
|
+
* Khi snap collapsible panel: tự set collapsed + reset size = 0.
|
|
217
|
+
* Khi expand collapsible panel đang collapsed: nếu delta đủ lớn → expand.
|
|
218
|
+
* Trả về delta thực sự đã áp.
|
|
219
|
+
*/
|
|
220
|
+
applyDelta(handleIndex, deltaPx, containerPx, snapThreshold = 0.5) {
|
|
221
|
+
const prev = this.#metas[handleIndex];
|
|
222
|
+
const next = this.#metas[handleIndex + 1];
|
|
223
|
+
if (!prev || !next)
|
|
224
|
+
return 0;
|
|
225
|
+
// Trường hợp 1: 1 trong 2 panel đang collapsed → expand khi delta đủ lớn.
|
|
226
|
+
// So sánh deltaPx (px) với minSize đã convert sang px (vì minSize có thể là flex weight).
|
|
227
|
+
const prevCollapsed = this.collapsedMap().get(prev.id) === true;
|
|
228
|
+
const nextCollapsed = this.collapsedMap().get(next.id) === true;
|
|
229
|
+
if (prevCollapsed || nextCollapsed) {
|
|
230
|
+
const flexBudgetPx = this.#flexBudgetPx(containerPx);
|
|
231
|
+
const totalFlexWeight = this.#totalFlexWeight();
|
|
232
|
+
if (prevCollapsed && prev.collapsible) {
|
|
233
|
+
const prevMinPx = this.#sizeToPx(prev, prev.minSize, flexBudgetPx, totalFlexWeight);
|
|
234
|
+
if (deltaPx >= prevMinPx) {
|
|
235
|
+
this.expandPanel(prev.id);
|
|
236
|
+
return prevMinPx;
|
|
237
|
+
}
|
|
238
|
+
}
|
|
239
|
+
if (nextCollapsed && next.collapsible) {
|
|
240
|
+
const nextMinPx = this.#sizeToPx(next, next.minSize, flexBudgetPx, totalFlexWeight);
|
|
241
|
+
if (-deltaPx >= nextMinPx) {
|
|
242
|
+
this.expandPanel(next.id);
|
|
243
|
+
return -nextMinPx;
|
|
244
|
+
}
|
|
245
|
+
}
|
|
246
|
+
return 0;
|
|
247
|
+
}
|
|
248
|
+
const sizes = this.liveSizes();
|
|
249
|
+
const prevSize = sizes.get(prev.id) ?? prev.declaredSize;
|
|
250
|
+
const nextSize = sizes.get(next.id) ?? next.declaredSize;
|
|
251
|
+
const flexBudgetPx = this.#flexBudgetPx(containerPx);
|
|
252
|
+
const totalFlexWeight = this.#totalFlexWeight();
|
|
253
|
+
const prevPx = prev.unit === 'px' ? prevSize : (flexBudgetPx * prevSize) / Math.max(totalFlexWeight, NEAR_ZERO);
|
|
254
|
+
const nextPx = next.unit === 'px' ? nextSize : (flexBudgetPx * nextSize) / Math.max(totalFlexWeight, NEAR_ZERO);
|
|
255
|
+
const rawNewPrevPx = prevPx + deltaPx;
|
|
256
|
+
const rawNewNextPx = nextPx - deltaPx;
|
|
257
|
+
const prevMinPx = this.#sizeToPx(prev, prev.minSize, flexBudgetPx, totalFlexWeight);
|
|
258
|
+
const nextMinPx = this.#sizeToPx(next, next.minSize, flexBudgetPx, totalFlexWeight);
|
|
259
|
+
// Snap check: panel kéo dưới minSize × snapThreshold + collapsible → snap collapse
|
|
260
|
+
if (prev.collapsible && prevMinPx > 0 && rawNewPrevPx < prevMinPx * snapThreshold) {
|
|
261
|
+
this.collapsePanel(prev.id);
|
|
262
|
+
this.setLiveSize(prev.id, 0);
|
|
263
|
+
return prevPx * -1;
|
|
264
|
+
}
|
|
265
|
+
if (next.collapsible && nextMinPx > 0 && rawNewNextPx < nextMinPx * snapThreshold) {
|
|
266
|
+
this.collapsePanel(next.id);
|
|
267
|
+
this.setLiveSize(next.id, 0);
|
|
268
|
+
return nextPx;
|
|
269
|
+
}
|
|
270
|
+
// Không snap → clamp logic cũ
|
|
271
|
+
const prevMaxPx = prev.maxSize != null ? this.#sizeToPx(prev, prev.maxSize, flexBudgetPx, totalFlexWeight) : Infinity;
|
|
272
|
+
const nextMaxPx = next.maxSize != null ? this.#sizeToPx(next, next.maxSize, flexBudgetPx, totalFlexWeight) : Infinity;
|
|
273
|
+
let delta = deltaPx;
|
|
274
|
+
delta = Math.max(delta, prevMinPx - prevPx);
|
|
275
|
+
delta = Math.min(delta, prevMaxPx - prevPx);
|
|
276
|
+
delta = Math.max(delta, nextPx - nextMaxPx);
|
|
277
|
+
delta = Math.min(delta, nextPx - nextMinPx);
|
|
278
|
+
if (delta === 0)
|
|
279
|
+
return 0;
|
|
280
|
+
const newPrevPx = prevPx + delta;
|
|
281
|
+
const newNextPx = nextPx - delta;
|
|
282
|
+
const liveNext = new Map(this.liveSizes());
|
|
283
|
+
liveNext.set(prev.id, prev.unit === 'px' ? newPrevPx : (newPrevPx * totalFlexWeight) / Math.max(flexBudgetPx, NEAR_ZERO));
|
|
284
|
+
liveNext.set(next.id, next.unit === 'px' ? newNextPx : (newNextPx * totalFlexWeight) / Math.max(flexBudgetPx, NEAR_ZERO));
|
|
285
|
+
this.liveSizes.set(liveNext);
|
|
286
|
+
return delta;
|
|
287
|
+
}
|
|
288
|
+
#flexBudgetPx(containerPx) {
|
|
289
|
+
let pxConsumed = 0;
|
|
290
|
+
const sizes = this.liveSizes();
|
|
291
|
+
for (const m of this.#metas) {
|
|
292
|
+
if (m.unit === 'px' && !this.collapsedMap().get(m.id)) {
|
|
293
|
+
pxConsumed += sizes.get(m.id) ?? m.declaredSize;
|
|
294
|
+
}
|
|
295
|
+
}
|
|
296
|
+
return Math.max(containerPx - pxConsumed, 0);
|
|
297
|
+
}
|
|
298
|
+
#totalFlexWeight() {
|
|
299
|
+
let total = 0;
|
|
300
|
+
const sizes = this.liveSizes();
|
|
301
|
+
for (const m of this.#metas) {
|
|
302
|
+
if (m.unit === 'flex' && !this.collapsedMap().get(m.id)) {
|
|
303
|
+
total += sizes.get(m.id) ?? m.declaredSize;
|
|
304
|
+
}
|
|
305
|
+
}
|
|
306
|
+
return total;
|
|
307
|
+
}
|
|
308
|
+
#sizeToPx(meta, value, flexBudgetPx, totalFlexWeight) {
|
|
309
|
+
return meta.unit === 'px' ? value : (flexBudgetPx * value) / Math.max(totalFlexWeight, NEAR_ZERO);
|
|
310
|
+
}
|
|
311
|
+
collapsePanel(id) {
|
|
312
|
+
const meta = this.#metas.find(m => m.id === id);
|
|
313
|
+
if (!meta || !meta.collapsible)
|
|
314
|
+
return;
|
|
315
|
+
// Lưu size hiện tại để expand sau
|
|
316
|
+
const current = this.liveSizes().get(id);
|
|
317
|
+
if (current !== undefined && current > 0) {
|
|
318
|
+
meta.lastSize = current;
|
|
319
|
+
}
|
|
320
|
+
this.setCollapsed(id, true);
|
|
321
|
+
}
|
|
322
|
+
expandPanel(id) {
|
|
323
|
+
const meta = this.#metas.find(m => m.id === id);
|
|
324
|
+
if (!meta)
|
|
325
|
+
return;
|
|
326
|
+
let restoreSize = meta.lastSize;
|
|
327
|
+
if (!restoreSize || restoreSize <= 0) {
|
|
328
|
+
// Fallback chain: lastSize → minSize → declaredSize. Giả định declaredSize > 0;
|
|
329
|
+
// nếu cả 3 đều ≤ 0 (template sai), panel expand về size 0 — visually invisible
|
|
330
|
+
// nhưng state nhất quán (collapsed=false). Caller chịu trách nhiệm khai báo size hợp lý.
|
|
331
|
+
restoreSize = meta.minSize > 0 ? meta.minSize : meta.declaredSize;
|
|
332
|
+
}
|
|
333
|
+
this.setLiveSize(id, restoreSize);
|
|
334
|
+
this.setCollapsed(id, false);
|
|
335
|
+
}
|
|
336
|
+
togglePanel(id) {
|
|
337
|
+
if (this.collapsedMap().get(id)) {
|
|
338
|
+
this.expandPanel(id);
|
|
339
|
+
}
|
|
340
|
+
else {
|
|
341
|
+
this.collapsePanel(id);
|
|
342
|
+
}
|
|
343
|
+
}
|
|
344
|
+
commit() {
|
|
345
|
+
const sizes = this.liveSizes();
|
|
346
|
+
const collapsed = this.collapsedMap();
|
|
347
|
+
const panels = this.#metas.map(meta => ({
|
|
348
|
+
id: meta.id,
|
|
349
|
+
size: sizes.get(meta.id) ?? meta.declaredSize,
|
|
350
|
+
unit: meta.unit,
|
|
351
|
+
collapsed: collapsed.get(meta.id) ?? false,
|
|
352
|
+
}));
|
|
353
|
+
this.committedLayout.set({ v: 1, panels });
|
|
354
|
+
}
|
|
355
|
+
static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "19.2.21", ngImport: i0, type: SplitterStateService, deps: [], target: i0.ɵɵFactoryTarget.Injectable });
|
|
356
|
+
static ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "19.2.21", ngImport: i0, type: SplitterStateService });
|
|
357
|
+
}
|
|
358
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "19.2.21", ngImport: i0, type: SplitterStateService, decorators: [{
|
|
359
|
+
type: Injectable
|
|
360
|
+
}] });
|
|
361
|
+
|
|
362
|
+
class SdSplitterComponent {
|
|
363
|
+
#host = inject(ElementRef);
|
|
364
|
+
// EnvironmentInjector: dùng cho createComponent (cần injector tree). Lifetime application-scope.
|
|
365
|
+
#envInjector = inject(EnvironmentInjector);
|
|
366
|
+
// Component-scoped Injector: gắn DestroyRef của component → afterNextRender callback tự cancel khi destroy
|
|
367
|
+
#injector = inject(Injector);
|
|
368
|
+
#destroyRef = inject(DestroyRef);
|
|
369
|
+
#state = inject(SplitterStateService);
|
|
370
|
+
#storage = inject(SdStorageService);
|
|
371
|
+
#storageHandle = computed(() => {
|
|
372
|
+
const key = this.storageKey();
|
|
373
|
+
return key ? this.#storage.create(key) : null;
|
|
374
|
+
});
|
|
375
|
+
orientation = input('horizontal');
|
|
376
|
+
disabled = input(false, { transform: booleanAttribute });
|
|
377
|
+
storageKey = input(undefined);
|
|
378
|
+
snapThreshold = input(0.5, { transform: numberAttribute });
|
|
379
|
+
keyboardStep = input(10, { transform: numberAttribute });
|
|
380
|
+
resizeEnd = output();
|
|
381
|
+
collapsedChange = output();
|
|
382
|
+
layoutChange = output();
|
|
383
|
+
panels = contentChildren(SdSplitterPanelComponent);
|
|
384
|
+
#handleRefs = [];
|
|
385
|
+
#dragStartSize = null;
|
|
386
|
+
#dragLastDelta = 0;
|
|
387
|
+
#prevCollapsedMap = new Map();
|
|
388
|
+
constructor() {
|
|
389
|
+
// 1. Reconcile state khi panels signal đổi (panel add/remove qua @if/@for)
|
|
390
|
+
effect(() => {
|
|
391
|
+
const panels = this.panels();
|
|
392
|
+
const stored = this.#storageHandle()?.get() ?? null;
|
|
393
|
+
const metas = panels.map((p, i) => this.#toMeta(p, i));
|
|
394
|
+
this.#state.reconcile(metas, stored);
|
|
395
|
+
});
|
|
396
|
+
// Auto-save vào storage khi committedLayout đổi (only commit triggers, không phải live drag)
|
|
397
|
+
effect(() => {
|
|
398
|
+
const layout = this.#state.committedLayout();
|
|
399
|
+
const handle = this.#storageHandle();
|
|
400
|
+
if (handle && layout.panels.length > 0) {
|
|
401
|
+
handle.setSilent(layout); // setSilent: không emit qua storage subject
|
|
402
|
+
}
|
|
403
|
+
});
|
|
404
|
+
// Emit layoutChange + collapsedChange (diff) khi committedLayout đổi
|
|
405
|
+
effect(() => {
|
|
406
|
+
const layout = this.#state.committedLayout();
|
|
407
|
+
if (layout.panels.length === 0)
|
|
408
|
+
return;
|
|
409
|
+
this.layoutChange.emit(layout);
|
|
410
|
+
// Detect collapsed change qua diff với prev map
|
|
411
|
+
const currMap = this.#state.collapsedMap();
|
|
412
|
+
for (const [id, isCollapsed] of currMap) {
|
|
413
|
+
const prev = this.#prevCollapsedMap.get(id) ?? false;
|
|
414
|
+
if (prev !== isCollapsed) {
|
|
415
|
+
this.collapsedChange.emit({ panelId: id, collapsed: isCollapsed });
|
|
416
|
+
}
|
|
417
|
+
}
|
|
418
|
+
this.#prevCollapsedMap = new Map(currMap);
|
|
419
|
+
});
|
|
420
|
+
// 2. Apply flex style lên panel host element dựa trên liveSizes + collapsedMap.
|
|
421
|
+
// Normalize flex-grow của các panel flex để sum = 1 → CSS phân phối hết free space.
|
|
422
|
+
// Nếu để raw weight (vd 0.7), sum < 1 → flexbox để lại khoảng trống bên rìa.
|
|
423
|
+
effect(() => {
|
|
424
|
+
const sizes = this.#state.liveSizes();
|
|
425
|
+
const collapsed = this.#state.collapsedMap();
|
|
426
|
+
const panels = this.panels();
|
|
427
|
+
// Tính tổng weight của panel flex đang không collapsed (để normalize)
|
|
428
|
+
let totalFlexWeight = 0;
|
|
429
|
+
for (let i = 0; i < panels.length; i++) {
|
|
430
|
+
const panel = panels[i];
|
|
431
|
+
const id = panel.panelId() ?? i;
|
|
432
|
+
if (panel.unit() === 'flex' && !collapsed.get(id)) {
|
|
433
|
+
totalFlexWeight += sizes.get(id) ?? 1;
|
|
434
|
+
}
|
|
435
|
+
}
|
|
436
|
+
for (let i = 0; i < panels.length; i++) {
|
|
437
|
+
const panel = panels[i];
|
|
438
|
+
const id = panel.panelId() ?? i;
|
|
439
|
+
const isCollapsed = collapsed.get(id) === true;
|
|
440
|
+
const size = sizes.get(id) ?? 1;
|
|
441
|
+
let flex;
|
|
442
|
+
if (isCollapsed) {
|
|
443
|
+
flex = '0 0 0';
|
|
444
|
+
}
|
|
445
|
+
else if (panel.unit() === 'px') {
|
|
446
|
+
flex = `0 0 ${size}px`;
|
|
447
|
+
}
|
|
448
|
+
else {
|
|
449
|
+
// Normalize: grow = weight / totalWeight → sum(grow) = 1
|
|
450
|
+
const grow = totalFlexWeight > 0 ? size / totalFlexWeight : 1;
|
|
451
|
+
flex = `${grow} 1 0`;
|
|
452
|
+
}
|
|
453
|
+
panel.elementRef.nativeElement.style.flex = flex;
|
|
454
|
+
}
|
|
455
|
+
});
|
|
456
|
+
// Sync handles sau khi DOM render xong (panels đã projected vào host)
|
|
457
|
+
effect(() => {
|
|
458
|
+
const panelCount = this.panels().length;
|
|
459
|
+
const orientation = this.orientation();
|
|
460
|
+
const disabled = this.disabled();
|
|
461
|
+
const keyboardStep = this.keyboardStep();
|
|
462
|
+
afterNextRender(() => this.#syncHandles(panelCount, orientation, disabled, keyboardStep), { injector: this.#injector } // component-scoped → auto-cancel khi component destroy
|
|
463
|
+
);
|
|
464
|
+
});
|
|
465
|
+
// Destroy handle ComponentRef khi container bị destroy (tránh leak)
|
|
466
|
+
this.#destroyRef.onDestroy(() => {
|
|
467
|
+
for (const ref of this.#handleRefs)
|
|
468
|
+
ref.destroy();
|
|
469
|
+
this.#handleRefs = [];
|
|
470
|
+
});
|
|
471
|
+
}
|
|
472
|
+
#toMeta(panel, index) {
|
|
473
|
+
return {
|
|
474
|
+
id: panel.panelId() ?? index,
|
|
475
|
+
index,
|
|
476
|
+
unit: panel.unit(),
|
|
477
|
+
minSize: panel.minSize(),
|
|
478
|
+
maxSize: panel.maxSize(),
|
|
479
|
+
collapsible: panel.collapsible(),
|
|
480
|
+
resizable: panel.resizable(),
|
|
481
|
+
declaredSize: panel.size(),
|
|
482
|
+
lastSize: panel.size(),
|
|
483
|
+
};
|
|
484
|
+
}
|
|
485
|
+
#syncHandles(panelCount, orientation, disabled, keyboardStep) {
|
|
486
|
+
const panels = this.panels();
|
|
487
|
+
const needed = Math.max(0, panelCount - 1);
|
|
488
|
+
// Remove excess
|
|
489
|
+
while (this.#handleRefs.length > needed) {
|
|
490
|
+
this.#handleRefs.pop().destroy();
|
|
491
|
+
}
|
|
492
|
+
// Create missing + wire events
|
|
493
|
+
while (this.#handleRefs.length < needed) {
|
|
494
|
+
const ref = createComponent(SdSplitterHandleComponent, { environmentInjector: this.#envInjector });
|
|
495
|
+
const handleIndex = this.#handleRefs.length;
|
|
496
|
+
ref.instance.dragStart.subscribe(() => this.#onDragStart(handleIndex));
|
|
497
|
+
ref.instance.dragMove.subscribe(delta => this.#onDragMove(handleIndex, delta));
|
|
498
|
+
ref.instance.dragEnd.subscribe(() => this.#onDragEnd(handleIndex));
|
|
499
|
+
ref.instance.toggleRequest.subscribe(() => this.#onHandleToggle(handleIndex));
|
|
500
|
+
this.#handleRefs.push(ref);
|
|
501
|
+
}
|
|
502
|
+
// Apply inputs với disabled tính theo per-panel resizable
|
|
503
|
+
for (let i = 0; i < this.#handleRefs.length; i++) {
|
|
504
|
+
const ref = this.#handleRefs[i];
|
|
505
|
+
const prev = panels[i];
|
|
506
|
+
const next = panels[i + 1];
|
|
507
|
+
const handleDisabled = disabled || !prev.resizable() || !next.resizable();
|
|
508
|
+
ref.setInput('orientation', orientation);
|
|
509
|
+
ref.setInput('disabled', handleDisabled);
|
|
510
|
+
ref.setInput('keyboardStep', keyboardStep);
|
|
511
|
+
ref.changeDetectorRef.detectChanges();
|
|
512
|
+
}
|
|
513
|
+
// Re-arrange DOM: panel0, handle0, panel1, handle1, ..., panelN
|
|
514
|
+
const host = this.#host.nativeElement;
|
|
515
|
+
for (let i = 0; i < panels.length; i++) {
|
|
516
|
+
host.appendChild(panels[i].elementRef.nativeElement);
|
|
517
|
+
if (i < this.#handleRefs.length)
|
|
518
|
+
host.appendChild(this.#handleRefs[i].location.nativeElement);
|
|
519
|
+
}
|
|
520
|
+
}
|
|
521
|
+
#onDragStart(handleIndex) {
|
|
522
|
+
const rect = this.#host.nativeElement.getBoundingClientRect();
|
|
523
|
+
const containerPx = this.orientation() === 'horizontal' ? rect.width : rect.height;
|
|
524
|
+
this.#dragStartSize = { handleIndex, containerPx };
|
|
525
|
+
this.#dragLastDelta = 0;
|
|
526
|
+
this.#host.nativeElement.classList.add('sd-splitter--dragging');
|
|
527
|
+
}
|
|
528
|
+
#onDragMove(handleIndex, deltaSinceStart) {
|
|
529
|
+
if (!this.#dragStartSize)
|
|
530
|
+
return;
|
|
531
|
+
const incrementalDelta = deltaSinceStart - this.#dragLastDelta;
|
|
532
|
+
this.#dragLastDelta = deltaSinceStart;
|
|
533
|
+
this.#state.applyDelta(handleIndex, incrementalDelta, this.#dragStartSize.containerPx, this.snapThreshold());
|
|
534
|
+
}
|
|
535
|
+
#onDragEnd(_handleIndex) {
|
|
536
|
+
this.#dragStartSize = null;
|
|
537
|
+
this.#host.nativeElement.classList.remove('sd-splitter--dragging');
|
|
538
|
+
this.#state.commit();
|
|
539
|
+
this.resizeEnd.emit(this.#state.committedLayout());
|
|
540
|
+
}
|
|
541
|
+
#onHandleToggle(handleIndex) {
|
|
542
|
+
// Double-click / Enter / Space — ưu tiên collapse panel collapsible ở phía prev, fallback next
|
|
543
|
+
const panels = this.panels();
|
|
544
|
+
const prev = panels[handleIndex];
|
|
545
|
+
const next = panels[handleIndex + 1];
|
|
546
|
+
const target = prev.collapsible() ? prev : next.collapsible() ? next : null;
|
|
547
|
+
if (!target)
|
|
548
|
+
return;
|
|
549
|
+
const id = target.panelId() ?? panels.indexOf(target);
|
|
550
|
+
this.#state.togglePanel(id);
|
|
551
|
+
this.#state.commit();
|
|
552
|
+
}
|
|
553
|
+
// --- Imperative API ---
|
|
554
|
+
getLayout() {
|
|
555
|
+
const metas = this.#state.getPanelMetas();
|
|
556
|
+
const sizes = this.#state.liveSizes();
|
|
557
|
+
const collapsed = this.#state.collapsedMap();
|
|
558
|
+
return {
|
|
559
|
+
v: 1,
|
|
560
|
+
panels: metas.map(m => ({
|
|
561
|
+
id: m.id,
|
|
562
|
+
size: sizes.get(m.id) ?? m.declaredSize,
|
|
563
|
+
unit: m.unit,
|
|
564
|
+
collapsed: collapsed.get(m.id) ?? false,
|
|
565
|
+
})),
|
|
566
|
+
};
|
|
567
|
+
}
|
|
568
|
+
setLayout(state) {
|
|
569
|
+
const metas = this.#state.getPanelMetas();
|
|
570
|
+
for (const stored of state.panels) {
|
|
571
|
+
const meta = metas.find(m => m.id === stored.id);
|
|
572
|
+
if (!meta || meta.unit !== stored.unit)
|
|
573
|
+
continue;
|
|
574
|
+
this.#state.setLiveSize(meta.id, stored.size);
|
|
575
|
+
this.#state.setCollapsed(meta.id, stored.collapsed);
|
|
576
|
+
}
|
|
577
|
+
this.#state.commit();
|
|
578
|
+
}
|
|
579
|
+
resetLayout() {
|
|
580
|
+
const metas = this.#state.getPanelMetas();
|
|
581
|
+
for (const m of metas) {
|
|
582
|
+
this.#state.setLiveSize(m.id, m.declaredSize);
|
|
583
|
+
this.#state.setCollapsed(m.id, false);
|
|
584
|
+
}
|
|
585
|
+
this.#state.commit();
|
|
586
|
+
}
|
|
587
|
+
collapse(target) {
|
|
588
|
+
const id = this.#resolveTarget(target);
|
|
589
|
+
this.#state.collapsePanel(id);
|
|
590
|
+
this.#state.commit();
|
|
591
|
+
}
|
|
592
|
+
expand(target) {
|
|
593
|
+
const id = this.#resolveTarget(target);
|
|
594
|
+
this.#state.expandPanel(id);
|
|
595
|
+
this.#state.commit();
|
|
596
|
+
}
|
|
597
|
+
toggle(target) {
|
|
598
|
+
const id = this.#resolveTarget(target);
|
|
599
|
+
this.#state.togglePanel(id);
|
|
600
|
+
this.#state.commit();
|
|
601
|
+
}
|
|
602
|
+
resizePanel(target, size) {
|
|
603
|
+
const id = this.#resolveTarget(target);
|
|
604
|
+
const meta = this.#state.getPanelMetas().find(m => m.id === id);
|
|
605
|
+
if (!meta)
|
|
606
|
+
return;
|
|
607
|
+
let clamped = Math.max(size, meta.minSize);
|
|
608
|
+
if (meta.maxSize != null)
|
|
609
|
+
clamped = Math.min(clamped, meta.maxSize);
|
|
610
|
+
this.#state.setLiveSize(id, clamped);
|
|
611
|
+
this.#state.commit();
|
|
612
|
+
}
|
|
613
|
+
#resolveTarget(target) {
|
|
614
|
+
const metas = this.#state.getPanelMetas();
|
|
615
|
+
if (typeof target === 'number') {
|
|
616
|
+
const meta = metas[target] ?? metas.find(m => m.id === target);
|
|
617
|
+
if (!meta)
|
|
618
|
+
throw new Error(`Splitter: no panel at index ${target}`);
|
|
619
|
+
return meta.id;
|
|
620
|
+
}
|
|
621
|
+
const meta = metas.find(m => m.id === target);
|
|
622
|
+
if (!meta)
|
|
623
|
+
throw new Error(`Splitter: no panel with id "${target}"`);
|
|
624
|
+
return meta.id;
|
|
625
|
+
}
|
|
626
|
+
static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "19.2.21", ngImport: i0, type: SdSplitterComponent, deps: [], target: i0.ɵɵFactoryTarget.Component });
|
|
627
|
+
static ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.2.0", version: "19.2.21", type: SdSplitterComponent, isStandalone: true, selector: "sd-splitter", inputs: { orientation: { classPropertyName: "orientation", publicName: "orientation", isSignal: true, isRequired: false, transformFunction: null }, disabled: { classPropertyName: "disabled", publicName: "disabled", isSignal: true, isRequired: false, transformFunction: null }, storageKey: { classPropertyName: "storageKey", publicName: "storageKey", isSignal: true, isRequired: false, transformFunction: null }, snapThreshold: { classPropertyName: "snapThreshold", publicName: "snapThreshold", isSignal: true, isRequired: false, transformFunction: null }, keyboardStep: { classPropertyName: "keyboardStep", publicName: "keyboardStep", isSignal: true, isRequired: false, transformFunction: null } }, outputs: { resizeEnd: "resizeEnd", collapsedChange: "collapsedChange", layoutChange: "layoutChange" }, host: { properties: { "class.sd-splitter--horizontal": "orientation() === \"horizontal\"", "class.sd-splitter--vertical": "orientation() === \"vertical\"", "class.sd-splitter--disabled": "disabled()" }, classAttribute: "sd-splitter" }, providers: [SplitterStateService], queries: [{ propertyName: "panels", predicate: SdSplitterPanelComponent, isSignal: true }], ngImport: i0, template: "<ng-content select=\"sd-splitter-panel\"></ng-content>\n", styles: [":host{display:flex;width:100%;height:100%;overflow:hidden;box-sizing:border-box}:host.sd-splitter--horizontal{flex-direction:row}:host.sd-splitter--vertical{flex-direction:column}:host.sd-splitter--disabled .sd-splitter__handle{pointer-events:none}:host.sd-splitter--dragging{-webkit-user-select:none;user-select:none}:host.sd-splitter--dragging .sd-splitter__panel{transition:none!important}\n"] });
|
|
628
|
+
}
|
|
629
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "19.2.21", ngImport: i0, type: SdSplitterComponent, decorators: [{
|
|
630
|
+
type: Component,
|
|
631
|
+
args: [{ selector: 'sd-splitter', standalone: true, providers: [SplitterStateService], host: {
|
|
632
|
+
'class': 'sd-splitter',
|
|
633
|
+
'[class.sd-splitter--horizontal]': 'orientation() === "horizontal"',
|
|
634
|
+
'[class.sd-splitter--vertical]': 'orientation() === "vertical"',
|
|
635
|
+
'[class.sd-splitter--disabled]': 'disabled()',
|
|
636
|
+
}, template: "<ng-content select=\"sd-splitter-panel\"></ng-content>\n", styles: [":host{display:flex;width:100%;height:100%;overflow:hidden;box-sizing:border-box}:host.sd-splitter--horizontal{flex-direction:row}:host.sd-splitter--vertical{flex-direction:column}:host.sd-splitter--disabled .sd-splitter__handle{pointer-events:none}:host.sd-splitter--dragging{-webkit-user-select:none;user-select:none}:host.sd-splitter--dragging .sd-splitter__panel{transition:none!important}\n"] }]
|
|
637
|
+
}], ctorParameters: () => [] });
|
|
638
|
+
|
|
639
|
+
// projects/sd-angular/components/splitter/index.ts
|
|
640
|
+
|
|
641
|
+
/**
|
|
642
|
+
* Generated bundle index. Do not edit.
|
|
643
|
+
*/
|
|
644
|
+
|
|
645
|
+
export { SdSplitterComponent, SdSplitterPanelComponent };
|
|
646
|
+
//# sourceMappingURL=sd-angular-core-components-splitter.mjs.map
|