@libs-ui/components-preview-text-data 0.2.350-1 → 0.2.352-0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
@@ -1,168 +1,222 @@
1
- /* eslint-disable @typescript-eslint/no-explicit-any */
2
- import { ChangeDetectionStrategy, Component, computed, inject, input, model, output, signal, viewChild } from '@angular/core';
3
- import { cpp } from '@codemirror/lang-cpp';
4
- import { css } from '@codemirror/lang-css';
5
- import { go } from '@codemirror/lang-go';
6
- import { html } from '@codemirror/lang-html';
7
- import { java } from '@codemirror/lang-java';
8
- import { javascript } from '@codemirror/lang-javascript';
9
- import { json } from '@codemirror/lang-json';
10
- import { markdown } from '@codemirror/lang-markdown';
11
- import { php } from '@codemirror/lang-php';
12
- import { python } from '@codemirror/lang-python';
13
- import { sql } from '@codemirror/lang-sql';
14
- import { xml } from '@codemirror/lang-xml';
15
- import { yaml } from '@codemirror/lang-yaml';
16
- import { StreamLanguage } from '@codemirror/language';
1
+ import { ChangeDetectionStrategy, Component, computed, DestroyRef, effect, inject, input, model, output, signal, viewChild } from '@angular/core';
17
2
  import { linter, lintGutter } from '@codemirror/lint';
18
- import { Compartment, EditorState } from '@codemirror/state';
19
- import { lineNumbers } from '@codemirror/view';
3
+ import { Compartment } from '@codemirror/state';
4
+ import { EditorView, lineNumbers } from '@codemirror/view';
20
5
  import { LibsUiComponentsButtonsButtonComponent } from '@libs-ui/components-buttons-button';
21
6
  import { LibsUiComponentsDropdownComponent } from '@libs-ui/components-dropdown';
22
7
  import { LibsUiNotificationService } from '@libs-ui/services-notification';
23
8
  import { get } from '@libs-ui/utils';
24
- import { basicSetup, EditorView } from 'codemirror';
25
- import { Parser } from 'node-sql-parser';
26
- import { httpRequestConfigGetOptionsLang, optionsLang } from './preview-text-data.define';
9
+ import { basicSetup } from 'codemirror6';
10
+ import { createDefaultLanguage, httpRequestConfigGetOptionsLang, languageRegistry, optionsLangData } from './preview-text-data.define';
27
11
  import * as i0 from "@angular/core";
