2

我在这里发布了一个问题,并建议我重组代码。现在已经不同了,我觉得有理由提出一个新问题。

无论如何,我正在尝试向列表框添加拖放功能,但我认为第一步是让事件绑定首先工作。目前,当我单击列表时,单击列表框时出现以下错误。出现列表框窗口,但是当我单击它时会发生错误。

 "AttributeError: make_list instance has no attribute 'nearest'.

此外,当我在 build_listbox 方法中打印列表框时,会打印以下小数点 .40720520L。这不应该打印列表框中的值吗?毕竟,方法相同。列表框是否未正确创建?

from Tkinter import *
import Tkinter

class make_list:

    def move_mouse(self, event):
        self.curIndex = event.nearest(event.y)
        print self.curIndex

    def click_button(self, event):
        w= event.widget
        self.curIndex = int(w.curselection()[0])
        #print self.curIndex
        value = w.get(self.curIndex)
        print value

    def build_main_window(self):
        self.build_listbox()

    def build_listbox(self):
        listbox = Listbox()
        listbox.bind('<<ListboxSelect>>', self.click_button)
        listbox.bind('<B1-Motion>', self.move_mouse)
        for item in ["one", "two", "three", "four"]:
            listbox.insert(END, item)    
        listbox.insert(END, "a list entry")
        listbox.pack()
        print listbox
        return

if __name__ == '__main__':
    start = make_list()
    start.build_main_window()
    mainloop()
4

1 回答 1

0

在 Tkinter 小部件上执行 a 时获得的值在print很大程度上与用户无关,它只是Tcl为您创建的变量的名称。

现在,由于显然您的代码正在运行(通过快速浏览问题中的评论),我建议以不同的方式执行拖放功能。例如,如果我们有一个与 中的值相关联的值Listbox列表,那么如果我们可以更新这个列表然后它会反映在Listbox显示上,那么执行任务会容易得多,对吧?太糟糕了,我们在 Python 中没有这样的东西,但目前考虑以下代码:

import Tkinter

def list_click(event):
    w = event.widget
    index = w.nearest(event.y)
    w._selection = index

def list_motion(event):
    w = event.widget
    if w._selection is None:
        return
    index = w.curselection()[0]
    w._var.swap(index, w._selection)
    w._selection = index

def list_clear(event):
    event.widget._selection = None

root = Tkinter.Tk()

v = ListVar(values=('one', 'two', 'three', 'four'))
v.append('a list entry')

listbox = Tkinter.Listbox(listvar=v)
listbox.pack()

listbox._selection = None
listbox._var = v
listbox.bind('<1>', list_click)
listbox.bind('<B1-Motion>', list_motion)
listbox.bind('<ButtonRelease-1>', list_clear)

root.mainloop()

此代码以下列方式工作:当在列表框中单击鼠标时,它将最近的项目标记为“已选择”,当您移动鼠标时,它交换项目并更新选择,当您释放鼠标按钮时,它清除选择. 那么缺少的是这个ListVar东西。这里是:

class ListVar(Tkinter.Variable):
    def __init__(self, master=None, name=None, **kwargs):
        Tkinter.Variable.__init__(self, master, kwargs.get('values'), name)

    def set(self, values):
        self._tk.call('set', self._name, values)

    def set_index(self, index, value):
        self._tk.call('lset', self._name, index, value)

    def get_index(self, index):
        return self._tk.eval('lindex $%s %d' % (self._name, index))

    def get(self, start=None, end=None):
        if start is None and end is None:
            res = self._tk.eval('lrange $%s 0 end' % self._name)
        elif end is None:
            res = self._tk.eval('lrange $%s %d end' % (self._name, start))
        else:
            res = self._tk.eval('lrange $%s %d %d' % (self._name, start, end))
        return self._tk.splitlist(res)

    def swap(self, a, b):
        if a != b:
            tmp = self.get_index(a)
            self.set_index(a, self.get_index(b))
            self.set_index(b, tmp)

    def append(self, value):
        self._tk.eval('lappend %s {%s}' % (self._name, value))

它非常粗糙,如果传递“hello{”等值会失败。这可以通过多种方式改进。

于 2013-01-23T21:24:24.650 回答