@sd-angular/core 19.0.0-beta.5 → 19.0.0-beta.50

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 (185) hide show
  1. package/README.md +686 -33
  2. package/assets/scss/ckeditor5.scss +60 -2
  3. package/assets/scss/core/bootstrap.scss +17 -0
  4. package/assets/scss/core/grid.scss +40 -0
  5. package/assets/scss/sd-core.scss +1 -0
  6. package/components/avatar/index.d.ts +1 -0
  7. package/components/avatar/src/avatar.component.d.ts +15 -0
  8. package/components/badge/src/badge.component.d.ts +77 -19
  9. package/components/button/src/button.component.d.ts +26 -28
  10. package/components/code-editor/index.d.ts +1 -0
  11. package/components/code-editor/src/code-editor.component.d.ts +25 -0
  12. package/components/document-builder/index.d.ts +1 -0
  13. package/components/document-builder/src/document-builder.component.d.ts +18 -36
  14. package/components/document-builder/src/document-builder.model.d.ts +10 -10
  15. package/components/document-builder/src/plugins/block-space/block-space.plugin.d.ts +9 -0
  16. package/components/document-builder/src/plugins/ck-comment/ck-comment.plugin.d.ts +44 -0
  17. package/components/document-builder/src/plugins/ck-comment/ck-comment.plugin.model.d.ts +56 -0
  18. package/components/document-builder/src/plugins/heading/heading.plugin.d.ts +1 -0
  19. package/components/document-builder/src/plugins/highlight-range/highlight-range.plugin.d.ts +4 -0
  20. package/components/document-builder/src/plugins/image-custom/image-custom.plugin.d.ts +31 -0
  21. package/components/document-builder/src/plugins/index.d.ts +7 -2
  22. package/components/document-builder/src/plugins/page-orientation/page-orientation.plugin.d.ts +2 -2
  23. package/components/document-builder/src/plugins/paste-handler/filters/bookmark.d.ts +14 -0
  24. package/components/document-builder/src/plugins/paste-handler/filters/br.d.ts +15 -0
  25. package/components/document-builder/src/plugins/paste-handler/filters/image.d.ts +25 -0
  26. package/components/document-builder/src/plugins/paste-handler/filters/list.d.ts +29 -0
  27. package/components/document-builder/src/plugins/paste-handler/filters/parse.d.ts +35 -0
  28. package/components/document-builder/src/plugins/paste-handler/filters/removeboldwrapper.d.ts +15 -0
  29. package/components/document-builder/src/plugins/paste-handler/filters/removegooglesheetstag.d.ts +15 -0
  30. package/components/document-builder/src/plugins/paste-handler/filters/removeinvalidtablewidth.d.ts +15 -0
  31. package/components/document-builder/src/plugins/paste-handler/filters/removemsattributes.d.ts +15 -0
  32. package/components/document-builder/src/plugins/paste-handler/filters/removestyleblock.d.ts +15 -0
  33. package/components/document-builder/src/plugins/paste-handler/filters/removexmlns.d.ts +15 -0
  34. package/components/document-builder/src/plugins/paste-handler/filters/replacemsfootnotes.d.ts +54 -0
  35. package/components/document-builder/src/plugins/paste-handler/filters/replacetabswithinprewithspaces.d.ts +24 -0
  36. package/components/document-builder/src/plugins/paste-handler/filters/space.d.ts +27 -0
  37. package/components/document-builder/src/plugins/paste-handler/filters/table.d.ts +16 -0
  38. package/components/document-builder/src/plugins/paste-handler/filters/utils.d.ts +25 -0
  39. package/components/document-builder/src/plugins/paste-handler/index.d.ts +35 -0
  40. package/components/document-builder/src/plugins/paste-handler/normalizers/googledocsnormalizer.d.ts +31 -0
  41. package/components/document-builder/src/plugins/paste-handler/normalizers/googlesheetsnormalizer.d.ts +31 -0
  42. package/components/document-builder/src/plugins/paste-handler/normalizers/mswordnormalizer.d.ts +29 -0
  43. package/components/document-builder/src/plugins/paste-handler/types.d.ts +30 -0
  44. package/components/document-builder/src/plugins/table-custom/index.d.ts +34 -0
  45. package/components/index.d.ts +4 -0
  46. package/components/mini-editor/index.d.ts +2 -0
  47. package/components/mini-editor/src/mini-editor.component.d.ts +90 -0
  48. package/components/mini-editor/src/mini-editor.model.d.ts +44 -0
  49. package/components/side-drawer/src/side-drawer.component.d.ts +1 -2
  50. package/components/tab-router/src/components/tab-router-item/tab-router-item.component.d.ts +4 -1
  51. package/components/tab-router/src/components/tab-router-outlet/tab-router-outlet.component.d.ts +3 -15
  52. package/components/table/src/components/selector-action/action-filter.pipe.d.ts +11 -10
  53. package/components/table/src/directives/index.d.ts +2 -0
  54. package/components/table/src/directives/sd-table-column-filter-def.directive.d.ts +9 -0
  55. package/components/table/src/directives/sticky-shadow.directive.d.ts +17 -0
  56. package/components/table/src/models/table-column.model.d.ts +32 -32
  57. package/components/table/src/models/table-command.model.d.ts +7 -3
  58. package/components/table/src/models/table-item.model.d.ts +5 -4
  59. package/components/table/src/models/table-option-export.model.d.ts +3 -2
  60. package/components/table/src/models/table-option-selector.model.d.ts +11 -10
  61. package/components/table/src/models/table-option.model.d.ts +10 -8
  62. package/components/table/src/services/table-filter/table-filter.model.d.ts +2 -2
  63. package/components/table/src/table.component.d.ts +33 -35
  64. package/components/view/index.d.ts +1 -0
  65. package/components/view/src/view.component.d.ts +16 -0
  66. package/components/workflow/src/models/index.d.ts +1 -0
  67. package/directives/index.d.ts +1 -0
  68. package/directives/src/sd-href.directive.d.ts +9 -0
  69. package/fesm2022/sd-angular-core-components-avatar.mjs +90 -0
  70. package/fesm2022/sd-angular-core-components-avatar.mjs.map +1 -0
  71. package/fesm2022/sd-angular-core-components-badge.mjs +101 -91
  72. package/fesm2022/sd-angular-core-components-badge.mjs.map +1 -1
  73. package/fesm2022/sd-angular-core-components-button.mjs +64 -96
  74. package/fesm2022/sd-angular-core-components-button.mjs.map +1 -1
  75. package/fesm2022/sd-angular-core-components-code-editor.mjs +129 -0
  76. package/fesm2022/sd-angular-core-components-code-editor.mjs.map +1 -0
  77. package/fesm2022/sd-angular-core-components-document-builder.mjs +3635 -521
  78. package/fesm2022/sd-angular-core-components-document-builder.mjs.map +1 -1
  79. package/fesm2022/sd-angular-core-components-history.mjs +1 -1
  80. package/fesm2022/sd-angular-core-components-history.mjs.map +1 -1
  81. package/fesm2022/sd-angular-core-components-import-excel.mjs +1 -1
  82. package/fesm2022/sd-angular-core-components-import-excel.mjs.map +1 -1
  83. package/fesm2022/sd-angular-core-components-mini-editor.mjs +326 -0
  84. package/fesm2022/sd-angular-core-components-mini-editor.mjs.map +1 -0
  85. package/fesm2022/sd-angular-core-components-preview.mjs +1 -1
  86. package/fesm2022/sd-angular-core-components-preview.mjs.map +1 -1
  87. package/fesm2022/sd-angular-core-components-side-drawer.mjs +21 -8
  88. package/fesm2022/sd-angular-core-components-side-drawer.mjs.map +1 -1
  89. package/fesm2022/sd-angular-core-components-tab-router.mjs +152 -226
  90. package/fesm2022/sd-angular-core-components-tab-router.mjs.map +1 -1
  91. package/fesm2022/sd-angular-core-components-table.mjs +646 -460
  92. package/fesm2022/sd-angular-core-components-table.mjs.map +1 -1
  93. package/fesm2022/sd-angular-core-components-upload-file.mjs +1 -1
  94. package/fesm2022/sd-angular-core-components-upload-file.mjs.map +1 -1
  95. package/fesm2022/sd-angular-core-components-view.mjs +45 -0
  96. package/fesm2022/sd-angular-core-components-view.mjs.map +1 -0
  97. package/fesm2022/sd-angular-core-components-workflow.mjs +33 -43
  98. package/fesm2022/sd-angular-core-components-workflow.mjs.map +1 -1
  99. package/fesm2022/sd-angular-core-components.mjs +4 -0
  100. package/fesm2022/sd-angular-core-components.mjs.map +1 -1
  101. package/fesm2022/sd-angular-core-directives.mjs +80 -27
  102. package/fesm2022/sd-angular-core-directives.mjs.map +1 -1
  103. package/fesm2022/sd-angular-core-forms-autocomplete.mjs +257 -363
  104. package/fesm2022/sd-angular-core-forms-autocomplete.mjs.map +1 -1
  105. package/fesm2022/sd-angular-core-forms-date-range.mjs +145 -245
  106. package/fesm2022/sd-angular-core-forms-date-range.mjs.map +1 -1
  107. package/fesm2022/sd-angular-core-forms-date.mjs +137 -273
  108. package/fesm2022/sd-angular-core-forms-date.mjs.map +1 -1
  109. package/fesm2022/sd-angular-core-forms-datetime.mjs +136 -288
  110. package/fesm2022/sd-angular-core-forms-datetime.mjs.map +1 -1
  111. package/fesm2022/sd-angular-core-forms-input-number.mjs +174 -337
  112. package/fesm2022/sd-angular-core-forms-input-number.mjs.map +1 -1
  113. package/fesm2022/sd-angular-core-forms-input.mjs +130 -285
  114. package/fesm2022/sd-angular-core-forms-input.mjs.map +1 -1
  115. package/fesm2022/sd-angular-core-forms-radio.mjs +3 -2
  116. package/fesm2022/sd-angular-core-forms-radio.mjs.map +1 -1
  117. package/fesm2022/sd-angular-core-forms-select.mjs +336 -421
  118. package/fesm2022/sd-angular-core-forms-select.mjs.map +1 -1
  119. package/fesm2022/sd-angular-core-forms-textarea.mjs +121 -226
  120. package/fesm2022/sd-angular-core-forms-textarea.mjs.map +1 -1
  121. package/fesm2022/sd-angular-core-modules-keycloak.mjs +126 -0
  122. package/fesm2022/sd-angular-core-modules-keycloak.mjs.map +1 -0
  123. package/fesm2022/sd-angular-core-modules-layout.mjs +440 -431
  124. package/fesm2022/sd-angular-core-modules-layout.mjs.map +1 -1
  125. package/fesm2022/sd-angular-core-modules.mjs +1 -1
  126. package/fesm2022/sd-angular-core-services-confirm.mjs +2 -2
  127. package/fesm2022/sd-angular-core-services-confirm.mjs.map +1 -1
  128. package/fesm2022/sd-angular-core-services-docx.mjs +173 -0
  129. package/fesm2022/sd-angular-core-services-docx.mjs.map +1 -0
  130. package/fesm2022/sd-angular-core-services-notify.mjs +2 -2
  131. package/fesm2022/sd-angular-core-services-notify.mjs.map +1 -1
  132. package/fesm2022/sd-angular-core-services.mjs +1 -0
  133. package/fesm2022/sd-angular-core-services.mjs.map +1 -1
  134. package/fesm2022/sd-angular-core-utilities-extensions.mjs +10 -6
  135. package/fesm2022/sd-angular-core-utilities-extensions.mjs.map +1 -1
  136. package/fesm2022/sd-angular-core-utilities-models.mjs +15 -1
  137. package/fesm2022/sd-angular-core-utilities-models.mjs.map +1 -1
  138. package/forms/autocomplete/src/autocomplete.component.d.ts +51 -56
  139. package/forms/date/src/date.component.d.ts +41 -45
  140. package/forms/date-range/src/date-range.component.d.ts +28 -33
  141. package/forms/datetime/src/datetime.component.d.ts +41 -48
  142. package/forms/input/src/input.component.d.ts +46 -56
  143. package/forms/input-number/src/input-number.component.d.ts +47 -54
  144. package/forms/select/src/select.component.d.ts +58 -58
  145. package/forms/textarea/src/textarea.component.d.ts +34 -41
  146. package/modules/index.d.ts +1 -1
  147. package/modules/keycloak/index.d.ts +4 -0
  148. package/modules/keycloak/keycloak.configuration.d.ts +11 -0
  149. package/modules/keycloak/keycloak.interceptor.d.ts +2 -0
  150. package/modules/keycloak/keycloak.module.d.ts +18 -0
  151. package/modules/keycloak/keycloak.service.d.ts +14 -0
  152. package/modules/layout/components/layout-main/layout-main.component.d.ts +7 -12
  153. package/modules/layout/components/sidebar-v1/components/sidebar/sidebar.component.d.ts +22 -29
  154. package/modules/layout/components/sidebar-v1/components/user/user.component.d.ts +11 -17
  155. package/modules/layout/components/sidebar-v1/main.component.d.ts +14 -14
  156. package/modules/layout/configurations/layout.configuration.d.ts +46 -3
  157. package/modules/layout/modules/forbidden/pages/root/root.component.d.ts +3 -8
  158. package/modules/layout/modules/home/components/home-page/home-page.component.d.ts +2 -5
  159. package/modules/layout/modules/not-found/pages/root/root.component.d.ts +3 -8
  160. package/modules/layout/services/index.d.ts +1 -0
  161. package/modules/layout/services/layout.service.d.ts +10 -0
  162. package/modules/layout/services/menu/menu.model.d.ts +2 -0
  163. package/modules/layout/services/storage/storage.service.d.ts +0 -3
  164. package/package.json +91 -68
  165. package/services/confirm/src/lib/confirm.service.d.ts +1 -0
  166. package/services/docx/index.d.ts +1 -0
  167. package/services/docx/src/lib/docx.model.d.ts +9 -0
  168. package/services/docx/src/lib/docx.service.d.ts +13 -0
  169. package/services/docx/src/public-api.d.ts +2 -0
  170. package/services/index.d.ts +1 -0
  171. package/utilities/extensions/src/string.extension.d.ts +2 -0
  172. package/utilities/models/index.d.ts +3 -0
  173. package/utilities/models/src/filter.model.d.ts +14 -2
  174. package/utilities/models/src/icon.model.d.ts +2 -0
  175. package/utilities/models/src/nested-key-of.model.d.ts +5 -0
  176. package/utilities/models/src/pattern.model.d.ts +1 -1
  177. package/utilities/models/src/unwrap-signal.model.d.ts +6 -0
  178. package/components/document-builder/src/plugins/comment/comment.plugin.d.ts +0 -4
  179. package/components/document-builder/src/plugins/table-fit/table-fit.plugin.d.ts +0 -4
  180. package/fesm2022/sd-angular-core-modules-oidc.mjs +0 -127
  181. package/fesm2022/sd-angular-core-modules-oidc.mjs.map +0 -1
  182. package/modules/oidc/dynamic-sts.loader.d.ts +0 -11
  183. package/modules/oidc/index.d.ts +0 -2
  184. package/modules/oidc/oidc.configuration.d.ts +0 -11
  185. package/modules/oidc/oidc.module.d.ts +0 -14
