@theseam/ui-common 0.4.24 → 0.4.26
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/esm2020/data-filters/filters/data-filter-search/data-filter-search.component.mjs +1 -1
- package/esm2020/data-filters/filters/data-filter-text/data-filter-text.component.mjs +1 -1
- package/esm2020/datatable/datatable/datatable.component.mjs +3 -3
- package/esm2020/datatable/datatable-action-menu/datatable-action-menu.component.mjs +3 -3
- package/esm2020/datatable/datatable-column-preferences/datatable-column-preferences.component.mjs +1 -1
- package/esm2020/form-field/input.directive.mjs +8 -4
- package/esm2020/framework/schema-form-controls/schema-form-input/schema-form-input.component.mjs +1 -1
- package/esm2020/framework/schema-form-controls/schema-form-number/schema-form-number.component.mjs +1 -1
- package/esm2020/framework/schema-form-controls/schema-form-select/schema-form-select.component.mjs +2 -2
- package/esm2020/framework/schema-form-controls/schema-form-tel/schema-form-tel.component.mjs +2 -2
- package/esm2020/google-maps/google-maps-places-autocomplete/google-maps-places-autocomplete.component.mjs +3 -3
- package/esm2020/rich-text/public-api.mjs +5 -0
- package/esm2020/rich-text/rich-text/rich-text.component.mjs +573 -0
- package/esm2020/rich-text/rich-text.module.mjs +32 -0
- package/esm2020/rich-text/theseam-ui-common-rich-text.mjs +5 -0
- package/esm2020/rich-text/utils/models.mjs +24 -0
- package/esm2020/rich-text/utils/utils.mjs +123 -0
- package/esm2020/tel-input/tel-input/tel-input.component.mjs +2 -2
- package/esm2020/utils/geo-json/coerce-feature-collection.mjs +1 -1
- package/esm2020/utils/geo-json/is-feature-collection.validator.mjs +1 -1
- package/esm2020/utils/geo-json/is-only-geometry-types.mjs +1 -1
- package/esm2020/utils/geo-json/merge-polygons.mjs +1 -1
- package/esm2020/utils/geo-json/no-inner-rings.validator.mjs +1 -1
- package/esm2020/utils/geo-json/split-multi-polygons.mjs +1 -1
- package/esm2020/utils/is-null-or-undefined-or-empty.mjs +4 -0
- package/esm2020/utils/not-null-or-undefined-or-empty.mjs +4 -0
- package/esm2020/utils/public-api.mjs +3 -1
- package/fesm2015/theseam-ui-common-data-filters.mjs +2 -2
- package/fesm2015/theseam-ui-common-data-filters.mjs.map +1 -1
- package/fesm2015/theseam-ui-common-datatable.mjs +5 -5
- package/fesm2015/theseam-ui-common-datatable.mjs.map +1 -1
- package/fesm2015/theseam-ui-common-form-field.mjs +7 -3
- package/fesm2015/theseam-ui-common-form-field.mjs.map +1 -1
- package/fesm2015/theseam-ui-common-framework.mjs +4 -4
- package/fesm2015/theseam-ui-common-framework.mjs.map +1 -1
- package/fesm2015/theseam-ui-common-google-maps.mjs +2 -2
- package/fesm2015/theseam-ui-common-google-maps.mjs.map +1 -1
- package/fesm2015/theseam-ui-common-rich-text.mjs +744 -0
- package/fesm2015/theseam-ui-common-rich-text.mjs.map +1 -0
- package/fesm2015/theseam-ui-common-tel-input.mjs +1 -1
- package/fesm2015/theseam-ui-common-tel-input.mjs.map +1 -1
- package/fesm2015/theseam-ui-common-utils.mjs +9 -1
- package/fesm2015/theseam-ui-common-utils.mjs.map +1 -1
- package/fesm2020/theseam-ui-common-data-filters.mjs +2 -2
- package/fesm2020/theseam-ui-common-data-filters.mjs.map +1 -1
- package/fesm2020/theseam-ui-common-datatable.mjs +5 -5
- package/fesm2020/theseam-ui-common-datatable.mjs.map +1 -1
- package/fesm2020/theseam-ui-common-form-field.mjs +7 -3
- package/fesm2020/theseam-ui-common-form-field.mjs.map +1 -1
- package/fesm2020/theseam-ui-common-framework.mjs +4 -4
- package/fesm2020/theseam-ui-common-framework.mjs.map +1 -1
- package/fesm2020/theseam-ui-common-google-maps.mjs +2 -2
- package/fesm2020/theseam-ui-common-google-maps.mjs.map +1 -1
- package/fesm2020/theseam-ui-common-rich-text.mjs +752 -0
- package/fesm2020/theseam-ui-common-rich-text.mjs.map +1 -0
- package/fesm2020/theseam-ui-common-tel-input.mjs +1 -1
- package/fesm2020/theseam-ui-common-tel-input.mjs.map +1 -1
- package/fesm2020/theseam-ui-common-utils.mjs +9 -1
- package/fesm2020/theseam-ui-common-utils.mjs.map +1 -1
- package/form-field/input.directive.d.ts +2 -1
- package/package.json +9 -1
- package/rich-text/index.d.ts +5 -0
- package/rich-text/public-api.d.ts +4 -0
- package/rich-text/rich-text/rich-text.component.d.ts +192 -0
- package/rich-text/rich-text.module.d.ts +10 -0
- package/rich-text/utils/models.d.ts +343 -0
- package/rich-text/utils/utils.d.ts +14 -0
- package/styles/vendor/quill/_quill.scss +162 -10
- package/utils/is-null-or-undefined-or-empty.d.ts +1 -0
- package/utils/not-null-or-undefined-or-empty.d.ts +1 -0
- package/utils/public-api.d.ts +2 -0
|
@@ -0,0 +1,573 @@
|
|
|
1
|
+
import { coerceBooleanProperty } from '@angular/cdk/coercion';
|
|
2
|
+
import { Component, EventEmitter, Inject, Input, Optional, Output, ViewChild, forwardRef } from '@angular/core';
|
|
3
|
+
import { FormControl, NG_VALUE_ACCESSOR } from '@angular/forms';
|
|
4
|
+
import 'quill-mention';
|
|
5
|
+
import { BehaviorSubject, Subject, combineLatest, filter, lastValueFrom, map, of, shareReplay, startWith, switchMap, take, tap } from 'rxjs';
|
|
6
|
+
import { isNullOrUndefined, notNullOrUndefined } from '@theseam/ui-common/utils';
|
|
7
|
+
import { THESEAM_QUILL_EDITOR_CONFIG, THESEAM_QUILL_EDITOR_CONFIG_DEFAULT, THESEAM_QUILL_MENTION_OPTIONS_DEFAULT, defaultHtmlCharacterCounterFn, defaultMentionRenderListFn, defaultMentionSearchFn, isMentionMenuOption } from '../utils/utils';
|
|
8
|
+
import * as i0 from "@angular/core";
|
|
9
|
+
import * as i1 from "@angular/common";
|
|
10
|
+
import * as i2 from "@angular/forms";
|
|
11
|
+
import * as i3 from "ngx-quill";
|
|
12
|
+
export const RICH_TEXT_VALUE_ACCESSOR = {
|
|
13
|
+
provide: NG_VALUE_ACCESSOR,
|
|
14
|
+
useExisting: forwardRef(() => RichTextComponent),
|
|
15
|
+
multi: true
|
|
16
|
+
};
|
|
17
|
+
export class RichTextComponent {
|
|
18
|
+
get required() {
|
|
19
|
+
return this._required;
|
|
20
|
+
}
|
|
21
|
+
set required(value) {
|
|
22
|
+
if (notNullOrUndefined(value)) {
|
|
23
|
+
this._required = coerceBooleanProperty(value);
|
|
24
|
+
}
|
|
25
|
+
}
|
|
26
|
+
/** @default '' (empty string) */
|
|
27
|
+
get placeholder() {
|
|
28
|
+
return this._placeholder;
|
|
29
|
+
}
|
|
30
|
+
set placeholder(value) {
|
|
31
|
+
if (notNullOrUndefined(value)) {
|
|
32
|
+
this._placeholder = value;
|
|
33
|
+
}
|
|
34
|
+
}
|
|
35
|
+
/**
|
|
36
|
+
* Initial rows visible in text area, set using `height` and `minHeight`.
|
|
37
|
+
*
|
|
38
|
+
* NOTE: Resizable editors will not be able to size below the height calculated from this value.
|
|
39
|
+
*
|
|
40
|
+
* @default 5
|
|
41
|
+
*/
|
|
42
|
+
get rows() {
|
|
43
|
+
return this._rows;
|
|
44
|
+
}
|
|
45
|
+
set rows(value) {
|
|
46
|
+
if (notNullOrUndefined(value)) {
|
|
47
|
+
this._rows = value;
|
|
48
|
+
this._pollCalculatedRowHeight.next();
|
|
49
|
+
}
|
|
50
|
+
}
|
|
51
|
+
/**
|
|
52
|
+
* When `true`, text area can be resized vertically.
|
|
53
|
+
*
|
|
54
|
+
* @default true
|
|
55
|
+
*/
|
|
56
|
+
get resizable() {
|
|
57
|
+
return this._resizable;
|
|
58
|
+
}
|
|
59
|
+
set resizable(value) {
|
|
60
|
+
if (notNullOrUndefined(value)) {
|
|
61
|
+
this._resizable = coerceBooleanProperty(value);
|
|
62
|
+
}
|
|
63
|
+
}
|
|
64
|
+
/** @default false */
|
|
65
|
+
get readOnly() {
|
|
66
|
+
return this._readOnly;
|
|
67
|
+
}
|
|
68
|
+
set readOnly(value) {
|
|
69
|
+
if (notNullOrUndefined(value)) {
|
|
70
|
+
this._readOnly = coerceBooleanProperty(value);
|
|
71
|
+
}
|
|
72
|
+
}
|
|
73
|
+
// TODO: fix bug where initial html value is rendered in plain text
|
|
74
|
+
/**
|
|
75
|
+
* When `true`, overrides all other provided settings to present the editor
|
|
76
|
+
* as a standard textbox.
|
|
77
|
+
*
|
|
78
|
+
* NOTE: For the moment, `<p></p>` tags are still allowed, triggered by the `Enter` key.
|
|
79
|
+
* https://github.com/slab/quill/issues/1432
|
|
80
|
+
*
|
|
81
|
+
* @default false
|
|
82
|
+
*/
|
|
83
|
+
get disableRichText() {
|
|
84
|
+
return this._disableRichText;
|
|
85
|
+
}
|
|
86
|
+
set disableRichText(value) {
|
|
87
|
+
if (notNullOrUndefined(value)) {
|
|
88
|
+
this._disableRichText = coerceBooleanProperty(value);
|
|
89
|
+
this._updateQuillConfig();
|
|
90
|
+
}
|
|
91
|
+
}
|
|
92
|
+
/**
|
|
93
|
+
* When `true`, displays a character counter at the bottom of the editor.
|
|
94
|
+
* Character counter text is determined by values provided in `minLength`
|
|
95
|
+
* and `maxLength`.
|
|
96
|
+
*
|
|
97
|
+
* The default count algorithm strips out html entities and replaces them
|
|
98
|
+
* with spaces to get the string length. To override this behavior, pass
|
|
99
|
+
* a custom function to `characterCounterFn`.
|
|
100
|
+
*
|
|
101
|
+
* To override default character counter display, pass a custom template to `characterCountTpl`.
|
|
102
|
+
*
|
|
103
|
+
* @default false
|
|
104
|
+
*/
|
|
105
|
+
get displayCharacterCounter() {
|
|
106
|
+
return this._displayCharacterCounter;
|
|
107
|
+
}
|
|
108
|
+
set displayCharacterCounter(value) {
|
|
109
|
+
if (notNullOrUndefined(value)) {
|
|
110
|
+
this._displayCharacterCounter = coerceBooleanProperty(value);
|
|
111
|
+
}
|
|
112
|
+
}
|
|
113
|
+
get minLength() {
|
|
114
|
+
return this._minLength;
|
|
115
|
+
}
|
|
116
|
+
set minLength(value) {
|
|
117
|
+
if (notNullOrUndefined(value)) {
|
|
118
|
+
this._minLength = value;
|
|
119
|
+
}
|
|
120
|
+
}
|
|
121
|
+
get maxLength() {
|
|
122
|
+
return this._maxLength;
|
|
123
|
+
}
|
|
124
|
+
set maxLength(value) {
|
|
125
|
+
if (notNullOrUndefined(value)) {
|
|
126
|
+
this._maxLength = value;
|
|
127
|
+
}
|
|
128
|
+
}
|
|
129
|
+
/**
|
|
130
|
+
* Set to override default character counter display.
|
|
131
|
+
*
|
|
132
|
+
* Template context includes variables `minLength`, `maxLength`, and `characterCount`.
|
|
133
|
+
*/
|
|
134
|
+
get characterCounterTpl() {
|
|
135
|
+
return this._characterCounterTpl;
|
|
136
|
+
}
|
|
137
|
+
set characterCounterTpl(value) {
|
|
138
|
+
if (notNullOrUndefined(value)) {
|
|
139
|
+
this._characterCounterTpl = value;
|
|
140
|
+
}
|
|
141
|
+
}
|
|
142
|
+
/**
|
|
143
|
+
* Set to override default counter function, which strips html entities and
|
|
144
|
+
* replaces them with a space.
|
|
145
|
+
*/
|
|
146
|
+
get characterCounterFn() {
|
|
147
|
+
return this._characterCounterFn;
|
|
148
|
+
}
|
|
149
|
+
set characterCounterFn(value) {
|
|
150
|
+
if (notNullOrUndefined(value)) {
|
|
151
|
+
this._characterCounterFn = value;
|
|
152
|
+
}
|
|
153
|
+
}
|
|
154
|
+
/**
|
|
155
|
+
* If `true`, component will configure the Quill editor with the quill-mentions extension
|
|
156
|
+
* and listen for values passed to `mentionItems` to populate the mentions menu.
|
|
157
|
+
*
|
|
158
|
+
* @default false
|
|
159
|
+
*/
|
|
160
|
+
get useMentions() {
|
|
161
|
+
return this._useMentions;
|
|
162
|
+
}
|
|
163
|
+
set useMentions(value) {
|
|
164
|
+
if (notNullOrUndefined(value)) {
|
|
165
|
+
this._useMentions = coerceBooleanProperty(value);
|
|
166
|
+
this._updateQuillConfig();
|
|
167
|
+
}
|
|
168
|
+
}
|
|
169
|
+
/**
|
|
170
|
+
* List of users, user groups, or other entities to display in mentions menu.
|
|
171
|
+
* Minimum required properties are `id` (unique) and `value`, which acts as the label.
|
|
172
|
+
*
|
|
173
|
+
* By default, the menu is triggered by typing the `@` symbol into the text area.
|
|
174
|
+
*/
|
|
175
|
+
get mentionItems() {
|
|
176
|
+
return this._mentionItems.value;
|
|
177
|
+
}
|
|
178
|
+
set mentionItems(value) {
|
|
179
|
+
if (notNullOrUndefined(value)) {
|
|
180
|
+
this._mentionItems.next(value);
|
|
181
|
+
}
|
|
182
|
+
}
|
|
183
|
+
/**
|
|
184
|
+
* Set to override default search function when user is typing a mention.
|
|
185
|
+
*/
|
|
186
|
+
get mentionSearchFn() {
|
|
187
|
+
return this._mentionSearchFn;
|
|
188
|
+
}
|
|
189
|
+
set mentionSearchFn(value) {
|
|
190
|
+
if (notNullOrUndefined(value)) {
|
|
191
|
+
this._mentionSearchFn = value;
|
|
192
|
+
}
|
|
193
|
+
}
|
|
194
|
+
/**
|
|
195
|
+
* Set to override default render function for mentions list.
|
|
196
|
+
*
|
|
197
|
+
* Function should call `renderList`.
|
|
198
|
+
*/
|
|
199
|
+
get mentionRenderListFn() {
|
|
200
|
+
return this._mentionRenderListFn;
|
|
201
|
+
}
|
|
202
|
+
set mentionRenderListFn(value) {
|
|
203
|
+
if (notNullOrUndefined(value)) {
|
|
204
|
+
this._mentionRenderListFn = value;
|
|
205
|
+
}
|
|
206
|
+
}
|
|
207
|
+
/**
|
|
208
|
+
* Set to override default text shown while mention items are loading.
|
|
209
|
+
*
|
|
210
|
+
* @default 'Loading...'
|
|
211
|
+
*/
|
|
212
|
+
get mentionListLoadingText() {
|
|
213
|
+
return this._mentionListLoadingText;
|
|
214
|
+
}
|
|
215
|
+
set mentionListLoadingText(value) {
|
|
216
|
+
if (notNullOrUndefined(value)) {
|
|
217
|
+
this._mentionListLoadingText = value;
|
|
218
|
+
}
|
|
219
|
+
}
|
|
220
|
+
/**
|
|
221
|
+
* Set to override default text shown when a mention search returns no items.
|
|
222
|
+
*
|
|
223
|
+
* @default 'No matches found.'
|
|
224
|
+
*/
|
|
225
|
+
get mentionListEmptyText() {
|
|
226
|
+
return this._mentionListEmptyText;
|
|
227
|
+
}
|
|
228
|
+
set mentionListEmptyText(value) {
|
|
229
|
+
if (notNullOrUndefined(value)) {
|
|
230
|
+
this._mentionListEmptyText = value;
|
|
231
|
+
}
|
|
232
|
+
}
|
|
233
|
+
get mentionListEmptyItem() {
|
|
234
|
+
return {
|
|
235
|
+
id: 'undefined',
|
|
236
|
+
value: this.mentionListEmptyText,
|
|
237
|
+
disabled: true,
|
|
238
|
+
emptyList: true
|
|
239
|
+
};
|
|
240
|
+
}
|
|
241
|
+
get quillEditor() {
|
|
242
|
+
return this._quillEditor;
|
|
243
|
+
}
|
|
244
|
+
set quillEditor(value) {
|
|
245
|
+
this._quillEditor = value;
|
|
246
|
+
// setTimeout because full html isn't available until the next tick
|
|
247
|
+
setTimeout(() => {
|
|
248
|
+
this._pollCalculatedRowHeight.next();
|
|
249
|
+
}, 0);
|
|
250
|
+
}
|
|
251
|
+
constructor(_renderer, _customConfig) {
|
|
252
|
+
this._renderer = _renderer;
|
|
253
|
+
this._customConfig = _customConfig;
|
|
254
|
+
this.formControl = new FormControl(null);
|
|
255
|
+
this._required = false;
|
|
256
|
+
this._placeholder = '';
|
|
257
|
+
this._rows = 5;
|
|
258
|
+
this._resizable = true;
|
|
259
|
+
this._readOnly = false;
|
|
260
|
+
this._disableRichText = false;
|
|
261
|
+
this._displayCharacterCounter = false;
|
|
262
|
+
this._characterCounterTpl = null;
|
|
263
|
+
this._characterCounterFn = defaultHtmlCharacterCounterFn;
|
|
264
|
+
this._useMentions = false;
|
|
265
|
+
this._mentionItems = new BehaviorSubject(undefined);
|
|
266
|
+
this.mentionItems$ = this._mentionItems.asObservable().pipe(filter(mentions => notNullOrUndefined(mentions) && mentions.length > 0), shareReplay({ bufferSize: 1, refCount: true }));
|
|
267
|
+
this._mentionSearchFn = defaultMentionSearchFn;
|
|
268
|
+
this._mentionRenderListFn = defaultMentionRenderListFn;
|
|
269
|
+
this._mentionListLoadingText = 'Loading...';
|
|
270
|
+
this._mentionListEmptyText = 'No matches found.';
|
|
271
|
+
this.quillEditorCreated = new EventEmitter();
|
|
272
|
+
this.quillEditorChanged = new EventEmitter();
|
|
273
|
+
this.quillContentChanged = new EventEmitter();
|
|
274
|
+
this.quillSelectionChanged = new EventEmitter();
|
|
275
|
+
this.quillFocus = new EventEmitter();
|
|
276
|
+
this.quillBlur = new EventEmitter();
|
|
277
|
+
this.mentionsUpdated = new EventEmitter();
|
|
278
|
+
this._isWritingValue = false;
|
|
279
|
+
this._pollCalculatedRowHeight = new Subject();
|
|
280
|
+
this._configSet = new BehaviorSubject(false);
|
|
281
|
+
this.configSet$ = this._configSet.asObservable();
|
|
282
|
+
this._stylesSet = new BehaviorSubject(false);
|
|
283
|
+
this._templateSet = new BehaviorSubject(false);
|
|
284
|
+
this._config = new BehaviorSubject(undefined);
|
|
285
|
+
this.config$ = this._config.asObservable();
|
|
286
|
+
this._selectedMentions = new BehaviorSubject([]);
|
|
287
|
+
this.selectedMentions$ = this._selectedMentions.asObservable();
|
|
288
|
+
this.initialized$ = combineLatest([
|
|
289
|
+
this._configSet.asObservable(),
|
|
290
|
+
this._stylesSet.asObservable(),
|
|
291
|
+
this._templateSet.asObservable(),
|
|
292
|
+
]).pipe(map(sets => sets.findIndex(s => !s) === -1));
|
|
293
|
+
this.characterCount$ = this.initialized$.pipe(filter(i => i), switchMap(() => {
|
|
294
|
+
if (notNullOrUndefined(this.formControl)) {
|
|
295
|
+
return this.formControl.valueChanges.pipe(startWith(this.formControl.value), map(v => this.characterCounterFn(v || '', this._config.value?.format)));
|
|
296
|
+
}
|
|
297
|
+
return of(0);
|
|
298
|
+
}));
|
|
299
|
+
this.selectedMentions$.pipe(tap(mentions => this.mentionsUpdated.emit(mentions))).subscribe();
|
|
300
|
+
}
|
|
301
|
+
ngOnInit() {
|
|
302
|
+
// TODO: test this more
|
|
303
|
+
// ignore quill initial valueChange event, to keep functionality in line with other inputs
|
|
304
|
+
let initialEmitComplete = false;
|
|
305
|
+
this._configSet.pipe(filter(s => s), switchMap(() => this.formControl.valueChanges.pipe(
|
|
306
|
+
// skip(1),
|
|
307
|
+
take(1), tap(() => initialEmitComplete = true)))).subscribe();
|
|
308
|
+
this.formControl.valueChanges.pipe(filter(() => !this._isWritingValue && initialEmitComplete), tap(value => this.value = value)).subscribe();
|
|
309
|
+
this._pollCalculatedRowHeight.asObservable().pipe(tap(() => {
|
|
310
|
+
this._stylesSet.next(false);
|
|
311
|
+
const calculatedRowsHeight = `${this.rows * 1.5}rem`;
|
|
312
|
+
if (notNullOrUndefined(this.rows) && notNullOrUndefined(this._quillEditor)) {
|
|
313
|
+
const editorEl = this._quillEditor.elementRef.nativeElement.querySelector('.ql-editor');
|
|
314
|
+
if (notNullOrUndefined(editorEl)) {
|
|
315
|
+
this._renderer.setStyle(editorEl, 'height', calculatedRowsHeight);
|
|
316
|
+
this._renderer.setStyle(editorEl, 'min-height', calculatedRowsHeight);
|
|
317
|
+
}
|
|
318
|
+
}
|
|
319
|
+
this._stylesSet.next(true);
|
|
320
|
+
})).subscribe();
|
|
321
|
+
this._buildQuillConfig();
|
|
322
|
+
}
|
|
323
|
+
ngAfterViewInit() {
|
|
324
|
+
if (isNullOrUndefined(this.characterCounterTpl)) {
|
|
325
|
+
this.characterCounterTpl = this.defaultCharacterCounterTpl || null;
|
|
326
|
+
}
|
|
327
|
+
this._templateSet.next(true);
|
|
328
|
+
}
|
|
329
|
+
_buildQuillConfig() {
|
|
330
|
+
this._configSet.next(false);
|
|
331
|
+
const config = {
|
|
332
|
+
...THESEAM_QUILL_EDITOR_CONFIG_DEFAULT
|
|
333
|
+
};
|
|
334
|
+
if (this.disableRichText) {
|
|
335
|
+
config.format = 'text';
|
|
336
|
+
config.formats = [];
|
|
337
|
+
config.modules = {
|
|
338
|
+
toolbar: false
|
|
339
|
+
};
|
|
340
|
+
}
|
|
341
|
+
else {
|
|
342
|
+
config.format = this._getConfigOrDefault('format');
|
|
343
|
+
config.formats = this._getConfigOrDefault('formats');
|
|
344
|
+
config.modules = {
|
|
345
|
+
...this._getConfigOrDefault('modules')
|
|
346
|
+
};
|
|
347
|
+
}
|
|
348
|
+
if (this.useMentions) {
|
|
349
|
+
const mentionModules = {
|
|
350
|
+
...config.modules,
|
|
351
|
+
mention: {
|
|
352
|
+
...THESEAM_QUILL_MENTION_OPTIONS_DEFAULT,
|
|
353
|
+
renderLoading: () => this.mentionListLoadingText,
|
|
354
|
+
source: async (searchTerm, renderList, mentionChar) => {
|
|
355
|
+
// this function is called every time the menu is triggered,
|
|
356
|
+
// so it will always have the latest value from mentionItems$
|
|
357
|
+
const mentionsAsync = await lastValueFrom(this.mentionItems$.pipe(take(1)));
|
|
358
|
+
if (notNullOrUndefined(mentionsAsync)) {
|
|
359
|
+
this.mentionRenderListFn(mentionsAsync, this.mentionSearchFn, this.mentionListEmptyItem, searchTerm, renderList, mentionChar);
|
|
360
|
+
}
|
|
361
|
+
},
|
|
362
|
+
...config.modules?.mention,
|
|
363
|
+
}
|
|
364
|
+
};
|
|
365
|
+
config.modules = mentionModules;
|
|
366
|
+
}
|
|
367
|
+
config.theme = this._getConfigOrDefault('theme');
|
|
368
|
+
config.debug = this._getConfigOrDefault('debug');
|
|
369
|
+
config.customToolbarPosition = this._getConfigOrDefault('customToolbarPosition');
|
|
370
|
+
config.sanitize = this._getConfigOrDefault('sanitize');
|
|
371
|
+
config.styles = this._getConfigOrDefault('styles');
|
|
372
|
+
config.strict = this._getConfigOrDefault('strict');
|
|
373
|
+
config.scrollingContainer = this._getConfigOrDefault('scrollingContainer');
|
|
374
|
+
config.bounds = this._getConfigOrDefault('bounds');
|
|
375
|
+
config.customOptions = this._getConfigOrDefault('customOptions');
|
|
376
|
+
config.customModules = this._getConfigOrDefault('customModules');
|
|
377
|
+
config.trackChanges = this._getConfigOrDefault('trackChanges');
|
|
378
|
+
config.preserveWhitespace = this._getConfigOrDefault('preserveWhitespace');
|
|
379
|
+
config.classes = this._getConfigOrDefault('classes');
|
|
380
|
+
config.trimOnValidation = this._getConfigOrDefault('trimOnValidation');
|
|
381
|
+
config.linkPlaceholder = this._getConfigOrDefault('linkPlaceholder');
|
|
382
|
+
config.compareValues = this._getConfigOrDefault('compareValues');
|
|
383
|
+
config.filterNull = this._getConfigOrDefault('filterNull');
|
|
384
|
+
config.debounceTime = this._getConfigOrDefault('debounceTime');
|
|
385
|
+
this._config.next(config);
|
|
386
|
+
// setTimeout bc ngx-quill library doesn't listen for input changes,
|
|
387
|
+
// so we must destroy and create the component each time config is updated
|
|
388
|
+
setTimeout(() => {
|
|
389
|
+
this._configSet.next(true);
|
|
390
|
+
}, 0);
|
|
391
|
+
}
|
|
392
|
+
_getConfigOrDefault(prop) {
|
|
393
|
+
if (this._customConfig && Object.prototype.hasOwnProperty.call(this._customConfig, prop)) {
|
|
394
|
+
return this._customConfig[prop];
|
|
395
|
+
}
|
|
396
|
+
return THESEAM_QUILL_EDITOR_CONFIG_DEFAULT[prop];
|
|
397
|
+
}
|
|
398
|
+
_updateQuillConfig() {
|
|
399
|
+
if (this._configSet.value) {
|
|
400
|
+
this._buildQuillConfig();
|
|
401
|
+
}
|
|
402
|
+
}
|
|
403
|
+
// Not usable, bc config has to be set immediately, potentially before mentions populate.
|
|
404
|
+
// To set customDataAttributes, provide a custom config.
|
|
405
|
+
// private _getCustomDataAttributes(mentions: TheSeamQuillMentionMenuOption[]): string[] {
|
|
406
|
+
// return mentions.reduce((acc, mention) => {
|
|
407
|
+
// const keys = Object.keys(mention)
|
|
408
|
+
// keys.forEach(key => {
|
|
409
|
+
// if (acc.findIndex(a => a === key) === -1) {
|
|
410
|
+
// acc.push(key)
|
|
411
|
+
// }
|
|
412
|
+
// })
|
|
413
|
+
// return acc
|
|
414
|
+
// }, <string[]>[] satisfies string[])
|
|
415
|
+
// }
|
|
416
|
+
get value() {
|
|
417
|
+
return this._value;
|
|
418
|
+
}
|
|
419
|
+
set value(value) {
|
|
420
|
+
this._value = value;
|
|
421
|
+
if (this.onChange) {
|
|
422
|
+
this.onChange(value);
|
|
423
|
+
}
|
|
424
|
+
if (this.onTouched) {
|
|
425
|
+
this.onTouched();
|
|
426
|
+
}
|
|
427
|
+
}
|
|
428
|
+
writeValue(value) {
|
|
429
|
+
this._isWritingValue = true;
|
|
430
|
+
this.formControl.setValue(value);
|
|
431
|
+
this.value = value;
|
|
432
|
+
this._isWritingValue = false;
|
|
433
|
+
}
|
|
434
|
+
registerOnChange(fn) {
|
|
435
|
+
this.onChange = fn;
|
|
436
|
+
}
|
|
437
|
+
registerOnTouched(fn) {
|
|
438
|
+
this.onTouched = fn;
|
|
439
|
+
}
|
|
440
|
+
setDisabledState(isDisabled) {
|
|
441
|
+
this.readOnly = isDisabled;
|
|
442
|
+
}
|
|
443
|
+
_onEditorCreated(event) {
|
|
444
|
+
this.quillEditorCreated.emit(event);
|
|
445
|
+
}
|
|
446
|
+
_onEditorChanged(event) {
|
|
447
|
+
this.quillEditorChanged.emit(event);
|
|
448
|
+
}
|
|
449
|
+
_onContentChanged(event) {
|
|
450
|
+
this.quillContentChanged.emit(event);
|
|
451
|
+
this._updateMentionsFromDelta(event.content);
|
|
452
|
+
}
|
|
453
|
+
_onSelectionChanged(event) {
|
|
454
|
+
this.quillSelectionChanged.emit(event);
|
|
455
|
+
}
|
|
456
|
+
_onFocus(event) {
|
|
457
|
+
this.quillFocus.emit(event);
|
|
458
|
+
}
|
|
459
|
+
_onBlur(event) {
|
|
460
|
+
this.quillBlur.emit(event);
|
|
461
|
+
if (this.onTouched) {
|
|
462
|
+
this.onTouched();
|
|
463
|
+
}
|
|
464
|
+
}
|
|
465
|
+
/**
|
|
466
|
+
* Hacky way to track mention inserts/deletes
|
|
467
|
+
*/
|
|
468
|
+
_updateMentionsFromDelta(content) {
|
|
469
|
+
if (notNullOrUndefined(content.ops)) {
|
|
470
|
+
const contentMentionIds = content.ops.map(o => o.insert?.mention?.id).filter(notNullOrUndefined);
|
|
471
|
+
const selectedMentions = [...this._selectedMentions.value];
|
|
472
|
+
const mentionOptions = [...this._mentionItems.value || []];
|
|
473
|
+
const newMentions = contentMentionIds.reduce((acc, mentionId) => {
|
|
474
|
+
const insertMention = mentionOptions.find(m => isMentionMenuOption(m) && m.id === mentionId);
|
|
475
|
+
if (notNullOrUndefined(insertMention)) {
|
|
476
|
+
acc.push(insertMention);
|
|
477
|
+
}
|
|
478
|
+
else {
|
|
479
|
+
console.warn('Mention addition failed! Selected mention option not found:', mentionId);
|
|
480
|
+
}
|
|
481
|
+
return acc;
|
|
482
|
+
}, []);
|
|
483
|
+
let emitUpdate = false;
|
|
484
|
+
if (selectedMentions.length !== newMentions.length) {
|
|
485
|
+
// if the length has changed, we know an update occurred
|
|
486
|
+
emitUpdate = true;
|
|
487
|
+
}
|
|
488
|
+
else {
|
|
489
|
+
// otherwise, test ids for old and new items to see if we need to emit a change
|
|
490
|
+
const selectedMentionIds = this._selectedMentions.value.map(m => m.id);
|
|
491
|
+
const newMentionIds = newMentions.map(m => m.id);
|
|
492
|
+
if (selectedMentionIds.findIndex(m => !newMentionIds.includes(m)) !== -1 ||
|
|
493
|
+
newMentionIds.findIndex(m => !selectedMentionIds.includes(m)) !== -1) {
|
|
494
|
+
emitUpdate = true;
|
|
495
|
+
}
|
|
496
|
+
}
|
|
497
|
+
if (emitUpdate) {
|
|
498
|
+
this._selectedMentions.next(newMentions);
|
|
499
|
+
}
|
|
500
|
+
}
|
|
501
|
+
}
|
|
502
|
+
}
|
|
503
|
+
RichTextComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "15.2.7", ngImport: i0, type: RichTextComponent, deps: [{ token: i0.Renderer2 }, { token: THESEAM_QUILL_EDITOR_CONFIG, optional: true }], target: i0.ɵɵFactoryTarget.Component });
|
|
504
|
+
RichTextComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "15.2.7", type: RichTextComponent, selector: "seam-rich-text", inputs: { val: ["value", "val"], required: "required", placeholder: "placeholder", rows: "rows", resizable: "resizable", disableRichText: "disableRichText", displayCharacterCounter: "displayCharacterCounter", minLength: "minLength", maxLength: "maxLength", characterCounterTpl: "characterCounterTpl", characterCounterFn: "characterCounterFn", useMentions: "useMentions", mentionItems: "mentionItems", mentionSearchFn: "mentionSearchFn", mentionRenderListFn: "mentionRenderListFn", mentionListLoadingText: "mentionListLoadingText", mentionListEmptyText: "mentionListEmptyText" }, outputs: { quillEditorCreated: "quillEditorCreated", quillEditorChanged: "quillEditorChanged", quillContentChanged: "quillContentChanged", quillSelectionChanged: "quillSelectionChanged", quillFocus: "quillFocus", quillBlur: "quillBlur", mentionsUpdated: "mentionsUpdated" }, providers: [
|
|
505
|
+
RICH_TEXT_VALUE_ACCESSOR
|
|
506
|
+
], viewQueries: [{ propertyName: "quillEditor", first: true, predicate: ["quillEditor"], descendants: true }, { propertyName: "defaultCharacterCounterTpl", first: true, predicate: ["characterCounter"], descendants: true }], ngImport: i0, template: "<div class=\"editor\" *ngIf=\"configSet$ | async\" [class.initializing]=\"!(initialized$ | async)\">\n <ng-container *ngIf=\"config$ | async as config\">\n <quill-editor\n #quillEditor\n seamInput\n [class.disabled]=\"readOnly\"\n [attr.disabled]=\"readOnly\"\n [formControl]=\"formControl\"\n [placeholder]=\"placeholder\"\n [readOnly]=\"readOnly\"\n [required]=\"required\"\n [minLength]=\"minLength\"\n [maxLength]=\"maxLength\"\n [format]=\"config.format\"\n [theme]=\"config.theme\"\n [modules]=\"config.modules\"\n [debug]=\"config.debug\"\n [formats]=\"config.formats\"\n [customToolbarPosition]=\"config.customToolbarPosition\"\n [sanitize]=\"config.sanitize\"\n [styles]=\"config.styles\"\n [strict]=\"config.strict\"\n [scrollingContainer]=\"config.scrollingContainer\"\n [bounds]=\"config.bounds\"\n [customOptions]=\"config.customOptions\"\n [customModules]=\"config.customModules\"\n [trackChanges]=\"config.trackChanges\"\n [preserveWhitespace]=\"config.preserveWhitespace\"\n [classes]=\"config.classes\"\n [trimOnValidation]=\"config.trimOnValidation\"\n [linkPlaceholder]=\"config.linkPlaceholder\"\n [compareValues]=\"config.compareValues\"\n [filterNull]=\"config.filterNull\"\n [debounceTime]=\"config.debounceTime\"\n [class.can-resize]=\"resizable\"\n (onEditorCreated)=\"_onEditorCreated($event)\"\n (onEditorChanged)=\"_onEditorChanged($event)\"\n (onContentChanged)=\"_onContentChanged($event)\"\n (onSelectionChanged)=\"_onSelectionChanged($event)\"\n (onFocus)=\"_onFocus($event)\"\n (onBlur)=\"_onBlur($event)\"></quill-editor>\n <ng-container *ngIf=\"displayCharacterCounter\">\n <ng-container\n [ngTemplateOutlet]=\"characterCounterTpl\"\n [ngTemplateOutletContext]=\"{ $implicit: characterCount$ | async, minLength: minLength, maxLength: maxLength, characterCount: characterCount$ | async }\"></ng-container>\n </ng-container>\n </ng-container>\n</div>\n\n<ng-template #characterCounter let-implicit let-minLength=\"minLength\" let-maxLength=\"maxLength\" let-characterCount=\"characterCount\">\n <div class=\"small text-black-50 text-right\">\n <ng-container *ngIf=\"characterCount || characterCount === 0\">\n Character Count:\n <span [class.text-danger]=\"characterCount && ((maxLength && characterCount > maxLength) || (minLength && characterCount < minLength))\">\n {{ characterCount | number }}\n </span>\n <ng-container *ngIf=\"maxLength\">\n / {{ maxLength }}\n </ng-container>\n <span *ngIf=\"minLength\" class=\"font-italic\">\n (minimum {{ minLength }} expected)\n </span>\n </ng-container>\n </div>\n</ng-template>\n", styles: [":host .editor{position:relative}:host .editor.initializing:before{content:\"\";position:absolute;top:0;left:0;width:100%;height:100%;background:#f8f9fa;z-index:1}:host quill-editor.can-resize ::ng-deep .ql-editor{overflow:auto;resize:vertical}\n"], dependencies: [{ kind: "directive", type: i1.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "directive", type: i1.NgTemplateOutlet, selector: "[ngTemplateOutlet]", inputs: ["ngTemplateOutletContext", "ngTemplateOutlet", "ngTemplateOutletInjector"] }, { kind: "directive", type: i2.NgControlStatus, selector: "[formControlName],[ngModel],[formControl]" }, { kind: "directive", type: i2.RequiredValidator, selector: ":not([type=checkbox])[required][formControlName],:not([type=checkbox])[required][formControl],:not([type=checkbox])[required][ngModel]", inputs: ["required"] }, { kind: "directive", type: i2.FormControlDirective, selector: "[formControl]", inputs: ["formControl", "disabled", "ngModel"], outputs: ["ngModelChange"], exportAs: ["ngForm"] }, { kind: "component", type: i3.QuillEditorComponent, selector: "quill-editor" }, { kind: "pipe", type: i1.AsyncPipe, name: "async" }, { kind: "pipe", type: i1.DecimalPipe, name: "number" }] });
|
|
507
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "15.2.7", ngImport: i0, type: RichTextComponent, decorators: [{
|
|
508
|
+
type: Component,
|
|
509
|
+
args: [{ selector: 'seam-rich-text', providers: [
|
|
510
|
+
RICH_TEXT_VALUE_ACCESSOR
|
|
511
|
+
], template: "<div class=\"editor\" *ngIf=\"configSet$ | async\" [class.initializing]=\"!(initialized$ | async)\">\n <ng-container *ngIf=\"config$ | async as config\">\n <quill-editor\n #quillEditor\n seamInput\n [class.disabled]=\"readOnly\"\n [attr.disabled]=\"readOnly\"\n [formControl]=\"formControl\"\n [placeholder]=\"placeholder\"\n [readOnly]=\"readOnly\"\n [required]=\"required\"\n [minLength]=\"minLength\"\n [maxLength]=\"maxLength\"\n [format]=\"config.format\"\n [theme]=\"config.theme\"\n [modules]=\"config.modules\"\n [debug]=\"config.debug\"\n [formats]=\"config.formats\"\n [customToolbarPosition]=\"config.customToolbarPosition\"\n [sanitize]=\"config.sanitize\"\n [styles]=\"config.styles\"\n [strict]=\"config.strict\"\n [scrollingContainer]=\"config.scrollingContainer\"\n [bounds]=\"config.bounds\"\n [customOptions]=\"config.customOptions\"\n [customModules]=\"config.customModules\"\n [trackChanges]=\"config.trackChanges\"\n [preserveWhitespace]=\"config.preserveWhitespace\"\n [classes]=\"config.classes\"\n [trimOnValidation]=\"config.trimOnValidation\"\n [linkPlaceholder]=\"config.linkPlaceholder\"\n [compareValues]=\"config.compareValues\"\n [filterNull]=\"config.filterNull\"\n [debounceTime]=\"config.debounceTime\"\n [class.can-resize]=\"resizable\"\n (onEditorCreated)=\"_onEditorCreated($event)\"\n (onEditorChanged)=\"_onEditorChanged($event)\"\n (onContentChanged)=\"_onContentChanged($event)\"\n (onSelectionChanged)=\"_onSelectionChanged($event)\"\n (onFocus)=\"_onFocus($event)\"\n (onBlur)=\"_onBlur($event)\"></quill-editor>\n <ng-container *ngIf=\"displayCharacterCounter\">\n <ng-container\n [ngTemplateOutlet]=\"characterCounterTpl\"\n [ngTemplateOutletContext]=\"{ $implicit: characterCount$ | async, minLength: minLength, maxLength: maxLength, characterCount: characterCount$ | async }\"></ng-container>\n </ng-container>\n </ng-container>\n</div>\n\n<ng-template #characterCounter let-implicit let-minLength=\"minLength\" let-maxLength=\"maxLength\" let-characterCount=\"characterCount\">\n <div class=\"small text-black-50 text-right\">\n <ng-container *ngIf=\"characterCount || characterCount === 0\">\n Character Count:\n <span [class.text-danger]=\"characterCount && ((maxLength && characterCount > maxLength) || (minLength && characterCount < minLength))\">\n {{ characterCount | number }}\n </span>\n <ng-container *ngIf=\"maxLength\">\n / {{ maxLength }}\n </ng-container>\n <span *ngIf=\"minLength\" class=\"font-italic\">\n (minimum {{ minLength }} expected)\n </span>\n </ng-container>\n </div>\n</ng-template>\n", styles: [":host .editor{position:relative}:host .editor.initializing:before{content:\"\";position:absolute;top:0;left:0;width:100%;height:100%;background:#f8f9fa;z-index:1}:host quill-editor.can-resize ::ng-deep .ql-editor{overflow:auto;resize:vertical}\n"] }]
|
|
512
|
+
}], ctorParameters: function () { return [{ type: i0.Renderer2 }, { type: undefined, decorators: [{
|
|
513
|
+
type: Optional
|
|
514
|
+
}, {
|
|
515
|
+
type: Inject,
|
|
516
|
+
args: [THESEAM_QUILL_EDITOR_CONFIG]
|
|
517
|
+
}] }]; }, propDecorators: { val: [{
|
|
518
|
+
type: Input,
|
|
519
|
+
args: ['value']
|
|
520
|
+
}], required: [{
|
|
521
|
+
type: Input
|
|
522
|
+
}], placeholder: [{
|
|
523
|
+
type: Input
|
|
524
|
+
}], rows: [{
|
|
525
|
+
type: Input
|
|
526
|
+
}], resizable: [{
|
|
527
|
+
type: Input
|
|
528
|
+
}], disableRichText: [{
|
|
529
|
+
type: Input
|
|
530
|
+
}], displayCharacterCounter: [{
|
|
531
|
+
type: Input
|
|
532
|
+
}], minLength: [{
|
|
533
|
+
type: Input
|
|
534
|
+
}], maxLength: [{
|
|
535
|
+
type: Input
|
|
536
|
+
}], characterCounterTpl: [{
|
|
537
|
+
type: Input
|
|
538
|
+
}], characterCounterFn: [{
|
|
539
|
+
type: Input
|
|
540
|
+
}], useMentions: [{
|
|
541
|
+
type: Input
|
|
542
|
+
}], mentionItems: [{
|
|
543
|
+
type: Input
|
|
544
|
+
}], mentionSearchFn: [{
|
|
545
|
+
type: Input
|
|
546
|
+
}], mentionRenderListFn: [{
|
|
547
|
+
type: Input
|
|
548
|
+
}], mentionListLoadingText: [{
|
|
549
|
+
type: Input
|
|
550
|
+
}], mentionListEmptyText: [{
|
|
551
|
+
type: Input
|
|
552
|
+
}], quillEditorCreated: [{
|
|
553
|
+
type: Output
|
|
554
|
+
}], quillEditorChanged: [{
|
|
555
|
+
type: Output
|
|
556
|
+
}], quillContentChanged: [{
|
|
557
|
+
type: Output
|
|
558
|
+
}], quillSelectionChanged: [{
|
|
559
|
+
type: Output
|
|
560
|
+
}], quillFocus: [{
|
|
561
|
+
type: Output
|
|
562
|
+
}], quillBlur: [{
|
|
563
|
+
type: Output
|
|
564
|
+
}], mentionsUpdated: [{
|
|
565
|
+
type: Output
|
|
566
|
+
}], quillEditor: [{
|
|
567
|
+
type: ViewChild,
|
|
568
|
+
args: ['quillEditor']
|
|
569
|
+
}], defaultCharacterCounterTpl: [{
|
|
570
|
+
type: ViewChild,
|
|
571
|
+
args: ['characterCounter']
|
|
572
|
+
}] } });
|
|
573
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"rich-text.component.js","sourceRoot":"","sources":["../../../../../projects/ui-common/rich-text/rich-text/rich-text.component.ts","../../../../../projects/ui-common/rich-text/rich-text/rich-text.component.html"],"names":[],"mappings":"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","sourcesContent":["import { coerceBooleanProperty } from '@angular/cdk/coercion';\nimport { AfterViewInit, Component, EventEmitter, Inject, Input, OnInit, Optional, Output, Provider, Renderer2, TemplateRef, ViewChild, forwardRef } from '@angular/core';\nimport { ControlValueAccessor, FormControl, NG_VALUE_ACCESSOR } from '@angular/forms';\n\nimport { Blur, ContentChange, EditorChangeContent, EditorChangeSelection, Focus, QuillEditorComponent, SelectionChange } from 'ngx-quill';\nimport { Delta } from 'quill';\nimport 'quill-mention';\nimport { BehaviorSubject, Observable, Subject, combineLatest, filter, lastValueFrom, map, of, shareReplay, startWith, switchMap, take, tap } from 'rxjs';\n\nimport { isNullOrUndefined, notNullOrUndefined } from '@theseam/ui-common/utils';\nimport { TheSeamCharacterCounterFn, TheSeamQuillEditorConfig, TheSeamQuillMentionMenuItem, TheSeamQuillMentionMenuOption, TheSeamQuillMentionSearchFn, TheSeamQuillMentionSourceFn, TheSeamQuillModules } from '../utils/models';\nimport { THESEAM_QUILL_EDITOR_CONFIG, THESEAM_QUILL_EDITOR_CONFIG_DEFAULT, THESEAM_QUILL_MENTION_OPTIONS_DEFAULT, defaultHtmlCharacterCounterFn, defaultMentionRenderListFn, defaultMentionSearchFn, isMentionMenuOption } from '../utils/utils';\n\nexport const RICH_TEXT_VALUE_ACCESSOR: Provider = {\n  provide: NG_VALUE_ACCESSOR,\n  useExisting: forwardRef(() => RichTextComponent),\n  multi: true\n}\n\n@Component({\n  selector: 'seam-rich-text',\n  templateUrl: './rich-text.component.html',\n  styleUrls: ['./rich-text.component.scss'],\n  providers: [\n    RICH_TEXT_VALUE_ACCESSOR\n  ]\n})\nexport class RichTextComponent implements OnInit, AfterViewInit, ControlValueAccessor {\n\n  onChange: any\n  onTouched: any\n\n  // eslint-disable-next-line @angular-eslint/no-input-rename\n  @Input('value') val: string | undefined\n\n  formControl = new FormControl<string | null>(null)\n\n  @Input()\n  get required(): boolean {\n    return this._required\n  }\n  set required(value: boolean | null | undefined) {\n    if (notNullOrUndefined(value)) {\n      this._required = coerceBooleanProperty(value)\n    }\n  }\n  private _required = false\n\n  /** @default '' (empty string) */\n  @Input()\n  get placeholder(): string {\n    return this._placeholder\n  }\n  set placeholder(value: string | null | undefined) {\n    if (notNullOrUndefined(value)) {\n      this._placeholder = value\n    }\n  }\n  private _placeholder = ''\n\n  /**\n   * Initial rows visible in text area, set using `height` and `minHeight`.\n   *\n   * NOTE: Resizable editors will not be able to size below the height calculated from this value.\n   *\n   * @default 5\n   */\n  @Input()\n  get rows(): number {\n    return this._rows\n  }\n  set rows(value: number | null | undefined) {\n    if (notNullOrUndefined(value)) {\n      this._rows = value\n\n      this._pollCalculatedRowHeight.next()\n    }\n  }\n  private _rows = 5\n\n  /**\n   * When `true`, text area can be resized vertically.\n   *\n   * @default true\n   */\n  @Input()\n  get resizable(): boolean {\n    return this._resizable\n  }\n  set resizable(value: boolean | null | undefined) {\n    if (notNullOrUndefined(value)) {\n      this._resizable = coerceBooleanProperty(value)\n    }\n  }\n  private _resizable = true\n\n  /** @default false */\n  get readOnly(): boolean {\n    return this._readOnly\n  }\n  set readOnly(value: boolean | null | undefined) {\n    if (notNullOrUndefined(value)) {\n      this._readOnly = coerceBooleanProperty(value)\n    }\n  }\n  private _readOnly = false\n\n  // TODO: fix bug where initial html value is rendered in plain text\n  /**\n   * When `true`, overrides all other provided settings to present the editor\n   * as a standard textbox.\n   *\n   * NOTE: For the moment, `<p></p>` tags are still allowed, triggered by the `Enter` key.\n   * https://github.com/slab/quill/issues/1432\n   *\n   * @default false\n   */\n  @Input()\n  get disableRichText(): boolean {\n    return this._disableRichText\n  }\n  set disableRichText(value: boolean | null | undefined) {\n    if (notNullOrUndefined(value)) {\n      this._disableRichText = coerceBooleanProperty(value)\n\n      this._updateQuillConfig()\n    }\n  }\n  private _disableRichText = false\n\n  /**\n   * When `true`, displays a character counter at the bottom of the editor.\n   * Character counter text is determined by values provided in `minLength`\n   * and `maxLength`.\n   *\n   * The default count algorithm strips out html entities and replaces them\n   * with spaces to get the string length. To override this behavior, pass\n   * a custom function to `characterCounterFn`.\n   *\n   * To override default character counter display, pass a custom template to `characterCountTpl`.\n   *\n   * @default false\n   */\n  @Input()\n  get displayCharacterCounter(): boolean {\n    return this._displayCharacterCounter\n  }\n  set displayCharacterCounter(value: boolean | null | undefined) {\n    if (notNullOrUndefined(value)) {\n      this._displayCharacterCounter = coerceBooleanProperty(value)\n    }\n  }\n  private _displayCharacterCounter = false\n\n  @Input()\n  get minLength(): number | undefined {\n    return this._minLength\n  }\n  set minLength(value: number | undefined) {\n    if (notNullOrUndefined(value)) {\n      this._minLength = value\n    }\n  }\n  private _minLength: number | undefined\n\n  @Input()\n  get maxLength(): number | undefined {\n    return this._maxLength\n  }\n  set maxLength(value: number | undefined) {\n    if (notNullOrUndefined(value)) {\n      this._maxLength = value\n    }\n  }\n  private _maxLength: number | undefined\n\n  /**\n   * Set to override default character counter display.\n   *\n   * Template context includes variables `minLength`, `maxLength`, and `characterCount`.\n   */\n  @Input()\n  get characterCounterTpl(): TemplateRef<any> | null {\n    return this._characterCounterTpl\n  }\n  set characterCounterTpl(value) {\n    if (notNullOrUndefined(value)) {\n      this._characterCounterTpl = value\n    }\n  }\n  private _characterCounterTpl: TemplateRef<any> | null = null\n\n  /**\n   * Set to override default counter function, which strips html entities and\n   * replaces them with a space.\n   */\n  @Input() get characterCounterFn(): TheSeamCharacterCounterFn {\n    return this._characterCounterFn\n  }\n  set characterCounterFn(value: TheSeamCharacterCounterFn | null | undefined) {\n    if (notNullOrUndefined(value)) {\n      this._characterCounterFn = value\n    }\n  }\n  private _characterCounterFn = defaultHtmlCharacterCounterFn\n\n  /**\n   * If `true`, component will configure the Quill editor with the quill-mentions extension\n   * and listen for values passed to `mentionItems` to populate the mentions menu.\n   *\n   * @default false\n   */\n  @Input()\n  get useMentions(): boolean {\n    return this._useMentions\n  }\n  set useMentions(value: boolean | null | undefined) {\n    if (notNullOrUndefined(value)) {\n      this._useMentions = coerceBooleanProperty(value)\n\n      this._updateQuillConfig()\n    }\n  }\n  private _useMentions: boolean = false\n\n  /**\n   * List of users, user groups, or other entities to display in mentions menu.\n   * Minimum required properties are `id` (unique) and `value`, which acts as the label.\n   *\n   * By default, the menu is triggered by typing the `@` symbol into the text area.\n   */\n  @Input()\n  get mentionItems(): TheSeamQuillMentionMenuItem[] | null | undefined {\n    return this._mentionItems.value\n  }\n  set mentionItems(value: TheSeamQuillMentionMenuItem[] | null | undefined) {\n    if (notNullOrUndefined(value)) {\n      this._mentionItems.next(value)\n    }\n  }\n  private _mentionItems = new BehaviorSubject<TheSeamQuillMentionMenuItem[] | null | undefined>(undefined)\n  public mentionItems$ = this._mentionItems.asObservable().pipe(\n    filter(mentions => notNullOrUndefined(mentions) && mentions.length > 0),\n    shareReplay({ bufferSize: 1, refCount: true })\n  )\n\n  /**\n   * Set to override default search function when user is typing a mention.\n   */\n  @Input()\n  get mentionSearchFn(): TheSeamQuillMentionSearchFn {\n    return this._mentionSearchFn\n  }\n  set mentionSearchFn(value: TheSeamQuillMentionSearchFn | null | undefined) {\n    if (notNullOrUndefined(value)) {\n      this._mentionSearchFn = value\n    }\n  }\n  private _mentionSearchFn: TheSeamQuillMentionSearchFn = defaultMentionSearchFn\n\n  /**\n   * Set to override default render function for mentions list.\n   *\n   * Function should call `renderList`.\n   */\n  @Input()\n  get mentionRenderListFn(): TheSeamQuillMentionSourceFn {\n    return this._mentionRenderListFn\n  }\n  set mentionRenderListFn(value: TheSeamQuillMentionSourceFn | null | undefined) {\n    if (notNullOrUndefined(value)) {\n      this._mentionRenderListFn = value\n    }\n  }\n  private _mentionRenderListFn: TheSeamQuillMentionSourceFn = defaultMentionRenderListFn\n\n  /**\n   * Set to override default text shown while mention items are loading.\n   *\n   * @default 'Loading...'\n   */\n  @Input()\n  get mentionListLoadingText(): string {\n    return this._mentionListLoadingText\n  }\n  set mentionListLoadingText(value: string | null | undefined) {\n    if (notNullOrUndefined(value)) {\n      this._mentionListLoadingText = value\n    }\n  }\n  private _mentionListLoadingText: string = 'Loading...'\n\n  /**\n   * Set to override default text shown when a mention search returns no items.\n   *\n   * @default 'No matches found.'\n   */\n  @Input()\n  get mentionListEmptyText(): string {\n    return this._mentionListEmptyText\n  }\n  set mentionListEmptyText(value: string | null | undefined) {\n    if (notNullOrUndefined(value)) {\n      this._mentionListEmptyText = value\n    }\n  }\n  private _mentionListEmptyText: string = 'No matches found.'\n\n  get mentionListEmptyItem(): TheSeamQuillMentionMenuOption {\n    return {\n      id: 'undefined',\n      value: this.mentionListEmptyText,\n      disabled: true,\n      emptyList: true\n    }\n  }\n\n  @Output() quillEditorCreated: EventEmitter<any> = new EventEmitter()\n  @Output() quillEditorChanged: EventEmitter<EditorChangeContent | EditorChangeSelection> = new EventEmitter()\n  @Output() quillContentChanged: EventEmitter<ContentChange> = new EventEmitter()\n  @Output() quillSelectionChanged: EventEmitter<SelectionChange> = new EventEmitter()\n  @Output() quillFocus: EventEmitter<Focus> = new EventEmitter()\n  @Output() quillBlur: EventEmitter<Blur> = new EventEmitter()\n\n  @Output() mentionsUpdated: EventEmitter<TheSeamQuillMentionMenuOption[]> = new EventEmitter()\n\n  @ViewChild('quillEditor')\n  get quillEditor(): TemplateRef<QuillEditorComponent> | undefined {\n    return this._quillEditor\n  }\n  set quillEditor(value: TemplateRef<QuillEditorComponent> | undefined) {\n    this._quillEditor = value\n\n    // setTimeout because full html isn't available until the next tick\n    setTimeout(() => {\n      this._pollCalculatedRowHeight.next()\n    }, 0)\n  }\n  private _quillEditor: TemplateRef<QuillEditorComponent> | undefined\n\n  @ViewChild('characterCounter') defaultCharacterCounterTpl: TemplateRef<any> | undefined\n\n  private _isWritingValue = false\n\n  private _pollCalculatedRowHeight = new Subject<void>()\n\n  private _configSet = new BehaviorSubject<boolean>(false)\n  public configSet$ = this._configSet.asObservable()\n\n  private _stylesSet = new BehaviorSubject<boolean>(false)\n  private _templateSet = new BehaviorSubject<boolean>(false)\n\n  public initialized$: Observable<boolean>\n\n  public characterCount$: Observable<number>\n\n  private _config = new BehaviorSubject<TheSeamQuillEditorConfig | undefined>(undefined)\n  public config$ = this._config.asObservable()\n\n  private _selectedMentions = new BehaviorSubject<TheSeamQuillMentionMenuOption[]>([])\n  public selectedMentions$ = this._selectedMentions.asObservable()\n\n  constructor(\n    private readonly _renderer: Renderer2,\n    @Optional() @Inject(THESEAM_QUILL_EDITOR_CONFIG) private _customConfig?: TheSeamQuillEditorConfig\n  ) {\n    this.initialized$ = combineLatest([\n      this._configSet.asObservable(),\n      this._stylesSet.asObservable(),\n      this._templateSet.asObservable(),\n    ]).pipe(\n      map(sets => sets.findIndex(s => !s) === -1)\n    )\n\n    this.characterCount$ = this.initialized$.pipe(\n      filter(i => i),\n      switchMap(() => {\n        if (notNullOrUndefined(this.formControl)) {\n          return this.formControl.valueChanges.pipe(\n            startWith(this.formControl.value),\n            map(v => this.characterCounterFn(v || '' as string, this._config.value?.format)),\n          )\n        }\n\n        return of(0)\n      })\n    )\n\n    this.selectedMentions$.pipe(\n      tap(mentions => this.mentionsUpdated.emit(mentions))\n    ).subscribe()\n  }\n\n  ngOnInit(): void {\n    // TODO: test this more\n    // ignore quill initial valueChange event, to keep functionality in line with other inputs\n    let initialEmitComplete = false\n    this._configSet.pipe(\n      filter(s => s),\n      switchMap(() => this.formControl.valueChanges.pipe(\n        // skip(1),\n        take(1),\n        tap(() => initialEmitComplete = true)\n      ))\n    ).subscribe()\n\n    this.formControl.valueChanges.pipe(\n      filter(() => !this._isWritingValue && initialEmitComplete),\n      tap(value => this.value = value)\n    ).subscribe()\n\n    this._pollCalculatedRowHeight.asObservable().pipe(\n      tap(() => {\n        this._stylesSet.next(false)\n\n        const calculatedRowsHeight = `${this.rows * 1.5}rem`\n\n        if (notNullOrUndefined(this.rows) && notNullOrUndefined(this._quillEditor)) {\n          const editorEl = this._quillEditor.elementRef.nativeElement.querySelector('.ql-editor')\n\n          if (notNullOrUndefined(editorEl)) {\n            this._renderer.setStyle(editorEl, 'height', calculatedRowsHeight)\n            this._renderer.setStyle(editorEl, 'min-height', calculatedRowsHeight)\n          }\n        }\n\n        this._stylesSet.next(true)\n      })\n    ).subscribe()\n\n    this._buildQuillConfig()\n  }\n\n  ngAfterViewInit(): void {\n    if (isNullOrUndefined(this.characterCounterTpl)) {\n      this.characterCounterTpl = this.defaultCharacterCounterTpl || null\n    }\n\n    this._templateSet.next(true)\n  }\n\n  private _buildQuillConfig() {\n    this._configSet.next(false)\n\n    const config: TheSeamQuillEditorConfig = {\n      ...THESEAM_QUILL_EDITOR_CONFIG_DEFAULT\n    }\n\n    if (this.disableRichText) {\n      config.format = 'text'\n      config.formats = []\n      config.modules = {\n        toolbar: false\n      }\n    }\n    else {\n      config.format = this._getConfigOrDefault('format')\n      config.formats = this._getConfigOrDefault('formats')\n      config.modules = {\n        ...this._getConfigOrDefault('modules')\n      }\n    }\n\n    if (this.useMentions) {\n      const mentionModules: TheSeamQuillModules = {\n        ...config.modules,\n        mention: {\n          ...THESEAM_QUILL_MENTION_OPTIONS_DEFAULT,\n          renderLoading: () => this.mentionListLoadingText,\n          source: async (searchTerm: string, renderList: (list: any[], searchTerm: string) => void, mentionChar: string) => {\n            // this function is called every time the menu is triggered,\n            // so it will always have the latest value from mentionItems$\n            const mentionsAsync = await lastValueFrom(this.mentionItems$.pipe(take(1)))\n\n            if (notNullOrUndefined(mentionsAsync)) {\n              this.mentionRenderListFn(mentionsAsync, this.mentionSearchFn, this.mentionListEmptyItem, searchTerm, renderList, mentionChar)\n            }\n          },\n          ...config.modules?.mention,\n        }\n      }\n\n      config.modules = mentionModules\n    }\n\n    config.theme = this._getConfigOrDefault('theme')\n    config.debug = this._getConfigOrDefault('debug')\n    config.customToolbarPosition = this._getConfigOrDefault('customToolbarPosition')\n    config.sanitize = this._getConfigOrDefault('sanitize')\n    config.styles = this._getConfigOrDefault('styles')\n    config.strict = this._getConfigOrDefault('strict')\n    config.scrollingContainer = this._getConfigOrDefault('scrollingContainer')\n    config.bounds = this._getConfigOrDefault('bounds')\n    config.customOptions = this._getConfigOrDefault('customOptions')\n    config.customModules = this._getConfigOrDefault('customModules')\n    config.trackChanges = this._getConfigOrDefault('trackChanges')\n    config.preserveWhitespace = this._getConfigOrDefault('preserveWhitespace')\n    config.classes = this._getConfigOrDefault('classes')\n    config.trimOnValidation = this._getConfigOrDefault('trimOnValidation')\n    config.linkPlaceholder = this._getConfigOrDefault('linkPlaceholder')\n    config.compareValues = this._getConfigOrDefault('compareValues')\n    config.filterNull = this._getConfigOrDefault('filterNull')\n    config.debounceTime = this._getConfigOrDefault('debounceTime')\n\n    this._config.next(config)\n\n    // setTimeout bc ngx-quill library doesn't listen for input changes,\n    // so we must destroy and create the component each time config is updated\n    setTimeout(() => {\n      this._configSet.next(true)\n    }, 0)\n  }\n\n  private _getConfigOrDefault<K extends keyof TheSeamQuillEditorConfig>(prop: K): TheSeamQuillEditorConfig[K] {\n    if (this._customConfig && Object.prototype.hasOwnProperty.call(this._customConfig, prop)) {\n      return this._customConfig[prop]\n    }\n    return THESEAM_QUILL_EDITOR_CONFIG_DEFAULT[prop]\n  }\n\n  private _updateQuillConfig() {\n    if (this._configSet.value) {\n      this._buildQuillConfig()\n    }\n  }\n\n  // Not usable, bc config has to be set immediately, potentially before mentions populate.\n  // To set customDataAttributes, provide a custom config.\n  // private _getCustomDataAttributes(mentions: TheSeamQuillMentionMenuOption[]): string[] {\n  //   return mentions.reduce((acc, mention) => {\n  //     const keys = Object.keys(mention)\n\n  //     keys.forEach(key => {\n  //       if (acc.findIndex(a => a === key) === -1) {\n  //         acc.push(key)\n  //       }\n  //     })\n\n  //     return acc\n  //   }, <string[]>[] satisfies string[])\n  // }\n\n  get value(): string | undefined | null {\n    return this._value\n  }\n  set value(value: string | undefined | null) {\n    this._value = value\n\n    if (this.onChange) { this.onChange(value) }\n    if (this.onTouched) { this.onTouched() }\n  }\n  private _value: string | undefined | null\n\n  writeValue(value: any): void {\n    this._isWritingValue = true\n\n    this.formControl.setValue(value)\n    this.value = value\n\n    this._isWritingValue = false\n  }\n\n  registerOnChange(fn: any): void {\n    this.onChange = fn\n  }\n\n  registerOnTouched(fn: any): void {\n    this.onTouched = fn\n  }\n\n  setDisabledState(isDisabled: boolean): void {\n    this.readOnly = isDisabled\n  }\n\n  _onEditorCreated(event: any) {\n    this.quillEditorCreated.emit(event)\n  }\n\n  _onEditorChanged(event: EditorChangeContent | EditorChangeSelection) {\n    this.quillEditorChanged.emit(event)\n  }\n\n  _onContentChanged(event: ContentChange) {\n    this.quillContentChanged.emit(event)\n\n    this._updateMentionsFromDelta(event.content)\n  }\n\n  _onSelectionChanged(event: SelectionChange) {\n    this.quillSelectionChanged.emit(event)\n  }\n\n  _onFocus(event: Focus) {\n    this.quillFocus.emit(event)\n  }\n\n  _onBlur(event: Blur) {\n    this.quillBlur.emit(event)\n\n    if (this.onTouched) {\n      this.onTouched()\n    }\n  }\n\n  /**\n   * Hacky way to track mention inserts/deletes\n   */\n  private _updateMentionsFromDelta(content: Delta) {\n    if (notNullOrUndefined(content.ops)) {\n      const contentMentionIds = content.ops.map(o => o.insert?.mention?.id).filter(notNullOrUndefined)\n      const selectedMentions = [ ...this._selectedMentions.value ]\n      const mentionOptions = [ ...this._mentionItems.value || [] ]\n\n      const newMentions: TheSeamQuillMentionMenuOption[] = contentMentionIds.reduce((acc, mentionId) => {\n        const insertMention = mentionOptions.find(m => isMentionMenuOption(m) && m.id === mentionId)\n\n        if (notNullOrUndefined(insertMention)) {\n          acc.push(insertMention)\n        }\n        else {\n          console.warn('Mention addition failed! Selected mention option not found:', mentionId)\n        }\n\n        return acc\n      }, <TheSeamQuillMentionMenuOption[]>[])\n\n      let emitUpdate = false\n      if (selectedMentions.length !== newMentions.length) {\n        // if the length has changed, we know an update occurred\n        emitUpdate = true\n      }\n      else {\n        // otherwise, test ids for old and new items to see if we need to emit a change\n        const selectedMentionIds = this._selectedMentions.value.map(m => m.id)\n        const newMentionIds = newMentions.map(m => m.id)\n\n        if (\n          selectedMentionIds.findIndex(m => !newMentionIds.includes(m)) !== -1 ||\n          newMentionIds.findIndex(m => !selectedMentionIds.includes(m)) !== -1\n        ) {\n          emitUpdate = true\n        }\n      }\n\n      if (emitUpdate) {\n        this._selectedMentions.next(newMentions)\n      }\n    }\n  }\n}\n","<div class=\"editor\" *ngIf=\"configSet$ | async\" [class.initializing]=\"!(initialized$ | async)\">\n  <ng-container *ngIf=\"config$ | async as config\">\n    <quill-editor\n      #quillEditor\n      seamInput\n      [class.disabled]=\"readOnly\"\n      [attr.disabled]=\"readOnly\"\n      [formControl]=\"formControl\"\n      [placeholder]=\"placeholder\"\n      [readOnly]=\"readOnly\"\n      [required]=\"required\"\n      [minLength]=\"minLength\"\n      [maxLength]=\"maxLength\"\n      [format]=\"config.format\"\n      [theme]=\"config.theme\"\n      [modules]=\"config.modules\"\n      [debug]=\"config.debug\"\n      [formats]=\"config.formats\"\n      [customToolbarPosition]=\"config.customToolbarPosition\"\n      [sanitize]=\"config.sanitize\"\n      [styles]=\"config.styles\"\n      [strict]=\"config.strict\"\n      [scrollingContainer]=\"config.scrollingContainer\"\n      [bounds]=\"config.bounds\"\n      [customOptions]=\"config.customOptions\"\n      [customModules]=\"config.customModules\"\n      [trackChanges]=\"config.trackChanges\"\n      [preserveWhitespace]=\"config.preserveWhitespace\"\n      [classes]=\"config.classes\"\n      [trimOnValidation]=\"config.trimOnValidation\"\n      [linkPlaceholder]=\"config.linkPlaceholder\"\n      [compareValues]=\"config.compareValues\"\n      [filterNull]=\"config.filterNull\"\n      [debounceTime]=\"config.debounceTime\"\n      [class.can-resize]=\"resizable\"\n      (onEditorCreated)=\"_onEditorCreated($event)\"\n      (onEditorChanged)=\"_onEditorChanged($event)\"\n      (onContentChanged)=\"_onContentChanged($event)\"\n      (onSelectionChanged)=\"_onSelectionChanged($event)\"\n      (onFocus)=\"_onFocus($event)\"\n      (onBlur)=\"_onBlur($event)\"></quill-editor>\n    <ng-container *ngIf=\"displayCharacterCounter\">\n      <ng-container\n        [ngTemplateOutlet]=\"characterCounterTpl\"\n        [ngTemplateOutletContext]=\"{ $implicit: characterCount$ | async, minLength: minLength, maxLength: maxLength, characterCount: characterCount$ | async }\"></ng-container>\n    </ng-container>\n  </ng-container>\n</div>\n\n<ng-template #characterCounter let-implicit let-minLength=\"minLength\" let-maxLength=\"maxLength\" let-characterCount=\"characterCount\">\n  <div class=\"small text-black-50 text-right\">\n    <ng-container *ngIf=\"characterCount || characterCount === 0\">\n      Character Count:\n      <span [class.text-danger]=\"characterCount && ((maxLength && characterCount > maxLength) || (minLength && characterCount < minLength))\">\n        {{ characterCount | number }}\n      </span>\n      <ng-container *ngIf=\"maxLength\">\n        / {{ maxLength }}\n      </ng-container>\n      <span *ngIf=\"minLength\" class=\"font-italic\">\n        (minimum {{ minLength }} expected)\n      </span>\n    </ng-container>\n  </div>\n</ng-template>\n"]}
|