@iobroker/adapter-react-v5 6.1.9 → 7.0.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 (210) hide show
  1. package/Components/404.js +13 -13
  2. package/Components/FileBrowser.js +24 -19
  3. package/Components/FileViewer.js +14 -5
  4. package/Components/Loader.js +223 -223
  5. package/Components/Loaders/PT.css +108 -108
  6. package/Components/Loaders/PT.js +103 -103
  7. package/Components/Loaders/Vendor.css +13 -13
  8. package/Components/Loaders/Vendor.js +7 -7
  9. package/Components/ObjectBrowser.d.ts +2 -0
  10. package/Components/ObjectBrowser.js +214 -115
  11. package/Components/UploadImage.js +305 -305
  12. package/Components/loader.css +221 -221
  13. package/Components/types.d.ts +82 -82
  14. package/GenericApp.js +49 -49
  15. package/LICENSE +22 -22
  16. package/Prompt.js +7 -7
  17. package/README.md +1004 -1008
  18. package/Theme.js +8 -7
  19. package/assets/devices/Alarm Systems.svg +18 -18
  20. package/assets/devices/Amplifier.svg +21 -21
  21. package/assets/devices/Awnings.svg +4 -4
  22. package/assets/devices/Battery Status.svg +4 -4
  23. package/assets/devices/Ceiling Spotlights.svg +15 -15
  24. package/assets/devices/Chandelier.svg +6 -6
  25. package/assets/devices/Climate.svg +11 -11
  26. package/assets/devices/Coffee Makers.svg +5 -5
  27. package/assets/devices/Cold Water.svg +31 -31
  28. package/assets/devices/Computer.svg +21 -21
  29. package/assets/devices/Consumption.svg +7 -7
  30. package/assets/devices/Curtains.svg +43 -43
  31. package/assets/devices/Dishwashers.svg +11 -11
  32. package/assets/devices/Doors.svg +5 -5
  33. package/assets/devices/Doorstep.svg +35 -35
  34. package/assets/devices/Dryer.svg +13 -13
  35. package/assets/devices/Fan.svg +20 -20
  36. package/assets/devices/Floor Lamps.svg +4 -4
  37. package/assets/devices/Garage Doors.svg +9 -9
  38. package/assets/devices/Gates.svg +32 -32
  39. package/assets/devices/Hairdryer.svg +23 -23
  40. package/assets/devices/Handle.svg +6 -6
  41. package/assets/devices/Hanging Lamps.svg +8 -8
  42. package/assets/devices/Heater.svg +44 -44
  43. package/assets/devices/Hoods.svg +11 -11
  44. package/assets/devices/Hot Water.svg +9 -9
  45. package/assets/devices/Humidity.svg +41 -41
  46. package/assets/devices/Iron.svg +4 -4
  47. package/assets/devices/Irrigation.svg +22 -22
  48. package/assets/devices/Led Strip.svg +30 -30
  49. package/assets/devices/Light.svg +29 -29
  50. package/assets/devices/Lightings.svg +46 -46
  51. package/assets/devices/Lock.svg +19 -19
  52. package/assets/devices/Louvre.svg +6 -6
  53. package/assets/devices/Mowing Machine.svg +8 -8
  54. package/assets/devices/Music.svg +12 -12
  55. package/assets/devices/Outdoor Blinds.svg +6 -6
  56. package/assets/devices/People.svg +19 -19
  57. package/assets/devices/Pool.svg +7 -7
  58. package/assets/devices/Power Consumption.svg +12 -12
  59. package/assets/devices/Printer.svg +9 -9
  60. package/assets/devices/Pump.svg +9 -9
  61. package/assets/devices/Receiver.svg +18 -18
  62. package/assets/devices/Sconces.svg +9 -9
  63. package/assets/devices/Security.svg +34 -34
  64. package/assets/devices/Shading.svg +4 -4
  65. package/assets/devices/Shutters.svg +10 -10
  66. package/assets/devices/SmokeDetector.svg +12 -12
  67. package/assets/devices/Sockets.svg +13 -13
  68. package/assets/devices/Speaker.svg +35 -35
  69. package/assets/devices/Stove.svg +11 -11
  70. package/assets/devices/Table Lamps.svg +11 -11
  71. package/assets/devices/Temperature Sensors.svg +28 -28
  72. package/assets/devices/Tv.svg +7 -7
  73. package/assets/devices/Vacuum Cleaner.svg +15 -15
  74. package/assets/devices/Ventilation.svg +12 -12
  75. package/assets/devices/Washing Machines.svg +15 -15
  76. package/assets/devices/Water Consumption.svg +5 -5
  77. package/assets/devices/Water Heater.svg +8 -8
  78. package/assets/devices/Water.svg +40 -40
  79. package/assets/devices/Weather.svg +28 -28
  80. package/assets/devices/Window.svg +7 -7
  81. package/assets/lamp_ceiling.svg +8 -8
  82. package/assets/lamp_table.svg +7 -7
  83. package/assets/no_icon.svg +9 -9
  84. package/assets/rooms/Anteroom.svg +52 -52
  85. package/assets/rooms/Attic.svg +21 -21
  86. package/assets/rooms/Balcony.svg +12 -12
  87. package/assets/rooms/Barn.svg +5 -5
  88. package/assets/rooms/Basement.svg +4 -4
  89. package/assets/rooms/Bathroom.svg +38 -38
  90. package/assets/rooms/Bedroom.svg +5 -5
  91. package/assets/rooms/Boiler Room.svg +12 -12
  92. package/assets/rooms/Carport.svg +17 -17
  93. package/assets/rooms/Cellar.svg +89 -89
  94. package/assets/rooms/Chamber.svg +9 -9
  95. package/assets/rooms/Corridor.svg +52 -52
  96. package/assets/rooms/Dining Area.svg +37 -37
  97. package/assets/rooms/Dining Room.svg +37 -37
  98. package/assets/rooms/Dining.svg +37 -37
  99. package/assets/rooms/Dressing Room.svg +4 -4
  100. package/assets/rooms/Driveway.svg +14 -14
  101. package/assets/rooms/Entrance.svg +44 -44
  102. package/assets/rooms/Equipment Room.svg +14 -14
  103. package/assets/rooms/Front Yard.svg +64 -64
  104. package/assets/rooms/Gallery.svg +13 -13
  105. package/assets/rooms/Garage.svg +20 -20
  106. package/assets/rooms/Garden.svg +12 -12
  107. package/assets/rooms/Ground Floor.svg +95 -95
  108. package/assets/rooms/Guest Bathroom.svg +32 -32
  109. package/assets/rooms/Guest Room.svg +5 -5
  110. package/assets/rooms/Gym.svg +4 -4
  111. package/assets/rooms/Hall.svg +19 -19
  112. package/assets/rooms/Home Theater.svg +7 -7
  113. package/assets/rooms/Kitchen.svg +17 -17
  114. package/assets/rooms/Laundry Room.svg +11 -11
  115. package/assets/rooms/Living Area.svg +10 -10
  116. package/assets/rooms/Living Room.svg +10 -10
  117. package/assets/rooms/Locker Room.svg +16 -16
  118. package/assets/rooms/Nursery.svg +4 -4
  119. package/assets/rooms/Office.svg +8 -8
  120. package/assets/rooms/Outdoors.svg +7 -7
  121. package/assets/rooms/Playroom.svg +5 -5
  122. package/assets/rooms/Pool.svg +7 -7
  123. package/assets/rooms/Rear Wall.svg +30 -30
  124. package/assets/rooms/Second Floor.svg +95 -95
  125. package/assets/rooms/Shed.svg +16 -16
  126. package/assets/rooms/Sleeping Area.svg +22 -22
  127. package/assets/rooms/Stairway.svg +4 -4
  128. package/assets/rooms/Stairwell.svg +15 -15
  129. package/assets/rooms/Storeroom.svg +4 -4
  130. package/assets/rooms/Summer House.svg +27 -27
  131. package/assets/rooms/Swimming Pool.svg +21 -21
  132. package/assets/rooms/Terrace.svg +6 -6
  133. package/assets/rooms/Toilet.svg +10 -10
  134. package/assets/rooms/Upstairs.svg +5 -5
  135. package/assets/rooms/Wardrobe.svg +60 -60
  136. package/assets/rooms/Washroom.svg +19 -19
  137. package/assets/rooms/Wc.svg +10 -10
  138. package/assets/rooms/Windscreen.svg +60 -60
  139. package/assets/rooms/Workshop.svg +22 -22
  140. package/assets/rooms/Workspace.svg +8 -8
  141. package/craco-module-federation.js +71 -71
  142. package/icons/IconFx.js +1 -1
  143. package/icons/IconLogout.js +1 -1
  144. package/index.css +54 -54
  145. package/modulefederation.admin.config.js +31 -31
  146. package/package.json +9 -9
  147. package/src/AdminConnection.tsx +3 -3
  148. package/src/Components/404.tsx +121 -121
  149. package/src/Components/ColorPicker.tsx +315 -315
  150. package/src/Components/ComplexCron.tsx +507 -507
  151. package/src/Components/CopyToClipboard.tsx +165 -165
  152. package/src/Components/CustomModal.tsx +163 -163
  153. package/src/Components/FileBrowser.tsx +2414 -2394
  154. package/src/Components/FileViewer.tsx +393 -384
  155. package/src/Components/Icon.tsx +210 -210
  156. package/src/Components/IconPicker.tsx +149 -149
  157. package/src/Components/IconSelector.tsx +2202 -2202
  158. package/src/Components/Image.tsx +176 -176
  159. package/src/Components/Loader.tsx +304 -304
  160. package/src/Components/Logo.tsx +166 -166
  161. package/src/Components/MDUtils.tsx +100 -100
  162. package/src/Components/ObjectBrowser.tsx +8032 -7915
  163. package/src/Components/Router.tsx +90 -90
  164. package/src/Components/SaveCloseButtons.tsx +113 -113
  165. package/src/Components/Schedule.tsx +1724 -1724
  166. package/src/Components/SelectWithIcon.tsx +197 -197
  167. package/src/Components/TabContainer.tsx +55 -55
  168. package/src/Components/TabContent.tsx +37 -37
  169. package/src/Components/TabHeader.tsx +19 -19
  170. package/src/Components/TableResize.tsx +259 -259
  171. package/src/Components/TextWithIcon.tsx +148 -148
  172. package/src/Components/ToggleThemeMenu.tsx +34 -34
  173. package/src/Components/TreeTable.tsx +919 -919
  174. package/src/Components/UploadImage.tsx +599 -599
  175. package/src/Components/Utils.tsx +1794 -1794
  176. package/src/Components/loader.css +221 -221
  177. package/src/Components/withWidth.tsx +21 -21
  178. package/src/Connection.tsx +7 -7
  179. package/src/Dialogs/ComplexCron.tsx +129 -129
  180. package/src/Dialogs/Confirm.tsx +162 -162
  181. package/src/Dialogs/Cron.tsx +182 -182
  182. package/src/Dialogs/Error.tsx +72 -72
  183. package/src/Dialogs/Message.tsx +71 -71
  184. package/src/Dialogs/SelectFile.tsx +270 -270
  185. package/src/Dialogs/SelectID.tsx +298 -298
  186. package/src/Dialogs/SimpleCron.tsx +100 -100
  187. package/src/Dialogs/TextInput.tsx +107 -107
  188. package/src/GenericApp.tsx +976 -976
  189. package/src/LegacyConnection.tsx +3589 -3589
  190. package/src/Prompt.tsx +20 -20
  191. package/src/Theme.tsx +479 -479
  192. package/src/icons/IconAdapter.tsx +20 -20
  193. package/src/icons/IconAlias.tsx +20 -20
  194. package/src/icons/IconChannel.tsx +21 -21
  195. package/src/icons/IconClearFilter.tsx +22 -22
  196. package/src/icons/IconClosed.tsx +17 -17
  197. package/src/icons/IconCopy.tsx +16 -16
  198. package/src/icons/IconDevice.tsx +27 -27
  199. package/src/icons/IconDocument.tsx +17 -17
  200. package/src/icons/IconDocumentReadOnly.tsx +18 -18
  201. package/src/icons/IconExpert.tsx +18 -18
  202. package/src/icons/IconFx.tsx +36 -36
  203. package/src/icons/IconInstance.tsx +20 -20
  204. package/src/icons/IconLogout.tsx +30 -30
  205. package/src/icons/IconNoIcon.tsx +19 -19
  206. package/src/icons/IconOpen.tsx +17 -17
  207. package/src/icons/IconProps.tsx +15 -15
  208. package/src/icons/IconState.tsx +17 -17
  209. package/src/index.css +54 -54
  210. package/types.d.ts +134 -134
