@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.
Files changed (211) hide show
  1. package/components/document-builder/src/components/header-footer-builder/header-footer-builder.component.d.ts +4 -1
  2. package/components/document-builder/src/document-builder.model.d.ts +2 -0
  3. package/components/editor/src/models/editor.model.d.ts +2 -0
  4. package/components/editor/src/plugins/image-upload/utils/validate.utils.d.ts +2 -1
  5. package/components/form-generic/index.d.ts +4 -0
  6. package/components/{workflow → form-generic}/src/components/form-builder/components/attribute-table/attribute-table.component.d.ts +3 -3
  7. package/components/{workflow → form-generic}/src/components/form-builder/components/attribute-template/attribute-template.component.d.ts +3 -3
  8. package/components/{workflow → form-generic}/src/components/form-builder/components/configure-validation/configure-validation.component.d.ts +3 -3
  9. package/components/{workflow → form-generic}/src/components/form-builder/components/expression-builder/expression-builder.component.d.ts +1 -1
  10. package/components/{workflow → form-generic}/src/components/form-render/form-render.component.d.ts +3 -3
  11. package/components/{workflow → form-generic}/src/components/sd-feel-expression/sd-feel-expression.component.d.ts +1 -1
  12. package/components/form-generic/src/configurations/form-generic.configuration.d.ts +6 -0
  13. package/components/form-generic/src/configurations/index.d.ts +2 -0
  14. package/components/{workflow → form-generic}/src/services/form-generic.service.d.ts +3 -3
  15. package/components/index.d.ts +1 -0
  16. package/components/splitter/index.d.ts +3 -0
  17. package/components/splitter/src/splitter-handle/splitter-handle.component.d.ts +24 -0
  18. package/components/splitter/src/splitter-panel/splitter-panel.component.d.ts +16 -0
  19. package/components/splitter/src/splitter-state.service.d.ts +26 -0
  20. package/components/splitter/src/splitter.component.d.ts +28 -0
  21. package/components/splitter/src/splitter.models.d.ts +23 -0
  22. package/components/table/src/components/selector-action/selector-action.component.d.ts +1 -0
  23. package/components/table/src/models/table-option-config.model.d.ts +1 -0
  24. package/components/table/src/services/column-width.util.d.ts +7 -0
  25. package/components/table/src/table.component.d.ts +1 -0
  26. package/components/workflow/index.d.ts +2 -4
  27. package/components/workflow/src/configurations/workflow.configuration.d.ts +9 -5
  28. package/configurations/src/sd-core.configuration.d.ts +1 -0
  29. package/fesm2022/sd-angular-core-components-anchor.mjs +4 -4
  30. package/fesm2022/sd-angular-core-components-anchor.mjs.map +1 -1
  31. package/fesm2022/sd-angular-core-components-code-editor.mjs +3 -2
  32. package/fesm2022/sd-angular-core-components-code-editor.mjs.map +1 -1
  33. package/fesm2022/sd-angular-core-components-document-builder.mjs +29 -8
  34. package/fesm2022/sd-angular-core-components-document-builder.mjs.map +1 -1
  35. package/fesm2022/sd-angular-core-components-editor.mjs +31 -16
  36. package/fesm2022/sd-angular-core-components-editor.mjs.map +1 -1
  37. package/fesm2022/sd-angular-core-components-form-generic.mjs +6404 -0
  38. package/fesm2022/sd-angular-core-components-form-generic.mjs.map +1 -0
  39. package/fesm2022/sd-angular-core-components-history.mjs +3 -2
  40. package/fesm2022/sd-angular-core-components-history.mjs.map +1 -1
  41. package/fesm2022/sd-angular-core-components-import-excel.mjs +25 -23
  42. package/fesm2022/sd-angular-core-components-import-excel.mjs.map +1 -1
  43. package/fesm2022/sd-angular-core-components-preview.mjs +6 -4
  44. package/fesm2022/sd-angular-core-components-preview.mjs.map +1 -1
  45. package/fesm2022/sd-angular-core-components-splitter.mjs +646 -0
  46. package/fesm2022/sd-angular-core-components-splitter.mjs.map +1 -0
  47. package/fesm2022/sd-angular-core-components-tab-router.mjs +7 -5
  48. package/fesm2022/sd-angular-core-components-tab-router.mjs.map +1 -1
  49. package/fesm2022/sd-angular-core-components-table.mjs +52 -23
  50. package/fesm2022/sd-angular-core-components-table.mjs.map +1 -1
  51. package/fesm2022/sd-angular-core-components-upload-file.mjs +24 -21
  52. package/fesm2022/sd-angular-core-components-upload-file.mjs.map +1 -1
  53. package/fesm2022/sd-angular-core-components-workflow.mjs +7 -6387
  54. package/fesm2022/sd-angular-core-components-workflow.mjs.map +1 -1
  55. package/fesm2022/sd-angular-core-components.mjs +1 -0
  56. package/fesm2022/sd-angular-core-components.mjs.map +1 -1
  57. package/fesm2022/sd-angular-core-configurations.mjs.map +1 -1
  58. package/fesm2022/sd-angular-core-directives.mjs +6 -2
  59. package/fesm2022/sd-angular-core-directives.mjs.map +1 -1
  60. package/fesm2022/sd-angular-core-forms-autocomplete.mjs +3 -1
  61. package/fesm2022/sd-angular-core-forms-autocomplete.mjs.map +1 -1
  62. package/fesm2022/sd-angular-core-forms-chip-calendar.mjs +5 -3
  63. package/fesm2022/sd-angular-core-forms-chip-calendar.mjs.map +1 -1
  64. package/fesm2022/sd-angular-core-forms-chip.mjs +5 -3
  65. package/fesm2022/sd-angular-core-forms-chip.mjs.map +1 -1
  66. package/fesm2022/sd-angular-core-forms-date-range.mjs +8 -5
  67. package/fesm2022/sd-angular-core-forms-date-range.mjs.map +1 -1
  68. package/fesm2022/sd-angular-core-forms-date.mjs +7 -5
  69. package/fesm2022/sd-angular-core-forms-date.mjs.map +1 -1
  70. package/fesm2022/sd-angular-core-forms-datetime.mjs +10 -8
  71. package/fesm2022/sd-angular-core-forms-datetime.mjs.map +1 -1
  72. package/fesm2022/sd-angular-core-forms-input-number.mjs +5 -3
  73. package/fesm2022/sd-angular-core-forms-input-number.mjs.map +1 -1
  74. package/fesm2022/sd-angular-core-forms-input.mjs +13 -6
  75. package/fesm2022/sd-angular-core-forms-input.mjs.map +1 -1
  76. package/fesm2022/sd-angular-core-forms-radio.mjs +3 -2
  77. package/fesm2022/sd-angular-core-forms-radio.mjs.map +1 -1
  78. package/fesm2022/sd-angular-core-forms-select.mjs +5 -3
  79. package/fesm2022/sd-angular-core-forms-select.mjs.map +1 -1
  80. package/fesm2022/sd-angular-core-forms-textarea.mjs +8 -5
  81. package/fesm2022/sd-angular-core-forms-textarea.mjs.map +1 -1
  82. package/fesm2022/sd-angular-core-handlers.mjs +7 -6
  83. package/fesm2022/sd-angular-core-handlers.mjs.map +1 -1
  84. package/fesm2022/sd-angular-core-i18n.mjs +790 -0
  85. package/fesm2022/sd-angular-core-i18n.mjs.map +1 -0
  86. package/fesm2022/sd-angular-core-interceptors.mjs +10 -6
  87. package/fesm2022/sd-angular-core-interceptors.mjs.map +1 -1
  88. package/fesm2022/sd-angular-core-modules-authom.mjs +1 -0
  89. package/fesm2022/sd-angular-core-modules-authom.mjs.map +1 -1
  90. package/fesm2022/sd-angular-core-modules-keycloak.mjs +1 -0
  91. package/fesm2022/sd-angular-core-modules-keycloak.mjs.map +1 -1
  92. package/fesm2022/sd-angular-core-modules-layout.mjs +47 -46
  93. package/fesm2022/sd-angular-core-modules-layout.mjs.map +1 -1
  94. package/fesm2022/sd-angular-core-services-confirm.mjs +15 -13
  95. package/fesm2022/sd-angular-core-services-confirm.mjs.map +1 -1
  96. package/fesm2022/sd-angular-core-services-docx.mjs +7 -7
  97. package/fesm2022/sd-angular-core-services-docx.mjs.map +1 -1
  98. package/fesm2022/sd-angular-core-services-excel.mjs +5 -3
  99. package/fesm2022/sd-angular-core-services-excel.mjs.map +1 -1
  100. package/fesm2022/sd-angular-core-services-storage.mjs.map +1 -1
  101. package/fesm2022/sd-angular-core-utilities-extensions.mjs +21 -10
  102. package/fesm2022/sd-angular-core-utilities-extensions.mjs.map +1 -1
  103. package/fesm2022/sd-angular-core-utilities-models.mjs +38 -24
  104. package/fesm2022/sd-angular-core-utilities-models.mjs.map +1 -1
  105. package/fesm2022/sd-angular-core.mjs +1 -0
  106. package/fesm2022/sd-angular-core.mjs.map +1 -1
  107. package/i18n/index.d.ts +5 -0
  108. package/i18n/src/en.d.ts +2 -0
  109. package/i18n/src/sd-i18n.messages.d.ts +2 -0
  110. package/i18n/src/sd-i18n.pipe.d.ts +9 -0
  111. package/i18n/src/sd-i18n.service.d.ts +12 -0
  112. package/i18n/src/sd-i18n.token.d.ts +1 -0
  113. package/i18n/src/sd-i18n.types.d.ts +5 -0
  114. package/i18n/src/vi.d.ts +312 -0
  115. package/package.json +52 -40
  116. package/public-api.d.ts +1 -0
  117. package/sd-angular-core-19.0.0-beta.94.tgz +0 -0
  118. package/services/confirm/src/lib/confirm.service.d.ts +1 -0
  119. package/utilities/extensions/src/string.extension.d.ts +2 -0
  120. package/utilities/models/src/pattern.model.d.ts +1 -1
  121. package/sd-angular-core-19.0.0-beta.92.tgz +0 -0
  122. /package/components/{workflow → form-generic}/src/components/form-builder/components/attribute-expression/attribute-expression.component.d.ts +0 -0
  123. /package/components/{workflow → form-generic}/src/components/form-builder/components/attribute-input/attribute-input.component.d.ts +0 -0
  124. /package/components/{workflow → form-generic}/src/components/form-builder/components/attribute-input-number/attribute-input-number.component.d.ts +0 -0
  125. /package/components/{workflow → form-generic}/src/components/form-builder/components/attribute-parameter/attribute-parameter.component.d.ts +0 -0
  126. /package/components/{workflow → form-generic}/src/components/form-builder/components/attribute-select/attribute-select.component.d.ts +0 -0
  127. /package/components/{workflow → form-generic}/src/components/form-builder/components/attribute-selection/attribute-selection.component.d.ts +0 -0
  128. /package/components/{workflow → form-generic}/src/components/form-builder/components/attribute-selection/components/build-queries/build-queries.component.d.ts +0 -0
  129. /package/components/{workflow → form-generic}/src/components/form-builder/components/attribute-selection/components/build-variables/build-variables.component.d.ts +0 -0
  130. /package/components/{workflow → form-generic}/src/components/form-builder/components/attribute-switch/attribute-switch.component.d.ts +0 -0
  131. /package/components/{workflow → form-generic}/src/components/form-builder/components/attribute-textarea/attribute-textarea.component.d.ts +0 -0
  132. /package/components/{workflow → form-generic}/src/components/form-builder/components/checkbox/attribute/checkbox-attribute.component.d.ts +0 -0
  133. /package/components/{workflow → form-generic}/src/components/form-builder/components/checkbox/control/checkbox-control.component.d.ts +0 -0
  134. /package/components/{workflow → form-generic}/src/components/form-builder/components/checkbox/index.d.ts +0 -0
  135. /package/components/{workflow → form-generic}/src/components/form-builder/components/chip-calendar/attribute/chip-calendar-attribute.component.d.ts +0 -0
  136. /package/components/{workflow → form-generic}/src/components/form-builder/components/chip-calendar/control/chip-calendar-control.component.d.ts +0 -0
  137. /package/components/{workflow → form-generic}/src/components/form-builder/components/chip-calendar/index.d.ts +0 -0
  138. /package/components/{workflow → form-generic}/src/components/form-builder/components/chip-string/attribute/chip-string-attribute.component.d.ts +0 -0
  139. /package/components/{workflow → form-generic}/src/components/form-builder/components/chip-string/control/chip-string-control.component.d.ts +0 -0
  140. /package/components/{workflow → form-generic}/src/components/form-builder/components/chip-string/index.d.ts +0 -0
  141. /package/components/{workflow → form-generic}/src/components/form-builder/components/datetime/attribute/datetime-attribute.component.d.ts +0 -0
  142. /package/components/{workflow → form-generic}/src/components/form-builder/components/datetime/control/datetime-control.component.d.ts +0 -0
  143. /package/components/{workflow → form-generic}/src/components/form-builder/components/datetime/index.d.ts +0 -0
  144. /package/components/{workflow → form-generic}/src/components/form-builder/components/html/attribute/components/build-queries/build-queries.component.d.ts +0 -0
  145. /package/components/{workflow → form-generic}/src/components/form-builder/components/html/attribute/html-attribute.component.d.ts +0 -0
  146. /package/components/{workflow → form-generic}/src/components/form-builder/components/html/control/html-control.component.d.ts +0 -0
  147. /package/components/{workflow → form-generic}/src/components/form-builder/components/html/index.d.ts +0 -0
  148. /package/components/{workflow → form-generic}/src/components/form-builder/components/index.d.ts +0 -0
  149. /package/components/{workflow → form-generic}/src/components/form-builder/components/number/attribute/number-attribute.component.d.ts +0 -0
  150. /package/components/{workflow → form-generic}/src/components/form-builder/components/number/control/number-control.component.d.ts +0 -0
  151. /package/components/{workflow → form-generic}/src/components/form-builder/components/number/index.d.ts +0 -0
  152. /package/components/{workflow → form-generic}/src/components/form-builder/components/radio/attribute/radio-attribute.component.d.ts +0 -0
  153. /package/components/{workflow → form-generic}/src/components/form-builder/components/radio/control/radio-control.component.d.ts +0 -0
  154. /package/components/{workflow → form-generic}/src/components/form-builder/components/radio/index.d.ts +0 -0
  155. /package/components/{workflow → form-generic}/src/components/form-builder/components/select/attribute/select-attribute.component.d.ts +0 -0
  156. /package/components/{workflow → form-generic}/src/components/form-builder/components/select/control/select-control.component.d.ts +0 -0
  157. /package/components/{workflow → form-generic}/src/components/form-builder/components/select/index.d.ts +0 -0
  158. /package/components/{workflow → form-generic}/src/components/form-builder/components/table/attribute/table-attribute.component.d.ts +0 -0
  159. /package/components/{workflow → form-generic}/src/components/form-builder/components/table/control/table-control.component.d.ts +0 -0
  160. /package/components/{workflow → form-generic}/src/components/form-builder/components/table/index.d.ts +0 -0
  161. /package/components/{workflow → form-generic}/src/components/form-builder/components/textarea/attribute/textarea-attribute.component.d.ts +0 -0
  162. /package/components/{workflow → form-generic}/src/components/form-builder/components/textarea/control/textarea-control.component.d.ts +0 -0
  163. /package/components/{workflow → form-generic}/src/components/form-builder/components/textarea/index.d.ts +0 -0
  164. /package/components/{workflow → form-generic}/src/components/form-builder/components/textfield/attribute/textfield-attribute.component.d.ts +0 -0
  165. /package/components/{workflow → form-generic}/src/components/form-builder/components/textfield/control/textfield-control.component.d.ts +0 -0
  166. /package/components/{workflow → form-generic}/src/components/form-builder/components/textfield/index.d.ts +0 -0
  167. /package/components/{workflow → form-generic}/src/components/form-builder/components/upload/attribute/upload-attribute.component.d.ts +0 -0
  168. /package/components/{workflow → form-generic}/src/components/form-builder/components/upload/control/upload-control.component.d.ts +0 -0
  169. /package/components/{workflow → form-generic}/src/components/form-builder/components/upload/index.d.ts +0 -0
  170. /package/components/{workflow → form-generic}/src/components/form-builder/form-builder.component.d.ts +0 -0
  171. /package/components/{workflow → form-generic}/src/components/form-builder/services/builder.service.d.ts +0 -0
  172. /package/components/{workflow → form-generic}/src/components/form-builder/services/index.d.ts +0 -0
  173. /package/components/{workflow → form-generic}/src/components/form-render/components/index.d.ts +0 -0
  174. /package/components/{workflow → form-generic}/src/components/form-render/components/item/components/chip-calendar/chip-calendar.component.d.ts +0 -0
  175. /package/components/{workflow → form-generic}/src/components/form-render/components/item/components/chip-string/chip-string.component.d.ts +0 -0
  176. /package/components/{workflow → form-generic}/src/components/form-render/components/item/components/datetime/datetime.component.d.ts +0 -0
  177. /package/components/{workflow → form-generic}/src/components/form-render/components/item/components/html/html.component.d.ts +0 -0
  178. /package/components/{workflow → form-generic}/src/components/form-render/components/item/components/index.d.ts +0 -0
  179. /package/components/{workflow → form-generic}/src/components/form-render/components/item/components/number/number.component.d.ts +0 -0
  180. /package/components/{workflow → form-generic}/src/components/form-render/components/item/components/radio/radio.component.d.ts +0 -0
  181. /package/components/{workflow → form-generic}/src/components/form-render/components/item/components/select/select.component.d.ts +0 -0
  182. /package/components/{workflow → form-generic}/src/components/form-render/components/item/components/table/table.component.d.ts +0 -0
  183. /package/components/{workflow → form-generic}/src/components/form-render/components/item/components/textarea/textarea.component.d.ts +0 -0
  184. /package/components/{workflow → form-generic}/src/components/form-render/components/item/components/textfield/textfield.component.d.ts +0 -0
  185. /package/components/{workflow → form-generic}/src/components/form-render/components/item/components/upload/upload.component.d.ts +0 -0
  186. /package/components/{workflow → form-generic}/src/components/form-render/components/item/item.component.d.ts +0 -0
  187. /package/components/{workflow → form-generic}/src/components/form-render/components/variable/variable.component.d.ts +0 -0
  188. /package/components/{workflow → form-generic}/src/components/index.d.ts +0 -0
  189. /package/components/{workflow → form-generic}/src/configurations/form.configuration.d.ts +0 -0
  190. /package/components/{workflow → form-generic}/src/models/form-generic-component.model.d.ts +0 -0
  191. /package/components/{workflow → form-generic}/src/models/form-generic-definition-html.model.d.ts +0 -0
  192. /package/components/{workflow → form-generic}/src/models/form-generic-definition-selection.model.d.ts +0 -0
  193. /package/components/{workflow → form-generic}/src/models/form-generic-definition-table.model.d.ts +0 -0
  194. /package/components/{workflow → form-generic}/src/models/form-generic-expression.model.d.ts +0 -0
  195. /package/components/{workflow → form-generic}/src/models/form-generic-template.model.d.ts +0 -0
  196. /package/components/{workflow → form-generic}/src/models/form-generic-validation.model.d.ts +0 -0
  197. /package/components/{workflow → form-generic}/src/models/form-generic.model.d.ts +0 -0
  198. /package/components/{workflow → form-generic}/src/models/form-render/form-render-args.model.d.ts +0 -0
  199. /package/components/{workflow → form-generic}/src/models/form-render/form-render-entity.model.d.ts +0 -0
  200. /package/components/{workflow → form-generic}/src/models/form-render/index.d.ts +0 -0
  201. /package/components/{workflow → form-generic}/src/models/index.d.ts +0 -0
  202. /package/components/{workflow → form-generic}/src/pipes/component-viewed.pipe.d.ts +0 -0
  203. /package/components/{workflow → form-generic}/src/pipes/expression-feel.pipe.d.ts +0 -0
  204. /package/components/{workflow → form-generic}/src/pipes/expression-query.pipe.d.ts +0 -0
  205. /package/components/{workflow → form-generic}/src/pipes/expression-view.pipe.d.ts +0 -0
  206. /package/components/{workflow → form-generic}/src/pipes/html.pipe.d.ts +0 -0
  207. /package/components/{workflow → form-generic}/src/pipes/hyperlink.pipe.d.ts +0 -0
  208. /package/components/{workflow → form-generic}/src/pipes/index.d.ts +0 -0
  209. /package/components/{workflow → form-generic}/src/pipes/when-expression.pipe.d.ts +0 -0
  210. /package/components/{workflow → form-generic}/src/services/form-render.service.d.ts +0 -0
  211. /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