如何在Windows 10控制台中使用ANSI转义序列的新支持?

时间:2016-04-21 05:07:18

标签: python cmd windows-10 windows-console

最新的Windows 10更新包括conhost.exe中的support for ANSI escape sequences

enter image description here

我已经能够确认在cmd.exe中正确地获取了转义序列,因此我有必要的更新。特别是,我尝试键入隐藏光标的prompt $e[?25l,然后键入prompt $e[?25h,再次显示光标。

但是,如果我启动Python解释器,然后执行以下操作:

>>> import sys
>>> sys.stdout.write("\033[?25l")

好吧,光标没有被隐藏。如何以正确的方式设置,以便控制台能够从Python获取转义序列?

3 个答案:

答案 0 :(得分:20)

问题是Python解释器无法启用ANSI转义序列的处理。 ANSI序列在Windows命令提示符下工作,因为cmd确实启用了它们。如果从命令提示符启动Python,您将发现ANSI序列确实有效,包括用于启用和禁用游标的序列。那是因为cmd已经为该控制台窗口启用了它们。

如果你想要点击某些内容来启动启用ANSI转义的Python解释器,你可以创建一个运行命令的快捷方式,如cmd /c C:\PythonXY\python

另一个更难的解决方案是通过调用设置了ENABLE_VIRTUAL_TERMINAL_PROCESSING标志的SetConsoleMode Windows API,使用ctypes为控制台窗口启用ANSI转义序列处理。例如:

import ctypes

kernel32 = ctypes.windll.kernel32
kernel32.SetConsoleMode(kernel32.GetStdHandle(-11), 7)

答案 1 :(得分:1)

我对here提出的一些代码的改编应该有助于您入门。在 Windows 10 上启用 ANSI VT 模式(虚拟终端处理)。为 stdout 1 stderr 传递参数值2

def _windows_enable_ANSI(std_id):
    """Enable Windows 10 cmd.exe ANSI VT Virtual Terminal Processing."""
    from ctypes import byref, POINTER, windll, WINFUNCTYPE
    from ctypes.wintypes import BOOL, DWORD, HANDLE

    GetStdHandle = compat_ctypes_WINFUNCTYPE(
        HANDLE,
        DWORD)(('GetStdHandle', windll.kernel32))

    GetFileType = compat_ctypes_WINFUNCTYPE(
        DWORD,
        HANDLE)(('GetFileType', windll.kernel32))

    GetConsoleMode = compat_ctypes_WINFUNCTYPE(
        BOOL,
        HANDLE,
        POINTER(DWORD))(('GetConsoleMode', windll.kernel32))

    SetConsoleMode = compat_ctypes_WINFUNCTYPE(
        BOOL,
        HANDLE,
        DWORD)(('SetConsoleMode', windll.kernel32))

    if std_id == 1:       # stdout
        h = GetStdHandle(-11)
    elif std_id == 2:     # stderr
        h = GetStdHandle(-12)
    else:
        return False

    if h is None or h == HANDLE(-1):
        return False

    FILE_TYPE_CHAR = 0x0002
    if (GetFileType(h) & 3) != FILE_TYPE_CHAR:
        return False

    mode = DWORD()
    if not GetConsoleMode(h, byref(mode)):
        return False

    ENABLE_VIRTUAL_TERMINAL_PROCESSING = 0x0004
    if (mode.value & ENABLE_VIRTUAL_TERMINAL_PROCESSING) == 0:
        SetConsoleMode(h, mode.value | ENABLE_VIRTUAL_TERMINAL_PROCESSING)
    return True

答案 2 :(得分:0)

colorama软件包在Windows中启用ANSI代码。

用法:

from colorama import init
init()

之后,ANSI代码应该可以工作了。