pythontkinter,使两个文本小部件的滚动同步

2024-06-16 09:55:57 发布

您现在位置:Python中文网/ 问答频道 /正文

我试图使两个文本小部件的滚动同步。到目前为止,我已经实现了使用滚动条,当使用滚动条时,它工作得很好。但例如,当我将焦点放在一个文本小部件上并使用鼠标滚轮滚动时,只有具有焦点的文本小部件被滚动,滚动条也会更新,但其他文本保持不变。同样的行为发生在使用pagedown或page up键时,正如我所知,对于不使用滚动条的每种形式的滚动。在

这是我的代码,我认为只有init才是绑定事件的相关部分,但为了以防万一,我决定把我所有的代码:

## HexText class
#
#
class HexText (tkk.Frame):

    __POS_TEXT = "00 01 02 03 04 05 06 07 08 09 0A 0B 0C 0D 0E 0F"
    __OFFSET_TEXT = "0x00000000"
    __LINE_LENGTH = len(__POS_TEXT)

    def __init__(self, master):

        super(HexText, self).__init__(master)

        self.__create_widgets()
        self.__organize_widgets()

    def __scrolls(self, *args):
        self.__data.yview(*args)
        self.__offset.yview(*args)

    def __create_widgets(self):

        self.__scrollbar = tkk.Scrollbar(self)
        self.__scrollbar["orient"] = tk.VERTICAL
        self.__scrollbar["command"] = self.__scrolls

        self.__data = tk.Text(self)
        self.__data["height"] = 8
        self.__data["width"] = HexText.__LINE_LENGTH
        self.__data["state"] = tk.DISABLED
        self.__data["relief"] = tk.GROOVE
        self.__data["yscrollcommand"] = self.__scrollbar.set

        self.__offset = tk.Text(self)
        self.__offset["height"] = 8
        self.__offset["width"] = len(HexText.__OFFSET_TEXT)
        self.__offset["state"] = tk.DISABLED
        self.__offset["relief"] = tk.FLAT
        self.__offset["bg"] = self.winfo_toplevel()["bg"]
        self.__offset["yscrollcommand"] = self.__scrollbar.set

        self.__pos = tk.Text(self)
        self.__pos.insert(tk.CURRENT, HexText.__POS_TEXT)
        self.__pos["height"] = 1
        self.__pos["width"] = HexText.__LINE_LENGTH
        self.__pos["state"] = tk.DISABLED
        self.__pos["relief"] = tk.FLAT
        self.__pos["bg"] = self.winfo_toplevel()["bg"]

    def __organize_widgets(self):

        self.__pos.grid(row = 0, column = 1, sticky = tk.N + tk.E + tk.W + tk.S)
        self.__offset.grid(row = 1, column = 0, sticky = tk.N + tk.E + tk.W + tk.S)
        self.__data.grid(row = 1, column = 1, sticky = tk.N + tk.E + tk.W + tk.S)
        self.__scrollbar.grid(row = 1, column = 2, sticky = tk.N + tk.E + tk.W + tk.S)

    @staticmethod
    def __get_char_index(string):
        i = str.find(string, '.')

        if i >= 0:
            i = int(string[i+1:])
        else:
            raise ValueError

        return i

    @staticmethod
    def __get_line_index(string):
        i = str.find(string, '.')

        if i >= 0:
            i = int(string[:i])
        else:
            raise ValueError

        return i

    @staticmethod
    def __get_hex_value(string):

        if (len(string) != 1):
            raise ValueError

        i = "%02X" % ord(string)

        return i

    def __update_offset(self, line_index):

        i = "0x%08X\n" % ((line_index) * 0x10)
        self.__offset["state"] = tk.NORMAL
        self.__offset.insert(tk.CURRENT, i)
        self.__offset["state"] = tk.DISABLED

    def __append(self, string):
        self.__data["state"] = tk.NORMAL
        self.__data.insert(tk.CURRENT, string)
        self.__data["state"] = tk.DISABLED

    def __write_char(self, string):
        str_index = self.__data.index(tk.CURRENT)
        i = HexText.__get_char_index(str_index)

        if (len(string) != 1):
            raise ValueError

        if (i == 0):
            self.__update_offset(HexText.__get_line_index(str_index) - 1)

        if (i == HexText.__LINE_LENGTH - 2):
            self.__append(HexText.__get_hex_value(string) + '\n')
        else:
            self.__append(HexText.__get_hex_value(string) + ' ')

    def write_str(self, string):

        for chars in string:
            self.__write_char(chars)

