This commit is contained in:
yuchenglong
2025-12-09 13:55:52 +08:00
7 changed files with 101 additions and 6 deletions

View File

@@ -300,16 +300,18 @@ ipcMain.handle("print-pdf", async (event, pdfDataOrUrl, printOptions = {}) => {
throw new Error(`PDF文件大小异常${stats.size} bytes可能文件损坏`); throw new Error(`PDF文件大小异常${stats.size} bytes可能文件损坏`);
} }
const duplexMode = printOptions?.duplex;
log.info( log.info(
`准备打印PDF: ${pdfFilePath}, 打印机: ${ `准备打印PDF: ${pdfFilePath}, 打印机: ${
targetPrinterName || "默认打印机" targetPrinterName || "默认打印机"
}` }, 双面打印: ${duplexMode ? "是" : "否"}`
); );
// 使用 pdf-to-printer 打印 // 使用 pdf-to-printer 打印
const printOptions_ptp = { const printOptions_ptp = {
printer: targetPrinterName || undefined, printer: targetPrinterName || undefined,
silent: true, // 静默打印 silent: true, // 静默打印
duplex: duplexMode || false, // 双面打印选项
}; };
await ptp.print(pdfFilePath, printOptions_ptp); await ptp.print(pdfFilePath, printOptions_ptp);

View File

@@ -0,0 +1,32 @@
.duplex-toggle {
display: flex;
align-items: center;
cursor: pointer;
user-select: none;
gap: 10px;
transition: opacity 0.2s ease;
}
.duplex-toggle:hover {
opacity: 0.8;
}
.duplex-toggle-icon {
width: 24px;
height: 24px;
display: flex;
align-items: center;
justify-content: center;
border-radius: 4px;
font-size: 16px;
transition: all 0.2s ease;
flex-shrink: 0;
font-family: NotoSansCJKsc-Medium;
}
.duplex-toggle-label {
font-size: 20px;
font-family: NotoSansCJKsc-Regular;
color: rgba(0, 45, 93, 1);
}

View File

@@ -0,0 +1,30 @@
import React from "react";
import "./DuplexToggle.css";
interface DuplexToggleProps {
checked: boolean;
onChange: (checked: boolean) => void;
label?: string;
}
const DuplexToggle: React.FC<DuplexToggleProps> = ({
checked,
onChange,
label = "",
}) => {
const handleClick = () => {
onChange(!checked);
};
return (
<div className="duplex-toggle" onClick={handleClick}>
<div className={`duplex-toggle-icon ${checked ? "checked" : "unchecked"}`}>
{checked ? "双" : "单"}
</div>
<span className="duplex-toggle-label">{label}</span>
</div>
);
};
export default DuplexToggle;

2
src/electron.d.ts vendored
View File

@@ -14,7 +14,7 @@ interface ElectronAPI {
) => Promise<{ success: boolean; data?: string; error?: string }>; ) => Promise<{ success: boolean; data?: string; error?: string }>;
printPdf: ( printPdf: (
pdfUrl: string, pdfUrl: string,
options?: { printerName?: string } options?: { printerName?: string; duplex?: boolean }
) => Promise<{ success: boolean; error?: string }>; ) => Promise<{ success: boolean; error?: string }>;
getPrinters: () => Promise<{ getPrinters: () => Promise<{
success: boolean; success: boolean;

View File

@@ -349,3 +349,10 @@
font-size: 22px; font-size: 22px;
color: #b12651; color: #b12651;
} }
.ui8-duplex-option {
margin-top: 15px;
display: flex;
justify-content: flex-end;
align-items: center;
}

View File

