🎨 MS Paint in your terminal.
Go to file
2023-07-18 14:03:51 -04:00
.vscode VS Code: mark files as readonly 2023-06-12 00:43:42 -04:00
samples Add smaller versions of the Pipe Strip (and sequel) 2023-07-13 17:32:03 -04:00
scripts Specify UTF-8 encoding, since it's not default on Windows 2023-07-17 20:04:24 -04:00
src/textual_paint Implement tiling vs centering wallpaper 2023-07-18 14:03:51 -04:00
typings Generate type stubs for stransi and ochre 2023-05-09 16:58:25 -04:00
.gitattributes Try a more heavy-handed way of getting GitHub to detect the language 2023-06-09 23:24:55 -04:00
.gitignore Ignore temporary backup files (*~) 2023-05-16 14:35:00 -04:00
cspell.json Rasterize before setting as wallpaper 2023-07-18 14:03:51 -04:00
LICENSE.txt Initial commit 2023-04-10 16:51:39 -04:00
pyproject.toml Add python packaging files 2023-04-29 17:30:46 -04:00
pyrightconfig.json Enable reporting unnecessary "type: ignore" comments 2023-07-16 00:47:15 -04:00
question_icon.ans Make question mark bubble rounder, more oval like 2023-05-04 21:13:06 -04:00
README.md Implement tiling vs centering wallpaper 2023-07-18 14:03:51 -04:00
requirements.txt Make psutil optional 2023-07-17 19:18:31 -04:00
screenshot.svg Update screenshot, and screenshot methodology, and remove table wrapper 2023-06-11 22:59:37 -04:00
setup.cfg Add more package metadata 2023-06-27 16:15:56 -04:00

Textual Paint

MS Paint in your terminal.

This is a TUI (Text User Interface) image editor, inspired by MS Paint, built with Textual.

MS Paint like interface

Features

  • Open and save images
    • Fancy file dialogs
    • Drag and drop files to open
    • Warnings when overwriting an existing file, or closing with unsaved changes
    • Automatically saves a temporary .ans~ backup file alongside the file you're editing, for recovery in case of a crash
    • File formats, chosen by typing a file extension in the Save As dialog:
      • ANSI (.ans) — Note that while it handles many more ANSI control codes when loading than those that it uses to save files, you may have limited success loading other ANSI files that you find on the web, or create with other tools. ANSI files can vary a lot and even encode animations!
      • mIRC codes (.irc, .mirc) — invented file extensions, and not to be confused with .mrc mIRC script files
      • Plain Text (.txt)
      • SVG (.svg) — can open SVGs saved by Textual Paint, which embed ANSI data; can also open some other SVGs that consist of a grid of rectangles and text elements. For fun, as a challenge, I made it quite flexible; it can handle uneven grids of unsorted rectangles. But that's only used as a fallback, because it's not perfect.
      • HTML (.htm, html) — write-only (opening not supported)
      • PNG (.png) — opens first frame of an APNG file
      • Bitmap (.bmp)
      • GIF (.gif) — opens first frame
      • TIFF (.tiff) — opens first frame
      • WebP (.webp) — opens first frame
      • JPEG (.jpg, .jpeg) — saving disabled because it's lossy (it would destroy your pixel art)
      • Windows Icon (.ico) — opens largest size in the file
      • Mac OS Icon (.icns) — opens largest size in the file; saving disabled because it requires specific sizes
      • Windows Cursor (.cur) — opens largest size in the file; saving not supported by Pillow (and it would need a defined hot spot)
      • See Pillow's documentation for more supported formats.
      • Note that metadata is not preserved when opening and saving image files. (This is however common for many image editors.)
  • Tools
    • Free-Form Select
    • Select
    • Eraser/Color Eraser
    • Fill With Color
    • Pick Color
    • Magnifier
    • Pencil
    • Brush
    • Airbrush
    • Text
    • Line
    • Curve
    • Rectangle
    • Polygon
    • Ellipse
    • Rounded Rectangle
  • Commands
    • Undo and Repeat (AKA redo)
    • Cut, Copy, Paste, Clear Selection (AKA delete), Select All
    • Paste From (insert file as selection), Copy To (save selection to file)
    • Flip/Rotate
    • Stretch/Skew
    • Invert Colors
    • Attributes (resize canvas; in the future may change color/text modes, maybe text encoding)
    • Clear Image
    • Edit Colors
    • most other commands too!
  • Color palette
  • Efficient screen updates and undo/redo history, by tracking regions affected by each action
    • You could totally use this program over SSH! Haha, this "what if" project could actually be useful. Of course, it should be mentioned that you can also run graphical programs over SSH, but this might be more responsive, or just fit your vibe better.
  • Brush previews
  • Menu bar
  • Status bar
  • Keyboard shortcuts
  • Localization into 26 languages: Arabic, Czech, Danish, German, Greek, English, Spanish, Finnish, French, Hebrew, Hungarian, Italian, Japanese, Korean, Dutch, Norwegian, Polish, Portuguese, Brazilian Portuguese, Russian, Slovak, Slovenian, Swedish, Turkish, Chinese, Simplified Chinese
  • Dark mode
  • Zooming works with text, despite running in the terminal :)