这是我尝试创建的小部件的图像,一个简单的十六进制查看器(两个文本小部件的行数相同):

http://i.stack.imgur.com/Yb8IH.png

所以我的问题是,我是否应该独立处理所有的向上翻页、向下翻页、鼠标滚轮和其他形式的滚动?难道没有更简单的方法让两个文本小部件一直都有相同的滚动吗?在


Tags: pos文本selfdatagetstringindex部件
2条回答

我知道这有点旧,但这个解决方案非常有效。默认情况下,文本小部件对滚动鼠标滚轮做出反应,因此不需要绑定任何内容。在

import sys
if sys.version[0] < '3':
    from Tkinter import *
else:
    from tkinter import *


class ScrolledTextPair(Frame):
    '''Two Text widgets and a Scrollbar in a Frame'''

    def __init__(self, master, **kwargs):
        Frame.__init__(self, master) # no need for super

        # Different default width
        if 'width' not in kwargs:
            kwargs['width'] = 30

        # Creating the widgets
        self.left = Text(self, **kwargs)
        self.left.pack(side=LEFT, fill=BOTH, expand=True)
        self.right = Text(self, **kwargs)
        self.right.pack(side=LEFT, fill=BOTH, expand=True)
        self.scrollbar = Scrollbar(self)
        self.scrollbar.pack(side=RIGHT, fill=Y)

        # Changing the settings to make the scrolling work
        self.scrollbar['command'] = self.on_scrollbar
        self.left['yscrollcommand'] = self.on_textscroll
        self.right['yscrollcommand'] = self.on_textscroll

    def on_scrollbar(self, *args):
        '''Scrolls both text widgets when the scrollbar is moved'''
        self.left.yview(*args)
        self.right.yview(*args)

    def on_textscroll(self, *args):
        '''Moves the scrollbar and scrolls text widgets when the mousewheel
        is moved on a text widget'''
        self.scrollbar.set(*args)
        self.on_scrollbar('moveto', args[0])


# Example
if __name__ == '__main__':
    root = Tk()

    t = ScrolledTextPair(root, bg='white', fg='black')
    t.pack(fill=BOTH, expand=True)
    for i in range(50):
        t.left.insert(END,"foo %s\n" % i)
        t.right.insert(END,"bar %s\n" % i)

    root.title("Text scrolling example")
    root.mainloop()

要回答您的问题,事件处理是否应该为每个滚动条独立完成-这是您必须做出的决定。 如果你只想让这个小部件达到这个目的,你可以一起处理,而不是单独处理。为此创建一个自定义事件处理程序,并相应地调用setters/getter。在

如果您有两个小部件(包括两个滚动条)在一个主小部件中,您希望在其中绑定事件,请使用 widget.bind(<EVENT>, handler)或使用widget.bind_all(<EVENT>, handler)将从子窗口小部件引发的事件绑定到这些处理程序。在

由于您已经在代码中使用了相同的处理程序(self.__scrollbar.set),所以您只需使用一个自定义事件处理程序来绑定Page Up/Page Down键以按特定偏移量滚动,而一个自定义则按MouseWheel-event滚动。在

如果您独立滚动并在这些处理程序中调用两个滚动函数,或者如果您在一个函数中同时滚动这两个函数,则取决于您,如上所述。在

由于滚动需要为每个tk.Text-小部件调用,我个人更喜欢在绑定到父小部件的一个处理程序中调用这两个滚动(因此使用bind_all),但我认为这是个人偏好的问题。在

相关问题 更多 >