@@ -0,0 +1,129 @@
1
+ import { Clipboard } from '@angular/cdk/clipboard';
2
+ import { CommonModule } from '@angular/common';
3
+ import * as i0 from '@angular/core';
4
+ import { inject, model, input, booleanAttribute, signal, computed, effect, untracked, ViewEncapsulation, ChangeDetectionStrategy, Component } from '@angular/core';
5
+ import * as i3 from '@angular/forms';
6
+ import { FormsModule } from '@angular/forms';
7
+ import * as i1 from '@angular/material/icon';
8
+ import { MatIconModule } from '@angular/material/icon';
9
+ import * as i2 from '@angular/material/tooltip';
10
+ import { MatTooltipModule } from '@angular/material/tooltip';
11
+ import { DomSanitizer } from '@angular/platform-browser';
12
+ import * as Prism from 'prismjs';
13
+ import 'prismjs/components/prism-typescript';
14
+ import 'prismjs/components/prism-json';
15
+ import 'prismjs/components/prism-css';
16
+ import 'prismjs/components/prism-scss';
17
+ import 'prismjs/components/prism-markup';
18
+
19
+ class SdCodeEditor {
20
+ #clipboard = inject(Clipboard);
21
+ #sanitizer = inject(DomSanitizer);
22
+ // ==========================================
23
+ // 1. SIGNAL INPUTS & MODEL
24
+ // ==========================================
25
+ // Nhận bất kỳ kiểu dữ liệu nào (string, array, object)
26
+ valueModel = model(undefined, { alias: 'model' });
27
+ language = input('typescript');
28
+ maxHeight = input('500px');
29
+ // Trạng thái Viewed (true = Read Only, false = Editable)
30
+ viewed = input(false, { transform: booleanAttribute });
31
+ // ==========================================
32
+ // 2. INTERNAL STATE
33
+ // ==========================================
34
+ copied = signal(false);
35
+ // Chuỗi text nội bộ dùng để map với thẻ <textarea>
36
+ textValue = signal('');
37
+ prismLang = computed(() => this.language() === 'html' ? 'markup' : this.language());
38
+ // Cờ lưu vết để chống Loop (Vòng lặp vô tận khi bắn 2 chiều)
39
+ _lastEmittedValue = undefined;
40
+ constructor() {
41
+ // ==========================================
42
+ // EFFECT 1: Dữ liệu từ CHA truyền vào (Model -> TextValue)
43
+ // ==========================================
44
+ effect(() => {
45
+ const extVal = this.valueModel();
46
+ const lang = this.language();
47
+ untracked(() => {
48
+ // Nếu giá trị này do chính component bắn ra, bỏ qua để tránh loop
49
+ if (extVal === this._lastEmittedValue)
50
+ return;
51
+ if (typeof extVal === 'string') {
52
+ this.textValue.set(extVal);
53
+ }
54
+ else if (extVal !== undefined && extVal !== null) {
55
+ // Tự động format Object -> String nếu là JSON
56
+ if (lang === 'json') {
57
+ try {
58
+ this.textValue.set(JSON.stringify(extVal, null, 2));
59
+ }
60
+ catch {
61
+ this.textValue.set('// Lỗi: Object có tham chiếu vòng (Circular Reference)');
62
+ }
63
+ }
64
+ else {
65
+ this.textValue.set(String(extVal));
66
+ }
67
+ }
68
+ else {
69
+ this.textValue.set('');
70
+ }
71
+ });
72
+ });
73
+ }
74
+ // ==========================================
75
+ // EFFECT 2: PrismJS render (TextValue -> HTML MÀU)
76
+ // ==========================================
77
+ highlightedCode = computed(() => {
78
+ // Dùng khoảng trắng để giữ độ cao cho thẻ pre nếu rỗng
79
+ const rawCode = this.textValue() || ' ';
80
+ const langKey = this.prismLang();
81
+ const grammar = Prism.languages[langKey] || Prism.languages['markup'];
82
+ // Cộng thêm \n ở cuối để chống lỗi con trỏ textarea ăn lẹm dòng cuối
83
+ const highlightedString = Prism.highlight(rawCode, grammar, langKey) + '\n';
84
+ return this.#sanitizer.bypassSecurityTrustHtml(highlightedString);
85
+ });
86
+ // ==========================================
87
+ // EVENTS
88
+ // ==========================================
89
+ // Khi người dùng gõ vào Textarea (TextValue -> Model)
90
+ onTextChange(newText) {
91
+ this.textValue.set(newText);
92
+ let valToEmit = newText;
93
+ // Nếu ngôn ngữ là JSON, cố gắng trả về Object thật
94
+ if (this.language() === 'json') {
95
+ try {
96
+ valToEmit = JSON.parse(newText);
97
+ }
98
+ catch {
99
+ // Nếu gõ dở ngoặc/sai cú pháp -> Trả về chuỗi String tạm
100
+ valToEmit = newText;
101
+ }
102
+ }
103
+ // Ghi sổ và bắn ra ngoài
104
+ this._lastEmittedValue = valToEmit;
105
+ this.valueModel.set(valToEmit);
106
+ }
107
+ copyToClipboard() {
108
+ const rawCode = this.textValue();
109
+ if (!rawCode)
110
+ return;
111
+ if (this.#clipboard.copy(rawCode)) {
112
+ this.copied.set(true);
113
+ setTimeout(() => this.copied.set(false), 2000);
114
+ }
115
+ }
116
+ static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "19.2.17", ngImport: i0, type: SdCodeEditor, deps: [], target: i0.ɵɵFactoryTarget.Component });
117
+ static ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "19.2.17", type: SdCodeEditor, isStandalone: true, selector: "sd-code-editor", inputs: { valueModel: { classPropertyName: "valueModel", publicName: "model", isSignal: true, isRequired: false, transformFunction: null }, language: { classPropertyName: "language", publicName: "language", isSignal: true, isRequired: false, transformFunction: null }, maxHeight: { classPropertyName: "maxHeight", publicName: "maxHeight", isSignal: true, isRequired: false, transformFunction: null }, viewed: { classPropertyName: "viewed", publicName: "viewed", isSignal: true, isRequired: false, transformFunction: null } }, outputs: { valueModel: "modelChange" }, ngImport: i0, template: "<div class=\"sd-code-wrapper\" [class.is-editable]=\"!viewed()\">\r\n \r\n <div class=\"sd-code-header\">\r\n <div class=\"mac-dots\">\r\n <span class=\"dot red\"></span>\r\n <span class=\"dot yellow\"></span>\r\n <span class=\"dot green\"></span>\r\n </div>\r\n \r\n <div class=\"lang-badge\">\r\n {{ language().toUpperCase() }} \r\n <span style=\"opacity: 0.6; margin-left: 4px; font-weight: normal;\">\r\n {{ viewed() ? '(READ ONLY)' : '(EDITING)' }}\r\n </span>\r\n </div>\r\n \r\n <button class=\"copy-btn\" [matTooltip]=\"copied() ? '\u0110\u00E3 copy!' : 'Copy code'\" (click)=\"copyToClipboard()\">\r\n <mat-icon [class.text-success]=\"copied()\">{{ copied() ? 'check' : 'content_copy' }}</mat-icon>\r\n <span>{{ copied() ? 'Copied' : 'Copy' }}</span>\r\n </button>\r\n </div>\r\n\r\n <div class=\"sd-code-content\" [style.maxHeight]=\"maxHeight()\">\r\n <div class=\"editor-overlay-container\">\r\n \r\n @if (!viewed()) {\r\n <textarea \r\n class=\"code-textarea\"\r\n [ngModel]=\"textValue()\"\r\n (ngModelChange)=\"onTextChange($event)\"\r\n spellcheck=\"false\"\r\n autocomplete=\"off\"\r\n autocorrect=\"off\"\r\n autocapitalize=\"off\">\r\n </textarea>\r\n }\r\n \r\n <pre aria-hidden=\"true\" class=\"code-display\"><code [class]=\"'language-' + prismLang()\" [innerHTML]=\"highlightedCode()\"></code></pre>\r\n \r\n </div>\r\n </div>\r\n</div>", styles: ["@charset \"UTF-8\";.sd-code-wrapper{background-color:#1e1e1e;border-radius:8px;overflow:hidden;box-shadow:0 4px 12px #00000026;margin:16px 0;font-family:Fira Code,Consolas,Monaco,monospace}.sd-code-wrapper .sd-code-header{display:flex;align-items:center;justify-content:space-between;background-color:#2d2d2d;padding:8px 16px;border-bottom:1px solid #404040}.sd-code-wrapper .sd-code-header .mac-dots{display:flex;gap:6px}.sd-code-wrapper .sd-code-header .mac-dots .dot{width:12px;height:12px;border-radius:50%}.sd-code-wrapper .sd-code-header .mac-dots .dot.red{background-color:#ff5f56}.sd-code-wrapper .sd-code-header .mac-dots .dot.yellow{background-color:#ffbd2e}.sd-code-wrapper .sd-code-header .mac-dots .dot.green{background-color:#27c93f}.sd-code-wrapper .sd-code-header .lang-badge{color:#858585;font-size:12px;font-weight:600;letter-spacing:.5px}.sd-code-wrapper .sd-code-header .copy-btn{display:flex;align-items:center;gap:4px;background:none;border:none;color:#ccc;cursor:pointer;font-size:13px;padding:4px 8px;border-radius:4px;transition:all .2s ease}.sd-code-wrapper .sd-code-header .copy-btn mat-icon{font-size:16px;width:16px;height:16px}.sd-code-wrapper .sd-code-header .copy-btn:hover{background-color:#404040;color:#fff}.sd-code-wrapper .sd-code-header .copy-btn .text-success{color:#27c93f!important}.sd-code-wrapper .sd-code-content{overflow-y:auto;position:relative}.sd-code-wrapper .sd-code-content::-webkit-scrollbar{width:8px;height:8px}.sd-code-wrapper .sd-code-content::-webkit-scrollbar-thumb{background:#555;border-radius:4px}.sd-code-wrapper .sd-code-content::-webkit-scrollbar-thumb:hover{background:#777}.sd-code-wrapper .sd-code-content::-webkit-scrollbar-track{background:#1e1e1e}.sd-code-wrapper .sd-code-content .editor-overlay-container{position:relative;min-height:100px}.sd-code-wrapper .sd-code-content .code-textarea,.sd-code-wrapper .sd-code-content .code-display{margin:0;padding:16px;border:0;width:100%;min-height:100%;font-family:inherit;font-size:14px;line-height:1.5;white-space:pre-wrap;word-break:break-all;tab-size:2}.sd-code-wrapper .sd-code-content .code-textarea{position:absolute;top:0;left:0;z-index:2;background:transparent;color:transparent;caret-color:#fff;resize:none;outline:none;overflow:hidden}.sd-code-wrapper .sd-code-content .code-display{position:relative;z-index:1;pointer-events:none}.sd-code-wrapper .sd-code-content .code-display code{color:#d4d4d4;font-family:inherit}.sd-code-wrapper .token.comment,.sd-code-wrapper .token.block-comment,.sd-code-wrapper .token.prolog,.sd-code-wrapper .token.doctype,.sd-code-wrapper .token.cdata{color:#999}.sd-code-wrapper .token.punctuation{color:#ccc}.sd-code-wrapper .token.tag,.sd-code-wrapper .token.attr-name,.sd-code-wrapper .token.namespace,.sd-code-wrapper .token.deleted{color:#e2777a}.sd-code-wrapper .token.function-name{color:#6196cc}.sd-code-wrapper .token.boolean,.sd-code-wrapper .token.number,.sd-code-wrapper .token.function{color:#f08d49}.sd-code-wrapper .token.property,.sd-code-wrapper .token.class-name,.sd-code-wrapper .token.constant,.sd-code-wrapper .token.symbol{color:#f8c555}.sd-code-wrapper .token.selector,.sd-code-wrapper .token.important,.sd-code-wrapper .token.atrule,.sd-code-wrapper .token.keyword,.sd-code-wrapper .token.builtin{color:#cc99cd}.sd-code-wrapper .token.string,.sd-code-wrapper .token.char,.sd-code-wrapper .token.attr-value,.sd-code-wrapper .token.regex,.sd-code-wrapper .token.variable{color:#7ec699}.sd-code-wrapper .token.operator,.sd-code-wrapper .token.entity,.sd-code-wrapper .token.url{color:#67cdcc}.sd-code-wrapper .token.important,.sd-code-wrapper .token.bold{font-weight:700}.sd-code-wrapper .token.italic{font-style:italic}.sd-code-wrapper .token.entity{cursor:help}.sd-code-wrapper .token.inserted{color:green}\n"], dependencies: [{ kind: "ngmodule", type: CommonModule }, { kind: "ngmodule", type: MatIconModule }, { kind: "component", type: i1.MatIcon, selector: "mat-icon", inputs: ["color", "inline", "svgIcon", "fontSet", "fontIcon"], exportAs: ["matIcon"] }, { kind: "ngmodule", type: MatTooltipModule }, { kind: "directive", type: i2.MatTooltip, selector: "[matTooltip]", inputs: ["matTooltipPosition", "matTooltipPositionAtOrigin", "matTooltipDisabled", "matTooltipShowDelay", "matTooltipHideDelay", "matTooltipTouchGestures", "matTooltip", "matTooltipClass"], exportAs: ["matTooltip"] }, { kind: "ngmodule", type: FormsModule }, { kind: "directive", type: i3.DefaultValueAccessor, selector: "input:not([type=checkbox])[formControlName],textarea[formControlName],input:not([type=checkbox])[formControl],textarea[formControl],input:not([type=checkbox])[ngModel],textarea[ngModel],[ngDefaultControl]" }, { kind: "directive", type: i3.NgControlStatus, selector: "[formControlName],[ngModel],[formControl]" }, { kind: "directive", type: i3.NgModel, selector: "[ngModel]:not([formControlName]):not([formControl])", inputs: ["name", "disabled", "ngModel", "ngModelOptions"], outputs: ["ngModelChange"], exportAs: ["ngModel"] }], changeDetection: i0.ChangeDetectionStrategy.OnPush, encapsulation: i0.ViewEncapsulation.None });
118
+ }
119
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "19.2.17", ngImport: i0, type: SdCodeEditor, decorators: [{
120
+ type: Component,
121
+ args: [{ selector: 'sd-code-editor', standalone: true, imports: [CommonModule, MatIconModule, MatTooltipModule, FormsModule], changeDetection: ChangeDetectionStrategy.OnPush, encapsulation: ViewEncapsulation.None, template: "<div class=\"sd-code-wrapper\" [class.is-editable]=\"!viewed()\">\r\n \r\n <div class=\"sd-code-header\">\r\n <div class=\"mac-dots\">\r\n <span class=\"dot red\"></span>\r\n <span class=\"dot yellow\"></span>\r\n <span class=\"dot green\"></span>\r\n </div>\r\n \r\n <div class=\"lang-badge\">\r\n {{ language().toUpperCase() }} \r\n <span style=\"opacity: 0.6; margin-left: 4px; font-weight: normal;\">\r\n {{ viewed() ? '(READ ONLY)' : '(EDITING)' }}\r\n </span>\r\n </div>\r\n \r\n <button class=\"copy-btn\" [matTooltip]=\"copied() ? '\u0110\u00E3 copy!' : 'Copy code'\" (click)=\"copyToClipboard()\">\r\n <mat-icon [class.text-success]=\"copied()\">{{ copied() ? 'check' : 'content_copy' }}</mat-icon>\r\n <span>{{ copied() ? 'Copied' : 'Copy' }}</span>\r\n </button>\r\n </div>\r\n\r\n <div class=\"sd-code-content\" [style.maxHeight]=\"maxHeight()\">\r\n <div class=\"editor-overlay-container\">\r\n \r\n @if (!viewed()) {\r\n <textarea \r\n class=\"code-textarea\"\r\n [ngModel]=\"textValue()\"\r\n (ngModelChange)=\"onTextChange($event)\"\r\n spellcheck=\"false\"\r\n autocomplete=\"off\"\r\n autocorrect=\"off\"\r\n autocapitalize=\"off\">\r\n </textarea>\r\n }\r\n \r\n <pre aria-hidden=\"true\" class=\"code-display\"><code [class]=\"'language-' + prismLang()\" [innerHTML]=\"highlightedCode()\"></code></pre>\r\n \r\n </div>\r\n </div>\r\n</div>", styles: ["@charset \"UTF-8\";.sd-code-wrapper{background-color:#1e1e1e;border-radius:8px;overflow:hidden;box-shadow:0 4px 12px #00000026;margin:16px 0;font-family:Fira Code,Consolas,Monaco,monospace}.sd-code-wrapper .sd-code-header{display:flex;align-items:center;justify-content:space-between;background-color:#2d2d2d;padding:8px 16px;border-bottom:1px solid #404040}.sd-code-wrapper .sd-code-header .mac-dots{display:flex;gap:6px}.sd-code-wrapper .sd-code-header .mac-dots .dot{width:12px;height:12px;border-radius:50%}.sd-code-wrapper .sd-code-header .mac-dots .dot.red{background-color:#ff5f56}.sd-code-wrapper .sd-code-header .mac-dots .dot.yellow{background-color:#ffbd2e}.sd-code-wrapper .sd-code-header .mac-dots .dot.green{background-color:#27c93f}.sd-code-wrapper .sd-code-header .lang-badge{color:#858585;font-size:12px;font-weight:600;letter-spacing:.5px}.sd-code-wrapper .sd-code-header .copy-btn{display:flex;align-items:center;gap:4px;background:none;border:none;color:#ccc;cursor:pointer;font-size:13px;padding:4px 8px;border-radius:4px;transition:all .2s ease}.sd-code-wrapper .sd-code-header .copy-btn mat-icon{font-size:16px;width:16px;height:16px}.sd-code-wrapper .sd-code-header .copy-btn:hover{background-color:#404040;color:#fff}.sd-code-wrapper .sd-code-header .copy-btn .text-success{color:#27c93f!important}.sd-code-wrapper .sd-code-content{overflow-y:auto;position:relative}.sd-code-wrapper .sd-code-content::-webkit-scrollbar{width:8px;height:8px}.sd-code-wrapper .sd-code-content::-webkit-scrollbar-thumb{background:#555;border-radius:4px}.sd-code-wrapper .sd-code-content::-webkit-scrollbar-thumb:hover{background:#777}.sd-code-wrapper .sd-code-content::-webkit-scrollbar-track{background:#1e1e1e}.sd-code-wrapper .sd-code-content .editor-overlay-container{position:relative;min-height:100px}.sd-code-wrapper .sd-code-content .code-textarea,.sd-code-wrapper .sd-code-content .code-display{margin:0;padding:16px;border:0;width:100%;min-height:100%;font-family:inherit;font-size:14px;line-height:1.5;white-space:pre-wrap;word-break:break-all;tab-size:2}.sd-code-wrapper .sd-code-content .code-textarea{position:absolute;top:0;left:0;z-index:2;background:transparent;color:transparent;caret-color:#fff;resize:none;outline:none;overflow:hidden}.sd-code-wrapper .sd-code-content .code-display{position:relative;z-index:1;pointer-events:none}.sd-code-wrapper .sd-code-content .code-display code{color:#d4d4d4;font-family:inherit}.sd-code-wrapper .token.comment,.sd-code-wrapper .token.block-comment,.sd-code-wrapper .token.prolog,.sd-code-wrapper .token.doctype,.sd-code-wrapper .token.cdata{color:#999}.sd-code-wrapper .token.punctuation{color:#ccc}.sd-code-wrapper .token.tag,.sd-code-wrapper .token.attr-name,.sd-code-wrapper .token.namespace,.sd-code-wrapper .token.deleted{color:#e2777a}.sd-code-wrapper .token.function-name{color:#6196cc}.sd-code-wrapper .token.boolean,.sd-code-wrapper .token.number,.sd-code-wrapper .token.function{color:#f08d49}.sd-code-wrapper .token.property,.sd-code-wrapper .token.class-name,.sd-code-wrapper .token.constant,.sd-code-wrapper .token.symbol{color:#f8c555}.sd-code-wrapper .token.selector,.sd-code-wrapper .token.important,.sd-code-wrapper .token.atrule,.sd-code-wrapper .token.keyword,.sd-code-wrapper .token.builtin{color:#cc99cd}.sd-code-wrapper .token.string,.sd-code-wrapper .token.char,.sd-code-wrapper .token.attr-value,.sd-code-wrapper .token.regex,.sd-code-wrapper .token.variable{color:#7ec699}.sd-code-wrapper .token.operator,.sd-code-wrapper .token.entity,.sd-code-wrapper .token.url{color:#67cdcc}.sd-code-wrapper .token.important,.sd-code-wrapper .token.bold{font-weight:700}.sd-code-wrapper .token.italic{font-style:italic}.sd-code-wrapper .token.entity{cursor:help}.sd-code-wrapper .token.inserted{color:green}\n"] }]
122
+ }], ctorParameters: () => [] });
123
+
124
+ /**
125
+ * Generated bundle index. Do not edit.
126
+ */
127
+
128
+ export { SdCodeEditor };
129
+ //# sourceMappingURL=sd-angular-core-components-code-editor.mjs.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"sd-angular-core-components-code-editor.mjs","sources":["../../../projects/sd-angular/components/code-editor/src/code-editor.component.ts","../../../projects/sd-angular/components/code-editor/src/code-editor.component.html","../../../projects/sd-angular/components/code-editor/sd-angular-core-components-code-editor.ts"],"sourcesContent":["import { Clipboard } from '@angular/cdk/clipboard';\r\nimport { CommonModule } from '@angular/common';\r\nimport {\r\n ChangeDetectionStrategy,\r\n Component,\r\n computed,\r\n effect,\r\n inject,\r\n input,\r\n model,\r\n signal,\r\n untracked,\r\n ViewEncapsulation,\r\n booleanAttribute\r\n} from '@angular/core';\r\nimport { FormsModule } from '@angular/forms';\r\nimport { MatIconModule } from '@angular/material/icon';\r\nimport { MatTooltipModule } from '@angular/material/tooltip';\r\nimport { DomSanitizer, SafeHtml } from '@angular/platform-browser';\r\n\r\nimport * as Prism from 'prismjs';\r\nimport 'prismjs/components/prism-typescript';\r\nimport 'prismjs/components/prism-json';\r\nimport 'prismjs/components/prism-css';\r\nimport 'prismjs/components/prism-scss';\r\nimport 'prismjs/components/prism-markup'; // HTML\r\n\r\nexport type CodeLanguage = 'html' | 'typescript' | 'json' | 'css' | 'scss';\r\n\r\n@Component({\r\n selector: 'sd-code-editor',\r\n standalone: true,\r\n imports: [CommonModule, MatIconModule, MatTooltipModule, FormsModule],\r\n changeDetection: ChangeDetectionStrategy.OnPush,\r\n templateUrl: './code-editor.component.html',\r\n styleUrls: ['./code-editor.component.scss'],\r\n encapsulation: ViewEncapsulation.None, // Bắt buộc để nhận màu Prism\r\n})\r\nexport class SdCodeEditor {\r\n #clipboard = inject(Clipboard);\r\n #sanitizer = inject(DomSanitizer);\r\n\r\n // ==========================================\r\n // 1. SIGNAL INPUTS & MODEL\r\n // ==========================================\r\n // Nhận bất kỳ kiểu dữ liệu nào (string, array, object)\r\n valueModel = model<any>(undefined, { alias: 'model' });\r\n \r\n language = input<CodeLanguage>('typescript');\r\n maxHeight = input<string>('500px');\r\n \r\n // Trạng thái Viewed (true = Read Only, false = Editable)\r\n viewed = input(false, { transform: booleanAttribute });\r\n\r\n // ==========================================\r\n // 2. INTERNAL STATE\r\n // ==========================================\r\n copied = signal<boolean>(false);\r\n \r\n // Chuỗi text nội bộ dùng để map với thẻ <textarea>\r\n textValue = signal<string>('');\r\n \r\n prismLang = computed(() => this.language() === 'html' ? 'markup' : this.language());\r\n\r\n // Cờ lưu vết để chống Loop (Vòng lặp vô tận khi bắn 2 chiều)\r\n private _lastEmittedValue: any = undefined;\r\n\r\n constructor() {\r\n // ==========================================\r\n // EFFECT 1: Dữ liệu từ CHA truyền vào (Model -> TextValue)\r\n // ==========================================\r\n effect(() => {\r\n const extVal = this.valueModel();\r\n const lang = this.language();\r\n \r\n untracked(() => {\r\n // Nếu giá trị này do chính component bắn ra, bỏ qua để tránh loop\r\n if (extVal === this._lastEmittedValue) return;\r\n\r\n if (typeof extVal === 'string') {\r\n this.textValue.set(extVal);\r\n } else if (extVal !== undefined && extVal !== null) {\r\n // Tự động format Object -> String nếu là JSON\r\n if (lang === 'json') {\r\n try {\r\n this.textValue.set(JSON.stringify(extVal, null, 2));\r\n } catch {\r\n this.textValue.set('// Lỗi: Object có tham chiếu vòng (Circular Reference)');\r\n }\r\n } else {\r\n this.textValue.set(String(extVal));\r\n }\r\n } else {\r\n this.textValue.set('');\r\n }\r\n });\r\n });\r\n }\r\n\r\n // ==========================================\r\n // EFFECT 2: PrismJS render (TextValue -> HTML MÀU)\r\n // ==========================================\r\n highlightedCode = computed<SafeHtml>(() => {\r\n // Dùng khoảng trắng để giữ độ cao cho thẻ pre nếu rỗng\r\n const rawCode = this.textValue() || ' '; \r\n const langKey = this.prismLang();\r\n const grammar = Prism.languages[langKey] || Prism.languages['markup'];\r\n \r\n // Cộng thêm \\n ở cuối để chống lỗi con trỏ textarea ăn lẹm dòng cuối\r\n const highlightedString = Prism.highlight(rawCode, grammar, langKey) + '\\n';\r\n return this.#sanitizer.bypassSecurityTrustHtml(highlightedString);\r\n });\r\n\r\n // ==========================================\r\n // EVENTS\r\n // ==========================================\r\n \r\n // Khi người dùng gõ vào Textarea (TextValue -> Model)\r\n onTextChange(newText: string) {\r\n this.textValue.set(newText);\r\n \r\n let valToEmit: any = newText;\r\n \r\n // Nếu ngôn ngữ là JSON, cố gắng trả về Object thật\r\n if (this.language() === 'json') {\r\n try {\r\n valToEmit = JSON.parse(newText);\r\n } catch {\r\n // Nếu gõ dở ngoặc/sai cú pháp -> Trả về chuỗi String tạm\r\n valToEmit = newText; \r\n }\r\n }\r\n \r\n // Ghi sổ và bắn ra ngoài\r\n this._lastEmittedValue = valToEmit;\r\n this.valueModel.set(valToEmit);\r\n }\r\n\r\n copyToClipboard() {\r\n const rawCode = this.textValue();\r\n if (!rawCode) return;\r\n \r\n if (this.#clipboard.copy(rawCode)) {\r\n this.copied.set(true);\r\n setTimeout(() => this.copied.set(false), 2000);\r\n }\r\n }\r\n}","<div class=\"sd-code-wrapper\" [class.is-editable]=\"!viewed()\">\r\n \r\n <div class=\"sd-code-header\">\r\n <div class=\"mac-dots\">\r\n <span class=\"dot red\"></span>\r\n <span class=\"dot yellow\"></span>\r\n <span class=\"dot green\"></span>\r\n </div>\r\n \r\n <div class=\"lang-badge\">\r\n {{ language().toUpperCase() }} \r\n <span style=\"opacity: 0.6; margin-left: 4px; font-weight: normal;\">\r\n {{ viewed() ? '(READ ONLY)' : '(EDITING)' }}\r\n </span>\r\n </div>\r\n \r\n <button class=\"copy-btn\" [matTooltip]=\"copied() ? 'Đã copy!' : 'Copy code'\" (click)=\"copyToClipboard()\">\r\n <mat-icon [class.text-success]=\"copied()\">{{ copied() ? 'check' : 'content_copy' }}</mat-icon>\r\n <span>{{ copied() ? 'Copied' : 'Copy' }}</span>\r\n </button>\r\n </div>\r\n\r\n <div class=\"sd-code-content\" [style.maxHeight]=\"maxHeight()\">\r\n <div class=\"editor-overlay-container\">\r\n \r\n @if (!viewed()) {\r\n <textarea \r\n class=\"code-textarea\"\r\n [ngModel]=\"textValue()\"\r\n (ngModelChange)=\"onTextChange($event)\"\r\n spellcheck=\"false\"\r\n autocomplete=\"off\"\r\n autocorrect=\"off\"\r\n autocapitalize=\"off\">\r\n </textarea>\r\n }\r\n \r\n <pre aria-hidden=\"true\" class=\"code-display\"><code [class]=\"'language-' + prismLang()\" [innerHTML]=\"highlightedCode()\"></code></pre>\r\n \r\n </div>\r\n </div>\r\n</div>","/**\n * Generated bundle index. Do not edit.\n */\n\nexport * from './index';\n"],"names":[],"mappings":"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"}