百度360必应搜狗淘宝本站头条
当前位置:网站首页 > IT知识 > 正文

CodeBuddy实现图片水印添加工具(代码加水印)

liuian 2025-06-13 14:46 5 浏览

如今,电商平台竞争激烈,辛苦拍摄的商品图极易被同行盗用。使用在线水印工具上传图片时,存在数据泄露风险,且第三方服务器存储的图片也可能被不法分子获取,导致商品图被盗用,辛苦打造的商品形象和销量被他人不劳而获。

电商日常运营中,上架新品、更新商品图时,往往需要处理大量图片。多数在线水印工具免费版无法满足批量处理需求,付费功能成本又高;而一些本地软件操作复杂,难以快速上手,大量时间浪费在图片处理上,严重影响商品上新速度。

在线水印工具依赖网络,当网络不稳定或处于无网络环境(如仓库盘点、外出拍摄商品图)时,无法及时给商品图添加水印,影响商品展示进度,甚至错过销售时机。

大部分还是要付费的

所以就想着实现一款本地的水印添加工具

使用本地工具添加水印时,所有的图片数据都在你的设备内处理,无需上传到外部服务器。这就极大地降低了图片信息泄露的风险,能更好地保障个人隐私和敏感信息。

本地工具无需依赖网络连接,无论你是处于离线状态,还是网络信号不佳的环境中,都可以随时对图片添加水印,操作更加灵活高效。

你能够依据自身的具体需求,自由地对水印的字体、颜色、大小、透明度以及位置等参数进行调整,从而实现独一无二的水印效果。

对于有大量图片需要添加水印的情况,本地工具支持批量操作,可一次性为多张图片添加相同或不同的水印,大大提高工作效率。

## 使用CodeBuddy开发的原因

CodeBuddy 可直接将你的需求描述(如 “添加批量图片处理功能”“实现水印透明度调节滑块”)转化为代码,无需手动编写复杂逻辑。例如,你只需说 “我需要一个界面让用户选择水印位置(上 / 中 / 下 / 左 / 右 / 居中)”,AI 就能自动生成对应的交互组件和位置计算代码。

配置或编写少量代码时,AI 会实时补全后续逻辑,减少重复劳动。比如输入 `for image in images:` 后,AI 可能自动补全图片遍历、水印添加和保存的完整代码块。

- CodeBuddy 的 AI 可根据你的需求生成直观的图形界面,如文件选择器、水印预览区、进度条等,且自动适配不同屏幕尺寸(PC 端、移动端)。

- **组件智能推荐**:当你需要实现特定功能(如 “水印位置选择”)时,AI 会推荐最合适的组件(如九宫格定位按钮),并自动完成组件间的逻辑连接。

## 使用CodeBuddy进行实操

我们打开vscode进入到拓展中直接搜索CodeBuddy,点击安装这个插件就行了

我们先在chat模式生成出我们需要的对应的README文件

对ai说:我想实现一款本地加水印的工具,请你帮我生成下README文件

然后在Craft模式让ai根据文件进行开发

下面来测试下效果

可以进行灵活的水印添加操作,并且可以实时预览效果

代码如下,感兴趣的可以试试

