gpt4 book ai didi

python - 动态设置 tkinter Spinbox 的范围

转载 作者:太空宇宙 更新时间:2023-11-04 03:52:36 36 4
gpt4 key购买 nike

我正在使用 Python 3.x 创建一个多语言“日历”小部件,该小部件具有用于年、月、日等的独立旋转框。要求是无论使用什么小部件,都绝不能隐藏 GUI 的任何部分时间,从而排除下拉列表等,使旋转框成为显而易见的选择。

如果有人知道在哪里可以找到纯 python/tkinter“日期和时间输入”widgit(理想地处理闰年和所有月末情况)的公共(public)域源代码,请指点我到那个存储库将不胜感激。

我能找到的最接近的解决方案是这个 OptionMenu 解决方案 (Change OptionMenu based on what is selected in another OptionMenu),但由于“不隐藏”的要求,它不能使用。以此为灵感(在查看 http://infohost.nmt.edu/tcc/help/pubs/tkinter/web/spinbox.html 上的 Spinbox 文档后)我想出了下面的代码。为了简单起见,我删除了处理闰年和月末问题的代码。除了两个问题外,该代码有效:

问题 1 – 我无法让“DaySpinBox”动态调整其范围以将其自身设置为“MonthSpinBox”中的月份:

根据我阅读 Mark Lutz 的“Python 编程”一书(第三版,第 383 和 384 页)的方式,我应该能够动态设置 DaySpinBox 的最大/最后一天(通过使用“values=”或“to= ”和“from_=”选项)以匹配出现在 MonthSpinBox 中的月份。我正在跟踪 MonthSpinBox 变量(“SelectedMonth”),并且根据书中的内容,我尝试使用这些方法更新 DaySpinBox:1 - 使用“to= SelectedMonth”。对此的一些尝试在创建 DaySpinBox 的调用中显示为注释。 (我是 lambda 的新手,也许我没有正确使用它们?),
2 - 使用“DaySpinBox[‘to’] = SelectedMonth”,以及3 - 使用“DaySpinBox.config( to = SelectedMonth )”。
我也尝试过在所有方法中使用“values=”但没有成功(首选“to=...”选项)。

问题 1:是否可以将 Spinbox 设置为使用动态范围,或者正如我开始怀疑的那样,它们的范围在创建 Spinbox 后是否无法更改?

问题 2:经过大量网络搜索后,我没有发现任何明确说明哪些 Spinbox 参数(“to=”、“from=”和“value=”关键字)可以动态更改而哪些不能动态更改的内容 – 哪里我可以找到此信息(适用于所有小部件类型)吗?

问题 2 – MonthSpinBox 总是初始化为一月而不是当前月份

我正在使用“textvariable”关键字将年、月和日旋转框初始化为“今天”。这适用于 YearSpinBox 和 DaySpinBox 但不适用于 MonthSpinBox。 (恼人的是,我认为 MonthSpinBox 正在工作,但我在尝试修复 DaySpinBox 时破坏了它)。唯一明显的区别是 Year 和 Day 旋转框使用整数,而 Month 旋转框使用字符串。有什么建议吗?

对于这两个问题,我都考虑了 LEGB 问题的可能性,但没有嵌套函数,因此变量隐藏应该不是问题 - 除非我的变量之一正在复制和隐藏在 tkinter 等中定义的变量。

我错过了什么?

顺便说一句,我知道使用全局变量的优缺点,也知道 PEP8 ( http://www.python.org/dev/peps/pep-0008/ ) - 请参阅目录中的第二个项目符号。 ;>)

谢谢。

from tkinter  import *
from tkinter import ttk
from datetime import datetime
from collections import OrderedDict

root = Tk() # put here so IntVar (etc.) is useable during initialization

StartUpDateTime = datetime.now()
DefaultYear = StartUpDateTime.year
DefaultMonthNumber = StartUpDateTime.month
DefaultDayOfMonth = StartUpDateTime.day

# These are only used to build the Month name & length dictionary
YearAndMonthLengths = [ 365, 31, 28, 31, 30, 31, 30, 31, 31, 30, 31, 30, 31 ]
EnglishMonthNames = ( 'Entire Year', 'January', 'February', 'March',
'April', 'May', 'June', 'July', 'August', 'September',
'October', 'November', 'December' )
FrenchMonthNames = ( 'année entière', 'janvier', 'février', 'mars',
'avril', 'mai', 'juin', 'juillet', 'août', 'septembre',
'octobre', 'novembre', 'décembre' )
#SpanishMonthNames = ....
#ItalianMonthNames = ....

Month_Names = FrenchMonthNames
Month_Names = EnglishMonthNames # comment out this line to use French

# Create dictionary containing month names that appear in the Month spinbox.
# OrderedDict used because Month name (=key) field must be language-independent.

DaysInMonth = OrderedDict()
for i in range( 0, len( Month_Names ) ) :
DaysInMonth[ Month_Names[ i ] ] = YearAndMonthLengths[ i ]

DefaultMonthName = Month_Names[ DefaultMonthNumber ]
DefaultMonthLength = DaysInMonth[ DefaultMonthName ]
# Initialize the Spinbox interface variables to todays date
SelectedYear = IntVar( value = DefaultYear )
SelectedMonthName = StringVar( value = DefaultMonthName )
SelectedMonthLength = IntVar( value = DefaultMonthLength )
SelectedDay = IntVar( value = DefaultDayOfMonth )

#-------------------------------------------------------------------------------

def GetChosenMonthLength( *args ) :

global SelectedMonthLength
SelectedMonthLength.set( DaysInMonth[ SelectedMonthName.get() ] )
return

#-------------------------------------------------------------------------------

mainframe = Frame( root )
mainframe.grid( row = 0, column = 0 )
# Show today's date. If it's July 17th, 2023 the spinboxes must be initialized
# to "July" ("julliet" if using French), "17" and "2023".
YearSpinBox = Spinbox( mainframe,
from_ = 2000, to = 2050,
textvariable = SelectedYear,
wrap = TRUE, state = 'readonly', width = 5 )
MonthSpinBox = Spinbox( mainframe,
values = tuple( DaysInMonth.keys() )[1:],
textvariable = SelectedMonthName,
wrap = TRUE, state = 'readonly', width = 10 )
DaySpinBox = Spinbox( mainframe,
from_ = 1,
to = SelectedMonthLength.get(),
# to = DaysInMonth[ SelectedMonthName.get() ],
# values = tuple( str( i ) for i in range( 1, SelectedMonthLength.get() + 1 ) ),
# values = lambda : tuple( str( i ) for i in range( 1, DaysInMonth[ SelectedMonthName.get() ] + 1 ) )
textvariable = SelectedDay,
wrap = TRUE, state = 'readonly', width = 16
)
MonthSpinBox.grid( row = 0, column = 0 )
DaySpinBox.grid( row = 0, column = 1 )
YearSpinBox.grid( row = 0, column = 2 )

SelectedMonthName.trace( 'w', GetChosenMonthLength )

mainloop()

顺便说一句,示例代码省略了证明所有 Spinbox 接口(interface)变量都被正确更新的调试代码 - 包括包含更改/新月份的最后一天的变量(使用“发送”到 Day Spinbox TO=”关键字)。

最佳答案

所有小部件的所有选项始终可以动态配置。如果我没记错的话,只有一个异常(exception),这是一个几乎没有人使用过的功能:框架上的 class 选项。

与更新日期旋转框相关,我没有看到您尝试更新它的位置,所以我不确定您认为它应该更新的原因。要更新旋转框,请将命令附加到月份旋转框,然后在回调中进行更新。

例如:

def update_days():
maxdays = int(DaysInMonth[SelectedMonthName.get()])
DaySpinBox.config(to=maxdays)

MonthSpinBox = Spinbox(...,command=update_days)

至于为什么月份没有设置为当前月份……我就不知道了。看起来你做的事情是对的。这个和其他小部件之间的唯一区别是这是一个 StringVar 而不是 IntVar,并且月份旋转框有一个 values 参数。也许这是由这种差异触发的 tkinter 中的错误或记录不完整的功能。

一个简单的解决方法是在创建月份旋转框后添加 SelectedMonthName.set(DefaultMonthName) 以将 stringvar 重置为正确的默认值。

关于python - 动态设置 tkinter Spinbox 的范围,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/20528596/

36 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com