mirror of
https://github.com/chubin/wttr.in
synced 2025-01-12 12:08:47 +00:00
283 lines
7.7 KiB
Python
283 lines
7.7 KiB
Python
#!/usr/bin/python
|
||
#vim: encoding=utf-8
|
||
# pylint: disable=wrong-import-position,wrong-import-order,redefined-builtin
|
||
|
||
"""
|
||
This module is used to generate png-files for wttr.in queries.
|
||
The only exported function is:
|
||
|
||
* render_ansi(png_file, text, options=None)
|
||
|
||
`render_ansi` is the main function of the module,
|
||
which does rendering of stream into a PNG-file.
|
||
|
||
The module uses PIL for graphical tasks, and pyte for rendering
|
||
of ANSI stream into terminal representation.
|
||
"""
|
||
|
||
from __future__ import print_function
|
||
|
||
import sys
|
||
import io
|
||
import os
|
||
import glob
|
||
|
||
from PIL import Image, ImageFont, ImageDraw
|
||
import pyte.screens
|
||
import emoji
|
||
import grapheme
|
||
|
||
from . import unicodedata2
|
||
|
||
sys.path.insert(0, "..")
|
||
import constants
|
||
import globals
|
||
|
||
COLS = 180
|
||
ROWS = 100
|
||
CHAR_WIDTH = 8
|
||
CHAR_HEIGHT = 14
|
||
FONT_SIZE = 13
|
||
FONT_CAT = {
|
||
'default': "/usr/share/fonts/truetype/dejavu/DejaVuSansMono.ttf",
|
||
'Cyrillic': "/usr/share/fonts/truetype/dejavu/DejaVuSansMono.ttf",
|
||
'Greek': "/usr/share/fonts/truetype/dejavu/DejaVuSansMono.ttf",
|
||
'Arabic': "/usr/share/fonts/truetype/dejavu/DejaVuSansMono.ttf",
|
||
'Hebrew': "/usr/share/fonts/truetype/dejavu/DejaVuSansMono.ttf",
|
||
'Han': "/usr/share/fonts/truetype/wqy/wqy-zenhei.ttc",
|
||
'Hiragana': "/usr/share/fonts/truetype/motoya-l-cedar/MTLc3m.ttf",
|
||
'Katakana': "/usr/share/fonts/truetype/motoya-l-cedar/MTLc3m.ttf",
|
||
'Hangul': "/usr/share/fonts/truetype/lexi/LexiGulim.ttf",
|
||
'Braille': "/usr/share/fonts/truetype/ancient-scripts/Symbola_hint.ttf",
|
||
'Emoji': "/usr/share/fonts/truetype/ancient-scripts/Symbola_hint.ttf",
|
||
}
|
||
|
||
#
|
||
# How to find font for non-standard scripts:
|
||
#
|
||
# $ fc-list :lang=ja
|
||
#
|
||
# GNU/Debian packages, that the fonts come from:
|
||
#
|
||
# * fonts-dejavu-core
|
||
# * fonts-wqy-zenhei (Han)
|
||
# * fonts-motoya-l-cedar (Hiragana/Katakana)
|
||
# * fonts-lexi-gulim (Hangul)
|
||
# * fonts-symbola (Braille/Emoji)
|
||
#
|
||
|
||
def render_ansi(text, options=None):
|
||
"""Render `text` (terminal sequence) in a PNG file
|
||
paying attention to passed command line `options`.
|
||
|
||
Return: file content
|
||
"""
|
||
|
||
screen = pyte.screens.Screen(COLS, ROWS)
|
||
screen.set_mode(pyte.modes.LNM)
|
||
stream = pyte.Stream(screen)
|
||
|
||
text, graphemes = _fix_graphemes(text)
|
||
stream.feed(text)
|
||
|
||
buf = sorted(screen.buffer.items(), key=lambda x: x[0])
|
||
buf = [[x[1] for x in sorted(line[1].items(), key=lambda x: x[0])] for line in buf]
|
||
|
||
return _gen_term(buf, graphemes, options=options)
|
||
|
||
def _color_mapping(color):
|
||
"""Convert pyte color to PIL color
|
||
|
||
Return: tuple of color values (R,G,B)
|
||
"""
|
||
|
||
if color == 'default':
|
||
return 'lightgray'
|
||
if color in ['green', 'black', 'cyan', 'blue', 'brown']:
|
||
return color
|
||
try:
|
||
return (
|
||
int(color[0:2], 16),
|
||
int(color[2:4], 16),
|
||
int(color[4:6], 16))
|
||
except (ValueError, IndexError):
|
||
# if we do not know this color and it can not be decoded as RGB,
|
||
# print it and return it as it is (will be displayed as black)
|
||
# print color
|
||
return color
|
||
return color
|
||
|
||
def _strip_buf(buf):
|
||
"""Strips empty spaces from behind and from the right side.
|
||
(from the right side is not yet implemented)
|
||
"""
|
||
|
||
def empty_line(line):
|
||
"Returns True if the line consists from spaces"
|
||
return all(x.data == ' ' for x in line)
|
||
|
||
def line_len(line):
|
||
"Returns len of the line excluding spaces from the right"
|
||
|
||
last_pos = len(line)
|
||
while last_pos > 0 and line[last_pos-1].data == ' ':
|
||
last_pos -= 1
|
||
return last_pos
|
||
|
||
number_of_lines = 0
|
||
for line in buf[::-1]:
|
||
if not empty_line(line):
|
||
break
|
||
number_of_lines += 1
|
||
|
||
if number_of_lines:
|
||
buf = buf[:-number_of_lines]
|
||
|
||
max_len = max(line_len(x) for x in buf)
|
||
buf = [line[:max_len] for line in buf]
|
||
|
||
return buf
|
||
|
||
def _script_category(char):
|
||
"""Returns category of a Unicode character
|
||
|
||
Possible values:
|
||
default, Cyrillic, Greek, Han, Hiragana
|
||
"""
|
||
|
||
if char in emoji.UNICODE_EMOJI:
|
||
return "Emoji"
|
||
|
||
cat = unicodedata2.script_cat(char)[0]
|
||
if char == u':':
|
||
return 'Han'
|
||
if cat in ['Latin', 'Common']:
|
||
return 'default'
|
||
return cat
|
||
|
||
def _load_emojilib():
|
||
"""Load known emojis from a directory, and return dictionary
|
||
of PIL Image objects correspodent to the loaded emojis.
|
||
Each emoji is resized to the CHAR_HEIGHT size.
|
||
"""
|
||
|
||
emojilib = {}
|
||
for filename in glob.glob("share/emoji/*.png"):
|
||
character = os.path.basename(filename)[:-4]
|
||
emojilib[character] = \
|
||
Image.open(filename).resize((CHAR_HEIGHT, CHAR_HEIGHT))
|
||
return emojilib
|
||
|
||
# pylint: disable=too-many-locals,too-many-branches,too-many-statements
|
||
def _gen_term(buf, graphemes, options=None):
|
||
"""Renders rendered pyte buffer `buf` and list of workaround `graphemes`
|
||
to a PNG file, and return its content
|
||
"""
|
||
|
||
if not options:
|
||
options = {}
|
||
|
||
current_grapheme = 0
|
||
|
||
buf = _strip_buf(buf)
|
||
cols = max(len(x) for x in buf)
|
||
rows = len(buf)
|
||
|
||
image = Image.new('RGB', (cols * CHAR_WIDTH, rows * CHAR_HEIGHT))
|
||
|
||
buf = buf[-ROWS:]
|
||
|
||
draw = ImageDraw.Draw(image)
|
||
font = {}
|
||
for cat in FONT_CAT:
|
||
font[cat] = ImageFont.truetype(FONT_CAT[cat], FONT_SIZE)
|
||
|
||
emojilib = _load_emojilib()
|
||
|
||
x_pos = 0
|
||
y_pos = 0
|
||
for line in buf:
|
||
x_pos = 0
|
||
for char in line:
|
||
current_color = _color_mapping(char.fg)
|
||
if char.bg != 'default':
|
||
draw.rectangle(
|
||
((x_pos, y_pos),
|
||
(x_pos+CHAR_WIDTH, y_pos+CHAR_HEIGHT)),
|
||
fill=_color_mapping(char.bg))
|
||
|
||
if char.data == "!":
|
||
try:
|
||
data = graphemes[current_grapheme]
|
||
except IndexError:
|
||
pass
|
||
current_grapheme += 1
|
||
else:
|
||
data = char.data
|
||
|
||
if data:
|
||
cat = _script_category(data[0])
|
||
if cat not in font:
|
||
globals.log("Unknown font category: %s" % cat)
|
||
if cat == 'Emoji' and emojilib.get(data):
|
||
image.paste(emojilib.get(data), (x_pos, y_pos))
|
||
else:
|
||
draw.text(
|
||
(x_pos, y_pos),
|
||
data,
|
||
font=font.get(cat, font.get('default')),
|
||
fill=current_color)
|
||
|
||
x_pos += CHAR_WIDTH * constants.WEATHER_SYMBOL_WIDTH_VTE.get(data, 1)
|
||
y_pos += CHAR_HEIGHT
|
||
|
||
if 'transparency' in options:
|
||
transparency = options.get('transparency', '255')
|
||
try:
|
||
transparency = int(transparency)
|
||
except ValueError:
|
||
transparency = 255
|
||
|
||
if transparency < 0:
|
||
transparency = 0
|
||
|
||
if transparency > 255:
|
||
transparency = 255
|
||
|
||
image = image.convert("RGBA")
|
||
datas = image.getdata()
|
||
|
||
new_data = []
|
||
for item in datas:
|
||
new_item = tuple(list(item[:3]) + [transparency])
|
||
new_data.append(new_item)
|
||
|
||
image.putdata(new_data)
|
||
|
||
img_bytes = io.BytesIO()
|
||
image.save(img_bytes, format="png")
|
||
return img_bytes.getvalue()
|
||
|
||
def _fix_graphemes(text):
|
||
"""
|
||
Extract long graphemes sequences that can't be handled
|
||
by pyte correctly because of the bug pyte#131.
|
||
Graphemes are omited and replaced with placeholders,
|
||
and returned as a list.
|
||
|
||
Return:
|
||
text_without_graphemes, graphemes
|
||
"""
|
||
|
||
output = ""
|
||
graphemes = []
|
||
|
||
for gra in grapheme.graphemes(text):
|
||
if len(gra) > 1:
|
||
character = "!"
|
||
graphemes.append(gra)
|
||
else:
|
||
character = gra
|
||
output += character
|
||
|
||
return output, graphemes
|