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

news2025/1/11 12:47:18

现在很多场合需要将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

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

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

PDF模板制作与填充(Java)

1.PDF模板制作 准备原始模板 准备一个原始PDF模板,可以编辑好Word,预留出要填充的部分,再转换成PDF格式。 设置表单域 用任意PDF编辑器打开PDF模板文件,设置表单域,下面以WPS为例: 拖动文本域到需要填充的…

kafka中节点如何服役和退役

服役新节点 1)新节点准备 (1)关闭 bigdata03,进行一个快照,并右键执行克隆操作。 (2)开启 bigdata04,并修改 IP 地址。 vi /etc/sysconfig/network-scripts/ifcfg-ens33修改完记…

笔记本怎么开启TPM2.0_笔记本开启TPM2.0教程(不同笔记本开启tpm2.0方法)

在win11最低要求是提示,电脑必须满足 TPM 2.0,并开需要开启TPM 才能正常安装windows11系统,有很多笔记本的用户问我,笔记本怎么开启tpm功能呢?下面小编就给大家详细介绍一下笔记本开启tpm功能的方法。 如何确认你笔记本…

【PyTorch项目实战】图像分割 —— U-Net:Semantic segmentation with PyTorch

文章目录 一、项目介绍二、项目实战2.1、搭建环境2.1.1、下载源码2.1.2、下载预训练模型2.1.3、下载训练集 2.2、环境配置2.3、模型预测 U-Net是一种用于生物医学图像分割的卷积神经网络架构,最初由Olaf Ronneberger等人于2015年提出。 论文: U-Net: Con…

开源竞争-大数据项目期末考核

开源竞争: 自己没有办法完全掌握技术的时候就开源这个技术,培养出更多的技术依赖,让更多人完善你的技术,那么这不就是在砸罐子吗?一个行业里面总会有人砸罐子的,你不如先砸还能听个想。 客观现实&#xf…

11月7日星期四今日早报简报微语报早读

11月7日星期四,农历十月初七,早报#微语早读。 1、河南:旅行社组织1000人次境外游客在豫住宿2夜以上,可申请激励奖补; 2、主播宣称下播后商品恢复原价构成欺诈,广州市监:罚款5万元;…

HTMLCSS:3D 旋转卡片的炫酷动画

效果演示 这段代码是一个HTML和CSS的组合&#xff0c;用于创建一个具有3D效果的动画卡片。 HTML <div class"obj"><div class"objchild"><span class"inn6"><h3 class"text">我是谁&#xff1f;我在那<…

词嵌入方法(Word Embedding)

词嵌入方法&#xff08;Word Embedding&#xff09; Word Embedding是NLP中的一种技术&#xff0c;通过将单词映射到一个空间向量来表示每个单词 ✨️常见的词嵌入方法&#xff1a; &#x1f31f;Word2Vec&#xff1a;由谷歌提出的方法&#xff0c;分为CBOW&#xff08;conti…

2024下半年系统架构师考试【回忆版】

2024年11月10日&#xff0c;系统架构师考试如期举行&#xff0c;屡战屡败的参试倒是把北京的学校转了好几所。 本次考试时间 考试科目考试时间综合知识、案例分析8:30 - 12:30论文14:30 - 16:30 案例分析 1、RESTful 对于前后端的优势&#xff1b; 2、心跳相对于ping/echo的…

最简单解决NET程序员在centos系统安装c#网站

目前随着技术栈转移&#xff0c;c#程序员如何在linux服务器中部署net程序呢&#xff1f; 我做了一次实验&#xff1a;一般来说runtime和sdk都要装。 1.centos系统内命令行输入命令 sudo yum install dotnet-sdk-6.0 安装6.0版 2.检测下是否成功&#xff1a;dotnet --versio…

参数估计理论

估计理论的主要任务是在某种信号假设下&#xff0c;估算该信号中某个参数&#xff08;比如幅度、相位、达到时间&#xff09;的具体取值。 参数估计&#xff1a;先假定研究的问题具有某种数学模型&#xff0c; 如正态分布&#xff0c;二项分布&#xff0c;再用已知类别的学习样…

java多线程stop() 和 suspend() 方法为何不推荐使用?

大家好&#xff0c;我是锋哥。今天分享关于【java多线程stop() 和 suspend() 方法为何不推荐使用&#xff1f;】面试题。希望对大家有帮助&#xff1b; java多线程stop() 和 suspend() 方法为何不推荐使用&#xff1f; 1000道 互联网大厂Java工程师 精选面试题-Java资源分享网…

嵌入式硬件电子电路设计(三)电源电路之负电源

引言&#xff1a;在对信号线性度放大要求非常高的应用需要使用双电源运放&#xff0c;比如高精度测量仪器、仪表等;那么就需要给双电源运放提供正负电源。 目录 负电源电路原理 负电源的作用 如何产生负电源 负电源能作功吗&#xff1f; 地的理解 负电压产生电路 BUCK电…

C++高级编程(8)

八、标准IO库 1.输入输出流类 1)非格式化输入输出 2)put #include <iostream> #include <string> ​ using namespace std; int main() {string str "123456789";for (int i str.length() - 1; i > 0; i--) {cout.put(str[i]); //从最后一个字符开…

Python 分子图分类,GNN Model for HIV Molecules Classification,HIV 分子图分类模型;整图分类问题,代码实战

一、分子图 分子图&#xff08;molecular graph&#xff09;是一种用来表示分子结构的图形方式&#xff0c;其中原子被表示为节点&#xff08;vertices&#xff09;&#xff0c;化学键被表示为边&#xff08;edges&#xff09;。对于HIV&#xff08;人类免疫缺陷病毒&#xff…

如何调整pdf的页面尺寸

用福昕阅读器打开pdf&#xff0c;进入打印页面&#xff0c;选择“属性”&#xff0c;在弹出的页面选择“高级” 选择你想调成的纸张尺寸&#xff0c;然后打印&#xff0c;打印出来的pdf就是调整尺寸后的pdf

查缺补漏----用户上网过程(HTTP,DNS与ARP)

&#xff08;1&#xff09;HTTP 来自湖科大计算机网络微课堂&#xff1a; ① HTTP/1.0采用非持续连接方式。在该方式下&#xff0c;每次浏览器要请求一个文件都要与服务器建立TCP连接当收到响应后就立即关闭连接。 每请求一个文档就要有两倍的RTT的开销。若一个网页上有很多引…

koa、vue安装与使用

koa官网&#xff1a;https://koajs.com/ 首选创建一个文件夹&#xff1a;mkdir koaDemo (cmd即可) 文件夹初始化&#xff1a;npm init (cmd即可) 初始化完成后就会产生一个package.json的文件。 安装&#xff1a; npm install koa --save (vscode的控制台中安装&a…

Linux:版本控制器git的简单使用+gdb/cgdb调试器的使用

一&#xff0c;版本控制器git 1.1概念 为了能够更方便我们管理不同版本的文件&#xff0c;便有了版本控制器。所谓的版本控制器&#xff0c;就是能让你 了解到⼀个文件的历史&#xff0c;以及它的发展过程的系统。通俗的讲就是⼀个可以记录工程的每⼀次改动和版本迭代的⼀个…

ML 系列:第 21 节 — 离散概率分布(二项分布)

一、说明 二项分布描述了在固定数量的独立伯努利试验中一定数量的成功的概率&#xff0c;其中每个试验只有两种可能的结果&#xff08;通常标记为成功和失败&#xff09;。 二、探讨伯努利模型 例如&#xff0c;假设您正在抛一枚公平的硬币 &#xff08;其中正面成功&#xff…