利用pythonstudio写的PDF、图片批量水印生成器,可同时为不同读者生成多组水印

现在很多场合需要将PDF或图片加水印,本程序利用pythonstudio编写。

第一步 界面

在这里插入图片描述
其中:

LstMask:列表框

  • PopupMenu:PmnMark

LstFiles:列表框

  • PopupMenu:PmnFiles

OdFiles:文件选择器

  • Filter:PDF文件(.PDF)|.PDF|图像文件(.JPG)|.JPG|图像文件(.png)|.png
  • Option-OfAllowMultiSelection:True

其余一眼都能看出来

代码:

模块1:m_water 用来打水印


import os
import io

from PyPDF2 import PdfWriter, PdfReader
from reportlab.lib import pagesizes  # 页面样式
from reportlab.lib.units import cm
from reportlab.pdfbase import pdfmetrics  # 注册字体
from reportlab.pdfbase.ttfonts import TTFont  # 字体类
from reportlab.pdfgen import canvas

from watermarker.marker import add_mark
from PIL import Image


pdfmetrics.registerFont(TTFont('SimHei', os.path.join(os.path.dirname(os.path.abspath(__file__)), "bird.ttf")))


# 生成水印文件
def create_water_mark(text):
    packet = io.BytesIO()
    # 创建一个带有水印的新PDF页
    my_canvas = canvas.Canvas(packet, pagesizes.A0)
    # 设置水印字体

    my_canvas.setFont("SimHei", 40)
    # 填充色
    my_canvas.setFillColorRGB(0, 0, 0)
    # 透明度
    my_canvas.setFillAlpha(0.1)
    # 设置字体旋转度数
    my_canvas.rotate(15)
    # x轴的3cm处,到24结束,步长是10
    for i in range(3, 24, 10):
        # y轴的
        for j in range(-5, 30, 5):
            my_canvas.drawString(i * cm, j * cm, text)
    my_canvas.save()
    packet.seek(0)
    return PdfReader(packet)


def add_watermark(input_pdf_path, output_pdf_path, watermark_text):
    # 创建水印
    watermark = create_water_mark(watermark_text)

    # 读取输入 PDF
    pdf_reader = PdfReader(input_pdf_path)
    pdf_writer = PdfWriter()

    # 遍历每一页,将水印添加到每一页
    for page in pdf_reader.pages:
        page.merge_page(watermark.pages[0])  # 将水印添加到当前页面
        pdf_writer.add_page(page)

    # 写入到输出 PDF 文件
    with open(output_pdf_path, "wb") as output_pdf:
        pdf_writer.write(output_pdf)


def add_pic_watermark(input_jpg_path,out_jpg_path, watermark_text):
    # 加水印
    add_mark(file=input_jpg_path,out=out_jpg_path,mark=watermark_text,opacity=0.3,angle=30,space=80)
##    add_mark(file=f"{cwd}\page_{i + 1}.jpg" , out=cwd, mark=EPdf.stamp,opacity=EPdf.tran,angle=30,space=80)



def main():
    pass


if __name__ == '__main__':
    main()


模块2:m_zip 用来打包文件

import zipfile
import os

def compress_folder(folder_path, output_path):
    with zipfile.ZipFile(output_path, 'w', zipfile.ZIP_DEFLATED) as zipf:
        for root, dirs, files in os.walk(folder_path):
            for file in files:
                file_path = os.path.join(root, file)
                arc_name = os.path.relpath(file_path, folder_path)
                zipf.write(file_path, arc_name)

def main():
    pass
    compress_folder("111","111.zip")

if __name__ == '__main__':
    main()

Unit1.py 主界面代码

import os
from glcl import *
from m_water import *
from m_zip import *
import shutil