Usage

Command Line Options

$ python3 paint.py --help
usage: textual-paint [options] [filename]

Paint in the terminal.

positional arguments:
  filename              Path to a file to open. File will be created if it
                        doesn't exist.

options:
  -h, --help            show this help message and exit
  --version             show program's version number and exit
  --theme {light,dark}  Theme to use, either "light" or "dark"
  --language {ar,cs,da,de,el,en,es,fi,fr,he,hu,it,ja,ko,nl,no,pl,pt,pt-br,ru,sk,sl,sv,tr,zh,zh-simplified}
                        Language to use
  --ascii-only-icons    Use only ASCII characters for tool icons, no emoji or
                        other Unicode symbols
  --backup-folder FOLDER
                        Folder to save backups to. By default a backup is saved
                        alongside the edited file.
  --inspect-layout      Enables DOM inspector (F12) and middle click highlight,
                        for development
  --clear-screen        Clear the screen before starting; useful for
                        development, to avoid seeing outdated errors
  --restart-on-changes  Restart the app when the source code is changed, for
                        development
  --recode-samples      Open and save each file in samples/, for testing

Keyboard Shortcuts

  • Ctrl+D: Toggle Dark Mode
  • Ctrl+Q: Quit
  • Ctrl+Shift+S: Save As IF SHIFT IS DETECTED⚠️ it might trigger Save instead, and overwrite the open file!
  • Ctrl+Shift+Z: Redo IF SHIFT IS DETECTED⚠️ it might trigger Undo instead.

The rest match MS Paint's keyboard shortcuts:

  • Ctrl+S: Save
  • Ctrl+O: Open
  • Ctrl+N: New
  • Ctrl+T: Toggle Tools Box
  • Ctrl+W: Toggle Colors Box
  • Ctrl+Z: Undo
  • Ctrl+Y: Redo
  • F4: Redo
  • Ctrl+A: Select All
  • Delete: Clear Selection
  • Ctrl+C: Copy
  • Ctrl+V: Paste
  • Ctrl+X: Cut
  • Ctrl+E: Image Attributes
  • Ctrl+PageUp: Large Size
  • Ctrl+PageDown: Normal Size

Tips

You can draw with a character by clicking the selected color display area in the palette and then typing the character, or by double clicking the same area to pick a character from a list.

You can set the text color by holding Ctrl while clicking a color in the palette, or while double clicking a color to open the Edit Colors dialog.

You can display a saved ANSI file in the terminal with cat:

cat samples/ship.ans

To view all the sample files, run:

find samples -type f -exec file --mime-type {} \; | grep -v -e "image/png" -e "image/svg" | cut -d: -f1 | sort | xargs -I{} sh -c 'echo "File: {}"; cat "{}"; echo "\n-----------------------\n"'
Command Explanation Let's break down the command:
  1. find samples -type f -exec file --mime-type {} \;: This part uses the find command to locate all files (-type f) within the "samples" folder and its subdirectories. For each file, it executes the file --mime-type command to determine the file's MIME type. This outputs a line like "samples/ship.ans: text/plain".

  2. grep -v -e "image/png" -e "image/svg": This filters out any lines containing the MIME types "image/png" or "image/svg", effectively excluding PNG and SVG files. -v means "invert the match", so it will only output lines that don't match the given patterns.

  3. cut -d: -f1: This extracts only the file paths from the output of the file command, removing the MIME type information.

  4. sort: This sorts the file paths alphabetically.

  5. xargs -I{} sh -c 'echo "File: {}"; cat "{}"; echo "\n-----------------------\n"': Finally, this executes the sh -c command for each file, echoing the filename, catting its content, and adding a separator line.

This command will sort and display the content of all non-PNG files within the "samples" folder and its subdirectories. Remember to run this command in the directory where the "samples" folder is located.

To preview ANSI art files in file managers like Nautilus, Thunar, Nemo, or Caja, you can install the ansi-art-thumbnailer program I made to go along with this project.

