ttrpg-tools/src/components/md-deck/hooks/deckStore.ts

424 lines
12 KiB
TypeScript
Raw Normal View History

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, CSV } from '../../utils/csv-loader';
2026-03-13 17:26:00 +08:00
import { initLayerConfigs, formatLayers, initLayerConfigsForSide } from './layer-parser';
2026-03-14 15:48:55 +08:00
import type { CardData, LayerConfig, Dimensions, CardSide, CardShape } from '../types';
2026-02-27 14:19:26 +08:00
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:40:52 +08:00
BLEED: 1,
2026-03-15 01:31:16 +08:00
PADDING: 2,
CORNER_RADIUS: 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 15:40:52 +08:00
bleed: number;
padding: number;
2026-03-15 01:31:16 +08:00
cornerRadius: number;
2026-03-14 15:48:55 +08:00
shape: CardShape;
2026-02-27 14:19:26 +08:00
fixed: boolean;
src: string;
2026-02-27 22:56:06 +08:00
rawSrc: string; // 原始 CSV 路径(用于生成代码时保持相对路径)
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: CSV<CardData>;
2026-02-27 14:19:26 +08:00
activeTab: number;
2026-02-27 14:32:43 +08:00
2026-02-27 14:19:26 +08:00
// 图层配置
2026-03-13 17:26:00 +08:00
frontLayerConfigs: LayerConfig[];
backLayerConfigs: LayerConfig[];
2026-02-27 14:32:43 +08:00
2026-02-27 14:19:26 +08:00
// 编辑状态
isEditing: boolean;
editingLayer: string | null;
2026-03-13 17:26:00 +08:00
activeSide: CardSide;
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 16:02:53 +08:00
2026-02-27 20:27:26 +08:00
// 导出状态
isExporting: boolean;
2026-02-27 21:02:33 +08:00
exportProgress: number; // 0-100
exportError: string | null;
2026-02-27 17:55:02 +08:00
// 打印设置
printOrientation: 'portrait' | 'landscape';
2026-03-13 17:26:00 +08:00
printFrontOddPageOffsetX: number;
printFrontOddPageOffsetY: number;
printDoubleSided: boolean;
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 15:40:52 +08:00
setBleed: (bleed: number) => void;
setPadding: (padding: number) => void;
2026-03-15 01:31:16 +08:00
setCornerRadius: (cornerRadius: number) => void;
2026-03-14 15:48:55 +08:00
setShape: (shape: CardShape) => void;
2026-02-27 14:32:43 +08:00
2026-02-27 14:19:26 +08:00
// 数据设置
2026-03-13 11:46:18 +08:00
setCards: (cards: CSV<CardData>) => void;
2026-02-27 14:19:26 +08:00
setActiveTab: (index: number) => void;
updateCardData: (index: number, key: string, value: string) => void;
2026-02-27 14:32:43 +08:00
2026-03-13 17:26:00 +08:00
// 图层操作 - 正面
setFrontLayerConfigs: (configs: LayerConfig[]) => void;
updateFrontLayerConfig: (prop: string, updates: Partial<LayerConfig>) => void;
toggleFrontLayerVisible: (prop: string) => void;
// 图层操作 - 背面
setBackLayerConfigs: (configs: LayerConfig[]) => void;
updateBackLayerConfig: (prop: string, updates: Partial<LayerConfig>) => void;
toggleBackLayerVisible: (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-03-13 17:26:00 +08:00
setActiveSide: (side: CardSide) => 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
// 数据加载
2026-03-13 17:26:00 +08:00
loadCardsFromPath: (path: string, rawSrc: string, layersStr?: string, backLayersStr?: 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
// 生成代码
2026-03-13 17:26:00 +08:00
generateCode: (backLayersStr?: string) => string;
copyCode: (backLayersStr?: string) => Promise<void>;
2026-02-27 16:02:53 +08:00
2026-02-27 20:27:26 +08:00
// 导出操作
setExporting: (exporting: boolean) => void;
exportDeck: () => void;
2026-02-27 21:02:33 +08:00
setExportProgress: (progress: number) => void;
setExportError: (error: string | null) => void;
clearExportError: () => void;
2026-02-27 17:55:02 +08:00
// 打印设置
setPrintOrientation: (orientation: 'portrait' | 'landscape') => void;
2026-03-13 17:26:00 +08:00
setPrintFrontOddPageOffsetX: (offset: number) => void;
setPrintFrontOddPageOffsetY: (offset: number) => void;
setPrintDoubleSided: (doubleSided: boolean) => 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 = '',
): 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-03-15 01:31:16 +08:00
cornerRadius: DECK_DEFAULTS.CORNER_RADIUS,
2026-03-14 15:48:55 +08:00
shape: 'rectangle',
2026-02-27 14:19:26 +08:00
fixed: false,
2026-02-27 15:21:21 +08:00
src: initialSrc,
2026-02-27 22:56:06 +08:00
rawSrc: initialSrc,
2026-02-27 14:19:26 +08:00
dimensions: null,
2026-03-13 11:46:18 +08:00
cards: [] as any,
2026-02-27 14:19:26 +08:00
activeTab: 0,
2026-03-13 17:26:00 +08:00
frontLayerConfigs: [],
backLayerConfigs: [],
2026-02-27 14:19:26 +08:00
isEditing: false,
editingLayer: null,
2026-03-13 17:26:00 +08:00
activeSide: 'front',
2026-02-27 14:19:26 +08:00
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 16:02:53 +08:00
error: null,
2026-02-27 20:27:26 +08:00
isExporting: false,
2026-02-27 21:02:33 +08:00
exportProgress: 0,
exportError: null,
2026-02-27 17:55:02 +08:00
printOrientation: 'portrait',
2026-03-13 17:26:00 +08:00
printFrontOddPageOffsetX: 0,
printFrontOddPageOffsetY: 0,
printDoubleSided: false
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-28 12:18:52 +08:00
padding: state.padding
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();
};
2026-02-27 15:40:52 +08:00
const setBleed = (bleed: number) => {
2026-02-27 14:32:43 +08:00
setState({ bleed });
updateDimensions();
};
2026-02-27 15:40:52 +08:00
const setPadding = (padding: number) => {
2026-02-27 14:32:43 +08:00
setState({ padding });
updateDimensions();
};
2026-03-15 01:31:16 +08:00
const setCornerRadius = (cornerRadius: number) => {
setState({ cornerRadius });
};
2026-03-14 15:48:55 +08:00
const setShape = (shape: CardShape) => {
setState({ shape });
};
2026-02-27 14:32:43 +08:00
2026-03-13 11:46:18 +08:00
const setCards = (cards: CSV<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-03-13 17:26:00 +08:00
// 正面图层操作
const setFrontLayerConfigs = (configs: LayerConfig[]) => setState({ frontLayerConfigs: configs });
const updateFrontLayerConfig = (prop: string, updates: Partial<LayerConfig>) => {
setState('frontLayerConfigs', (prev) => prev.map((config) => config.prop === prop ? { ...config, ...updates } : config));
};
const toggleFrontLayerVisible = (prop: string) => {
setState('frontLayerConfigs', (prev) => prev.map((config) =>
config.prop === prop ? { ...config, visible: !config.visible } : config
));
};
// 背面图层操作
const setBackLayerConfigs = (configs: LayerConfig[]) => setState({ backLayerConfigs: configs });
const updateBackLayerConfig = (prop: string, updates: Partial<LayerConfig>) => {
setState('backLayerConfigs', (prev) => prev.map((config) => config.prop === prop ? { ...config, ...updates } : config));
2026-02-27 14:19:26 +08:00
};
2026-03-13 17:26:00 +08:00
const toggleBackLayerVisible = (prop: string) => {
setState('backLayerConfigs', (prev) => prev.map((config) =>
2026-02-27 14:19:26 +08:00
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 });
2026-03-13 17:26:00 +08:00
const setActiveSide = (side: CardSide) => setState({ activeSide: side });
2026-02-27 14:19:26 +08:00
const updateLayerPosition = (x1: number, y1: number, x2: number, y2: number) => {
const layer = state.editingLayer;
if (!layer) return;
2026-03-13 17:26:00 +08:00
const currentSide = state.activeSide;
const configs = currentSide === 'front' ? state.frontLayerConfigs : state.backLayerConfigs;
const setter = currentSide === 'front' ? setFrontLayerConfigs : setBackLayerConfigs;
setter(configs.map((config) =>
2026-02-27 14:19:26 +08:00
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
// 加载卡牌数据(核心逻辑)
2026-03-13 17:26:00 +08:00
const loadCardsFromPath = async (path: string, rawSrc: string, layersStr: string = '', backLayersStr: string = '') => {
2026-02-27 15:21:21 +08:00
if (!path) {
setState({ error: '未指定 CSV 文件路径' });
2026-02-27 14:32:43 +08:00
return;
}
2026-02-27 15:21:21 +08:00
2026-02-27 22:56:06 +08:00
setState({ isLoading: true, error: null, src: path, rawSrc: rawSrc });
2026-02-27 15:21:21 +08:00
try {
const data = await loadCSV(path);
2026-03-13 17:26:00 +08:00
2026-02-27 15:21:21 +08:00
if (data.length === 0) {
2026-03-13 17:26:00 +08:00
setState({
2026-02-27 15:21:21 +08:00
error: 'CSV 文件为空或格式不正确',
2026-03-13 17:26:00 +08:00
isLoading: false
2026-02-27 15:21:21 +08:00
});
return;
}
2026-03-13 17:26:00 +08:00
setState({
cards: data,
2026-02-27 15:21:21 +08:00
activeTab: 0,
2026-03-13 17:26:00 +08:00
frontLayerConfigs: initLayerConfigsForSide(data, layersStr),
backLayerConfigs: initLayerConfigsForSide(data, backLayersStr),
isLoading: false
2026-02-27 15:21:21 +08:00
});
updateDimensions();
} catch (err) {
2026-03-13 17:26:00 +08:00
setState({
2026-02-27 15:21:21 +08:00
error: `加载 CSV 失败:${err instanceof Error ? err.message : '未知错误'}`,
2026-03-13 17:26:00 +08:00
isLoading: false
2026-02-27 15:21:21 +08:00
});
}
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-03-13 17:26:00 +08:00
const generateCode = (backLayersStr?: string) => {
const frontLayersStr = formatLayers(state.frontLayerConfigs);
const backLayersString = backLayersStr || formatLayers(state.backLayerConfigs);
2026-03-13 11:14:01 +08:00
const parts = [
`:md-deck[${state.rawSrc || state.src}]`,
2026-03-13 17:29:21 +08:00
`{size="${state.sizeW}x${state.sizeH}" `,
`grid="${state.gridW}x${state.gridH}" `
2026-03-13 11:14:01 +08:00
];
2026-03-13 17:26:00 +08:00
2026-03-13 11:14:01 +08:00
// 仅在非默认值时添加 bleed 和 padding
if (state.bleed !== DECK_DEFAULTS.BLEED) {
2026-03-13 17:29:21 +08:00
parts.push(`bleed="${state.bleed}" `);
2026-03-13 11:14:01 +08:00
}
if (state.padding !== DECK_DEFAULTS.PADDING) {
2026-03-13 17:29:21 +08:00
parts.push(`padding="${state.padding}" `);
2026-03-13 11:14:01 +08:00
}
2026-03-14 15:48:55 +08:00
if (state.shape !== 'rectangle') {
parts.push(`shape="${state.shape}" `);
}
2026-03-13 17:26:00 +08:00
2026-03-13 17:29:21 +08:00
parts.push(`layers="${frontLayersStr}" `);
2026-03-13 17:26:00 +08:00
if (backLayersString) {
2026-03-13 17:29:21 +08:00
parts.push(`back-layers="${backLayersString}" `);
2026-03-13 17:26:00 +08:00
}
parts.push('}');
2026-03-13 11:46:18 +08:00
return parts.join('');
2026-02-27 14:19:26 +08:00
};
2026-02-27 14:32:43 +08:00
2026-03-13 17:26:00 +08:00
const copyCode = async (backLayersStr?: string) => {
const code = generateCode(backLayersStr);
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 20:27:26 +08:00
const setExporting = (exporting: boolean) => setState({ isExporting: exporting });
2026-02-27 16:02:53 +08:00
2026-02-27 20:27:26 +08:00
const exportDeck = () => {
2026-02-27 21:02:33 +08:00
setState({ isExporting: true, exportProgress: 0, exportError: null });
2026-02-27 16:02:53 +08:00
};
2026-02-27 21:02:33 +08:00
const setExportProgress = (progress: number) => setState({ exportProgress: progress });
const setExportError = (error: string | null) => setState({ exportError: error });
const clearExportError = () => setState({ exportError: null });
2026-02-27 17:55:02 +08:00
const setPrintOrientation = (orientation: 'portrait' | 'landscape') => {
setState({ printOrientation: orientation });
};
2026-03-13 17:26:00 +08:00
const setPrintFrontOddPageOffsetX = (offset: number) => {
setState({ printFrontOddPageOffsetX: offset });
};
const setPrintFrontOddPageOffsetY = (offset: number) => {
setState({ printFrontOddPageOffsetY: offset });
2026-02-27 17:55:02 +08:00
};
2026-03-13 17:26:00 +08:00
const setPrintDoubleSided = (doubleSided: boolean) => {
setState({ printDoubleSided: doubleSided });
2026-02-27 17:55:02 +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-03-15 01:31:16 +08:00
setCornerRadius,
2026-03-14 15:48:55 +08:00
setShape,
2026-02-27 14:19:26 +08:00
setCards,
setActiveTab,
updateCardData,
2026-03-13 17:26:00 +08:00
setFrontLayerConfigs,
updateFrontLayerConfig,
toggleFrontLayerVisible,
setBackLayerConfigs,
updateBackLayerConfig,
toggleBackLayerVisible,
2026-02-27 14:19:26 +08:00
setIsEditing,
setEditingLayer,
updateLayerPosition,
2026-03-13 17:26:00 +08:00
setActiveSide,
2026-02-27 14:19:26 +08:00
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,
2026-02-27 16:02:53 +08:00
copyCode,
2026-02-27 20:27:26 +08:00
setExporting,
exportDeck,
2026-02-27 21:02:33 +08:00
setExportProgress,
setExportError,
clearExportError,
2026-02-27 17:55:02 +08:00
setPrintOrientation,
2026-03-13 17:26:00 +08:00
setPrintFrontOddPageOffsetX,
setPrintFrontOddPageOffsetY,
setPrintDoubleSided
2026-02-27 14:19:26 +08:00
};
2026-02-27 14:58:44 +08:00
return { state, actions };
2026-02-27 14:19:26 +08:00
}