From 0b47db00ea993114cb6d820d8637ca6a4318c4b8 Mon Sep 17 00:00:00 2001 From: Frost Ming Date: Thu, 24 Jun 2021 16:52:40 +0800 Subject: [PATCH] Fix the truecolor display on windows Close #42 --- cfonts/cli.py | 7 +++--- cfonts/colors.py | 65 ++++++++++++++++++++++++++++++++++++++++++++---- cfonts/core.py | 4 --- 3 files changed, 63 insertions(+), 13 deletions(-) diff --git a/cfonts/cli.py b/cfonts/cli.py index b746c2c..dd536d1 100644 --- a/cfonts/cli.py +++ b/cfonts/cli.py @@ -63,9 +63,8 @@ def parse_args() -> argparse.Namespace: parser.add_argument( "-f", "--font", - default=consts.FontFaces.block, - choices=consts.FontFaces, - type=consts.FontFaces, + default=consts.FontFaces.block.value, + choices=[f.value for f in consts.FontFaces], help="Use to define the font face", ) parser.add_argument( @@ -146,7 +145,7 @@ def main() -> None: else: gradient = None options = { - "font": args.font.value, + "font": args.font, "colors": colors, "background": args.background, "align": args.align, diff --git a/cfonts/colors.py b/cfonts/colors.py index 30ede93..d9e5b39 100644 --- a/cfonts/colors.py +++ b/cfonts/colors.py @@ -1,9 +1,10 @@ """ Utility functions for handling terminal colors """ -import colorsys import os -from typing import Iterable, List, Mapping, NamedTuple, Tuple +from typing import Iterable, List, Mapping, NamedTuple, Tuple, no_type_check + +import colorama from .consts import ANSI_COLORS, ANSI_RGB @@ -17,6 +18,15 @@ class Style(NamedTuple): _Hsv = Tuple[float, float, float] +def support_truecolor() -> bool: + return os.name != "nt" or ( + os.getenv("ANSICON") is not None + or os.getenv("WT_SESSION") is not None + or "ON" == os.getenv("ConEmuANSI") + or "xterm" == os.getenv("Term") + ) + + def hex_to_rgb(hex_string: str) -> _Rgb: """Return a tuple of red, green and blue components for the color given as #rrggbb. @@ -33,12 +43,56 @@ def rgb_to_hex(rgb: _Rgb) -> str: return "#" + "".join("%02x" % c for c in rgb) +@no_type_check def rgb_to_hsv(rgb: _Rgb) -> _Hsv: - return colorsys.rgb_to_hsv(*rgb) + r, g, b = rgb + r /= 255 + g /= 255 + b /= 255 + + max_value = max(r, g, b) + min_value = min(r, g, b) + diff = max_value - min_value + + h, s, v = 0, diff / max_value if max_value > 0 else 0, max_value + + if max_value == min_value: + h = 0 + elif max_value == r: + h = 60 * (g - b) / diff + if g < b: + h += 360 + elif max_value == g: + h = 60 * (b - r) / diff + 120 + else: + h = 60 * (r - g) / diff + 240 + + return h, (s * 100), (v * 100) def hsv_to_rgb(hsv: _Hsv) -> _Rgb: - return tuple(int(c) for c in colorsys.hsv_to_rgb(*hsv)) # type: ignore + h, s, v = hsv + h /= 60 + s /= 100 + v /= 100 + hi = int(h) % 6 + + f = h - int(h) + p = 255 * v * (1 - s) + q = 255 * v * (1 - (s * f)) + t = 255 * v * (1 - (s * (1 - f))) + v *= 255 + + result = { + 0: (v, t, p), + 1: (q, v, p), + 2: (p, v, t), + 3: (p, q, v), + 4: (t, p, v), + 5: (v, p, q), + }[hi] + r, g, b = result + return int(r), int(g), int(b) def _color_distance(left: _Hsv, right: _Hsv) -> float: @@ -147,10 +201,11 @@ def rgb_style(self, color: _Rgb, background: bool) -> Style: return Style("\x1b[{};2;{};{};{}m".format(open_bit, r, g, b), close) -if (os.getenv("DISABLE_TRUECOLOR") or os.name == "nt") and not os.getenv( +if (os.getenv("DISABLE_TRUECOLOR") or not support_truecolor()) and not os.getenv( "ENABLE_TRUECOLOR" ): # Disable truecolor for windows pen = AnsiPen() + colorama.init() else: pen = TrueColorPen() diff --git a/cfonts/core.py b/cfonts/core.py index ff991ed..d3a1c20 100644 --- a/cfonts/core.py +++ b/cfonts/core.py @@ -13,13 +13,9 @@ import re from typing import List, Mapping, Optional, Tuple -import colorama - from .colors import pen from .consts import ALIGNMENT, CHARS, SIZE, BgColors, CandyColors, Colors, FontFaces -colorama.init() - class Font: colors: int