@churchapps/apphelper 0.0.7 → 0.0.9
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/.eslintignore +3 -3
- package/.eslintrc.json +22 -22
- package/LICENSE +21 -21
- package/README.md +16 -16
- package/dist/components/ExportLink.d.ts.map +1 -1
- package/dist/components/ExportLink.js +9 -7
- package/dist/components/ExportLink.js.map +1 -1
- package/dist/components/FormSubmissionEdit.d.ts +0 -1
- package/dist/components/FormSubmissionEdit.d.ts.map +1 -1
- package/dist/components/FormSubmissionEdit.js +1 -2
- package/dist/components/FormSubmissionEdit.js.map +1 -1
- package/dist/components/markdownEditor/editor.css +787 -787
- package/dist/components/markdownEditor/images/icons/arrow-clockwise.svg +3 -3
- package/dist/components/markdownEditor/images/icons/arrow-counterclockwise.svg +3 -3
- package/dist/components/markdownEditor/images/icons/chat-square-quote.svg +3 -3
- package/dist/components/markdownEditor/images/icons/chevron-down.svg +2 -2
- package/dist/components/markdownEditor/images/icons/code.svg +2 -2
- package/dist/components/markdownEditor/images/icons/journal-code.svg +4 -4
- package/dist/components/markdownEditor/images/icons/journal-text.svg +4 -4
- package/dist/components/markdownEditor/images/icons/justify.svg +2 -2
- package/dist/components/markdownEditor/images/icons/link.svg +3 -3
- package/dist/components/markdownEditor/images/icons/list-ol.svg +3 -3
- package/dist/components/markdownEditor/images/icons/list-ul.svg +2 -2
- package/dist/components/markdownEditor/images/icons/pencil-fill.svg +2 -2
- package/dist/components/markdownEditor/images/icons/text-center.svg +2 -2
- package/dist/components/markdownEditor/images/icons/text-left.svg +2 -2
- package/dist/components/markdownEditor/images/icons/text-paragraph.svg +2 -2
- package/dist/components/markdownEditor/images/icons/text-right.svg +2 -2
- package/dist/components/markdownEditor/images/icons/type-bold.svg +2 -2
- package/dist/components/markdownEditor/images/icons/type-h1.svg +2 -2
- package/dist/components/markdownEditor/images/icons/type-h2.svg +2 -2
- package/dist/components/markdownEditor/images/icons/type-h3.svg +2 -2
- package/dist/components/markdownEditor/images/icons/type-h4.svg +12 -12
- package/dist/components/markdownEditor/images/icons/type-italic.svg +2 -2
- package/dist/components/markdownEditor/images/icons/type-strikethrough.svg +2 -2
- package/dist/components/markdownEditor/images/icons/type-underline.svg +2 -2
- package/dist/donationComponents/components/NonAuthDonationInner.d.ts.map +1 -1
- package/dist/donationComponents/components/NonAuthDonationInner.js +38 -5
- package/dist/donationComponents/components/NonAuthDonationInner.js.map +1 -1
- package/package.json +84 -84
- package/src/components/CreatePerson.tsx +80 -80
- package/src/components/DisplayBox.tsx +68 -68
- package/src/components/ErrorMessages.tsx +26 -26
- package/src/components/ExportLink.tsx +66 -61
- package/src/components/FloatingSupport.tsx +16 -16
- package/src/components/FormSubmissionEdit.tsx +120 -122
- package/src/components/HelpIcon.tsx +10 -10
- package/src/components/ImageEditor.tsx +126 -126
- package/src/components/InputBox.tsx +73 -73
- package/src/components/Loading.tsx +29 -29
- package/src/components/PersonAdd.tsx +75 -75
- package/src/components/QuestionEdit.tsx +62 -62
- package/src/components/SmallButton.tsx +39 -39
- package/src/components/SupportModal.tsx +26 -26
- package/src/components/TabPanel.tsx +34 -34
- package/src/components/gallery/GalleryModal.tsx +102 -102
- package/src/components/gallery/StockPhotos.tsx +74 -74
- package/src/components/gallery/index.ts +1 -1
- package/src/components/index.tsx +23 -23
- package/src/components/markdownEditor/Editor.tsx +132 -132
- package/src/components/markdownEditor/MarkdownEditor.tsx +16 -16
- package/src/components/markdownEditor/MarkdownModal.tsx +46 -46
- package/src/components/markdownEditor/MarkdownPreview.tsx +14 -14
- package/src/components/markdownEditor/editor.css +787 -787
- package/src/components/markdownEditor/images/icons/arrow-clockwise.svg +3 -3
- package/src/components/markdownEditor/images/icons/arrow-counterclockwise.svg +3 -3
- package/src/components/markdownEditor/images/icons/chat-square-quote.svg +3 -3
- package/src/components/markdownEditor/images/icons/chevron-down.svg +2 -2
- package/src/components/markdownEditor/images/icons/code.svg +2 -2
- package/src/components/markdownEditor/images/icons/journal-code.svg +4 -4
- package/src/components/markdownEditor/images/icons/journal-text.svg +4 -4
- package/src/components/markdownEditor/images/icons/justify.svg +2 -2
- package/src/components/markdownEditor/images/icons/link.svg +3 -3
- package/src/components/markdownEditor/images/icons/list-ol.svg +3 -3
- package/src/components/markdownEditor/images/icons/list-ul.svg +2 -2
- package/src/components/markdownEditor/images/icons/pencil-fill.svg +2 -2
- package/src/components/markdownEditor/images/icons/text-center.svg +2 -2
- package/src/components/markdownEditor/images/icons/text-left.svg +2 -2
- package/src/components/markdownEditor/images/icons/text-paragraph.svg +2 -2
- package/src/components/markdownEditor/images/icons/text-right.svg +2 -2
- package/src/components/markdownEditor/images/icons/type-bold.svg +2 -2
- package/src/components/markdownEditor/images/icons/type-h1.svg +2 -2
- package/src/components/markdownEditor/images/icons/type-h2.svg +2 -2
- package/src/components/markdownEditor/images/icons/type-h3.svg +2 -2
- package/src/components/markdownEditor/images/icons/type-h4.svg +12 -12
- package/src/components/markdownEditor/images/icons/type-italic.svg +2 -2
- package/src/components/markdownEditor/images/icons/type-strikethrough.svg +2 -2
- package/src/components/markdownEditor/images/icons/type-underline.svg +2 -2
- package/src/components/markdownEditor/index.ts +2 -2
- package/src/components/markdownEditor/plugins/AutoLinkPlugin.tsx +35 -35
- package/src/components/markdownEditor/plugins/ControlledEditorPlugin.tsx +24 -24
- package/src/components/markdownEditor/plugins/ListMaxIndentLevelPlugin.tsx +68 -68
- package/src/components/markdownEditor/plugins/MarkdownTransformers.ts +106 -106
- package/src/components/markdownEditor/plugins/ReadOnlyPlugin.tsx +15 -15
- package/src/components/markdownEditor/plugins/ToolbarPlugin.tsx +401 -401
- package/src/components/markdownEditor/plugins/customLink/CustomLinkNode.tsx +224 -224
- package/src/components/markdownEditor/plugins/customLink/CustomLinkNodePlugin.tsx +32 -32
- package/src/components/markdownEditor/plugins/customLink/CustomLinkNodeTransformer.tsx +102 -102
- package/src/components/markdownEditor/plugins/customLink/FloatingLinkEditor.tsx +243 -243
- package/src/components/markdownEditor/plugins/customLink/FloatingLinkEditor.types.ts +11 -11
- package/src/components/markdownEditor/plugins/emoji/EmojiNode.tsx +95 -95
- package/src/components/markdownEditor/plugins/emoji/EmojiNodeTransform.ts +41 -41
- package/src/components/markdownEditor/plugins/emoji/EmojiPickerPlugin.tsx +152 -152
- package/src/components/markdownEditor/plugins/emoji/EmojisPlugin.tsx +65 -65
- package/src/components/markdownEditor/plugins/index.ts +6 -6
- package/src/components/markdownEditor/theme.ts +65 -65
- package/src/components/material/AppList.tsx +20 -20
- package/src/components/material/ChurchList.tsx +22 -22
- package/src/components/material/NavItem.tsx +41 -41
- package/src/components/material/NewPrivateMessage.tsx +103 -103
- package/src/components/material/PrivateMessageDetails.tsx +23 -23
- package/src/components/material/PrivateMessages.tsx +87 -87
- package/src/components/material/SiteWrapper.tsx +140 -140
- package/src/components/material/UserMenu.tsx +141 -141
- package/src/components/material/iconPicker/IconNamesList.ts +2240 -2240
- package/src/components/material/iconPicker/IconPicker.tsx +153 -153
- package/src/components/material/index.tsx +6 -6
- package/src/components/notes/AddNote.tsx +90 -90
- package/src/components/notes/Conversation.tsx +82 -82
- package/src/components/notes/Conversations.tsx +58 -58
- package/src/components/notes/NewConversation.tsx +78 -78
- package/src/components/notes/Note.tsx +44 -44
- package/src/components/notes/Notes.tsx +52 -52
- package/src/components/notes/index.ts +5 -5
- package/src/components/reporting/ChartReport.tsx +98 -98
- package/src/components/reporting/ReportFilter.tsx +54 -54
- package/src/components/reporting/ReportFilterField.tsx +160 -160
- package/src/components/reporting/ReportOutput.tsx +79 -79
- package/src/components/reporting/ReportWithFilter.tsx +70 -70
- package/src/components/reporting/TableReport.tsx +57 -57
- package/src/components/reporting/TreeReport.tsx +111 -111
- package/src/components/reporting/index.ts +4 -4
- package/src/donationComponents/DonationPage.tsx +136 -136
- package/src/donationComponents/components/BankForm.tsx +159 -159
- package/src/donationComponents/components/CardForm.tsx +104 -104
- package/src/donationComponents/components/DonationForm.tsx +235 -235
- package/src/donationComponents/components/FundDonation.tsx +49 -49
- package/src/donationComponents/components/FundDonations.tsx +39 -39
- package/src/donationComponents/components/NonAuthDonation.tsx +31 -31
- package/src/donationComponents/components/NonAuthDonationInner.tsx +259 -221
- package/src/donationComponents/components/PaymentMethods.tsx +135 -135
- package/src/donationComponents/components/RecurringDonations.tsx +121 -121
- package/src/donationComponents/components/RecurringDonationsEdit.tsx +93 -93
- package/src/donationComponents/components/index.tsx +9 -9
- package/src/donationComponents/index.ts +3 -3
- package/src/donationComponents/modals/DonationPreviewModal.tsx +66 -66
- package/src/helpers/AnalyticsHelper.ts +33 -33
- package/src/helpers/ApiHelper.ts +125 -125
- package/src/helpers/AppearanceHelper.ts +69 -69
- package/src/helpers/ArrayHelper.ts +81 -81
- package/src/helpers/CommonEnvironmentHelper.ts +80 -80
- package/src/helpers/CurrencyHelper.ts +10 -10
- package/src/helpers/DateHelper.ts +108 -108
- package/src/helpers/DonationHelper.ts +26 -26
- package/src/helpers/ErrorHelper.ts +36 -36
- package/src/helpers/EventHelper.ts +52 -52
- package/src/helpers/FileHelper.ts +31 -31
- package/src/helpers/PersonHelper.ts +60 -60
- package/src/helpers/SocketHelper.ts +76 -76
- package/src/helpers/Themes.ts +14 -14
- package/src/helpers/UniqueIdHelper.ts +36 -36
- package/src/helpers/UserHelper.ts +59 -59
- package/src/helpers/createEmotionCache.ts +17 -17
- package/src/helpers/index.ts +18 -18
- package/src/hooks/index.ts +1 -1
- package/src/hooks/useMountedState.ts +16 -16
- package/src/index.ts +6 -6
- package/src/interfaces/Access.ts +24 -24
- package/src/interfaces/Attendance.ts +8 -8
- package/src/interfaces/Content.ts +10 -10
- package/src/interfaces/Doing.ts +24 -24
- package/src/interfaces/Donation.ts +45 -45
- package/src/interfaces/Error.ts +17 -17
- package/src/interfaces/Membership.ts +51 -51
- package/src/interfaces/Messaging.ts +20 -20
- package/src/interfaces/Permissions.ts +68 -68
- package/src/interfaces/Reporting.ts +7 -7
- package/src/interfaces/UserContextInterface.ts +13 -13
- package/src/interfaces/index.ts +13 -13
- package/src/pageComponents/LoginPage.tsx +244 -244
- package/src/pageComponents/LogoutPage.tsx +28 -28
- package/src/pageComponents/components/Forgot.tsx +79 -79
- package/src/pageComponents/components/Login.tsx +54 -54
- package/src/pageComponents/components/LoginSetPassword.tsx +63 -63
- package/src/pageComponents/components/Register.tsx +107 -107
- package/src/pageComponents/components/SelectChurchModal.tsx +41 -41
- package/src/pageComponents/components/SelectChurchRegister.tsx +88 -88
- package/src/pageComponents/components/SelectChurchSearch.tsx +69 -69
- package/src/pageComponents/components/SelectableChurch.tsx +38 -38
- package/src/pageComponents/index.ts +3 -3
- package/tsconfig.json +34 -34
- package/tslint.json +14 -14
|
@@ -1,95 +1,95 @@
|
|
|
1
|
-
/**
|
|
2
|
-
* Copyright (c) Meta Platforms, Inc. and affiliates.
|
|
3
|
-
*
|
|
4
|
-
* This source code is licensed under the MIT license found in the
|
|
5
|
-
* LICENSE file in the root directory of this source tree.
|
|
6
|
-
*
|
|
7
|
-
*/
|
|
8
|
-
|
|
9
|
-
import type {
|
|
10
|
-
EditorConfig,
|
|
11
|
-
LexicalNode,
|
|
12
|
-
NodeKey,
|
|
13
|
-
SerializedTextNode,
|
|
14
|
-
Spread,
|
|
15
|
-
} from 'lexical';
|
|
16
|
-
|
|
17
|
-
import {$applyNodeReplacement, TextNode} from 'lexical';
|
|
18
|
-
|
|
19
|
-
export type SerializedEmojiNode = Spread<
|
|
20
|
-
{
|
|
21
|
-
type: string;
|
|
22
|
-
},
|
|
23
|
-
SerializedTextNode
|
|
24
|
-
>;
|
|
25
|
-
|
|
26
|
-
export class EmojiNode extends TextNode {
|
|
27
|
-
exportJSON() : SerializedEmojiNode {
|
|
28
|
-
const json = {
|
|
29
|
-
...super.exportJSON(),
|
|
30
|
-
type: 'emoji'
|
|
31
|
-
};
|
|
32
|
-
return json;
|
|
33
|
-
}
|
|
34
|
-
|
|
35
|
-
static getType(): string {
|
|
36
|
-
return 'emoji';
|
|
37
|
-
}
|
|
38
|
-
|
|
39
|
-
static clone(node: EmojiNode): EmojiNode {
|
|
40
|
-
return new EmojiNode(node.__text, node.__key);
|
|
41
|
-
}
|
|
42
|
-
|
|
43
|
-
static importJSON(serializedNode: SerializedEmojiNode): EmojiNode {
|
|
44
|
-
const node = new EmojiNode(serializedNode.text);
|
|
45
|
-
return node;
|
|
46
|
-
}
|
|
47
|
-
|
|
48
|
-
/*
|
|
49
|
-
constructor(text: string, key?: NodeKey) {
|
|
50
|
-
super(text, key);
|
|
51
|
-
}
|
|
52
|
-
*/
|
|
53
|
-
|
|
54
|
-
createDOM(config: EditorConfig): HTMLElement {
|
|
55
|
-
const dom = document.createElement('span');
|
|
56
|
-
const inner = super.createDOM(config);
|
|
57
|
-
dom.className = 'material-symbols-outlined';
|
|
58
|
-
dom.appendChild(inner);
|
|
59
|
-
return dom;
|
|
60
|
-
}
|
|
61
|
-
|
|
62
|
-
updateDOM(
|
|
63
|
-
prevNode: TextNode,
|
|
64
|
-
dom: HTMLElement,
|
|
65
|
-
config: EditorConfig,
|
|
66
|
-
): boolean {
|
|
67
|
-
const inner = dom.firstChild;
|
|
68
|
-
if (inner === null) {
|
|
69
|
-
return true;
|
|
70
|
-
}
|
|
71
|
-
super.updateDOM(prevNode, inner as HTMLElement, config);
|
|
72
|
-
return false;
|
|
73
|
-
}
|
|
74
|
-
}
|
|
75
|
-
|
|
76
|
-
export function $isEmojiNode(
|
|
77
|
-
node: LexicalNode | null | undefined,
|
|
78
|
-
): node is EmojiNode {
|
|
79
|
-
return node instanceof EmojiNode;
|
|
80
|
-
}
|
|
81
|
-
|
|
82
|
-
export function $createEmojiNode(
|
|
83
|
-
emojiText: string
|
|
84
|
-
): EmojiNode {
|
|
85
|
-
const node = new EmojiNode(emojiText).setMode('token');
|
|
86
|
-
return $applyNodeReplacement(node);
|
|
87
|
-
}
|
|
88
|
-
|
|
89
|
-
|
|
90
|
-
export function $toggleEmojiNode(
|
|
91
|
-
emojiText: string,
|
|
92
|
-
): EmojiNode {
|
|
93
|
-
const node = new EmojiNode(emojiText).setMode('token');
|
|
94
|
-
return node;
|
|
95
|
-
}
|
|
1
|
+
/**
|
|
2
|
+
* Copyright (c) Meta Platforms, Inc. and affiliates.
|
|
3
|
+
*
|
|
4
|
+
* This source code is licensed under the MIT license found in the
|
|
5
|
+
* LICENSE file in the root directory of this source tree.
|
|
6
|
+
*
|
|
7
|
+
*/
|
|
8
|
+
|
|
9
|
+
import type {
|
|
10
|
+
EditorConfig,
|
|
11
|
+
LexicalNode,
|
|
12
|
+
NodeKey,
|
|
13
|
+
SerializedTextNode,
|
|
14
|
+
Spread,
|
|
15
|
+
} from 'lexical';
|
|
16
|
+
|
|
17
|
+
import {$applyNodeReplacement, TextNode} from 'lexical';
|
|
18
|
+
|
|
19
|
+
export type SerializedEmojiNode = Spread<
|
|
20
|
+
{
|
|
21
|
+
type: string;
|
|
22
|
+
},
|
|
23
|
+
SerializedTextNode
|
|
24
|
+
>;
|
|
25
|
+
|
|
26
|
+
export class EmojiNode extends TextNode {
|
|
27
|
+
exportJSON() : SerializedEmojiNode {
|
|
28
|
+
const json = {
|
|
29
|
+
...super.exportJSON(),
|
|
30
|
+
type: 'emoji'
|
|
31
|
+
};
|
|
32
|
+
return json;
|
|
33
|
+
}
|
|
34
|
+
|
|
35
|
+
static getType(): string {
|
|
36
|
+
return 'emoji';
|
|
37
|
+
}
|
|
38
|
+
|
|
39
|
+
static clone(node: EmojiNode): EmojiNode {
|
|
40
|
+
return new EmojiNode(node.__text, node.__key);
|
|
41
|
+
}
|
|
42
|
+
|
|
43
|
+
static importJSON(serializedNode: SerializedEmojiNode): EmojiNode {
|
|
44
|
+
const node = new EmojiNode(serializedNode.text);
|
|
45
|
+
return node;
|
|
46
|
+
}
|
|
47
|
+
|
|
48
|
+
/*
|
|
49
|
+
constructor(text: string, key?: NodeKey) {
|
|
50
|
+
super(text, key);
|
|
51
|
+
}
|
|
52
|
+
*/
|
|
53
|
+
|
|
54
|
+
createDOM(config: EditorConfig): HTMLElement {
|
|
55
|
+
const dom = document.createElement('span');
|
|
56
|
+
const inner = super.createDOM(config);
|
|
57
|
+
dom.className = 'material-symbols-outlined';
|
|
58
|
+
dom.appendChild(inner);
|
|
59
|
+
return dom;
|
|
60
|
+
}
|
|
61
|
+
|
|
62
|
+
updateDOM(
|
|
63
|
+
prevNode: TextNode,
|
|
64
|
+
dom: HTMLElement,
|
|
65
|
+
config: EditorConfig,
|
|
66
|
+
): boolean {
|
|
67
|
+
const inner = dom.firstChild;
|
|
68
|
+
if (inner === null) {
|
|
69
|
+
return true;
|
|
70
|
+
}
|
|
71
|
+
super.updateDOM(prevNode, inner as HTMLElement, config);
|
|
72
|
+
return false;
|
|
73
|
+
}
|
|
74
|
+
}
|
|
75
|
+
|
|
76
|
+
export function $isEmojiNode(
|
|
77
|
+
node: LexicalNode | null | undefined,
|
|
78
|
+
): node is EmojiNode {
|
|
79
|
+
return node instanceof EmojiNode;
|
|
80
|
+
}
|
|
81
|
+
|
|
82
|
+
export function $createEmojiNode(
|
|
83
|
+
emojiText: string
|
|
84
|
+
): EmojiNode {
|
|
85
|
+
const node = new EmojiNode(emojiText).setMode('token');
|
|
86
|
+
return $applyNodeReplacement(node);
|
|
87
|
+
}
|
|
88
|
+
|
|
89
|
+
|
|
90
|
+
export function $toggleEmojiNode(
|
|
91
|
+
emojiText: string,
|
|
92
|
+
): EmojiNode {
|
|
93
|
+
const node = new EmojiNode(emojiText).setMode('token');
|
|
94
|
+
return node;
|
|
95
|
+
}
|
|
@@ -1,41 +1,41 @@
|
|
|
1
|
-
import { EmojiNode, $isEmojiNode, $createEmojiNode, $toggleEmojiNode } from './EmojiNode';
|
|
2
|
-
import materialIcons from '../../../material/iconPicker/IconNamesList';
|
|
3
|
-
|
|
4
|
-
import type { TextMatchTransformer } from "@lexical/markdown";
|
|
5
|
-
import { TextNode, $isTextNode } from "lexical";
|
|
6
|
-
|
|
7
|
-
const EMOJI_NODE_MARKDOWN_REGEX_QUERY = /:[A-Za-z_]+:$/;
|
|
8
|
-
|
|
9
|
-
export const EMOJI_NODE_MARKDOWN_REGEX = new RegExp(EMOJI_NODE_MARKDOWN_REGEX_QUERY);
|
|
10
|
-
|
|
11
|
-
const replaceEmojiMarkdownWithNode = (textNode: TextNode) => {
|
|
12
|
-
const emojiNode = $createEmojiNode(
|
|
13
|
-
textNode.__text.replaceAll(':', '')
|
|
14
|
-
);
|
|
15
|
-
|
|
16
|
-
textNode.replace(emojiNode);
|
|
17
|
-
|
|
18
|
-
return emojiNode;
|
|
19
|
-
};
|
|
20
|
-
|
|
21
|
-
export const EMOJI_NODE_MARKDOWN_TRANSFORM: TextMatchTransformer = {
|
|
22
|
-
dependencies: [EmojiNode],
|
|
23
|
-
export: (node, exportChildren, exportFormat) => {
|
|
24
|
-
if (!($isEmojiNode(node) || $isTextNode(node))) {
|
|
25
|
-
return null;
|
|
26
|
-
}
|
|
27
|
-
|
|
28
|
-
if (!materialIcons.includes(node.getTextContent().replaceAll(':', ''))) {
|
|
29
|
-
return null;
|
|
30
|
-
}
|
|
31
|
-
|
|
32
|
-
const linkContent = ':' + node.__text.replaceAll(':', '') + ':';
|
|
33
|
-
|
|
34
|
-
return linkContent;
|
|
35
|
-
},
|
|
36
|
-
importRegExp: EMOJI_NODE_MARKDOWN_REGEX,
|
|
37
|
-
regExp: EMOJI_NODE_MARKDOWN_REGEX,
|
|
38
|
-
replace: replaceEmojiMarkdownWithNode,
|
|
39
|
-
trigger: ":",
|
|
40
|
-
type: "text-match"
|
|
41
|
-
};
|
|
1
|
+
import { EmojiNode, $isEmojiNode, $createEmojiNode, $toggleEmojiNode } from './EmojiNode';
|
|
2
|
+
import materialIcons from '../../../material/iconPicker/IconNamesList';
|
|
3
|
+
|
|
4
|
+
import type { TextMatchTransformer } from "@lexical/markdown";
|
|
5
|
+
import { TextNode, $isTextNode } from "lexical";
|
|
6
|
+
|
|
7
|
+
const EMOJI_NODE_MARKDOWN_REGEX_QUERY = /:[A-Za-z_]+:$/;
|
|
8
|
+
|
|
9
|
+
export const EMOJI_NODE_MARKDOWN_REGEX = new RegExp(EMOJI_NODE_MARKDOWN_REGEX_QUERY);
|
|
10
|
+
|
|
11
|
+
const replaceEmojiMarkdownWithNode = (textNode: TextNode) => {
|
|
12
|
+
const emojiNode = $createEmojiNode(
|
|
13
|
+
textNode.__text.replaceAll(':', '')
|
|
14
|
+
);
|
|
15
|
+
|
|
16
|
+
textNode.replace(emojiNode);
|
|
17
|
+
|
|
18
|
+
return emojiNode;
|
|
19
|
+
};
|
|
20
|
+
|
|
21
|
+
export const EMOJI_NODE_MARKDOWN_TRANSFORM: TextMatchTransformer = {
|
|
22
|
+
dependencies: [EmojiNode],
|
|
23
|
+
export: (node, exportChildren, exportFormat) => {
|
|
24
|
+
if (!($isEmojiNode(node) || $isTextNode(node))) {
|
|
25
|
+
return null;
|
|
26
|
+
}
|
|
27
|
+
|
|
28
|
+
if (!materialIcons.includes(node.getTextContent().replaceAll(':', ''))) {
|
|
29
|
+
return null;
|
|
30
|
+
}
|
|
31
|
+
|
|
32
|
+
const linkContent = ':' + node.__text.replaceAll(':', '') + ':';
|
|
33
|
+
|
|
34
|
+
return linkContent;
|
|
35
|
+
},
|
|
36
|
+
importRegExp: EMOJI_NODE_MARKDOWN_REGEX,
|
|
37
|
+
regExp: EMOJI_NODE_MARKDOWN_REGEX,
|
|
38
|
+
replace: replaceEmojiMarkdownWithNode,
|
|
39
|
+
trigger: ":",
|
|
40
|
+
type: "text-match"
|
|
41
|
+
};
|
|
@@ -1,152 +1,152 @@
|
|
|
1
|
-
/**
|
|
2
|
-
* Copyright (c) Meta Platforms, Inc. and affiliates.
|
|
3
|
-
*
|
|
4
|
-
* This source code is licensed under the MIT license found in the
|
|
5
|
-
* LICENSE file in the root directory of this source tree.
|
|
6
|
-
*
|
|
7
|
-
*/
|
|
8
|
-
|
|
9
|
-
import {useLexicalComposerContext} from '@lexical/react/LexicalComposerContext';
|
|
10
|
-
import {
|
|
11
|
-
LexicalTypeaheadMenuPlugin,
|
|
12
|
-
useBasicTypeaheadTriggerMatch,
|
|
13
|
-
} from '@lexical/react/LexicalTypeaheadMenuPlugin';
|
|
14
|
-
import {
|
|
15
|
-
$createTextNode,
|
|
16
|
-
$getSelection,
|
|
17
|
-
$isRangeSelection,
|
|
18
|
-
TextNode,
|
|
19
|
-
} from 'lexical';
|
|
20
|
-
import { $createEmojiNode } from './EmojiNode';
|
|
21
|
-
import materialIcons from '../../../material/iconPicker/IconNamesList';
|
|
22
|
-
import * as React from 'react';
|
|
23
|
-
import {useCallback, useEffect, useMemo, useState} from 'react';
|
|
24
|
-
import * as ReactDOM from 'react-dom';
|
|
25
|
-
|
|
26
|
-
|
|
27
|
-
function EmojiMenuItem({
|
|
28
|
-
index,
|
|
29
|
-
isSelected,
|
|
30
|
-
onClick,
|
|
31
|
-
onMouseEnter,
|
|
32
|
-
option,
|
|
33
|
-
}: {
|
|
34
|
-
index: number;
|
|
35
|
-
isSelected: boolean;
|
|
36
|
-
onClick: () => void;
|
|
37
|
-
onMouseEnter: () => void;
|
|
38
|
-
option: string;
|
|
39
|
-
}) {
|
|
40
|
-
let className = 'item';
|
|
41
|
-
if (isSelected) {
|
|
42
|
-
className += ' selected';
|
|
43
|
-
}
|
|
44
|
-
return (
|
|
45
|
-
<li
|
|
46
|
-
key={option}
|
|
47
|
-
tabIndex={-1}
|
|
48
|
-
className={className}
|
|
49
|
-
role="option"
|
|
50
|
-
aria-selected={isSelected}
|
|
51
|
-
id={'typeahead-item-' + index}
|
|
52
|
-
onMouseEnter={onMouseEnter}
|
|
53
|
-
onClick={onClick}>
|
|
54
|
-
<span className="text">
|
|
55
|
-
<span className="material-symbols-outlined">{option}</span> {option}
|
|
56
|
-
</span>
|
|
57
|
-
</li>
|
|
58
|
-
);
|
|
59
|
-
}
|
|
60
|
-
|
|
61
|
-
|
|
62
|
-
const MAX_EMOJI_SUGGESTION_COUNT = 10;
|
|
63
|
-
|
|
64
|
-
export default function EmojiPickerPlugin() {
|
|
65
|
-
const [editor] = useLexicalComposerContext();
|
|
66
|
-
const [queryString, setQueryString] = useState<string | null>(null);
|
|
67
|
-
|
|
68
|
-
|
|
69
|
-
|
|
70
|
-
const checkForTriggerMatch = useBasicTypeaheadTriggerMatch(':', {
|
|
71
|
-
minLength: 0,
|
|
72
|
-
});
|
|
73
|
-
|
|
74
|
-
const options: Array<any> = useMemo(() => {
|
|
75
|
-
let result = materialIcons.filter((option: string) =>
|
|
76
|
-
(queryString != null) ? new RegExp(queryString, 'gi').exec(option) : materialIcons
|
|
77
|
-
);
|
|
78
|
-
return result.slice(0, MAX_EMOJI_SUGGESTION_COUNT);
|
|
79
|
-
}, [queryString]);
|
|
80
|
-
|
|
81
|
-
const onSelectOption = useCallback(
|
|
82
|
-
(
|
|
83
|
-
selectedOption: any,
|
|
84
|
-
nodeToRemove: TextNode | null,
|
|
85
|
-
closeMenu: () => void,
|
|
86
|
-
) => {
|
|
87
|
-
editor.update(() => {
|
|
88
|
-
const selection = $getSelection();
|
|
89
|
-
|
|
90
|
-
if (!$isRangeSelection(selection) || selectedOption == null) {
|
|
91
|
-
return;
|
|
92
|
-
}
|
|
93
|
-
|
|
94
|
-
if (nodeToRemove) {
|
|
95
|
-
nodeToRemove.remove();
|
|
96
|
-
}
|
|
97
|
-
|
|
98
|
-
selection.insertNodes([$createEmojiNode(selectedOption)]);
|
|
99
|
-
|
|
100
|
-
closeMenu();
|
|
101
|
-
});
|
|
102
|
-
},
|
|
103
|
-
[editor],
|
|
104
|
-
);
|
|
105
|
-
|
|
106
|
-
if (typeof document === "undefined") {
|
|
107
|
-
return null;
|
|
108
|
-
}
|
|
109
|
-
|
|
110
|
-
return (
|
|
111
|
-
<LexicalTypeaheadMenuPlugin
|
|
112
|
-
onQueryChange={setQueryString}
|
|
113
|
-
onSelectOption={onSelectOption}
|
|
114
|
-
triggerFn={checkForTriggerMatch}
|
|
115
|
-
options={options}
|
|
116
|
-
menuRenderFn={(
|
|
117
|
-
anchorElementRef,
|
|
118
|
-
{selectedIndex, selectOptionAndCleanUp, setHighlightedIndex},
|
|
119
|
-
) => {
|
|
120
|
-
if (anchorElementRef.current == null || options.length === 0) {
|
|
121
|
-
return null;
|
|
122
|
-
}
|
|
123
|
-
|
|
124
|
-
return anchorElementRef.current && options.length
|
|
125
|
-
? ReactDOM.createPortal(
|
|
126
|
-
<div className="typeahead-popover emoji-menu">
|
|
127
|
-
<ul>
|
|
128
|
-
{options.map((option: string, index) => (
|
|
129
|
-
<div key={option}>
|
|
130
|
-
<EmojiMenuItem
|
|
131
|
-
index={index}
|
|
132
|
-
isSelected={selectedIndex === index}
|
|
133
|
-
onClick={() => {
|
|
134
|
-
setHighlightedIndex(index);
|
|
135
|
-
selectOptionAndCleanUp(option);
|
|
136
|
-
}}
|
|
137
|
-
onMouseEnter={() => {
|
|
138
|
-
setHighlightedIndex(index);
|
|
139
|
-
}}
|
|
140
|
-
option={option}
|
|
141
|
-
/>
|
|
142
|
-
</div>
|
|
143
|
-
))}
|
|
144
|
-
</ul>
|
|
145
|
-
</div>,
|
|
146
|
-
anchorElementRef.current,
|
|
147
|
-
)
|
|
148
|
-
: null;
|
|
149
|
-
}}
|
|
150
|
-
/>
|
|
151
|
-
);
|
|
152
|
-
}
|
|
1
|
+
/**
|
|
2
|
+
* Copyright (c) Meta Platforms, Inc. and affiliates.
|
|
3
|
+
*
|
|
4
|
+
* This source code is licensed under the MIT license found in the
|
|
5
|
+
* LICENSE file in the root directory of this source tree.
|
|
6
|
+
*
|
|
7
|
+
*/
|
|
8
|
+
|
|
9
|
+
import {useLexicalComposerContext} from '@lexical/react/LexicalComposerContext';
|
|
10
|
+
import {
|
|
11
|
+
LexicalTypeaheadMenuPlugin,
|
|
12
|
+
useBasicTypeaheadTriggerMatch,
|
|
13
|
+
} from '@lexical/react/LexicalTypeaheadMenuPlugin';
|
|
14
|
+
import {
|
|
15
|
+
$createTextNode,
|
|
16
|
+
$getSelection,
|
|
17
|
+
$isRangeSelection,
|
|
18
|
+
TextNode,
|
|
19
|
+
} from 'lexical';
|
|
20
|
+
import { $createEmojiNode } from './EmojiNode';
|
|
21
|
+
import materialIcons from '../../../material/iconPicker/IconNamesList';
|
|
22
|
+
import * as React from 'react';
|
|
23
|
+
import {useCallback, useEffect, useMemo, useState} from 'react';
|
|
24
|
+
import * as ReactDOM from 'react-dom';
|
|
25
|
+
|
|
26
|
+
|
|
27
|
+
function EmojiMenuItem({
|
|
28
|
+
index,
|
|
29
|
+
isSelected,
|
|
30
|
+
onClick,
|
|
31
|
+
onMouseEnter,
|
|
32
|
+
option,
|
|
33
|
+
}: {
|
|
34
|
+
index: number;
|
|
35
|
+
isSelected: boolean;
|
|
36
|
+
onClick: () => void;
|
|
37
|
+
onMouseEnter: () => void;
|
|
38
|
+
option: string;
|
|
39
|
+
}) {
|
|
40
|
+
let className = 'item';
|
|
41
|
+
if (isSelected) {
|
|
42
|
+
className += ' selected';
|
|
43
|
+
}
|
|
44
|
+
return (
|
|
45
|
+
<li
|
|
46
|
+
key={option}
|
|
47
|
+
tabIndex={-1}
|
|
48
|
+
className={className}
|
|
49
|
+
role="option"
|
|
50
|
+
aria-selected={isSelected}
|
|
51
|
+
id={'typeahead-item-' + index}
|
|
52
|
+
onMouseEnter={onMouseEnter}
|
|
53
|
+
onClick={onClick}>
|
|
54
|
+
<span className="text">
|
|
55
|
+
<span className="material-symbols-outlined">{option}</span> {option}
|
|
56
|
+
</span>
|
|
57
|
+
</li>
|
|
58
|
+
);
|
|
59
|
+
}
|
|
60
|
+
|
|
61
|
+
|
|
62
|
+
const MAX_EMOJI_SUGGESTION_COUNT = 10;
|
|
63
|
+
|
|
64
|
+
export default function EmojiPickerPlugin() {
|
|
65
|
+
const [editor] = useLexicalComposerContext();
|
|
66
|
+
const [queryString, setQueryString] = useState<string | null>(null);
|
|
67
|
+
|
|
68
|
+
|
|
69
|
+
|
|
70
|
+
const checkForTriggerMatch = useBasicTypeaheadTriggerMatch(':', {
|
|
71
|
+
minLength: 0,
|
|
72
|
+
});
|
|
73
|
+
|
|
74
|
+
const options: Array<any> = useMemo(() => {
|
|
75
|
+
let result = materialIcons.filter((option: string) =>
|
|
76
|
+
(queryString != null) ? new RegExp(queryString, 'gi').exec(option) : materialIcons
|
|
77
|
+
);
|
|
78
|
+
return result.slice(0, MAX_EMOJI_SUGGESTION_COUNT);
|
|
79
|
+
}, [queryString]);
|
|
80
|
+
|
|
81
|
+
const onSelectOption = useCallback(
|
|
82
|
+
(
|
|
83
|
+
selectedOption: any,
|
|
84
|
+
nodeToRemove: TextNode | null,
|
|
85
|
+
closeMenu: () => void,
|
|
86
|
+
) => {
|
|
87
|
+
editor.update(() => {
|
|
88
|
+
const selection = $getSelection();
|
|
89
|
+
|
|
90
|
+
if (!$isRangeSelection(selection) || selectedOption == null) {
|
|
91
|
+
return;
|
|
92
|
+
}
|
|
93
|
+
|
|
94
|
+
if (nodeToRemove) {
|
|
95
|
+
nodeToRemove.remove();
|
|
96
|
+
}
|
|
97
|
+
|
|
98
|
+
selection.insertNodes([$createEmojiNode(selectedOption)]);
|
|
99
|
+
|
|
100
|
+
closeMenu();
|
|
101
|
+
});
|
|
102
|
+
},
|
|
103
|
+
[editor],
|
|
104
|
+
);
|
|
105
|
+
|
|
106
|
+
if (typeof document === "undefined") {
|
|
107
|
+
return null;
|
|
108
|
+
}
|
|
109
|
+
|
|
110
|
+
return (
|
|
111
|
+
<LexicalTypeaheadMenuPlugin
|
|
112
|
+
onQueryChange={setQueryString}
|
|
113
|
+
onSelectOption={onSelectOption}
|
|
114
|
+
triggerFn={checkForTriggerMatch}
|
|
115
|
+
options={options}
|
|
116
|
+
menuRenderFn={(
|
|
117
|
+
anchorElementRef,
|
|
118
|
+
{selectedIndex, selectOptionAndCleanUp, setHighlightedIndex},
|
|
119
|
+
) => {
|
|
120
|
+
if (anchorElementRef.current == null || options.length === 0) {
|
|
121
|
+
return null;
|
|
122
|
+
}
|
|
123
|
+
|
|
124
|
+
return anchorElementRef.current && options.length
|
|
125
|
+
? ReactDOM.createPortal(
|
|
126
|
+
<div className="typeahead-popover emoji-menu">
|
|
127
|
+
<ul>
|
|
128
|
+
{options.map((option: string, index) => (
|
|
129
|
+
<div key={option}>
|
|
130
|
+
<EmojiMenuItem
|
|
131
|
+
index={index}
|
|
132
|
+
isSelected={selectedIndex === index}
|
|
133
|
+
onClick={() => {
|
|
134
|
+
setHighlightedIndex(index);
|
|
135
|
+
selectOptionAndCleanUp(option);
|
|
136
|
+
}}
|
|
137
|
+
onMouseEnter={() => {
|
|
138
|
+
setHighlightedIndex(index);
|
|
139
|
+
}}
|
|
140
|
+
option={option}
|
|
141
|
+
/>
|
|
142
|
+
</div>
|
|
143
|
+
))}
|
|
144
|
+
</ul>
|
|
145
|
+
</div>,
|
|
146
|
+
anchorElementRef.current,
|
|
147
|
+
)
|
|
148
|
+
: null;
|
|
149
|
+
}}
|
|
150
|
+
/>
|
|
151
|
+
);
|
|
152
|
+
}
|