Source code for qwt.text

# -*- coding: utf-8 -*-
#
# Licensed under the terms of the Qwt License
# Copyright (c) 2002 Uwe Rathmann, for the original C++ code
# Copyright (c) 2015 Pierre Raybaut, for the Python translation/optimization
# (see LICENSE file for more details)

"""
Text widgets
------------

QwtText
~~~~~~~

.. autoclass:: QwtText
   :members:

QwtTextLabel
~~~~~~~~~~~~

.. autoclass:: QwtTextLabel
   :members:

Text engines
------------

QwtTextEngine
~~~~~~~~~~~~~

.. autoclass:: QwtTextEngine
   :members:

QwtPlainTextEngine
~~~~~~~~~~~~~~~~~~

.. autoclass:: QwtPlainTextEngine
   :members:

QwtRichTextEngine
~~~~~~~~~~~~~~~~~

.. autoclass:: QwtRichTextEngine
   :members:
"""

import math
import os
import struct

from qtpy.QtCore import QRectF, QSize, QSizeF, Qt
from qtpy.QtGui import (
    QAbstractTextDocumentLayout,
    QColor,
    QFont,
    QFontInfo,
    QFontMetrics,
    QFontMetricsF,
    QPainter,
    QPalette,
    QPixmap,
    QTextDocument,
    QTextOption,
    QTransform,
)
from qtpy.QtWidgets import QApplication, QFrame, QSizePolicy, QWidget

from qwt.painter import QwtPainter
from qwt.qthelpers import qcolor_from_str

QWIDGETSIZE_MAX = (1 << 24) - 1

QT_API = os.environ["QT_API"]


def taggedRichText(text, flags):
    richText = text
    if flags & Qt.AlignJustify:
        richText = '<div align="justify">' + richText + "</div>"
    elif flags & Qt.AlignRight:
        richText = '<div align="right">' + richText + "</div>"
    elif flags & Qt.AlignHCenter:
        richText = '<div align="center">' + richText + "</div>"
    return richText


class QwtRichTextDocument(QTextDocument):
    def __init__(self, text, flags, font):
        super(QwtRichTextDocument, self).__init__(None)
        self.setUndoRedoEnabled(False)
        self.setDefaultFont(font)
        self.setHtml(text)

        option = self.defaultTextOption()
        if flags & Qt.TextWordWrap:
            option.setWrapMode(QTextOption.WordWrap)
        else:
            option.setWrapMode(QTextOption.NoWrap)

        option.setAlignment(flags)
        self.setDefaultTextOption(option)

        root = self.rootFrame()
        fm = root.frameFormat()
        fm.setBorder(0)
        fm.setMargin(0)
        fm.setPadding(0)
        fm.setBottomMargin(0)
        fm.setLeftMargin(0)
        root.setFrameFormat(fm)

        self.adjustSize()


