发票批量打印

[复制链接]
114 |10
发表于 5 天前 | 显示全部楼层 |阅读模式
发票批量打印工具
2月12日更新:

更新内容:
1、纸张大小新增A5四版和A5两版(A5进纸方向为竖向)
2、增加打印预览功能(有个小bug,部分自定义纸张预览图与实际打印效果会不一致)
3、可选OFD格式打印(下载Spire.Pdf.dll,不下载则pdf和图片格式依旧可以正常打印)(单个OFD文件页面不超过3页)。
4、增加多版打印图片之间的间隙。
5、评论区反馈部分pdf打印机出现卡死,经测试是因为这些打印机调用了windows文件对话框保存pdf文件,由于软件使用了多线程,在打印子线程中调用模态对话框就会出现线程冲突,现修复Adobe PDF和Microsoft Print to PDF打印机出现的冲突;
其他打印机解决方式1:点击界面最下方:使用系统打印对话框进行打印 。
方式2:找到该打印机设置界面-打印机首选项-设置指定保存位置不弹出对话框(如果有)。
方式3:从打印预览界面进行打印。
方式4:更换虚拟打印机(论坛搜索的几款都还不错)。
6、应评论区需求增加同一发票打印两版在一张A4纸上的功能,使用方法:勾选‘复制页面*2’,设置纸张大小‘A4两版’。
7、文件列表中可按Delete键删减选中。
重复一遍:如果使用虚拟打印机推荐勾选合并页面(生成一个文件)。






1月16日更新:
可选A4三版 和自定义纸张大小打印
自定义纸张大小最多保留3个,超过3个最后添加的会覆盖最先添加的





1月12日更新:
可选A4 四版  和  A4 六版 打印
可选添加裁剪线








PDF/图片打印
文件路径传入方式:拖入文件或者文件夹,双击列表输入路径等。
虚拟打印机推荐合并页面(生成一个文件),否则不合并页面(打印机响应速度会快那么一点)
A5进纸方向为竖向:148mmX210mm(宽X高)
请先少量测试页面方向是否正确,设置是否生效,再批量打印。
64位,不支持win7
蓝奏:https://wwvv.lanzout.com/b00l1hq06d 密码:52pj

本帖子中包含更多资源

您需要 登录 才可以下载或查看,没有账号?立即注册

x
回复

使用道具 举报

