@pareto-engineering/design-system 4.0.0-alpha.77 → 4.0.0-alpha.78
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/dist/cjs/a/AnimatedBlobs/AnimatedBlobs.js +1 -1
- package/dist/cjs/a/AnimatedGradient/AnimatedGradient.js +1 -1
- package/dist/cjs/a/AppContext/Context.js +1 -1
- package/dist/cjs/a/AppContext/ContextProvider.js +1 -1
- package/dist/cjs/a/BlurOverlay/BlurOverlay.js +1 -1
- package/dist/cjs/a/ContentTree/ContentTree.js +1 -1
- package/dist/cjs/a/ContentTree/common/Tree/Tree.js +1 -1
- package/dist/cjs/a/Conversation/Context.js +1 -1
- package/dist/cjs/a/Conversation/Conversation.js +1 -1
- package/dist/cjs/a/Conversation/common/Message/Message.js +1 -1
- package/dist/cjs/a/DatePicker/DatePicker.js +1 -1
- package/dist/cjs/a/DotInfo/DotInfo.js +1 -1
- package/dist/cjs/a/Figure/Figure.js +1 -1
- package/dist/cjs/a/HamburgerButton/HamburgerButton.js +1 -1
- package/dist/cjs/a/IconList/IconList.js +1 -1
- package/dist/cjs/a/IconList/common/Item/Item.js +1 -1
- package/dist/cjs/a/Label/Label.js +1 -1
- package/dist/cjs/a/LexicalPreview/LexicalPreview.js +1 -1
- package/dist/cjs/a/LoadingCircle/LoadingCircle.js +1 -1
- package/dist/cjs/a/MetaCard/MetaCard.js +1 -1
- package/dist/cjs/a/People/People.js +1 -1
- package/dist/cjs/a/People/common/Person/Person.js +1 -1
- package/dist/cjs/a/People/styles.scss +1 -1
- package/dist/cjs/a/Popover/Popover.js +1 -1
- package/dist/cjs/a/ProgressBar/ProgressBar.js +1 -1
- package/dist/cjs/a/Quote/Quote.js +16 -6
- package/dist/cjs/a/Quote/styles.scss +16 -37
- package/dist/cjs/a/Removable/Removable.js +1 -1
- package/dist/cjs/a/SVG/SVG.js +1 -1
- package/dist/cjs/a/SVG/common/UseSVG/UseSVG.js +1 -1
- package/dist/cjs/a/SnapScroller/SnapScroller.js +1 -1
- package/dist/cjs/a/Spinner/Spinner.js +1 -1
- package/dist/cjs/a/TextSteps/TextSteps.js +1 -1
- package/dist/cjs/a/Timestamp/Timestamp.js +1 -1
- package/dist/cjs/a/Tip/Tip.js +1 -1
- package/dist/cjs/a/ToggleSwitch/ToggleSwitch.js +1 -1
- package/dist/cjs/a/Tooltip/Tooltip.js +71 -0
- package/dist/cjs/a/Tooltip/index.js +13 -0
- package/dist/cjs/a/Tooltip/styles.scss +53 -0
- package/dist/cjs/a/XMLEditor/XMLEditor.js +1 -1
- package/dist/cjs/a/index.js +8 -1
- package/dist/cjs/b/Button/Button.js +1 -1
- package/dist/cjs/b/Button/common/Group/Group.js +1 -1
- package/dist/cjs/b/Card/Card.js +1 -1
- package/dist/cjs/b/Card/common/Group/Group.js +1 -1
- package/dist/cjs/b/Card/common/Section/Section.js +1 -1
- package/dist/cjs/b/ExpandableLexicalPreview/ExpandableLexicalPreview.js +3 -3
- package/dist/cjs/b/ExpandableLexicalPreview/common/ExpandButton/ExpandButton.js +3 -3
- package/dist/cjs/b/Logo/Logo.js +1 -1
- package/dist/cjs/b/Page/Context.js +1 -1
- package/dist/cjs/b/Page/Page.js +1 -1
- package/dist/cjs/b/Page/common/PageHelmet/PageHelmet.js +1 -1
- package/dist/cjs/b/Page/common/Section/Section.js +1 -1
- package/dist/cjs/b/SocialMediaButton/SocialMediaButton.js +1 -1
- package/dist/cjs/b/ThemeSelector/ThemeSelector.js +1 -1
- package/dist/cjs/b/Title/Title.js +1 -1
- package/dist/cjs/c/Modal/Modal.js +1 -1
- package/dist/cjs/c/Modal/common/ModalHeader/ModalHeader.js +1 -1
- package/dist/cjs/c/Shortener/Shortener.js +1 -1
- package/dist/cjs/c/SocialMediaShareButton/SocialMediaShareButton.js +1 -1
- package/dist/cjs/f/FormInput/FormInput.js +1 -1
- package/dist/cjs/f/common/Debugger/Debugger.js +1 -1
- package/dist/cjs/f/common/Description/Description.js +1 -1
- package/dist/cjs/f/common/Label/Label.js +1 -1
- package/dist/cjs/f/fields/Checkbox/Checkbox.js +1 -1
- package/dist/cjs/f/fields/ChoicesInput/ChoicesInput.js +1 -1
- package/dist/cjs/f/fields/ChoicesInput/common/Choice/Choice.js +1 -1
- package/dist/cjs/f/fields/EditorInput/EditorInput.js +20 -9
- package/dist/cjs/f/fields/EditorInput/common/BlockFormatDropDown/BlockFormatDropDown.js +256 -0
- package/dist/cjs/f/fields/EditorInput/common/BlockFormatDropDown/index.js +13 -0
- package/dist/cjs/f/fields/EditorInput/common/BlockFormatDropDown/styles.scss +55 -0
- package/dist/cjs/f/fields/EditorInput/common/ColorPicker/ColorPicker.js +94 -0
- package/dist/cjs/f/fields/EditorInput/common/ColorPicker/index.js +13 -0
- package/dist/cjs/f/fields/EditorInput/common/ColorPicker/styles.scss +36 -0
- package/dist/cjs/f/fields/EditorInput/common/FontSizeDropDown/FontSizeDropDown.js +94 -0
- package/dist/cjs/f/fields/EditorInput/common/FontSizeDropDown/index.js +13 -0
- package/dist/cjs/f/fields/EditorInput/common/FontSizeDropDown/styles.scss +55 -0
- package/dist/cjs/f/fields/EditorInput/common/ToolbarPlugin/ToolbarPlugin.js +295 -0
- package/dist/cjs/f/fields/EditorInput/common/ToolbarPlugin/index.js +13 -0
- package/dist/cjs/f/fields/EditorInput/common/ToolbarPlugin/styles.scss +56 -0
- package/dist/cjs/f/fields/EditorInput/common/index.js +24 -3
- package/dist/cjs/f/fields/EditorInput/styles.scss +109 -25
- package/dist/cjs/f/fields/LinkInput/LinkInput.js +1 -1
- package/dist/cjs/f/fields/QueryChoices/QueryChoices.js +1 -1
- package/dist/cjs/f/fields/QueryCombobox/QueryCombobox.js +1 -1
- package/dist/cjs/f/fields/QueryCombobox/common/Combobox/Combobox.js +1 -1
- package/dist/cjs/f/fields/QueryCombobox/common/Menu/Menu.js +1 -1
- package/dist/cjs/f/fields/QueryCombobox/common/MultipleCombobox/MultipleCombobox.js +1 -1
- package/dist/cjs/f/fields/QuerySelect/QuerySelect.js +1 -1
- package/dist/cjs/f/fields/RatingsInput/RatingsInput.js +1 -1
- package/dist/cjs/f/fields/RatingsInput/common/Rating/Rating.js +1 -1
- package/dist/cjs/f/fields/SelectInput/SelectInput.js +1 -1
- package/dist/cjs/f/fields/SelectInput/common/Menu/Menu.js +1 -1
- package/dist/cjs/f/fields/SelectInput/common/Multiple/Multiple.js +1 -1
- package/dist/cjs/f/fields/SelectInput/common/Single/Single.js +1 -1
- package/dist/cjs/f/fields/TextInput/TextInput.js +1 -1
- package/dist/cjs/f/fields/TextareaInput/TextareaInput.js +1 -1
- package/dist/cjs/test/QueryLoader/QueryLoader.js +1 -1
- package/dist/cjs/test/QueryLoader/common/PreloadedTestData/PreloadedTestData.js +1 -1
- package/dist/cjs/utils/hooks/index.js +7 -0
- package/dist/cjs/utils/hooks/useOutsideClick.js +25 -0
- package/dist/cjs/utils/index.js +6 -0
- package/dist/es/a/People/styles.scss +1 -1
- package/dist/es/a/Quote/Quote.js +15 -5
- package/dist/es/a/Quote/styles.scss +16 -37
- package/dist/es/a/Tooltip/Tooltip.js +59 -0
- package/dist/es/a/Tooltip/index.js +2 -0
- package/dist/es/a/Tooltip/styles.scss +53 -0
- package/dist/es/a/index.js +2 -1
- package/dist/es/b/ExpandableLexicalPreview/ExpandableLexicalPreview.js +1 -1
- package/dist/es/b/ExpandableLexicalPreview/common/ExpandButton/ExpandButton.js +1 -1
- package/dist/es/f/fields/EditorInput/EditorInput.js +20 -9
- package/dist/es/f/fields/EditorInput/common/BlockFormatDropDown/BlockFormatDropDown.js +246 -0
- package/dist/es/f/fields/EditorInput/common/BlockFormatDropDown/index.js +2 -0
- package/dist/es/f/fields/EditorInput/common/BlockFormatDropDown/styles.scss +55 -0
- package/dist/es/f/fields/EditorInput/common/ColorPicker/ColorPicker.js +83 -0
- package/dist/es/f/fields/EditorInput/common/ColorPicker/index.js +2 -0
- package/dist/es/f/fields/EditorInput/common/ColorPicker/styles.scss +36 -0
- package/dist/es/f/fields/EditorInput/common/FontSizeDropDown/FontSizeDropDown.js +81 -0
- package/dist/es/f/fields/EditorInput/common/FontSizeDropDown/index.js +2 -0
- package/dist/es/f/fields/EditorInput/common/FontSizeDropDown/styles.scss +55 -0
- package/dist/es/f/fields/EditorInput/common/ToolbarPlugin/ToolbarPlugin.js +283 -0
- package/dist/es/f/fields/EditorInput/common/ToolbarPlugin/index.js +2 -0
- package/dist/es/f/fields/EditorInput/common/ToolbarPlugin/styles.scss +56 -0
- package/dist/es/f/fields/EditorInput/common/index.js +5 -2
- package/dist/es/f/fields/EditorInput/styles.scss +109 -25
- package/dist/es/utils/hooks/index.js +2 -1
- package/dist/es/utils/hooks/useOutsideClick.js +19 -0
- package/dist/es/utils/index.js +1 -1
- package/package.json +13 -5
- package/src/stories/a/Quote.stories.jsx +30 -10
- package/src/stories/a/Tooltip.stories.jsx +80 -0
- package/src/stories/a/XMLEditor.stories.jsx +6 -1
- package/src/ui/a/People/styles.scss +1 -1
- package/src/ui/a/Quote/Quote.jsx +16 -9
- package/src/ui/a/Quote/styles.scss +16 -37
- package/src/ui/a/Tooltip/Tooltip.jsx +83 -0
- package/src/ui/a/Tooltip/index.js +2 -0
- package/src/ui/a/Tooltip/styles.scss +53 -0
- package/src/ui/a/index.js +1 -0
- package/src/ui/b/ExpandableLexicalPreview/ExpandableLexicalPreview.jsx +1 -1
- package/src/ui/b/ExpandableLexicalPreview/common/ExpandButton/ExpandButton.jsx +1 -1
- package/src/ui/f/fields/EditorInput/EditorInput.jsx +21 -4
- package/src/ui/f/fields/EditorInput/common/BlockFormatDropDown/BlockFormatDropDown.jsx +292 -0
- package/src/ui/f/fields/EditorInput/common/BlockFormatDropDown/index.js +2 -0
- package/src/ui/f/fields/EditorInput/common/BlockFormatDropDown/styles.scss +55 -0
- package/src/ui/f/fields/EditorInput/common/ColorPicker/ColorPicker.jsx +110 -0
- package/src/ui/f/fields/EditorInput/common/ColorPicker/index.js +2 -0
- package/src/ui/f/fields/EditorInput/common/ColorPicker/styles.scss +36 -0
- package/src/ui/f/fields/EditorInput/common/FontSizeDropDown/FontSizeDropDown.jsx +134 -0
- package/src/ui/f/fields/EditorInput/common/FontSizeDropDown/index.js +2 -0
- package/src/ui/f/fields/EditorInput/common/FontSizeDropDown/styles.scss +55 -0
- package/src/ui/f/fields/EditorInput/common/ToolbarPlugin/ToolbarPlugin.jsx +431 -0
- package/src/ui/f/fields/EditorInput/common/ToolbarPlugin/index.js +2 -0
- package/src/ui/f/fields/EditorInput/common/ToolbarPlugin/styles.scss +56 -0
- package/src/ui/f/fields/EditorInput/common/index.js +6 -0
- package/src/ui/f/fields/EditorInput/styles.scss +109 -25
- package/src/ui/utils/hooks/index.js +1 -0
- package/src/ui/utils/hooks/useOutsideClick.js +23 -0
- package/src/ui/utils/index.js +1 -1
- package/tests/__snapshots__/Storyshots.test.js.snap +746 -378
- package/tests/mockResizeObserver.js +9 -0
- package/tests/test-setup.js +3 -0
- package/dist/cjs/f/fields/EditorInput/common/Toolbar.js +0 -257
- package/dist/es/f/fields/EditorInput/common/Toolbar.js +0 -246
- package/src/ui/f/fields/EditorInput/common/Toolbar.jsx +0 -356
- package/src/ui/f/fields/EditorInput/common/index.jsx +0 -3
|
@@ -0,0 +1,283 @@
|
|
|
1
|
+
/* eslint-disable import/no-extraneous-dependencies -- required here */
|
|
2
|
+
import * as React from 'react';
|
|
3
|
+
import { useEffect, useState, useCallback } from 'react';
|
|
4
|
+
import PropTypes from 'prop-types';
|
|
5
|
+
import { mergeRegister, $getNearestNodeOfType, $findMatchingParent } from '@lexical/utils';
|
|
6
|
+
import { $isHeadingNode } from '@lexical/rich-text';
|
|
7
|
+
import { $getSelection, $isRangeSelection, $isRootOrShadowRoot, FORMAT_TEXT_COMMAND, FORMAT_ELEMENT_COMMAND, UNDO_COMMAND, REDO_COMMAND, SELECTION_CHANGE_COMMAND, COMMAND_PRIORITY_CRITICAL, CAN_UNDO_COMMAND, CAN_REDO_COMMAND, DEPRECATED_$isGridSelection } from 'lexical';
|
|
8
|
+
import { $isListNode, ListNode } from '@lexical/list';
|
|
9
|
+
import { $getSelectionStyleValueForProperty, $isAtNodeEnd, $patchStyleText } from '@lexical/selection';
|
|
10
|
+
import { $isLinkNode, TOGGLE_LINK_COMMAND } from '@lexical/link';
|
|
11
|
+
import { useLexicalComposerContext } from '@lexical/react/LexicalComposerContext';
|
|
12
|
+
import { $isTableNode } from '@lexical/table';
|
|
13
|
+
import styleNames from '@pareto-engineering/bem/exports';
|
|
14
|
+
import { BlockFormatDropDown } from "../BlockFormatDropDown";
|
|
15
|
+
import { ColorPicker } from "../ColorPicker";
|
|
16
|
+
import { FontSizeDropDown } from "../FontSizeDropDown";
|
|
17
|
+
|
|
18
|
+
// Local Definitions
|
|
19
|
+
import "./styles.scss";
|
|
20
|
+
const getSelectedNode = selection => {
|
|
21
|
+
const {
|
|
22
|
+
anchor,
|
|
23
|
+
focus
|
|
24
|
+
} = selection;
|
|
25
|
+
const anchorNode = selection.anchor.getNode();
|
|
26
|
+
const focusNode = selection.focus.getNode();
|
|
27
|
+
if (anchorNode === focusNode) {
|
|
28
|
+
return anchorNode;
|
|
29
|
+
}
|
|
30
|
+
const isBackward = selection.isBackward();
|
|
31
|
+
if (isBackward) {
|
|
32
|
+
return $isAtNodeEnd(focus) ? anchorNode : focusNode;
|
|
33
|
+
}
|
|
34
|
+
return $isAtNodeEnd(anchor) ? focusNode : anchorNode;
|
|
35
|
+
};
|
|
36
|
+
export const blockTypeToBlockName = {
|
|
37
|
+
bullet: 'Bulleted List',
|
|
38
|
+
check: 'Check List',
|
|
39
|
+
code: 'Code Block',
|
|
40
|
+
h1: 'Heading 1',
|
|
41
|
+
h2: 'Heading 2',
|
|
42
|
+
h3: 'Heading 3',
|
|
43
|
+
number: 'Numbered List',
|
|
44
|
+
paragraph: 'Normal',
|
|
45
|
+
quote: 'Quote'
|
|
46
|
+
};
|
|
47
|
+
const baseClassName = styleNames.base;
|
|
48
|
+
const componentClassName = 'toolbar-plugin';
|
|
49
|
+
|
|
50
|
+
/**
|
|
51
|
+
* This is the component description
|
|
52
|
+
*/
|
|
53
|
+
const ToolbarPlugin = ({
|
|
54
|
+
id,
|
|
55
|
+
className: userClassName,
|
|
56
|
+
style: inlineStyles,
|
|
57
|
+
baseBgColor
|
|
58
|
+
// ...otherProps
|
|
59
|
+
}) => {
|
|
60
|
+
const [editor] = useLexicalComposerContext();
|
|
61
|
+
const [activeEditor, setActiveEditor] = useState(editor);
|
|
62
|
+
const [blockType, setBlockType] = useState('paragraph');
|
|
63
|
+
const [rootType, setRootType] = useState('root');
|
|
64
|
+
const [fontColor, setFontColor] = useState('#000000');
|
|
65
|
+
const [bgColor, setBgColor] = useState('#ffffff');
|
|
66
|
+
const [fontSize, setFontSize] = useState('15px');
|
|
67
|
+
const [isEditable, setIsEditable] = useState(() => editor.isEditable());
|
|
68
|
+
const [isBold, setIsBold] = useState(false);
|
|
69
|
+
const [isItalic, setIsItalic] = useState(false);
|
|
70
|
+
const [isStrikethrough, setIsStrikethrough] = useState(false);
|
|
71
|
+
const [isLink, setIsLink] = useState(false);
|
|
72
|
+
const [isUnderline, setIsUnderline] = useState(false);
|
|
73
|
+
const [canRedo, setCanRedo] = useState(false);
|
|
74
|
+
const [canUndo, setCanUndo] = useState(false);
|
|
75
|
+
const formatLink = useCallback(() => {
|
|
76
|
+
if (!isLink) {
|
|
77
|
+
// eslint-disable-next-line no-alert
|
|
78
|
+
const path = prompt('Enter the full URL. Ex: https://www.example.com');
|
|
79
|
+
editor.dispatchCommand(TOGGLE_LINK_COMMAND, path);
|
|
80
|
+
} else {
|
|
81
|
+
editor.dispatchCommand(TOGGLE_LINK_COMMAND, null);
|
|
82
|
+
}
|
|
83
|
+
}, [editor, isLink]);
|
|
84
|
+
const $updateToolbar = useCallback(() => {
|
|
85
|
+
const selection = $getSelection();
|
|
86
|
+
if ($isRangeSelection(selection)) {
|
|
87
|
+
const anchorNode = selection.anchor.getNode();
|
|
88
|
+
let element = anchorNode.getKey() === 'root' ? anchorNode : $findMatchingParent(anchorNode, e => {
|
|
89
|
+
const parent = e.getParent();
|
|
90
|
+
return parent !== null && $isRootOrShadowRoot(parent);
|
|
91
|
+
});
|
|
92
|
+
if (element === null) {
|
|
93
|
+
element = anchorNode.getTopLevelElementOrThrow();
|
|
94
|
+
}
|
|
95
|
+
const elementKey = element.getKey();
|
|
96
|
+
const elementDOM = activeEditor.getElementByKey(elementKey);
|
|
97
|
+
|
|
98
|
+
// Update text format
|
|
99
|
+
setIsBold(selection.hasFormat('bold'));
|
|
100
|
+
setIsItalic(selection.hasFormat('italic'));
|
|
101
|
+
setIsUnderline(selection.hasFormat('underline'));
|
|
102
|
+
setIsStrikethrough(selection.hasFormat('strikethrough'));
|
|
103
|
+
|
|
104
|
+
// Update links
|
|
105
|
+
const node = getSelectedNode(selection);
|
|
106
|
+
const parent = node.getParent();
|
|
107
|
+
if ($isLinkNode(parent) || $isLinkNode(node)) {
|
|
108
|
+
setIsLink(true);
|
|
109
|
+
} else {
|
|
110
|
+
setIsLink(false);
|
|
111
|
+
}
|
|
112
|
+
const tableNode = $findMatchingParent(node, $isTableNode);
|
|
113
|
+
if ($isTableNode(tableNode)) {
|
|
114
|
+
setRootType('table');
|
|
115
|
+
} else {
|
|
116
|
+
setRootType('root');
|
|
117
|
+
}
|
|
118
|
+
if (elementDOM !== null) {
|
|
119
|
+
if ($isListNode(element)) {
|
|
120
|
+
const parentList = $getNearestNodeOfType(anchorNode, ListNode);
|
|
121
|
+
const type = parentList ? parentList.getListType() : element.getListType();
|
|
122
|
+
setBlockType(type);
|
|
123
|
+
} else {
|
|
124
|
+
const type = $isHeadingNode(element) ? element.getTag() : element.getType();
|
|
125
|
+
if (type in blockTypeToBlockName) {
|
|
126
|
+
setBlockType(type);
|
|
127
|
+
}
|
|
128
|
+
}
|
|
129
|
+
}
|
|
130
|
+
setFontSize($getSelectionStyleValueForProperty(selection, 'font-size', '15px'));
|
|
131
|
+
setFontColor($getSelectionStyleValueForProperty(selection, 'color', '#000'));
|
|
132
|
+
setBgColor($getSelectionStyleValueForProperty(selection, 'background-color', '#fff'));
|
|
133
|
+
}
|
|
134
|
+
}, [activeEditor]);
|
|
135
|
+
useEffect(() => editor.registerCommand(SELECTION_CHANGE_COMMAND, (_payload, newEditor) => {
|
|
136
|
+
$updateToolbar();
|
|
137
|
+
setActiveEditor(newEditor);
|
|
138
|
+
return false;
|
|
139
|
+
}, COMMAND_PRIORITY_CRITICAL), [editor, $updateToolbar]);
|
|
140
|
+
useEffect(() => mergeRegister(editor.registerEditableListener(editable => {
|
|
141
|
+
setIsEditable(editable);
|
|
142
|
+
}), activeEditor.registerUpdateListener(({
|
|
143
|
+
editorState
|
|
144
|
+
}) => {
|
|
145
|
+
editorState.read(() => {
|
|
146
|
+
$updateToolbar();
|
|
147
|
+
});
|
|
148
|
+
}), activeEditor.registerCommand(CAN_UNDO_COMMAND, payload => {
|
|
149
|
+
setCanUndo(payload);
|
|
150
|
+
return false;
|
|
151
|
+
}, COMMAND_PRIORITY_CRITICAL), activeEditor.registerCommand(CAN_REDO_COMMAND, payload => {
|
|
152
|
+
setCanRedo(payload);
|
|
153
|
+
return false;
|
|
154
|
+
}, COMMAND_PRIORITY_CRITICAL)), [$updateToolbar, activeEditor, editor]);
|
|
155
|
+
const applyStyleText = useCallback(styles => {
|
|
156
|
+
activeEditor.update(() => {
|
|
157
|
+
const selection = $getSelection();
|
|
158
|
+
if ($isRangeSelection(selection) || DEPRECATED_$isGridSelection(selection)) {
|
|
159
|
+
$patchStyleText(selection, styles);
|
|
160
|
+
}
|
|
161
|
+
});
|
|
162
|
+
}, [activeEditor]);
|
|
163
|
+
const onFontColorSelect = useCallback(value => {
|
|
164
|
+
applyStyleText({
|
|
165
|
+
color: value
|
|
166
|
+
});
|
|
167
|
+
}, [applyStyleText]);
|
|
168
|
+
const onBgColorSelect = useCallback(value => {
|
|
169
|
+
applyStyleText({
|
|
170
|
+
'background-color': value
|
|
171
|
+
});
|
|
172
|
+
}, [applyStyleText]);
|
|
173
|
+
return /*#__PURE__*/React.createElement("div", {
|
|
174
|
+
id: id,
|
|
175
|
+
className: [baseClassName, componentClassName, userClassName, `x-${baseBgColor}`].filter(e => e).join(' '),
|
|
176
|
+
style: inlineStyles
|
|
177
|
+
}, /*#__PURE__*/React.createElement("div", {
|
|
178
|
+
className: "group"
|
|
179
|
+
}, /*#__PURE__*/React.createElement("button", {
|
|
180
|
+
type: "button",
|
|
181
|
+
onClick: () => editor.dispatchCommand(UNDO_COMMAND),
|
|
182
|
+
className: "ai-icon",
|
|
183
|
+
disabled: !canUndo || !isEditable
|
|
184
|
+
}, "A"), /*#__PURE__*/React.createElement("button", {
|
|
185
|
+
type: "button",
|
|
186
|
+
className: "ai-icon",
|
|
187
|
+
onClick: () => editor.dispatchCommand(REDO_COMMAND),
|
|
188
|
+
disabled: !canRedo || !isEditable
|
|
189
|
+
}, "B")), /*#__PURE__*/React.createElement(BlockFormatDropDown, {
|
|
190
|
+
blockType: blockType,
|
|
191
|
+
editor: editor,
|
|
192
|
+
rootType: rootType
|
|
193
|
+
}), /*#__PURE__*/React.createElement("div", {
|
|
194
|
+
className: "group"
|
|
195
|
+
}, /*#__PURE__*/React.createElement("button", {
|
|
196
|
+
type: "button",
|
|
197
|
+
className: isBold ? 'active' : undefined,
|
|
198
|
+
onClick: () => editor.dispatchCommand(FORMAT_TEXT_COMMAND, 'bold')
|
|
199
|
+
}, /*#__PURE__*/React.createElement("span", {
|
|
200
|
+
className: "icon"
|
|
201
|
+
}, "|")), /*#__PURE__*/React.createElement("button", {
|
|
202
|
+
type: "button",
|
|
203
|
+
className: isItalic ? 'active' : undefined,
|
|
204
|
+
onClick: () => editor.dispatchCommand(FORMAT_TEXT_COMMAND, 'italic')
|
|
205
|
+
}, /*#__PURE__*/React.createElement("span", {
|
|
206
|
+
className: "icon"
|
|
207
|
+
}, "}")), /*#__PURE__*/React.createElement("button", {
|
|
208
|
+
type: "button",
|
|
209
|
+
className: isUnderline ? 'active' : undefined,
|
|
210
|
+
onClick: () => editor.dispatchCommand(FORMAT_TEXT_COMMAND, 'underline')
|
|
211
|
+
}, /*#__PURE__*/React.createElement("span", {
|
|
212
|
+
className: "icon"
|
|
213
|
+
}, "~")), /*#__PURE__*/React.createElement("button", {
|
|
214
|
+
type: "button",
|
|
215
|
+
className: isStrikethrough ? 'active' : undefined,
|
|
216
|
+
onClick: () => editor.dispatchCommand(FORMAT_TEXT_COMMAND, 'strikethrough')
|
|
217
|
+
}, /*#__PURE__*/React.createElement("span", {
|
|
218
|
+
className: "icon"
|
|
219
|
+
}, "?")), /*#__PURE__*/React.createElement("button", {
|
|
220
|
+
type: "button",
|
|
221
|
+
className: isLink ? 'active' : undefined,
|
|
222
|
+
onClick: formatLink
|
|
223
|
+
}, /*#__PURE__*/React.createElement("span", {
|
|
224
|
+
className: "icon"
|
|
225
|
+
}, "]"))), /*#__PURE__*/React.createElement(FontSizeDropDown, {
|
|
226
|
+
editor: editor,
|
|
227
|
+
value: fontSize
|
|
228
|
+
}), /*#__PURE__*/React.createElement("div", {
|
|
229
|
+
className: "group"
|
|
230
|
+
}, /*#__PURE__*/React.createElement(ColorPicker, {
|
|
231
|
+
icon: "'",
|
|
232
|
+
color: fontColor,
|
|
233
|
+
onChange: onFontColorSelect
|
|
234
|
+
}), /*#__PURE__*/React.createElement(ColorPicker, {
|
|
235
|
+
icon: "#",
|
|
236
|
+
color: bgColor,
|
|
237
|
+
onChange: onBgColorSelect
|
|
238
|
+
})), /*#__PURE__*/React.createElement("div", {
|
|
239
|
+
className: "group"
|
|
240
|
+
}, /*#__PURE__*/React.createElement("button", {
|
|
241
|
+
type: "button",
|
|
242
|
+
onClick: () => editor.dispatchCommand(FORMAT_ELEMENT_COMMAND, 'left')
|
|
243
|
+
}, /*#__PURE__*/React.createElement("span", {
|
|
244
|
+
className: "icon"
|
|
245
|
+
}, "^")), /*#__PURE__*/React.createElement("button", {
|
|
246
|
+
type: "button",
|
|
247
|
+
onClick: () => editor.dispatchCommand(FORMAT_ELEMENT_COMMAND, 'center')
|
|
248
|
+
}, /*#__PURE__*/React.createElement("span", {
|
|
249
|
+
className: "icon"
|
|
250
|
+
}, "_")), /*#__PURE__*/React.createElement("button", {
|
|
251
|
+
type: "button",
|
|
252
|
+
onClick: () => editor.dispatchCommand(FORMAT_ELEMENT_COMMAND, 'right')
|
|
253
|
+
}, /*#__PURE__*/React.createElement("span", {
|
|
254
|
+
className: "icon"
|
|
255
|
+
}, "`")), /*#__PURE__*/React.createElement("button", {
|
|
256
|
+
type: "button",
|
|
257
|
+
onClick: () => editor.dispatchCommand(FORMAT_ELEMENT_COMMAND, 'justify')
|
|
258
|
+
}, /*#__PURE__*/React.createElement("span", {
|
|
259
|
+
className: "icon"
|
|
260
|
+
}, "{"))));
|
|
261
|
+
};
|
|
262
|
+
ToolbarPlugin.propTypes = {
|
|
263
|
+
/**
|
|
264
|
+
* The HTML id for this element
|
|
265
|
+
*/
|
|
266
|
+
id: PropTypes.string,
|
|
267
|
+
/**
|
|
268
|
+
* The HTML class names for this element
|
|
269
|
+
*/
|
|
270
|
+
className: PropTypes.string,
|
|
271
|
+
/**
|
|
272
|
+
* The React-written, css properties for this element.
|
|
273
|
+
*/
|
|
274
|
+
style: PropTypes.objectOf(PropTypes.string),
|
|
275
|
+
/**
|
|
276
|
+
* Base background color
|
|
277
|
+
*/
|
|
278
|
+
baseBgColor: PropTypes.string
|
|
279
|
+
};
|
|
280
|
+
ToolbarPlugin.defaultProps = {
|
|
281
|
+
baseBgColor: 'background-far'
|
|
282
|
+
};
|
|
283
|
+
export default ToolbarPlugin;
|
|
@@ -0,0 +1,56 @@
|
|
|
1
|
+
/* @pareto-engineering/generator-front 1.0.12 */
|
|
2
|
+
|
|
3
|
+
@use "@pareto-engineering/bem";
|
|
4
|
+
@use "@pareto-engineering/styles/src/mixins";
|
|
5
|
+
@use "@pareto-engineering/styles/src/globals" as *;
|
|
6
|
+
|
|
7
|
+
|
|
8
|
+
$default-gap: calc(var(--gap) / 2);
|
|
9
|
+
$default-padding: .55em .75em;
|
|
10
|
+
$default-input-border-radius: var(--theme-default-input-border-radius);
|
|
11
|
+
$default-border: var(--theme-default-input-border);
|
|
12
|
+
$default-icon-color: var(--soft-ui-icons);
|
|
13
|
+
|
|
14
|
+
.#{bem.$base}.toolbar-plugin {
|
|
15
|
+
align-items: center;
|
|
16
|
+
border-inline: $default-border;
|
|
17
|
+
border-top: $default-border;
|
|
18
|
+
border-top-left-radius: $default-input-border-radius;
|
|
19
|
+
border-top-right-radius: $default-input-border-radius;
|
|
20
|
+
display: flex;
|
|
21
|
+
flex-wrap: wrap;
|
|
22
|
+
padding-block: calc(var(--theme-default-padding) / 4);
|
|
23
|
+
|
|
24
|
+
> .ds + .group {
|
|
25
|
+
border-left: 1px solid var(--soft-background-inputs);
|
|
26
|
+
}
|
|
27
|
+
|
|
28
|
+
> .group {
|
|
29
|
+
display: flex;
|
|
30
|
+
gap: calc(var(--gap) / 2);
|
|
31
|
+
|
|
32
|
+
> button {
|
|
33
|
+
background: var(--x);
|
|
34
|
+
border: 0;
|
|
35
|
+
border-radius: calc(var(--theme-default-border-radius) / 2);
|
|
36
|
+
color: $default-icon-color;
|
|
37
|
+
cursor: pointer;
|
|
38
|
+
margin: 0;
|
|
39
|
+
padding: $default-padding;
|
|
40
|
+
|
|
41
|
+
&:hover,
|
|
42
|
+
&.active {
|
|
43
|
+
background: var(--hard-x);
|
|
44
|
+
}
|
|
45
|
+
}
|
|
46
|
+
|
|
47
|
+
&:not(.#{bem.$base}) {
|
|
48
|
+
padding-inline: calc(var(--theme-default-padding) / 4);
|
|
49
|
+
}
|
|
50
|
+
|
|
51
|
+
|
|
52
|
+
&:not(:last-child, .#{bem.$base}) {
|
|
53
|
+
border-right: 1px solid var(--soft-background-inputs);
|
|
54
|
+
}
|
|
55
|
+
}
|
|
56
|
+
}
|
|
@@ -1,3 +1,6 @@
|
|
|
1
|
-
export {
|
|
1
|
+
export { BlockFormatDropDown } from "./BlockFormatDropDown";
|
|
2
2
|
export { default as TreeViewPlugin } from "./TreeViewPlugin";
|
|
3
|
-
export { default as StopPropagationPlugin } from "./StopPropagationPlugin";
|
|
3
|
+
export { default as StopPropagationPlugin } from "./StopPropagationPlugin";
|
|
4
|
+
export { ToolbarPlugin } from "./ToolbarPlugin";
|
|
5
|
+
export { ColorPicker } from "./ColorPicker";
|
|
6
|
+
export { FontSizeDropDown } from "./FontSizeDropDown";
|
|
@@ -6,7 +6,6 @@
|
|
|
6
6
|
|
|
7
7
|
$default-gap: calc(var(--gap) / 2);
|
|
8
8
|
$default-padding: .55em .75em;
|
|
9
|
-
$default-border: 1px solid var(--outline-inputs);
|
|
10
9
|
$default-input-border-radius: var(--theme-default-input-border-radius);
|
|
11
10
|
$default-border: var(--theme-default-input-border);
|
|
12
11
|
$focus-border: var(--theme-focus-input-border);
|
|
@@ -15,6 +14,7 @@ $default-background: var(--background-inputs);
|
|
|
15
14
|
$default-icon-color: var(--on-background-inputs);
|
|
16
15
|
$disabled-background: var(--background-inputs-30);
|
|
17
16
|
$default-color-menu-padding: .5em .25em;
|
|
17
|
+
$default-check-mark-dimensions: 1rem;
|
|
18
18
|
|
|
19
19
|
.#{bem.$base}.editor-input {
|
|
20
20
|
display: flex;
|
|
@@ -24,29 +24,6 @@ $default-color-menu-padding: .5em .25em;
|
|
|
24
24
|
margin-bottom: var(--gap);
|
|
25
25
|
}
|
|
26
26
|
|
|
27
|
-
> .#{bem.$base}.toolbar {
|
|
28
|
-
display: flex;
|
|
29
|
-
gap: $default-gap;
|
|
30
|
-
margin-bottom: .25em;
|
|
31
|
-
|
|
32
|
-
> .group {
|
|
33
|
-
> button {
|
|
34
|
-
background: $default-background;
|
|
35
|
-
border: $default-border;
|
|
36
|
-
color: $default-icon-color;
|
|
37
|
-
padding: $default-padding;
|
|
38
|
-
|
|
39
|
-
&.active {
|
|
40
|
-
background-color: $active-background;
|
|
41
|
-
}
|
|
42
|
-
}
|
|
43
|
-
}
|
|
44
|
-
|
|
45
|
-
.flip {
|
|
46
|
-
transform: scaleX(-1);
|
|
47
|
-
}
|
|
48
|
-
}
|
|
49
|
-
|
|
50
27
|
&:not(.disabled) {
|
|
51
28
|
&:hover,
|
|
52
29
|
&:focus,
|
|
@@ -93,7 +70,8 @@ $default-color-menu-padding: .5em .25em;
|
|
|
93
70
|
> .content-editable {
|
|
94
71
|
background: $default-background;
|
|
95
72
|
border: $default-border;
|
|
96
|
-
border-radius: $default-input-border-radius;
|
|
73
|
+
border-bottom-left-radius: $default-input-border-radius;
|
|
74
|
+
border-bottom-right-radius: $default-input-border-radius;
|
|
97
75
|
color: var(--y);
|
|
98
76
|
height: var(--rows);
|
|
99
77
|
outline: none;
|
|
@@ -102,10 +80,116 @@ $default-color-menu-padding: .5em .25em;
|
|
|
102
80
|
resize: var(--resize);
|
|
103
81
|
width: 100%;
|
|
104
82
|
|
|
83
|
+
.lexical-list-item-checked,
|
|
84
|
+
.lexical-list-item-unchecked {
|
|
85
|
+
list-style-type: none;
|
|
86
|
+
margin-inline: calc(var(--gap) / 2);
|
|
87
|
+
outline: none;
|
|
88
|
+
padding-inline: calc(var(--theme-default-padding) * 1.5);
|
|
89
|
+
position: relative;
|
|
90
|
+
}
|
|
91
|
+
|
|
92
|
+
.lexical-list-item-checked {
|
|
93
|
+
text-decoration: line-through;
|
|
94
|
+
}
|
|
95
|
+
|
|
96
|
+
.lexical-list-item-unchecked::before,
|
|
97
|
+
.lexical-list-item-checked::before {
|
|
98
|
+
background-size: cover;
|
|
99
|
+
content: "";
|
|
100
|
+
cursor: pointer;
|
|
101
|
+
display: block;
|
|
102
|
+
height: $default-check-mark-dimensions;
|
|
103
|
+
left: 0;
|
|
104
|
+
position: absolute;
|
|
105
|
+
top: 2px;
|
|
106
|
+
width: $default-check-mark-dimensions;
|
|
107
|
+
}
|
|
108
|
+
|
|
109
|
+
.lexical-list-item-unchecked[dir="rtl"]::before,
|
|
110
|
+
.lexical-list-item-checked[dir="rtl"]::before {
|
|
111
|
+
left: auto;
|
|
112
|
+
right: 0;
|
|
113
|
+
}
|
|
114
|
+
|
|
115
|
+
.lexical-list-item-unchecked:focus::before,
|
|
116
|
+
.lexical-list-item-checked:focus::before {
|
|
117
|
+
border-radius: 2px;
|
|
118
|
+
box-shadow: 0 0 0 2px var(--light-blue);
|
|
119
|
+
}
|
|
120
|
+
|
|
121
|
+
.lexical-list-item-unchecked::before {
|
|
122
|
+
border: var(--theme-default-border-style) var(--ui-icons);
|
|
123
|
+
border-radius: 2px;
|
|
124
|
+
}
|
|
125
|
+
|
|
126
|
+
.lexical-list-item-checked::before {
|
|
127
|
+
background-color: var(--blue);
|
|
128
|
+
background-repeat: no-repeat;
|
|
129
|
+
border: var(--theme-default-border-style) var(--soft-blue);
|
|
130
|
+
border-radius: 2px;
|
|
131
|
+
}
|
|
132
|
+
|
|
133
|
+
.lexical-list-item-checked::after {
|
|
134
|
+
border-color: var(--background-far);
|
|
135
|
+
border-style: solid;
|
|
136
|
+
border-width: 0 2px 2px 0;
|
|
137
|
+
content: "";
|
|
138
|
+
cursor: pointer;
|
|
139
|
+
display: block;
|
|
140
|
+
height: 6px;
|
|
141
|
+
left: 7px;
|
|
142
|
+
position: absolute;
|
|
143
|
+
right: 7px;
|
|
144
|
+
top: 6px;
|
|
145
|
+
transform: rotate(45deg);
|
|
146
|
+
width: 3px;
|
|
147
|
+
}
|
|
148
|
+
|
|
149
|
+
.lexical-code {
|
|
150
|
+
background-color: var(--hard-background-cards);
|
|
151
|
+
display: block;
|
|
152
|
+
font-size: calc(var(--s-1) * 1rem);
|
|
153
|
+
margin: 0;
|
|
154
|
+
margin-bottom: calc(var(--gap) / 2);
|
|
155
|
+
margin-top: calc(var(--gap) / 2);
|
|
156
|
+
overflow-x: auto;
|
|
157
|
+
padding-block: calc(var(--theme-default-padding) / 2);
|
|
158
|
+
padding-bottom: calc(var(--theme-default-padding) / 2);
|
|
159
|
+
padding-left: calc(var(--theme-default-padding) * 2);
|
|
160
|
+
position: relative;
|
|
161
|
+
tab-size: 2;
|
|
162
|
+
}
|
|
163
|
+
|
|
164
|
+
.lexical-code::before {
|
|
165
|
+
background-color: var(--hard-background-cards);
|
|
166
|
+
border-right: var(--theme-default-border-style) var(--ui-icons);
|
|
167
|
+
color: var(--pagraph);
|
|
168
|
+
content: attr(data-gutter);
|
|
169
|
+
left: 0;
|
|
170
|
+
padding: calc(var(--theme-default-padding) / 2);
|
|
171
|
+
position: absolute;
|
|
172
|
+
text-align: right;
|
|
173
|
+
top: 0;
|
|
174
|
+
white-space: pre-wrap;
|
|
175
|
+
}
|
|
176
|
+
|
|
105
177
|
li:has(ol, ul) {
|
|
106
178
|
list-style-type: none;
|
|
107
179
|
}
|
|
108
180
|
|
|
181
|
+
a {
|
|
182
|
+
cursor: pointer;
|
|
183
|
+
}
|
|
184
|
+
|
|
185
|
+
blockquote {
|
|
186
|
+
border-left: 4px solid var(--ui-icons);
|
|
187
|
+
font-size: calc(var(--s0) * 1rem);
|
|
188
|
+
margin: 0;
|
|
189
|
+
margin-left: var(--gap);
|
|
190
|
+
padding-left: calc(var(--theme-default-padding) / 2);
|
|
191
|
+
}
|
|
192
|
+
|
|
109
193
|
p,
|
|
110
194
|
span,
|
|
111
195
|
strong,
|
|
@@ -0,0 +1,19 @@
|
|
|
1
|
+
import { useEffect } from 'react';
|
|
2
|
+
const useOutsideClick = (ref, callback) => {
|
|
3
|
+
// const ref = useRef(null)
|
|
4
|
+
|
|
5
|
+
useEffect(() => {
|
|
6
|
+
const handleClickOutside = event => {
|
|
7
|
+
if (ref.current && !ref.current.contains(event.target)) {
|
|
8
|
+
callback();
|
|
9
|
+
}
|
|
10
|
+
};
|
|
11
|
+
document.addEventListener('mousedown', handleClickOutside);
|
|
12
|
+
document.addEventListener('touchstart', handleClickOutside);
|
|
13
|
+
return () => {
|
|
14
|
+
document.removeEventListener('mousedown', handleClickOutside);
|
|
15
|
+
document.removeEventListener('touchstart', handleClickOutside);
|
|
16
|
+
};
|
|
17
|
+
}, [callback]);
|
|
18
|
+
};
|
|
19
|
+
export default useOutsideClick;
|
package/dist/es/utils/index.js
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
export { useWindowSize, useDynamicPosition } from "./hooks";
|
|
1
|
+
export { useWindowSize, useDynamicPosition, useOutsideClick } from "./hooks";
|
package/package.json
CHANGED
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
{
|
|
2
2
|
"name": "@pareto-engineering/design-system",
|
|
3
|
-
"version": "4.0.0-alpha.
|
|
3
|
+
"version": "4.0.0-alpha.78",
|
|
4
4
|
"description": "",
|
|
5
5
|
"main": "dist/cjs/index.js",
|
|
6
6
|
"module": "dist/es/index.js",
|
|
@@ -51,8 +51,15 @@
|
|
|
51
51
|
},
|
|
52
52
|
"dependencies": {
|
|
53
53
|
"@codemirror/lang-xml": "^6.0.2",
|
|
54
|
-
"@lexical/
|
|
55
|
-
"@
|
|
54
|
+
"@lexical/code": "0.12.2",
|
|
55
|
+
"@lexical/link": "0.12.2",
|
|
56
|
+
"@lexical/list": "0.12.2",
|
|
57
|
+
"@lexical/react": "0.12.2",
|
|
58
|
+
"@lexical/rich-text": "0.12.2",
|
|
59
|
+
"@lexical/selection": "0.12.2",
|
|
60
|
+
"@lexical/table": "0.12.2",
|
|
61
|
+
"@lexical/utils": "0.12.2",
|
|
62
|
+
"@pareto-engineering/assets": "^4.0.0-alpha.78",
|
|
56
63
|
"@pareto-engineering/bem": "^4.0.0-alpha.20",
|
|
57
64
|
"@pareto-engineering/styles": "^4.0.0-alpha.73",
|
|
58
65
|
"@pareto-engineering/utils": "^4.0.0-alpha.73",
|
|
@@ -62,10 +69,11 @@
|
|
|
62
69
|
"formik": "^2.2.9",
|
|
63
70
|
"fuse.js": "^7.0.0",
|
|
64
71
|
"hamburgers": "^1.2.1",
|
|
65
|
-
"lexical": "
|
|
72
|
+
"lexical": "0.12.2",
|
|
66
73
|
"lodash": "^4.17.21",
|
|
67
74
|
"prop-types": "^15.8.1",
|
|
68
75
|
"react": "^18.2.0",
|
|
76
|
+
"react-color-palette": "^7.1.0",
|
|
69
77
|
"react-day-picker": "^8.10.0",
|
|
70
78
|
"react-dom": "^18.2.0",
|
|
71
79
|
"react-helmet-async": "^1.3.0",
|
|
@@ -74,5 +82,5 @@
|
|
|
74
82
|
"relay-test-utils": "^15.0.0"
|
|
75
83
|
},
|
|
76
84
|
"browserslist": "> 2%",
|
|
77
|
-
"gitHead": "
|
|
85
|
+
"gitHead": "2232d74831fee703583d210b9d083f357440b21a"
|
|
78
86
|
}
|
|
@@ -13,17 +13,37 @@ export default {
|
|
|
13
13
|
// storyfn => <div className="">{ storyfn() }</div>,
|
|
14
14
|
],
|
|
15
15
|
argTypes:{
|
|
16
|
-
|
|
16
|
+
children:{
|
|
17
|
+
control:{
|
|
18
|
+
type:'text',
|
|
19
|
+
},
|
|
20
|
+
},
|
|
21
|
+
source:{
|
|
22
|
+
control:{
|
|
23
|
+
type:'text',
|
|
24
|
+
},
|
|
25
|
+
},
|
|
26
|
+
author:{
|
|
27
|
+
control:{
|
|
28
|
+
type:'text',
|
|
29
|
+
},
|
|
30
|
+
},
|
|
31
|
+
color:{
|
|
32
|
+
control:{
|
|
33
|
+
type:'text',
|
|
34
|
+
},
|
|
35
|
+
},
|
|
17
36
|
},
|
|
18
37
|
}
|
|
19
38
|
|
|
20
|
-
|
|
21
|
-
<
|
|
22
|
-
<Quote
|
|
23
|
-
author="Austen Spoonts"
|
|
24
|
-
>
|
|
25
|
-
Pareto was like having a full-time employee without having to hire somebody.
|
|
26
|
-
That’s how I would explain it. An on-call, on-demand full-time employee.
|
|
27
|
-
</Quote>
|
|
28
|
-
</div>
|
|
39
|
+
const Template = (args) => (
|
|
40
|
+
<Quote {...args} />
|
|
29
41
|
)
|
|
42
|
+
|
|
43
|
+
export const Base = Template.bind({})
|
|
44
|
+
Base.args = {
|
|
45
|
+
children:'Lorem ipsum dolor sit amet, consectetur adipiscing elit. Integer posuere erat a ante.',
|
|
46
|
+
source :'https://example.com',
|
|
47
|
+
author :'John Doe',
|
|
48
|
+
color :'highlighted',
|
|
49
|
+
}
|