@fde-desktop/fde-core 0.3.8 → 0.4.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 (148) hide show
  1. package/README.md +419 -68
  2. package/dist/CalendarApp-CHLUCAI7.css +744 -0
  3. package/dist/CalendarApp-K7ZOUZ6C.js +2 -0
  4. package/dist/CalendarApp-OTGEERSS.cjs +8 -0
  5. package/dist/CodeServerApp-5KZGO7HL.css +75 -0
  6. package/dist/CodeServerApp-LUZFCQBI.js +3 -0
  7. package/dist/CodeServerApp-P3TMJPLY.cjs +9 -0
  8. package/dist/CreateItemApp-NAZMXOPK.cjs +14 -0
  9. package/dist/CreateItemApp-PQB5GTFG.css +107 -0
  10. package/dist/CreateItemApp-ZHCTSPQE.js +8 -0
  11. package/dist/DeviceInfoApp-R6YNVIGX.cjs +11 -0
  12. package/dist/DeviceInfoApp-YHCYAO6N.js +5 -0
  13. package/dist/DeviceInfoApp-ZSMRSITP.css +7 -0
  14. package/dist/FilesApp-AKCVRTXR.js +8 -0
  15. package/dist/FilesApp-E6L5W3T2.css +1817 -0
  16. package/dist/FilesApp-RW3Y6ILO.cjs +14 -0
  17. package/dist/ImageViewerApp-5UXNSW2O.js +11 -0
  18. package/dist/ImageViewerApp-N2Q7E7WZ.css +215 -0
  19. package/dist/ImageViewerApp-RRRRKSFN.cjs +17 -0
  20. package/dist/ImageViewerMenuBar-I3TFKQPS.cjs +14 -0
  21. package/dist/ImageViewerMenuBar-TV5C6TM2.js +5 -0
  22. package/dist/ImageViewerMenuBar-XLK4LIHW.css +56 -0
  23. package/dist/MenuEditApp-HUZRFEHE.js +9 -0
  24. package/dist/MenuEditApp-MCUHGTKQ.cjs +15 -0
  25. package/dist/MenuEditApp-YA6HSAMJ.css +94 -0
  26. package/dist/MenuEditMenuBar-7VHMZNRM.css +56 -0
  27. package/dist/MenuEditMenuBar-GF6L4PGZ.cjs +15 -0
  28. package/dist/MenuEditMenuBar-IUXFPZE5.js +6 -0
  29. package/dist/NotesApp-37BV33C6.js +10 -0
  30. package/dist/NotesApp-4EVUQEFZ.cjs +16 -0
  31. package/dist/NotesApp-TQ6IHDNX.css +302 -0
  32. package/dist/NotesMenuBar-25LKN3SE.cjs +15 -0
  33. package/dist/NotesMenuBar-MXLOX7OT.css +56 -0
  34. package/dist/NotesMenuBar-SRV3AIAL.js +6 -0
  35. package/dist/PdfApp-5VBDNRMC.cjs +16 -0
  36. package/dist/PdfApp-BUIC5U5H.css +206 -0
  37. package/dist/PdfApp-RH6MZZX5.js +10 -0
  38. package/dist/PdfMenuBar-NLZC6JHS.js +4 -0
  39. package/dist/PdfMenuBar-QUM72EE4.css +56 -0
  40. package/dist/PdfMenuBar-WBRTKMLN.cjs +13 -0
  41. package/dist/SettingsApp-5LDHEHYV.cjs +20 -0
  42. package/dist/SettingsApp-JVOSEFH3.css +283 -0
  43. package/dist/SettingsApp-X6764D7T.js +14 -0
  44. package/dist/SettingsMenuBar-5CBSSMVM.css +56 -0
  45. package/dist/SettingsMenuBar-VLT6TTCM.js +6 -0
  46. package/dist/SettingsMenuBar-Y5QEXDEO.cjs +15 -0
  47. package/dist/StorybookApp-NQ244BER.css +7 -0
  48. package/dist/StorybookApp-NZDV4X3Y.js +1 -0
  49. package/dist/StorybookApp-VF3KIMU3.cjs +7 -0
  50. package/dist/TerminalApp-CDGWRBFJ.cjs +10 -0
  51. package/dist/TerminalApp-EAATMIMX.css +77 -0
  52. package/dist/TerminalApp-GCKJCM55.js +4 -0
  53. package/dist/TerminalMenuBar-3J26O26Q.css +56 -0
  54. package/dist/TerminalMenuBar-7BH7MGNJ.cjs +14 -0
  55. package/dist/TerminalMenuBar-7JAEQUZ4.js +5 -0
  56. package/dist/UploaderApp-2WYRCUQV.js +10 -0
  57. package/dist/UploaderApp-6KV3TGCT.css +1817 -0
  58. package/dist/UploaderApp-EYFC36PM.cjs +16 -0
  59. package/dist/chunk-2FO445RM.cjs +449 -0
  60. package/dist/chunk-2PSTHGTD.cjs +42 -0
  61. package/dist/chunk-2RQX7QBP.cjs +148 -0
  62. package/dist/chunk-3IICBLEA.js +442 -0
  63. package/dist/chunk-43W6UDUZ.cjs +19 -0
  64. package/dist/chunk-4E45FBAH.js +223 -0
  65. package/dist/chunk-4MCFQPKY.js +444 -0
  66. package/dist/chunk-4OH5RPSQ.cjs +38 -0
  67. package/dist/chunk-4XURSNM4.js +43 -0
  68. package/dist/chunk-4ZCRYHL6.js +407 -0
  69. package/dist/chunk-54PYEQLK.js +283 -0
  70. package/dist/chunk-5C6IQE42.cjs +35 -0
  71. package/dist/chunk-5NOHYJNH.js +84 -0
  72. package/dist/chunk-5PYK5ASL.js +162 -0
  73. package/dist/chunk-5YH6AKEO.js +146 -0
  74. package/dist/chunk-657BJOY5.cjs +324 -0
  75. package/dist/chunk-6QOUYSEE.cjs +2303 -0
  76. package/dist/chunk-7SAFECOJ.js +215 -0
  77. package/dist/chunk-7Y7HB7FB.cjs +53 -0
  78. package/dist/chunk-ABIAPZ6S.cjs +45 -0
  79. package/dist/chunk-AQL372JF.cjs +219 -0
  80. package/dist/chunk-AXDUVZVP.cjs +88 -0
  81. package/dist/chunk-AYFNYHMP.js +541 -0
  82. package/dist/chunk-BDO6B7MZ.cjs +451 -0
  83. package/dist/chunk-BKXEA2BK.cjs +286 -0
  84. package/dist/chunk-BLV47DX2.js +47 -0
  85. package/dist/chunk-BQCD5RAF.cjs +48 -0
  86. package/dist/chunk-BQL3YXMV.js +17429 -0
  87. package/dist/chunk-C6BEZNAM.cjs +45 -0
  88. package/dist/chunk-CFWV2JMR.js +234 -0
  89. package/dist/chunk-CV5PUHAE.cjs +86 -0
  90. package/dist/chunk-D5MVFFID.js +42 -0
  91. package/dist/chunk-D7R55WWT.js +1601 -0
  92. package/dist/chunk-DMNF4CNN.cjs +49 -0
  93. package/dist/chunk-DWP2SYF7.js +55 -0
  94. package/dist/chunk-E55VXNLK.cjs +17498 -0
  95. package/dist/chunk-EAELL43F.js +42 -0
  96. package/dist/chunk-EUQLZW6P.js +48 -0
  97. package/dist/chunk-EX5V2ZTU.js +40 -0
  98. package/dist/chunk-FH4ILMKF.js +38 -0
  99. package/dist/chunk-FRHBM2U7.js +33 -0
  100. package/dist/chunk-FX2TPX3L.cjs +45 -0
  101. package/dist/chunk-GCYD6T52.js +32 -0
  102. package/dist/chunk-GRYCUBJZ.js +9 -0
  103. package/dist/chunk-HWHBSAUC.js +40 -0
  104. package/dist/chunk-ICUE6T7J.cjs +50 -0
  105. package/dist/chunk-IDHP3R4I.js +31 -0
  106. package/dist/chunk-IUOQPOEN.js +2293 -0
  107. package/dist/chunk-J7L2S2GT.cjs +34 -0
  108. package/dist/chunk-JEBKLIMU.cjs +123 -0
  109. package/dist/chunk-KQHICFX3.js +121 -0
  110. package/dist/chunk-LMJE6V4N.cjs +42 -0
  111. package/dist/chunk-MVDGM5Y4.js +68 -0
  112. package/dist/chunk-NVEGEK3N.js +31 -0
  113. package/dist/chunk-NWMSWRUD.js +2236 -0
  114. package/dist/chunk-ODXL6BR3.js +77 -0
  115. package/dist/chunk-OJIDKDKF.js +68 -0
  116. package/dist/chunk-PKPQA5NR.js +15 -0
  117. package/dist/chunk-PNDBLFJW.cjs +50 -0
  118. package/dist/chunk-PYTKNRGM.js +280 -0
  119. package/dist/chunk-Q3WA72BF.cjs +70 -0
  120. package/dist/chunk-QB72BLCJ.cjs +237 -0
  121. package/dist/chunk-QHBBLML3.js +86 -0
  122. package/dist/chunk-RDIDAZ3S.cjs +9 -0
  123. package/dist/chunk-RGJPRXYY.js +48 -0
  124. package/dist/chunk-RQ6OZRUW.cjs +41 -0
  125. package/dist/chunk-SBE4SZAN.cjs +226 -0
  126. package/dist/chunk-SYGUWGWK.cjs +2329 -0
  127. package/dist/chunk-TDZ43MUX.cjs +165 -0
  128. package/dist/chunk-TGWMOHAO.js +17 -0
  129. package/dist/chunk-U4RYIS6Z.cjs +548 -0
  130. package/dist/chunk-UIQCTAVM.cjs +59 -0
  131. package/dist/chunk-XVASHRCE.cjs +70 -0
  132. package/dist/chunk-XYSMVQQD.cjs +1608 -0
  133. package/dist/chunk-YAIWI4Z5.js +7 -0
  134. package/dist/chunk-YP2PLNOF.cjs +34 -0
  135. package/dist/chunk-YSOLW4FS.cjs +11 -0
  136. package/dist/chunk-YY6OUR2U.js +44 -0
  137. package/dist/chunk-YZWS7FDT.cjs +409 -0
  138. package/dist/chunk-Z5YGWL65.cjs +39 -0
  139. package/dist/chunk-ZBGWYTCU.cjs +83 -0
  140. package/dist/chunk-ZHB5Q2M6.js +36 -0
  141. package/dist/chunk-ZHNDXNL4.js +45 -0
  142. package/dist/chunk-ZX3EDZ5C.cjs +17 -0
  143. package/dist/index.cjs +4405 -5156
  144. package/dist/index.css +9192 -0
  145. package/dist/index.d.cts +1324 -762
  146. package/dist/index.d.ts +1324 -762
  147. package/dist/index.js +3648 -5038
  148. package/package.json +14 -6
