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

13

我想自动选择列表框中的第一项。通过选择第一项,我不是指仅将其默认设置为第一项或在其上设置焦点。我已经通过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个回答

27

最简单的解决方案是在您更改选择时同时生成<<ListboxSelect>>事件:

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

太棒了!你让这些事情看起来轻而易举。 - sedeh
简单而高效...谢谢。 - Fejs

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

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

Tkinter.Listbox()的MVC-Model-Part行为相当复杂。因此,其Controller-Part .methods()较难处理。

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

这些模式有微妙的差别。

例如,BROWSE类似于SINGLE,但还允许选择被拖动

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

EXTENDED模式允许进行多个选择,并且类似于Windows文件资源管理器GUI - 您可以使用简单的单击选择一个项目,使用Ctrl-click组合选择多个项目,并用Shift-click选择一系列项目。

可以使用以下代码编程实现多个选择:

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 的值。
证毕,以上为初始帖子中的内容。

3
谢谢您的回答。在发布代码时,请尽量解释它,以便OP能够理解所给出的代码。 - Eduard Luca
2
所有与模式相关的评论与选择第一个项目有什么关系?这个答案似乎对于这样一个简单的问题过于复杂。 - Bryan Oakley
@user3666197 谢谢你的解释。目前我不想使用MULTIPLE模式。无论如何,我按照你的建议在self.master.mainloop()之前添加了self.listbox.selection_set( first = 0 ),但行为没有改变。你可以从http://ideone.com/8VzU3p复制修改后的代码并进行测试。还有其他想法吗?谢谢。 - sedeh
@BryanOakley 你有什么想法可以解决这个问题吗?谢谢。 - sedeh

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接