我正在用Python构建Tkinter应用程序,我想为用户添加一个文本小部件以添加注释,同时在每个注释的注释旁边添加数字。我使用第二个小的Text小部件添加数字,然后按照this question中的建议使两个小部件一起滚动,以使数字保持其正确位置。
问题是,当Text的高度超过窗口小部件的高度并且滚动条应“跟随”以显示 INSERT 索引时,滚动条会出现意外现象每次输入新的字符时,都会交替上下。
每次输入新的char时,我都会监视赋予 yscrollcommand 的值,它们遵循以下顺序,而我在同一行中编写时,最后一行在低端小部件的:
f/l: 0.0 0.9732142857142857
---
f/l: 0.026785714285714284 1.0
---
f/l: 0.0 0.9732142857142857
---
f/l: 0.026785714285714284 1.0
---
f/l: 0.0 0.9732142857142857
---
f/l: 0.026785714285714284 1.0
---
f/l: 0.0 0.9732142857142857
---
f/l: 0.026785714285714284 1.0
这是相关代码:
from tkinter import *
from tkinter.ttk import *
class MyApp(Frame):
def __init__(self, parent, *args, **kwargs):
Frame.__init__(self, parent)
self.pack()
self.parent = parent
self.action_frame = Frame(self, name="action")
self.vis_frame = Panedwindow(self.action_frame, orient=HORIZONTAL)
self.action_frame.pack(fill=BOTH, expand=YES)
self.vis_frame.pack(fill=BOTH, expand=YES)
fr = LabelFrame(self.vis_frame, text="Comments")
fr1 = LabelFrame(self.vis_frame, text="Draft")
self.text = Text(fr)
self.lbl_fr = Text(fr, width=4, state=DISABLED, background="SystemMenu")
self.scrl = Scrollbar(fr, orient=VERTICAL)
Button(fr, text="Submit", command=self.submit).pack(side=BOTTOM)
self.scrl.pack(side=RIGHT, fill=Y)
self.text.pack(side=RIGHT, fill=BOTH)
self.lbl_fr.pack(side=RIGHT, fill=Y)
self.scrl.configure(command=self.__scrollBoth)
self.text.configure(yscrollcommand=self.__updateScroll)
self.lbl_fr.configure(yscrollcommand=self.__updateScroll)
self.vis_frame.add(fr1, weight=1)
self.vis_frame.add(fr)
bindtags = list(self.text.bindtags())
bindtags.insert(2, "custom")
self.text.bindtags(tuple(bindtags))
self.text.bind_class("custom", "<Return>", self.lbls)
self.text.bind_class("custom", "<Key>", self.lbls)
self.text.bind_class("custom", "<BackSpace>", self.lbls)
self.text.bind_class("custom", "<Delete>", self.lbls)
self.cur = self.lbl_fr.index(INSERT)
def lbls(self, e=None):
self.text.see("{}+1c".format(INSERT))
self.lbl_fr["state"] = NORMAL
self.lbl_fr.delete("0.0", END)
i = "1.0"
while True:
if i == self.text.index(END):
break
linenum = str(i).split(".")[0]
if i[:-2] == self.text.index(INSERT).split(".")[0]:
self.cur = self.lbl_fr.index(INSERT)
what = "{}.\n".format(linenum)
self.lbl_fr.insert(INSERT, what)
for j in range(len(self.text.get(i, i+"+1l-1c"))//self.text.cget("width")):
if i[:-2] == self.text.index(INSERT).split(".")[0]:
if j+1 == int(self.text.index(INSERT).split(".")[1]) // self.text.cget("width"):
self.cur = self.lbl_fr.index(INSERT)
self.lbl_fr.insert(INSERT, " \n")
i = self.text.index("{}+1l".format(i))
self.lbl_fr.delete(INSERT+"-1c", INSERT+" lineend")
self.lbl_fr["state"] = DISABLED
print("---------------------------")
# print("--->", self.cur)
def __scrollBoth(self, act, pos, type=None):
self.text.yview_moveto(pos)
self.lbl_fr.yview_moveto(pos)
print("pos: ", pos)
def __updateScroll(self, first, last, type=None):
self.text.yview_moveto(first)
self.lbl_fr.yview_moveto(first)
self.scrl.set(first, last)
# self.text.see(INSERT)
# self.lbl_fr.see(self.cur)
print("f/l: ", first, last)
def submit(self):
pass
if __name__ == "__main__":
root = Tk()
MyApp(root)
root.mainloop()
有人碰巧遇到任何类似的问题吗?我尝试了一种变通方法,方法是在 __ updateScroll 方法中添加两条注释行,以将两个小部件明确地放置在 INSERT 所在的位置( self.cur 变量总是使用 lbl_fr文本小部件中的 INSERT 标签的相应位置进行更新),但这当然不是一个完美的解决方案,因为它不会滚动小部件时无需更改 INSERT 即可正常工作。