2026-02-27 14:32:43 +08:00
|
|
|
|
import { createStore } from 'solid-js/store';
|
2026-02-27 15:21:21 +08:00
|
|
|
|
import { calculateDimensions } from './dimensions';
|
|
|
|
|
|
import { loadCSV } from '../../utils/csv-loader';
|
|
|
|
|
|
import { initLayerConfigs } from './layer-parser';
|
2026-02-27 14:19:26 +08:00
|
|
|
|
import type { CardData, LayerConfig, Dimensions } from '../types';
|
|
|
|
|
|
|
2026-02-27 15:21:21 +08:00
|
|
|
|
/**
|
|
|
|
|
|
* 默认配置常量
|
|
|
|
|
|
*/
|
|
|
|
|
|
export const DECK_DEFAULTS = {
|
2026-02-27 15:32:04 +08:00
|
|
|
|
SIZE_W: 54,
|
|
|
|
|
|
SIZE_H: 86,
|
|
|
|
|
|
GRID_W: 5,
|
|
|
|
|
|
GRID_H: 8,
|
2026-02-27 15:21:21 +08:00
|
|
|
|
BLEED: '1',
|
2026-02-27 15:32:04 +08:00
|
|
|
|
PADDING: '2',
|
|
|
|
|
|
FONT_SIZE: 3
|
2026-02-27 15:21:21 +08:00
|
|
|
|
} as const;
|
|
|
|
|
|
|
2026-02-27 14:19:26 +08:00
|
|
|
|
export interface DeckState {
|
|
|
|
|
|
// 基本属性
|
2026-02-27 15:32:04 +08:00
|
|
|
|
sizeW: number;
|
|
|
|
|
|
sizeH: number;
|
|
|
|
|
|
gridW: number;
|
|
|
|
|
|
gridH: number;
|
2026-02-27 14:19:26 +08:00
|
|
|
|
bleed: string;
|
|
|
|
|
|
padding: string;
|
2026-02-27 15:32:04 +08:00
|
|
|
|
fontSize: number;
|
2026-02-27 14:19:26 +08:00
|
|
|
|
fixed: boolean;
|
|
|
|
|
|
src: string;
|
2026-02-27 14:32:43 +08:00
|
|
|
|
|
2026-02-27 14:19:26 +08:00
|
|
|
|
// 解析后的尺寸
|
|
|
|
|
|
dimensions: Dimensions | null;
|
2026-02-27 14:32:43 +08:00
|
|
|
|
|
2026-02-27 14:19:26 +08:00
|
|
|
|
// 卡牌数据
|
|
|
|
|
|
cards: CardData[];
|
|
|
|
|
|
activeTab: number;
|
2026-02-27 14:32:43 +08:00
|
|
|
|
|
2026-02-27 14:19:26 +08:00
|
|
|
|
// 图层配置
|
|
|
|
|
|
layerConfigs: LayerConfig[];
|
2026-02-27 14:32:43 +08:00
|
|
|
|
|
2026-02-27 14:19:26 +08:00
|
|
|
|
// 编辑状态
|
|
|
|
|
|
isEditing: boolean;
|
|
|
|
|
|
editingLayer: string | null;
|
2026-02-27 14:32:43 +08:00
|
|
|
|
|
2026-02-27 14:19:26 +08:00
|
|
|
|
// 框选状态
|
|
|
|
|
|
isSelecting: boolean;
|
|
|
|
|
|
selectStart: { x: number; y: number } | null;
|
|
|
|
|
|
selectEnd: { x: number; y: number } | null;
|
2026-02-27 14:32:43 +08:00
|
|
|
|
|
2026-02-27 15:21:21 +08:00
|
|
|
|
// 加载状态
|
|
|
|
|
|
isLoading: boolean;
|
|
|
|
|
|
|
2026-02-27 14:32:43 +08:00
|
|
|
|
// 错误状态
|
|
|
|
|
|
error: string | null;
|
2026-02-27 14:19:26 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
export interface DeckActions {
|
|
|
|
|
|
// 基本属性设置
|
2026-02-27 15:32:04 +08:00
|
|
|
|
setSizeW: (size: number) => void;
|
|
|
|
|
|
setSizeH: (size: number) => void;
|
|
|
|
|
|
setGridW: (grid: number) => void;
|
|
|
|
|
|
setGridH: (grid: number) => void;
|
2026-02-27 14:19:26 +08:00
|
|
|
|
setBleed: (bleed: string) => void;
|
|
|
|
|
|
setPadding: (padding: string) => void;
|
2026-02-27 15:32:04 +08:00
|
|
|
|
setFontSize: (size: number) => void;
|
2026-02-27 14:32:43 +08:00
|
|
|
|
|
2026-02-27 14:19:26 +08:00
|
|
|
|
// 数据设置
|
|
|
|
|
|
setCards: (cards: CardData[]) => void;
|
|
|
|
|
|
setActiveTab: (index: number) => void;
|
|
|
|
|
|
updateCardData: (index: number, key: string, value: string) => void;
|
2026-02-27 14:32:43 +08:00
|
|
|
|
|
2026-02-27 14:19:26 +08:00
|
|
|
|
// 图层操作
|
|
|
|
|
|
setLayerConfigs: (configs: LayerConfig[]) => void;
|
|
|
|
|
|
updateLayerConfig: (prop: string, updates: Partial<LayerConfig>) => void;
|
|
|
|
|
|
toggleLayerVisible: (prop: string) => void;
|
2026-02-27 14:32:43 +08:00
|
|
|
|
|
2026-02-27 14:19:26 +08:00
|
|
|
|
// 编辑状态
|
|
|
|
|
|
setIsEditing: (editing: boolean) => void;
|
|
|
|
|
|
setEditingLayer: (layer: string | null) => void;
|
|
|
|
|
|
updateLayerPosition: (x1: number, y1: number, x2: number, y2: number) => void;
|
2026-02-27 14:32:43 +08:00
|
|
|
|
|
2026-02-27 14:19:26 +08:00
|
|
|
|
// 框选操作
|
|
|
|
|
|
setIsSelecting: (selecting: boolean) => void;
|
|
|
|
|
|
setSelectStart: (pos: { x: number; y: number } | null) => void;
|
|
|
|
|
|
setSelectEnd: (pos: { x: number; y: number } | null) => void;
|
|
|
|
|
|
cancelSelection: () => void;
|
2026-02-27 14:32:43 +08:00
|
|
|
|
|
2026-02-27 15:21:21 +08:00
|
|
|
|
// 数据加载
|
|
|
|
|
|
loadCardsFromPath: (path: string, layersStr?: string) => Promise<void>;
|
2026-02-27 14:32:43 +08:00
|
|
|
|
setError: (error: string | null) => void;
|
2026-02-27 15:21:21 +08:00
|
|
|
|
clearError: () => void;
|
2026-02-27 14:32:43 +08:00
|
|
|
|
|
2026-02-27 14:19:26 +08:00
|
|
|
|
// 生成代码
|
|
|
|
|
|
generateCode: () => string;
|
2026-02-27 15:21:21 +08:00
|
|
|
|
copyCode: () => Promise<void>;
|
2026-02-27 14:19:26 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
2026-02-27 14:58:44 +08:00
|
|
|
|
export interface DeckStore {
|
|
|
|
|
|
state: DeckState;
|
|
|
|
|
|
actions: DeckActions;
|
|
|
|
|
|
}
|
2026-02-27 14:19:26 +08:00
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
|
* 创建 deck store
|
|
|
|
|
|
*/
|
2026-02-27 15:21:21 +08:00
|
|
|
|
export function createDeckStore(
|
|
|
|
|
|
initialSrc: string = '',
|
|
|
|
|
|
initialLayers: string = ''
|
|
|
|
|
|
): DeckStore {
|
2026-02-27 14:19:26 +08:00
|
|
|
|
const [state, setState] = createStore<DeckState>({
|
2026-02-27 15:32:04 +08:00
|
|
|
|
sizeW: DECK_DEFAULTS.SIZE_W,
|
|
|
|
|
|
sizeH: DECK_DEFAULTS.SIZE_H,
|
|
|
|
|
|
gridW: DECK_DEFAULTS.GRID_W,
|
|
|
|
|
|
gridH: DECK_DEFAULTS.GRID_H,
|
2026-02-27 15:21:21 +08:00
|
|
|
|
bleed: DECK_DEFAULTS.BLEED,
|
|
|
|
|
|
padding: DECK_DEFAULTS.PADDING,
|
2026-02-27 15:32:04 +08:00
|
|
|
|
fontSize: DECK_DEFAULTS.FONT_SIZE,
|
2026-02-27 14:19:26 +08:00
|
|
|
|
fixed: false,
|
2026-02-27 15:21:21 +08:00
|
|
|
|
src: initialSrc,
|
2026-02-27 14:19:26 +08:00
|
|
|
|
dimensions: null,
|
|
|
|
|
|
cards: [],
|
|
|
|
|
|
activeTab: 0,
|
|
|
|
|
|
layerConfigs: [],
|
|
|
|
|
|
isEditing: false,
|
|
|
|
|
|
editingLayer: null,
|
|
|
|
|
|
isSelecting: false,
|
|
|
|
|
|
selectStart: null,
|
2026-02-27 14:32:43 +08:00
|
|
|
|
selectEnd: null,
|
2026-02-27 15:21:21 +08:00
|
|
|
|
isLoading: false,
|
2026-02-27 14:32:43 +08:00
|
|
|
|
error: null
|
2026-02-27 14:19:26 +08:00
|
|
|
|
});
|
|
|
|
|
|
|
2026-02-27 14:32:43 +08:00
|
|
|
|
// 更新尺寸并重新计算 dimensions
|
|
|
|
|
|
const updateDimensions = () => {
|
|
|
|
|
|
const dims = calculateDimensions({
|
2026-02-27 15:32:04 +08:00
|
|
|
|
sizeW: state.sizeW,
|
|
|
|
|
|
sizeH: state.sizeH,
|
|
|
|
|
|
gridW: state.gridW,
|
|
|
|
|
|
gridH: state.gridH,
|
2026-02-27 14:32:43 +08:00
|
|
|
|
bleed: state.bleed,
|
2026-02-27 15:32:04 +08:00
|
|
|
|
padding: state.padding,
|
|
|
|
|
|
fontSize: state.fontSize
|
2026-02-27 14:32:43 +08:00
|
|
|
|
});
|
|
|
|
|
|
setState({ dimensions: dims });
|
|
|
|
|
|
};
|
|
|
|
|
|
|
2026-02-27 15:32:04 +08:00
|
|
|
|
const setSizeW = (size: number) => {
|
|
|
|
|
|
setState({ sizeW: size });
|
2026-02-27 14:32:43 +08:00
|
|
|
|
updateDimensions();
|
|
|
|
|
|
};
|
2026-02-27 15:32:04 +08:00
|
|
|
|
const setSizeH = (size: number) => {
|
|
|
|
|
|
setState({ sizeH: size });
|
|
|
|
|
|
updateDimensions();
|
|
|
|
|
|
};
|
|
|
|
|
|
const setGridW = (grid: number) => {
|
|
|
|
|
|
setState({ gridW: grid });
|
|
|
|
|
|
updateDimensions();
|
|
|
|
|
|
};
|
|
|
|
|
|
const setGridH = (grid: number) => {
|
|
|
|
|
|
setState({ gridH: grid });
|
2026-02-27 14:32:43 +08:00
|
|
|
|
updateDimensions();
|
|
|
|
|
|
};
|
|
|
|
|
|
const setBleed = (bleed: string) => {
|
|
|
|
|
|
setState({ bleed });
|
|
|
|
|
|
updateDimensions();
|
|
|
|
|
|
};
|
|
|
|
|
|
const setPadding = (padding: string) => {
|
|
|
|
|
|
setState({ padding });
|
|
|
|
|
|
updateDimensions();
|
|
|
|
|
|
};
|
2026-02-27 15:32:04 +08:00
|
|
|
|
const setFontSize = (size: number) => {
|
|
|
|
|
|
setState({ fontSize: size });
|
|
|
|
|
|
updateDimensions();
|
|
|
|
|
|
};
|
2026-02-27 14:32:43 +08:00
|
|
|
|
|
|
|
|
|
|
const setCards = (cards: CardData[]) => setState({ cards, activeTab: 0 });
|
2026-02-27 14:19:26 +08:00
|
|
|
|
const setActiveTab = (index: number) => setState({ activeTab: index });
|
|
|
|
|
|
const updateCardData = (index: number, key: string, value: string) => {
|
|
|
|
|
|
setState('cards', index, key, value);
|
|
|
|
|
|
};
|
2026-02-27 14:32:43 +08:00
|
|
|
|
|
2026-02-27 14:19:26 +08:00
|
|
|
|
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
|
|
|
|
|
|
));
|
|
|
|
|
|
};
|
2026-02-27 14:32:43 +08:00
|
|
|
|
|
2026-02-27 14:19:26 +08:00
|
|
|
|
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 });
|
|
|
|
|
|
};
|
2026-02-27 14:32:43 +08:00
|
|
|
|
|
2026-02-27 14:19:26 +08:00
|
|
|
|
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 });
|
|
|
|
|
|
};
|
2026-02-27 14:32:43 +08:00
|
|
|
|
|
2026-02-27 15:21:21 +08:00
|
|
|
|
// 加载卡牌数据(核心逻辑)
|
|
|
|
|
|
const loadCardsFromPath = async (path: string, layersStr: string = '') => {
|
|
|
|
|
|
if (!path) {
|
|
|
|
|
|
setState({ error: '未指定 CSV 文件路径' });
|
2026-02-27 14:32:43 +08:00
|
|
|
|
return;
|
|
|
|
|
|
}
|
2026-02-27 15:21:21 +08:00
|
|
|
|
|
|
|
|
|
|
setState({ isLoading: true, error: null, src: path });
|
|
|
|
|
|
|
|
|
|
|
|
try {
|
|
|
|
|
|
const data = await loadCSV(path);
|
|
|
|
|
|
|
|
|
|
|
|
if (data.length === 0) {
|
|
|
|
|
|
setState({
|
|
|
|
|
|
error: 'CSV 文件为空或格式不正确',
|
|
|
|
|
|
isLoading: false
|
|
|
|
|
|
});
|
|
|
|
|
|
return;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
setState({
|
|
|
|
|
|
cards: data,
|
|
|
|
|
|
activeTab: 0,
|
|
|
|
|
|
layerConfigs: initLayerConfigs(data, layersStr),
|
|
|
|
|
|
isLoading: false
|
|
|
|
|
|
});
|
|
|
|
|
|
updateDimensions();
|
|
|
|
|
|
} catch (err) {
|
|
|
|
|
|
setState({
|
|
|
|
|
|
error: `加载 CSV 失败:${err instanceof Error ? err.message : '未知错误'}`,
|
|
|
|
|
|
isLoading: false
|
|
|
|
|
|
});
|
|
|
|
|
|
}
|
2026-02-27 14:19:26 +08:00
|
|
|
|
};
|
2026-02-27 14:32:43 +08:00
|
|
|
|
|
|
|
|
|
|
const setError = (error: string | null) => setState({ error });
|
2026-02-27 15:21:21 +08:00
|
|
|
|
const clearError = () => setState({ error: null });
|
2026-02-27 14:32:43 +08:00
|
|
|
|
|
2026-02-27 14:19:26 +08:00
|
|
|
|
const generateCode = () => {
|
|
|
|
|
|
const layersStr = state.layerConfigs
|
2026-02-27 14:32:43 +08:00
|
|
|
|
.filter(l => l.visible)
|
|
|
|
|
|
.map(l => `${l.prop}:${l.x1},${l.y1}-${l.x2},${l.y2}`)
|
|
|
|
|
|
.join(' ');
|
2026-02-27 15:32:04 +08:00
|
|
|
|
return `:md-deck[${state.src}]{size="${state.sizeW}x${state.sizeH}" grid="${state.gridW}x${state.gridH}" bleed="${state.bleed}" padding="${state.padding}" fontSize="${state.fontSize}" layers="${layersStr}"}`;
|
2026-02-27 14:19:26 +08:00
|
|
|
|
};
|
2026-02-27 14:32:43 +08:00
|
|
|
|
|
2026-02-27 15:21:21 +08:00
|
|
|
|
const copyCode = async () => {
|
2026-02-27 14:19:26 +08:00
|
|
|
|
const code = generateCode();
|
2026-02-27 15:21:21 +08:00
|
|
|
|
try {
|
|
|
|
|
|
await navigator.clipboard.writeText(code);
|
2026-02-27 14:19:26 +08:00
|
|
|
|
alert('已复制到剪贴板!');
|
2026-02-27 15:21:21 +08:00
|
|
|
|
} catch (err) {
|
2026-02-27 14:19:26 +08:00
|
|
|
|
console.error('复制失败:', err);
|
2026-02-27 15:21:21 +08:00
|
|
|
|
alert('复制失败,请手动复制');
|
|
|
|
|
|
}
|
2026-02-27 14:19:26 +08:00
|
|
|
|
};
|
|
|
|
|
|
|
2026-02-27 14:58:44 +08:00
|
|
|
|
const actions: DeckActions = {
|
2026-02-27 15:32:04 +08:00
|
|
|
|
setSizeW,
|
|
|
|
|
|
setSizeH,
|
|
|
|
|
|
setGridW,
|
|
|
|
|
|
setGridH,
|
2026-02-27 14:19:26 +08:00
|
|
|
|
setBleed,
|
|
|
|
|
|
setPadding,
|
2026-02-27 15:32:04 +08:00
|
|
|
|
setFontSize,
|
2026-02-27 14:19:26 +08:00
|
|
|
|
setCards,
|
|
|
|
|
|
setActiveTab,
|
|
|
|
|
|
updateCardData,
|
|
|
|
|
|
setLayerConfigs,
|
|
|
|
|
|
updateLayerConfig,
|
|
|
|
|
|
toggleLayerVisible,
|
|
|
|
|
|
setIsEditing,
|
|
|
|
|
|
setEditingLayer,
|
|
|
|
|
|
updateLayerPosition,
|
|
|
|
|
|
setIsSelecting,
|
|
|
|
|
|
setSelectStart,
|
|
|
|
|
|
setSelectEnd,
|
|
|
|
|
|
cancelSelection,
|
2026-02-27 15:21:21 +08:00
|
|
|
|
loadCardsFromPath,
|
2026-02-27 14:32:43 +08:00
|
|
|
|
setError,
|
2026-02-27 15:21:21 +08:00
|
|
|
|
clearError,
|
2026-02-27 14:19:26 +08:00
|
|
|
|
generateCode,
|
|
|
|
|
|
copyCode
|
|
|
|
|
|
};
|
2026-02-27 14:58:44 +08:00
|
|
|
|
|
|
|
|
|
|
return { state, actions };
|
2026-02-27 14:19:26 +08:00
|
|
|
|
}
|