Python tkinter Notebook标签添加关闭按钮元素及存储状态提示图标案例

这篇具有很好参考价值的文章主要介绍了Python tkinter Notebook标签添加关闭按钮元素及存储状态提示图标案例。希望对大家有所帮助。如果存在错误或未考虑完全的地方,请大家不吝赐教,您也可以点击"举报违法"按钮提交疑问。

通过Python tkinter实现类似Notepad++页面的效果,包括在Notebook标签中添加关闭按钮元素和存储状态提示图标,展示当前页面和鼠标交互效果。

效果图展示

Python tkinter Notebook标签添加关闭按钮元素

粉色框是当前页面,橙色框是鼠标经过,红色框是按下按钮,灰色按钮是其他页面的效果;
存储标识可以用来识别页面是否存储:例如当前页面已经保存用蓝色,未保存用红色,其他页面已经保存用灰色,未保存用淡红色。

首先导入模块

import tkinter as tkfrom tkinter import ttk

定义一个名为CustomNotebook的类,继承自ttk.Notebook

class CustomNotebook(ttk.Notebook):
    """定义一个名为CustomNotebook的类,继承自ttk.Notebook"""

    __initialized = False  # 初始化一个私有变量,用于标记是否已经初始化

    def __init__(self, *args, **kwargs):
        # 如果尚未初始化,则调用自定义初始化方法,并设置已初始化标志
        if not self.__initialized:
            self.__initialize_custom_style()
            CustomNotebook.__initialized = True # 标记__initialized,防止第二次调用出错


        # 设置notebook的样式为"CustomNotebook"
        kwargs["style"] = "CustomNotebook"
        # 调用父类的初始化方法
        ttk.Notebook.__init__(self, *args, **kwargs)

        self._active = None # 初始化一个私有变量,用于存储当前活动的tab

        # 绑定鼠标左键按下事件到on_close_press方法
        self.bind("<ButtonPress-1>", self.on_close_press, True)
        # 绑定鼠标左键释放事件到on_close_release方法
        self.bind("<ButtonRelease-1>", self.on_close_release)

    def on_close_press(self, event):
        """当按钮被按下时触发,位于关闭按钮上方"""

        # 获取鼠标点击位置的元素
        element = self.identify(event.x, event.y)

        # 如果元素包含"close",则执行以下操作
        if "close" in element:
            # 获取鼠标点击位置的索引值
            index = self.index("@%d,%d" % (event.x, event.y))
            # 将按钮状态设置为按下
            self.state(['pressed'])
            # 将_active属性设置为点击的索引值
            self._active = index    def on_close_release(self, event):
        """ 
        当鼠标在关闭按钮上释放时调用此方法。
        event:  包含鼠标事件信息的对象。
        """
        if not self.instate(['pressed']): # 如果按钮没有按下状态,直接返回
            return
        try:
            element =  self.identify(event.x, event.y) # 获取鼠标释放位置的元素
            index = self.index("@%d,%d" % (event.x, event.y)) # 获取元素在列表中的索引
            if "close" in element and self._active == index: # 如果元素是关闭按钮,并且当前激活的标签页与释放位置的标签页相同
                self.forget(index)  # 删除该标签页
                self.event_generate("<<NotebookTabClosed>>") # 生成一个表示标签页关闭的事件
        except: pass
        self.state(["!pressed"]) # 将按钮状态设置为非按下状态
        self._active = None # 将当前激活的标签页设置为None

    def __initialize_custom_style(self):
        # 创建一个ttk样式对象
        style = ttk.Style()
        # 定义四个图片对象,分别表示关闭按钮的不同状态
        self.images = (
            # 元素普通状态时图标
            tk.PhotoImage("img_closenormal", data='''
                R0lGODdhCwALAIMAAJKSkpeXl5ubm5+fn6CgoKampqqqqq2trbGxsba2tr29vcHBwdnZ2QAAAAAAAAAA
                ACwAAAAACwALAAAIXQAXJEBwwEDBAgUGHEigYOCBhwYMEBCAIAEDBgYQXhwg4ACCiwwKgOQIEeRGjgUK
                GgBJYABKgyYZuBRAQORFmwwEBBhgE6FNAQAEDCBAtCVHoAcC6AzANAAAAAYCAgA7
                '''),
            # 选中的选项卡的关闭图标
            tk.PhotoImage("img_closeselected", data='''
                R0lGODdhCwALAIVUAJ9dcptfe6NfcpthfZ5hfJ9ifp5nfaNjda9lc6Rmea1nfqRpe6Fqfq9ofK1ofqxr
                frFndbJqeLJufZ9rgKpngaBqgKtpg6tphK5uhKxvia5xhq91iq1xjK1zj692jLFyg7N1hrJ3i7B2jbF4
                ja51ka94kq97l698mq9+m7B8mbB/nLCDn6aEoayMprCForGMq7KNq7KPrrOXt7WfwP///wAAAAAAAAAA
                AAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAACwAAAAACwALAAAIcgBnwFiBwoQJEh0yUGARQ8YLFwRPHLwQ
                wEUMGjQOcsB4YYAKFxhpbMRooYCJFSlCYlTgoAAJFSlMhHTgQEGBDilSqoyggACGEiFFhGxwAINQGiNC
                gMAIQcADDR48bAjxQUIEBABaGJgwoQKDBQkOCGAREAA7
                '''),
            # 鼠标经过时选项卡的关闭图标
            tk.PhotoImage("img_closeactive", data='''
                R0lGODdhCwALAIVSAN5IN+lRNeBQP+5bPvtSP+FSQuJXR+VbRu5dQetfRuVbSf9dS/xeTO5iR+hhTOxl
                TP1wTP1iUP1kUv9lVPxnVv9rVP1pWP9rWv9vXv5/WP54Xf9yYv11YP90ZP99YP94af97bP99bv6BW/6C
                Yv6FYP+BZf6JZv+Eb/6Jav+Oa+eJe/+Bcf+Edf+Hef+KfP+Rd/+WfP+bev+Pgu+Rhv+RhP+ViP+Zjf+h
                lv+lmv+1rP+/twAAAAAAAAAAAAAAACwAAAAACwALAAAIcwB13KDhgkWIEB8wRJiBIwcOGzRktFjx
                wUKBGjh27GABYoNGCQZc2NC4owNJkCsIktRYoYKCDywmkizhQYODDSFWrNyRYsSDCSY1wiBJosGCCxpP
                vIihMcMAAgwoXOBQAoUJERACzAAAoICAAwkaIAigIiAAOw==
                '''),
            # 按下关闭按钮时选项卡的关闭图标
            tk.PhotoImage("img_closepressed", data='''
                R0lGODdhCwALAIUAAGsiD20jEHIkEHYlEXsnEX4oEogrE4orFIwsFJEtFJUvFZoxFpwwFp4yF6EzF6Yz
                F642GbU3GbY4Grk5Grk6G7s7G7w6G7w8G8A+HcJAHcVDH8ZEH8dEIMhGIMtJIsxKIs1KIpdxZ8eAbtqI
                cuKdiQAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAA
                AAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAACwAAAAACwALAAAIcABDABBA4AACBQscNBARIAGDBxAkVKhw
                oeIABiRIRJhQIeMFDAUeZCTRMWOGCwYgRBhpMgMGBBEqWBipYcOGDAoolBz5wcOGBRdGchjps0FQEhk6
                eMgIgoODihgycOgAAoQHDiIsYMBwoeYGDhtGBAQAOw==
                '''),
            )

        # 使用element_create方法创建一个名为"close"的元素,类型为"image",图像文件名为"img_closenormal"
        style.element_create("close", "image", "img_closenormal",
                            # 下面设置各状态图标,激活顺序越往后的越要写在前面,比如按下要先鼠标经过激活,按下就要写在鼠标经过前
                            # 当元素处于激活、按下、未禁用状态时,显示"img_closepressed"图片
                            ("active", "pressed", "!disabled", "img_closepressed"),
                            # 当元素处于激活且未禁用状态时,显示"img_closeactive"图片
                            ("active", "!disabled", "img_closeactive"), 
                            # 当选项卡处于选中状态时,显示"img_closeselected"图片
                            ("selected", "img_closeselected"), 
                            # 设置元素的边框宽度为10像素,无边框;设置元素的粘性属性为空字符串,表示不粘附在其他元素上。
                            border=10, sticky='')
        '''
        notebook有如下状态可以设置
            disabled    禁用状态,该状态下的控件无法接收用户输入。
            normal      正常状态,该状态下的控件可以接收用户输入。
            active      激活状态(鼠标经过),该状态下的控件可以接收用户输入,并且会显示特殊效果(如闪烁)。
            selected    选中状态,该状态下的控件会显示特殊效果(如高亮)。
            insensitive 不敏感状态,该状态下的控件不会响应用户的键盘操作。
            focus       聚焦状态,该状态下的控件会显示特殊效果(如边框)。
        '''
        # 设置Notebook的样式为"CustomNotebook",并为"CustomNotebook.client"添加一个样式选项,设置其"sticky"属性为"nswe"
        style.layout("CustomNotebook", [("CustomNotebook.client", {"sticky": "nswe"})])
        # 设置CustomNotebook.Tab的布局样式
        style.layout("CustomNotebook.Tab", [
            ("CustomNotebook.tab", { # 设置 CustomNotebook.tab 的样式
                "sticky": "nswe", # 设置 tab 的粘性属性为 NSWE,表示在水平方向上可拉伸,垂直方向上可滚动
                "children": [     # 设置 tab 的子元素
                    ("CustomNotebook.padding", { # 设置 CustomNotebook.padding 的样式
                        "side": "top",      # 设置 padding 的侧边距在顶部
                        "sticky": "nswe",   # 设置 padding 的粘性属性为 NSWE
                        "children": [       # 设置 padding 的子元素
                            ("CustomNotebook.focus", { # 设置 CustomNotebook.focus 的样式
                                "side": "top",    # 设置 focus 的侧边距在顶部
                                "sticky": "nswe", # 设置 focus 的粘性属性为 NSWE
                                "children": [     # 设置 focus 的子元素
                                    # 设置 CustomNotebook.label 的样式
                                    ("CustomNotebook.label", {"side": "left", "sticky": ''}), # 设置 label 的侧边距在左侧,无粘性
                                    # 设置 CustomNotebook.close 的样式
                                    ("CustomNotebook.close", {"side": "left", "sticky": ''}), # 设置 close 的侧边距在左侧,无粘性
                                ]
                            })
                        ]
                    })
                ]
            })
        ])