@@ -1,599 +1,599 @@
1
- import React, { Component, createRef } from 'react';
2
- import Dropzone from 'react-dropzone';
3
- import { Cropper, ReactCropperElement } from 'react-cropper';
4
-
5
- import {
6
- Menu, MenuItem, Tooltip, IconButton,
7
- } from '@mui/material';
8
-
9
- import {
10
- Close as IconClose,
11
- Crop as CropIcon,
12
- } from '@mui/icons-material';
13
- import { FaFileUpload as UploadIcon } from 'react-icons/fa';
14
-
15
- import I18n from '../i18n';
16
- import Icon from './Icon';
17
-
18
- // import 'cropperjs/dist/cropper.css';
19
- const cropperStyles = `
20
- /*!
21
- * Cropper.js v1.5.12
22
- * https://fengyuanchen.github.io/cropperjs
23
- *
24
- * Copyright 2015-present Chen Fengyuan
25
- * Released under the MIT license
26
- *
27
- * Date: 2021-06-12T08:00:11.623Z
28
- */
29
-
30
- .cropper-container {
31
- direction: ltr;
32
- font-size: 0;
33
- line-height: 0;
34
- position: relative;
35
- -ms-touch-action: none;
36
- touch-action: none;
37
- -webkit-user-select: none;
38
- -moz-user-select: none;
39
- -ms-user-select: none;
40
- user-select: none;
41
- }
42
-
43
- .cropper-container img {
44
- display: block;
45
- height: 100%;
46
- image-orientation: 0deg;
47
- max-height: none !important;
48
- max-width: none !important;
49
- min-height: 0 !important;
50
- min-width: 0 !important;
51
- width: 100%;
52
- }
53
-
54
- .cropper-wrap-box,
55
- .cropper-canvas,
56
- .cropper-drag-box,
57
- .cropper-crop-box,
58
- .cropper-modal {
59
- bottom: 0;
60
- left: 0;
61
- position: absolute;
62
- right: 0;
63
- top: 0;
64
- }
65
-
66
- .cropper-wrap-box,
67
- .cropper-canvas {
68
- overflow: hidden;
69
- }
70
-
71
- .cropper-drag-box {
72
- background-color: #fff;
73
- opacity: 0;
74
- }
75
-
76
- .cropper-modal {
77
- background-color: #000;
78
- opacity: 0.5;
79
- }
80
-
81
- .cropper-view-box {
82
- display: block;
83
- height: 100%;
84
- outline: 1px solid #39f;
85
- outline-color: rgba(51, 153, 255, 0.75);
86
- overflow: hidden;
87
- width: 100%;
88
- }
89
-
90
- .cropper-dashed {
91
- border: 0 dashed #eee;
92
- display: block;
93
- opacity: 0.5;
94
- position: absolute;
95
- }
96
-
97
- .cropper-dashed.dashed-h {
98
- border-bottom-width: 1px;
99
- border-top-width: 1px;
100
- height: calc(100% / 3);
101
- left: 0;
102
- top: calc(100% / 3);
103
- width: 100%;
104
- }
105
-
106
- .cropper-dashed.dashed-v {
107
- border-left-width: 1px;
108
- border-right-width: 1px;
109
- height: 100%;
110
- left: calc(100% / 3);
111
- top: 0;
112
- width: calc(100% / 3);
113
- }
114
-
115
- .cropper-center {
116
- display: block;
117
- height: 0;
118
- left: 50%;
119
- opacity: 0.75;
120
- position: absolute;
121
- top: 50%;
122
- width: 0;
123
- }
124
-
125
- .cropper-center::before,
126
- .cropper-center::after {
127
- background-color: #eee;
128
- content: ' ';
129
- display: block;
130
- position: absolute;
131
- }
132
-
133
- .cropper-center::before {
134
- height: 1px;
135
- left: -3px;
136
- top: 0;
137
- width: 7px;
138
- }
139
-
140
- .cropper-center::after {
141
- height: 7px;
142
- left: 0;
143
- top: -3px;
144
- width: 1px;
145
- }
146
-
147
- .cropper-face,
148
- .cropper-line,
149
- .cropper-point {
150
- display: block;
151
- height: 100%;
152
- opacity: 0.1;
153
- position: absolute;
154
- width: 100%;
155
- }
156
-
157
- .cropper-face {
158
- background-color: #fff;
159
- left: 0;
160
- top: 0;
161
- }
162
-
163
- .cropper-line {
164
- background-color: #39f;
165
- }
166
-
167
- .cropper-line.line-e {
168
- cursor: ew-resize;
169
- right: -3px;
170
- top: 0;
171
- width: 5px;
172
- }
173
-
174
- .cropper-line.line-n {
175
- cursor: ns-resize;
176
- height: 5px;
177
- left: 0;
178
- top: -3px;
179
- }
180
-
181
- .cropper-line.line-w {
182
- cursor: ew-resize;
183
- left: -3px;
184
- top: 0;
185
- width: 5px;
186
- }
187
-
188
- .cropper-line.line-s {
189
- bottom: -3px;
190
- cursor: ns-resize;
191
- height: 5px;
192
- left: 0;
193
- }
194
-
195
- .cropper-point {
196
- background-color: #39f;
197
- height: 5px;
198
- opacity: 0.75;
199
- width: 5px;
200
- }
201
-
202
- .cropper-point.point-e {
203
- cursor: ew-resize;
204
- margin-top: -3px;
205
- right: -3px;
206
- top: 50%;
207
- }
208
-
209
- .cropper-point.point-n {
210
- cursor: ns-resize;
211
- left: 50%;
212
- margin-left: -3px;
213
- top: -3px;
214
- }
215
-
216
- .cropper-point.point-w {
217
- cursor: ew-resize;
218
- left: -3px;
219
- margin-top: -3px;
220
- top: 50%;
221
- }
222
-
223
- .cropper-point.point-s {
224
- bottom: -3px;
225
- cursor: s-resize;
226
- left: 50%;
227
- margin-left: -3px;
228
- }
229
-
230
- .cropper-point.point-ne {
231
- cursor: nesw-resize;
232
- right: -3px;
233
- top: -3px;
234
- }
235
-
236
- .cropper-point.point-nw {
237
- cursor: nwse-resize;
238
- left: -3px;
239
- top: -3px;
240
- }
241
-
242
- .cropper-point.point-sw {
243
- bottom: -3px;
244
- cursor: nesw-resize;
245
- left: -3px;
246
- }
247
-
248
- .cropper-point.point-se {
249
- bottom: -3px;
250
- cursor: nwse-resize;
251
- height: 20px;
252
- opacity: 1;
253
- right: -3px;
254
- width: 20px;
255
- }
256
-
257
- @media (min-width: 768px) {
258
- .cropper-point.point-se {
259
- height: 15px;
260
- width: 15px;
261
- }
262
- }
263
-
264
- @media (min-width: 992px) {
265
- .cropper-point.point-se {
266
- height: 10px;
267
- width: 10px;
268
- }
269
- }
270
-
271
- @media (min-width: 1200px) {
272
- .cropper-point.point-se {
273
- height: 5px;
274
- opacity: 0.75;
275
- width: 5px;
276
- }
277
- }
278
-
279
- .cropper-point.point-se::before {
280
- background-color: #39f;
281
- bottom: -50%;
282
- content: ' ';
283
- display: block;
284
- height: 200%;
285
- opacity: 0;
286
- position: absolute;
287
- right: -50%;
288
- width: 200%;
289
- }
290
-
291
- .cropper-invisible {
292
- opacity: 0;
293
- }
294
-
295
- .cropper-bg {
296
- background-image: url('');
297
- }
298
-
299
- .cropper-hide {
300
- display: block;
301
- height: 0;
302
- position: absolute;
303
- width: 0;
304
- }
305
-
306
- .cropper-hidden {
307
- display: none !important;
308
- }
309
-
310
- .cropper-move {
311
- cursor: move;
312
- }
313
-
314
- .cropper-crop {
315
- cursor: crosshair;
316
- }
317
-
318
- .cropper-disabled .cropper-drag-box,
319
- .cropper-disabled .cropper-face,
320
- .cropper-disabled .cropper-line,
321
- .cropper-disabled .cropper-point {
322
- cursor: not-allowed;
323
- }
324
- `;
325
-
326
- const styles: Record<string, React.CSSProperties> = {
327
- dropZone: {
328
- width: '100%',
329
- height: 100,
330
- position: 'relative',
331
- },
332
- dropZoneEmpty: {
333
-
334
- },
335
- image: {
336
- objectFit: 'contain',
337
- margin: 'auto',
338
- display: 'flex',
339
- width: '100%',
340
- height: '100%',
341
- },
342
-
343
- uploadDiv: {
344
- position: 'relative',
345
- width: '100%',
346
- height: 300,
347
- opacity: 0.9,
348
- marginTop: 30,
349
- cursor: 'pointer',
350
- outline: 'none',
351
- },
352
- uploadDivDragging: {
353
- opacity: 1,
354
- background: 'rgba(128,255,128,0.1)',
355
- },
356
-
357
- uploadCenterDiv: {
358
- margin: 5,
359
- border: '3px dashed grey',
360
- borderRadius: 5,
361
- width: 'calc(100% - 10px)',
362
- height: 'calc(100% - 10px)',
363
- position: 'relative',
364
- display: 'flex',
365
- },
366
- uploadCenterIcon: {
367
- paddingTop: 10,
368
- width: 48,
369
- height: 48,
370
- },
371
- uploadCenterText: {
372
- fontSize: 16,
373
- },
374
- uploadCenterTextAndIcon: {
375
- textAlign: 'center',
376
- position: 'absolute',
377
- top: 0,
378
- bottom: 0,
379
- left: 0,
380
- right: 0,
381
- display: 'flex',
382
- flexDirection: 'column',
383
- alignItems: 'center',
384
- justifyContent: 'center',
385
-
386
- },
387
- disabledOpacity: {
388
- opacity: 0.3,
389
- cursor: 'default',
390
- },
391
- buttonRemoveWrapper: {
392
- position: 'absolute',
393
- zIndex: 222,
394
- right: 0,
395
- },
396
- buttonCropWrapper: {
397
- position: 'absolute',
398
- zIndex: 222,
399
- right: 0,
400
- top: 50,
401
- },
402
- error: {
403
- border: '2px solid red',
404
- boxSizing: 'border-box',
405
- },
406
- };
407
-
408
- interface UploadImageProps {
409
- maxSize?: number;
410
- disabled?: boolean;
411
- crop?: boolean;
412
- error?: boolean;
413
- onChange: (base64: string) => void | undefined;
414
- icon: string | null;
415
- removeIconFunc: () => void | null;
416
- accept?: Record<string, string[]>;
417
- }
418
-
419
- interface UploadImageState {
420
- uploadFile: boolean | 'dragging';
421
- anchorEl: HTMLElement | null;
422
- cropHandler: boolean;
423
- }
424
-
425
- class UploadImage extends Component<UploadImageProps, UploadImageState> {
426
- private readonly cropperRef: React.RefObject<ReactCropperElement>;
427
-
428
- constructor(props: UploadImageProps) {
429
- super(props);
430
-
431
- this.state = {
432
- uploadFile: false,
433
- anchorEl: null,
434
- cropHandler: false,
435
- };
436
- this.cropperRef = createRef();
437
-
438
- if (!window.document.getElementById('cropper-style-json-component')) {
439
- const style = window.document.createElement('style');
440
- style.setAttribute('id', 'cropper-style-json-component');
441
- style.innerHTML = cropperStyles;
442
- window.document.head.appendChild(style);
443
- }
444
- }
445
-
446
- onDrop(acceptedFiles: File[]) {
447
- const onChange = this.props.onChange;
448
- const maxSize = this.props.maxSize || 10 * 1024;
449
-
450
- const file = acceptedFiles[0];
451
- const reader = new FileReader();
452
-
453
- reader.onabort = () => console.log('file reading was aborted');
454
- reader.onerror = () => console.log('file reading has failed');
455
- reader.onload = () => {
456
- if (!file || !file.name) {
457
- return;
458
- }
459
- const parts = file.name?.split('.');
460
- let ext = parts?.length ? `image/${parts.pop()?.toLowerCase()}` : 'image/jpeg';
461
- if (ext === 'image/jpg') {
462
- ext = 'image/jpeg';
463
- } else if (ext.includes('svg')) {
464
- ext = 'image/svg+xml';
465
- }
466
- if (file.size > maxSize) {
467
- window.alert(I18n.t('ra_File is too big. Max %sk allowed. Try use SVG.', Math.round(maxSize / 1024)));
468
- } else {
469
- const base64 = `data:${ext};base64,${btoa(
470
- new Uint8Array(reader.result as ArrayBufferLike)
471
- .reduce((data, byte) => data + String.fromCharCode(byte), ''),
472
- )}`;
473
-
474
- if (onChange) {
475
- onChange(base64);
476
- } else {
477
- console.log(base64);
478
- }
479
- }
480
- };
481
- reader.readAsArrayBuffer(file);
482
- }
483
-
484
- render() {
485
- const {
486
- disabled, icon, removeIconFunc, error, crop, onChange,
487
- } = this.props;
488
- const maxSize = this.props.maxSize || 10 * 1024;
489
- const accept = this.props.accept || { 'image/*': [] };
490
- const { uploadFile, anchorEl, cropHandler } = this.state;
491
- return <Dropzone
492
- disabled={!!disabled || cropHandler}
493
- key="dropzone"
494
- multiple={false}
495
- accept={accept}
496
- maxSize={maxSize}
497
- onDragEnter={() => this.setState({ uploadFile: 'dragging' })}
498
- onDragLeave={() => this.setState({ uploadFile: true })}
499
- onDrop={(acceptedFiles: File[], errors) => {
500
- this.setState({ uploadFile: false });
501
- if (!acceptedFiles.length) {
502
- window.alert((errors && errors[0] && errors[0].errors && errors[0].errors[0] && errors[0].errors[0].message) || I18n.t('ra_Cannot upload'));
503
- } else {
504
- this.onDrop(acceptedFiles);
505
- }
506
- }}
507
- >
508
- {({ getRootProps, getInputProps }) => <div
509
- style={{
510
- ...styles.uploadDiv,
511
- ...(uploadFile === 'dragging' ? styles.uploadDivDragging : undefined),
512
- ...styles.dropZone,
513
- ...(disabled ? styles.disabledOpacity : undefined),
514
- ...(!icon ? styles.dropZoneEmpty : undefined),
515
- }}
516
- {...getRootProps()}
517
- >
518
- <input {...getInputProps()} />
519
- <div style={{ ...styles.uploadCenterDiv, ...(error ? styles.error : undefined) }}>
520
- {!icon ? <div style={styles.uploadCenterTextAndIcon}>
521
- <UploadIcon style={styles.uploadCenterIcon} />
522
- <div style={styles.uploadCenterText}>
523
- {uploadFile === 'dragging' ? I18n.t('ra_Drop file here') :
524
- I18n.t('ra_Place your files here or click here to open the browse dialog')}
525
- </div>
526
- </div>
527
- :
528
- removeIconFunc && !cropHandler && <div style={styles.buttonRemoveWrapper}>
529
- <Tooltip title={I18n.t('ra_Clear')} componentsProps={{ popper: { sx: { pointerEvents: 'none' } } }}>
530
- <IconButton
531
- size="large"
532
- onClick={e => {
533
- removeIconFunc && removeIconFunc();
534
- e.stopPropagation();
535
- }}
536
- >
537
- <IconClose />
538
- </IconButton>
539
- </Tooltip>
540
- </div>}
541
- {icon && crop && <div style={styles.buttonCropWrapper}>
542
- <Tooltip title={I18n.t('ra_Crop')} componentsProps={{ popper: { sx: { pointerEvents: 'none' } } }}>
543
- <IconButton
544
- size="large"
545
- onClick={e => {
546
- if (!cropHandler) {
547
- this.setState({ cropHandler: true });
548
- } else {
549
- this.setState({ anchorEl: e.currentTarget });
550
- }
551
- e.stopPropagation();
552
- }}
553
- >
554
- <CropIcon color={cropHandler ? 'primary' : 'inherit'} />
555
- </IconButton>
556
- </Tooltip>
557
- <Menu
558
- anchorEl={anchorEl}
559
- keepMounted
560
- open={Boolean(anchorEl)}
561
- onClose={() => this.setState({ anchorEl: null })}
562
- >
563
- <MenuItem onClick={() => this.setState({ anchorEl: null, cropHandler: false }, () => {
564
- const imageElement = this.cropperRef?.current?.cropper;
565
- if (imageElement) {
566
- if (onChange) {
567
- onChange(imageElement.getCroppedCanvas().toDataURL());
568
- } else {
569
- console.log(imageElement.getCroppedCanvas().toDataURL());
570
- }
571
- }
572
- })}
573
- >
574
- {I18n.t('ra_Save')}
575
- </MenuItem>
576
- <MenuItem onClick={() => this.setState({ anchorEl: null, cropHandler: false })}>{I18n.t('ra_Close')}</MenuItem>
577
- </Menu>
578
- </div>}
579
- {icon && !cropHandler ? <Icon src={icon} style={styles.image} alt="icon" /> : null}
580
-
581
- {icon && crop && cropHandler ? <Cropper
582
- ref={this.cropperRef}
583
- style={styles.image}
584
- src={icon}
585
- initialAspectRatio={1}
586
- viewMode={1}
587
- guides={false}
588
- minCropBoxHeight={10}
589
- minCropBoxWidth={10}
590
- background={false}
591
- checkOrientation={false}
592
- /> : null}
593
- </div>
594
- </div>}
595
- </Dropzone>;
596
- }
597
- }
598
-
599
- export default UploadImage;
1
+ import React, { Component, createRef } from 'react';
2
+ import Dropzone from 'react-dropzone';
3
+ import { Cropper, ReactCropperElement } from 'react-cropper';
4
+
5
+ import {
6
+ Menu, MenuItem, Tooltip, IconButton,
7
+ } from '@mui/material';
8
+
9
+ import {
10
+ Close as IconClose,
11
+ Crop as CropIcon,
12
+ } from '@mui/icons-material';
13
+ import { FaFileUpload as UploadIcon } from 'react-icons/fa';
14
+
15
+ import I18n from '../i18n';
16
+ import Icon from './Icon';
17
+
18
+ // import 'cropperjs/dist/cropper.css';
19
+ const cropperStyles = `
20
+ /*!
21
+ * Cropper.js v1.5.12
22
+ * https://fengyuanchen.github.io/cropperjs
23
+ *
24
+ * Copyright 2015-present Chen Fengyuan
25
+ * Released under the MIT license
26
+ *
27
+ * Date: 2021-06-12T08:00:11.623Z
28
+ */
29
+
30
+ .cropper-container {
31
+ direction: ltr;
32
+ font-size: 0;
33
+ line-height: 0;
34
+ position: relative;
35
+ -ms-touch-action: none;
36
+ touch-action: none;
37
+ -webkit-user-select: none;
38
+ -moz-user-select: none;
39
+ -ms-user-select: none;
40
+ user-select: none;
41
+ }
42
+
43
+ .cropper-container img {
44
+ display: block;
45
+ height: 100%;
46
+ image-orientation: 0deg;
47
+ max-height: none !important;
48
+ max-width: none !important;
49
+ min-height: 0 !important;
50
+ min-width: 0 !important;
51
+ width: 100%;
52
+ }
53
+
54
+ .cropper-wrap-box,
55
+ .cropper-canvas,
56
+ .cropper-drag-box,
57
+ .cropper-crop-box,
58
+ .cropper-modal {
59
+ bottom: 0;
60
+ left: 0;
61
+ position: absolute;
62
+ right: 0;
63
+ top: 0;
64
+ }
65
+
66
+ .cropper-wrap-box,
67
+ .cropper-canvas {
68
+ overflow: hidden;
69
+ }
70
+
71
+ .cropper-drag-box {
72
+ background-color: #fff;
73
+ opacity: 0;
74
+ }
75
+
76
+ .cropper-modal {
77
+ background-color: #000;
78
+ opacity: 0.5;
79
+ }
80
+
81
+ .cropper-view-box {
82
+ display: block;
83
+ height: 100%;
84
+ outline: 1px solid #39f;
85
+ outline-color: rgba(51, 153, 255, 0.75);
86
+ overflow: hidden;
87
+ width: 100%;
88
+ }
89
+
90
+ .cropper-dashed {
91
+ border: 0 dashed #eee;
92
+ display: block;
93
+ opacity: 0.5;
94
+ position: absolute;
95
+ }
96
+
97
+ .cropper-dashed.dashed-h {
98
+ border-bottom-width: 1px;
99
+ border-top-width: 1px;
100
+ height: calc(100% / 3);
101
+ left: 0;
102
+ top: calc(100% / 3);
103
+ width: 100%;
104
+ }
105
+
106
+ .cropper-dashed.dashed-v {
107
+ border-left-width: 1px;
108
+ border-right-width: 1px;
109
+ height: 100%;
110
+ left: calc(100% / 3);
111
+ top: 0;
112
+ width: calc(100% / 3);
113
+ }
114
+
115
+ .cropper-center {
116
+ display: block;
117
+ height: 0;
118
+ left: 50%;
119
+ opacity: 0.75;
120
+ position: absolute;
121
+ top: 50%;
122
+ width: 0;
123
+ }
124
+
125
+ .cropper-center::before,
126
+ .cropper-center::after {
127
+ background-color: #eee;
128
+ content: ' ';
129
+ display: block;
130
+ position: absolute;
131
+ }
132
+
133
+ .cropper-center::before {
134
+ height: 1px;
135
+ left: -3px;
136
+ top: 0;
137
+ width: 7px;
138
+ }
139
+
140
+ .cropper-center::after {
141
+ height: 7px;
142
+ left: 0;
143
+ top: -3px;
144
+ width: 1px;
145
+ }
146
+
147
+ .cropper-face,
148
+ .cropper-line,
149
+ .cropper-point {
150
+ display: block;
151
+ height: 100%;
152
+ opacity: 0.1;
153
+ position: absolute;
154
+ width: 100%;
155
+ }
156
+
157
+ .cropper-face {
158
+ background-color: #fff;
159
+ left: 0;
160
+ top: 0;
161
+ }
162
+
163
+ .cropper-line {
164
+ background-color: #39f;
165
+ }
166
+
167
+ .cropper-line.line-e {
168
+ cursor: ew-resize;
169
+ right: -3px;
170
+ top: 0;
171
+ width: 5px;
172
+ }
173
+
174
+ .cropper-line.line-n {
175
+ cursor: ns-resize;
176
+ height: 5px;
177
+ left: 0;
178
+ top: -3px;
179
+ }
180
+
181
+ .cropper-line.line-w {
182
+ cursor: ew-resize;
183
+ left: -3px;
184
+ top: 0;
185
+ width: 5px;
186
+ }
187
+
188
+ .cropper-line.line-s {
189
+ bottom: -3px;
190
+ cursor: ns-resize;
191
+ height: 5px;
192
+ left: 0;
193
+ }
194
+
195
+ .cropper-point {
196
+ background-color: #39f;
197
+ height: 5px;
198
+ opacity: 0.75;
199
+ width: 5px;
200
+ }
201
+
202
+ .cropper-point.point-e {
203
+ cursor: ew-resize;
204
+ margin-top: -3px;
205
+ right: -3px;
206
+ top: 50%;
207
+ }
208
+
209
+ .cropper-point.point-n {
210
+ cursor: ns-resize;
211
+ left: 50%;
212
+ margin-left: -3px;
213
+ top: -3px;
214
+ }
215
+
216
+ .cropper-point.point-w {
217
+ cursor: ew-resize;
218
+ left: -3px;
219
+ margin-top: -3px;
220
+ top: 50%;
221
+ }
222
+
223
+ .cropper-point.point-s {
224
+ bottom: -3px;
225
+ cursor: s-resize;
226
+ left: 50%;
227
+ margin-left: -3px;
228
+ }
229
+
230
+ .cropper-point.point-ne {
231
+ cursor: nesw-resize;
232
+ right: -3px;
233
+ top: -3px;
234
+ }
235
+
236
+ .cropper-point.point-nw {
237
+ cursor: nwse-resize;
238
+ left: -3px;
239
+ top: -3px;
240
+ }
241
+
242
+ .cropper-point.point-sw {
243
+ bottom: -3px;
244
+ cursor: nesw-resize;
245
+ left: -3px;
246
+ }
247
+
248
+ .cropper-point.point-se {
249
+ bottom: -3px;
250
+ cursor: nwse-resize;
251
+ height: 20px;
252
+ opacity: 1;
253
+ right: -3px;
254
+ width: 20px;
255
+ }
256
+
257
+ @media (min-width: 768px) {
258
+ .cropper-point.point-se {
259
+ height: 15px;
260
+ width: 15px;
261
+ }
262
+ }
263
+
264
+ @media (min-width: 992px) {
265
+ .cropper-point.point-se {
266
+ height: 10px;
267
+ width: 10px;
268
+ }
269
+ }
270
+
271
+ @media (min-width: 1200px) {
272
+ .cropper-point.point-se {
273
+ height: 5px;
274
+ opacity: 0.75;
275
+ width: 5px;
276
+ }
277
+ }
278
+
279
+ .cropper-point.point-se::before {
280
+ background-color: #39f;
281
+ bottom: -50%;
282
+ content: ' ';
283
+ display: block;
284
+ height: 200%;
285
+ opacity: 0;
286
+ position: absolute;
287
+ right: -50%;
288
+ width: 200%;
289
+ }
290
+
291
+ .cropper-invisible {
292
+ opacity: 0;
293
+ }
294
+
295
+ .cropper-bg {
296
+ background-image: url('');
297
+ }
298
+
299
+ .cropper-hide {
300
+ display: block;
301
+ height: 0;
302
+ position: absolute;
303
+ width: 0;
304
+ }
305
+
306
+ .cropper-hidden {
307
+ display: none !important;
308
+ }
309
+
310
+ .cropper-move {
311
+ cursor: move;
312
+ }
313
+
314
+ .cropper-crop {
315
+ cursor: crosshair;
316
+ }
317
+
318
+ .cropper-disabled .cropper-drag-box,
319
+ .cropper-disabled .cropper-face,
320
+ .cropper-disabled .cropper-line,
321
+ .cropper-disabled .cropper-point {
322
+ cursor: not-allowed;
323
+ }
324
+ `;
325
+
326
+ const styles: Record<string, React.CSSProperties> = {
327
+ dropZone: {
328
+ width: '100%',
329
+ height: 100,
330
+ position: 'relative',
331
+ },
332
+ dropZoneEmpty: {
333
+
334
+ },
335
+ image: {
336
+ objectFit: 'contain',
337
+ margin: 'auto',
338
+ display: 'flex',
339
+ width: '100%',
340
+ height: '100%',
341
+ },
342
+
343
+ uploadDiv: {
344
+ position: 'relative',
345
+ width: '100%',
346
+ height: 300,
347
+ opacity: 0.9,
348
+ marginTop: 30,
349
+ cursor: 'pointer',
350
+ outline: 'none',
351
+ },
352
+ uploadDivDragging: {
353
+ opacity: 1,
354
+ background: 'rgba(128,255,128,0.1)',
355
+ },
356
+
357
+ uploadCenterDiv: {
358
+ margin: 5,
359
+ border: '3px dashed grey',
360
+ borderRadius: 5,
361
+ width: 'calc(100% - 10px)',
362
+ height: 'calc(100% - 10px)',
363
+ position: 'relative',
364
+ display: 'flex',
365
+ },
366
+ uploadCenterIcon: {
367
+ paddingTop: 10,
368
+ width: 48,
369
+ height: 48,
370
+ },
371
+ uploadCenterText: {
372
+ fontSize: 16,
373
+ },
374
+ uploadCenterTextAndIcon: {
375
+ textAlign: 'center',
376
+ position: 'absolute',
377
+ top: 0,
378
+ bottom: 0,
379
+ left: 0,
380
+ right: 0,
381
+ display: 'flex',
382
+ flexDirection: 'column',
383
+ alignItems: 'center',
384
+ justifyContent: 'center',
385
+
386
+ },
387
+ disabledOpacity: {
388
+ opacity: 0.3,
389
+ cursor: 'default',
390
+ },
391
+ buttonRemoveWrapper: {
392
+ position: 'absolute',
393
+ zIndex: 222,
394
+ right: 0,
395
+ },
396
+ buttonCropWrapper: {
397
+ position: 'absolute',
398
+ zIndex: 222,
399
+ right: 0,
400
+ top: 50,
401
+ },
402
+ error: {
403
+ border: '2px solid red',
404
+ boxSizing: 'border-box',
405
+ },
406
+ };
407
+
408
+ interface UploadImageProps {
409
+ maxSize?: number;
410
+ disabled?: boolean;
411
+ crop?: boolean;
412
+ error?: boolean;
413
+ onChange: (base64: string) => void | undefined;
414
+ icon: string | null;
415
+ removeIconFunc: () => void | null;
416
+ accept?: Record<string, string[]>;
417
+ }
418
+
419
+ interface UploadImageState {
420
+ uploadFile: boolean | 'dragging';
421
+ anchorEl: HTMLElement | null;
422
+ cropHandler: boolean;
423
+ }
424
+
425
+ class UploadImage extends Component<UploadImageProps, UploadImageState> {
426
+ private readonly cropperRef: React.RefObject<ReactCropperElement>;
427
+
428
+ constructor(props: UploadImageProps) {
429
+ super(props);
430
+
431
+ this.state = {
432
+ uploadFile: false,
433
+ anchorEl: null,
434
+ cropHandler: false,
435
+ };
436
+ this.cropperRef = createRef();
437
+
438
+ if (!window.document.getElementById('cropper-style-json-component')) {
439
+ const style = window.document.createElement('style');
440
+ style.setAttribute('id', 'cropper-style-json-component');
441
+ style.innerHTML = cropperStyles;
442
+ window.document.head.appendChild(style);
443
+ }
444
+ }
445
+
446
+ onDrop(acceptedFiles: File[]) {
447
+ const onChange = this.props.onChange;
448
+ const maxSize = this.props.maxSize || 10 * 1024;
449
+
450
+ const file = acceptedFiles[0];
451
+ const reader = new FileReader();
452
+
453
+ reader.onabort = () => console.log('file reading was aborted');
454
+ reader.onerror = () => console.log('file reading has failed');
455
+ reader.onload = () => {
456
+ if (!file || !file.name) {
457
+ return;
458
+ }
459
+ const parts = file.name?.split('.');
460
+ let ext = parts?.length ? `image/${parts.pop()?.toLowerCase()}` : 'image/jpeg';
461
+ if (ext === 'image/jpg') {
462
+ ext = 'image/jpeg';
463
+ } else if (ext.includes('svg')) {
464
+ ext = 'image/svg+xml';
465
+ }
466
+ if (file.size > maxSize) {
467
+ window.alert(I18n.t('ra_File is too big. Max %sk allowed. Try use SVG.', Math.round(maxSize / 1024)));
468
+ } else {
469
+ const base64 = `data:${ext};base64,${btoa(
470
+ new Uint8Array(reader.result as ArrayBufferLike)
471
+ .reduce((data, byte) => data + String.fromCharCode(byte), ''),
472
+ )}`;
473
+
474
+ if (onChange) {
475
+ onChange(base64);
476
+ } else {
477
+ console.log(base64);
478
+ }
479
+ }
480
+ };
481
+ reader.readAsArrayBuffer(file);
482
+ }
483
+
484
+ render() {
485
+ const {
486
+ disabled, icon, removeIconFunc, error, crop, onChange,
487
+ } = this.props;
488
+ const maxSize = this.props.maxSize || 10 * 1024;
489
+ const accept = this.props.accept || { 'image/*': [] };
490
+ const { uploadFile, anchorEl, cropHandler } = this.state;
491
+ return <Dropzone
492
+ disabled={!!disabled || cropHandler}
493
+ key="dropzone"
494
+ multiple={false}
495
+ accept={accept}
496
+ maxSize={maxSize}
497
+ onDragEnter={() => this.setState({ uploadFile: 'dragging' })}
498
+ onDragLeave={() => this.setState({ uploadFile: true })}
499
+ onDrop={(acceptedFiles: File[], errors) => {
500
+ this.setState({ uploadFile: false });
501
+ if (!acceptedFiles.length) {
502
+ window.alert((errors && errors[0] && errors[0].errors && errors[0].errors[0] && errors[0].errors[0].message) || I18n.t('ra_Cannot upload'));
503
+ } else {
504
+ this.onDrop(acceptedFiles);
505
+ }
506
+ }}
507
+ >
508
+ {({ getRootProps, getInputProps }) => <div
509
+ style={{
510
+ ...styles.uploadDiv,
511
+ ...(uploadFile === 'dragging' ? styles.uploadDivDragging : undefined),
512
+ ...styles.dropZone,
513
+ ...(disabled ? styles.disabledOpacity : undefined),
514
+ ...(!icon ? styles.dropZoneEmpty : undefined),
515
+ }}
516
+ {...getRootProps()}
517
+ >
518
+ <input {...getInputProps()} />
519
+ <div style={{ ...styles.uploadCenterDiv, ...(error ? styles.error : undefined) }}>
520
+ {!icon ? <div style={styles.uploadCenterTextAndIcon}>
521
+ <UploadIcon style={styles.uploadCenterIcon} />
522
+ <div style={styles.uploadCenterText}>
523
+ {uploadFile === 'dragging' ? I18n.t('ra_Drop file here') :
524
+ I18n.t('ra_Place your files here or click here to open the browse dialog')}
525
+ </div>
526
+ </div>
527
+ :
528
+ removeIconFunc && !cropHandler && <div style={styles.buttonRemoveWrapper}>
529
+ <Tooltip title={I18n.t('ra_Clear')} componentsProps={{ popper: { sx: { pointerEvents: 'none' } } }}>
530
+ <IconButton
531
+ size="large"
532
+ onClick={e => {
533
+ removeIconFunc && removeIconFunc();
534
+ e.stopPropagation();
535
+ }}
536
+ >
537
+ <IconClose />
538
+ </IconButton>
539
+ </Tooltip>
540
+ </div>}
541
+ {icon && crop && <div style={styles.buttonCropWrapper}>
542
+ <Tooltip title={I18n.t('ra_Crop')} componentsProps={{ popper: { sx: { pointerEvents: 'none' } } }}>
543
+ <IconButton
544
+ size="large"
545
+ onClick={e => {
546
+ if (!cropHandler) {
547
+ this.setState({ cropHandler: true });
548
+ } else {
549
+ this.setState({ anchorEl: e.currentTarget });
550
+ }
551
+ e.stopPropagation();
552
+ }}
553
+ >
554
+ <CropIcon color={cropHandler ? 'primary' : 'inherit'} />
555
+ </IconButton>
556
+ </Tooltip>
557
+ <Menu
558
+ anchorEl={anchorEl}
559
+ keepMounted
560
+ open={Boolean(anchorEl)}
561
+ onClose={() => this.setState({ anchorEl: null })}
562
+ >
563
+ <MenuItem onClick={() => this.setState({ anchorEl: null, cropHandler: false }, () => {
564
+ const imageElement = this.cropperRef?.current?.cropper;
565
+ if (imageElement) {
566
+ if (onChange) {
567
+ onChange(imageElement.getCroppedCanvas().toDataURL());
568
+ } else {
569
+ console.log(imageElement.getCroppedCanvas().toDataURL());
570
+ }
571
+ }
572
+ })}
573
+ >
574
+ {I18n.t('ra_Save')}
575
+ </MenuItem>
576
+ <MenuItem onClick={() => this.setState({ anchorEl: null, cropHandler: false })}>{I18n.t('ra_Close')}</MenuItem>
577
+ </Menu>
578
+ </div>}
579
+ {icon && !cropHandler ? <Icon src={icon} style={styles.image} alt="icon" /> : null}
580
+
581
+ {icon && crop && cropHandler ? <Cropper
582
+ ref={this.cropperRef}
583
+ style={styles.image}
584
+ src={icon}
585
+ initialAspectRatio={1}
586
+ viewMode={1}
587
+ guides={false}
588
+ minCropBoxHeight={10}
589
+ minCropBoxWidth={10}
590
+ background={false}
591
+ checkOrientation={false}
592
+ /> : null}
593
+ </div>
594
+ </div>}
595
+ </Dropzone>;
596
+ }
597
+ }
598
+
599
+ export default UploadImage;