decap-cms-widget-markdown 2.16.0-beta.0

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.
Files changed (215) hide show
  1. package/CHANGELOG.md +820 -0
  2. package/LICENSE +22 -0
  3. package/README.md +9 -0
  4. package/dist/decap-cms-widget-markdown.js +9 -0
  5. package/dist/decap-cms-widget-markdown.js.LICENSE.txt +37 -0
  6. package/dist/decap-cms-widget-markdown.js.map +1 -0
  7. package/dist/esm/MarkdownControl/RawEditor.js +106 -0
  8. package/dist/esm/MarkdownControl/Toolbar.js +216 -0
  9. package/dist/esm/MarkdownControl/ToolbarButton.js +43 -0
  10. package/dist/esm/MarkdownControl/VisualEditor.js +268 -0
  11. package/dist/esm/MarkdownControl/components/Shortcode.js +78 -0
  12. package/dist/esm/MarkdownControl/components/VoidBlock.js +55 -0
  13. package/dist/esm/MarkdownControl/index.js +129 -0
  14. package/dist/esm/MarkdownControl/plugins/BreakToDefaultBlock.js +34 -0
  15. package/dist/esm/MarkdownControl/plugins/CloseBlock.js +36 -0
  16. package/dist/esm/MarkdownControl/plugins/CommandsAndQueries.js +188 -0
  17. package/dist/esm/MarkdownControl/plugins/ForceInsert.js +49 -0
  18. package/dist/esm/MarkdownControl/plugins/Hotkey.js +35 -0
  19. package/dist/esm/MarkdownControl/plugins/LineBreak.js +21 -0
  20. package/dist/esm/MarkdownControl/plugins/Link.js +56 -0
  21. package/dist/esm/MarkdownControl/plugins/blocks/defaultEmptyBlock.js +16 -0
  22. package/dist/esm/MarkdownControl/plugins/blocks/events/keyDown.js +47 -0
  23. package/dist/esm/MarkdownControl/plugins/blocks/events/keyDownBackspace.js +26 -0
  24. package/dist/esm/MarkdownControl/plugins/blocks/events/keyDownEnter.js +28 -0
  25. package/dist/esm/MarkdownControl/plugins/blocks/events/toggleBlock.js +46 -0
  26. package/dist/esm/MarkdownControl/plugins/blocks/locations/areCurrentAndPreviousBlocksOfType.js +20 -0
  27. package/dist/esm/MarkdownControl/plugins/blocks/locations/getListTypeAtCursor.js +16 -0
  28. package/dist/esm/MarkdownControl/plugins/blocks/locations/isCursorAtEndOfParagraph.js +19 -0
  29. package/dist/esm/MarkdownControl/plugins/blocks/locations/isCursorAtStartOfBlockType.js +19 -0
  30. package/dist/esm/MarkdownControl/plugins/blocks/locations/isCursorAtStartOfNonEmptyHeading.js +20 -0
  31. package/dist/esm/MarkdownControl/plugins/blocks/locations/isCursorInBlockType.js +20 -0
  32. package/dist/esm/MarkdownControl/plugins/blocks/locations/isCursorInNonDefaultBlock.js +20 -0
  33. package/dist/esm/MarkdownControl/plugins/blocks/transforms/splitIntoParagraph.js +23 -0
  34. package/dist/esm/MarkdownControl/plugins/blocks/transforms/unwrapIfCursorAtStart.js +36 -0
  35. package/dist/esm/MarkdownControl/plugins/blocks/transforms/wrapListItemsInBlock.js +27 -0
  36. package/dist/esm/MarkdownControl/plugins/blocks/withBlocks.js +19 -0
  37. package/dist/esm/MarkdownControl/plugins/inlines/events/keyDown.js +39 -0
  38. package/dist/esm/MarkdownControl/plugins/inlines/events/keyDownShiftEnter.js +27 -0
  39. package/dist/esm/MarkdownControl/plugins/inlines/events/toggleLink.js +24 -0
  40. package/dist/esm/MarkdownControl/plugins/inlines/events/toggleMark.js +18 -0
  41. package/dist/esm/MarkdownControl/plugins/inlines/locations/isMarkActive.js +17 -0
  42. package/dist/esm/MarkdownControl/plugins/inlines/selectors/getActiveLink.js +15 -0
  43. package/dist/esm/MarkdownControl/plugins/inlines/transforms/unwrapLink.js +14 -0
  44. package/dist/esm/MarkdownControl/plugins/inlines/transforms/wrapLink.js +45 -0
  45. package/dist/esm/MarkdownControl/plugins/inlines/withInlines.js +23 -0
  46. package/dist/esm/MarkdownControl/plugins/lists/events/keyDown.js +38 -0
  47. package/dist/esm/MarkdownControl/plugins/lists/events/keyDownBackspace.js +29 -0
  48. package/dist/esm/MarkdownControl/plugins/lists/events/keyDownEnter.js +44 -0
  49. package/dist/esm/MarkdownControl/plugins/lists/events/keyDownShiftTab.js +32 -0
  50. package/dist/esm/MarkdownControl/plugins/lists/events/keyDownTab.js +57 -0
  51. package/dist/esm/MarkdownControl/plugins/lists/events/toggleListType.js +29 -0
  52. package/dist/esm/MarkdownControl/plugins/lists/locations/isCursorAtListItemStart.js +18 -0
  53. package/dist/esm/MarkdownControl/plugins/lists/locations/isCursorAtNoninitialParagraphStart.js +17 -0
  54. package/dist/esm/MarkdownControl/plugins/lists/locations/isCursorInItemContainingNestedList.js +14 -0
  55. package/dist/esm/MarkdownControl/plugins/lists/locations/isCursorInListItem.js +20 -0
  56. package/dist/esm/MarkdownControl/plugins/lists/locations/isSelectionWithinNoninitialListItem.js +18 -0
  57. package/dist/esm/MarkdownControl/plugins/lists/selectors/getListContainedInListItem.js +18 -0
  58. package/dist/esm/MarkdownControl/plugins/lists/selectors/getLowestAncestorList.js +15 -0
  59. package/dist/esm/MarkdownControl/plugins/lists/selectors/getLowestAncestorQuote.js +15 -0
  60. package/dist/esm/MarkdownControl/plugins/lists/transforms/changeListType.js +24 -0
  61. package/dist/esm/MarkdownControl/plugins/lists/transforms/convertParagraphToListItem.js +27 -0
  62. package/dist/esm/MarkdownControl/plugins/lists/transforms/liftFirstMatchedParent.js +20 -0
  63. package/dist/esm/MarkdownControl/plugins/lists/transforms/liftListItem.js +37 -0
  64. package/dist/esm/MarkdownControl/plugins/lists/transforms/mergeWithPreviousListItem.js +19 -0
  65. package/dist/esm/MarkdownControl/plugins/lists/transforms/moveListToListItem.js +18 -0
  66. package/dist/esm/MarkdownControl/plugins/lists/transforms/splitListItem.js +37 -0
  67. package/dist/esm/MarkdownControl/plugins/lists/transforms/splitToNestedList.js +37 -0
  68. package/dist/esm/MarkdownControl/plugins/lists/transforms/unwrapFirstMatchedParent.js +14 -0
  69. package/dist/esm/MarkdownControl/plugins/lists/transforms/unwrapSelectionFromList.js +25 -0
  70. package/dist/esm/MarkdownControl/plugins/lists/transforms/wrapFirstMatchedParent.js +14 -0
  71. package/dist/esm/MarkdownControl/plugins/lists/transforms/wrapSelectionInList.js +27 -0
  72. package/dist/esm/MarkdownControl/plugins/lists/withLists.js +67 -0
  73. package/dist/esm/MarkdownControl/plugins/matchers/lowestMatchedAncestor.js +13 -0
  74. package/dist/esm/MarkdownControl/plugins/matchers/matchLink.js +14 -0
  75. package/dist/esm/MarkdownControl/plugins/matchers/matchedAncestors.js +15 -0
  76. package/dist/esm/MarkdownControl/plugins/shortcodes/insertShortcode.js +35 -0
  77. package/dist/esm/MarkdownControl/plugins/shortcodes/locations/isCursorInEmptyParagraph.js +20 -0
  78. package/dist/esm/MarkdownControl/plugins/shortcodes/withShortcodes.js +30 -0
  79. package/dist/esm/MarkdownControl/plugins/util.js +18 -0
  80. package/dist/esm/MarkdownControl/renderers.js +326 -0
  81. package/dist/esm/MarkdownPreview.js +47 -0
  82. package/dist/esm/index.js +31 -0
  83. package/dist/esm/regexHelper.js +151 -0
  84. package/dist/esm/schema.js +35 -0
  85. package/dist/esm/serializers/index.js +233 -0
  86. package/dist/esm/serializers/rehypePaperEmoji.js +24 -0
  87. package/dist/esm/serializers/remarkAllowHtmlEntities.js +55 -0
  88. package/dist/esm/serializers/remarkAssertParents.js +89 -0
  89. package/dist/esm/serializers/remarkEscapeMarkdownEntities.js +271 -0
  90. package/dist/esm/serializers/remarkImagesToText.js +41 -0
  91. package/dist/esm/serializers/remarkPaddedLinks.js +127 -0
  92. package/dist/esm/serializers/remarkRehypeShortcodes.js +93 -0
  93. package/dist/esm/serializers/remarkShortcodes.js +123 -0
  94. package/dist/esm/serializers/remarkSlate.js +518 -0
  95. package/dist/esm/serializers/remarkSquashReferences.js +88 -0
  96. package/dist/esm/serializers/remarkStripTrailingBreaks.js +61 -0
  97. package/dist/esm/serializers/remarkWrapHtml.js +24 -0
  98. package/dist/esm/serializers/slateRemark.js +512 -0
  99. package/dist/esm/styles.js +18 -0
  100. package/dist/esm/types.js +10 -0
  101. package/package.json +64 -0
  102. package/src/MarkdownControl/RawEditor.js +100 -0
  103. package/src/MarkdownControl/Toolbar.js +278 -0
  104. package/src/MarkdownControl/ToolbarButton.js +48 -0
  105. package/src/MarkdownControl/VisualEditor.js +293 -0
  106. package/src/MarkdownControl/__tests__/VisualEditor.spec.js +56 -0
  107. package/src/MarkdownControl/__tests__/__snapshots__/parser.spec.js.snap +543 -0
  108. package/src/MarkdownControl/__tests__/parser.spec.js +668 -0
  109. package/src/MarkdownControl/components/Shortcode.js +74 -0
  110. package/src/MarkdownControl/components/VoidBlock.js +58 -0
  111. package/src/MarkdownControl/index.js +125 -0
  112. package/src/MarkdownControl/plugins/BreakToDefaultBlock.js +23 -0
  113. package/src/MarkdownControl/plugins/CloseBlock.js +25 -0
  114. package/src/MarkdownControl/plugins/CommandsAndQueries.js +156 -0
  115. package/src/MarkdownControl/plugins/ForceInsert.js +38 -0
  116. package/src/MarkdownControl/plugins/Hotkey.js +29 -0
  117. package/src/MarkdownControl/plugins/LineBreak.js +15 -0
  118. package/src/MarkdownControl/plugins/Link.js +40 -0
  119. package/src/MarkdownControl/plugins/blocks/defaultEmptyBlock.js +8 -0
  120. package/src/MarkdownControl/plugins/blocks/events/keyDown.js +47 -0
  121. package/src/MarkdownControl/plugins/blocks/events/keyDownBackspace.js +27 -0
  122. package/src/MarkdownControl/plugins/blocks/events/keyDownEnter.js +26 -0
  123. package/src/MarkdownControl/plugins/blocks/events/toggleBlock.js +39 -0
  124. package/src/MarkdownControl/plugins/blocks/locations/areCurrentAndPreviousBlocksOfType.js +15 -0
  125. package/src/MarkdownControl/plugins/blocks/locations/getListTypeAtCursor.js +11 -0
  126. package/src/MarkdownControl/plugins/blocks/locations/isCursorAtEndOfParagraph.js +14 -0
  127. package/src/MarkdownControl/plugins/blocks/locations/isCursorAtStartOfBlockType.js +14 -0
  128. package/src/MarkdownControl/plugins/blocks/locations/isCursorAtStartOfNonEmptyHeading.js +22 -0
  129. package/src/MarkdownControl/plugins/blocks/locations/isCursorInBlockType.js +27 -0
  130. package/src/MarkdownControl/plugins/blocks/locations/isCursorInNonDefaultBlock.js +17 -0
  131. package/src/MarkdownControl/plugins/blocks/transforms/splitIntoParagraph.js +13 -0
  132. package/src/MarkdownControl/plugins/blocks/transforms/unwrapIfCursorAtStart.js +30 -0
  133. package/src/MarkdownControl/plugins/blocks/transforms/wrapListItemsInBlock.js +11 -0
  134. package/src/MarkdownControl/plugins/blocks/withBlocks.js +15 -0
  135. package/src/MarkdownControl/plugins/inlines/events/keyDown.js +38 -0
  136. package/src/MarkdownControl/plugins/inlines/events/keyDownShiftEnter.js +28 -0
  137. package/src/MarkdownControl/plugins/inlines/events/toggleLink.js +17 -0
  138. package/src/MarkdownControl/plugins/inlines/events/toggleMark.js +13 -0
  139. package/src/MarkdownControl/plugins/inlines/locations/isMarkActive.js +11 -0
  140. package/src/MarkdownControl/plugins/inlines/selectors/getActiveLink.js +10 -0
  141. package/src/MarkdownControl/plugins/inlines/transforms/unwrapLink.js +9 -0
  142. package/src/MarkdownControl/plugins/inlines/transforms/wrapLink.js +30 -0
  143. package/src/MarkdownControl/plugins/inlines/withInlines.js +20 -0
  144. package/src/MarkdownControl/plugins/lists/events/keyDown.js +34 -0
  145. package/src/MarkdownControl/plugins/lists/events/keyDownBackspace.js +31 -0
  146. package/src/MarkdownControl/plugins/lists/events/keyDownEnter.js +39 -0
  147. package/src/MarkdownControl/plugins/lists/events/keyDownShiftTab.js +27 -0
  148. package/src/MarkdownControl/plugins/lists/events/keyDownTab.js +55 -0
  149. package/src/MarkdownControl/plugins/lists/events/toggleListType.js +23 -0
  150. package/src/MarkdownControl/plugins/lists/locations/isCursorAtListItemStart.js +13 -0
  151. package/src/MarkdownControl/plugins/lists/locations/isCursorAtNoninitialParagraphStart.js +11 -0
  152. package/src/MarkdownControl/plugins/lists/locations/isCursorInItemContainingNestedList.js +9 -0
  153. package/src/MarkdownControl/plugins/lists/locations/isCursorInListItem.js +21 -0
  154. package/src/MarkdownControl/plugins/lists/locations/isSelectionWithinNoninitialListItem.js +14 -0
  155. package/src/MarkdownControl/plugins/lists/selectors/getListContainedInListItem.js +12 -0
  156. package/src/MarkdownControl/plugins/lists/selectors/getLowestAncestorList.js +11 -0
  157. package/src/MarkdownControl/plugins/lists/selectors/getLowestAncestorQuote.js +11 -0
  158. package/src/MarkdownControl/plugins/lists/transforms/changeListType.js +16 -0
  159. package/src/MarkdownControl/plugins/lists/transforms/convertParagraphToListItem.js +19 -0
  160. package/src/MarkdownControl/plugins/lists/transforms/liftFirstMatchedParent.js +11 -0
  161. package/src/MarkdownControl/plugins/lists/transforms/liftListItem.js +30 -0
  162. package/src/MarkdownControl/plugins/lists/transforms/mergeWithPreviousListItem.js +13 -0
  163. package/src/MarkdownControl/plugins/lists/transforms/moveListToListItem.js +13 -0
  164. package/src/MarkdownControl/plugins/lists/transforms/splitListItem.js +33 -0
  165. package/src/MarkdownControl/plugins/lists/transforms/splitToNestedList.js +33 -0
  166. package/src/MarkdownControl/plugins/lists/transforms/unwrapFirstMatchedParent.js +9 -0
  167. package/src/MarkdownControl/plugins/lists/transforms/unwrapSelectionFromList.js +14 -0
  168. package/src/MarkdownControl/plugins/lists/transforms/wrapFirstMatchedParent.js +9 -0
  169. package/src/MarkdownControl/plugins/lists/transforms/wrapSelectionInList.js +17 -0
  170. package/src/MarkdownControl/plugins/lists/withLists.js +66 -0
  171. package/src/MarkdownControl/plugins/matchers/lowestMatchedAncestor.js +6 -0
  172. package/src/MarkdownControl/plugins/matchers/matchLink.js +9 -0
  173. package/src/MarkdownControl/plugins/matchers/matchedAncestors.js +18 -0
  174. package/src/MarkdownControl/plugins/shortcodes/insertShortcode.js +34 -0
  175. package/src/MarkdownControl/plugins/shortcodes/locations/isCursorInEmptyParagraph.js +17 -0
  176. package/src/MarkdownControl/plugins/shortcodes/withShortcodes.js +26 -0
  177. package/src/MarkdownControl/plugins/util.js +11 -0
  178. package/src/MarkdownControl/renderers.js +352 -0
  179. package/src/MarkdownPreview.js +27 -0
  180. package/src/__tests__/__snapshots__/renderer.spec.js.snap +239 -0
  181. package/src/__tests__/renderer.spec.js +261 -0
  182. package/src/index.js +16 -0
  183. package/src/regexHelper.js +137 -0
  184. package/src/schema.js +35 -0
  185. package/src/serializers/__tests__/__fixtures__/commonmarkExpected.json +625 -0
  186. package/src/serializers/__tests__/__fixtures__/duplicate_marks_github_issue_3280.md +1 -0
  187. package/src/serializers/__tests__/commonmark.spec.js +110 -0
  188. package/src/serializers/__tests__/index.spec.js +52 -0
  189. package/src/serializers/__tests__/remarkAllowHtmlEntities.spec.js +25 -0
  190. package/src/serializers/__tests__/remarkAssertParents.spec.js +171 -0
  191. package/src/serializers/__tests__/remarkEscapeMarkdownEntities.spec.js +84 -0
  192. package/src/serializers/__tests__/remarkPaddedLinks.spec.js +43 -0
  193. package/src/serializers/__tests__/remarkPlugins.spec.js +299 -0
  194. package/src/serializers/__tests__/remarkShortcodes.spec.js +106 -0
  195. package/src/serializers/__tests__/remarkSlate.spec.js +67 -0
  196. package/src/serializers/__tests__/remarkStripTrailingBreaks.spec.js +23 -0
  197. package/src/serializers/__tests__/slate.spec.js +300 -0
  198. package/src/serializers/index.js +226 -0
  199. package/src/serializers/rehypePaperEmoji.js +16 -0
  200. package/src/serializers/remarkAllowHtmlEntities.js +58 -0
  201. package/src/serializers/remarkAssertParents.js +83 -0
  202. package/src/serializers/remarkEscapeMarkdownEntities.js +269 -0
  203. package/src/serializers/remarkImagesToText.js +26 -0
  204. package/src/serializers/remarkPaddedLinks.js +120 -0
  205. package/src/serializers/remarkRehypeShortcodes.js +67 -0
  206. package/src/serializers/remarkShortcodes.js +106 -0
  207. package/src/serializers/remarkSlate.js +425 -0
  208. package/src/serializers/remarkSquashReferences.js +73 -0
  209. package/src/serializers/remarkStripTrailingBreaks.js +56 -0
  210. package/src/serializers/remarkWrapHtml.js +20 -0
  211. package/src/serializers/slateRemark.js +468 -0
  212. package/src/styles.js +13 -0
  213. package/src/types.js +3 -0
  214. package/test-helpers/h.js +32 -0
  215. package/webpack.config.js +3 -0
