tkinter Multilistbox自定义事件绑定问题

时间:2020-01-13 20:44:37

标签: python tkinter listbox

所以我的项目是与图像一起显示来自数据库的信息。 这个想法如下:

我有一个描述图像的数据库。一个表具有常规信息,例如哪个图像中包含哪种颜色,而另一个表具有更详细的信息,例如可以在哪里找到该颜色,等等。

启动搜索时,我希望打开两个窗口(ClassTwoClassThree)。 我的完整应用程序包含用于过滤数据库请求的字段,而不是第一个仅包含用于启动搜索的按钮的窗口。当我启动该搜索时,例如所有包含绿色的图像,因此ClassTwo会列出所有具有该颜色的图像以及其他元数据。

然后

ClassThree将列出所有包含相同颜色但有更多细节的区域,例如图像中的位置和大小等。 单击任一MultiListbox后,我想打开一个显示图像的ImageViewer。 对于ClassThree,我还将直接突出显示我单击的区域,因此两个类都将具有绑定到MultiListbox的函数。

我的问题是与Listboxes的绑定有关,无法正常工作。当我使用例如image_parts_info_lb.bind()该功能根本没有被触发。当我使用image_parts_info_lb.bind_all()时,只会触发最后一个MultiListbox的功能。

您可以在课程注释中找到Python2的原始MultiListbox版本。

这是我的代码

import tkinter as tk

class MultiListbox(tk.Frame):
#original Python2 code here:
#https://www.oreilly.com/library/view/python-cookbook/0596001673/ch09s05.html
    def __init__(self, master, lists):
        tk.Frame.__init__(self, master)
        self.lists = []
        print(lists)
        for l,w in lists:
            frame = tk.Frame(self)
            frame.pack(side='left', expand='yes', fill='both')
            tk.Label(frame, text=l, borderwidth=1, relief='raised').pack(fill='x')
            lb = tk.Listbox(frame, width=w, borderwidth=0, selectborderwidth=0,
                         relief='flat', exportselection=False, height=16)
            lb.pack(expand='yes', fill='both')
            self.lists.append(lb)
            #commented out functions that were not necessary, as suggested in the comments
            #lb.bind('<B1-Motion>', self._select)
            lb.bind('<<ListboxSelect>>', self._select)
            #lb.bind('<Leave>', lambda e: 'break')
            lb.bind('<MouseWheel>', lambda e, s=self: s._scroll_mouse(e))
            #lb.bind('<Button-2>', lambda e, s=self: s._button2(e.x, e.y))
        frame = tk.Frame(self)
        frame.pack(side='left', fill='y')
        tk.Label(frame, borderwidth=1, relief='raised').pack(fill='x')
        sb = tk.Scrollbar(frame, orient='vertical', command=self._scroll)
        sb.pack(expand='yes', fill='y')
        self.lists[0]['yscrollcommand']=sb.set

    def _select(self, event):
        w = event.widget
        curselection = w.curselection()

        if curselection:
            self.selection_clear(0, self.size())
            self.selection_set(curselection[0])

    def _button2(self, x, y):
        for l in self.lists:
            l.scan_mark(x, y)
        return 'break'

    def _b2motion(self, x, y):
        for l in self.lists: l.scan_dragto(x, y)
        return 'break'

    def _scroll(self, *args):
        for l in self.lists:
            l.yview(*args)
        return 'break'

    def _scroll_mouse(self, event):
        for l in self.lists:
            l.yview_scroll(int(-1*(event.delta/120)), 'units')
        return 'break'

    def curselection(self):
        return self.lists[0].curselection()

    def delete(self, first, last=None):
        for l in self.lists:
            l.delete(first, last)

    def get(self, first, last=None):
        result = []
        for l in self.lists:
            result.append(l.get(first,last))
        if last: return apply(map, [None] + result)
        return result

    def index(self, index):
        self.lists[0].index(index)

    def insert(self, index, *elements):
        for e in elements:
            for i, l in enumerate(self.lists):
                l.insert(index, e[i])

    def size(self):
        return self.lists[0].size()

    def see(self, index):
        for l in self.lists:
            l.see(index)

    def selection_anchor(self, index):
        for l in self.lists:
            l.selection_anchor(index)

    def selection_clear(self, first, last=None):
        for l in self.lists:
            l.selection_clear(first, last)

    def selection_includes(self, index):
        return self.lists[0].selection_includes(index)

    def selection_set(self, first, last=None):
        for l in self.lists:
            l.selection_set(first, last)

