cloud-ide-element 1.0.4 → 1.0.5

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 (54) hide show
  1. package/README.md +63 -271
  2. package/fesm2022/cloud-ide-element.mjs +614 -618
  3. package/fesm2022/cloud-ide-element.mjs.map +1 -1
  4. package/index.d.ts +1362 -3
  5. package/package.json +3 -30
  6. package/esm2022/cloud-ide-element.mjs +0 -5
  7. package/esm2022/lib/components/confirmation-modal/confirmation-modal.component.mjs +0 -182
  8. package/esm2022/lib/components/data-grid/data-grid.component.mjs +0 -1363
  9. package/esm2022/lib/components/data-grid/data-grid.types.mjs +0 -37
  10. package/esm2022/lib/components/dropdown/dropdown.component.mjs +0 -396
  11. package/esm2022/lib/components/global-notifications/global-notifications.component.mjs +0 -30
  12. package/esm2022/lib/components/json-editor/json-editor.component.mjs +0 -521
  13. package/esm2022/lib/components/skeleton-loader/skeleton-loader.component.mjs +0 -33
  14. package/esm2022/lib/components/toast-notification/toast-notification.component.mjs +0 -152
  15. package/esm2022/lib/elements/button/cide-ele-button.component.mjs +0 -249
  16. package/esm2022/lib/elements/file-input/file-input.component.mjs +0 -83
  17. package/esm2022/lib/elements/icon/icon.component.mjs +0 -26
  18. package/esm2022/lib/elements/input/input.component.mjs +0 -467
  19. package/esm2022/lib/elements/select/select.component.mjs +0 -471
  20. package/esm2022/lib/elements/spinner/spinner.component.mjs +0 -32
  21. package/esm2022/lib/elements/tab/cide-ele-tab.component.mjs +0 -74
  22. package/esm2022/lib/elements/textarea/textarea.component.mjs +0 -157
  23. package/esm2022/lib/services/confirmation.service.mjs +0 -151
  24. package/esm2022/lib/services/dropdown-manager.service.mjs +0 -93
  25. package/esm2022/lib/services/notification.service.mjs +0 -196
  26. package/esm2022/lib/utils/directives/resizer/resizer.directive.mjs +0 -231
  27. package/esm2022/lib/utils/directives/tooltip/tooltip.directive.mjs +0 -294
  28. package/esm2022/lib/utils/pipes/capitalize/capitalize.pipe.mjs +0 -38
  29. package/esm2022/lib/utils/services/elements/elements.service.mjs +0 -39
  30. package/esm2022/public-api.mjs +0 -31
  31. package/lib/components/confirmation-modal/confirmation-modal.component.d.ts +0 -16
  32. package/lib/components/data-grid/data-grid.component.d.ts +0 -244
  33. package/lib/components/data-grid/data-grid.types.d.ts +0 -146
  34. package/lib/components/dropdown/dropdown.component.d.ts +0 -75
  35. package/lib/components/global-notifications/global-notifications.component.d.ts +0 -5
  36. package/lib/components/json-editor/json-editor.component.d.ts +0 -116
  37. package/lib/components/skeleton-loader/skeleton-loader.component.d.ts +0 -11
  38. package/lib/components/toast-notification/toast-notification.component.d.ts +0 -13
  39. package/lib/elements/button/cide-ele-button.component.d.ts +0 -85
  40. package/lib/elements/file-input/file-input.component.d.ts +0 -25
  41. package/lib/elements/icon/icon.component.d.ts +0 -9
  42. package/lib/elements/input/input.component.d.ts +0 -182
  43. package/lib/elements/select/select.component.d.ts +0 -91
  44. package/lib/elements/spinner/spinner.component.d.ts +0 -19
  45. package/lib/elements/tab/cide-ele-tab.component.d.ts +0 -26
  46. package/lib/elements/textarea/textarea.component.d.ts +0 -47
  47. package/lib/services/confirmation.service.d.ts +0 -65
  48. package/lib/services/dropdown-manager.service.d.ts +0 -22
  49. package/lib/services/notification.service.d.ts +0 -81
  50. package/lib/utils/directives/resizer/resizer.directive.d.ts +0 -44
  51. package/lib/utils/directives/tooltip/tooltip.directive.d.ts +0 -43
  52. package/lib/utils/pipes/capitalize/capitalize.pipe.d.ts +0 -7
  53. package/lib/utils/services/elements/elements.service.d.ts +0 -12
  54. package/public-api.d.ts +0 -22
