Heim > Backend-Entwicklung > Python-Tutorial > „Wie man mit Python und Tkinter einfache Zeichensoftware implementiert'

„Wie man mit Python und Tkinter einfache Zeichensoftware implementiert'

WBOYWBOYWBOYWBOYWBOYWBOYWBOYWBOYWBOYWBOYWBOYWBOYWB
Freigeben: 2023-05-09 15:49:16
nach vorne
1486 Leute haben es durchsucht

一起画图吧

为什么突然想搞这个画图软件呢

不瞒各位,是因为最近接到了一个很小很小很小小得不能再小的小项目

就是基于Tkinter,做一个简易的画图软件,要求不高,能画就行,能保存就行,能撤回就行,能导入就行!

于是,遇到项目就精神抖擞的俺,三下五除二的就夸夸夸的写,终于!花了将近两个小时多的时间,写出来了一个还用得过去得画图软件,虽然这个画图软件是写出来了,但是,俺意犹未尽呀(贪婪!太贪婪了!),于是想搞一个更加NB一点的画图软件,于是我打开浏览器(哦不),打开俺滴大脑,想着提升一下画图软件的功能,于是就写了以下画图软件

画图软件

基本介绍:构造一个GUI图形界面,主菜单有导入图片、保存截图、清屏、撤销、工具栏等功能,工具栏中有铅笔画图、画直线、画矩形、画圆形、添加文本、橡皮擦、颜色填充、设置前景色和设置背景色等功能。

用到的模块:Tkinter、PIL

画不多说,展示一波

„Wie man mit Python und Tkinter einfache Zeichensoftware implementiert

话不多说,介绍一波

导入。可导入后缀名为jpg、png、gif的图片,可在软件上呈现图片,可进行绘画。

保存。任意截取屏幕上的部分,截取好后按下回车键,即可保存,若想退出,则按下esc键

清屏。顾名思义,咱就不多说了

撤销。即返回上一步,但是里面有一个小bug,具体是啥,各位猜猜

工具栏。工具栏里有啥呢?进来看看就知道咯

实现代码

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

252

253

254

255

256

257

258

259

260

261

262

263

264

265

266

267

268

269

270

271

272

273

274

275

276

277

278

279

280

281

282

283

284

285

286

287

288

289

290

291

292

293

294

295

296

297

298

299

300

301

302

303

304

305

306

307

308

309

310

311

312

313

314

315

316

317

318

319

320

321

322

323

324

325

326

327

328

329

330

331

332

333

334

335

336

337

338

339

340

341

342

343

344

345

346

347

348

349

350

351

352

353

354

355

356

357

358

359

360

361

362

363

364

365

366

367

368

369

370

371

372

import tkinter as tk

from tkinter import *

import tkinter.simpledialog

import tkinter.colorchooser

import tkinter.filedialog

from PIL import Image, ImageTk, ImageGrab

from tkinter.colorchooser import askcolor

from win32 import win32api, win32gui, win32print

from win32.lib import win32con

 

from win32.win32api import GetSystemMetrics

 

 