将定义好的Notebook应用到tk窗口,并添加存储标识,不需要的删除即可

if __name__ == "__main__":
    root = tk.Tk()
    notebook = CustomNotebook(width=400, height=400, padding=(2,5,2,2))
    notebook.pack(side="top", fill="both", expand=True)

    images = (
        # 原色
        tk.PhotoImage('save_original', data = '''
            R0lGODdhDgAOAIU9ADFjpTFjrTFjtTFrtTljrTlrrcDAwEJzvUpzvUp7vVJ7rVJ7vVqEvWOEvWOMzmuU
            zmuU1nOUxnOc1nOc3nuUxnucxnul53ut54Slzoylxoyl1oyt1ozGY5StxpS13py13qWtxqW9563G57XO
            773O78bW78bW987e9zlrvQAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAA
            AAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAACwAAAAADgAOAAAIjAAzUFhwIIXBgykYgDAQ4cQJFBAjQiTh
            oEOCExdQOJA4kcSBAycsaOSIwkKJASlCkoQ4IYSAFCYsOHDwoOZMBxI+vIxpoWfPCRKC6kxRwqdPoEE9
            BEhBwqiFoFA3LG26EgUEDUtHiOTAtSuHBxiybvXK1UEFAAhEiKhagUIBCg0ODBBAN4DdAAAIKAgIADs=
            '''),
        # 黑白
        tk.PhotoImage('save_gray', data = '''
            R0lGODdhDgAOAIU9AFxcXF1dXV9fX2JiYmRkZGVlZW1tbW9vb3R0dHZ2dn5+foGBgYeHh46Ojo+Pj5CQ
            kJKSkpaWlpeXl6CgoKGhoaOjo6WlpaioqKmpqa2trbCwsLKysru7u8DAwMLCwsrKys3NzdTU1NXV1dzc
            3P///wAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAA
            AAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAACwAAAAADgAOAAAIiwApQEhgoIDBgwUUZOjwYMQIEhAjQgTB
            4AKCERZIMJA4EYQBAyMmaORIYkKIAQVCkoQogUOAAiImMGDQoOZMBhE2vIw5oWdPCRGC6iwQwqdPoEE1
            ACgAwuiEoFAvLG26koSDCks/iMTAtSuGBhOybvXKlYEEAAc8eKgqAQIBCAsMDAhAF4BduwIQBAQAOw==
            '''),
        # 红色
        tk.PhotoImage('save_red', data = '''
            R0lGODdhDgAOAIU9AP/h4f++vv+3t/+2tv+vr/+srP+kpP+iov+dnf+UlP+Skv+Pj/+Li/+Kiv+Hh/+F
            hf+Dg/+Cgv95ef94eP90dP9ycv9xcf9wcP9paf9jY/9gYP9YWP9WVv9RUf9PT/tHR/pGRvhERPVBQfM/
            P/I+PgAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAA
            AAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAACwAAAAADgAOAAAIiwAhUNjg4YPBgx80LDhQIUAAABAjQiSA
            oQGHAA4AYJA4kYAHDwEiaOQIIMKAEB9CkoQoAcGIDwIiYMBwoeZMDBMSvIwZoWdPCROC6vwwwKdPoEEV
            kPhAwGiEoFAbLG26EoCFB0sLiGTAtSuDCxGybvXKFYMEEh0MGKgqgQIIChk8hBhBl4RduyI4BAQAOw==
            '''),
        # 淡红色
        tk.PhotoImage('save_lowred', data = '''
            R0lGODdhDgAOAIU9AP/w8P/p6f/o6P/h4f/e3v/W1v/U1P/Pz//Gxv/ExP/Bwf+9vf+8vP+5uf+3
            t/+1tf+0tP+rq/+qqv+mpv+kpP+jo/+iov+bm/+Vlf+Skv+Kiv+IiP+Dg/+Bgft5efp4ePh2dvVzc/Nx
            cfJwcAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAA
            AAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAACwAAAAADgAOAAAIiwAhUNjg4YPBgx80LDhQIUAAABAjQiSA
            oQGHAA4AYJA4kYAHDwEiaOQIIMKAEB9CkoQoAcGIDwIiYMBwoeZMDBMSvIwZoWdPCROC6vwwwKdPoEEV
            kPhAwGiEoFAbLG26EoCFB0sLiGTAtSuDCxGybvXKFYMEEh0MGKgqgQIIChk8hBhBl4RduyI4BAQAOw==
            '''),
        )

    colour = {"blue":None, "red":None, "violet":None, "orange":None, "green":None}
    for color in colour.keys():
        colour[color] = tk.Frame(notebook, background=color) # 新建Frame作为标签内容
        notebook.add(colour[color], text=color) # 将新建的Frame添加到notebook生成一个标签

    colour['white'] = tk.Frame(notebook) 
    notebook.add(colour['white'], text='white', image='save_gray' , compound='left') # 新建一个直接带image元素的标签,图标元素位于左侧
    notebook.insert(3, colour['white'])  #将'white'标签移动到3的位置
    notebook.select(colour['white'])

    # 修改标签的image元素
    notebook.tab(colour['blue']  , image='save_original', compound='left')
    notebook.tab(colour['red' ]  , image='save_red'     , compound='left')
    notebook.tab(colour['violet'], image='save_lowred'  , compound='left')

    root.mainloop()

