可以做公司公章的软件(一个非常实用的加盖电子公章的工具)
可以做公司公章的软件(一个非常实用的加盖电子公章的工具)
2024-06-30 05:40:39  作者:寂寞如歌啊  网址:https://m.xinb2b.cn/tech/mjt323731.html

大家好,我是野生程序员一灯。

今天分享一个python开发的小GUI项目:电子签章器。

客户的需求是这样的:

客户是一个不小的中药生产和批发企业,每天都需要为它的客户提供大量的药品质量检测纸质文件,并且需要在纸质上加盖公章。据客户说,每年都为此消耗大量的A4纸,而且需要人工处理,长期积累下来也浪费了大量的人力成本。该客户想直接为它的客户发带公章的pdf文件,这样一年能节省不少的成本。只需要一个员工,通过一个桌面软件,批量选择word文件,并且可随时选择电子公章文件(pdf格式),直接将word文件转成pdf并自动加盖公章。

这个需求有几个关键点:

1、考虑到word的格式有doc和docx,这里可以使用win32com,同时非常方便地将word转成pdf;

2、将word转成pdf后,为pdf加盖公章,本质上是加水印的行为,在代码中的逻辑是两页pdf叠加在一起;

3、使用pyqt6,为了提高使用体验,需要使用多线程,即UI线程和业务线程分开,避免在处理文件时UI界面僵死。

成品的样子

可以做公司公章的软件(一个非常实用的加盖电子公章的工具)(1)

打包后的样子

word处理部分的核心代码

from win32com.client import Dispatchimport osclass WordHandler: def __init__(self, file, only_read=True): self.file = file self.only_read = only_read self.word = Dispatch('kwps.Application') self.word.Visible = 0 self.word.DisplayAlerts = 0 self.doc = self.__open() def __open(self): doc = self.word.Documents.Open(self.file, ReadOnly=self.only_read) return doc def save_as_pdf(self): suffix = os.path.splitext(self.file)[-1] if suffix == '.doc': save_to = self.file.replace('.doc', '.pdf') elif suffix == '.docx': save_to = self.file.replace('.docx', '.pdf') self.doc.ExportAsFixedFormat(save_to, 17) return save_to def close(self): self.doc.Close() def quit(self): self.word.Quit()

啰嗦一句,在word转pdf这里,`ExportAsFixedFormat`是`Document`的方法(Open方法返回的是一个Document对象),除了这个方法,也有`SaveAs`和`SaveAs2`方法可以实现,它们是`Application`的方法,即它们是客户端提供的方法,对应着客户端 -> 文件 -> 另存为。

pdf加水印部分

这里使用`PyPDF2`,这是个声名在外的库了,无需多介绍,这里只分析加水印最基本的逻辑。

1、使用`PdfFileReader`分别打开水印pdf文件和要处理的pdf文件

2、水印文件只需要一页就ok,所以要使用PdfFileReader.getPage(0)来获取到第一页;

3、使用`PdfFileWriter`创建一个新的pdf对象,用来保存最后处理的结果

3、要处理的pdf文件在打开后,按照页数循环,逐个循环,将循环到的单页和水印单页合并,然后保存到`PdfFileWriter`所创建的对象中

至此,添加水印的工作便完成了。

pyqt的多线程部分

其实整个需求中,最耗费时间的应该是pyqt的多线程这部分了,使用多线程可以将UI线程和业务线程有效地分开,好处多多。先捋一捋最基本的流程。

可以做公司公章的软件(一个非常实用的加盖电子公章的工具)(2)

1、点击签章后,手动开启一个子线程,UI主线程需要将1和2的数据传递到子线程

2、子线程拿到数据后,就开始工作,并且每完成一个文件,还需要将信号发回到主线程,主线程可以告知用户处理的进度

3、所有文件处理完毕时,手动关闭子线程

这是子线程的核心代码

class MyDocumentThread(QObject): # 发送信号 signal_for_send = pyqtSignal(str) # 接收信号 接收一个文件列表 signal_for_accept = pyqtSignal(list) def __init__(self, parent=None): super(MyDocumentThread, self).__init__() self.lock = QMutex() def work(self, file_list): """ 文档处理逻辑 """ print('子线程接受到的数据:', file_list) # 上锁 self.lock.lock() temp_pdf_list = [] for file in file_list[0]: pdf_file = word2pdf(file) temp_pdf_list.append(pdf_file) # 发送信号给UI主线程 print('pdf处理后', temp_pdf_list) self.lock.unlock() # 将temp_pdf_list增加水印 self.lock.lock() for index, pdf in enumerate(temp_pdf_list, 1): done = add_mark(pdf, file_list[-1]) print(index, pdf) if done == 'success': self.signal_for_send.emit(f'{pdf}处理完成...') if index == len(temp_pdf_list): self.signal_for_send.emit('done') # 删除临时文件 self.lock.unlock()

注:

官方强烈推荐使用QObject来自定义自己的线程,在该线程中实现自己的业务逻辑,而非在QThread中实现执行业务逻辑。所以以上逻辑在继承QObject来实现的,并且在实例化自定义自定义线程时,需要手动实例化一个QThread,使用moveToThread将自定义线程对象交给QThread对象来处理。

关于pyqt的多线程,既有官网文档,也有各界道友的各种实现,我这里也说说自己的心得。

