micropython-samples/tft_fonts/TFTfont.py

71 wiersze
3.1 KiB
Python

class TFTFont(object):
def __init__(self, font, index, vert, horiz, nchars):
self.firstchar = 32 # ord(first_character) Future use: absent from C file.
self.nchars = nchars # No. of chars in font
self.bits_horiz = horiz # Width in pixels of char if rendered as monospaced
self.bits_vert = vert # Height in pixels
self.bytes_vert = (self.bits_vert + 7) // 8 # Height in bytes
self.bytes_per_ch = self.bytes_vert * horiz # Total bytes per monospaced character
self.monospaced = False
self._index = index
self._font = font
def get_idx(self, relch):
offset = relch * 2 # index is 2 bytes/char
return self._index[offset] + (self._index[offset + 1] << 8)
def get_ch(self, ch):
from uctypes import addressof
relch = ch - self.firstchar
if relch > self.nchars or relch < 0:
raise ValueError('Character value {:} is unsupported.'.format(ch))
offset = self.get_idx(relch)
delta = self.get_idx(relch + 1) - offset
if self.monospaced:
return addressof(self._font) + offse, self.bits_vert, delta, self.bits_horiz
else:
return addressof(self._font) + offset, self.bits_vert, delta, delta // self.bytes_vert
# *************** TEST CODE ***************
# This runs on a PC and allows font creation and cfonts_to_py.py to be tested without hardware
# It can be deleted. Usage:
# import fonts
# fonts.fonts['freesans23x25'].render(0x41)
# dict allows access to multiple fonts in fonts file
def get_ch_test(self, ch):
self.char = bytearray(self.bytes_per_ch)
self.mv = memoryview(self.char)
self._zero = bytearray(self.bytes_per_ch)
self._mvzero = memoryview(self._zero)
relch = ch - self.firstchar
if relch > self.nchars or relch < 0:
raise ValueError('Character value {:} is unsupported.'.format(ch))
offset = self.get_idx(relch)
delta = self.get_idx(relch + 1) - offset
bv = self.bits_vert
mv = self.mv
if self.monospaced:
bh = self.bits_horiz # Char width in bits
mv[: delta] = self._font[offset : offset + delta]
mv[delta : self.bytes_per_ch] = self._mvzero[delta : self.bytes_per_ch]
else:
mv[: delta] = self._font[offset : offset + delta]
bh = delta // self.bytes_vert
return bh # horizontal increment for character location
def render(self, ch): # enter with ord(ch)
bh = self.get_ch_test(ch)
bv = self.bits_vert # Cache for speed
mv = self.mv
for bit_vert in range(bv): # for each vertical line
bytenum, bitnum = divmod(bit_vert, 8)
bit = 1 << (7 - bitnum) # bits are reversed 1 << bitnum
for bit_horiz in range(bh): # horizontal line
fontbyte = self.mv[self.bytes_vert * bit_horiz + bytenum]
print('*' if fontbyte & bit else ' ', end = '')
print('|')
return bh # Caller accounts for horzontal increment