@@ -0,0 +1,548 @@
1
+ 'use strict';
2
+
3
+ var chunkFX2TPX3L_cjs = require('./chunk-FX2TPX3L.cjs');
4
+ var chunkYSOLW4FS_cjs = require('./chunk-YSOLW4FS.cjs');
5
+ var chunkJ7L2S2GT_cjs = require('./chunk-J7L2S2GT.cjs');
6
+ var chunkZBGWYTCU_cjs = require('./chunk-ZBGWYTCU.cjs');
7
+ var chunk6QOUYSEE_cjs = require('./chunk-6QOUYSEE.cjs');
8
+ var chunkZ5YGWL65_cjs = require('./chunk-Z5YGWL65.cjs');
9
+ var chunkE55VXNLK_cjs = require('./chunk-E55VXNLK.cjs');
10
+ var react = require('react');
11
+ var reactI18next = require('react-i18next');
12
+ var core = require('@dnd-kit/core');
13
+ var jsxRuntime = require('react/jsx-runtime');
14
+
15
+ var FOLDER_ID_TO_TRANSLATION_KEY = {
16
+ "launcher-folder-social-links": "socialLinks",
17
+ "launcher-folder-office": "office",
18
+ "launcher-folder-programming": "programming",
19
+ "launcher-folder-games": "games",
20
+ "launcher-folder-custom-apps": "customApps"
21
+ };
22
+ function useFolderName(folder) {
23
+ const { t } = reactI18next.useTranslation("apps");
24
+ if (!folder.isPredefined) {
25
+ return folder.name;
26
+ }
27
+ const translationKey = FOLDER_ID_TO_TRANSLATION_KEY[folder.id];
28
+ if (!translationKey) {
29
+ return folder.name;
30
+ }
31
+ return t(`folders.${translationKey}`, folder.name);
32
+ }
33
+ var DragDropStateContext = react.createContext(null);
34
+ var DragDropProvider = ({ children, onDrop, onCancel }) => {
35
+ const [activeItem, setActiveItem] = react.useState(null);
36
+ const [overTarget, setOverTarget] = react.useState(null);
37
+ const handleDragStart = react.useCallback((event) => {
38
+ const item = event.active.data.current;
39
+ if (item) {
40
+ setActiveItem(item);
41
+ }
42
+ }, []);
43
+ const handleDragOver = react.useCallback((event) => {
44
+ if (event.over) {
45
+ const target = event.over.data.current;
46
+ if (target) {
47
+ setOverTarget(target);
48
+ }
49
+ } else {
50
+ setOverTarget(null);
51
+ }
52
+ }, []);
53
+ const handleDragEnd = react.useCallback(
54
+ (event) => {
55
+ const { active, over } = event;
56
+ if (over && active.data.current) {
57
+ const item = active.data.current;
58
+ const target = over.data.current;
59
+ if (target.accepts.includes(item.type)) {
60
+ onDrop(item, target);
61
+ }
62
+ } else if (!over && onCancel) {
63
+ onCancel();
64
+ }
65
+ setActiveItem(null);
66
+ setOverTarget(null);
67
+ },
68
+ [onDrop, onCancel]
69
+ );
70
+ const value = react.useMemo(() => ({ activeItem, overTarget }), [activeItem, overTarget]);
71
+ return /* @__PURE__ */ jsxRuntime.jsx(DragDropStateContext.Provider, { value, children: /* @__PURE__ */ jsxRuntime.jsx(
72
+ core.DndContext,
73
+ {
74
+ onDragStart: handleDragStart,
75
+ onDragOver: handleDragOver,
76
+ onDragEnd: handleDragEnd,
77
+ collisionDetection: core.pointerWithin,
78
+ children
79
+ }
80
+ ) });
81
+ };
82
+ function useDraggableItem(options) {
83
+ const { item, disabled = false } = options;
84
+ const { attributes, listeners, setNodeRef, transform, isDragging } = core.useDraggable({
85
+ id: `${item.sourceId}-${item.id}`,
86
+ data: item,
87
+ disabled
88
+ });
89
+ const style = {
90
+ opacity: isDragging ? 0 : 1,
91
+ cursor: disabled ? "default" : "grab"
92
+ };
93
+ return {
94
+ attributes,
95
+ listeners,
96
+ setNodeRef,
97
+ isDragging,
98
+ transform: transform ? { x: transform.x, y: transform.y } : null,
99
+ style
100
+ };
101
+ }
102
+ function useDroppableArea(options) {
103
+ const { target, disabled = false } = options;
104
+ const context = react.useContext(DragDropStateContext);
105
+ const { setNodeRef, isOver } = core.useDroppable({
106
+ id: target.id,
107
+ data: target,
108
+ disabled
109
+ });
110
+ return {
111
+ setNodeRef,
112
+ isOver,
113
+ active: context?.activeItem ?? null
114
+ };
115
+ }
116
+ function useDragDropState() {
117
+ const context = react.useContext(DragDropStateContext);
118
+ if (!context) {
119
+ throw new Error("useDragDropState must be used within a DragDropProvider");
120
+ }
121
+ return context;
122
+ }
123
+
124
+ // src/hooks/useDragDrop/modifiers/snapCenterToCursor.ts
125
+ var snapCenterToCursor = ({ activatorEvent, transform, draggingNodeRect }) => {
126
+ if (!activatorEvent || !draggingNodeRect) {
127
+ return transform;
128
+ }
129
+ const event = activatorEvent;
130
+ let cursorX = 0;
131
+ let cursorY = 0;
132
+ if ("touches" in event && event.touches.length > 0) {
133
+ cursorX = event.touches[0].clientX;
134
+ cursorY = event.touches[0].clientY;
135
+ } else if ("clientX" in event) {
136
+ cursorX = event.clientX;
137
+ cursorY = event.clientY;
138
+ }
139
+ const offsetX = cursorX - draggingNodeRect.left - draggingNodeRect.width / 2;
140
+ const offsetY = cursorY - draggingNodeRect.top - draggingNodeRect.height / 2;
141
+ return {
142
+ ...transform,
143
+ x: transform.x + offsetX,
144
+ y: transform.y + offsetY
145
+ };
146
+ };
147
+
148
+ // src/hooks/useDragDrop/DragOverlay.module.css
149
+ var DragOverlay_default = {};
150
+ var DragOverlayComponent = () => {
151
+ const { activeItem } = useDragDropState();
152
+ if (!activeItem) return null;
153
+ return /* @__PURE__ */ jsxRuntime.jsx(core.DragOverlay, { dropAnimation: core.defaultDropAnimation, modifiers: [snapCenterToCursor], children: /* @__PURE__ */ jsxRuntime.jsxs(chunkE55VXNLK_cjs.Box, { className: DragOverlay_default.dragOverlay, children: [
154
+ /* @__PURE__ */ jsxRuntime.jsx(
155
+ chunkJ7L2S2GT_cjs.AppIcon_default,
156
+ {
157
+ fcIcon: activeItem.metadata?.fcIcon,
158
+ iconUrl: activeItem.metadata?.iconUrl,
159
+ fallback: activeItem.metadata?.icon,
160
+ size: 20
161
+ }
162
+ ),
163
+ /* @__PURE__ */ jsxRuntime.jsx(chunkE55VXNLK_cjs.Text, { size: "xs", children: activeItem.name })
164
+ ] }) });
165
+ };
166
+
167
+ // src/hooks/useDragDrop/constants.ts
168
+ var ACCEPTED_TYPES = {
169
+ "launcher-folder": ["app"],
170
+ "available-apps": ["app"]
171
+ };
172
+
173
+ // src/components/Apps/MenuEditApp/MenuEditApp.module.css
174
+ var MenuEditApp_default = {};
175
+ var FolderItem = ({
176
+ folder,
177
+ isEditing,
178
+ editingName,
179
+ onEdit,
180
+ onDelete,
181
+ onRename,
182
+ onSetEditingName,
183
+ onRemoveApp
184
+ }) => {
185
+ const { t } = reactI18next.useTranslation("menuedit");
186
+ const folderName = useFolderName(folder);
187
+ const { activeItem } = useDragDropState();
188
+ const { setNodeRef, isOver } = useDroppableArea({
189
+ target: {
190
+ type: "launcher-folder",
191
+ id: folder.id,
192
+ name: folderName,
193
+ accepts: ACCEPTED_TYPES["launcher-folder"]
194
+ }
195
+ });
196
+ return /* @__PURE__ */ jsxRuntime.jsxs(
197
+ chunkE55VXNLK_cjs.Card,
198
+ {
199
+ ref: setNodeRef,
200
+ className: MenuEditApp_default.folderCard,
201
+ shadow: "xs",
202
+ padding: "xs",
203
+ "data-is-over": isOver && activeItem?.type === "app" ? "true" : void 0,
204
+ children: [
205
+ /* @__PURE__ */ jsxRuntime.jsxs(chunkE55VXNLK_cjs.Group, { justify: "space-between", mb: 4, children: [
206
+ isEditing ? /* @__PURE__ */ jsxRuntime.jsx(
207
+ chunkE55VXNLK_cjs.TextInput,
208
+ {
209
+ value: editingName,
210
+ onChange: (e) => onSetEditingName(e.target.value),
211
+ onBlur: () => onRename(folder.id),
212
+ onKeyDown: (e) => {
213
+ if (e.key === "Enter") {
214
+ e.preventDefault();
215
+ onRename(folder.id);
216
+ }
217
+ },
218
+ autoFocus: true,
219
+ size: "xs",
220
+ style: { flex: 1 }
221
+ }
222
+ ) : /* @__PURE__ */ jsxRuntime.jsxs(chunkE55VXNLK_cjs.Group, { gap: "xs", children: [
223
+ /* @__PURE__ */ jsxRuntime.jsx(chunkJ7L2S2GT_cjs.AppIcon_default, { fcIcon: folder.fcIcon, size: 16 }),
224
+ /* @__PURE__ */ jsxRuntime.jsx(chunkE55VXNLK_cjs.Text, { size: "sm", fw: 500, children: folderName }),
225
+ folder.isPredefined && /* @__PURE__ */ jsxRuntime.jsxs(chunkE55VXNLK_cjs.Text, { size: "xs", c: "dimmed", children: [
226
+ "(",
227
+ t("predefined"),
228
+ ")"
229
+ ] })
230
+ ] }),
231
+ !folder.isPredefined && !isEditing && /* @__PURE__ */ jsxRuntime.jsxs(chunkE55VXNLK_cjs.Group, { gap: 4, children: [
232
+ /* @__PURE__ */ jsxRuntime.jsx(
233
+ chunkE55VXNLK_cjs.ActionIcon,
234
+ {
235
+ variant: "subtle",
236
+ size: "xs",
237
+ onClick: () => onEdit(folder.id),
238
+ "aria-label": t("rename"),
239
+ children: /* @__PURE__ */ jsxRuntime.jsx(chunkYSOLW4FS_cjs.VscIcon_default, { name: "VscEdit", size: 14 })
240
+ }
241
+ ),
242
+ onDelete && /* @__PURE__ */ jsxRuntime.jsx(
243
+ chunkE55VXNLK_cjs.ActionIcon,
244
+ {
245
+ variant: "subtle",
246
+ size: "xs",
247
+ color: "red",
248
+ onClick: () => onDelete(folder.id),
249
+ "aria-label": t("delete"),
250
+ children: /* @__PURE__ */ jsxRuntime.jsx(chunkYSOLW4FS_cjs.VscIcon_default, { name: "VscTrash", size: 14 })
251
+ }
252
+ )
253
+ ] })
254
+ ] }),
255
+ /* @__PURE__ */ jsxRuntime.jsx(chunkE55VXNLK_cjs.Box, { className: MenuEditApp_default.appList, children: folder.appIds.map((appId) => {
256
+ const app = chunk6QOUYSEE_cjs.APPS.find((a) => a.id === appId);
257
+ if (!app) return null;
258
+ return /* @__PURE__ */ jsxRuntime.jsx(
259
+ DraggableApp,
260
+ {
261
+ app,
262
+ sourceId: folder.id,
263
+ onContextMenu: (e) => {
264
+ e.preventDefault();
265
+ e.stopPropagation();
266
+ onRemoveApp(appId, folder.id);
267
+ }
268
+ },
269
+ appId
270
+ );
271
+ }) })
272
+ ]
273
+ }
274
+ );
275
+ };
276
+ var DraggableApp = ({ app, sourceId, onContextMenu }) => {
277
+ const { attributes, listeners, setNodeRef, isDragging, style } = useDraggableItem({
278
+ item: {
279
+ type: "app",
280
+ id: app.id,
281
+ name: app.name,
282
+ sourceId,
283
+ sourceType: "launcher-folder",
284
+ metadata: {
285
+ fcIcon: app.fcIcon,
286
+ iconUrl: app.iconUrl,
287
+ icon: app.icon
288
+ }
289
+ }
290
+ });
291
+ return /* @__PURE__ */ jsxRuntime.jsxs(
292
+ chunkE55VXNLK_cjs.UnstyledButton,
293
+ {
294
+ ref: setNodeRef,
295
+ className: MenuEditApp_default.appItem,
296
+ style: { ...style, opacity: isDragging ? 0.5 : 1 },
297
+ onContextMenu,
298
+ ...attributes,
299
+ ...listeners,
300
+ children: [
301
+ /* @__PURE__ */ jsxRuntime.jsx(chunkJ7L2S2GT_cjs.AppIcon_default, { fcIcon: app.fcIcon, iconUrl: app.iconUrl, fallback: app.icon, size: 16 }),
302
+ /* @__PURE__ */ jsxRuntime.jsx(chunkE55VXNLK_cjs.Text, { size: "xs", children: app.name })
303
+ ]
304
+ }
305
+ );
306
+ };
307
+ var AvailableApps = ({ customFolderId }) => {
308
+ const { t } = reactI18next.useTranslation("menuedit");
309
+ const { activeItem } = useDragDropState();
310
+ const { setNodeRef, isOver } = useDroppableArea({
311
+ target: {
312
+ type: "available-apps",
313
+ id: "available-apps",
314
+ name: "Available Apps",
315
+ accepts: ACCEPTED_TYPES["available-apps"]
316
+ }
317
+ });
318
+ const appsInFolders = new Set(chunk6QOUYSEE_cjs.useDesktopStore.getState().launcherFolders.flatMap((f) => f.appIds));
319
+ const availableApps = chunk6QOUYSEE_cjs.APPS.filter((app) => app.id !== "settings" && app.id !== "menuedit");
320
+ const addAppToFolder = chunk6QOUYSEE_cjs.useDesktopStore((state) => state.addAppToFolder);
321
+ return /* @__PURE__ */ jsxRuntime.jsxs(
322
+ chunkE55VXNLK_cjs.Box,
323
+ {
324
+ ref: setNodeRef,
325
+ className: MenuEditApp_default.availableApps,
326
+ "data-is-over": isOver && activeItem?.type === "app" ? "true" : void 0,
327
+ children: [
328
+ /* @__PURE__ */ jsxRuntime.jsx(chunkE55VXNLK_cjs.Text, { size: "xs", c: "dimmed", mb: 8, children: t("availableApps") }),
329
+ customFolderId === null && /* @__PURE__ */ jsxRuntime.jsx(chunkE55VXNLK_cjs.Text, { size: "xs", c: "orange", mb: 4, children: t("canCreateNewFolder") }),
330
+ /* @__PURE__ */ jsxRuntime.jsx(chunkE55VXNLK_cjs.Box, { className: MenuEditApp_default.appGrid, children: availableApps.filter((app) => !appsInFolders.has(app.id)).map((app) => /* @__PURE__ */ jsxRuntime.jsx(
331
+ DraggableAppFromAvailable,
332
+ {
333
+ app,
334
+ targetFolderId: customFolderId,
335
+ onAdd: addAppToFolder
336
+ },
337
+ app.id
338
+ )) })
339
+ ]
340
+ }
341
+ );
342
+ };
343
+ var DraggableAppFromAvailable = ({ app, targetFolderId, onAdd }) => {
344
+ const { attributes, listeners, setNodeRef, isDragging, style } = useDraggableItem({
345
+ item: {
346
+ type: "app",
347
+ id: app.id,
348
+ name: app.name,
349
+ sourceId: "available-apps",
350
+ sourceType: "available-apps",
351
+ metadata: {
352
+ fcIcon: app.fcIcon,
353
+ iconUrl: app.iconUrl,
354
+ icon: app.icon
355
+ }
356
+ }
357
+ });
358
+ const handleClick = () => {
359
+ if (targetFolderId) {
360
+ onAdd(app.id, targetFolderId);
361
+ }
362
+ };
363
+ return /* @__PURE__ */ jsxRuntime.jsxs(
364
+ chunkE55VXNLK_cjs.UnstyledButton,
365
+ {
366
+ ref: setNodeRef,
367
+ className: MenuEditApp_default.appItem,
368
+ style: { ...style, opacity: isDragging ? 0 : 1 },
369
+ onClick: handleClick,
370
+ disabled: !targetFolderId,
371
+ ...attributes,
372
+ ...listeners,
373
+ children: [
374
+ /* @__PURE__ */ jsxRuntime.jsx(chunkJ7L2S2GT_cjs.AppIcon_default, { fcIcon: app.fcIcon, iconUrl: app.iconUrl, fallback: app.icon, size: 16 }),
375
+ /* @__PURE__ */ jsxRuntime.jsx(chunkE55VXNLK_cjs.Text, { size: "xs", children: app.name })
376
+ ]
377
+ }
378
+ );
379
+ };
380
+ var MenuEditAppContent = ({ window: win, notifyReady }) => {
381
+ const { t } = reactI18next.useTranslation("menuedit");
382
+ const windowId = win?.id;
383
+ const launcherFolders = chunk6QOUYSEE_cjs.useDesktopStore((state) => state.launcherFolders);
384
+ const setLauncherFolders = chunk6QOUYSEE_cjs.useDesktopStore((state) => state.setLauncherFolders);
385
+ const createLauncherFolder = chunk6QOUYSEE_cjs.useDesktopStore((state) => state.createLauncherFolder);
386
+ const updateLauncherFolder = chunk6QOUYSEE_cjs.useDesktopStore((state) => state.updateLauncherFolder);
387
+ const deleteLauncherFolder = chunk6QOUYSEE_cjs.useDesktopStore((state) => state.deleteLauncherFolder);
388
+ const removeAppFromFolder = chunk6QOUYSEE_cjs.useDesktopStore((state) => state.removeAppFromFolder);
389
+ const [editingFolderId, setEditingFolderId] = react.useState(null);
390
+ const [editingName, setEditingName] = react.useState("");
391
+ const initializedRef = react.useRef(false);
392
+ react.useEffect(() => {
393
+ if (!windowId) return;
394
+ if (initializedRef.current) return;
395
+ initializedRef.current = true;
396
+ chunkFX2TPX3L_cjs.useMenuEditStore.getState().reset(windowId);
397
+ }, [windowId]);
398
+ react.useEffect(() => {
399
+ if (!windowId) return;
400
+ const store = chunkFX2TPX3L_cjs.useMenuEditStore.getState();
401
+ if (store.getSnapshot(windowId) === null) {
402
+ store.takeSnapshot(windowId, launcherFolders);
403
+ } else {
404
+ const snapshot = store.getSnapshot(windowId);
405
+ if (snapshot) {
406
+ const isDirty = JSON.stringify(launcherFolders) !== JSON.stringify(snapshot);
407
+ store.setIsDirty(windowId, isDirty);
408
+ }
409
+ }
410
+ }, [windowId, launcherFolders]);
411
+ const handleSave = react.useCallback(() => {
412
+ if (!windowId) return;
413
+ chunkFX2TPX3L_cjs.useMenuEditStore.getState().takeSnapshot(windowId, launcherFolders);
414
+ }, [windowId, launcherFolders]);
415
+ const revertToSnapshot = react.useCallback(() => {
416
+ if (!windowId) return;
417
+ const snapshot = chunkFX2TPX3L_cjs.useMenuEditStore.getState().getSnapshot(windowId);
418
+ if (snapshot) {
419
+ setLauncherFolders(snapshot);
420
+ }
421
+ }, [windowId, setLauncherFolders]);
422
+ const isDirtyGetter = react.useCallback(() => {
423
+ if (!windowId) return false;
424
+ const snapshot = chunkFX2TPX3L_cjs.useMenuEditStore.getState().getSnapshot(windowId);
425
+ return JSON.stringify(launcherFolders) !== JSON.stringify(snapshot ?? []);
426
+ }, [windowId, launcherFolders]);
427
+ chunkZBGWYTCU_cjs.useCloseInterceptor({
428
+ isDirtyGetter,
429
+ windowId: win?.id,
430
+ onDiscard: revertToSnapshot,
431
+ onSave: handleSave
432
+ });
433
+ const handleCreateFolder = react.useCallback(() => {
434
+ const folder = createLauncherFolder({
435
+ name: t("newFolder"),
436
+ fcIcon: "VscFolder",
437
+ order: launcherFolders.length
438
+ });
439
+ setEditingFolderId(folder.id);
440
+ setEditingName(folder.name);
441
+ }, [createLauncherFolder, launcherFolders.length, t]);
442
+ react.useEffect(() => {
443
+ notifyReady?.({
444
+ createFolder: handleCreateFolder,
445
+ save: handleSave,
446
+ discard: revertToSnapshot
447
+ });
448
+ }, [handleCreateFolder, handleSave, revertToSnapshot, notifyReady]);
449
+ const handleEditFolder = react.useCallback(
450
+ (folderId) => {
451
+ const folder = launcherFolders.find((f) => f.id === folderId);
452
+ if (folder) {
453
+ setEditingFolderId(folder.id);
454
+ setEditingName(folder.name);
455
+ }
456
+ },
457
+ [launcherFolders]
458
+ );
459
+ const handleRename = react.useCallback(
460
+ (folderId) => {
461
+ if (editingName.trim()) {
462
+ updateLauncherFolder(folderId, { name: editingName.trim() });
463
+ }
464
+ setEditingFolderId(null);
465
+ },
466
+ [editingName, updateLauncherFolder]
467
+ );
468
+ const dockerMode = chunkZ5YGWL65_cjs.isDocker();
469
+ const predefinedFolders = launcherFolders.filter((f) => f.isPredefined);
470
+ const customFolders = launcherFolders.filter((f) => !f.isPredefined);
471
+ const firstCustomFolderId = customFolders.length > 0 ? customFolders[0].id : null;
472
+ return /* @__PURE__ */ jsxRuntime.jsx(chunkE55VXNLK_cjs.Box, { className: MenuEditApp_default.container, "data-windowid": win?.id, "data-menuedit-container": true, children: /* @__PURE__ */ jsxRuntime.jsxs(chunkE55VXNLK_cjs.Box, { className: MenuEditApp_default.content, children: [
473
+ predefinedFolders.length > 0 && /* @__PURE__ */ jsxRuntime.jsxs(chunkE55VXNLK_cjs.Box, { mb: "md", children: [
474
+ /* @__PURE__ */ jsxRuntime.jsx(chunkE55VXNLK_cjs.Text, { size: "xs", c: "dimmed", mb: 8, children: t("predefinedFolders") }),
475
+ /* @__PURE__ */ jsxRuntime.jsx(chunkE55VXNLK_cjs.Box, { className: MenuEditApp_default.folderList, children: predefinedFolders.map((folder) => /* @__PURE__ */ jsxRuntime.jsx(
476
+ FolderItem,
477
+ {
478
+ folder,
479
+ isEditing: false,
480
+ editingName: "",
481
+ onEdit: () => {
482
+ },
483
+ onRemoveApp: removeAppFromFolder,
484
+ onRename: () => {
485
+ },
486
+ onSetEditingName: () => {
487
+ }
488
+ },
489
+ folder.id
490
+ )) })
491
+ ] }),
492
+ dockerMode && customFolders.length > 0 && /* @__PURE__ */ jsxRuntime.jsxs(chunkE55VXNLK_cjs.Box, { mb: "md", children: [
493
+ /* @__PURE__ */ jsxRuntime.jsx(chunkE55VXNLK_cjs.Text, { size: "xs", c: "dimmed", mb: 8, children: t("customFolders") }),
494
+ /* @__PURE__ */ jsxRuntime.jsx(chunkE55VXNLK_cjs.Box, { className: MenuEditApp_default.folderList, children: customFolders.map((folder) => /* @__PURE__ */ jsxRuntime.jsx(
495
+ FolderItem,
496
+ {
497
+ folder,
498
+ isEditing: editingFolderId === folder.id,
499
+ editingName,
500
+ onEdit: handleEditFolder,
501
+ onDelete: deleteLauncherFolder,
502
+ onRename: handleRename,
503
+ onSetEditingName: setEditingName,
504
+ onRemoveApp: removeAppFromFolder
505
+ },
506
+ folder.id
507
+ )) })
508
+ ] }),
509
+ /* @__PURE__ */ jsxRuntime.jsxs(chunkE55VXNLK_cjs.Group, { justify: "space-between", mb: "md", children: [
510
+ /* @__PURE__ */ jsxRuntime.jsx(AvailableApps, { customFolderId: firstCustomFolderId }),
511
+ /* @__PURE__ */ jsxRuntime.jsx(chunkE55VXNLK_cjs.Text, { size: "xs", c: "dimmed", mt: "md", children: t("instructions") }),
512
+ dockerMode && /* @__PURE__ */ jsxRuntime.jsx(chunkE55VXNLK_cjs.ActionIcon, { variant: "light", onClick: handleCreateFolder, "aria-label": t("addFolder"), children: /* @__PURE__ */ jsxRuntime.jsx(chunkYSOLW4FS_cjs.VscIcon_default, { name: "VscAdd", size: 16 }) })
513
+ ] })
514
+ ] }) });
515
+ };
516
+ var MenuEditApp = (props) => {
517
+ const addAppToFolder = chunk6QOUYSEE_cjs.useDesktopStore((state) => state.addAppToFolder);
518
+ const removeAppFromFolder = chunk6QOUYSEE_cjs.useDesktopStore((state) => state.removeAppFromFolder);
519
+ const handleDrop = react.useCallback(
520
+ (item, target) => {
521
+ if (item.type !== "app") return;
522
+ if (target.type === "available-apps") {
523
+ if (item.sourceType === "launcher-folder") {
524
+ removeAppFromFolder(item.id, item.sourceId);
525
+ }
526
+ return;
527
+ }
528
+ if (target.type !== "launcher-folder") return;
529
+ if (item.sourceId !== target.id && item.sourceType === "launcher-folder") {
530
+ removeAppFromFolder(item.id, item.sourceId);
531
+ }
532
+ addAppToFolder(item.id, target.id);
533
+ },
534
+ [addAppToFolder, removeAppFromFolder]
535
+ );
536
+ return /* @__PURE__ */ jsxRuntime.jsxs(DragDropProvider, { onDrop: handleDrop, children: [
537
+ /* @__PURE__ */ jsxRuntime.jsx(MenuEditAppContent, { ...props }),
538
+ /* @__PURE__ */ jsxRuntime.jsx(DragOverlayComponent, {})
539
+ ] });
540
+ };
541
+ var MenuEditApp_default2 = MenuEditApp;
542
+
543
+ exports.DragDropProvider = DragDropProvider;
544
+ exports.MenuEditApp_default = MenuEditApp_default2;
545
+ exports.useDragDropState = useDragDropState;
546
+ exports.useDraggableItem = useDraggableItem;
547
+ exports.useDroppableArea = useDroppableArea;
548
+ exports.useFolderName = useFolderName;
@@ -0,0 +1,59 @@
1
+ 'use strict';
2
+
3
+ var zustand = require('zustand');
4
+
5
+ // src/components/Apps/ImageViewerApp/imageViewerStore.ts
6
+ var DEFAULT_STATE = {
7
+ userZoom: 100,
8
+ calculatedZoom: 100,
9
+ fitToWindow: true
10
+ };
11
+ var useImageViewerStore = zustand.create((set, get) => ({
12
+ states: {},
13
+ getUserZoom: (id) => get().states[id]?.userZoom ?? 100,
14
+ getCalculatedZoom: (id) => get().states[id]?.calculatedZoom ?? 100,
15
+ getFitToWindow: (id) => get().states[id]?.fitToWindow ?? true,
16
+ getDisplayZoom: (id) => {
17
+ const state = get().states[id];
18
+ if (!state) return 100;
19
+ return state.fitToWindow ? state.calculatedZoom : state.userZoom;
20
+ },
21
+ setUserZoom: (id, zoom) => set((s) => ({
22
+ states: {
23
+ ...s.states,
24
+ [id]: { ...s.states[id] ?? DEFAULT_STATE, userZoom: zoom, fitToWindow: false }
25
+ }
26
+ })),
27
+ setFitToWindow: (id, fitToWindow) => set((s) => ({
28
+ states: {
29
+ ...s.states,
30
+ [id]: { ...s.states[id] ?? DEFAULT_STATE, fitToWindow }
31
+ }
32
+ })),
33
+ updateCalculatedZoom: (id, imageDims, containerDims) => {
34
+ const scale = Math.min(
35
+ containerDims.width / imageDims.width,
36
+ containerDims.height / imageDims.height
37
+ );
38
+ const calculatedZoom = Math.round(scale * 100);
39
+ set((s) => ({
40
+ states: {
41
+ ...s.states,
42
+ [id]: { ...s.states[id] ?? DEFAULT_STATE, calculatedZoom }
43
+ }
44
+ }));
45
+ },
46
+ reset: (id) => set((s) => ({
47
+ states: { ...s.states, [id]: DEFAULT_STATE }
48
+ }))
49
+ }));
50
+ var useFitToWindow = (windowId) => useImageViewerStore((s) => s.states[windowId]?.fitToWindow ?? true);
51
+ var useDisplayZoom = (windowId) => useImageViewerStore((s) => {
52
+ const state = s.states[windowId];
53
+ if (!state) return 100;
54
+ return state.fitToWindow ? state.calculatedZoom : state.userZoom;
55
+ });
56
+
57
+ exports.useDisplayZoom = useDisplayZoom;
58
+ exports.useFitToWindow = useFitToWindow;
59
+ exports.useImageViewerStore = useImageViewerStore;
@@ -0,0 +1,70 @@
1
+ 'use strict';
2
+
3
+ var chunkRDIDAZ3S_cjs = require('./chunk-RDIDAZ3S.cjs');
4
+ var chunkSBE4SZAN_cjs = require('./chunk-SBE4SZAN.cjs');
5
+ var chunkFX2TPX3L_cjs = require('./chunk-FX2TPX3L.cjs');
6
+ var react = require('react');
7
+ var reactI18next = require('react-i18next');
8
+ var jsxRuntime = require('react/jsx-runtime');
9
+
10
+ var useMenuEditMenuBarElements = (options) => {
11
+ const { windowId, contentData, onExit } = options;
12
+ const { t } = reactI18next.useTranslation("menuedit");
13
+ const { t: tCommon } = reactI18next.useTranslation("common");
14
+ const isDirty = chunkFX2TPX3L_cjs.useIsDirty(windowId);
15
+ const createFolder = contentData?.createFolder;
16
+ const save = contentData?.save;
17
+ const discard = contentData?.discard;
18
+ return react.useMemo(() => {
19
+ const items = [
20
+ {
21
+ type: "menu",
22
+ label: tCommon("menu.file"),
23
+ items: [
24
+ {
25
+ type: "item",
26
+ label: t("newFolder"),
27
+ icon: "FcPlus",
28
+ onClick: () => createFolder?.()
29
+ },
30
+ { type: "divider" },
31
+ {
32
+ type: "item",
33
+ label: tCommon("actions.save"),
34
+ icon: "FcInspection",
35
+ onClick: () => save?.(),
36
+ disabled: !isDirty
37
+ },
38
+ {
39
+ type: "item",
40
+ label: tCommon("actions.discard"),
41
+ icon: "FcCancel",
42
+ onClick: () => discard?.(),
43
+ disabled: !isDirty
44
+ },
45
+ { type: "divider" },
46
+ {
47
+ type: "item",
48
+ label: tCommon("actions.exit"),
49
+ icon: "FcLeft",
50
+ onClick: onExit
51
+ }
52
+ ]
53
+ }
54
+ ];
55
+ return chunkRDIDAZ3S_cjs.appendDirtyIndicator(items, isDirty);
56
+ }, [t, tCommon, isDirty, createFolder, save, discard, onExit]);
57
+ };
58
+ var MenuEditMenuBar = react.memo(
59
+ ({ windowId, contentData, onExit }) => {
60
+ const elements = useMenuEditMenuBarElements({
61
+ windowId,
62
+ contentData,
63
+ onExit
64
+ });
65
+ return /* @__PURE__ */ jsxRuntime.jsx(chunkSBE4SZAN_cjs.AppMenuBar_default, { elements });
66
+ }
67
+ );
68
+ MenuEditMenuBar.displayName = "MenuEditMenuBar";
69
+
70
+ exports.MenuEditMenuBar = MenuEditMenuBar;