```Python

import os

import tkinter as tk

from tkinter import filedialog, messagebox, ttk

from PIL import Image, ImageTk

import time

def add_watermark(source_image_path, watermark_path, output_folder_path, position, opacity, watermark_scale_percent):

try:

# 获取原始图片文件名(包含扩展名)

source_image_name = os.path.basename(source_image_path)

file_name, file_extension = os.path.splitext(source_image_name)

# 构建输出文件名,在原始文件名后添加_时间戳后缀

# 获取当前时间的时间元组

current_time = time.localtime()

# 将时间元组格式化为字符串,比如 20241202103000 这样的格式(表示2024年12月2日10时30分0秒)

time_str = time.strftime("%Y%m%d%H%M%S", current_time)

output_image_name = file_name + "_" + time_str + file_extension

# 构建完整的输出文件路径

output_path = os.path.join(output_folder_path, output_image_name)

# 打开原始图片

image = Image.open(source_image_path)

image_width, image_height = image.size

# 打开水印图片并确保其为RGBA模式

watermark_image = Image.open(watermark_path)

if watermark_image.mode!= 'RGBA':

watermark_image = watermark_image.convert('RGBA')

watermark_width, watermark_height = watermark_image.size

scale_percent = watermark_scale_percent / 100 # 将百分比转换为小数

new_watermark_width = int(watermark_width * scale_percent)

new_watermark_height = int(watermark_height * scale_percent)

resized_watermark = watermark_image.resize((new_watermark_width, new_watermark_height))

print(f"水印图片已缩放到: {new_watermark_width}x{new_watermark_height} 像素")

# 根据选择的位置调整水印位置

if position == "左上":

paste_position = (0, 0)

elif position == "右上":

paste_position = (image_width - new_watermark_width, 0)

elif position == "左下":

paste_position = (0, image_height - new_watermark_height)

elif position == "右下":

paste_position = (image_width - new_watermark_width, image_height - new_watermark_height)

else:

paste_position = (0, 0)

# 处理透明度

try:

if opacity < 100:

alpha = resized_watermark.split()[3]

alpha = alpha.point(lambda p: p * opacity // 100)

resized_watermark.putalpha(alpha)

except Exception as e:

print("Error in transparency processing:", e)

messagebox.showerror("透明度处理错误", "在处理透明度时出现错误,请检查水印图像。")

# 添加水印

image.paste(resized_watermark, paste_position, resized_watermark)

# 保存结果图片

image.save(output_path, quality=85) # 压缩质量为85%

print(f"保存添加水印后的图片到: {output_path}")

messagebox.showinfo("完成", "水印已成功添加!")

except Exception as e:

print("Error in adding watermark:", e)

messagebox.showerror("错误", "发生错误: " + str(e))

def browse_source_image():

file_selected = filedialog.askopenfilename()

if file_selected:

source_image_entry.delete(0, tk.END)

source_image_entry.insert(0, file_selected)

apply_watermark_preview()

def browse_watermark_image():

file_selected = filedialog.askopenfilename()

if file_selected:

watermark_entry.delete(0, tk.END)

watermark_entry.insert(0, file_selected)

watermark_image = preprocess_watermark_image(file_selected)

if watermark_image:

apply_watermark_preview()

def browse_output_image():

folder_selected = filedialog.askdirectory()

if folder_selected:

output_entry.delete(0, tk.END)

output_entry.insert(0, folder_selected)

def add_watermark_process():

source_image_path = source_image_entry.get()

watermark_path = watermark_entry.get()

output_folder_path = output_entry.get()

position = position_var.get()

opacity = opacity_var.get().rstrip('%')

opacity = int(opacity)

watermark_scale_percent = scale_var.get().rstrip('%')

watermark_scale_percent = int(watermark_scale_percent)

if not source_image_path or not watermark_path or not output_folder_path:

messagebox.showwarning("警告", "请确保所有字段都已填写。")

return

add_watermark(source_image_path, watermark_path, output_folder_path, position, opacity, watermark_scale_percent)

def apply_watermark_preview():

source_image_path = source_image_entry.get()

watermark_path = watermark_entry.get()

position = position_var.get()

opacity = opacity_var.get().rstrip('%')

opacity = int(opacity)

watermark_scale_percent = scale_var.get().rstrip('%')

watermark_scale_percent = int(watermark_scale_percent)

if not source_image_path or not watermark_path:

return

# 创建加载弹窗

loading_window = tk.Toplevel(root)

loading_window.title("加载中")

loading_window.geometry("50x50")

loading_label = tk.Label(loading_window, text="正在生成水印预览,请稍候...")

loading_label.pack(pady=10)

progress_var = tk.IntVar()

progress_bar = ttk.Progressbar(loading_window, orient="horizontal", length=150, mode="indeterminate",variable=progress_var)

progress_bar.pack(pady=5)

progress_bar.start(10) # 启动进度条动画,这里设置每10毫秒更新一次

root.update_idletasks() # 立即更新界面,确保加载弹窗显示出来

try:

# 打开原始图片

source_image = Image.open(source_image_path)

source_width, source_height = source_image.size

# 打开水印图片并确保其为RGBA模式

watermark_image = Image.open(watermark_path)

if watermark_image.mode!= 'RGBA':

watermark_image = watermark_image.convert('RGBA')

watermark_width, watermark_height = watermark_image.size

scale_percent = watermark_scale_percent / 100

new_watermark_width = int(watermark_width * scale_percent)

new_watermark_height = int(watermark_height * scale_percent)

resized_watermark = watermark_image.resize((new_watermark_width, new_watermark_height))

# 处理透明度

try:

if opacity < 100:

alpha = resized_watermark.split()[3]

alpha = alpha.point(lambda p: p * opacity // 100)

resized_watermark.putalpha(alpha)

except Exception as e:

print("Error in transparency processing:", e)

messagebox.showerror("透明度处理错误", "在处理透明度时出现错误,请检查水印图像。")

# 根据选择的位置调整水印位置

if position == "左上":

paste_position = (0, 0)

elif position == "右上":

paste_position = (source_width - new_watermark_width, 0)

elif position == "左下":

paste_position = (0, source_height - new_watermark_height)

elif position == "右下":

paste_position = (source_width - new_watermark_width, source_height - new_watermark_height)

else:

paste_position = (0, 0)

# 创建带有水印的预览图像

preview_image = source_image.copy()

preview_image.paste(resized_watermark, paste_position, resized_watermark)

# 设置固定的预览框尺寸

target_width = 300

target_height = 250

# 获取原始图片宽高比

width_ratio = preview_image.width / target_width

height_ratio = preview_image.height / target_height

# 根据宽高比计算缩放后的尺寸,保持图片比例不变

if width_ratio > height_ratio:

new_width = target_width

new_height = int(preview_image.height / width_ratio)

else:

new_height = target_height

new_width = int(preview_image.width / height_ratio)

# 缩放图片

preview_image = preview_image.resize((new_width, new_height), Image.Resampling.LANCZOS)

# 计算在预览框中居中显示的偏移量

offset_x = (target_width - new_width) // 2

offset_y = (target_height - new_height) // 2

# 创建一个新的白色背景的图片(尺寸为预览框大小)

final_preview_image = Image.new('RGB', (target_width, target_height), (255, 255, 255))

final_preview_image.paste(preview_image, (offset_x, offset_y))

# 转换为Tkinter可以显示的格式

preview_photo = ImageTk.PhotoImage(final_preview_image)

preview_canvas.create_image(0, 0, anchor='nw', image=preview_photo)

preview_canvas.image = preview_photo

# 关闭加载弹窗

loading_window.destroy()

except Exception as e:

print("Error in preview:", e)

messagebox.showerror("预览错误", "无法显示水印预览: " + str(e))

# 如果出现错误也关闭加载弹窗

loading_window.destroy()

def on_position_change(event):

apply_watermark_preview()

def on_opacity_change(event):

apply_watermark_preview()

def on_scale_change(event):

apply_watermark_preview()

def preprocess_watermark_image(watermark_path):

try:

watermark_image = Image.open(watermark_path)

if watermark_image.mode!= 'RGBA':

watermark_image = watermark_image.convert('RGBA')

# 可以添加更多的预处理操作,如检查图像数据完整性等

return watermark_image

except Exception as e:

print("Error in preprocessing watermark image:", e)

messagebox.showerror("水印图像预处理错误", "在处理水印图像时出现错误,请选择正确的水印图像。")

return None

def about():

messagebox.showinfo("关于", "这是一个简单的用python制作的添加水印工具。\n版本: 1.0")

def exit_app():

root.quit()

if __name__ == "__main__":

# 创建主窗口

root = tk.Tk()

root.title("图片水印添加工具")

root.geometry("500x700")

root.resizable(False, False)

# 设置主窗口居中

root.update_idletasks()

window_width = root.winfo_width()

window_height = root.winfo_height()

screen_width = root.winfo_screenwidth()

screen_height = root.winfo_screenheight()

x = (screen_width - window_width) // 2

y = (screen_height - window_height) // 2

root.geometry(f"{window_width}x{window_height}+{x}+{y}")

# 待加水印图片选择

source_image_frame = tk.Frame(root)

source_image_frame.pack(pady=10, padx=20, fill="x")

source_image_label = tk.Label(source_image_frame, text="待加水印图片:")

source_image_label.pack(side="left")

source_image_entry = tk.Entry(source_image_frame, width=40)

source_image_entry.pack(side="left", padx=5)


source_image_browse_button = tk.Button(source_image_frame, text="浏览", command=browse_source_image)

source_image_browse_button.pack(side="left")

# 水印图片选择

watermark_frame = tk.Frame(root)

watermark_frame.pack(pady=10, padx=20, fill="x")

watermark_label = tk.Label(watermark_frame, text="水印图片:")

watermark_label.pack(side="left")

watermark_entry = tk.Entry(watermark_frame, width=40)

watermark_entry.pack(side="left", padx=5)

watermark_browse_button = tk.Button(watermark_frame, text="浏览", command=browse_watermark_image)

watermark_browse_button.pack(side="left")

# 输出图片选择

output_frame = tk.Frame(root)

output_frame.pack(pady=10, padx=20, fill="x")

output_label = tk.Label(output_frame, text="输出文件夹:")

output_label.pack(side="left")

output_entry = tk.Entry(output_frame, width=40)

output_entry.pack(side="left", padx=5)

output_browse_button = tk.Button(output_frame, text="浏览", command=browse_output_image)

output_browse_button.pack(side="left")

# 水印位置选择

position_frame = tk.Frame(root)

position_frame.pack(pady=10, padx=20, fill="x")

position_label = tk.Label(position_frame, text="水印位置:")

position_label.pack(side="left")

position_var = tk.StringVar()

position_option = ttk.Combobox(

position_frame,

textvariable=position_var,

values=["左上", "右上", "左下", "右下"],

width=10,

state="readonly",

)

position_option.current(0)

position_option.pack(side="left", padx=5)

position_option.bind("<<ComboboxSelected>>", on_position_change)

# 水印透明度选择

opacity_frame = tk.Frame(root)

opacity_frame.pack(pady=10, padx=20, fill="x")

opacity_label = tk.Label(opacity_frame, text="水印透明度:")

opacity_label.pack(side="left")

opacity_var = tk.StringVar()

opacity_option = ttk.Combobox(

opacity_frame,

textvariable=opacity_var,

values=["0%", "10%", "20%", "30%", "40%", "50%", "60%", "70%", "80%", "90%", "100%"],

width=10,

state="readonly",

)

opacity_option.current(10) # 默认选择100%

opacity_option.pack(side="left", padx=5)

opacity_option.bind("<<ComboboxSelected>>", on_opacity_change)

# 水印缩放比例选择

scale_frame = tk.Frame(root)

scale_frame.pack(pady=10, padx=20, fill="x")

scale_label = tk.Label(scale_frame, text="水印缩放比例:")

scale_label.pack(side="left")

scale_var = tk.StringVar()

scale_option = ttk.Combobox(

scale_frame,

textvariable=scale_var,

values=["3%", "5%", "10%", "20%", "30%", "40%", "50%", "60%", "70%", "80%", "90%", "100%", "150%", "200%", "250%", "300%"],

width=10,

state="readonly",

)

scale_option.current(2) # 默认选择10%

scale_option.pack(side="left", padx=5)

scale_option.bind("<<ComboboxSelected>>", on_scale_change)

# 添加进度条

progress_frame = tk.Frame(root)

progress_frame.pack(pady=10, padx=20, fill="x")

progress_label = tk.Label(progress_frame, text="处理进度:")

progress_label.pack(side="left")

progress_var = tk.IntVar()

progress_bar = ttk.Progressbar(

progress_frame, orient="horizontal", length=300, mode="determinate", variable=progress_var

)

progress_bar.pack(side="left", padx=5)

# 添加水印预览区域

preview_frame = tk.Frame(root)

preview_frame.pack(pady=10, padx=20, fill="x")

preview_label = tk.Label(preview_frame, text="水印预览:")

preview_label.pack(side="left")

preview_canvas = tk.Canvas(preview_frame, width=300, height=250, bg="white")

preview_canvas.pack(side="left", padx=5)

# 开始添加水印按钮

start_button = tk.Button(root, text="开始添加水印", command=add_watermark_process)

start_button.pack(pady=20)

# 创建菜单栏

menu_bar = tk.Menu(root)

# 文件菜单

"""

file_menu = tk.Menu(menu_bar, tearoff=0)

file_menu.add_command(label="打开待加水印图片", command=browse_source_image)

file_menu.add_command(label="选择水印图片", command=browse_watermark_image)

file_menu.add_command(label="选择输出文件夹", command=browse_output_image)

file_menu.add_separator()

file_menu.add_command(label="退出", command=exit_app)

menu_bar.add_cascade(label="文件", menu=file_menu)

"""

# 关于菜单

about_menu = tk.Menu(menu_bar, tearoff=0)

about_menu.add_command(label="关于本工具", command=about)

menu_bar.add_cascade(label="关于", menu=about_menu)

# 将菜单栏配置到根窗口上

root.config(menu=menu_bar)

# 运行主循环

root.mainloop()

```