@@ -1,521 +0,0 @@
1
- import { Component, Input, Output, EventEmitter, signal, computed, inject, DestroyRef, forwardRef, viewChild } from '@angular/core';
2
- import { CommonModule } from '@angular/common';
3
- import { NG_VALUE_ACCESSOR } from '@angular/forms';
4
- import { FormsModule } from '@angular/forms';
5
- import * as i0 from "@angular/core";
6
- export class CideEleJsonEditorComponent {
7
- constructor() {
8
- this.required = false;
9
- this.disabled = signal(false);
10
- this.showCharacterCount = false;
11
- this.config = {};
12
- this.configSignal = signal({});
13
- // Outputs
14
- this.valueChange = new EventEmitter();
15
- this.objectChange = new EventEmitter();
16
- this.errorsChange = new EventEmitter();
17
- this.validChange = new EventEmitter();
18
- // ViewChild
19
- this.jsonTextarea = viewChild.required('jsonTextarea');
20
- // Dependency injection
21
- this.destroyRef = inject(DestroyRef);
22
- // Internal state
23
- this.jsonString = signal('');
24
- this.jsonObject = signal(null);
25
- this.errors = signal([]);
26
- this.editorId = `json-editor-${Math.random().toString(36).substr(2, 9)}`;
27
- // Computed values
28
- this.hasErrors = computed(() => this.errors().length > 0);
29
- this.isValid = computed(() => !this.hasErrors() && this.jsonObject() !== null);
30
- this.lineNumbers = computed(() => {
31
- const lines = this.jsonString().split('\n');
32
- return Array.from({ length: Math.max(lines.length, 1) }, (_, i) => i + 1);
33
- });
34
- this.containerClass = computed(() => {
35
- const classes = ['tw-w-full'];
36
- if (this.disabled()) {
37
- classes.push('tw-opacity-50');
38
- }
39
- return classes.join(' ');
40
- });
41
- // Default configuration
42
- this.defaultConfig = {
43
- syntaxHighlighting: true,
44
- showLineNumbers: true,
45
- autoFormat: true,
46
- validateOnChange: true,
47
- theme: 'light',
48
- minHeight: 120,
49
- maxHeight: 400,
50
- placeholder: 'Enter JSON configuration...',
51
- readOnly: false,
52
- showErrors: true
53
- };
54
- this.onChange = () => { };
55
- this.onTouched = () => { };
56
- }
57
- ngOnInit() {
58
- // Merge default config with input config
59
- const mergedConfig = { ...this.defaultConfig, ...this.config };
60
- this.configSignal.set(mergedConfig);
61
- // Initial validation
62
- if (mergedConfig.validateOnChange && this.jsonString()) {
63
- this.validateJson();
64
- }
65
- }
66
- /**
67
- * Handle input changes
68
- */
69
- onInput(event) {
70
- const target = event.target;
71
- const value = target.value;
72
- this.jsonString.set(value);
73
- this.valueChange.emit(value);
74
- // Call ControlValueAccessor onChange callback
75
- this.onChange(value);
76
- if (this.configSignal().validateOnChange) {
77
- this.validateJson();
78
- }
79
- this.parseJson(value);
80
- }
81
- /**
82
- * Handle blur event
83
- */
84
- onBlur() {
85
- // Call ControlValueAccessor onTouched callback
86
- this.onTouched();
87
- if (this.configSignal().autoFormat && this.isValid()) {
88
- this.formatJson();
89
- }
90
- }
91
- /**
92
- * Parse JSON string and emit object
93
- */
94
- parseJson(jsonStr) {
95
- if (!jsonStr.trim()) {
96
- this.jsonObject.set(null);
97
- this.objectChange.emit(null);
98
- return;
99
- }
100
- try {
101
- const parsed = JSON.parse(jsonStr);
102
- this.jsonObject.set(parsed);
103
- this.objectChange.emit(parsed);
104
- }
105
- catch (error) {
106
- // Error will be handled by validateJson
107
- this.jsonObject.set(null);
108
- this.objectChange.emit(null);
109
- }
110
- }
111
- /**
112
- * Validate JSON and update errors
113
- */
114
- validateJson() {
115
- const jsonStr = this.jsonString();
116
- const newErrors = [];
117
- if (!jsonStr.trim()) {
118
- this.errors.set(newErrors);
119
- this.errorsChange.emit(newErrors);
120
- this.validChange.emit(true);
121
- return;
122
- }
123
- try {
124
- JSON.parse(jsonStr);
125
- // JSON is valid
126
- this.errors.set(newErrors);
127
- this.errorsChange.emit(newErrors);
128
- this.validChange.emit(true);
129
- }
130
- catch (error) {
131
- // Parse error - extract line and column information
132
- const errorMessage = error.message;
133
- const lineMatch = errorMessage.match(/position (\d+)/);
134
- if (lineMatch) {
135
- const position = parseInt(lineMatch[1]);
136
- const lines = jsonStr.substring(0, position).split('\n');
137
- const line = lines.length;
138
- const column = lines[lines.length - 1].length + 1;
139
- newErrors.push({
140
- line,
141
- column,
142
- message: 'Invalid JSON syntax',
143
- type: 'syntax'
144
- });
145
- }
146
- else {
147
- newErrors.push({
148
- line: 1,
149
- column: 1,
150
- message: 'Invalid JSON format',
151
- type: 'syntax'
152
- });
153
- }
154
- this.errors.set(newErrors);
155
- this.errorsChange.emit(newErrors);
156
- this.validChange.emit(false);
157
- }
158
- }
159
- /**
160
- * Format JSON with proper indentation
161
- */
162
- formatJson() {
163
- if (!this.jsonString().trim()) {
164
- return;
165
- }
166
- try {
167
- const parsed = JSON.parse(this.jsonString());
168
- const formatted = JSON.stringify(parsed, null, 2);
169
- this.jsonString.set(formatted);
170
- this.valueChange.emit(formatted);
171
- this.jsonObject.set(parsed);
172
- this.objectChange.emit(parsed);
173
- this.errors.set([]);
174
- this.errorsChange.emit([]);
175
- this.validChange.emit(true);
176
- }
177
- catch (error) {
178
- // Don't format if JSON is invalid
179
- console.warn('Cannot format invalid JSON');
180
- }
181
- }
182
- /**
183
- * Check if there's an error on a specific line
184
- */
185
- hasErrorOnLine(line) {
186
- return this.errors().some(error => error.line === line);
187
- }
188
- /**
189
- * Get error ID for accessibility
190
- */
191
- errorId() {
192
- return `${this.editorId}-errors`;
193
- }
194
- /**
195
- * Focus the editor
196
- */
197
- focus() {
198
- this.jsonTextarea().nativeElement.focus();
199
- }
200
- /**
201
- * Clear the editor
202
- */
203
- clear() {
204
- this.jsonString.set('');
205
- this.jsonObject.set(null);
206
- this.errors.set([]);
207
- this.valueChange.emit('');
208
- this.objectChange.emit(null);
209
- this.errorsChange.emit([]);
210
- this.validChange.emit(true);
211
- }
212
- /**
213
- * Get current JSON object
214
- */
215
- getValue() {
216
- return this.jsonObject();
217
- }
218
- /**
219
- * Get current JSON string
220
- */
221
- getStringValue() {
222
- return this.jsonString();
223
- }
224
- /**
225
- * Check if JSON is valid
226
- */
227
- isValidJson() {
228
- return this.isValid();
229
- }
230
- /**
231
- * Get validation errors
232
- */
233
- getErrors() {
234
- return this.errors();
235
- }
236
- // ControlValueAccessor methods
237
- writeValue(value) {
238
- if (value === null || value === undefined) {
239
- this.jsonString.set('');
240
- this.jsonObject.set(null);
241
- }
242
- else if (typeof value === 'string') {
243
- this.jsonString.set(value);
244
- this.parseJson(value);
245
- }
246
- else if (typeof value === 'object') {
247
- this.jsonObject.set(value);
248
- this.jsonString.set(JSON.stringify(value, null, 2));
249
- }
250
- }
251
- registerOnChange(fn) {
252
- this.onChange = fn;
253
- }
254
- registerOnTouched(fn) {
255
- this.onTouched = fn;
256
- }
257
- setDisabledState(isDisabled) {
258
- this.disabled.set(isDisabled);
259
- }
260
- static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.7", ngImport: i0, type: CideEleJsonEditorComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
261
- static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "18.2.7", type: CideEleJsonEditorComponent, isStandalone: true, selector: "cide-ele-json-editor", inputs: { label: "label", helperText: "helperText", required: "required", disabled: "disabled", showCharacterCount: "showCharacterCount", config: "config" }, outputs: { valueChange: "valueChange", objectChange: "objectChange", errorsChange: "errorsChange", validChange: "validChange" }, providers: [
262
- {
263
- provide: NG_VALUE_ACCESSOR,
264
- useExisting: forwardRef(() => CideEleJsonEditorComponent),
265
- multi: true
266
- }
267
- ], viewQueries: [{ propertyName: "jsonTextarea", first: true, predicate: ["jsonTextarea"], descendants: true, isSignal: true }], ngImport: i0, template: `
268
- <div class="tw-relative tw-w-full" [class]="containerClass()">
269
- <!-- Label -->
270
- @if (label) {
271
- <label
272
- [for]="editorId"
273
- class="tw-block tw-text-sm tw-font-medium tw-text-gray-700 tw-mb-2"
274
- [class.tw-text-red-600]="hasErrors()">
275
- {{ label }}
276
- @if (required) {
277
- <span class="tw-text-red-500 tw-ml-1">*</span>
278
- }
279
- </label>
280
- }
281
-
282
- <!-- Editor Container -->
283
- <div
284
- class="tw-relative tw-border tw-rounded-md tw-bg-white tw-overflow-hidden"
285
- [class.tw-border-red-300]="hasErrors()"
286
- [class.tw-border-gray-300]="!hasErrors()"
287
- [class.tw-focus-within:tw-border-blue-500]="!hasErrors()"
288
- [class.tw-focus-within:tw-ring-1]="!hasErrors()"
289
- [class.tw-focus-within:tw-ring-blue-500]="!hasErrors()"
290
- [class.tw-opacity-50]="disabled()"
291
- [class.tw-cursor-not-allowed]="disabled()">
292
-
293
- <!-- Line Numbers -->
294
- @if (configSignal().showLineNumbers) {
295
- <div
296
- class="tw-absolute tw-left-0 tw-top-0 tw-bottom-0 tw-w-12 tw-bg-gray-50 tw-border-r tw-border-gray-200 tw-text-xs tw-text-gray-500 tw-font-mono tw-select-none tw-overflow-hidden"
297
- [class.tw-bg-gray-800]="configSignal().theme === 'dark'"
298
- [class.tw-border-gray-600]="configSignal().theme === 'dark'"
299
- [class.tw-text-gray-400]="configSignal().theme === 'dark'">
300
- @for (line of lineNumbers(); track line) {
301
- <div
302
- class="tw-px-2 tw-py-1 tw-text-right"
303
- [class.tw-bg-red-100]="hasErrorOnLine(line)"
304
- [class.tw-text-red-600]="hasErrorOnLine(line)"
305
- [class.tw-bg-red-900]="configSignal().theme === 'dark' && hasErrorOnLine(line)"
306
- [class.tw-text-red-400]="configSignal().theme === 'dark' && hasErrorOnLine(line)">
307
- {{ line }}
308
- </div>
309
- }
310
- </div>
311
- }
312
-
313
- <!-- Textarea -->
314
- <textarea
315
- #jsonTextarea
316
- [id]="editorId"
317
- [value]="jsonString()"
318
- (input)="onInput($event)"
319
- (blur)="onBlur()"
320
- [placeholder]="configSignal().placeholder || 'Enter JSON configuration...'"
321
- [readonly]="configSignal().readOnly || disabled()"
322
- [disabled]="disabled()"
323
- class="tw-w-full tw-p-3 tw-text-sm tw-font-mono tw-leading-relaxed tw-resize-none tw-outline-none tw-bg-transparent"
324
- [class.tw-pl-12]="configSignal().showLineNumbers"
325
- [class.tw-text-gray-900]="configSignal().theme === 'light'"
326
- [class.tw-text-gray-100]="configSignal().theme === 'dark'"
327
- [class.tw-bg-gray-900]="configSignal().theme === 'dark'"
328
- [style.min-height.px]="configSignal().minHeight || 120"
329
- [style.max-height.px]="configSignal().maxHeight || 400"
330
- [attr.aria-describedby]="errorId()"
331
- [attr.aria-invalid]="hasErrors()">
332
- </textarea>
333
-
334
- <!-- Format Button -->
335
- @if (!configSignal().readOnly && !disabled()) {
336
- <button
337
- type="button"
338
- (click)="formatJson()"
339
- class="tw-absolute tw-top-2 tw-right-2 tw-p-1 tw-text-gray-400 hover:tw-text-gray-600 tw-rounded tw-transition-colors"
340
- [class.tw-text-gray-500]="configSignal().theme === 'dark'"
341
- [class.hover:tw-text-gray-300]="configSignal().theme === 'dark'"
342
- title="Format JSON">
343
- <svg class="tw-w-4 tw-h-4" fill="none" stroke="currentColor" viewBox="0 0 24 24">
344
- <path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M4 6h16M4 12h16M4 18h7"></path>
345
- </svg>
346
- </button>
347
- }
348
- </div>
349
-
350
- <!-- Helper Text -->
351
- @if (helperText && !hasErrors()) {
352
- <p class="tw-mt-1 tw-text-xs tw-text-gray-500">{{ helperText }}</p>
353
- }
354
-
355
- <!-- Error Messages -->
356
- @if (configSignal().showErrors && hasErrors()) {
357
- <div [id]="errorId()" class="tw-mt-1 tw-space-y-1">
358
- @for (error of errors(); track error.line + '-' + error.column) {
359
- <p class="tw-text-xs tw-text-red-600 tw-flex tw-items-start tw-gap-1">
360
- <svg class="tw-w-3 tw-h-3 tw-mt-0.5 tw-flex-shrink-0" fill="currentColor" viewBox="0 0 20 20">
361
- <path fill-rule="evenodd" d="M18 10a8 8 0 11-16 0 8 8 0 0116 0zm-7 4a1 1 0 11-2 0 1 1 0 012 0zm-1-9a1 1 0 00-1 1v4a1 1 0 102 0V6a1 1 0 00-1-1z" clip-rule="evenodd"></path>
362
- </svg>
363
- <span>
364
- Line {{ error.line }}, Column {{ error.column }}: {{ error.message }}
365
- </span>
366
- </p>
367
- }
368
- </div>
369
- }
370
-
371
- <!-- Character Count -->
372
- @if (showCharacterCount) {
373
- <p class="tw-mt-1 tw-text-xs tw-text-gray-400 tw-text-right">
374
- {{ jsonString().length }} characters
375
- </p>
376
- }
377
- </div>
378
- `, isInline: true, styles: [":host{display:block}textarea{font-family:Monaco,Menlo,Ubuntu Mono,monospace;tab-size:2}textarea:focus{box-shadow:none}.tw-cursor-not-allowed{cursor:not-allowed}\n"], dependencies: [{ kind: "ngmodule", type: CommonModule }, { kind: "ngmodule", type: FormsModule }] }); }
379
- }
380
- i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.7", ngImport: i0, type: CideEleJsonEditorComponent, decorators: [{
381
- type: Component,
382
- args: [{ selector: 'cide-ele-json-editor', standalone: true, imports: [CommonModule, FormsModule], template: `
383
- <div class="tw-relative tw-w-full" [class]="containerClass()">
384
- <!-- Label -->
385
- @if (label) {
386
- <label
387
- [for]="editorId"
388
- class="tw-block tw-text-sm tw-font-medium tw-text-gray-700 tw-mb-2"
389
- [class.tw-text-red-600]="hasErrors()">
390
- {{ label }}
391
- @if (required) {
392
- <span class="tw-text-red-500 tw-ml-1">*</span>
393
- }
394
- </label>
395
- }
396
-
397
- <!-- Editor Container -->
398
- <div
399
- class="tw-relative tw-border tw-rounded-md tw-bg-white tw-overflow-hidden"
400
- [class.tw-border-red-300]="hasErrors()"
401
- [class.tw-border-gray-300]="!hasErrors()"
402
- [class.tw-focus-within:tw-border-blue-500]="!hasErrors()"
403
- [class.tw-focus-within:tw-ring-1]="!hasErrors()"
404
- [class.tw-focus-within:tw-ring-blue-500]="!hasErrors()"
405
- [class.tw-opacity-50]="disabled()"
406
- [class.tw-cursor-not-allowed]="disabled()">
407
-
408
- <!-- Line Numbers -->
409
- @if (configSignal().showLineNumbers) {
410
- <div
411
- class="tw-absolute tw-left-0 tw-top-0 tw-bottom-0 tw-w-12 tw-bg-gray-50 tw-border-r tw-border-gray-200 tw-text-xs tw-text-gray-500 tw-font-mono tw-select-none tw-overflow-hidden"
412
- [class.tw-bg-gray-800]="configSignal().theme === 'dark'"
413
- [class.tw-border-gray-600]="configSignal().theme === 'dark'"
414
- [class.tw-text-gray-400]="configSignal().theme === 'dark'">
415
- @for (line of lineNumbers(); track line) {
416
- <div
417
- class="tw-px-2 tw-py-1 tw-text-right"
418
- [class.tw-bg-red-100]="hasErrorOnLine(line)"
419
- [class.tw-text-red-600]="hasErrorOnLine(line)"
420
- [class.tw-bg-red-900]="configSignal().theme === 'dark' && hasErrorOnLine(line)"
421
- [class.tw-text-red-400]="configSignal().theme === 'dark' && hasErrorOnLine(line)">
422
- {{ line }}
423
- </div>
424
- }
425
- </div>
426
- }
427
-
428
- <!-- Textarea -->
429
- <textarea
430
- #jsonTextarea
431
- [id]="editorId"
432
- [value]="jsonString()"
433
- (input)="onInput($event)"
434
- (blur)="onBlur()"
435
- [placeholder]="configSignal().placeholder || 'Enter JSON configuration...'"
436
- [readonly]="configSignal().readOnly || disabled()"
437
- [disabled]="disabled()"
438
- class="tw-w-full tw-p-3 tw-text-sm tw-font-mono tw-leading-relaxed tw-resize-none tw-outline-none tw-bg-transparent"
439
- [class.tw-pl-12]="configSignal().showLineNumbers"
440
- [class.tw-text-gray-900]="configSignal().theme === 'light'"
441
- [class.tw-text-gray-100]="configSignal().theme === 'dark'"
442
- [class.tw-bg-gray-900]="configSignal().theme === 'dark'"
443
- [style.min-height.px]="configSignal().minHeight || 120"
444
- [style.max-height.px]="configSignal().maxHeight || 400"
445
- [attr.aria-describedby]="errorId()"
446
- [attr.aria-invalid]="hasErrors()">
447
- </textarea>
448
-
449
- <!-- Format Button -->
450
- @if (!configSignal().readOnly && !disabled()) {
451
- <button
452
- type="button"
453
- (click)="formatJson()"
454
- class="tw-absolute tw-top-2 tw-right-2 tw-p-1 tw-text-gray-400 hover:tw-text-gray-600 tw-rounded tw-transition-colors"
455
- [class.tw-text-gray-500]="configSignal().theme === 'dark'"
456
- [class.hover:tw-text-gray-300]="configSignal().theme === 'dark'"
457
- title="Format JSON">
458
- <svg class="tw-w-4 tw-h-4" fill="none" stroke="currentColor" viewBox="0 0 24 24">
459
- <path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M4 6h16M4 12h16M4 18h7"></path>
460
- </svg>
461
- </button>
462
- }
463
- </div>
464
-
465
- <!-- Helper Text -->
466
- @if (helperText && !hasErrors()) {
467
- <p class="tw-mt-1 tw-text-xs tw-text-gray-500">{{ helperText }}</p>
468
- }
469
-
470
- <!-- Error Messages -->
471
- @if (configSignal().showErrors && hasErrors()) {
472
- <div [id]="errorId()" class="tw-mt-1 tw-space-y-1">
473
- @for (error of errors(); track error.line + '-' + error.column) {
474
- <p class="tw-text-xs tw-text-red-600 tw-flex tw-items-start tw-gap-1">
475
- <svg class="tw-w-3 tw-h-3 tw-mt-0.5 tw-flex-shrink-0" fill="currentColor" viewBox="0 0 20 20">
476
- <path fill-rule="evenodd" d="M18 10a8 8 0 11-16 0 8 8 0 0116 0zm-7 4a1 1 0 11-2 0 1 1 0 012 0zm-1-9a1 1 0 00-1 1v4a1 1 0 102 0V6a1 1 0 00-1-1z" clip-rule="evenodd"></path>
477
- </svg>
478
- <span>
479
- Line {{ error.line }}, Column {{ error.column }}: {{ error.message }}
480
- </span>
481
- </p>
482
- }
483
- </div>
484
- }
485
-
486
- <!-- Character Count -->
487
- @if (showCharacterCount) {
488
- <p class="tw-mt-1 tw-text-xs tw-text-gray-400 tw-text-right">
489
- {{ jsonString().length }} characters
490
- </p>
491
- }
492
- </div>
493
- `, providers: [
494
- {
495
- provide: NG_VALUE_ACCESSOR,
496
- useExisting: forwardRef(() => CideEleJsonEditorComponent),
497
- multi: true
498
- }
499
- ], styles: [":host{display:block}textarea{font-family:Monaco,Menlo,Ubuntu Mono,monospace;tab-size:2}textarea:focus{box-shadow:none}.tw-cursor-not-allowed{cursor:not-allowed}\n"] }]
500
- }], propDecorators: { label: [{
501
- type: Input
502
- }], helperText: [{
503
- type: Input
504
- }], required: [{
505
- type: Input
506
- }], disabled: [{
507
- type: Input
508
- }], showCharacterCount: [{
509
- type: Input
510
- }], config: [{
511
- type: Input
512
- }], valueChange: [{
513
- type: Output
514
- }], objectChange: [{
515
- type: Output
516
- }], errorsChange: [{
517
- type: Output
518
- }], validChange: [{
519
- type: Output
520
- }] } });
521
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"json-editor.component.js","sourceRoot":"","sources":["../../../../../../projects/cloud-ide-element/src/lib/components/json-editor/json-editor.component.ts"],"names":[],"mappings":"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","sourcesContent":["import { Component, Input, Output, EventEmitter, OnInit, ElementRef, signal, computed, inject, DestroyRef, forwardRef, viewChild } from '@angular/core';\r\nimport { CommonModule } from '@angular/common';\r\nimport { ControlValueAccessor, NG_VALUE_ACCESSOR } from '@angular/forms';\r\nimport { FormsModule } from '@angular/forms';\r\n\r\nexport interface JsonEditorConfig {\r\n  /** Enable syntax highlighting */\r\n  syntaxHighlighting?: boolean;\r\n  /** Enable line numbers */\r\n  showLineNumbers?: boolean;\r\n  /** Enable auto-formatting */\r\n  autoFormat?: boolean;\r\n  /** Enable validation */\r\n  validateOnChange?: boolean;\r\n  /** Theme for the editor */\r\n  theme?: 'light' | 'dark';\r\n  /** Minimum height in pixels */\r\n  minHeight?: number;\r\n  /** Maximum height in pixels */\r\n  maxHeight?: number;\r\n  /** Placeholder text */\r\n  placeholder?: string;\r\n  /** Read-only mode */\r\n  readOnly?: boolean;\r\n  /** Show error messages */\r\n  showErrors?: boolean;\r\n}\r\n\r\nexport interface JsonEditorError {\r\n  line: number;\r\n  column: number;\r\n  message: string;\r\n  type: 'syntax' | 'validation';\r\n}\r\n\r\n@Component({\r\n  selector: 'cide-ele-json-editor',\r\n  standalone: true,\r\n  imports: [CommonModule, FormsModule],\r\n  template: `\r\n    <div class=\"tw-relative tw-w-full\" [class]=\"containerClass()\">\r\n      <!-- Label -->\r\n      @if (label) {\r\n        <label \r\n          [for]=\"editorId\" \r\n          class=\"tw-block tw-text-sm tw-font-medium tw-text-gray-700 tw-mb-2\"\r\n          [class.tw-text-red-600]=\"hasErrors()\">\r\n          {{ label }}\r\n          @if (required) {\r\n            <span class=\"tw-text-red-500 tw-ml-1\">*</span>\r\n          }\r\n        </label>\r\n      }\r\n\r\n      <!-- Editor Container -->\r\n      <div \r\n        class=\"tw-relative tw-border tw-rounded-md tw-bg-white tw-overflow-hidden\"\r\n        [class.tw-border-red-300]=\"hasErrors()\"\r\n        [class.tw-border-gray-300]=\"!hasErrors()\"\r\n        [class.tw-focus-within:tw-border-blue-500]=\"!hasErrors()\"\r\n        [class.tw-focus-within:tw-ring-1]=\"!hasErrors()\"\r\n        [class.tw-focus-within:tw-ring-blue-500]=\"!hasErrors()\"\r\n        [class.tw-opacity-50]=\"disabled()\"\r\n        [class.tw-cursor-not-allowed]=\"disabled()\">\r\n        \r\n        <!-- Line Numbers -->\r\n        @if (configSignal().showLineNumbers) {\r\n          <div \r\n            class=\"tw-absolute tw-left-0 tw-top-0 tw-bottom-0 tw-w-12 tw-bg-gray-50 tw-border-r tw-border-gray-200 tw-text-xs tw-text-gray-500 tw-font-mono tw-select-none tw-overflow-hidden\"\r\n            [class.tw-bg-gray-800]=\"configSignal().theme === 'dark'\"\r\n            [class.tw-border-gray-600]=\"configSignal().theme === 'dark'\"\r\n            [class.tw-text-gray-400]=\"configSignal().theme === 'dark'\">\r\n            @for (line of lineNumbers(); track line) {\r\n              <div \r\n                class=\"tw-px-2 tw-py-1 tw-text-right\"\r\n                [class.tw-bg-red-100]=\"hasErrorOnLine(line)\"\r\n                [class.tw-text-red-600]=\"hasErrorOnLine(line)\"\r\n                [class.tw-bg-red-900]=\"configSignal().theme === 'dark' && hasErrorOnLine(line)\"\r\n                [class.tw-text-red-400]=\"configSignal().theme === 'dark' && hasErrorOnLine(line)\">\r\n                {{ line }}\r\n              </div>\r\n            }\r\n          </div>\r\n        }\r\n\r\n        <!-- Textarea -->\r\n        <textarea\r\n          #jsonTextarea\r\n          [id]=\"editorId\"\r\n          [value]=\"jsonString()\"\r\n          (input)=\"onInput($event)\"\r\n          (blur)=\"onBlur()\"\r\n          [placeholder]=\"configSignal().placeholder || 'Enter JSON configuration...'\"\r\n          [readonly]=\"configSignal().readOnly || disabled()\"\r\n          [disabled]=\"disabled()\"\r\n          class=\"tw-w-full tw-p-3 tw-text-sm tw-font-mono tw-leading-relaxed tw-resize-none tw-outline-none tw-bg-transparent\"\r\n          [class.tw-pl-12]=\"configSignal().showLineNumbers\"\r\n          [class.tw-text-gray-900]=\"configSignal().theme === 'light'\"\r\n          [class.tw-text-gray-100]=\"configSignal().theme === 'dark'\"\r\n          [class.tw-bg-gray-900]=\"configSignal().theme === 'dark'\"\r\n          [style.min-height.px]=\"configSignal().minHeight || 120\"\r\n          [style.max-height.px]=\"configSignal().maxHeight || 400\"\r\n          [attr.aria-describedby]=\"errorId()\"\r\n          [attr.aria-invalid]=\"hasErrors()\">\r\n        </textarea>\r\n\r\n        <!-- Format Button -->\r\n        @if (!configSignal().readOnly && !disabled()) {\r\n          <button\r\n            type=\"button\"\r\n            (click)=\"formatJson()\"\r\n            class=\"tw-absolute tw-top-2 tw-right-2 tw-p-1 tw-text-gray-400 hover:tw-text-gray-600 tw-rounded tw-transition-colors\"\r\n            [class.tw-text-gray-500]=\"configSignal().theme === 'dark'\"\r\n            [class.hover:tw-text-gray-300]=\"configSignal().theme === 'dark'\"\r\n            title=\"Format JSON\">\r\n            <svg class=\"tw-w-4 tw-h-4\" fill=\"none\" stroke=\"currentColor\" viewBox=\"0 0 24 24\">\r\n              <path stroke-linecap=\"round\" stroke-linejoin=\"round\" stroke-width=\"2\" d=\"M4 6h16M4 12h16M4 18h7\"></path>\r\n            </svg>\r\n          </button>\r\n        }\r\n      </div>\r\n\r\n      <!-- Helper Text -->\r\n      @if (helperText && !hasErrors()) {\r\n        <p class=\"tw-mt-1 tw-text-xs tw-text-gray-500\">{{ helperText }}</p>\r\n      }\r\n\r\n      <!-- Error Messages -->\r\n      @if (configSignal().showErrors && hasErrors()) {\r\n        <div [id]=\"errorId()\" class=\"tw-mt-1 tw-space-y-1\">\r\n          @for (error of errors(); track error.line + '-' + error.column) {\r\n            <p class=\"tw-text-xs tw-text-red-600 tw-flex tw-items-start tw-gap-1\">\r\n              <svg class=\"tw-w-3 tw-h-3 tw-mt-0.5 tw-flex-shrink-0\" fill=\"currentColor\" viewBox=\"0 0 20 20\">\r\n                <path fill-rule=\"evenodd\" d=\"M18 10a8 8 0 11-16 0 8 8 0 0116 0zm-7 4a1 1 0 11-2 0 1 1 0 012 0zm-1-9a1 1 0 00-1 1v4a1 1 0 102 0V6a1 1 0 00-1-1z\" clip-rule=\"evenodd\"></path>\r\n              </svg>\r\n              <span>\r\n                Line {{ error.line }}, Column {{ error.column }}: {{ error.message }}\r\n              </span>\r\n            </p>\r\n          }\r\n        </div>\r\n      }\r\n\r\n      <!-- Character Count -->\r\n      @if (showCharacterCount) {\r\n        <p class=\"tw-mt-1 tw-text-xs tw-text-gray-400 tw-text-right\">\r\n          {{ jsonString().length }} characters\r\n        </p>\r\n      }\r\n    </div>\r\n  `,\r\n  styles: [`\r\n    :host {\r\n      display: block;\r\n    }\r\n    \r\n    textarea {\r\n      font-family: 'Monaco', 'Menlo', 'Ubuntu Mono', monospace;\r\n      tab-size: 2;\r\n    }\r\n    \r\n    textarea:focus {\r\n      box-shadow: none;\r\n    }\r\n    \r\n    .tw-cursor-not-allowed {\r\n      cursor: not-allowed;\r\n    }\r\n  `],\r\n  providers: [\r\n    {\r\n      provide: NG_VALUE_ACCESSOR,\r\n      useExisting: forwardRef(() => CideEleJsonEditorComponent),\r\n      multi: true\r\n    }\r\n  ]\r\n})\r\nexport class CideEleJsonEditorComponent implements OnInit, ControlValueAccessor {\r\n  // Inputs\r\n  @Input() label?: string;\r\n  @Input() helperText?: string;\r\n  @Input() required = false;\r\n  @Input() disabled = signal(false);\r\n  @Input() showCharacterCount = false;\r\n  @Input() config: JsonEditorConfig = {};\r\n  configSignal = signal<JsonEditorConfig>({});\r\n\r\n  // Outputs\r\n  @Output() valueChange = new EventEmitter<string>();\r\n  @Output() objectChange = new EventEmitter<object | null>();\r\n  @Output() errorsChange = new EventEmitter<JsonEditorError[]>();\r\n  @Output() validChange = new EventEmitter<boolean>();\r\n\r\n  // ViewChild\r\n  jsonTextarea = viewChild.required<ElementRef<HTMLTextAreaElement>>('jsonTextarea');\r\n\r\n  // Dependency injection\r\n  private destroyRef = inject(DestroyRef);\r\n\r\n  // Internal state\r\n  jsonString = signal('');\r\n  jsonObject = signal<object | null>(null);\r\n  errors = signal<JsonEditorError[]>([]);\r\n  editorId = `json-editor-${Math.random().toString(36).substr(2, 9)}`;\r\n\r\n  // Computed values\r\n  hasErrors = computed(() => this.errors().length > 0);\r\n  isValid = computed(() => !this.hasErrors() && this.jsonObject() !== null);\r\n  lineNumbers = computed(() => {\r\n    const lines = this.jsonString().split('\\n');\r\n    return Array.from({ length: Math.max(lines.length, 1) }, (_, i) => i + 1);\r\n  });\r\n  containerClass = computed(() => {\r\n    const classes = ['tw-w-full'];\r\n    if (this.disabled()) {\r\n      classes.push('tw-opacity-50');\r\n    }\r\n    return classes.join(' ');\r\n  });\r\n\r\n  // Default configuration\r\n  private defaultConfig: JsonEditorConfig = {\r\n    syntaxHighlighting: true,\r\n    showLineNumbers: true,\r\n    autoFormat: true,\r\n    validateOnChange: true,\r\n    theme: 'light',\r\n    minHeight: 120,\r\n    maxHeight: 400,\r\n    placeholder: 'Enter JSON configuration...',\r\n    readOnly: false,\r\n    showErrors: true\r\n  };\r\n\r\n  ngOnInit(): void {\r\n    // Merge default config with input config\r\n    const mergedConfig = { ...this.defaultConfig, ...this.config };\r\n    this.configSignal.set(mergedConfig);\r\n    \r\n    // Initial validation\r\n    if (mergedConfig.validateOnChange && this.jsonString()) {\r\n      this.validateJson();\r\n    }\r\n  }\r\n\r\n  /**\r\n   * Handle input changes\r\n   */\r\n  onInput(event: Event): void {\r\n    const target = event.target as HTMLTextAreaElement;\r\n    const value = target.value;\r\n    \r\n    this.jsonString.set(value);\r\n    this.valueChange.emit(value);\r\n    \r\n    // Call ControlValueAccessor onChange callback\r\n    this.onChange(value);\r\n    \r\n    if (this.configSignal().validateOnChange) {\r\n      this.validateJson();\r\n    }\r\n    \r\n    this.parseJson(value);\r\n  }\r\n\r\n  /**\r\n   * Handle blur event\r\n   */\r\n  onBlur(): void {\r\n    // Call ControlValueAccessor onTouched callback\r\n    this.onTouched();\r\n    \r\n    if (this.configSignal().autoFormat && this.isValid()) {\r\n      this.formatJson();\r\n    }\r\n  }\r\n\r\n  /**\r\n   * Parse JSON string and emit object\r\n   */\r\n  private parseJson(jsonStr: string): void {\r\n    if (!jsonStr.trim()) {\r\n      this.jsonObject.set(null);\r\n      this.objectChange.emit(null);\r\n      return;\r\n    }\r\n\r\n    try {\r\n      const parsed = JSON.parse(jsonStr);\r\n      this.jsonObject.set(parsed);\r\n      this.objectChange.emit(parsed);\r\n    } catch (error) {\r\n      // Error will be handled by validateJson\r\n      this.jsonObject.set(null);\r\n      this.objectChange.emit(null);\r\n    }\r\n  }\r\n\r\n  /**\r\n   * Validate JSON and update errors\r\n   */\r\n  private validateJson(): void {\r\n    const jsonStr = this.jsonString();\r\n    const newErrors: JsonEditorError[] = [];\r\n\r\n    if (!jsonStr.trim()) {\r\n      this.errors.set(newErrors);\r\n      this.errorsChange.emit(newErrors);\r\n      this.validChange.emit(true);\r\n      return;\r\n    }\r\n\r\n    try {\r\n      JSON.parse(jsonStr);\r\n      // JSON is valid\r\n      this.errors.set(newErrors);\r\n      this.errorsChange.emit(newErrors);\r\n      this.validChange.emit(true);\r\n    } catch (error) {\r\n      // Parse error - extract line and column information\r\n      const errorMessage = (error as Error).message;\r\n      const lineMatch = errorMessage.match(/position (\\d+)/);\r\n      \r\n      if (lineMatch) {\r\n        const position = parseInt(lineMatch[1]);\r\n        const lines = jsonStr.substring(0, position).split('\\n');\r\n        const line = lines.length;\r\n        const column = lines[lines.length - 1].length + 1;\r\n        \r\n        newErrors.push({\r\n          line,\r\n          column,\r\n          message: 'Invalid JSON syntax',\r\n          type: 'syntax'\r\n        });\r\n      } else {\r\n        newErrors.push({\r\n          line: 1,\r\n          column: 1,\r\n          message: 'Invalid JSON format',\r\n          type: 'syntax'\r\n        });\r\n      }\r\n      \r\n      this.errors.set(newErrors);\r\n      this.errorsChange.emit(newErrors);\r\n      this.validChange.emit(false);\r\n    }\r\n  }\r\n\r\n  /**\r\n   * Format JSON with proper indentation\r\n   */\r\n  formatJson(): void {\r\n    if (!this.jsonString().trim()) {\r\n      return;\r\n    }\r\n\r\n    try {\r\n      const parsed = JSON.parse(this.jsonString());\r\n      const formatted = JSON.stringify(parsed, null, 2);\r\n      this.jsonString.set(formatted);\r\n      this.valueChange.emit(formatted);\r\n      this.jsonObject.set(parsed);\r\n      this.objectChange.emit(parsed);\r\n      this.errors.set([]);\r\n      this.errorsChange.emit([]);\r\n      this.validChange.emit(true);\r\n    } catch (error) {\r\n      // Don't format if JSON is invalid\r\n      console.warn('Cannot format invalid JSON');\r\n    }\r\n  }\r\n\r\n  /**\r\n   * Check if there's an error on a specific line\r\n   */\r\n  hasErrorOnLine(line: number): boolean {\r\n    return this.errors().some(error => error.line === line);\r\n  }\r\n\r\n  /**\r\n   * Get error ID for accessibility\r\n   */\r\n  errorId(): string {\r\n    return `${this.editorId}-errors`;\r\n  }\r\n\r\n  /**\r\n   * Focus the editor\r\n   */\r\n  focus(): void {\r\n    this.jsonTextarea().nativeElement.focus();\r\n  }\r\n\r\n  /**\r\n   * Clear the editor\r\n   */\r\n  clear(): void {\r\n    this.jsonString.set('');\r\n    this.jsonObject.set(null);\r\n    this.errors.set([]);\r\n    this.valueChange.emit('');\r\n    this.objectChange.emit(null);\r\n    this.errorsChange.emit([]);\r\n    this.validChange.emit(true);\r\n  }\r\n\r\n  /**\r\n   * Get current JSON object\r\n   */\r\n  getValue(): object | null {\r\n    return this.jsonObject();\r\n  }\r\n\r\n  /**\r\n   * Get current JSON string\r\n   */\r\n  getStringValue(): string {\r\n    return this.jsonString();\r\n  }\r\n\r\n  /**\r\n   * Check if JSON is valid\r\n   */\r\n  isValidJson(): boolean {\r\n    return this.isValid();\r\n  }\r\n\r\n  /**\r\n   * Get validation errors\r\n   */\r\n  getErrors(): JsonEditorError[] {\r\n    return this.errors();\r\n  }\r\n\r\n  // ControlValueAccessor methods\r\n  writeValue(value: string | object | null | undefined): void {\r\n    if (value === null || value === undefined) {\r\n      this.jsonString.set('');\r\n      this.jsonObject.set(null);\r\n    } else if (typeof value === 'string') {\r\n      this.jsonString.set(value);\r\n      this.parseJson(value);\r\n    } else if (typeof value === 'object') {\r\n      this.jsonObject.set(value);\r\n      this.jsonString.set(JSON.stringify(value, null, 2));\r\n    }\r\n  }\r\n\r\n  registerOnChange(fn: (value: string) => void): void {\r\n    this.onChange = fn;\r\n  }\r\n\r\n  registerOnTouched(fn: () => void): void {\r\n    this.onTouched = fn;\r\n  }\r\n\r\n  setDisabledState(isDisabled: boolean): void {\r\n    this.disabled.set(isDisabled);\r\n  }\r\n\r\n  private onChange: (value: string) => void = () => {};\r\n  private onTouched: () => void = () => {};\r\n} "]}
@@ -1,33 +0,0 @@
1
- import { CommonModule } from '@angular/common';
2
- import { Component, Input } from '@angular/core';
3
- import * as i0 from "@angular/core";
4
- import * as i1 from "@angular/common";
5
- export class CideEleSkeletonLoaderComponent {
6
- constructor() {
7
- this.width = '100%';
8
- this.height = '1rem';
9
- this.borderRadius = '0.375rem'; // rounded-md
10
- this.count = 1;
11
- this.circle = false;
12
- this.animation = 'pulse';
13
- }
14
- static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.7", ngImport: i0, type: CideEleSkeletonLoaderComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
15
- static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "18.2.7", type: CideEleSkeletonLoaderComponent, isStandalone: true, selector: "cide-ele-skeleton-loader", inputs: { width: "width", height: "height", borderRadius: "borderRadius", count: "count", circle: "circle", animation: "animation" }, ngImport: i0, template: "<ng-container *ngFor=\"let i of [].constructor(count)\">\r\n <div\r\n [ngClass]=\"[\r\n 'tw-bg-gray-200',\r\n animation === 'pulse' ? 'tw-animate-pulse' : '',\r\n circle ? 'tw-rounded-full' : 'tw-rounded-md'\r\n ]\"\r\n [style.width]=\"width\"\r\n [style.height]=\"height\"\r\n [style.borderRadius]=\"circle ? '50%' : borderRadius\"\r\n class=\"tw-mb-2\"\r\n ></div>\r\n</ng-container>\r\n", styles: [""], dependencies: [{ kind: "ngmodule", type: CommonModule }, { kind: "directive", type: i1.NgClass, selector: "[ngClass]", inputs: ["class", "ngClass"] }, { kind: "directive", type: i1.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }] }); }
16
- }
17
- i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.7", ngImport: i0, type: CideEleSkeletonLoaderComponent, decorators: [{
18
- type: Component,
19
- args: [{ selector: 'cide-ele-skeleton-loader', standalone: true, imports: [CommonModule], template: "<ng-container *ngFor=\"let i of [].constructor(count)\">\r\n <div\r\n [ngClass]=\"[\r\n 'tw-bg-gray-200',\r\n animation === 'pulse' ? 'tw-animate-pulse' : '',\r\n circle ? 'tw-rounded-full' : 'tw-rounded-md'\r\n ]\"\r\n [style.width]=\"width\"\r\n [style.height]=\"height\"\r\n [style.borderRadius]=\"circle ? '50%' : borderRadius\"\r\n class=\"tw-mb-2\"\r\n ></div>\r\n</ng-container>\r\n" }]
20
- }], propDecorators: { width: [{
21
- type: Input
22
- }], height: [{
23
- type: Input
24
- }], borderRadius: [{
25
- type: Input
26
- }], count: [{
27
- type: Input
28
- }], circle: [{
29
- type: Input
30
- }], animation: [{
31
- type: Input
32
- }] } });
33
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoic2tlbGV0b24tbG9hZGVyLmNvbXBvbmVudC5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uLy4uLy4uL3Byb2plY3RzL2Nsb3VkLWlkZS1lbGVtZW50L3NyYy9saWIvY29tcG9uZW50cy9za2VsZXRvbi1sb2FkZXIvc2tlbGV0b24tbG9hZGVyLmNvbXBvbmVudC50cyIsIi4uLy4uLy4uLy4uLy4uLy4uL3Byb2plY3RzL2Nsb3VkLWlkZS1lbGVtZW50L3NyYy9saWIvY29tcG9uZW50cy9za2VsZXRvbi1sb2FkZXIvc2tlbGV0b24tbG9hZGVyLmNvbXBvbmVudC5odG1sIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLE9BQU8sRUFBRSxZQUFZLEVBQUUsTUFBTSxpQkFBaUIsQ0FBQztBQUMvQyxPQUFPLEVBQUUsU0FBUyxFQUFFLEtBQUssRUFBRSxNQUFNLGVBQWUsQ0FBQzs7O0FBU2pELE1BQU0sT0FBTyw4QkFBOEI7SUFQM0M7UUFRVyxVQUFLLEdBQVcsTUFBTSxDQUFDO1FBQ3ZCLFdBQU0sR0FBVyxNQUFNLENBQUM7UUFDeEIsaUJBQVksR0FBVyxVQUFVLENBQUMsQ0FBQyxhQUFhO1FBQ2hELFVBQUssR0FBVyxDQUFDLENBQUM7UUFDbEIsV0FBTSxHQUFZLEtBQUssQ0FBQztRQUN4QixjQUFTLEdBQThCLE9BQU8sQ0FBQztLQUN6RDs4R0FQWSw4QkFBOEI7a0dBQTlCLDhCQUE4QiwwTkNWM0MsMGFBYUEseURETlksWUFBWTs7MkZBR1gsOEJBQThCO2tCQVAxQyxTQUFTOytCQUNFLDBCQUEwQixjQUV4QixJQUFJLFdBQ1AsQ0FBQyxZQUFZLENBQUM7OEJBSWQsS0FBSztzQkFBYixLQUFLO2dCQUNHLE1BQU07c0JBQWQsS0FBSztnQkFDRyxZQUFZO3NCQUFwQixLQUFLO2dCQUNHLEtBQUs7c0JBQWIsS0FBSztnQkFDRyxNQUFNO3NCQUFkLEtBQUs7Z0JBQ0csU0FBUztzQkFBakIsS0FBSyIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IENvbW1vbk1vZHVsZSB9IGZyb20gJ0Bhbmd1bGFyL2NvbW1vbic7XHJcbmltcG9ydCB7IENvbXBvbmVudCwgSW5wdXQgfSBmcm9tICdAYW5ndWxhci9jb3JlJztcclxuXHJcbkBDb21wb25lbnQoe1xyXG4gIHNlbGVjdG9yOiAnY2lkZS1lbGUtc2tlbGV0b24tbG9hZGVyJyxcclxuICB0ZW1wbGF0ZVVybDogJy4vc2tlbGV0b24tbG9hZGVyLmNvbXBvbmVudC5odG1sJyxcclxuICBzdGFuZGFsb25lOiB0cnVlLFxyXG4gIGltcG9ydHM6IFtDb21tb25Nb2R1bGVdLFxyXG4gIHN0eWxlVXJsczogWycuL3NrZWxldG9uLWxvYWRlci5jb21wb25lbnQuc2NzcyddXHJcbn0pXHJcbmV4cG9ydCBjbGFzcyBDaWRlRWxlU2tlbGV0b25Mb2FkZXJDb21wb25lbnQge1xyXG4gIEBJbnB1dCgpIHdpZHRoOiBzdHJpbmcgPSAnMTAwJSc7XHJcbiAgQElucHV0KCkgaGVpZ2h0OiBzdHJpbmcgPSAnMXJlbSc7XHJcbiAgQElucHV0KCkgYm9yZGVyUmFkaXVzOiBzdHJpbmcgPSAnMC4zNzVyZW0nOyAvLyByb3VuZGVkLW1kXHJcbiAgQElucHV0KCkgY291bnQ6IG51bWJlciA9IDE7XHJcbiAgQElucHV0KCkgY2lyY2xlOiBib29sZWFuID0gZmFsc2U7XHJcbiAgQElucHV0KCkgYW5pbWF0aW9uOiAncHVsc2UnIHwgJ3dhdmUnIHwgJ25vbmUnID0gJ3B1bHNlJztcclxufVxyXG4iLCI8bmctY29udGFpbmVyICpuZ0Zvcj1cImxldCBpIG9mIFtdLmNvbnN0cnVjdG9yKGNvdW50KVwiPlxyXG4gIDxkaXZcclxuICAgIFtuZ0NsYXNzXT1cIltcclxuICAgICAgJ3R3LWJnLWdyYXktMjAwJyxcclxuICAgICAgYW5pbWF0aW9uID09PSAncHVsc2UnID8gJ3R3LWFuaW1hdGUtcHVsc2UnIDogJycsXHJcbiAgICAgIGNpcmNsZSA/ICd0dy1yb3VuZGVkLWZ1bGwnIDogJ3R3LXJvdW5kZWQtbWQnXHJcbiAgICBdXCJcclxuICAgIFtzdHlsZS53aWR0aF09XCJ3aWR0aFwiXHJcbiAgICBbc3R5bGUuaGVpZ2h0XT1cImhlaWdodFwiXHJcbiAgICBbc3R5bGUuYm9yZGVyUmFkaXVzXT1cImNpcmNsZSA/ICc1MCUnIDogYm9yZGVyUmFkaXVzXCJcclxuICAgIGNsYXNzPVwidHctbWItMlwiXHJcbiAgPjwvZGl2PlxyXG48L25nLWNvbnRhaW5lcj5cclxuIl19