-
Notifications
You must be signed in to change notification settings - Fork 86
/
Copy pathutils.ts
468 lines (395 loc) · 14.4 KB
/
utils.ts
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
import { Change, diffChars } from 'diff';
import { StyleProp, TextStyle } from 'react-native';
// @ts-ignore the lib do not have TS declarations yet
import matchAll from 'string.prototype.matchall';
import { MentionData, MentionPartType, Part, PartType, Position, RegexMatchResult, Suggestion } from '../types';
const mentionRegEx = /(?<original>(?<trigger>.)\[(?<name>[^[]*)]\((?<id>[^(]*)\))/gi;
const defaultMentionTextStyle: StyleProp<TextStyle> = {fontWeight: 'bold', color: 'blue'};
const defaultPlainStringGenerator = ({trigger}: MentionPartType, {name}: MentionData) => `${trigger}${name}`;
type CharactersDiffChange = Omit<Change, 'count'> & { count: number };
const isMentionPartType = (partType: PartType): partType is MentionPartType => {
return (partType as MentionPartType).trigger != null;
};
const getPartIndexByCursor = (parts: Part[], cursor: number, isIncludeEnd?: boolean) => {
return parts.findIndex(one => cursor >= one.position.start && isIncludeEnd ? cursor <= one.position.end : cursor < one.position.end);
};
/**
* The method for getting parts between two cursor positions.
* ```
* | part1 | part2 | part3 |
* a b c|d e f g h i j h k|l m n o
* ```
* We will get 3 parts here:
* 1. Part included 'd'
* 2. Part included 'efghij'
* 3. Part included 'hk'
* Cursor will move to position after 'k'
*
* @param parts full part list
* @param cursor current cursor position
* @param count count of characters that didn't change
*/
const getPartsInterval = (parts: Part[], cursor: number, count: number): Part[] => {
const newCursor = cursor + count;
const currentPartIndex = getPartIndexByCursor(parts, cursor);
const currentPart = parts[currentPartIndex];
const newPartIndex = getPartIndexByCursor(parts, newCursor, true);
const newPart = parts[newPartIndex];
let partsInterval: Part[] = [];
if (!currentPart || !newPart) {
return partsInterval;
}
// Push whole first affected part or sub-part of the first affected part
if (currentPart.position.start === cursor && currentPart.position.end <= newCursor) {
partsInterval.push(currentPart);
} else {
partsInterval.push(generatePlainTextPart(currentPart.text.substr(cursor - currentPart.position.start, count)));
}
if (newPartIndex > currentPartIndex) {
// Concat fully included parts
partsInterval = partsInterval.concat(parts.slice(currentPartIndex + 1, newPartIndex));
// Push whole last affected part or sub-part of the last affected part
if (newPart.position.end === newCursor && newPart.position.start >= cursor) {
partsInterval.push(newPart);
} else {
partsInterval.push(generatePlainTextPart(newPart.text.substr(0, newCursor - newPart.position.start)));
}
}
return partsInterval;
};
/**
* Function for getting object with keyword for each mention part type
*
* If keyword is undefined then we don't tracking mention typing and shouldn't show suggestions.
* If keyword is not undefined (even empty string '') then we are tracking mention typing.
*
* Examples where @name is just plain text yet, not mention:
* '|abc @name dfg' - keyword is undefined
* 'abc @| dfg' - keyword is ''
* 'abc @name| dfg' - keyword is 'name'
* 'abc @na|me dfg' - keyword is 'na'
* 'abc @|name dfg' - keyword is against ''
* 'abc @name |dfg' - keyword is 'name '
* 'abc @name dfg|' - keyword is 'name dfg'
* 'abc @name dfg |' - keyword is undefined (we have more than one space)
* 'abc @name dfg he|' - keyword is undefined (we have more than one space)
*/
const getMentionPartSuggestionKeywords = (
parts: Part[],
plainText: string,
selection: Position,
partTypes: PartType[],
): { [trigger: string]: string | undefined } => {
const keywordByTrigger: { [trigger: string]: string | undefined } = {};
partTypes.filter(isMentionPartType).forEach((
{
trigger,
allowedSpacesCount = 1,
},
) => {
keywordByTrigger[trigger] = undefined;
// Check if we don't have selection range
if (selection.end != selection.start) {
return;
}
// Find the part with the cursor
const part = parts.find(one => selection.end > one.position.start && selection.end <= one.position.end);
// Check if the cursor is not in mention type part
if (part == null || part.data != null) {
return;
}
const triggerIndex = plainText.lastIndexOf(trigger, selection.end);
// Return undefined in case when:
if (
// - the trigger index is not event found
triggerIndex == -1
// - the trigger index is out of found part with selection cursor
|| triggerIndex < part.position.start
) {
return;
}
// Looking for break lines and spaces between the current cursor and trigger
let spacesCount = 0;
for (let cursor = selection.end - 1; cursor >= triggerIndex; cursor -= 1) {
// Mention cannot have new line
if (plainText[cursor] === '\n') {
return;
}
// Incrementing space counter if the next symbol is space
if (plainText[cursor] === ' ') {
spacesCount += 1;
// Check maximum allowed spaces in trigger word
if (spacesCount > allowedSpacesCount) {
return;
}
}
}
keywordByTrigger[trigger] = plainText.substring(
triggerIndex + 1,
selection.end,
);
});
return keywordByTrigger;
};
/**
* Generates new value when we changing text.
*
* @param parts full parts list
* @param originalText original plain text
* @param changedText changed plain text
*/
const generateValueFromPartsAndChangedText = (parts: Part[], originalText: string, changedText: string) => {
const changes = diffChars(originalText, changedText) as CharactersDiffChange[];
let newParts: Part[] = [];
let cursor = 0;
changes.forEach(change => {
switch (true) {
/**
* We should:
* - Move cursor forward on the changed text length
*/
case change.removed: {
cursor += change.count;
break;
}
/**
* We should:
* - Push new part to the parts with that new text
*/
case change.added: {
newParts.push(generatePlainTextPart(change.value));
break;
}
/**
* We should concat parts that didn't change.
* - In case when we have only one affected part we should push only that one sub-part
* - In case we have two affected parts we should push first
*/
default: {
if (change.count !== 0) {
newParts = newParts.concat(getPartsInterval(parts, cursor, change.count));
cursor += change.count;
}
break;
}
}
});
return getValueFromParts(newParts);
};
/**
* Method for adding suggestion to the parts and generating value. We should:
* - Find part with plain text where we were tracking mention typing using selection state
* - Split the part to next parts:
* -* Before new mention
* -* With new mention
* -* After mention with space at the beginning
* - Generate new parts array and convert it to value
*
* @param parts - full part list
* @param mentionType - actually the mention type
* @param plainText - current plain text
* @param selection - current selection
* @param suggestion - suggestion that should be added
*/
const generateValueWithAddedSuggestion = (
parts: Part[],
mentionType: MentionPartType,
plainText: string,
selection: Position,
suggestion: Suggestion,
): string | undefined => {
const currentPartIndex = parts.findIndex(one => selection.end >= one.position.start && selection.end <= one.position.end);
const currentPart = parts[currentPartIndex];
if (!currentPart) {
return;
}
const triggerPartIndex = currentPart.text.lastIndexOf(mentionType.trigger, selection.end - currentPart.position.start);
const newMentionPartPosition: Position = {
start: triggerPartIndex,
end: selection.end - currentPart.position.start,
};
const isInsertSpaceToNextPart = mentionType.isInsertSpaceAfterMention
// Cursor is at the very end of parts or text row
&& (plainText.length === selection.end || parts[currentPartIndex]?.text.startsWith('\n', newMentionPartPosition.end));
return getValueFromParts([
...parts.slice(0, currentPartIndex),
// Create part with string before mention
generatePlainTextPart(currentPart.text.substring(0, newMentionPartPosition.start)),
generateMentionPart(mentionType, {
original: getMentionValue(mentionType.trigger, suggestion),
trigger: mentionType.trigger,
...suggestion,
}),
// Create part with rest of string after mention and add a space if needed
generatePlainTextPart(`${isInsertSpaceToNextPart ? ' ' : ''}${currentPart.text.substring(newMentionPartPosition.end)}`),
...parts.slice(currentPartIndex + 1),
]);
};
/**
* Method for generating part for plain text
*
* @param text - plain text that will be added to the part
* @param positionOffset - position offset from the very beginning of text
*/
const generatePlainTextPart = (text: string, positionOffset = 0): Part => ({
text,
position: {
start: positionOffset,
end: positionOffset + text.length,
},
});
/**
* Method for generating part for mention
*
* @param mentionPartType
* @param mention - mention data
* @param positionOffset - position offset from the very beginning of text
*/
const generateMentionPart = (mentionPartType: MentionPartType, mention: MentionData, positionOffset = 0): Part => {
const text = mentionPartType.getPlainString
? mentionPartType.getPlainString(mention)
: defaultPlainStringGenerator(mentionPartType, mention);
return {
text,
position: {
start: positionOffset,
end: positionOffset + text.length,
},
partType: mentionPartType,
data: mention,
};
};
/**
* Generates part for matched regex result
*
* @param partType - current part type (pattern or mention)
* @param result - matched regex result
* @param positionOffset - position offset from the very beginning of text
*/
const generateRegexResultPart = (partType: PartType, result: RegexMatchResult, positionOffset = 0): Part => {
if (isMentionPartType(partType)) {
return generateMentionPart(partType, result.groups, positionOffset);
}
return {
text: result[0],
position: {
start: positionOffset,
end: positionOffset + result[0].length,
},
partType,
};
};
/**
* Method for generation mention value that accepts mention regex
*
* @param trigger
* @param suggestion
*/
const getMentionValue = (trigger: string, suggestion: Suggestion) => `${trigger}[${suggestion.name}](${suggestion.id})`;
/**
* Recursive function for deep parse MentionInput's value and get plainText with parts
*
* @param value - the MentionInput's value
* @param partTypes - All provided part types
* @param positionOffset - offset from the very beginning of plain text
*/
const parseValue = (
value: string = '',
partTypes: PartType[],
positionOffset = 0,
): { plainText: string; parts: Part[] } => {
let plainText = '';
let parts: Part[] = [];
// We don't have any part types so adding just plain text part
if (partTypes.length === 0) {
plainText += value;
parts.push(generatePlainTextPart(value, positionOffset));
} else {
const [partType, ...restPartTypes] = partTypes;
const regex = isMentionPartType(partType) ? mentionRegEx : partType.pattern;
const matches: RegexMatchResult[] = Array.from(matchAll(value ?? '', regex));
// In case when we didn't get any matches continue parsing value with rest part types
if (matches.length === 0) {
return parseValue(value, restPartTypes, positionOffset);
}
// In case when we have some text before matched part parsing the text with rest part types
if (matches[0].index != 0) {
const text = value.substr(0, matches[0].index);
const plainTextAndParts = parseValue(text, restPartTypes, positionOffset);
parts = parts.concat(plainTextAndParts.parts);
plainText += plainTextAndParts.plainText;
}
// Iterating over all found pattern matches
for (let i = 0; i < matches.length; i++) {
const result = matches[i];
// Matched pattern is a mention and the mention doesn't match current mention type
// We should parse the mention with rest part types
if (isMentionPartType(partType) && result.groups.trigger !== partType.trigger) {
const plainTextAndParts = parseValue(result['0'], restPartTypes, positionOffset + plainText.length);
parts = parts.concat(plainTextAndParts.parts);
plainText += plainTextAndParts.plainText;
} else {
const part = generateRegexResultPart(partType, result, positionOffset + plainText.length);
parts.push(part);
plainText += part.text;
}
// Check if the result is not at the end of whole value so we have a text after matched part
// We should parse the text with rest part types
if ((result.index + result[0].length) !== value.length) {
// Check if it is the last result
const isLastResult = i === matches.length - 1;
// So we should to add the last substring of value after matched mention
const text = value.slice(
result.index + result[0].length,
isLastResult ? undefined : matches[i + 1].index,
);
const plainTextAndParts = parseValue(text, restPartTypes, positionOffset + plainText.length);
parts = parts.concat(plainTextAndParts.parts);
plainText += plainTextAndParts.plainText;
}
}
}
// Exiting from generatePartsFromValue
return {
plainText,
parts,
};
};
/**
* Function for generation value from parts array
*
* @param parts
*/
const getValueFromParts = (parts: Part[]) => parts
.map(item => (item.data ? item.data.original : item.text))
.join('');
/**
* Replace all mention values in value to some specified format
*
* @param value - value that is generated by MentionInput component
* @param replacer - function that takes mention object as parameter and returns string
*/
const replaceMentionValues = (
value: string,
replacer: (mention: MentionData) => string,
) => value.replace(mentionRegEx, (fullMatch, original, trigger, name, id) => replacer({
original,
trigger,
name,
id,
}));
export {
mentionRegEx,
defaultMentionTextStyle,
isMentionPartType,
getMentionPartSuggestionKeywords,
generateValueFromPartsAndChangedText,
generateValueWithAddedSuggestion,
generatePlainTextPart,
generateMentionPart,
generateRegexResultPart,
getMentionValue,
parseValue,
getValueFromParts,
replaceMentionValues,
};