将Tkinter.Text子类化以创建自定义小部件 [英] Subclassing Tkinter.Text to create a custom widget

查看:78
本文介绍了将Tkinter.Text子类化以创建自定义小部件的处理方法,对大家解决问题具有一定的参考价值,需要的朋友们下面随着小编来一起学习吧!

问题描述

我试图创建一个带有垂直滚动条的文本小部件,同时保留Tkinter.Text中的所有方法/功能.

I am trying to create a text widget with a vertical scrollbar, while retaining all the methods / functions from Tkinter.Text.

到目前为止,我有以下代码:

So far I have the following code:

class ScrollableTextWidget(Tkinter.Text):
    def __init__(self, parent):
        self.parent = parent
        self.Frame = ttk.Frame(self.parent)
        Tkinter.Text.__init__(self, self.Frame, width=1, height=1)
        self.__initWidget()

    def __initWidget(self):
        self.Frame.grid(sticky="NSEW")
        self.ScrollbarY = ttk.Scrollbar(self.Frame, orient="vertical", 
                                        command=self.yview)
        self.configure(yscrollcommand=self.ScrollbarY.set)
        self.grid(column=0, row=0, sticky="NSEW")
        self.ScrollbarY.grid(column=1, row=0, sticky="NS")
        self.Frame.columnconfigure(0, weight=1)
        self.Frame.rowconfigure(0, weight=1)

是否可以像这样创建我的自定义小部件,还是应该将其放在Tkinter.frame中并编写自己的方法?

Is it OK to create my custom widget like this or should I rather put it in a Tkinter.frame and write my own methods?

推荐答案

对小部件进行子类化以创建自定义小部件是很正常的.但是,如果此自定义窗口小部件由多个窗口小部件组成,则通常应将Frame子类化.例如,要创建一个带有滚动条的文本小部件,我将执行以下操作:

It's very normal to subclass a widget to create a custom one. However, if this custom widget is made up of more than one widget, you would normally subclass Frame. For example, to create a widget that is a text widget with a scrollbar I would do something like this:

import Tkinter as tk

class ScrolledText(tk.Frame):
    def __init__(self, parent, *args, **kwargs):
        tk.Frame.__init__(self, parent)
        self.text = tk.Text(self, *args, **kwargs)
        self.vsb = tk.Scrollbar(self, orient="vertical", command=self.text.yview)
        self.text.configure(yscrollcommand=self.vsb.set)
        self.vsb.pack(side="right", fill="y")
        self.text.pack(side="left", fill="both", expand=True)

class Example(tk.Frame):
    def __init__(self, parent):
        tk.Frame.__init__(self, parent)
        self.scrolled_text = ScrolledText(self)
        self.scrolled_text.pack(side="top", fill="both", expand=True)
        with open(__file__, "r") as f:
            self.scrolled_text.text.insert("1.0", f.read())

root = tk.Tk()
Example(root).pack(side="top", fill="both", expand=True)
root.mainloop()

使用这种方法,请注意在插入文本时如何引用内部文本窗口小部件.如果希望此窗口小部件看起来更像真实文本窗口小部件,则可以创建到某些或所有文本窗口小部件功能的映射.例如:

With this approach, notice how you need to reference the inner text widget when inserting text. If you want this widget to look more like a real text widget, you can create a mapping to some or all of the text widget functions. For example:

import Tkinter as tk

class ScrolledText(tk.Frame):
    def __init__(self, parent, *args, **kwargs):
        tk.Frame.__init__(self, parent)
        self.text = tk.Text(self, *args, **kwargs)
        self.vsb = tk.Scrollbar(self, orient="vertical", command=self.text.yview)
        self.text.configure(yscrollcommand=self.vsb.set)
        self.vsb.pack(side="right", fill="y")
        self.text.pack(side="left", fill="both", expand=True)

        # expose some text methods as methods on this object
        self.insert = self.text.insert
        self.delete = self.text.delete
        self.mark_set = self.text.mark_set
        self.get = self.text.get
        self.index = self.text.index
        self.search = self.text.search

class Example(tk.Frame):
    def __init__(self, parent):
        tk.Frame.__init__(self, parent)
        self.scrolled_text = ScrolledText(self)
        self.scrolled_text.pack(side="top", fill="both", expand=True)
        with open(__file__, "r") as f:
            self.scrolled_text.insert("1.0", f.read())

root = tk.Tk()
Example(root).pack(side="top", fill="both", expand=True)
root.mainloop()

这篇关于将Tkinter.Text子类化以创建自定义小部件的文章就介绍到这了,希望我们推荐的答案对大家有所帮助,也希望大家多多支持IT屋!

查看全文
登录 关闭
扫码关注1秒登录
发送“验证码”获取 | 15天全站免登陆