1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
|
import tkinter as tk
from tkinter import filedialog, messagebox, Scale
from PIL import Image, ImageTk, ImageFilter, ImageDraw
import numpy as np
class BlurRegionGUI:
"""可视化模糊区域选择 GUI"""
def __init__(self, root):
self.root = root
self.root.title("图片区域模糊工具")
self.root.geometry("1200x800")
self.image_path = None
self.original_image = None
self.display_image = None
self.regions = [] # [(x1, y1, x2, y2), ...]
self.current_region = None
self.start_x = None
self.start_y = None
self.blur_radius = 15
self.setup_ui()
def setup_ui(self):
"""设置UI界面"""
# 顶部按钮栏
button_frame = tk.Frame(self.root)
button_frame.pack(pady=10)
tk.Button(button_frame, text="打开图片", command=self.load_image,
width=15, height=2).pack(side=tk.LEFT, padx=5)
tk.Button(button_frame, text="应用模糊", command=self.apply_blur,
width=15, height=2).pack(side=tk.LEFT, padx=5)
tk.Button(button_frame, text="重置", command=self.reset_image,
width=15, height=2).pack(side=tk.LEFT, padx=5)
tk.Button(button_frame, text="保存", command=self.save_image,
width=15, height=2).pack(side=tk.LEFT, padx=5)
tk.Button(button_frame, text="清除选择", command=self.clear_regions,
width=15, height=2).pack(side=tk.LEFT, padx=5)
# 模糊半径调节
radius_frame = tk.Frame(self.root)
radius_frame.pack(pady=5)
tk.Label(radius_frame, text="模糊半径:").pack(side=tk.LEFT, padx=5)
self.radius_scale = Scale(radius_frame, from_=1, to=50, orient=tk.HORIZONTAL,
length=200, command=self.update_blur_radius)
self.radius_scale.set(15)
self.radius_scale.pack(side=tk.LEFT, padx=5)
self.radius_label = tk.Label(radius_frame, text="15")
self.radius_label.pack(side=tk.LEFT, padx=5)
# 说明文字
info_label = tk.Label(self.root,
text="操作说明: 鼠标左键拖拽选择区域,可以多次选择多个区域",
font=("Arial", 10))
info_label.pack(pady=5)
# 图片显示区域
canvas_frame = tk.Frame(self.root)
canvas_frame.pack(fill=tk.BOTH, expand=True, padx=10, pady=10)
self.canvas = tk.Canvas(canvas_frame, bg="gray", cursor="crosshair")
self.canvas.pack(fill=tk.BOTH, expand=True)
# 绑定鼠标事件
self.canvas.bind("<Button-1>", self.on_mouse_press)
self.canvas.bind("<B1-Motion>", self.on_mouse_drag)
self.canvas.bind("<ButtonRelease-1>", self.on_mouse_release)
def update_blur_radius(self, value):
"""更新模糊半径"""
self.blur_radius = int(float(value))
self.radius_label.config(text=str(self.blur_radius))
def load_image(self):
"""加载图片"""
file_path = filedialog.askopenfilename(
filetypes=[("output_frame_0001.png", "*.jpg *.jpeg *.png *.bmp *.gif")]
)
if file_path:
self.image_path = file_path
self.original_image = Image.open(file_path)
self.display_image = self.original_image.copy()
self.regions = []
self.display_image_on_canvas()
def display_image_on_canvas(self):
"""在画布上显示图片"""
if self.display_image is None:
return
# 获取画布大小
canvas_width = self.canvas.winfo_width()
canvas_height = self.canvas.winfo_height()
if canvas_width <= 1 or canvas_height <= 1:
self.root.after(100, self.display_image_on_canvas)
return
# 计算缩放比例
img_width, img_height = self.display_image.size
scale = min(canvas_width / img_width, canvas_height / img_height)
# 缩放图片
display_width = int(img_width * scale)
display_height = int(img_height * scale)
display_img = self.display_image.resize((display_width, display_height), Image.Resampling.LANCZOS)
# 转换为 PhotoImage
self.photo = ImageTk.PhotoImage(display_img)
# 清除画布并显示图片
self.canvas.delete("all")
self.canvas.create_image(canvas_width // 2, canvas_height // 2,
image=self.photo, anchor=tk.CENTER)
# 绘制已选择的区域
for x1, y1, x2, y2 in self.regions:
# 转换坐标到显示坐标
offset_x = (canvas_width - display_width) // 2
offset_y = (canvas_height - display_height) // 2
self.canvas.create_rectangle(
x1 * scale + offset_x, y1 * scale + offset_y,
x2 * scale + offset_x, y2 * scale + offset_y,
outline="green", width=2
)
def on_mouse_press(self, event):
"""鼠标按下"""
self.start_x = event.x
self.start_y = event.y
def on_mouse_drag(self, event):
"""鼠标拖拽"""
if self.start_x is None or self.start_y is None:
return
# 清除当前临时矩形
self.canvas.delete("temp_rect")
# 绘制临时矩形
self.canvas.create_rectangle(
self.start_x, self.start_y, event.x, event.y,
outline="red", width=2, tags="temp_rect"
)
def on_mouse_release(self, event):
"""鼠标释放"""
if self.start_x is None or self.start_y is None:
return
if self.display_image is None:
return
# 获取画布和图片信息
canvas_width = self.canvas.winfo_width()
canvas_height = self.canvas.winfo_height()
img_width, img_height = self.display_image.size
scale = min(canvas_width / img_width, canvas_height / img_height)
offset_x = (canvas_width - int(img_width * scale)) // 2
offset_y = (canvas_height - int(img_height * scale)) // 2
# 转换坐标到图片坐标
x1 = int((self.start_x - offset_x) / scale)
y1 = int((self.start_y - offset_y) / scale)
x2 = int((event.x - offset_x) / scale)
y2 = int((event.y - offset_y) / scale)
# 确保坐标在图片范围内
x1 = max(0, min(x1, img_width))
y1 = max(0, min(y1, img_height))
x2 = max(0, min(x2, img_width))
y2 = max(0, min(y2, img_height))
# 确保 x1 < x2, y1 < y2
if x1 > x2:
x1, x2 = x2, x1
if y1 > y2:
y1, y2 = y2, y1
# 检查区域是否有效
if abs(x2 - x1) > 5 and abs(y2 - y1) > 5:
self.regions.append((x1, y1, x2, y2))
print(f"已选择区域: ({x1}, {y1}) -> ({x2}, {y2})")
self.display_image_on_canvas()
self.start_x = None
self.start_y = None
def apply_blur(self):
"""应用模糊效果"""
if self.original_image is None:
messagebox.showwarning("警告", "请先加载图片!")
return
if not self.regions:
messagebox.showwarning("警告", "请先选择要模糊的区域!")
return
# 创建模糊版本
blurred = self.original_image.filter(ImageFilter.GaussianBlur(radius=self.blur_radius))
# 创建遮罩
mask = Image.new('L', self.original_image.size, 0)
draw = ImageDraw.Draw(mask)
# 绘制所有选择的区域
for x1, y1, x2, y2 in self.regions:
draw.rectangle([x1, y1, x2, y2], fill=255)
# 合成
self.display_image = Image.composite(blurred, self.original_image, mask)
self.regions = [] # 清空选择
self.display_image_on_canvas()
messagebox.showinfo("成功", "模糊效果已应用!")
def reset_image(self):
"""重置图片"""
if self.original_image:
self.display_image = self.original_image.copy()
self.regions = []
self.display_image_on_canvas()
def clear_regions(self):
"""清除所有选择"""
self.regions = []
self.display_image_on_canvas()
def save_image(self):
"""保存图片"""
if self.display_image is None:
messagebox.showwarning("警告", "没有可保存的图片!")
return
file_path = filedialog.asksaveasfilename(
defaultextension=".jpg",
filetypes=[("JPEG", "*.jpg"), ("PNG", "*.png"), ("所有文件", "*.*")]
)
if file_path:
self.display_image.save(file_path)
messagebox.showinfo("成功", f"图片已保存到: {file_path}")
# 使用示例
if __name__ == '__main__':
root = tk.Tk()
app = BlurRegionGUI(root)
root.mainloop()
|