在Python tkinter中,“文本”小部件的yscroll命令无法按预期运行

时间:2019-01-24 11:07:24

标签: python tkinter

我正在用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 即可正常工作。

0 个答案:

没有答案