@pranaysahith/decap-cms-core 3.9.1

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 (299) hide show
  1. package/README.md +9 -0
  2. package/dist/@pranaysahith/decap-cms-core.js +52 -0
  3. package/dist/@pranaysahith/decap-cms-core.js.LICENSE.txt +141 -0
  4. package/dist/@pranaysahith/decap-cms-core.js.map +1 -0
  5. package/dist/decap-cms-core.js +47 -0
  6. package/dist/decap-cms-core.js.LICENSE.txt +116 -0
  7. package/dist/decap-cms-core.js.map +1 -0
  8. package/dist/esm/actions/auth.js +97 -0
  9. package/dist/esm/actions/collections.js +15 -0
  10. package/dist/esm/actions/config.js +493 -0
  11. package/dist/esm/actions/deploys.js +79 -0
  12. package/dist/esm/actions/editorialWorkflow.js +480 -0
  13. package/dist/esm/actions/entries.js +865 -0
  14. package/dist/esm/actions/media.js +147 -0
  15. package/dist/esm/actions/mediaLibrary.js +552 -0
  16. package/dist/esm/actions/notifications.js +21 -0
  17. package/dist/esm/actions/search.js +149 -0
  18. package/dist/esm/actions/status.js +74 -0
  19. package/dist/esm/actions/waitUntil.js +32 -0
  20. package/dist/esm/backend.js +1082 -0
  21. package/dist/esm/bootstrap.js +101 -0
  22. package/dist/esm/components/App/App.js +289 -0
  23. package/dist/esm/components/App/Header.js +172 -0
  24. package/dist/esm/components/App/NotFoundPage.js +19 -0
  25. package/dist/esm/components/Collection/Collection.js +198 -0
  26. package/dist/esm/components/Collection/CollectionControls.js +46 -0
  27. package/dist/esm/components/Collection/CollectionSearch.js +222 -0
  28. package/dist/esm/components/Collection/CollectionTop.js +68 -0
  29. package/dist/esm/components/Collection/ControlButton.js +17 -0
  30. package/dist/esm/components/Collection/Entries/Entries.js +73 -0
  31. package/dist/esm/components/Collection/Entries/EntriesCollection.js +241 -0
  32. package/dist/esm/components/Collection/Entries/EntriesSearch.js +113 -0
  33. package/dist/esm/components/Collection/Entries/EntryCard.js +177 -0
  34. package/dist/esm/components/Collection/Entries/EntryListing.js +143 -0
  35. package/dist/esm/components/Collection/FilterControl.js +33 -0
  36. package/dist/esm/components/Collection/FolderRenameControl.js +403 -0
  37. package/dist/esm/components/Collection/GroupControl.js +33 -0
  38. package/dist/esm/components/Collection/NestedCollection.js +308 -0
  39. package/dist/esm/components/Collection/Sidebar.js +91 -0
  40. package/dist/esm/components/Collection/SortControl.js +59 -0
  41. package/dist/esm/components/Collection/ViewStyleControl.js +38 -0
  42. package/dist/esm/components/Editor/Editor.js +466 -0
  43. package/dist/esm/components/Editor/EditorControlPane/EditorControl.js +395 -0
  44. package/dist/esm/components/Editor/EditorControlPane/EditorControlPane.js +254 -0
  45. package/dist/esm/components/Editor/EditorControlPane/Widget.js +374 -0
  46. package/dist/esm/components/Editor/EditorInterface.js +386 -0
  47. package/dist/esm/components/Editor/EditorPreviewPane/EditorPreview.js +47 -0
  48. package/dist/esm/components/Editor/EditorPreviewPane/EditorPreviewContent.js +66 -0
  49. package/dist/esm/components/Editor/EditorPreviewPane/EditorPreviewPane.js +288 -0
  50. package/dist/esm/components/Editor/EditorPreviewPane/PreviewHOC.js +27 -0
  51. package/dist/esm/components/Editor/EditorToolbar.js +536 -0
  52. package/dist/esm/components/Editor/EntryPathEditor.js +272 -0
  53. package/dist/esm/components/Editor/withWorkflow.js +56 -0
  54. package/dist/esm/components/EditorWidgets/Unknown/UnknownControl.js +18 -0
  55. package/dist/esm/components/EditorWidgets/Unknown/UnknownPreview.js +20 -0
  56. package/dist/esm/components/EditorWidgets/index.js +4 -0
  57. package/dist/esm/components/MediaLibrary/EmptyMessage.js +22 -0
  58. package/dist/esm/components/MediaLibrary/MediaLibrary.js +446 -0
  59. package/dist/esm/components/MediaLibrary/MediaLibraryButtons.js +93 -0
  60. package/dist/esm/components/MediaLibrary/MediaLibraryCard.js +99 -0
  61. package/dist/esm/components/MediaLibrary/MediaLibraryCardGrid.js +198 -0
  62. package/dist/esm/components/MediaLibrary/MediaLibraryHeader.js +32 -0
  63. package/dist/esm/components/MediaLibrary/MediaLibraryModal.js +156 -0
  64. package/dist/esm/components/MediaLibrary/MediaLibrarySearch.js +51 -0
  65. package/dist/esm/components/MediaLibrary/MediaLibraryTop.js +123 -0
  66. package/dist/esm/components/UI/DragDrop.js +67 -0
  67. package/dist/esm/components/UI/ErrorBoundary.js +173 -0
  68. package/dist/esm/components/UI/FileUploadButton.js +27 -0
  69. package/dist/esm/components/UI/Modal.js +104 -0
  70. package/dist/esm/components/UI/Notifications.js +62 -0
  71. package/dist/esm/components/UI/SettingsDropdown.js +107 -0
  72. package/dist/esm/components/UI/index.js +6 -0
  73. package/dist/esm/components/Workflow/Workflow.js +133 -0
  74. package/dist/esm/components/Workflow/WorkflowCard.js +128 -0
  75. package/dist/esm/components/Workflow/WorkflowList.js +204 -0
  76. package/dist/esm/constants/collectionTypes.js +2 -0
  77. package/dist/esm/constants/collectionViews.js +2 -0
  78. package/dist/esm/constants/commitProps.js +2 -0
  79. package/dist/esm/constants/configSchema.js +644 -0
  80. package/dist/esm/constants/fieldInference.js +57 -0
  81. package/dist/esm/constants/publishModes.js +18 -0
  82. package/dist/esm/constants/validationErrorTypes.js +6 -0
  83. package/dist/esm/formats/formats.js +83 -0
  84. package/dist/esm/formats/frontmatter.js +146 -0
  85. package/dist/esm/formats/helpers.js +12 -0
  86. package/dist/esm/formats/json.js +8 -0
  87. package/dist/esm/formats/toml.js +32 -0
  88. package/dist/esm/formats/yaml.js +51 -0
  89. package/dist/esm/index.js +7 -0
  90. package/dist/esm/integrations/index.js +28 -0
  91. package/dist/esm/integrations/providers/algolia/implementation.js +174 -0
  92. package/dist/esm/integrations/providers/assetStore/implementation.js +165 -0
  93. package/dist/esm/lib/consoleError.js +3 -0
  94. package/dist/esm/lib/formatters.js +191 -0
  95. package/dist/esm/lib/i18n.js +367 -0
  96. package/dist/esm/lib/phrases.js +6 -0
  97. package/dist/esm/lib/polyfill.js +8 -0
  98. package/dist/esm/lib/registry.js +329 -0
  99. package/dist/esm/lib/serializeEntryValues.js +67 -0
  100. package/dist/esm/lib/stega.js +142 -0
  101. package/dist/esm/lib/textHelper.js +9 -0
  102. package/dist/esm/lib/urlHelper.js +111 -0
  103. package/dist/esm/mediaLibrary.js +37 -0
  104. package/dist/esm/reducers/auth.js +27 -0
  105. package/dist/esm/reducers/collections.js +428 -0
  106. package/dist/esm/reducers/combinedReducer.js +8 -0
  107. package/dist/esm/reducers/config.js +29 -0
  108. package/dist/esm/reducers/cursors.js +31 -0
  109. package/dist/esm/reducers/deploys.js +45 -0
  110. package/dist/esm/reducers/editorialWorkflow.js +83 -0
  111. package/dist/esm/reducers/entries.js +568 -0
  112. package/dist/esm/reducers/entryDraft.js +212 -0
  113. package/dist/esm/reducers/globalUI.js +25 -0
  114. package/dist/esm/reducers/index.js +66 -0
  115. package/dist/esm/reducers/integrations.js +53 -0
  116. package/dist/esm/reducers/mediaLibrary.js +252 -0
  117. package/dist/esm/reducers/medias.js +68 -0
  118. package/dist/esm/reducers/notifications.js +23 -0
  119. package/dist/esm/reducers/search.js +92 -0
  120. package/dist/esm/reducers/status.js +30 -0
  121. package/dist/esm/redux/index.js +7 -0
  122. package/dist/esm/redux/middleware/waitUntilAction.js +48 -0
  123. package/dist/esm/routing/history.js +12 -0
  124. package/dist/esm/types/diacritics.d.js +0 -0
  125. package/dist/esm/types/global.d.js +1 -0
  126. package/dist/esm/types/immutable.js +7 -0
  127. package/dist/esm/types/redux.js +14 -0
  128. package/dist/esm/types/tomlify-j0.4.d.js +0 -0
  129. package/dist/esm/valueObjects/AssetProxy.js +44 -0
  130. package/dist/esm/valueObjects/EditorComponent.js +34 -0
  131. package/dist/esm/valueObjects/Entry.js +20 -0
  132. package/index.d.ts +618 -0
  133. package/package.json +106 -0
  134. package/src/__tests__/backend.spec.js +1161 -0
  135. package/src/actions/__tests__/config.spec.js +1009 -0
  136. package/src/actions/__tests__/editorialWorkflow.spec.js +216 -0
  137. package/src/actions/__tests__/entries.spec.js +596 -0
  138. package/src/actions/__tests__/media.spec.ts +171 -0
  139. package/src/actions/__tests__/mediaLibrary.spec.js +327 -0
  140. package/src/actions/__tests__/search.spec.js +209 -0
  141. package/src/actions/auth.ts +127 -0
  142. package/src/actions/collections.ts +18 -0
  143. package/src/actions/config.ts +565 -0
  144. package/src/actions/deploys.ts +104 -0
  145. package/src/actions/editorialWorkflow.ts +567 -0
  146. package/src/actions/entries.ts +1055 -0
  147. package/src/actions/media.ts +139 -0
  148. package/src/actions/mediaLibrary.ts +574 -0
  149. package/src/actions/notifications.ts +36 -0
  150. package/src/actions/search.ts +221 -0
  151. package/src/actions/status.ts +99 -0
  152. package/src/actions/waitUntil.ts +49 -0
  153. package/src/backend.ts +1400 -0
  154. package/src/bootstrap.js +104 -0
  155. package/src/components/App/App.js +286 -0
  156. package/src/components/App/Header.js +266 -0
  157. package/src/components/App/NotFoundPage.js +23 -0
  158. package/src/components/Collection/Collection.js +210 -0
  159. package/src/components/Collection/CollectionControls.js +58 -0
  160. package/src/components/Collection/CollectionSearch.js +243 -0
  161. package/src/components/Collection/CollectionTop.js +81 -0
  162. package/src/components/Collection/ControlButton.js +27 -0
  163. package/src/components/Collection/Entries/Entries.js +82 -0
  164. package/src/components/Collection/Entries/EntriesCollection.js +277 -0
  165. package/src/components/Collection/Entries/EntriesSearch.js +102 -0
  166. package/src/components/Collection/Entries/EntryCard.js +246 -0
  167. package/src/components/Collection/Entries/EntryListing.js +151 -0
  168. package/src/components/Collection/Entries/__tests__/EntriesCollection.spec.js +163 -0
  169. package/src/components/Collection/Entries/__tests__/__snapshots__/EntriesCollection.spec.js.snap +46 -0
  170. package/src/components/Collection/FilterControl.js +39 -0
  171. package/src/components/Collection/GroupControl.js +39 -0
  172. package/src/components/Collection/NestedCollection.js +330 -0
  173. package/src/components/Collection/Sidebar.js +136 -0
  174. package/src/components/Collection/SortControl.js +68 -0
  175. package/src/components/Collection/ViewStyleControl.js +50 -0
  176. package/src/components/Collection/__tests__/Collection.spec.js +75 -0
  177. package/src/components/Collection/__tests__/NestedCollection.spec.js +445 -0
  178. package/src/components/Collection/__tests__/Sidebar.spec.js +87 -0
  179. package/src/components/Collection/__tests__/__snapshots__/Collection.spec.js.snap +144 -0
  180. package/src/components/Collection/__tests__/__snapshots__/NestedCollection.spec.js.snap +550 -0
  181. package/src/components/Collection/__tests__/__snapshots__/Sidebar.spec.js.snap +312 -0
  182. package/src/components/Editor/Editor.js +497 -0
  183. package/src/components/Editor/EditorControlPane/EditorControl.js +452 -0
  184. package/src/components/Editor/EditorControlPane/EditorControlPane.js +269 -0
  185. package/src/components/Editor/EditorControlPane/Widget.js +384 -0
  186. package/src/components/Editor/EditorInterface.js +444 -0
  187. package/src/components/Editor/EditorPreviewPane/EditorPreview.js +40 -0
  188. package/src/components/Editor/EditorPreviewPane/EditorPreviewContent.js +74 -0
  189. package/src/components/Editor/EditorPreviewPane/EditorPreviewPane.js +333 -0
  190. package/src/components/Editor/EditorPreviewPane/PreviewHOC.js +33 -0
  191. package/src/components/Editor/EditorToolbar.js +691 -0
  192. package/src/components/Editor/__tests__/Editor.spec.js +221 -0
  193. package/src/components/Editor/__tests__/EditorToolbar.spec.js +120 -0
  194. package/src/components/Editor/__tests__/__snapshots__/Editor.spec.js.snap +45 -0
  195. package/src/components/Editor/__tests__/__snapshots__/EditorToolbar.spec.js.snap +4233 -0
  196. package/src/components/Editor/withWorkflow.js +61 -0
  197. package/src/components/EditorWidgets/Unknown/UnknownControl.js +17 -0
  198. package/src/components/EditorWidgets/Unknown/UnknownPreview.js +19 -0
  199. package/src/components/EditorWidgets/index.js +5 -0
  200. package/src/components/MediaLibrary/EmptyMessage.js +28 -0
  201. package/src/components/MediaLibrary/MediaLibrary.js +411 -0
  202. package/src/components/MediaLibrary/MediaLibraryButtons.js +136 -0
  203. package/src/components/MediaLibrary/MediaLibraryCard.js +128 -0
  204. package/src/components/MediaLibrary/MediaLibraryCardGrid.js +199 -0
  205. package/src/components/MediaLibrary/MediaLibraryHeader.js +48 -0
  206. package/src/components/MediaLibrary/MediaLibraryModal.js +200 -0
  207. package/src/components/MediaLibrary/MediaLibrarySearch.js +61 -0
  208. package/src/components/MediaLibrary/MediaLibraryTop.js +143 -0
  209. package/src/components/MediaLibrary/__tests__/MediaLibraryButtons.spec.js +45 -0
  210. package/src/components/MediaLibrary/__tests__/MediaLibraryCard.spec.js +49 -0
  211. package/src/components/MediaLibrary/__tests__/__snapshots__/MediaLibraryCard.spec.js.snap +264 -0
  212. package/src/components/UI/DragDrop.js +66 -0
  213. package/src/components/UI/ErrorBoundary.js +214 -0
  214. package/src/components/UI/FileUploadButton.js +24 -0
  215. package/src/components/UI/Modal.js +112 -0
  216. package/src/components/UI/Notifications.tsx +83 -0
  217. package/src/components/UI/SettingsDropdown.js +103 -0
  218. package/src/components/UI/__tests__/ErrorBoundary.spec.js +57 -0
  219. package/src/components/UI/index.js +6 -0
  220. package/src/components/Workflow/Workflow.js +169 -0
  221. package/src/components/Workflow/WorkflowCard.js +177 -0
  222. package/src/components/Workflow/WorkflowList.js +272 -0
  223. package/src/constants/__tests__/configSchema.spec.js +611 -0
  224. package/src/constants/collectionTypes.ts +2 -0
  225. package/src/constants/collectionViews.js +2 -0
  226. package/src/constants/commitProps.ts +2 -0
  227. package/src/constants/configSchema.js +441 -0
  228. package/src/constants/fieldInference.tsx +78 -0
  229. package/src/constants/publishModes.ts +22 -0
  230. package/src/constants/validationErrorTypes.js +6 -0
  231. package/src/formats/__tests__/formats.spec.js +87 -0
  232. package/src/formats/__tests__/frontmatter.spec.js +429 -0
  233. package/src/formats/__tests__/toml.spec.js +9 -0
  234. package/src/formats/__tests__/yaml.spec.js +162 -0
  235. package/src/formats/formats.ts +97 -0
  236. package/src/formats/frontmatter.ts +150 -0
  237. package/src/formats/helpers.ts +14 -0
  238. package/src/formats/json.ts +9 -0
  239. package/src/formats/toml.ts +33 -0
  240. package/src/formats/yaml.ts +58 -0
  241. package/src/index.js +8 -0
  242. package/src/integrations/index.js +35 -0
  243. package/src/integrations/providers/algolia/implementation.js +176 -0
  244. package/src/integrations/providers/assetStore/implementation.js +148 -0
  245. package/src/lib/__tests__/formatters.spec.js +751 -0
  246. package/src/lib/__tests__/i18n.spec.js +792 -0
  247. package/src/lib/__tests__/phrases.spec.js +119 -0
  248. package/src/lib/__tests__/registry.spec.js +261 -0
  249. package/src/lib/__tests__/serializeEntryValues.spec.js +22 -0
  250. package/src/lib/__tests__/urlHelper.spec.js +138 -0
  251. package/src/lib/consoleError.js +7 -0
  252. package/src/lib/formatters.ts +286 -0
  253. package/src/lib/i18n.ts +454 -0
  254. package/src/lib/phrases.js +8 -0
  255. package/src/lib/polyfill.js +9 -0
  256. package/src/lib/registry.js +312 -0
  257. package/src/lib/serializeEntryValues.js +75 -0
  258. package/src/lib/stega.ts +145 -0
  259. package/src/lib/textHelper.js +11 -0
  260. package/src/lib/urlHelper.ts +128 -0
  261. package/src/mediaLibrary.ts +51 -0
  262. package/src/reducers/__tests__/auth.spec.ts +38 -0
  263. package/src/reducers/__tests__/collections.spec.js +610 -0
  264. package/src/reducers/__tests__/config.spec.js +38 -0
  265. package/src/reducers/__tests__/entries.spec.js +694 -0
  266. package/src/reducers/__tests__/entryDraft.spec.js +297 -0
  267. package/src/reducers/__tests__/globalUI.js +43 -0
  268. package/src/reducers/__tests__/integrations.spec.ts +76 -0
  269. package/src/reducers/__tests__/mediaLibrary.spec.js +154 -0
  270. package/src/reducers/__tests__/medias.spec.ts +49 -0
  271. package/src/reducers/auth.ts +46 -0
  272. package/src/reducers/collections.ts +535 -0
  273. package/src/reducers/combinedReducer.ts +11 -0
  274. package/src/reducers/config.ts +38 -0
  275. package/src/reducers/cursors.js +36 -0
  276. package/src/reducers/deploys.ts +52 -0
  277. package/src/reducers/editorialWorkflow.ts +163 -0
  278. package/src/reducers/entries.ts +819 -0
  279. package/src/reducers/entryDraft.js +260 -0
  280. package/src/reducers/globalUI.ts +45 -0
  281. package/src/reducers/index.ts +82 -0
  282. package/src/reducers/integrations.ts +59 -0
  283. package/src/reducers/mediaLibrary.ts +296 -0
  284. package/src/reducers/medias.ts +66 -0
  285. package/src/reducers/notifications.ts +52 -0
  286. package/src/reducers/search.ts +111 -0
  287. package/src/reducers/status.ts +40 -0
  288. package/src/redux/index.ts +18 -0
  289. package/src/redux/middleware/waitUntilAction.ts +64 -0
  290. package/src/routing/__tests__/history.spec.ts +49 -0
  291. package/src/routing/history.ts +17 -0
  292. package/src/types/diacritics.d.ts +1 -0
  293. package/src/types/global.d.ts +8 -0
  294. package/src/types/immutable.ts +49 -0
  295. package/src/types/redux.ts +827 -0
  296. package/src/types/tomlify-j0.4.d.ts +13 -0
  297. package/src/valueObjects/AssetProxy.ts +48 -0
  298. package/src/valueObjects/EditorComponent.js +38 -0
  299. package/src/valueObjects/Entry.ts +63 -0
