Как передать аргументы лямбда-функции в команде yscroll в списке tkinter?

#python-3.x #tkinter #lambda #arguments #command

Вопрос:

6 Объекты списка создаются в цикле for в классе приложений. И его команда yscroll, 6 функций жестко закодированы. (функция def для 0, функция def для 1…)

Похоже, если я смогу передать аргумент индекса в лямбда-функцию, 6 функций для прокрутки списка могут быть сжаты до 1 для цикла.

Но функция в классе, у нее есть аргумент «я». Это сбивает меня с толку.

Как я могу передать аргумент индекса в лямбда-функцию в команде yscroll?

 class app(tk.Frame):
    def __init__(self):
        self.root = tk.Tk()
        self.root.title('title something')

        # showing data frame
        self.data_frame = tk.LabelFrame(self.root, text='')
        self.data_frame.pack(fill='x')
        
        self.scrollbar = tk.Scrollbar(self.data_frame)
        self.scrollbar.pack(side='right', fill='y')
        
        self.listboxes = []
        self.listboxes_column = 6 # This can be vary.
        
        # listboxes are in a list.
        for i in range(self.listboxes_column):
            self.listboxes.append(tk.Listbox(self.data_frame, selectmode='extended', height=20, width=0, yscrollcommand = self.scrollbar.set))
            self.listboxes[i].pack(side='left')
            
        # when self.listboxes_column == 3
        # self.list_indexes == [[1, 2, 3], [0, 2, 3], [0, 1, 3], [0, 1, 2]]
        self.list_indexes = []
        for i in range(self.listboxes_column):
            indexes = [j for j in range(self.listboxes_column)]
            indexes.remove(i)
            self.list_indexes.append(indexes)

        # a listbox can scroll the others     
        # with lambda function and argument passing,
        # I want to make these 6 lines be in a for loop.
        # The from like this
        # for i in range(6):
        #     self.listboxes[index_argument].config(yscrollcommand = lambda ????? : self.list_scrolls_all(index_argument  ????? ))

        self.listboxes[0].config(yscrollcommand = self.list0_scrolls_all)
        self.listboxes[1].config(yscrollcommand = self.list1_scrolls_all)
        self.listboxes[2].config(yscrollcommand = self.list2_scrolls_all)
        self.listboxes[3].config(yscrollcommand = self.list3_scrolls_all)
        self.listboxes[4].config(yscrollcommand = self.list4_scrolls_all)
        self.listboxes[5].config(yscrollcommand = self.list4_scrolls_all)
        
        self.scrollbar.config(command=self.bar_scrolls_all)
        self.root.mainloop()


   # functions for lists scroll from 0 to 5.
   # I don't know how to pass argument via yscrollcommand in Listbox.
   # I want a form like this.
   #
   # def list_scrolls_all(self, index, *args):
   #     for i in self.list_indexes[index] :
   #         self.listboxes[i].yview_moveto(args[0])
   #     self.scrollbar.set(*args)

   def list0_scrolls_all(self, *args):
        for i in self.list_indexes[0] :
            self.listboxes[i].yview_moveto(args[0])
        self.scrollbar.set(*args)

    def list1_scrolls_all(self, *args):
        for i in self.list_indexes[1] :
            self.listboxes[i].yview_moveto(args[0])
        self.scrollbar.set(*args)


   # scroll bar
   def bar_scrolls_all(self,*args):
        for i in range(self.listboxes_column):
            self.listboxes[i].yview(*args)
 

Комментарии:

1.Это должно правильно использоваться lambda для правильной привязки функции: for i in range(6): self.listboxes[i].config(yscrollcommand=lambda index_argument=i: self.list_scrolls_all(index_argument)) . self Аргумент в методах (функции в классе называются методами) просто позволяет методу использовать переменные класса. Например, когда вы используете self.listboxes внутри метода, self. это self аргумент в вашем методе.

2. Спасибо. Я думал, что могу передать int и *args в функции через команду yscroll. Но, похоже, yscrollcommand может передавать только *аргументы.

Ответ №1:

Я просто случайно работаю над программой с несколькими списками, поэтому я внес несколько изменений в ваш код.

Я включил flexx функцию, чтобы все объекты были полностью гибкими, хотя мне пришлось заменить pack grid их .

 import tkinter as tk

def flexx(m, r = 0, c = 0, rw = 1, cw = 1):
    if r !=  None:
        m.rowconfigure(r, weight = rw)
    if c !=  None:
        m.columnconfigure(c, weight = cw)

class app:

    def __init__(self):
        self.root = tk.Tk()
        self.root.title('title something')
        # make root flexible
        flexx(self.root)

        # showing data frame
        self.data_frame = tk.LabelFrame(self.root, text='Multiple Linked Listboxes')
        self.data_frame.grid(row = 0, column = 0, sticky = tk.NSEW)

        self.listboxes = []
        self.listboxes_column = 6 # This can be vary.

        self.scrollbar = tk.Scrollbar(self.data_frame)
        self.scrollbar.grid(row = 0, column = self.listboxes_column, sticky = tk.NS)

        # listboxes are in a list.
        for i in range(self.listboxes_column):
            self.listboxes.append(tk.Listbox(
                self.data_frame, selectmode='extended',
                height=20, width=0))
            self.listboxes[i].grid(row = 0, column = i, sticky = tk.NSEW)

            # make data_frame flexible
            flexx(self.data_frame, c = i)

            # populate listbox with some data for testing purposes
            for b in dir(self):
                self.listboxes[i].insert("end", b)

        # connect yscollcommand to all listboxes
        for a in self.listboxes:
            a["yscrollcommand"] = self.move_to

        if False: # Not sure waht this does
            # when self.listboxes_column == 3
            # self.list_indexes == [[1, 2, 3], [0, 2, 3], [0, 1, 3], [0, 1, 2]]
            self.list_indexes = []
            for i in range(self.listboxes_column):
                indexes = [j for j in range(self.listboxes_column)]
                indexes.remove(i)
                self.list_indexes.append(indexes)

        # connect scrollbar command to all y_views
        self.scrollbar.config(command = self.y_view)

    def y_view(self, *args):
        for a in self.listboxes:
            a.yview(*args)

    def move_to(self, *args):
        self.scrollbar.set(*args)
        self.y_view("moveto", args[0])

if __name__ == "__main__":
    demo = app()
    demo.root.mainloop()
 

Комментарии:

1. О, ваш код работает именно так, как я хочу.

2. Хороший @4001jh, счастливого кодирования.