@@ -5,11 +5,13 @@ import "./UI8.css";
import "../../assets/css/basic.css"; import "../../assets/css/basic.css";
import { useNavigate } from "react-router-dom"; import { useNavigate } from "react-router-dom";
import ConfirmButton from "../../components/ConfirmButton"; import ConfirmButton from "../../components/ConfirmButton";
import DuplexToggle from "../../components/DuplexToggle";
import ui8A from "../../assets/ui8A.png"; import ui8A from "../../assets/ui8A.png";
import ui8B from "../../assets/ui8B.png"; import ui8B from "../../assets/ui8B.png";
import { getDaojiandanPdf } from "../../api/hisApi"; import { getDaojiandanPdf } from "../../api/hisApi";
const PREFERRED_PRINTER_KEY = "preferredPrinterName"; const PREFERRED_PRINTER_KEY = "preferredPrinterName";
const PREFERRED_DUPLEX_KEY = "preferredDuplex";
type PrinterInfo = ElectronPrinterInfo; type PrinterInfo = ElectronPrinterInfo;
@@ -74,6 +76,7 @@ const UI8: React.FC = () => {
const printerDropdownRef = useRef<HTMLDivElement | null>(null); const printerDropdownRef = useRef<HTMLDivElement | null>(null);
const pdfContainerRef = useRef<HTMLDivElement | null>(null); const pdfContainerRef = useRef<HTMLDivElement | null>(null);
const [isRightSectionHidden, setIsRightSectionHidden] = useState(false); const [isRightSectionHidden, setIsRightSectionHidden] = useState(false);
const [duplexEnabled, setDuplexEnabled] = useState<boolean>(false);
const getExamId = () => { const getExamId = () => {
const storedId = localStorage.getItem("selectedExamId"); const storedId = localStorage.getItem("selectedExamId");
@@ -221,6 +224,12 @@ const UI8: React.FC = () => {
fetchPrinters(); fetchPrinters();
// 恢复双面打印设置
const storedDuplex = localStorage.getItem(PREFERRED_DUPLEX_KEY);
if (storedDuplex === "true") {
setDuplexEnabled(true);
}
return () => { return () => {
isMounted = false; isMounted = false;
}; };
@@ -343,7 +352,10 @@ const UI8: React.FC = () => {
const printResult = await window.electronAPI.printPdf( const printResult = await window.electronAPI.printPdf(
printData, printData,
selectedPrinter ? { printerName: selectedPrinter } : undefined {
...(selectedPrinter ? { printerName: selectedPrinter } : {}),
duplex: duplexEnabled,
}
); );
if (!printResult.success) { if (!printResult.success) {
@@ -363,7 +375,7 @@ const UI8: React.FC = () => {
} finally { } finally {
setIsPrinting(false); setIsPrinting(false);
} }
}, [originPdfUrls, pdfFiles, printers.length, selectedPrinter, navigate, isPrinting]); }, [originPdfUrls, pdfFiles, printers.length, selectedPrinter, navigate, isPrinting, duplexEnabled]);
const handlePageCountUpdate = useCallback((index: number, numPages: number) => { const handlePageCountUpdate = useCallback((index: number, numPages: number) => {
window.electronAPI?.log("info", `[UI8] PDF渲染成功 (index=${index}),共 ${numPages}`); window.electronAPI?.log("info", `[UI8] PDF渲染成功 (index=${index}),共 ${numPages}`);
@@ -399,6 +411,17 @@ const UI8: React.FC = () => {
} }
}} }}
> >
<DuplexToggle
checked={duplexEnabled}
onChange={(enabled) => {
setDuplexEnabled(enabled);
localStorage.setItem(PREFERRED_DUPLEX_KEY, enabled.toString());
window.electronAPI?.log(
"info",
`[UI8] 双面打印设置为: ${enabled ? "启用" : "禁用"}`
);
}}
/>
<span> <span>
{printersLoading {printersLoading
? "打印机加载中..." ? "打印机加载中..."
@@ -436,6 +459,7 @@ const UI8: React.FC = () => {
{printerError && ( {printerError && (
<div className="ui8-printer-error">{printerError}</div> <div className="ui8-printer-error">{printerError}</div>
)} )}
</div> </div>
<div className="basic-white-block"> <div className="basic-white-block">

View File

@@ -27,8 +27,8 @@ const UI9: React.FC = () => {
navigate("/"); navigate("/");
}, [navigate]); }, [navigate]);
const [countdown, setCountdown] = useState(30); const [countdown, setCountdown] = useState(45);
const [backTime, setBackTime] = useState("确认(30S)"); const [backTime, setBackTime] = useState("确认(45S)");
useEffect(() => { useEffect(() => {
if (countdown > 0) { if (countdown > 0) {