繁体   English   中英

如何在不知道小部件的字体系列/大小的情况下更改小部件的字体样式?

[英]How to change a widget's font style without knowing the widget's font family/size?

有没有办法在不知道小部件的字体系列和字体大小的情况下更改Tkinter小部件的字体样式?

用例:我们使用标准的Tkinter小部件( LabelEntryText等)创建我们的 UI。 当我们的应用程序运行时,我们可能希望使用.config()方法将这些小部件的字体样式动态更改为粗体和/或斜体。 不幸的是,如果不指定字体的系列和大小,似乎无法指定字体规范。

以下是我们想要做的示例,但这些示例都不起作用:

widget.config(font='bold')

widget.config(font=( None, None, 'bold' ))

有比使用.config()更改应用程序字体更好的方法,尤其是当您的目标是更改整组小部件(或所有小部件)的字体时。

Tk 真正伟大的特性之一是“命名字体”的概念。 命名字体的美妙之处在于,如果您更新字体,所有使用该字体的小部件都会自动更新。 因此,配置您的小部件一次以使用这些自定义字体,然后更改属性是微不足道的。

这是一个快速示例:

# python 2 imports
# import Tkinter as tk
# import tkFont

# python 3 imports
import tkinter as tk
import tkinter.font as tkFont


class App:
    def __init__(self):
        root=tk.Tk()
        # create a custom font
        self.customFont = tkFont.Font(family="Helvetica", size=12)

        # create a couple widgets that use that font
        buttonframe = tk.Frame()
        label = tk.Label(root, text="Hello, world", font=self.customFont)
        text = tk.Text(root, width=20, height=2, font=self.customFont)
        buttonframe.pack(side="top", fill="x")
        label.pack()
        text.pack()
        text.insert("end","press +/- buttons to change\nfont size")

        # create buttons to adjust the font
        bigger = tk.Button(root, text="+", command=self.OnBigger)
        smaller = tk.Button(root, text="-", command=self.OnSmaller)
        bigger.pack(in_=buttonframe, side="left")
        smaller.pack(in_=buttonframe, side="left")

        root.mainloop()

    def OnBigger(self):
        '''Make the font 2 points bigger'''
        size = self.customFont['size']
        self.customFont.configure(size=size+2)

    def OnSmaller(self):
        '''Make the font 2 points smaller'''
        size = self.customFont['size']
        self.customFont.configure(size=size-2)

app=App()

如果您不喜欢这种方法,或者您想将自定义字体基于默认字体,或者您只是更改一两种字体来表示状态,则可以使用font.actual来获取实际大小给定小部件的字体。 例如:

import Tkinter as tk
import tkFont

root = tk.Tk()
label = tk.Label(root, text="Hello, world")
font = tkFont.Font(font=label['font'])
print font.actual()

当我运行上面的我得到以下输出:

{'family': 'Lucida Grande', 
 'weight': 'normal', 
 'slant': 'roman', 
 'overstrike': False, 
 'underline': False, 
 'size': 13}

一个标签甚至更短:

from Tkinter import *
import Tkinter as tk
root = tk.Tk()

# font="-weight bold" does your thing
example = Label(root, text="This is a bold example.", font="-weight bold")
example.pack()

root.mainloop()

只需使用特定小部件的基本属性,假设您要更改标签的字体。 您可以使用以下语法:

mlabel = Label(text="Your text", font=("Name of your font",size))

此代码适用于 python 3.4

如果您使用的是命名字体,您可以使用几个语句来获得您想要的:

import tkFont
wfont = tkFont.nametofont(widget['font'])
wfont.config(weight='bold')

编辑以纳入 B. Oakley 的评论。

要在不接触或没有小部件的情况下获取默认字体,您可以使用默认字体的通用名称。

#!/usr/bin/env python3
import tkinter
import tkinter.font  # Python3!

tkinter.Tk()
default_font = tkinter.font.Font(font='TkDefaultFont')
print(default_font.actual())

将上述大部分信息归结为单个代码片段:

lbl = ttk.Label(blah, blah)   # Make a label
font = tkFont(lbl['font'])    # Get its font
font.config(weight='bold')    # Modify font attributes
lbl['font'] = font            # Tell the label to use the modified font

这允许独立于使用的字体更改字体属性(只要字体支持该属性)。

您也可以即时执行此操作,以创建真正令人作呕的字体效果。

虽然问这个问题已经有一段时间了,但我最近不得不为此实施一个解决方案,我认为值得分享。 函数 widget_font_config(...) 在 Python 2 和 3 上运行。

本质上,widget 字体的“当前值”被抓取、修改,然后放回去。 支持命名字体,默认的inplace_f值为True意味着命名字体将被保留并就地修改。 但是该标志也可以设置为False ,这将导致命名字体被替换为不同的命名字体,以防用户不希望小部件字体的更改渗透到所有其他使用命名字体。

def widget_font_config(widget, inplace_f=True, **kwargs):
    import sys
    if sys.version_info[0] == 2:
        import tkFont as tk_font
    else:
        import tkinter.font as tk_font
    inplace_f = kwargs.pop('inplace', inplace_f)
    font = None    
    if widget and 'font' in widget.config():
        current_font = widget.cget('font') #grabs current value
        namedfont_f = False
        try:
            font = tk_font.nametofont(current_font)
            namedfont_f = True
        except:
            font = current_font
        if namedfont_f and inplace_f:
            font.config(**kwargs)
        else:
            font_d = tk_font.Font(font=font).actual()
            font_d.update(**kwargs)
            font = tk_font.Font(**font_d)
            widget.config(font=font)
        widget.update_idletasks()
    return font

if __name__ == '__main__':
    import sys
    pyVers = sys.version_info[0] # .major
    if pyVers == 2:
        import Tkinter as tk, tkFont as tk_font
    else:
        import tkinter as tk, tkinter.font as tk_font
    def go():
        print(widget_font_config(root.label,  slant='roman', underline=1).actual())
        print(widget_font_config(root.button, overstrike=1).actual())
    root = tk.Tk()
    font_s = 'Courier 20 italic'
    font_d = dict(family='Courier', size=10, weight='normal', slant='italic')
    font = tk_font.Font(**font_d)
    root.label = tk.Label(text='Label {}'.format(font_s), font=font_s)
    root.label.pack()
    root.button = tk.Button(text='Button {}'.format(font), font=font, command=go)
    root.button.pack()
    root.mainloop()

我知道这个问题真的很老了,但为了谷歌的人,我还是要回答一下。

如果你只是想让文本更大一点,你可以做font=15 请注意,无论输入什么数字,这似乎总是将字体大小设置为 15。

如果想要一个确切的大小并且没有改变字体,你可以做font=('TkDefaultFont', 15)

'TkDefaultFont''TkDefaultFont'的默认字体)

您可以在创建时在小部件的参数中使用其中任何一个,或者稍后使用.config()


在 python 3.6.4 中工作

暂无
暂无

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

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