class Form1(Form):

    def __init__(self, owner):
        self.PgbFiles = ProgressBar(self)
        self.LblStat = Label(self)
        self.PmnFiles = PopupMenu(self)
        self.PmnMark = PopupMenu(self)
        self.OdFiles = OpenDialog(self)
        self.BtnAddFiles = Button(self)
        self.Label3 = Label(self)
        self.Label2 = Label(self)
        self.BtnClose = Button(self)
        self.BtnMark = Button(self)
        self.Label1 = Label(self)
        self.EdtMark = Edit(self)
        self.LstFiles = ListBox(self)
        self.LstMark = ListBox(self)
        self.BtnBeginMark = Button(self)
        self.LoadProps(os.path.join(os.path.dirname(os.path.abspath(__file__)), "Unit1.pydfm"))
        self.BtnClose.OnClick = self.BtnCloseClick
        self.BtnBeginMark.OnClick = self.BtnBeginMarkClick
        self.LstFiles.OnClick = self.LstFilesClick
        self.MniFiles.OnClick = self.MniFilesClick
        self.PmnFiles.OnPopup = self.PmnFilesPopup
        self.BtnAddFiles.OnClick = self.BtnAddFilesClick
        self.MniDel.OnClick = self.MniDelClick
        self.PmnMark.OnPopup = self.PmnMarkPopup
        self.BtnMark.OnClick = self.BtnMarkClick




    # 把水印名加入到列表
    def BtnMarkClick(self, Sender):
        # 如果 EDT不为空,则加至列表
        if self.EdtMark.Text.strip()!="":
            # 没有重复
            if self.LstMark.Items.IndexOf(self.EdtMark.Text.strip())>-1:
                ShowMessage("当前水印已存在,请检查后重新输入。")
                return
            self.LstMark.Items.Add(self.EdtMark.Text.strip())
            # 清除EDT
            self.EdtMark.Text=""
        else:
            ShowMessage("请先填写需要添加的水印内容。")


    # 如果水印名有值,就显示删除
    def PmnMarkPopup(self, Sender):
        self.MniDel.Enabled=True if self.LstMark.ItemIndex>-1 else False

    # 如果文件有值,就显示删除
    def PmnFilesPopup(self, Sender):
        self.MniDel.Enabled=True if self.LstFiles.ItemIndex>-1 else False

    # 确认是否要删除水印,然后删除
    def MniDelClick(self, Sender):
        if Application.MessageBox("是否要删除"+ self.LstMark.Items[self.LstMark.ItemIndex]+"?","请确认", MB_YESNO)==IDYES:
            self.LstMark.DeleteSelected()

    # 确认是否要删除文件,然后删除
    def MniFilesClick(self, Sender):
        if Application.MessageBox("是否要删除"+ self.LstFiles.Items[self.LstFiles.ItemIndex]+"?","请确认", MB_YESNO)==IDYES:
            self.LstFiles.DeleteSelected()

    # 选择文件并添加到列表中
    def BtnAddFilesClick(self, Sender):
        if self.OdFiles.Execute():
            for item in self.OdFiles.Files:
                # 如果不重复就添加
                if self.LstFiles.Items.IndexOf(item)==-1:
                    self.LstFiles.Items.Add(item)


    # 显示选中项的提示
    def LstFilesClick(self, Sender):
        self.LstFiles.Hint=self.LstFiles.Items[self.LstFiles.ItemIndex]

    # 循环打水印
    def BtnBeginMarkClick(self, Sender):
        # 如果没有文件或水印,退出
        if self.LstFiles.Count==0 or self.LstMark.Count==0:
            ShowMessage("请确认添加好水印和文件后再进行操作。")
            return
        # 将所有按钮都禁用
        self.BtnAddFiles.Enabled=False
        self.BtnBeginMark.Enabled=False
        self.BtnMark.Enabled=False
        self.BtnClose.Enabled=False

        # 提示需要较长时间
        ShowMessage("共有"+str(self.LstFiles.Count * self.LstMark.Count)+"个文件要打水印并打包,所需时间较长,请确认后开始操作。")
        # 进度条最大值
        self.PgbFiles.max=self.LstFiles.Count * self.LstMark.Count
        self.PgbFiles.Position=0
        self.LblStat.Caption="正在建立文件夹"

        # 建立以水印名为名字的文件夹
        try:
            for folder in self.LstMark.Items:
                if os.path.exists(folder)==False:
                    os.mkdir(folder)
        except:
            ShowMessage("建立文件夹失败")
            return

        self.LblStat.Caption="开始打水印"


        # 循环水印
        for mark in self.LstMark.Items:
            # 循环文件
            for markfile in self.LstFiles.Items:
                # 如果是pdf,打PDF水印,存在水印文件夹
                if markfile[-4:].upper()==".PDF":
                    try:
                        outfilename=os.path.join( mark, markfile[markfile.rfind("\\")+1:])
                        add_watermark(markfile,outfilename,mark)
                    except:
                        ShowMessage(markfile[markfile.rfind("\\")+1:]+"文件建立失败")
                else:
                # 如果是图片,打图片水印,存在水印文件夹下
                    try:
                        add_pic_watermark(markfile,mark,mark)
                    except:
                        ShowMessage(markfile[markfile.rfind("\\")+1:]+"文件建立失败")
                self.PgbFiles.Position+=1
        # 打包进度
        self.PgbFiles.Max=self.LstMark.Count
        self.PgbFiles.Position=0

        # 所有水印打完后,每个水印文件夹打包
        if os.path.exists("output")==False:
            os.mkdir("output")
        self.LblStat.Caption="开始文件打包"
        for folder in self.LstMark.Items:
            compress_folder(folder,"output\\"+folder+".zip")
            self.PgbFiles.Position+=1

        # 删除临时文件
        for folder in self.LstMark.Items:
            shutil.rmtree(folder)


        # 恢复所有按钮
        self.BtnAddFiles.Enabled=True
        self.BtnBeginMark.Enabled=True
        self.BtnMark.Enabled=True
        self.BtnClose.Enabled=True
        self.LblStat.Caption="打水印完成"
        ShowMessage("打水印、打包完成,请在当前文件夹下检查ZIP文件。")
        self.LblStat.Caption=""

        self.LstMark.Clear()
        self.LstFiles.Clear()


    def BtnCloseClick(self, Sender):
        self.Close()






