地面站终端 App
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

691 lines
18 KiB

/* -*- mode: C++ ; c-file-style: "stroustrup" -*- *****************************
* Qwt Widget Library
* Copyright (C) 1997 Josef Wilgen
* Copyright (C) 2002 Uwe Rathmann
*
* This library is free software; you can redistribute it and/or
* modify it under the terms of the Qwt License, Version 1.0
*****************************************************************************/
// vim: expandtab
#include <qwindowdefs.h>
#include <qwidget.h>
#include <qrect.h>
#include <qpainter.h>
#include <qpalette.h>
#include <qpaintdevice.h>
#include <qpixmap.h>
#include <qstyle.h>
#if QT_VERSION < 0x040000
#include <qsimplerichtext.h>
#else
#include <qtextdocument.h>
#include <qabstracttextdocumentlayout.h>
#include <qstyleoption.h>
#include <qpaintengine.h>
#endif
#include "qwt_clipper.h"
#include "qwt_math.h"
#include "qwt_color_map.h"
#include "qwt_scale_map.h"
#include "qwt_painter.h"
QwtMetricsMap QwtPainter::d_metricsMap;
#if defined(Q_WS_X11)
bool QwtPainter::d_deviceClipping = true;
#else
bool QwtPainter::d_deviceClipping = false;
#endif
#if QT_VERSION < 0x040000
bool QwtPainter::d_SVGMode = false;
#endif
static inline bool needDeviceClipping(
const QPainter *painter, bool deviceClipping)
{
return deviceClipping &&
(painter->device()->devType() == QInternal::Widget ||
painter->device()->devType() == QInternal::Pixmap );
}
/*!
\brief En/Disable device clipping.
On X11 the default for device clipping is enabled,
otherwise it is disabled.
\sa QwtPainter::deviceClipping()
*/
void QwtPainter::setDeviceClipping(bool enable)
{
d_deviceClipping = enable;
}
/*!
Returns whether device clipping is enabled. On X11 the default
is enabled, otherwise it is disabled.
\sa QwtPainter::setDeviceClipping()
*/
bool QwtPainter::deviceClipping()
{
return d_deviceClipping;
}
/*!
Returns rect for device clipping
\sa QwtPainter::setDeviceClipping()
*/
const QRect &QwtPainter::deviceClipRect()
{
static QRect clip;
if ( !clip.isValid() ) {
clip.setCoords(QWT_COORD_MIN, QWT_COORD_MIN,
QWT_COORD_MAX, QWT_COORD_MAX);
}
return clip;
}
//! Clip a point array
QwtPolygon QwtPainter::clip(const QwtPolygon &pa)
{
return QwtClipper::clipPolygon(deviceClipRect(), pa);
}
#if QT_VERSION < 0x040000
/*!
\brief En/Disable SVG mode.
When saving a QPicture to a SVG some texts are misaligned.
In SVGMode QwtPainter tries to fix them.
\sa QwtPainter::isSVGMode()
\note A QPicture that is created in SVG mode and saved to the
native format, will be misaligned. Also it is not possible to
reload and play a SVG document, that was created in SVG mode.
*/
void QwtPainter::setSVGMode(bool on)
{
d_SVGMode = on;
}
bool QwtPainter::isSVGMode()
{
return d_SVGMode;
}
#endif // QT_VERSION < 0x040000
/*!
Scale all QwtPainter drawing operations using the ratio
QwtPaintMetrics(from).logicalDpiX() / QwtPaintMetrics(to).logicalDpiX()
and QwtPaintMetrics(from).logicalDpiY() / QwtPaintMetrics(to).logicalDpiY()
\sa QwtPainter::resetScaleMetrics(), QwtPainter::scaleMetricsX,
QwtPainter::scaleMetricsY()
*/
void QwtPainter::setMetricsMap(const QPaintDevice *layout,
const QPaintDevice *device)
{
d_metricsMap.setMetrics(layout, device);
}
/*!
Change the metrics map
\sa QwtPainter::resetMetricsMap, QwtPainter::metricsMap
*/
void QwtPainter::setMetricsMap(const QwtMetricsMap &map)
{
d_metricsMap = map;
}
/*!
Reset the metrics map to the ratio 1:1
\sa QwtPainter::setMetricsMap, QwtPainter::resetMetricsMap
*/
void QwtPainter::resetMetricsMap()
{
d_metricsMap = QwtMetricsMap();
}
/*!
\return Metrics map
*/
const QwtMetricsMap &QwtPainter::metricsMap()
{
return d_metricsMap;
}
/*!
Wrapper for QPainter::setClipRect()
*/
void QwtPainter::setClipRect(QPainter *painter, const QRect &rect)
{
painter->setClipRect(d_metricsMap.layoutToDevice(rect, painter));
}
/*!
Wrapper for QPainter::drawRect()
*/
void QwtPainter::drawRect(QPainter *painter, int x, int y, int w, int h)
{
drawRect(painter, QRect(x, y, w, h));
}
/*!
Wrapper for QPainter::drawRect()
*/
void QwtPainter::drawRect(QPainter *painter, const QRect &rect)
{
const QRect r = d_metricsMap.layoutToDevice(rect, painter);
QRect clipRect;
const bool deviceClipping = needDeviceClipping(painter, d_deviceClipping);
if ( deviceClipping )
clipRect = deviceClipRect();
if ( clipRect.isValid() ) {
if ( !clipRect.intersects(r) )
return;
if ( !clipRect.contains(r) ) {
fillRect(painter, r & clipRect, painter->brush());
int pw = painter->pen().width();
pw = pw % 2 + pw / 2;
QwtPolygon pa(5);
pa.setPoint(0, r.left(), r.top());
pa.setPoint(1, r.right() - pw, r.top());
pa.setPoint(2, r.right() - pw, r.bottom() - pw);
pa.setPoint(3, r.left(), r.bottom() - pw);
pa.setPoint(4, r.left(), r.top());
painter->save();
painter->setBrush(Qt::NoBrush);
drawPolyline(painter, pa);
painter->restore();
return;
}
}
painter->drawRect(r);
}
/*!
Wrapper for QPainter::fillRect()
*/
void QwtPainter::fillRect(QPainter *painter,
const QRect &rect, const QBrush &brush)
{
if ( !rect.isValid() )
return;
const bool deviceClipping = needDeviceClipping(painter, d_deviceClipping);
QRect clipRect;
#if QT_VERSION >= 0x040000
/*
Performance of Qt4 is horrible for non trivial brushs. Without
clipping expect minutes or hours for repainting large rects
(might result from zooming)
*/
clipRect = painter->window();
if ( painter->hasClipping() )
clipRect &= painter->clipRegion().boundingRect();
if ( deviceClipping )
clipRect &= deviceClipRect();
#else
if ( deviceClipping )
clipRect = deviceClipRect();
#endif
QRect r = d_metricsMap.layoutToDevice(rect, painter);
if ( clipRect.isValid() )
r = r.intersect(clipRect);
if ( r.isValid() )
painter->fillRect(r, brush);
}
/*!
Wrapper for QPainter::drawPie()
*/
void QwtPainter::drawPie(QPainter *painter, const QRect &rect,
int a, int alen)
{
QRect r = d_metricsMap.layoutToDevice(rect, painter);
const bool deviceClipping = needDeviceClipping(painter, d_deviceClipping);
if ( deviceClipping && !deviceClipRect().contains(rect) )
return;
painter->drawPie(r, a, alen);
}
/*!
Wrapper for QPainter::drawEllipse()
*/
void QwtPainter::drawEllipse(QPainter *painter, const QRect &rect)
{
QRect r = d_metricsMap.layoutToDevice(rect, painter);
const bool deviceClipping = needDeviceClipping(painter, d_deviceClipping);
if ( deviceClipping && !deviceClipRect().contains(rect) )
return;
#if QT_VERSION >= 0x040000
if ( painter->pen().style() != Qt::NoPen &&
painter->pen().color().isValid() ) {
// Qt4 adds the pen to the rect, Qt3 not.
int pw = painter->pen().width();
if ( pw == 0 )
pw = 1;
r.setWidth(r.width() - pw);
r.setHeight(r.height() - pw);
}
#endif
painter->drawEllipse(r);
}
/*!
Wrapper for QPainter::drawText()
*/
void QwtPainter::drawText(QPainter *painter, int x, int y,
const QString &text)
{
drawText(painter, QPoint(x, y), text);
}
/*!
Wrapper for QPainter::drawText()
*/
void QwtPainter::drawText(QPainter *painter, const QPoint &pos,
const QString &text)
{
const QPoint p = d_metricsMap.layoutToDevice(pos, painter);
const bool deviceClipping = needDeviceClipping(painter, d_deviceClipping);
if ( deviceClipping && !deviceClipRect().contains(p) )
return;
painter->drawText(p, text);
}
/*!
Wrapper for QPainter::drawText()
*/
void QwtPainter::drawText(QPainter *painter, int x, int y, int w, int h,
int flags, const QString &text)
{
drawText(painter, QRect(x, y, w, h), flags, text);
}
/*!
Wrapper for QPainter::drawText()
*/
void QwtPainter::drawText(QPainter *painter, const QRect &rect,
int flags, const QString &text)
{
QRect textRect = d_metricsMap.layoutToDevice(rect, painter);
#if QT_VERSION < 0x040000
if ( d_SVGMode &&
( flags == 0 || flags & Qt::AlignVCenter )
&& painter->device()->devType() == QInternal::Picture ) {
/*
Qt3 misalignes texts, when saving a text
to a SVG image.
*/
textRect.setY(textRect.y() - painter->fontMetrics().height() / 4);
}
#endif
painter->drawText(textRect, flags, text);
}
#ifndef QT_NO_RICHTEXT
/*!
Wrapper for QSimpleRichText::draw()
*/
#if QT_VERSION < 0x040000
void QwtPainter::drawSimpleRichText(QPainter *painter, const QRect &rect,
int flags, QSimpleRichText &text)
{
QColorGroup cg;
cg.setColor(QColorGroup::Text, painter->pen().color());
const QRect scaledRect = d_metricsMap.layoutToDevice(rect, painter);
text.setWidth(painter, scaledRect.width());
// QSimpleRichText is Qt::AlignTop by default
int y = scaledRect.y();
if (flags & Qt::AlignBottom)
y += (scaledRect.height() - text.height());
else if (flags & Qt::AlignVCenter)
y += (scaledRect.height() - text.height())/2;
text.draw(painter, scaledRect.x(), y, scaledRect, cg);
}
#else
void QwtPainter::drawSimpleRichText(QPainter *painter, const QRect &rect,
int flags, QTextDocument &text)
{
const QRect scaledRect = d_metricsMap.layoutToDevice(rect, painter);
text.setPageSize(QSize(scaledRect.width(), QWIDGETSIZE_MAX));
QAbstractTextDocumentLayout* layout = text.documentLayout();
const int height = qRound(layout->documentSize().height());
int y = scaledRect.y();
if (flags & Qt::AlignBottom)
y += (scaledRect.height() - height);
else if (flags & Qt::AlignVCenter)
y += (scaledRect.height() - height)/2;
QAbstractTextDocumentLayout::PaintContext context;
context.palette.setColor(QPalette::Text, painter->pen().color());
painter->save();
painter->translate(scaledRect.x(), y);
layout->draw(painter, context);
painter->restore();
}
#endif
#endif // !QT_NO_RICHTEXT
/*!
Wrapper for QPainter::drawLine()
*/
void QwtPainter::drawLine(QPainter *painter, int x1, int y1, int x2, int y2)
{
const bool deviceClipping = needDeviceClipping(painter, d_deviceClipping);
if ( deviceClipping &&
!(deviceClipRect().contains(x1, y1) && deviceClipRect().contains(x2, y2)) ) {
QwtPolygon pa(2);
pa.setPoint(0, x1, y1);
pa.setPoint(1, x2, y2);
drawPolyline(painter, pa);
return;
}
if ( d_metricsMap.isIdentity() ) {
#if QT_VERSION >= 0x030200 && QT_VERSION < 0x040000
if ( !painter->device()->isExtDev() )
#endif
{
painter->drawLine(x1, y1, x2, y2);
return;
}
}
const QPoint p1 = d_metricsMap.layoutToDevice(QPoint(x1, y1));
const QPoint p2 = d_metricsMap.layoutToDevice(QPoint(x2, y2));
#if QT_VERSION >= 0x030200 && QT_VERSION < 0x040000
if ( painter->device()->isExtDev() ) {
// Strange: the postscript driver of QPrinter adds an offset
// of 0.5 to the start/endpoint when using drawLine, but not
// for lines painted with drawLineSegments.
QwtPolygon pa(2);
pa.setPoint(0, p1);
pa.setPoint(1, p2);
painter->drawLineSegments(pa);
} else
painter->drawLine(p1, p2);
#else
painter->drawLine(p1, p2);
#endif
}
/*!
Wrapper for QPainter::drawPolygon()
*/
void QwtPainter::drawPolygon(QPainter *painter, const QwtPolygon &pa)
{
const bool deviceClipping = needDeviceClipping(painter, d_deviceClipping);
QwtPolygon cpa = d_metricsMap.layoutToDevice(pa);
if ( deviceClipping ) {
#ifdef __GNUC__
#endif
cpa = clip(cpa);
}
painter->drawPolygon(cpa);
}
/*!
Wrapper for QPainter::drawPolyline()
*/
void QwtPainter::drawPolyline(QPainter *painter, const QwtPolygon &pa)
{
const bool deviceClipping = needDeviceClipping(painter, d_deviceClipping);
QwtPolygon cpa = d_metricsMap.layoutToDevice(pa);
if ( deviceClipping )
cpa = clip(cpa);
#if QT_VERSION >= 0x040000 && QT_VERSION < 0x040400
bool doSplit = false;
if ( painter->paintEngine()->type() == QPaintEngine::Raster &&
painter->pen().width() >= 2 ) {
/*
The raster paint engine seems to use some algo with O(n*n).
( Qt 4.3 is better than Qt 4.2, but remains unacceptable)
To work around this problem, we have to split the polygon into
smaller pieces.
*/
doSplit = true;
}
if ( doSplit ) {
const int numPoints = cpa.size();
const QPoint *points = cpa.data();
const int splitSize = 20;
for ( int i = 0; i < numPoints; i += splitSize ) {
const int n = qwtMin(splitSize + 1, cpa.size() - i);
painter->drawPolyline(points + i, n);
}
} else
#endif
painter->drawPolyline(cpa);
}
/*!
Wrapper for QPainter::drawPoint()
*/
void QwtPainter::drawPoint(QPainter *painter, int x, int y)
{
const bool deviceClipping = needDeviceClipping(painter, d_deviceClipping);
const QPoint pos = d_metricsMap.layoutToDevice(QPoint(x, y));
if ( deviceClipping && !deviceClipRect().contains(pos) )
return;
painter->drawPoint(pos);
}
void QwtPainter::drawColoredArc(QPainter *painter, const QRect &rect,
int peak, int arc, int interval, const QColor &c1, const QColor &c2)
{
int h1, s1, v1;
int h2, s2, v2;
#if QT_VERSION < 0x040000
c1.hsv(&h1, &s1, &v1);
c2.hsv(&h2, &s2, &v2);
#else
c1.getHsv(&h1, &s1, &v1);
c2.getHsv(&h2, &s2, &v2);
#endif
arc /= 2;
for ( int angle = -arc; angle < arc; angle += interval) {
double ratio;
if ( angle >= 0 )
ratio = 1.0 - angle / double(arc);
else
ratio = 1.0 + angle / double(arc);
QColor c;
c.setHsv( h1 + qRound(ratio * (h2 - h1)),
s1 + qRound(ratio * (s2 - s1)),
v1 + qRound(ratio * (v2 - v1)) );
painter->setPen(QPen(c, painter->pen().width()));
painter->drawArc(rect, (peak + angle) * 16, interval * 16);
}
}
void QwtPainter::drawFocusRect(QPainter *painter, QWidget *widget)
{
drawFocusRect(painter, widget, widget->rect());
}
void QwtPainter::drawFocusRect(QPainter *painter, QWidget *widget,
const QRect &rect)
{
#if QT_VERSION < 0x040000
widget->style().drawPrimitive(QStyle::PE_FocusRect, painter,
rect, widget->colorGroup());
#else
QStyleOptionFocusRect opt;
opt.init(widget);
opt.rect = rect;
opt.state |= QStyle::State_HasFocus;
widget->style()->drawPrimitive(QStyle::PE_FrameFocusRect,
&opt, painter, widget);
#endif
}
//! Draw a round frame
#if QT_VERSION < 0x040000
void QwtPainter::drawRoundFrame(QPainter *painter, const QRect &rect,
int width, const QColorGroup &cg, bool sunken)
#else
void QwtPainter::drawRoundFrame(QPainter *painter, const QRect &rect,
int width, const QPalette &palette, bool sunken)
#endif
{
#if QT_VERSION < 0x040000
QColor c0 = cg.mid();
QColor c1, c2;
if ( sunken ) {
c1 = cg.dark();
c2 = cg.light();
} else {
c1 = cg.light();
c2 = cg.dark();
}
#else
QColor c0 = palette.color(QPalette::Mid);
QColor c1, c2;
if ( sunken ) {
c1 = palette.color(QPalette::Dark);
c2 = palette.color(QPalette::Light);
} else {
c1 = palette.color(QPalette::Light);
c2 = palette.color(QPalette::Dark);
}
#endif
painter->setPen(QPen(c0, width));
painter->drawArc(rect, 0, 360 * 16); // full
const int peak = 150;
const int interval = 2;
if ( c0 != c1 )
drawColoredArc(painter, rect, peak, 160, interval, c0, c1);
if ( c0 != c2 )
drawColoredArc(painter, rect, peak + 180, 120, interval, c0, c2);
}
void QwtPainter::drawColorBar(QPainter *painter,
const QwtColorMap &colorMap, const QwtDoubleInterval &interval,
const QwtScaleMap &scaleMap, Qt::Orientation orientation,
const QRect &rect)
{
#if QT_VERSION < 0x040000
QValueVector<QRgb> colorTable;
#else
QVector<QRgb> colorTable;
#endif
if ( colorMap.format() == QwtColorMap::Indexed )
colorTable = colorMap.colorTable(interval);
QColor c;
const QRect devRect = d_metricsMap.layoutToDevice(rect);
/*
We paint to a pixmap first to have something scalable for printing
( f.e. in a Pdf document )
*/
QPixmap pixmap(devRect.size());
QPainter pmPainter(&pixmap);
pmPainter.translate(-devRect.x(), -devRect.y());
if ( orientation == Qt::Horizontal ) {
QwtScaleMap sMap = scaleMap;
sMap.setPaintInterval(devRect.left(), devRect.right());
for ( int x = devRect.left(); x <= devRect.right(); x++ ) {
const double value = sMap.invTransform(x);
if ( colorMap.format() == QwtColorMap::RGB )
c.setRgb(colorMap.rgb(interval, value));
else
c = colorTable[colorMap.colorIndex(interval, value)];
pmPainter.setPen(c);
pmPainter.drawLine(x, devRect.top(), x, devRect.bottom());
}
} else { // Vertical
QwtScaleMap sMap = scaleMap;
sMap.setPaintInterval(devRect.bottom(), devRect.top());
for ( int y = devRect.top(); y <= devRect.bottom(); y++ ) {
const double value = sMap.invTransform(y);
if ( colorMap.format() == QwtColorMap::RGB )
c.setRgb(colorMap.rgb(interval, value));
else
c = colorTable[colorMap.colorIndex(interval, value)];
pmPainter.setPen(c);
pmPainter.drawLine(devRect.left(), y, devRect.right(), y);
}
}
pmPainter.end();
painter->drawPixmap(devRect, pixmap);
}