class Draw_designs(tk.Frame):

    def __init__(self, master=None):

        tk.Frame.__init__(self, master)

        self.master = master

        self.pack()

 

        self.temp = []  # 保存图形的类型

        self.li = []    # 保存所画图形的坐标

        self.fill_color = None # 保存填充的颜色

 

        self.lastDraw = 0

        self.end = [0]

        self.size = "12"    # 字体大小

 

        self.yesno = 0

        self.function = 1   # 默认铅笔

        self.X = 0

        self.Y = 0

 

        self.foreColor = '#000000'

        self.backColor = '#FFFFFF'

 

        self.create_widget()

        self.setMenu()

 

    def create_widget(self):

        self.image = PhotoImage()

        self.canvas = Canvas(root, bg='white', width=x, height=y)   # 创建画布

        self.canvas.create_image(x, y, image=self.image)

 

        self.canvas.bind(&#39;<Button-1>&#39;, self.onLeftButtonDown)

        self.canvas.bind(&#39;<B1-Motion>&#39;, self.onLeftButtonMove)

        self.canvas.bind(&#39;<ButtonRelease-1>&#39;, self.onLeftButtonUp)

        self.canvas.bind(&#39;<ButtonRelease-3>&#39;, self.onRightButtonUp)

        self.canvas.pack(fill=tk.BOTH, expand=tk.YES)

 

    def setMenu(self):

        &#39;&#39;&#39;主菜单及其关联的函数&#39;&#39;&#39;

        self.menu = tk.Menu(self, bg="red")

        root.config(menu=self.menu)

        self.menu.add_command(label=&#39;导入&#39;, command=self.Import)

        self.menu.add_command(label=&#39;保存&#39;, command=self.SavePicture)

        self.menu.add_command(label=&#39;清屏&#39;, command=self.Clear)

        self.menu.add_command(label=&#39;撤销&#39;, command=self.Back)

 

        &#39;&#39;&#39;子菜单及其关联的函数&#39;&#39;&#39;

        self.menuType = tk.Menu(self.menu, tearoff=0)   # tearoff=0 - 表示无法将下拉菜单从“工具栏”窗口分离

        self.menu.add_cascade(label=&#39;工具栏&#39;, menu=self.menuType)  # add_cascade建立菜单类别对象

        # 在"工具栏"内建立菜单列表

        self.menuType.add_command(label=&#39;铅笔&#39;, command=self.drawCurve)

        self.menuType.add_command(label=&#39;直线&#39;, command=self.drawLine)

        self.menuType.add_command(label=&#39;矩形&#39;, command=self.drawRectangle)

        self.menuType.add_command(label=&#39;圆形&#39;, command=self.drawCircle)

        self.menuType.add_command(label=&#39;文本&#39;, command=self.drawText)

        self.menuType.add_command(label=&#39;橡皮擦&#39;, command=self.onErase)

        self.menuType.add_command(label=&#39;颜色填充&#39;, command=self.fill)

        self.menuType.add_separator()       # 建立分隔线

        self.menuType.add_command(label=&#39;选择前景色&#39;, command=self.chooseForeColor)

        self.menuType.add_command(label=&#39;选择背景色&#39;, command=self.chooseBackColor)

 

    def Import(self):     # 导入文件

        filename = tk.filedialog.askopenfilename(title=&#39;导入图片&#39;, filetypes=[(&#39;image&#39;, &#39;*.jpg *.png *.gif&#39;)])

        if filename:

            self.image = Image.open(filename)

            self.image = self.image.resize((800, 600), Image.ANTIALIAS)

            self.image = ImageTk.PhotoImage(self.image)

            self.canvas.create_image(400, 300, image=self.image)

 

    def SavePicture(self):      # 保存画布

        ScreenShot()

 

    def Clear(self):    # 清屏

        for item in self.canvas.find_all():

            self.canvas.delete(item)

        # 清屏后对数据进行初始化

        self.end = [0]

        self.lastDraw = 0

 

    def Back(self):     # 撤回

        try:

            for i in range(self.end[-2], self.end[-1] + 1):

                self.canvas.delete(i)

            self.end.pop()

            self.li.pop()

 

        except:

            self.end = [0]

 

    def onLeftButtonDown(self, event):  # 点击鼠标左键后运行此函数

        self.yesno = 1

        self.X = event.x

        self.Y = event.y

 

        if self.function == 7:  # 颜色填充

            for i in range(len(self.li)):

                if (self.X >= self.li[i][0] and self.X <= self.li[i][2]) and (self.Y >= self.li[i][1] and self.Y <= self.li[i][3]):

                    if self.temp[i] == &#39;rect&#39;:

                        rect = self.canvas.create_rectangle(self.li[i][0], self.li[i][1], self.li[i][2], self.li[i][3])

                        self.canvas.itemconfig(rect, fill=self.fill_color)

                        self.end.append(rect)   # 加入撤销列表

 

                    elif self.temp[i] == &#39;oval&#39;:

                        oval = self.canvas.create_oval(self.li[i][0], self.li[i][1], self.li[i][2], self.li[i][3])

                        self.canvas.itemconfig(oval, fill=self.fill_color)

                        self.end.append(oval)  # 加入撤销列表

 

                    break

 

        if self.function == 4:

            self.canvas.create_text(event.x, event.y, font=("等线", int(self.size)), text=self.text, fill=self.foreColor)

            self.function = 1

 

    def onLeftButtonMove(self, event):  # 按下鼠标左键并移动后运行此函数

        if self.yesno == 0:

            return

 

        if self.function == 1:    # 铅笔

            self.lastDraw = self.canvas.create_line(self.X, self.Y, event.x, event.y, fill=self.foreColor)

            self.X = event.x

            self.Y = event.y

 

        elif self.function == 2:  # 画直线

            try:

                self.canvas.delete(self.lastDraw)

            except Exception:

                pass

            self.lastDraw = self.canvas.create_line(self.X, self.Y, event.x, event.y, fill=self.foreColor)

 

        elif self.function == 3:  # 画矩形

            try:

                self.canvas.delete(self.lastDraw)

            except Exception:

                pass

            self.lastDraw = self.canvas.create_rectangle(self.X, self.Y, event.x, event.y, outline=self.foreColor)

 

        elif self.function == 5:  # 橡皮擦

            self.lastDraw = self.canvas.create_rectangle(event.x - 10, event.y - 10, event.x + 10, event.y + 10, outline=self.backColor)

 

        elif self.function == 6:  # 画圆

            try:

                self.canvas.delete(self.lastDraw)

            except Exception:

                pass

            self.lastDraw = self.canvas.create_oval(self.X, self.Y, event.x, event.y, fill=self.backColor, outline=self.foreColor)

 

    def onLeftButtonUp(self, event):    # 左键鼠标释放后运行此函数

        if self.function == 2:

            self.lastDraw = self.canvas.create_line(self.X, self.Y, event.x, event.y, fill=self.foreColor)

 

        elif self.function == 3:    # 正方形

            self.lastDraw = self.canvas.create_rectangle(self.X, self.Y, event.x, event.y, outline=self.foreColor)

            self.li.append((self.X, self.Y, event.x, event.y))  # 保存图型的坐标

            self.temp.append(&#39;rect&#39;)

 

        elif self.function == 6:    # 圆形

            self.lastDraw = self.canvas.create_oval(self.X, self.Y, event.x, event.y, outline=self.foreColor)

            self.li.append((self.X, self.Y, event.x, event.y))  # 保存图型的坐标

            self.temp.append(&#39;oval&#39;)

 

        self.yesno = 0

        if self.function != 7:

            self.end.append(self.lastDraw)

 

    def onRightButtonUp(self, event):   # 在画布中鼠标右键按下并松开时,弹出菜单

        self.menu.post(event.x_root, event.y_root)

 

    def drawCurve(self):    # 铅笔

        self.function = 1

 

    def drawLine(self):     # 直线

        self.function = 2

 

    def drawRectangle(self):    # 矩形

        self.function = 3

 

    def drawCircle(self):   # 画圆

        self.function = 6

 

    def drawText(self):     # 文字

        self.text = tk.simpledialog.askstring(title=&#39;输入文本&#39;, prompt=&#39;&#39;)

        if self.text is not None:

            self.size = tk.simpledialog.askinteger(&#39;输入字号&#39;, prompt=&#39;&#39;, initialvalue=20)

            if self.size is None:

                self.size = "20"

        self.function = 4

 

    def onErase(self):  # 橡皮擦

        self.function = 5

 

    def fill(self):

        c = askcolor(color=self.foreColor, title="选择画笔颜色")

        self.fill_color = c[1]

        self.function = 7

 

    def chooseForeColor(self):  # 设置前景色

        self.foreColor = tk.colorchooser.askcolor()[1]

 

    def chooseBackColor(self):  # 设置背景色

        self.backColor = tk.colorchooser.askcolor()[1]

 

"""

------------- 截图 -----------------

"""

def get_real_resolution():

    """获取真实的分辨率"""

    hDC = win32gui.GetDC(0)

    # 横向分辨率

    w = win32print.GetDeviceCaps(hDC, win32con.DESKTOPHORZRES)

    # 纵向分辨率

    h = win32print.GetDeviceCaps(hDC, win32con.DESKTOPVERTRES)

    return w, h

 

 

def get_screen_size():

    """获取缩放后的分辨率"""

    w = GetSystemMetrics(0)

    h = GetSystemMetrics(1)

    return w, h

 

 

real_resolution = get_real_resolution()

screen_size = get_screen_size()

 

# Windows 设置的屏幕缩放率

# ImageGrab 的参数是基于显示分辨率的坐标,而 tkinter 获取到的是基于缩放后的分辨率的坐标

screen_scale_rate = round(real_resolution[0] / screen_size[0], 2)

 

 

class Box:

 

    def __init__(self):

        self.start_x = None

        self.start_y = None

        self.end_x = None

        self.end_y = None

 

    def isNone(self):

        return self.start_x is None or self.end_x is None

 

    def setStart(self, x, y):

        self.start_x = x

        self.start_y = y

 

    def setEnd(self, x, y):

        self.end_x = x

        self.end_y = y

 

    def box(self):

        lt_x = min(self.start_x, self.end_x)

        lt_y = min(self.start_y, self.end_y)

        rb_x = max(self.start_x, self.end_x)

        rb_y = max(self.start_y, self.end_y)

        return lt_x, lt_y, rb_x, rb_y

 

    def center(self):

        center_x = (self.start_x + self.end_x) / 2

        center_y = (self.start_y + self.end_y) / 2

        return center_x, center_y

 

 

class SelectionArea:

 

    def __init__(self, canvas: tk.Canvas):

        self.canvas = canvas

        self.area_box = Box()

 

    def empty(self):

        return self.area_box.isNone()

 

    def setStartPoint(self, x, y):

        self.canvas.delete(&#39;area&#39;, &#39;lt_txt&#39;, &#39;rb_txt&#39;)

        self.area_box.setStart(x, y)

        # 开始坐标文字

        self.canvas.create_text(

            x, y - 10, text=f&#39;({x}, {y})&#39;, fill=&#39;red&#39;, tag=&#39;lt_txt&#39;)

 

    def updateEndPoint(self, x, y):

        self.area_box.setEnd(x, y)

        self.canvas.delete(&#39;area&#39;, &#39;rb_txt&#39;)

        box_area = self.area_box.box()

        # 选择区域

        self.canvas.create_rectangle(

            *box_area, fill=&#39;black&#39;, outline=&#39;red&#39;, width=2, tags="area")

        self.canvas.create_text(

            x, y + 10, text=f&#39;({x}, {y})&#39;, fill=&#39;red&#39;, tag=&#39;rb_txt&#39;)

 

 

class ScreenShot():

 

    def __init__(self, scaling_factor=2):

        self.win = tk.Tk()

        # self.win.tk.call(&#39;tk&#39;, &#39;scaling&#39;, scaling_factor)

        self.width = self.win.winfo_screenwidth()

        self.height = self.win.winfo_screenheight()

 

        # 无边框,没有最小化最大化关闭这几个按钮,也无法拖动这个窗体,程序的窗体在Windows系统任务栏上也消失

        self.win.overrideredirect(True)

        self.win.attributes(&#39;-alpha&#39;, 0.25)

 

        self.is_selecting = False

 

        # 绑定按 Enter 确认, Esc 退出

        self.win.bind(&#39;<KeyPress-Escape>&#39;, self.exit)

        self.win.bind(&#39;<KeyPress-Return>&#39;, self.confirmScreenShot)

        self.win.bind(&#39;<Button-1>&#39;, self.selectStart)

        self.win.bind(&#39;<ButtonRelease-1>&#39;, self.selectDone)

        self.win.bind(&#39;<Motion>&#39;, self.changeSelectionArea)

 

        self.canvas = tk.Canvas(self.win, width=self.width,

                                height=self.height)

        self.canvas.pack()

        self.area = SelectionArea(self.canvas)

        self.win.mainloop()

 

    def exit(self, event):

        self.win.destroy()

 

    def clear(self):

        self.canvas.delete(&#39;area&#39;, &#39;lt_txt&#39;, &#39;rb_txt&#39;)

        self.win.attributes(&#39;-alpha&#39;, 0)

 

    def captureImage(self):

        if self.area.empty():

            return None

        else:

            filename = tk.filedialog.asksaveasfilename(filetypes=[(&#39;.jpg&#39;, &#39;JPG&#39;)],

                                                       initialdir=&#39;C:\\Users\\lin042\\Desktop\\&#39;)

            box_area = [x * screen_scale_rate for x in self.area.area_box.box()]

            self.clear()

            img = ImageGrab.grab(box_area).save(filename)

            return img

 

    def confirmScreenShot(self, event):

        img = self.captureImage()

        if img is not None:

            img.show()

        self.win.destroy()

 

    def selectStart(self, event):

        self.is_selecting = True

        self.area.setStartPoint(event.x, event.y)

        # print(&#39;Select&#39;, event)

 

    def changeSelectionArea(self, event):

        if self.is_selecting:

            self.area.updateEndPoint(event.x, event.y)

            # print(event)

 

    def selectDone(self, event):

        self.is_selecting = False

 

if __name__ == &#39;__main__&#39;:

    x = 1200    # 宽

    y = 600     # 高

    root = tk.Tk()

    root.title(&#39;街三仔画图&#39;)   # 软件名

    root.geometry(&#39;1200x600&#39;)    # 设置软件大小 - 宽x高

    Draw_designs(root)

    root.mainloop()

Nach dem Login kopieren

Das obige ist der detaillierte Inhalt von„Wie man mit Python und Tkinter einfache Zeichensoftware implementiert'. Für weitere Informationen folgen Sie bitte anderen verwandten Artikeln auf der PHP chinesischen Website!

Verwandte Etiketten:
Erklärung dieser Website
Der Inhalt dieses Artikels wird freiwillig von Internetnutzern beigesteuert und das Urheberrecht liegt beim ursprünglichen Autor. Diese Website übernimmt keine entsprechende rechtliche Verantwortung. Wenn Sie Inhalte finden, bei denen der Verdacht eines Plagiats oder einer Rechtsverletzung besteht, wenden Sie sich bitte an admin@php.cn
Beliebte Tutorials
Mehr>
Neueste Downloads
Mehr>
Web-Effekte
Quellcode der Website
Website-Materialien
Frontend-Vorlage