伸手党可直接下载可执行文件
https://download.csdn.net/download/gxchai/89956703

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:/a/912743.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

基于python深度学习技术矩阵分解的推荐系统,通过学习隐含特征,实现推荐

实现了一个基于矩阵分解的推荐系统,用于预测用户对电影的评分。具体来说,该程序通过TensorFlow构建和训练一个模型,来学习用户和电影之间的隐含特征,并根据这些特征预测评分。以下是代码的主要功能和步骤的详细描述: …

[vulnhub] DarkHole: 1

https://www.vulnhub.com/entry/darkhole-1,724/ 端口扫描主机发现 探测存活主机,184是靶机 nmap -sP 192.168.75.0/24 Starting Nmap 7.94SVN ( https://nmap.org ) at 2024-11-08 09:59 CST Nmap scan report for 192.168.75.1 Host is up (0.00027s latency). MA…

4.1 WINDOWS XP,ReactOS对象与对象目录----1

系列文章目录 文章目录 系列文章目录4.1 对象与对象目录OBJECT_HEADERObpLookupEntryDirectory()NtCreateTimer() 4.1 对象与对象目录 “对象(Object)”这个词现在大家都已耳熟能详了,但是对象到底是什么呢?广义地说,对象就是“目标”,行为…

STM32H503开发(2)----STM32CubeProgrammer烧录

STM32H503开发----2.STM32CubeProgrammer烧录 概述硬件准备视频教学样品申请源码下载参考程序自举模式BOOT0设置UART烧录USB烧录 概述 STM32CubeProgrammer (STM32CubeProg) 是一款用于编程STM32产品的全功能多操作系统软件工具。 它提供了一个易用高效的环境,通过…

“双十一”电商狂欢进行时,在AI的加持下看网易云信IM、RTC如何助力商家!

作为一年一度的消费盛会,2024年“双十一”购物狂欢节早已拉开帷幕。蹲守直播间、在主播热情介绍中点开链接并加购,也已成为大多数人打开“双11”的重要方式。然而,在这火热的购物氛围背后,主播频频“翻车”、优质主播稀缺、客服响…

debian系统安装qt的时候 显示xcb相关文件缺失

如果是安装之后的问题 我们可以选择使用ldd的命令查看当前依赖的so那些文件确实 ldd /home/yinsir/Qt/5.15.2/gcc_64/plugins/platforms/libqxcb.so 本人在进行打包的时候 出现则会个报错 ERROR: ldd outputLine: “libxcb-util.so.1 > not found” ERROR: for binary: “/…

A023-基于SpringBoot的冷链物流系统的设计与实现

🙊作者简介:在校研究生,拥有计算机专业的研究生开发团队,分享技术代码帮助学生学习,独立完成自己的网站项目。 代码可以查看文章末尾⬇️联系方式获取,记得注明来意哦~🌹 赠送计算机毕业设计600…

【数据分析】如何构建指标体系?

