@akemona-org/strapi-plugin-upload 3.7.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 (246) hide show
  1. package/LICENSE +22 -0
  2. package/README.md +19 -0
  3. package/admin/src/assets/images/logo.svg +1 -0
  4. package/admin/src/components/BrowseAssets/Wrapper.js +8 -0
  5. package/admin/src/components/BrowseAssets/index.js +182 -0
  6. package/admin/src/components/Card/index.js +146 -0
  7. package/admin/src/components/CardBorder/index.js +28 -0
  8. package/admin/src/components/CardControl/Wrapper.js +89 -0
  9. package/admin/src/components/CardControl/index.js +61 -0
  10. package/admin/src/components/CardControlsWrapper/index.js +22 -0
  11. package/admin/src/components/CardEmpty/Bar.js +19 -0
  12. package/admin/src/components/CardEmpty/Wrapper.js +7 -0
  13. package/admin/src/components/CardEmpty/index.js +16 -0
  14. package/admin/src/components/CardErrorMessage/index.js +9 -0
  15. package/admin/src/components/CardImgWrapper/index.js +52 -0
  16. package/admin/src/components/CardPreview/Image.js +12 -0
  17. package/admin/src/components/CardPreview/Wrapper.js +25 -0
  18. package/admin/src/components/CardPreview/index.js +66 -0
  19. package/admin/src/components/CardTitle/index.js +14 -0
  20. package/admin/src/components/CardWrapper/index.js +23 -0
  21. package/admin/src/components/CheckControl/Button.js +49 -0
  22. package/admin/src/components/CheckControl/CustomDropdownSection.js +9 -0
  23. package/admin/src/components/CheckControl/Spacer.js +7 -0
  24. package/admin/src/components/CheckControl/index.js +74 -0
  25. package/admin/src/components/Container/index.js +7 -0
  26. package/admin/src/components/ContainerFluid/index.js +12 -0
  27. package/admin/src/components/ControlsWrapper/index.js +9 -0
  28. package/admin/src/components/DragLayer/Wrapper.js +13 -0
  29. package/admin/src/components/DragLayer/index.js +68 -0
  30. package/admin/src/components/DropdownButton/index.js +63 -0
  31. package/admin/src/components/DropdownSection/index.js +27 -0
  32. package/admin/src/components/Duration/index.js +25 -0
  33. package/admin/src/components/Duration/utils/formatDuration.js +13 -0
  34. package/admin/src/components/EditForm/CropWrapper.js +21 -0
  35. package/admin/src/components/EditForm/ErrorMessage.js +10 -0
  36. package/admin/src/components/EditForm/FileDetailsBox.js +95 -0
  37. package/admin/src/components/EditForm/FileDetailsBoxWrapper.js +14 -0
  38. package/admin/src/components/EditForm/FileWrapper.js +41 -0
  39. package/admin/src/components/EditForm/FormWrapper.js +7 -0
  40. package/admin/src/components/EditForm/SizeBox.js +13 -0
  41. package/admin/src/components/EditForm/Wrapper.js +8 -0
  42. package/admin/src/components/EditForm/index.js +425 -0
  43. package/admin/src/components/EditForm/utils/form.js +34 -0
  44. package/admin/src/components/EditForm/utils/isImageType.js +3 -0
  45. package/admin/src/components/EditForm/utils/isVideoType.js +3 -0
  46. package/admin/src/components/FileIcon/Wrapper.js +20 -0
  47. package/admin/src/components/FileIcon/index.js +33 -0
  48. package/admin/src/components/FileIcon/utils/extensions.json +4 -0
  49. package/admin/src/components/FileInfos/index.js +32 -0
  50. package/admin/src/components/Filters/index.js +30 -0
  51. package/admin/src/components/FiltersList/index.js +34 -0
  52. package/admin/src/components/FiltersList/utils/formatFilter.js +36 -0
  53. package/admin/src/components/FiltersPicker/FiltersCard/FilterButton.js +8 -0
  54. package/admin/src/components/FiltersPicker/FiltersCard/FilterInput.js +35 -0
  55. package/admin/src/components/FiltersPicker/FiltersCard/InputWrapper.js +10 -0
  56. package/admin/src/components/FiltersPicker/FiltersCard/SizeInput.js +57 -0
  57. package/admin/src/components/FiltersPicker/FiltersCard/Wrapper.js +8 -0
  58. package/admin/src/components/FiltersPicker/FiltersCard/index.js +97 -0
  59. package/admin/src/components/FiltersPicker/FiltersCard/init.js +21 -0
  60. package/admin/src/components/FiltersPicker/FiltersCard/reducer.js +47 -0
  61. package/admin/src/components/FiltersPicker/FiltersCard/utils/filtersForm.js +27 -0
  62. package/admin/src/components/FiltersPicker/index.js +43 -0
  63. package/admin/src/components/FiltersPicker/utils/formatFilter.js +18 -0
  64. package/admin/src/components/InfiniteLoadingIndicator/Button.js +20 -0
  65. package/admin/src/components/InfiniteLoadingIndicator/Container.js +12 -0
  66. package/admin/src/components/InfiniteLoadingIndicator/Wrapper.js +18 -0
  67. package/admin/src/components/InfiniteLoadingIndicator/index.js +39 -0
  68. package/admin/src/components/Input/index.js +7 -0
  69. package/admin/src/components/InputFileModal/Label.js +47 -0
  70. package/admin/src/components/InputFileModal/P.js +15 -0
  71. package/admin/src/components/InputFileModal/index.js +73 -0
  72. package/admin/src/components/InputMedia/CardControlWrapper.js +12 -0
  73. package/admin/src/components/InputMedia/CardPreviewWrapper.js +13 -0
  74. package/admin/src/components/InputMedia/Chevron/Wrapper.js +10 -0
  75. package/admin/src/components/InputMedia/Chevron/index.js +21 -0
  76. package/admin/src/components/InputMedia/EmptyInputMedia.js +60 -0
  77. package/admin/src/components/InputMedia/EmptyText.js +13 -0
  78. package/admin/src/components/InputMedia/ErrorMessage.js +12 -0
  79. package/admin/src/components/InputMedia/IconUpload.js +48 -0
  80. package/admin/src/components/InputMedia/InputFilePreview.js +37 -0
  81. package/admin/src/components/InputMedia/Name.js +10 -0
  82. package/admin/src/components/InputMedia/Wrapper.js +11 -0
  83. package/admin/src/components/InputMedia/index.js +238 -0
  84. package/admin/src/components/InputUploadURL/Wrapper.js +13 -0
  85. package/admin/src/components/InputUploadURL/index.js +46 -0
  86. package/admin/src/components/IntlText/index.js +26 -0
  87. package/admin/src/components/List/ListCell.js +19 -0
  88. package/admin/src/components/List/ListRow.js +8 -0
  89. package/admin/src/components/List/index.js +114 -0
  90. package/admin/src/components/ListEmpty/Wrapper.js +15 -0
  91. package/admin/src/components/ListEmpty/index.js +70 -0
  92. package/admin/src/components/ListEmpty/utils/generateRows.js +12 -0
  93. package/admin/src/components/ListModal/BaselineAlignmentWrapper.js +12 -0
  94. package/admin/src/components/ListModal/index.js +66 -0
  95. package/admin/src/components/ListWrapper/index.js +16 -0
  96. package/admin/src/components/LoadingIndicator/index.js +44 -0
  97. package/admin/src/components/ModalHeader/BackButton.js +33 -0
  98. package/admin/src/components/ModalHeader/Wrapper.js +24 -0
  99. package/admin/src/components/ModalHeader/index.js +47 -0
  100. package/admin/src/components/ModalNavWrapper/BaselineAlignment.js +12 -0
  101. package/admin/src/components/ModalNavWrapper/Hr.js +14 -0
  102. package/admin/src/components/ModalNavWrapper/Wrapper.js +8 -0
  103. package/admin/src/components/ModalNavWrapper/index.js +70 -0
  104. package/admin/src/components/ModalSection/index.js +20 -0
  105. package/admin/src/components/ModalTab/Count/Wrapper.js +16 -0
  106. package/admin/src/components/ModalTab/Count/index.js +30 -0
  107. package/admin/src/components/ModalTab/Wrapper.js +37 -0
  108. package/admin/src/components/ModalTab/index.js +60 -0
  109. package/admin/src/components/PlayIcon/Play.js +65 -0
  110. package/admin/src/components/PlayIcon/Wrapper.js +24 -0
  111. package/admin/src/components/PlayIcon/index.js +23 -0
  112. package/admin/src/components/SelectAll/Wrapper.js +19 -0
  113. package/admin/src/components/SelectAll/index.js +26 -0
  114. package/admin/src/components/SelectedAssets/DraggableCard.js +149 -0
  115. package/admin/src/components/SelectedAssets/ListTitleWrapper.js +7 -0
  116. package/admin/src/components/SelectedAssets/ListWrapper.js +7 -0
  117. package/admin/src/components/SelectedAssets/SortableList.js +102 -0
  118. package/admin/src/components/SelectedAssets/Wrapper.js +7 -0
  119. package/admin/src/components/SelectedAssets/index.js +56 -0
  120. package/admin/src/components/SortList/Wrapper.js +19 -0
  121. package/admin/src/components/SortList/index.js +37 -0
  122. package/admin/src/components/SortListItem/Wrapper.js +28 -0
  123. package/admin/src/components/SortListItem/index.js +35 -0
  124. package/admin/src/components/SortPicker/index.js +54 -0
  125. package/admin/src/components/Tag/Wrapper.js +16 -0
  126. package/admin/src/components/Tag/index.js +25 -0
  127. package/admin/src/components/UploadForm/index.js +61 -0
  128. package/admin/src/components/UploadList/ButtonWrapper.js +7 -0
  129. package/admin/src/components/UploadList/Container.js +10 -0
  130. package/admin/src/components/UploadList/CustomModalSection.js +9 -0
  131. package/admin/src/components/UploadList/ListTitle.js +10 -0
  132. package/admin/src/components/UploadList/ListTitleWrapper.js +7 -0
  133. package/admin/src/components/UploadList/RowItem.js +98 -0
  134. package/admin/src/components/UploadList/index.js +90 -0
  135. package/admin/src/components/VideoPlayer/Wrapper.js +24 -0
  136. package/admin/src/components/VideoPlayer/index.js +57 -0
  137. package/admin/src/components/VideoPreview/CanvasWrapper.js +22 -0
  138. package/admin/src/components/VideoPreview/Thumbnail.js +12 -0
  139. package/admin/src/components/VideoPreview/Wrapper.js +15 -0
  140. package/admin/src/components/VideoPreview/index.js +165 -0
  141. package/admin/src/components/VideoPreview/reducer.js +33 -0
  142. package/admin/src/containers/App/index.js +31 -0
  143. package/admin/src/containers/HomePage/HomePageContent/HomePageList.js +98 -0
  144. package/admin/src/containers/HomePage/HomePageContent/HomePageSettings.js +103 -0
  145. package/admin/src/containers/HomePage/HomePageContent/index.js +90 -0
  146. package/admin/src/containers/HomePage/components.js +18 -0
  147. package/admin/src/containers/HomePage/index.js +373 -0
  148. package/admin/src/containers/HomePage/init.js +5 -0
  149. package/admin/src/containers/HomePage/reducer.js +88 -0
  150. package/admin/src/containers/HomePage/utils/generateParamsFromQuery.js +14 -0
  151. package/admin/src/containers/HomePage/utils/generateStringFromParams.js +27 -0
  152. package/admin/src/containers/HomePage/utils/getHeaderLabel.js +13 -0
  153. package/admin/src/containers/HomePage/utils/index.js +2 -0
  154. package/admin/src/containers/Initializer/index.js +44 -0
  155. package/admin/src/containers/InputModalStepper/HeaderSearch.js +12 -0
  156. package/admin/src/containers/InputModalStepper/InputModalStepper.js +480 -0
  157. package/admin/src/containers/InputModalStepper/Search.js +48 -0
  158. package/admin/src/containers/InputModalStepper/index.js +86 -0
  159. package/admin/src/containers/InputModalStepper/stepper.js +52 -0
  160. package/admin/src/containers/InputModalStepperProvider/index.js +564 -0
  161. package/admin/src/containers/InputModalStepperProvider/init.js +5 -0
  162. package/admin/src/containers/InputModalStepperProvider/reducer.js +301 -0
  163. package/admin/src/containers/ModalStepper/index.js +629 -0
  164. package/admin/src/containers/ModalStepper/init.js +5 -0
  165. package/admin/src/containers/ModalStepper/reducer.js +132 -0
  166. package/admin/src/containers/ModalStepper/stepper.js +46 -0
  167. package/admin/src/containers/SettingsPage/Divider.js +11 -0
  168. package/admin/src/containers/SettingsPage/SectionTitleWrapper.js +7 -0
  169. package/admin/src/containers/SettingsPage/Wrapper.js +17 -0
  170. package/admin/src/containers/SettingsPage/index.js +198 -0
  171. package/admin/src/containers/SettingsPage/init.js +5 -0
  172. package/admin/src/containers/SettingsPage/reducer.js +38 -0
  173. package/admin/src/contexts/AppContext/index.js +5 -0
  174. package/admin/src/contexts/InputModal/InputModalDataManager.js +5 -0
  175. package/admin/src/contexts/index.js +2 -0
  176. package/admin/src/hooks/index.js +2 -0
  177. package/admin/src/hooks/useAppContext/index.js +6 -0
  178. package/admin/src/hooks/useModalContext.js +6 -0
  179. package/admin/src/icons/BrokenFile/index.js +91 -0
  180. package/admin/src/icons/Cloud/index.js +35 -0
  181. package/admin/src/icons/DoubleFile/index.js +35 -0
  182. package/admin/src/icons/Download/index.js +36 -0
  183. package/admin/src/icons/File/index.js +35 -0
  184. package/admin/src/index.js +85 -0
  185. package/admin/src/lifecycles.js +13 -0
  186. package/admin/src/permissions.js +40 -0
  187. package/admin/src/pluginId.js +5 -0
  188. package/admin/src/translations/de.json +81 -0
  189. package/admin/src/translations/dk.json +81 -0
  190. package/admin/src/translations/en.json +86 -0
  191. package/admin/src/translations/es.json +81 -0
  192. package/admin/src/translations/fr.json +85 -0
  193. package/admin/src/translations/he.json +81 -0
  194. package/admin/src/translations/index.js +39 -0
  195. package/admin/src/translations/it.json +82 -0
  196. package/admin/src/translations/ja.json +81 -0
  197. package/admin/src/translations/ms.json +73 -0
  198. package/admin/src/translations/pl.json +81 -0
  199. package/admin/src/translations/pt-BR.json +81 -0
  200. package/admin/src/translations/ru.json +81 -0
  201. package/admin/src/translations/sk.json +81 -0
  202. package/admin/src/translations/th.json +81 -0
  203. package/admin/src/translations/uk.json +78 -0
  204. package/admin/src/translations/zh-Hans.json +86 -0
  205. package/admin/src/translations/zh.json +81 -0
  206. package/admin/src/utils/ItemTypes.js +3 -0
  207. package/admin/src/utils/canDownloadFile.js +3 -0
  208. package/admin/src/utils/compactParams.js +13 -0
  209. package/admin/src/utils/createFileToDownloadName.js +9 -0
  210. package/admin/src/utils/createMatrix.js +32 -0
  211. package/admin/src/utils/createNewFilesToDownloadArray.js +57 -0
  212. package/admin/src/utils/createNewFilesToUploadArray.js +25 -0
  213. package/admin/src/utils/formatBytes.js +13 -0
  214. package/admin/src/utils/formatFileForEditing.js +20 -0
  215. package/admin/src/utils/formatFilters.js +23 -0
  216. package/admin/src/utils/generatePageFromStart.js +5 -0
  217. package/admin/src/utils/generateStartFromPage.js +5 -0
  218. package/admin/src/utils/getFileModelTimestamps.js +15 -0
  219. package/admin/src/utils/getFilesToDownload.js +5 -0
  220. package/admin/src/utils/getRequestUrl.js +5 -0
  221. package/admin/src/utils/getTrad.js +5 -0
  222. package/admin/src/utils/getType.js +15 -0
  223. package/admin/src/utils/getYupError.js +17 -0
  224. package/admin/src/utils/index.js +20 -0
  225. package/admin/src/utils/unformatBytes.js +16 -0
  226. package/admin/src/utils/urlYupSchema.js +49 -0
  227. package/config/config.json +8 -0
  228. package/config/functions/bootstrap.js +143 -0
  229. package/config/routes.json +98 -0
  230. package/config/schema.graphql.js +96 -0
  231. package/controllers/Upload.js +103 -0
  232. package/controllers/upload/admin.js +213 -0
  233. package/controllers/upload/api.js +125 -0
  234. package/controllers/validation/settings.js +22 -0
  235. package/controllers/validation/upload.js +27 -0
  236. package/documentation/1.0.0/overrides/upload-File.json +263 -0
  237. package/errors.js +48 -0
  238. package/index.js +7 -0
  239. package/middlewares/upload/defaults.json +5 -0
  240. package/middlewares/upload/index.js +35 -0
  241. package/models/File.js +7 -0
  242. package/models/File.settings.json +86 -0
  243. package/package.json +71 -0
  244. package/services/Upload.js +397 -0
  245. package/services/image-manipulation.js +167 -0
  246. package/utils/file.js +23 -0
