@firestitch/markdown-editor 18.0.0 → 18.0.1
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/app/modules/markdown-editor/components/markdown-editor/markdown-editor.component.d.ts +4 -4
- package/app/modules/markdown-editor/services/milkdown-loader.service.d.ts +2 -2
- package/assets/milkdown/milkdown.js +993 -0
- package/assets/milkdown/theme/common/block-edit.css +131 -0
- package/assets/milkdown/theme/common/code-mirror.css +333 -0
- package/assets/milkdown/theme/common/cursor.css +22 -0
- package/assets/milkdown/theme/common/image-block.css +384 -0
- package/assets/milkdown/theme/common/latex.css +44 -0
- package/assets/milkdown/theme/common/link-tooltip.css +107 -0
- package/assets/milkdown/theme/common/list-item.css +50 -0
- package/assets/milkdown/theme/common/placeholder.css +11 -0
- package/assets/milkdown/theme/common/table.css +219 -0
- package/assets/milkdown/theme/common/toolbar.css +50 -0
- package/assets/milkdown/theme/nord/style.css +29 -0
- package/esm2022/app/modules/markdown-editor/components/markdown-editor/markdown-editor.component.mjs +16 -15
- package/esm2022/app/modules/markdown-editor/services/milkdown-loader.service.mjs +52 -43
- package/fesm2022/firestitch-markdown-editor.mjs +66 -56
- package/fesm2022/firestitch-markdown-editor.mjs.map +1 -1
- package/package.json +1 -3
|
@@ -0,0 +1,219 @@
|
|
|
1
|
+
@import '@milkdown/kit/prose/tables/style/tables.css';
|
|
2
|
+
|
|
3
|
+
.milkdown .milkdown-table-block {
|
|
4
|
+
display: block;
|
|
5
|
+
margin: 4px 0;
|
|
6
|
+
}
|
|
7
|
+
|
|
8
|
+
.milkdown .milkdown-table-block th,
|
|
9
|
+
.milkdown .milkdown-table-block td {
|
|
10
|
+
border: 1px solid
|
|
11
|
+
color-mix(in srgb, var(--crepe-color-outline), transparent 80%);
|
|
12
|
+
padding: 4px 16px;
|
|
13
|
+
}
|
|
14
|
+
|
|
15
|
+
.milkdown .milkdown-table-block th .ProseMirror-selectednode, .milkdown .milkdown-table-block td .ProseMirror-selectednode {
|
|
16
|
+
background-color: transparent !important;
|
|
17
|
+
}
|
|
18
|
+
|
|
19
|
+
.milkdown .milkdown-table-block th:has(.ProseMirror-selectednode), .milkdown .milkdown-table-block td:has(.ProseMirror-selectednode) {
|
|
20
|
+
outline: 1px solid var(--crepe-color-primary);
|
|
21
|
+
outline-offset: -1px;
|
|
22
|
+
}
|
|
23
|
+
|
|
24
|
+
.milkdown .milkdown-table-block .selectedCell::after {
|
|
25
|
+
background-color: var(--crepe-color-selected);
|
|
26
|
+
opacity: 0.4;
|
|
27
|
+
}
|
|
28
|
+
|
|
29
|
+
.milkdown .milkdown-table-block .selectedCell ::-moz-selection {
|
|
30
|
+
background: transparent;
|
|
31
|
+
}
|
|
32
|
+
|
|
33
|
+
.milkdown .milkdown-table-block .selectedCell ::selection {
|
|
34
|
+
background: transparent;
|
|
35
|
+
}
|
|
36
|
+
|
|
37
|
+
.milkdown .milkdown-table-block .drag-preview {
|
|
38
|
+
background-color: var(--crepe-color-surface);
|
|
39
|
+
opacity: 0.4;
|
|
40
|
+
position: absolute;
|
|
41
|
+
z-index: 100;
|
|
42
|
+
display: flex;
|
|
43
|
+
flex-direction: column;
|
|
44
|
+
outline: 1px solid var(--crepe-color-primary);
|
|
45
|
+
outline-offset: -1px;
|
|
46
|
+
}
|
|
47
|
+
|
|
48
|
+
.milkdown .milkdown-table-block .drag-preview[data-show='false'] {
|
|
49
|
+
display: none;
|
|
50
|
+
}
|
|
51
|
+
|
|
52
|
+
.milkdown .milkdown-table-block .drag-preview th:has(.ProseMirror-selectednode), .milkdown .milkdown-table-block .drag-preview td:has(.ProseMirror-selectednode) {
|
|
53
|
+
outline: none;
|
|
54
|
+
}
|
|
55
|
+
|
|
56
|
+
.milkdown .milkdown-table-block .handle {
|
|
57
|
+
position: absolute;
|
|
58
|
+
font-size: 14px;
|
|
59
|
+
transition: opacity ease-in-out 0.2s;
|
|
60
|
+
}
|
|
61
|
+
|
|
62
|
+
.milkdown .milkdown-table-block .handle[data-show='false'] {
|
|
63
|
+
opacity: 0;
|
|
64
|
+
}
|
|
65
|
+
|
|
66
|
+
.milkdown .milkdown-table-block svg {
|
|
67
|
+
fill: var(--crepe-color-outline);
|
|
68
|
+
}
|
|
69
|
+
|
|
70
|
+
.milkdown .milkdown-table-block .cell-handle {
|
|
71
|
+
z-index: 50;
|
|
72
|
+
left: -999px;
|
|
73
|
+
top: -999px;
|
|
74
|
+
cursor: grab;
|
|
75
|
+
background-color: var(--crepe-color-surface);
|
|
76
|
+
color: var(--crepe-color-outline);
|
|
77
|
+
border-radius: 100px;
|
|
78
|
+
box-shadow: var(--crepe-shadow-1);
|
|
79
|
+
transition: background-color 0.2s ease-in-out;
|
|
80
|
+
}
|
|
81
|
+
|
|
82
|
+
.milkdown .milkdown-table-block .cell-handle:hover {
|
|
83
|
+
background-color: var(--crepe-color-hover);
|
|
84
|
+
}
|
|
85
|
+
|
|
86
|
+
.milkdown .milkdown-table-block .cell-handle:has(.button-group:hover) {
|
|
87
|
+
background-color: var(--crepe-color-surface);
|
|
88
|
+
}
|
|
89
|
+
|
|
90
|
+
.milkdown .milkdown-table-block .cell-handle[data-role='col-drag-handle'] {
|
|
91
|
+
transform: translateY(50%);
|
|
92
|
+
padding: 0 6px;
|
|
93
|
+
width: 28px;
|
|
94
|
+
height: 16px;
|
|
95
|
+
}
|
|
96
|
+
|
|
97
|
+
.milkdown .milkdown-table-block .cell-handle[data-role='row-drag-handle'] {
|
|
98
|
+
transform: translateX(50%);
|
|
99
|
+
padding: 6px 0;
|
|
100
|
+
width: 16px;
|
|
101
|
+
height: 28px;
|
|
102
|
+
}
|
|
103
|
+
|
|
104
|
+
.milkdown .milkdown-table-block .cell-handle .button-group {
|
|
105
|
+
position: absolute;
|
|
106
|
+
transform: translateX(-50%);
|
|
107
|
+
left: 50%;
|
|
108
|
+
top: -52px;
|
|
109
|
+
display: flex;
|
|
110
|
+
background-color: var(--crepe-color-surface);
|
|
111
|
+
border-radius: 8px;
|
|
112
|
+
box-shadow: var(--crepe-shadow-1);
|
|
113
|
+
}
|
|
114
|
+
|
|
115
|
+
.milkdown .milkdown-table-block .cell-handle .button-group::after {
|
|
116
|
+
content: '';
|
|
117
|
+
position: absolute;
|
|
118
|
+
bottom: -8px;
|
|
119
|
+
height: 8px;
|
|
120
|
+
background-color: transparent;
|
|
121
|
+
width: 100%;
|
|
122
|
+
}
|
|
123
|
+
|
|
124
|
+
.milkdown .milkdown-table-block .cell-handle .button-group[data-show='false'] {
|
|
125
|
+
display: none;
|
|
126
|
+
}
|
|
127
|
+
|
|
128
|
+
.milkdown .milkdown-table-block .cell-handle .button-group button {
|
|
129
|
+
cursor: pointer;
|
|
130
|
+
margin: 6px;
|
|
131
|
+
padding: 4px;
|
|
132
|
+
display: flex;
|
|
133
|
+
justify-content: center;
|
|
134
|
+
align-items: center;
|
|
135
|
+
border-radius: 4px;
|
|
136
|
+
}
|
|
137
|
+
|
|
138
|
+
.milkdown .milkdown-table-block .cell-handle .button-group button svg {
|
|
139
|
+
width: 24px;
|
|
140
|
+
height: 24px;
|
|
141
|
+
}
|
|
142
|
+
|
|
143
|
+
.milkdown .milkdown-table-block .cell-handle .button-group button:hover {
|
|
144
|
+
border-radius: 8px;
|
|
145
|
+
background-color: var(--crepe-color-hover);
|
|
146
|
+
}
|
|
147
|
+
|
|
148
|
+
.milkdown .milkdown-table-block .cell-handle .button-group button:active {
|
|
149
|
+
background: var(--crepe-color-selected);
|
|
150
|
+
}
|
|
151
|
+
|
|
152
|
+
.milkdown .milkdown-table-block .cell-handle:hover {
|
|
153
|
+
opacity: 1;
|
|
154
|
+
}
|
|
155
|
+
|
|
156
|
+
.milkdown .milkdown-table-block .line-handle {
|
|
157
|
+
z-index: 20;
|
|
158
|
+
background-color: var(--crepe-color-primary);
|
|
159
|
+
}
|
|
160
|
+
|
|
161
|
+
.milkdown .milkdown-table-block .line-handle:hover {
|
|
162
|
+
opacity: 1;
|
|
163
|
+
}
|
|
164
|
+
|
|
165
|
+
.milkdown .milkdown-table-block .line-handle .add-button {
|
|
166
|
+
cursor: pointer;
|
|
167
|
+
background-color: var(--crepe-color-surface);
|
|
168
|
+
color: var(--crepe-color-outline);
|
|
169
|
+
border-radius: 100px;
|
|
170
|
+
box-shadow: var(--crepe-shadow-1);
|
|
171
|
+
transition: background-color 0.2s ease-in-out;
|
|
172
|
+
}
|
|
173
|
+
|
|
174
|
+
.milkdown .milkdown-table-block .line-handle .add-button svg {
|
|
175
|
+
width: 16px;
|
|
176
|
+
height: 16px;
|
|
177
|
+
}
|
|
178
|
+
|
|
179
|
+
.milkdown .milkdown-table-block .line-handle .add-button:hover {
|
|
180
|
+
background-color: var(--crepe-color-hover);
|
|
181
|
+
}
|
|
182
|
+
|
|
183
|
+
.milkdown .milkdown-table-block .line-handle .add-button:active {
|
|
184
|
+
background: var(--crepe-color-selected);
|
|
185
|
+
}
|
|
186
|
+
|
|
187
|
+
.milkdown .milkdown-table-block .line-handle[data-role='x-line-drag-handle'] {
|
|
188
|
+
height: 1px;
|
|
189
|
+
z-index: 2;
|
|
190
|
+
}
|
|
191
|
+
|
|
192
|
+
.milkdown .milkdown-table-block .line-handle[data-role='x-line-drag-handle'] .add-button {
|
|
193
|
+
position: absolute;
|
|
194
|
+
transform: translateX(-50%) translateY(-50%);
|
|
195
|
+
padding: 6px 0;
|
|
196
|
+
width: 16px;
|
|
197
|
+
height: 28px;
|
|
198
|
+
}
|
|
199
|
+
|
|
200
|
+
.milkdown .milkdown-table-block .line-handle[data-role='y-line-drag-handle'] {
|
|
201
|
+
width: 1px;
|
|
202
|
+
z-index: 1;
|
|
203
|
+
}
|
|
204
|
+
|
|
205
|
+
.milkdown .milkdown-table-block .line-handle[data-role='y-line-drag-handle'] .add-button {
|
|
206
|
+
position: absolute;
|
|
207
|
+
transform: translateY(-50%) translateX(-50%);
|
|
208
|
+
padding: 0 6px;
|
|
209
|
+
width: 28px;
|
|
210
|
+
height: 16px;
|
|
211
|
+
}
|
|
212
|
+
|
|
213
|
+
.milkdown .milkdown-table-block .line-handle[data-display-type='indicator'] .add-button {
|
|
214
|
+
display: none;
|
|
215
|
+
}
|
|
216
|
+
|
|
217
|
+
.milkdown .milkdown-table-block.readonly .handle {
|
|
218
|
+
display: none;
|
|
219
|
+
}
|
|
@@ -0,0 +1,50 @@
|
|
|
1
|
+
.milkdown:has(.milkdown-link-preview[data-show='true']) .milkdown-toolbar,
|
|
2
|
+
.milkdown:has(.milkdown-link-edit[data-show='true']) .milkdown-toolbar {
|
|
3
|
+
display: none;
|
|
4
|
+
}
|
|
5
|
+
.milkdown .milkdown-toolbar[data-show='false'] {
|
|
6
|
+
display: none;
|
|
7
|
+
}
|
|
8
|
+
.milkdown .milkdown-toolbar {
|
|
9
|
+
z-index: 10;
|
|
10
|
+
position: absolute;
|
|
11
|
+
display: flex;
|
|
12
|
+
background: var(--crepe-color-surface);
|
|
13
|
+
box-shadow: var(--crepe-shadow-1);
|
|
14
|
+
border-radius: 8px;
|
|
15
|
+
overflow: hidden;
|
|
16
|
+
}
|
|
17
|
+
.milkdown .milkdown-toolbar .divider {
|
|
18
|
+
width: 1px;
|
|
19
|
+
background: color-mix(
|
|
20
|
+
in srgb,
|
|
21
|
+
var(--crepe-color-outline),
|
|
22
|
+
transparent 80%
|
|
23
|
+
);
|
|
24
|
+
height: 24px;
|
|
25
|
+
margin: 10px;
|
|
26
|
+
}
|
|
27
|
+
.milkdown .milkdown-toolbar .toolbar-item {
|
|
28
|
+
width: 32px;
|
|
29
|
+
height: 32px;
|
|
30
|
+
margin: 6px;
|
|
31
|
+
padding: 4px;
|
|
32
|
+
cursor: pointer;
|
|
33
|
+
border-radius: 4px;
|
|
34
|
+
}
|
|
35
|
+
.milkdown .milkdown-toolbar .toolbar-item:hover {
|
|
36
|
+
background: var(--crepe-color-hover);
|
|
37
|
+
}
|
|
38
|
+
.milkdown .milkdown-toolbar .toolbar-item:active {
|
|
39
|
+
background: var(--crepe-color-selected);
|
|
40
|
+
}
|
|
41
|
+
.milkdown .milkdown-toolbar .toolbar-item svg {
|
|
42
|
+
height: 24px;
|
|
43
|
+
width: 24px;
|
|
44
|
+
color: var(--crepe-color-outline);
|
|
45
|
+
fill: var(--crepe-color-outline);
|
|
46
|
+
}
|
|
47
|
+
.milkdown .milkdown-toolbar .toolbar-item.active svg {
|
|
48
|
+
color: var(--crepe-color-primary);
|
|
49
|
+
fill: var(--crepe-color-primary);
|
|
50
|
+
}
|
|
@@ -0,0 +1,29 @@
|
|
|
1
|
+
.milkdown {
|
|
2
|
+
--crepe-color-background: #fdfcff;
|
|
3
|
+
--crepe-color-on-background: #1b1c1d;
|
|
4
|
+
--crepe-color-surface: #f8f9ff;
|
|
5
|
+
--crepe-color-surface-low: #f2f3fa;
|
|
6
|
+
--crepe-color-on-surface: #191c20;
|
|
7
|
+
--crepe-color-on-surface-variant: #43474e;
|
|
8
|
+
--crepe-color-outline: #73777f;
|
|
9
|
+
--crepe-color-primary: #37618e;
|
|
10
|
+
--crepe-color-secondary: #d7e3f8;
|
|
11
|
+
--crepe-color-on-secondary: #101c2b;
|
|
12
|
+
--crepe-color-inverse: #2e3135;
|
|
13
|
+
--crepe-color-on-inverse: #eff0f7;
|
|
14
|
+
--crepe-color-inline-code: #ba1a1a;
|
|
15
|
+
--crepe-color-error: #ba1a1a;
|
|
16
|
+
--crepe-color-hover: #eceef4;
|
|
17
|
+
--crepe-color-selected: #e1e2e8;
|
|
18
|
+
--crepe-color-inline-area: #d8dae0;
|
|
19
|
+
|
|
20
|
+
--crepe-font-title: Rubik, Cambria, 'Times New Roman', Times, serif;
|
|
21
|
+
--crepe-font-default: Inter, Arial, Helvetica, sans-serif;
|
|
22
|
+
--crepe-font-code:
|
|
23
|
+
'JetBrains Mono', Menlo, Monaco, 'Courier New', Courier, monospace;
|
|
24
|
+
|
|
25
|
+
--crepe-shadow-1:
|
|
26
|
+
0px 1px 3px 1px rgba(0, 0, 0, 0.15), 0px 1px 2px 0px rgba(0, 0, 0, 0.3);
|
|
27
|
+
--crepe-shadow-2:
|
|
28
|
+
0px 2px 6px 2px rgba(0, 0, 0, 0.15), 0px 1px 2px 0px rgba(0, 0, 0, 0.3);
|
|
29
|
+
}
|
package/esm2022/app/modules/markdown-editor/components/markdown-editor/markdown-editor.component.mjs
CHANGED
|
@@ -9,10 +9,6 @@ import { FsMilkdownLoaderService } from '../../services/milkdown-loader.service'
|
|
|
9
9
|
import * as i0 from "@angular/core";
|
|
10
10
|
import * as i1 from "@firestitch/label";
|
|
11
11
|
export class FsMarkdownEditorComponent {
|
|
12
|
-
_defaultConfig = inject(FS_MARKDOWN_EDITOR_CONFIG, { optional: true });
|
|
13
|
-
_cdRef = inject(ChangeDetectorRef);
|
|
14
|
-
_loader = inject(FsMilkdownLoaderService);
|
|
15
|
-
_zone = inject(NgZone);
|
|
16
12
|
editorRef;
|
|
17
13
|
classFocused = false;
|
|
18
14
|
config = {};
|
|
@@ -27,12 +23,16 @@ export class FsMarkdownEditorComponent {
|
|
|
27
23
|
_markdown = '';
|
|
28
24
|
_editorReady = false;
|
|
29
25
|
_destroy$ = new Subject();
|
|
26
|
+
_defaultConfig = inject(FS_MARKDOWN_EDITOR_CONFIG, { optional: true });
|
|
27
|
+
_cdRef = inject(ChangeDetectorRef);
|
|
28
|
+
_loader = inject(FsMilkdownLoaderService);
|
|
29
|
+
_zone = inject(NgZone);
|
|
30
30
|
get markdown() {
|
|
31
31
|
return this._markdown;
|
|
32
32
|
}
|
|
33
33
|
ngOnInit() {
|
|
34
34
|
this.config = { ...(this._defaultConfig || {}), ...(this.config || {}) };
|
|
35
|
-
this._loader.
|
|
35
|
+
this._loader.load();
|
|
36
36
|
}
|
|
37
37
|
ngAfterViewInit() {
|
|
38
38
|
this._loader.loaded$
|
|
@@ -114,19 +114,20 @@ export class FsMarkdownEditorComponent {
|
|
|
114
114
|
if (this._crepe || !this.editorRef) {
|
|
115
115
|
return;
|
|
116
116
|
}
|
|
117
|
-
|
|
118
|
-
|
|
119
|
-
|
|
120
|
-
|
|
121
|
-
|
|
122
|
-
|
|
123
|
-
|
|
124
|
-
|
|
117
|
+
const milkdown = window.__MILKDOWN__;
|
|
118
|
+
if (!milkdown) {
|
|
119
|
+
console.error('Milkdown not loaded. Ensure assets are configured.');
|
|
120
|
+
return;
|
|
121
|
+
}
|
|
122
|
+
this._editorViewCtx = milkdown.editorViewCtx;
|
|
123
|
+
this._parserCtx = milkdown.parserCtx;
|
|
124
|
+
this._zone.runOutsideAngular(() => {
|
|
125
|
+
this._crepe = new milkdown.Crepe({
|
|
125
126
|
root: this.editorRef.nativeElement,
|
|
126
127
|
defaultValue: this._markdown,
|
|
127
128
|
features: {},
|
|
128
129
|
featureConfigs: {
|
|
129
|
-
[Crepe.Feature.Placeholder]: {
|
|
130
|
+
[milkdown.Crepe.Feature.Placeholder]: {
|
|
130
131
|
text: this.config.placeholder || 'Start typing...',
|
|
131
132
|
},
|
|
132
133
|
},
|
|
@@ -235,4 +236,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.7", ngImpor
|
|
|
235
236
|
type: HostBinding,
|
|
236
237
|
args: ['class.initialized']
|
|
237
238
|
}] } });
|
|
238
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"markdown-editor.component.js","sourceRoot":"","sources":["../../../../../../../src/app/modules/markdown-editor/components/markdown-editor/markdown-editor.component.ts","../../../../../../../src/app/modules/markdown-editor/components/markdown-editor/markdown-editor.component.html"],"names":[],"mappings":"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","sourcesContent":["import {\n  AfterViewInit,\n  ChangeDetectionStrategy,\n  ChangeDetectorRef,\n  Component,\n  ElementRef,\n  forwardRef,\n  HostBinding,\n  inject,\n  Input,\n  NgZone,\n  OnDestroy,\n  OnInit,\n  Optional,\n  ViewChild,\n} from '@angular/core';\nimport {\n  ControlContainer,\n  ControlValueAccessor,\n  NG_VALIDATORS,\n  NG_VALUE_ACCESSOR,\n  NgForm,\n  ValidationErrors,\n  Validator,\n} from '@angular/forms';\n\nimport { guid } from '@firestitch/common';\nimport { FsLabelModule } from '@firestitch/label';\n\nimport { Subject } from 'rxjs';\nimport { takeUntil } from 'rxjs/operators';\n\nimport type { Crepe } from '@milkdown/crepe';\n\nimport { FS_MARKDOWN_EDITOR_CONFIG } from '../../injects/config.inject';\nimport { FsMarkdownEditorConfig } from '../../interfaces/markdown-editor-config';\nimport { FsMilkdownLoaderService } from '../../services/milkdown-loader.service';\n\n\n@Component({\n  selector: 'fs-markdown-editor',\n  templateUrl: './markdown-editor.component.html',\n  styleUrls: ['./markdown-editor.component.scss'],\n  providers: [\n    {\n      provide: NG_VALUE_ACCESSOR,\n      useExisting: forwardRef(() => FsMarkdownEditorComponent),\n      multi: true,\n    },\n    {\n      provide: NG_VALIDATORS,\n      useExisting: forwardRef(() => FsMarkdownEditorComponent),\n      multi: true,\n    },\n  ],\n  changeDetection: ChangeDetectionStrategy.OnPush,\n  viewProviders: [\n    {\n      provide: ControlContainer,\n      deps: [[Optional, NgForm]],\n      useFactory: (ngForm: NgForm) => ngForm,\n    },\n  ],\n  standalone: true,\n  imports: [\n    FsLabelModule,\n  ],\n})\nexport class FsMarkdownEditorComponent\n  implements OnInit, AfterViewInit, ControlValueAccessor, Validator, OnDestroy {\n\n  private _defaultConfig = inject<FsMarkdownEditorConfig>(FS_MARKDOWN_EDITOR_CONFIG, { optional: true });\n  private _cdRef = inject(ChangeDetectorRef);\n  private _loader = inject(FsMilkdownLoaderService);\n  private _zone = inject(NgZone);\n\n  @ViewChild('editorRef') public editorRef: ElementRef;\n\n  @HostBinding('class.focused') public classFocused = false;\n\n  @Input() public config: FsMarkdownEditorConfig = {};\n\n  @Input()\n  @HostBinding('class.disabled')\n  public disabled = false;\n\n  @HostBinding('class.initialized')\n  public initialized = false;\n\n  public readonly containerID = `fs-markdown-editor-${guid('xxx')}`;\n  public onTouched: () => void;\n  public onChange: (data: any) => void;\n\n  private _crepe: Crepe;\n  private _editorViewCtx: any;\n  private _parserCtx: any;\n  private _markdown = '';\n  private _editorReady = false;\n  private _destroy$ = new Subject<void>();\n\n  public get markdown(): string {\n    return this._markdown;\n  }\n\n  public ngOnInit(): void {\n    this.config = { ...(this._defaultConfig || {}), ...(this.config || {}) };\n    this._loader.loadStyles();\n  }\n\n  public ngAfterViewInit(): void {\n    this._loader.loaded$\n      .pipe(takeUntil(this._destroy$))\n      .subscribe((loaded) => {\n        if (loaded) {\n          this._initEditor();\n        }\n      });\n  }\n\n  public writeValue(markdown: string): void {\n    this._markdown = markdown || '';\n\n    if (this._editorReady) {\n      this._setEditorContent(this._markdown);\n    }\n\n    this._cdRef.markForCheck();\n  }\n\n  public registerOnChange(fn: (data: any) => void): void {\n    this.onChange = fn;\n  }\n\n  public registerOnTouched(fn: () => void): void {\n    this.onTouched = fn;\n  }\n\n  public setDisabledState(isDisabled: boolean): void {\n    this.disabled = isDisabled;\n  }\n\n  public validate(): ValidationErrors | null {\n    const err: any = {};\n    if (this.config.maxLength && this._markdown) {\n      const length = this._markdown.length;\n      if (length > this.config.maxLength) {\n        err.maxLengthError = `Must be ${this.config.maxLength} characters or fewer. You entered ${length} characters.`;\n      }\n    }\n\n    return Object.keys(err).length ? err : null;\n  }\n\n  public clear(): void {\n    this.writeValue('');\n    if (this.onChange) {\n      this.onChange('');\n    }\n  }\n\n  public setMarkdown(markdown: string): void {\n    this.writeValue(markdown);\n  }\n\n  public getMarkdown(): string {\n    if (!this._crepe) {\n      return this._markdown;\n    }\n\n    return this._crepe.getMarkdown();\n  }\n\n  public focus(): void {\n    if (this._crepe && this._editorViewCtx) {\n      this._crepe.editor.action((ctx) => {\n        const view = ctx.get(this._editorViewCtx) as any;\n        view.focus();\n      });\n    }\n  }\n\n  public disable(): void {\n    this.disabled = true;\n    this._cdRef.markForCheck();\n  }\n\n  public destroy(): void {\n    if (this._crepe) {\n      this._crepe.destroy();\n      this._crepe = null;\n    }\n    this._editorReady = false;\n    this.initialized = false;\n    this._cdRef.markForCheck();\n  }\n\n  public ngOnDestroy(): void {\n    this._destroy$.next();\n    this._destroy$.complete();\n    this.destroy();\n  }\n\n  private _initEditor(): void {\n    if (this._crepe || !this.editorRef) {\n      return;\n    }\n\n    this._zone.runOutsideAngular(async () => {\n      const [{ Crepe }, { editorViewCtx, parserCtx }] = await Promise.all([\n        import('@milkdown/crepe'),\n        import('@milkdown/kit/core'),\n      ]);\n\n      this._editorViewCtx = editorViewCtx;\n      this._parserCtx = parserCtx;\n\n      this._crepe = new Crepe({\n        root: this.editorRef.nativeElement,\n        defaultValue: this._markdown,\n        features: {},\n        featureConfigs: {\n          [Crepe.Feature.Placeholder]: {\n            text: this.config.placeholder || 'Start typing...',\n          },\n        },\n      });\n\n      this._crepe.on((listener) => {\n        listener.markdownUpdated((_ctx, markdown, prevMarkdown) => {\n          if (markdown !== prevMarkdown) {\n            this._zone.run(() => {\n              this._markdown = markdown;\n              if (this.onChange) {\n                this.onChange(markdown);\n              }\n              if (this.onTouched) {\n                this.onTouched();\n              }\n              this._cdRef.markForCheck();\n            });\n          }\n        });\n      });\n\n      this._crepe.create().then(() => {\n        this._zone.run(() => {\n          this._editorReady = true;\n          this.initialized = true;\n\n          if (this._markdown) {\n            this._setEditorContent(this._markdown);\n          }\n\n          this._cdRef.markForCheck();\n\n          if (this.config.autofocus) {\n            this.focus();\n          }\n\n          if (this.config.initialized) {\n            this.config.initialized();\n          }\n        });\n      });\n    });\n  }\n\n  private _setEditorContent(markdown: string): void {\n    this._crepe.editor.action((ctx) => {\n      const view = ctx.get(this._editorViewCtx) as any;\n      const parser = ctx.get(this._parserCtx) as any;\n      const doc = parser(markdown);\n      if (doc) {\n        const tr = view.state.tr.replaceWith(\n          0, view.state.doc.content.size, doc.content,\n        );\n        view.dispatch(tr);\n      }\n    });\n  }\n}\n","<fs-label-field\n  [appearance]=\"'outline'\"\n  [hoverable]=\"true\"\n  [showOutline]=\"initialized\"\n  [disabled]=\"disabled\"\n  [padless]=\"config.padless\"\n  [class.initialized]=\"initialized\">\n  @if (config.label) {\n    <fs-label>\n      {{ config.label }}\n    </fs-label>\n  }\n  <div\n    class=\"markdown-editor-container\"\n    [id]=\"containerID\">\n    <div #editorRef></div>\n  </div>\n  <fs-label-message>\n    <div class=\"fs-form-message\">\n      <div class=\"fs-form-hint\">\n        {{ config.hint }}\n      </div>\n    </div>\n  </fs-label-message>\n</fs-label-field>\n"]}
|
|
239
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"markdown-editor.component.js","sourceRoot":"","sources":["../../../../../../../src/app/modules/markdown-editor/components/markdown-editor/markdown-editor.component.ts","../../../../../../../src/app/modules/markdown-editor/components/markdown-editor/markdown-editor.component.html"],"names":[],"mappings":"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","sourcesContent":["import {\n  AfterViewInit,\n  ChangeDetectionStrategy,\n  ChangeDetectorRef,\n  Component,\n  ElementRef,\n  forwardRef,\n  HostBinding,\n  inject,\n  Input,\n  NgZone,\n  OnDestroy,\n  OnInit,\n  Optional,\n  ViewChild,\n} from '@angular/core';\nimport {\n  ControlContainer,\n  ControlValueAccessor,\n  NG_VALIDATORS,\n  NG_VALUE_ACCESSOR,\n  NgForm,\n  ValidationErrors,\n  Validator,\n} from '@angular/forms';\n\nimport { guid } from '@firestitch/common';\nimport { FsLabelModule } from '@firestitch/label';\n\nimport { Subject } from 'rxjs';\nimport { takeUntil } from 'rxjs/operators';\n\nimport { FS_MARKDOWN_EDITOR_CONFIG } from '../../injects/config.inject';\nimport { FsMarkdownEditorConfig } from '../../interfaces/markdown-editor-config';\nimport { FsMilkdownLoaderService } from '../../services/milkdown-loader.service';\n\n\n@Component({\n  selector: 'fs-markdown-editor',\n  templateUrl: './markdown-editor.component.html',\n  styleUrls: ['./markdown-editor.component.scss'],\n  providers: [\n    {\n      provide: NG_VALUE_ACCESSOR,\n      useExisting: forwardRef(() => FsMarkdownEditorComponent),\n      multi: true,\n    },\n    {\n      provide: NG_VALIDATORS,\n      useExisting: forwardRef(() => FsMarkdownEditorComponent),\n      multi: true,\n    },\n  ],\n  changeDetection: ChangeDetectionStrategy.OnPush,\n  viewProviders: [\n    {\n      provide: ControlContainer,\n      deps: [[Optional, NgForm]],\n      useFactory: (ngForm: NgForm) => ngForm,\n    },\n  ],\n  standalone: true,\n  imports: [\n    FsLabelModule,\n  ],\n})\nexport class FsMarkdownEditorComponent\nimplements OnInit, AfterViewInit, ControlValueAccessor, Validator, OnDestroy {\n\n  @ViewChild('editorRef') public editorRef: ElementRef;\n\n  @HostBinding('class.focused') public classFocused = false;\n\n  @Input() public config: FsMarkdownEditorConfig = {};\n\n  @Input()\n  @HostBinding('class.disabled')\n  public disabled = false;\n\n  @HostBinding('class.initialized')\n  public initialized = false;\n\n  public readonly containerID = `fs-markdown-editor-${guid('xxx')}`;\n  public onTouched: () => void;\n  public onChange: (data: any) => void;\n\n  private _crepe: any;\n  private _editorViewCtx: any;\n  private _parserCtx: any;\n  private _markdown = '';\n  private _editorReady = false;\n  private _destroy$ = new Subject<void>();\n  private _defaultConfig = inject<FsMarkdownEditorConfig>(FS_MARKDOWN_EDITOR_CONFIG, { optional: true });\n  private _cdRef = inject(ChangeDetectorRef);\n  private _loader = inject(FsMilkdownLoaderService);\n  private _zone = inject(NgZone);\n\n\n  public get markdown(): string {\n    return this._markdown;\n  }\n\n  public ngOnInit(): void {\n    this.config = { ...(this._defaultConfig || {}), ...(this.config || {}) };\n    this._loader.load();\n  }\n\n  public ngAfterViewInit(): void {\n    this._loader.loaded$\n      .pipe(takeUntil(this._destroy$))\n      .subscribe((loaded) => {\n        if (loaded) {\n          this._initEditor();\n        }\n      });\n  }\n\n  public writeValue(markdown: string): void {\n    this._markdown = markdown || '';\n\n    if (this._editorReady) {\n      this._setEditorContent(this._markdown);\n    }\n\n    this._cdRef.markForCheck();\n  }\n\n  public registerOnChange(fn: (data: any) => void): void {\n    this.onChange = fn;\n  }\n\n  public registerOnTouched(fn: () => void): void {\n    this.onTouched = fn;\n  }\n\n  public setDisabledState(isDisabled: boolean): void {\n    this.disabled = isDisabled;\n  }\n\n  public validate(): ValidationErrors | null {\n    const err: any = {};\n    if (this.config.maxLength && this._markdown) {\n      const length = this._markdown.length;\n      if (length > this.config.maxLength) {\n        err.maxLengthError = `Must be ${this.config.maxLength} characters or fewer. You entered ${length} characters.`;\n      }\n    }\n\n    return Object.keys(err).length ? err : null;\n  }\n\n  public clear(): void {\n    this.writeValue('');\n    if (this.onChange) {\n      this.onChange('');\n    }\n  }\n\n  public setMarkdown(markdown: string): void {\n    this.writeValue(markdown);\n  }\n\n  public getMarkdown(): string {\n    if (!this._crepe) {\n      return this._markdown;\n    }\n\n    return this._crepe.getMarkdown();\n  }\n\n  public focus(): void {\n    if (this._crepe && this._editorViewCtx) {\n      this._crepe.editor.action((ctx: any) => {\n        const view = ctx.get(this._editorViewCtx);\n        view.focus();\n      });\n    }\n  }\n\n  public disable(): void {\n    this.disabled = true;\n    this._cdRef.markForCheck();\n  }\n\n  public destroy(): void {\n    if (this._crepe) {\n      this._crepe.destroy();\n      this._crepe = null;\n    }\n    this._editorReady = false;\n    this.initialized = false;\n    this._cdRef.markForCheck();\n  }\n\n  public ngOnDestroy(): void {\n    this._destroy$.next();\n    this._destroy$.complete();\n    this.destroy();\n  }\n\n  private _initEditor(): void {\n    if (this._crepe || !this.editorRef) {\n      return;\n    }\n\n    const milkdown = (window as any).__MILKDOWN__;\n    if (!milkdown) {\n      console.error('Milkdown not loaded. Ensure assets are configured.');\n\n      return;\n    }\n\n    this._editorViewCtx = milkdown.editorViewCtx;\n    this._parserCtx = milkdown.parserCtx;\n\n    this._zone.runOutsideAngular(() => {\n      this._crepe = new milkdown.Crepe({\n        root: this.editorRef.nativeElement,\n        defaultValue: this._markdown,\n        features: {},\n        featureConfigs: {\n          [milkdown.Crepe.Feature.Placeholder]: {\n            text: this.config.placeholder || 'Start typing...',\n          },\n        },\n      });\n\n      this._crepe.on((listener: any) => {\n        listener.markdownUpdated((_ctx: any, markdown: string, prevMarkdown: string) => {\n          if (markdown !== prevMarkdown) {\n            this._zone.run(() => {\n              this._markdown = markdown;\n              if (this.onChange) {\n                this.onChange(markdown);\n              }\n              if (this.onTouched) {\n                this.onTouched();\n              }\n              this._cdRef.markForCheck();\n            });\n          }\n        });\n      });\n\n      this._crepe.create().then(() => {\n        this._zone.run(() => {\n          this._editorReady = true;\n          this.initialized = true;\n\n          if (this._markdown) {\n            this._setEditorContent(this._markdown);\n          }\n\n          this._cdRef.markForCheck();\n\n          if (this.config.autofocus) {\n            this.focus();\n          }\n\n          if (this.config.initialized) {\n            this.config.initialized();\n          }\n        });\n      });\n    });\n  }\n\n  private _setEditorContent(markdown: string): void {\n    this._crepe.editor.action((ctx: any) => {\n      const view = ctx.get(this._editorViewCtx);\n      const parser = ctx.get(this._parserCtx);\n      const doc = parser(markdown);\n      if (doc) {\n        const tr = view.state.tr.replaceWith(\n          0, view.state.doc.content.size, doc.content,\n        );\n        view.dispatch(tr);\n      }\n    });\n  }\n}\n","<fs-label-field\n  [appearance]=\"'outline'\"\n  [hoverable]=\"true\"\n  [showOutline]=\"initialized\"\n  [disabled]=\"disabled\"\n  [padless]=\"config.padless\"\n  [class.initialized]=\"initialized\">\n  @if (config.label) {\n    <fs-label>\n      {{ config.label }}\n    </fs-label>\n  }\n  <div\n    class=\"markdown-editor-container\"\n    [id]=\"containerID\">\n    <div #editorRef></div>\n  </div>\n  <fs-label-message>\n    <div class=\"fs-form-message\">\n      <div class=\"fs-form-hint\">\n        {{ config.hint }}\n      </div>\n    </div>\n  </fs-label-message>\n</fs-label-field>\n"]}
|
|
@@ -3,68 +3,77 @@ import { Injectable, inject } from '@angular/core';
|
|
|
3
3
|
import { BehaviorSubject } from 'rxjs';
|
|
4
4
|
import { shareReplay } from 'rxjs/operators';
|
|
5
5
|
import * as i0 from "@angular/core";
|
|
6
|
-
const
|
|
6
|
+
const ASSET_BASE = '/assets/markdown-editor/';
|
|
7
7
|
const STYLESHEETS = [
|
|
8
|
-
`${
|
|
9
|
-
`${
|
|
10
|
-
`${
|
|
11
|
-
`${
|
|
12
|
-
`${
|
|
13
|
-
`${
|
|
14
|
-
`${
|
|
15
|
-
`${
|
|
16
|
-
`${
|
|
17
|
-
`${
|
|
8
|
+
`${ASSET_BASE}theme/common/block-edit.css`,
|
|
9
|
+
`${ASSET_BASE}theme/common/code-mirror.css`,
|
|
10
|
+
`${ASSET_BASE}theme/common/cursor.css`,
|
|
11
|
+
`${ASSET_BASE}theme/common/image-block.css`,
|
|
12
|
+
`${ASSET_BASE}theme/common/latex.css`,
|
|
13
|
+
`${ASSET_BASE}theme/common/link-tooltip.css`,
|
|
14
|
+
`${ASSET_BASE}theme/common/list-item.css`,
|
|
15
|
+
`${ASSET_BASE}theme/common/placeholder.css`,
|
|
16
|
+
`${ASSET_BASE}theme/common/toolbar.css`,
|
|
17
|
+
`${ASSET_BASE}theme/common/table.css`,
|
|
18
18
|
];
|
|
19
|
+
const SCRIPT = `${ASSET_BASE}milkdown.js`;
|
|
19
20
|
export class FsMilkdownLoaderService {
|
|
20
21
|
_document = inject(DOCUMENT);
|
|
21
22
|
_loaded = new BehaviorSubject(false);
|
|
22
|
-
|
|
23
|
+
_loading = false;
|
|
23
24
|
_loaded$ = this._loaded.asObservable()
|
|
24
25
|
.pipe(shareReplay(1));
|
|
25
26
|
get loaded$() {
|
|
26
27
|
return this._loaded$;
|
|
27
28
|
}
|
|
28
|
-
|
|
29
|
-
if (this.
|
|
30
|
-
this._loaded.
|
|
31
|
-
|
|
32
|
-
|
|
33
|
-
let remaining = STYLESHEETS.length;
|
|
34
|
-
if (remaining === 0) {
|
|
35
|
-
this._stylesLoaded = true;
|
|
36
|
-
this._loaded.next(true);
|
|
29
|
+
load() {
|
|
30
|
+
if (this._loaded.value || this._loading) {
|
|
31
|
+
if (this._loaded.value) {
|
|
32
|
+
this._loaded.next(true);
|
|
33
|
+
}
|
|
37
34
|
return;
|
|
38
35
|
}
|
|
36
|
+
this._loading = true;
|
|
37
|
+
let stylesRemaining = STYLESHEETS.length;
|
|
38
|
+
let scriptDone = false;
|
|
39
|
+
const checkComplete = () => {
|
|
40
|
+
if (stylesRemaining === 0 && scriptDone) {
|
|
41
|
+
this._loading = false;
|
|
42
|
+
this._loaded.next(true);
|
|
43
|
+
}
|
|
44
|
+
};
|
|
45
|
+
const onStyleDone = () => {
|
|
46
|
+
stylesRemaining--;
|
|
47
|
+
checkComplete();
|
|
48
|
+
};
|
|
39
49
|
STYLESHEETS.forEach((href) => {
|
|
40
|
-
|
|
41
|
-
|
|
42
|
-
remaining--;
|
|
43
|
-
if (remaining === 0) {
|
|
44
|
-
this._stylesLoaded = true;
|
|
45
|
-
this._loaded.next(true);
|
|
46
|
-
}
|
|
50
|
+
if (this._document.querySelector(`link[href="${href}"]`)) {
|
|
51
|
+
onStyleDone();
|
|
47
52
|
return;
|
|
48
53
|
}
|
|
49
54
|
const link = this._document.createElement('link');
|
|
50
55
|
link.rel = 'stylesheet';
|
|
51
56
|
link.href = href;
|
|
52
|
-
link.onload =
|
|
53
|
-
|
|
54
|
-
if (remaining === 0) {
|
|
55
|
-
this._stylesLoaded = true;
|
|
56
|
-
this._loaded.next(true);
|
|
57
|
-
}
|
|
58
|
-
};
|
|
59
|
-
link.onerror = () => {
|
|
60
|
-
remaining--;
|
|
61
|
-
if (remaining === 0) {
|
|
62
|
-
this._stylesLoaded = true;
|
|
63
|
-
this._loaded.next(true);
|
|
64
|
-
}
|
|
65
|
-
};
|
|
57
|
+
link.onload = onStyleDone;
|
|
58
|
+
link.onerror = onStyleDone;
|
|
66
59
|
this._document.head.appendChild(link);
|
|
67
60
|
});
|
|
61
|
+
if (this._document.querySelector(`script[src="${SCRIPT}"]`)) {
|
|
62
|
+
scriptDone = true;
|
|
63
|
+
checkComplete();
|
|
64
|
+
return;
|
|
65
|
+
}
|
|
66
|
+
const script = this._document.createElement('script');
|
|
67
|
+
script.src = SCRIPT;
|
|
68
|
+
script.onload = () => {
|
|
69
|
+
scriptDone = true;
|
|
70
|
+
checkComplete();
|
|
71
|
+
};
|
|
72
|
+
script.onerror = () => {
|
|
73
|
+
scriptDone = true;
|
|
74
|
+
checkComplete();
|
|
75
|
+
};
|
|
76
|
+
this._document.head.appendChild(script);
|
|
68
77
|
}
|
|
69
78
|
static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.7", ngImport: i0, type: FsMilkdownLoaderService, deps: [], target: i0.ɵɵFactoryTarget.Injectable });
|
|
70
79
|
static ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "18.2.7", ngImport: i0, type: FsMilkdownLoaderService, providedIn: 'root' });
|
|
@@ -75,4 +84,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.7", ngImpor
|
|
|
75
84
|
providedIn: 'root',
|
|
76
85
|
}]
|
|
77
86
|
}] });
|
|
78
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
87
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoibWlsa2Rvd24tbG9hZGVyLnNlcnZpY2UuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi8uLi9zcmMvYXBwL21vZHVsZXMvbWFya2Rvd24tZWRpdG9yL3NlcnZpY2VzL21pbGtkb3duLWxvYWRlci5zZXJ2aWNlLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLE9BQU8sRUFBRSxRQUFRLEVBQUUsTUFBTSxpQkFBaUIsQ0FBQztBQUMzQyxPQUFPLEVBQUUsVUFBVSxFQUFFLE1BQU0sRUFBRSxNQUFNLGVBQWUsQ0FBQztBQUVuRCxPQUFPLEVBQUUsZUFBZSxFQUFjLE1BQU0sTUFBTSxDQUFDO0FBQ25ELE9BQU8sRUFBRSxXQUFXLEVBQUUsTUFBTSxnQkFBZ0IsQ0FBQzs7QUFFN0MsTUFBTSxVQUFVLEdBQUcsMEJBQTBCLENBQUM7QUFFOUMsTUFBTSxXQUFXLEdBQUc7SUFDbEIsR0FBRyxVQUFVLDZCQUE2QjtJQUMxQyxHQUFHLFVBQVUsOEJBQThCO0lBQzNDLEdBQUcsVUFBVSx5QkFBeUI7SUFDdEMsR0FBRyxVQUFVLDhCQUE4QjtJQUMzQyxHQUFHLFVBQVUsd0JBQXdCO0lBQ3JDLEdBQUcsVUFBVSwrQkFBK0I7SUFDNUMsR0FBRyxVQUFVLDRCQUE0QjtJQUN6QyxHQUFHLFVBQVUsOEJBQThCO0lBQzNDLEdBQUcsVUFBVSwwQkFBMEI7SUFDdkMsR0FBRyxVQUFVLHdCQUF3QjtDQUN0QyxDQUFDO0FBRUYsTUFBTSxNQUFNLEdBQUcsR0FBRyxVQUFVLGFBQWEsQ0FBQztBQUsxQyxNQUFNLE9BQU8sdUJBQXVCO0lBQzFCLFNBQVMsR0FBRyxNQUFNLENBQVcsUUFBUSxDQUFDLENBQUM7SUFDdkMsT0FBTyxHQUFHLElBQUksZUFBZSxDQUFDLEtBQUssQ0FBQyxDQUFDO0lBQ3JDLFFBQVEsR0FBRyxLQUFLLENBQUM7SUFFakIsUUFBUSxHQUFHLElBQUksQ0FBQyxPQUFPLENBQUMsWUFBWSxFQUFFO1NBQzNDLElBQUksQ0FDSCxX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cG9ydCB7IEluamVjdGFibGUsIGluamVjdCB9IGZyb20gJ0Bhbmd1bGFyL2NvcmUnO1xuXG5pbXBvcnQgeyBCZWhhdmlvclN1YmplY3QsIE9ic2VydmFibGUgfSBmcm9tICdyeGpzJztcbmltcG9ydCB7IHNoYXJlUmVwbGF5IH0gZnJvbSAncnhqcy9vcGVyYXRvcnMnO1xuXG5jb25zdCBBU1NFVF9CQVNFID0gJy9hc3NldHMvbWFya2Rvd24tZWRpdG9yLyc7XG5cbmNvbnN0IFNUWUxFU0hFRVRTID0gW1xuICBgJHtBU1NFVF9CQVNFfXRoZW1lL2NvbW1vbi9ibG9jay1lZGl0LmNzc2AsXG4gIGAke0FTU0VUX0JBU0V9dGhlbWUvY29tbW9uL2NvZGUtbWlycm9yLmNzc2AsXG4gIGAke0FTU0VUX0JBU0V9dGhlbWUvY29tbW9uL2N1cnNvci5jc3NgLFxuICBgJHtBU1NFVF9CQVNFfXRoZW1lL2NvbW1vbi9pbWFnZS1ibG9jay5jc3NgLFxuICBgJHtBU1NFVF9CQVNFfXRoZW1lL2NvbW1vbi9sYXRleC5jc3NgLFxuICBgJHtBU1NFVF9CQVNFfXRoZW1lL2NvbW1vbi9saW5rLXRvb2x0aXAuY3NzYCxcbiAgYCR7QVNTRVRfQkFTRX10aGVtZS9jb21tb24vbGlzdC1pdGVtLmNzc2AsXG4gIGAke0FTU0VUX0JBU0V9dGhlbWUvY29tbW9uL3BsYWNlaG9sZGVyLmNzc2AsXG4gIGAke0FTU0VUX0JBU0V9dGhlbWUvY29tbW9uL3Rvb2xiYXIuY3NzYCxcbiAgYCR7QVNTRVRfQkFTRX10aGVtZS9jb21tb24vdGFibGUuY3NzYCxcbl07XG5cbmNvbnN0IFNDUklQVCA9IGAke0FTU0VUX0JBU0V9bWlsa2Rvd24uanNgO1xuXG5ASW5qZWN0YWJsZSh7XG4gIHByb3ZpZGVkSW46ICdyb290Jyxcbn0pXG5leHBvcnQgY2xhc3MgRnNNaWxrZG93bkxvYWRlclNlcnZpY2Uge1xuICBwcml2YXRlIF9kb2N1bWVudCA9IGluamVjdDxEb2N1bWVudD4oRE9DVU1FTlQpO1xuICBwcml2YXRlIF9sb2FkZWQgPSBuZXcgQmVoYXZpb3JTdWJqZWN0KGZhbHNlKTtcbiAgcHJpdmF0ZSBfbG9hZGluZyA9IGZhbHNlO1xuXG4gIHByaXZhdGUgX2xvYWRlZCQgPSB0aGlzLl9sb2FkZWQuYXNPYnNlcnZhYmxlKClcbiAgICAucGlwZShcbiAgICAgIHNoYXJlUmVwbGF5KDEpLFxuICAgICk7XG5cbiAgcHVibGljIGdldCBsb2FkZWQkKCk6IE9ic2VydmFibGU8Ym9vbGVhbj4ge1xuICAgIHJldHVybiB0aGlzLl9sb2FkZWQkO1xuICB9XG5cbiAgcHVibGljIGxvYWQoKTogdm9pZCB7XG4gICAgaWYgKHRoaXMuX2xvYWRlZC52YWx1ZSB8fCB0aGlzLl9sb2FkaW5nKSB7XG4gICAgICBpZiAodGhpcy5fbG9hZGVkLnZhbHVlKSB7XG4gICAgICAgIHRoaXMuX2xvYWRlZC5uZXh0KHRydWUpO1xuICAgICAgfVxuXG4gICAgICByZXR1cm47XG4gICAgfVxuXG4gICAgdGhpcy5fbG9hZGluZyA9IHRydWU7XG5cbiAgICBsZXQgc3R5bGVzUmVtYWluaW5nID0gU1RZTEVTSEVFVFMubGVuZ3RoO1xuICAgIGxldCBzY3JpcHREb25lID0gZmFsc2U7XG5cbiAgICBjb25zdCBjaGVja0NvbXBsZXRlID0gKCkgPT4ge1xuICAgICAgaWYgKHN0eWxlc1JlbWFpbmluZyA9PT0gMCAmJiBzY3JpcHREb25lKSB7XG4gICAgICAgIHRoaXMuX2xvYWRpbmcgPSBmYWxzZTtcbiAgICAgICAgdGhpcy5fbG9hZGVkLm5leHQodHJ1ZSk7XG4gICAgICB9XG4gICAgfTtcblxuICAgIGNvbnN0IG9uU3R5bGVEb25lID0gKCkgPT4ge1xuICAgICAgc3R5bGVzUmVtYWluaW5nLS07XG4gICAgICBjaGVja0NvbXBsZXRlKCk7XG4gICAgfTtcblxuICAgIFNUWUxFU0hFRVRTLmZvckVhY2goKGhyZWYpID0+IHtcbiAgICAgIGlmICh0aGlzLl9kb2N1bWVudC5xdWVyeVNlbGVjdG9yKGBsaW5rW2hyZWY9XCIke2hyZWZ9XCJdYCkpIHtcbiAgICAgICAgb25TdHlsZURvbmUoKTtcblxuICAgICAgICByZXR1cm47XG4gICAgICB9XG5cbiAgICAgIGNvbnN0IGxpbmsgPSB0aGlzLl9kb2N1bWVudC5jcmVhdGVFbGVtZW50KCdsaW5rJyk7XG4gICAgICBsaW5rLnJlbCA9ICdzdHlsZXNoZWV0JztcbiAgICAgIGxpbmsuaHJlZiA9IGhyZWY7XG4gICAgICBsaW5rLm9ubG9hZCA9IG9uU3R5bGVEb25lO1xuICAgICAgbGluay5vbmVycm9yID0gb25TdHlsZURvbmU7XG4gICAgICB0aGlzLl9kb2N1bWVudC5oZWFkLmFwcGVuZENoaWxkKGxpbmspO1xuICAgIH0pO1xuXG4gICAgaWYgKHRoaXMuX2RvY3VtZW50LnF1ZXJ5U2VsZWN0b3IoYHNjcmlwdFtzcmM9XCIke1NDUklQVH1cIl1gKSkge1xuICAgICAgc2NyaXB0RG9uZSA9IHRydWU7XG4gICAgICBjaGVja0NvbXBsZXRlKCk7XG5cbiAgICAgIHJldHVybjtcbiAgICB9XG5cbiAgICBjb25zdCBzY3JpcHQgPSB0aGlzLl9kb2N1bWVudC5jcmVhdGVFbGVtZW50KCdzY3JpcHQnKTtcbiAgICBzY3JpcHQuc3JjID0gU0NSSVBUO1xuICAgIHNjcmlwdC5vbmxvYWQgPSAoKSA9PiB7XG4gICAgICBzY3JpcHREb25lID0gdHJ1ZTtcbiAgICAgIGNoZWNrQ29tcGxldGUoKTtcbiAgICB9O1xuICAgIHNjcmlwdC5vbmVycm9yID0gKCkgPT4ge1xuICAgICAgc2NyaXB0RG9uZSA9IHRydWU7XG4gICAgICBjaGVja0NvbXBsZXRlKCk7XG4gICAgfTtcbiAgICB0aGlzLl9kb2N1bWVudC5oZWFkLmFwcGVuZENoaWxkKHNjcmlwdCk7XG4gIH1cbn1cbiJdfQ==
|