## 总结

CodeBuddy 的 **AI 编程能力** 让电商从业者无需成为技术专家,就能快速开发出功能强大、安全可靠的本地水印工具。AI 不仅能自动实现核心功能,还能根据电商场景特点进行优化,大幅缩短开发周期,降低技术门槛,让你专注于业务价值而非技术实现。

相关推荐

办公小技巧:定时提醒不慌张 Excel制作智能提醒器

平时工作或学习中总有许多事情需要提醒,比如参加高考的日子、女友的生日、每周例会等。我们可以利用Excel的函数制作实用的智能提醒器,就不用担心忽略这些重要的日子了。每周例会轻松提醒公司规定每个月的最后...

使用vba将Excel 文件转成 CSV 文件

使用vba将Excel中的内容写入文本文件,保存为txt或csv格式。方法一:SubtoCSV()DimshtAsWorksheetDimiAsInteger,j...

用Excel编写俄罗斯方块(vb做俄罗斯方块)

看到这个题目,想必你一定会感到非常惊讶,什么,Excel居然能开发游戏?没错,Excel的强大取决于使用者,遇强则强,遇弱则弱。但我这篇文章并不是为了展示Excel使用过程中的奇技淫巧,而是主要写给那...

该死的手抄滚开(该死的手抄滚开表情包)

