CodeBuddy实现图片水印添加工具(代码加水印)
liuian 2025-06-13 14:46 73 浏览
如今,电商平台竞争激烈,辛苦拍摄的商品图极易被同行盗用。使用在线水印工具上传图片时,存在数据泄露风险,且第三方服务器存储的图片也可能被不法分子获取,导致商品图被盗用,辛苦打造的商品形象和销量被他人不劳而获。
电商日常运营中,上架新品、更新商品图时,往往需要处理大量图片。多数在线水印工具免费版无法满足批量处理需求,付费功能成本又高;而一些本地软件操作复杂,难以快速上手,大量时间浪费在图片处理上,严重影响商品上新速度。
在线水印工具依赖网络,当网络不稳定或处于无网络环境(如仓库盘点、外出拍摄商品图)时,无法及时给商品图添加水印,影响商品展示进度,甚至错过销售时机。
大部分还是要付费的
所以就想着实现一款本地的水印添加工具
使用本地工具添加水印时,所有的图片数据都在你的设备内处理,无需上传到外部服务器。这就极大地降低了图片信息泄露的风险,能更好地保障个人隐私和敏感信息。
本地工具无需依赖网络连接,无论你是处于离线状态,还是网络信号不佳的环境中,都可以随时对图片添加水印,操作更加灵活高效。
你能够依据自身的具体需求,自由地对水印的字体、颜色、大小、透明度以及位置等参数进行调整,从而实现独一无二的水印效果。
对于有大量图片需要添加水印的情况,本地工具支持批量操作,可一次性为多张图片添加相同或不同的水印,大大提高工作效率。
## 使用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 不仅能自动实现核心功能,还能根据电商场景特点进行优化,大幅缩短开发周期,降低技术门槛,让你专注于业务价值而非技术实现。
相关推荐
-
- 驱动网卡(怎么从新驱动网卡)
-
网卡一般是指为电脑主机提供有线无线网络功能的适配器。而网卡驱动指的就是电脑连接识别这些网卡型号的桥梁。网卡只有打上了网卡驱动才能正常使用。并不是说所有的网卡一插到电脑上面就能进行数据传输了,他都需要里面芯片组的驱动文件才能支持他进行数据传输...
-
2026-01-30 00:37 liuian
- win10更新助手装系统(微软win10更新助手)
-
1、点击首页“系统升级”的按钮,给出弹框,告诉用户需要上传IMEI码才能使用升级服务。同时给出同意和取消按钮。华为手机助手2、点击同意,则进入到“系统升级”功能华为手机助手华为手机助手3、在检测界面,...
- windows11专业版密钥最新(windows11专业版激活码永久)
-
Windows11专业版的正版密钥,我们是对windows的激活所必备的工具。该密钥我们可以通过微软商城或者通过计算机的硬件供应商去购买获得。获得了windows11专业版的正版密钥后,我...
-
- 手机删过的软件恢复(手机删除过的软件怎么恢复)
-
操作步骤:1、首先,我们需要先打开手机。然后在许多图标中找到带有[文件管理]文本的图标,然后单击“文件管理”进入页面。2、进入页面后,我们将在顶部看到一行文本:手机,最新信息,文档,视频,图片,音乐,收藏,最后是我们正在寻找的[更多],单击...
-
2026-01-29 23:55 liuian
- 一键ghost手动备份系统步骤(一键ghost 备份)
-
步骤1、首先把装有一键GHOST装系统的U盘插在电脑上,然后打开电脑马上按F2或DEL键入BIOS界面,然后就选择BOOT打USDHDD模式选择好,然后按F10键保存,电脑就会马上重启。 步骤...
- 怎么创建局域网(怎么创建局域网打游戏)
-
1、购买路由器一台。进入路由器把dhcp功能打开 2、购买一台交换机。从路由器lan端口拉出一条网线查到交换机的任意一个端口上。 3、两台以上电脑。从交换机任意端口拉出网线插到电脑上(电脑设置...
- 精灵驱动器官方下载(精灵驱动手机版下载)
-
是的。驱动精灵是一款集驱动管理和硬件检测于一体的、专业级的驱动管理和维护工具。驱动精灵为用户提供驱动备份、恢复、安装、删除、在线更新等实用功能。1、全新驱动精灵2012引擎,大幅提升硬件和驱动辨识能力...
- 一键还原系统步骤(一键还原系统有哪些)
-
1、首先需要下载安装一下Windows一键还原程序,在安装程序窗口中,点击“下一步”,弹出“用户许可协议”窗口,选择“我同意该许可协议的条款”,并点击“下一步”。 2、在弹出的“准备安装”窗口中,可...
- 电脑加速器哪个好(电脑加速器哪款好)
-
我认为pp加速器最好用,飞速土豆太懒,急速酷六根本不工作。pp加速器什么网页都加速,太任劳任怨了!以上是个人观点,具体性能请自己试。ps:我家电脑性能很好。迅游加速盒子是可以加速电脑的。因为有过之...
- 任何u盘都可以做启动盘吗(u盘必须做成启动盘才能装系统吗)
-
是的,需要注意,U盘的大小要在4G以上,最好是8G以上,因为启动盘里面需要装系统,内存小的话,不能用来安装系统。内存卡或者U盘或者移动硬盘都可以用来做启动盘安装系统。普通的U盘就可以,不过最好U盘...
- u盘怎么恢复文件(u盘文件恢复的方法)
-
开360安全卫士,点击上面的“功能大全”。点击文件恢复然后点击“数据”下的“文件恢复”功能。选择驱动接着选择需要恢复的驱动,选择接入的U盘。点击开始扫描选好就点击中间的“开始扫描”,开始扫描U盘数据。...
- 系统虚拟内存太低怎么办(系统虚拟内存占用过高什么原因)
-
1.检查系统虚拟内存使用情况,如果发现有大量的空闲内存,可以尝试释放一些不必要的进程,以释放内存空间。2.如果系统虚拟内存使用率较高,可以尝试增加系统虚拟内存的大小,以便更多的应用程序可以使用更多...
-
- 剪贴板权限设置方法(剪贴板访问权限)
-
1、首先打开iphone手机,触碰并按住单词或图像直到显示选择选项。2、其次,然后选取“拷贝”或“剪贴板”。3、勾选需要的“权限”,最后选择开启,即可完成苹果剪贴板权限设置。仅参考1.打开苹果手机设置按钮,点击【通用】。2.点击【键盘】,再...
-
2026-01-29 21:37 liuian
- 平板系统重装大师(平板重装win系统)
-
如果你的平板开不了机,但可以连接上电脑,那就能好办,楼主下载安装个平板刷机王到你的个人电脑上,然后连接你的平板,平板刷机王会自动识别你的平板,平板刷机王上有你平板的我刷机包,楼主点击下载一个,下载完成...
- 联想官网售后服务网点(联想官网售后服务热线)
-
联想3c服务中心是联想旗下的官方售后,是基于互联网O2O模式开发的全新服务平台。可以为终端用户提供多品牌手机、电脑以及其他3C类产品的维修、保养和保险服务。根据客户需求层次,联想服务针对个人及家庭客户...
- 一周热门
-
-
用什么工具在Win中查看8G大的log文件?
-
如何在 Windows 10 或 11 上通过命令行安装 Node.js 和 NPM
-
Trae IDE 如何与 GitHub 无缝对接?
-
如何修改图片拍摄日期?快速修改图片拍摄日期的6种方法
-
5步搞定动态考勤表!标记节假日、调休日?Excel自动变色!
-
RK3588-HDMIRX(瑞芯微rk3588芯片手册)
-
用纯Python轻松构建Web UI:Remi 动态更新,实时刷新界面内容
-
tplink无线路由器桥接教程(tplink路由器如何进行无线桥接)
-
都说Feign是RPC,没有侵入性,为什么我的代码越来越像 C++
-
R语言 | CNS绘图第1款——linkET万物皆可连
-
- 最近发表
- 标签列表
-
- python判断字典是否为空 (50)
- crontab每周一执行 (48)
- aes和des区别 (43)
- bash脚本和shell脚本的区别 (35)
- canvas库 (33)
- dataframe筛选满足条件的行 (35)
- gitlab日志 (33)
- lua xpcall (36)
- blob转json (33)
- python判断是否在列表中 (34)
- python html转pdf (36)
- 安装指定版本npm (37)
- idea搜索jar包内容 (33)
- css鼠标悬停出现隐藏的文字 (34)
- linux nacos启动命令 (33)
- gitlab 日志 (36)
- adb pull (37)
- python判断元素在不在列表里 (34)
- python 字典删除元素 (34)
- vscode切换git分支 (35)
- python bytes转16进制 (35)
- grep前后几行 (34)
- hashmap转list (35)
- c++ 字符串查找 (35)
- mysql刷新权限 (34)