[docs] class QwtTextEngine(object): """ Abstract base class for rendering text strings A text engine is responsible for rendering texts for a specific text format. They are used by `QwtText` to render a text. `QwtPlainTextEngine` and `QwtRichTextEngine` are part of the `PythonQwt` library. .. seealso:: :py:meth:`qwt.text.QwtText.setTextEngine()` """ def __init__(self): pass
[docs] def heightForWidth(self, font, flags, text, width): """ Find the height for a given width :param QFont font: Font of the text :param int flags: Bitwise OR of the flags used like in QPainter::drawText :param str text: Text to be rendered :param float width: Width :return: Calculated height """ pass
[docs] def textSize(self, font, flags, text): """ Returns the size, that is needed to render text :param QFont font: Font of the text :param int flags: Bitwise OR of the flags like in for QPainter::drawText :param str text: Text to be rendered :return: Calculated size """ pass
[docs] def mightRender(self, text): """ Test if a string can be rendered by this text engine :param str text: Text to be tested :return: True, if it can be rendered """ pass
[docs] def textMargins(self, font): """ Return margins around the texts The textSize might include margins around the text, like QFontMetrics::descent(). In situations where texts need to be aligned in detail, knowing these margins might improve the layout calculations. :param QFont font: Font of the text :return: tuple (left, right, top, bottom) representing margins """ pass
[docs] def draw(self, painter, rect, flags, text): """ Draw the text in a clipping rectangle :param QPainter painter: Painter :param QRectF rect: Clipping rectangle :param int flags: Bitwise OR of the flags like in for QPainter::drawText() :param str text: Text to be rendered """ pass
ASCENTCACHE = {} def get_screen_resolution(): """Return screen resolution: tuple of floats (DPIx, DPIy)""" try: desktop = QApplication.desktop() return (desktop.logicalDpiX(), desktop.logicalDpiY()) except AttributeError: screen = QApplication.primaryScreen() return (screen.logicalDotsPerInchX(), screen.logicalDotsPerInchY()) def qwtUnscaleFont(painter): if painter.font().pixelSize() >= 0: return dpix, dpiy = get_screen_resolution() pd = painter.device() if pd.logicalDpiX() != dpix or pd.logicalDpiY() != dpiy: try: pixelFont = QFont(painter.font(), QApplication.desktop()) except AttributeError: pixelFont = QFont(painter.font()) pixelFont.setPixelSize(QFontInfo(pixelFont).pixelSize()) painter.setFont(pixelFont)
[docs] class QwtPlainTextEngine(QwtTextEngine): """ A text engine for plain texts `QwtPlainTextEngine` renders texts using the basic `Qt` classes `QPainter` and `QFontMetrics`. """ def __init__(self): self.qrectf_max = QRectF(0, 0, QWIDGETSIZE_MAX, QWIDGETSIZE_MAX) self._fm_cache = {} self._fm_cache_f = {} def fontmetrics(self, font): fid = font.toString() try: return self._fm_cache[fid] except KeyError: return self._fm_cache.setdefault(fid, QFontMetrics(font)) def fontmetrics_f(self, font): fid = font.toString() try: return self._fm_cache_f[fid] except KeyError: return self._fm_cache_f.setdefault(fid, QFontMetricsF(font))
[docs] def heightForWidth(self, font, flags, text, width): """ Find the height for a given width :param QFont font: Font of the text :param int flags: Bitwise OR of the flags used like in QPainter::drawText :param str text: Text to be rendered :param float width: Width :return: Calculated height """ fm = self.fontmetrics_f(font) rect = fm.boundingRect(QRectF(0, 0, width, QWIDGETSIZE_MAX), flags, text) return rect.height()
[docs] def textSize(self, font, flags, text): """ Returns the size, that is needed to render text :param QFont font: Font of the text :param int flags: Bitwise OR of the flags like in for QPainter::drawText :param str text: Text to be rendered :return: Calculated size """ fm = self.fontmetrics_f(font) rect = fm.boundingRect(self.qrectf_max, flags, text) return rect.size()
def effectiveAscent(self, font): global ASCENTCACHE fontKey = font.key() ascent = ASCENTCACHE.get(fontKey) if ascent is not None: return ascent return ASCENTCACHE.setdefault(fontKey, self.findAscent(font)) def findAscent(self, font): dummy = "E" white = QColor(Qt.white) fm = self.fontmetrics(font) boundingr = fm.boundingRect(dummy) pm = QPixmap(boundingr.width(), boundingr.height()) pm.fill(white) p = QPainter(pm) p.setFont(font) p.drawText(0, 0, pm.width(), pm.height(), 0, dummy) p.end() img = pm.toImage() w = pm.width() linebytes = w * 4 for row in range(img.height()): if QT_API.startswith("pyside"): line = bytes(img.scanLine(row)) else: line = img.scanLine(row).asstring(linebytes) for col in range(w): color = struct.unpack("I", line[col * 4 : (col + 1) * 4])[0] if color != white.rgb(): return fm.ascent() - row + 1 return fm.ascent()
[docs] def textMargins(self, font): """ Return margins around the texts The textSize might include margins around the text, like QFontMetrics::descent(). In situations where texts need to be aligned in detail, knowing these margins might improve the layout calculations. :param QFont font: Font of the text :return: tuple (left, right, top, bottom) representing margins """ left = right = 0 fm = self.fontmetrics(font) top = fm.ascent() - self.effectiveAscent(font) bottom = fm.descent() return left, right, top, bottom
[docs] def draw(self, painter, rect, flags, text): """ Draw the text in a clipping rectangle :param QPainter painter: Painter :param QRectF rect: Clipping rectangle :param int flags: Bitwise OR of the flags like in for QPainter::drawText() :param str text: Text to be rendered """ painter.save() qwtUnscaleFont(painter) painter.drawText(rect, flags, text) painter.restore()
[docs] def mightRender(self, text): """ Test if a string can be rendered by this text engine :param str text: Text to be tested :return: True, if it can be rendered """ return True
[docs] class QwtRichTextEngine(QwtTextEngine): """ A text engine for `Qt` rich texts `QwtRichTextEngine` renders `Qt` rich texts using the classes of the Scribe framework of `Qt`. """ def __init__(self): pass
[docs] def heightForWidth(self, font, flags, text, width): """ Find the height for a given width :param QFont font: Font of the text :param int flags: Bitwise OR of the flags used like in QPainter::drawText :param str text: Text to be rendered :param float width: Width :return: Calculated height """ doc = QwtRichTextDocument(text, flags, font) doc.setPageSize(QSizeF(width, QWIDGETSIZE_MAX)) return doc.documentLayout().documentSize().height()
[docs] def textSize(self, font, flags, text): """ Returns the size, that is needed to render text :param QFont font: Font of the text :param int flags: Bitwise OR of the flags like in for QPainter::drawText :param str text: Text to be rendered :return: Calculated size """ doc = QwtRichTextDocument(text, flags, font) option = doc.defaultTextOption() if option.wrapMode() != QTextOption.NoWrap: option.setWrapMode(QTextOption.NoWrap) doc.setDefaultTextOption(option) doc.adjustSize() return doc.size()
[docs] def draw(self, painter, rect, flags, text): """ Draw the text in a clipping rectangle :param QPainter painter: Painter :param QRectF rect: Clipping rectangle :param int flags: Bitwise OR of the flags like in for QPainter::drawText() :param str text: Text to be rendered """ txt = QwtRichTextDocument(text, flags, painter.font()) painter.save() unscaledRect = QRectF(rect) if painter.font().pixelSize() < 0: dpix, dpiy = get_screen_resolution() pd = painter.device() if pd.logicalDpiX() != dpix or pd.logicalDpiY() != dpiy: transform = QTransform() transform.scale( dpix / float(pd.logicalDpiX()), dpiy / float(pd.logicalDpiY()) ) painter.setWorldTransform(transform, True) invtrans, _ok = transform.inverted() unscaledRect = invtrans.mapRect(rect) txt.setDefaultFont(painter.font()) txt.setPageSize(QSizeF(unscaledRect.width(), QWIDGETSIZE_MAX)) layout = txt.documentLayout() height = layout.documentSize().height() y = unscaledRect.y() if flags & Qt.AlignBottom: y += unscaledRect.height() - height elif flags & Qt.AlignVCenter: y += (unscaledRect.height() - height) / 2 context = QAbstractTextDocumentLayout.PaintContext() context.palette.setColor(QPalette.Text, painter.pen().color()) painter.translate(unscaledRect.x(), y) layout.draw(painter, context) painter.restore()
def taggedText(self, text, flags): return taggedRichText(text, flags)
[docs] def mightRender(self, text): """ Test if a string can be rendered by this text engine :param str text: Text to be tested :return: True, if it can be rendered """ try: return Qt.mightBeRichText(text) except AttributeError: return True
[docs] def textMargins(self, font): """ Return margins around the texts The textSize might include margins around the text, like QFontMetrics::descent(). In situations where texts need to be aligned in detail, knowing these margins might improve the layout calculations. :param QFont font: Font of the text :return: tuple (left, right, top, bottom) representing margins """ return 0, 0, 0, 0
class QwtText_PrivateData(object): def __init__(self): self.renderFlags = Qt.AlignCenter self.borderRadius = 0 self.borderPen = Qt.NoPen self.backgroundBrush = Qt.NoBrush self.paintAttributes = 0 self.layoutAttributes = 0 self.textEngine = None self.text = None self.font = None self.color = None class QwtText_LayoutCache(object): def __init__(self): self.textSize = QSizeF() self.font = None def invalidate(self): self.textSize = QSizeF()
[docs] class QwtText(object): """ A class representing a text A `QwtText` is a text including a set of attributes how to render it. - Format: A text might include control sequences (f.e tags) describing how to render it. Each format (f.e MathML, TeX, Qt Rich Text) has its own set of control sequences, that can be handles by a special `QwtTextEngine` for this format. - Background: A text might have a background, defined by a `QPen` and `QBrush` to improve its visibility. The corners of the background might be rounded. - Font: A text might have an individual font. - Color A text might have an individual color. - Render Flags Flags from `Qt.AlignmentFlag` and `Qt.TextFlag` used like in `QPainter.drawText()`. ..seealso:: :py:meth:`qwt.text.QwtTextEngine`, :py:meth:`qwt.text.QwtTextLabel` Text formats: * `QwtText.AutoText`: The text format is determined using `QwtTextEngine.mightRender()` for all available text engines in increasing order > PlainText. If none of the text engines can render the text is rendered like `QwtText.PlainText`. * `QwtText.PlainText`: Draw the text as it is, using a QwtPlainTextEngine. * `QwtText.RichText`: Use the Scribe framework (Qt Rich Text) to render the text. * `QwtText.OtherFormat`: The number of text formats can be extended using `setTextEngine`. Formats >= `QwtText.OtherFormat` are not used by Qwt. Paint attributes: * `QwtText.PaintUsingTextFont`: The text has an individual font. * `QwtText.PaintUsingTextColor`: The text has an individual color. * `QwtText.PaintBackground`: The text has an individual background. Layout attributes: * `QwtText.MinimumLayout`: Layout the text without its margins. This mode is useful if a text needs to be aligned accurately, like the tick labels of a scale. If `QwtTextEngine.textMargins` is not implemented for the format of the text, `MinimumLayout` has no effect. .. py:class:: QwtText([text=None], [textFormat=None], [other=None]) :param str text: Text content :param int textFormat: Text format :param qwt.text.QwtText other: Object to copy (text and textFormat arguments are ignored) """ # enum TextFormat AutoText, PlainText, RichText = list(range(3)) OtherFormat = 100 # enum PaintAttribute PaintUsingTextFont = 0x01 PaintUsingTextColor = 0x02 PaintBackground = 0x04 # enum LayoutAttribute MinimumLayout = 0x01 # Optimization: a single text engine for all QwtText objects # (this is not how it's implemented in Qwt6 C++ library) __map = {PlainText: QwtPlainTextEngine(), RichText: QwtRichTextEngine()} def __init__(self, text=None, textFormat=None, other=None): if text is None: text = "" if textFormat is None: textFormat = self.AutoText if other is not None: text = other if isinstance(text, QwtText): self.__data = text.__data self.__layoutCache = text.__layoutCache else: self.__data = QwtText_PrivateData() self.__data.text = text self.__data.textEngine = self.textEngine(text, textFormat) self.__layoutCache = QwtText_LayoutCache()
[docs] @classmethod def make( cls, text=None, textformat=None, renderflags=None, font=None, family=None, pointsize=None, weight=None, color=None, borderradius=None, borderpen=None, brush=None, ): """ Create and setup a new `QwtText` object (convenience function). :param str text: Text content :param int textformat: Text format :param int renderflags: Flags from `Qt.AlignmentFlag` and `Qt.TextFlag` :param font: Font :type font: QFont or None :param family: Font family (default: Helvetica) :type family: str or None :param pointsize: Font point size (default: 10) :type pointsize: int or None :param weight: Font weight (default: QFont.Normal) :type weight: int or None :param color: Pen color :type color: QColor or str or None :param borderradius: Radius for the corners of the border frame :type borderradius: float or None :param borderpen: Background pen :type borderpen: QPen or None :param brush: Background brush :type brush: QBrush or None .. seealso:: :py:meth:`setText()` """ item = cls(text=text, textFormat=textformat) if renderflags is not None: item.setRenderFlags(renderflags) if font is not None: item.setFont(font) elif family is not None or pointsize is not None or weight is not None: family = "Helvetica" if family is None else family pointsize = 10 if pointsize is None else pointsize weight = QFont.Normal if weight is None else weight item.setFont(QFont(family, pointsize, weight)) if color is not None: item.setColor(qcolor_from_str(color, Qt.black)) if borderradius is not None: item.setBorderRadius(borderradius) if borderpen is not None: item.setBorderPen(borderpen) if brush is not None: item.setBackgroundBrush(brush) return item
def __eq__(self, other): return ( self.__data.renderFlags == other.__data.renderFlags and self.__data.text == other.__data.text and self.__data.font == other.__data.font and self.__data.color == other.__data.color and self.__data.borderRadius == other.__data.borderRadius and self.__data.borderPen == other.__data.borderPen and self.__data.backgroundBrush == other.__data.backgroundBrush and self.__data.paintAttributes == other.__data.paintAttributes and self.__data.textEngine == other.__data.textEngine ) def __ne__(self, other): return not self.__eq__(other)
[docs] def isEmpty(self): """ :return: True if text is empty """ return len(self.text()) == 0
[docs] def setText(self, text, textFormat=None): """ Assign a new text content :param str text: Text content :param int textFormat: Text format .. seealso:: :py:meth:`text()` """ if textFormat is None: textFormat = self.AutoText self.__data.text = text self.__data.textEngine = self.textEngine(text, textFormat) self.__layoutCache.invalidate()
[docs] def text(self): """ :return: Text content .. seealso:: :py:meth:`setText()` """ return self.__data.text
[docs] def setRenderFlags(self, renderFlags): """ Change the render flags The default setting is `Qt.AlignCenter` :param int renderFlags: Bitwise OR of the flags used like in `QPainter.drawText()` .. seealso:: :py:meth:`renderFlags()`, :py:meth:`qwt.text.QwtTextEngine.draw()` """ renderFlags = Qt.AlignmentFlag(renderFlags) if renderFlags != self.__data.renderFlags: self.__data.renderFlags = renderFlags self.__layoutCache.invalidate()
[docs] def renderFlags(self): """ :return: Render flags .. seealso:: :py:meth:`setRenderFlags()` """ return self.__data.renderFlags
[docs] def setFont(self, font): """ Set the font. :param QFont font: Font .. note:: Setting the font might have no effect, when the text contains control sequences for setting fonts. .. seealso:: :py:meth:`font()`, :py:meth:`usedFont()` """ self.__data.font = font self.setPaintAttribute(self.PaintUsingTextFont)
[docs] def font(self): """ :return: Return the font .. seealso:: :py:meth:`setFont()`, :py:meth:`usedFont()` """ return self.__data.font
[docs] def usedFont(self, defaultFont): """ Return the font of the text, if it has one. Otherwise return defaultFont. :param QFont defaultFont: Default font :return: Font used for drawing the text .. seealso:: :py:meth:`setFont()`, :py:meth:`font()` """ if self.__data.paintAttributes & self.PaintUsingTextFont: return self.__data.font return defaultFont
[docs] def setColor(self, color): """ Set the pen color used for drawing the text. :param QColor color: Color .. note:: Setting the color might have no effect, when the text contains control sequences for setting colors. .. seealso:: :py:meth:`color()`, :py:meth:`usedColor()` """ self.__data.color = QColor(color) self.setPaintAttribute(self.PaintUsingTextColor)
[docs] def color(self): """ :return: Return the pen color, used for painting the text .. seealso:: :py:meth:`setColor()`, :py:meth:`usedColor()` """ return self.__data.color
[docs] def usedColor(self, defaultColor): """ Return the color of the text, if it has one. Otherwise return defaultColor. :param QColor defaultColor: Default color :return: Color used for drawing the text .. seealso:: :py:meth:`setColor()`, :py:meth:`color()` """ if self.__data.paintAttributes & self.PaintUsingTextColor: return self.__data.color return defaultColor
[docs] def setBorderRadius(self, radius): """ Set the radius for the corners of the border frame :param float radius: Radius of a rounded corner .. seealso:: :py:meth:`borderRadius()`, :py:meth:`setBorderPen()`, :py:meth:`setBackgroundBrush()` """ self.__data.borderRadius = max([0.0, radius])
[docs] def borderRadius(self): """ :return: Radius for the corners of the border frame .. seealso:: :py:meth:`setBorderRadius()`, :py:meth:`borderPen()`, :py:meth:`backgroundBrush()` """ return self.__data.borderRadius
[docs] def setBorderPen(self, pen): """ Set the background pen :param QPen pen: Background pen .. seealso:: :py:meth:`borderPen()`, :py:meth:`setBackgroundBrush()` """ self.__data.borderPen = pen self.setPaintAttribute(self.PaintBackground)
[docs] def borderPen(self): """ :return: Background pen .. seealso:: :py:meth:`setBorderPen()`, :py:meth:`backgroundBrush()` """ return self.__data.borderPen
[docs] def setBackgroundBrush(self, brush): """ Set the background brush :param QBrush brush: Background brush .. seealso:: :py:meth:`backgroundBrush()`, :py:meth:`setBorderPen()` """ self.__data.backgroundBrush = brush self.setPaintAttribute(self.PaintBackground)
[docs] def backgroundBrush(self): """ :return: Background brush .. seealso:: :py:meth:`setBackgroundBrush()`, :py:meth:`borderPen()` """ return self.__data.backgroundBrush
[docs] def setPaintAttribute(self, attribute, on=True): """ Change a paint attribute :param int attribute: Paint attribute :param bool on: On/Off .. note:: Used by `setFont()`, `setColor()`, `setBorderPen()` and `setBackgroundBrush()` .. seealso:: :py:meth:`testPaintAttribute()` """ if on: self.__data.paintAttributes |= attribute else: self.__data.paintAttributes &= ~attribute
[docs] def testPaintAttribute(self, attribute): """ Test a paint attribute :param int attribute: Paint attribute :return: True, if attribute is enabled .. seealso:: :py:meth:`setPaintAttribute()` """ return self.__data.paintAttributes & attribute
[docs] def setLayoutAttribute(self, attribute, on=True): """ Change a layout attribute :param int attribute: Layout attribute :param bool on: On/Off .. seealso:: :py:meth:`testLayoutAttribute()` """ if on: self.__data.layoutAttributes |= attribute else: self.__data.layoutAttributes &= ~attribute
[docs] def testLayoutAttribute(self, attribute): """ Test a layout attribute :param int attribute: Layout attribute :return: True, if attribute is enabled .. seealso:: :py:meth:`setLayoutAttribute()` """ return self.__data.layoutAttributes & attribute
[docs] def heightForWidth(self, width, defaultFont=None): """ Find the height for a given width :param float width: Width :param QFont defaultFont: Font, used for the calculation if the text has no font :return: Calculated height """ if defaultFont is None: defaultFont = QFont() font = QFont(self.usedFont(defaultFont)) h = 0 if self.__data.layoutAttributes & self.MinimumLayout: (left, right, top, bottom) = self.__data.textEngine.textMargins(font) h = self.__data.textEngine.heightForWidth( font, self.__data.renderFlags, self.__data.text, width + left + right ) h -= top + bottom else: h = self.__data.textEngine.heightForWidth( font, self.__data.renderFlags, self.__data.text, width ) return h
[docs] def textSize(self, defaultFont): """ Returns the size, that is needed to render text :param QFont defaultFont Font, used for the calculation if the text has no font :return: Caluclated size """ font = QFont(self.usedFont(defaultFont)) if ( not self.__layoutCache.textSize.isValid() or self.__layoutCache.font is not font ): self.__layoutCache.textSize = self.__data.textEngine.textSize( font, self.__data.renderFlags, self.__data.text ) self.__layoutCache.font = font sz = self.__layoutCache.textSize if self.__data.layoutAttributes & self.MinimumLayout: (left, right, top, bottom) = self.__data.textEngine.textMargins(font) sz -= QSizeF(left + right, top + bottom) return sz
[docs] def draw(self, painter, rect): """ Draw a text into a rectangle :param QPainter painter: Painter :param QRectF rect: Rectangle """ if self.__data.paintAttributes & self.PaintBackground: if ( self.__data.borderPen != Qt.NoPen or self.__data.backgroundBrush != Qt.NoBrush ): painter.save() painter.setPen(self.__data.borderPen) painter.setBrush(self.__data.backgroundBrush) if self.__data.borderRadius == 0: painter.drawRect(rect) else: painter.setRenderHint(QPainter.Antialiasing, True) painter.drawRoundedRect( rect, self.__data.borderRadius, self.__data.borderRadius ) painter.restore() painter.save() if self.__data.paintAttributes & self.PaintUsingTextFont: painter.setFont(self.__data.font) if self.__data.paintAttributes & self.PaintUsingTextColor: if self.__data.color.isValid(): painter.setPen(self.__data.color) expandedRect = rect if self.__data.layoutAttributes & self.MinimumLayout: font = QFont(painter.font()) (left, right, top, bottom) = self.__data.textEngine.textMargins(font) expandedRect.setTop(rect.top() - top) expandedRect.setBottom(rect.bottom() + bottom) expandedRect.setLeft(rect.left() - left) expandedRect.setRight(rect.right() + right) self.__data.textEngine.draw( painter, expandedRect, self.__data.renderFlags, self.__data.text ) painter.restore()
[docs] def textEngine(self, text=None, format_=None): """ Find the text engine for a text format In case of `QwtText.AutoText` the first text engine (beside `QwtPlainTextEngine`) is returned, where `QwtTextEngine.mightRender` returns true. If there is none `QwtPlainTextEngine` is returned. If no text engine is registered for the format `QwtPlainTextEngine` is returned. :param str text: Text, needed in case of AutoText :param int format: Text format :return: Corresponding text engine """ if text is None: return self.__map.get(format_) elif format_ is not None: if format_ == QwtText.AutoText: for key, engine in list(self.__map.items()): if key != QwtText.PlainText: if engine and engine.mightRender(text): return engine engine = self.__map.get(format_) if engine is not None: return engine return self.__map[QwtText.PlainText] else: raise TypeError( "%s().textEngine() takes 1 or 2 argument(s) (none" " given)" % self.__class__.__name__ )
[docs] def setTextEngine(self, format_, engine): """ Assign/Replace a text engine for a text format With setTextEngine it is possible to extend `PythonQwt` with other types of text formats. For `QwtText.PlainText` it is not allowed to assign a engine to None. :param int format_: Text format :param qwt.text.QwtTextEngine engine: Text engine .. seealso:: :py:meth:`setPaintAttribute()` .. warning:: Using `QwtText.AutoText` does nothing. """ if format_ == QwtText.AutoText: return if format_ == QwtText.PlainText and engine is None: return self.__map.setdefault(format_, engine)
class QwtTextLabel_PrivateData(object): def __init__(self): self.indent = 4 self.margin = 0 self.text = QwtText()
[docs] class QwtTextLabel(QFrame): """ A Widget which displays a QwtText .. py:class:: QwtTextLabel(parent) :param QWidget parent: Parent widget .. py:class:: QwtTextLabel([text=None], [parent=None]) :noindex: :param str text: Text :param QWidget parent: Parent widget """ def __init__(self, *args): if len(args) == 0: text, parent = None, None elif len(args) == 1: if isinstance(args[0], QWidget): text = None (parent,) = args else: parent = None (text,) = args elif len(args) == 2: text, parent = args else: raise TypeError( "%s() takes 0, 1 or 2 argument(s) (%s given)" % (self.__class__.__name__, len(args)) ) super(QwtTextLabel, self).__init__(parent) self.init() if text is not None: self.__data.text = text def init(self): self.__data = QwtTextLabel_PrivateData() self.setSizePolicy(QSizePolicy.Preferred, QSizePolicy.Preferred)
[docs] def setPlainText(self, text): """ Interface for the designer plugin - does the same as setText() :param str text: Text .. seealso:: :py:meth:`plainText()` """ self.setText(QwtText(text))
[docs] def plainText(self): """ Interface for the designer plugin :return: Text as plain text .. seealso:: :py:meth:`setPlainText()` """ return self.__data.text.text()
[docs] def setText(self, text, textFormat=QwtText.AutoText): """ Change the label's text, keeping all other QwtText attributes :param text: New text :type text: qwt.text.QwtText or str :param int textFormat: Format of text .. seealso:: :py:meth:`text()` """ if isinstance(text, QwtText): self.__data.text = text else: self.__data.text.setText(text, textFormat) self.update() self.updateGeometry()
[docs] def text(self): """ :return: Return the text .. seealso:: :py:meth:`setText()` """ return self.__data.text
[docs] def clear(self): """ Clear the text and all `QwtText` attributes """ self.__data.text = QwtText() self.update() self.updateGeometry()
[docs] def indent(self): """ :return: Label's text indent in pixels .. seealso:: :py:meth:`setIndent()` """ return self.__data.indent
[docs] def setIndent(self, indent): """ Set label's text indent in pixels :param int indent: Indentation in pixels .. seealso:: :py:meth:`indent()` """ if indent < 0: indent = 0 self.__data.indent = indent self.update() self.updateGeometry()
[docs] def margin(self): """ :return: Label's text indent in pixels .. seealso:: :py:meth:`setMargin()` """ return self.__data.margin
[docs] def setMargin(self, margin): """ Set label's margin in pixels :param int margin: Margin in pixels .. seealso:: :py:meth:`margin()` """ self.__data.margin = margin self.update() self.updateGeometry()
[docs] def sizeHint(self): """ Return a size hint """ return self.minimumSizeHint()
[docs] def minimumSizeHint(self): """ Return a minimum size hint """ sz = self.__data.text.textSize(self.font()) mw = 2 * (self.frameWidth() + self.__data.margin) mh = mw indent = self.__data.indent if indent <= 0: indent = self.defaultIndent() if indent > 0: align = self.__data.text.renderFlags() if align & Qt.AlignLeft or align & Qt.AlignRight: mw += self.__data.indent elif align & Qt.AlignTop or align & Qt.AlignBottom: mh += self.__data.indent sz += QSizeF(mw, mh) return QSize(math.ceil(sz.width()), math.ceil(sz.height()))
[docs] def heightForWidth(self, width): """ :param int width: Width :return: Preferred height for this widget, given the width. """ renderFlags = self.__data.text.renderFlags() indent = self.__data.indent if indent <= 0: indent = self.defaultIndent() width -= 2 * self.frameWidth() if renderFlags & Qt.AlignLeft or renderFlags & Qt.AlignRight: width -= indent height = math.ceil(self.__data.text.heightForWidth(width, self.font())) if renderFlags & Qt.AlignTop or renderFlags & Qt.AlignBottom: height += indent height += 2 * self.frameWidth() return height
[docs] def paintEvent(self, event): painter = QPainter(self) if not self.contentsRect().contains(event.rect()): painter.save() painter.setClipRegion(event.region() & self.frameRect()) self.drawFrame(painter) painter.restore() painter.setClipRegion(event.region() & self.contentsRect()) self.drawContents(painter)
[docs] def drawContents(self, painter): """ Redraw the text and focus indicator :param QPainter painter: Painter """ r = self.textRect() if r.isEmpty(): return painter.setFont(self.font()) painter.setPen(self.palette().color(QPalette.Active, QPalette.Text)) self.drawText(painter, QRectF(r)) if self.hasFocus(): m = 2 focusRect = self.contentsRect().adjusted(m, m, -m + 1, -m + 1) QwtPainter.drawFocusRect(painter, self, focusRect)
[docs] def drawText(self, painter, textRect): """ Redraw the text :param QPainter painter: Painter :param QRectF textRect: Text rectangle """ self.__data.text.draw(painter, textRect)
[docs] def textRect(self): """ Calculate geometry for the text in widget coordinates :return: Geometry for the text """ r = self.contentsRect() if not r.isEmpty() and self.__data.margin > 0: r.setRect( r.x() + self.__data.margin, r.y() + self.__data.margin, r.width() - 2 * self.__data.margin, r.height() - 2 * self.__data.margin, ) if not r.isEmpty(): indent = self.__data.indent if indent <= 0: indent = self.defaultIndent() if indent > 0: renderFlags = self.__data.text.renderFlags() if renderFlags & Qt.AlignLeft: r.setX(r.x() + indent) elif renderFlags & Qt.AlignRight: r.setWidth(r.width() - indent) elif renderFlags & Qt.AlignTop: r.setY(r.y() + indent) elif renderFlags & Qt.AlignBottom: r.setHeight(r.height() - indent) return r
def defaultIndent(self): if self.frameWidth() <= 0: return 0 if self.__data.text.testPaintAttribute(QwtText.PaintUsingTextFont): fnt = self.__data.text.font() else: fnt = self.font() return QFontMetrics(fnt).boundingRect("x").width() / 2