Python 以编程方式更改控制台字体大小

Rob*_*bin 4 python fonts ctypes cmd

我发现下面的代码应该以编程方式更改控制台字体大小。我在 Windows 10 上。

但是,无论我调整什么值,我似乎都无法控制字体大小,而且由于某种原因,运行此脚本时打开的控制台非常宽。

我不知道 ctypes 是如何工作的 - 我想要的只是从 Python 内部修改控制台字体的大小。

任何实际的工作解决方案?

import ctypes

LF_FACESIZE = 32
STD_OUTPUT_HANDLE = -11

class COORD(ctypes.Structure):
    _fields_ = [("X", ctypes.c_short), ("Y", ctypes.c_short)]

class CONSOLE_FONT_INFOEX(ctypes.Structure):
    _fields_ = [("cbSize", ctypes.c_ulong),
                ("nFont", ctypes.c_ulong),
                ("dwFontSize", COORD),
                ("FontFamily", ctypes.c_uint),
                ("FontWeight", ctypes.c_uint),
                ("FaceName", ctypes.c_wchar * LF_FACESIZE)]

font = CONSOLE_FONT_INFOEX()
font.cbSize = ctypes.sizeof(CONSOLE_FONT_INFOEX)
font.nFont = 12
font.dwFontSize.X = 11
font.dwFontSize.Y = 18
font.FontFamily = 54
font.FontWeight = 400
font.FaceName = "Lucida Console"

handle = ctypes.windll.kernel32.GetStdHandle(STD_OUTPUT_HANDLE)
ctypes.windll.kernel32.SetCurrentConsoleFontEx(
        handle, ctypes.c_long(False), ctypes.pointer(font))


print("Foo")
Run Code Online (Sandbox Code Playgroud)

Cri*_*ati 6

我想首先指出[SO]:通过 ctypes 从 Python 调用的 C 函数返回不正确的值(@CristiFati 的回答),它描述了大多数CTypes用户遇到的问题。

ctypes的主页(在以上列出的URL):[Python.Docs]:ctypes的-外国函数库的Python

我“稍微”更改了您的代码。

代码.py

#!/usr/bin/env python

import sys
from ctypes import POINTER, WinDLL, Structure, sizeof, byref
from ctypes.wintypes import BOOL, SHORT, WCHAR, UINT, ULONG, DWORD, HANDLE


LF_FACESIZE = 32
STD_OUTPUT_HANDLE = -11


class COORD(Structure):
    _fields_ = [
        ("X", SHORT),
        ("Y", SHORT),
    ]


class CONSOLE_FONT_INFOEX(Structure):
    _fields_ = [
        ("cbSize", ULONG),
        ("nFont", DWORD),
        ("dwFontSize", COORD),
        ("FontFamily", UINT),
        ("FontWeight", UINT),
        ("FaceName", WCHAR * LF_FACESIZE)
    ]


kernel32_dll = WinDLL("kernel32.dll")

get_last_error_func = kernel32_dll.GetLastError
get_last_error_func.argtypes = []
get_last_error_func.restype = DWORD

get_std_handle_func = kernel32_dll.GetStdHandle
get_std_handle_func.argtypes = [DWORD]
get_std_handle_func.restype = HANDLE

get_current_console_font_ex_func = kernel32_dll.GetCurrentConsoleFontEx
get_current_console_font_ex_func.argtypes = [HANDLE, BOOL, POINTER(CONSOLE_FONT_INFOEX)]
get_current_console_font_ex_func.restype = BOOL

set_current_console_font_ex_func = kernel32_dll.SetCurrentConsoleFontEx
set_current_console_font_ex_func.argtypes = [HANDLE, BOOL, POINTER(CONSOLE_FONT_INFOEX)]
set_current_console_font_ex_func.restype = BOOL


def main():
    # Get stdout handle
    stdout = get_std_handle_func(STD_OUTPUT_HANDLE)
    if not stdout:
        print("{:s} error: {:d}".format(get_std_handle_func.__name__, get_last_error_func()))
        return
    # Get current font characteristics
    font = CONSOLE_FONT_INFOEX()
    font.cbSize = sizeof(CONSOLE_FONT_INFOEX)
    res = get_current_console_font_ex_func(stdout, False, byref(font))
    if not res:
        print("{:s} error: {:d}".format(get_current_console_font_ex_func.__name__, get_last_error_func()))
        return
    # Display font information
    print("Console information for {:}".format(font))
    for field_name, _ in font._fields_:
        field_data = getattr(font, field_name)
        if field_name == "dwFontSize":
            print("    {:s}: {{X: {:d}, Y: {:d}}}".format(field_name, field_data.X, field_data.Y))
        else:
            print("    {:s}: {:}".format(field_name, field_data))
    while 1:
        try:
            height = int(input("\nEnter font height (invalid to exit): "))
        except:
            break
        # Alter font height
        font.dwFontSize.X = 10  # Changing X has no effect (at least on my machine)
        font.dwFontSize.Y = height
        # Apply changes
        res = set_current_console_font_ex_func(stdout, False, byref(font))
        if not res:
            print("{:s} error: {:d}".format(set_current_console_font_ex_func.__name__, get_last_error_func()))
            return
        print("OMG! The window changed :)")
        # Get current font characteristics again and display font size
        res = get_current_console_font_ex_func(stdout, False, byref(font))
        if not res:
            print("{:s} error: {:d}".format(get_current_console_font_ex_func.__name__, get_last_error_func()))
            return
        print("\nNew sizes    X: {:d}, Y: {:d}".format(font.dwFontSize.X, font.dwFontSize.Y))


if __name__ == "__main__":
    print("Python {:s} on {:s}\n".format(sys.version, sys.platform))
    main()
Run Code Online (Sandbox Code Playgroud)

注意事项

  • CTypes允许类似于C 的低级别访问(只有语法是Python

  • 代码使用[MS.Docs]: SetConsoleTextAttribute 函数

  • 使用ctypes.wintypes常量(引用标准CTypes类型)(使代码具有类似Win 的风格)

  • 它非常(几乎痛苦)长(也是因为我添加了正确的错误处理)

  • 替代地,所建议的在的答案之一[SO]:在Windows更改控制台字体(其中复制从代码),将安装有3三方模块(例如[GitHub上]:mhammond / pywin32 -的Python对于 Windows (pywin32) 扩展,它是WINAPI 上Python包装器),它需要编写更少的代码,因为PythonC之间的桥接已经实现,而且上述功能可能只需几行即可完成

  • 正如我在代码中评论的那样,设置COORD.X似乎被忽略了。但它会在设置COORD.Y时自动设置(接近于COORD.Y // 2- 可能是为了保持纵横比)。在我的机器上(Win 10 pc064),默认值为16。您可能希望在最后将其重新设置,以避免使控制台处于“挑战”状态(显然,Win调整了cmd窗口大小,以(某种程度)与字体大小同步):

    图像0