@@ -0,0 +1,373 @@
1
+ import React, { useCallback, useReducer, useRef, useState, useEffect } from 'react';
2
+ import { get, includes, toString, isEqual, intersectionWith } from 'lodash';
3
+ import { useHistory, useLocation } from 'react-router-dom';
4
+ import { Header } from '@buffetjs/custom';
5
+ import { Button, Padded } from '@buffetjs/core';
6
+ import {
7
+ PopUpWarning,
8
+ useGlobalContext,
9
+ generateFiltersFromSearch,
10
+ generateSearchFromFilters,
11
+ request,
12
+ useQuery,
13
+ } from 'strapi-helper-plugin';
14
+ import { formatFileForEditing, getRequestUrl, getTrad, getFileModelTimestamps } from '../../utils';
15
+ import Container from '../../components/Container';
16
+ import HomePageContent from './HomePageContent';
17
+ import { useAppContext } from '../../hooks';
18
+ import ModalStepper from '../ModalStepper';
19
+ import { generateStringFromParams, getHeaderLabel } from './utils';
20
+ import init from './init';
21
+ import reducer, { initialState } from './reducer';
22
+
23
+ const HomePage = () => {
24
+ const { allowedActions } = useAppContext();
25
+ const { canRead } = allowedActions;
26
+ const { formatMessage, plugins } = useGlobalContext();
27
+ const [, updated_at] = getFileModelTimestamps(plugins);
28
+ const [reducerState, dispatch] = useReducer(reducer, initialState, () =>
29
+ init(initialState, allowedActions)
30
+ );
31
+ const query = useQuery();
32
+ const [isModalOpen, setIsModalOpen] = useState(false);
33
+ const [isPopupOpen, setIsPopupOpen] = useState(false);
34
+ const [fileToEdit, setFileToEdit] = useState(null);
35
+ const [shouldRefetch, setShouldRefetch] = useState(false);
36
+ const [modalInitialStep, setModalInitialStep] = useState('browse');
37
+ const { push } = useHistory();
38
+ const { search } = useLocation();
39
+ const isMounted = useRef(true);
40
+ const {
41
+ data,
42
+ dataCount,
43
+ dataToDelete,
44
+ isLoading,
45
+ shouldRefetchData,
46
+ showModalConfirmButtonLoading,
47
+ } = reducerState.toJS();
48
+ const pluginName = formatMessage({ id: getTrad('plugin.name') });
49
+ const paramsKeys = ['_limit', '_start', '_q', '_sort'];
50
+
51
+ useEffect(() => {
52
+ return () => (isMounted.current = false);
53
+ }, []);
54
+
55
+ useEffect(() => {
56
+ fetchListData();
57
+
58
+ // eslint-disable-next-line react-hooks/exhaustive-deps
59
+ }, [search]);
60
+
61
+ const deleteMedia = async id => {
62
+ const requestURL = getRequestUrl(`files/${id}`);
63
+
64
+ try {
65
+ await request(requestURL, {
66
+ method: 'DELETE',
67
+ });
68
+
69
+ return Promise.resolve();
70
+ } catch (err) {
71
+ const errorMessage = get(err, 'response.payload.message', 'An error occured');
72
+
73
+ return Promise.reject(errorMessage);
74
+ }
75
+ };
76
+
77
+ const fetchData = async () => {
78
+ const dataRequestURL = getRequestUrl('files');
79
+ const params = generateStringFromParams(query);
80
+
81
+ const paramsToSend = params.includes('_sort')
82
+ ? params
83
+ : params.concat(`&_sort=${updated_at}:DESC`);
84
+
85
+ try {
86
+ const data = await request(`${dataRequestURL}?${paramsToSend}`, {
87
+ method: 'GET',
88
+ });
89
+
90
+ return Promise.resolve(data);
91
+ } catch (err) {
92
+ if (isMounted.current) {
93
+ dispatch({ type: 'GET_DATA_ERROR' });
94
+ strapi.notification.toggle({
95
+ type: 'warning',
96
+ message: { id: 'notification.error' },
97
+ });
98
+ }
99
+ }
100
+
101
+ return [];
102
+ };
103
+
104
+ const fetchDataCount = async () => {
105
+ const params = generateStringFromParams(query, ['_limit', '_sort', '_start']);
106
+ const requestURL = getRequestUrl('files/count');
107
+
108
+ try {
109
+ const { count } = await request(`${requestURL}?${params}`, {
110
+ method: 'GET',
111
+ });
112
+
113
+ return Promise.resolve(count);
114
+ } catch (err) {
115
+ if (isMounted.current) {
116
+ dispatch({ type: 'GET_DATA_ERROR' });
117
+ strapi.notification.toggle({
118
+ type: 'warning',
119
+ message: { id: 'notification.error' },
120
+ });
121
+ }
122
+ }
123
+
124
+ return null;
125
+ };
126
+
127
+ const fetchListData = async () => {
128
+ if (canRead) {
129
+ dispatch({ type: 'GET_DATA' });
130
+
131
+ const [data, count] = await Promise.all([fetchData(), fetchDataCount()]);
132
+
133
+ if (isMounted.current) {
134
+ dispatch({
135
+ type: 'GET_DATA_SUCCEEDED',
136
+ data,
137
+ count,
138
+ });
139
+ }
140
+ }
141
+ };
142
+
143
+ const generateNewSearch = (updatedParams = {}) => {
144
+ return {
145
+ ...getSearchParams(),
146
+ filters: generateFiltersFromSearch(search),
147
+ ...updatedParams,
148
+ };
149
+ };
150
+
151
+ const getSearchParams = () => {
152
+ const params = {};
153
+
154
+ query.forEach((value, key) => {
155
+ if (includes(paramsKeys, key)) {
156
+ params[key] = value;
157
+ }
158
+ });
159
+
160
+ return params;
161
+ };
162
+
163
+ const handleChangeCheck = ({ target: { name } }) => {
164
+ dispatch({
165
+ type: 'ON_CHANGE_DATA_TO_DELETE',
166
+ id: name,
167
+ });
168
+ };
169
+
170
+ const handleChangeParams = ({ target: { name, value } }) => {
171
+ let updatedQueryParams = generateNewSearch({ [name]: value });
172
+
173
+ if (name === 'filters') {
174
+ const existingFilters = generateFiltersFromSearch(search);
175
+ const canAddFilter = intersectionWith(existingFilters, [value], isEqual).length === 0;
176
+ updatedQueryParams = generateNewSearch({ [name]: existingFilters });
177
+
178
+ if (canAddFilter) {
179
+ const filters = [...existingFilters, value];
180
+
181
+ updatedQueryParams = generateNewSearch({ [name]: filters });
182
+ }
183
+ }
184
+
185
+ if (name === '_limit') {
186
+ updatedQueryParams = generateNewSearch({ [name]: value, _start: 0 });
187
+ }
188
+
189
+ const newSearch = generateSearchFromFilters(updatedQueryParams);
190
+
191
+ push({ search: newSearch });
192
+ };
193
+
194
+ const handleClickEditFile = id => {
195
+ if (allowedActions.canUpdate) {
196
+ const file = formatFileForEditing(data.find(file => toString(file.id) === toString(id)));
197
+
198
+ setFileToEdit(file);
199
+ setModalInitialStep('edit');
200
+ handleClickToggleModal();
201
+ }
202
+ };
203
+
204
+ const handleClickToggleModal = (refetch = false) => {
205
+ setIsModalOpen(prev => !prev);
206
+ setShouldRefetch(refetch);
207
+ };
208
+
209
+ const handleClickTogglePopup = () => {
210
+ setIsPopupOpen(prev => !prev);
211
+ };
212
+
213
+ const handleDeleteFilter = index => {
214
+ const filters = generateFiltersFromSearch(search).filter(
215
+ (filter, filterIndex) => filterIndex !== index
216
+ );
217
+
218
+ const updatedQueryParams = generateNewSearch({ filters });
219
+
220
+ const newSearch = generateSearchFromFilters(updatedQueryParams);
221
+
222
+ push({ search: newSearch });
223
+ };
224
+
225
+ const handleConfirmDeleteMedias = useCallback(async () => {
226
+ dispatch({ type: 'ON_DELETE_MEDIAS' });
227
+
228
+ try {
229
+ await Promise.all(dataToDelete.map(item => deleteMedia(item.id)));
230
+
231
+ dispatch({
232
+ type: 'ON_DELETE_MEDIAS_SUCCEEDED',
233
+ });
234
+ } catch (err) {
235
+ strapi.notification.toggle({
236
+ type: 'warning',
237
+ message: err,
238
+ });
239
+
240
+ dispatch({
241
+ type: 'ON_DELETE_MEDIAS_ERROR',
242
+ });
243
+ } finally {
244
+ setIsPopupOpen(false);
245
+ }
246
+ }, [dataToDelete]);
247
+
248
+ const handleClosedModalDeleteAll = useCallback(() => {
249
+ if (shouldRefetchData) {
250
+ fetchListData();
251
+ } else {
252
+ dispatch({ type: 'RESET_DATA_TO_DELETE' });
253
+ }
254
+ // eslint-disable-next-line react-hooks/exhaustive-deps
255
+ }, [shouldRefetchData]);
256
+
257
+ const handleModalClose = () => {
258
+ resetModalState();
259
+
260
+ if (shouldRefetch) {
261
+ fetchListData();
262
+ setShouldRefetch(false);
263
+ }
264
+ };
265
+
266
+ const handleSelectAll = () => {
267
+ dispatch({
268
+ type: 'TOGGLE_SELECT_ALL',
269
+ });
270
+ };
271
+
272
+ const resetModalState = () => {
273
+ setModalInitialStep('browse');
274
+ setFileToEdit(null);
275
+ };
276
+
277
+ const headerProps = {
278
+ title: {
279
+ label: pluginName,
280
+ },
281
+ /* eslint-disable indent */
282
+ content: canRead
283
+ ? formatMessage(
284
+ {
285
+ id: getTrad(getHeaderLabel(dataCount)),
286
+ },
287
+ { number: dataCount }
288
+ )
289
+ : null,
290
+ /* eslint-enable indent */
291
+ actions: [
292
+ {
293
+ disabled: dataToDelete.length === 0,
294
+ color: 'delete',
295
+ // TradId from the strapi-admin package
296
+ label: formatMessage({ id: 'app.utils.delete' }),
297
+ onClick: () => setIsPopupOpen(true),
298
+ type: 'button',
299
+ Component: buttonProps => {
300
+ if (!allowedActions.canUpdate) {
301
+ return null;
302
+ }
303
+
304
+ return <Button {...buttonProps} />;
305
+ },
306
+ },
307
+ {
308
+ disabled: false,
309
+ color: 'primary',
310
+ label: formatMessage({ id: getTrad('header.actions.upload-assets') }),
311
+ onClick: () => handleClickToggleModal(),
312
+ type: 'button',
313
+ Component: buttonProps => {
314
+ if (!allowedActions.canCreate) {
315
+ return null;
316
+ }
317
+
318
+ return <Button {...buttonProps} />;
319
+ },
320
+ },
321
+ ],
322
+ };
323
+
324
+ const handleRemoveFileFromDataToDelete = useCallback(id => {
325
+ dispatch({
326
+ type: 'ON_CHANGE_DATA_TO_DELETE',
327
+ id,
328
+ });
329
+ }, []);
330
+
331
+ const content = canRead ? (
332
+ <HomePageContent
333
+ data={data}
334
+ dataCount={dataCount}
335
+ dataToDelete={dataToDelete}
336
+ isLoading={isLoading}
337
+ onCardCheck={handleChangeCheck}
338
+ onCardClick={handleClickEditFile}
339
+ onClick={handleClickToggleModal}
340
+ onFilterDelete={handleDeleteFilter}
341
+ onParamsChange={handleChangeParams}
342
+ onSelectAll={handleSelectAll}
343
+ />
344
+ ) : null;
345
+
346
+ return (
347
+ <Container>
348
+ <Header {...headerProps} isLoading={isLoading} />
349
+ {content}
350
+ <ModalStepper
351
+ initialFileToEdit={fileToEdit}
352
+ initialStep={modalInitialStep}
353
+ isOpen={isModalOpen}
354
+ onClosed={handleModalClose}
355
+ onRemoveFileFromDataToDelete={handleRemoveFileFromDataToDelete}
356
+ onToggle={handleClickToggleModal}
357
+ refetchData={fetchListData}
358
+ />
359
+ <PopUpWarning
360
+ isOpen={isPopupOpen}
361
+ isConfirmButtonLoading={showModalConfirmButtonLoading}
362
+ onConfirm={handleConfirmDeleteMedias}
363
+ onClosed={handleClosedModalDeleteAll}
364
+ toggleModal={handleClickTogglePopup}
365
+ popUpWarningType="danger"
366
+ />
367
+ <Padded bottom size="md" />
368
+ <Padded bottom size="md" />
369
+ </Container>
370
+ );
371
+ };
372
+
373
+ export default HomePage;
@@ -0,0 +1,5 @@
1
+ const init = (initialState, allowedActions) => {
2
+ return initialState.set('isLoading', allowedActions.canRead);
3
+ };
4
+
5
+ export default init;
@@ -0,0 +1,88 @@
1
+ import { fromJS } from 'immutable';
2
+
3
+ const initialState = fromJS({
4
+ data: [],
5
+ dataCount: 0,
6
+ dataToDelete: [],
7
+ isLoading: true,
8
+ showModalConfirmButtonLoading: false,
9
+ shouldRefetchData: false,
10
+ });
11
+
12
+ const reducer = (state, action) => {
13
+ switch (action.type) {
14
+ case 'CLEAR_DATA_TO_DELETE':
15
+ return state.update('dataToDelete', () => fromJS([]));
16
+ case 'GET_DATA':
17
+ return state.update('isLoading', () => true);
18
+ case 'GET_DATA_ERROR':
19
+ return state.update('isLoading', () => false);
20
+ case 'GET_DATA_SUCCEEDED':
21
+ return state
22
+ .update('data', () => fromJS(action.data))
23
+ .update('dataCount', () => action.count)
24
+ .update('isLoading', () => false);
25
+ case 'ON_CHANGE_DATA_TO_DELETE': {
26
+ const { id } = action;
27
+ const isSelected = state
28
+ .get('dataToDelete')
29
+ .find(item => item.get('id').toString() === id.toString());
30
+
31
+ if (!isSelected) {
32
+ const item = state.get('data').find(item => item.get('id').toString() === id.toString());
33
+
34
+ return state.update('dataToDelete', dataToDelete => {
35
+ return dataToDelete.push(item);
36
+ });
37
+ }
38
+
39
+ const index = state
40
+ .get('dataToDelete')
41
+ .findIndex(item => item.get('id').toString() === id.toString());
42
+
43
+ return state.removeIn(['dataToDelete', index]);
44
+ }
45
+ case 'ON_DELETE_MEDIAS': {
46
+ return state.update('showModalConfirmButtonLoading', () => true);
47
+ }
48
+ case 'ON_DELETE_MEDIAS_SUCCEEDED': {
49
+ return state
50
+ .update('dataToDelete', () => fromJS([]))
51
+ .update('shouldRefetchData', () => true)
52
+ .update('showModalConfirmButtonLoading', () => false);
53
+ }
54
+ case 'ON_DELETE_MEDIAS_ERROR': {
55
+ return state
56
+ .update('dataToDelete', () => fromJS([]))
57
+ .update('showModalConfirmButtonLoading', () => false);
58
+ }
59
+ case 'RESET_DATA_TO_DELETE': {
60
+ return state
61
+ .update('dataToDelete', () => fromJS([]))
62
+ .update('shouldRefetchData', () => false)
63
+ .update('showModalConfirmButtonLoading', () => false);
64
+ }
65
+ case 'TOGGLE_SELECT_ALL': {
66
+ const isSelected = state.get('data').every(item => state.get('dataToDelete').includes(item));
67
+
68
+ if (isSelected) {
69
+ return state.update('dataToDelete', dataToDelete =>
70
+ dataToDelete.filter(item => !state.get('data').includes(item))
71
+ );
72
+ }
73
+
74
+ return state.update('dataToDelete', dataToDelete => {
75
+ const newItems = state.get('data').filter(item => {
76
+ return !state.get('dataToDelete').includes(item);
77
+ });
78
+
79
+ return dataToDelete.concat(newItems);
80
+ });
81
+ }
82
+ default:
83
+ return state;
84
+ }
85
+ };
86
+
87
+ export default reducer;
88
+ export { initialState };
@@ -0,0 +1,14 @@
1
+ const generateParamsFromQuery = query => {
2
+ const params = {
3
+ _limit: 10,
4
+ _start: 0,
5
+ };
6
+
7
+ query.forEach((value, key) => {
8
+ params[key] = value;
9
+ });
10
+
11
+ return params;
12
+ };
13
+
14
+ export default generateParamsFromQuery;
@@ -0,0 +1,27 @@
1
+ import { isEmpty, toString } from 'lodash';
2
+ import generateParamsFromQuery from './generateParamsFromQuery';
3
+
4
+ const generateStringFromParams = (query, paramsToFilter = []) => {
5
+ let paramsString = '';
6
+ const paramsObject = generateParamsFromQuery(query);
7
+
8
+ Object.keys(paramsObject)
9
+ .filter(key => {
10
+ return !paramsToFilter.includes(key) && !isEmpty(toString(paramsObject[key]));
11
+ })
12
+ .forEach(key => {
13
+ const value = paramsObject[key];
14
+
15
+ if (key.includes('mime') && value === 'file') {
16
+ const revertedKey = key.includes('_ncontains') ? 'mime_contains' : 'mime_ncontains';
17
+
18
+ paramsString += `&${revertedKey}=image&${revertedKey}=video`;
19
+ } else {
20
+ paramsString += `&${key}=${value}`;
21
+ }
22
+ });
23
+
24
+ return paramsString.substring(1);
25
+ };
26
+
27
+ export default generateStringFromParams;
@@ -0,0 +1,13 @@
1
+ const getHeaderLabel = count => {
2
+ let headerLabel = 'header.content.assets-single';
3
+
4
+ if (count === 0) {
5
+ headerLabel = 'header.content.assets-empty';
6
+ } else if (count > 1) {
7
+ headerLabel = 'header.content.assets-multiple';
8
+ }
9
+
10
+ return headerLabel;
11
+ };
12
+
13
+ export default getHeaderLabel;
@@ -0,0 +1,2 @@
1
+ export { default as generateStringFromParams } from './generateStringFromParams';
2
+ export { default as getHeaderLabel } from './getHeaderLabel';
@@ -0,0 +1,44 @@
1
+ /**
2
+ *
3
+ * Initializer
4
+ *
5
+ */
6
+
7
+ import { useEffect, useRef } from 'react';
8
+ import PropTypes from 'prop-types';
9
+ import { request } from 'strapi-helper-plugin';
10
+ import pluginId from '../../pluginId';
11
+
12
+ const Initializer = ({ updatePlugin }) => {
13
+ const ref = useRef();
14
+ ref.current = updatePlugin;
15
+
16
+ useEffect(() => {
17
+ const getData = async () => {
18
+ const requestURL = '/content-manager/content-types';
19
+
20
+ try {
21
+ const { data } = await request(requestURL, { method: 'GET' });
22
+ const fileModel = data.find(model => model.uid === 'plugins::upload.file');
23
+
24
+ ref.current(pluginId, 'fileModel', fileModel);
25
+ ref.current(pluginId, 'isReady', true);
26
+ } catch (err) {
27
+ strapi.notification.toggle({
28
+ type: 'warning',
29
+ message: { id: 'content-manager.error.model.fetch' },
30
+ });
31
+ }
32
+ };
33
+
34
+ getData();
35
+ }, []);
36
+
37
+ return null;
38
+ };
39
+
40
+ Initializer.propTypes = {
41
+ updatePlugin: PropTypes.func.isRequired,
42
+ };
43
+
44
+ export default Initializer;
@@ -0,0 +1,12 @@
1
+ import styled from 'styled-components';
2
+ import { HeaderSearch as BaseHeaderSearch } from 'strapi-helper-plugin';
3
+
4
+ const HeaderSearch = styled(BaseHeaderSearch)`
5
+ position: relative;
6
+ margin-top: -1px;
7
+ line-height: 5.8rem;
8
+ border-right: none;
9
+ background-color: transparent;
10
+ `;
11
+
12
+ export default HeaderSearch;