@@ -0,0 +1,100 @@
1
+ import React, { useEffect, useMemo, useState } from 'react';
2
+ import PropTypes from 'prop-types';
3
+ import ImmutablePropTypes from 'react-immutable-proptypes';
4
+ import { ClassNames } from '@emotion/core';
5
+ import styled from '@emotion/styled';
6
+ import { lengths, fonts } from 'decap-cms-ui-default';
7
+ import { createEditor } from 'slate';
8
+ import { Editable, ReactEditor, Slate, withReact } from 'slate-react';
9
+ import { withHistory } from 'slate-history';
10
+
11
+ import { editorStyleVars, EditorControlBar } from '../styles';
12
+ import Toolbar from './Toolbar';
13
+ import defaultEmptyBlock from './plugins/blocks/defaultEmptyBlock';
14
+
15
+ function rawEditorStyles({ minimal }) {
16
+ return `
17
+ position: relative;
18
+ overflow: hidden;
19
+ overflow-x: auto;
20
+ min-height: ${minimal ? 'auto' : lengths.richTextEditorMinHeight};
21
+ font-family: ${fonts.mono};
22
+ border-top-left-radius: 0;
23
+ border-top-right-radius: 0;
24
+ border-top: 0;
25
+ margin-top: -${editorStyleVars.stickyDistanceBottom};
26
+ `;
27
+ }
28
+
29
+ const RawEditorContainer = styled.div`
30
+ position: relative;
31
+ `;
32
+ function RawEditor(props) {
33
+ const { className, field, isShowModeToggle, t, onChange } = props;
34
+
35
+ const editor = useMemo(() => withReact(withHistory(createEditor())), []);
36
+
37
+ const [value, setValue] = useState(
38
+ props.value
39
+ ? props.value.split('\n').map(line => defaultEmptyBlock(line))
40
+ : [defaultEmptyBlock()],
41
+ );
42
+
43
+ useEffect(() => {
44
+ if (props.pendingFocus) {
45
+ ReactEditor.focus(editor);
46
+ }
47
+ }, []);
48
+
49
+ function handleToggleMode() {
50
+ props.onMode('rich_text');
51
+ }
52
+
53
+ function handleChange(value) {
54
+ onChange(value.map(line => line.children[0].text).join('\n'));
55
+ setValue(value);
56
+ }
57
+
58
+ return (
59
+ <Slate editor={editor} value={value} onChange={handleChange}>
60
+ <RawEditorContainer>
61
+ <EditorControlBar>
62
+ <Toolbar
63
+ onToggleMode={handleToggleMode}
64
+ buttons={field.get('buttons')}
65
+ disabled
66
+ rawMode
67
+ isShowModeToggle={isShowModeToggle}
68
+ t={t}
69
+ />
70
+ </EditorControlBar>
71
+ <ClassNames>
72
+ {({ css, cx }) => (
73
+ <Editable
74
+ className={cx(
75
+ className,
76
+ css`
77
+ ${rawEditorStyles({ minimal: field.get('minimal') })}
78
+ `,
79
+ )}
80
+ value={value}
81
+ onChange={handleChange}
82
+ />
83
+ )}
84
+ </ClassNames>
85
+ </RawEditorContainer>
86
+ </Slate>
87
+ );
88
+ }
89
+
90
+ RawEditor.propTypes = {
91
+ onChange: PropTypes.func.isRequired,
92
+ onMode: PropTypes.func.isRequired,
93
+ className: PropTypes.string.isRequired,
94
+ value: PropTypes.string,
95
+ field: ImmutablePropTypes.map.isRequired,
96
+ isShowModeToggle: PropTypes.bool.isRequired,
97
+ t: PropTypes.func.isRequired,
98
+ };
99
+
100
+ export default RawEditor;
@@ -0,0 +1,278 @@
1
+ import React from 'react';
2
+ import PropTypes from 'prop-types';
3
+ import ImmutablePropTypes from 'react-immutable-proptypes';
4
+ import styled from '@emotion/styled';
5
+ import { css } from '@emotion/core';
6
+ import { List } from 'immutable';
7
+ import {
8
+ Toggle,
9
+ Dropdown,
10
+ DropdownItem,
11
+ DropdownButton,
12
+ colors,
13
+ transitions,
14
+ lengths,
15
+ } from 'decap-cms-ui-default';
16
+
17
+ import ToolbarButton from './ToolbarButton';
18
+
19
+ const ToolbarContainer = styled.div`
20
+ background-color: ${colors.textFieldBorder};
21
+ border-top-right-radius: ${lengths.borderRadius};
22
+ position: relative;
23
+ display: flex;
24
+ justify-content: space-between;
25
+ align-items: center;
26
+ padding: 11px 14px;
27
+ min-height: 58px;
28
+ transition: background-color ${transitions.main}, color ${transitions.main};
29
+ color: ${colors.text};
30
+ `;
31
+
32
+ const ToolbarDropdownWrapper = styled.div`
33
+ display: inline-block;
34
+ position: relative;
35
+ `;
36
+
37
+ const ToolbarToggle = styled.div`
38
+ flex-shrink: 0;
39
+ display: flex;
40
+ align-items: center;
41
+ font-size: 14px;
42
+ margin: 0 10px;
43
+ `;
44
+
45
+ const StyledToggle = ToolbarToggle.withComponent(Toggle);
46
+
47
+ const ToolbarToggleLabel = styled.span`
48
+ display: inline-block;
49
+ text-align: center;
50
+ white-space: nowrap;
51
+ line-height: 20px;
52
+ min-width: ${props => (props.offPosition ? '62px' : '70px')};
53
+
54
+ ${props =>
55
+ props.isActive &&
56
+ css`
57
+ font-weight: 600;
58
+ color: ${colors.active};
59
+ `};
60
+ `;
61
+
62
+ export default class Toolbar extends React.Component {
63
+ static propTypes = {
64
+ buttons: ImmutablePropTypes.list,
65
+ editorComponents: ImmutablePropTypes.list,
66
+ onToggleMode: PropTypes.func.isRequired,
67
+ rawMode: PropTypes.bool,
68
+ isShowModeToggle: PropTypes.bool.isRequired,
69
+ plugins: ImmutablePropTypes.map,
70
+ onSubmit: PropTypes.func,
71
+ onAddAsset: PropTypes.func,
72
+ getAsset: PropTypes.func,
73
+ disabled: PropTypes.bool,
74
+ onMarkClick: PropTypes.func,
75
+ onBlockClick: PropTypes.func,
76
+ onLinkClick: PropTypes.func,
77
+ hasMark: PropTypes.func,
78
+ hasInline: PropTypes.func,
79
+ hasBlock: PropTypes.func,
80
+ t: PropTypes.func.isRequired,
81
+ };
82
+
83
+ isVisible = button => {
84
+ const { buttons } = this.props;
85
+ return !List.isList(buttons) || buttons.includes(button);
86
+ };
87
+
88
+ handleBlockClick = (event, type) => {
89
+ if (event) {
90
+ event.preventDefault();
91
+ }
92
+ this.props.onBlockClick(type);
93
+ };
94
+
95
+ handleMarkClick = (event, type) => {
96
+ event.preventDefault();
97
+ this.props.onMarkClick(type);
98
+ };
99
+
100
+ render() {
101
+ const {
102
+ onLinkClick,
103
+ onToggleMode,
104
+ rawMode,
105
+ isShowModeToggle,
106
+ plugins,
107
+ disabled,
108
+ onSubmit,
109
+ hasMark = () => {},
110
+ hasInline = () => {},
111
+ hasBlock = () => {},
112
+ hasQuote = () => {},
113
+ hasListItems = () => {},
114
+ editorComponents,
115
+ t,
116
+ } = this.props;
117
+ const isVisible = this.isVisible;
118
+ const showEditorComponents = !editorComponents || editorComponents.size >= 1;
119
+
120
+ function showPlugin({ id }) {
121
+ return editorComponents ? editorComponents.includes(id) : true;
122
+ }
123
+
124
+ const pluginsList = plugins ? plugins.toList().filter(showPlugin) : List();
125
+
126
+ const headingOptions = {
127
+ 'heading-one': t('editor.editorWidgets.headingOptions.headingOne'),
128
+ 'heading-two': t('editor.editorWidgets.headingOptions.headingTwo'),
129
+ 'heading-three': t('editor.editorWidgets.headingOptions.headingThree'),
130
+ 'heading-four': t('editor.editorWidgets.headingOptions.headingFour'),
131
+ 'heading-five': t('editor.editorWidgets.headingOptions.headingFive'),
132
+ 'heading-six': t('editor.editorWidgets.headingOptions.headingSix'),
133
+ };
134
+
135
+ return (
136
+ <ToolbarContainer>
137
+ <div>
138
+ {isVisible('bold') && (
139
+ <ToolbarButton
140
+ type="bold"
141
+ label={t('editor.editorWidgets.markdown.bold')}
142
+ icon="bold"
143
+ onClick={this.handleMarkClick}
144
+ isActive={hasMark('bold')}
145
+ disabled={disabled}
146
+ />
147
+ )}
148
+ {isVisible('italic') && (
149
+ <ToolbarButton
150
+ type="italic"
151
+ label={t('editor.editorWidgets.markdown.italic')}
152
+ icon="italic"
153
+ onClick={this.handleMarkClick}
154
+ isActive={hasMark('italic')}
155
+ disabled={disabled}
156
+ />
157
+ )}
158
+ {isVisible('code') && (
159
+ <ToolbarButton
160
+ type="code"
161
+ label={t('editor.editorWidgets.markdown.code')}
162
+ icon="code"
163
+ onClick={this.handleMarkClick}
164
+ isActive={hasMark('code')}
165
+ disabled={disabled}
166
+ />
167
+ )}
168
+ {isVisible('link') && (
169
+ <ToolbarButton
170
+ type="link"
171
+ label={t('editor.editorWidgets.markdown.link')}
172
+ icon="link"
173
+ onClick={onLinkClick}
174
+ isActive={hasInline('link')}
175
+ disabled={disabled}
176
+ />
177
+ )}
178
+ {/* Show dropdown if at least one heading is not hidden */}
179
+ {Object.keys(headingOptions).some(isVisible) && (
180
+ <ToolbarDropdownWrapper>
181
+ <Dropdown
182
+ dropdownWidth="max-content"
183
+ dropdownTopOverlap="36px"
184
+ renderButton={() => (
185
+ <DropdownButton>
186
+ <ToolbarButton
187
+ type="headings"
188
+ label={t('editor.editorWidgets.markdown.headings')}
189
+ icon="hOptions"
190
+ disabled={disabled}
191
+ isActive={!disabled && Object.keys(headingOptions).some(hasBlock)}
192
+ />
193
+ </DropdownButton>
194
+ )}
195
+ >
196
+ {!disabled &&
197
+ Object.keys(headingOptions).map(
198
+ (optionKey, idx) =>
199
+ isVisible(optionKey) && (
200
+ <DropdownItem
201
+ key={idx}
202
+ label={headingOptions[optionKey]}
203
+ className={hasBlock(optionKey) ? 'active' : ''}
204
+ onClick={() => this.handleBlockClick(null, optionKey)}
205
+ />
206
+ ),
207
+ )}
208
+ </Dropdown>
209
+ </ToolbarDropdownWrapper>
210
+ )}
211
+ {isVisible('quote') && (
212
+ <ToolbarButton
213
+ type="quote"
214
+ label={t('editor.editorWidgets.markdown.quote')}
215
+ icon="quote"
216
+ onClick={this.handleBlockClick}
217
+ isActive={hasQuote('quote')}
218
+ disabled={disabled}
219
+ />
220
+ )}
221
+ {isVisible('bulleted-list') && (
222
+ <ToolbarButton
223
+ type="bulleted-list"
224
+ label={t('editor.editorWidgets.markdown.bulletedList')}
225
+ icon="list-bulleted"
226
+ onClick={this.handleBlockClick}
227
+ isActive={hasListItems('bulleted-list')}
228
+ disabled={disabled}
229
+ />
230
+ )}
231
+ {isVisible('numbered-list') && (
232
+ <ToolbarButton
233
+ type="numbered-list"
234
+ label={t('editor.editorWidgets.markdown.numberedList')}
235
+ icon="list-numbered"
236
+ onClick={this.handleBlockClick}
237
+ isActive={hasListItems('numbered-list')}
238
+ disabled={disabled}
239
+ />
240
+ )}
241
+ {showEditorComponents && (
242
+ <ToolbarDropdownWrapper>
243
+ <Dropdown
244
+ dropdownTopOverlap="36px"
245
+ dropdownWidth="max-content"
246
+ renderButton={() => (
247
+ <DropdownButton>
248
+ <ToolbarButton
249
+ label={t('editor.editorWidgets.markdown.addComponent')}
250
+ icon="add-with"
251
+ onClick={this.handleComponentsMenuToggle}
252
+ disabled={disabled}
253
+ />
254
+ </DropdownButton>
255
+ )}
256
+ >
257
+ {pluginsList.map((plugin, idx) => (
258
+ <DropdownItem key={idx} label={plugin.label} onClick={() => onSubmit(plugin)} />
259
+ ))}
260
+ </Dropdown>
261
+ </ToolbarDropdownWrapper>
262
+ )}
263
+ </div>
264
+ {isShowModeToggle && (
265
+ <ToolbarToggle>
266
+ <ToolbarToggleLabel isActive={!rawMode} offPosition>
267
+ {t('editor.editorWidgets.markdown.richText')}
268
+ </ToolbarToggleLabel>
269
+ <StyledToggle active={rawMode} onChange={onToggleMode} />
270
+ <ToolbarToggleLabel isActive={rawMode}>
271
+ {t('editor.editorWidgets.markdown.markdown')}
272
+ </ToolbarToggleLabel>
273
+ </ToolbarToggle>
274
+ )}
275
+ </ToolbarContainer>
276
+ );
277
+ }
278
+ }
@@ -0,0 +1,48 @@
1
+ import React from 'react';
2
+ import PropTypes from 'prop-types';
3
+ import styled from '@emotion/styled';
4
+ import { Icon, buttons } from 'decap-cms-ui-default';
5
+
6
+ const StyledToolbarButton = styled.button`
7
+ ${buttons.button};
8
+ display: inline-block;
9
+ padding: 6px;
10
+ border: none;
11
+ background-color: transparent;
12
+ font-size: 16px;
13
+ color: ${props => (props.isActive ? '#1e2532' : 'inherit')};
14
+ cursor: pointer;
15
+
16
+ &:disabled {
17
+ cursor: auto;
18
+ opacity: 0.5;
19
+ }
20
+
21
+ ${Icon} {
22
+ display: block;
23
+ }
24
+ `;
25
+
26
+ function ToolbarButton({ type, label, icon, onClick, isActive, disabled }) {
27
+ return (
28
+ <StyledToolbarButton
29
+ isActive={isActive}
30
+ onClick={e => onClick && onClick(e, type)}
31
+ title={label}
32
+ disabled={disabled}
33
+ >
34
+ {icon ? <Icon type={icon} /> : label}
35
+ </StyledToolbarButton>
36
+ );
37
+ }
38
+
39
+ ToolbarButton.propTypes = {
40
+ type: PropTypes.string,
41
+ label: PropTypes.string.isRequired,
42
+ icon: PropTypes.string,
43
+ onClick: PropTypes.func,
44
+ isActive: PropTypes.bool,
45
+ disabled: PropTypes.bool,
46
+ };
47
+
48
+ export default ToolbarButton;