feat: print outline

This commit is contained in:
hypercross 2026-02-27 16:32:45 +08:00
parent bab09a2561
commit dad93d06e3
2 changed files with 158 additions and 8 deletions

View File

@ -1,4 +1,4 @@
import { Show, For, createMemo } from 'solid-js'; import { For, createMemo } from 'solid-js';
import { marked } from '../../markdown'; import { marked } from '../../markdown';
import { getLayerStyle } from './hooks/dimensions'; import { getLayerStyle } from './hooks/dimensions';
import type { DeckStore } from './hooks/deckStore'; import type { DeckStore } from './hooks/deckStore';
@ -28,7 +28,7 @@ export function PrintPreview(props: PrintPreviewProps) {
const A4_HEIGHT = 297; const A4_HEIGHT = 297;
const PRINT_MARGIN = 5; // 打印边距 const PRINT_MARGIN = 5; // 打印边距
// 计算每张卡牌在 A4 纸上的位置 // 计算每张卡牌在 A4 纸上的位置(居中布局)
const pages = createMemo(() => { const pages = createMemo(() => {
const cards = store.state.cards; const cards = store.state.cards;
const cardWidth = store.state.dimensions?.cardWidth || 56; const cardWidth = store.state.dimensions?.cardWidth || 56;
@ -45,9 +45,21 @@ export function PrintPreview(props: PrintPreviewProps) {
// 每页的卡牌数量 // 每页的卡牌数量
const cardsPerPage = cardsPerRow * rowsPerPage; const cardsPerPage = cardsPerRow * rowsPerPage;
// 计算最大卡牌区域的尺寸(用于居中和外围框)
const maxGridWidth = cardsPerRow * cardWidth;
const maxGridHeight = rowsPerPage * cardHeight;
// 居中偏移量(使卡牌区域在 A4 纸上居中)
const offsetX = (A4_WIDTH - maxGridWidth) / 2;
const offsetY = (A4_HEIGHT - maxGridHeight) / 2;
// 分页 // 分页
const result: { pageIndex: number; cards: Array<{ data: typeof cards[0]; x: number; y: number }> }[] = []; const result: {
let currentPage: typeof result[0] = { pageIndex: 0, cards: [] }; pageIndex: number;
cards: Array<{ data: typeof cards[0]; x: number; y: number }>;
bounds: { minX: number; minY: number; maxX: number; maxY: number };
}[] = [];
let currentPage: typeof result[0] = { pageIndex: 0, cards: [], bounds: { minX: Infinity, minY: Infinity, maxX: -Infinity, maxY: -Infinity } };
for (let i = 0; i < cards.length; i++) { for (let i = 0; i < cards.length; i++) {
const pageIndex = Math.floor(i / cardsPerPage); const pageIndex = Math.floor(i / cardsPerPage);
@ -57,21 +69,91 @@ export function PrintPreview(props: PrintPreviewProps) {
if (pageIndex !== currentPage.pageIndex) { if (pageIndex !== currentPage.pageIndex) {
result.push(currentPage); result.push(currentPage);
currentPage = { pageIndex, cards: [] }; currentPage = { pageIndex, cards: [], bounds: { minX: Infinity, minY: Infinity, maxX: -Infinity, maxY: -Infinity } };
} }
// 使用居中偏移量计算卡牌位置
const cardX = offsetX + col * cardWidth;
const cardY = offsetY + row * cardHeight;
currentPage.cards.push({ currentPage.cards.push({
data: cards[i], data: cards[i],
x: PRINT_MARGIN + col * cardWidth, x: cardX,
y: PRINT_MARGIN + row * cardHeight y: cardY
}); });
// 更新边界(含 1mm 边距)
currentPage.bounds.minX = Math.min(currentPage.bounds.minX, cardX);
currentPage.bounds.minY = Math.min(currentPage.bounds.minY, cardY);
currentPage.bounds.maxX = Math.max(currentPage.bounds.maxX, cardX + cardWidth);
currentPage.bounds.maxY = Math.max(currentPage.bounds.maxY, cardY + cardHeight);
} }
if (currentPage.cards.length > 0) { if (currentPage.cards.length > 0) {
result.push(currentPage); result.push(currentPage);
} }
return result; // 为每页添加固定的外围框尺寸(基于最大网格)
return result.map(page => ({
...page,
frameBounds: {
minX: offsetX,
minY: offsetY,
maxX: offsetX + maxGridWidth,
maxY: offsetY + maxGridHeight
}
}));
});
// 计算裁切线和外围框位置
const cropMarks = createMemo(() => {
const pagesData = pages();
return pagesData.map(page => {
const { frameBounds, cards } = page;
const cardWidth = store.state.dimensions?.cardWidth || 56;
const cardHeight = store.state.dimensions?.cardHeight || 88;
// 收集所有唯一的裁切线位置
const xPositions = new Set<number>();
const yPositions = new Set<number>();
cards.forEach(card => {
xPositions.add(card.x);
xPositions.add(card.x + cardWidth);
yPositions.add(card.y);
yPositions.add(card.y + cardHeight);
});
const sortedX = Array.from(xPositions).sort((a, b) => a - b);
const sortedY = Array.from(yPositions).sort((a, b) => a - b);
// 裁切线超出外围框的距离
const OVERLAP = 3; // 3mm
// 生成水平裁切线(沿 Y 轴)
const horizontalLines = sortedY.map(y => ({
y,
xStart: frameBounds.minX - OVERLAP,
xEnd: frameBounds.maxX + OVERLAP
}));
// 生成垂直裁切线(沿 X 轴)
const verticalLines = sortedX.map(x => ({
x,
yStart: frameBounds.minY - OVERLAP,
yEnd: frameBounds.maxY + OVERLAP
}));
// 外围框边界(离卡牌区域边缘 1mm
const frameBoundsWithMargin = {
x: frameBounds.minX - 1,
y: frameBounds.minY - 1,
width: frameBounds.maxX - frameBounds.minX + 2,
height: frameBounds.maxY - frameBounds.minY + 2
};
return { horizontalLines, verticalLines, frameBounds, frameBoundsWithMargin };
});
}); });
const visibleLayers = createMemo(() => store.state.layerConfigs.filter((l) => l.visible)); const visibleLayers = createMemo(() => store.state.layerConfigs.filter((l) => l.visible));
@ -116,6 +198,71 @@ export function PrintPreview(props: PrintPreviewProps) {
> >
{/* 渲染该页的所有卡牌 */} {/* 渲染该页的所有卡牌 */}
<div class="relative w-full h-full"> <div class="relative w-full h-full">
{/* 裁切线和外围框层 */}
<svg class="absolute inset-0 w-full h-full pointer-events-none" style={{ overflow: 'visible' }}>
{/* 外围边框:黑色 0.2mm */}
<rect
x={`${cropMarks()[page.pageIndex]?.frameBoundsWithMargin.x}mm`}
y={`${cropMarks()[page.pageIndex]?.frameBoundsWithMargin.y}mm`}
width={`${cropMarks()[page.pageIndex]?.frameBoundsWithMargin.width}mm`}
height={`${cropMarks()[page.pageIndex]?.frameBoundsWithMargin.height}mm`}
fill="none"
stroke="black"
stroke-width="0.2"
/>
{/* 水平裁切线 */}
<For each={cropMarks()[page.pageIndex]?.horizontalLines}>
{(line) => (
<>
{/* 左侧裁切线(外围框外部) */}
<line
x1={`${line.xStart}mm`}
y1={`${line.y}mm`}
x2={`${page.frameBounds.minX}mm`}
y2={`${line.y}mm`}
stroke="#888"
stroke-width="0.1"
/>
{/* 右侧裁切线(外围框外部) */}
<line
x1={`${page.frameBounds.maxX}mm`}
y1={`${line.y}mm`}
x2={`${line.xEnd}mm`}
y2={`${line.y}mm`}
stroke="#888"
stroke-width="0.1"
/>
</>
)}
</For>
{/* 垂直裁切线 */}
<For each={cropMarks()[page.pageIndex]?.verticalLines}>
{(line) => (
<>
{/* 上方裁切线(外围框外部) */}
<line
x1={`${line.x}mm`}
y1={`${line.yStart}mm`}
x2={`${line.x}mm`}
y2={`${page.frameBounds.minY}mm`}
stroke="#888"
stroke-width="0.1"
/>
{/* 下方裁切线(外围框外部) */}
<line
x1={`${line.x}mm`}
y1={`${page.frameBounds.maxY}mm`}
x2={`${line.x}mm`}
y2={`${line.yEnd}mm`}
stroke="#888"
stroke-width="0.1"
/>
</>
)}
</For>
</svg>
<For each={page.cards}> <For each={page.cards}>
{(card) => ( {(card) => (
<div <div

View File

@ -11,6 +11,9 @@
visibility: visible; visibility: visible;
-webkit-print-color-adjust: exact !important; -webkit-print-color-adjust: exact !important;
print-color-adjust: exact !important; print-color-adjust: exact !important;
position: absolute;
top: 0;
left: 0;
} }
@page { @page {