发表于 5 天前 | 显示全部楼层
[Python]  
  1. import sys
  2. import os
  3. import time
  4. import fitz
  5. from print import Ui_PrintForm
  6. from itertools import count
  7. from pathlib import Path
  8. import configparser
  9. from PIL.ImageQt import ImageQt
  10. from PIL import Image
  11. from PySide6.QtGui import QPainter, QPageSize, QPageLayout, QColor
  12. from PySide6.QtPrintSupport import QPrinter, QPrintDialog, QPrinterInfo
  13. from PySide6.QtCore import QRect, QObject, QThread, Signal, QEvent, Qt
  14. from PySide6.QtWidgets import QApplication, QWidget, QListWidgetItem, QFileDialog, QStyleFactory
  15. class printPdfWorker(QObject):
  16.     finished = Signal()
  17.     progress = Signal(str, str)
  18.     def __init__(self, pdf=None, parent=None):
  19.         super().__init__()
  20.         self.win = parent
  21.         self.paths = pdf
  22.         self.cup = self.win.cup
  23.         self.paper = self.win.paper
  24.         self.printname = self.win.printName
  25.         self.dpi = int(str(self.win.dpi)[:3])
  26.         self.orientation = self.win.dirtion()
  27.         ali = self.win.ali
  28.         if '水平居中' in ali:
  29.             self.inter = 2
  30.         else:  # inserted
  31.             if '靠右居中' in ali:
  32.                 self.inter = 1
  33.         self.double = self.win.double
  34.         self._printer = QPrinter(QPrinter.PrinterMode.HighResolution)
  35.         self._printer.setPrinterName(self.printname)
  36.         if 'A4' in self.paper:
  37.             self._printer.setPageSize(QPageSize(QPageSize.A4))
  38.             self.height_dpx, self.width_dpx = self.a4_size(self.dpi, 210, 297)
  39.         else:  # inserted
  40.             if 'A5' in self.paper:
  41.                 self._printer.setPageSize(QPageSize(QPageSize.A5))
  42.                 self.height_dpx, self.width_dpx = self.a4_size(self.dpi, 148, 210)
  43.         self._printer.setPrintRange(QPrinter.PrintRange.AllPages)
  44.         if self.win.checkbox.isChecked():
  45.             self._printer.setColorMode(QPrinter.ColorMode.GrayScale)
  46.         else:  # inserted
  47.             self._printer.setColorMode(QPrinter.ColorMode.Color)
  48.         try:
  49.             self._printer.setDuplex(self.double)
  50.         except Exception as e:
  51.             pass  # postinserted
  52.         else:  # inserted
  53.             if self.orientation:
  54.                 self._printer.setPageOrientation(self.orientation)
  55.         self._printer.setCopyCount(self.cup)
  56.             print(e)
  57.     def setprinton(self):
  58.         if 'A4' in self.paper:
  59.             self._printer.setPageSize(QPageSize(QPageSize.A4))
  60.             self.height_dpx, self.width_dpx = self.a4_size(self.dpi, 210, 297)
  61.         else:  # inserted
  62.             if 'A5' in self.paper:
  63.                 self._printer.setPageSize(QPageSize(QPageSize.A5))
  64.                 self.height_dpx, self.width_dpx = self.a4_size(self.dpi, 148, 210)
  65.         self._printer.setPrintRange(QPrinter.PrintRange.AllPages)
  66.         if self.win.checkbox.isChecked():
  67.             self._printer.setColorMode(QPrinter.ColorMode.GrayScale)
  68.         else:  # inserted
  69.             self._printer.setColorMode(QPrinter.ColorMode.Color)
  70.         try:
  71.             self._printer.setDuplex(self.double)
  72.         except Exception as e:
  73.             pass  # postinserted
  74.         else:  # inserted
  75.             if self.orientation:
  76.                 self._printer.setPageOrientation(self.orientation)
  77.         self._printer.setCopyCount(self.cup)
  78.             print(e)
  79.         else:  # inserted
  80.             pass
  81.     def rundialog(self):
  82.         self.dialog = QPrintDialog(self._printer)
  83.         self.dialog.setOptions(QPrintDialog.PrintToFile | QPrintDialog.PrintSelection)
  84.         if self.dialog.exec():
  85.             self.runprint()
  86.         else:  # inserted
  87.             self.finished.emit()
  88.     def runprint(self):
  89.         """长时间运行的打印任务。"""  # inserted
  90.         try:
  91.             self.setprinton()
  92.             if self.paper!= 'A4两版':
  93.                 if self.win.mergebox.isChecked():
  94.                     painter = QPainter(self._printer)
  95.                     rect = painter.viewport()
  96.                     images = self.add_image(self.paths)
  97.                     for pil_image, pageNumber in zip(images, count(1)):
  98.                         if pageNumber > 1:
  99.                             self._printer.newPage()
  100.                         self.print_image(pil_image, rect, painter)
  101.                     painter.end()
  102.                 else:  # inserted
  103.                     for index, path in enumerate(self.paths):
  104.                         painter = QPainter(self._printer)
  105.                         rect = painter.viewport()
  106.                         images = []
  107.                         path = Path(path)
  108.                         suffix = path.suffix.lower()
  109.                         if suffix == '.pdf':
  110.                             images = self.open_pdf(path, images)
  111.                             for pil_image, pageNumber in zip(images, count(1)):
  112.                                 if pageNumber > 1:
  113.                                     self._printer.newPage()
  114.                                 self.print_image(pil_image, rect, painter)
  115.                         else:  # inserted
  116.                             with Image.open(path) as image:
  117.                                 pass  # postinserted
  118.         except Exception as e:
  119.                                 pil_image = image.copy()
  120.                                 self.print_image(pil_image, rect, painter)
  121.                         painter.end()
  122.             else:  # inserted
  123.                 if self.win.mergebox.isChecked():
  124.                     images = self.add_image(self.paths)
  125.                     self.A4_sep(images)
  126.                 else:  # inserted
  127.                     batch_size = 10
  128.                     for i in range(0, len(self.paths), batch_size):
  129.                         batch_paths = self.paths[i:i + batch_size]
  130.                         images = self.add_image(batch_paths)
  131.                         self.A4_sep(images)
  132.             self.progress.emit('文件已发送至打印机', 'green')
  133.         else:  # inserted
  134.             self.finished.emit()
  135.                 print(f'打印出错{e}0')
  136.     def open_pdf(self, path, images):
  137.         with fitz.open(path) as pdf:
  138.             num_pages = len(pdf)
  139.             printRange = range(num_pages)
  140.             page_indices = [i for i in printRange]
  141.             for index in page_indices:
  142.                 pixmap = pdf[index].get_pixmap(dpi=self.dpi)
  143.                 pil_image = Image.frombytes('RGB', [pixmap.width, pixmap.height], pixmap.samples)
  144.                 images.append(pil_image)
  145.             return images
  146.     def add_image(self, paths=None):
  147.         images = []
  148.         file_paths = paths
  149.         file_paths = [path for path in file_paths if path.endswith('.pdf')] + [path for path in file_paths if path.endswith(('.jpg', '.jpeg', '.png'))]
  150.         for index, path in enumerate(file_paths):
  151.             path = Path(path)
  152.             suffix = path.suffix.lower()
  153.             if suffix == '.pdf':
  154.                 images = self.open_pdf(path, images)
  155.             else:  # inserted
  156.                 with Image.open(path) as image:
  157.                     images.append(image.copy())
  158.         return images
  159.     def A4_sep(self, images):
  160.         if len(images) % 2!= 0:
  161.             images.append(None)
  162.         painter = QPainter(self._printer)
  163.         for index, pageNumber in zip(range(0, len(images), 2), count(1)):
  164.             image1 = images[index]
  165.             image2 = images[index + 1]
  166.             if pageNumber > 1:
  167.                 self._printer.newPage()
  168.             self.join_pic(image1, image2, painter)
  169.         painter.end()
  170.     def print_image(self, pil_image, rect, painter):
  171.         pilWidth, pilHeight = pil_image.size
  172.         imageRatio = pilHeight / pilWidth
  173.         viewportRatio = rect.height() / rect.width()
  174.         A4Ratio = self.height_dpx / self.width_dpx
  175.         if self.win.up == '自动旋转':
  176.             if viewportRatio < 1 and imageRatio > 1 or (viewportRatio > 1 and imageRatio < 1):
  177.                 pil_image = pil_image.transpose(Image.ROTATE_90)
  178.                 pilWidth, pilHeight = pil_image.size
  179.                 imageRatio = pilHeight / pilWidth
  180.             if A4Ratio < imageRatio:
  181.                 x = int(pilHeight / viewportRatio - pilWidth)
  182.                 xOffset = int(x / 2)
  183.                 yOffset = 0
  184.             else:  # inserted
  185.                 xOffset = 0
  186.                 y = int(rect.height() - rect.width() / pilWidth * pilHeight)
  187.                 yOffset = int(y / self.inter)
  188.         else:  # inserted
  189.             xOffset, yOffset, x, y = (0, 0, 0, 0)
  190.         if viewportRatio > imageRatio:
  191.             y = int(rect.width() / (pilWidth / pilHeight))
  192.             printArea = QRect(xOffset, yOffset, rect.width(), y)
  193.         else:  # inserted
  194.             x = int(pilWidth / pilHeight * rect.height())
  195.             printArea = QRect(xOffset, yOffset, x, rect.height())
  196.         image = ImageQt(pil_image)
  197.         painter.drawImage(printArea, image)
  198.         return painter
  199.     def join_pic(self, image1, image2, painter):
  200.         if image2 == None:
  201.             image2 = Image.new('RGB', image1.size, 'white')
  202.         for image in (image1, image2):
  203.             pilWidth, pilHeight = image.size
  204.             imageRatio = pilHeight / pilWidth
  205.             if imageRatio > 1:
  206.                 if image == image1:
  207.                     image1 = image.transpose(Image.ROTATE_90)
  208.                 if image == image2:
  209.                     image2 = image.transpose(Image.ROTATE_90)
  210.         def resize_image(image):
  211.             height = int(self.height_dpx / 2)
  212.             ratio = height / image.size[1]
  213.             max_width = int(image.size[0] * ratio)
  214.             max_height = int(height)
  215.             if max_width > self.width_dpx:
  216.                 max_width = self.width_dpx
  217.                 ratio = self.width_dpx / image.size[0]
  218.                 max_height = int(image.size[1] * ratio)
  219.             new_width = max_width
  220.             new_height = max_height
  221.             resized_image = image.resize((new_width, new_height))
  222.             return resized_image
  223.         image1 = resize_image(image1)
  224.         image2 = resize_image(image2)
  225.         half_hight = int(self.height_dpx / 2)
  226.         merged_image = Image.new('RGB', (self.width_dpx, self.height_dpx), 'white')
  227.         if image1.size[0] < self.width_dpx:
  228.             x1 = int((self.width_dpx - image1.size[0]) / self.inter)
  229.         else:  # inserted
  230.             x1 = 0
  231.         if image1.size[1] < half_hight:
  232.             y1 = int((half_hight - image1.size[1]) / 2)
  233.         else:  # inserted
  234.             y1 = 0
  235.         if image2.size[0] < self.width_dpx:
  236.             x2 = int((self.width_dpx - image2.size[0]) / self.inter)
  237.         else:  # inserted
  238.             x2 = 0
  239.         if image2.size[1] < half_hight:
  240.             y2 = int((half_hight - image2.size[1]) / 2)
  241.         else:  # inserted
  242.             y2 = 0
  243.         merged_image.paste(image1, (x1, y1))
  244.         merged_image.paste(image2, (x2, half_hight + y2))
  245.         rect = painter.viewport()
  246.         self.print_image(merged_image, rect, painter)
  247.     def a4_size(self, dpi, width, height):
  248.         a4_width = width / 25.4
  249.         a4_height = height / 25.4
  250.         height_dpx = int(a4_height * dpi)
  251.         width_dpx = int(a4_width * dpi)
  252.         return (height_dpx, width_dpx)
  253. class Window(QWidget):
  254.     def __init__(self):
  255.         super(Window, self).__init__()
  256.         self.ui = Ui_PrintForm()
  257.         self.ui_win = self.windowFlags()
  258.         self.ui.setupUi(self)
  259.         self.longRunningBtn = self.ui.pushButton
  260.         self.longRunningBtn.clicked.connect(self.runPrintTask)
  261.         self.addfile = self.ui.pushButton_2
  262.         self.addfile.clicked.connect(self.getFile)
  263.         self.clearfile = self.ui.pushButton_3
  264.         self.clearfile.clicked.connect(self.clearFile)
  265.         self.sysPrint = self.ui.toolButton
  266.         self.sysPrint.clicked.connect(self.rundio)
  267.         self.spinbox = self.ui.doubleSpinBox
  268.         self.paper_box = self.ui.comboBox
  269.         self.dpi_box = self.ui.comboBox_2
  270.         self.double_box = self.ui.comboBox_3
  271.         self.alignment = self.ui.comboBox_4
  272.         self.direction = self.ui.comboBox_6
  273.         self.paper_box.currentIndexChanged.connect(self.setdirection)
  274.         self.bar = self.ui.label_8
  275.         self.listwidget = self.ui.listWidget
  276.         self.checkbox = self.ui.checkBox
  277.         self.mergebox = self.ui.checkBox_2
  278.         self.printbox = self.ui.comboBox_5
  279.         self.load_printers()
  280.         self.small_win = self.ui.dockWidget
  281.         self.small_win.hide()
  282.         self.textbox = self.ui.textEdit
  283.         self.textbox.textChanged.connect(self.changedText)
  284.         self.load_config()
  285.         self.setdirection()
  286.         self.file_path = []
  287.         self.listwidget.viewport().installEventFilter(self)
  288.     def eventFilter(self, source, event):
  289.         if event.type() == QEvent.MouseButtonDblClick and source is self.listwidget.viewport():
  290.             self.small_win.show()
  291.             return True
  292.         return super().eventFilter(source, event)
  293.     def setdirection(self):
  294.         if self.paper_box.currentText()!= 'A4':
  295.             self.direction.setCurrentIndex(0)
  296.             self.direction.setEnabled(False)
  297.         else:  # inserted
  298.             self.direction.setEnabled(True)
  299.     def changedText(self):
  300.         self.clearFile()
  301.         text = self.textbox.toPlainText()
  302.         lines = text.splitlines()
  303.         for line in lines:
  304.             if line.strip():
  305.                 self.showListwidget(line)
  306.     def load_config(self):
  307.         config = configparser.ConfigParser()
  308.         config.read('printConfig.ini')
  309.         self.spinbox.setValue(int(config.get('Print', 'Series', fallback=1)))
  310.         self.paper_box.setCurrentIndex(int(config.get('Print', 'Paper', fallback=0)))
  311.         self.dpi_box.setCurrentIndex(int(config.get('Print', 'Dpi', fallback=1)))
  312.         self.double_box.setCurrentIndex(int(config.get('Print', 'Double', fallback=0)))
  313.         self.double_box.setCurrentIndex(int(config.get('Print', 'Center', fallback=0)))
  314.         self.printbox.setCurrentText(config.get('Print', 'PrintName', fallback=''))
  315.         self.direction.setCurrentIndex(int(config.get('Print', 'PageDirection', fallback=0)))
  316.         self.checkbox.setCheckState(Qt.CheckState.Checked if config.getboolean('Print', 'Color', fallback=False) else Qt.CheckState.Unchecked)
  317.         self.mergebox.setCheckState(Qt.CheckState.Checked if config.getboolean('Print', 'Mergebox', fallback=False) else Qt.CheckState.Unchecked)
  318.     def doublePrint(self):
  319.         double = self.double_box.currentIndex()
  320.         if double == 0:
  321.             return QPrinter.DuplexMode.DuplexNone
  322.         if double == 1:
  323.             return QPrinter.DuplexLongSide
  324.         if double == 2:
  325.             return QPrinter.DuplexShortSide
  326.         if double == 3:
  327.             return QPrinter.DuplexAuto
  328.     def dirtion(self):
  329.         self.up = self.direction.currentText()
  330.         if self.up == '纵向':
  331.             return QPageLayout.Portrait
  332.         if self.up == '横向':
  333.             return QPageLayout.Landscape
  334.     def clearFile(self):
  335.         self.file_path = []
  336.         self.listwidget.clear()
  337.         self.runBar('准备就绪......', 'black')
  338.     def getFile(self):
  339.         response = QFileDialog.getOpenFileNames(parent=self, caption='选择文件', filter='文件类型 (*.pdf *.jpg *.png *.jpeg *.bmp);;Images (*.png *.jpg *.jpeg *.bmp);;PDF Files (*.pdf)')
  340.         if response:
  341.             file_paths = response[0]
  342.             for path in file_paths:
  343.                 self.showListwidget(path)
  344.     def showListwidget(self, path):
  345.         self.file_path.append(path)
  346.         item_widget = QListWidgetItem(path)
  347.         self.listwidget.addItem(item_widget)
  348.         self.bar.setText(f'已添加文件:{len(self.file_path)}个')
  349.         return self.file_path
  350.     def lianjie(self, paths):
  351.         self.clearFile()
  352.         valid_extensions = {'.jpg', '.png', '.jpeg', 'bmp', '.pdf'}
  353.         for path in paths:
  354.             _, extensions = os.path.splitext(path)
  355.             if extensions.lower() in valid_extensions:
  356.                 self.showListwidget(path)
  357.     def load_printers(self):
  358.         printers = QPrinterInfo.availablePrinters()
  359.         printer_names = [printer.printerName() for printer in printers]
  360.         self.printbox.addItems(printer_names)
  361.     def printdata(self):
  362.         self.cup = self.spinbox.value()
  363.         self.paper = self.paper_box.currentText()
  364.         self.dpi = self.dpi_box.currentText()
  365.         self.ali = self.alignment.currentText()
  366.         self.double = self.doublePrint()
  367.         self.printName = self.printbox.currentText()
  368.         self.runBar('正在发送页面到打印机\n请勿关闭程序...', 'red')
  369.     def rundio(self):
  370.         pdf_file = self.file_path
  371.         if not pdf_file:
  372.             self.runBar('没有待打印的文件', 'blue')
  373.             return
  374.         self.printdata()
  375.         self.thread = QThread()
  376.         self.worker = printPdfWorker(pdf_file, self)
  377.         self.worker.moveToThread(self.thread)
  378.         self.thread.started.connect(self.worker.rundialog)
  379.         self.worker.finished.connect(self.thread.quit)
  380.         self.worker.finished.connect(self.worker.deleteLater)
  381.         self.worker.progress.connect(self.runBar)
  382.         self.thread.start()
  383.     def runPrintTask(self):
  384.         pdf_file = self.file_path
  385.         if not pdf_file:
  386.             self.runBar('没有待打印的文件', 'blue')
  387.             return
  388.         self.printdata()
  389.         self.thread = QThread()
  390.         self.worker = printPdfWorker(pdf_file, self)
  391.         self.worker.moveToThread(self.thread)
  392.         self.thread.started.connect(self.worker.runprint)
  393.         self.worker.finished.connect(self.thread.quit)
  394.         self.worker.finished.connect(self.worker.deleteLater)
  395.         self.worker.progress.connect(self.runBar)
  396.         self.thread.start()
  397.     def runBar(self, text, color='black'):
  398.         palette = self.bar.palette()
  399.         palette.setColor(self.bar.foregroundRole(), QColor(color))
  400.         self.bar.setPalette(palette)
  401.         self.bar.setText(text)
  402.     def dragEnterEvent(self, event):
  403.         if event.mimeData().hasUrls():
  404.             event.accept()
  405.         else:  # inserted
  406.             event.ignore()
  407.     def dropEvent(self, event):
  408.         valid_extensions = {'.jpg', '.png', '.jpeg', 'bmp', '.pdf'}
  409.         dropped_files = []
  410.         for url in event.mimeData().urls():
  411.             file_path = url.toLocalFile()
  412.             if os.path.isdir(file_path):
  413.                 for root, dirs, files in os.walk(file_path):
  414.                     for file in files:
  415.                         full_file_path = os.path.join(root, file)
  416.                         _, extension = os.path.splitext(full_file_path)
  417.                         if extension.lower() in valid_extensions:
  418.                             dropped_files.append(full_file_path)
  419.             else:  # inserted
  420.                 _, extension = os.path.splitext(file_path)
  421.                 if extension.lower() in valid_extensions:
  422.                     dropped_files.append(file_path)
  423.         for file_path in dropped_files:
  424.             self.showListwidget(file_path)
  425.     def closeEvent(self, event):
  426.         self.save_combobox()
  427.         event.accept()
  428.     def save_combobox(self):
  429.         config = configparser.ConfigParser()
  430.         config.read('printConfig.ini')
  431.         if 'Print' not in config:
  432.             config.add_section('Print')
  433.         cup = self.spinbox.value()
  434.         paper = self.paper_box.currentIndex()
  435.         dpi = self.dpi_box.currentIndex()
  436.         double = self.double_box.currentIndex()
  437.         center = self.alignment.currentIndex()
  438.         printName = self.printbox.currentText()
  439.         direction = self.direction.currentIndex()
  440.         mergebox = int(self.mergebox.checkState() == Qt.CheckState.Checked)
  441.         config['Print'] = {'Series': int(cup), 'Paper': str(paper), 'Dpi': str(dpi), 'Double': str(double), 'Center': str(center), 'Color': str(int(self.checkbox.checkState() == Qt.CheckState.Checked)), 'PrintName': str(printName), 'PageDirection': str(direction), 'Mergebox': str(mergebox)}
  442.         with open('printConfig.ini', 'w') as configfile:
  443.             config.write(configfile)
  444. if __name__ == '__main__':
  445.     app = QApplication(sys.argv)
  446.     app.setStyle(QStyleFactory.create('Fusion'))
  447.     win = Window()
  448.     win.show()
  449.     sys.exit(app.exec())