@@ -0,0 +1,214 @@
1
+ import React from 'react';
2
+ import PropTypes from 'prop-types';
3
+ import { translate } from 'react-polyglot';
4
+ import styled from '@emotion/styled';
5
+ import yaml from 'yaml';
6
+ import truncate from 'lodash/truncate';
7
+ import copyToClipboard from 'copy-text-to-clipboard';
8
+ import { localForage } from 'decap-cms-lib-util';
9
+ import { buttons, colors } from 'decap-cms-ui-default';
10
+ import cleanStack from 'clean-stack';
11
+
12
+ const ISSUE_URL = 'https://github.com/decaporg/decap-cms/issues/new?';
13
+
14
+ function getIssueTemplate({ version, provider, browser, config }) {
15
+ return `
16
+ **Describe the bug**
17
+
18
+ **To Reproduce**
19
+
20
+ **Expected behavior**
21
+
22
+ **Screenshots**
23
+
24
+ **Applicable Versions:**
25
+ - Decap CMS version: \`${version}\`
26
+ - Git provider: \`${provider}\`
27
+ - Browser version: \`${browser}\`
28
+
29
+ **CMS configuration**
30
+ \`\`\`
31
+ ${config}
32
+ \`\`\`
33
+
34
+ **Additional context**
35
+ `;
36
+ }
37
+
38
+ function buildIssueTemplate({ config }) {
39
+ let version = '';
40
+ if (typeof DECAP_CMS_VERSION === 'string') {
41
+ version = `decap-cms@${DECAP_CMS_VERSION}`;
42
+ } else if (typeof DECAP_CMS_APP_VERSION === 'string') {
43
+ version = `decap-cms-app@${DECAP_CMS_APP_VERSION}`;
44
+ }
45
+ const template = getIssueTemplate({
46
+ version,
47
+ provider: config.backend.name,
48
+ browser: navigator.userAgent,
49
+ config: yaml.stringify(config),
50
+ });
51
+
52
+ return template;
53
+ }
54
+
55
+ function buildIssueUrl({ title, config }) {
56
+ try {
57
+ const body = buildIssueTemplate({ config });
58
+
59
+ const params = new URLSearchParams();
60
+ params.append('title', truncate(title, { length: 100 }));
61
+ params.append('body', truncate(body, { length: 4000, omission: '\n...' }));
62
+ params.append('labels', 'type: bug');
63
+
64
+ return `${ISSUE_URL}${params.toString()}`;
65
+ } catch (e) {
66
+ console.log(e);
67
+ return `${ISSUE_URL}template=bug_report.md`;
68
+ }
69
+ }
70
+
71
+ const ErrorBoundaryContainer = styled.div`
72
+ padding: 40px;
73
+
74
+ h1 {
75
+ font-size: 28px;
76
+ color: ${colors.text};
77
+ }
78
+
79
+ h2 {
80
+ font-size: 20px;
81
+ }
82
+
83
+ strong {
84
+ color: ${colors.textLead};
85
+ font-weight: 500;
86
+ }
87
+
88
+ hr {
89
+ width: 200px;
90
+ margin: 30px 0;
91
+ border: 0;
92
+ height: 1px;
93
+ background-color: ${colors.text};
94
+ }
95
+
96
+ a {
97
+ color: ${colors.active};
98
+ }
99
+ `;
100
+
101
+ const PrivacyWarning = styled.span`
102
+ color: ${colors.text};
103
+ `;
104
+
105
+ const CopyButton = styled.button`
106
+ ${buttons.button};
107
+ ${buttons.default};
108
+ ${buttons.gray};
109
+ display: block;
110
+ margin: 12px 0;
111
+ `;
112
+
113
+ function RecoveredEntry({ entry, t }) {
114
+ console.log(entry);
115
+ return (
116
+ <>
117
+ <hr />
118
+ <h2>{t('ui.errorBoundary.recoveredEntry.heading')}</h2>
119
+ <strong>{t('ui.errorBoundary.recoveredEntry.warning')}</strong>
120
+ <CopyButton onClick={() => copyToClipboard(entry)}>
121
+ {t('ui.errorBoundary.recoveredEntry.copyButtonLabel')}
122
+ </CopyButton>
123
+ <pre>
124
+ <code>{entry}</code>
125
+ </pre>
126
+ </>
127
+ );
128
+ }
129
+
130
+ export class ErrorBoundary extends React.Component {
131
+ static propTypes = {
132
+ children: PropTypes.node,
133
+ t: PropTypes.func.isRequired,
134
+ config: PropTypes.object.isRequired,
135
+ };
136
+
137
+ state = {
138
+ hasError: false,
139
+ errorMessage: '',
140
+ errorTitle: '',
141
+ backup: '',
142
+ };
143
+
144
+ static getDerivedStateFromError(error) {
145
+ console.error(error);
146
+ return {
147
+ hasError: true,
148
+ errorMessage: cleanStack(error.stack, { basePath: window.location.origin || '' }),
149
+ errorTitle: error.toString(),
150
+ };
151
+ }
152
+
153
+ componentDidMount() {
154
+ // Manually validate PropTypes - React 19 breaking change
155
+ PropTypes.checkPropTypes(ErrorBoundary.propTypes, this.props, 'prop', 'ErrorBoundary');
156
+ }
157
+
158
+ shouldComponentUpdate(nextProps, nextState) {
159
+ if (this.props.showBackup) {
160
+ return (
161
+ this.state.errorMessage !== nextState.errorMessage || this.state.backup !== nextState.backup
162
+ );
163
+ }
164
+ return true;
165
+ }
166
+
167
+ async componentDidUpdate() {
168
+ if (this.props.showBackup) {
169
+ const backup = await localForage.getItem('backup');
170
+ backup && console.log(backup);
171
+ this.setState({ backup });
172
+ }
173
+ }
174
+
175
+ render() {
176
+ const { hasError, errorMessage, backup, errorTitle } = this.state;
177
+ const { showBackup, t } = this.props;
178
+ if (!hasError) {
179
+ return this.props.children;
180
+ }
181
+ return (
182
+ <ErrorBoundaryContainer>
183
+ <h1>{t('ui.errorBoundary.title')}</h1>
184
+ <p>
185
+ <span>{t('ui.errorBoundary.details')}</span>
186
+ <a
187
+ href={buildIssueUrl({ title: errorTitle, config: this.props.config })}
188
+ target="_blank"
189
+ rel="noopener noreferrer"
190
+ data-testid="issue-url"
191
+ >
192
+ {t('ui.errorBoundary.reportIt')}
193
+ </a>
194
+ </p>
195
+ <p>
196
+ {t('ui.errorBoundary.privacyWarning')
197
+ .split('\n')
198
+ .map((item, index) => (
199
+ <>
200
+ <PrivacyWarning key={index}>{item}</PrivacyWarning>
201
+ <br />
202
+ </>
203
+ ))}
204
+ </p>
205
+ <hr />
206
+ <h2>{t('ui.errorBoundary.detailsHeading')}</h2>
207
+ <p>{errorMessage}</p>
208
+ {backup && showBackup && <RecoveredEntry entry={backup} t={t} />}
209
+ </ErrorBoundaryContainer>
210
+ );
211
+ }
212
+ }
213
+
214
+ export default translate()(ErrorBoundary);
@@ -0,0 +1,24 @@
1
+ import React from 'react';
2
+ import PropTypes from 'prop-types';
3
+
4
+ export function FileUploadButton({ label, imagesOnly, onChange, disabled, className }) {
5
+ return (
6
+ <label tabIndex={'0'} className={`nc-fileUploadButton ${className || ''}`}>
7
+ <span>{label}</span>
8
+ <input
9
+ type="file"
10
+ accept={imagesOnly ? 'image/*' : '*/*'}
11
+ onChange={onChange}
12
+ disabled={disabled}
13
+ />
14
+ </label>
15
+ );
16
+ }
17
+
18
+ FileUploadButton.propTypes = {
19
+ className: PropTypes.string,
20
+ label: PropTypes.string.isRequired,
21
+ imagesOnly: PropTypes.bool,
22
+ onChange: PropTypes.func.isRequired,
23
+ disabled: PropTypes.bool,
24
+ };
@@ -0,0 +1,112 @@
1
+ import React from 'react';
2
+ import PropTypes from 'prop-types';
3
+ import { css, Global, ClassNames } from '@emotion/react';
4
+ import ReactModal from 'react-modal';
5
+ import { transitions, shadows, lengths, zIndex } from 'decap-cms-ui-default';
6
+
7
+ function ReactModalGlobalStyles() {
8
+ return (
9
+ <Global
10
+ styles={css`
11
+ .ReactModal__Body--open {
12
+ overflow: hidden;
13
+ }
14
+ `}
15
+ />
16
+ );
17
+ }
18
+
19
+ const styleStrings = {
20
+ modalBody: `
21
+ ${shadows.dropDeep};
22
+ background-color: #fff;
23
+ border-radius: ${lengths.borderRadius};
24
+ height: 80%;
25
+ text-align: center;
26
+ max-width: 2200px;
27
+ padding: 20px;
28
+
29
+ &:focus {
30
+ outline: none;
31
+ }
32
+ `,
33
+ overlay: `
34
+ z-index: ${zIndex.zIndex99999};
35
+ position: fixed;
36
+ top: 0;
37
+ left: 0;
38
+ right: 0;
39
+ bottom: 0;
40
+ display: flex;
41
+ justify-content: center;
42
+ align-items: center;
43
+ opacity: 0;
44
+ background-color: rgba(0, 0, 0, 0);
45
+ transition: background-color ${transitions.main}, opacity ${transitions.main};
46
+ `,
47
+ overlayAfterOpen: `
48
+ background-color: rgba(0, 0, 0, 0.6);
49
+ opacity: 1;
50
+ `,
51
+ overlayBeforeClose: `
52
+ background-color: rgba(0, 0, 0, 0);
53
+ opacity: 0;
54
+ `,
55
+ };
56
+
57
+ export class Modal extends React.Component {
58
+ static propTypes = {
59
+ children: PropTypes.node.isRequired,
60
+ isOpen: PropTypes.bool.isRequired,
61
+ className: PropTypes.string,
62
+ onClose: PropTypes.func.isRequired,
63
+ };
64
+
65
+ componentDidMount() {
66
+ // Manually validate PropTypes - React 19 breaking change
67
+ PropTypes.checkPropTypes(Modal.propTypes, this.props, 'prop', 'Modal');
68
+
69
+ ReactModal.setAppElement('#nc-root');
70
+ }
71
+
72
+ render() {
73
+ const { isOpen, children, className, onClose } = this.props;
74
+ return (
75
+ <>
76
+ <ReactModalGlobalStyles />
77
+ <ClassNames>
78
+ {({ css, cx }) => (
79
+ <ReactModal
80
+ isOpen={isOpen}
81
+ onRequestClose={onClose}
82
+ closeTimeoutMS={300}
83
+ className={{
84
+ base: cx(
85
+ css`
86
+ ${styleStrings.modalBody};
87
+ `,
88
+ className,
89
+ ),
90
+ afterOpen: '',
91
+ beforeClose: '',
92
+ }}
93
+ overlayClassName={{
94
+ base: css`
95
+ ${styleStrings.overlay};
96
+ `,
97
+ afterOpen: css`
98
+ ${styleStrings.overlayAfterOpen};
99
+ `,
100
+ beforeClose: css`
101
+ ${styleStrings.overlayBeforeClose};
102
+ `,
103
+ }}
104
+ >
105
+ {children}
106
+ </ReactModal>
107
+ )}
108
+ </ClassNames>
109
+ </>
110
+ );
111
+ }
112
+ }
@@ -0,0 +1,83 @@
1
+ // eslint-disable-next-line no-unused-vars
2
+ import React, { useEffect } from 'react';
3
+ // import { translate } from 'react-polyglot';
4
+ import { injectStyle } from 'react-toastify/dist/inject-style';
5
+ import { toast, ToastContainer } from 'react-toastify';
6
+ import { connect, useDispatch } from 'react-redux';
7
+ import { useTranslate } from 'react-polyglot';
8
+
9
+ import { dismissNotification } from '../../actions/notifications';
10
+
11
+ import type { Id, ToastItem } from 'react-toastify';
12
+ import type { State } from '../../types/redux';
13
+ import type { Notification } from '../../reducers/notifications';
14
+
15
+ injectStyle();
16
+
17
+ interface Props {
18
+ notifications: Notification[];
19
+ }
20
+
21
+ type IdMap = {
22
+ [id: string]: Id;
23
+ };
24
+
25
+ function Notifications({ notifications }: Props) {
26
+ const t = useTranslate();
27
+ const dispatch = useDispatch();
28
+ const [idMap, setIdMap] = React.useState<IdMap>({});
29
+
30
+ useEffect(() => {
31
+ notifications
32
+ .filter(notification => !idMap[notification.id])
33
+ .forEach(notification => {
34
+ const toastId = toast(
35
+ typeof notification.message == 'string'
36
+ ? notification.message
37
+ : t(notification.message.key, { ...notification.message }),
38
+ {
39
+ autoClose: notification.dismissAfter,
40
+ type: notification.type,
41
+ },
42
+ );
43
+
44
+ idMap[notification.id] = toastId;
45
+ setIdMap(idMap);
46
+
47
+ if (notification.dismissAfter) {
48
+ setTimeout(() => {
49
+ dispatch(dismissNotification(notification.id));
50
+ }, notification.dismissAfter);
51
+ }
52
+ });
53
+
54
+ Object.entries(idMap).forEach(([id, toastId]) => {
55
+ if (!notifications.find(notification => notification.id === id)) {
56
+ toast.dismiss(toastId);
57
+ delete idMap[id];
58
+ setIdMap(idMap);
59
+ }
60
+ });
61
+ }, [notifications]);
62
+
63
+ toast.onChange((payload: ToastItem) => {
64
+ if (payload.status == 'removed') {
65
+ const id = Object.entries(idMap).find(([, toastId]) => toastId === payload.id)?.[0];
66
+ if (id) {
67
+ dispatch(dismissNotification(id));
68
+ }
69
+ }
70
+ });
71
+
72
+ return (
73
+ <>
74
+ <ToastContainer position="top-right" theme="colored" className="notif__container" />
75
+ </>
76
+ );
77
+ }
78
+
79
+ function mapStateToProps({ notifications }: State): Props {
80
+ return { notifications: notifications.notifications };
81
+ }
82
+
83
+ export default connect(mapStateToProps)(Notifications);
@@ -0,0 +1,103 @@
1
+ import React from 'react';
2
+ import PropTypes from 'prop-types';
3
+ import { css } from '@emotion/react';
4
+ import styled from '@emotion/styled';
5
+ import { translate } from 'react-polyglot';
6
+ import { Icon, Dropdown, DropdownItem, DropdownButton, colors } from 'decap-cms-ui-default';
7
+
8
+ import { stripProtocol } from '../../lib/urlHelper';
9
+
10
+ const styles = {
11
+ avatarImage: css`
12
+ width: 32px;
13
+ border-radius: 32px;
14
+ `,
15
+ };
16
+
17
+ const AvatarDropdownButton = styled(DropdownButton)`
18
+ display: inline-block;
19
+ padding: 8px;
20
+ cursor: pointer;
21
+ color: #1e2532;
22
+ background-color: transparent;
23
+ `;
24
+
25
+ const AvatarImage = styled.img`
26
+ ${styles.avatarImage};
27
+ `;
28
+
29
+ const AvatarPlaceholderIcon = styled(Icon)`
30
+ ${styles.avatarImage};
31
+ height: 32px;
32
+ color: #1e2532;
33
+ background-color: ${colors.textFieldBorder};
34
+ `;
35
+
36
+ const AppHeaderSiteLink = styled.a`
37
+ font-size: 14px;
38
+ font-weight: 400;
39
+ color: #7b8290;
40
+ padding: 10px 16px;
41
+ `;
42
+
43
+ const AppHeaderTestRepoIndicator = styled.a`
44
+ font-size: 14px;
45
+ font-weight: 400;
46
+ color: #7b8290;
47
+ padding: 10px 16px;
48
+ `;
49
+
50
+ function Avatar({ imageUrl }) {
51
+ return imageUrl ? (
52
+ <AvatarImage src={imageUrl} />
53
+ ) : (
54
+ <AvatarPlaceholderIcon type="user" size="large" />
55
+ );
56
+ }
57
+
58
+ Avatar.propTypes = {
59
+ imageUrl: PropTypes.string,
60
+ };
61
+
62
+ function SettingsDropdown({ displayUrl, isTestRepo, imageUrl, onLogoutClick, t }) {
63
+ return (
64
+ <React.Fragment>
65
+ {isTestRepo && (
66
+ <AppHeaderTestRepoIndicator
67
+ href="https://www.decapcms.org/docs/test-backend"
68
+ target="_blank"
69
+ rel="noopener noreferrer"
70
+ >
71
+ Test Backend ↗
72
+ </AppHeaderTestRepoIndicator>
73
+ )}
74
+ {displayUrl ? (
75
+ <AppHeaderSiteLink href={displayUrl} target="_blank">
76
+ {stripProtocol(displayUrl)}
77
+ </AppHeaderSiteLink>
78
+ ) : null}
79
+ <Dropdown
80
+ dropdownTopOverlap="50px"
81
+ dropdownWidth="100px"
82
+ dropdownPosition="right"
83
+ renderButton={() => (
84
+ <AvatarDropdownButton>
85
+ <Avatar imageUrl={imageUrl} />
86
+ </AvatarDropdownButton>
87
+ )}
88
+ >
89
+ <DropdownItem label={t('ui.settingsDropdown.logOut')} onClick={onLogoutClick} />
90
+ </Dropdown>
91
+ </React.Fragment>
92
+ );
93
+ }
94
+
95
+ SettingsDropdown.propTypes = {
96
+ displayUrl: PropTypes.string,
97
+ isTestRepo: PropTypes.bool,
98
+ imageUrl: PropTypes.string,
99
+ onLogoutClick: PropTypes.func.isRequired,
100
+ t: PropTypes.func.isRequired,
101
+ };
102
+
103
+ export default translate()(SettingsDropdown);
@@ -0,0 +1,57 @@
1
+ import React from 'react';
2
+ import { render } from '@testing-library/react';
3
+ import { oneLineTrim } from 'common-tags';
4
+
5
+ import { ErrorBoundary } from '../ErrorBoundary';
6
+
7
+ function WithError() {
8
+ throw new Error('Some unknown error');
9
+ }
10
+
11
+ jest.spyOn(console, 'error').mockImplementation(() => ({}));
12
+
13
+ Object.defineProperty(
14
+ window.navigator,
15
+ 'userAgent',
16
+ (value => ({
17
+ get() {
18
+ return value;
19
+ },
20
+ set(v) {
21
+ value = v;
22
+ },
23
+ }))(window.navigator['userAgent']),
24
+ );
25
+
26
+ describe('Editor', () => {
27
+ const config = { backend: { name: 'github' } };
28
+
29
+ const props = { t: jest.fn(key => key), config };
30
+
31
+ beforeEach(() => {
32
+ jest.clearAllMocks();
33
+ });
34
+
35
+ it('should match snapshot with issue URL', () => {
36
+ global.navigator.userAgent = 'Test User Agent';
37
+ const { getByTestId } = render(
38
+ <ErrorBoundary {...props}>
39
+ <WithError />
40
+ </ErrorBoundary>,
41
+ );
42
+
43
+ expect(console.error).toHaveBeenCalledWith(new Error('Some unknown error'));
44
+ expect(getByTestId('issue-url').getAttribute('href')).toEqual(
45
+ oneLineTrim`https://github.com/decaporg/decap-cms/issues/new?
46
+ title=Error%3A+Some+unknown+error&
47
+ body=%0A**Describe+the+bug**%0A%0A**To+Reproduce**%0A%0A**Expected+behavior**%0A%0A**
48
+ Screenshots**%0A%0A**Applicable+Versions%3A**%0A+-+
49
+ Decap+CMS+version%3A+%60%60%0A+-+
50
+ Git+provider%3A+%60github%60%0A+-+
51
+ Browser+version%3A+%60Test+User+Agent%60%0A%0A**
52
+ CMS+configuration**%0A%60%60%60%0Abackend%3A%0A++name%3A+github%0A%0A%60%60%60%0A%0A**
53
+ Additional+context**%0A&labels=type%3A+bug
54
+ `,
55
+ );
56
+ });
57
+ });
@@ -0,0 +1,6 @@
1
+ export { DragSource, DropTarget, HTML5DragDrop } from './DragDrop';
2
+ export { default as ErrorBoundary } from './ErrorBoundary';
3
+ export { FileUploadButton } from './FileUploadButton';
4
+ export { Modal } from './Modal';
5
+ export { default as Notifications } from './Notifications';
6
+ export { default as SettingsDropdown } from './SettingsDropdown';