有哪些指标体系搭建模型?五个步骤教你从0开始搭建指标体系 一、企业指标体系搭建存在什么问题 许多企业在搭建数据指标体系时遇到了诸多难题,如问题定位不准确、数据采集不完整、目标不一致、报表无序、指标覆盖不全面以及报表价值未充分利用等。 1、…

C++20 概念与约束(1)—— SFINAE

1、从模板说起 众所周知,C在使用模板时,如果有多个模板匹配,则编译器会选择最匹配的一个模板进行实例化,这也正是模板特化和偏特化的依据。 根据上面这张图中的现象,列举下面几个示例: 1、不存在模板的情况…

基于Spring Boot的在线装修管理系统的设计与实现,LW+源码+讲解

摘 要 互联网发展至今,无论是其理论还是技术都已经成熟,而且它广泛参与在社会中的方方面面。它让信息都可以通过网络传播,搭配信息管理工具可以很好地为人们提供服务。针对信息管理混乱,出错率高,信息安全性差&#…

原生鸿蒙应用市场:赋能开发者全生命周期服务体验

文章目录 背景自动化检测前移:早发现,早解决技术细节:静态代码分析与兼容性测试应用场景 按需加载:优化性能,提升用户体验技术细节:模块化与懒加载实现应用场景 应用加密:保护应用代码安全&…

RDD 算子全面解析:从基础到进阶与面试要点

Spark 的介绍与搭建:从理论到实践_spark环境搭建-CSDN博客 Spark 的Standalone集群环境安装与测试-CSDN博客 PySpark 本地开发环境搭建与实践-CSDN博客 Spark 程序开发与提交:本地与集群模式全解析-CSDN博客 Spark on YARN:Spark集群模式…

配置多公钥在多平台使用Git

步骤很详细,直接上教程 当我们想在不同远程仓库平台配置不同公钥时会发现不进行额外配置是不行的,只会使用默认的公钥,本篇文章便是为了解决这个问题 进入C:\Users\[你的用户名]\.ssh文件夹 如果没有这个文件夹可以新建一下 在上述文件夹新建…

如何在 Android 上增加 SELinux 权限

SELinux(Security-Enhanced Linux)是一种强制访问控制(MAC)机制,它为 Android 系统提供了额外的安全层。通过 SELinux,系统管理员可以定义细粒度的安全策略,限制进程对文件、网络和其他资源的访…

新能源汽车与公共充电桩布局

近年来,全球范围内对新能源汽车产业的推动力度不断增强,中国新能源汽车市场也呈现蓬勃发展的势头,在政策与市场的共同推动下,新能源汽车销量持续增长。然而,据中国充电联盟数据显示,充电基础设施建设滞后于新能源汽车数量增长的现状导致充电桩供需不平衡,公共充电桩服务空白区域…

中科大:LLM知识遗忘评估与优化

📖标题:A Closer Look at Machine Unlearning for Large Language Models 🌐来源:arXiv, 2410.08109 🌟摘要 🔸大型语言模型(LLM)可能会记住敏感或受版权保护的内容,从…

django+postgresql

PostgreSQL概述 PostgreSQL 是一个功能强大的开源关系数据库管理系统(RDBMS),以其高度的稳定性、扩展性和社区支持而闻名。PostgreSQL 支持 SQL 标准并具有很多先进特性,如 ACID 合规、复杂查询、外键支持、事务处理、表分区、JS…

Flink CEP 入门

1.复杂事件处理 大数据应用领域存在业务逻辑非常复杂的应用系统,比如,一个应用要检测特定顺序先后发生的一组事件,对事件组进行分析或报警提示,若使用SQL 或者DataStream API 处理这类应用,过程相对来说比较…

CSS教程(三)- CSS 三大特性

1. 层叠性 介绍 多组CSS样式共同作用于一个元素,就会出现 覆盖(层叠) 另一个冲突的样式。 层叠原则 样式冲突:遵循就近原则(哪个样式离结构近,就执行哪个样式) 样式不冲突,就不会重…

粒子群优化双向深度学习!PSO-BiTCN-BiGRU-Attention多输入单输出回归预测

粒子群优化双向深度学习!PSO-BiTCN-BiGRU-Attention多输入单输出回归预测 目录 粒子群优化双向深度学习!PSO-BiTCN-BiGRU-Attention多输入单输出回归预测效果一览基本介绍程序设计参考资料 效果一览 基本介绍 1.Matlab实现PSO-BiTCN-BiGRU-Attention粒子…