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.
qt3/src/kernel/qpainter.cpp

3969 lines
100 KiB

/****************************************************************************
**
** Implementation of QPainter, QPen and QBrush classes
**
** Created : 940112
**
** Copyright (C) 1992-2008 Trolltech ASA. All rights reserved.
**
** This file is part of the kernel module of the Qt GUI Toolkit.
**
** This file may be used under the terms of the GNU General
** Public License versions 2.0 or 3.0 as published by the Free
** Software Foundation and appearing in the files LICENSE.GPL2
** and LICENSE.GPL3 included in the packaging of this file.
** Alternatively you may (at your option) use any later version
** of the GNU General Public License if such license has been
** publicly approved by Trolltech ASA (or its successors, if any)
** and the KDE Free Qt Foundation.
**
** Please review the following information to ensure GNU General
** Public Licensing requirements will be met:
** http://trolltech.com/products/qt/licenses/licensing/opensource/.
** If you are unsure which license is appropriate for your use, please
** review the following information:
** http://trolltech.com/products/qt/licenses/licensing/licensingoverview
** or contact the sales department at sales@trolltech.com.
**
** This file may be used under the terms of the Q Public License as
** defined by Trolltech ASA and appearing in the file LICENSE.QPL
** included in the packaging of this file. Licensees holding valid Qt
** Commercial licenses may use this file in accordance with the Qt
** Commercial License Agreement provided with the Software.
**
** This file is provided "AS IS" with NO WARRANTY OF ANY KIND,
** INCLUDING THE WARRANTIES OF DESIGN, MERCHANTABILITY AND FITNESS FOR
** A PARTICULAR PURPOSE. Trolltech reserves all rights not granted
** herein.
**
**********************************************************************/
#include "qpainter.h"
#include "qpainter_p.h"
#include "qbitmap.h"
#include "qptrstack.h"
#include "qptrdict.h"
#include "qdatastream.h"
#include "qwidget.h"
#include "qimage.h"
#include "qpaintdevicemetrics.h"
#include "qapplication.h"
#include "qrichtext_p.h"
#include "qregexp.h"
#include "qcleanuphandler.h"
#ifdef Q_WS_QWS
#include "qgfx_qws.h"
#endif
#include <string.h>
#include "qtextlayout_p.h"
#include "qfontengine_p.h"
#ifndef QT_NO_TRANSFORMATIONS
typedef QPtrStack<QWMatrix> QWMatrixStack;
#endif
// POSIX Large File Support redefines truncate -> truncate64
#if defined(truncate)
# undef truncate
#endif
/*!
\class QPainter qpainter.h
\brief The QPainter class does low-level painting e.g. on widgets.
\ingroup graphics
\ingroup images
\mainclass
The painter provides highly optimized functions to do most of the
drawing GUI programs require. QPainter can draw everything from
simple lines to complex shapes like pies and chords. It can also
draw aligned text and pixmaps. Normally, it draws in a "natural"
coordinate system, but it can also do view and world
transformation.
The typical use of a painter is:
\list
\i Construct a painter.
\i Set a pen, a brush etc.
\i Draw.
\i Destroy the painter.
\endlist
Mostly, all this is done inside a paint event. (In fact, 99% of
all QPainter use is in a reimplementation of
QWidget::paintEvent(), and the painter is heavily optimized for
such use.) Here's one very simple example:
\code
void SimpleExampleWidget::paintEvent()
{
QPainter paint( this );
paint.setPen( Qt::blue );
paint.drawText( rect(), AlignCenter, "The Text" );
}
\endcode
Usage is simple, and there are many settings you can use:
\list
\i font() is the currently set font. If you set a font that isn't
available, Qt finds a close match. In fact font() returns what
you set using setFont() and fontInfo() returns the font actually
being used (which may be the same).
\i brush() is the currently set brush; the color or pattern that's
used for filling e.g. circles.
\i pen() is the currently set pen; the color or stipple that's
used for drawing lines or boundaries.
\i backgroundMode() is \c Opaque or \c Transparent, i.e. whether
backgroundColor() is used or not.
\i backgroundColor() only applies when backgroundMode() is Opaque
and pen() is a stipple. In that case, it describes the color of
the background pixels in the stipple.
\i rasterOp() is how pixels drawn interact with the pixels already
there.
\i brushOrigin() is the origin of the tiled brushes, normally the
origin of the window.
\i viewport(), window(), worldMatrix() and many more make up the
painter's coordinate transformation system. See \link
coordsys.html The Coordinate System \endlink for an explanation of
this, or see below for a very brief overview of the functions.
\i hasClipping() is whether the painter clips at all. (The paint
device clips, too.) If the painter clips, it clips to clipRegion().
\i pos() is the current position, set by moveTo() and used by
lineTo().
\endlist
Note that some of these settings mirror settings in some paint
devices, e.g. QWidget::font(). QPainter::begin() (or the QPainter
constructor) copies these attributes from the paint device.
Calling, for example, QWidget::setFont() doesn't take effect until
the next time a painter begins painting on it.
save() saves all of these settings on an internal stack, restore()
pops them back.
The core functionality of QPainter is drawing, and there are
functions to draw most primitives: drawPoint(), drawPoints(),
drawLine(), drawRect(), drawWinFocusRect(), drawRoundRect(),
drawEllipse(), drawArc(), drawPie(), drawChord(),
drawLineSegments(), drawPolyline(), drawPolygon(),
drawConvexPolygon() and drawCubicBezier(). All of these functions
take integer coordinates; there are no floating-point versions
since we want drawing to be as fast as possible.
There are functions to draw pixmaps/images, namely drawPixmap(),
drawImage() and drawTiledPixmap(). drawPixmap() and drawImage()
produce the same result, except that drawPixmap() is faster
on-screen and drawImage() faster and sometimes better on QPrinter
and QPicture.
Text drawing is done using drawText(), and when you need
fine-grained positioning, boundingRect() tells you where a given
drawText() command would draw.
There is a drawPicture() function that draws the contents of an
entire QPicture using this painter. drawPicture() is the only
function that disregards all the painter's settings: the QPicture
has its own settings.
Normally, the QPainter operates on the device's own coordinate
system (usually pixels), but QPainter has good support for
coordinate transformation. See \link coordsys.html The Coordinate
System \endlink for a more general overview and a simple example.
The most common functions used are scale(), rotate(), translate()
and shear(), all of which operate on the worldMatrix().
setWorldMatrix() can replace or add to the currently set
worldMatrix().
setViewport() sets the rectangle on which QPainter operates. The
default is the entire device, which is usually fine, except on
printers. setWindow() sets the coordinate system, that is, the
rectangle that maps to viewport(). What's drawn inside the
window() ends up being inside the viewport(). The window's
default is the same as the viewport, and if you don't use the
transformations, they are optimized away, gaining another little
bit of speed.
After all the coordinate transformation is done, QPainter can clip
the drawing to an arbitrary rectangle or region. hasClipping() is
TRUE if QPainter clips, and clipRegion() returns the clip region.
You can set it using either setClipRegion() or setClipRect().
Note that the clipping can be slow. It's all system-dependent,
but as a rule of thumb, you can assume that drawing speed is
inversely proportional to the number of rectangles in the clip
region.
After QPainter's clipping, the paint device may also clip. For
example, most widgets clip away the pixels used by child widgets,
and most printers clip away an area near the edges of the paper.
This additional clipping is not reflected by the return value of
clipRegion() or hasClipping().
QPainter also includes some less-used functions that are very
useful on those occasions when they're needed.
isActive() indicates whether the painter is active. begin() (and
the most usual constructor) makes it active. end() (and the
destructor) deactivates it. If the painter is active, device()
returns the paint device on which the painter paints.
Sometimes it is desirable to make someone else paint on an unusual
QPaintDevice. QPainter supports a static function to do this,
redirect(). We recommend not using it, but for some hacks it's
perfect.
setTabStops() and setTabArray() can change where the tab stops
are, but these are very seldomly used.
\warning Note that QPainter does not attempt to work around
coordinate limitations in the underlying window system. Some
platforms may behave incorrectly with coordinates as small as
+/-4000.
\headerfile qdrawutil.h
\sa QPaintDevice QWidget QPixmap QPrinter QPicture
\link simple-application.html Application Walkthrough \endlink
\link coordsys.html Coordinate System Overview \endlink
*/
/*!
\fn QGfx * QPainter::internalGfx()
\internal
*/
/*!
\enum QPainter::CoordinateMode
\value CoordDevice
\value CoordPainter
\sa clipRegion()
*/
/*!
\enum QPainter::TextDirection
\value Auto
\value RTL right to left
\value LTR left to right
\sa drawText()
*/
/*!
\enum Qt::PaintUnit
\value PixelUnit
\value LoMetricUnit \e obsolete
\value HiMetricUnit \e obsolete
\value LoEnglishUnit \e obsolete
\value HiEnglishUnit \e obsolete
\value TwipsUnit \e obsolete
*/
/*!
\enum Qt::BrushStyle
\value NoBrush
\value SolidPattern
\value Dense1Pattern
\value Dense2Pattern
\value Dense3Pattern
\value Dense4Pattern
\value Dense5Pattern
\value Dense6Pattern
\value Dense7Pattern
\value HorPattern
\value VerPattern
\value CrossPattern
\value BDiagPattern
\value FDiagPattern
\value DiagCrossPattern
\value CustomPattern
\img brush-styles.png Brush Styles
*/
/*!
\enum Qt::RasterOp
This enum type is used to describe the way things are written to
the paint device. Each bit of the \e src (what you write)
interacts with the corresponding bit of the \e dst pixel.
\value CopyROP dst = src
\value OrROP dst = src OR dst
\value XorROP dst = src XOR dst
\value NotAndROP dst = (NOT src) AND dst
\value EraseROP an alias for \c NotAndROP
\value NotCopyROP dst = NOT src
\value NotOrROP dst = (NOT src) OR dst
\value NotXorROP dst = (NOT src) XOR dst
\value AndROP dst = src AND dst
\value NotEraseROP an alias for \c AndROP
\value NotROP dst = NOT dst
\value ClearROP dst = 0
\value SetROP dst = 1
\value NopROP dst = dst
\value AndNotROP dst = src AND (NOT dst)
\value OrNotROP dst = src OR (NOT dst)
\value NandROP dst = NOT (src AND dst)
\value NorROP dst = NOT (src OR dst)
By far the most useful ones are \c CopyROP and \c XorROP.
On Qt/Embedded, only \c CopyROP, \c XorROP, and \c NotROP are supported.
*/
/*!
\enum Qt::AlignmentFlags
This enum type is used to describe alignment. It contains
horizontal and vertical flags.
The horizontal flags are:
\value AlignAuto Aligns according to the language. Left for most,
right for Arabic and Hebrew.
\value AlignLeft Aligns with the left edge.
\value AlignRight Aligns with the right edge.
\value AlignHCenter Centers horizontally in the available space.
\value AlignJustify Justifies the text in the available space.
Does not work for everything and may be interpreted as
AlignAuto in some cases.
The vertical flags are:
\value AlignTop Aligns with the top.
\value AlignBottom Aligns with the bottom.
\value AlignVCenter Centers vertically in the available space.
You can use only one of the horizontal flags at a time. There is
one two-dimensional flag:
\value AlignCenter Centers in both dimensions.
You can use at most one horizontal and one vertical flag at a time. \c
AlignCenter counts as both horizontal and vertical.
Masks:
\value AlignHorizontal_Mask
\value AlignVertical_Mask
Conflicting combinations of flags have undefined meanings.
*/
/*!
\enum Qt::TextFlags
This enum type is used to define some modifier flags. Some of
these flags only make sense in the context of printing:
\value SingleLine Treats all whitespace as spaces and prints just
one line.
\value DontClip If it's impossible to stay within the given bounds,
it prints outside.
\value ExpandTabs Makes the U+0009 (ASCII tab) character move to
the next tab stop.
\value ShowPrefix Displays the string "\&P" as <u>P</u>
(see QButton for an example). For an ampersand, use "\&\&".
\value WordBreak Breaks lines at appropriate points, e.g. at word
boundaries.
\value BreakAnywhere Breaks lines anywhere, even within words.
\value NoAccel Same as ShowPrefix but doesn't draw the underlines.
You can use as many modifier flags as you want, except that \c
SingleLine and \c WordBreak cannot be combined.
Flags that are inappropriate for a given use (e.g. ShowPrefix to
QGridLayout::addWidget()) are generally ignored.
*/
/*!
\enum Qt::PenStyle
This enum type defines the pen styles that can be drawn using
QPainter. The styles are
\value NoPen no line at all. For example, QPainter::drawRect()
fills but does not draw any boundary line.
\value SolidLine a simple line.
\value DashLine dashes separated by a few pixels.
\value DotLine dots separated by a few pixels.
\value DashDotLine alternate dots and dashes.
\value DashDotDotLine one dash, two dots, one dash, two dots.
\value FineDotLine single pixel dots separated by single pixels.
\value MPenStyle mask of the pen styles.
\img pen-styles.png Pen Styles
*/
/*!
\enum Qt::PenCapStyle
This enum type defines the pen cap styles supported by Qt, i.e.
the line end caps that can be drawn using QPainter.
\value FlatCap a square line end that does not cover the end
point of the line.
\value SquareCap a square line end that covers the end point and
extends beyond it with half the line width.
\value RoundCap a rounded line end.
\value MPenCapStyle mask of the pen cap styles.
\img pen-cap-styles.png Pen Cap Styles
*/
/*!
\enum Qt::PenJoinStyle
This enum type defines the pen join styles supported by Qt, i.e.
which joins between two connected lines can be drawn using
QPainter.
\value MiterJoin The outer edges of the lines are extended to
meet at an angle, and this area is filled.
\value BevelJoin The triangular notch between the two lines is filled.
\value RoundJoin A circular arc between the two lines is filled.
\value MPenJoinStyle mask of the pen join styles.
\img pen-join-styles.png Pen Join Styles
*/
/*!
\enum Qt::BGMode
Background mode
\value TransparentMode
\value OpaqueMode
*/
/*!
Constructs a painter.
Notice that all painter settings (setPen, setBrush etc.) are reset
to default values when begin() is called.
\sa begin(), end()
*/
QPainter::QPainter()
{
init();
}
/*!
Constructs a painter that begins painting the paint device \a pd
immediately. Depending on the underlying graphic system the
painter will paint over children of the paintdevice if \a
unclipped is TRUE.
This constructor is convenient for short-lived painters, e.g. in a
\link QWidget::paintEvent() paint event\endlink and should be used
only once. The constructor calls begin() for you and the QPainter
destructor automatically calls end().
Here's an example using begin() and end():
\code
void MyWidget::paintEvent( QPaintEvent * )
{
QPainter p;
p.begin( this );
p.drawLine( ... ); // drawing code
p.end();
}
\endcode
The same example using this constructor:
\code
void MyWidget::paintEvent( QPaintEvent * )
{
QPainter p( this );
p.drawLine( ... ); // drawing code
}
\endcode
Since the constructor cannot provide feedback when the initialization
of the painter failed you should rather use begin() and end() to paint
on external devices, e.g. printers.
\sa begin(), end()
*/
QPainter::QPainter( const QPaintDevice *pd, bool unclipped )
{
init();
if ( begin( pd, unclipped ) )
flags |= CtorBegin;
}
/*!
Constructs a painter that begins painting the paint device \a pd
immediately, with the default arguments taken from \a
copyAttributes. The painter will paint over children of the paint
device if \a unclipped is TRUE (although this is not supported on
all platforms).
\sa begin()
*/
QPainter::QPainter( const QPaintDevice *pd,
const QWidget *copyAttributes, bool unclipped )
{
init();
if ( begin( pd, copyAttributes, unclipped ) )
flags |= CtorBegin;
}
/*!
Destroys the painter.
*/
QPainter::~QPainter()
{
if ( isActive() )
end();
else
killPStack();
if ( tabarray ) // delete tab array
delete [] tabarray;
#ifndef QT_NO_TRANSFORMATIONS
if ( wm_stack )
delete (QWMatrixStack *)wm_stack;
#endif
destroy();
}
/*!
\overload bool QPainter::begin( const QPaintDevice *pd, const QWidget *copyAttributes, bool unclipped )
This version opens the painter on a paint device \a pd and sets
the initial pen, background color and font from \a copyAttributes,
painting over the paint device's children when \a unclipped is
TRUE. This is equivalent to:
\code
QPainter p;
p.begin( pd );
p.setPen( copyAttributes->foregroundColor() );
p.setBackgroundColor( copyAttributes->backgroundColor() );
p.setFont( copyAttributes->font() );
\endcode
This begin function is convenient for double buffering. When you
draw in a pixmap instead of directly in a widget (to later bitBlt
the pixmap into the widget) you will need to set the widget's
font etc. This function does exactly that.
Example:
\code
void MyWidget::paintEvent( QPaintEvent * )
{
QPixmap pm(size());
QPainter p;
p.begin(&pm, this);
// ... potentially flickering paint operation ...
p.end();
bitBlt(this, 0, 0, &pm);
}
\endcode
\sa end()
*/
bool QPainter::begin( const QPaintDevice *pd, const QWidget *copyAttributes, bool unclipped )
{
if ( copyAttributes == 0 ) {
#if defined(QT_CHECK_NULL)
qWarning( "QPainter::begin: The widget to copy attributes from cannot "
"be null" );
#endif
return FALSE;
}
if ( begin( pd, unclipped ) ) {
setPen( copyAttributes->foregroundColor() );
setBackgroundColor( copyAttributes->backgroundColor() );
setFont( copyAttributes->font() );
return TRUE;
}
return FALSE;
}
/*!
\internal
Sets or clears a pointer flag.
*/
void QPainter::setf( uint b, bool v )
{
if ( v )
setf( b );
else
clearf( b );
}
/*!
\fn bool QPainter::isActive() const
Returns TRUE if the painter is active painting, i.e. begin() has
been called and end() has not yet been called; otherwise returns
FALSE.
\sa QPaintDevice::paintingActive()
*/
/*!
\fn QPaintDevice *QPainter::device() const
Returns the paint device on which this painter is currently
painting, or 0 if the painter is not active.
\sa QPaintDevice::paintingActive()
*/
struct QPState { // painter state
QFont font;
QPen pen;
QPoint curPt;
QBrush brush;
QColor bgc;
uchar bgm;
uchar rop;
QPoint bro;
QRect wr, vr;
#ifndef QT_NO_TRANSFORMATIONS
QWMatrix wm;
#else
int xlatex;
int xlatey;
#endif
bool vxf;
bool wxf;
QRegion rgn;
bool clip;
int ts;
int *ta;
void* wm_stack;
};
//TODO lose the worldmatrix stack
typedef QPtrStack<QPState> QPStateStack;
void QPainter::killPStack()
{
#if defined(QT_CHECK_STATE)
if ( ps_stack && !((QPStateStack *)ps_stack)->isEmpty() )
qWarning( "QPainter::killPStack: non-empty save/restore stack when "
"end() was called" );
#endif
delete (QPStateStack *)ps_stack;
ps_stack = 0;
}
/*!
Saves the current painter state (pushes the state onto a stack). A
save() must be followed by a corresponding restore(). end()
unwinds the stack.
\sa restore()
*/
void QPainter::save()
{
if ( testf(ExtDev) ) {
if ( testf(DirtyFont) )
updateFont();
if ( testf(DirtyPen) )
updatePen();
if ( testf(DirtyBrush) )
updateBrush();
pdev->cmd( QPaintDevice::PdcSave, this, 0 );
}
QPStateStack *pss = (QPStateStack *)ps_stack;
if ( pss == 0 ) {
pss = new QPtrStack<QPState>;
Q_CHECK_PTR( pss );
pss->setAutoDelete( TRUE );
ps_stack = pss;
}
QPState *ps = new QPState;
Q_CHECK_PTR( ps );
ps->font = cfont;
ps->pen = cpen;
ps->curPt = pos();
ps->brush = cbrush;
ps->bgc = bg_col;
ps->bgm = bg_mode;
ps->rop = rop;
ps->bro = bro;
#ifndef QT_NO_TRANSFORMATIONS
ps->wr = QRect( wx, wy, ww, wh );
ps->vr = QRect( vx, vy, vw, vh );
ps->wm = wxmat;
ps->vxf = testf(VxF);
ps->wxf = testf(WxF);
#else
ps->xlatex = xlatex;
ps->xlatey = xlatey;
#endif
ps->rgn = crgn;
ps->clip = testf(ClipOn);
ps->ts = tabstops;
ps->ta = tabarray;
ps->wm_stack = wm_stack;
wm_stack = 0;
pss->push( ps );
}
/*!
Restores the current painter state (pops a saved state off the
stack).
\sa save()
*/
void QPainter::restore()
{
if ( testf(ExtDev) ) {
pdev->cmd( QPaintDevice::PdcRestore, this, 0 );
if ( pdev->devType() == QInternal::Picture )
block_ext = TRUE;
}
QPStateStack *pss = (QPStateStack *)ps_stack;
if ( pss == 0 || pss->isEmpty() ) {
#if defined(QT_CHECK_STATE)
qWarning( "QPainter::restore: Empty stack error" );
#endif
return;
}
QPState *ps = pss->pop();
bool hardRestore = testf(VolatileDC);
if ( ps->font != cfont || hardRestore )
setFont( ps->font );
if ( ps->pen != cpen || hardRestore )
setPen( ps->pen );
if ( ps->brush != cbrush || hardRestore )
setBrush( ps->brush );
if ( ps->bgc != bg_col || hardRestore )
setBackgroundColor( ps->bgc );
if ( ps->bgm != bg_mode || hardRestore )
setBackgroundMode( (BGMode)ps->bgm );
if ( ps->rop != rop || hardRestore )
setRasterOp( (RasterOp)ps->rop );
if ( ps->bro != bro || hardRestore )
setBrushOrigin( ps->bro );
#ifndef QT_NO_TRANSFORMATIONS
QRect wr( wx, wy, ww, wh );
QRect vr( vx, vy, vw, vh );
if ( ps->wr != wr || hardRestore )
setWindow( ps->wr );
if ( ps->vr != vr || hardRestore )
setViewport( ps->vr );
if ( ps->wm != wxmat || hardRestore )
setWorldMatrix( ps->wm );
if ( ps->vxf != testf(VxF) || hardRestore )
setViewXForm( ps->vxf );
if ( ps->wxf != testf(WxF) || hardRestore )
setWorldXForm( ps->wxf );
#else
xlatex = ps->xlatex;
xlatey = ps->xlatey;
setf( VxF, xlatex || xlatey );
#endif
if ( ps->curPt != pos() || hardRestore )
moveTo( ps->curPt );
if ( ps->rgn != crgn || hardRestore )
setClipRegion( ps->rgn );
if ( ps->clip != testf(ClipOn) || hardRestore )
setClipping( ps->clip );
tabstops = ps->ts;
tabarray = ps->ta;
#ifndef QT_NO_TRANSFORMATIONS
if ( wm_stack )
delete (QWMatrixStack *)wm_stack;
wm_stack = ps->wm_stack;
#endif
delete ps;
block_ext = FALSE;
}
typedef QPtrDict<QPaintDevice> QPaintDeviceDict;
static QPaintDeviceDict *pdev_dict = 0;
/*!
Redirects all paint commands for a paint device, \a pdev, to
another paint device, \a replacement, unless \a replacement is 0.
If \a replacement is 0, the redirection for \a pdev is removed.
In general, you'll probably find calling QPixmap::grabWidget() or
QPixmap::grabWindow() is an easier solution.
*/
void QPainter::redirect( QPaintDevice *pdev, QPaintDevice *replacement )
{
if ( pdev_dict == 0 ) {
if ( replacement == 0 )
return;
pdev_dict = new QPaintDeviceDict;
Q_CHECK_PTR( pdev_dict );
}
#if defined(QT_CHECK_NULL)
if ( pdev == 0 )
qWarning( "QPainter::redirect: The pdev argument cannot be 0" );
#endif
if ( replacement ) {
pdev_dict->insert( pdev, replacement );
} else {
pdev_dict->remove( pdev );
if ( pdev_dict->count() == 0 ) {
delete pdev_dict;
pdev_dict = 0;
}
}
}
/*!
\internal
Returns the replacement for \a pdev, or 0 if there is no replacement.
*/
QPaintDevice *QPainter::redirect( QPaintDevice *pdev )
{
return pdev_dict ? pdev_dict->find( pdev ) : 0;
}
/*!
Returns the font metrics for the painter, if the painter is
active. It is not possible to obtain metrics for an inactive
painter, so the return value is undefined if the painter is not
active.
\sa fontInfo(), isActive()
*/
QFontMetrics QPainter::fontMetrics() const
{
if ( pdev && pdev->devType() == QInternal::Picture )
return QFontMetrics( cfont );
return QFontMetrics(this);
}
/*!
Returns the font info for the painter, if the painter is active.
It is not possible to obtain font information for an inactive
painter, so the return value is undefined if the painter is not
active.
\sa fontMetrics(), isActive()
*/
QFontInfo QPainter::fontInfo() const
{
if ( pdev && pdev->devType() == QInternal::Picture )
return QFontInfo( cfont );
return QFontInfo(this);
}
/*!
\fn const QPen &QPainter::pen() const
Returns the painter's current pen.
\sa setPen()
*/
/*!
Sets a new painter pen.
The \a pen defines how to draw lines and outlines, and it also
defines the text color.
\sa pen()
*/
void QPainter::setPen( const QPen &pen )
{
#if defined(QT_CHECK_STATE)
if ( !isActive() )
qWarning( "QPainter::setPen: Will be reset by begin()" );
#endif
if ( cpen == pen )
return;
cpen = pen;
updatePen();
}
/*!
\overload
Sets the painter's pen to have style \a style, width 0 and black
color.
\sa pen(), QPen
*/
void QPainter::setPen( PenStyle style )
{
#if defined(QT_CHECK_STATE)
if ( !isActive() )
qWarning( "QPainter::setPen: Will be reset by begin()" );
#endif
QPen::QPenData *d = cpen.data; // low level access
if ( d->style == style && d->linest == style && !d->width && d->color == Qt::black )
return;
if ( d->count != 1 ) {
cpen.detach();
d = cpen.data;
}
d->style = style;
d->width = 0;
d->color = Qt::black;
d->linest = style;
updatePen();
}
/*!
\overload
Sets the painter's pen to have style \c SolidLine, width 0 and the
specified \a color.
\sa pen(), QPen
*/
void QPainter::setPen( const QColor &color )
{
#if defined(QT_CHECK_STATE)
if ( !isActive() )
qWarning( "QPainter::setPen: Will be reset by begin()" );
#endif
QPen::QPenData *d = cpen.data; // low level access
if ( d->color == color && !d->width && d->style == SolidLine && d->linest == SolidLine )
return;
if ( d->count != 1 ) {
cpen.detach();
d = cpen.data;
}
d->style = SolidLine;
d->width = 0;
d->color = color;
d->linest = SolidLine;
updatePen();
}
/*!
\fn const QBrush &QPainter::brush() const
Returns the painter's current brush.
\sa QPainter::setBrush()
*/
/*!
\overload
Sets the painter's brush to \a brush.
The \a brush defines how shapes are filled.
\sa brush()
*/
void QPainter::setBrush( const QBrush &brush )
{
#if defined(QT_CHECK_STATE)
if ( !isActive() )
qWarning( "QPainter::setBrush: Will be reset by begin()" );
#endif
if ( cbrush == brush )
return;
cbrush = brush;
updateBrush();
}
/*!
Sets the painter's brush to black color and the specified \a
style.
\sa brush(), QBrush
*/
void QPainter::setBrush( BrushStyle style )
{
#if defined(QT_CHECK_STATE)
if ( !isActive() )
qWarning( "QPainter::setBrush: Will be reset by begin()" );
#endif
QBrush::QBrushData *d = cbrush.data; // low level access
if ( d->style == style && d->color == Qt::black && !d->pixmap )
return;
if ( d->count != 1 ) {
cbrush.detach();
d = cbrush.data;
}
d->style = style;
d->color = Qt::black;
if ( d->pixmap ) {
delete d->pixmap;
d->pixmap = 0;
}
updateBrush();
}
/*!
\overload
Sets the painter's brush to have style \c SolidPattern and the
specified \a color.
\sa brush(), QBrush
*/
void QPainter::setBrush( const QColor &color )
{
#if defined(QT_CHECK_STATE)
if ( !isActive() )
qWarning( "QPainter::setBrush: Will be reset by begin()" );
#endif
QBrush::QBrushData *d = cbrush.data; // low level access
if ( d->color == color && d->style == SolidPattern && !d->pixmap )
return;
if ( d->count != 1 ) {
cbrush.detach();
d = cbrush.data;
}
d->style = SolidPattern;
d->color = color;
if ( d->pixmap ) {
delete d->pixmap;
d->pixmap = 0;
}
updateBrush();
}
/*!
\fn const QColor &QPainter::backgroundColor() const
Returns the current background color.
\sa setBackgroundColor() QColor
*/
/*!
\fn BGMode QPainter::backgroundMode() const
Returns the current background mode.
\sa setBackgroundMode() BGMode
*/
/*!
\fn RasterOp QPainter::rasterOp() const
Returns the current \link Qt::RasterOp raster operation \endlink.
\sa setRasterOp() RasterOp
*/
/*!
\fn const QPoint &QPainter::brushOrigin() const
Returns the brush origin currently set.
\sa setBrushOrigin()
*/
/*!
\fn int QPainter::tabStops() const
Returns the tab stop setting.
\sa setTabStops()
*/
/*!
Set the tab stop width to \a ts, i.e. locates tab stops at \a ts,
2*\a ts, 3*\a ts and so on.
Tab stops are used when drawing formatted text with \c ExpandTabs
set. This fixed tab stop value is used only if no tab array is set
(which is the default case).
A value of 0 (the default) implies a tabstop setting of 8 times the width of the
character 'x' in the font currently set on the painter.
\sa tabStops(), setTabArray(), drawText(), fontMetrics()
*/
void QPainter::setTabStops( int ts )
{
#if defined(QT_CHECK_STATE)
if ( !isActive() )
qWarning( "QPainter::setTabStops: Will be reset by begin()" );
#endif
tabstops = ts;
if ( isActive() && testf(ExtDev) ) { // tell extended device
QPDevCmdParam param[1];
param[0].ival = ts;
pdev->cmd( QPaintDevice::PdcSetTabStops, this, param );
}
}
/*!
\fn int *QPainter::tabArray() const
Returns the currently set tab stop array.
\sa setTabArray()
*/
/*!
Sets the tab stop array to \a ta. This puts tab stops at \a ta[0],
\a ta[1] and so on. The array is null-terminated.
If both a tab array and a tab top size is set, the tab array wins.
\sa tabArray(), setTabStops(), drawText(), fontMetrics()
*/
void QPainter::setTabArray( int *ta )
{
#if defined(QT_CHECK_STATE)
if ( !isActive() )
qWarning( "QPainter::setTabArray: Will be reset by begin()" );
#endif
if ( ta != tabarray ) {
tabarraylen = 0;
if ( tabarray ) // Avoid purify complaint
delete [] tabarray; // delete old array
if ( ta ) { // tabarray = copy of 'ta'
while ( ta[tabarraylen] )
tabarraylen++;
tabarraylen++; // and 0 terminator
tabarray = new int[tabarraylen]; // duplicate ta
memcpy( tabarray, ta, sizeof(int)*tabarraylen );
} else {
tabarray = 0;
}
}
if ( isActive() && testf(ExtDev) ) { // tell extended device
QPDevCmdParam param[2];
param[0].ival = tabarraylen;
param[1].ivec = tabarray;
pdev->cmd( QPaintDevice::PdcSetTabArray, this, param );
}
}
/*!
\fn HANDLE QPainter::handle() const
Returns the platform-dependent handle used for drawing. Using this
function is not portable.
*/
/*****************************************************************************
QPainter xform settings
*****************************************************************************/
#ifndef QT_NO_TRANSFORMATIONS
/*!
Enables view transformations if \a enable is TRUE, or disables
view transformations if \a enable is FALSE.
\sa hasViewXForm(), setWindow(), setViewport(), setWorldMatrix(),
setWorldXForm(), xForm()
*/
void QPainter::setViewXForm( bool enable )
{
#if defined(QT_CHECK_STATE)
if ( !isActive() )
qWarning( "QPainter::setViewXForm: Will be reset by begin()" );
#endif
if ( !isActive() || enable == testf(VxF) )
return;
setf( VxF, enable );
if ( testf(ExtDev) ) {
QPDevCmdParam param[1];
param[0].ival = enable;
pdev->cmd( QPaintDevice::PdcSetVXform, this, param );
}
updateXForm();
}
/*!
\fn bool QPainter::hasViewXForm() const
Returns TRUE if view transformation is enabled; otherwise returns
FALSE.
\sa setViewXForm(), xForm()
*/
/*!
Returns the window rectangle.
\sa setWindow(), setViewXForm()
*/
QRect QPainter::window() const
{
return QRect( wx, wy, ww, wh );
}
/*!
Sets the window rectangle view transformation for the painter and
enables view transformation.
The window rectangle is part of the view transformation. The
window specifies the logical coordinate system and is specified by
the \a x, \a y, \a w width and \a h height parameters. Its sister,
the viewport(), specifies the device coordinate system.
The default window rectangle is the same as the device's
rectangle. See the \link coordsys.html Coordinate System Overview
\endlink for an overview of coordinate transformation.
\sa window(), setViewport(), setViewXForm(), setWorldMatrix(),
setWorldXForm()
*/
void QPainter::setWindow( int x, int y, int w, int h )
{
#if defined(QT_CHECK_STATE)
if ( !isActive() )
qWarning( "QPainter::setWindow: Will be reset by begin()" );
#endif
wx = x;
wy = y;
ww = w;
wh = h;
if ( testf(ExtDev) ) {
QRect r( x, y, w, h );
QPDevCmdParam param[1];
param[0].rect = (QRect*)&r;
pdev->cmd( QPaintDevice::PdcSetWindow, this, param );
}
if ( testf(VxF) )
updateXForm();
else
setViewXForm( TRUE );
}
/*!
Returns the viewport rectangle.
\sa setViewport(), setViewXForm()
*/
QRect QPainter::viewport() const // get viewport
{
return QRect( vx, vy, vw, vh );
}
/*!
Sets the viewport rectangle view transformation for the painter
and enables view transformation.
The viewport rectangle is part of the view transformation. The
viewport specifies the device coordinate system and is specified
by the \a x, \a y, \a w width and \a h height parameters. Its
sister, the window(), specifies the logical coordinate system.
The default viewport rectangle is the same as the device's
rectangle. See the \link coordsys.html Coordinate System Overview
\endlink for an overview of coordinate transformation.
\sa viewport(), setWindow(), setViewXForm(), setWorldMatrix(),
setWorldXForm(), xForm()
*/
void QPainter::setViewport( int x, int y, int w, int h )
{
#if defined(QT_CHECK_STATE)
if ( !isActive() )
qWarning( "QPainter::setViewport: Will be reset by begin()" );
#endif
vx = x;
vy = y;
vw = w;
vh = h;
if ( testf(ExtDev) ) {
QRect r( x, y, w, h );
QPDevCmdParam param[1];
param[0].rect = (QRect*)&r;
pdev->cmd( QPaintDevice::PdcSetViewport, this, param );
}
if ( testf(VxF) )
updateXForm();
else
setViewXForm( TRUE );
}
/*!
Enables world transformations if \a enable is TRUE, or disables
world transformations if \a enable is FALSE. The world
transformation matrix is not changed.
\sa setWorldMatrix(), setWindow(), setViewport(), setViewXForm(),
xForm()
*/
void QPainter::setWorldXForm( bool enable )
{
#if defined(QT_CHECK_STATE)
if ( !isActive() )
qWarning( "QPainter::setWorldXForm: Will be reset by begin()" );
#endif
if ( !isActive() || enable == testf(WxF) )
return;
setf( WxF, enable );
if ( testf(ExtDev) && !block_ext ) {
QPDevCmdParam param[1];
param[0].ival = enable;
pdev->cmd( QPaintDevice::PdcSetWXform, this, param );
}
updateXForm();
}
/*!
\fn bool QPainter::hasWorldXForm() const
Returns TRUE if world transformation is enabled; otherwise returns
FALSE.
\sa setWorldXForm()
*/
/*!
Returns the world transformation matrix.
\sa setWorldMatrix()
*/
const QWMatrix &QPainter::worldMatrix() const
{
return wxmat;
}
/*!
Sets the world transformation matrix to \a m and enables world
transformation.
If \a combine is TRUE, then \a m is combined with the current
transformation matrix, otherwise \a m replaces the current
transformation matrix.
If \a m is the identity matrix and \a combine is FALSE, this
function calls setWorldXForm(FALSE). (The identity matrix is the
matrix where QWMatrix::m11() and QWMatrix::m22() are 1.0 and the
rest are 0.0.)
World transformations are applied after the view transformations
(i.e. \link setWindow() window\endlink and \link setViewport()
viewport\endlink).
The following functions can transform the coordinate system without using
a QWMatrix:
\list
\i translate()
\i scale()
\i shear()
\i rotate()
\endlist
They operate on the painter's worldMatrix() and are implemented like this:
\code
void QPainter::rotate( double a )
{
QWMatrix m;
m.rotate( a );
setWorldMatrix( m, TRUE );
}
\endcode
Note that you should always use \a combine when you are drawing
into a QPicture. Otherwise it may not be possible to replay the
picture with additional transformations. Using translate(),
scale(), etc., is safe.
For a brief overview of coordinate transformation, see the \link
coordsys.html Coordinate System Overview. \endlink
\sa worldMatrix() setWorldXForm() setWindow() setViewport()
setViewXForm() xForm() QWMatrix
*/
void QPainter::setWorldMatrix( const QWMatrix &m, bool combine )
{
if ( !isActive() ) {
#if defined(QT_CHECK_STATE)
qWarning( "QPainter::setWorldMatrix: Will be reset by begin()" );
#endif
return;
}
if ( combine )
wxmat = m * wxmat; // combines
else
wxmat = m; // set new matrix
bool identity = wxmat.m11() == 1.0F && wxmat.m22() == 1.0F &&
wxmat.m12() == 0.0F && wxmat.m21() == 0.0F &&
wxmat.dx() == 0.0F && wxmat.dy() == 0.0F;
if ( testf(ExtDev) && !block_ext ) {
QPDevCmdParam param[2];
param[0].matrix = &m;
param[1].ival = combine;
pdev->cmd( QPaintDevice::PdcSetWMatrix, this, param );
}
if ( identity && pdev->devType() != QInternal::Picture )
setWorldXForm( FALSE );
else if ( !testf(WxF) )
setWorldXForm( TRUE );
else
updateXForm();
}
/*! \obsolete
We recommend using save() instead.
*/
void QPainter::saveWorldMatrix()
{
QWMatrixStack *stack = (QWMatrixStack *)wm_stack;
if ( stack == 0 ) {
stack = new QPtrStack<QWMatrix>;
Q_CHECK_PTR( stack );
stack->setAutoDelete( TRUE );
wm_stack = stack;
}
stack->push( new QWMatrix( wxmat ) );
}
/*! \obsolete
We recommend using restore() instead.
*/
void QPainter::restoreWorldMatrix()
{
QWMatrixStack *stack = (QWMatrixStack *)wm_stack;
if ( stack == 0 || stack->isEmpty() ) {
#if defined(QT_CHECK_STATE)
qWarning( "QPainter::restoreWorldMatrix: Empty stack error" );
#endif
return;
}
QWMatrix* m = stack->pop();
setWorldMatrix( *m );
delete m;
}
#endif // QT_NO_TRANSFORMATIONS
/*!
Translates the coordinate system by \a (dx, dy). After this call,
\a (dx, dy) is added to points.
For example, the following code draws the same point twice:
\code
void MyWidget::paintEvent()
{
QPainter paint( this );
paint.drawPoint( 0, 0 );
paint.translate( 100.0, 40.0 );
paint.drawPoint( -100, -40 );
}
\endcode
\sa scale(), shear(), rotate(), resetXForm(), setWorldMatrix(), xForm()
*/
void QPainter::translate( double dx, double dy )
{
#ifndef QT_NO_TRANSFORMATIONS
QWMatrix m;
m.translate( dx, dy );
setWorldMatrix( m, TRUE );
#else
xlatex += (int)dx;
xlatey += (int)dy;
setf( VxF, xlatex || xlatey );
#endif
}
#ifndef QT_NO_TRANSFORMATIONS
/*!
Scales the coordinate system by \a (sx, sy).
\sa translate(), shear(), rotate(), resetXForm(), setWorldMatrix(),
xForm()
*/
void QPainter::scale( double sx, double sy )
{
QWMatrix m;
m.scale( sx, sy );
setWorldMatrix( m, TRUE );
}
/*!
Shears the coordinate system by \a (sh, sv).
\sa translate(), scale(), rotate(), resetXForm(), setWorldMatrix(),
xForm()
*/
void QPainter::shear( double sh, double sv )
{
QWMatrix m;
m.shear( sv, sh );
setWorldMatrix( m, TRUE );
}
/*!
Rotates the coordinate system \a a degrees counterclockwise.
\sa translate(), scale(), shear(), resetXForm(), setWorldMatrix(),
xForm()
*/
void QPainter::rotate( double a )
{
QWMatrix m;
m.rotate( a );
setWorldMatrix( m, TRUE );
}
/*!
Resets any transformations that were made using translate(), scale(),
shear(), rotate(), setWorldMatrix(), setViewport() and
setWindow().
\sa worldMatrix(), viewport(), window()
*/
void QPainter::resetXForm()
{
if ( !isActive() )
return;
wx = wy = vx = vy = 0; // default view origins
ww = vw = pdev->metric( QPaintDeviceMetrics::PdmWidth );
wh = vh = pdev->metric( QPaintDeviceMetrics::PdmHeight );
wxmat = QWMatrix();
setWorldXForm( FALSE );
setViewXForm( FALSE );
}
/*!
\internal
Updates an internal integer transformation matrix.
*/
void QPainter::updateXForm()
{
QWMatrix m;
if ( testf(VxF) ) {
double scaleW = (double)vw/(double)ww;
double scaleH = (double)vh/(double)wh;
m.setMatrix( scaleW, 0, 0, scaleH, vx - wx*scaleW, vy - wy*scaleH );
}
if ( testf(WxF) ) {
if ( testf(VxF) )
m = wxmat * m;
else
m = wxmat;
}
xmat = m;
txinv = FALSE; // no inverted matrix
txop = TxNone;
if ( m12()==0.0 && m21()==0.0 && m11() >= 0.0 && m22() >= 0.0 ) {
if ( m11()==1.0 && m22()==1.0 ) {
if ( dx()!=0.0 || dy()!=0.0 )
txop = TxTranslate;
} else {
txop = TxScale;
#if defined(Q_WS_WIN)
setf(DirtyFont);
#endif
}
} else {
txop = TxRotShear;
#if defined(Q_WS_WIN)
setf(DirtyFont);
#endif
}
}
/*!
\internal
Updates an internal integer inverse transformation matrix.
*/
void QPainter::updateInvXForm()
{
#if defined(QT_CHECK_STATE)
Q_ASSERT( txinv == FALSE );
#endif
txinv = TRUE; // creating inverted matrix
bool invertible;
QWMatrix m;
if ( testf(VxF) ) {
m.translate( vx, vy );
m.scale( 1.0*vw/ww, 1.0*vh/wh );
m.translate( -wx, -wy );
}
if ( testf(WxF) ) {
if ( testf(VxF) )
m = wxmat * m;
else
m = wxmat;
}
ixmat = m.invert( &invertible ); // invert matrix
}
#else
void QPainter::resetXForm()
{
xlatex = 0;
xlatey = 0;
clearf( VxF );
}
#endif // QT_NO_TRANSFORMATIONS
extern bool qt_old_transformations;
/*!
\internal
Maps a point from logical coordinates to device coordinates.
*/
void QPainter::map( int x, int y, int *rx, int *ry ) const
{
#ifndef QT_NO_TRANSFORMATIONS
if ( qt_old_transformations ) {
switch ( txop ) {
case TxNone:
*rx = x; *ry = y;
break;
case TxTranslate:
// #### "Why no rounding here?", Warwick asked of Haavard.
*rx = int(x + dx());
*ry = int(y + dy());
break;
case TxScale: {
double tx = m11()*x + dx();
double ty = m22()*y + dy();
*rx = tx >= 0 ? int(tx + 0.5) : int(tx - 0.5);
*ry = ty >= 0 ? int(ty + 0.5) : int(ty - 0.5);
} break;
default: {
double tx = m11()*x + m21()*y+dx();
double ty = m12()*x + m22()*y+dy();
*rx = tx >= 0 ? int(tx + 0.5) : int(tx - 0.5);
*ry = ty >= 0 ? int(ty + 0.5) : int(ty - 0.5);
} break;
}
} else {
switch ( txop ) {
case TxNone:
*rx = x;
*ry = y;
break;
case TxTranslate:
*rx = qRound( x + dx() );
*ry = qRound( y + dy() );
break;
case TxScale:
*rx = qRound( m11()*x + dx() );
*ry = qRound( m22()*y + dy() );
break;
default:
*rx = qRound( m11()*x + m21()*y+dx() );
*ry = qRound( m12()*x + m22()*y+dy() );
break;
}
}
#else
*rx = x + xlatex;
*ry = y + xlatey;
#endif
}
/*!
\internal
Maps a rectangle from logical coordinates to device coordinates.
This internal function does not handle rotation and/or shear.
*/
void QPainter::map( int x, int y, int w, int h,
int *rx, int *ry, int *rw, int *rh ) const
{
#ifndef QT_NO_TRANSFORMATIONS
if ( qt_old_transformations ) {
switch ( txop ) {
case TxNone:
*rx = x; *ry = y;
*rw = w; *rh = h;
break;
case TxTranslate:
// #### "Why no rounding here?", Warwick asked of Haavard.
*rx = int(x + dx());
*ry = int(y + dy());
*rw = w; *rh = h;
break;
case TxScale: {
double tx1 = m11()*x + dx();
double ty1 = m22()*y + dy();
double tx2 = m11()*(x + w - 1) + dx();
double ty2 = m22()*(y + h - 1) + dy();
*rx = qRound( tx1 );
*ry = qRound( ty1 );
*rw = qRound( tx2 ) - *rx + 1;
*rh = qRound( ty2 ) - *ry + 1;
} break;
default:
#if defined(QT_CHECK_STATE)
qWarning( "QPainter::map: Internal error" );
#endif
break;
}
} else {
switch ( txop ) {
case TxNone:
*rx = x; *ry = y;
*rw = w; *rh = h;
break;
case TxTranslate:
*rx = qRound(x + dx() );
*ry = qRound(y + dy() );
*rw = w; *rh = h;
break;
case TxScale:
*rx = qRound( m11()*x + dx() );
*ry = qRound( m22()*y + dy() );
*rw = qRound( m11()*w );
*rh = qRound( m22()*h );
break;
default:
#if defined(QT_CHECK_STATE)
qWarning( "QPainter::map: Internal error" );
#endif
break;
}
}
#else
*rx = x + xlatex;
*ry = y + xlatey;
*rw = w; *rh = h;
#endif
}
/*!
\internal
Maps a point from device coordinates to logical coordinates.
*/
void QPainter::mapInv( int x, int y, int *rx, int *ry ) const
{
#ifndef QT_NO_TRANSFORMATIONS
#if defined(QT_CHECK_STATE)
if ( !txinv )
qWarning( "QPainter::mapInv: Internal error" );
#endif
if ( qt_old_transformations ) {
double tx = im11()*x + im21()*y+idx();
double ty = im12()*x + im22()*y+idy();
*rx = tx >= 0 ? int(tx + 0.5) : int(tx - 0.5);
*ry = ty >= 0 ? int(ty + 0.5) : int(ty - 0.5);
} else {
*rx = qRound( im11()*x + im21()*y + idx() );
*ry = qRound( im12()*x + im22()*y + idy() );
}
#else
*rx = x - xlatex;
*ry = y - xlatey;
#endif
}
/*!
\internal
Maps a rectangle from device coordinates to logical coordinates.
Cannot handle rotation and/or shear.
*/
void QPainter::mapInv( int x, int y, int w, int h,
int *rx, int *ry, int *rw, int *rh ) const
{
#ifndef QT_NO_TRANSFORMATIONS
#if defined(QT_CHECK_STATE)
if ( !txinv || txop == TxRotShear )
qWarning( "QPainter::mapInv: Internal error" );
#endif
if ( qt_old_transformations ) {
double tx = im11()*x + idx();
double ty = im22()*y + idy();
double tw = im11()*w;
double th = im22()*h;
*rx = tx >= 0 ? int(tx + 0.5) : int(tx - 0.5);
*ry = ty >= 0 ? int(ty + 0.5) : int(ty - 0.5);
*rw = tw >= 0 ? int(tw + 0.5) : int(tw - 0.5);
*rh = th >= 0 ? int(th + 0.5) : int(th - 0.5);
} else {
*rx = qRound( im11()*x + idx() );
*ry = qRound( im22()*y + idy() );
*rw = qRound( im11()*w );
*rh = qRound( im22()*h );
}
#else
*rx = x - xlatex;
*ry = y - xlatey;
*rw = w;
*rh = h;
#endif
}
/*!
Returns the point \a pv transformed from model coordinates to
device coordinates.
\sa xFormDev(), QWMatrix::map()
*/
QPoint QPainter::xForm( const QPoint &pv ) const
{
#ifndef QT_NO_TRANSFORMATIONS
if ( txop == TxNone )
return pv;
int x=pv.x(), y=pv.y();
map( x, y, &x, &y );
return QPoint( x, y );
#else
return QPoint( pv.x()+xlatex, pv.y()+xlatey );
#endif
}
/*!
\overload
Returns the rectangle \a rv transformed from model coordinates to
device coordinates.
If world transformation is enabled and rotation or shearing has
been specified, then the bounding rectangle is returned.
\sa xFormDev(), QWMatrix::map()
*/
QRect QPainter::xForm( const QRect &rv ) const
{
#ifndef QT_NO_TRANSFORMATIONS
if ( txop == TxNone )
return rv;
if ( txop == TxRotShear ) { // rotation/shear
return xmat.mapRect( rv );
}
// Just translation/scale
int x, y, w, h;
rv.rect( &x, &y, &w, &h );
map( x, y, w, h, &x, &y, &w, &h );
return QRect( x, y, w, h );
#else
return QRect( rv.x()+xlatex, rv.y()+xlatey, rv.width(), rv.height() );
#endif
}
/*!
\overload
Returns the point array \a av transformed from model coordinates
to device coordinates.
\sa xFormDev(), QWMatrix::map()
*/
QPointArray QPainter::xForm( const QPointArray &av ) const
{
QPointArray a = av;
#ifndef QT_NO_TRANSFORMATIONS
if ( txop != TxNone )
{
return xmat * av;
}
#else
a.translate( xlatex, xlatey );
#endif
return a;
}
/*!
\overload
Returns the point array \a av transformed from model coordinates
to device coordinates. The \a index is the first point in the
array and \a npoints denotes the number of points to be
transformed. If \a npoints is negative, all points from \a
av[index] until the last point in the array are transformed.
The returned point array consists of the number of points that
were transformed.
Example:
\code
QPointArray a(10);
QPointArray b;
b = painter.xForm(a, 2, 4); // b.size() == 4
b = painter.xForm(a, 2, -1); // b.size() == 8
\endcode
\sa xFormDev(), QWMatrix::map()
*/
QPointArray QPainter::xForm( const QPointArray &av, int index,
int npoints ) const
{
int lastPoint = npoints < 0 ? av.size() : index+npoints;
QPointArray a( lastPoint-index );
memcpy( a.data(), av.data()+index, (lastPoint-index)*sizeof( QPoint ) );
#ifndef QT_NO_TRANSFORMATIONS
return xmat*a;
#else
a.translate( xlatex, xlatey );
return a;
#endif
}
/*!
\overload
Returns the point \a pd transformed from device coordinates to
model coordinates.
\sa xForm(), QWMatrix::map()
*/
QPoint QPainter::xFormDev( const QPoint &pd ) const
{
#ifndef QT_NO_TRANSFORMATIONS
if ( txop == TxNone )
return pd;
if ( !txinv ) {
QPainter *that = (QPainter*)this; // mutable
that->updateInvXForm();
}
#endif
int x=pd.x(), y=pd.y();
mapInv( x, y, &x, &y );
return QPoint( x, y );
}
/*!
Returns the rectangle \a rd transformed from device coordinates to
model coordinates.
If world transformation is enabled and rotation or shearing is
used, then the bounding rectangle is returned.
\sa xForm(), QWMatrix::map()
*/
QRect QPainter::xFormDev( const QRect &rd ) const
{
#ifndef QT_NO_TRANSFORMATIONS
if ( txop == TxNone )
return rd;
if ( !txinv ) {
QPainter *that = (QPainter*)this; // mutable
that->updateInvXForm();
}
if ( txop == TxRotShear ) { // rotation/shear
return ixmat.mapRect( rd );
}
#endif
// Just translation/scale
int x, y, w, h;
rd.rect( &x, &y, &w, &h );
mapInv( x, y, w, h, &x, &y, &w, &h );
return QRect( x, y, w, h );
}
/*!
\overload
Returns the point array \a ad transformed from device coordinates
to model coordinates.
\sa xForm(), QWMatrix::map()
*/
QPointArray QPainter::xFormDev( const QPointArray &ad ) const
{
#ifndef QT_NO_TRANSFORMATIONS
if ( txop == TxNone )
return ad;
if ( !txinv ) {
QPainter *that = (QPainter*)this; // mutable
that->updateInvXForm();
}
return ixmat * ad;
#else
// ###
return ad;
#endif
}
/*!
\overload
Returns the point array \a ad transformed from device coordinates
to model coordinates. The \a index is the first point in the array
and \a npoints denotes the number of points to be transformed. If
\a npoints is negative, all points from \a ad[index] until the
last point in the array are transformed.
The returned point array consists of the number of points that
were transformed.
Example:
\code
QPointArray a(10);
QPointArray b;
b = painter.xFormDev(a, 1, 3); // b.size() == 3
b = painter.xFormDev(a, 1, -1); // b.size() == 9
\endcode
\sa xForm(), QWMatrix::map()
*/
QPointArray QPainter::xFormDev( const QPointArray &ad, int index,
int npoints ) const
{
int lastPoint = npoints < 0 ? ad.size() : index+npoints;
QPointArray a( lastPoint-index );
memcpy( a.data(), ad.data()+index, (lastPoint-index)*sizeof( QPoint ) );
#ifndef QT_NO_TRANSFORMATIONS
if ( txop == TxNone )
return a;
if ( !txinv ) {
QPainter *that = (QPainter*)this; // mutable
that->updateInvXForm();
}
return ixmat * a;
#else
// ###
return a;
#endif
}
/*!
Fills the rectangle \a (x, y, w, h) with the \a brush.
You can specify a QColor as \a brush, since there is a QBrush
constructor that takes a QColor argument and creates a solid
pattern brush.
\sa drawRect()
*/
void QPainter::fillRect( int x, int y, int w, int h, const QBrush &brush )
{
QPen oldPen = pen(); // save pen
QBrush oldBrush = this->brush(); // save brush
setPen( NoPen );
setBrush( brush );
drawRect( x, y, w, h ); // draw filled rect
setBrush( oldBrush ); // restore brush
setPen( oldPen ); // restore pen
}
/*!
\overload void QPainter::setBrushOrigin( const QPoint &p )
Sets the brush origin to point \a p.
*/
/*!
\overload void QPainter::setWindow( const QRect &r )
Sets the painter's window to rectangle \a r.
*/
/*!
\overload void QPainter::setViewport( const QRect &r )
Sets the painter's viewport to rectangle \a r.
*/
/*!
\fn bool QPainter::hasClipping() const
Returns TRUE if clipping has been set; otherwise returns FALSE.
\sa setClipping()
*/
/*!
Returns the currently set clip region. Note that the clip region
is given in physical device coordinates and \e not subject to any
\link coordsys.html coordinate transformation \endlink if \a m is
equal to \c CoordDevice (the default). If \a m equals \c
CoordPainter the returned region is in model coordinates.
\sa setClipRegion(), setClipRect(), setClipping() QPainter::CoordinateMode
*/
QRegion QPainter::clipRegion( CoordinateMode m ) const
{
// ### FIXME in 4.0:
// If the transformation mode is CoordPainter, we should transform the
// clip region with painter transformations.
#ifndef QT_NO_TRANSFORMATIONS
QRegion r;
if ( m == CoordDevice ) {
r = crgn;
} else {
if ( !txinv ) {
QPainter *that = (QPainter*)this; // mutable
that->updateInvXForm();
}
r = ixmat * crgn;
}
return r;
#else
return crgn;
#endif
}
/*!
\fn void QPainter::setClipRect( int x, int y, int w, int h, CoordinateMode m)
Sets the clip region to the rectangle \a x, \a y, \a w, \a h and
enables clipping. The clip mode is set to \a m.
If \a m is \c CoordDevice (the default), the coordinates given for
the clip region are taken to be physical device coordinates and
are \e not subject to any \link coordsys.html coordinate
transformations\endlink. If \a m is \c CoordPainter, the
coordinates given for the clip region are taken to be model
coordinates.
\sa setClipRegion(), clipRegion(), setClipping() QPainter::CoordinateMode
*/
/*!
\overload void QPainter::drawPoint( const QPoint &p )
Draws the point \a p.
*/
/*!
\overload void QPainter::moveTo( const QPoint &p )
Moves to the point \a p.
*/
/*!
\overload void QPainter::lineTo( const QPoint &p )
Draws a line to the point \a p.
*/
/*!
\overload void QPainter::drawLine( const QPoint &p1, const QPoint &p2 )
Draws a line from point \a p1 to point \a p2.
*/
/*!
\overload void QPainter::drawRect( const QRect &r )
Draws the rectangle \a r.
*/
/*!
\overload void QPainter::drawWinFocusRect( const QRect &r )
Draws rectangle \a r as a window focus rectangle.
*/
/*!
\overload void QPainter::drawWinFocusRect( const QRect &r, const QColor &bgColor )
Draws rectangle \a r as a window focus rectangle using background
color \a bgColor.
*/
#if !defined(Q_WS_X11) && !defined(Q_WS_QWS) && !defined(Q_WS_MAC)
// The doc and X implementation of this functions is in qpainter_x11.cpp
void QPainter::drawWinFocusRect( int, int, int, int,
bool, const QColor & )
{
// do nothing, only called from X11 specific functions
}
#endif
/*!
\overload void QPainter::drawRoundRect( const QRect &r, int xRnd, int yRnd )
Draws a rounded rectangle \a r, rounding to the x position \a xRnd
and the y position \a yRnd on each corner.
*/
/*!
\overload void QPainter::drawEllipse( const QRect &r )
Draws the ellipse that fits inside rectangle \a r.
*/
/*!
\overload void QPainter::drawArc( const QRect &r, int a, int alen )
Draws the arc that fits inside the rectangle \a r with start angle
\a a and arc length \a alen.
*/
/*!
\overload void QPainter::drawPie( const QRect &r, int a, int alen )
Draws a pie segment that fits inside the rectangle \a r with start
angle \a a and arc length \a alen.
*/
/*!
\overload void QPainter::drawChord( const QRect &r, int a, int alen )
Draws a chord that fits inside the rectangle \a r with start angle
\a a and arc length \a alen.
*/
/*!
\overload void QPainter::drawPixmap( const QPoint &p, const QPixmap &pm, const QRect &sr )
Draws the rectangle \a sr of pixmap \a pm with its origin at point
\a p.
*/
/*!
\overload void QPainter::drawPixmap( const QPoint &p, const QPixmap &pm )
Draws the pixmap \a pm with its origin at point \a p.
*/
void QPainter::drawPixmap( const QPoint &p, const QPixmap &pm )
{
drawPixmap( p.x(), p.y(), pm, 0, 0, pm.width(), pm.height() );
}
#if !defined(QT_NO_IMAGE_SMOOTHSCALE) || !defined(QT_NO_PIXMAP_TRANSFORMATION)
/*!
\overload
Draws the pixmap \a pm into the rectangle \a r. The pixmap is
scaled to fit the rectangle, if image and rectangle size disagree.
*/
void QPainter::drawPixmap( const QRect &r, const QPixmap &pm )
{
int rw = r.width();
int rh = r.height();
int iw= pm.width();
int ih = pm.height();
if ( rw <= 0 || rh <= 0 || iw <= 0 || ih <= 0 )
return;
bool scale = ( rw != iw || rh != ih );
float scaleX = (float)rw/(float)iw;
float scaleY = (float)rh/(float)ih;
bool smooth = ( scaleX < 1.5 || scaleY < 1.5 );
if ( testf(ExtDev) ) {
QPDevCmdParam param[2];
param[0].rect = &r;
param[1].pixmap = &pm;
#if defined(Q_WS_WIN)
if ( !pdev->cmd( QPaintDevice::PdcDrawPixmap, this, param ) || !hdc )
return;
#elif defined(Q_WS_QWS)
pdev->cmd( QPaintDevice::PdcDrawPixmap, this, param );
return;
#elif defined(Q_WS_MAC)
if ( !pdev->cmd( QPaintDevice::PdcDrawPixmap, this, param ) || !pdev->handle())
return;
#else
if ( !pdev->cmd( QPaintDevice::PdcDrawPixmap, this, param ) || !hd )
return;
#endif
}
QPixmap pixmap = pm;
if ( scale ) {
#ifndef QT_NO_IMAGE_SMOOTHSCALE
# ifndef QT_NO_PIXMAP_TRANSFORMATION
if ( smooth )
# endif
{
QImage i = pm.convertToImage();
pixmap = QPixmap( i.smoothScale( rw, rh ) );
}
# ifndef QT_NO_PIXMAP_TRANSFORMATION
else
# endif
#endif
#ifndef QT_NO_PIXMAP_TRANSFORMATION
{
pixmap = pm.xForm( QWMatrix( scaleX, 0, 0, scaleY, 0, 0 ) );
}
#endif
}
drawPixmap( r.x(), r.y(), pixmap );
}
#endif
/*!
\overload void QPainter::drawImage( const QPoint &, const QImage &, const QRect &sr, int conversionFlags = 0 );
Draws the rectangle \a sr from the image at the given point.
*/
/*
Draws at point \a p the \sr rect from image \a pm, using \a
conversionFlags if the image needs to be converted to a pixmap.
The default value for \a conversionFlags is 0; see
convertFromImage() for information about what other values do.
This function may convert \a image to a pixmap and then draw it, if
device() is a QPixmap or a QWidget, or else draw it directly, if
device() is a QPrinter or QPicture.
*/
/*!
Draws at (\a x, \a y) the \a sw by \a sh area of pixels from (\a
sx, \a sy) in \a image, using \a conversionFlags if the image
needs to be converted to a pixmap. The default value for \a
conversionFlags is 0; see convertFromImage() for information about
what other values do.
This function may convert \a image to a pixmap and then draw it,
if device() is a QPixmap or a QWidget, or else draw it directly,
if device() is a QPrinter or QPicture.
Currently alpha masks of the image are ignored when painting on a QPrinter.
\sa drawPixmap() QPixmap::convertFromImage()
*/
void QPainter::drawImage( int x, int y, const QImage & image,
int sx, int sy, int sw, int sh,
int conversionFlags )
{
#ifdef Q_WS_QWS
//### Hackish
# ifndef QT_NO_TRANSFORMATIONS
if ( !image.isNull() && gfx &&
(txop==TxNone||txop==TxTranslate) && !testf(ExtDev) )
# else
if ( !image.isNull() && gfx && !testf(ExtDev) )
# endif
{
if(sw<0)
sw=image.width();
if(sh<0)
sh=image.height();
QImage image2 = qt_screen->mapToDevice( image );
// This is a bit dubious
if(image2.depth()==1) {
image2.setNumColors( 2 );
image2.setColor( 0, qRgb(255,255,255) );
image2.setColor( 1, qRgb(0,0,0) );
}
if ( image2.hasAlphaBuffer() )
gfx->setAlphaType(QGfx::InlineAlpha);
else
gfx->setAlphaType(QGfx::IgnoreAlpha);
gfx->setSource(&image2);
if ( testf(VxF|WxF) ) {
map( x, y, &x, &y );
}
gfx->blt(x,y,sw,sh,sx,sy);
return;
}
#endif
if ( !isActive() || image.isNull() )
return;
// right/bottom
if ( sw < 0 )
sw = image.width() - sx;
if ( sh < 0 )
sh = image.height() - sy;
// Sanity-check clipping
if ( sx < 0 ) {
x -= sx;
sw += sx;
sx = 0;
}
if ( sw + sx > image.width() )
sw = image.width() - sx;
if ( sy < 0 ) {
y -= sy;
sh += sy;
sy = 0;
}
if ( sh + sy > image.height() )
sh = image.height() - sy;
if ( sw <= 0 || sh <= 0 )
return;
bool all = image.rect().intersect(QRect(sx,sy,sw,sh)) == image.rect();
QImage subimage = all ? image : image.copy(sx,sy,sw,sh);
if ( testf(ExtDev) ) {
QPDevCmdParam param[2];
QRect r( x, y, subimage.width(), subimage.height() );
param[0].rect = &r;
param[1].image = &subimage;
#if defined(Q_WS_WIN)
if ( !pdev->cmd( QPaintDevice::PdcDrawImage, this, param ) || !hdc )
return;
#elif defined (Q_WS_QWS)
pdev->cmd( QPaintDevice::PdcDrawImage, this, param );
return;
#elif defined(Q_WS_MAC)
if(!pdev->cmd( QPaintDevice::PdcDrawImage, this, param ) || !pdev->handle() )
return;
#else
if ( !pdev->cmd( QPaintDevice::PdcDrawImage, this, param ) || !hd )
return;
#endif
}
QPixmap pm;
pm.convertFromImage( subimage, conversionFlags );
drawPixmap( x, y, pm );
}
/*!
\overload void QPainter::drawImage( const QPoint &p, const QImage &i, int conversion_flags )
Draws the image \a i at point \a p.
If the image needs to be modified to fit in a lower-resolution
result (e.g. converting from 32-bit to 8-bit), use the \a
conversion_flags to specify how you'd prefer this to happen.
\sa Qt::ImageConversionFlags
*/
void QPainter::drawImage( const QPoint & p, const QImage & i,
int conversion_flags )
{
drawImage(p, i, i.rect(), conversion_flags);
}
#if !defined(QT_NO_IMAGE_TRANSFORMATION) || !defined(QT_NO_IMAGE_SMOOTHSCALE)
/*!
\overload
Draws the image \a i into the rectangle \a r. The image will be
scaled to fit the rectangle if image and rectangle dimensions
differ.
*/
void QPainter::drawImage( const QRect &r, const QImage &i )
{
int rw = r.width();
int rh = r.height();
int iw= i.width();
int ih = i.height();
if ( rw <= 0 || rh <= 0 || iw <= 0 || ih <= 0 )
return;
if ( testf(ExtDev) ) {
QPDevCmdParam param[2];
param[0].rect = &r;
param[1].image = &i;
#if defined(Q_WS_WIN)
if ( !pdev->cmd( QPaintDevice::PdcDrawImage, this, param ) || !hdc )
return;
#elif defined(Q_WS_QWS)
pdev->cmd( QPaintDevice::PdcDrawImage, this, param );
return;
#elif defined(Q_WS_MAC)
if ( !pdev->cmd( QPaintDevice::PdcDrawImage, this, param ) || !pdev->handle() )
return;
#else
if ( !pdev->cmd( QPaintDevice::PdcDrawImage, this, param ) || !hd )
return;
#endif
}
bool scale = ( rw != iw || rh != ih );
float scaleX = (float)rw/(float)iw;
float scaleY = (float)rh/(float)ih;
bool smooth = ( scaleX < 1.5 || scaleY < 1.5 );
QImage img = scale
? (
#if defined(QT_NO_IMAGE_TRANSFORMATION)
i.smoothScale( rw, rh )
#elif defined(QT_NO_IMAGE_SMOOTHSCALE)
i.scale( rw, rh )
#else
smooth ? i.smoothScale( rw, rh ) : i.scale( rw, rh )
#endif
)
: i;
drawImage( r.x(), r.y(), img );
}
#endif
void bitBlt( QPaintDevice *dst, int dx, int dy,
const QImage *src, int sx, int sy, int sw, int sh,
int conversion_flags )
{
QPixmap tmp;
if ( sx == 0 && sy == 0
&& (sw<0 || sw==src->width()) && (sh<0 || sh==src->height()) )
{
tmp.convertFromImage( *src, conversion_flags );
} else {
tmp.convertFromImage( src->copy( sx, sy, sw, sh, conversion_flags),
conversion_flags );
}
bitBlt( dst, dx, dy, &tmp );
}
/*!
\overload void QPainter::drawTiledPixmap( const QRect &r, const QPixmap &pm, const QPoint &sp )
Draws a tiled pixmap, \a pm, inside rectangle \a r with its origin
at point \a sp.
*/
/*!
\overload void QPainter::drawTiledPixmap( const QRect &r, const QPixmap &pm )
Draws a tiled pixmap, \a pm, inside rectangle \a r.
*/
/*!
\overload void QPainter::fillRect( const QRect &r, const QBrush &brush )
Fills the rectangle \a r using brush \a brush.
*/
/*!
\fn void QPainter::eraseRect( int x, int y, int w, int h )
Erases the area inside \a x, \a y, \a w, \a h. Equivalent to
\c{fillRect( x, y, w, h, backgroundColor() )}.
*/
/*!
\overload void QPainter::eraseRect( const QRect &r )
Erases the area inside the rectangle \a r.
*/
/*!
\fn QPainter::drawText( int x, int y, const QString &, int len = -1, TextDirection dir = Auto )
\overload
Draws the given text at position \a x, \a y. If \a len is -1 (the
default) all the text is drawn, otherwise the first \a len
characters are drawn. The text's direction is given by \a dir.
\sa QPainter::TextDirection
*/
/*!
\fn void QPainter::drawText( int x, int y, int w, int h, int flags,
const QString&, int len = -1, QRect *br=0,
QTextParag **internal=0 )
\overload
Draws the given text within the rectangle starting at \a x, \a y,
with width \a w and height \a h. If \a len is -1 (the default) all
the text is drawn, otherwise the first \a len characters are
drawn. The text's flags that are given in the \a flags parameter
are \l{Qt::AlignmentFlags} and \l{Qt::TextFlags} OR'd together. \a
br (if not null) is set to the actual bounding rectangle of the
output. The \a internal parameter is for internal use only.
*/
/*!
\fn void QPainter::drawText( const QPoint &, const QString &, int len = -1, TextDirection dir = Auto );
\overload
Draws the text at the given point.
\sa QPainter::TextDirection
*/
/*
Draws the text in \a s at point \a p. If \a len is -1 the entire
string is drawn, otherwise just the first \a len characters. The
text's direction is specified by \a dir.
*/
/*!
\fn void QPainter::drawText( int x, int y, const QString &, int pos, int len, TextDirection dir = Auto );
\overload
Draws the text from position \a pos, at point \a (x, y). If \a len is
-1 the entire string is drawn, otherwise just the first \a len
characters. The text's direction is specified by \a dir.
*/
/*!
\fn void QPainter::drawText( const QPoint &p, const QString &, int pos, int len, TextDirection dir = Auto );
Draws the text from position \a pos, at point \a p. If \a len is
-1 the entire string is drawn, otherwise just the first \a len
characters. The text's direction is specified by \a dir.
Note that the meaning of \e y is not the same for the two
drawText() varieties. For overloads that take a simple \e x, \e y
pair (or a point), the \e y value is the text's baseline; for
overloads that take a rectangle, \e rect.y() is the top of the
rectangle and the text is aligned within that rectangle in
accordance with the alignment flags.
\sa QPainter::TextDirection
*/
/*!
\fn void QPainter::drawTextItem(const QPoint &, const QTextItem &, int)
\internal
*/
static inline void fix_neg_rect( int *x, int *y, int *w, int *h )
{
if ( *w < 0 ) {
*w = -*w + 2;
*x -= *w - 1;
}
if ( *h < 0 ) {
*h = -*h + 2;
*y -= *h - 1;
}
}
void QPainter::fix_neg_rect( int *x, int *y, int *w, int *h )
{
::fix_neg_rect(x,y,w,h);
}
//
// The drawText function takes two special parameters; 'internal' and 'brect'.
//
// The 'internal' parameter contains a pointer to an array of encoded
// information that keeps internal geometry data.
// If the drawText function is called repeatedly to display the same text,
// it makes sense to calculate text width and linebreaks the first time,
// and use these parameters later to print the text because we save a lot of
// CPU time.
// The 'internal' parameter will not be used if it is a null pointer.
// The 'internal' parameter will be generated if it is not null, but points
// to a null pointer, i.e. internal != 0 && *internal == 0.
// The 'internal' parameter will be used if it contains a non-null pointer.
//
// If the 'brect parameter is a non-null pointer, then the bounding rectangle
// of the text will be returned in 'brect'.
//
/*!
\overload
Draws at most \a len characters from \a str in the rectangle \a r.
This function draws formatted text. The \a tf text format is
really of type \l Qt::AlignmentFlags and \l Qt::TextFlags OR'd
together.
Horizontal alignment defaults to AlignAuto and vertical alignment
defaults to AlignTop.
\a brect (if not null) is set to the actual bounding rectangle of
the output. \a internal is, yes, internal.
\sa boundingRect()
*/
void QPainter::drawText( const QRect &r, int tf,
const QString& str, int len, QRect *brect,
QTextParag **internal )
{
if ( !isActive() )
return;
if ( len < 0 )
len = str.length();
if ( len == 0 ) // empty string
return;
if ( testf(DirtyFont|ExtDev) ) {
if ( testf(DirtyFont) )
updateFont();
if ( testf(ExtDev) && (tf & DontPrint) == 0 ) {
QPDevCmdParam param[3];
QString newstr = str;
newstr.truncate( len );
param[0].rect = &r;
param[1].ival = tf;
param[2].str = &newstr;
if ( pdev->devType() != QInternal::Printer ) {
#if defined(Q_WS_WIN)
if ( !pdev->cmd( QPaintDevice::PdcDrawText2Formatted,
this, param) ||
!hdc )
return; // QPrinter wants PdcDrawText2
#elif defined(Q_WS_QWS)
pdev->cmd( QPaintDevice::PdcDrawText2Formatted, this, param);
return;
#elif defined(Q_WS_MAC)
if ( !pdev->cmd( QPaintDevice::PdcDrawText2Formatted, this, param) ||
!pdev->handle())
return; // QPrinter wants PdcDrawText2
#else
if ( !pdev->cmd( QPaintDevice::PdcDrawText2Formatted,
this, param) ||
!hd )
return; // QPrinter wants PdcDrawText2
#endif
}
}
}
qt_format_text(font(), r, tf, str, len, brect,
tabstops, tabarray, tabarraylen, internal, this);
}
//#define QT_FORMAT_TEXT_DEBUG
#define QChar_linesep QChar(0x2028U)
void qt_format_text( const QFont& font, const QRect &_r,
int tf, const QString& str, int len, QRect *brect,
int tabstops, int* tabarray, int tabarraylen,
QTextParag **, QPainter* painter )
{
// we need to copy r here to protect against the case (&r == brect).
QRect r( _r );
bool dontclip = (tf & Qt::DontClip) == Qt::DontClip;
bool wordbreak = (tf & Qt::WordBreak) == Qt::WordBreak;
bool singleline = (tf & Qt::SingleLine) == Qt::SingleLine;
bool showprefix = (tf & Qt::ShowPrefix) == Qt::ShowPrefix;
bool noaccel = ( tf & Qt::NoAccel ) == Qt::NoAccel;
bool isRightToLeft = str.isRightToLeft();
if ( ( tf & Qt::AlignHorizontal_Mask ) == Qt::AlignAuto )
tf |= isRightToLeft ? Qt::AlignRight : Qt::AlignLeft;
bool expandtabs = ( (tf & Qt::ExpandTabs) &&
( ( (tf & Qt::AlignLeft) && !isRightToLeft ) ||
( (tf & Qt::AlignRight) && isRightToLeft ) ) );
if ( !painter )
tf |= Qt::DontPrint;
int maxUnderlines = 0;
int numUnderlines = 0;
int underlinePositionStack[32];
int *underlinePositions = underlinePositionStack;
QFont fnt(painter ? (painter->pfont ? *painter->pfont : painter->cfont) : font);
QFontMetrics fm( fnt );
QString text = str;
// str.setLength() always does a deep copy, so the replacement
// code below is safe.
text.setLength( len );
// compatible behaviour to the old implementation. Replace
// tabs by spaces
QChar *chr = (QChar*)text.unicode();
const QChar *end = chr + len;
bool haveLineSep = FALSE;
while ( chr != end ) {
if ( *chr == '\r' || ( singleline && *chr == '\n' ) ) {
*chr = ' ';
} else if ( *chr == '\n' ) {
*chr = QChar_linesep;
haveLineSep = TRUE;
} else if ( *chr == '&' ) {
++maxUnderlines;
}
++chr;
}
if ( !expandtabs ) {
chr = (QChar*)text.unicode();
while ( chr != end ) {
if ( *chr == '\t' )
*chr = ' ';
++chr;
}
} else if (!tabarraylen && !tabstops) {
tabstops = fm.width('x')*8;
}
if ( noaccel || showprefix ) {
if ( maxUnderlines > 32 )
underlinePositions = new int[maxUnderlines];
QChar *cout = (QChar*)text.unicode();
QChar *cin = cout;
int l = len;
while ( l ) {
if ( *cin == '&' ) {
++cin;
--l;
if ( !l )
break;
if ( *cin != '&' )
underlinePositions[numUnderlines++] = cout - text.unicode();
}
*cout = *cin;
++cout;
++cin;
--l;
}
uint newlen = cout - text.unicode();
if ( newlen != text.length())
text.setLength( newlen );
}
// no need to do extra work for underlines if we don't paint
if ( tf & Qt::DontPrint )
numUnderlines = 0;
int height = 0;
int left = r.width();
int right = 0;
QTextLayout textLayout( text, fnt );
int rb = QMAX( 0, -fm.minRightBearing() );
int lb = QMAX( 0, -fm.minLeftBearing() );
if ( text.isEmpty() ) {
height = fm.height();
left = right = 0;
tf |= QPainter::DontPrint;
} else {
textLayout.beginLayout((haveLineSep || expandtabs || wordbreak) ?
QTextLayout::MultiLine :
(tf & Qt::DontPrint) ? QTextLayout::NoBidi : QTextLayout::SingleLine );
// break underline chars into items of their own
for( int i = 0; i < numUnderlines; i++ ) {
textLayout.setBoundary( underlinePositions[i] );
textLayout.setBoundary( underlinePositions[i]+1 );
}
int lineWidth = wordbreak ? QMAX(0, r.width()-rb-lb) : INT_MAX;
if(!wordbreak)
tf |= Qt::IncludeTrailingSpaces;
int leading = fm.leading();
int asc = fm.ascent();
int desc = fm.descent();
height = -leading;
//qDebug("\n\nbeginLayout: lw = %d, rectwidth=%d", lineWidth , r.width());
while ( !textLayout.atEnd() ) {
height += leading;
textLayout.beginLine( lineWidth == INT_MAX ? lineWidth : lineWidth );
//qDebug("-----beginLine( %d )-----", lineWidth );
bool linesep = FALSE;
while ( 1 ) {
QTextItem ti = textLayout.currentItem();
//qDebug("item: from=%d, ch=%x", ti.from(), text.unicode()[ti.from()].unicode() );
if ( expandtabs && ti.isTab() ) {
int tw = 0;
int x = textLayout.widthUsed();
if ( tabarraylen ) {
// qDebug("tabarraylen=%d", tabarraylen );
int tab = 0;
while ( tab < tabarraylen ) {
if ( tabarray[tab] > x ) {
tw = tabarray[tab] - x;
break;
}
++tab;
}
} else {
tw = tabstops - (x % tabstops);
}
//qDebug("tw = %d", tw );
if ( tw )
ti.setWidth( tw );
}
if ( ti.isObject() && text.unicode()[ti.from()] == QChar_linesep )
linesep = TRUE;
if ( linesep || textLayout.addCurrentItem() != QTextLayout::Ok || textLayout.atEnd() )
break;
}
int ascent = asc, descent = desc, lineLeft, lineRight;
textLayout.setLineWidth( r.width()-rb-lb );
textLayout.endLine( 0, height, tf, &ascent, &descent,
&lineLeft, &lineRight );
//qDebug("finalizing line: lw=%d ascent = %d, descent=%d lineleft=%d lineright=%d", lineWidth, ascent, descent,lineLeft, lineRight );
left = QMIN( left, lineLeft );
right = QMAX( right, lineRight );
height += ascent + descent + 1;
if ( linesep )
textLayout.nextItem();
}
}
int yoff = 0;
if ( tf & Qt::AlignBottom )
yoff = r.height() - height;
else if ( tf & Qt::AlignVCenter )
yoff = (r.height() - height)/2;
if ( brect ) {
*brect = QRect( r.x() + left, r.y() + yoff, right-left + lb+rb, height );
//qDebug("br = %d %d %d/%d, left=%d, right=%d", brect->x(), brect->y(), brect->width(), brect->height(), left, right);
}
if (!(tf & QPainter::DontPrint)) {
bool restoreClipping = FALSE;
bool painterHasClip = FALSE;
QRegion painterClipRegion;
if ( !dontclip ) {
#ifndef QT_NO_TRANSFORMATIONS
QRegion reg = painter->xmat * r;
#else
QRegion reg = r;
reg.translate( painter->xlatex, painter->xlatey );
#endif
if ( painter->hasClipping() )
reg &= painter->clipRegion();
painterHasClip = painter->hasClipping();
painterClipRegion = painter->clipRegion();
restoreClipping = TRUE;
painter->setClipRegion( reg );
} else {
if ( painter->hasClipping() ){
painterHasClip = painter->hasClipping();
painterClipRegion = painter->clipRegion();
restoreClipping = TRUE;
painter->setClipping( FALSE );
}
}
int cUlChar = 0;
int _tf = 0;
if (fnt.underline()) _tf |= Qt::Underline;
if (fnt.overline()) _tf |= Qt::Overline;
if (fnt.strikeOut()) _tf |= Qt::StrikeOut;
//qDebug("have %d items",textLayout.numItems());
for ( int i = 0; i < textLayout.numItems(); i++ ) {
QTextItem ti = textLayout.itemAt( i );
//qDebug("Item %d: from=%d, length=%d, space=%d x=%d", i, ti.from(), ti.length(), ti.isSpace(), ti.x() );
if ( ti.isTab() || ti.isObject() )
continue;
int textFlags = _tf;
if ( !noaccel && numUnderlines > cUlChar && ti.from() == underlinePositions[cUlChar] ) {
textFlags |= Qt::Underline;
cUlChar++;
}
#if defined(Q_WS_X11) || defined(Q_WS_QWS)
if ( painter->bg_mode == Qt::OpaqueMode ) {
int h = ti.ascent() + ti.descent() + 1;
if (ti.y() + h < height)
// don't add leading to last line
h += fm.leading();
qt_draw_background( painter, r.x()+lb + ti.x(), r.y() + yoff + ti.y() - ti.ascent(),
ti.width(), h);
}
#endif
painter->drawTextItem( r.x()+lb, r.y() + yoff, ti, textFlags );
}
if ( restoreClipping ) {
painter->setClipRegion( painterClipRegion );
painter->setClipping( painterHasClip );
}
}
if ( underlinePositions != underlinePositionStack )
delete [] underlinePositions;
}
/*!
\overload
Returns the bounding rectangle of the aligned text that would be
printed with the corresponding drawText() function using the first
\a len characters from \a str if \a len is > -1, or the whole of
\a str if \a len is -1. The drawing, and hence the bounding
rectangle, is constrained to the rectangle \a r, or to the
rectangle required to draw the text, whichever is the larger.
The \a internal parameter should not be used.
\sa drawText(), fontMetrics(), QFontMetrics::boundingRect(), Qt::TextFlags
*/
QRect QPainter::boundingRect( const QRect &r, int flags,
const QString& str, int len, QTextParag **internal )
{
QRect brect;
if ( str.isEmpty() )
brect.setRect( r.x(),r.y(), 0,0 );
else
drawText( r, flags | DontPrint, str, len, &brect, internal );
return brect;
}
/*!
\fn QRect QPainter::boundingRect( int x, int y, int w, int h, int flags, const QString&, int len = -1, QTextParag **intern=0 );
Returns the bounding rectangle of the aligned text that would be
printed with the corresponding drawText() function using the first
\a len characters of the string if \a len is > -1, or the whole of
the string if \a len is -1. The drawing, and hence the bounding
rectangle, is constrained to the rectangle that begins at point \a
(x, y) with width \a w and hight \a h, or to the
rectangle required to draw the text, whichever is the larger.
The \a flags argument is
the bitwise OR of the following flags:
\table
\header \i Flag \i Meaning
\row \i \c AlignAuto \i aligns according to the language, usually left.
\row \i \c AlignLeft \i aligns to the left border.
\row \i \c AlignRight \i aligns to the right border.
\row \i \c AlignHCenter \i aligns horizontally centered.
\row \i \c AlignTop \i aligns to the top border.
\row \i \c AlignBottom \i aligns to the bottom border.
\row \i \c AlignVCenter \i aligns vertically centered.
\row \i \c AlignCenter \i (== \c AlignHCenter | \c AlignVCenter).
\row \i \c SingleLine \i ignores newline characters in the text.
\row \i \c ExpandTabs \i expands tabs.
\row \i \c ShowPrefix \i interprets "&x" as "<u>x</u>".
\row \i \c WordBreak \i breaks the text to fit the rectangle.
\endtable
Horizontal alignment defaults to \c AlignLeft and vertical
alignment defaults to \c AlignTop.
If several of the horizontal or several of the vertical alignment flags
are set, the resulting alignment is undefined.
The \a intern parameter should not be used.
\sa Qt::TextFlags
*/
/*****************************************************************************
QPen member functions
*****************************************************************************/
/*!
\class QPen qpen.h
\brief The QPen class defines how a QPainter should draw lines and outlines
of shapes.
\ingroup graphics
\ingroup images
\ingroup shared
\mainclass
A pen has a style, width, color, cap style and join style.
The pen style defines the line type. The default pen style is \c
Qt::SolidLine. Setting the style to \c NoPen tells the painter to
not draw lines or outlines.
When drawing 1 pixel wide diagonal lines you can either use a very
fast algorithm (specified by a line width of 0, which is the
default), or a slower but more accurate algorithm (specified by a
line width of 1). For horizontal and vertical lines a line width
of 0 is the same as a line width of 1. The cap and join style have
no effect on 0-width lines.
The pen color defines the color of lines and text. The default
line color is black. The QColor documentation lists predefined
colors.
The cap style defines how the end points of lines are drawn. The
join style defines how the joins between two lines are drawn when
multiple connected lines are drawn (QPainter::drawPolyline()
etc.). The cap and join styles only apply to wide lines, i.e. when
the width is 1 or greater.
Use the QBrush class to specify fill styles.
Example:
\code
QPainter painter;
QPen pen( red, 2 ); // red solid line, 2 pixels wide
painter.begin( &anyPaintDevice ); // paint something
painter.setPen( pen ); // set the red, wide pen
painter.drawRect( 40,30, 200,100 ); // draw a rectangle
painter.setPen( blue ); // set blue pen, 0 pixel width
painter.drawLine( 40,30, 240,130 ); // draw a diagonal in rectangle
painter.end(); // painting done
\endcode
See the \l Qt::PenStyle enum type for a complete list of pen
styles.
With reference to the end points of lines, for wide (non-0-width)
pens it depends on the cap style whether the end point is drawn or
not. QPainter will try to make sure that the end point is drawn
for 0-width pens, but this cannot be absolutely guaranteed because
the underlying drawing engine is free to use any (typically
accelerated) algorithm for drawing 0-width lines. On all tested
systems, however, the end point of at least all non-diagonal lines
are drawn.
A pen's color(), width(), style(), capStyle() and joinStyle() can
be set in the constructor or later with setColor(), setWidth(),
setStyle(), setCapStyle() and setJoinStyle(). Pens may also be
compared and streamed.
\img pen-styles.png Pen styles
\sa QPainter, QPainter::setPen()
*/
/*!
\internal
Initializes the pen.
*/
void QPen::init( const QColor &color, uint width, uint linestyle )
{
data = new QPenData;
Q_CHECK_PTR( data );
data->style = (PenStyle)(linestyle & MPenStyle);
data->width = width;
data->color = color;
data->linest = linestyle;
}
/*!
Constructs a default black solid line pen with 0 width, which
renders lines 1 pixel wide (fast diagonals).
*/
QPen::QPen()
{
init( Qt::black, 0, SolidLine ); // default pen
}
/*!
Constructs a black pen with 0 width (fast diagonals) and style \a
style.
\sa setStyle()
*/
QPen::QPen( PenStyle style )
{
init( Qt::black, 0, style );
}
/*!
Constructs a pen with the specified \a color, \a width and \a
style.
\sa setWidth(), setStyle(), setColor()
*/
QPen::QPen( const QColor &color, uint width, PenStyle style )
{
init( color, width, style );
}
/*!
Constructs a pen with the specified color \a cl and width \a w.
The pen style is set to \a s, the pen cap style to \a c and the
pen join style to \a j.
A line width of 0 will produce a 1 pixel wide line using a fast
algorithm for diagonals. A line width of 1 will also produce a 1
pixel wide line, but uses a slower more accurate algorithm for
diagonals. For horizontal and vertical lines a line width of 0 is
the same as a line width of 1. The cap and join style have no
effect on 0-width lines.
\sa setWidth(), setStyle(), setColor()
*/
QPen::QPen( const QColor &cl, uint w, PenStyle s, PenCapStyle c,
PenJoinStyle j )
{
init( cl, w, s | c | j );
}
/*!
Constructs a pen that is a copy of \a p.
*/
QPen::QPen( const QPen &p )
{
data = p.data;
data->ref();
}
/*!
Destroys the pen.
*/
QPen::~QPen()
{
if ( data->deref() )
delete data;
}
/*!
Detaches from shared pen data to make sure that this pen is the
only one referring the data.
If multiple pens share common data, this pen dereferences the data
and gets a copy of the data. Nothing is done if there is just a
single reference.
*/
void QPen::detach()
{
if ( data->count != 1 )
*this = copy();
}
/*!
Assigns \a p to this pen and returns a reference to this pen.
*/
QPen &QPen::operator=( const QPen &p )
{
p.data->ref();
if ( data->deref() )
delete data;
data = p.data;
return *this;
}
/*!
Returns a \link shclass.html deep copy\endlink of the pen.
*/
QPen QPen::copy() const
{
QPen p( data->color, data->width, data->style, capStyle(), joinStyle() );
return p;
}
/*!
\fn PenStyle QPen::style() const
Returns the pen style.
\sa setStyle()
*/
/*!
Sets the pen style to \a s.
See the \l Qt::PenStyle documentation for a list of all the
styles.
\warning On Mac OS X the style setting (other than \c NoPen and \c
SolidLine) have no effect as they are not implemented by the
underlying system.
\warning On Windows 95/98, the style setting (other than \c NoPen
and \c SolidLine) has no effect for lines with width greater than
1.
\sa style()
*/
void QPen::setStyle( PenStyle s )
{
if ( data->style == s )
return;
detach();
data->style = s;
data->linest = (data->linest & ~MPenStyle) | s;
}
/*!
\fn uint QPen::width() const
Returns the pen width.
\sa setWidth()
*/
/*!
Sets the pen width to \a w.
A line width of 0 will produce a 1 pixel wide line using a fast
algorithm for diagonals. A line width of 1 will also produce a 1
pixel wide line, but uses a slower more accurate algorithm for
diagonals. For horizontal and vertical lines a line width of 0 is
the same as a line width of 1. The cap and join style have no
effect on 0-width lines.
\sa width()
*/
void QPen::setWidth( uint w )
{
if ( data->width == w )
return;
detach();
data->width = w;
}
/*!
Returns the pen's cap style.
\sa setCapStyle()
*/
Qt::PenCapStyle QPen::capStyle() const
{
return (PenCapStyle)(data->linest & MPenCapStyle);
}
/*!
Sets the pen's cap style to \a c.
The default value is \c FlatCap. The cap style has no effect on
0-width pens.
\img pen-cap-styles.png Pen Cap Styles
\warning On Windows 95/98 and Macintosh, the cap style setting has
no effect. Wide lines are rendered as if the cap style was \c
SquareCap.
\sa capStyle()
*/
void QPen::setCapStyle( PenCapStyle c )
{
if ( (data->linest & MPenCapStyle) == c )
return;
detach();
data->linest = (data->linest & ~MPenCapStyle) | c;
}
/*!
Returns the pen's join style.
\sa setJoinStyle()
*/
Qt::PenJoinStyle QPen::joinStyle() const
{
return (PenJoinStyle)(data->linest & MPenJoinStyle);
}
/*!
Sets the pen's join style to \a j.
The default value is \c MiterJoin. The join style has no effect on
0-width pens.
\img pen-join-styles.png Pen Join Styles
\warning On Windows 95/98 and Macintosh, the join style setting
has no effect. Wide lines are rendered as if the join style was \c
BevelJoin.
\sa joinStyle()
*/
void QPen::setJoinStyle( PenJoinStyle j )
{
if ( (data->linest & MPenJoinStyle) == j )
return;
detach();
data->linest = (data->linest & ~MPenJoinStyle) | j;
}
/*!
\fn const QColor &QPen::color() const
Returns the pen color.
\sa setColor()
*/
/*!
Sets the pen color to \a c.
\sa color()
*/
void QPen::setColor( const QColor &c )
{
detach();
data->color = c;
}
/*!
\fn bool QPen::operator!=( const QPen &p ) const
Returns TRUE if the pen is different from \a p; otherwise returns
FALSE.
Two pens are different if they have different styles, widths or
colors.
\sa operator==()
*/
/*!
Returns TRUE if the pen is equal to \a p; otherwise returns FALSE.
Two pens are equal if they have equal styles, widths and colors.
\sa operator!=()
*/
bool QPen::operator==( const QPen &p ) const
{
return (p.data == data) || (p.data->linest == data->linest &&
p.data->width == data->width && p.data->color == data->color);
}
/*****************************************************************************
QPen stream functions
*****************************************************************************/
#ifndef QT_NO_DATASTREAM
/*!
\relates QPen
Writes the pen \a p to the stream \a s and returns a reference to
the stream.
\sa \link datastreamformat.html Format of the QDataStream operators \endlink
*/
QDataStream &operator<<( QDataStream &s, const QPen &p )
{
// ### width() should not be restricted to 8-bit values
if ( s.version() < 3 )
return s << (Q_UINT8)p.style() << (Q_UINT8)p.width() << p.color();
else
return s << (Q_UINT8)( p.style() | p.capStyle() | p.joinStyle() )
<< (Q_UINT8)p.width() << p.color();
}
/*!
\relates QPen
Reads a pen from the stream \a s into \a p and returns a reference
to the stream.
\sa \link datastreamformat.html Format of the QDataStream operators \endlink
*/
QDataStream &operator>>( QDataStream &s, QPen &p )
{
Q_UINT8 style, width;
QColor color;
s >> style;
s >> width;
s >> color;
p = QPen( color, (uint)width, (Qt::PenStyle)style ); // owl
return s;
}
#endif //QT_NO_DATASTREAM
/*****************************************************************************
QBrush member functions
*****************************************************************************/
/*!
\class QBrush qbrush.h
\brief The QBrush class defines the fill pattern of shapes drawn by a QPainter.
\ingroup graphics
\ingroup images
\ingroup shared
A brush has a style and a color. One of the brush styles is a
custom pattern, which is defined by a QPixmap.
The brush style defines the fill pattern. The default brush style
is \c NoBrush (depending on how you construct a brush). This style
tells the painter to not fill shapes. The standard style for
filling is \c SolidPattern.
The brush color defines the color of the fill pattern. The QColor
documentation lists the predefined colors.
Use the QPen class for specifying line/outline styles.
Example:
\code
QPainter painter;
QBrush brush( yellow ); // yellow solid pattern
painter.begin( &anyPaintDevice ); // paint something
painter.setBrush( brush ); // set the yellow brush
painter.setPen( NoPen ); // do not draw outline
painter.drawRect( 40,30, 200,100 ); // draw filled rectangle
painter.setBrush( NoBrush ); // do not fill
painter.setPen( black ); // set black pen, 0 pixel width
painter.drawRect( 10,10, 30,20 ); // draw rectangle outline
painter.end(); // painting done
\endcode
See the setStyle() function for a complete list of brush styles.
\img brush-styles.png Brush Styles
\sa QPainter, QPainter::setBrush(), QPainter::setBrushOrigin()
*/
/*!
\internal
Initializes the brush.
*/
void QBrush::init( const QColor &color, BrushStyle style )
{
data = new QBrushData;
Q_CHECK_PTR( data );
data->style = style;
data->color = color;
data->pixmap = 0;
}
/*!
Constructs a default black brush with the style \c NoBrush (will
not fill shapes).
*/
QBrush::QBrush()
{
static QBrushData* defBrushData = 0;
if ( !defBrushData ) {
static QSharedCleanupHandler<QBrushData> defBrushCleanup;
defBrushData = new QBrushData;
defBrushData->style = NoBrush;
defBrushData->color = Qt::black;
defBrushData->pixmap = 0;
defBrushCleanup.set( &defBrushData );
}
data = defBrushData;
data->ref();
}
/*!
Constructs a black brush with the style \a style.
\sa setStyle()
*/
QBrush::QBrush( BrushStyle style )
{
init( Qt::black, style );
}
/*!
Constructs a brush with the color \a color and the style \a style.
\sa setColor(), setStyle()
*/
QBrush::QBrush( const QColor &color, BrushStyle style )
{
init( color, style );
}
/*!
Constructs a brush with the color \a color and a custom pattern
stored in \a pixmap.
The color will only have an effect for monochrome pixmaps, i.e.
for QPixmap::depth() == 1.
Pixmap brushes are currently not supported when printing on X11.
\sa setColor(), setPixmap()
*/
QBrush::QBrush( const QColor &color, const QPixmap &pixmap )
{
init( color, CustomPattern );
setPixmap( pixmap );
}
/*!
Constructs a brush that is a \link shclass.html shallow
copy\endlink of \a b.
*/
QBrush::QBrush( const QBrush &b )
{
data = b.data;
data->ref();
}
/*!
Destroys the brush.
*/
QBrush::~QBrush()
{
if ( data->deref() ) {
delete data->pixmap;
delete data;
}
}
/*!
Detaches from shared brush data to make sure that this brush is
the only one referring the data.
If multiple brushes share common data, this brush dereferences the
data and gets a copy of the data. Nothing is done if there is just
a single reference.
*/
void QBrush::detach()
{
if ( data->count != 1 )
*this = copy();
}
/*!
Assigns \a b to this brush and returns a reference to this brush.
*/
QBrush &QBrush::operator=( const QBrush &b )
{
b.data->ref(); // beware of b = b
if ( data->deref() ) {
delete data->pixmap;
delete data;
}
data = b.data;
return *this;
}
/*!
Returns a \link shclass.html deep copy\endlink of the brush.
*/
QBrush QBrush::copy() const
{
if ( data->style == CustomPattern ) { // brush has pixmap
QBrush b( data->color, *data->pixmap );
return b;
} else { // brush has std pattern
QBrush b( data->color, data->style );
return b;
}
}
/*!
\fn BrushStyle QBrush::style() const
Returns the brush style.
\sa setStyle()
*/
/*!
Sets the brush style to \a s.
The brush styles are:
\table
\header \i Pattern \i Meaning
\row \i NoBrush \i will not fill shapes (default).
\row \i SolidPattern \i solid (100%) fill pattern.
\row \i Dense1Pattern \i11 94% fill pattern.
\row \i Dense2Pattern \i11 88% fill pattern.
\row \i Dense3Pattern \i11 63% fill pattern.
\row \i Dense4Pattern \i11 50% fill pattern.
\row \i Dense5Pattern \i11 37% fill pattern.
\row \i Dense6Pattern \i11 12% fill pattern.
\row \i Dense7Pattern \i11 6% fill pattern.
\row \i HorPattern \i horizontal lines pattern.
\row \i VerPattern \i vertical lines pattern.
\row \i CrossPattern \i crossing lines pattern.
\row \i BDiagPattern \i diagonal lines (directed /) pattern.
\row \i FDiagPattern \i diagonal lines (directed \) pattern.
\row \i DiagCrossPattern \i diagonal crossing lines pattern.
\row \i CustomPattern \i set when a pixmap pattern is being used.
\endtable
On Windows, dense and custom patterns cannot be transparent.
See the \link #details Detailed Description\endlink for a picture
of all the styles.
\sa style()
*/
void QBrush::setStyle( BrushStyle s ) // set brush style
{
if ( data->style == s )
return;
#if defined(QT_CHECK_RANGE)
if ( s == CustomPattern )
qWarning( "QBrush::setStyle: CustomPattern is for internal use" );
#endif
detach();
data->style = s;
}
/*!
\fn const QColor &QBrush::color() const
Returns the brush color.
\sa setColor()
*/
/*!
Sets the brush color to \a c.
\sa color(), setStyle()
*/
void QBrush::setColor( const QColor &c )
{
detach();
data->color = c;
}
/*!
\fn QPixmap *QBrush::pixmap() const
Returns a pointer to the custom brush pattern, or 0 if no custom
brush pattern has been set.
\sa setPixmap()
*/
/*!
Sets the brush pixmap to \a pixmap. The style is set to \c
CustomPattern.
The current brush color will only have an effect for monochrome
pixmaps, i.e. for QPixmap::depth() == 1.
Pixmap brushes are currently not supported when printing on X11.
\sa pixmap(), color()
*/
void QBrush::setPixmap( const QPixmap &pixmap )
{
detach();
if ( data->pixmap )
delete data->pixmap;
if ( pixmap.isNull() ) {
data->style = NoBrush;
data->pixmap = 0;
} else {
data->style = CustomPattern;
data->pixmap = new QPixmap( pixmap );
if ( data->pixmap->optimization() == QPixmap::MemoryOptim )
data->pixmap->setOptimization( QPixmap::NormalOptim );
}
}
/*!
\fn bool QBrush::operator!=( const QBrush &b ) const
Returns TRUE if the brush is different from \a b; otherwise
returns FALSE.
Two brushes are different if they have different styles, colors or
pixmaps.
\sa operator==()
*/
/*!
Returns TRUE if the brush is equal to \a b; otherwise returns
FALSE.
Two brushes are equal if they have equal styles, colors and
pixmaps.
\sa operator!=()
*/
bool QBrush::operator==( const QBrush &b ) const
{
return (b.data == data) || (b.data->style == data->style &&
b.data->color == data->color &&
b.data->pixmap == data->pixmap);
}
/*!
\fn inline double QPainter::translationX() const
\internal
*/
/*!
\fn inline double QPainter::translationY() const
\internal
*/
/*****************************************************************************
QBrush stream functions
*****************************************************************************/
#ifndef QT_NO_DATASTREAM
/*!
\relates QBrush
Writes the brush \a b to the stream \a s and returns a reference
to the stream.
\sa \link datastreamformat.html Format of the QDataStream operators \endlink
*/
QDataStream &operator<<( QDataStream &s, const QBrush &b )
{
s << (Q_UINT8)b.style() << b.color();
if ( b.style() == Qt::CustomPattern )
#ifndef QT_NO_IMAGEIO
s << *b.pixmap();
#else
qWarning("No Image Brush I/O");
#endif
return s;
}
/*!
\relates QBrush
Reads the brush \a b from the stream \a s and returns a reference
to the stream.
\sa \link datastreamformat.html Format of the QDataStream operators \endlink
*/
QDataStream &operator>>( QDataStream &s, QBrush &b )
{
Q_UINT8 style;
QColor color;
s >> style;
s >> color;
if ( style == Qt::CustomPattern ) {
#ifndef QT_NO_IMAGEIO
QPixmap pm;
s >> pm;
b = QBrush( color, pm );
#else
qWarning("No Image Brush I/O");
#endif
}
else
b = QBrush( color, (Qt::BrushStyle)style );
return s;
}
#endif // QT_NO_DATASTREAM