@nice2dev/ui-graphics 1.0.4 → 1.0.5
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.
- package/dist/cjs/animation/AnimationEditor.js +1 -1
- package/dist/cjs/animation/Audience.js +1 -1
- package/dist/cjs/core/LocalUI.js +1 -1
- package/dist/cjs/core/yjsCollaboration.js +2 -2
- package/dist/cjs/core/yjsCollaboration.js.map +1 -1
- package/dist/cjs/game/GameAsset2dEditor.js +1 -1
- package/dist/cjs/node_modules/lib0/array.js +103 -0
- package/dist/cjs/node_modules/lib0/array.js.map +1 -0
- package/dist/cjs/node_modules/lib0/binary.js +42 -0
- package/dist/cjs/node_modules/lib0/binary.js.map +1 -0
- package/dist/cjs/node_modules/lib0/broadcastchannel.js +121 -0
- package/dist/cjs/node_modules/lib0/broadcastchannel.js.map +1 -0
- package/dist/cjs/node_modules/lib0/buffer.js +103 -0
- package/dist/cjs/node_modules/lib0/buffer.js.map +1 -0
- package/dist/cjs/node_modules/lib0/conditions.js +18 -0
- package/dist/cjs/node_modules/lib0/conditions.js.map +1 -0
- package/dist/cjs/node_modules/lib0/decoding.js +460 -0
- package/dist/cjs/node_modules/lib0/decoding.js.map +1 -0
- package/dist/cjs/node_modules/lib0/dom.js +55 -0
- package/dist/cjs/node_modules/lib0/dom.js.map +1 -0
- package/dist/cjs/node_modules/lib0/encoding.js +688 -0
- package/dist/cjs/node_modules/lib0/encoding.js.map +1 -0
- package/dist/cjs/node_modules/lib0/environment.js +129 -0
- package/dist/cjs/node_modules/lib0/environment.js.map +1 -0
- package/dist/cjs/node_modules/lib0/error.js +37 -0
- package/dist/cjs/node_modules/lib0/error.js.map +1 -0
- package/dist/cjs/node_modules/lib0/function.js +135 -0
- package/dist/cjs/node_modules/lib0/function.js.map +1 -0
- package/dist/cjs/node_modules/lib0/iterator.js +52 -0
- package/dist/cjs/node_modules/lib0/iterator.js.map +1 -0
- package/dist/cjs/node_modules/lib0/logging.common.js +63 -0
- package/dist/cjs/node_modules/lib0/logging.common.js.map +1 -0
- package/dist/cjs/node_modules/lib0/logging.js +128 -0
- package/dist/cjs/node_modules/lib0/logging.js.map +1 -0
- package/dist/cjs/node_modules/lib0/map.js +108 -0
- package/dist/cjs/node_modules/lib0/map.js.map +1 -0
- package/dist/cjs/node_modules/lib0/math.js +44 -0
- package/dist/cjs/node_modules/lib0/math.js.map +1 -0
- package/dist/cjs/node_modules/lib0/number.js +21 -0
- package/dist/cjs/node_modules/lib0/number.js.map +1 -0
- package/dist/cjs/node_modules/lib0/object.js +132 -0
- package/dist/cjs/node_modules/lib0/object.js.map +1 -0
- package/dist/cjs/node_modules/lib0/observable.js +168 -0
- package/dist/cjs/node_modules/lib0/observable.js.map +1 -0
- package/dist/cjs/node_modules/lib0/pair.js +33 -0
- package/dist/cjs/node_modules/lib0/pair.js.map +1 -0
- package/dist/cjs/node_modules/lib0/prng.js +95 -0
- package/dist/cjs/node_modules/lib0/prng.js.map +1 -0
- package/dist/cjs/node_modules/lib0/promise.js +33 -0
- package/dist/cjs/node_modules/lib0/promise.js.map +1 -0
- package/dist/cjs/node_modules/lib0/random.js +28 -0
- package/dist/cjs/node_modules/lib0/random.js.map +1 -0
- package/dist/cjs/node_modules/lib0/schema.js +1168 -0
- package/dist/cjs/node_modules/lib0/schema.js.map +1 -0
- package/dist/cjs/node_modules/lib0/set.js +12 -0
- package/dist/cjs/node_modules/lib0/set.js.map +1 -0
- package/dist/cjs/node_modules/lib0/storage.js +79 -0
- package/dist/cjs/node_modules/lib0/storage.js.map +1 -0
- package/dist/cjs/node_modules/lib0/string.js +94 -0
- package/dist/cjs/node_modules/lib0/string.js.map +1 -0
- package/dist/cjs/node_modules/lib0/symbol.js +15 -0
- package/dist/cjs/node_modules/lib0/symbol.js.map +1 -0
- package/dist/cjs/node_modules/lib0/time.js +18 -0
- package/dist/cjs/node_modules/lib0/time.js.map +1 -0
- package/dist/cjs/node_modules/lib0/trait/equality.js +32 -0
- package/dist/cjs/node_modules/lib0/trait/equality.js.map +1 -0
- package/dist/cjs/node_modules/lib0/url.js +20 -0
- package/dist/cjs/node_modules/lib0/url.js.map +1 -0
- package/dist/cjs/node_modules/lib0/webcrypto.js +8 -0
- package/dist/cjs/node_modules/lib0/webcrypto.js.map +1 -0
- package/dist/cjs/node_modules/y-protocols/auth.js +27 -0
- package/dist/cjs/node_modules/y-protocols/auth.js.map +1 -0
- package/dist/cjs/node_modules/y-protocols/awareness.js +277 -0
- package/dist/cjs/node_modules/y-protocols/awareness.js.map +1 -0
- package/dist/cjs/node_modules/y-protocols/sync.js +149 -0
- package/dist/cjs/node_modules/y-protocols/sync.js.map +1 -0
- package/dist/cjs/node_modules/y-websocket/src/y-websocket.js +521 -0
- package/dist/cjs/node_modules/y-websocket/src/y-websocket.js.map +1 -0
- package/dist/cjs/node_modules/yjs/dist/yjs.js +10399 -0
- package/dist/cjs/node_modules/yjs/dist/yjs.js.map +1 -0
- package/dist/cjs/packages/ui/dist/index.js +65997 -0
- package/dist/cjs/packages/ui/dist/index.js.map +1 -0
- package/dist/cjs/pixel/PixelEditor.js +1 -1
- package/dist/cjs/pixel/PixelEditorMenuBar.js +1 -1
- package/dist/cjs/pixel/PixelEditorRightPanel.js +1 -1
- package/dist/cjs/pixel/PixelEditorTimeline.js +1 -1
- package/dist/cjs/pixel/PixelEditorToolbar.js +1 -1
- package/dist/cjs/pixel/SpriteGeneratorPanel.js +1 -1
- package/dist/cjs/vector/VectorEditor.js +1 -1
- package/dist/cjs/vector/VectorEditorMenuBar.js +1 -1
- package/dist/cjs/vector/VectorEditorRightPanel.js +1 -1
- package/dist/esm/animation/AnimationEditor.js +5 -5
- package/dist/esm/animation/Audience.js +2 -2
- package/dist/esm/core/LocalUI.js +4 -4
- package/dist/esm/core/yjsCollaboration.js +2 -2
- package/dist/esm/core/yjsCollaboration.js.map +1 -1
- package/dist/esm/game/GameAsset2dEditor.js +19 -19
- package/dist/esm/node_modules/lib0/array.js +95 -0
- package/dist/esm/node_modules/lib0/array.js.map +1 -0
- package/dist/esm/node_modules/lib0/binary.js +30 -0
- package/dist/esm/node_modules/lib0/binary.js.map +1 -0
- package/dist/esm/node_modules/lib0/broadcastchannel.js +117 -0
- package/dist/esm/node_modules/lib0/broadcastchannel.js.map +1 -0
- package/dist/esm/node_modules/lib0/buffer.js +96 -0
- package/dist/esm/node_modules/lib0/buffer.js.map +1 -0
- package/dist/esm/node_modules/lib0/conditions.js +16 -0
- package/dist/esm/node_modules/lib0/conditions.js.map +1 -0
- package/dist/esm/node_modules/lib0/decoding.js +439 -0
- package/dist/esm/node_modules/lib0/decoding.js.map +1 -0
- package/dist/esm/node_modules/lib0/dom.js +48 -0
- package/dist/esm/node_modules/lib0/dom.js.map +1 -0
- package/dist/esm/node_modules/lib0/encoding.js +663 -0
- package/dist/esm/node_modules/lib0/encoding.js.map +1 -0
- package/dist/esm/node_modules/lib0/environment.js +121 -0
- package/dist/esm/node_modules/lib0/environment.js.map +1 -0
- package/dist/esm/node_modules/lib0/error.js +33 -0
- package/dist/esm/node_modules/lib0/error.js.map +1 -0
- package/dist/esm/node_modules/lib0/function.js +130 -0
- package/dist/esm/node_modules/lib0/function.js.map +1 -0
- package/dist/esm/node_modules/lib0/iterator.js +48 -0
- package/dist/esm/node_modules/lib0/iterator.js.map +1 -0
- package/dist/esm/node_modules/lib0/logging.common.js +52 -0
- package/dist/esm/node_modules/lib0/logging.common.js.map +1 -0
- package/dist/esm/node_modules/lib0/logging.js +115 -0
- package/dist/esm/node_modules/lib0/logging.js.map +1 -0
- package/dist/esm/node_modules/lib0/map.js +102 -0
- package/dist/esm/node_modules/lib0/map.js.map +1 -0
- package/dist/esm/node_modules/lib0/math.js +37 -0
- package/dist/esm/node_modules/lib0/math.js.map +1 -0
- package/dist/esm/node_modules/lib0/number.js +17 -0
- package/dist/esm/node_modules/lib0/number.js.map +1 -0
- package/dist/esm/node_modules/lib0/object.js +119 -0
- package/dist/esm/node_modules/lib0/object.js.map +1 -0
- package/dist/esm/node_modules/lib0/observable.js +165 -0
- package/dist/esm/node_modules/lib0/observable.js.map +1 -0
- package/dist/esm/node_modules/lib0/pair.js +30 -0
- package/dist/esm/node_modules/lib0/pair.js.map +1 -0
- package/dist/esm/node_modules/lib0/prng.js +87 -0
- package/dist/esm/node_modules/lib0/prng.js.map +1 -0
- package/dist/esm/node_modules/lib0/promise.js +31 -0
- package/dist/esm/node_modules/lib0/promise.js.map +1 -0
- package/dist/esm/node_modules/lib0/random.js +25 -0
- package/dist/esm/node_modules/lib0/random.js.map +1 -0
- package/dist/esm/node_modules/lib0/schema.js +1113 -0
- package/dist/esm/node_modules/lib0/schema.js.map +1 -0
- package/dist/esm/node_modules/lib0/set.js +10 -0
- package/dist/esm/node_modules/lib0/set.js.map +1 -0
- package/dist/esm/node_modules/lib0/storage.js +75 -0
- package/dist/esm/node_modules/lib0/storage.js.map +1 -0
- package/dist/esm/node_modules/lib0/string.js +85 -0
- package/dist/esm/node_modules/lib0/string.js.map +1 -0
- package/dist/esm/node_modules/lib0/symbol.js +13 -0
- package/dist/esm/node_modules/lib0/symbol.js.map +1 -0
- package/dist/esm/node_modules/lib0/time.js +16 -0
- package/dist/esm/node_modules/lib0/time.js.map +1 -0
- package/dist/esm/node_modules/lib0/trait/equality.js +29 -0
- package/dist/esm/node_modules/lib0/trait/equality.js.map +1 -0
- package/dist/esm/node_modules/lib0/url.js +18 -0
- package/dist/esm/node_modules/lib0/url.js.map +1 -0
- package/dist/esm/node_modules/lib0/webcrypto.js +6 -0
- package/dist/esm/node_modules/lib0/webcrypto.js.map +1 -0
- package/dist/esm/node_modules/y-protocols/auth.js +24 -0
- package/dist/esm/node_modules/y-protocols/auth.js.map +1 -0
- package/dist/esm/node_modules/y-protocols/awareness.js +271 -0
- package/dist/esm/node_modules/y-protocols/awareness.js.map +1 -0
- package/dist/esm/node_modules/y-protocols/sync.js +138 -0
- package/dist/esm/node_modules/y-protocols/sync.js.map +1 -0
- package/dist/esm/node_modules/y-websocket/src/y-websocket.js +515 -0
- package/dist/esm/node_modules/y-websocket/src/y-websocket.js.map +1 -0
- package/dist/esm/node_modules/yjs/dist/yjs.js +10295 -0
- package/dist/esm/node_modules/yjs/dist/yjs.js.map +1 -0
- package/dist/esm/packages/ui/dist/index.js +65571 -0
- package/dist/esm/packages/ui/dist/index.js.map +1 -0
- package/dist/esm/pixel/PixelEditor.js +3 -3
- package/dist/esm/pixel/PixelEditorMenuBar.js +2 -2
- package/dist/esm/pixel/PixelEditorRightPanel.js +3 -3
- package/dist/esm/pixel/PixelEditorTimeline.js +2 -2
- package/dist/esm/pixel/PixelEditorToolbar.js +2 -2
- package/dist/esm/pixel/SpriteGeneratorPanel.js +8 -8
- package/dist/esm/vector/VectorEditor.js +3 -3
- package/dist/esm/vector/VectorEditorMenuBar.js +3 -3
- package/dist/esm/vector/VectorEditorRightPanel.js +13 -13
- package/package.json +1 -1
|
@@ -1,7 +1,7 @@
|
|
|
1
1
|
import { jsxs, jsx } from 'react/jsx-runtime';
|
|
2
2
|
import { useState, useCallback } from 'react';
|
|
3
3
|
import styles from './PixelEditor.module.css.js';
|
|
4
|
-
import { NiceModal as
|
|
4
|
+
import { NiceModal as Ys, NiceNumberInput as ir, NiceButton as We } from '../packages/ui/dist/index.js';
|
|
5
5
|
import { usePixelEditor } from './usePixelEditor.js';
|
|
6
6
|
import PixelEditorMenuBar from './PixelEditorMenuBar.js';
|
|
7
7
|
import PixelEditorToolbar from './PixelEditorToolbar.js';
|
|
@@ -24,10 +24,10 @@ const NicePixelEditor = ({ initialWidth = 64, initialHeight = 64, className, onS
|
|
|
24
24
|
if (f)
|
|
25
25
|
api.openFile(f);
|
|
26
26
|
e.target.value = "";
|
|
27
|
-
} }), jsx(PixelEditorMenuBar, { api: api, hasOnSaveToLibrary: !!onSaveToLibrary, onOpenGenerator: () => setShowGenerator(true) }), jsxs("div", { className: styles.mainArea, children: [jsx(PixelEditorToolbar, { api: api }), jsx(PixelEditorCanvas, { api: api }), jsx(PixelEditorRightPanel, { api: api })] }), jsx(PixelEditorTimeline, { api: api }), jsx(PixelEditorStatusBar, { api: api }), showGenerator && (jsx(NiceSpriteGeneratorPanel, { onSendToCanvas: handleGeneratorSend, onClose: () => setShowGenerator(false) })), jsxs(
|
|
27
|
+
} }), jsx(PixelEditorMenuBar, { api: api, hasOnSaveToLibrary: !!onSaveToLibrary, onOpenGenerator: () => setShowGenerator(true) }), jsxs("div", { className: styles.mainArea, children: [jsx(PixelEditorToolbar, { api: api }), jsx(PixelEditorCanvas, { api: api }), jsx(PixelEditorRightPanel, { api: api })] }), jsx(PixelEditorTimeline, { api: api }), jsx(PixelEditorStatusBar, { api: api }), showGenerator && (jsx(NiceSpriteGeneratorPanel, { onSendToCanvas: handleGeneratorSend, onClose: () => setShowGenerator(false) })), jsxs(Ys, { open: api.dlgNew, onClose: () => api.setDlgNew(false), title: "New Document", children: [jsx("label", { children: "Width (px)" }), jsx(ir, { min: 1, max: 4096, value: api.nw, onChange: (val) => api.setNw(val !== null && val !== void 0 ? val : 64), "aria-label": "Document width" }), jsx("label", { children: "Height (px)" }), jsx(ir, { min: 1, max: 4096, value: api.nh, onChange: (val) => api.setNh(val !== null && val !== void 0 ? val : 64), "aria-label": "Document height" }), jsx("div", { style: { display: "flex", gap: 4, flexWrap: "wrap", marginTop: 8 }, children: [
|
|
28
28
|
[16, 16], [32, 32], [48, 48], [64, 64], [128, 128],
|
|
29
29
|
[256, 256], [16, 32], [32, 64], [48, 16], [64, 32],
|
|
30
|
-
].map(([w, h]) => (jsxs(
|
|
30
|
+
].map(([w, h]) => (jsxs(We, { className: styles.btnSecondary, variant: "ghost", size: "sm", style: { fontSize: 10, padding: "2px 6px" }, onClick: () => { api.setNw(w); api.setNh(h); }, children: [w, "x", h] }, `${w}x${h}`))) }), jsxs("div", { className: styles.dialogActions, children: [jsx(We, { className: styles.btnSecondary, variant: "ghost", onClick: () => api.setDlgNew(false), children: "Cancel" }), jsx(We, { className: styles.btnPrimary, variant: "primary", onClick: () => api.newDoc(api.nw, api.nh), children: "Create" })] })] })] }));
|
|
31
31
|
};
|
|
32
32
|
/** @deprecated Use NicePixelEditor */
|
|
33
33
|
const PixelEditor = NicePixelEditor;
|
|
@@ -1,9 +1,9 @@
|
|
|
1
1
|
import { jsxs, jsx, Fragment } from 'react/jsx-runtime';
|
|
2
2
|
import React from 'react';
|
|
3
3
|
import styles from './PixelEditor.module.css.js';
|
|
4
|
-
import { NiceButton as
|
|
4
|
+
import { NiceButton as We } from '../packages/ui/dist/index.js';
|
|
5
5
|
|
|
6
|
-
const PixelEditorMenuBar = React.memo(({ api, hasOnSaveToLibrary, onOpenGenerator }) => (jsxs("div", { className: styles.menuBar, children: [jsx(
|
|
6
|
+
const PixelEditorMenuBar = React.memo(({ api, hasOnSaveToLibrary, onOpenGenerator }) => (jsxs("div", { className: styles.menuBar, children: [jsx(We, { className: styles.menuBtn, variant: "ghost", size: "sm", onClick: () => api.setDlgNew(true), children: "New" }), jsx(We, { className: styles.menuBtn, variant: "ghost", size: "sm", onClick: () => { var _a; return (_a = api.fInp.current) === null || _a === void 0 ? void 0 : _a.click(); }, children: "Open" }), jsx(We, { className: styles.menuBtn, variant: "ghost", size: "sm", onClick: api.exportAse, children: "Save .ase" }), jsx("div", { className: styles.menuSep }), jsx(We, { className: styles.menuBtn, variant: "ghost", size: "sm", onClick: api.exportPNG, children: "PNG" }), jsx(We, { className: styles.menuBtn, variant: "ghost", size: "sm", onClick: api.exportJPG, children: "JPG" }), jsx(We, { className: styles.menuBtn, variant: "ghost", size: "sm", onClick: api.exportWebP, children: "WebP" }), jsx(We, { className: styles.menuBtn, variant: "ghost", size: "sm", onClick: api.exportBMP, children: "BMP" }), jsx(We, { className: styles.menuBtn, variant: "ghost", size: "sm", onClick: api.exportSVG, children: "SVG" }), jsx(We, { className: styles.menuBtn, variant: "ghost", size: "sm", onClick: api.exportSheet, children: "Sheet" }), hasOnSaveToLibrary && (jsxs(Fragment, { children: [jsx("div", { className: styles.menuSep }), jsx(We, { className: styles.menuBtn, variant: "ghost", size: "sm", onClick: api.saveToLibrary, children: "\uD83D\uDCE6 Save to Library" })] })), jsx("div", { className: styles.menuSep }), jsx(We, { className: styles.menuBtn, variant: "ghost", size: "sm", onClick: api.undo, children: "Undo" }), jsx(We, { className: styles.menuBtn, variant: "ghost", size: "sm", onClick: api.redo, children: "Redo" }), jsx("div", { className: styles.menuSep }), jsx(We, { className: styles.menuBtn, variant: "ghost", size: "sm", onClick: () => api.flipCv("h"), children: "Flip H" }), jsx(We, { className: styles.menuBtn, variant: "ghost", size: "sm", onClick: () => api.flipCv("v"), children: "Flip V" }), jsx(We, { className: styles.menuBtn, variant: "ghost", size: "sm", onClick: api.rotateCv, children: "Rot 90\u00B0" }), jsx(We, { className: styles.menuBtn, variant: "ghost", size: "sm", onClick: api.clearCv, children: "Clear" }), jsx("div", { className: styles.menuSep }), jsxs(We, { className: styles.menuBtn, variant: "ghost", size: "sm", onClick: () => api.setGrid((g) => !g), children: ["Grid: ", api.grid ? "ON" : "OFF"] }), jsx("div", { className: styles.menuSep }), jsx(We, { className: styles.menuBtn, variant: "ghost", size: "sm", onClick: api.doCopy, disabled: !api.sel, children: "Copy" }), jsx(We, { className: styles.menuBtn, variant: "ghost", size: "sm", onClick: api.doCut, disabled: !api.sel, children: "Cut" }), jsx(We, { className: styles.menuBtn, variant: "ghost", size: "sm", onClick: api.doPaste, disabled: !api.clipboard.current, children: "Paste" }), jsx(We, { className: styles.menuBtn, variant: "ghost", size: "sm", onClick: api.doSelectAll, children: "Sel All" }), jsx("div", { className: styles.menuSep }), jsx(We, { className: styles.menuBtn, variant: "ghost", size: "sm", onClick: onOpenGenerator, children: "\uD83C\uDFA8 Generator" })] })));
|
|
7
7
|
PixelEditorMenuBar.displayName = "PixelEditorMenuBar";
|
|
8
8
|
|
|
9
9
|
export { PixelEditorMenuBar as default };
|
|
@@ -1,12 +1,12 @@
|
|
|
1
1
|
import { jsxs, jsx } from 'react/jsx-runtime';
|
|
2
2
|
import React from 'react';
|
|
3
3
|
import styles from './PixelEditor.module.css.js';
|
|
4
|
-
import { NiceSlider as
|
|
4
|
+
import { NiceSlider as so, NiceButton as We, NiceCheckbox as Fr, NiceTextInput as Tt, NiceSelect as fn } from '../packages/ui/dist/index.js';
|
|
5
5
|
import { BRUSH_SHAPES, BLEND_MODES } from './pixelEditorTypes.js';
|
|
6
6
|
import { HSVPicker } from './HSVPicker.js';
|
|
7
7
|
|
|
8
|
-
const PixelEditorRightPanel = React.memo(({ api }) => (jsxs("div", { className: styles.rightPanel, children: [jsxs("div", { className: styles.panelSection, children: [jsx("div", { className: styles.panelTitle, children: jsx("span", { children: "Brush" }) }), jsxs("div", { className: styles.brushSlider, children: [jsx("label", { children: "Size" }), jsx(
|
|
9
|
-
api.setEditingLayerName(null); }, autoFocus: true, "aria-label": "Layer name" })) : (jsx("span", { className: styles.layerName, onDoubleClick: (e) => { e.stopPropagation(); api.setEditingLayerName(l.id); }, children: l.name })), jsxs("span", { className: styles.layerOpacity, children: [Math.round((l.opacity / 255) * 100), "%"] })] }, l.id))), api.layers[api.aLi] && (jsxs("div", { className: styles.layerControls, children: [jsx(
|
|
8
|
+
const PixelEditorRightPanel = React.memo(({ api }) => (jsxs("div", { className: styles.rightPanel, children: [jsxs("div", { className: styles.panelSection, children: [jsx("div", { className: styles.panelTitle, children: jsx("span", { children: "Brush" }) }), jsxs("div", { className: styles.brushSlider, children: [jsx("label", { children: "Size" }), jsx(so, { min: 1, max: 64, value: api.brush.size, onChange: (val) => api.setBrush((b) => ({ ...b, size: val })), "aria-label": "Brush size" }), jsx("span", { children: api.brush.size })] }), jsxs("div", { className: styles.brushSlider, children: [jsx("label", { children: "Opacity" }), jsx(so, { min: 1, max: 100, value: api.brush.opacity, onChange: (val) => api.setBrush((b) => ({ ...b, opacity: val })), "aria-label": "Brush opacity" }), jsxs("span", { children: [api.brush.opacity, "%"] })] }), jsxs("div", { className: styles.brushSlider, children: [jsx("label", { children: "Flow" }), jsx(so, { min: 1, max: 100, value: api.brush.flow, onChange: (val) => api.setBrush((b) => ({ ...b, flow: val })), "aria-label": "Brush flow" }), jsxs("span", { children: [api.brush.flow, "%"] })] }), jsxs("div", { className: styles.brushSlider, children: [jsx("label", { children: "Hard" }), jsx(so, { min: 0, max: 100, value: api.brush.hardness, onChange: (val) => api.setBrush((b) => ({ ...b, hardness: val })), "aria-label": "Brush hardness" }), jsxs("span", { children: [api.brush.hardness, "%"] })] }), jsxs("div", { className: styles.brushSlider, children: [jsx("label", { children: "Space" }), jsx(so, { min: 1, max: 100, value: api.brush.spacing, onChange: (val) => api.setBrush((b) => ({ ...b, spacing: val })), "aria-label": "Brush spacing" }), jsxs("span", { children: [api.brush.spacing, "%"] })] }), jsx("div", { className: styles.brushShapes, children: BRUSH_SHAPES.map((s) => (jsx(We, { className: `${styles.miniBtn} ${api.brush.shape === s.id ? styles.miniBtnActive : ""}`, variant: api.brush.shape === s.id ? 'primary' : 'ghost', size: "sm", onClick: () => api.setBrush((b) => ({ ...b, shape: s.id })), title: s.label, children: s.icon }, s.id))) }), jsxs("div", { className: styles.pressureToggles, children: [jsx(Fr, { checked: api.brush.pressureSize, onChange: (checked) => api.setBrush((b) => ({ ...b, pressureSize: checked })), label: "Pressure\u2192Size", className: styles.checkLabel }), jsx(Fr, { checked: api.brush.pressureOpacity, onChange: (checked) => api.setBrush((b) => ({ ...b, pressureOpacity: checked })), label: "Pressure\u2192Opacity", className: styles.checkLabel })] })] }), jsxs("div", { className: styles.panelSection, children: [jsxs("div", { className: styles.panelTitle, children: [jsx("span", { children: "Colour" }), jsx(We, { variant: "ghost", size: "sm", onClick: () => api.setShowHSV(!api.showHSV), title: "Toggle HSV Picker", children: api.showHSV ? "▾" : "▸" })] }), api.showHSV && jsx(HSVPicker, { color: api.col1, onChange: api.setCol1 })] }), jsxs("div", { className: styles.panelSection, children: [jsxs("div", { className: styles.panelTitle, children: [jsx("span", { children: "Layers" }), jsxs("span", { children: [jsx(We, { variant: "ghost", size: "sm", onClick: api.addLayer, title: "Add Layer", "aria-label": "Add Layer", children: "+" }), jsx(We, { variant: "ghost", size: "sm", onClick: api.dupLayer, title: "Duplicate Layer", "aria-label": "Duplicate Layer", children: "\u29C9" }), jsx(We, { variant: "ghost", size: "sm", onClick: api.rmLayer, title: "Remove Layer", "aria-label": "Remove Layer", children: "\u2212" }), jsx(We, { variant: "ghost", size: "sm", onClick: api.mergeDown, title: "Merge Down", "aria-label": "Merge Down", children: "\u2913" }), jsx(We, { variant: "ghost", size: "sm", onClick: api.moveLayerUp, title: "Move Up", "aria-label": "Move Up", children: "\u2191" }), jsx(We, { variant: "ghost", size: "sm", onClick: api.moveLayerDown, title: "Move Down", "aria-label": "Move Down", children: "\u2193" })] })] }), [...api.layers].map((l, i) => ({ l, i })).reverse().map(({ l, i }) => (jsxs("div", { className: `${styles.layerItem} ${i === api.aLi ? styles.layerItemActive : ""}`, onClick: () => api.setALi(i), children: [jsx("span", { className: styles.layerVis, onClick: (e) => { e.stopPropagation(); api.toggleVis(i); }, children: l.visible ? "👁" : "─" }), jsx("span", { className: styles.layerLock, onClick: (e) => { e.stopPropagation(); api.toggleLock(i); }, title: l.locked ? "Locked" : "Unlocked", children: l.locked ? "🔒" : "🔓" }), api.editingLayerName === l.id ? (jsx(Tt, { className: styles.layerNameInput, value: l.name, onChange: (val) => api.renameLayer(i, val), onBlur: () => api.setEditingLayerName(null), onKeyDown: (e) => { if (e.key === "Enter")
|
|
9
|
+
api.setEditingLayerName(null); }, autoFocus: true, "aria-label": "Layer name" })) : (jsx("span", { className: styles.layerName, onDoubleClick: (e) => { e.stopPropagation(); api.setEditingLayerName(l.id); }, children: l.name })), jsxs("span", { className: styles.layerOpacity, children: [Math.round((l.opacity / 255) * 100), "%"] })] }, l.id))), api.layers[api.aLi] && (jsxs("div", { className: styles.layerControls, children: [jsx(fn, { value: api.layers[api.aLi].blendMode, onChange: (val) => api.setLayerBlendMode(api.aLi, val), className: styles.blendSelect, "aria-label": "Blend mode", options: BLEND_MODES.map((m) => ({ value: m, label: m })) }), jsx(so, { min: 0, max: 255, value: api.layers[api.aLi].opacity, onChange: (val) => api.setLayerOpacity(api.aLi, val), className: styles.layerOpSlider, title: `Opacity: ${Math.round((api.layers[api.aLi].opacity / 255) * 100)}%` })] }))] }), jsxs("div", { className: styles.panelSection, children: [jsxs("div", { className: styles.panelTitle, children: [jsx("span", { children: "Palette" }), jsx(We, { variant: "ghost", size: "sm", onClick: () => { if (!api.pal.includes(api.col1))
|
|
10
10
|
api.setPal((p) => [...p, api.col1]); }, title: "Add current colour", children: "+" })] }), jsx("div", { className: styles.paletteGrid, children: api.pal.map((c, i) => (jsx("div", { className: `${styles.paletteSwatch} ${c === api.col1 ? styles.paletteSwatchActive : ""}`, style: { backgroundColor: c }, onClick: () => api.setCol1(c), onContextMenu: (e) => { e.preventDefault(); api.setCol2(c); }, title: c }, i))) })] })] })));
|
|
11
11
|
PixelEditorRightPanel.displayName = "PixelEditorRightPanel";
|
|
12
12
|
|
|
@@ -1,9 +1,9 @@
|
|
|
1
1
|
import { jsxs, jsx } from 'react/jsx-runtime';
|
|
2
2
|
import React from 'react';
|
|
3
3
|
import styles from './PixelEditor.module.css.js';
|
|
4
|
-
import { NiceButton as
|
|
4
|
+
import { NiceButton as We, NiceNumberInput as ir, NiceCheckbox as Fr } from '../packages/ui/dist/index.js';
|
|
5
5
|
|
|
6
|
-
const PixelEditorTimeline = React.memo(({ api }) => (jsxs("div", { className: styles.bottomPanel, children: [jsxs("div", { className: styles.timelineControls, children: [jsx(
|
|
6
|
+
const PixelEditorTimeline = React.memo(({ api }) => (jsxs("div", { className: styles.bottomPanel, children: [jsxs("div", { className: styles.timelineControls, children: [jsx(We, { variant: "ghost", size: "sm", onClick: api.addFrame, children: "+ Frame" }), jsx(We, { variant: "ghost", size: "sm", onClick: api.dupFrame, "aria-label": "Duplicate frame", children: "Dup" }), jsx(We, { variant: "ghost", size: "sm", onClick: api.rmFrame, "aria-label": "Remove frame", children: "\u2212 Frame" }), jsx("div", { className: styles.menuSep }), jsx(We, { variant: "ghost", size: "sm", onClick: () => api.setPlaying((p) => !p), children: api.playing ? "⏸" : "▶" }), jsx("label", { children: "FPS:" }), jsx(ir, { min: 1, max: 60, value: api.fps, onChange: (val) => api.setFps(val !== null && val !== void 0 ? val : 8), "aria-label": "Frames per second" }), jsx("div", { className: styles.menuSep }), jsx(Fr, { checked: api.onion, onChange: (checked) => api.setOnion(checked), label: "Onion" }), jsx("div", { style: { flex: 1 } }), jsxs("span", { children: [api.fc, " frame", api.fc !== 1 ? "s" : ""] })] }), jsx("div", { className: styles.timelineFrames, children: Array.from({ length: api.fc }, (_, fi) => (jsxs("div", { className: `${styles.frameThumb} ${fi === api.afi ? styles.frameThumbActive : ""}`, onClick: () => api.setAfi(fi), children: [api.thumbs[fi] && (jsx("img", { src: api.thumbs[fi], alt: `Frame ${fi + 1}`, style: { width: 36, height: 36, imageRendering: "pixelated", objectFit: "contain" } })), jsx("span", { className: styles.frameNum, children: fi + 1 })] }, fi))) })] })));
|
|
7
7
|
PixelEditorTimeline.displayName = "PixelEditorTimeline";
|
|
8
8
|
|
|
9
9
|
export { PixelEditorTimeline as default };
|
|
@@ -1,10 +1,10 @@
|
|
|
1
1
|
import { jsxs, jsx } from 'react/jsx-runtime';
|
|
2
2
|
import React from 'react';
|
|
3
3
|
import styles from './PixelEditor.module.css.js';
|
|
4
|
-
import { NiceButton as
|
|
4
|
+
import { NiceButton as We, NiceColorPicker as Rn } from '../packages/ui/dist/index.js';
|
|
5
5
|
import { TOOLS, SYMMETRY_MODES } from './pixelEditorTypes.js';
|
|
6
6
|
|
|
7
|
-
const PixelEditorToolbar = React.memo(({ api }) => (jsxs("div", { className: styles.toolbar, children: [TOOLS.map((t) => (jsx(
|
|
7
|
+
const PixelEditorToolbar = React.memo(({ api }) => (jsxs("div", { className: styles.toolbar, children: [TOOLS.map((t) => (jsx(We, { className: `${styles.toolBtn} ${api.tool === t.id ? styles.toolBtnActive : ""}`, variant: api.tool === t.id ? 'primary' : 'ghost', size: "sm", onClick: () => api.setTool(t.id), title: `${t.label} (${t.key})`, children: t.icon }, t.id))), jsx("div", { className: styles.toolSep }), jsx(We, { className: `${styles.toolBtn} ${api.shapeFilled ? styles.toolBtnActive : ""}`, variant: api.shapeFilled ? 'primary' : 'ghost', size: "sm", onClick: () => api.setShapeFilled((f) => !f), title: `${api.shapeFilled ? "Filled" : "Outline"} (F)`, children: api.shapeFilled ? "■" : "□" }), jsx("div", { className: styles.toolGroup, children: SYMMETRY_MODES.map((m) => (jsx(We, { className: `${styles.miniBtn} ${api.symmetry === m.id ? styles.miniBtnActive : ""}`, variant: api.symmetry === m.id ? 'primary' : 'ghost', size: "sm", onClick: () => api.setSymmetry(m.id), title: `Symmetry: ${m.label}`, children: m.label.slice(0, 2) }, m.id))) }), jsxs("div", { className: styles.colorSwatches, children: [jsx("label", { style: { position: "relative", width: 28, height: 28, cursor: "pointer" }, title: "Primary Colour (click for HSV)", children: jsx(Rn, { value: api.col1, onChange: (c) => api.setCol1(c) }) }), jsx("label", { style: { position: "relative", width: 22, height: 22, cursor: "pointer" }, title: "Secondary Colour", children: jsx(Rn, { value: api.col2, onChange: (c) => api.setCol2(c) }) }), jsx(We, { className: styles.toolBtn, variant: "ghost", size: "sm", onClick: () => {
|
|
8
8
|
const tmp = api.col1;
|
|
9
9
|
api.setCol1(api.col2);
|
|
10
10
|
api.setCol2(tmp);
|
|
@@ -1,7 +1,7 @@
|
|
|
1
1
|
import { jsx, jsxs } from 'react/jsx-runtime';
|
|
2
2
|
import { useState, useRef, useEffect, useMemo, useCallback } from 'react';
|
|
3
3
|
import styles from './PixelEditor.module.css.js';
|
|
4
|
-
import { NiceModal as
|
|
4
|
+
import { NiceModal as Ys, NiceButton as We, NiceSelect as fn, NiceNumberInput as ir, NiceCheckbox as Fr } from '../packages/ui/dist/index.js';
|
|
5
5
|
import { getAllParts, assembleSprite, applyIsometricProjection, getPart, generateAllDirections, render8DirSpriteSheet, getPartsByCategory, ALL_DIRECTIONS } from './spriteGeneratorCore.js';
|
|
6
6
|
import { CHARACTER_PRESETS, initCharacterParts, SKIN_PALETTES, HAIR_PALETTES, CLOTHING_PALETTES } from './characterGeneratorPresets.js';
|
|
7
7
|
import { BUILDING_PRESETS, assembleBuildingFromConfig, initBuildingParts, WALL_PALETTES, ROOF_PALETTES } from './buildingGeneratorPresets.js';
|
|
@@ -195,26 +195,26 @@ const NiceSpriteGeneratorPanel = ({ onSendToCanvas, onClose, }) => {
|
|
|
195
195
|
}
|
|
196
196
|
}, [tab, skinPalette, hairPalette, clothPalette, wallPalette, roofPalette, furPalette]);
|
|
197
197
|
/* ── Render ── */
|
|
198
|
-
return (jsx(
|
|
198
|
+
return (jsx(Ys, { open: true, onClose: onClose, title: "Sprite Generator", children: jsxs("div", { className: styles.genPanel, children: [jsxs("div", { className: styles.genHeader, children: [jsx("h3", { children: "Sprite Generator" }), jsx(We, { className: styles.btnIcon, variant: "ghost", size: "sm", onClick: onClose, title: "Close", children: "\u2715" })] }), jsx("div", { className: styles.genTabs, children: ['character', 'building', 'animal'].map(t => (jsxs(We, { className: `${styles.genTabBtn} ${tab === t ? styles.genTabBtnActive : ''}`, variant: tab === t ? 'primary' : 'ghost', size: "sm", onClick: () => setTab(t), children: [t === 'character' ? '🧑' : t === 'building' ? '🏠' : '🐾', ' ', t.charAt(0).toUpperCase() + t.slice(1)] }, t))) }), jsxs("div", { className: styles.genBody, children: [jsxs("div", { className: styles.genLeft, children: [jsxs("div", { className: styles.panelSection, children: [jsx("div", { className: styles.panelTitle, children: jsx("span", { children: "Presets" }) }), jsx("div", { className: styles.genPresetGrid, children: presets.map((preset, i) => (jsx(We, { className: `${styles.genPresetBtn} ${i === selectedPresetIdx ? styles.genPresetBtnActive : ''}`, variant: i === selectedPresetIdx ? 'primary' : 'ghost', size: "sm", onClick: () => applyPreset(i), title: preset.name, children: preset.name }, i))) })] }), jsxs("div", { className: styles.panelSection, style: { flex: 1, overflowY: 'auto' }, children: [jsx("div", { className: styles.panelTitle, children: jsx("span", { children: "Parts" }) }), categories.map(cat => {
|
|
199
199
|
const parts = getPartsByCategory(cat);
|
|
200
200
|
if (parts.length === 0)
|
|
201
201
|
return null;
|
|
202
202
|
return (jsxs("div", { style: { marginBottom: 6 }, children: [jsx("div", { style: { fontSize: 9, color: 'var(--gray-600)', marginBottom: 2, textTransform: 'uppercase' }, children: cat.replace(/([A-Z])/g, ' $1').trim() }), jsx("div", { className: styles.genPartGrid, children: parts.map(part => {
|
|
203
203
|
const active = selectedParts.some(p => { var _a; return p.partId === part.id && ((_a = p.visible) !== null && _a !== void 0 ? _a : true); });
|
|
204
|
-
return (jsx(
|
|
204
|
+
return (jsx(We, { className: `${styles.miniBtn} ${active ? styles.miniBtnActive : ''}`, variant: active ? 'primary' : 'ghost', size: "sm", onClick: () => active ? removePart(part.id) : addPart(part.id), title: part.name, children: part.name.slice(0, 8) }, part.id));
|
|
205
205
|
}) })] }, cat));
|
|
206
|
-
})] })] }), jsxs("div", { className: styles.genCenter, children: [jsx("div", { className: styles.genPreviewWrap, children: jsx("canvas", { ref: previewRef, className: styles.genPreviewCanvas }) }), jsx("div", { className: styles.genDirGrid, children: ALL_DIRECTIONS.map(d => (jsx(
|
|
206
|
+
})] })] }), jsxs("div", { className: styles.genCenter, children: [jsx("div", { className: styles.genPreviewWrap, children: jsx("canvas", { ref: previewRef, className: styles.genPreviewCanvas }) }), jsx("div", { className: styles.genDirGrid, children: ALL_DIRECTIONS.map(d => (jsx(We, { className: `${styles.miniBtn} ${direction === d ? styles.miniBtnActive : ''}`, variant: direction === d ? 'primary' : 'ghost', size: "sm", onClick: () => setDirection(d), children: d }, d))) }), jsxs("div", { className: styles.genRow, children: [jsx("label", { style: { fontSize: 10, color: 'var(--gray-600)' }, children: "Projection" }), jsx(fn, { className: styles.propSelect, value: projection, onChange: val => setProjection(val), options: [
|
|
207
207
|
{ value: 'topDown', label: 'Top-Down' },
|
|
208
208
|
{ value: 'sideView', label: 'Side View' },
|
|
209
209
|
{ value: 'iso25d', label: '2.5D Isometric' },
|
|
210
|
-
] })] }), jsxs("div", { className: styles.genRow, children: [jsx("label", { style: { fontSize: 10, color: 'var(--gray-600)' }, children: "Size" }), jsx(
|
|
210
|
+
] })] }), jsxs("div", { className: styles.genRow, children: [jsx("label", { style: { fontSize: 10, color: 'var(--gray-600)' }, children: "Size" }), jsx(ir, { className: styles.propInput, style: { width: 48 }, min: 8, max: 256, value: canvasW, onChange: val => setCanvasW(val !== null && val !== void 0 ? val : 32) }), jsx("span", { style: { fontSize: 10, color: 'var(--gray-600)' }, children: "\u00D7" }), jsx(ir, { className: styles.propInput, style: { width: 48 }, min: 8, max: 256, value: canvasH, onChange: val => setCanvasH(val !== null && val !== void 0 ? val : 32) })] }), jsx(Fr, { checked: generate8Dir, onChange: checked => setGenerate8Dir(checked), label: "Generate 8-direction sheet", className: styles.checkLabel })] }), jsxs("div", { className: styles.genRight, children: [jsxs("div", { className: styles.panelSection, children: [jsx("div", { className: styles.panelTitle, children: jsx("span", { children: "Colors" }) }), paletteGroups.map(group => {
|
|
211
211
|
var _a;
|
|
212
|
-
return (jsxs("div", { style: { marginBottom: 6 }, children: [jsx("div", { style: { fontSize: 9, color: 'var(--gray-600)', marginBottom: 2 }, children: group.label }), jsx(
|
|
212
|
+
return (jsxs("div", { style: { marginBottom: 6 }, children: [jsx("div", { style: { fontSize: 9, color: 'var(--gray-600)', marginBottom: 2 }, children: group.label }), jsx(fn, { className: styles.propSelect, value: group.value, onChange: val => group.set(val), options: group.options.map(opt => ({ value: opt.id, label: opt.name })) }), jsx("div", { className: styles.genColorSwatches, children: (_a = group.options.find(o => o.id === group.value)) === null || _a === void 0 ? void 0 : _a.colors.map((c, i) => (jsx("div", { className: styles.genColorSwatch, style: { backgroundColor: c }, title: c }, i))) })] }, group.label));
|
|
213
213
|
})] }), jsxs("div", { className: styles.panelSection, style: { flex: 1, overflowY: 'auto' }, children: [jsx("div", { className: styles.panelTitle, children: jsx("span", { children: "Assembly" }) }), selectedParts.map((ap, i) => {
|
|
214
214
|
var _a, _b, _c, _d;
|
|
215
215
|
const part = partsMapRef.current.get(ap.partId);
|
|
216
|
-
return (jsxs("div", { className: styles.genAssemblyItem, children: [jsx(
|
|
217
|
-
})] })] })] }), jsxs("div", { className: styles.genFooter, children: [jsx(
|
|
216
|
+
return (jsxs("div", { className: styles.genAssemblyItem, children: [jsx(We, { className: styles.btnIcon, variant: "ghost", size: "sm", onClick: () => togglePart(ap.partId), title: ((_a = ap.visible) !== null && _a !== void 0 ? _a : true) ? 'Hide' : 'Show', children: ((_b = ap.visible) !== null && _b !== void 0 ? _b : true) ? '👁' : '─' }), jsx("span", { className: styles.genAssemblyName, children: (_c = part === null || part === void 0 ? void 0 : part.name) !== null && _c !== void 0 ? _c : ap.partId }), jsxs("span", { style: { fontSize: 9, color: 'var(--gray-700)' }, children: ["z:", (_d = part === null || part === void 0 ? void 0 : part.zOrder) !== null && _d !== void 0 ? _d : 0] }), jsx(We, { className: styles.btnIcon, variant: "ghost", size: "sm", onClick: () => removePart(ap.partId), title: "Remove", children: "\u2715" })] }, ap.partId));
|
|
217
|
+
})] })] })] }), jsxs("div", { className: styles.genFooter, children: [jsx(We, { className: styles.btnSecondary, variant: "ghost", onClick: onClose, children: "Cancel" }), jsx(We, { className: styles.btnPrimary, variant: "primary", onClick: handleSendToCanvas, children: "\uD83D\uDCCB Send to Canvas" })] })] }) }));
|
|
218
218
|
};
|
|
219
219
|
NiceSpriteGeneratorPanel.displayName = 'NiceSpriteGeneratorPanel';
|
|
220
220
|
|
|
@@ -1,5 +1,5 @@
|
|
|
1
1
|
import { jsxs, jsx } from 'react/jsx-runtime';
|
|
2
|
-
import { NiceButton as
|
|
2
|
+
import { NiceButton as We, NiceModal as Ys, NiceTextArea as Os } from '../packages/ui/dist/index.js';
|
|
3
3
|
import styles from './VectorEditor.module.css.js';
|
|
4
4
|
import { TOOLS } from './vectorEditorTypes.js';
|
|
5
5
|
import { ShapeRenderer } from './VectorEditorShapeRenderer.js';
|
|
@@ -17,13 +17,13 @@ const NiceVectorEditor = ({ width: artW = 512, height: artH = 512, className, on
|
|
|
17
17
|
if (f)
|
|
18
18
|
openFile(f);
|
|
19
19
|
e.target.value = "";
|
|
20
|
-
} }), jsx(VectorEditorMenuBar, { api: api, hasOnSaveToLibrary: !!onSaveToLibrary }), jsxs("div", { className: styles.mainArea, children: [jsx("div", { className: styles.toolbar, children: TOOLS.map((t) => (jsx(
|
|
20
|
+
} }), jsx(VectorEditorMenuBar, { api: api, hasOnSaveToLibrary: !!onSaveToLibrary }), jsxs("div", { className: styles.mainArea, children: [jsx("div", { className: styles.toolbar, children: TOOLS.map((t) => (jsx(We, { className: `${styles.toolBtn} ${tool === t.id ? styles.toolBtnActive : ""}`, onClick: () => switchTool(t.id), "aria-label": `${t.label} (${t.key})`, variant: tool === t.id ? "primary" : "ghost", size: "sm", children: t.icon }, t.id))) }), jsx("div", { ref: boxRef, className: styles.canvasContainer, children: jsxs("svg", { viewBox: `0 0 ${artW} ${artH}`, onMouseDown: onDown, onMouseMove: onMove, onMouseUp: onUp, onMouseLeave: () => {
|
|
21
21
|
/* handled internally via refs */
|
|
22
22
|
}, onDoubleClick: onDblClick, onWheel: onWheel, onContextMenu: (e) => e.preventDefault(), style: {
|
|
23
23
|
transform: `translate(${px}px, ${py}px) scale(${zoom})`,
|
|
24
24
|
transformOrigin: "center center",
|
|
25
25
|
cursor: tool === "pointer" ? "default" : "crosshair",
|
|
26
|
-
}, children: [jsx("rect", { x: 0, y: 0, width: artW, height: artH, fill: "#222" }), zoom >= 2 && (jsxs("g", { opacity: 0.08, children: [Array.from({ length: Math.floor(artW / gridSize) + 1 }, (_, i) => (jsx("line", { x1: i * gridSize, y1: 0, x2: i * gridSize, y2: artH, stroke: "#fff", strokeWidth: 1 / zoom }, `gx${i}`))), Array.from({ length: Math.floor(artH / gridSize) + 1 }, (_, i) => (jsx("line", { x1: 0, y1: i * gridSize, x2: artW, y2: i * gridSize, stroke: "#fff", strokeWidth: 1 / zoom }, `gy${i}`)))] })), shapes.map((s) => (jsx(ShapeRenderer, { shape: s, selected: s.id === selId, zoom: zoom }, s.id)))] }) }), jsx(VectorEditorRightPanel, { api: api })] }), jsx(VectorEditorStatusBar, { api: api, artW: artW, artH: artH }), jsxs(
|
|
26
|
+
}, children: [jsx("rect", { x: 0, y: 0, width: artW, height: artH, fill: "#222" }), zoom >= 2 && (jsxs("g", { opacity: 0.08, children: [Array.from({ length: Math.floor(artW / gridSize) + 1 }, (_, i) => (jsx("line", { x1: i * gridSize, y1: 0, x2: i * gridSize, y2: artH, stroke: "#fff", strokeWidth: 1 / zoom }, `gx${i}`))), Array.from({ length: Math.floor(artH / gridSize) + 1 }, (_, i) => (jsx("line", { x1: 0, y1: i * gridSize, x2: artW, y2: i * gridSize, stroke: "#fff", strokeWidth: 1 / zoom }, `gy${i}`)))] })), shapes.map((s) => (jsx(ShapeRenderer, { shape: s, selected: s.id === selId, zoom: zoom }, s.id)))] }) }), jsx(VectorEditorRightPanel, { api: api })] }), jsx(VectorEditorStatusBar, { api: api, artW: artW, artH: artH }), jsxs(Ys, { open: dlgSvg, onClose: () => setDlgSvg(false), title: "Paste SVG", children: [jsx("label", { children: "Paste SVG source code below:" }), jsx(Os, { value: svgText, onChange: (val) => setSvgText(val), placeholder: '<svg xmlns="http://www.w3.org/2000/svg" ...>' }), jsxs("div", { className: styles.dialogActions, children: [jsx(We, { variant: "ghost", onClick: () => setDlgSvg(false), children: "Cancel" }), jsx(We, { variant: "primary", onClick: () => importSVGString(svgText), children: "Import" })] })] })] }));
|
|
27
27
|
};
|
|
28
28
|
/** @deprecated Use NiceVectorEditor */
|
|
29
29
|
const VectorEditor = NiceVectorEditor;
|
|
@@ -1,16 +1,16 @@
|
|
|
1
1
|
import { jsxs, jsx, Fragment } from 'react/jsx-runtime';
|
|
2
2
|
import React from 'react';
|
|
3
|
-
import { NiceButton as
|
|
3
|
+
import { NiceButton as We, NiceSelect as fn, NiceCheckbox as Fr } from '../packages/ui/dist/index.js';
|
|
4
4
|
import styles from './VectorEditor.module.css.js';
|
|
5
5
|
|
|
6
6
|
const VectorEditorMenuBar$1 = ({ api, hasOnSaveToLibrary }) => {
|
|
7
7
|
const { selId, hasClipboard, snapGrid, fInp, shapes, boolSecondShape, newCanvas, setDlgSvg, setBoolSecondShape, exportSVG, exportPNG, exportJPG, exportWebP, exportBMP, undo, redo, duplicateShape, copyShape, pasteShape, deleteShape, bringToFront, sendToBack, flipH, flipV, alignShapes, setSnapGrid, saveToLibrary, unionShapes, subtractShapes, intersectShapes, excludeShapes, outlineStroke, } = api;
|
|
8
8
|
// Check if we can perform boolean ops (need 2 shapes selected)
|
|
9
9
|
const canBool = selId && boolSecondShape && selId !== boolSecondShape;
|
|
10
|
-
return (jsxs("div", { className: styles.menuBar, children: [jsx(
|
|
10
|
+
return (jsxs("div", { className: styles.menuBar, children: [jsx(We, { className: styles.menuBtn, variant: "ghost", size: "sm", onClick: newCanvas, children: "New" }), jsx(We, { className: styles.menuBtn, variant: "ghost", size: "sm", onClick: () => { var _a; return (_a = fInp.current) === null || _a === void 0 ? void 0 : _a.click(); }, children: "Open SVG" }), jsx(We, { className: styles.menuBtn, variant: "ghost", size: "sm", onClick: () => setDlgSvg(true), children: "Paste SVG" }), jsx("div", { className: styles.menuSep }), jsx(We, { className: styles.menuBtn, variant: "ghost", size: "sm", onClick: exportSVG, children: "SVG" }), jsx(We, { className: styles.menuBtn, variant: "ghost", size: "sm", onClick: exportPNG, children: "PNG" }), jsx(We, { className: styles.menuBtn, variant: "ghost", size: "sm", onClick: exportJPG, children: "JPG" }), jsx(We, { className: styles.menuBtn, variant: "ghost", size: "sm", onClick: exportWebP, children: "WebP" }), jsx(We, { className: styles.menuBtn, variant: "ghost", size: "sm", onClick: exportBMP, children: "BMP" }), jsx("div", { className: styles.menuSep }), jsx(We, { className: styles.menuBtn, variant: "ghost", size: "sm", onClick: undo, children: "Undo" }), jsx(We, { className: styles.menuBtn, variant: "ghost", size: "sm", onClick: redo, children: "Redo" }), jsx("div", { className: styles.menuSep }), jsx(We, { className: styles.menuBtn, variant: "ghost", size: "sm", onClick: duplicateShape, disabled: !selId, "aria-label": "Ctrl+D", children: "Dup" }), jsx(We, { className: styles.menuBtn, variant: "ghost", size: "sm", onClick: copyShape, disabled: !selId, "aria-label": "Ctrl+C", children: "Copy" }), jsx(We, { className: styles.menuBtn, variant: "ghost", size: "sm", onClick: pasteShape, disabled: !hasClipboard, "aria-label": "Ctrl+V", children: "Paste" }), jsx("div", { className: styles.menuSep }), jsx(We, { className: styles.menuBtn, variant: "ghost", size: "sm", onClick: deleteShape, disabled: !selId, children: "Delete" }), jsx(We, { className: styles.menuBtn, variant: "ghost", size: "sm", onClick: bringToFront, disabled: !selId, children: "\u2191 Front" }), jsx(We, { className: styles.menuBtn, variant: "ghost", size: "sm", onClick: sendToBack, disabled: !selId, children: "\u2193 Back" }), jsx("div", { className: styles.menuSep }), jsx(We, { className: styles.menuBtn, variant: "ghost", size: "sm", onClick: flipH, disabled: !selId, "aria-label": "Flip H", children: "\u21D4" }), jsx(We, { className: styles.menuBtn, variant: "ghost", size: "sm", onClick: flipV, disabled: !selId, "aria-label": "Flip V", children: "\u21D5" }), jsx("div", { className: styles.menuSep }), jsx(We, { className: styles.menuBtn, variant: "ghost", size: "sm", onClick: () => alignShapes("left"), disabled: !selId, "aria-label": "Align Left", children: "\u25E7" }), jsx(We, { className: styles.menuBtn, variant: "ghost", size: "sm", onClick: () => alignShapes("center"), disabled: !selId, "aria-label": "Align Center", children: "\u25EB" }), jsx(We, { className: styles.menuBtn, variant: "ghost", size: "sm", onClick: () => alignShapes("right"), disabled: !selId, "aria-label": "Align Right", children: "\u25E8" }), jsx("div", { className: styles.menuSep }), shapes.length >= 2 && (jsxs(Fragment, { children: [jsx(fn, { className: styles.menuBtn, value: boolSecondShape || "", onChange: (val) => setBoolSecondShape(val || null), options: [
|
|
11
11
|
{ value: "", label: "2nd Shape" },
|
|
12
12
|
...shapes.filter(s => s.id !== selId).map((s, i) => ({ value: s.id, label: `Shape ${i + 1}` })),
|
|
13
|
-
], style: { minWidth: 80 } }), jsx(
|
|
13
|
+
], style: { minWidth: 80 } }), jsx(We, { className: styles.menuBtn, variant: "ghost", size: "sm", onClick: unionShapes, disabled: !canBool, "aria-label": "Union", children: "\u222A" }), jsx(We, { className: styles.menuBtn, variant: "ghost", size: "sm", onClick: subtractShapes, disabled: !canBool, "aria-label": "Subtract", children: "\u2212" }), jsx(We, { className: styles.menuBtn, variant: "ghost", size: "sm", onClick: intersectShapes, disabled: !canBool, "aria-label": "Intersect", children: "\u2229" }), jsx(We, { className: styles.menuBtn, variant: "ghost", size: "sm", onClick: excludeShapes, disabled: !canBool, "aria-label": "Exclude", children: "\u2295" }), jsx(We, { className: styles.menuBtn, variant: "ghost", size: "sm", onClick: outlineStroke, disabled: !selId, "aria-label": "Outline Stroke", children: "\u25AD" }), jsx("div", { className: styles.menuSep })] })), jsxs("label", { className: styles.menuLabel, children: [jsx(Fr, { checked: snapGrid, onChange: () => setSnapGrid((v) => !v) }), "Snap"] }), hasOnSaveToLibrary && (jsxs(Fragment, { children: [jsx("div", { className: styles.menuSep }), jsx(We, { className: styles.menuBtn, variant: "ghost", size: "sm", onClick: saveToLibrary, children: "\uD83D\uDCE6 Save to Library" })] }))] }));
|
|
14
14
|
};
|
|
15
15
|
var VectorEditorMenuBar = React.memo(VectorEditorMenuBar$1);
|
|
16
16
|
|
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
import { jsxs, jsx, Fragment } from 'react/jsx-runtime';
|
|
2
2
|
import React, { useState } from 'react';
|
|
3
|
-
import { NiceButton as
|
|
3
|
+
import { NiceButton as We, NiceColorPicker as Rn, NiceTextInput as Tt, NiceSelect as fn, NiceSlider as so, NiceNumberInput as ir } from '../packages/ui/dist/index.js';
|
|
4
4
|
import styles from './VectorEditor.module.css.js';
|
|
5
5
|
import { FONTS } from './vectorEditorTypes.js';
|
|
6
6
|
import { PRESET_GRADIENTS, gradientToCSS, gradientId, cloneGradient } from './vectorGradients.js';
|
|
@@ -21,14 +21,14 @@ const VectorEditorRightPanel$1 = ({ api }) => {
|
|
|
21
21
|
// Store gradient data as custom property (will be used in SVG export)
|
|
22
22
|
});
|
|
23
23
|
};
|
|
24
|
-
return (jsxs("div", { className: styles.rightPanel, children: [jsxs("div", { className: styles.panelSection, children: [jsx("div", { className: styles.panelTitle, children: jsx("span", { children: "Appearance" }) }), jsxs("div", { className: styles.propRow, children: [jsx("span", { className: styles.propLabel, children: "Fill" }), jsxs("div", { style: { display: "flex", gap: 4 }, children: [jsx(
|
|
24
|
+
return (jsxs("div", { className: styles.rightPanel, children: [jsxs("div", { className: styles.panelSection, children: [jsx("div", { className: styles.panelTitle, children: jsx("span", { children: "Appearance" }) }), jsxs("div", { className: styles.propRow, children: [jsx("span", { className: styles.propLabel, children: "Fill" }), jsxs("div", { style: { display: "flex", gap: 4 }, children: [jsx(We, { className: `${styles.menuBtn} ${fillMode === "solid" ? styles.menuBtnActive : ""}`, style: { padding: "2px 6px", fontSize: 10 }, variant: fillMode === "solid" ? "primary" : "ghost", size: "sm", onClick: () => setFillMode("solid"), children: "Solid" }), jsx(We, { className: `${styles.menuBtn} ${fillMode === "gradient" ? styles.menuBtnActive : ""}`, style: { padding: "2px 6px", fontSize: 10 }, variant: fillMode === "gradient" ? "primary" : "ghost", size: "sm", onClick: () => setFillMode("gradient"), children: "Gradient" })] })] }), fillMode === "solid" ? (jsxs("div", { className: styles.propRow, children: [jsx("span", { className: styles.propLabel }), jsx("div", { className: styles.propColorSwatch, style: { background: fillCol }, children: jsx(Rn, { value: fillCol, onChange: (c) => { setFillCol(c); updateSel({ fill: c }); } }) }), jsx(Tt, { className: styles.propInput, value: (selShape === null || selShape === void 0 ? void 0 : selShape.fill) || fillCol, onChange: (val) => { setFillCol(val); updateSel({ fill: val }); } })] })) : (jsxs(Fragment, { children: [jsx("div", { className: styles.propRow, style: { flexWrap: "wrap", gap: 4 }, children: PRESET_GRADIENTS.map((grad, i) => (jsx("div", { style: {
|
|
25
25
|
width: 24,
|
|
26
26
|
height: 24,
|
|
27
27
|
borderRadius: 4,
|
|
28
28
|
background: gradientToCSS(grad),
|
|
29
29
|
cursor: "pointer",
|
|
30
30
|
border: (selectedGradient === null || selectedGradient === void 0 ? void 0 : selectedGradient.id) === grad.id ? "2px solid #fff" : "1px solid #444",
|
|
31
|
-
}, onClick: () => applyGradient(grad), title: `${grad.type} gradient` }, i))) }), jsx("div", { className: styles.propRow, children: jsx(
|
|
31
|
+
}, onClick: () => applyGradient(grad), title: `${grad.type} gradient` }, i))) }), jsx("div", { className: styles.propRow, children: jsx(We, { className: styles.menuBtn, variant: "ghost", size: "sm", style: { width: "100%", fontSize: 10 }, onClick: () => setShowGradientEditor(!showGradientEditor), children: showGradientEditor ? "▼ Hide Editor" : "► Edit Gradient" }) }), showGradientEditor && selectedGradient && (jsxs("div", { style: { padding: 4, background: "#252525", borderRadius: 4, margin: "4px 0" }, children: [jsxs("div", { className: styles.propRow, children: [jsx("span", { className: styles.propLabel, children: "Type" }), jsx(fn, { className: styles.propInput, value: selectedGradient.type, onChange: (val) => {
|
|
32
32
|
const type = val;
|
|
33
33
|
let newGrad;
|
|
34
34
|
if (type === "linear") {
|
|
@@ -54,7 +54,7 @@ const VectorEditorRightPanel$1 = ({ api }) => {
|
|
|
54
54
|
}, options: [
|
|
55
55
|
{ value: "linear", label: "Linear" },
|
|
56
56
|
{ value: "radial", label: "Radial" },
|
|
57
|
-
] })] }), selectedGradient.type === "linear" && (jsxs("div", { className: styles.propRow, children: [jsx("span", { className: styles.propLabel, children: "Angle" }), jsx(
|
|
57
|
+
] })] }), selectedGradient.type === "linear" && (jsxs("div", { className: styles.propRow, children: [jsx("span", { className: styles.propLabel, children: "Angle" }), jsx(so, { className: styles.propInput, min: 0, max: 360, value: selectedGradient.angle, onChange: (val) => {
|
|
58
58
|
const newGrad = {
|
|
59
59
|
...selectedGradient,
|
|
60
60
|
id: gradientId(),
|
|
@@ -62,7 +62,7 @@ const VectorEditorRightPanel$1 = ({ api }) => {
|
|
|
62
62
|
};
|
|
63
63
|
setSelectedGradient(newGrad);
|
|
64
64
|
updateSel({ fill: `url(#${newGrad.id})` });
|
|
65
|
-
} }), jsxs("span", { style: { fontSize: 10, minWidth: 28 }, children: [selectedGradient.angle, "\u00B0"] })] })), selectedGradient.type === "radial" && (jsxs(Fragment, { children: [jsxs("div", { className: styles.propRow, children: [jsx("span", { className: styles.propLabel, children: "CX" }), jsx(
|
|
65
|
+
} }), jsxs("span", { style: { fontSize: 10, minWidth: 28 }, children: [selectedGradient.angle, "\u00B0"] })] })), selectedGradient.type === "radial" && (jsxs(Fragment, { children: [jsxs("div", { className: styles.propRow, children: [jsx("span", { className: styles.propLabel, children: "CX" }), jsx(so, { className: styles.propInput, min: 0, max: 100, value: selectedGradient.cx * 100, onChange: (val) => {
|
|
66
66
|
const newGrad = {
|
|
67
67
|
...selectedGradient,
|
|
68
68
|
id: gradientId(),
|
|
@@ -70,7 +70,7 @@ const VectorEditorRightPanel$1 = ({ api }) => {
|
|
|
70
70
|
};
|
|
71
71
|
setSelectedGradient(newGrad);
|
|
72
72
|
updateSel({ fill: `url(#${newGrad.id})` });
|
|
73
|
-
} })] }), jsxs("div", { className: styles.propRow, children: [jsx("span", { className: styles.propLabel, children: "CY" }), jsx(
|
|
73
|
+
} })] }), jsxs("div", { className: styles.propRow, children: [jsx("span", { className: styles.propLabel, children: "CY" }), jsx(so, { className: styles.propInput, min: 0, max: 100, value: selectedGradient.cy * 100, onChange: (val) => {
|
|
74
74
|
const newGrad = {
|
|
75
75
|
...selectedGradient,
|
|
76
76
|
id: gradientId(),
|
|
@@ -78,7 +78,7 @@ const VectorEditorRightPanel$1 = ({ api }) => {
|
|
|
78
78
|
};
|
|
79
79
|
setSelectedGradient(newGrad);
|
|
80
80
|
updateSel({ fill: `url(#${newGrad.id})` });
|
|
81
|
-
} })] }), jsxs("div", { className: styles.propRow, children: [jsx("span", { className: styles.propLabel, children: "R" }), jsx(
|
|
81
|
+
} })] }), jsxs("div", { className: styles.propRow, children: [jsx("span", { className: styles.propLabel, children: "R" }), jsx(so, { className: styles.propInput, min: 10, max: 100, value: selectedGradient.r * 100, onChange: (val) => {
|
|
82
82
|
const newGrad = {
|
|
83
83
|
...selectedGradient,
|
|
84
84
|
id: gradientId(),
|
|
@@ -86,33 +86,33 @@ const VectorEditorRightPanel$1 = ({ api }) => {
|
|
|
86
86
|
};
|
|
87
87
|
setSelectedGradient(newGrad);
|
|
88
88
|
updateSel({ fill: `url(#${newGrad.id})` });
|
|
89
|
-
} })] })] })), jsx("div", { style: { fontSize: 10, color: "#888", marginTop: 4 }, children: "Color Stops:" }), selectedGradient.stops.map((stop, i) => (jsxs("div", { className: styles.propRow, children: [jsx(
|
|
89
|
+
} })] })] })), jsx("div", { style: { fontSize: 10, color: "#888", marginTop: 4 }, children: "Color Stops:" }), selectedGradient.stops.map((stop, i) => (jsxs("div", { className: styles.propRow, children: [jsx(Rn, { value: stop.color, style: { width: 24, height: 20 }, onChange: (color) => {
|
|
90
90
|
const newStops = [...selectedGradient.stops];
|
|
91
91
|
newStops[i] = { ...newStops[i], color };
|
|
92
92
|
const newGrad = { ...selectedGradient, id: gradientId(), stops: newStops };
|
|
93
93
|
setSelectedGradient(newGrad);
|
|
94
94
|
updateSel({ fill: `url(#${newGrad.id})` });
|
|
95
|
-
} }), jsx(
|
|
95
|
+
} }), jsx(so, { min: 0, max: 100, value: stop.offset * 100, style: { flex: 1 }, onChange: (val) => {
|
|
96
96
|
const newStops = [...selectedGradient.stops];
|
|
97
97
|
newStops[i] = { ...newStops[i], offset: val / 100 };
|
|
98
98
|
newStops.sort((a, b) => a.offset - b.offset);
|
|
99
99
|
const newGrad = { ...selectedGradient, id: gradientId(), stops: newStops };
|
|
100
100
|
setSelectedGradient(newGrad);
|
|
101
101
|
updateSel({ fill: `url(#${newGrad.id})` });
|
|
102
|
-
} }), jsxs("span", { style: { fontSize: 9, minWidth: 24 }, children: [Math.round(stop.offset * 100), "%"] })] }, i))), jsxs("div", { className: styles.propRow, style: { gap: 4 }, children: [jsx(
|
|
102
|
+
} }), jsxs("span", { style: { fontSize: 9, minWidth: 24 }, children: [Math.round(stop.offset * 100), "%"] })] }, i))), jsxs("div", { className: styles.propRow, style: { gap: 4 }, children: [jsx(We, { className: styles.menuBtn, variant: "ghost", size: "sm", style: { fontSize: 10, flex: 1 }, onClick: () => {
|
|
103
103
|
const newStops = [...selectedGradient.stops, { offset: 0.5, color: "#888888" }];
|
|
104
104
|
newStops.sort((a, b) => a.offset - b.offset);
|
|
105
105
|
const newGrad = { ...selectedGradient, id: gradientId(), stops: newStops };
|
|
106
106
|
setSelectedGradient(newGrad);
|
|
107
107
|
updateSel({ fill: `url(#${newGrad.id})` });
|
|
108
|
-
}, children: "+ Add Stop" }), jsx(
|
|
108
|
+
}, children: "+ Add Stop" }), jsx(We, { className: styles.menuBtn, variant: "ghost", size: "sm", style: { fontSize: 10, flex: 1 }, disabled: selectedGradient.stops.length <= 2, onClick: () => {
|
|
109
109
|
if (selectedGradient.stops.length <= 2)
|
|
110
110
|
return;
|
|
111
111
|
const newStops = selectedGradient.stops.slice(0, -1);
|
|
112
112
|
const newGrad = { ...selectedGradient, id: gradientId(), stops: newStops };
|
|
113
113
|
setSelectedGradient(newGrad);
|
|
114
114
|
updateSel({ fill: `url(#${newGrad.id})` });
|
|
115
|
-
}, children: "- Remove" })] })] }))] })), jsxs("div", { className: styles.propRow, children: [jsx("span", { className: styles.propLabel, children: "Stroke" }), jsx("div", { className: styles.propColorSwatch, style: { background: strokeCol }, children: jsx(
|
|
115
|
+
}, children: "- Remove" })] })] }))] })), jsxs("div", { className: styles.propRow, children: [jsx("span", { className: styles.propLabel, children: "Stroke" }), jsx("div", { className: styles.propColorSwatch, style: { background: strokeCol }, children: jsx(Rn, { value: strokeCol, onChange: (c) => { setStrokeCol(c); updateSel({ stroke: c }); } }) }), jsx(Tt, { className: styles.propInput, value: (selShape === null || selShape === void 0 ? void 0 : selShape.stroke) || strokeCol, onChange: (val) => { setStrokeCol(val); updateSel({ stroke: val }); } })] }), jsxs("div", { className: styles.propRow, children: [jsx("span", { className: styles.propLabel, children: "Width" }), jsx(ir, { className: styles.propInput, min: 0, max: 100, value: (_a = selShape === null || selShape === void 0 ? void 0 : selShape.strokeWidth) !== null && _a !== void 0 ? _a : strokeW, onChange: (val) => { const v = val !== null && val !== void 0 ? val : 0; setStrokeW(v); updateSel({ strokeWidth: v }); } })] }), jsxs("div", { className: styles.propRow, children: [jsx("span", { className: styles.propLabel, children: "Opacity" }), jsx(so, { className: styles.propInput, min: 0, max: 1, step: 0.01, value: (_b = selShape === null || selShape === void 0 ? void 0 : selShape.opacity) !== null && _b !== void 0 ? _b : shapeOpacity, onChange: (val) => { setShapeOpacity(val); updateSel({ opacity: val }); } }), jsxs("span", { style: { fontSize: 10, minWidth: 28 }, children: [Math.round(((_c = selShape === null || selShape === void 0 ? void 0 : selShape.opacity) !== null && _c !== void 0 ? _c : shapeOpacity) * 100), "%"] })] }), jsxs("div", { className: styles.propRow, children: [jsx("span", { className: styles.propLabel, children: "Rotation" }), jsx(ir, { className: styles.propInput, min: -360, max: 360, value: (_d = selShape === null || selShape === void 0 ? void 0 : selShape.rotation) !== null && _d !== void 0 ? _d : 0, onChange: (val) => updateSel({ rotation: val !== null && val !== void 0 ? val : 0 }) }), jsx("span", { style: { fontSize: 10 }, children: "\u00B0" })] })] }), jsxs("div", { className: styles.panelSection, children: [jsx("div", { className: styles.panelTitle, children: jsx("span", { children: "Tool Options" }) }), (tool === "roundrect" || (selShape === null || selShape === void 0 ? void 0 : selShape.type) === "roundrect") && (jsxs("div", { className: styles.propRow, children: [jsx("span", { className: styles.propLabel, children: "Radius" }), jsx(ir, { className: styles.propInput, min: 0, max: 200, value: (_e = selShape === null || selShape === void 0 ? void 0 : selShape.rx) !== null && _e !== void 0 ? _e : cornerRadius, onChange: (val) => { const v = val !== null && val !== void 0 ? val : 0; setCornerRadius(v); updateSel({ rx: v }); } })] })), (tool === "polygon" || (selShape === null || selShape === void 0 ? void 0 : selShape.type) === "polygon") && (jsxs("div", { className: styles.propRow, children: [jsx("span", { className: styles.propLabel, children: "Sides" }), jsx(ir, { className: styles.propInput, min: 3, max: 32, value: (_f = selShape === null || selShape === void 0 ? void 0 : selShape.sides) !== null && _f !== void 0 ? _f : polygonSides, onChange: (val) => { const v = val !== null && val !== void 0 ? val : 3; setPolygonSides(v); updateSel({ sides: v }); } })] })), (tool === "star" || (selShape === null || selShape === void 0 ? void 0 : selShape.type) === "star") && (jsxs(Fragment, { children: [jsxs("div", { className: styles.propRow, children: [jsx("span", { className: styles.propLabel, children: "Points" }), jsx(ir, { className: styles.propInput, min: 3, max: 32, value: (_g = selShape === null || selShape === void 0 ? void 0 : selShape.sides) !== null && _g !== void 0 ? _g : starPoints2, onChange: (val) => { const v = val !== null && val !== void 0 ? val : 3; setStarPoints2(v); updateSel({ sides: v }); } })] }), jsxs("div", { className: styles.propRow, children: [jsx("span", { className: styles.propLabel, children: "Inner R" }), jsx(so, { className: styles.propInput, min: 0.1, max: 0.9, step: 0.05, value: (_h = selShape === null || selShape === void 0 ? void 0 : selShape.innerRadius) !== null && _h !== void 0 ? _h : starInnerRatio, onChange: (val) => { setStarInnerRatio(val); updateSel({ innerRadius: val }); } }), jsxs("span", { style: { fontSize: 10 }, children: [Math.round(((_j = selShape === null || selShape === void 0 ? void 0 : selShape.innerRadius) !== null && _j !== void 0 ? _j : starInnerRatio) * 100), "%"] })] })] })), (tool === "text" || (selShape === null || selShape === void 0 ? void 0 : selShape.type) === "text") && (jsxs(Fragment, { children: [jsxs("div", { className: styles.propRow, children: [jsx("span", { className: styles.propLabel, children: "Size" }), jsx(ir, { className: styles.propInput, min: 8, max: 200, value: (_k = selShape === null || selShape === void 0 ? void 0 : selShape.fontSize) !== null && _k !== void 0 ? _k : fontSize, onChange: (val) => { const v = val !== null && val !== void 0 ? val : 12; setFontSize(v); updateSel({ fontSize: v }); } })] }), jsxs("div", { className: styles.propRow, children: [jsx("span", { className: styles.propLabel, children: "Font" }), jsx(fn, { className: styles.propInput, value: (_l = selShape === null || selShape === void 0 ? void 0 : selShape.fontFamily) !== null && _l !== void 0 ? _l : fontFamily, onChange: (val) => { setFontFamily(val); updateSel({ fontFamily: val }); }, options: FONTS.map((f) => ({ value: f, label: f })) })] })] })), snapGrid && (jsxs("div", { className: styles.propRow, children: [jsx("span", { className: styles.propLabel, children: "Grid" }), jsx(ir, { className: styles.propInput, min: 4, max: 128, value: gridSize, onChange: (val) => setGridSize(val !== null && val !== void 0 ? val : 8) })] }))] }), jsxs("div", { className: styles.panelSection, children: [jsx("div", { className: styles.panelTitle, children: jsxs("span", { children: ["Shapes (", shapes.length, ")"] }) }), [...shapes].reverse().map((s) => (jsxs("div", { className: `${styles.shapeItem} ${s.id === selId ? styles.shapeItemActive : ""}`, onClick: () => { setSelId(s.id); setTool("pointer"); }, children: [jsx("span", { className: styles.shapeIcon, children: s.type === "rect" ? "▢"
|
|
116
116
|
: s.type === "roundrect" ? "▣"
|
|
117
117
|
: s.type === "ellipse" ? "◯"
|
|
118
118
|
: s.type === "line" ? "╱"
|
|
@@ -120,7 +120,7 @@ const VectorEditorRightPanel$1 = ({ api }) => {
|
|
|
120
120
|
: s.type === "polygon" ? "⬡"
|
|
121
121
|
: s.type === "star" ? "★"
|
|
122
122
|
: s.type === "polyline" ? "✎"
|
|
123
|
-
: "T" }), jsxs("span", { className: styles.shapeName, children: [s.type, s.text ? ` "${s.text}"` : ""] })] }, s.id))), shapes.length === 0 && (jsx("div", { style: { color: "#555", fontSize: 11, padding: 4 }, children: "No shapes yet. Use the tools to draw." }))] }), selShape && (jsxs("div", { className: styles.panelSection, children: [jsx("div", { className: styles.panelTitle, children: jsx("span", { children: "Transform" }) }), jsxs("div", { className: styles.propRow, children: [jsx("span", { className: styles.propLabel, children: "X" }), jsx(
|
|
123
|
+
: "T" }), jsxs("span", { className: styles.shapeName, children: [s.type, s.text ? ` "${s.text}"` : ""] })] }, s.id))), shapes.length === 0 && (jsx("div", { style: { color: "#555", fontSize: 11, padding: 4 }, children: "No shapes yet. Use the tools to draw." }))] }), selShape && (jsxs("div", { className: styles.panelSection, children: [jsx("div", { className: styles.panelTitle, children: jsx("span", { children: "Transform" }) }), jsxs("div", { className: styles.propRow, children: [jsx("span", { className: styles.propLabel, children: "X" }), jsx(ir, { className: styles.propInput, value: Math.round(selShape.x), onChange: (val) => updateSel({ x: val !== null && val !== void 0 ? val : 0 }) })] }), jsxs("div", { className: styles.propRow, children: [jsx("span", { className: styles.propLabel, children: "Y" }), jsx(ir, { className: styles.propInput, value: Math.round(selShape.y), onChange: (val) => updateSel({ y: val !== null && val !== void 0 ? val : 0 }) })] }), jsxs("div", { className: styles.propRow, children: [jsx("span", { className: styles.propLabel, children: "W" }), jsx(ir, { className: styles.propInput, value: Math.round(selShape.w), onChange: (val) => updateSel({ w: val !== null && val !== void 0 ? val : 0 }) })] }), jsxs("div", { className: styles.propRow, children: [jsx("span", { className: styles.propLabel, children: "H" }), jsx(ir, { className: styles.propInput, value: Math.round(selShape.h), onChange: (val) => updateSel({ h: val !== null && val !== void 0 ? val : 0 }) })] })] }))] }));
|
|
124
124
|
};
|
|
125
125
|
var VectorEditorRightPanel = React.memo(VectorEditorRightPanel$1);
|
|
126
126
|
|
package/package.json
CHANGED
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
{
|
|
2
2
|
"name": "@nice2dev/ui-graphics",
|
|
3
|
-
"version": "1.0.
|
|
3
|
+
"version": "1.0.5",
|
|
4
4
|
"description": "Nice2Dev Graphics Editors — Pixel, Vector, Photo and Animation editors for React",
|
|
5
5
|
"author": "NiceToDev <contact@nicetodev.com>",
|
|
6
6
|
"license": "SEE LICENSE IN LICENSE",
|