碰到形式主义惯犯或硬拔存在感的,顶不住要手抄怎么办?电脑时代这一切都不是问题,除了纸张背面少点中性笔划下的凸痕。网络上有大量的手写体,比较大众化(就是丑)的有李国夫手写体、陈静的字体、萌妹子等,我们也...

Excel二维表转换一维表,三种方法一网打尽!

今天高顿网校小编向大家分享二维表格转一维表的三种方法,分别用到函数、数据透视表和VBA代码。三种方法各有利弊,表亲可以自行选择。如下图,A1:E5是数据源,A7:C22是最终要实现的结果样式:一起动手...

PMC用到的所有函数,大咖总结,实用干货,PMC必备神器!

PMC用到的所有函数你和PMC大咖之间的差异,就是这些Excel函数!1.PMC日常用到的所有函数如下,大咖总结,实用干货,PMC必备神器!每个函数都有实例讲解,不怕你不会。序号PMC工作需求EXCE...

Excel VBA应用-13:统计业务员业绩,目标完成率分析表

在评价业务员销售业绩时,往往会给业务员设定销售目标,根据实际业务计算业务员的目标完成率。报表格式如下图:要计算目标完成率,首先要有销售目标的数据,可以在Excel表中建立一个销售目标表,这种方式的好处...

