如何在不影响其他元素中的相同选项的情况下更改ttk stylename中元素的选项?
我发布了一个脚本here,其中tkinter程序员可以使用它来公开关于给定ttk stylename的所有元素及其选项。例如。对于stylename my.Vertical.TScrollbar
,你可以看到这个stylename有3个元素,每个元素都有一个名为background
的类似选项。
Stylename = my.Vertical.TScrollbar
Layout = [('Vertical.Scrollbar.trough', {'children': [('Vertical.Scrollbar.uparrow', {'side': 'top', 'sticky': ''}), ('Vertical.Scrollbar.downarrow', {'side': 'bottom', 'sticky': ''}), ('Vertical.Scrollbar.thumb', {'sticky': 'nswe', 'expand': '1'})], 'sticky': 'ns'})]
Element(s) = ['Vertical.Scrollbar.trough', 'Vertical.Scrollbar.uparrow', 'Vertical.Scrollbar.downarrow', 'Vertical.Scrollbar.thumb']
Vertical.Scrollbar.trough options: ('-borderwidth', '-troughcolor', '-troughrelief')
Vertical.Scrollbar.uparrow options: ('-background', '-relief', '-borderwidth', '-arrowcolor', '-arrowsize')
Vertical.Scrollbar.downarrow options: ('-background', '-relief', '-borderwidth', '-arrowcolor', '-arrowsize')
Vertical.Scrollbar.thumb options: ('-orient', '-width', '-relief', '-background', '-borderwidth')
tkinter文档教导可以使用.configure
方法更改样式名称的选项。下面是一个说明问题的脚本。当我发出命令
style.configure('my.Vertical.TScrollbar',
background=color1, troughcolor=color2,
arrowcolor=color3, borderwidth=10, width=40)
元素拇指,uparrow和downarrow的背景都将变为蓝色,并且所有4个元素的边界宽度变为10像素宽。如果我需要更好的控制,例如要仅更改拇指元素的背景或仅更改拇指元素的边框宽度,我该怎么做?
测试脚本:
#!/usr/bin/python3
# -*- coding: utf-8 -*-
import tkinter as tk
import tkinter.ttk as ttk
class App(ttk.Frame):
def __init__(self, parent=None, *args, **kwargs):
ttk.Frame.__init__(self, parent, *args, **kwargs)
color1 = 'blue'
color2 = 'light green'
color3 = 'white'
style=ttk.Style()
style.configure('my.Vertical.TScrollbar',
background=color1, troughcolor=color2,
arrowcolor=color3, borderwidth=10, width=40)
style.map('my.Vertical.TScrollbar',
background=[('active',color1),('!active',color1)],
arrowcolor=[('active',color3),('!active',color3)])
vscrollbar = ttk.Scrollbar(self,
orient='vertical',
style='my.Vertical.TScrollbar')
vscrollbar.pack(fill='y', side='right', expand='false')
canvas = tk.Canvas(self,
bd=0,
highlightthickness=0,
yscrollcommand=vscrollbar.set)
canvas.pack(side='left', fill='both', expand='true')
vscrollbar.config(command=canvas.yview)
if __name__ == "__main__":
root=tk.Tk()
app = App(root)
app.pack(side='left', fill='both', expand='true')
root.mainloop()
ttk中的元素使用flyweight pattern并且不存储值本身。相反,该样式包含与窗口小部件布局关联的元素使用的所有值。因此,使用后台配置值的所有元素都会从当前窗口小部件样式中读取相同的值。如果支持自定义颜色,则使用不同的选项名称。因此,从troughcolor选项中读取槽背景颜色。对于按钮,按钮本身有一个字段,因为背景用于聚焦环下方按钮后面的区域(如果可见)。
如果您真的想要获得更好的控制,您必须编写新的主题元素引擎,或者您可以使用image element engine使用图像定义新元素。如果您想尝试,有一些基于图像的主题导入到github的tcl示例。但是,ttk主题的重点是尝试尽可能多地使用系统提供的外观和感觉,并避免让程序员自定义UI。作为一般规则,您制作的自定义越多,就越有可能将废话视为其他人的系统。