Skip to main content

A powerful Python library for terminal string styling, ANSI color manipulation, ASCII art generation, and SVG terminal rendering.

Reason this release was yanked:

The project modules were not included because of an incorrect pyproject.toml configuration.

Project description

pyansistring

pyansistring Banner

CI Build Coverage PyPI Version PyPI Python version PyPI downloads License Ruff

ANSI-aware string formatting for Python CLIs.

pyansistring gives you a string type that keeps styling attached while you keep using familiar string operations. You can color text, apply SGR attributes, generate gradients, color ASCII art, and export styled output to SVG.

Why pyansistring

  • Keeps styling aligned with text during common string operations.
  • Supports 4-bit, 8-bit, and 24-bit (truecolor) foreground/background/underline colors.
  • Works well for CLI tools, logs, dashboards, and terminal UX.
  • Includes practical APIs for gradients, ASCII art coloring, and SVG export.

Features

  • ANSIString class that subclasses Python str.
  • Style-preserving operations for concatenation, slicing, splitting, joining, replacing, stripping, case transforms, and formatting (f-strings, .format()).
  • SGR styling and attributes: bold, dim, italic, underline, strikethrough, invert, and advanced underline modes (single, double, curly, dotted, dashed).
  • Color channels: 4-bit ANSI, 8-bit palette, and 24-bit RGB for foreground, background, and underlines.
  • Targeting modes: apply to full strings, slice ranges, or word matches (case-sensitive or insensitive).
  • Gradient engine: RGB or HSL interpolation, coordinate-based gradients for multiline text, and out-of-bounds handling.
  • SVG export: render text or path modes with per-character coloring and optional custom fonts.
  • Art registries: built-in ASCII art, load from TOML, custom color generators, and an optional cowsay adapter.
  • ANSI parsing: convert raw ANSI-encoded strings back into ANSIString instances with styles intact.
  • Large constants base: extensive predefined color constants and palettes, plus SGR/regex helpers for easy access.
  • Terminal-oriented formatting controls: supports modern and compatibility SGR formatting modes to improve behavior across ANSI-capable terminals.
  • Performance-minded internals: cached line-start indexing, style object caching, and change-tracked re-rendering to reduce repeated work.
  • Comprehensive test suite covering edge cases around string operations, style preservation, gradient calculations, and terminal rendering.

Requirements

  • Python 3.11+

Installation

Install the base package:

pip install pyansistring

Install extras when needed:

pip install pyansistring[img]            # For SVG export (installs fontTools)
pip install pyansistring[adapter-cowsay] # For the cowsay adapter
pip install pyansistring[adapters]       # All adapters
pip install pyansistring[all]            # Install everything

Quick Start

from pyansistring import ANSIString, Foreground, Background, SGR

text = (
    ANSIString("Hello, World!")
    .fg_4b(Foreground.YELLOW)
    .bg_4b(Background.BLUE)
    .style(SGR.BOLD)
)

print(text)

Usage Examples

The examples below are generated by examples/generate_usage_svg.py.

Unstyled text

from pyansistring import ANSIString

print(ANSIString("Hello, World!"))

unstyled

Whole-string styling

from pyansistring import ANSIString, Foreground, Background, SGR

print(
    ANSIString("Hello, World!")
    .fg_4b(Foreground.YELLOW)
    .bg_4b(Background.BLUE)
    .style(SGR.BOLD)
)

whole

Style by slice

from pyansistring import ANSIString, Foreground, Background, SGR

print(
    ANSIString("Hello, World!")
    .fg_4b(Foreground.YELLOW, (0, 5), (7, 12))
    .bg_4b(Background.BLUE, (7, 12))
    .style(SGR.BOLD, (7, 12))
)

slice

Style by words

from pyansistring import ANSIString, Foreground, Background, SGR

print(
    ANSIString("Hello, World!")
    .fg_4b_words(Foreground.YELLOW, "Hello", "World")
    .bg_4b_words(Background.BLUE, "World")
    .style_words(SGR.BOLD, "Hello", "World")
)

words

SGR attributes

from pyansistring import ANSIString, SGR

print(ANSIString("Hello, World!").style(SGR.BOLD).style(SGR.UNDERLINE))

sgr

4-bit, 8-bit, and 24-bit colors

from pyansistring import ANSIString, Foreground, Background

print(ANSIString("Hello, World!").fg_4b(Foreground.YELLOW).bg_4b(Background.BLUE))
print(ANSIString("Hello, World!").fg_8b(11).bg_8b(4).ul_8b(74))
print(ANSIString("Hello, World!").fg_24b(255, 255, 0).bg_24b(0, 0, 238).ul_24b(135, 175, 215))

4bit 8bit rgb

Underline modes

from pyansistring import ANSIString, UnderlineMode

print(
    ANSIString("Hello, World!")
    .bg_24b(255, 255, 255)
    .ul_24b(255, 0, 0)
    .style(UnderlineMode.DOUBLE)
)

underline

Rainbow

from pyansistring import ANSIString

print(ANSIString("Hello, World! This is rainbow text!").rainbow(fg=True))

rainbow

Gradient APIs

from pyansistring import ANSIString

print(
    ANSIString("Hello, World! This is gradient text!")
    .gradient([(84, 161, 255), (233, 200, 216)], fg=True)
)

print(
    ANSIString("Hello, colorful gradient world!")
    .gradient_words([(255, 99, 71), (255, 215, 0)], "Hello", "world", case_sensitive=False, fg=True)
)

