refactor: deckStore

This commit is contained in:
hypercross 2026-02-27 14:19:26 +08:00
parent 14ce2e1a6b
commit 72285e093f
6 changed files with 294 additions and 301 deletions

View File

@ -1,56 +1,48 @@
import { Show, For } from 'solid-js'; import { Show, For } from 'solid-js';
import { marked } from '../markdown'; import { marked } from '../markdown';
import type { CardData, LayerConfig, Dimensions } from './types';
import { getLayerStyle } from './utils/dimensions'; import { getLayerStyle } from './utils/dimensions';
import { getSelectionBoxStyle } from './hooks/use-selection'; import {getSelectionBoxStyle, useSelection} from './hooks/use-selection';
import {DeckStore} from "./stores/deckStore";
export interface CardPreviewProps { export interface CardPreviewProps {
cards: CardData[]; store: DeckStore;
activeTab: number;
layerConfigs: LayerConfig[];
dimensions: Dimensions;
isEditing: boolean;
isFixed: boolean;
editingLayer: string | null;
isSelecting: boolean;
selectStart: { x: number; y: number } | null;
selectEnd: { x: number; y: number } | null;
onMouseDown: (e: MouseEvent) => void;
onMouseMove: (e: MouseEvent) => void;
onMouseUp: () => void;
onMouseLeave: () => void;
} }
/** /**
* layer * layer
*/ */
function renderLayer(layer: { prop: string }, cardData: CardData): string { function renderLayer(layer: { prop: string }, cardData: DeckStore['cards'][number]): string {
const content = cardData[layer.prop] || ''; const content = cardData[layer.prop] || '';
return marked.parse(content) as string; return marked.parse(content) as string;
} }
export function CardPreview(props: CardPreviewProps) { export function CardPreview(props: CardPreviewProps) {
const currentCard = () => props.cards[props.activeTab]; const currentCard = () => props.store.cards[props.store.activeTab];
const visibleLayers = () => props.layerConfigs.filter(l => l.visible); const visibleLayers = () => props.store.layerConfigs.filter((l) => l.visible);
const selectionStyle = () => const selectionStyle = () =>
getSelectionBoxStyle(props.selectStart, props.selectEnd, props.dimensions); getSelectionBoxStyle(props.store.selectStart, props.store.selectEnd, props.store.dimensions);
const selection = useSelection(props.store);
let cardRef: HTMLDivElement | undefined;
return ( return (
<div class="flex justify-center"> <div class="flex justify-center">
<Show when={props.activeTab < props.cards.length}> <Show when={props.store.activeTab < props.store.cards.length}>
<div <div
ref={cardRef}
class="relative bg-white border border-gray-300 shadow-lg" class="relative bg-white border border-gray-300 shadow-lg"
style={{ style={{
width: `${props.dimensions.cardWidth}mm`, width: `${props.store.dimensions?.cardWidth}mm`,
height: `${props.dimensions.cardHeight}mm` height: `${props.store.dimensions?.cardHeight}mm`
}} }}
onMouseDown={props.onMouseDown} onMouseDown={(e) => selection.onMouseDown(e, cardRef!)}
onMouseMove={props.onMouseMove} onMouseMove={(e) => selection.onMouseMove(e, cardRef!)}
onMouseUp={props.onMouseUp} onMouseUp={selection.onMouseUp}
onMouseLeave={props.onMouseLeave} onMouseLeave={selection.onMouseLeave}
> >
{/* 框选遮罩 */} {/* 框选遮罩 */}
<Show when={props.isSelecting && selectionStyle()}> <Show when={props.store.isSelecting && selectionStyle()}>
<div <div
class="absolute bg-blue-500/30 border-2 border-blue-500 pointer-events-none" class="absolute bg-blue-500/30 border-2 border-blue-500 pointer-events-none"
style={selectionStyle()!} style={selectionStyle()!}
@ -61,28 +53,28 @@ export function CardPreview(props: CardPreviewProps) {
<div <div
class="absolute" class="absolute"
style={{ style={{
left: `${props.dimensions.gridOriginX}mm`, left: `${props.store.dimensions?.gridOriginX}mm`,
top: `${props.dimensions.gridOriginY}mm`, top: `${props.store.dimensions?.gridOriginY}mm`,
width: `${props.dimensions.gridAreaWidth}mm`, width: `${props.store.dimensions?.gridAreaWidth}mm`,
height: `${props.dimensions.gridAreaHeight}mm` height: `${props.store.dimensions?.gridAreaHeight}mm`
}} }}
> >
{/* 编辑模式下的网格线 */} {/* 编辑模式下的网格线 */}
<Show when={props.isEditing && !props.isFixed}> <Show when={props.store.isEditing && !props.store.fixed}>
<div class="absolute inset-0 pointer-events-none"> <div class="absolute inset-0 pointer-events-none">
<For each={Array.from({ length: props.dimensions.gridW - 1 })}> <For each={Array.from({ length: (props.store.dimensions?.gridW || 0) - 1 })}>
{(_, i) => ( {(_, i) => (
<div <div
class="absolute top-0 bottom-0 border-r border-dashed border-gray-300" class="absolute top-0 bottom-0 border-r border-dashed border-gray-300"
style={{ left: `${(i() + 1) * props.dimensions.cellWidth}mm` }} style={{ left: `${(i() + 1) * (props.store.dimensions?.cellWidth || 0)}mm` }}
/> />
)} )}
</For> </For>
<For each={Array.from({ length: props.dimensions.gridH - 1 })}> <For each={Array.from({ length: (props.store.dimensions?.gridH || 0) - 1 })}>
{(_, i) => ( {(_, i) => (
<div <div
class="absolute left-0 right-0 border-b border-dashed border-gray-300" class="absolute left-0 right-0 border-b border-dashed border-gray-300"
style={{ top: `${(i() + 1) * props.dimensions.cellHeight}mm` }} style={{ top: `${(i() + 1) * (props.store.dimensions?.cellHeight || 0)}mm` }}
/> />
)} )}
</For> </For>
@ -92,12 +84,12 @@ export function CardPreview(props: CardPreviewProps) {
{/* 渲染每个 layer */} {/* 渲染每个 layer */}
<For each={visibleLayers()}> <For each={visibleLayers()}>
{(layer) => { {(layer) => {
const style = getLayerStyle(layer, props.dimensions); const style = getLayerStyle(layer, props.store.dimensions!);
return ( return (
<div <div
class={`absolute flex items-center justify-center text-center prose prose-sm ${ class={`absolute flex items-center justify-center text-center prose prose-sm ${
props.isEditing ? 'bg-blue-500/20 ring-2 ring-blue-500' : '' props.store.isEditing ? 'bg-blue-500/20 ring-2 ring-blue-500' : ''
}`} }`}
style={style} style={style}
innerHTML={renderLayer(layer, currentCard())} innerHTML={renderLayer(layer, currentCard())}

View File

@ -1,26 +1,14 @@
import { Show, For } from 'solid-js'; import { For } from 'solid-js';
import type { CardData, LayerConfig } from './types'; import { DeckStore } from './stores/deckStore';
export interface DataEditorPanelProps { export interface DataEditorPanelProps {
cards: CardData[];
activeTab: number; activeTab: number;
updateCardData: (key: string, value: string) => void; cards: DeckStore['cards'];
updateCardData: DeckStore['updateCardData'];
} }
export interface PropertiesEditorPanelProps { export interface PropertiesEditorPanelProps {
localSize: string; store: DeckStore;
localGrid: string;
localBleed: string;
localPadding: string;
layerConfigs: LayerConfig[];
editingLayer: string | null;
onSizeChange: (value: string) => void;
onGridChange: (value: string) => void;
onBleedChange: (value: string) => void;
onPaddingChange: (value: string) => void;
onToggleLayerVisible: (prop: string) => void;
onStartEditingLayer: (prop: string) => void;
onCopyCode: () => void;
} }
/** /**
@ -39,7 +27,7 @@ export function DataEditorPanel(props: DataEditorPanelProps) {
class="w-full border border-gray-300 rounded px-2 py-1 text-sm" class="w-full border border-gray-300 rounded px-2 py-1 text-sm"
rows={3} rows={3}
value={props.cards[props.activeTab]?.[key] || ''} value={props.cards[props.activeTab]?.[key] || ''}
onInput={(e) => props.updateCardData(key, e.target.value)} onInput={(e) => props.updateCardData(props.activeTab, key, e.target.value)}
/> />
</div> </div>
)} )}
@ -53,6 +41,8 @@ export function DataEditorPanel(props: DataEditorPanelProps) {
* *
*/ */
export function PropertiesEditorPanel(props: PropertiesEditorPanelProps) { export function PropertiesEditorPanel(props: PropertiesEditorPanelProps) {
const { store } = props;
return ( return (
<div class="w-64 flex-shrink-0"> <div class="w-64 flex-shrink-0">
<h3 class="font-bold mb-2"></h3> <h3 class="font-bold mb-2"></h3>
@ -63,8 +53,8 @@ export function PropertiesEditorPanel(props: PropertiesEditorPanelProps) {
<input <input
type="text" type="text"
class="w-full border border-gray-300 rounded px-2 py-1 text-sm" class="w-full border border-gray-300 rounded px-2 py-1 text-sm"
value={props.localSize} value={store.size}
onInput={(e) => props.onSizeChange(e.target.value)} onInput={(e) => store.setSize(e.target.value)}
/> />
</div> </div>
@ -73,8 +63,8 @@ export function PropertiesEditorPanel(props: PropertiesEditorPanelProps) {
<input <input
type="text" type="text"
class="w-full border border-gray-300 rounded px-2 py-1 text-sm" class="w-full border border-gray-300 rounded px-2 py-1 text-sm"
value={props.localGrid} value={store.grid}
onInput={(e) => props.onGridChange(e.target.value)} onInput={(e) => store.setGrid(e.target.value)}
/> />
</div> </div>
@ -83,8 +73,8 @@ export function PropertiesEditorPanel(props: PropertiesEditorPanelProps) {
<input <input
type="text" type="text"
class="w-full border border-gray-300 rounded px-2 py-1 text-sm" class="w-full border border-gray-300 rounded px-2 py-1 text-sm"
value={props.localBleed} value={store.bleed}
onInput={(e) => props.onBleedChange(e.target.value)} onInput={(e) => store.setBleed(e.target.value)}
/> />
</div> </div>
@ -93,33 +83,33 @@ export function PropertiesEditorPanel(props: PropertiesEditorPanelProps) {
<input <input
type="text" type="text"
class="w-full border border-gray-300 rounded px-2 py-1 text-sm" class="w-full border border-gray-300 rounded px-2 py-1 text-sm"
value={props.localPadding} value={store.padding}
onInput={(e) => props.onPaddingChange(e.target.value)} onInput={(e) => store.setPadding(e.target.value)}
/> />
</div> </div>
<hr class="my-4" /> <hr class="my-4" />
<h4 class="font-medium text-sm text-gray-700"></h4> <h4 class="font-medium text-sm text-gray-700"></h4>
<For each={props.layerConfigs}> <For each={store.layerConfigs}>
{(layer) => ( {(layer) => (
<div class="flex items-center gap-2"> <div class="flex items-center gap-2">
<input <input
type="checkbox" type="checkbox"
checked={layer.visible} checked={layer.visible}
onChange={() => props.onToggleLayerVisible(layer.prop)} onChange={() => store.toggleLayerVisible(layer.prop)}
class="cursor-pointer" class="cursor-pointer"
/> />
<span class="text-sm flex-1">{layer.prop}</span> <span class="text-sm flex-1">{layer.prop}</span>
<button <button
onClick={() => props.onStartEditingLayer(layer.prop)} onClick={() => store.setEditingLayer(layer.prop)}
class={`text-xs px-2 py-0.5 rounded cursor-pointer ${ class={`text-xs px-2 py-0.5 rounded cursor-pointer ${
props.editingLayer === layer.prop store.editingLayer === layer.prop
? 'bg-blue-500 text-white' ? 'bg-blue-500 text-white'
: 'bg-gray-200 text-gray-700 hover:bg-gray-300' : 'bg-gray-200 text-gray-700 hover:bg-gray-300'
}`} }`}
> >
{props.editingLayer === layer.prop ? '✓ 框选' : '编辑位置'} {store.editingLayer === layer.prop ? '✓ 框选' : '编辑位置'}
</button> </button>
</div> </div>
)} )}
@ -128,7 +118,7 @@ export function PropertiesEditorPanel(props: PropertiesEditorPanelProps) {
<hr class="my-4" /> <hr class="my-4" />
<button <button
onClick={props.onCopyCode} onClick={store.copyCode}
class="w-full bg-blue-600 hover:bg-blue-700 text-white px-3 py-2 rounded text-sm font-medium cursor-pointer" class="w-full bg-blue-600 hover:bg-blue-700 text-white px-3 py-2 rounded text-sm font-medium cursor-pointer"
> >
📋 📋

View File

@ -1,70 +0,0 @@
import { createSignal } from 'solid-js';
import type { LayerConfig } from '../types';
import { formatLayers } from '../utils/layer-parser';
/**
*
*/
export function useLayerEditor(
props: any,
src: string,
localSize: () => string,
localGrid: () => string,
localBleed: () => string,
localPadding: () => string
) {
const [isEditing, setIsEditing] = createSignal(false);
const [editingLayer, setEditingLayer] = createSignal<string | null>(null);
const [layerConfigs, setLayerConfigs] = createSignal<LayerConfig[]>([]);
const toggleLayerVisible = (prop: string) => {
setLayerConfigs(configs => configs.map(c =>
c.prop === prop ? { ...c, visible: !c.visible } : c
));
};
const startEditingLayer = (prop: string) => {
setEditingLayer(prop);
};
const updateLayerPosition = (x1: number, y1: number, x2: number, y2: number) => {
const layer = editingLayer();
if (!layer) return;
setLayerConfigs(configs => configs.map(c =>
c.prop === layer ? { ...c, x1, y1, x2, y2 } : c
));
setEditingLayer(null);
};
const generateCode = () => {
const layersStr = formatLayers(layerConfigs());
return `:md-deck[${src}]{size="${localSize()}" grid="${localGrid()}" bleed="${localBleed()}" padding="${localPadding()}" layers="${layersStr}"}`;
};
const copyCode = () => {
const code = generateCode();
navigator.clipboard.writeText(code).then(() => {
alert('已复制到剪贴板!');
}).catch(err => {
console.error('复制失败:', err);
});
};
const isFixed = () => props.fixed === true || props.fixed === 'true';
return {
isEditing,
setIsEditing,
editingLayer,
setEditingLayer,
layerConfigs,
setLayerConfigs,
toggleLayerVisible,
startEditingLayer,
updateLayerPosition,
generateCode,
copyCode,
isFixed
};
}

View File

@ -1,82 +1,65 @@
import { createSignal } from 'solid-js'; import type { DeckStore } from '../stores/deckStore';
import type { Dimensions, LayerConfig } from '../types';
/** /**
* * deckStore
* hook
*/ */
export function useSelection( export function useSelection(store: DeckStore) {
isEditing: () => boolean, const calculateGridCoords = (e: MouseEvent, cardEl: HTMLElement, dimensions: DeckStore['dimensions']) => {
editingLayer: () => string | null, if (!dimensions) return { gridX: 1, gridY: 1 };
dimensions: () => Dimensions
) {
const [isSelecting, setIsSelecting] = createSignal(false);
const [selectStart, setSelectStart] = createSignal<{ x: number; y: number } | null>(null);
const [selectEnd, setSelectEnd] = createSignal<{ x: number; y: number } | null>(null);
const calculateGridCoords = (e: MouseEvent, cardEl: HTMLElement) => {
const rect = cardEl.getBoundingClientRect(); const rect = cardEl.getBoundingClientRect();
const dims = dimensions();
const offsetX = (e.clientX - rect.left) / rect.width * dims.cardWidth; const offsetX = (e.clientX - rect.left) / rect.width * dimensions.cardWidth;
const offsetY = (e.clientY - rect.top) / rect.height * dims.cardHeight; const offsetY = (e.clientY - rect.top) / rect.height * dimensions.cardHeight;
const gridX = Math.max(1, Math.floor((offsetX - dims.gridOriginX) / dims.cellWidth) + 1); const gridX = Math.max(1, Math.floor((offsetX - dimensions.gridOriginX) / dimensions.cellWidth) + 1);
const gridY = Math.max(1, Math.floor((offsetY - dims.gridOriginY) / dims.cellHeight) + 1); const gridY = Math.max(1, Math.floor((offsetY - dimensions.gridOriginY) / dimensions.cellHeight) + 1);
return { return {
gridX: Math.max(1, Math.min(dims.gridW, gridX)), gridX: Math.max(1, Math.min(dimensions.gridW, gridX)),
gridY: Math.max(1, Math.min(dims.gridH, gridY)) gridY: Math.max(1, Math.min(dimensions.gridH, gridY))
}; };
}; };
const handleMouseDown = (e: MouseEvent) => { const handleMouseDown = (e: MouseEvent, cardEl: HTMLElement) => {
if (!isEditing() || !editingLayer()) return; if (!store.isEditing || !store.editingLayer) return;
const cardEl = e.currentTarget as HTMLElement; const { gridX, gridY } = calculateGridCoords(e, cardEl, store.dimensions);
const { gridX, gridY } = calculateGridCoords(e, cardEl);
setSelectStart({ x: gridX, y: gridY }); store.setSelectStart({ x: gridX, y: gridY });
setSelectEnd({ x: gridX, y: gridY }); store.setSelectEnd({ x: gridX, y: gridY });
setIsSelecting(true); store.setIsSelecting(true);
}; };
const handleMouseMove = (e: MouseEvent) => { const handleMouseMove = (e: MouseEvent, cardEl: HTMLElement) => {
if (!isSelecting()) return; if (!store.isSelecting) return;
const cardEl = e.currentTarget as HTMLElement; const { gridX, gridY } = calculateGridCoords(e, cardEl, store.dimensions);
const { gridX, gridY } = calculateGridCoords(e, cardEl);
setSelectEnd({ x: gridX, y: gridY }); store.setSelectEnd({ x: gridX, y: gridY });
}; };
const handleMouseUp = () => { const handleMouseUp = () => {
if (!isSelecting() || !editingLayer()) return; if (!store.isSelecting || !store.editingLayer) return;
const start = selectStart()!; const start = store.selectStart!;
const end = selectEnd()!; const end = store.selectEnd!;
const x1 = Math.min(start.x, end.x); const x1 = Math.min(start.x, end.x);
const y1 = Math.min(start.y, end.y); const y1 = Math.min(start.y, end.y);
const x2 = Math.max(start.x, end.x); const x2 = Math.max(start.x, end.x);
const y2 = Math.max(start.y, end.y); const y2 = Math.max(start.y, end.y);
return { x1, y1, x2, y2 }; store.updateLayerPosition(x1, y1, x2, y2);
}; store.cancelSelection();
const cancelSelection = () => {
setIsSelecting(false);
setSelectStart(null);
setSelectEnd(null);
}; };
return { return {
isSelecting, onMouseDown: handleMouseDown,
selectStart, onMouseMove: handleMouseMove,
selectEnd, onMouseUp: handleMouseUp,
handleMouseDown, onMouseLeave: handleMouseUp
handleMouseMove,
handleMouseUp,
cancelSelection
}; };
} }
@ -86,9 +69,9 @@ export function useSelection(
export function getSelectionBoxStyle( export function getSelectionBoxStyle(
selectStart: { x: number; y: number } | null, selectStart: { x: number; y: number } | null,
selectEnd: { x: number; y: number } | null, selectEnd: { x: number; y: number } | null,
dims: Dimensions dims: { gridOriginX: number; gridOriginY: number; cellWidth: number; cellHeight: number } | null
): { left: string; top: string; width: string; height: string } | null { ): { left: string; top: string; width: string; height: string } | null {
if (!selectStart || !selectEnd) return null; if (!selectStart || !selectEnd || !dims) return null;
const x1 = Math.min(selectStart.x, selectEnd.x); const x1 = Math.min(selectStart.x, selectEnd.x);
const y1 = Math.min(selectStart.y, selectEnd.y); const y1 = Math.min(selectStart.y, selectEnd.y);

View File

@ -1,12 +1,9 @@
import { customElement, noShadowDOM } from 'solid-element'; import { customElement, noShadowDOM } from 'solid-element';
import { createSignal, For, Show, createEffect, createMemo, createResource } from 'solid-js'; import { Show, createEffect, createResource, For } from 'solid-js';
import { resolvePath } from './utils/path'; import { resolvePath } from './utils/path';
import type { CardData, Dimensions } from './types';
import { loadCSV } from './utils/csv-loader'; import { loadCSV } from './utils/csv-loader';
import { initLayerConfigs } from './utils/layer-parser'; import { initLayerConfigs } from './utils/layer-parser';
import { calculateDimensions } from './utils/dimensions'; import { createDeckStore } from './stores/deckStore';
import { useSelection } from './hooks/use-selection';
import { useLayerEditor } from './hooks/use-layer-editor';
import { CardPreview } from './card-preview'; import { CardPreview } from './card-preview';
import { DataEditorPanel, PropertiesEditorPanel } from './editor-panel'; import { DataEditorPanel, PropertiesEditorPanel } from './editor-panel';
@ -20,25 +17,8 @@ customElement('md-deck', {
}, (props, { element }) => { }, (props, { element }) => {
noShadowDOM(); noShadowDOM();
const [cards, setCards] = createSignal<CardData[]>([]); // 创建统一的 store
const [activeTab, setActiveTab] = createSignal(0); const store = createDeckStore();
let tabsContainer: HTMLDivElement | undefined;
// 本地编辑的属性
const [localSize, setLocalSize] = createSignal(props.size as string || '54x86');
const [localGrid, setLocalGrid] = createSignal(props.grid as string || '5x8');
const [localBleed, setLocalBleed] = createSignal(props.bleed as string || '1');
const [localPadding, setLocalPadding] = createSignal(props.padding as string || '2');
// 使用图层编辑器 hook
const layerEditor = useLayerEditor(
props,
'',
localSize,
localGrid,
localBleed,
localPadding
);
// 从 element 的 textContent 获取 CSV 路径 // 从 element 的 textContent 获取 CSV 路径
const src = element?.textContent?.trim() || ''; const src = element?.textContent?.trim() || '';
@ -55,63 +35,28 @@ customElement('md-deck', {
// 解析相对路径 // 解析相对路径
const resolvedSrc = resolvePath(articlePath, src); const resolvedSrc = resolvePath(articlePath, src);
// 初始化 store
store.initialize(props, src);
// 加载 CSV 文件 // 加载 CSV 文件
const [csvData] = createResource(() => resolvedSrc, loadCSV); const [csvData] = createResource(() => resolvedSrc, loadCSV);
createEffect(() => { createEffect(() => {
const data = csvData(); const data = !csvData.loading && csvData();
if (data) { if (data) {
setCards(data); store.setCards(data);
layerEditor.setLayerConfigs(initLayerConfigs(data, props.layers as string || '')); store.setLayerConfigs(initLayerConfigs(data, props.layers as string || ''));
} }
}); });
// 更新 src 到 layerEditor
createEffect(() => {
(layerEditor as any).src = src;
});
// 解析尺寸
const dimensions = createMemo((): Dimensions => {
return calculateDimensions({
size: localSize(),
bleed: localBleed(),
padding: localPadding(),
grid: localGrid()
});
});
// 使用框选 hook
const selection = useSelection(
layerEditor.isEditing,
layerEditor.editingLayer,
dimensions
);
// 处理框选完成
const handleMouseUp = () => {
const result = selection.handleMouseUp();
if (result) {
layerEditor.updateLayerPosition(result.x1, result.y1, result.x2, result.y2);
selection.cancelSelection();
}
};
// 更新 CSV 数据
const updateCardData = (key: string, value: string) => {
setCards(cards => cards.map((card, i) =>
i === activeTab() ? { ...card, [key]: value } : card
));
};
return ( return (
<div class="md-deck flex gap-4"> <div class="md-deck flex gap-4">
{/* 左侧CSV 数据编辑 */} {/* 左侧CSV 数据编辑 */}
<Show when={layerEditor.isEditing() && !layerEditor.isFixed()}> <Show when={store.isEditing && !store.fixed}>
<DataEditorPanel <DataEditorPanel
cards={cards()} activeTab={store.activeTab}
activeTab={activeTab()} cards={store.cards}
updateCardData={updateCardData} updateCardData={store.updateCardData}
/> />
</Show> </Show>
@ -120,22 +65,22 @@ customElement('md-deck', {
{/* Tab 选择器 */} {/* Tab 选择器 */}
<div class="flex items-center gap-2 border-b border-gray-200 pb-2 mb-4"> <div class="flex items-center gap-2 border-b border-gray-200 pb-2 mb-4">
<button <button
onClick={() => layerEditor.setIsEditing(!layerEditor.isEditing())} onClick={() => store.setIsEditing(!store.isEditing)}
class={`px-3 py-1 rounded text-sm font-medium transition-colors ${ class={`px-3 py-1 rounded text-sm font-medium transition-colors ${
layerEditor.isEditing() && !layerEditor.isFixed() store.isEditing && !store.fixed
? 'bg-blue-100 text-blue-600' ? 'bg-blue-100 text-blue-600'
: 'bg-gray-100 text-gray-600 hover:bg-gray-200' : 'bg-gray-100 text-gray-600 hover:bg-gray-200'
} cursor-pointer`} } cursor-pointer`}
> >
{layerEditor.isEditing() ? '✓ 编辑中' : '✏️ 编辑'} {store.isEditing ? '✓ 编辑中' : '✏️ 编辑'}
</button> </button>
<div ref={tabsContainer} class="flex gap-1 overflow-x-auto flex-1 min-w-0 flex-wrap"> <div class="flex gap-1 overflow-x-auto flex-1 min-w-0 flex-wrap">
<For each={cards()}> <For each={store.cards}>
{(card, index) => ( {(card, index) => (
<button <button
onClick={() => setActiveTab(index())} onClick={() => store.setActiveTab(index())}
class={`font-medium transition-colors flex-shrink-0 min-w-[1.6em] cursor-pointer px-2 py-1 rounded ${ class={`font-medium transition-colors flex-shrink-0 min-w-[1.6em] cursor-pointer px-2 py-1 rounded ${
activeTab() === index() store.activeTab === index()
? 'bg-blue-100 text-blue-600 border-b-2 border-blue-600' ? 'bg-blue-100 text-blue-600 border-b-2 border-blue-600'
: 'text-gray-500 hover:text-gray-700 hover:bg-gray-100' : 'text-gray-500 hover:text-gray-700 hover:bg-gray-100'
}`} }`}
@ -148,43 +93,14 @@ customElement('md-deck', {
</div> </div>
{/* 卡牌预览 */} {/* 卡牌预览 */}
<Show when={!csvData.loading && cards().length > 0}> <Show when={!csvData.loading && store.cards.length > 0}>
<CardPreview <CardPreview store={store}/>
cards={cards()}
activeTab={activeTab()}
layerConfigs={layerEditor.layerConfigs()}
dimensions={dimensions()}
isEditing={layerEditor.isEditing()}
isFixed={layerEditor.isFixed()}
editingLayer={layerEditor.editingLayer()}
isSelecting={selection.isSelecting()}
selectStart={selection.selectStart()}
selectEnd={selection.selectEnd()}
onMouseDown={selection.handleMouseDown}
onMouseMove={selection.handleMouseMove}
onMouseUp={handleMouseUp}
onMouseLeave={handleMouseUp}
/>
</Show> </Show>
</div> </div>
{/* 右侧:属性编辑表单 */} {/* 右侧:属性编辑表单 */}
<Show when={layerEditor.isEditing() && !layerEditor.isFixed()}> <Show when={store.isEditing && !store.fixed}>
<PropertiesEditorPanel <PropertiesEditorPanel store={store} />
localSize={localSize()}
localGrid={localGrid()}
localBleed={localBleed()}
localPadding={localPadding()}
layerConfigs={layerEditor.layerConfigs()}
editingLayer={layerEditor.editingLayer()}
onSizeChange={setLocalSize}
onGridChange={setLocalGrid}
onBleedChange={setLocalBleed}
onPaddingChange={setLocalPadding}
onToggleLayerVisible={layerEditor.toggleLayerVisible}
onStartEditingLayer={layerEditor.startEditingLayer}
onCopyCode={layerEditor.copyCode}
/>
</Show> </Show>
</div> </div>
); );

View File

@ -0,0 +1,182 @@
import { createStore, produce } from 'solid-js/store';
import type { CardData, LayerConfig, Dimensions } from '../types';
export interface DeckState {
// 基本属性
size: string;
grid: string;
bleed: string;
padding: string;
fixed: boolean;
src: string;
// 解析后的尺寸
dimensions: Dimensions | null;
// 卡牌数据
cards: CardData[];
activeTab: number;
// 图层配置
layerConfigs: LayerConfig[];
// 编辑状态
isEditing: boolean;
editingLayer: string | null;
// 框选状态
isSelecting: boolean;
selectStart: { x: number; y: number } | null;
selectEnd: { x: number; y: number } | null;
}
export interface DeckActions {
// 基本属性设置
setSize: (size: string) => void;
setGrid: (grid: string) => void;
setBleed: (bleed: string) => void;
setPadding: (padding: string) => void;
// 数据设置
setCards: (cards: CardData[]) => void;
setActiveTab: (index: number) => void;
updateCardData: (index: number, key: string, value: string) => void;
// 图层操作
setLayerConfigs: (configs: LayerConfig[]) => void;
updateLayerConfig: (prop: string, updates: Partial<LayerConfig>) => void;
toggleLayerVisible: (prop: string) => void;
// 编辑状态
setIsEditing: (editing: boolean) => void;
setEditingLayer: (layer: string | null) => void;
updateLayerPosition: (x1: number, y1: number, x2: number, y2: number) => void;
// 框选操作
setIsSelecting: (selecting: boolean) => void;
setSelectStart: (pos: { x: number; y: number } | null) => void;
setSelectEnd: (pos: { x: number; y: number } | null) => void;
cancelSelection: () => void;
// 初始化
initialize: (props: Record<string, any>, csvPath: string) => void;
// 生成代码
generateCode: () => string;
copyCode: () => void;
}
export interface DeckStore extends DeckState, DeckActions {}
/**
* deck store
*/
export function createDeckStore(): DeckStore {
const [state, setState] = createStore<DeckState>({
size: '54x86',
grid: '5x8',
bleed: '1',
padding: '2',
fixed: false,
src: '',
dimensions: null,
cards: [],
activeTab: 0,
layerConfigs: [],
isEditing: false,
editingLayer: null,
isSelecting: false,
selectStart: null,
selectEnd: null
});
const setSize = (size: string) => setState({ size });
const setGrid = (grid: string) => setState({ grid });
const setBleed = (bleed: string) => setState({ bleed });
const setPadding = (padding: string) => setState({ padding });
const setCards = (cards: CardData[]) => setState({ cards });
const setActiveTab = (index: number) => setState({ activeTab: index });
const updateCardData = (index: number, key: string, value: string) => {
setState('cards', index, key, value);
};
const setLayerConfigs = (configs: LayerConfig[]) => setState({ layerConfigs: configs });
const updateLayerConfig = (prop: string, updates: Partial<LayerConfig>) => {
setState('layerConfigs', (prev) => prev.map((config) => config.prop === prop ? { ...config, ...updates } : config));
};
const toggleLayerVisible = (prop: string) => {
setState('layerConfigs', (prev) => prev.map((config) =>
config.prop === prop ? { ...config, visible: !config.visible } : config
));
};
const setIsEditing = (editing: boolean) => setState({ isEditing: editing });
const setEditingLayer = (layer: string | null) => setState({ editingLayer: layer });
const updateLayerPosition = (x1: number, y1: number, x2: number, y2: number) => {
const layer = state.editingLayer;
if (!layer) return;
setState('layerConfigs', (prev) => prev.map((config) =>
config.prop === layer ? { ...config, x1, y1, x2, y2 } : config
));
setState({ editingLayer: null });
};
const setIsSelecting = (selecting: boolean) => setState({ isSelecting: selecting });
const setSelectStart = (pos: { x: number; y: number } | null) => setState({ selectStart: pos });
const setSelectEnd = (pos: { x: number; y: number } | null) => setState({ selectEnd: pos });
const cancelSelection = () => {
setState({ isSelecting: false, selectStart: null, selectEnd: null });
};
const initialize = (props: Record<string, any>, csvPath: string) => {
setState({
size: props.size as string || '54x86',
grid: props.grid as string || '5x8',
bleed: props.bleed as string || '1',
padding: props.padding as string || '2',
fixed: props.fixed === true || props.fixed === 'true',
src: csvPath
});
};
const generateCode = () => {
const layersStr = state.layerConfigs
.map(l => `${l.prop}=${l.x1},${l.y1},${l.x2},${l.y2}`)
.join('|');
return `:md-deck[${state.src}]{size="${state.size}" grid="${state.grid}" bleed="${state.bleed}" padding="${state.padding}" layers="${layersStr}"}`;
};
const copyCode = () => {
const code = generateCode();
navigator.clipboard.writeText(code).then(() => {
alert('已复制到剪贴板!');
}).catch(err => {
console.error('复制失败:', err);
});
};
return {
...state,
setSize,
setGrid,
setBleed,
setPadding,
setCards,
setActiveTab,
updateCardData,
setLayerConfigs,
updateLayerConfig,
toggleLayerVisible,
setIsEditing,
setEditingLayer,
updateLayerPosition,
setIsSelecting,
setSelectStart,
setSelectEnd,
cancelSelection,
initialize,
generateCode,
copyCode
};
}