@plone/volto 17.0.0-alpha.2 → 17.0.0-alpha.20

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 (297) hide show
  1. package/.yarn/install-state.gz +0 -0
  2. package/CHANGELOG.md +530 -15
  3. package/CONTRIBUTING.md +1 -1
  4. package/README.md +11 -14
  5. package/addon-registry.js +34 -0
  6. package/create-theme-addons-loader.js +79 -0
  7. package/cypress/support/commands.js +56 -4
  8. package/cypress/support/volto-slate.js +4 -5
  9. package/docker-compose.yml +1 -1
  10. package/locales/ca/LC_MESSAGES/volto.po +272 -6
  11. package/locales/ca.json +1 -1
  12. package/locales/de/LC_MESSAGES/volto.po +291 -25
  13. package/locales/de.json +1 -1
  14. package/locales/en/LC_MESSAGES/volto.po +271 -5
  15. package/locales/en.json +1 -1
  16. package/locales/es/LC_MESSAGES/volto.po +281 -15
  17. package/locales/es.json +1 -1
  18. package/locales/eu/LC_MESSAGES/volto.po +272 -6
  19. package/locales/eu.json +1 -1
  20. package/locales/fi/LC_MESSAGES/volto.po +4882 -0
  21. package/locales/fi.json +1 -1
  22. package/locales/fr/LC_MESSAGES/volto.po +272 -6
  23. package/locales/fr.json +1 -1
  24. package/locales/it/LC_MESSAGES/volto.po +273 -7
  25. package/locales/it.json +1 -1
  26. package/locales/ja/LC_MESSAGES/volto.po +272 -6
  27. package/locales/ja.json +1 -1
  28. package/locales/nl/LC_MESSAGES/volto.po +927 -649
  29. package/locales/nl.json +1 -1
  30. package/locales/pt/LC_MESSAGES/volto.po +272 -6
  31. package/locales/pt.json +1 -1
  32. package/locales/pt_BR/LC_MESSAGES/volto.po +281 -15
  33. package/locales/pt_BR.json +1 -1
  34. package/locales/ro/LC_MESSAGES/volto.po +272 -6
  35. package/locales/ro.json +1 -1
  36. package/locales/volto.pot +272 -6
  37. package/locales/zh_CN/LC_MESSAGES/volto.po +272 -6
  38. package/locales/zh_CN.json +1 -1
  39. package/package.json +5 -3
  40. package/packages/volto-slate/build/messages/src/blocks/Table/TableBlockEdit.json +1 -1
  41. package/packages/volto-slate/build/messages/src/blocks/Text/DefaultTextBlockEditor.json +1 -1
  42. package/packages/volto-slate/build/messages/src/blocks/Text/DetachedTextBlockEditor.json +1 -1
  43. package/packages/volto-slate/build/messages/src/blocks/Text/SlashMenu.json +1 -1
  44. package/packages/volto-slate/build/messages/src/editor/plugins/AdvancedLink/index.json +1 -1
  45. package/packages/volto-slate/build/messages/src/editor/plugins/Link/index.json +1 -1
  46. package/packages/volto-slate/build/messages/src/editor/plugins/Table/index.json +1 -1
  47. package/packages/volto-slate/build/messages/src/elementEditor/messages.json +1 -1
  48. package/packages/volto-slate/build/messages/src/widgets/HtmlSlateWidget.json +1 -1
  49. package/packages/volto-slate/build/messages/src/widgets/RichTextWidgetView.json +1 -1
  50. package/packages/volto-slate/package.json +1 -1
  51. package/packages/volto-slate/src/actions/index.js +1 -1
  52. package/packages/volto-slate/src/blocks/Table/TableBlockEdit.jsx +21 -212
  53. package/packages/volto-slate/src/blocks/Table/schema.js +122 -0
  54. package/packages/volto-slate/src/blocks/Text/DefaultTextBlockEditor.jsx +8 -3
  55. package/packages/volto-slate/src/blocks/Text/SlashMenu.jsx +4 -3
  56. package/packages/volto-slate/src/blocks/Text/TextBlockView.jsx +20 -16
  57. package/packages/volto-slate/src/blocks/Text/extensions/withDeserializers.js +3 -1
  58. package/packages/volto-slate/src/blocks/Text/index.js +10 -2
  59. package/packages/volto-slate/src/editor/config.jsx +5 -4
  60. package/packages/volto-slate/src/editor/deserialize.js +0 -1
  61. package/packages/volto-slate/src/editor/index.js +4 -4
  62. package/packages/volto-slate/src/editor/less/slate.less +28 -0
  63. package/packages/volto-slate/src/editor/plugins/StyleMenu/StyleMenu.jsx +14 -4
  64. package/packages/volto-slate/src/editor/plugins/StyleMenu/utils.js +14 -5
  65. package/packages/volto-slate/src/editor/render.jsx +68 -8
  66. package/packages/volto-slate/src/editor/ui/SlateContextToolbar.jsx +2 -2
  67. package/packages/volto-slate/src/editor/ui/index.js +15 -15
  68. package/packages/volto-slate/src/index.js +2 -2
  69. package/packages/volto-slate/src/utils/blocks.js +7 -0
  70. package/packages/volto-slate/src/widgets/RichTextWidget.jsx +15 -8
  71. package/razzle.config.js +28 -0
  72. package/src/actions/index.js +6 -0
  73. package/src/actions/language/language.js +9 -8
  74. package/src/actions/querystringsearch/querystringsearch.js +20 -14
  75. package/src/actions/relations/rebuild.js +25 -0
  76. package/src/actions/relations/relations.js +86 -0
  77. package/src/actions/relations/relations.test.js +15 -0
  78. package/src/components/index.js +2 -0
  79. package/src/components/manage/Add/Add.jsx +2 -2
  80. package/src/components/manage/AnchorPlugin/index.jsx +2 -2
  81. package/src/components/manage/AnchorPlugin/utils/EditorUtils.js +3 -1
  82. package/src/components/manage/BlockChooser/BlockChooser.jsx +14 -5
  83. package/src/components/manage/BlockChooser/BlockChooser.test.jsx +5 -0
  84. package/src/components/manage/Blocks/Block/BlocksForm.jsx +19 -2
  85. package/src/components/manage/Blocks/Block/Edit.jsx +1 -1
  86. package/src/components/manage/Blocks/Block/Style.jsx +2 -2
  87. package/src/components/manage/Blocks/Container/Data.jsx +32 -0
  88. package/src/components/manage/Blocks/Container/Edit.jsx +174 -0
  89. package/src/components/manage/Blocks/Container/EditBlockWrapper.jsx +120 -0
  90. package/src/components/manage/Blocks/Container/NewBlockAddButton.jsx +84 -0
  91. package/src/components/manage/Blocks/Container/SimpleContainerToolbar.jsx +54 -0
  92. package/src/components/manage/Blocks/Grid/Edit.jsx +33 -0
  93. package/src/components/manage/Blocks/Grid/View.jsx +43 -0
  94. package/src/components/manage/Blocks/Grid/adapter.js +14 -0
  95. package/src/components/manage/Blocks/Grid/grid-1.svg +6 -0
  96. package/src/components/manage/Blocks/Grid/grid-2.svg +9 -0
  97. package/src/components/manage/Blocks/Grid/grid-3.svg +10 -0
  98. package/src/components/manage/Blocks/Grid/grid-4.svg +11 -0
  99. package/src/components/manage/Blocks/Grid/schema.js +35 -0
  100. package/src/components/manage/Blocks/Grid/templates.js +47 -0
  101. package/src/components/manage/Blocks/HeroImageLeft/Edit.jsx +6 -1
  102. package/src/components/manage/Blocks/Image/Edit.jsx +11 -7
  103. package/src/components/manage/Blocks/Image/ImageSidebar.jsx +2 -1
  104. package/src/components/manage/Blocks/Image/schema.js +11 -0
  105. package/src/components/manage/Blocks/Listing/DefaultTemplate.jsx +18 -3
  106. package/src/components/manage/Blocks/Listing/Edit.jsx +0 -14
  107. package/src/components/manage/Blocks/Listing/ListingBody.jsx +30 -8
  108. package/src/components/manage/Blocks/Listing/ListingBody.test.jsx +20 -0
  109. package/src/components/manage/Blocks/Listing/getAsyncData.js +9 -3
  110. package/src/components/manage/Blocks/Listing/withQuerystringResults.jsx +26 -18
  111. package/src/components/manage/Blocks/Search/SearchBlockEdit.jsx +5 -4
  112. package/src/components/manage/Blocks/Search/SearchBlockView.jsx +2 -1
  113. package/src/components/manage/Blocks/Search/components/DateRangeFacet.jsx +4 -1
  114. package/src/components/manage/Blocks/Search/components/Facets.jsx +64 -4
  115. package/src/components/manage/Blocks/Search/components/SearchInput.jsx +9 -2
  116. package/src/components/manage/Blocks/Search/components/index.js +13 -13
  117. package/src/components/manage/Blocks/Search/hocs/index.js +2 -2
  118. package/src/components/manage/Blocks/Search/hocs/withQueryString.jsx +2 -2
  119. package/src/components/manage/Blocks/Search/hocs/withSearch.jsx +43 -15
  120. package/src/components/manage/Blocks/Search/layout/LeftColumnFacets.jsx +17 -5
  121. package/src/components/manage/Blocks/Search/layout/RightColumnFacets.jsx +17 -5
  122. package/src/components/manage/Blocks/Search/layout/TopSideFacets.jsx +21 -5
  123. package/src/components/manage/Blocks/Search/schema.js +16 -1
  124. package/src/components/manage/Blocks/Teaser/Body.jsx +0 -1
  125. package/src/components/manage/Blocks/Teaser/DefaultBody.jsx +20 -15
  126. package/src/components/manage/Blocks/Teaser/schema.js +5 -0
  127. package/src/components/manage/Blocks/Title/View.jsx +15 -5
  128. package/src/components/manage/Blocks/Title/View.test.jsx +16 -1
  129. package/src/components/manage/Blocks/ToC/Schema.jsx +5 -1
  130. package/src/components/manage/Blocks/ToC/View.jsx +8 -1
  131. package/src/components/manage/Blocks/ToC/variations/DefaultTocRenderer.jsx +17 -4
  132. package/src/components/manage/Blocks/ToC/variations/HorizontalMenu.jsx +148 -10
  133. package/src/components/manage/Blocks/ToC/variations/index.js +3 -1
  134. package/src/components/manage/Contents/Contents.jsx +39 -26
  135. package/src/components/manage/Contents/ContentsItem.jsx +6 -0
  136. package/src/components/manage/Contents/ContentsUploadModal.jsx +10 -5
  137. package/src/components/manage/Controlpanels/AddonsControlpanel.jsx +3 -3
  138. package/src/components/manage/Controlpanels/Controlpanels.jsx +199 -224
  139. package/src/components/manage/Controlpanels/Controlpanels.test.jsx +46 -7
  140. package/src/components/manage/Controlpanels/Relations/BrokenRelations.jsx +66 -0
  141. package/src/components/manage/Controlpanels/Relations/Relations.jsx +114 -0
  142. package/src/components/manage/Controlpanels/Relations/RelationsListing.jsx +479 -0
  143. package/src/components/manage/Controlpanels/Relations/RelationsMatrix.jsx +531 -0
  144. package/src/components/manage/Controlpanels/Users/UserGroupMembershipControlPanel.jsx +3 -3
  145. package/src/components/manage/Controlpanels/Users/UserGroupMembershipListing.jsx +51 -82
  146. package/src/components/manage/Controlpanels/Users/UserGroupMembershipMatrix.jsx +79 -75
  147. package/src/components/manage/DragDropList/DragDropList.jsx +18 -13
  148. package/src/components/manage/Form/Form.jsx +5 -3
  149. package/src/components/manage/Form/InlineForm.jsx +39 -9
  150. package/src/components/manage/Form/InlineFormState.js +8 -0
  151. package/src/components/manage/History/History.jsx +11 -1
  152. package/src/components/manage/LinksToItem/LinksToItem.jsx +209 -0
  153. package/src/components/manage/LinksToItem/LinksToItem.test.jsx +97 -0
  154. package/src/components/manage/Multilingual/CreateTranslation.jsx +2 -2
  155. package/src/components/manage/Multilingual/TranslationObject.jsx +4 -3
  156. package/src/components/manage/Preferences/ChangePassword.jsx +2 -2
  157. package/src/components/manage/Preferences/PersonalPreferences.jsx +2 -2
  158. package/src/components/manage/Sharing/Sharing.jsx +5 -1
  159. package/src/components/manage/TemplateChooser/TemplateChooser.jsx +38 -0
  160. package/src/components/manage/TemplateChooser/TemplateChooser.test.jsx +34 -0
  161. package/src/components/manage/TemplateChooser/template.svg +10 -0
  162. package/src/components/manage/Toast/Toast.jsx +2 -2
  163. package/src/components/manage/Toolbar/More.jsx +15 -0
  164. package/src/components/manage/Toolbar/Types.jsx +2 -2
  165. package/src/components/manage/UniversalLink/UniversalLink.jsx +2 -6
  166. package/src/components/manage/UniversalLink/UniversalLink.test.jsx +36 -0
  167. package/src/components/manage/Widgets/ColorPickerWidget.jsx +6 -1
  168. package/src/components/manage/Widgets/DatetimeWidget.jsx +9 -5
  169. package/src/components/manage/Widgets/FileWidget.jsx +2 -1
  170. package/src/components/manage/Widgets/ObjectListWidget.jsx +3 -8
  171. package/src/components/manage/Widgets/RecurrenceWidget/ByDayField.jsx +2 -1
  172. package/src/components/manage/Widgets/RecurrenceWidget/MonthOfTheYearField.jsx +2 -1
  173. package/src/components/manage/Widgets/RecurrenceWidget/Occurences.jsx +2 -1
  174. package/src/components/manage/Widgets/RecurrenceWidget/RecurrenceWidget.jsx +8 -3
  175. package/src/components/manage/Widgets/RecurrenceWidget/WeekdayOfTheMonthField.jsx +2 -1
  176. package/src/components/manage/Widgets/SelectUtils.js +1 -1
  177. package/src/components/manage/Widgets/SelectWidget.jsx +1 -1
  178. package/src/components/theme/Anontools/Anontools.jsx +44 -72
  179. package/src/components/theme/Anontools/Anontools.stories.jsx +16 -6
  180. package/src/components/theme/Anontools/Anontools.test.jsx +16 -2
  181. package/src/components/theme/Breadcrumbs/Breadcrumbs.jsx +52 -99
  182. package/src/components/theme/Breadcrumbs/Breadcrumbs.stories.jsx +14 -13
  183. package/src/components/theme/Comments/CommentEditModal.jsx +63 -115
  184. package/src/components/theme/Component/Component.jsx +1 -1
  185. package/src/components/theme/ContactForm/ContactForm.jsx +108 -192
  186. package/src/components/theme/ContactForm/ContactForm.stories.jsx +1 -1
  187. package/src/components/theme/ContactForm/ContactForm.test.jsx +2 -3
  188. package/src/components/theme/Footer/Footer.jsx +2 -13
  189. package/src/components/theme/Header/Header.jsx +37 -63
  190. package/src/components/theme/Header/Header.test.jsx +18 -0
  191. package/src/components/theme/Icon/Icon.jsx +2 -2
  192. package/src/components/theme/LanguageSelector/LanguageSelector.js +8 -3
  193. package/src/components/theme/Login/Login.jsx +1 -0
  194. package/src/components/theme/Logo/Logo.jsx +2 -1
  195. package/src/components/theme/MultilingualRedirector/MultilingualRedirector.jsx +2 -2
  196. package/src/components/theme/Navigation/NavItem.jsx +4 -2
  197. package/src/components/theme/NotFound/NotFound.jsx +55 -41
  198. package/src/components/theme/PasswordReset/PasswordReset.jsx +7 -4
  199. package/src/components/theme/PasswordReset/RequestPasswordReset.jsx +1 -1
  200. package/src/components/theme/SearchWidget/SearchWidget.jsx +38 -98
  201. package/src/components/theme/Sitemap/Sitemap.jsx +5 -3
  202. package/src/components/theme/View/AlbumView.jsx +9 -1
  203. package/src/components/theme/View/DefaultView.jsx +1 -1
  204. package/src/components/theme/View/EventDatesInfo.jsx +2 -1
  205. package/src/components/theme/View/EventView.jsx +6 -2
  206. package/src/components/theme/View/FileView.jsx +23 -18
  207. package/src/components/theme/View/ImageView.jsx +37 -32
  208. package/src/components/theme/View/LinkView.jsx +53 -78
  209. package/src/components/theme/View/ListingView.jsx +33 -27
  210. package/src/components/theme/View/NewsItemView.jsx +10 -5
  211. package/src/components/theme/View/RenderBlocks.jsx +56 -21
  212. package/src/components/theme/View/RenderEmptyBlock.jsx +5 -0
  213. package/src/components/theme/View/SummaryView.jsx +47 -38
  214. package/src/components/theme/View/TabularView.jsx +59 -53
  215. package/src/components/theme/Widgets/DateWidget.jsx +2 -1
  216. package/src/components/theme/Widgets/DatetimeWidget.jsx +2 -1
  217. package/src/components/theme/Widgets/RelationsWidget.jsx +13 -11
  218. package/src/config/Blocks.jsx +44 -0
  219. package/src/config/ControlPanels.js +2 -0
  220. package/src/config/NonContentRoutes.jsx +1 -0
  221. package/src/config/RichTextEditor/Blocks.jsx +2 -2
  222. package/src/config/RichTextEditor/FromHTML.jsx +2 -2
  223. package/src/config/RichTextEditor/Styles.jsx +1 -1
  224. package/src/config/Widgets.jsx +2 -0
  225. package/src/config/index.js +24 -0
  226. package/src/config/server.js +2 -0
  227. package/src/constants/ActionTypes.js +4 -0
  228. package/src/constants/Indexes.js +3 -1
  229. package/src/constants/Languages.js +8 -4
  230. package/src/express-middleware/devproxy.js +1 -1
  231. package/src/express-middleware/files.js +3 -3
  232. package/src/express-middleware/images.js +4 -4
  233. package/src/express-middleware/ok.js +16 -0
  234. package/src/express-middleware/robotstxt.js +1 -1
  235. package/src/express-middleware/sitemap.js +37 -5
  236. package/src/express-middleware/static.js +3 -3
  237. package/src/helpers/Api/Api.js +1 -1
  238. package/src/helpers/Blocks/Blocks.js +48 -0
  239. package/src/helpers/Blocks/Blocks.test.js +79 -0
  240. package/src/helpers/Extensions/index.js +2 -1
  241. package/src/helpers/Extensions/withBlockSchemaEnhancer.js +15 -11
  242. package/src/helpers/Extensions/withBlockSchemaEnhancer.test.js +145 -0
  243. package/src/helpers/FormValidation/FormValidation.js +40 -2
  244. package/src/helpers/FormValidation/FormValidation.test.js +73 -0
  245. package/src/helpers/Html/Html.jsx +3 -1
  246. package/src/helpers/Html/Html.test.jsx +5 -0
  247. package/src/helpers/MessageLabels/MessageLabels.js +80 -0
  248. package/src/helpers/Robots/Robots.js +24 -6
  249. package/src/helpers/ScrollToTop/ScrollToTop.jsx +5 -3
  250. package/src/helpers/Sitemap/Sitemap.js +44 -2
  251. package/src/helpers/Url/Url.js +27 -6
  252. package/src/helpers/Url/Url.test.js +26 -0
  253. package/src/helpers/Utils/UseDetectClickOutside.stories.jsx +191 -0
  254. package/src/helpers/Utils/Utils.js +63 -13
  255. package/src/helpers/Utils/Utils.test.js +4 -4
  256. package/src/helpers/Utils/usePagination.js +67 -14
  257. package/src/helpers/Utils/usePagination.test.js +115 -0
  258. package/src/helpers/index.js +20 -10
  259. package/src/hooks/client/useClient.js +11 -0
  260. package/src/hooks/clipboard/useClipboard.js +26 -0
  261. package/src/hooks/index.js +2 -0
  262. package/src/icons/grid-block.svg +11 -0
  263. package/src/middleware/Api.test.js +54 -0
  264. package/src/middleware/api.js +24 -6
  265. package/src/middleware/index.js +2 -2
  266. package/src/reducers/actions/actions.js +8 -6
  267. package/src/reducers/actions/actions.test.js +70 -0
  268. package/src/reducers/breadcrumbs/breadcrumbs.js +1 -1
  269. package/src/reducers/index.js +2 -0
  270. package/src/reducers/navigation/navigation.js +1 -1
  271. package/src/reducers/relations/relations.js +173 -0
  272. package/src/reducers/types/types.js +1 -1
  273. package/src/routes.js +14 -0
  274. package/src/server.jsx +28 -23
  275. package/src/start-server.js +2 -2
  276. package/test-setup-config.js +2 -0
  277. package/theme/themes/pastanaga/extras/blocks.less +3 -1
  278. package/theme/themes/pastanaga/extras/contents.less +1 -0
  279. package/theme/themes/pastanaga/extras/grid.less +426 -0
  280. package/theme/themes/pastanaga/extras/main.less +3 -1
  281. package/theme/themes/pastanaga/extras/search.less +6 -0
  282. package/theme/themes/pastanaga/extras/sidebar.less +4 -0
  283. package/theme/themes/pastanaga/extras/toc.less +29 -0
  284. package/theme/themes/pastanaga/extras/userscontrolpanel.less +99 -76
  285. package/.changelog.draft +0 -31
  286. package/.editorconfig +0 -36
  287. package/.storybook/main.js +0 -127
  288. package/.storybook/manager.js +0 -15
  289. package/.storybook/preview.js +0 -21
  290. package/.storybook/static/previewImage.svg +0 -48
  291. package/.yarnrc.yml +0 -5
  292. package/jsdoc.json +0 -16
  293. package/netlify.toml +0 -5
  294. package/pyvenv.cfg +0 -3
  295. package/share/man/man1/ttx.1 +0 -225
  296. package/src/components/theme/Header/Header.md +0 -27
  297. package/towncrier.toml +0 -33