VBA高级应用30例应用2实现在列表框内及列表框间实现数据拖动

《VBA高级应用30例》(版权10178985),是我推出的第十套教程,教程是专门针对高级学员在学习VBA过程中提高路途上的案例展开,这套教程案例与理论结合,紧贴“实战”,并做“战术总结”,以便大家能...

技术分析:一款流行的VBA宏病毒(vba宏是什么)

1.通过邮件传播的宏病毒近期流行的一个宏病毒通过邮件进行传播,捕捉到的一个样本,其邮件头如下:邮件的内容是这样子的(为节省篇幅,省略号处省略部分内容):YourbillsummaryAccount...

Excel规划求解怎么用?最简单的3*3不同数字填充技...

Excel有什么样的功能,它到底有多强大?Excel表格中有函数公式、图表、操作技巧、vba操作等不同类型的使用方法,但还有一种更加重要的用法就是数据分析,需要用到的就是我们的Excel规划求解。规划...

excel vba vb.net考勤时间处理通用方法(2)

接上期:excelvbavb.net考勤时间处理通用方法(1)DimstartTicksAsIntegerstartTicks=My.Computer.Clock.Ti...

aardio + VBA ( Excel ) 快速开发,3 分钟可入门

aardio很小,下载就可以使用,没有任何复杂步骤。aardio与ExcelVBA(兼容WPSJSA)可以相互调用函数,不需要任何复杂的封装。3分钟够用,下面开始:一、aardio...

VBA+SQL实战:根据时间智能匹配比赛场次,获取最近3场比赛数据

“每天有大量比赛数据,如何快速判断某时间属于哪一场比赛,并自动获取最近3场记录?”本文介绍VBA和SQL双方案,实现:自动匹配时间所属场次(如“第一场/第二场/第三场”)智能返回最近3场比赛数据(跨...

Excel VBA 每天一段代码:自定义分页函数

3月12日视频《ExcelVBA网友答疑:ActiveX控件实现数据分页、翻页》中,我自定义了函数窗体数据查询(),现将代码公布如下:参数说明:1、ItemNum每页显示的数据数量(由复合框下...

VBA布尔矩阵筛选在财会场景的六大高阶应用

基于内存位运算与矩阵预编译技术,布尔矩阵可大幅提升财务数据处理效率。以下是针对2025年财务自动化需求的创新实践案例,覆盖审计、税务、合并报表等核心场景:1.多维度税务异常检测场景:在500万条交易...