欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

Python+Tkinter實現(xiàn)簡單的畫圖軟件

 更新時間:2023年04月21日 09:22:38   作者:街 三 仔  
這篇文章主要為大家詳細介紹了如何利用Python和Tkinter實現(xiàn)編寫一個簡單的畫圖軟件,可以實現(xiàn)畫圖、保存、撤銷等功能,感興趣的可以了解一下

一起畫圖吧

為什么突然想搞這個畫圖軟件呢

不瞞各位,是因為最近接到了一個很小很小很小小得不能再小的小項目

就是基于Tkinter,做一個簡易的畫圖軟件,要求不高,能畫就行,能保存就行,能撤回就行,能導入就行!

于是,遇到項目就精神抖擻的俺,三下五除二的就夸夸夸的寫,終于!花了將近兩個小時多的時間,寫出來了一個還用得過去得畫圖軟件,雖然這個畫圖軟件是寫出來了,但是,俺意猶未盡呀(貪婪!太貪婪了?。谑窍敫阋粋€更加NB一點的畫圖軟件,于是我打開瀏覽器(哦不),打開俺滴大腦,想著提升一下畫圖軟件的功能,于是就寫了以下畫圖軟件

畫圖軟件

基本介紹:構造一個GUI圖形界面,主菜單有導入圖片、保存截圖、清屏、撤銷、工具欄等功能,工具欄中有鉛筆畫圖、畫直線、畫矩形、畫圓形、添加文本、橡皮擦、顏色填充、設置前景色和設置背景色等功能。

用到的模塊:Tkinter、PIL

畫不多說,展示一波

話不多說,介紹一波

導入??蓪牒缶Y名為jpg、png、gif的圖片,可在軟件上呈現(xiàn)圖片,可進行繪畫。

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

清屏。顧名思義,咱就不多說了

撤銷。即返回上一步,但是里面有一個小bug,具體是啥,各位猜猜

工具欄。工具欄里有啥呢?進來看看就知道咯

實現(xiàn)代碼

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)   # 創(chuàng)建畫布
        self.canvas.create_image(x, y, image=self.image)

        self.canvas.bind('<Button-1>', self.onLeftButtonDown)
        self.canvas.bind('<B1-Motion>', self.onLeftButtonMove)
        self.canvas.bind('<ButtonRelease-1>', self.onLeftButtonUp)
        self.canvas.bind('<ButtonRelease-3>', self.onRightButtonUp)
        self.canvas.pack(fill=tk.BOTH, expand=tk.YES)

    def setMenu(self):
        '''主菜單及其關聯(lián)的函數(shù)'''
        self.menu = tk.Menu(self, bg="red")
        root.config(menu=self.menu)
        self.menu.add_command(label='導入', command=self.Import)
        self.menu.add_command(label='保存', command=self.SavePicture)
        self.menu.add_command(label='清屏', command=self.Clear)
        self.menu.add_command(label='撤銷', command=self.Back)

        '''子菜單及其關聯(lián)的函數(shù)'''
        self.menuType = tk.Menu(self.menu, tearoff=0)   # tearoff=0 - 表示無法將下拉菜單從“工具欄”窗口分離
        self.menu.add_cascade(label='工具欄', menu=self.menuType)  # add_cascade建立菜單類別對象
        # 在"工具欄"內建立菜單列表
        self.menuType.add_command(label='鉛筆', command=self.drawCurve)
        self.menuType.add_command(label='直線', command=self.drawLine)
        self.menuType.add_command(label='矩形', command=self.drawRectangle)
        self.menuType.add_command(label='圓形', command=self.drawCircle)
        self.menuType.add_command(label='文本', command=self.drawText)
        self.menuType.add_command(label='橡皮擦', command=self.onErase)
        self.menuType.add_command(label='顏色填充', command=self.fill)
        self.menuType.add_separator()       # 建立分隔線
        self.menuType.add_command(label='選擇前景色', command=self.chooseForeColor)
        self.menuType.add_command(label='選擇背景色', command=self.chooseBackColor)

    def Import(self):     # 導入文件
        filename = tk.filedialog.askopenfilename(title='導入圖片', filetypes=[('image', '*.jpg *.png *.gif')])
        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)
        # 清屏后對數(shù)據(jù)進行初始化
        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):  # 點擊鼠標左鍵后運行此函數(shù)
        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] == 'rect':
                        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] == 'oval':
                        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):  # 按下鼠標左鍵并移動后運行此函數(shù)
        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):    # 左鍵鼠標釋放后運行此函數(shù)
        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('rect')

        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('oval')

        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='輸入文本', prompt='')
        if self.text is not None:
            self.size = tk.simpledialog.askinteger('輸入字號', prompt='', 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 的參數(shù)是基于顯示分辨率的坐標,而 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('area', 'lt_txt', 'rb_txt')
        self.area_box.setStart(x, y)
        # 開始坐標文字
        self.canvas.create_text(
            x, y - 10, text=f'({x}, {y})', fill='red', tag='lt_txt')

    def updateEndPoint(self, x, y):
        self.area_box.setEnd(x, y)
        self.canvas.delete('area', 'rb_txt')
        box_area = self.area_box.box()
        # 選擇區(qū)域
        self.canvas.create_rectangle(
            *box_area, fill='black', outline='red', width=2, tags="area")
        self.canvas.create_text(
            x, y + 10, text=f'({x}, {y})', fill='red', tag='rb_txt')


class ScreenShot():

    def __init__(self, scaling_factor=2):
        self.win = tk.Tk()
        # self.win.tk.call('tk', 'scaling', scaling_factor)
        self.width = self.win.winfo_screenwidth()
        self.height = self.win.winfo_screenheight()

        # 無邊框,沒有最小化最大化關閉這幾個按鈕,也無法拖動這個窗體,程序的窗體在Windows系統(tǒng)任務欄上也消失
        self.win.overrideredirect(True)
        self.win.attributes('-alpha', 0.25)

        self.is_selecting = False

        # 綁定按 Enter 確認, Esc 退出
        self.win.bind('<KeyPress-Escape>', self.exit)
        self.win.bind('<KeyPress-Return>', self.confirmScreenShot)
        self.win.bind('<Button-1>', self.selectStart)
        self.win.bind('<ButtonRelease-1>', self.selectDone)
        self.win.bind('<Motion>', 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('area', 'lt_txt', 'rb_txt')
        self.win.attributes('-alpha', 0)

    def captureImage(self):
        if self.area.empty():
            return None
        else:
            filename = tk.filedialog.asksaveasfilename(filetypes=[('.jpg', 'JPG')],
                                                       initialdir='C:\\Users\\lin042\\Desktop\\')
            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('Select', 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__ == '__main__':
    x = 1200    # 寬
    y = 600     # 高
    root = tk.Tk()
    root.title('街三仔畫圖')   # 軟件名
    root.geometry('1200x600')    # 設置軟件大小 - 寬x高
    Draw_designs(root)
    root.mainloop()

項目地址

街三仔畫圖 ~ github

到此這篇關于Python+Tkinter實現(xiàn)簡單的畫圖軟件的文章就介紹到這了,更多相關Python Tkinter畫圖軟件內容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關文章希望大家以后多多支持腳本之家!

相關文章

  • python格式化字符串實例總結

    python格式化字符串實例總結

    這篇文章主要介紹了python格式化字符串的方法,實例展示了常見的幾類Python針對字符串的格式方法,非常實用,需要的朋友可以參考下
    2014-09-09
  • python爬蟲框架scrapy下載中間件的編寫方法

    python爬蟲框架scrapy下載中間件的編寫方法

    這篇文章主要介紹了python爬蟲框架scrapy下載中間件,在每一個scrapy工程中都有一個名為 middlewares.py 的文件,這個就是中間件文件,本文通過示例代碼給大家介紹的非常詳細,需要的朋友參考下吧
    2022-03-03
  • python安裝包出現(xiàn)Retrying?(Retry(total=4,?connect=None,?read=None,?redirect=None,?status=None))問題解決

    python安裝包出現(xiàn)Retrying?(Retry(total=4,?connect=None,?read=No

    這篇文章主要給大家介紹了關于python安裝包出現(xiàn)Retrying?(Retry(total=4,?connect=None,?read=None,?redirect=None,?status=None))問題的解決方法,需要的朋友可以參考下
    2022-09-09
  • Python對接支付寶支付自實現(xiàn)功能

    Python對接支付寶支付自實現(xiàn)功能

    這篇文章主要介紹了Python對接支付寶支付自實現(xiàn)功能,本文通過實例代碼給大家介紹的非常詳細,具有一定的參考借鑒價值,需要的朋友可以參考下
    2019-10-10
  • python中subprocess批量執(zhí)行l(wèi)inux命令

    python中subprocess批量執(zhí)行l(wèi)inux命令

    本篇文章給大家詳細講述了python中使用subprocess批量執(zhí)行l(wèi)inux命令的方法,有興趣的朋友參考學習下。
    2018-04-04
  • Matplotlib繪制子圖的常見幾種方法

    Matplotlib繪制子圖的常見幾種方法

    Matplotlib的可以把很多張圖畫到一個顯示界面,在作對比分析的時候非常有用。本文就介紹了幾種方法,需要的朋友們下面隨著小編來一起學習學習吧
    2021-05-05
  • 教你怎么用python爬取愛奇藝熱門電影

    教你怎么用python爬取愛奇藝熱門電影

    突然心血來潮想看看電影,特地整理了這篇文章,文中有非常詳細的代碼示例,對正在學習python爬蟲的小伙伴們有很好的幫助,需要的朋友可以參考下
    2021-05-05
  • 利用Python破解斗地主殘局詳解

    利用Python破解斗地主殘局詳解

    斗地主應該對大家來說都不陌生,下面這篇文章主要跟大家分享了關于利用Python破解斗地主殘局的相關資料,文中介紹的非常詳細,對大家具有一定的參考學習價值,需要的朋友們下面來一起看看吧。
    2017-06-06
  • 詳解如何利用Python繪制迷宮小游戲

    詳解如何利用Python繪制迷宮小游戲

    這篇文章主要為大家介紹了如何用Python制作一個迷宮游戲,文中的示例代碼講解詳細,對大家更好的理解和學習python有一定幫助,感興趣的朋友可以了解下
    2022-02-02
  • Python實現(xiàn)刪除重復視頻文件的方法詳解

    Python實現(xiàn)刪除重復視頻文件的方法詳解

    這篇文章主要為大家詳細介紹了如何利用Python實現(xiàn)刪除重復視頻文件功能,文中的示例代碼講解詳細,對我們學習Python有一定幫助,需要的可以參考一下
    2022-10-10

最新評論