在Tkinter列表框中默认选择第一个项

13 投票
2 回答
31908 浏览
提问于 2025-04-18 17:59

我想要在列表框中自动选择第一个项目。这里的“选择第一个项目”不是说只是让第一个项目默认被选中或者让它获得焦点。我已经通过执行 self.listbox.select_set(0) 达到了这一点。我希望默认的项目也能被真正地 选中。换句话说,当我运行下面的代码时,我希望 print(value) 能打印出默认选择的值。如果从选项菜单中选择了亚洲,那么控制台应该自动打印出日本。如果选择了非洲,那么应该打印出尼日利亚,欧洲则打印德国。

有没有什么建议可以让我实现这个呢?谢谢。

from tkinter import *
from tkinter import ttk
import tkinter.messagebox

class App:
    def __init__(self):
        self.master = Tk()
        self.di = {'Asia': ['Japan', 'China', 'Malaysia', 'India', 'Korea',
                            'Vietnam', 'Laos', 'Thailand', 'Singapore',
                            'Indonesia', 'Taiwan'],
                     'Europe': ['Germany', 'France', 'Switzerland'],
                     'Africa': ['Nigeria', 'Kenya', 'Ethiopia', 'Ghana',
                                'Congo', 'Senegal', 'Guinea', 'Mali', 'Cameroun',
                                'Benin', 'Tanzania', 'South Africa', 'Zimbabwe']}
        self.variable_a = StringVar()
        self.frame_optionmenu = ttk.Frame(self.master)
        self.frame_optionmenu.pack()
        options = sorted(self.di.keys())
        self.optionmenu = ttk.OptionMenu(self.frame_optionmenu, self.variable_a, options[0], *options)

        self.variable_a.set('Asia')
        self.optionmenu.pack()
        self.btn = ttk.Button(self.master, text="Submit", width=8, command=self.submit)
        self.btn.pack()

        self.frame_listbox = ttk.Frame(self.master)

        self.frame_listbox.pack(side=RIGHT, fill=Y)
        self.scrollbar = Scrollbar(self.frame_listbox )
        self.scrollbar.pack(side=RIGHT, fill=Y)
        self.listbox = Listbox(self.frame_listbox, selectmode=SINGLE, yscrollcommand=self.scrollbar.set)
        self.variable_a.trace('w', self.updateoptions)

        self.scrollbar.config(command=self.listbox.yview)
        self.listbox.pack()

        #Populate listbox
        for each in self.di[self.variable_a.get()]:
            self.listbox.insert(END, each)
            self.listbox.select_set(0) #This only sets focus on the first item.
        self.listbox.bind("<<ListboxSelect>>", self.OnSelect)

        self.master.mainloop()

    def updateoptions(self, *args):
        #countries = self.di[self.variable_a.get()]
        self.listbox.delete(0, 'end')
        for each in self.di[self.variable_a.get()]:
            self.listbox.insert(END, each)
            self.listbox.select_set(0) #This only sets focus on the first item.
        self.listbox.pack()

    def submit(self, *args):
        var = self.variable_a.get()
        if messagebox.askokcancel("Selection", "Confirm selection: " + var):
            print(var)

    def OnSelect(self, event):
        widget = event.widget
        value = widget.get(widget.curselection()[0])
        print(value)

App()

运行 Python 3.4.1

2 个回答

5
# add before .mainloop()
self.listbox.selection_set( first = 0 )

编辑#1 2014-08-21 13:50 [UTC+0000]

Tkinter.Listbox() 的行为有点复杂,涉及到MVC模型中的一些部分。因此,它的控制部分 .methods() 处理起来会稍微复杂一些。

默认情况下,Listbox()select-mode 只允许选择一个项目,但 select-mode 参数支持四种设置:SINGLEBROWSEMULTIPLEEXTENDED(默认是 BROWSE)。其中前两种是单选模式,后两种则允许多选。

这些模式在细节上有所不同。

比如,BROWSE 模式类似于 SINGLE它还允许你拖动选择

MULTIPLE 模式下,点击一个项目会切换它的状态,而不会影响其他已选择的项目。

EXTENDED 模式允许多项选择,使用起来就像Windows文件管理器的界面——你可以通过简单的 点击 选择一个项目,通过 Ctrl-点击 组合选择多个项目,或者通过 Shift-点击 选择一系列项目。

多项选择可以用这样的代码来实现:

listbox = Listbox( aWindow, bg = 'white', font = ( 'courier', fontsz ) )
listbox.config( selectmode = EXTENDED )                         # see above
listbox.bind( '<Double-1>', ( lambda event: onDoubleClick() ) ) # a lambda-wrapped CallBackHANDLER()
# onDoubleClick: get messages selected in listbox               # not listed here
selections = listbox.curselection()                             # tuple of digit-string(s), aTupleOfSTRINGs, where digit-string(s) range from { 0, 1, .., N-1 }
selections = [ int( x ) + 1 for x in selections ]               # transform string(s) to shifted int(s), make 'em { 1, 2, .., N }

当启用多项选择时,.curselection() 方法会返回一个包含所选项目相对编号的字符串列表,如果没有选择任何项目,则返回一个空的元组。

需要注意的是,这个方法总是返回一个字符串元组,即使在单选模式下也是如此。

因此,Listbox().selection_set() 方法必须功能丰富,以便能够配置所有可能的状态,针对 aSelectionSET 的值。

以上内容在最初的帖子中已经说明。

27

最简单的解决办法就是在你改变选择的时候,同时生成一个 <<ListboxSelect>> 事件。

def updateoptions(self, *args):
    ...
    self.listbox.select_set(0) #This only sets focus on the first item.
    self.listbox.event_generate("<<ListboxSelect>>")
    ...

撰写回答