Version

Quick search

Table Of Contents

Source code for kivy.core.text

'''
Text
====

An abstraction of text creation. Depending of the selected backend, the
accuracy of text rendering may vary.

.. versionchanged:: 1.10.1
    :meth:`LabelBase.find_base_direction` added.

.. versionchanged:: 1.5.0
    :attr:`LabelBase.line_height` added.

.. versionchanged:: 1.0.7
    The :class:`LabelBase` does not generate any texture if the text has a
    width <= 1.

This is the backend layer for rendering text with different text providers,
you should only be using this directly if your needs aren't fulfilled by the
:class:`~kivy.uix.label.Label`.

Usage example::

    from kivy.core.text import Label as CoreLabel

    ...
    ...
    my_label = CoreLabel()
    my_label.text = 'hello'
    # the label is usually not drawn until needed, so force it to draw.
    my_label.refresh()
    # Now access the texture of the label and use it wherever and
    # however you may please.
    hello_texture = my_label.texture


Font Context Manager
====================

A font context is a namespace where multiple fonts are loaded; if a font is
missing a glyph needed to render text, it can fall back to a different font in
the same context. The font context manager can be used to query and manipulate
the state of font contexts when using the Pango text provider (no other
provider currently implements it).

.. versionadded:: 1.11.0

.. warning:: This feature requires the Pango text provider.

Font contexts can be created automatically by :class:`kivy.uix.label.Label` or
:class:`kivy.uix.textinput.TextInput`; if a non-existent context is used in
one of these classes, it will be created automatically, or if a font file is
specified without a context (this creates an isolated context, without
support for fallback).

Usage example::

    from kivy.uix.label import Label
    from kivy.core.text import FontContextManager as FCM

    # Create a font context containing system fonts + one custom TTF
    FCM.create('system://myapp')
    family = FCM.add_font('/path/to/file.ttf')

    # These are now interchangeable ways to refer to the custom font:
    lbl1 = Label(font_context='system://myapp', family_name=family)
    lbl2 = Label(font_context='system://myapp', font_name='/path/to/file.ttf')

    # You could also refer to a system font by family, since this is a
    # system:// font context
    lbl3 = Label(font_context='system://myapp', family_name='Arial')
'''

__all__ = ('LabelBase', 'Label',
           'FontContextManagerBase', 'FontContextManager')

import re
import os
from ast import literal_eval
from functools import partial
from copy import copy
from kivy import kivy_data_dir
from kivy.config import Config
from kivy.utils import platform
from kivy.graphics.texture import Texture
from kivy.core import core_select_lib
from kivy.core.text.text_layout import layout_text, LayoutWord
from kivy.resources import resource_find, resource_add_path
from kivy.setupconfig import USE_SDL2, USE_PANGOFT2
from kivy.logger import Logger


if 'KIVY_DOC' not in os.environ:
    _default_font_paths = literal_eval(Config.get('kivy', 'default_font'))
    DEFAULT_FONT = _default_font_paths.pop(0)
else:
    DEFAULT_FONT = None

FONT_REGULAR = 0
FONT_ITALIC = 1
FONT_BOLD = 2
FONT_BOLDITALIC = 3

whitespace_pat = re.compile('( +)')