class ClassOne:
    def __init__(self, master):
        self.master = master
        self.frame = tk.Frame(self.master)
        self.create_elements()
        self.frame.pack()

    def create_elements(self):
        search_button = tk.Button(self.frame, text = "Launch searches", command = \
        self.call_other_classes)
        search_button.grid(row = 2, column = 0, padx = 20, pady = 10)
        exit_button = tk.Button(self.frame, text = "Exit", command = self.master.quit)
        exit_button.grid(row = 2, column = 1, padx = 20, pady = 10)

    def call_other_classes(self):
        self.classes_list = []
        self.classes_list.append(ClassTwo(self.master))
        self.classes_list.append(ClassThree(self.master))

class ClassTwo:
    def __init__(self, master):
        self.master = master
        self.frame = tk.Frame(tk.Toplevel(self.master))
        self.frame.pack()
        self.create_elements()
        self.fill_image_listbox()       

    def create_elements(self):
        #placement of the custom MultiListbox
        self.available_images_lb = MultiListbox(self.frame, (('stuff1', 0), ('stuff1', 0), \
        ('stuff1', 0), ('stuff1', 0), ('stuff1', 0) ))
        self.available_images_lb.grid(row = 1, column = 1)
        self.available_images_lb.bind('<<ListboxSelect>>', self.print_stuff_two)
        #Button
        exit_button = tk.Button(self.frame, text = "Quit", command = self.frame.quit)
        exit_button.grid(row = 2, column = 1, padx = 20, pady = 10)


    def fill_image_listbox(self):
        image_info = [5*['ABCD'],5*['EFGH'],5*['JKLM'],5*['NOPQ'],5*['RSTU'], 5*['VWXY']]
        for item in image_info:
            self.available_images_lb.insert('end', item)

    def print_stuff_two(self, event):
        print('Class Two active')

class ClassThree:
    def __init__(self, master):
        self.master = master
        self.frame = tk.Frame(tk.Toplevel(self.master))
        self.create_elements()
        self.frame.pack()
        self.fill_imageparts_listbox()

    def create_elements(self):  
        self.image_parts_info_lb = MultiListbox(self.frame, (('stuff1', 0), ('stuff1', 0), \
        ('stuff1', 0), ('stuff1', 0), ('stuff1', 0) ))
        self.image_parts_info_lb.grid(row = 1, column = 1)
        self.image_parts_info_lb.bind('<<ListboxSelect>>', self.print_stuff_three)

    def fill_imageparts_listbox(self):
        self.image_parts_info_lb.delete(0, 'end')
        image_part_info = [5*['1234'],5*['5678'],5*['91011']]
        for item in image_part_info:
            self.image_parts_info_lb.insert('end', item)

    def print_stuff_three(self, event):
        print('Class Three active')     

def main():
    root = tk.Tk()
    root.title('Image Viewer')
    root.geometry('500x150+300+300')
    my_class_one = ClassOne(root)
    root.mainloop()


if __name__ == "__main__":
    main()

代替打印功能,我将启动一个简单的Image Viewer,然后使用Pillow突出显示图像中的区域。该功能已实现且运行良好,只有绑定到自定义Listbox的功能无法正常工作。

我愿意接受任何意见。另外,如果您对编码模式有任何建议,请随时告诉我。

1 个答案:

答案 0 :(得分:1)

问题MultiListbox个自定义事件绑定问题,单击任何列表元素,触发print_stuff_x function

实施自定义事件'<<MultiListboxSelect>>',该事件将在处理'<<ListboxSelect>>'事件并对其进行.bind(...时触发。


参考


关键点

self.event_generate('<<MultiListboxSelect>>', when='tail')

class MultiListbox(tk.Frame):
    def __init__(self, master, lists):
        ...
        for l,w in lists:
            ...
            lb = tk.Listbox(frame, ...
            lb.bind('<<ListboxSelect>>', self._select)

    def _select(self, event):
        w = event.widget
        curselection = w.curselection()

        if curselection:
            self.event_generate('<<MultiListboxSelect>>', when='tail')
            ...

class ClassTwo:
    ...

    def create_elements(self):
        self.available_images_lb = MultiListbox(self.frame, ...
        ...
        self.available_images_lb.bind('<<MultiListboxSelect>>', self.print_stuff_two)
        ...

class ClassThree:
    ...

    def create_elements(self):  
        self.image_parts_info_lb = MultiListbox(self.frame, ...
        ...
        self.image_parts_info_lb.bind('<<MultiListboxSelect>>', self.print_stuff_three)