复制代码
回复

使用道具 举报

发表于 5 天前 | 显示全部楼层
你的需求版本已经上传了网盘了,打印居中方式选择‘’垂直两端‘’
回复

使用道具 举报

发表于 5 天前 | 显示全部楼层
实测 A4两版 从中间裁剪后 上下俩部分 过大
提个意见
A4两版 时候,上下部分分别 向上和向下靠齐 也就是两端缩进些
中间区域留白增加
会更好

本帖子中包含更多资源

您需要 登录 才可以下载或查看,没有账号?立即注册

x
回复

使用道具 举报

发表于 5 天前 | 显示全部楼层
支持楼主的分享精神,

本帖子中包含更多资源

您需要 登录 才可以下载或查看,没有账号?立即注册

x
回复

使用道具 举报

发表于 5 天前 | 显示全部楼层
2张发票合并页面打印,并不是选择合并页面,而是纸张大小选A4两版。还有2个问题,能不能自由删除添加进去序列的文件,和输出PDF版自己保存。
回复

使用道具 举报

发表于 5 天前 | 显示全部楼层
感谢,这个是神级需求啊,谢谢
回复

使用道具 举报

发表于 5 天前 | 显示全部楼层
有空用一下谢谢分享
回复

使用道具 举报

发表于 5 天前 | 显示全部楼层
省时省力,感谢
回复

使用道具 举报

发表于 5 天前 | 显示全部楼层
非常好,谢谢,收藏了
回复

使用道具 举报

您需要登录后才可以回帖 登录 | 立即注册

本版积分规则

快速回复 返回顶部 返回列表