1、根据以上的描述,主线程和子线程需要双向通信,我们都知道,使用`pyqtSignal`可以通信,这是一个发送信号的方法,但如何实现双向通信呢?网上大部分的案例都是单向通信,即子线程将处理的结果或其他信息传回主线程。这里我们需要注意到一个关键的动作,即:`pyqtSignal().emit()`,这是发送信号的工作,如何双向通信呢?本质上,在主线程中使用`emit`,那么就是主线程向子线程发送信号,在子线程中使用`emit`就是子线程向主线程发送信号。要注意的是,需要提前定义好这两种信号,发送时可以在需要的地方执行`emit`就可以了。

可以做公司公章的软件(一个非常实用的加盖电子公章的工具)(3)

根据子线程传回的信号,向用户展示“全部处理完成”

2、如何在使用多线程

(1)以上的代码仅仅是核心代码,是实现多线程的方法,但是如何在pyqt整个流程中调用呢?这里谈谈关键点,首先要在初始化界面时,实例化咱们自定义好的线程对象。

可以做公司公章的软件(一个非常实用的加盖电子公章的工具)(4)

(2)在点击签章时,才开启线程

可以做公司公章的软件(一个非常实用的加盖电子公章的工具)(5)

有些道友在处理化界面时就启动了线程,其实要看具体需求哈,手工控制线程的启动和关闭,可以实现软件循环使用,比如处理完一批软件文档时,接着可以处理第二批,这就是手工控制的好处。如果在界面初始化自动开启线程,那么关闭后就无法再开线程了,得关闭软件才能再次开启。

好啦,今天的分享就到此为止。欢迎私信交流。

  • 中年女人时尚穿搭大全(这才是中年女人该有的穿搭)
  • 2024-06-30这才是中年女人该有的穿搭有人说,四五十岁,已经到了等待退休的年纪,哪需要打扮得光鲜靓丽,便放任老态,甚至往老气的方向打扮,有的爱美的女性则更偏向于扮嫩结果却适得其反但仍有部分女性在打扮上深谙其道,而她们的状态看上去不比年轻的。
  • 适合亲子看的56部电影(国庆假期宅家里怎么过)
  • 2024-06-30国庆假期宅家里怎么过国庆假期来啦和孩子一起看高分电影少不了好的电影不仅能带来观影的愉悦更能带给我们教育的启发和思考小编为大家推荐近5年来的10部高分电影快和孩子们一起看看这些电影里的精彩故事吧~《寻梦环游记》家庭梦想原谅。
  • 双枪女匪首被执行枪决(双枪女匪首领程莲珍逃亡3年)
  • 2024-06-30双枪女匪首领程莲珍逃亡3年新中国成立初期,贵州有一个双枪女匪首,她叫程莲珍,逃亡3年才被抓获子女被捕后,程莲珍本应被严肃处罚,主席却下急令:不能杀这是为何呢?日理万机的主席,怎么会关注一个贵州地区的女匪首,还亲自下令网开一面,。
  • 正宗烤鱼的做法窍门(在家也能做大厨)
  • 2024-06-30在家也能做大厨大家好,这里是【刘一手美食】,关注老刘,每天分享一道好吃又实用的家常菜1、烤鱼,一种发源于重庆巫溪县,而发扬于万州的特色美食2、今天老刘就来分享一下“铁板烤鱼”做法,喜欢的朋友可以先收藏,有空自己试一。
  • 史料越详细名将越少(青未了古人揭榜挂帅)
  • 2024-06-30青未了古人揭榜挂帅如今,“揭榜挂帅”一词,经常现身新闻、见诸报端,因其具有四字形态、古文语感和联合动宾结构,很容易让人误当作成语但细究起来,这个词最早进入大众视野却是在2016年4月召开的网络安全和信息化工作座谈会上,。
  • 加勒比海盗九位首领(加勒比海十三太保)
  • 2024-06-30加勒比海十三太保安提瓜和巴布达安提瓜和巴布达(英语:AntiguaandBarbuda),简称“安巴”,是中美洲的一个岛国,位于加勒比海和大西洋之间,往南邻近法属瓜德罗普,西南方向靠近英属蒙特塞拉特,西面则为圣基茨和。
  • 如此好玩的方言(有趣的方言个把滴)
  • 2024-06-30有趣的方言个把滴方言,每一个地方都有着自己的区域特点和趣味,也最能体现一方水土和一方文化,以及风俗人情了正因为有着区域性,所说出来,往往会令他乡之客是丈二的和尚摸不着头脑我的老家是萧县的,自然也有着很多方言,有一句很。
  • omakase怎么吃才专业
  • 2024-06-30omakase怎么吃才专业按照日本餐饮礼仪,omakase需以正式、恭敬的姿态品尝明确结论是,要专业地享用omakase,需要注意以下几点首先,要尊重厨师的创意和完成的菜品其次,要遵循日本餐饮礼仪,如使用筷子、过桥等礼仪最重要。
  • 社交恐惧症该怎么克服(教你如何走出社交恐惧症)
  • 2024-06-30教你如何走出社交恐惧症慢跑组社交恐惧症(SocialPhobia),又名社交焦虑症(socialanxiety)、见人恐怖症,是一种对社交或公开场合感到强烈恐惧或忧虑的精神疾病以下是小编为大家整理的有关走出社交恐惧症的内容。
  • 事业运很旺的嘴型面相(事业运很旺的嘴型面相有哪些)
  • 2024-06-30事业运很旺的嘴型面相有哪些厚唇大口:是个精力旺盛,有时像哥儿门,有时又像大姐大自尊心很强,是个慢熟之人在事业上為了某个观点,会据理力争,坚持到底,甚至奔波劳碌,自我本為观念强但是只要交情够,你会為他人赴汤蹈火,在所不辞代表:巩。