print(
    ANSIString("HELLO\nworld")
    .gradient_coordinates(
        [(255, 0, 120), (0, 200, 255)],
        (1, 1),
        (2, 1),
        (3, 1),
        (4, 1),
        (5, 1),
        index_base=1,
        fg=True,
    )
)

gradient
gradient_words
gradient_coordinates

ArtRegistry and custom generators

from pyansistring import (
    ArtRegistry,
    ColorGeneratorContext,
    register_color_generator,
    unregister_color_generator,
)


def zigzag_generator(context: ColorGeneratorContext) -> list[tuple[int, int, int]]:
    palette = [
        (84, 161, 255),
        (255, 99, 71),
        (255, 215, 0),
        (120, 220, 160),
    ]
    return [palette[i % len(palette)] for i in range(max(2, context["step_count"]))]


register_color_generator("zigzag_readme_v1", zigzag_generator)
try:
    registry = ArtRegistry()
    registry.register(
        "ZIGZAG",
        " /\\/\\/\\/\\\n \\/\\/\\/\\/",
        colorings=(
            {
                "mode": "gradient",
                "colors": {
                    "generator": "zigzag_readme_v1",
                    "mode": "seeded",
                    "seed": 42,
                },
                "skip_whitespace": True,
                "fg": True,
            },
        ),
    )
    print(registry.get_colored_art("ZIGZAG"))
finally:
    unregister_color_generator("zigzag_readme_v1")

art_registry_zigzag

Parse ANSI text back into ANSIString

from pyansistring import ANSIString

raw = "\x1b[31mError\x1b[0m: file not found"
parsed = ANSIString.from_ansi(raw)

print(parsed.plain_text)
print(parsed)

Export ANSIString to SVG

from fontTools.ttLib import TTFont
from pyansistring import ANSIString, SGR

font = TTFont("path/to/font.ttf")
styled = ANSIString("SVG output").style(SGR.BOLD).fg_24b(90, 170, 255)

svg_code = styled.to_svg(
    font=font,
    font_size_px=16,
    convert_text_to_path=False,
)

For a complete terminal tour, run examples/showcase.py.
For a focused ArtRegistry walkthrough, run examples/art_registry_demo.py.

Contributing

Contributions are what make the open source community such an amazing place to learn, inspire, and create. Any contributions you make are greatly appreciated.

  1. Fork the Project
  2. Create your Feature Branch (git checkout -b feature/AmazingFeature)
  3. Commit your Changes (git commit -m 'Add some AmazingFeature')
  4. Push to the Branch (git push origin feature/AmazingFeature)
  5. Open a Pull Request

License

Distributed under the MIT License. See LICENSE for more information.

Project details


Download files

Download the file for your platform. If you're not sure which to choose, learn more about installing packages.

Source Distribution

pyansistring-0.3.0.tar.gz (8.9 kB view details)

Uploaded Source

Built Distribution

If you're not sure about the file name format, learn more about wheel file names.

pyansistring-0.3.0-py3-none-any.whl (7.6 kB view details)

Uploaded Python 3

File details

Details for the file pyansistring-0.3.0.tar.gz.

File metadata

  • Download URL: pyansistring-0.3.0.tar.gz
  • Upload date:
  • Size: 8.9 kB
  • Tags: Source
  • Uploaded using Trusted Publishing? Yes
  • Uploaded via: twine/6.1.0 CPython/3.13.7

File hashes

Hashes for pyansistring-0.3.0.tar.gz
Algorithm Hash digest
SHA256 04028db2fbb3f873120e5f7d388e78c34887616422cf0414de9b9a13cda59e1c
MD5 eabe9c2ec9c465fd9d49f66548b30aaa
BLAKE2b-256 de0887183f73e914ae7b025a07113b0cfdf9cc26c5f869e77d80031f9806a4c0

See more details on using hashes here.

Provenance

The following attestation bundles were made for pyansistring-0.3.0.tar.gz:

Publisher: publish-to-pypi.yml on l1asis/pyansistring

Attestations: Values shown here reflect the state when the release was signed and may no longer be current.

File details

Details for the file pyansistring-0.3.0-py3-none-any.whl.

File metadata

  • Download URL: pyansistring-0.3.0-py3-none-any.whl
  • Upload date:
  • Size: 7.6 kB
  • Tags: Python 3
  • Uploaded using Trusted Publishing? Yes
  • Uploaded via: twine/6.1.0 CPython/3.13.7

File hashes

Hashes for pyansistring-0.3.0-py3-none-any.whl
Algorithm Hash digest
SHA256 6b5311a84bcebd9290e82a2941bb407ca04bbd1e927f53264fbe0d4099c892e9
MD5 0c7ea5d39996bb73d0e31f12a17e7169
BLAKE2b-256 ae68d2902fa26d66ff537c3701ccb21df218e25c20d76b109091c49e8140067b

See more details on using hashes here.

Provenance

The following attestation bundles were made for pyansistring-0.3.0-py3-none-any.whl:

Publisher: publish-to-pypi.yml on l1asis/pyansistring

Attestations: Values shown here reflect the state when the release was signed and may no longer be current.

Supported by

AWS Cloud computing and Security Sponsor Datadog Monitoring Depot Continuous Integration Fastly CDN Google Download Analytics Pingdom Monitoring Sentry Error logging StatusPage Status page