[docs]class LabelBase(object): '''Core text label. This is the abstract class used by different backends to render text. .. warning:: The core text label can't be changed at runtime. You must recreate one. :Parameters: `font_size`: int, defaults to 12 Font size of the text `font_context`: str, defaults to None Context for the specified font (see :class:`kivy.uix.label.Label` for details). `None` will autocreate an isolated context named after the resolved font file. `font_name`: str, defaults to DEFAULT_FONT Font name of the text `font_family`: str, defaults to None Font family name to request for drawing, this can only be used with `font_context`. `bold`: bool, defaults to False Activate "bold" text style `italic`: bool, defaults to False Activate "italic" text style `text_size`: tuple, defaults to (None, None) Add constraint to render the text (inside a bounding box). If no size is given, the label size will be set to the text size. `padding`: int|float or list|tuple, defaults to [0, 0, 0, 0]. Padding of the text in the format [padding_left, padding_top, padding_right, padding_bottom]. ``padding`` should be int|float or a list|tuple with 1, 2 or 4 elements. `padding_x`: float, defaults to 0.0 Left/right padding `padding_y`: float, defaults to 0.0 Top/bottom padding `halign`: str, defaults to "left" Horizontal text alignment inside the bounding box `valign`: str, defaults to "bottom" Vertical text alignment inside the bounding box `shorten`: bool, defaults to False Indicate whether the label should attempt to shorten its textual contents as much as possible if a `size` is given. Setting this to True without an appropriately set size will lead to unexpected results. `shorten_from`: str, defaults to `center` The side from which we should shorten the text from, can be left, right, or center. E.g. if left, the ellipsis will appear towards the left side and it will display as much text starting from the right as possible. `split_str`: string, defaults to `' '` (space) The string to use to split the words by when shortening. If empty, we can split after every character filling up the line as much as possible. `max_lines`: int, defaults to 0 (unlimited) If set, this indicate how maximum line are allowed to render the text. Works only if a limitation on text_size is set. `mipmap`: bool, defaults to False Create a mipmap for the texture `strip`: bool, defaults to False Whether each row of text has its leading and trailing spaces stripped. If `halign` is `justify` it is implicitly True. `strip_reflow`: bool, defaults to True Whether text that has been reflowed into a second line should be stripped, even if `strip` is False. This is only in effect when `size_hint_x` is not None, because otherwise lines are never split. `unicode_errors`: str, defaults to `'replace'` How to handle unicode decode errors. Can be `'strict'`, `'replace'` or `'ignore'`. `outline_width`: int, defaults to None Width in pixels for the outline. `outline_color`: tuple, defaults to (0, 0, 0) Color of the outline. `font_features`: str, defaults to None OpenType font features in CSS format (Pango only) `base_direction`: str, defaults to None (auto) Text direction, one of `None`, `'ltr'`, `'rtl'`, `'weak_ltr'`, or `'weak_rtl'` (Pango only) `text_language`: str, defaults to None (user locale) RFC-3066 format language tag as a string (Pango only) `limit_render_to_text_bbox`: bool, defaults to False. PIL only. If set to ``True``, this parameter indicates that rendering should be limited to the bounding box of the text, excluding any additional white spaces designated for ascent and descent. By limiting the rendering to the bounding box of the text, it ensures a more precise alignment with surrounding elements when utilizing properties such as `valign`, `y`, `pos`, `pos_hint`, etc. .. versionadded:: 2.3.0 `limit_render_to_text_bbox` was added to allow to limit text rendering to the text bounding box (PIL only). .. deprecated:: 2.2.0 `padding_x` and `padding_y` have been deprecated. Please use `padding` instead. .. versionchanged:: 2.2.0 `padding` is now a list and defaults to [0, 0, 0, 0]. `padding` accepts int|float or a list|tuple with 1, 2 or 4 elements. .. versionchanged:: 1.10.1 `font_context`, `font_family`, `font_features`, `base_direction` and `text_language` were added. .. versionchanged:: 1.10.0 `outline_width` and `outline_color` were added. .. versionchanged:: 1.9.0 `strip`, `strip_reflow`, `shorten_from`, `split_str`, and `unicode_errors` were added. .. versionchanged:: 1.9.0 `padding_x` and `padding_y` has been fixed to work as expected. In the past, the text was padded by the negative of their values. .. versionchanged:: 1.8.0 `max_lines` parameters has been added. .. versionchanged:: 1.0.8 `size` have been deprecated and replaced with `text_size`. .. versionchanged:: 1.0.7 The `valign` is now respected. This wasn't the case previously so you might have an issue in your application if you have not considered this. ''' __slots__ = ('options', 'texture', '_label', '_text_size') _cached_lines = [] _fonts = {} _fonts_cache = {} _fonts_dirs = [] _font_dirs_files = [] _texture_1px = None _font_family_support = False def __init__( self, text='', font_size=12, font_name=DEFAULT_FONT, bold=False, italic=False, underline=False, strikethrough=False, font_family=None, halign='left', valign='bottom', shorten=False, text_size=None, mipmap=False, color=None, line_height=1.0, strip=False, strip_reflow=True, shorten_from='center', split_str=' ', unicode_errors='replace', font_hinting='normal', font_kerning=True, font_blended=True, outline_width=None, outline_color=None, font_context=None, font_features=None, base_direction=None, font_direction='ltr', font_script_name='Latin', text_language=None, limit_render_to_text_bbox=False, **kwargs): # Include system fonts_dir in resource paths. # This allows us to specify a font from those dirs. LabelBase.get_system_fonts_dir() options = {'text': text, 'font_size': font_size, 'font_name': font_name, 'bold': bold, 'italic': italic, 'underline': underline, 'strikethrough': strikethrough, 'font_family': font_family, 'halign': halign, 'valign': valign, 'shorten': shorten, 'mipmap': mipmap, 'line_height': line_height, 'strip': strip, 'strip_reflow': strip_reflow, 'shorten_from': shorten_from, 'split_str': split_str, 'unicode_errors': unicode_errors, 'font_hinting': font_hinting, 'font_kerning': font_kerning, 'font_blended': font_blended, 'outline_width': outline_width, 'font_context': font_context, 'font_features': font_features, 'base_direction': base_direction, 'font_direction': font_direction, 'font_script_name': font_script_name, 'text_language': text_language, 'limit_render_to_text_bbox': limit_render_to_text_bbox} kwargs_get = kwargs.get options['color'] = color or (1, 1, 1, 1) options['outline_color'] = outline_color or (0, 0, 0, 1) options['padding'] = kwargs_get('padding', [0, 0, 0, 0]) if isinstance(options['padding'], (int, float)): options['padding'] = [options['padding']] * 4 elif ( isinstance(options['padding'], (list, tuple)) and len(options['padding']) != 4 ): if len(options['padding']) == 1: options['padding'] = options['padding'] * 4 elif len(options['padding']) == 2: options['padding'] = options['padding'] * 2 else: raise ValueError( "padding should be int|float or a list|tuple with 1, 2 or " f"4 elements, got {type(options['padding'])} with " f"{len(options['padding'])} elements." ) options['padding_x'] = kwargs_get('padding_x') options['padding_y'] = kwargs_get('padding_y') for padding_option in ('padding_x', 'padding_y'): if kwargs_get(padding_option): Logger.warning( f"LabelBase: The use of the {padding_option} parameter is " "deprecated, and will be removed in future versions. Use " "padding instead." ) if 'size' in kwargs: options['text_size'] = kwargs['size'] else: if text_size is None: options['text_size'] = (None, None) else: options['text_size'] = text_size self._text_size = options['text_size'] self._text = options['text'] self._internal_size = 0, 0 # the real computed text size (inclds pad) self._cached_lines = [] self.options = options self.texture = None self.is_shortened = False self.resolve_font_name() self._migrate_deprecated_padding_xy() def _migrate_deprecated_padding_xy(self): options = self.options self.options['padding'] = list(self.options['padding']) if options['padding_x']: self.options['padding'][::2] = [options['padding_x']] * 2 if options['padding_y']: self.options['padding'][1::2] = [options['padding_y']] * 2
[docs] @staticmethod def register(name, fn_regular, fn_italic=None, fn_bold=None, fn_bolditalic=None): '''Register an alias for a Font. .. versionadded:: 1.1.0 If you're using a ttf directly, you might not be able to use the bold/italic properties of the ttf version. If the font is delivered in multiple files (one regular, one italic and one bold), then you need to register these files and use the alias instead. All the fn_regular/fn_italic/fn_bold parameters are resolved with :func:`kivy.resources.resource_find`. If fn_italic/fn_bold are None, fn_regular will be used instead. ''' if fn_regular is None: raise ValueError("font_regular cannot be None") fonts = [] for font_type in fn_regular, fn_italic, fn_bold, fn_bolditalic: if font_type is not None: font = resource_find(font_type) if font is None: raise IOError('File {0} not found'.format(font_type)) else: fonts.append(font) else: fonts.append(fonts[0]) # add regular font to list again LabelBase._fonts[name] = tuple(fonts)
def resolve_font_name(self): options = self.options fontname = options['font_name'] fonts = self._fonts fontscache = self._fonts_cache if self._font_family_support and options['font_family']: options['font_name_r'] = None return # is the font registered? if fontname in fonts: # return the preferred font for the current bold/italic combination italic = int(options['italic']) if options['bold']: bold = FONT_BOLD else: bold = FONT_REGULAR options['font_name_r'] = fonts[fontname][italic | bold] elif fontname in fontscache: options['font_name_r'] = fontscache[fontname] else: filename = resource_find(fontname) if not filename and not fontname.endswith('.ttf'): fontname = '{}.ttf'.format(fontname) filename = resource_find(fontname) if filename is None: # XXX for compatibility, check directly in the data dir filename = pep8_fn = os.path.join(kivy_data_dir, fontname) if not os.path.exists(pep8_fn) or not os.path.isfile(pep8_fn): raise IOError('Label: File %r not found' % fontname) fontscache[fontname] = filename options['font_name_r'] = filename
[docs] @staticmethod def get_system_fonts_dir(): '''Return the directories used by the system for fonts. ''' if LabelBase._fonts_dirs: return LabelBase._fonts_dirs fdirs = [] if platform == 'linux': fdirs = [ '/usr/share/fonts', '/usr/local/share/fonts', os.path.expanduser('~/.fonts'), os.path.expanduser('~/.local/share/fonts')] elif platform == 'macosx': fdirs = ['/Library/Fonts', '/System/Library/Fonts', os.path.expanduser('~/Library/Fonts')] elif platform == 'win': fdirs = [os.path.join(os.environ['SYSTEMROOT'], 'Fonts')] elif platform == 'ios': fdirs = ['/System/Library/Fonts'] elif platform == 'android': fdirs = ['/system/fonts'] else: raise Exception("Unknown platform: {}".format(platform)) fdirs.append(os.path.join(kivy_data_dir, 'fonts')) # register the font dirs rdirs = [] _font_dir_files = [] for fdir in fdirs: for _dir, dirs, files in os.walk(fdir): _font_dir_files.extend(files) resource_add_path(_dir) rdirs.append(_dir) LabelBase._fonts_dirs = rdirs LabelBase._font_dirs_files = _font_dir_files return rdirs
[docs] def get_extents(self, text): '''Return a tuple (width, height) indicating the size of the specified text''' return (0, 0)
[docs] def get_cached_extents(self): '''Returns a cached version of the :meth:`get_extents` function. :: >>> func = self._get_cached_extents() >>> func <built-in method size of PROVIDER.font.Font object at 0x01E45650> >>> func('a line') (36, 18) .. warning:: This method returns a size measuring function that is valid for the font settings used at the time :meth:`get_cached_extents` was called. Any change in the font settings will render the returned function incorrect. You should only use this if you know what you're doing. .. versionadded:: 1.9.0 ''' return self.get_extents
def _render_begin(self): pass def _render_text(self, text, x, y): pass def _render_end(self): pass
[docs] def shorten(self, text, margin=2): ''' Shortens the text to fit into a single line by the width specified by :attr:`text_size` [0]. If :attr:`text_size` [0] is None, it returns text unchanged. :attr:`split_str` and :attr:`shorten_from` determines how the text is shortened. :params: `text` str, the text to be shortened. `margin` int, the amount of space to leave between the margins and the text. This is in addition to :attr:`padding_x`. :returns: the text shortened to fit into a single line. ''' textwidth = self.get_cached_extents() uw = self.text_size[0] if uw is None or not text: return text opts = self.options uw = max(0, int(uw - opts['padding'][0] - opts['padding'][2] - margin)) # if larger, it won't fit so don't even try extents chr = type(text) text = text.replace(chr('\n'), chr(' ')) if len(text) <= uw and textwidth(text)[0] <= uw: return text c = opts['split_str'] offset = 0 if len(c) else 1 dir = opts['shorten_from'][0] elps = textwidth('...')[0] if elps > uw: self.is_shortened = True if textwidth('..')[0] <= uw: return '..' else: return '.' uw -= elps f = partial(text.find, c) f_rev = partial(text.rfind, c) # now find the first and last word e1, s2 = f(), f_rev() if dir != 'l': # center or right # no split, or the first word doesn't even fit if e1 != -1: l1 = textwidth(text[:e1])[0] l2 = textwidth(text[s2 + 1:])[0] if e1 == -1 or l1 + l2 > uw: self.is_shortened = True if len(c): opts['split_str'] = '' res = self.shorten(text, margin) opts['split_str'] = c return res # at this point we do char by char so e1 must be zero if l1 <= uw: return chr('{0}...').format(text[:e1]) return chr('...') # both word fits, and there's at least on split_str if s2 == e1: # there's only on split_str self.is_shortened = True return chr('{0}...{1}').format(text[:e1], text[s2 + 1:]) # both the first and last word fits, and they start/end at diff pos if dir == 'r': ee1 = f(e1 + 1) while l2 + textwidth(text[:ee1])[0] <= uw: e1 = ee1 if e1 == s2: break ee1 = f(e1 + 1) else: while True: if l1 <= l2: ee1 = f(e1 + 1) l1 = textwidth(text[:ee1])[0] if l2 + l1 > uw: break e1 = ee1 if e1 == s2: break else: ss2 = f_rev(0, s2 - offset) l2 = textwidth(text[ss2 + 1:])[0] if l2 + l1 > uw: break s2 = ss2 if e1 == s2: break else: # left # no split, or the last word doesn't even fit if s2 != -1: l2 = textwidth(text[s2 + (1 if len(c) else -1):])[0] l1 = textwidth(text[:max(0, e1)])[0] # if split_str if s2 == -1 or l2 + l1 > uw: self.is_shortened = True if len(c): opts['split_str'] = '' res = self.shorten(text, margin) opts['split_str'] = c return res return chr('...') # both word fits, and there's at least on split_str if s2 == e1: # there's only on split_str self.is_shortened = True return chr('{0}...{1}').format(text[:e1], text[s2 + 1:]) # both the first and last word fits, and they start/end at diff pos ss2 = f_rev(0, s2 - offset) while l1 + textwidth(text[ss2 + 1:])[0] <= uw: s2 = ss2 if s2 == e1: break ss2 = f_rev(0, s2 - offset) self.is_shortened = True return chr('{0}...{1}').format(text[:e1], text[s2 + 1:])
def _default_line_options(self, lines): for line in lines: if len(line.words): # get opts from first line, first word return line.words[0].options return None def clear_texture(self): self._render_begin() data = self._render_end() assert data if data is not None and data.width > 1: self.texture.blit_data(data) return # FIXME: This should possibly use a Config value, and possibly we should # expose pango_unichar_direction() / pango_bidi_type_for_unichar()
[docs] @staticmethod def find_base_direction(text): '''Searches a string the first character that has a strong direction, according to the Unicode bidirectional algorithm. Returns `None` if the base direction cannot be determined, or one of `'ltr'` or `'rtl'`. .. versionadded: 1.10.1 .. note:: This feature requires the Pango text provider. ''' return 'ltr'
def render_lines(self, lines, options, render_text, y, size): get_extents = self.get_cached_extents() uw, uh = options['text_size'] padding_left = options['padding'][0] padding_right = options['padding'][2] if uw is not None: uww = uw - padding_left - padding_right # real width of just text w = size[0] sw = options['space_width'] halign = options['halign'] split = re.split find_base_dir = self.find_base_direction cur_base_dir = options['base_direction'] for layout_line in lines: # for plain label each line has only one str lw, lh = layout_line.w, layout_line.h line = '' assert len(layout_line.words) < 2 if len(layout_line.words): last_word = layout_line.words[0] line = last_word.text if not cur_base_dir: cur_base_dir = find_base_dir(line) x = padding_left if halign == 'auto': if cur_base_dir and 'rtl' in cur_base_dir: # right-align RTL text x = max(0, int(w - lw - padding_right)) elif halign == 'center': x = min( int(w - lw), max( int(padding_left), int((w - lw + padding_left - padding_right) / 2.0) ) ) elif halign == 'right': x = max(0, int(w - lw - padding_right)) # right left justify # divide left over space between `spaces` # TODO implement a better method of stretching glyphs? if (uw is not None and halign == 'justify' and line and not layout_line.is_last_line): # number spaces needed to fill, and remainder n, rem = divmod(max(uww - lw, 0), sw) n = int(n) words = None if n or rem: # there's no trailing space when justify is selected words = split(whitespace_pat, line) if words is not None and len(words) > 1: space = type(line)(' ') # words: every even index is spaces, just add ltr n spaces for i in range(n): idx = (2 * i + 1) % (len(words) - 1) words[idx] = words[idx] + space if rem: # render the last word at the edge, also add it to line ext = get_extents(words[-1]) word = LayoutWord(last_word.options, ext[0], ext[1], words[-1]) layout_line.words.append(word) last_word.lw = uww - ext[0] # word was stretched render_text(words[-1], x + last_word.lw, y) last_word.text = line = ''.join(words[:-2]) else: last_word.lw = uww # word was stretched last_word.text = line = ''.join(words) layout_line.w = uww # the line occupies full width if len(line): layout_line.x = x layout_line.y = y render_text(line, x, y) y += lh return y def _render_real(self): lines = self._cached_lines options = self._default_line_options(lines) if options is None: # there was no text to render return self.clear_texture() old_opts = self.options ih = self._internal_size[1] # the real size of text, not texture size = self.size valign = options['valign'] padding_top = options['padding'][1] if valign == 'bottom': y = int(size[1] - ih + padding_top) elif valign == 'top': y = int(padding_top) elif valign in ('middle', 'center'): y = int((size[1] - ih + 2 * padding_top) / 2) self._render_begin() self.render_lines(lines, options, self._render_text, y, size) # get data from provider data = self._render_end() assert data self.options = old_opts # If the text is 1px width, usually, the data is black. # Don't blit that kind of data, otherwise, you have a little black bar. if data is not None and data.width > 1: self.texture.blit_data(data)
[docs] def render(self, real=False): '''Return a tuple (width, height) to create the image with the user constraints. (width, height) includes the padding. ''' if real: return self._render_real() options = copy(self.options) options['space_width'] = self.get_extents(' ')[0] options['strip'] = strip = (options['strip'] or options['halign'] == 'justify') uw, uh = options['text_size'] = self._text_size text = self.text if strip: text = text.strip() self.is_shortened = False if uw is not None and options['shorten']: text = self.shorten(text) self._cached_lines = lines = [] if not text: return 0, 0 if uh is not None and (options['valign'] == 'middle' or options['valign'] == 'center'): center = -1 # pos of newline if len(text) > 1: middle = int(len(text) // 2) l, r = text.rfind('\n', 0, middle), text.find('\n', middle) if l != -1 and r != -1: center = l if center - l <= r - center else r elif l != -1: center = l elif r != -1: center = r # if a newline split text, render from center down and up til uh if center != -1: # layout from center down until half uh w, h, clipped = layout_text(text[center + 1:], lines, (0, 0), (uw, uh / 2), options, self.get_cached_extents(), True, True) # now layout from center upwards until uh is reached w, h, clipped = layout_text(text[:center + 1], lines, (w, h), (uw, uh), options, self.get_cached_extents(), False, True) else: # if there's no new line, layout everything w, h, clipped = layout_text(text, lines, (0, 0), (uw, None), options, self.get_cached_extents(), True, True) else: # top or bottom w, h, clipped = layout_text(text, lines, (0, 0), (uw, uh), options, self.get_cached_extents(), options['valign'] == 'top', True) self._internal_size = w, h if uw: w = uw if uh: h = uh if h > 1 and w < 2: w = 2 return int(w), int(h)
def _texture_refresh(self, *l): self.refresh() def _texture_fill(self, texture): # second pass, render for real self.render(real=True)
[docs] def refresh(self): '''Force re-rendering of the text ''' self.resolve_font_name() # first pass, calculating width/height sz = self.render() self._size_texture = sz self._size = (sz[0], sz[1]) # if no text are rendered, return nothing. width, height = self._size if width <= 1 or height <= 1: self.texture = self.texture_1px return # create a delayed texture texture = self.texture if texture is None or \ width != texture.width or \ height != texture.height: texture = Texture.create(size=(width, height), mipmap=self.options['mipmap'], callback=self._texture_fill) texture.flip_vertical() texture.add_reload_observer(self._texture_refresh) self.texture = texture else: texture.ask_update(self._texture_fill)
def _get_text(self): return self._text def _set_text(self, text): if text != self._text: self._text = text text = property(_get_text, _set_text, doc='Get/Set the text') label = property(_get_text, _set_text, doc='Get/Set the text') @property def texture_1px(self): if LabelBase._texture_1px is None: tex = Texture.create(size=(1, 1), colorfmt='rgba') tex.blit_buffer(b'\x00\x00\x00\x00', colorfmt='rgba') LabelBase._texture_1px = tex return LabelBase._texture_1px @property def size(self): return self._size @property def width(self): return self._size[0] @property def height(self): return self._size[1] @property def content_width(self): '''Return the content width; i.e. the width of the text without any padding.''' if self.texture is None: return 0 return self.texture.width - ( self.options['padding'][0] + self.options['padding'][2] ) @property def content_height(self): '''Return the content height; i.e. the height of the text without any padding.''' if self.texture is None: return 0 return self.texture.height - ( self.options['padding'][1] + self.options['padding'][3] ) @property def content_size(self): '''Return the content size (width, height)''' if self.texture is None: return (0, 0) return (self.content_width, self.content_height) @property def fontid(self): '''Return a unique id for all font parameters''' return str([self.options[x] for x in ( 'font_size', 'font_name_r', 'bold', 'italic', 'underline', 'strikethrough')]) def _get_text_size(self): return self._text_size def _set_text_size(self, x): self._text_size = x text_size = property(_get_text_size, _set_text_size, doc='''Get/set the (width, height) of the ' 'contrained rendering box''') usersize = property(_get_text_size, _set_text_size, doc='''(deprecated) Use text_size instead.''')
class FontContextManagerBase(object): @staticmethod def create(font_context): '''Create a font context, you must specify a unique name (string). Returns `True` on success and `False` on failure. If `font_context` starts with one of the reserved words `'system://'`, `'directory://'`, `'fontconfig://'` or `'systemconfig://'`, the context is setup accordingly (exact results of this depends on your platform, environment and configuration). * `'system://'` loads the default system's FontConfig configuration and all fonts (usually including user fonts). * `directory://` contexts preload a directory of font files (specified in the context name), `systemconfig://` loads the system's FontConfig configuration (but no fonts), and `fontconfig://` loads FontConfig configuration file (specified in the context name!). These are for advanced users only, check the source code and FontConfig documentation for details. * Fonts automatically loaded to an isolated context (ie when no font context was specified) start with `'isolated://'`. This has no special effect, and only serves to help you identify them in the results returned from :meth:`list`. * Any other string is a context that will only draw with the font file(s) you explicitly add to it. .. versionadded:: 1.11.0 .. note:: Font contexts are created automatically by specifying a name in the `font_context` property of :class:`kivy.uix.label.Label` or :class:`kivy.uix.textinput.TextInput`. They are also auto-created by :meth:`add_font` by default, so you normally don't need to call this directly. .. note:: This feature requires the Pango text provider. ''' raise NotImplementedError("No font_context support in text provider") @staticmethod def exists(font_context): '''Returns True if a font context with the given name exists. .. versionadded:: 1.11.0 .. note:: This feature requires the Pango text provider. ''' raise NotImplementedError("No font_context support in text provider") @staticmethod def destroy(font_context): '''Destroy a named font context (if it exists) .. versionadded:: 1.11.0 .. note:: This feature requires the Pango text provider. ''' raise NotImplementedError("No font_context support in text provider") @staticmethod def list(): '''Returns a list of `bytes` objects, each representing a cached font context name. Note that entries that start with `isolated://` were autocreated by loading a font file with no font_context specified. .. versionadded:: 1.11.0 .. note:: This feature requires the Pango text provider. ''' raise NotImplementedError("No font_context support in text provider") @staticmethod def list_families(font_context): '''Returns a list of `bytes` objects, each representing a font family name that is available in the given `font_context`. .. versionadded:: 1.11.0 .. note:: Pango adds static "Serif", "Sans" and "Monospace" to the list in current versions, even if only a single custom font file is added to the context. .. note:: This feature requires the Pango text provider. ''' raise NotImplementedError("No font_context support in text provider") @staticmethod def list_custom(font_context): '''Returns a dictionary representing all the custom-loaded fonts in the context. The key is a `bytes` object representing the full path to the font file, the value is a `bytes` object representing the font family name used to request drawing with the font. .. versionadded:: 1.11.0 .. note:: This feature requires the Pango text provider. ''' raise NotImplementedError("No font_context support in text provider") @staticmethod def add_font(font_context, filename, autocreate=True, family=None): '''Add a font file to a named font context. If `autocreate` is true, the context will be created if it does not exist (this is the default). You can specify the `family` argument (string) to skip auto-detecting the font family name. .. warning:: The `family` argument is slated for removal if the underlying implementation can be fixed, It is offered as a way to optimize startup time for deployed applications (it avoids opening the file with FreeType2 to determine its family name). To use this, first load the font file without specifying `family`, and hardcode the returned (autodetected) `family` value in your font context initialization. .. versionadded:: 1.11.0 .. note:: This feature requires the Pango text provider. ''' raise NotImplementedError("No font_context support in text provider") # Load the appropriate provider label_libs = [] if USE_PANGOFT2: label_libs += [('pango', 'text_pango', 'LabelPango')] if USE_SDL2: label_libs += [('sdl2', 'text_sdl2', 'LabelSDL2')] label_libs += [ ('pil', 'text_pil', 'LabelPIL')] Text = Label = core_select_lib('text', label_libs) if 'KIVY_DOC' not in os.environ: if not Label: import sys Logger.critical('App: Unable to get a Text provider, abort.') sys.exit(1) # FIXME: Better way to do this if Label.__name__ == 'LabelPango': from kivy.core.text.text_pango import PangoFontContextManager FontContextManager = PangoFontContextManager() else: FontContextManager = FontContextManagerBase() # For the first initialization, register the default font Label.register(DEFAULT_FONT, *_default_font_paths)