上面代码中的base64编码的字符串,可以用下面代码获取,要用gif格式文件转换,传入文件路径即可。

from PIL import Imageimport base64, iodef bmp_to_base64(img_path):
    "将gif图片转为base64编码的字符串"
    # 打开图片
    img = Image.open(img_path)
    # 将图片数据转为字节流
    byte_arr = io.BytesIO()
    img.save(byte_arr, format='GIF')
    img_bytes = byte_arr.getvalue()
    # 将字节流转为base64编码的字符串
    base64_str = base64.b64encode(img_bytes).decode()
    return base64_strprint (bmp_to_base64('gif文件路径'))

#想要拖动标签切换位置的可以绑定鼠标事件,自己试试吧。文章来源地址https://www.toymoban.com/news/detail-654019.html

到了这里,关于Python tkinter Notebook标签添加关闭按钮元素及存储状态提示图标案例的文章就介绍完了。如果您还想了解更多内容,请在右上角搜索TOY模板网以前的文章或继续浏览下面的相关文章,希望大家以后多多支持TOY模板网!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处: 如若内容造成侵权/违法违规/事实不符,请点击违法举报进行投诉反馈,一经查实,立即删除!

领支付宝红包 赞助服务器费用

相关文章

  • Python+Tkinter 图形化界面基础篇:添加图形和图像

    在 Python 图形化界面开发中,添加图形和图像可以使你的应用程序更具吸引力和可交互性。本篇博客将介绍如何在 Tkinter 中添加图形元素、绘制基本图形以及显示图像。我们将详细讨论这些概念,并提供示例代码以帮助你更好地理解。 在 Tkinter 中,可以使用 Canvas 小部件来添

    2024年02月03日
    浏览(48)
  • 如何在web-view页添加悬浮按钮,在web-view页嵌套自定义元素

    效果如图,模拟器无法显示,真机调试正常

    2024年02月03日
    浏览(31)
  • selenium 新建、切换、关闭标签页(python)

    注:python 调用 selenium 版本 4.1.0 这里我安装过两个版本的selenium,调用方式是不一样的,可以查看(dir(driver)) 修改源文件: python_path + Libsite-packagesseleniumwebdrivercommonservice.py, 注意是common文件夹下的service.py 这里48行,self.creationflags 改成 134217728. 再运行就不会弹出黑色的运

    2024年02月11日
    浏览(41)
  • python绘制柱状图,并添加标签

    利用下面的代码,绘制柱状图。 难点:绘制标签 bar.get_height() 用于获取柱子的高度,也就是柱状图中每个柱子的高度值。在上面的代码中,我们使用这个高度值来确定要在柱子上方添加的数据标签的位置。 具体来说,这行代码: 将 height 设置为当前柱子的高度。然后,在

    2024年01月18日
    浏览(50)
  • tkinter模块高级操作(一)—— 透明按钮、透明文本框、自定义按钮及自定义文本框

    —— 众所周知,tkinter模块中自带的Button类是无法做到使其透明的(至少我无法做到) 【tip:透明是指让背景颜色或图片穿过按钮而显示出来】 —— 找遍了Button类的所有参数和操作,都无法解决这个问题! —— 但!真的就没有办法做出一个透明的按钮吗? —— 如果你认为

    2024年02月09日
    浏览(38)
  • python:给List添加元素

    List 是 Python 中常用的数据类型,它一个有序集合,即其中的元素始终保持着初始时的定义的顺序(除非你对它们进行排序或其他修改操作)。 在Python中,向List添加元素,方法有如下4种方法(append(),extend(),insert(), +加号) 此元素如果是一个list,那么这个list将作为一个整体进

    2024年02月08日
    浏览(34)
  • Python 中如何向列表或数组添加元素

    编程中的数组是一个有序的项目集合,所有的项目都需要是相同的数据类型。 然而,与其它编程语言不同,数组在 Python 中不是一个内置的数据结构。Python 使用列表取代传统的数组。 列表本质上是动态数组,是 Python 中最常见的和最强大的数据结构之一。 你可以把它们想象

    2024年02月11日
    浏览(42)
  • python List添加元素的4种方法

    在Python中,向List添加元素,方法有如下4种:append(),extend(),insert(), 加号+ 【1】 append() 追加单个元素到List的尾部,只接受一个参数,参数可以是任何数据类型,被追加的元素在List中保持着原结构类型。 此元素如果是一个list,那么这个list将作为一个整体进行追加,注意append

    2024年02月06日
    浏览(42)
  • chatgpt赋能python:Python中添加SEO元素的实践

    在SEO(Search Engine Optimization)优化中,添加正确的SEO元素对网站的排名和可见性至关重要。Python作为一种广泛应用于网络开发的编程语言,提供了许多用于添加和管理SEO元素的工具和技术。在这篇文章中,我们将探讨如何使用Python添加常用的SEO元素,包括meta标签、title标签和

    2024年02月09日
    浏览(42)
  • Python 集合 add()函数使用详解,集合添加元素

    「作者主页」: 士别三日wyx 「作者简介」: CSDN top100、阿里云博客专家、华为云享专家、网络安全领域优质创作者 「推荐专栏」: 小白零基础《Python入门到精通》 add() 函数可以给集合 「添加」 元素。 语法 参数 element :(必选)需要添加的元素 返回值 None,没有返回值,

    2024年02月17日
    浏览(44)

觉得文章有用就打赏一下文章作者

支付宝扫一扫打赏

博客赞助

微信扫一扫打赏

请作者喝杯咖啡吧~博客赞助

支付宝扫一扫领取红包,优惠每天领

二维码1

领取红包

二维码2

领红包