繁体   English   中英

Python tkinter:对OptionMenu小部件的内容进行排序

[英]Python tkinter: Sorting the contents of an OptionMenu widget

大家好,谢谢!

我在Google周围搜索了所有内容,并阅读了几乎所有的结果,但仍然无法确定
出来,所以至少请指出我的方向!
我了解了pmw,但我想先看看是否有任何方法可以使用tkinter。

我正在为DnD骰子写一个足够简单的程序,我有一个OptionMenu
包含某些人需要玩的骰子。 我也有一个输入字段
我的默认选项中未包含的模具。 我的问题是,即使新
选项添加成功,选项未排序。

我通过添加新选项时销毁OptionMenu解决了这一问题,
排序我的列表,然后从头开始重建OptionMenu,但是我正在使用
当时的位置管理器方法,我后来不得不重写程序,因为我有
一些解决问题。 我现在正在使用包管理器并销毁/重建
除非我想“重新”打包我所有的小部件或为其制作专属标签,否则这不是一个选择!

这是我的代码的工作示例:

from tkinter import *

class DropdownExample(Frame):
    def __init__(self, master = None):
        Frame.__init__(self, master)
        self.pack(fill = 'both', expand = True)

        # Add Option Button
        self.addOptBtn = Button(self, text = "Add Option", command = self.add_option)

        # Option Input Field
        self.newOpt = IntVar()
        self.newOpt.set("Type a number")

        self.optIn = Entry(self)
        self.optIn['textvariable'] = self.newOpt

        # Dropdown Menu
        self.myOptions = [0, 1, 2]

        self.selOpt = IntVar()
        self.selOpt.set("Options")

        self.optMenu = OptionMenu(self, self.selOpt, *self.myOptions)

        # Positioning
        self.addOptBtn.pack(side = 'left', padx = 5)
        self.optIn.pack(side = 'left', padx = 5)
        self.optMenu.pack(side = 'left', padx = 5)


    def add_option(self):
        self.numToAdd = ""
        self.counter = 0

        try:
            self.numToAdd = int(self.optIn.get())                                           # Integer validation

            while self.counter < len(self.myOptions):                                       # Comparison loop & error handling
                if self.numToAdd == self.myOptions[self.counter]:
                    print("Already exists!")                    
                    break;

                elif self.numToAdd < 0:
                    print("No less than 0!")
                    break;

                elif self.counter < len(self.myOptions)-1:
                    self.counter += 1

                else:                                                                       # Dropdown menu option addition
                    self.myOptions.append(self.numToAdd)
                    self.myOptions.sort()

                    self.optMenu['menu'].add_command(label = self.numToAdd)

                    self.selOpt.set(self.numToAdd)

                    print("Added succesfully!")

                    self.counter += 2

        except ValueError:
            print("Type ONLY numbers!")


def runme():
    app = DropdownExample()
    app.master.title("Dropdown Menu Example")
    app.master.resizable(0, 0)
    app.mainloop()

runme()

我在Windows 7上使用Python 3.3

Menu有一组insert_something()方法。 您必须使列表与每个插入物( bisect模块)一起排序。

from tkinter import *
import bisect

...
                else:                                                                       # Dropdown menu option addition
                    index = bisect.bisect(self.myOptions, self.numToAdd)
                    self.myOptions.insert(index, self.numToAdd)
                    self.optMenu['menu'].insert_command(index, label=self.numToAdd)
                    self.selOpt.set(self.numToAdd)
                    print("Added succesfully!", self.myOptions)
                    self.counter += 2

替换行:

self.optMenu['menu'].add_command(label = self.numToAdd)

有:

for dit in self.myOptions:
    self.optMenu['menu'].delete(0)
for dat in self.myOptions:
    self.optMenu['menu'].add_command(label = dat)

需要注意的是,“ add_command”将该项添加到菜单,而“ delete”则采用该项的索引。

看来,无论我选择遵循哪种方式,都归结为更新
小部件。
如果需要,更新框架通常会重绘部分或整个框架。
所以我回到了第一广场。 我销毁了小部件,更新了列表,然后再次创建了它。
至于位置,我仅将标签用作OptionMenu的背景。
这样,我可以随意删除/替换OptionMenu,而无需移动任何小部件,只要它停留在标签内,并且是其中唯一的一个,当然,只要我不移动任何标签即可。

这可能是一种解决方法,而不是解决方案,但可以完成工作,并且需要一点点
可以将其用于具有相同问题的任何其他小部件。

from tkinter import *

class DropdownExample(Frame):
    def __init__(self, master = None):
        Frame.__init__(self, master)
        self.pack(fill = 'both', expand = True)

        # Add Option Button
        self.addOptBtn = Button(self, text = "Add Option", command = self.add_option)

        # Option Input Field
        self.newOpt = IntVar()
        self.newOpt.set("Type a number")

        self.optIn = Entry(self)
        self.optIn['textvariable'] = self.newOpt

        # Dropdown Menu
        # menu's label
        self.optMenuLabel = Label(self)

        # option menu
        self.myOptions = [0, 1, 2]

        self.selOpt = IntVar()
        self.selOpt.set("Options")

        self.optMenu = OptionMenu(self.optMenuLabel, self.selOpt, *self.myOptions)

        # Positioning
        self.addOptBtn.pack(side = 'left', padx = 5)
        self.optIn.pack(side = 'left', padx = 5)
        self.optMenuLabel.pack(side = 'left', padx = 5)
        self.optMenu.pack(side = 'left', padx = 5)


    def add_option(self):
        self.numToAdd = ""
        self.counter = 0

        try:
            self.numToAdd = int(self.optIn.get())                                           # Integer validation

            while self.counter < len(self.myOptions):                                       # Comparison loop & error handling
                if self.numToAdd == self.myOptions[self.counter]:
                    print("Already exists!")                    
                    break;

                elif self.numToAdd < 0:
                    print("No less than 0!")
                    break;

                elif self.counter < len(self.myOptions)-1:
                    self.counter += 1

                else:                                                                       # Dropdown menu option addition
                    self.myOptions.append(self.numToAdd)
                    self.myOptions.sort()

                    self.selOpt.set(self.numToAdd)

                    self.optMenu.destroy()
                    self.optMenu = OptionMenu(self.optMenuLabel, self.selOpt, *self.myOptions)
                    self.optMenu.pack(side = 'left', padx = 5)

                    print("Added succesfully!", self.myOptions)
                    break;

        except ValueError:
            print("Type ONLY numbers!")


def runme():
    app = DropdownExample()
    app.master.title("Dropdown Menu Example")
    app.master.resizable(0, 0)
    app.mainloop()

runme()

暂无
暂无

声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.

 
粤ICP备18138465号  © 2020-2024 STACKOOM.COM