@@ -0,0 +1,114 @@
1
+ /**
2
+ * Relations Control Panel
3
+ */
4
+ import React, { useEffect } from 'react';
5
+ import { find } from 'lodash';
6
+ import { useSelector } from 'react-redux';
7
+ import { Portal } from 'react-portal';
8
+ import { useHistory } from 'react-router';
9
+ import { Link, useLocation } from 'react-router-dom';
10
+ import { FormattedMessage, useIntl } from 'react-intl';
11
+ import { useDispatch } from 'react-redux';
12
+ import { Divider, Message, Segment } from 'semantic-ui-react';
13
+ import { Helmet, messages } from '@plone/volto/helpers';
14
+ import { listActions } from '@plone/volto/actions';
15
+ import { Icon, Toolbar } from '@plone/volto/components';
16
+ import { getParentUrl } from '@plone/volto/helpers';
17
+ import RelationsMatrix from '@plone/volto/components/manage/Controlpanels/Relations/RelationsMatrix';
18
+ import backSVG from '@plone/volto/icons/back.svg';
19
+
20
+ const RelationsControlPanel = () => {
21
+ const intl = useIntl();
22
+ const history = useHistory();
23
+ const location = useLocation();
24
+ const dispatch = useDispatch();
25
+
26
+ const brokenRelations = useSelector(
27
+ (state) => state.relations?.stats?.broken,
28
+ );
29
+
30
+ const relations_stats = useSelector((state) => state.relations?.stats?.stats);
31
+ const actions = useSelector((state) => state.actions?.actions ?? {});
32
+ const can_edit = find(actions.object, {
33
+ id: 'edit',
34
+ });
35
+
36
+ useEffect(() => {
37
+ dispatch(listActions('/'));
38
+ }, [dispatch]);
39
+
40
+ return (
41
+ <>
42
+ <div className="relations-control-panel">
43
+ <Helmet title={intl.formatMessage(messages.relations)} />
44
+ {can_edit ? (
45
+ <Segment.Group raised>
46
+ <Segment className="primary">
47
+ {brokenRelations && Object.keys(brokenRelations).length > 0 ? (
48
+ <React.Fragment>
49
+ <Message warning>
50
+ <FormattedMessage
51
+ id="Some relations are broken. Please fix."
52
+ defaultMessage="Some relations are broken. Please fix."
53
+ />
54
+ </Message>
55
+ <Divider hidden />
56
+ </React.Fragment>
57
+ ) : null}
58
+ <h1>
59
+ <FormattedMessage id="Relations" defaultMessage="Relations" />
60
+ </h1>
61
+ {!relations_stats ? (
62
+ <React.Fragment>
63
+ <Divider hidden />
64
+ <Message warning>
65
+ <FormattedMessage
66
+ id="Please upgrade to plone.restapi >= 8.35.3."
67
+ defaultMessage="Please upgrade to plone.restapi >= 8.35.3."
68
+ />
69
+ </Message>
70
+ </React.Fragment>
71
+ ) : null}
72
+ </Segment>
73
+ <Segment>
74
+ <RelationsMatrix />
75
+ </Segment>
76
+ </Segment.Group>
77
+ ) : (
78
+ <Segment.Group>
79
+ <Segment>
80
+ <FormattedMessage id="Relations" defaultMessage="Relations" />
81
+ <Divider hidden />
82
+ <FormattedMessage
83
+ id="You have not the required permission for this control panel."
84
+ defaultMessage="You have not the required permission for this control panel."
85
+ />
86
+ </Segment>
87
+ </Segment.Group>
88
+ )}
89
+ </div>
90
+
91
+ {__CLIENT__ && (
92
+ <Portal node={document.getElementById('toolbar')}>
93
+ <Toolbar
94
+ pathname={location.pathname}
95
+ hideDefaultViewButtons
96
+ inner={
97
+ <Link
98
+ className="item"
99
+ to="#"
100
+ onClick={() => {
101
+ history.push(getParentUrl(location.pathname));
102
+ }}
103
+ >
104
+ <Icon name={backSVG} className="contents circled" size="30px" />
105
+ </Link>
106
+ }
107
+ />
108
+ </Portal>
109
+ )}
110
+ </>
111
+ );
112
+ };
113
+
114
+ export default RelationsControlPanel;
@@ -0,0 +1,479 @@
1
+ import React, { useEffect } from 'react';
2
+ import useDeepCompareEffect from 'use-deep-compare-effect';
3
+ import { FormattedMessage, useIntl } from 'react-intl';
4
+ import { useSelector, useDispatch } from 'react-redux';
5
+ import { toast } from 'react-toastify';
6
+ import { uniqBy } from 'lodash';
7
+ import { Checkbox, Message } from 'semantic-ui-react';
8
+ import { messages } from '@plone/volto/helpers';
9
+ import { Toast, UniversalLink } from '@plone/volto/components';
10
+ import {
11
+ createRelations,
12
+ deleteRelations,
13
+ queryRelations,
14
+ searchContent,
15
+ } from '@plone/volto/actions';
16
+
17
+ const ListingTemplate = ({
18
+ relationtype,
19
+ query_source,
20
+ query_target,
21
+ potential_sources_path,
22
+ potential_targets_path,
23
+ // target_filter,
24
+ }) => {
25
+ const intl = useIntl();
26
+ const dispatch = useDispatch();
27
+
28
+ const MAX = 40; // Maximum of rows and columns
29
+ const MAX_RELATIONS = 1000;
30
+
31
+ const stats = useSelector((state) => state.relations?.stats || null);
32
+ let relations = useSelector(
33
+ (state) => state.relations?.relations?.[relationtype]?.items || [],
34
+ );
35
+
36
+ let potential_targets_objects = useSelector(
37
+ (state) => state.search.subrequests.potential_targets?.items || [],
38
+ );
39
+
40
+ let potential_sources_objects = useSelector(
41
+ (state) => state.search.subrequests.potential_sources?.items || [],
42
+ );
43
+
44
+ const staticCatalogVocabularyQuery = useSelector(
45
+ (state) =>
46
+ state.relations?.relations?.[relationtype]
47
+ ?.staticCatalogVocabularyQuery || {},
48
+ );
49
+
50
+ // Editable if plone.api.relations available
51
+ const editable = useSelector(
52
+ (state) => state.relations?.relations?.[relationtype]?.readonly !== true,
53
+ );
54
+
55
+ let relationMatrix = {};
56
+ relations.map((tpl) => {
57
+ if (relationMatrix[tpl.source.UID]) {
58
+ relationMatrix[tpl.source.UID].targets.push(tpl.target);
59
+ } else {
60
+ relationMatrix[tpl.source.UID] = {
61
+ source: tpl.source,
62
+ targets: [tpl.target],
63
+ };
64
+ }
65
+ return relationMatrix;
66
+ });
67
+
68
+ // x-axis: relation targets
69
+ // ************************
70
+ let matrix_options = relations.map((relation) => ({
71
+ value: relation.target.UID,
72
+ label: relation.target.title,
73
+ url: relation.target['@id'],
74
+ review_state: relation.target.review_state,
75
+ uid: relation.target.UID,
76
+ }));
77
+ matrix_options = uniqBy(matrix_options, function (el) {
78
+ return el.value;
79
+ });
80
+
81
+ // Add potential targets
82
+ const potential_targets = potential_targets_objects.map((obj) => ({
83
+ value: obj.UID,
84
+ label: obj.title,
85
+ url: obj['@id'],
86
+ review_state: obj.review_state,
87
+ uid: obj.UID,
88
+ }));
89
+ // Just show potential targets if no querying
90
+ matrix_options =
91
+ query_source === '' &&
92
+ query_target === '' &&
93
+ potential_sources_path !== '' &&
94
+ potential_targets_path !== ''
95
+ ? potential_targets
96
+ : [...matrix_options, ...potential_targets];
97
+ matrix_options = uniqBy(matrix_options, function (el) {
98
+ return el.value;
99
+ });
100
+ matrix_options.sort(function (a, b) {
101
+ var labelA = a.label.toUpperCase();
102
+ var labelB = b.label.toUpperCase();
103
+ if (labelA < labelB) {
104
+ return -1;
105
+ }
106
+ if (labelA > labelB) {
107
+ return 1;
108
+ }
109
+ return 0;
110
+ });
111
+
112
+ // y-axis: relation sources
113
+ // ************************
114
+ let items = Object.keys(relationMatrix).map((key) => ({
115
+ value: key,
116
+ label: relationMatrix[key].source.title,
117
+ targets: relationMatrix[key].targets.map((el) => el.UID),
118
+ url: relationMatrix[key].source['@id'],
119
+ review_state: relationMatrix[key].source.review_state,
120
+ }));
121
+ // Add potential sources
122
+ const potential_sources = potential_sources_objects.map((obj) => ({
123
+ value: obj.UID,
124
+ label: obj.title,
125
+ targets: relationMatrix[obj.UID]?.targets?.map((el) => el.UID) || [],
126
+ url: obj['@id'],
127
+ review_state: obj.review_state,
128
+ }));
129
+ items =
130
+ query_source === '' &&
131
+ query_target === '' &&
132
+ potential_sources_path !== '' &&
133
+ potential_targets_path !== ''
134
+ ? potential_sources
135
+ : [...items, ...potential_sources];
136
+ items = uniqBy(items, function (el) {
137
+ return el.value;
138
+ });
139
+ items.sort(function (a, b) {
140
+ var labelA = a.label.toUpperCase();
141
+ var labelB = b.label.toUpperCase();
142
+ if (labelA < labelB) {
143
+ return -1;
144
+ }
145
+ if (labelA > labelB) {
146
+ return 1;
147
+ }
148
+ return 0;
149
+ });
150
+
151
+ useEffect(() => {
152
+ // If many relations, then fetch relations only with search query on source or target
153
+ if (stats?.stats[relationtype] <= MAX_RELATIONS) {
154
+ dispatch(queryRelations(relationtype));
155
+ } else {
156
+ dispatch(
157
+ queryRelations(
158
+ relationtype,
159
+ false,
160
+ null,
161
+ null,
162
+ null,
163
+ query_source
164
+ ? query_source.startsWith('/')
165
+ ? query_source
166
+ : `${query_source}*`
167
+ : null,
168
+ query_target
169
+ ? query_target.startsWith('/')
170
+ ? query_target
171
+ : `${query_target}*`
172
+ : null,
173
+ ),
174
+ );
175
+ }
176
+ }, [dispatch, stats, relationtype, query_source, query_target]);
177
+
178
+ // Get potential source and target objects
179
+ useDeepCompareEffect(() => {
180
+ // Fetch fresh potential targets
181
+ if (potential_targets_path !== '/' && potential_targets_path !== '') {
182
+ dispatch(
183
+ searchContent(
184
+ potential_targets_path,
185
+ {
186
+ SearchableText: query_target,
187
+ metadata_fields: ['UID'],
188
+ sort_on: 'sortable_title',
189
+ ...staticCatalogVocabularyQuery,
190
+ },
191
+ 'potential_targets',
192
+ ),
193
+ );
194
+ } else {
195
+ // TODO Better just reset redux store
196
+ dispatch(searchContent('/findstenichätsch', null, 'potential_targets'));
197
+ }
198
+
199
+ // Fetch fresh potential sources
200
+ if (potential_sources_path !== '/' && potential_sources_path !== '') {
201
+ dispatch(
202
+ searchContent(
203
+ potential_sources_path,
204
+ {
205
+ SearchableText: query_source,
206
+ metadata_fields: ['UID'],
207
+ sort_on: 'sortable_title',
208
+ // No need to restrict here. ...staticCatalogVocabularyQuery,
209
+ },
210
+ 'potential_sources',
211
+ ),
212
+ );
213
+ } else {
214
+ // TODO Better just reset redux store
215
+ dispatch(searchContent('/findstenichätsch', null, 'potential_sources'));
216
+ }
217
+ }, [
218
+ dispatch,
219
+ potential_targets_path,
220
+ potential_sources_path,
221
+ staticCatalogVocabularyQuery,
222
+ query_source,
223
+ query_target,
224
+ ]);
225
+
226
+ function fetchRelations() {
227
+ dispatch(
228
+ queryRelations(
229
+ relationtype,
230
+ false,
231
+ null,
232
+ null,
233
+ null,
234
+ query_source
235
+ ? query_source.startsWith('/')
236
+ ? query_source
237
+ : `${query_source}*`
238
+ : null,
239
+ query_target
240
+ ? query_target.startsWith('/')
241
+ ? query_target
242
+ : `${query_target}*`
243
+ : null,
244
+ ),
245
+ );
246
+ }
247
+
248
+ const onSelectOptionHandler = (relation, selectedvalue, checked) => {
249
+ let source = selectedvalue.y;
250
+ let target = selectedvalue.x;
251
+ const relation_data = [
252
+ {
253
+ source: source,
254
+ target: target,
255
+ relation: relation,
256
+ },
257
+ ];
258
+ dispatch(
259
+ checked ? createRelations(relation_data) : deleteRelations(relation_data),
260
+ )
261
+ .then((resp) => {
262
+ fetchRelations();
263
+ })
264
+ .then(() => {
265
+ toast.success(
266
+ <Toast
267
+ success
268
+ title={intl.formatMessage(messages.success)}
269
+ content={intl.formatMessage(messages.relationsUpdated)}
270
+ />,
271
+ );
272
+ });
273
+ };
274
+
275
+ const onSelectAllHandler = (target, items_ids, checked) => {
276
+ let relation_data = [];
277
+ items_ids.forEach((el) => {
278
+ relation_data.push({
279
+ source: el,
280
+ target: target,
281
+ relation: relationtype,
282
+ });
283
+ });
284
+ dispatch(
285
+ checked ? createRelations(relation_data) : deleteRelations(relation_data),
286
+ )
287
+ .then((resp) => {
288
+ fetchRelations();
289
+ })
290
+ .then(() => {
291
+ toast.success(
292
+ <Toast
293
+ success
294
+ title={intl.formatMessage(messages.success)}
295
+ content={intl.formatMessage(messages.relationsUpdated)}
296
+ />,
297
+ );
298
+ });
299
+ };
300
+
301
+ return (
302
+ <>
303
+ {/* <div>
304
+ <div>
305
+ <div>{items.length} sources</div>
306
+ <div>{matrix_options.length} targets</div>
307
+ </div>
308
+ <div>
309
+ <div>query_source <b>{query_source}</b></div>
310
+ <div>query_target <b>{query_target}</b></div>
311
+ <div>potential_sources_path <b>{potential_sources_path}</b></div>
312
+ <div>potential_targets_path <b>{potential_targets_path}</b></div>
313
+ </div>
314
+ </div> */}
315
+ {matrix_options.length <= MAX &&
316
+ (items.length <= MAX) & (matrix_options.length > 0) &&
317
+ items.length > 0 ? (
318
+ <div className="administration_matrix">
319
+ <div className="label-options">
320
+ <div className="target-labels">
321
+ <div></div>
322
+ <div>
323
+ {matrix_options?.map((matrix_option) => (
324
+ <div
325
+ className="label-options-label inclined"
326
+ id={`label-options-label-${matrix_option.value}`}
327
+ key={matrix_option.value}
328
+ >
329
+ <div>
330
+ <UniversalLink
331
+ href={matrix_option.url}
332
+ className={
333
+ matrix_option.review_state !== 'published'
334
+ ? 'not-published'
335
+ : ''
336
+ }
337
+ target="_blank"
338
+ >
339
+ <span className="label" title={matrix_option.label}>
340
+ {matrix_option.label.length > 30
341
+ ? matrix_option.label.slice(0, 27) + '...'
342
+ : matrix_option.label}
343
+ </span>
344
+ </UniversalLink>
345
+ </div>
346
+ </div>
347
+ ))}
348
+ </div>
349
+ </div>
350
+ <div className="listing-row selectall" key="selectall">
351
+ <div className="listing-item">
352
+ <div />
353
+ <div className="matrix_options">
354
+ {!(
355
+ relationtype === 'isReferencing' ||
356
+ relationtype === 'iterate-working-copy' ||
357
+ !editable
358
+ ) ? (
359
+ matrix_options?.map((matrix_option) => (
360
+ <div
361
+ key={matrix_option.value}
362
+ title={
363
+ intl.formatMessage(
364
+ messages.createOrDeleteRelationsToTarget,
365
+ ) + ` '${matrix_option.label}'`
366
+ }
367
+ >
368
+ <Checkbox
369
+ className="toggle-target"
370
+ defaultChecked={false}
371
+ onChange={(event, { checked }) =>
372
+ onSelectAllHandler(
373
+ matrix_option.value,
374
+ items.map((el) => el.value),
375
+ checked,
376
+ )
377
+ }
378
+ />
379
+ </div>
380
+ ))
381
+ ) : (
382
+ <FormattedMessage
383
+ id="Read only for this type of relation."
384
+ defaultMessage="Read only for this type of relation."
385
+ />
386
+ )}
387
+ </div>
388
+ </div>
389
+ </div>
390
+ </div>
391
+
392
+ <div className="items" key="items">
393
+ <>
394
+ {!editable && (
395
+ <Message warning>
396
+ <FormattedMessage
397
+ id="Relations are editable with plone.api >= 2.0.3."
398
+ defaultMessage="Relations are editable with plone.api >= 2.0.3."
399
+ />
400
+ </Message>
401
+ )}
402
+ {items.map((item) => (
403
+ <div
404
+ className="listing-row"
405
+ key={item.id}
406
+ id={`source-row-${item.value}`}
407
+ >
408
+ <div className="listing-item" key={item['@id']}>
409
+ <div>
410
+ <span title={item.label} className="item-title">
411
+ <UniversalLink
412
+ href={item.url}
413
+ className={
414
+ item.review_state !== 'published'
415
+ ? 'not-published'
416
+ : ''
417
+ }
418
+ target="_blank"
419
+ >
420
+ {item.label.length > 25
421
+ ? item.label.slice(0, 22) + '...'
422
+ : item.label}
423
+ </UniversalLink>
424
+ {/* <span>targets: {item.targets.join(', ')}</span> */}
425
+ </span>
426
+ </div>
427
+ <div className="matrix_options">
428
+ {matrix_options?.map((matrix_option) => (
429
+ <React.Fragment key={matrix_option.value}>
430
+ <Checkbox
431
+ className={`checkbox_${matrix_option.value}`}
432
+ key={matrix_option.value}
433
+ title={matrix_option.title}
434
+ disabled={
435
+ relationtype === 'isReferencing' ||
436
+ relationtype === 'iterate-working-copy' ||
437
+ !editable
438
+ }
439
+ checked={item.targets.includes(matrix_option.value)}
440
+ onChange={(event, { checked }) => {
441
+ onSelectOptionHandler(
442
+ relationtype,
443
+ { x: matrix_option.value, y: item.value },
444
+ checked,
445
+ );
446
+ }}
447
+ />
448
+ </React.Fragment>
449
+ ))}
450
+ </div>
451
+ </div>
452
+ </div>
453
+ ))}
454
+ </>
455
+ </div>
456
+ </div>
457
+ ) : (
458
+ <div className="administration_matrix">
459
+ {matrix_options.length > MAX || items.length > MAX ? (
460
+ <FormattedMessage
461
+ id="narrowDownRelations"
462
+ defaultMessage="Found {sources} sources and {targets} targets. Narrow down to {max}!"
463
+ values={{
464
+ sources: items.length,
465
+ targets: matrix_options.length,
466
+ max: MAX,
467
+ }}
468
+ />
469
+ ) : query_source || query_target ? (
470
+ <div>{intl.formatMessage(messages.norelationfound)}</div>
471
+ ) : (
472
+ <div>{intl.formatMessage(messages.toomanyrelationsfound)}</div>
473
+ )}
474
+ </div>
475
+ )}
476
+ </>
477
+ );
478
+ };
479
+ export default ListingTemplate;