Known Issues

  • Undo/Redo doesn't work inside the Text tool's textbox. Ctrl+Z will delete the textbox. (Also note that the Text tool works differently from MS Paint; it will overwrite characters and the cursor can move freely, which makes it better for ASCII art, but worse for prose.)
  • The Text tool's cursor doesn't blink.
  • The selection box border appears inside instead of outside (and lacks dashes). For the text box, I hid the border because it was too visually confusing, but it should also have an outer border.
  • Pick Color can't be cancelled (with Esc or by pressing both mouse buttons), since it samples the color continuously.
  • Pressing both mouse buttons stops the current tool, but doesn't undo the current action.
  • Due to limitations of the terminal, shortcuts using Shift or Alt might not work.
  • Menus are not keyboard navigable.
  • The Zoom submenu flickers as it opens, and may not always open in the right place.
  • The canvas flickers when zooming in with the Magnifier tool.
  • Some languages don't display correctly.
  • Large files can make the program very slow, as can magnifying the canvas. There is a 500 KB limit when opening files to prevent it from freezing.
  • Free-Form Select stamping/finalizing is incorrect when the selection is off-screen to the left or top.
  • Status bar description can be left blank when selecting a menu item. (I think the Leave event can come after closing, once the mouse moves.)
  • Menu items like Copy/Cut/Paste are not grayed out when inapplicable. Only unimplemented items are grayed out.
  • ANSI files (.ans) are treated as UTF-8 when saving and loading, rather than CP437 or Windows-1252 or any other encodings. Unicode is nice and modern terminals support it, but it's not the standard for ANSI files. There isn't really a standard for ANSI files.
  • ANSI files are loaded with a white background. This may make sense as a default for text files, but ANSI files either draw a background or assume a black background, being designed for terminals.
  • Hitting Enter in View Bitmap mode exits the mode but may also trigger a menu item. Menu items ought to be disabled when hidden, and View Bitmap should prevent the key event from taking other actions if possible.
  • Airbrush is continuous in space instead of time. It should keep spraying while the mouse stays still.
  • Error messages may not show up when opening a file fails.
  • Edit Colors dialog
    • Focus ring shows even while grid is not focused
    • Can show two cells as selected, instead of one across both grids
    • Custom colors order X/Y is different from MS Paint
    • Pressing enter in color grid should select color and close
    • Selection ring is hard to see in dark mode
    • Focus ring is invisible on a black color cell
    • When dragging on the color field or luminosity slider, the cursor can be seen to jump back to earlier places where the mouse was, before settling at the current position. (This may only be visible when the program is running slowly, such while debugging. I haven't observed this on the canvas, so maybe it has something to do with the dialog being on a separate layer.)
    • When opening the Edit Colors dialog, it may immediately close, if the mouse lines up with the "OK" or "Cancel" buttons. (This doesn't seem to currently happen, but I haven't knowingly fixed it. A git bisect turned up a bogus commit, possibly due to reproducing the behavior being unreliable. It also seems like it might depend on the specific layout of the dialog, which changed during development, and maybe even the terminal size.)

Compatibility

Python 3.10 or later is required.

Linux

Tested on Ubuntu 22, with GNOME Terminal, and VS Code's integrated terminal.

GNOME Terminal works best, with crisp triangles used for icons in dialogs, emoji support, and true color support.

macOS

Tested on OSX 10.14 (Mojave), with iTerm2, and VS Code's integrated terminal.

iTerm2 mostly works, but two tool icons are missing (Free-Form Select and Fill With Color). They show as a square with a question mark in it, and may cause the rest of the row of characters to be misaligned, including the canvas. (I carefully picked the symbols to avoid this on Ubuntu, so I may need to do the same for macOS, conditioning on the TERM_PROGRAM environment variable.)

In VS Code, only Free-Form Select shows as tofu (a missing character symbol), and there's no misalignment.

The default Terminal app has the same problems as iTerm2, plus borders are not rendered nicely, giving it a sort of frayed fabric look, and it's limited to 256 colors.

Windows

Textual Paint works with the new Windows Terminal, however, the Fill With Color tool icon is missing (shows as tofu), and the Pencil emoji causes misalignment of everything to the right of it, including the canvas.

It will not work properly with the old Windows console, which lacks emoji/Unicode support and true color support.

VS Code

Note that VS Code's integrated terminal tries to fix the contrast of text, including in the canvas, which is entirely inappropriate for an ANSI art editor, as it obscures the colors, and can indeed harm the contrast of the resulting document, by tricking you into thinking there's more contrast than there actually is.

To disable this, you can add this to your settings.json:

"terminal.integrated.minimumContrastRatio": 1

I ran into some issues with this, and had to set it to 1.1, but I'm not sure if that's necessary for everyone.

Development

Recommended: first, create a virtual environment:

python -m venv .venv
source .venv/bin/activate

Install Textual and other dependencies:

pip install -r requirements.txt

Run the app via Textual's CLI for live-reloading CSS support, and enable other development features:

textual run --dev "src.textual_paint.paint --clear-screen --inspect-layout --restart-on-changes"

Or run more basically:

python -m src.textual_paint.paint

Or install the CLI globally*:

pip install --editable .

Then run:

textual-paint

*If you use a virtual environment, it will only install within that environment.

--clear-screen is useful for development, because it's sometimes jarring or perplexing to see error messages that have actually been fixed, when exiting the program.

--inspect-layout enables a DOM inspector accessible with F12, which I built. It also lets you apply a rainbow highlight and labels to all ancestors under the mouse with middle click, but this is mostly obsolete/redundant with the DOM inspector now. The labels affect the layout, so you can also hold Ctrl to only colorize, and you can remember how the colors correspond to the labels, to build a mental model of the layout.

--restart-on-changes automatically restarts the program when any Python files change. This works by the program restarting itself directly. (Programs like modd or nodemon that run your program in a subprocess don't work well with Textual's escape sequences.)

There are also launch tasks configured for VS Code, so you can run the program from the Run and Debug panel. Note that it runs slower in VS Code's debugger.

To see logs, run textual console and then run the program via textual run --dev. This also makes it run slower.

Often it's useful to exclude events with textual console -x EVENT.

To test file encoding, run textual run --dev "src.textual_paint.paint --recode-samples".

If there are differences in the ANSI files, you can set up a special diff like this:

git config --local "diff.cat-show-all.textconv" "cat --show-all"

but you should check that cat --show-all samples/2x2.ans works on your system first. Also, note that it might not work with your Git GUI of choice; you may need to use the command line.

Troubleshooting

Unable to import 'app' from module 'src.textual_paint.paint'

  • Make sure to activate the virtual environment, if you're using one.
  • Make sure to run the program from the root directory of the repository.

ModuleNotFoundError: No module named 'src'

  • Make sure to run the program from the root directory of the repository.

ImportError: attempted relative import with no known parent package

  • paint.py can only be run as a module, not as a script. I just... I haven't had the heart to remove the shebang line.

Linting

pyright  # type checking
cspell-cli lint .  # spell checking

Update Dependencies

python -m pipreqs.pipreqs --ignore .history --force

License

MIT

Unicode Symbols and Emojis for Paint Tools

The first thing I did in this project was to collect possible characters to represent all the tool icons in MS Paint, to gauge how good of a recreation it would be possible to achieve, starting from looks. As it turns out, I didn't run into any significant roadblocks, so I ended up recreating MS Paint. Again.

These are the symbols I've found so far:

  • Free-Form Select: ✂️📐🆓🕸⚝⛤⛥⛦⛧🫥🇫/🇸◌⁛⁘ ⢼⠮
  • Select: ⬚▧🔲 ⣏⣹ ⛶
  • Eraser/Color Eraser: 🧼🧽🧹🚫👋🗑️▰▱
  • Fill With Color: 🌊💦💧🌈🎉🎊🪣🫗
  • Pick Color: 🎨💉💅💧📌📍⤤𝀃🝯🍶
  • Magnifier: 🔍🔎👀🔬🔭🧐🕵️‍♂️🕵️‍♀️
  • Pencil: ✏️✍️🖎🖊️🖋️✒️🖆📝🖍️
  • Brush: 🖌️🖌👨‍🎨🧑‍🎨💅
  • Airbrush: 💨ᖜ╔🧴🥤🫠
  • Text: 🆎📝📄📃🔤📜A
  • Line: 📏📉📈⟍𝈏╲⧹\
  • Curve: ↪️🪝🌙〰️◡◠~∼≈∽∿〜〰﹋﹏≈≋~⁓
  • Rectangle: ▭▬▮▯🟥🟧🟨🟩🟦🟪🟫◼️◻️▪️▫️
  • Polygon: ▙𝗟𝙇﹄』𓊋⬣⬟🔶🔷🔸🔹🔺🔻△▲
  • Ellipse: ⬭🔴🟠🟡🟢🔵🟣🟤🫧
  • Rounded Rectangle: ▢

The default symbols used may not be the best on your particular system, so I may add some kind of configuration for this in the future.

Cursor

A crosshair cursor could use one of +✜✛⊹✚╋╬⁘⁛⌖⯐, but whilst that imitates the look, it might be better to show the pixel under the cursor, i.e. character cell, surrounded by dashes, like this:

 ╻
╺█╸
 ╹

See Also

  • JavE, an advanced Java-based ASCII art editor
  • Playscii, a beautiful ASCII/ANSI art editor. This is also written in Python and MIT licensed, so I might take some code from it, for converting images to ANSI, for example. Who knows, maybe I could even try to support its file format.
  • cmdpxl, a pixel art editor for the terminal using the keyboard
  • pypixelart, a pixel art editor using vim-like keybindings, inspired by cmdpxl but not terminal-based