我正在尝试让两个文本小部件滚动同步。到目前为止,我已经使用滚动条实现了这一点,当使用滚动条时,它可以正常工作。但是,例如,当我关注其中一个文本小部件并使用鼠标滚轮滚动时,只滚动具有焦点的文本小部件,滚动条也会更新,但其他文本保持不变。使用向下翻页或向上翻页键时会出现相同的行为,并且对于不使用滚动条的每种滚动形式我都知道。
这是我的代码,我认为只有 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
所以我的问题是,我应该独立处理所有页面,页面向下,鼠标滚轮和其他所有形式的滚动吗?是不是有一种更简单的方法让两个文本小部件一直有相同的滚动?
答案 0 :(得分:6)
我知道这有点旧,但这个解决方案非常有效。 Text小部件默认响应滚动鼠标滚轮,因此无需绑定任何内容。
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()
答案 1 :(得分:0)
如果事件处理应该针对每个滚动条独立完成,请回答您的问题 - 这是您必须做出的决定。 如果你只想为此目的制作这个小部件,你可以一起处理它们,而不是单独处理它们。为此创建一个自定义事件处理程序并相应地调用setter / getters。
如果您有两个小部件(包括两个滚动条)位于您想要绑定事件的主窗口小部件中,请使用它们将它们绑定在那里
widget.bind(<EVENT>, handler)
或使用widget.bind_all(<EVENT>, handler)
将从子窗口小部件引发的事件绑定到这些处理程序。
由于您已在代码中使用相同的处理程序(self.__scrollbar.set
),因此您可以使用自定义事件处理程序绑定Page Up
/ Page Down
键以按特定偏移量滚动和一个自定义按MouseWheel
滚动 - 事件。
如果您单独滚动并在这些处理程序中调用两个滚动功能,或者如果您在一个功能中同时滚动,则由您决定,如上所述。
由于需要为每个tk.Text
调用滚动 - Widget我个人更喜欢在一个处理程序中调用两个滚动到父窗口小部件以及所有下面的部分(所以使用bind_all
),但这是一个我认为个人喜好的问题。