28
- const sqlParser = new Parser();
29
12
  export class LibsUiComponentsPreviewTextDataComponent {
13
+ // ==========================================================================
14
+ // PRIVATE PROPERTIES
15
+ // ==========================================================================
30
16
  editorViewInstance;
31
17
  wrapCompartment = new Compartment();
32
18
  languageCompartment = new Compartment();
33
19
  lineNumberCompartment = new Compartment();
20
+ linterCompartment = new Compartment();
21
+ /** Cache để tránh load lại language đã load */
22
+ languageCache = new Map();
23
+ /** Flag để tránh update editor khi đang khởi tạo */
24
+ isInitialized = false;
25
+ // ==========================================================================
26
+ // INJECTED SERVICES
27
+ // ==========================================================================
28
+ notificationService = inject(LibsUiNotificationService);
29
+ destroyRef = inject(DestroyRef);
30
+ // ==========================================================================
31
+ // PROTECTED SIGNALS
32
+ // ==========================================================================
34
33
  isWrap = signal(true);
35
34
  configLoadDataIsHttpConfig = httpRequestConfigGetOptionsLang();
36
- labelLang = computed(() => optionsLang().find((item) => item.id === this.langSelected())?.label);
37
- content = input('', { transform: (value) => value ?? '' });
35
+ labelLang = computed(() => optionsLangData.find((item) => item.id === this.langSelected())?.label);
36
+ // ==========================================================================
37
+ // INPUTS
38
+ // ==========================================================================
39
+ content = input('', {
40
+ transform: (value) => value ?? '',
41
+ });
38
42
  langSelected = model.required();
39
- editable = input(false, { transform: (value) => value ?? false });
40
- hiddenAction = input(false, { transform: (value) => value ?? false });
43
+ editable = input(false, {
44
+ transform: (value) => value ?? false,
45
+ });
46
+ hiddenAction = input(false, {
47
+ transform: (value) => value ?? false,
48
+ });
41
49
  lintIgnorePatterns = input(['Cannot use import statement outside a module', 'Unexpected token export', 'import ', '@angular/core'], {
42
50
  transform: (value) => value ?? ['Cannot use import statement outside a module', 'Unexpected token export', 'import ', '@angular/core'],
43
51
  });
44
- background = input('#f8f9fa', { transform: (value) => value ?? '#f8f9fa' });
52
+ background = input('#f8f9fa', {
53
+ transform: (value) => value ?? '#f8f9fa',
54
+ });
55
+ // ==========================================================================
56
+ // OUTPUTS
57
+ // ==========================================================================
45
58
  outChange = output();
46
59
  syntaxErrors = output();
60
+ // ==========================================================================
61
+ // VIEW CHILDREN
62
+ // ==========================================================================
47
63
  containerPreview = viewChild.required('containerPreview');
48
- notificationService = inject(LibsUiNotificationService);
49
- ngAfterViewInit() {
50
- this.editorViewInstance = new EditorView({ ...this.getExtensionsByLanguage(this.langSelected()), doc: this.content(), parent: this.containerPreview().nativeElement });
64
+ // ==========================================================================
65
+ // CONSTRUCTOR - Setup effects
66
+ // ==========================================================================
67
+ constructor() {
68
+ // Effect để sync content từ input vào editor
69
+ effect(() => {
70
+ const newContent = this.content();
71
+ if (this.isInitialized && this.editorViewInstance) {
72
+ const currentContent = this.editorViewInstance.state.doc.toString();
73
+ if (newContent !== currentContent) {
74
+ this.editorViewInstance.dispatch({
75
+ changes: {
76
+ from: 0,
77
+ to: currentContent.length,
78
+ insert: newContent,
79
+ },
80
+ });
81
+ }
82
+ }
83
+ });
84
+ // Cleanup khi component bị destroy
85
+ this.destroyRef.onDestroy(() => {
86
+ this.editorViewInstance?.destroy();
87
+ this.languageCache.clear();
88
+ });
51
89
  }
52
- getExtensionsByLanguage(lang) {
53
- let formatByLangFunction = StreamLanguage.define({
54
- token(stream) {
55
- stream.skipToEnd();
56
- return null;
57
- },
90
+ // ==========================================================================
91
+ // LIFECYCLE HOOKS
92
+ // ==========================================================================
93
+ async ngAfterViewInit() {
94
+ await this.initializeEditor();
95
+ this.isInitialized = true;
96
+ }
97
+ // ==========================================================================
98
+ // PRIVATE METHODS - Editor initialization
99
+ // ==========================================================================
100
+ /**
101
+ * Khởi tạo editor với các extensions cơ bản
102
+ * Language được load async để tối ưu bundle size
103
+ */
104
+ async initializeEditor() {
105
+ const languageExtension = await this.loadLanguageExtension(this.langSelected());
106
+ const linterExtension = await this.loadLinterExtension(this.langSelected());
107
+ this.editorViewInstance = new EditorView({
108
+ doc: this.content(),
109
+ parent: this.containerPreview().nativeElement,
110
+ extensions: this.createExtensions(languageExtension, linterExtension),
58
111
  });
59
- let lintFunction = linter(() => []);
112
+ }
113
+ /**
114
+ * Tạo danh sách extensions cho editor
115
+ */
116
+ createExtensions(languageExtension, linterExtension) {
117
+ return [
118
+ basicSetup,
119
+ this.languageCompartment.of(languageExtension),
120
+ this.linterCompartment.of(linterExtension),
121
+ this.createLightTheme(),
122
+ EditorView.editable.of(this.editable()),
123
+ this.wrapCompartment.of(EditorView.lineWrapping),
124
+ this.lineNumberCompartment.of(lineNumbers()),
125
+ lintGutter(),
126
+ this.createBracketTheme(),
127
+ this.createUpdateListener(),
128
+ ];
129
+ }
130
+ /**
131
+ * Load language extension với caching
132
+ * Mỗi language chỉ được load 1 lần
133
+ */
134
+ async loadLanguageExtension(lang) {
60
135
  const lowerLang = lang.toLowerCase();
61
- switch (lowerLang) {
62
- case 'javascript':
63
- formatByLangFunction = javascript({ jsx: true, typescript: true });
64
- lintFunction = this.jsLinterFn();
65
- break;
66
- case 'html':
67
- formatByLangFunction = html();
68
- break;
69
- case 'css':
70
- formatByLangFunction = css();
71
- break;
72
- case 'markdown':
73
- formatByLangFunction = markdown();
74
- break;
75
- case 'json':
76
- formatByLangFunction = json();
77
- lintFunction = this.jsonLinterFn();
78
- break;
79
- case 'sql':
80
- formatByLangFunction = sql();
81
- lintFunction = this.sqlLinterFn();
82
- break;
83
- case 'xml':
84
- formatByLangFunction = xml();
85
- break;
86
- case 'yaml':
87
- formatByLangFunction = yaml();
88
- break;
89
- case 'python':
90
- formatByLangFunction = python();
91
- break;
92
- case 'java':
93
- formatByLangFunction = java();
94
- break;
95
- case 'cpp':
96
- formatByLangFunction = cpp();
97
- break;
98
- case 'php':
99
- formatByLangFunction = php();
100
- break;
101
- case 'go':
102
- formatByLangFunction = go();
103
- break;
136
+ const cached = this.languageCache.get(lowerLang);
137
+ if (cached) {
138
+ return cached;
104
139
  }
105
- return {
106
- extensions: [
107
- basicSetup,
108
- this.languageCompartment.of(formatByLangFunction),
109
- this.lightTheme(),
110
- EditorView.editable.of(this.editable()),
111
- this.wrapCompartment.of(EditorView.lineWrapping),
112
- this.lineNumberCompartment.of(lineNumbers()),
113
- lintGutter(),
114
- this.customBracketTheme(),
115
- lintFunction,
116
- EditorView.updateListener.of((update) => {
117
- if (update.docChanged && this.editable()) {
118
- const newValue = update.state.doc.toString();
119
- this.outChange.emit({
120
- content: newValue,
121
- isWrap: this.isWrap(),
122
- language: this.langSelected(),
123
- contextChange: 'content',
124
- });
125
- }
126
- }),
127
- ],
128
- };
140
+ const registryItem = languageRegistry[lowerLang];
141
+ const extension = registryItem ? await registryItem.loader() : await createDefaultLanguage();
142
+ this.languageCache.set(lowerLang, extension);
143
+ return extension;
129
144
  }
130
- lightTheme() {
145
+ /**
146
+ * Load linter extension nếu có
147
+ */
148
+ async loadLinterExtension(lang) {
149
+ const lowerLang = lang.toLowerCase();
150
+ const registryItem = languageRegistry[lowerLang];
151
+ if (registryItem?.linter) {
152
+ return registryItem.linter(this);
153
+ }
154
+ return linter(() => []);
155
+ }
156
+ // ==========================================================================
157
+ // PRIVATE METHODS - Theme configuration
158
+ // ==========================================================================
159
+ createLightTheme() {
131
160
  return EditorView.theme({
132
161
  '&': { backgroundColor: this.background(), color: '#333' },
133
162
  '.cm-content': { caretColor: '#111' },
134
163
  '.cm-activeLine': { backgroundColor: '#f0f0f0' },
135
164
  });
136
165
  }
137
- customBracketTheme() {
166
+ createBracketTheme() {
138
167
  return EditorView.theme({
139
168
  '.cm-matchingBracket': {
140
- backgroundColor: 'rgba(100, 200, 255, 0.15)', // xanh nhẹ trong suốt
169
+ backgroundColor: 'rgba(100, 200, 255, 0.15)',
141
170
  border: '1px solid rgba(100, 200, 255, 0.4)',
142
171
  borderRadius: '3px',
143
172
  transition: 'all 0.2s ease',
144
173
  },
145
174
  '.cm-nonmatchingBracket': {
146
- backgroundColor: 'rgba(255, 100, 100, 0.15)', // đỏ nhẹ
175
+ backgroundColor: 'rgba(255, 100, 100, 0.15)',
147
176
  border: '1px solid rgba(255, 100, 100, 0.3)',
148
177
  borderRadius: '3px',
149
178
  },
150
179
  });
151
180
  }
152
- jsLinterFn() {
181
+ // ==========================================================================
182
+ // PRIVATE METHODS - Update listener
183
+ // ==========================================================================
184
+ createUpdateListener() {
185
+ return EditorView.updateListener.of((update) => {
186
+ if (update.docChanged && this.editable()) {
187
+ const newValue = update.state.doc.toString();
188
+ this.outChange.emit({
189
+ content: newValue,
190
+ isWrap: this.isWrap(),
191
+ language: this.langSelected(),
192
+ contextChange: 'content',
193
+ });
194
+ }
195
+ });
196
+ }
197
+ // ==========================================================================
198
+ // PUBLIC METHODS - Linter factories (được gọi từ registry)
199
+ // ==========================================================================
200
+ /**
201
+ * Tạo JavaScript linter
202
+ * Sử dụng Function constructor để validate syntax
203
+ */
204
+ createJsLinter() {
153
205
  return linter((view) => {
154
206
  const text = view.state.doc.toString();
155
207
  const diagnostics = [];
156
208
  try {
157
- new Function(text); // test parse code
209
+ new Function(text);
158
210
  }
159
211
  catch (err) {
160
- if (!this.lintIgnorePatterns().some((pattern) => err.message.includes(pattern))) {
212
+ const message = get(err, 'message', '');
213
+ const shouldIgnore = this.lintIgnorePatterns().some((pattern) => message.includes(pattern));
214
+ if (!shouldIgnore) {
161
215
  diagnostics.push({
162
216
  from: 0,
163
217
  to: text.length,
164
218
  severity: 'error',
165
- message: err.message,
219
+ message,
166
220
  });
167
221
  this.syntaxErrors.emit(diagnostics);
168
222
  }
@@ -170,7 +224,11 @@ export class LibsUiComponentsPreviewTextDataComponent {
170
224
  return diagnostics;
171
225
  });
172
226
  }
173
- jsonLinterFn() {
227
+ /**
228
+ * Tạo JSON linter
229
+ * Sử dụng JSON.parse để validate
230
+ */
231
+ createJsonLinter() {
174
232
  return linter((view) => {
175
233
  const text = view.state.doc.toString();
176
234
  const diagnostics = [];
@@ -189,11 +247,18 @@ export class LibsUiComponentsPreviewTextDataComponent {
189
247
  return diagnostics;
190
248
  });
191
249
  }
192
- sqlLinterFn() {
193
- return linter((view) => {
250
+ /**
251
+ * Tạo SQL linter
252
+ * Lazy load node-sql-parser chỉ khi cần
253
+ */
254
+ createSqlLinter() {
255
+ return linter(async (view) => {
194
256
  const text = view.state.doc.toString();
195
257
  const diagnostics = [];
196
258
  try {
259
+ // Lazy load SQL parser chỉ khi thực sự cần
260
+ const { Parser } = await import('node-sql-parser');
261
+ const sqlParser = new Parser();
197
262
  sqlParser.astify(text);
198
263
  }
199
264
  catch (err) {
@@ -208,36 +273,50 @@ export class LibsUiComponentsPreviewTextDataComponent {
208
273
  return diagnostics;
209
274
  });
210
275
  }
276
+ // ==========================================================================
277
+ // PROTECTED METHODS - Event handlers
278
+ // ==========================================================================
211
279
  handlerCopy() {
212
280
  navigator.clipboard.writeText(this.content() || '');
213
281
  this.notificationService.showCompTypeTextInfo('Sao chép thành công');
214
282
  }
215
283
  handlerLineWrap() {
284
+ console.log('handlerLineWrap', this.isWrap());
285
+ // Lấy nội dung mới nhất từ editor sau khi đã xóa newlines
286
+ const currentContent = this.editorViewInstance?.state.doc.toString() || '';
287
+ const updatedContent = this.isWrap() && currentContent ? currentContent.replace(/\n/g, ' ') : this.content();
216
288
  this.isWrap.update((val) => !val);
289
+ console.log('updatedContent', this.isWrap());
290
+ console.log('currentContent', currentContent);
291
+ console.log('updatedContent', updatedContent);
292
+ if (currentContent !== updatedContent) {
293
+ this.editorViewInstance?.dispatch({
294
+ changes: { from: 0, to: currentContent.length, insert: updatedContent },
295
+ });
296
+ }
297
+ // Batch cả 2 dispatch thành 1 để tối ưu performance
217
298
  this.editorViewInstance?.dispatch({
218
- effects: this.wrapCompartment.reconfigure(this.isWrap() ? EditorView.lineWrapping : []),
219
- });
220
- this.editorViewInstance?.dispatch({
221
- effects: this.lineNumberCompartment.reconfigure(this.isWrap() ? lineNumbers() : [] // 👈 nếu tắt thì remove extension
222
- ),
299
+ effects: [this.wrapCompartment.reconfigure(this.isWrap() ? EditorView.lineWrapping : []), this.lineNumberCompartment.reconfigure(this.isWrap() ? lineNumbers() : [])],
223
300
  });
224
301
  this.outChange.emit({
225
- content: this.content(),
302
+ content: updatedContent,
226
303
  isWrap: this.isWrap(),
227
304
  language: this.langSelected(),
228
305
  contextChange: 'isWrap',
229
306
  });
230
307
  }
231
- handlerSelectKey(data) {
232
- if (!data || !data.key) {
308
+ async handlerSelectKey(data) {
309
+ if (!data?.key) {
233
310
  return;
234
311
  }
235
- this.langSelected.set(data.key);
236
- const extensions = this.getExtensionsByLanguage(this.langSelected());
237
- this.editorViewInstance?.setState(EditorState.create({
238
- doc: this.editorViewInstance?.state.doc.toString(),
239
- ...extensions,
240
- }));
312
+ const newLang = data.key;
313
+ this.langSelected.set(newLang);
314
+ // Load language và linter mới
315
+ const [languageExtension, linterExtension] = await Promise.all([this.loadLanguageExtension(newLang), this.loadLinterExtension(newLang)]);
316
+ // Reconfigure thay vì tạo mới state để giữ lại document content
317
+ this.editorViewInstance?.dispatch({
318
+ effects: [this.languageCompartment.reconfigure(languageExtension), this.linterCompartment.reconfigure(linterExtension)],
319
+ });
241
320
  this.outChange.emit({
242
321
  content: this.content(),
243
322
  isWrap: this.isWrap(),
@@ -251,5 +330,5 @@ export class LibsUiComponentsPreviewTextDataComponent {
251
330
  i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: LibsUiComponentsPreviewTextDataComponent, decorators: [{
252
331
  type: Component,
253
332
  args: [{ selector: 'libs_ui-components-preview_text_data', standalone: true, changeDetection: ChangeDetectionStrategy.OnPush, imports: [LibsUiComponentsDropdownComponent, LibsUiComponentsButtonsButtonComponent], template: "<div\n class=\"libs-ui-preview-data-container flex flex-col w-full h-auto rounded-[8px] libs-ui-border-general px-[8px]\"\n [style.--background-color]=\"background()\"\n [class.pt-[8px]]=\"!hiddenAction()\">\n @if (!hiddenAction()) {\n <div class=\"flex items-center content-between color-[#6a7383]\">\n <libs_ui-components-dropdown\n classInclude=\"w-[200px]\"\n [listConfig]=\"configLoadDataIsHttpConfig\"\n [listMaxItemShow]=\"5\"\n [isNgContent]=\"true\"\n [readonly]=\"!editable()\"\n [listHasButtonUnSelectOption]=\"false\"\n (outSelectKey)=\"handlerSelectKey($event)\">\n <libs_ui-components-buttons-button\n [type]=\"'button-link-third'\"\n [label]=\"labelLang() || ''\"\n [sizeButton]=\"'small'\"\n [classIconRight]=\"editable() ? 'libs-ui-icon-move-right rotate-90' : ''\"\n [classInclude]=\"'!p-[0px]' + (editable() ? '' : '!pointer-events-none !cursor-default hover:!text-[#6A7383]')\" />\n </libs_ui-components-dropdown>\n <div class=\"flex items-center\">\n <libs_ui-components-buttons-button\n [type]=\"'button-link-third'\"\n [label]=\"isWrap() ? 'i18n_remove_line_wrap' : 'i18n_line_wrap'\"\n [sizeButton]=\"'small'\"\n [classIconLeft]=\"isWrap() ? 'libs-ui-icon-unwrap' : 'libs-ui-icon-wrap'\"\n [classInclude]=\"'mo-lib-p-0px mo-lib-mr-16px'\"\n (outClick)=\"handlerLineWrap()\" />\n <libs_ui-components-buttons-button\n [type]=\"'button-link-third'\"\n [label]=\"'i18n_copy'\"\n [sizeButton]=\"'small'\"\n [classIconLeft]=\"'libs-ui-icon-copy'\"\n [classInclude]=\"'mo-lib-p-0px'\"\n (outClick)=\"handlerCopy()\" />\n </div>\n </div>\n }\n <div #containerPreview></div>\n</div>\n", styles: [":host ::ng-deep .libs-ui-preview-data-container{background-color:var(--background-color)!important}:host ::ng-deep .libs-ui-preview-data-container .cm-line{white-space:pre-wrap}:host ::ng-deep .libs-ui-preview-data-container .cm-lintRange-error{background-color:#ff323233;border-bottom:2px solid red}:host ::ng-deep .libs-ui-preview-data-container .cm-lintRange-warning{background-color:#ffc80026}:host ::ng-deep .libs-ui-preview-data-container .cm-tooltip-lint{background:#fff8f8;color:#d32f2f;border:1px solid #f44336;padding:8px 10px;font-size:13px;font-family:Inter,sans-serif;border-radius:6px;box-shadow:0 2px 8px #ff000026}:host ::ng-deep .libs-ui-preview-data-container .cm-focused{outline:none!important}:host ::ng-deep .libs-ui-preview-data-container .cm-gutters{background-color:var(--background-color)!important}\n"] }]
254
- }] });
255
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"preview-text-data.component.js","sourceRoot":"","sources":["../../../../../libs-ui/components/preview-text-data/src/preview-text-data.component.ts","../../../../../libs-ui/components/preview-text-data/src/preview-text-data.component.html"],"names":[],"mappings":"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","sourcesContent":["/* eslint-disable @typescript-eslint/no-explicit-any */\nimport { AfterViewInit, ChangeDetectionStrategy, Component, computed, ElementRef, inject, input, model, output, signal, viewChild } from '@angular/core';\nimport { cpp } from '@codemirror/lang-cpp';\nimport { css } from '@codemirror/lang-css';\nimport { go } from '@codemirror/lang-go';\nimport { html } from '@codemirror/lang-html';\nimport { java } from '@codemirror/lang-java';\nimport { javascript } from '@codemirror/lang-javascript';\nimport { json } from '@codemirror/lang-json';\nimport { markdown } from '@codemirror/lang-markdown';\nimport { php } from '@codemirror/lang-php';\nimport { python } from '@codemirror/lang-python';\nimport { sql } from '@codemirror/lang-sql';\nimport { xml } from '@codemirror/lang-xml';\nimport { yaml } from '@codemirror/lang-yaml';\nimport { StreamLanguage } from '@codemirror/language';\nimport { Diagnostic, linter, lintGutter } from '@codemirror/lint';\nimport { Compartment, EditorState } from '@codemirror/state';\nimport { lineNumbers } from '@codemirror/view';\nimport { LibsUiComponentsButtonsButtonComponent } from '@libs-ui/components-buttons-button';\nimport { IEmitSelectKey, LibsUiComponentsDropdownComponent } from '@libs-ui/components-dropdown';\nimport { LibsUiNotificationService } from '@libs-ui/services-notification';\nimport { get } from '@libs-ui/utils';\nimport { basicSetup, EditorView } from 'codemirror';\nimport { Parser } from 'node-sql-parser';\nimport { httpRequestConfigGetOptionsLang, optionsLang } from './preview-text-data.define';\nimport { IPreviewTextDataChange, PREVIEW_TEXT_DATA_LANGUAGE_SUPPORT } from './preview-text-data.interfaces';\n\nconst sqlParser = new Parser();\n\n@Component({\n  // eslint-disable-next-line @angular-eslint/component-selector\n  selector: 'libs_ui-components-preview_text_data',\n  templateUrl: './preview-text-data.component.html',\n  styleUrls: ['./preview-text-data.component.scss'],\n  standalone: true,\n  changeDetection: ChangeDetectionStrategy.OnPush,\n  imports: [LibsUiComponentsDropdownComponent, LibsUiComponentsButtonsButtonComponent],\n})\nexport class LibsUiComponentsPreviewTextDataComponent implements AfterViewInit {\n  private editorViewInstance?: EditorView;\n  private wrapCompartment = new Compartment();\n  private languageCompartment = new Compartment();\n  private lineNumberCompartment = new Compartment();\n\n  protected isWrap = signal<boolean>(true);\n  protected configLoadDataIsHttpConfig = httpRequestConfigGetOptionsLang();\n  protected labelLang = computed(() => optionsLang().find((item) => item.id === this.langSelected())?.label);\n\n  readonly content = input<string, string | undefined>('', { transform: (value?: string) => value ?? '' });\n  readonly langSelected = model.required<PREVIEW_TEXT_DATA_LANGUAGE_SUPPORT>();\n  readonly editable = input<boolean, boolean | undefined>(false, { transform: (value?: boolean) => value ?? false });\n  readonly hiddenAction = input<boolean, boolean | undefined>(false, { transform: (value?: boolean) => value ?? false });\n  readonly lintIgnorePatterns = input<Array<string>, Array<string> | undefined>(['Cannot use import statement outside a module', 'Unexpected token export', 'import ', '@angular/core'], {\n    transform: (value?: Array<string>) => value ?? ['Cannot use import statement outside a module', 'Unexpected token export', 'import ', '@angular/core'],\n  });\n\n  readonly background = input<string, string | undefined>('#f8f9fa', { transform: (value?: string) => value ?? '#f8f9fa' });\n\n  readonly outChange = output<IPreviewTextDataChange>();\n  readonly syntaxErrors = output<Diagnostic[]>();\n\n  private containerPreview = viewChild.required<ElementRef<HTMLDivElement>>('containerPreview');\n\n  private notificationService = inject(LibsUiNotificationService);\n  ngAfterViewInit() {\n    this.editorViewInstance = new EditorView({ ...this.getExtensionsByLanguage(this.langSelected()), doc: this.content(), parent: this.containerPreview().nativeElement });\n  }\n\n  private getExtensionsByLanguage(lang: PREVIEW_TEXT_DATA_LANGUAGE_SUPPORT): any {\n    let formatByLangFunction: any = StreamLanguage.define({\n      token(stream: any) {\n        stream.skipToEnd();\n        return null;\n      },\n    });\n    let lintFunction = linter(() => []);\n\n    const lowerLang = lang.toLowerCase();\n\n    switch (lowerLang) {\n      case 'javascript':\n        formatByLangFunction = javascript({ jsx: true, typescript: true });\n        lintFunction = this.jsLinterFn();\n        break;\n\n      case 'html':\n        formatByLangFunction = html();\n        break;\n\n      case 'css':\n        formatByLangFunction = css();\n        break;\n\n      case 'markdown':\n        formatByLangFunction = markdown();\n        break;\n\n      case 'json':\n        formatByLangFunction = json();\n        lintFunction = this.jsonLinterFn();\n        break;\n\n      case 'sql':\n        formatByLangFunction = sql();\n        lintFunction = this.sqlLinterFn();\n        break;\n\n      case 'xml':\n        formatByLangFunction = xml();\n        break;\n\n      case 'yaml':\n        formatByLangFunction = yaml();\n        break;\n\n      case 'python':\n        formatByLangFunction = python();\n        break;\n\n      case 'java':\n        formatByLangFunction = java();\n        break;\n\n      case 'cpp':\n        formatByLangFunction = cpp();\n        break;\n\n      case 'php':\n        formatByLangFunction = php();\n        break;\n\n      case 'go':\n        formatByLangFunction = go();\n        break;\n    }\n\n    return {\n      extensions: [\n        basicSetup,\n        this.languageCompartment.of(formatByLangFunction),\n        this.lightTheme(),\n        EditorView.editable.of(this.editable()),\n        this.wrapCompartment.of(EditorView.lineWrapping),\n        this.lineNumberCompartment.of(lineNumbers()),\n        lintGutter(),\n        this.customBracketTheme(),\n        lintFunction,\n        EditorView.updateListener.of((update: any) => {\n          if (update.docChanged && this.editable()) {\n            const newValue = update.state.doc.toString();\n\n            this.outChange.emit({\n              content: newValue,\n              isWrap: this.isWrap(),\n              language: this.langSelected(),\n              contextChange: 'content',\n            });\n          }\n        }),\n      ],\n    };\n  }\n\n  private lightTheme() {\n    return EditorView.theme({\n      '&': { backgroundColor: this.background(), color: '#333' },\n      '.cm-content': { caretColor: '#111' },\n      '.cm-activeLine': { backgroundColor: '#f0f0f0' },\n    });\n  }\n\n  private customBracketTheme() {\n    return EditorView.theme({\n      '.cm-matchingBracket': {\n        backgroundColor: 'rgba(100, 200, 255, 0.15)', // xanh nhẹ trong suốt\n        border: '1px solid rgba(100, 200, 255, 0.4)',\n        borderRadius: '3px',\n        transition: 'all 0.2s ease',\n      },\n      '.cm-nonmatchingBracket': {\n        backgroundColor: 'rgba(255, 100, 100, 0.15)', // đỏ nhẹ\n        border: '1px solid rgba(255, 100, 100, 0.3)',\n        borderRadius: '3px',\n      },\n    });\n  }\n\n  private jsLinterFn() {\n    return linter((view: any) => {\n      const text = view.state.doc.toString();\n      const diagnostics: Diagnostic[] = [];\n      try {\n        new Function(text); // test parse code\n      } catch (err: any) {\n        if (!this.lintIgnorePatterns().some((pattern: any) => err.message.includes(pattern))) {\n          diagnostics.push({\n            from: 0,\n            to: text.length,\n            severity: 'error',\n            message: err.message,\n          });\n          this.syntaxErrors.emit(diagnostics);\n        }\n      }\n      return diagnostics;\n    });\n  }\n\n  private jsonLinterFn() {\n    return linter((view: any) => {\n      const text = view.state.doc.toString();\n      const diagnostics: Diagnostic[] = [];\n      try {\n        JSON.parse(text);\n      } catch (err: unknown) {\n        diagnostics.push({\n          from: 0,\n          to: text.length,\n          severity: 'error',\n          message: get(err, 'message', ''),\n        });\n      }\n      this.syntaxErrors.emit(diagnostics);\n      return diagnostics;\n    });\n  }\n\n  private sqlLinterFn() {\n    return linter((view: any) => {\n      const text = view.state.doc.toString();\n      const diagnostics: Diagnostic[] = [];\n      try {\n        sqlParser.astify(text);\n      } catch (err: unknown) {\n        diagnostics.push({\n          from: 0,\n          to: text.length,\n          severity: 'error',\n          message: get(err, 'message', ''),\n        });\n      }\n      this.syntaxErrors.emit(diagnostics);\n      return diagnostics;\n    });\n  }\n\n  protected handlerCopy() {\n    navigator.clipboard.writeText(this.content() || '');\n    this.notificationService.showCompTypeTextInfo('Sao chép thành công');\n  }\n\n  protected handlerLineWrap() {\n    this.isWrap.update((val: any) => !val);\n    this.editorViewInstance?.dispatch({\n      effects: this.wrapCompartment.reconfigure(this.isWrap() ? EditorView.lineWrapping : []),\n    });\n\n    this.editorViewInstance?.dispatch({\n      effects: this.lineNumberCompartment.reconfigure(\n        this.isWrap() ? lineNumbers() : [] // 👈 nếu tắt thì remove extension\n      ),\n    });\n    this.outChange.emit({\n      content: this.content(),\n      isWrap: this.isWrap(),\n      language: this.langSelected(),\n      contextChange: 'isWrap',\n    });\n  }\n\n  protected handlerSelectKey(data?: IEmitSelectKey) {\n    if (!data || !data.key) {\n      return;\n    }\n    this.langSelected.set(data.key as PREVIEW_TEXT_DATA_LANGUAGE_SUPPORT);\n    const extensions = this.getExtensionsByLanguage(this.langSelected());\n    this.editorViewInstance?.setState(\n      EditorState.create({\n        doc: this.editorViewInstance?.state.doc.toString(),\n        ...extensions,\n      })\n    );\n    this.outChange.emit({\n      content: this.content(),\n      isWrap: this.isWrap(),\n      language: this.langSelected(),\n      contextChange: 'language',\n    });\n  }\n}\n","<div\n  class=\"libs-ui-preview-data-container flex flex-col w-full h-auto rounded-[8px] libs-ui-border-general px-[8px]\"\n  [style.--background-color]=\"background()\"\n  [class.pt-[8px]]=\"!hiddenAction()\">\n  @if (!hiddenAction()) {\n    <div class=\"flex items-center content-between color-[#6a7383]\">\n      <libs_ui-components-dropdown\n        classInclude=\"w-[200px]\"\n        [listConfig]=\"configLoadDataIsHttpConfig\"\n        [listMaxItemShow]=\"5\"\n        [isNgContent]=\"true\"\n        [readonly]=\"!editable()\"\n        [listHasButtonUnSelectOption]=\"false\"\n        (outSelectKey)=\"handlerSelectKey($event)\">\n        <libs_ui-components-buttons-button\n          [type]=\"'button-link-third'\"\n          [label]=\"labelLang() || ''\"\n          [sizeButton]=\"'small'\"\n          [classIconRight]=\"editable() ? 'libs-ui-icon-move-right rotate-90' : ''\"\n          [classInclude]=\"'!p-[0px]' + (editable() ? '' : '!pointer-events-none !cursor-default hover:!text-[#6A7383]')\" />\n      </libs_ui-components-dropdown>\n      <div class=\"flex items-center\">\n        <libs_ui-components-buttons-button\n          [type]=\"'button-link-third'\"\n          [label]=\"isWrap() ? 'i18n_remove_line_wrap' : 'i18n_line_wrap'\"\n          [sizeButton]=\"'small'\"\n          [classIconLeft]=\"isWrap() ? 'libs-ui-icon-unwrap' : 'libs-ui-icon-wrap'\"\n          [classInclude]=\"'mo-lib-p-0px mo-lib-mr-16px'\"\n          (outClick)=\"handlerLineWrap()\" />\n        <libs_ui-components-buttons-button\n          [type]=\"'button-link-third'\"\n          [label]=\"'i18n_copy'\"\n          [sizeButton]=\"'small'\"\n          [classIconLeft]=\"'libs-ui-icon-copy'\"\n          [classInclude]=\"'mo-lib-p-0px'\"\n          (outClick)=\"handlerCopy()\" />\n      </div>\n    </div>\n  }\n  <div #containerPreview></div>\n</div>\n"]}
333
+ }], ctorParameters: () => [] });
334
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"preview-text-data.component.js","sourceRoot":"","sources":["../../../../../libs-ui/components/preview-text-data/src/preview-text-data.component.ts","../../../../../libs-ui/components/preview-text-data/src/preview-text-data.component.html"],"names":[],"mappings":"AAAA,OAAO,EAAiB,uBAAuB,EAAE,SAAS,EAAE,QAAQ,EAAE,UAAU,EAAE,MAAM,EAAc,MAAM,EAAE,KAAK,EAAE,KAAK,EAAE,MAAM,EAAE,MAAM,EAAE,SAAS,EAAE,MAAM,eAAe,CAAC;AAC7K,OAAO,EAAc,MAAM,EAAE,UAAU,EAAE,MAAM,kBAAkB,CAAC;AAClE,OAAO,EAAE,WAAW,EAAa,MAAM,mBAAmB,CAAC;AAC3D,OAAO,EAAE,UAAU,EAAE,WAAW,EAAc,MAAM,kBAAkB,CAAC;AACvE,OAAO,EAAE,sCAAsC,EAAE,MAAM,oCAAoC,CAAC;AAC5F,OAAO,EAAkB,iCAAiC,EAAE,MAAM,8BAA8B,CAAC;AACjG,OAAO,EAAE,yBAAyB,EAAE,MAAM,gCAAgC,CAAC;AAC3E,OAAO,EAAE,GAAG,EAAE,MAAM,gBAAgB,CAAC;AACrC,OAAO,EAAE,UAAU,EAAE,MAAM,aAAa,CAAC;AACzC,OAAO,EAAE,qBAAqB,EAAE,+BAA+B,EAAE,gBAAgB,EAAE,eAAe,EAAE,MAAM,4BAA4B,CAAC;;AAWvI,MAAM,OAAO,wCAAwC;IACnD,6EAA6E;IAC7E,qBAAqB;IACrB,6EAA6E;IAErE,kBAAkB,CAAc;IACvB,eAAe,GAAG,IAAI,WAAW,EAAE,CAAC;IACpC,mBAAmB,GAAG,IAAI,WAAW,EAAE,CAAC;IACxC,qBAAqB,GAAG,IAAI,WAAW,EAAE,CAAC;IAC1C,iBAAiB,GAAG,IAAI,WAAW,EAAE,CAAC;IAEvD,+CAA+C;IAC9B,aAAa,GAAG,IAAI,GAAG,EAAqB,CAAC;IAE9D,oDAAoD;IAC5C,aAAa,GAAG,KAAK,CAAC;IAE9B,6EAA6E;IAC7E,oBAAoB;IACpB,6EAA6E;IAE5D,mBAAmB,GAAG,MAAM,CAAC,yBAAyB,CAAC,CAAC;IACxD,UAAU,GAAG,MAAM,CAAC,UAAU,CAAC,CAAC;IAEjD,6EAA6E;IAC7E,oBAAoB;IACpB,6EAA6E;IAE1D,MAAM,GAAG,MAAM,CAAU,IAAI,CAAC,CAAC;IAC/B,0BAA0B,GAAG,+BAA+B,EAAE,CAAC;IAC/D,SAAS,GAAG,QAAQ,CAAC,GAAG,EAAE,CAAC,eAAe,CAAC,IAAI,CAAC,CAAC,IAAI,EAAE,EAAE,CAAC,IAAI,CAAC,EAAE,KAAK,IAAI,CAAC,YAAY,EAAE,CAAC,EAAE,KAAK,CAAC,CAAC;IAEtH,6EAA6E;IAC7E,SAAS;IACT,6EAA6E;IAEpE,OAAO,GAAG,KAAK,CAA6B,EAAE,EAAE;QACvD,SAAS,EAAE,CAAC,KAAc,EAAE,EAAE,CAAC,KAAK,IAAI,EAAE;KAC3C,CAAC,CAAC;IAEM,YAAY,GAAG,KAAK,CAAC,QAAQ,EAAsC,CAAC;IAEpE,QAAQ,GAAG,KAAK,CAA+B,KAAK,EAAE;QAC7D,SAAS,EAAE,CAAC,KAAe,EAAE,EAAE,CAAC,KAAK,IAAI,KAAK;KAC/C,CAAC,CAAC;IAEM,YAAY,GAAG,KAAK,CAA+B,KAAK,EAAE;QACjE,SAAS,EAAE,CAAC,KAAe,EAAE,EAAE,CAAC,KAAK,IAAI,KAAK;KAC/C,CAAC,CAAC;IAEM,kBAAkB,GAAG,KAAK,CAA2C,CAAC,8CAA8C,EAAE,yBAAyB,EAAE,SAAS,EAAE,eAAe,CAAC,EAAE;QACrL,SAAS,EAAE,CAAC,KAAqB,EAAE,EAAE,CAAC,KAAK,IAAI,CAAC,8CAA8C,EAAE,yBAAyB,EAAE,SAAS,EAAE,eAAe,CAAC;KACvJ,CAAC,CAAC;IAEM,UAAU,GAAG,KAAK,CAA6B,SAAS,EAAE;QACjE,SAAS,EAAE,CAAC,KAAc,EAAE,EAAE,CAAC,KAAK,IAAI,SAAS;KAClD,CAAC,CAAC;IAEH,6EAA6E;IAC7E,UAAU;IACV,6EAA6E;IAEpE,SAAS,GAAG,MAAM,EAA0B,CAAC;IAC7C,YAAY,GAAG,MAAM,EAAgB,CAAC;IAE/C,6EAA6E;IAC7E,gBAAgB;IAChB,6EAA6E;IAE5D,gBAAgB,GAAG,SAAS,CAAC,QAAQ,CAA6B,kBAAkB,CAAC,CAAC;IAEvG,6EAA6E;IAC7E,8BAA8B;IAC9B,6EAA6E;IAE7E;QACE,6CAA6C;QAC7C,MAAM,CAAC,GAAG,EAAE;YACV,MAAM,UAAU,GAAG,IAAI,CAAC,OAAO,EAAE,CAAC;YAClC,IAAI,IAAI,CAAC,aAAa,IAAI,IAAI,CAAC,kBAAkB,EAAE,CAAC;gBAClD,MAAM,cAAc,GAAG,IAAI,CAAC,kBAAkB,CAAC,KAAK,CAAC,GAAG,CAAC,QAAQ,EAAE,CAAC;gBACpE,IAAI,UAAU,KAAK,cAAc,EAAE,CAAC;oBAClC,IAAI,CAAC,kBAAkB,CAAC,QAAQ,CAAC;wBAC/B,OAAO,EAAE;4BACP,IAAI,EAAE,CAAC;4BACP,EAAE,EAAE,cAAc,CAAC,MAAM;4BACzB,MAAM,EAAE,UAAU;yBACnB;qBACF,CAAC,CAAC;gBACL,CAAC;YACH,CAAC;QACH,CAAC,CAAC,CAAC;QAEH,mCAAmC;QACnC,IAAI,CAAC,UAAU,CAAC,SAAS,CAAC,GAAG,EAAE;YAC7B,IAAI,CAAC,kBAAkB,EAAE,OAAO,EAAE,CAAC;YACnC,IAAI,CAAC,aAAa,CAAC,KAAK,EAAE,CAAC;QAC7B,CAAC,CAAC,CAAC;IACL,CAAC;IAED,6EAA6E;IAC7E,kBAAkB;IAClB,6EAA6E;IAE7E,KAAK,CAAC,eAAe;QACnB,MAAM,IAAI,CAAC,gBAAgB,EAAE,CAAC;QAC9B,IAAI,CAAC,aAAa,GAAG,IAAI,CAAC;IAC5B,CAAC;IAED,6EAA6E;IAC7E,0CAA0C;IAC1C,6EAA6E;IAE7E;;;OAGG;IACK,KAAK,CAAC,gBAAgB;QAC5B,MAAM,iBAAiB,GAAG,MAAM,IAAI,CAAC,qBAAqB,CAAC,IAAI,CAAC,YAAY,EAAE,CAAC,CAAC;QAChF,MAAM,eAAe,GAAG,MAAM,IAAI,CAAC,mBAAmB,CAAC,IAAI,CAAC,YAAY,EAAE,CAAC,CAAC;QAE5E,IAAI,CAAC,kBAAkB,GAAG,IAAI,UAAU,CAAC;YACvC,GAAG,EAAE,IAAI,CAAC,OAAO,EAAE;YACnB,MAAM,EAAE,IAAI,CAAC,gBAAgB,EAAE,CAAC,aAAa;YAC7C,UAAU,EAAE,IAAI,CAAC,gBAAgB,CAAC,iBAAiB,EAAE,eAAe,CAAC;SACtE,CAAC,CAAC;IACL,CAAC;IAED;;OAEG;IACK,gBAAgB,CAAC,iBAA4B,EAAE,eAA0B;QAC/E,OAAO;YACL,UAAU;YACV,IAAI,CAAC,mBAAmB,CAAC,EAAE,CAAC,iBAAiB,CAAC;YAC9C,IAAI,CAAC,iBAAiB,CAAC,EAAE,CAAC,eAAe,CAAC;YAC1C,IAAI,CAAC,gBAAgB,EAAE;YACvB,UAAU,CAAC,QAAQ,CAAC,EAAE,CAAC,IAAI,CAAC,QAAQ,EAAE,CAAC;YACvC,IAAI,CAAC,eAAe,CAAC,EAAE,CAAC,UAAU,CAAC,YAAY,CAAC;YAChD,IAAI,CAAC,qBAAqB,CAAC,EAAE,CAAC,WAAW,EAAE,CAAC;YAC5C,UAAU,EAAE;YACZ,IAAI,CAAC,kBAAkB,EAAE;YACzB,IAAI,CAAC,oBAAoB,EAAE;SAC5B,CAAC;IACJ,CAAC;IAED;;;OAGG;IACK,KAAK,CAAC,qBAAqB,CAAC,IAAwC;QAC1E,MAAM,SAAS,GAAG,IAAI,CAAC,WAAW,EAAE,CAAC;QAErC,MAAM,MAAM,GAAG,IAAI,CAAC,aAAa,CAAC,GAAG,CAAC,SAAS,CAAC,CAAC;QACjD,IAAI,MAAM,EAAE,CAAC;YACX,OAAO,MAAM,CAAC;QAChB,CAAC;QACD,MAAM,YAAY,GAAG,gBAAgB,CAAC,SAAS,CAAC,CAAC;QACjD,MAAM,SAAS,GAAG,YAAY,CAAC,CAAC,CAAC,MAAM,YAAY,CAAC,MAAM,EAAE,CAAC,CAAC,CAAC,MAAM,qBAAqB,EAAE,CAAC;QAE7F,IAAI,CAAC,aAAa,CAAC,GAAG,CAAC,SAAS,EAAE,SAAS,CAAC,CAAC;QAC7C,OAAO,SAAS,CAAC;IACnB,CAAC;IAED;;OAEG;IACK,KAAK,CAAC,mBAAmB,CAAC,IAAwC;QACxE,MAAM,SAAS,GAAG,IAAI,CAAC,WAAW,EAAE,CAAC;QACrC,MAAM,YAAY,GAAG,gBAAgB,CAAC,SAAS,CAAC,CAAC;QAEjD,IAAI,YAAY,EAAE,MAAM,EAAE,CAAC;YACzB,OAAO,YAAY,CAAC,MAAM,CAAC,IAAI,CAAC,CAAC;QACnC,CAAC;QAED,OAAO,MAAM,CAAC,GAAG,EAAE,CAAC,EAAE,CAAC,CAAC;IAC1B,CAAC;IAED,6EAA6E;IAC7E,wCAAwC;IACxC,6EAA6E;IAErE,gBAAgB;QACtB,OAAO,UAAU,CAAC,KAAK,CAAC;YACtB,GAAG,EAAE,EAAE,eAAe,EAAE,IAAI,CAAC,UAAU,EAAE,EAAE,KAAK,EAAE,MAAM,EAAE;YAC1D,aAAa,EAAE,EAAE,UAAU,EAAE,MAAM,EAAE;YACrC,gBAAgB,EAAE,EAAE,eAAe,EAAE,SAAS,EAAE;SACjD,CAAC,CAAC;IACL,CAAC;IAEO,kBAAkB;QACxB,OAAO,UAAU,CAAC,KAAK,CAAC;YACtB,qBAAqB,EAAE;gBACrB,eAAe,EAAE,2BAA2B;gBAC5C,MAAM,EAAE,oCAAoC;gBAC5C,YAAY,EAAE,KAAK;gBACnB,UAAU,EAAE,eAAe;aAC5B;YACD,wBAAwB,EAAE;gBACxB,eAAe,EAAE,2BAA2B;gBAC5C,MAAM,EAAE,oCAAoC;gBAC5C,YAAY,EAAE,KAAK;aACpB;SACF,CAAC,CAAC;IACL,CAAC;IAED,6EAA6E;IAC7E,oCAAoC;IACpC,6EAA6E;IAErE,oBAAoB;QAC1B,OAAO,UAAU,CAAC,cAAc,CAAC,EAAE,CAAC,CAAC,MAAkB,EAAE,EAAE;YACzD,IAAI,MAAM,CAAC,UAAU,IAAI,IAAI,CAAC,QAAQ,EAAE,EAAE,CAAC;gBACzC,MAAM,QAAQ,GAAG,MAAM,CAAC,KAAK,CAAC,GAAG,CAAC,QAAQ,EAAE,CAAC;gBAC7C,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC;oBAClB,OAAO,EAAE,QAAQ;oBACjB,MAAM,EAAE,IAAI,CAAC,MAAM,EAAE;oBACrB,QAAQ,EAAE,IAAI,CAAC,YAAY,EAAE;oBAC7B,aAAa,EAAE,SAAS;iBACzB,CAAC,CAAC;YACL,CAAC;QACH,CAAC,CAAC,CAAC;IACL,CAAC;IAED,6EAA6E;IAC7E,2DAA2D;IAC3D,6EAA6E;IAE7E;;;OAGG;IACH,cAAc;QACZ,OAAO,MAAM,CAAC,CAAC,IAAI,EAAE,EAAE;YACrB,MAAM,IAAI,GAAG,IAAI,CAAC,KAAK,CAAC,GAAG,CAAC,QAAQ,EAAE,CAAC;YACvC,MAAM,WAAW,GAAiB,EAAE,CAAC;YAErC,IAAI,CAAC;gBACH,IAAI,QAAQ,CAAC,IAAI,CAAC,CAAC;YACrB,CAAC;YAAC,OAAO,GAAY,EAAE,CAAC;gBACtB,MAAM,OAAO,GAAG,GAAG,CAAC,GAAG,EAAE,SAAS,EAAE,EAAE,CAAC,CAAC;gBACxC,MAAM,YAAY,GAAG,IAAI,CAAC,kBAAkB,EAAE,CAAC,IAAI,CAAC,CAAC,OAAO,EAAE,EAAE,CAAC,OAAO,CAAC,QAAQ,CAAC,OAAO,CAAC,CAAC,CAAC;gBAE5F,IAAI,CAAC,YAAY,EAAE,CAAC;oBAClB,WAAW,CAAC,IAAI,CAAC;wBACf,IAAI,EAAE,CAAC;wBACP,EAAE,EAAE,IAAI,CAAC,MAAM;wBACf,QAAQ,EAAE,OAAO;wBACjB,OAAO;qBACR,CAAC,CAAC;oBACH,IAAI,CAAC,YAAY,CAAC,IAAI,CAAC,WAAW,CAAC,CAAC;gBACtC,CAAC;YACH,CAAC;YAED,OAAO,WAAW,CAAC;QACrB,CAAC,CAAC,CAAC;IACL,CAAC;IAED;;;OAGG;IACH,gBAAgB;QACd,OAAO,MAAM,CAAC,CAAC,IAAI,EAAE,EAAE;YACrB,MAAM,IAAI,GAAG,IAAI,CAAC,KAAK,CAAC,GAAG,CAAC,QAAQ,EAAE,CAAC;YACvC,MAAM,WAAW,GAAiB,EAAE,CAAC;YAErC,IAAI,CAAC;gBACH,IAAI,CAAC,KAAK,CAAC,IAAI,CAAC,CAAC;YACnB,CAAC;YAAC,OAAO,GAAY,EAAE,CAAC;gBACtB,WAAW,CAAC,IAAI,CAAC;oBACf,IAAI,EAAE,CAAC;oBACP,EAAE,EAAE,IAAI,CAAC,MAAM;oBACf,QAAQ,EAAE,OAAO;oBACjB,OAAO,EAAE,GAAG,CAAC,GAAG,EAAE,SAAS,EAAE,EAAE,CAAC;iBACjC,CAAC,CAAC;YACL,CAAC;YAED,IAAI,CAAC,YAAY,CAAC,IAAI,CAAC,WAAW,CAAC,CAAC;YACpC,OAAO,WAAW,CAAC;QACrB,CAAC,CAAC,CAAC;IACL,CAAC;IAED;;;OAGG;IACH,eAAe;QACb,OAAO,MAAM,CAAC,KAAK,EAAE,IAAI,EAAE,EAAE;YAC3B,MAAM,IAAI,GAAG,IAAI,CAAC,KAAK,CAAC,GAAG,CAAC,QAAQ,EAAE,CAAC;YACvC,MAAM,WAAW,GAAiB,EAAE,CAAC;YAErC,IAAI,CAAC;gBACH,2CAA2C;gBAC3C,MAAM,EAAE,MAAM,EAAE,GAAG,MAAM,MAAM,CAAC,iBAAiB,CAAC,CAAC;gBACnD,MAAM,SAAS,GAAG,IAAI,MAAM,EAAE,CAAC;gBAC/B,SAAS,CAAC,MAAM,CAAC,IAAI,CAAC,CAAC;YACzB,CAAC;YAAC,OAAO,GAAY,EAAE,CAAC;gBACtB,WAAW,CAAC,IAAI,CAAC;oBACf,IAAI,EAAE,CAAC;oBACP,EAAE,EAAE,IAAI,CAAC,MAAM;oBACf,QAAQ,EAAE,OAAO;oBACjB,OAAO,EAAE,GAAG,CAAC,GAAG,EAAE,SAAS,EAAE,EAAE,CAAC;iBACjC,CAAC,CAAC;YACL,CAAC;YAED,IAAI,CAAC,YAAY,CAAC,IAAI,CAAC,WAAW,CAAC,CAAC;YACpC,OAAO,WAAW,CAAC;QACrB,CAAC,CAAC,CAAC;IACL,CAAC;IAED,6EAA6E;IAC7E,qCAAqC;IACrC,6EAA6E;IAEnE,WAAW;QACnB,SAAS,CAAC,SAAS,CAAC,SAAS,CAAC,IAAI,CAAC,OAAO,EAAE,IAAI,EAAE,CAAC,CAAC;QACpD,IAAI,CAAC,mBAAmB,CAAC,oBAAoB,CAAC,qBAAqB,CAAC,CAAC;IACvE,CAAC;IAES,eAAe;QACvB,OAAO,CAAC,GAAG,CAAC,iBAAiB,EAAE,IAAI,CAAC,MAAM,EAAE,CAAC,CAAC;QAC9C,0DAA0D;QAC1D,MAAM,cAAc,GAAG,IAAI,CAAC,kBAAkB,EAAE,KAAK,CAAC,GAAG,CAAC,QAAQ,EAAE,IAAI,EAAE,CAAC;QAC3E,MAAM,cAAc,GAAG,IAAI,CAAC,MAAM,EAAE,IAAI,cAAc,CAAC,CAAC,CAAC,cAAc,CAAC,OAAO,CAAC,KAAK,EAAE,GAAG,CAAC,CAAC,CAAC,CAAC,IAAI,CAAC,OAAO,EAAE,CAAC;QAC7G,IAAI,CAAC,MAAM,CAAC,MAAM,CAAC,CAAC,GAAG,EAAE,EAAE,CAAC,CAAC,GAAG,CAAC,CAAC;QAClC,OAAO,CAAC,GAAG,CAAC,gBAAgB,EAAE,IAAI,CAAC,MAAM,EAAE,CAAC,CAAC;QAC7C,OAAO,CAAC,GAAG,CAAC,gBAAgB,EAAE,cAAc,CAAC,CAAC;QAC9C,OAAO,CAAC,GAAG,CAAC,gBAAgB,EAAE,cAAc,CAAC,CAAC;QAE9C,IAAI,cAAc,KAAK,cAAc,EAAE,CAAC;YACtC,IAAI,CAAC,kBAAkB,EAAE,QAAQ,CAAC;gBAChC,OAAO,EAAE,EAAE,IAAI,EAAE,CAAC,EAAE,EAAE,EAAE,cAAc,CAAC,MAAM,EAAE,MAAM,EAAE,cAAc,EAAE;aACxE,CAAC,CAAC;QACL,CAAC;QAED,oDAAoD;QACpD,IAAI,CAAC,kBAAkB,EAAE,QAAQ,CAAC;YAChC,OAAO,EAAE,CAAC,IAAI,CAAC,eAAe,CAAC,WAAW,CAAC,IAAI,CAAC,MAAM,EAAE,CAAC,CAAC,CAAC,UAAU,CAAC,YAAY,CAAC,CAAC,CAAC,EAAE,CAAC,EAAE,IAAI,CAAC,qBAAqB,CAAC,WAAW,CAAC,IAAI,CAAC,MAAM,EAAE,CAAC,CAAC,CAAC,WAAW,EAAE,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC;SACtK,CAAC,CAAC;QAEH,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC;YAClB,OAAO,EAAE,cAAc;YACvB,MAAM,EAAE,IAAI,CAAC,MAAM,EAAE;YACrB,QAAQ,EAAE,IAAI,CAAC,YAAY,EAAE;YAC7B,aAAa,EAAE,QAAQ;SACxB,CAAC,CAAC;IACL,CAAC;IAES,KAAK,CAAC,gBAAgB,CAAC,IAAqB;QACpD,IAAI,CAAC,IAAI,EAAE,GAAG,EAAE,CAAC;YACf,OAAO;QACT,CAAC;QAED,MAAM,OAAO,GAAG,IAAI,CAAC,GAAyC,CAAC;QAC/D,IAAI,CAAC,YAAY,CAAC,GAAG,CAAC,OAAO,CAAC,CAAC;QAE/B,8BAA8B;QAC9B,MAAM,CAAC,iBAAiB,EAAE,eAAe,CAAC,GAAG,MAAM,OAAO,CAAC,GAAG,CAAC,CAAC,IAAI,CAAC,qBAAqB,CAAC,OAAO,CAAC,EAAE,IAAI,CAAC,mBAAmB,CAAC,OAAO,CAAC,CAAC,CAAC,CAAC;QAEzI,gEAAgE;QAChE,IAAI,CAAC,kBAAkB,EAAE,QAAQ,CAAC;YAChC,OAAO,EAAE,CAAC,IAAI,CAAC,mBAAmB,CAAC,WAAW,CAAC,iBAAiB,CAAC,EAAE,IAAI,CAAC,iBAAiB,CAAC,WAAW,CAAC,eAAe,CAAC,CAAC;SACxH,CAAC,CAAC;QAEH,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC;YAClB,OAAO,EAAE,IAAI,CAAC,OAAO,EAAE;YACvB,MAAM,EAAE,IAAI,CAAC,MAAM,EAAE;YACrB,QAAQ,EAAE,IAAI,CAAC,YAAY,EAAE;YAC7B,aAAa,EAAE,UAAU;SAC1B,CAAC,CAAC;IACL,CAAC;wGAnXU,wCAAwC;4FAAxC,wCAAwC,upCCpBrD,g0DAyCA,o3BDvBY,iCAAiC,i1DAAE,sCAAsC;;4FAExE,wCAAwC;kBATpD,SAAS;+BAEE,sCAAsC,cAGpC,IAAI,mBACC,uBAAuB,CAAC,MAAM,WACtC,CAAC,iCAAiC,EAAE,sCAAsC,CAAC","sourcesContent":["import { AfterViewInit, ChangeDetectionStrategy, Component, computed, DestroyRef, effect, ElementRef, inject, input, model, output, signal, viewChild } from '@angular/core';\nimport { Diagnostic, linter, lintGutter } from '@codemirror/lint';\nimport { Compartment, Extension } from '@codemirror/state';\nimport { EditorView, lineNumbers, ViewUpdate } from '@codemirror/view';\nimport { LibsUiComponentsButtonsButtonComponent } from '@libs-ui/components-buttons-button';\nimport { IEmitSelectKey, LibsUiComponentsDropdownComponent } from '@libs-ui/components-dropdown';\nimport { LibsUiNotificationService } from '@libs-ui/services-notification';\nimport { get } from '@libs-ui/utils';\nimport { basicSetup } from 'codemirror6';\nimport { createDefaultLanguage, httpRequestConfigGetOptionsLang, languageRegistry, optionsLangData } from './preview-text-data.define';\nimport { IPreviewTextDataChange, PREVIEW_TEXT_DATA_LANGUAGE_SUPPORT } from './preview-text-data.interfaces';\n@Component({\n  // eslint-disable-next-line @angular-eslint/component-selector\n  selector: 'libs_ui-components-preview_text_data',\n  templateUrl: './preview-text-data.component.html',\n  styleUrls: ['./preview-text-data.component.scss'],\n  standalone: true,\n  changeDetection: ChangeDetectionStrategy.OnPush,\n  imports: [LibsUiComponentsDropdownComponent, LibsUiComponentsButtonsButtonComponent],\n})\nexport class LibsUiComponentsPreviewTextDataComponent implements AfterViewInit {\n  // ==========================================================================\n  // PRIVATE PROPERTIES\n  // ==========================================================================\n\n  private editorViewInstance?: EditorView;\n  private readonly wrapCompartment = new Compartment();\n  private readonly languageCompartment = new Compartment();\n  private readonly lineNumberCompartment = new Compartment();\n  private readonly linterCompartment = new Compartment();\n\n  /** Cache để tránh load lại language đã load */\n  private readonly languageCache = new Map<string, Extension>();\n\n  /** Flag để tránh update editor khi đang khởi tạo */\n  private isInitialized = false;\n\n  // ==========================================================================\n  // INJECTED SERVICES\n  // ==========================================================================\n\n  private readonly notificationService = inject(LibsUiNotificationService);\n  private readonly destroyRef = inject(DestroyRef);\n\n  // ==========================================================================\n  // PROTECTED SIGNALS\n  // ==========================================================================\n\n  protected readonly isWrap = signal<boolean>(true);\n  protected readonly configLoadDataIsHttpConfig = httpRequestConfigGetOptionsLang();\n  protected readonly labelLang = computed(() => optionsLangData.find((item) => item.id === this.langSelected())?.label);\n\n  // ==========================================================================\n  // INPUTS\n  // ==========================================================================\n\n  readonly content = input<string, string | undefined>('', {\n    transform: (value?: string) => value ?? '',\n  });\n\n  readonly langSelected = model.required<PREVIEW_TEXT_DATA_LANGUAGE_SUPPORT>();\n\n  readonly editable = input<boolean, boolean | undefined>(false, {\n    transform: (value?: boolean) => value ?? false,\n  });\n\n  readonly hiddenAction = input<boolean, boolean | undefined>(false, {\n    transform: (value?: boolean) => value ?? false,\n  });\n\n  readonly lintIgnorePatterns = input<Array<string>, Array<string> | undefined>(['Cannot use import statement outside a module', 'Unexpected token export', 'import ', '@angular/core'], {\n    transform: (value?: Array<string>) => value ?? ['Cannot use import statement outside a module', 'Unexpected token export', 'import ', '@angular/core'],\n  });\n\n  readonly background = input<string, string | undefined>('#f8f9fa', {\n    transform: (value?: string) => value ?? '#f8f9fa',\n  });\n\n  // ==========================================================================\n  // OUTPUTS\n  // ==========================================================================\n\n  readonly outChange = output<IPreviewTextDataChange>();\n  readonly syntaxErrors = output<Diagnostic[]>();\n\n  // ==========================================================================\n  // VIEW CHILDREN\n  // ==========================================================================\n\n  private readonly containerPreview = viewChild.required<ElementRef<HTMLDivElement>>('containerPreview');\n\n  // ==========================================================================\n  // CONSTRUCTOR - Setup effects\n  // ==========================================================================\n\n  constructor() {\n    // Effect để sync content từ input vào editor\n    effect(() => {\n      const newContent = this.content();\n      if (this.isInitialized && this.editorViewInstance) {\n        const currentContent = this.editorViewInstance.state.doc.toString();\n        if (newContent !== currentContent) {\n          this.editorViewInstance.dispatch({\n            changes: {\n              from: 0,\n              to: currentContent.length,\n              insert: newContent,\n            },\n          });\n        }\n      }\n    });\n\n    // Cleanup khi component bị destroy\n    this.destroyRef.onDestroy(() => {\n      this.editorViewInstance?.destroy();\n      this.languageCache.clear();\n    });\n  }\n\n  // ==========================================================================\n  // LIFECYCLE HOOKS\n  // ==========================================================================\n\n  async ngAfterViewInit(): Promise<void> {\n    await this.initializeEditor();\n    this.isInitialized = true;\n  }\n\n  // ==========================================================================\n  // PRIVATE METHODS - Editor initialization\n  // ==========================================================================\n\n  /**\n   * Khởi tạo editor với các extensions cơ bản\n   * Language được load async để tối ưu bundle size\n   */\n  private async initializeEditor(): Promise<void> {\n    const languageExtension = await this.loadLanguageExtension(this.langSelected());\n    const linterExtension = await this.loadLinterExtension(this.langSelected());\n\n    this.editorViewInstance = new EditorView({\n      doc: this.content(),\n      parent: this.containerPreview().nativeElement,\n      extensions: this.createExtensions(languageExtension, linterExtension),\n    });\n  }\n\n  /**\n   * Tạo danh sách extensions cho editor\n   */\n  private createExtensions(languageExtension: Extension, linterExtension: Extension): Extension[] {\n    return [\n      basicSetup,\n      this.languageCompartment.of(languageExtension),\n      this.linterCompartment.of(linterExtension),\n      this.createLightTheme(),\n      EditorView.editable.of(this.editable()),\n      this.wrapCompartment.of(EditorView.lineWrapping),\n      this.lineNumberCompartment.of(lineNumbers()),\n      lintGutter(),\n      this.createBracketTheme(),\n      this.createUpdateListener(),\n    ];\n  }\n\n  /**\n   * Load language extension với caching\n   * Mỗi language chỉ được load 1 lần\n   */\n  private async loadLanguageExtension(lang: PREVIEW_TEXT_DATA_LANGUAGE_SUPPORT): Promise<Extension> {\n    const lowerLang = lang.toLowerCase();\n\n    const cached = this.languageCache.get(lowerLang);\n    if (cached) {\n      return cached;\n    }\n    const registryItem = languageRegistry[lowerLang];\n    const extension = registryItem ? await registryItem.loader() : await createDefaultLanguage();\n\n    this.languageCache.set(lowerLang, extension);\n    return extension;\n  }\n\n  /**\n   * Load linter extension nếu có\n   */\n  private async loadLinterExtension(lang: PREVIEW_TEXT_DATA_LANGUAGE_SUPPORT): Promise<Extension> {\n    const lowerLang = lang.toLowerCase();\n    const registryItem = languageRegistry[lowerLang];\n\n    if (registryItem?.linter) {\n      return registryItem.linter(this);\n    }\n\n    return linter(() => []);\n  }\n\n  // ==========================================================================\n  // PRIVATE METHODS - Theme configuration\n  // ==========================================================================\n\n  private createLightTheme(): Extension {\n    return EditorView.theme({\n      '&': { backgroundColor: this.background(), color: '#333' },\n      '.cm-content': { caretColor: '#111' },\n      '.cm-activeLine': { backgroundColor: '#f0f0f0' },\n    });\n  }\n\n  private createBracketTheme(): Extension {\n    return EditorView.theme({\n      '.cm-matchingBracket': {\n        backgroundColor: 'rgba(100, 200, 255, 0.15)',\n        border: '1px solid rgba(100, 200, 255, 0.4)',\n        borderRadius: '3px',\n        transition: 'all 0.2s ease',\n      },\n      '.cm-nonmatchingBracket': {\n        backgroundColor: 'rgba(255, 100, 100, 0.15)',\n        border: '1px solid rgba(255, 100, 100, 0.3)',\n        borderRadius: '3px',\n      },\n    });\n  }\n\n  // ==========================================================================\n  // PRIVATE METHODS - Update listener\n  // ==========================================================================\n\n  private createUpdateListener(): Extension {\n    return EditorView.updateListener.of((update: ViewUpdate) => {\n      if (update.docChanged && this.editable()) {\n        const newValue = update.state.doc.toString();\n        this.outChange.emit({\n          content: newValue,\n          isWrap: this.isWrap(),\n          language: this.langSelected(),\n          contextChange: 'content',\n        });\n      }\n    });\n  }\n\n  // ==========================================================================\n  // PUBLIC METHODS - Linter factories (được gọi từ registry)\n  // ==========================================================================\n\n  /**\n   * Tạo JavaScript linter\n   * Sử dụng Function constructor để validate syntax\n   */\n  createJsLinter(): Extension {\n    return linter((view) => {\n      const text = view.state.doc.toString();\n      const diagnostics: Diagnostic[] = [];\n\n      try {\n        new Function(text);\n      } catch (err: unknown) {\n        const message = get(err, 'message', '');\n        const shouldIgnore = this.lintIgnorePatterns().some((pattern) => message.includes(pattern));\n\n        if (!shouldIgnore) {\n          diagnostics.push({\n            from: 0,\n            to: text.length,\n            severity: 'error',\n            message,\n          });\n          this.syntaxErrors.emit(diagnostics);\n        }\n      }\n\n      return diagnostics;\n    });\n  }\n\n  /**\n   * Tạo JSON linter\n   * Sử dụng JSON.parse để validate\n   */\n  createJsonLinter(): Extension {\n    return linter((view) => {\n      const text = view.state.doc.toString();\n      const diagnostics: Diagnostic[] = [];\n\n      try {\n        JSON.parse(text);\n      } catch (err: unknown) {\n        diagnostics.push({\n          from: 0,\n          to: text.length,\n          severity: 'error',\n          message: get(err, 'message', ''),\n        });\n      }\n\n      this.syntaxErrors.emit(diagnostics);\n      return diagnostics;\n    });\n  }\n\n  /**\n   * Tạo SQL linter\n   * Lazy load node-sql-parser chỉ khi cần\n   */\n  createSqlLinter(): Extension {\n    return linter(async (view) => {\n      const text = view.state.doc.toString();\n      const diagnostics: Diagnostic[] = [];\n\n      try {\n        // Lazy load SQL parser chỉ khi thực sự cần\n        const { Parser } = await import('node-sql-parser');\n        const sqlParser = new Parser();\n        sqlParser.astify(text);\n      } catch (err: unknown) {\n        diagnostics.push({\n          from: 0,\n          to: text.length,\n          severity: 'error',\n          message: get(err, 'message', ''),\n        });\n      }\n\n      this.syntaxErrors.emit(diagnostics);\n      return diagnostics;\n    });\n  }\n\n  // ==========================================================================\n  // PROTECTED METHODS - Event handlers\n  // ==========================================================================\n\n  protected handlerCopy(): void {\n    navigator.clipboard.writeText(this.content() || '');\n    this.notificationService.showCompTypeTextInfo('Sao chép thành công');\n  }\n\n  protected handlerLineWrap(): void {\n    console.log('handlerLineWrap', this.isWrap());\n    // Lấy nội dung mới nhất từ editor sau khi đã xóa newlines\n    const currentContent = this.editorViewInstance?.state.doc.toString() || '';\n    const updatedContent = this.isWrap() && currentContent ? currentContent.replace(/\\n/g, ' ') : this.content();\n    this.isWrap.update((val) => !val);\n    console.log('updatedContent', this.isWrap());\n    console.log('currentContent', currentContent);\n    console.log('updatedContent', updatedContent);\n\n    if (currentContent !== updatedContent) {\n      this.editorViewInstance?.dispatch({\n        changes: { from: 0, to: currentContent.length, insert: updatedContent },\n      });\n    }\n\n    // Batch cả 2 dispatch thành 1 để tối ưu performance\n    this.editorViewInstance?.dispatch({\n      effects: [this.wrapCompartment.reconfigure(this.isWrap() ? EditorView.lineWrapping : []), this.lineNumberCompartment.reconfigure(this.isWrap() ? lineNumbers() : [])],\n    });\n\n    this.outChange.emit({\n      content: updatedContent,\n      isWrap: this.isWrap(),\n      language: this.langSelected(),\n      contextChange: 'isWrap',\n    });\n  }\n\n  protected async handlerSelectKey(data?: IEmitSelectKey): Promise<void> {\n    if (!data?.key) {\n      return;\n    }\n\n    const newLang = data.key as PREVIEW_TEXT_DATA_LANGUAGE_SUPPORT;\n    this.langSelected.set(newLang);\n\n    // Load language và linter mới\n    const [languageExtension, linterExtension] = await Promise.all([this.loadLanguageExtension(newLang), this.loadLinterExtension(newLang)]);\n\n    // Reconfigure thay vì tạo mới state để giữ lại document content\n    this.editorViewInstance?.dispatch({\n      effects: [this.languageCompartment.reconfigure(languageExtension), this.linterCompartment.reconfigure(linterExtension)],\n    });\n\n    this.outChange.emit({\n      content: this.content(),\n      isWrap: this.isWrap(),\n      language: this.langSelected(),\n      contextChange: 'language',\n    });\n  }\n}\n","<div\n  class=\"libs-ui-preview-data-container flex flex-col w-full h-auto rounded-[8px] libs-ui-border-general px-[8px]\"\n  [style.--background-color]=\"background()\"\n  [class.pt-[8px]]=\"!hiddenAction()\">\n  @if (!hiddenAction()) {\n    <div class=\"flex items-center content-between color-[#6a7383]\">\n      <libs_ui-components-dropdown\n        classInclude=\"w-[200px]\"\n        [listConfig]=\"configLoadDataIsHttpConfig\"\n        [listMaxItemShow]=\"5\"\n        [isNgContent]=\"true\"\n        [readonly]=\"!editable()\"\n        [listHasButtonUnSelectOption]=\"false\"\n        (outSelectKey)=\"handlerSelectKey($event)\">\n        <libs_ui-components-buttons-button\n          [type]=\"'button-link-third'\"\n          [label]=\"labelLang() || ''\"\n          [sizeButton]=\"'small'\"\n          [classIconRight]=\"editable() ? 'libs-ui-icon-move-right rotate-90' : ''\"\n          [classInclude]=\"'!p-[0px]' + (editable() ? '' : '!pointer-events-none !cursor-default hover:!text-[#6A7383]')\" />\n      </libs_ui-components-dropdown>\n      <div class=\"flex items-center\">\n        <libs_ui-components-buttons-button\n          [type]=\"'button-link-third'\"\n          [label]=\"isWrap() ? 'i18n_remove_line_wrap' : 'i18n_line_wrap'\"\n          [sizeButton]=\"'small'\"\n          [classIconLeft]=\"isWrap() ? 'libs-ui-icon-unwrap' : 'libs-ui-icon-wrap'\"\n          [classInclude]=\"'mo-lib-p-0px mo-lib-mr-16px'\"\n          (outClick)=\"handlerLineWrap()\" />\n        <libs_ui-components-buttons-button\n          [type]=\"'button-link-third'\"\n          [label]=\"'i18n_copy'\"\n          [sizeButton]=\"'small'\"\n          [classIconLeft]=\"'libs-ui-icon-copy'\"\n          [classInclude]=\"'mo-lib-p-0px'\"\n          (outClick)=\"handlerCopy()\" />\n      </div>\n    </div>\n  }\n  <div #containerPreview></div>\n</div>\n"]}