summaryrefslogtreecommitdiffstats
path: root/src/kernel/qpainter_x11.cpp
diff options
context:
space:
mode:
authorTimothy Pearson <[email protected]>2011-07-10 15:24:15 -0500
committerTimothy Pearson <[email protected]>2011-07-10 15:24:15 -0500
commitbd0f3345a938b35ce6a12f6150373b0955b8dd12 (patch)
tree7a520322212d48ebcb9fbe1087e7fca28b76185c /src/kernel/qpainter_x11.cpp
downloadqt3-bd0f3345a938b35ce6a12f6150373b0955b8dd12.tar.gz
qt3-bd0f3345a938b35ce6a12f6150373b0955b8dd12.zip
Add Qt3 development HEAD version
Diffstat (limited to 'src/kernel/qpainter_x11.cpp')
-rw-r--r--src/kernel/qpainter_x11.cpp3153
1 files changed, 3153 insertions, 0 deletions
diff --git a/src/kernel/qpainter_x11.cpp b/src/kernel/qpainter_x11.cpp
new file mode 100644
index 0000000..206bffc
--- /dev/null
+++ b/src/kernel/qpainter_x11.cpp
@@ -0,0 +1,3153 @@
+/****************************************************************************
+**
+** Implementation of QPainter class for X11
+**
+** 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 [email protected].
+**
+** 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 "qplatformdefs.h"
+
+#include "qfont.h"
+#include "qpainter.h"
+#include "qwidget.h"
+#include "qbitmap.h"
+#include "qpixmapcache.h"
+#include "qtextcodec.h"
+#include "qpaintdevicemetrics.h"
+
+#include "qt_x11_p.h"
+
+#include "qtextlayout_p.h"
+#include "qfontdata_p.h"
+#include "qfontengine_p.h"
+#include "qtextengine_p.h"
+
+#include <math.h>
+
+// paintevent magic to provide Windows semantics on X11
+static QRegion* paintEventClipRegion = 0;
+static QPaintDevice* paintEventDevice = 0;
+
+void qt_set_paintevent_clipping( QPaintDevice* dev, const QRegion& region)
+{
+ if ( !paintEventClipRegion )
+ paintEventClipRegion = new QRegion( region );
+ else
+ *paintEventClipRegion = region;
+ paintEventDevice = dev;
+}
+
+void qt_clear_paintevent_clipping()
+{
+ delete paintEventClipRegion;
+ paintEventClipRegion = 0;
+ paintEventDevice = 0;
+}
+
+class QWFlagWidget : public QWidget
+{
+public:
+ void setWState( WFlags f ) { QWidget::setWState(f); }
+ void clearWState( WFlags f ) { QWidget::clearWState(f); }
+ void setWFlags( WFlags f ) { QWidget::setWFlags(f); }
+ void clearWFlags( WFlags f ) { QWidget::clearWFlags(f); }
+};
+
+void qt_erase_region( QWidget* w, const QRegion& region)
+{
+ QRegion reg = region;
+
+ if ( QPainter::redirect(w) || (!w->isTopLevel() && w->backgroundPixmap()
+ && w->backgroundOrigin() != QWidget::WidgetOrigin) ) {
+ QPoint offset = w->backgroundOffset();
+ int ox = offset.x();
+ int oy = offset.y();
+
+ bool unclipped = w->testWFlags( Qt::WPaintUnclipped );
+ if ( unclipped )
+ ((QWFlagWidget*)w)->clearWFlags( Qt::WPaintUnclipped );
+ QPainter p( w );
+ p.setClipRegion( region ); // automatically includes paintEventDevice if required
+ if ( w->backgroundPixmap() )
+ p.drawTiledPixmap( 0, 0, w->width(), w->height(),
+ *w->backgroundPixmap(), ox, oy );
+ else
+ p.fillRect( w->rect(), w->eraseColor() );
+ if ( unclipped )
+ ((QWFlagWidget*)w)->setWFlags( Qt::WPaintUnclipped );
+ return;
+ }
+
+ if ( w == paintEventDevice && paintEventClipRegion )
+ reg = paintEventClipRegion->intersect( reg );
+
+ QMemArray<QRect> r = reg.rects();
+ for (uint i=0; i<r.size(); i++) {
+ const QRect& rr = r[(int)i];
+ XClearArea( w->x11Display(), w->winId(),
+ rr.x(), rr.y(), rr.width(), rr.height(), False );
+ }
+}
+
+void qt_erase_rect( QWidget* w, const QRect& r)
+{
+ if ( QPainter::redirect(w) || w == paintEventDevice
+ || w->backgroundOrigin() != QWidget::WidgetOrigin )
+ qt_erase_region( w, r );
+ else
+ XClearArea( w->x11Display(), w->winId(), r.x(), r.y(), r.width(), r.height(), False );
+
+}
+
+#ifdef QT_NO_XFTFREETYPE
+static const Qt::HANDLE rendhd = 0;
+#endif
+
+// hack, so we don't have to make QRegion::clipRectangles() public or include
+// X11 headers in qregion.h
+inline void *qt_getClipRects( const QRegion &r, int &num )
+{
+ return r.clipRectangles( num );
+}
+
+static inline void x11SetClipRegion(Display *dpy, GC gc, GC gc2, Qt::HANDLE draw, const QRegion &r)
+{
+ int num;
+ XRectangle *rects = (XRectangle *)qt_getClipRects( r, num );
+
+ if (gc)
+ XSetClipRectangles( dpy, gc, 0, 0, rects, num, YXBanded );
+ if (gc2)
+ XSetClipRectangles( dpy, gc2, 0, 0, rects, num, YXBanded );
+
+#ifndef QT_NO_XFTFREETYPE
+ if (draw)
+ XftDrawSetClipRectangles((XftDraw *) draw, 0, 0, rects, num);
+#else
+ Q_UNUSED(draw);
+#endif // QT_NO_XFTFREETYPE
+}
+
+static inline void x11ClearClipRegion(Display *dpy, GC gc, GC gc2, Qt::HANDLE draw)
+{
+ if (gc)
+ XSetClipMask(dpy, gc, None);
+ if (gc2)
+ XSetClipMask(dpy, gc2, None);
+
+#ifndef QT_NO_XFTFREETYPE
+ if (draw) {
+# ifdef QT_XFT2
+ XftDrawSetClip((XftDraw *) draw, None);
+# else
+ // stupid Xft1
+ Picture pict = XftDrawPicture((XftDraw *) draw);
+ XRenderPictureAttributes pattr;
+ pattr.clip_mask = None;
+ XRenderChangePicture(dpy, pict, CPClipMask, &pattr);
+# endif // QT_XFT2
+ }
+#else
+ Q_UNUSED(draw);
+#endif // QT_NO_XFTFREETYPE
+}
+
+
+/*****************************************************************************
+ Trigonometric function for QPainter
+
+ We have implemented simple sine and cosine function that are called from
+ QPainter::drawPie() and QPainter::drawChord() when drawing the outline of
+ pies and chords.
+ These functions are slower and less accurate than math.h sin() and cos(),
+ but with still around 1/70000th sec. execution time (on a 486DX2-66) and
+ 8 digits accuracy, it should not be the bottleneck in drawing these shapes.
+ The advantage is that you don't have to link in the math library.
+ *****************************************************************************/
+
+const double Q_PI = 3.14159265358979323846; // pi
+const double Q_2PI = 6.28318530717958647693; // 2*pi
+const double Q_PI2 = 1.57079632679489661923; // pi/2
+
+
+#if defined(Q_CC_GNU) && defined(Q_OS_AIX)
+// AIX 4.2 gcc 2.7.2.3 gets internal error.
+static int qRoundAIX( double d )
+{
+ return qRound(d);
+}
+#define qRound qRoundAIX
+#endif
+
+
+#if defined(Q_CC_GNU) && defined(__i386__)
+
+inline double qcos( double a )
+{
+ double r;
+ __asm__ (
+ "fcos"
+ : "=t" (r) : "0" (a) );
+ return(r);
+}
+
+inline double qsin( double a )
+{
+ double r;
+ __asm__ (
+ "fsin"
+ : "=t" (r) : "0" (a) );
+ return(r);
+}
+
+double qsincos( double a, bool calcCos=FALSE )
+{
+ return calcCos ? qcos(a) : qsin(a);
+}
+
+#else
+
+double qsincos( double a, bool calcCos=FALSE )
+{
+ if ( calcCos ) // calculate cosine
+ a -= Q_PI2;
+ if ( a >= Q_2PI || a <= -Q_2PI ) { // fix range: -2*pi < a < 2*pi
+ int m = (int)(a/Q_2PI);
+ a -= Q_2PI*m;
+ }
+ if ( a < 0.0 ) // 0 <= a < 2*pi
+ a += Q_2PI;
+ int sign = a > Q_PI ? -1 : 1;
+ if ( a >= Q_PI )
+ a = Q_2PI - a;
+ if ( a >= Q_PI2 )
+ a = Q_PI - a;
+ if ( calcCos )
+ sign = -sign;
+ double a2 = a*a; // here: 0 <= a < pi/4
+ double a3 = a2*a; // make taylor sin sum
+ double a5 = a3*a2;
+ double a7 = a5*a2;
+ double a9 = a7*a2;
+ double a11 = a9*a2;
+ return (a-a3/6+a5/120-a7/5040+a9/362880-a11/39916800)*sign;
+}
+
+inline double qsin( double a ) { return qsincos(a, FALSE); }
+inline double qcos( double a ) { return qsincos(a, TRUE); }
+
+#endif
+
+
+/*****************************************************************************
+ QPainter internal GC (Graphics Context) allocator.
+
+ The GC allocator offers two functions; alloc_gc() and free_gc() that
+ reuse GC objects instead of calling XCreateGC() and XFreeGC(), which
+ are a whole lot slower.
+ *****************************************************************************/
+
+struct QGC
+{
+ GC gc;
+ char in_use;
+ bool mono;
+ int scrn;
+};
+
+const int gc_array_size = 256;
+static QGC gc_array[gc_array_size]; // array of GCs
+static bool gc_array_init = FALSE;
+
+
+static void init_gc_array()
+{
+ if ( !gc_array_init ) {
+ memset( gc_array, 0, gc_array_size*sizeof(QGC) );
+ gc_array_init = TRUE;
+ }
+}
+
+static void cleanup_gc_array( Display *dpy )
+{
+ register QGC *p = gc_array;
+ int i = gc_array_size;
+ if ( gc_array_init ) {
+ while ( i-- ) {
+ if ( p->gc ) // destroy GC
+ XFreeGC( dpy, p->gc );
+ p++;
+ }
+ gc_array_init = FALSE;
+ }
+}
+
+// #define DONT_USE_GC_ARRAY
+
+static GC alloc_gc( Display *dpy, int scrn, Drawable hd, bool monochrome=FALSE,
+ bool privateGC = FALSE )
+{
+#if defined(DONT_USE_GC_ARRAY)
+ privateGC = TRUE; // will be slower
+#endif
+ if ( privateGC ) {
+ GC gc = XCreateGC( dpy, hd, 0, 0 );
+ XSetGraphicsExposures( dpy, gc, False );
+ return gc;
+ }
+ register QGC *p = gc_array;
+ int i = gc_array_size;
+ if ( !gc_array_init ) // not initialized
+ init_gc_array();
+ while ( i-- ) {
+ if ( !p->gc ) { // create GC (once)
+ p->gc = XCreateGC( dpy, hd, 0, 0 );
+ p->scrn = scrn;
+ XSetGraphicsExposures( dpy, p->gc, False );
+ p->in_use = FALSE;
+ p->mono = monochrome;
+ }
+ if ( !p->in_use && p->mono == monochrome && p->scrn == scrn ) {
+ p->in_use = TRUE; // available/compatible GC
+ return p->gc;
+ }
+ p++;
+ }
+#if defined(QT_CHECK_NULL)
+ qWarning( "QPainter: Internal error; no available GC" );
+#endif
+ GC gc = XCreateGC( dpy, hd, 0, 0 );
+ XSetGraphicsExposures( dpy, gc, False );
+ return gc;
+}
+
+static void free_gc( Display *dpy, GC gc, bool privateGC = FALSE )
+{
+#if defined(DONT_USE_GC_ARRAY)
+ privateGC = TRUE; // will be slower
+#endif
+ if ( privateGC ) {
+ Q_ASSERT( dpy != 0 );
+ XFreeGC( dpy, gc );
+ return;
+ }
+ register QGC *p = gc_array;
+ int i = gc_array_size;
+ if ( gc_array_init ) {
+ while ( i-- ) {
+ if ( p->gc == gc ) {
+ p->in_use = FALSE; // set available
+ XSetClipMask( dpy, gc, None ); // make it reusable
+ XSetFunction( dpy, gc, GXcopy );
+ XSetFillStyle( dpy, gc, FillSolid );
+ XSetTSOrigin( dpy, gc, 0, 0 );
+ return;
+ }
+ p++;
+ }
+ }
+
+ // not found in gc_array
+ XFreeGC(dpy, gc);
+}
+
+
+/*****************************************************************************
+ QPainter internal GC (Graphics Context) cache for solid pens and
+ brushes.
+
+ The GC cache makes a significant contribution to speeding up
+ drawing. Setting new pen and brush colors will make the painter
+ look for another GC with the same color instead of changing the
+ color value of the GC currently in use. The cache structure is
+ optimized for fast lookup. Only solid line pens with line width 0
+ and solid brushes are cached.
+
+ In addition, stored GCs may have an implicit clipping region
+ set. This prevents any drawing outside paint events. Both
+ updatePen() and updateBrush() keep track of the validity of this
+ clipping region by storing the clip_serial number in the cache.
+
+*****************************************************************************/
+
+struct QGCC // cached GC
+{
+ GC gc;
+ uint pix;
+ int count;
+ int hits;
+ uint clip_serial;
+ int scrn;
+};
+
+const int gc_cache_size = 29; // multiply by 4
+static QGCC *gc_cache_buf;
+static QGCC *gc_cache[4*gc_cache_size];
+static bool gc_cache_init = FALSE;
+static uint gc_cache_clip_serial = 0;
+
+
+static void init_gc_cache()
+{
+ if ( !gc_cache_init ) {
+ gc_cache_init = TRUE;
+ gc_cache_clip_serial = 0;
+ QGCC *g = gc_cache_buf = new QGCC[4*gc_cache_size];
+ memset( g, 0, 4*gc_cache_size*sizeof(QGCC) );
+ for ( int i=0; i<4*gc_cache_size; i++ )
+ gc_cache[i] = g++;
+ }
+}
+
+
+// #define GC_CACHE_STAT
+#if defined(GC_CACHE_STAT)
+#include "qtextstream.h"
+#include "qbuffer.h"
+
+static int g_numhits = 0;
+static int g_numcreates = 0;
+static int g_numfaults = 0;
+#endif
+
+
+static void cleanup_gc_cache()
+{
+ if ( !gc_cache_init )
+ return;
+#if defined(GC_CACHE_STAT)
+ qDebug( "Number of cache hits = %d", g_numhits );
+ qDebug( "Number of cache creates = %d", g_numcreates );
+ qDebug( "Number of cache faults = %d", g_numfaults );
+ for ( int i=0; i<gc_cache_size; i++ ) {
+ QCString str;
+ QBuffer buf( str );
+ buf.open(IO_ReadWrite);
+ QTextStream s(&buf);
+ s << i << ": ";
+ for ( int j=0; j<4; j++ ) {
+ QGCC *g = gc_cache[i*4+j];
+ s << (g->gc ? 'X' : '-') << ',' << g->hits << ','
+ << g->count << '\t';
+ }
+ s << '\0';
+ qDebug( str );
+ buf.close();
+ }
+#endif
+ delete [] gc_cache_buf;
+ gc_cache_init = FALSE;
+}
+
+
+static bool obtain_gc( void **ref, GC *gc, uint pix, Display *dpy, int scrn,
+ Qt::HANDLE hd, uint painter_clip_serial )
+{
+ if ( !gc_cache_init )
+ init_gc_cache();
+
+ int k = (pix % gc_cache_size) * 4;
+ QGCC *g = gc_cache[k];
+ QGCC *prev = 0;
+
+#define NOMATCH (g->gc && (g->pix != pix || g->scrn != scrn || \
+ (g->clip_serial > 0 && g->clip_serial != painter_clip_serial)))
+
+ if ( NOMATCH ) {
+ prev = g;
+ g = gc_cache[++k];
+ if ( NOMATCH ) {
+ prev = g;
+ g = gc_cache[++k];
+ if ( NOMATCH ) {
+ prev = g;
+ g = gc_cache[++k];
+ if ( NOMATCH ) {
+ if ( g->count == 0 && g->scrn == scrn) { // steal this GC
+ g->pix = pix;
+ g->count = 1;
+ g->hits = 1;
+ g->clip_serial = 0;
+ XSetForeground( dpy, g->gc, pix );
+ XSetClipMask(dpy, g->gc, None);
+ gc_cache[k] = prev;
+ gc_cache[k-1] = g;
+ *ref = (void *)g;
+ *gc = g->gc;
+ return TRUE;
+ } else { // all GCs in use
+#if defined(GC_CACHE_STAT)
+ g_numfaults++;
+#endif
+ *ref = 0;
+ return FALSE;
+ }
+ }
+ }
+ }
+ }
+
+#undef NOMATCH
+
+ *ref = (void *)g;
+
+ if ( g->gc ) { // reuse existing GC
+#if defined(GC_CACHE_STAT)
+ g_numhits++;
+#endif
+ *gc = g->gc;
+ g->count++;
+ g->hits++;
+ if ( prev && g->hits > prev->hits ) { // maintain LRU order
+ gc_cache[k] = prev;
+ gc_cache[k-1] = g;
+ }
+ return TRUE;
+ } else { // create new GC
+#if defined(GC_CACHE_STAT)
+ g_numcreates++;
+#endif
+ g->gc = alloc_gc( dpy, scrn, hd, FALSE );
+ g->scrn = scrn;
+ g->pix = pix;
+ g->count = 1;
+ g->hits = 1;
+ g->clip_serial = 0;
+ *gc = g->gc;
+ return FALSE;
+ }
+}
+
+static inline void release_gc( void *ref )
+{
+ ((QGCC*)ref)->count--;
+}
+
+/*****************************************************************************
+ QPainter member functions
+ *****************************************************************************/
+
+/*!
+ \internal
+
+ Internal function that initializes the painter.
+*/
+
+void QPainter::initialize()
+{
+ init_gc_array();
+ init_gc_cache();
+}
+
+/*!
+ \internal
+
+ Internal function that cleans up the painter.
+*/
+
+void QPainter::cleanup()
+{
+ cleanup_gc_cache();
+ cleanup_gc_array( QPaintDevice::x11AppDisplay() );
+ QPointArray::cleanBuffers();
+}
+
+/*!
+ \internal
+
+ Internal function that destroys up the painter.
+*/
+
+void QPainter::destroy()
+{
+
+}
+
+void QPainter::init()
+{
+ d = 0;
+ flags = IsStartingUp;
+ bg_col = white; // default background color
+ bg_mode = TransparentMode; // default background mode
+ rop = CopyROP; // default ROP
+ tabstops = 0; // default tabbing
+ tabarray = 0;
+ tabarraylen = 0;
+ ps_stack = 0;
+ wm_stack = 0;
+ gc = gc_brush = 0;
+ pdev = 0;
+ dpy = 0;
+ txop = txinv = 0;
+ penRef = brushRef = 0;
+ clip_serial = 0;
+ pfont = 0;
+ block_ext = FALSE;
+}
+
+
+/*!
+ \fn const QFont &QPainter::font() const
+
+ Returns the currently set painter font.
+
+ \sa setFont(), QFont
+*/
+
+/*!
+ Sets the painter's font to \a font.
+
+ This font is used by subsequent drawText() functions. The text
+ color is the same as the pen color.
+
+ \sa font(), drawText()
+*/
+
+void QPainter::setFont( const QFont &font )
+{
+#if defined(QT_CHECK_STATE)
+ if ( !isActive() )
+ qWarning( "QPainter::setFont: Will be reset by begin()" );
+#endif
+ if ( cfont.d != font.d ) {
+ cfont = font;
+ cfont.x11SetScreen( scrn );
+ setf(DirtyFont);
+ }
+}
+
+
+void QPainter::updateFont()
+{
+ if (!isActive())
+ return;
+
+ clearf(DirtyFont);
+ if ( testf(ExtDev) ) {
+ if (pdev->devType() == QInternal::Printer) {
+ if ( pfont ) delete pfont;
+ pfont = new QFont( cfont.d, pdev );
+ }
+ QPDevCmdParam param[1];
+ param[0].font = &cfont;
+ if ( !pdev->cmd( QPaintDevice::PdcSetFont, this, param ) || !hd )
+ return;
+ }
+ setf(NoCache);
+ if ( penRef )
+ updatePen(); // force a non-cached GC
+}
+
+
+void QPainter::updatePen()
+{
+ if (!isActive())
+ return;
+
+ if ( testf(ExtDev) ) {
+ QPDevCmdParam param[1];
+ param[0].pen = &cpen;
+ if ( !pdev->cmd( QPaintDevice::PdcSetPen, this, param ) || !hd )
+ return;
+ }
+
+ int ps = cpen.style();
+ bool cacheIt = !testf(ClipOn|MonoDev|NoCache) &&
+ (ps == NoPen || ps == SolidLine) &&
+ cpen.width() == 0 && rop == CopyROP;
+
+ bool obtained = FALSE;
+ bool internclipok = hasClipping();
+ if ( cacheIt ) {
+ if ( gc ) {
+ if ( penRef )
+ release_gc( penRef );
+ else
+ free_gc( dpy, gc );
+ }
+ obtained = obtain_gc(&penRef, &gc, cpen.color().pixel(scrn), dpy, scrn,
+ hd, clip_serial);
+ if ( !obtained && !penRef )
+ gc = alloc_gc( dpy, scrn, hd, FALSE );
+ } else {
+ if ( gc ) {
+ if ( penRef ) {
+ release_gc( penRef );
+ penRef = 0;
+ gc = alloc_gc( dpy, scrn, hd, testf(MonoDev) );
+ } else {
+ internclipok = TRUE;
+ }
+ } else {
+ gc = alloc_gc( dpy, scrn, hd, testf(MonoDev), testf(UsePrivateCx) );
+ }
+ }
+
+ if ( !internclipok ) {
+ if ( pdev == paintEventDevice && paintEventClipRegion ) {
+ if ( penRef &&((QGCC*)penRef)->clip_serial < gc_cache_clip_serial ) {
+ x11SetClipRegion( dpy, gc, 0, rendhd, *paintEventClipRegion );
+ ((QGCC*)penRef)->clip_serial = gc_cache_clip_serial;
+ } else if ( !penRef ) {
+ x11SetClipRegion( dpy, gc, 0, rendhd, *paintEventClipRegion );
+ }
+ } else if (penRef && ((QGCC*)penRef)->clip_serial ) {
+ x11ClearClipRegion(dpy, gc, 0, rendhd);
+ ((QGCC*)penRef)->clip_serial = 0;
+ }
+ }
+
+ if ( obtained )
+ return;
+
+ char dashes[10]; // custom pen dashes
+ int dash_len = 0; // length of dash list
+ int s = LineSolid;
+ int cp = CapButt;
+ int jn = JoinMiter;
+
+ /*
+ We are emulating Windows here. Windows treats cpen.width() == 1
+ (or 0) as a very special case. The fudge variable unifies this
+ case with the general case.
+ */
+ int dot = cpen.width(); // width of a dot
+ int fudge = 1;
+ bool allow_zero_lw = TRUE;
+ if ( dot <= 1 ) {
+ dot = 3;
+ fudge = 2;
+ }
+
+ switch( ps ) {
+ case NoPen:
+ case SolidLine:
+ s = LineSolid;
+ break;
+ case DashLine:
+ dashes[0] = fudge * 3 * dot;
+ dashes[1] = fudge * dot;
+ dash_len = 2;
+ allow_zero_lw = FALSE;
+ break;
+ case DotLine:
+ dashes[0] = dot;
+ dashes[1] = dot;
+ dash_len = 2;
+ allow_zero_lw = FALSE;
+ break;
+ case DashDotLine:
+ dashes[0] = 3 * dot;
+ dashes[1] = fudge * dot;
+ dashes[2] = dot;
+ dashes[3] = fudge * dot;
+ dash_len = 4;
+ allow_zero_lw = FALSE;
+ break;
+ case DashDotDotLine:
+ dashes[0] = 3 * dot;
+ dashes[1] = dot;
+ dashes[2] = dot;
+ dashes[3] = dot;
+ dashes[4] = dot;
+ dashes[5] = dot;
+ dash_len = 6;
+ allow_zero_lw = FALSE;
+ }
+ Q_ASSERT( dash_len <= (int) sizeof(dashes) );
+
+ switch ( cpen.capStyle() ) {
+ case SquareCap:
+ cp = CapProjecting;
+ break;
+ case RoundCap:
+ cp = CapRound;
+ break;
+ case FlatCap:
+ default:
+ cp = CapButt;
+ break;
+ }
+ switch ( cpen.joinStyle() ) {
+ case BevelJoin:
+ jn = JoinBevel;
+ break;
+ case RoundJoin:
+ jn = JoinRound;
+ break;
+ case MiterJoin:
+ default:
+ jn = JoinMiter;
+ break;
+ }
+
+ XSetForeground( dpy, gc, cpen.color().pixel(scrn) );
+ XSetBackground( dpy, gc, bg_col.pixel(scrn) );
+
+ if ( dash_len ) { // make dash list
+ XSetDashes( dpy, gc, 0, dashes, dash_len );
+ s = bg_mode == TransparentMode ? LineOnOffDash : LineDoubleDash;
+ }
+ XSetLineAttributes( dpy, gc,
+ (! allow_zero_lw && cpen.width() == 0) ? 1 : cpen.width(),
+ s, cp, jn );
+}
+
+
+void QPainter::updateBrush()
+{
+ if (!isActive())
+ return;
+
+ static const uchar dense1_pat[] = { 0xff, 0xbb, 0xff, 0xff, 0xff, 0xbb, 0xff, 0xff };
+ static const uchar dense2_pat[] = { 0x77, 0xff, 0xdd, 0xff, 0x77, 0xff, 0xdd, 0xff };
+ static const uchar dense3_pat[] = { 0x55, 0xbb, 0x55, 0xee, 0x55, 0xbb, 0x55, 0xee };
+ static const uchar dense4_pat[] = { 0x55, 0xaa, 0x55, 0xaa, 0x55, 0xaa, 0x55, 0xaa };
+ static const uchar dense5_pat[] = { 0xaa, 0x44, 0xaa, 0x11, 0xaa, 0x44, 0xaa, 0x11 };
+ static const uchar dense6_pat[] = { 0x88, 0x00, 0x22, 0x00, 0x88, 0x00, 0x22, 0x00 };
+ static const uchar dense7_pat[] = { 0x00, 0x44, 0x00, 0x00, 0x00, 0x44, 0x00, 0x00 };
+ static const uchar hor_pat[] = { // horizontal pattern
+ 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0xff, 0xff, 0xff,
+ 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
+ 0x00, 0x00, 0x00, 0xff, 0xff, 0xff, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
+ 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0xff, 0xff, 0xff,
+ 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
+ 0x00, 0x00, 0x00, 0xff, 0xff, 0xff, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00 };
+ static const uchar ver_pat[] = { // vertical pattern
+ 0x08, 0x82, 0x20, 0x08, 0x82, 0x20, 0x08, 0x82, 0x20, 0x08, 0x82, 0x20,
+ 0x08, 0x82, 0x20, 0x08, 0x82, 0x20, 0x08, 0x82, 0x20, 0x08, 0x82, 0x20,
+ 0x08, 0x82, 0x20, 0x08, 0x82, 0x20, 0x08, 0x82, 0x20, 0x08, 0x82, 0x20,
+ 0x08, 0x82, 0x20, 0x08, 0x82, 0x20, 0x08, 0x82, 0x20, 0x08, 0x82, 0x20,
+ 0x08, 0x82, 0x20, 0x08, 0x82, 0x20, 0x08, 0x82, 0x20, 0x08, 0x82, 0x20,
+ 0x08, 0x82, 0x20, 0x08, 0x82, 0x20, 0x08, 0x82, 0x20, 0x08, 0x82, 0x20 };
+ static const uchar cross_pat[] = { // cross pattern
+ 0x08, 0x82, 0x20, 0x08, 0x82, 0x20, 0x08, 0x82, 0x20, 0xff, 0xff, 0xff,
+ 0x08, 0x82, 0x20, 0x08, 0x82, 0x20, 0x08, 0x82, 0x20, 0x08, 0x82, 0x20,
+ 0x08, 0x82, 0x20, 0xff, 0xff, 0xff, 0x08, 0x82, 0x20, 0x08, 0x82, 0x20,
+ 0x08, 0x82, 0x20, 0x08, 0x82, 0x20, 0x08, 0x82, 0x20, 0xff, 0xff, 0xff,
+ 0x08, 0x82, 0x20, 0x08, 0x82, 0x20, 0x08, 0x82, 0x20, 0x08, 0x82, 0x20,
+ 0x08, 0x82, 0x20, 0xff, 0xff, 0xff, 0x08, 0x82, 0x20, 0x08, 0x82, 0x20 };
+ static const uchar bdiag_pat[] = { // backward diagonal pattern
+ 0x20, 0x20, 0x10, 0x10, 0x08, 0x08, 0x04, 0x04, 0x02, 0x02, 0x01, 0x01,
+ 0x80, 0x80, 0x40, 0x40, 0x20, 0x20, 0x10, 0x10, 0x08, 0x08, 0x04, 0x04,
+ 0x02, 0x02, 0x01, 0x01, 0x80, 0x80, 0x40, 0x40 };
+ static const uchar fdiag_pat[] = { // forward diagonal pattern
+ 0x02, 0x02, 0x04, 0x04, 0x08, 0x08, 0x10, 0x10, 0x20, 0x20, 0x40, 0x40,
+ 0x80, 0x80, 0x01, 0x01, 0x02, 0x02, 0x04, 0x04, 0x08, 0x08, 0x10, 0x10,
+ 0x20, 0x20, 0x40, 0x40, 0x80, 0x80, 0x01, 0x01 };
+ static const uchar dcross_pat[] = { // diagonal cross pattern
+ 0x22, 0x22, 0x14, 0x14, 0x08, 0x08, 0x14, 0x14, 0x22, 0x22, 0x41, 0x41,
+ 0x80, 0x80, 0x41, 0x41, 0x22, 0x22, 0x14, 0x14, 0x08, 0x08, 0x14, 0x14,
+ 0x22, 0x22, 0x41, 0x41, 0x80, 0x80, 0x41, 0x41 };
+ static const uchar * const pat_tbl[] = {
+ dense1_pat, dense2_pat, dense3_pat, dense4_pat, dense5_pat,
+ dense6_pat, dense7_pat,
+ hor_pat, ver_pat, cross_pat, bdiag_pat, fdiag_pat, dcross_pat };
+
+ if ( testf(ExtDev) ) {
+ QPDevCmdParam param[1];
+ param[0].brush = &cbrush;
+ if ( !pdev->cmd( QPaintDevice::PdcSetBrush, this, param ) || !hd )
+ return;
+ }
+
+ int bs = cbrush.style();
+ bool cacheIt = !testf(ClipOn|MonoDev|NoCache) &&
+ (bs == NoBrush || bs == SolidPattern) &&
+ bro.x() == 0 && bro.y() == 0 && rop == CopyROP;
+
+ bool obtained = FALSE;
+ bool internclipok = hasClipping();
+ if ( cacheIt ) {
+ if ( gc_brush ) {
+ if ( brushRef )
+ release_gc( brushRef );
+ else
+ free_gc( dpy, gc_brush );
+ }
+ obtained = obtain_gc(&brushRef, &gc_brush, cbrush.color().pixel(scrn), dpy,
+ scrn, hd, clip_serial);
+ if ( !obtained && !brushRef )
+ gc_brush = alloc_gc( dpy, scrn, hd, FALSE );
+ } else {
+ if ( gc_brush ) {
+ if ( brushRef ) {
+ release_gc( brushRef );
+ brushRef = 0;
+ gc_brush = alloc_gc( dpy, scrn, hd, testf(MonoDev) );
+ } else {
+ internclipok = TRUE;
+ }
+ } else {
+ gc_brush = alloc_gc( dpy, scrn, hd, testf(MonoDev), testf(UsePrivateCx));
+ }
+ }
+
+ if ( !internclipok ) {
+ if ( pdev == paintEventDevice && paintEventClipRegion ) {
+ if ( brushRef &&((QGCC*)brushRef)->clip_serial < gc_cache_clip_serial ) {
+ x11SetClipRegion( dpy, gc_brush, 0, rendhd, *paintEventClipRegion );
+ ((QGCC*)brushRef)->clip_serial = gc_cache_clip_serial;
+ } else if ( !brushRef ){
+ x11SetClipRegion( dpy, gc_brush, 0, rendhd, *paintEventClipRegion );
+ }
+ } else if (brushRef && ((QGCC*)brushRef)->clip_serial ) {
+ x11ClearClipRegion(dpy, gc_brush, 0, rendhd);
+ ((QGCC*)brushRef)->clip_serial = 0;
+ }
+ }
+
+ if ( obtained )
+ return;
+
+ const uchar *pat = 0; // pattern
+ int d = 0; // defalt pattern size: d*d
+ int s = FillSolid;
+ if ( bs >= Dense1Pattern && bs <= DiagCrossPattern ) {
+ pat = pat_tbl[ bs-Dense1Pattern ];
+ if ( bs <= Dense7Pattern )
+ d = 8;
+ else if ( bs <= CrossPattern )
+ d = 24;
+ else
+ d = 16;
+ }
+
+ XSetLineAttributes( dpy, gc_brush, 0, LineSolid, CapButt, JoinMiter );
+ XSetForeground( dpy, gc_brush, cbrush.color().pixel(scrn) );
+ XSetBackground( dpy, gc_brush, bg_col.pixel(scrn) );
+
+ if ( bs == CustomPattern || pat ) {
+ QPixmap *pm;
+ if ( pat ) {
+ QString key;
+ key.sprintf( "$qt-brush$%d", bs );
+ pm = QPixmapCache::find( key );
+ bool del = FALSE;
+ if ( !pm ) { // not already in pm dict
+ pm = new QBitmap( d, d, pat, TRUE );
+ Q_CHECK_PTR( pm );
+ del = !QPixmapCache::insert( key, pm );
+ }
+ if ( cbrush.data->pixmap )
+ delete cbrush.data->pixmap;
+ cbrush.data->pixmap = new QPixmap( *pm );
+ if (del) delete pm;
+ }
+ pm = cbrush.data->pixmap;
+ pm->x11SetScreen( scrn );
+ if ( pm->depth() == 1 ) {
+ XSetStipple( dpy, gc_brush, pm->handle() );
+ s = bg_mode == TransparentMode ? FillStippled : FillOpaqueStippled;
+ } else {
+ XSetTile( dpy, gc_brush, pm->handle() );
+ s = FillTiled;
+ }
+ }
+ XSetFillStyle( dpy, gc_brush, s );
+}
+
+
+/*!
+ Begins painting the paint device \a pd and returns TRUE if
+ successful; otherwise returns FALSE. If \a unclipped is TRUE, the
+ painting will not be clipped at the paint device's boundaries,
+ (although this is not supported by all platforms).
+
+ The errors that can occur are serious problems, such as these:
+
+ \code
+ p->begin( 0 ); // impossible - paint device cannot be 0
+
+ QPixmap pm( 0, 0 );
+ p->begin( pm ); // impossible - pm.isNull();
+
+ p->begin( myWidget );
+ p2->begin( myWidget ); // impossible - only one painter at a time
+ \endcode
+
+ Note that most of the time, you can use one of the constructors
+ instead of begin(), and that end() is automatically done at
+ destruction.
+
+ \warning A paint device can only be painted by one painter at a
+ time.
+
+ \sa end(), flush()
+*/
+
+bool QPainter::begin( const QPaintDevice *pd, bool unclipped )
+{
+ if ( isActive() ) { // already active painting
+#if defined(QT_CHECK_STATE)
+ qWarning( "QPainter::begin: Painter is already active."
+ "\n\tYou must end() the painter before a second begin()" );
+#endif
+ return FALSE;
+ }
+ if ( pd == 0 ) {
+#if defined(QT_CHECK_NULL)
+ qWarning( "QPainter::begin: Paint device cannot be null" );
+#endif
+ return FALSE;
+ }
+
+ QPixmap::x11SetDefaultScreen( pd->x11Screen() );
+
+ const QWidget *copyFrom = 0;
+ pdev = redirect( (QPaintDevice*)pd );
+ if ( pdev ) { // redirected paint device?
+ if ( pd->devType() == QInternal::Widget )
+ copyFrom = (const QWidget *)pd; // copy widget settings
+ } else {
+ pdev = (QPaintDevice*)pd;
+ }
+
+ if ( pdev->isExtDev() && pdev->paintingActive() ) {
+ // somebody else is already painting
+#if defined(QT_CHECK_STATE)
+ qWarning( "QPainter::begin: Another QPainter is already painting "
+ "this device;\n\tAn extended paint device can only be "
+ "painted by one QPainter at a time." );
+#endif
+ return FALSE;
+ }
+
+ bool reinit = flags != IsStartingUp; // 2nd or 3rd etc. time called
+ flags = IsActive | DirtyFont; // init flags
+ int dt = pdev->devType(); // get the device type
+
+ if ( (pdev->devFlags & QInternal::ExternalDevice) != 0 )
+ setf(ExtDev);
+ else if ( dt == QInternal::Pixmap ) // device is a pixmap
+ ((QPixmap*)pdev)->detach(); // will modify it
+
+ dpy = pdev->x11Display(); // get display variable
+ scrn = pdev->x11Screen(); // get screen variable
+ hd = pdev->handle(); // get handle to drawable
+ rendhd = pdev->rendhd;
+
+ if ( testf(ExtDev) ) { // external device
+ if ( !pdev->cmd( QPaintDevice::PdcBegin, this, 0 ) ) {
+ // could not begin painting
+ if ( reinit )
+ clearf( IsActive | DirtyFont );
+ else
+ flags = IsStartingUp;
+ pdev = 0;
+ return FALSE;
+ }
+ if ( tabstops ) // update tabstops for device
+ setTabStops( tabstops );
+ if ( tabarray ) // update tabarray for device
+ setTabArray( tabarray );
+ }
+
+ if ( pdev->x11Depth() != pdev->x11AppDepth( scrn ) ) { // non-standard depth
+ setf(NoCache);
+ setf(UsePrivateCx);
+ }
+
+ pdev->painters++; // also tell paint device
+ bro = curPt = QPoint( 0, 0 );
+ if ( reinit ) {
+ bg_mode = TransparentMode; // default background mode
+ rop = CopyROP; // default ROP
+ wxmat.reset(); // reset world xform matrix
+ xmat.reset();
+ ixmat.reset();
+ txop = txinv = 0;
+ if ( dt != QInternal::Widget ) {
+ QFont defaultFont; // default drawing tools
+ QPen defaultPen;
+ QBrush defaultBrush;
+ cfont = defaultFont; // set these drawing tools
+ cpen = defaultPen;
+ cbrush = defaultBrush;
+ bg_col = white; // default background color
+ }
+ }
+ wx = wy = vx = vy = 0; // default view origins
+
+ if ( dt == QInternal::Widget ) { // device is a widget
+ QWidget *w = (QWidget*)pdev;
+ cfont = w->font(); // use widget font
+ cpen = QPen( w->foregroundColor() ); // use widget fg color
+ if ( reinit ) {
+ QBrush defaultBrush;
+ cbrush = defaultBrush;
+ }
+ bg_col = w->backgroundColor(); // use widget bg color
+ ww = vw = w->width(); // default view size
+ wh = vh = w->height();
+ if ( unclipped || w->testWFlags( WPaintUnclipped ) ) { // paint direct on device
+ setf( NoCache );
+ setf(UsePrivateCx);
+ updatePen();
+ updateBrush();
+ XSetSubwindowMode( dpy, gc, IncludeInferiors );
+ XSetSubwindowMode( dpy, gc_brush, IncludeInferiors );
+#ifndef QT_NO_XFTFREETYPE
+ if (rendhd)
+ XftDrawSetSubwindowMode((XftDraw *) rendhd, IncludeInferiors);
+#endif
+ }
+ } else if ( dt == QInternal::Pixmap ) { // device is a pixmap
+ QPixmap *pm = (QPixmap*)pdev;
+ if ( pm->isNull() ) {
+#if defined(QT_CHECK_NULL)
+ qWarning( "QPainter::begin: Cannot paint null pixmap" );
+#endif
+ end();
+ return FALSE;
+ }
+ bool mono = pm->depth() == 1; // monochrome bitmap
+ if ( mono ) {
+ setf( MonoDev );
+ bg_col = color0;
+ cpen.setColor( color1 );
+ }
+ ww = vw = pm->width(); // default view size
+ wh = vh = pm->height();
+ } else if ( testf(ExtDev) ) { // external device
+ ww = vw = pdev->metric( QPaintDeviceMetrics::PdmWidth );
+ wh = vh = pdev->metric( QPaintDeviceMetrics::PdmHeight );
+ }
+ if ( ww == 0 )
+ ww = wh = vw = vh = 1024;
+ if ( copyFrom ) { // copy redirected widget
+ cfont = copyFrom->font();
+ cpen = QPen( copyFrom->foregroundColor() );
+ bg_col = copyFrom->backgroundColor();
+ }
+ if ( testf(ExtDev) ) { // external device
+ setBackgroundColor( bg_col ); // default background color
+ setBackgroundMode( TransparentMode ); // default background mode
+ setRasterOp( CopyROP ); // default raster operation
+ }
+ clip_serial = gc_cache_clip_serial++;
+ updateBrush();
+ updatePen();
+ return TRUE;
+}
+
+/*!
+ Ends painting. Any resources used while painting are released.
+
+ Note that while you mostly don't need to call end(), the
+ destructor will do it, there is at least one common case when it
+ is needed, namely double buffering.
+
+ \code
+ QPainter p( myPixmap, this )
+ // ...
+ p.end(); // stops drawing on myPixmap
+ p.begin( this );
+ p.drawPixmap( 0, 0, myPixmap );
+ \endcode
+
+ Since you can't draw a QPixmap while it is being painted, it is
+ necessary to close the active painter.
+
+ \sa begin(), isActive()
+*/
+
+bool QPainter::end() // end painting
+{
+ if ( !isActive() ) {
+#if defined(QT_CHECK_STATE)
+ qWarning( "QPainter::end: Missing begin() or begin() failed" );
+#endif
+ return FALSE;
+ }
+ killPStack();
+
+ //#### This should not be necessary:
+ if ( pdev->devType() == QInternal::Widget && // #####
+ ((QWidget*)pdev)->testWFlags(WPaintUnclipped) ) {
+ if ( gc )
+ XSetSubwindowMode( dpy, gc, ClipByChildren );
+ if ( gc_brush )
+ XSetSubwindowMode( dpy, gc_brush, ClipByChildren );
+ }
+
+ if ( gc_brush ) { // restore brush gc
+ if ( brushRef ) {
+ release_gc( brushRef );
+ brushRef = 0;
+ } else {
+ free_gc( dpy, gc_brush, testf(UsePrivateCx) );
+ }
+ gc_brush = 0;
+
+ }
+ if ( gc ) { // restore pen gc
+ if ( penRef ) {
+ release_gc( penRef );
+ penRef = 0;
+ } else {
+ free_gc( dpy, gc, testf(UsePrivateCx) );
+ }
+ gc = 0;
+ }
+
+ if ( testf(ExtDev) )
+ pdev->cmd( QPaintDevice::PdcEnd, this, 0 );
+
+#ifndef QT_NO_XFTFREETYPE
+ if (rendhd) {
+ // reset clipping/subwindow mode on our render picture
+ XftDrawSetClip((XftDraw *) rendhd, None);
+ XftDrawSetSubwindowMode((XftDraw *) rendhd, ClipByChildren);
+ }
+#endif // QT_NO_XFTFREETYPE
+
+ if ( pfont ) {
+ delete pfont;
+ pfont = 0;
+ }
+
+ flags = 0;
+ pdev->painters--;
+ pdev = 0;
+ dpy = 0;
+ return TRUE;
+}
+
+/*!
+ Flushes any buffered drawing operations inside the region \a
+ region using clipping mode \a cm.
+
+ The flush may update the whole device if the platform does not
+ support flushing to a specified region.
+
+ \sa flush() CoordinateMode
+*/
+
+void QPainter::flush(const QRegion &, CoordinateMode)
+{
+ flush();
+}
+
+
+/*!
+ \overload
+
+ Flushes any buffered drawing operations.
+*/
+
+void QPainter::flush()
+{
+ if ( isActive() && dpy )
+ XFlush( dpy );
+}
+
+
+/*!
+ Sets the background color of the painter to \a c.
+
+ The background color is the color that is filled in when drawing
+ opaque text, stippled lines and bitmaps. The background color has
+ no effect in transparent background mode (which is the default).
+
+ \sa backgroundColor() setBackgroundMode() BackgroundMode
+*/
+
+void QPainter::setBackgroundColor( const QColor &c )
+{
+ if ( !isActive() ) {
+#if defined(QT_CHECK_STATE)
+ qWarning( "QPainter::setBackgroundColor: Call begin() first" );
+#endif
+ return;
+ }
+ bg_col = c;
+ if ( testf(ExtDev) ) {
+ QPDevCmdParam param[1];
+ param[0].color = &bg_col;
+ if ( !pdev->cmd( QPaintDevice::PdcSetBkColor, this, param ) || !hd )
+ return;
+ }
+ if ( !penRef )
+ updatePen(); // update pen setting
+ if ( !brushRef )
+ updateBrush(); // update brush setting
+}
+
+/*!
+ Sets the background mode of the painter to \a m, which must be
+ either \c TransparentMode (the default) or \c OpaqueMode.
+
+ Transparent mode draws stippled lines and text without setting the
+ background pixels. Opaque mode fills these space with the current
+ background color.
+
+ Note that in order to draw a bitmap or pixmap transparently, you
+ must use QPixmap::setMask().
+
+ \sa backgroundMode(), setBackgroundColor()
+*/
+
+void QPainter::setBackgroundMode( BGMode m )
+{
+ if ( !isActive() ) {
+#if defined(QT_CHECK_STATE)
+ qWarning( "QPainter::setBackgroundMode: Call begin() first" );
+#endif
+ return;
+ }
+ if ( m != TransparentMode && m != OpaqueMode ) {
+#if defined(QT_CHECK_RANGE)
+ qWarning( "QPainter::setBackgroundMode: Invalid mode" );
+#endif
+ return;
+ }
+ bg_mode = m;
+ if ( testf(ExtDev) ) {
+ QPDevCmdParam param[1];
+ param[0].ival = m;
+ if ( !pdev->cmd( QPaintDevice::PdcSetBkMode, this, param ) || !hd )
+ return;
+ }
+ if ( !penRef )
+ updatePen(); // update pen setting
+ if ( !brushRef )
+ updateBrush(); // update brush setting
+}
+
+static const short ropCodes[] = { // ROP translation table
+ GXcopy, // CopyROP
+ GXor, // OrROP
+ GXxor, // XorROP
+ GXandInverted, // NotAndROP EraseROP
+ GXcopyInverted, // NotCopyROP
+ GXorInverted, // NotOrROP
+ GXequiv, // NotXorROP
+ GXand, // AndROP
+ GXinvert, // NotROP
+ GXclear, // ClearROP
+ GXset, // SetROP
+ GXnoop, // NopROP
+ GXandReverse, // AndNotROP
+ GXorReverse, // OrNotROP
+ GXnand, // NandROP
+ GXnor // NorROP
+};
+
+
+/*!
+ Sets the \link Qt::RasterOp raster operation \endlink to \a r.
+ The default is \c CopyROP.
+
+ \sa rasterOp() Qt::RasterOp
+*/
+
+void QPainter::setRasterOp( RasterOp r )
+{
+ if ( !isActive() ) {
+#if defined(QT_CHECK_STATE)
+ qWarning( "QPainter::setRasterOp: Call begin() first" );
+#endif
+ return;
+ }
+ if ( (uint)r > LastROP ) {
+#if defined(QT_CHECK_RANGE)
+ qWarning( "QPainter::setRasterOp: Invalid ROP code" );
+#endif
+ return;
+ }
+ rop = r;
+ if ( testf(ExtDev) ) {
+ QPDevCmdParam param[1];
+ param[0].ival = r;
+ if ( !pdev->cmd( QPaintDevice::PdcSetROP, this, param ) || !hd )
+ return;
+ }
+ if ( penRef )
+ updatePen(); // get non-cached pen GC
+ if ( brushRef )
+ updateBrush(); // get non-cached brush GC
+ XSetFunction( dpy, gc, ropCodes[rop] );
+ XSetFunction( dpy, gc_brush, ropCodes[rop] );
+}
+
+// ### matthias - true?
+
+/*!
+ Sets the brush origin to \a (x, y).
+
+ The brush origin specifies the (0, 0) coordinate of the painter's
+ brush. This setting only applies to pattern brushes and pixmap
+ brushes.
+
+ \sa brushOrigin()
+*/
+
+void QPainter::setBrushOrigin( int x, int y )
+{
+ if ( !isActive() ) {
+#if defined(QT_CHECK_STATE)
+ qWarning( "QPainter::setBrushOrigin: Call begin() first" );
+#endif
+ return;
+ }
+ bro = QPoint(x, y);
+ if ( testf(ExtDev) ) {
+ QPDevCmdParam param[1];
+ param[0].point = &bro;
+ if ( !pdev->cmd( QPaintDevice::PdcSetBrushOrigin, this, param ) ||
+ !hd )
+ return;
+ }
+ if ( brushRef )
+ updateBrush(); // get non-cached brush GC
+ XSetTSOrigin( dpy, gc_brush, x, y );
+}
+
+
+/*!
+ Enables clipping if \a enable is TRUE, or disables clipping if \a
+ enable is FALSE.
+
+ \sa hasClipping(), setClipRect(), setClipRegion()
+*/
+
+void QPainter::setClipping( bool enable )
+{
+ if ( !isActive() ) {
+#if defined(QT_CHECK_STATE)
+ qWarning( "QPainter::setClipping: Will be reset by begin()" );
+#endif
+ return;
+ }
+
+ if ( enable == testf(ClipOn) )
+ return;
+
+ setf( ClipOn, enable );
+ if ( testf(ExtDev) ) {
+ if ( block_ext )
+ return;
+ QPDevCmdParam param[1];
+ param[0].ival = enable;
+ if ( !pdev->cmd( QPaintDevice::PdcSetClip, this, param ) || !hd )
+ return;
+ }
+ if ( enable ) {
+ QRegion rgn = crgn;
+ if ( pdev == paintEventDevice && paintEventClipRegion )
+ rgn = rgn.intersect( *paintEventClipRegion );
+ if ( penRef )
+ updatePen();
+ if ( brushRef )
+ updateBrush();
+ x11SetClipRegion( dpy, gc, gc_brush, rendhd, rgn );
+ } else {
+ if ( pdev == paintEventDevice && paintEventClipRegion ) {
+ x11SetClipRegion( dpy, gc, gc_brush , rendhd, *paintEventClipRegion );
+ } else {
+ x11ClearClipRegion(dpy, gc, gc_brush, rendhd);
+ }
+ }
+}
+
+
+/*!
+ \overload
+
+ Sets the clip region to the rectangle \a r and enables clipping.
+ The clip mode is set to \a m.
+
+ \sa CoordinateMode
+*/
+
+void QPainter::setClipRect( const QRect &r, CoordinateMode m )
+{
+ setClipRegion( QRegion( r ), m );
+}
+
+/*!
+ Sets the clip region to \a rgn and enables clipping. The clip mode
+ is set to \a m.
+
+ Note that the clip region is given in physical device coordinates
+ and \e not subject to any \link coordsys.html coordinate
+ transformation.\endlink
+
+ \sa setClipRect(), clipRegion(), setClipping() CoordinateMode
+*/
+
+void QPainter::setClipRegion( const QRegion &rgn, CoordinateMode m )
+{
+#if defined(QT_CHECK_STATE)
+ if ( !isActive() )
+ qWarning( "QPainter::setClipRegion: Will be reset by begin()" );
+#endif
+ if ( m == CoordDevice )
+ crgn = rgn;
+ else
+ crgn = xmat * rgn;
+
+ if ( testf(ExtDev) ) {
+ if ( block_ext )
+ return;
+ QPDevCmdParam param[2];
+ param[0].rgn = &rgn;
+ param[1].ival = m;
+ if ( !pdev->cmd( QPaintDevice::PdcSetClipRegion, this, param ) )
+ return; // device cannot clip
+ }
+ clearf( ClipOn ); // be sure to update clip rgn
+ setClipping( TRUE );
+}
+
+
+/*!
+ \internal
+
+ Internal function for drawing a polygon.
+*/
+
+void QPainter::drawPolyInternal( const QPointArray &a, bool close )
+{
+ if ( a.size() < 2 )
+ return;
+
+ int x1, y1, x2, y2; // connect last to first point
+ a.point( a.size()-1, &x1, &y1 );
+ a.point( 0, &x2, &y2 );
+ bool do_close = close && !(x1 == x2 && y1 == y2);
+
+ if ( close && cbrush.style() != NoBrush ) { // draw filled polygon
+ XFillPolygon( dpy, hd, gc_brush, (XPoint*)a.shortPoints(), a.size(),
+ Nonconvex, CoordModeOrigin );
+ if ( cpen.style() == NoPen ) { // draw fake outline
+ XDrawLines( dpy, hd, gc_brush, (XPoint*)a.shortPoints(), a.size(),
+ CoordModeOrigin );
+ if ( do_close )
+ XDrawLine( dpy, hd, gc_brush, x1, y1, x2, y2 );
+ }
+ }
+ if ( cpen.style() != NoPen ) { // draw outline
+ XDrawLines( dpy, hd, gc, (XPoint*)a.shortPoints(), a.size(),
+ CoordModeOrigin);
+ if ( do_close )
+ XDrawLine( dpy, hd, gc, x1, y1, x2, y2 );
+ }
+}
+
+
+/*!
+ Draws/plots a single point at \a (x, y) using the current pen.
+
+ \sa QPen
+*/
+
+void QPainter::drawPoint( int x, int y )
+{
+ if ( !isActive() )
+ return;
+ if ( testf(ExtDev|VxF|WxF) ) {
+ if ( testf(ExtDev) ) {
+ QPDevCmdParam param[1];
+ QPoint p( x, y );
+ param[0].point = &p;
+ if ( !pdev->cmd( QPaintDevice::PdcDrawPoint, this, param ) ||
+ !hd )
+ return;
+ }
+ map( x, y, &x, &y );
+ }
+ if ( cpen.style() != NoPen )
+ XDrawPoint( dpy, hd, gc, x, y );
+}
+
+
+/*!
+ Draws/plots an array of points, \a a, using the current pen.
+
+ If \a index is non-zero (the default is zero) only points from \a
+ index are drawn. If \a npoints is negative (the default) the rest
+ of the points from \a index are drawn. If \a npoints is zero or
+ greater, \a npoints points are drawn.
+
+ \warning On X11, coordinates that do not fit into 16-bit signed
+ values are truncated. This limitation is expected to go away in
+ Qt 4.
+*/
+
+void QPainter::drawPoints( const QPointArray& a, int index, int npoints )
+{
+ if ( npoints < 0 )
+ npoints = a.size() - index;
+ if ( index + npoints > (int)a.size() )
+ npoints = a.size() - index;
+ if ( !isActive() || npoints < 1 || index < 0 )
+ return;
+ QPointArray pa = a;
+ if ( testf(ExtDev|VxF|WxF) ) {
+ if ( testf(ExtDev) ) {
+ QPDevCmdParam param[1];
+ for (int i=0; i<npoints; i++) {
+ QPoint p( pa[index+i].x(), pa[index+i].y() );
+ param[0].point = &p;
+ if ( !pdev->cmd( QPaintDevice::PdcDrawPoint, this, param ))
+ return;
+ }
+ if ( !hd ) return;
+ }
+ if ( txop != TxNone ) {
+ pa = xForm( a, index, npoints );
+ if ( pa.size() != a.size() ) {
+ index = 0;
+ npoints = pa.size();
+ }
+ }
+ }
+ if ( cpen.style() != NoPen )
+ XDrawPoints( dpy, hd, gc, (XPoint*)(pa.shortPoints( index, npoints )),
+ npoints, CoordModeOrigin );
+}
+
+
+/*! \obsolete
+ Sets the current pen position to \a (x, y)
+
+ \sa lineTo(), pos()
+*/
+
+void QPainter::moveTo( int x, int y )
+{
+ if ( !isActive() )
+ return;
+ if ( testf(ExtDev|VxF|WxF) ) {
+ if ( testf(ExtDev) ) {
+ QPDevCmdParam param[1];
+ QPoint p( x, y );
+ param[0].point = &p;
+ if ( !pdev->cmd( QPaintDevice::PdcMoveTo, this, param ) || !hd )
+ return;
+ }
+ }
+ curPt = QPoint( x, y );
+}
+
+/*! \obsolete
+ Use drawLine() instead.
+
+ Draws a line from the current pen position to \a (x, y) and sets
+ \a (x, y) to be the new current pen position.
+
+ \sa QPen moveTo(), drawLine(), pos()
+*/
+
+void QPainter::lineTo( int x, int y )
+{
+ if ( !isActive() )
+ return;
+ int cx = curPt.x(), cy = curPt.y();
+ curPt = QPoint( x, y );
+ if ( testf(ExtDev|VxF|WxF) ) {
+ if ( testf(ExtDev) ) {
+ QPDevCmdParam param[1];
+ QPoint p( x, y );
+ param[0].point = &p;
+ if ( !pdev->cmd( QPaintDevice::PdcLineTo, this, param ) || !hd )
+ return;
+ }
+ map( x, y, &x, &y );
+ map( cx, cy, &cx, &cy );
+ }
+ if ( cpen.style() != NoPen )
+ XDrawLine( dpy, hd, gc, cx, cy, x, y );
+}
+
+/*!
+ Draws a line from (\a x1, \a y1) to (\a x2, \a y2) and sets the
+ current pen position to (\a x2, \a y2).
+
+ \sa pen()
+*/
+
+void QPainter::drawLine( int x1, int y1, int x2, int y2 )
+{
+ if ( !isActive() )
+ return;
+ curPt = QPoint( x2, y2 );
+ if ( testf(ExtDev|VxF|WxF) ) {
+ if ( testf(ExtDev) ) {
+ QPDevCmdParam param[2];
+ QPoint p1(x1, y1), p2(x2, y2);
+ param[0].point = &p1;
+ param[1].point = &p2;
+ if ( !pdev->cmd( QPaintDevice::PdcDrawLine, this, param ) || !hd )
+ return;
+ }
+ map( x1, y1, &x1, &y1 );
+ map( x2, y2, &x2, &y2 );
+ }
+ if ( cpen.style() != NoPen )
+ XDrawLine( dpy, hd, gc, x1, y1, x2, y2 );
+}
+
+
+
+/*!
+ Draws a rectangle with upper left corner at \a (x, y) and with
+ width \a w and height \a h.
+
+ \sa QPen, drawRoundRect()
+*/
+
+void QPainter::drawRect( int x, int y, int w, int h )
+{
+ if ( !isActive() )
+ return;
+ if ( testf(ExtDev|VxF|WxF) ) {
+ if ( testf(ExtDev) ) {
+ QPDevCmdParam param[1];
+ QRect r( x, y, w, h );
+ param[0].rect = &r;
+ if ( !pdev->cmd( QPaintDevice::PdcDrawRect, this, param ) || !hd )
+ return;
+ }
+ if ( txop == TxRotShear ) { // rotate/shear polygon
+ QPointArray pa = xmat.mapToPolygon( QRect(x, y, w, h) );
+ pa.resize( 5 );
+ pa.setPoint( 4, pa.point( 0 ) );
+ drawPolyInternal( pa );
+ return;
+ }
+ map( x, y, w, h, &x, &y, &w, &h );
+ }
+ if ( w <= 0 || h <= 0 ) {
+ if ( w == 0 || h == 0 )
+ return;
+ fix_neg_rect( &x, &y, &w, &h );
+ }
+ if ( cbrush.style() != NoBrush ) {
+ if ( cpen.style() == NoPen ) {
+ XFillRectangle( dpy, hd, gc_brush, x, y, w, h );
+ return;
+ }
+ int lw = cpen.width();
+ int lw2 = (lw+1)/2;
+ if ( w > lw && h > lw )
+ XFillRectangle( dpy, hd, gc_brush, x+lw2, y+lw2, w-lw-1, h-lw-1 );
+ }
+ if ( cpen.style() != NoPen )
+ XDrawRectangle( dpy, hd, gc, x, y, w-1, h-1 );
+}
+
+/*!
+ \overload
+
+ Draws a Windows focus rectangle with upper left corner at (\a x,
+ \a y) and with width \a w and height \a h.
+
+ This function draws a stippled XOR rectangle that is used to
+ indicate keyboard focus (when QApplication::style() is \c
+ WindowStyle).
+
+ \warning This function draws nothing if the coordinate system has
+ been \link rotate() rotated\endlink or \link shear()
+ sheared\endlink.
+
+ \sa drawRect(), QApplication::style()
+*/
+
+void QPainter::drawWinFocusRect( int x, int y, int w, int h )
+{
+ drawWinFocusRect( x, y, w, h, TRUE, color0 );
+}
+
+/*!
+ Draws a Windows focus rectangle with upper left corner at (\a x,
+ \a y) and with width \a w and height \a h using a pen color that
+ contrasts with \a bgColor.
+
+ This function draws a stippled rectangle (XOR is not used) that is
+ used to indicate keyboard focus (when the QApplication::style() is
+ \c WindowStyle).
+
+ The pen color used to draw the rectangle is either white or black
+ depending on the color of \a bgColor (see QColor::gray()).
+
+ \warning This function draws nothing if the coordinate system has
+ been \link rotate() rotated\endlink or \link shear()
+ sheared\endlink.
+
+ \sa drawRect(), QApplication::style()
+*/
+
+void QPainter::drawWinFocusRect( int x, int y, int w, int h,
+ const QColor &bgColor )
+{
+ drawWinFocusRect( x, y, w, h, FALSE, bgColor );
+}
+
+
+/*!
+ \internal
+*/
+
+void QPainter::drawWinFocusRect( int x, int y, int w, int h,
+ bool xorPaint, const QColor &bgColor )
+{
+ if ( !isActive() || txop == TxRotShear )
+ return;
+ static char winfocus_line[] = { 1, 1 };
+
+ QPen old_pen = cpen;
+ RasterOp old_rop = (RasterOp)rop;
+
+ if ( xorPaint ) {
+ if ( QColor::numBitPlanes() <= 8 )
+ setPen( color1 );
+ else
+ setPen( white );
+ setRasterOp( XorROP );
+ } else {
+ if ( qGray( bgColor.rgb() ) < 128 )
+ setPen( white );
+ else
+ setPen( black );
+ }
+
+ if ( testf(ExtDev|VxF|WxF) ) {
+ if ( testf(ExtDev) ) {
+ QPDevCmdParam param[1];
+ QRect r( x, y, w, h );
+ param[0].rect = &r;
+ if ( !pdev->cmd( QPaintDevice::PdcDrawRect, this, param ) || !hd) {
+ setRasterOp( old_rop );
+ setPen( old_pen );
+ return;
+ }
+ }
+ map( x, y, w, h, &x, &y, &w, &h );
+ }
+ if ( w <= 0 || h <= 0 ) {
+ if ( w == 0 || h == 0 )
+ return;
+ fix_neg_rect( &x, &y, &w, &h );
+ }
+ XSetDashes( dpy, gc, 0, winfocus_line, 2 );
+ XSetLineAttributes( dpy, gc, 1, LineOnOffDash, CapButt, JoinMiter );
+
+ XDrawRectangle( dpy, hd, gc, x, y, w-1, h-1 );
+ XSetLineAttributes( dpy, gc, 0, LineSolid, CapButt, JoinMiter );
+ setRasterOp( old_rop );
+ setPen( old_pen );
+}
+
+
+/*!
+ Draws a rectangle with rounded corners at \a (x, y), with width \a
+ w and height \a h.
+
+ The \a xRnd and \a yRnd arguments specify how rounded the corners
+ should be. 0 is angled corners, 99 is maximum roundedness.
+
+ The width and height include all of the drawn lines.
+
+ \sa drawRect(), QPen
+*/
+
+void QPainter::drawRoundRect( int x, int y, int w, int h, int xRnd, int yRnd )
+{
+ if ( !isActive() )
+ return;
+ if ( xRnd <= 0 || yRnd <= 0 ) {
+ drawRect( x, y, w, h ); // draw normal rectangle
+ return;
+ }
+ if ( xRnd >= 100 ) // fix ranges
+ xRnd = 99;
+ if ( yRnd >= 100 )
+ yRnd = 99;
+ if ( testf(ExtDev|VxF|WxF) ) {
+ if ( testf(ExtDev) ) {
+ QPDevCmdParam param[3];
+ QRect r( x, y, w, h );
+ param[0].rect = &r;
+ param[1].ival = xRnd;
+ param[2].ival = yRnd;
+ if ( !pdev->cmd( QPaintDevice::PdcDrawRoundRect, this, param ) ||
+ !hd )
+ return;
+ }
+ if ( txop == TxRotShear ) { // rotate/shear polygon
+ if ( w <= 0 || h <= 0 )
+ fix_neg_rect( &x, &y, &w, &h );
+ w--;
+ h--;
+ int rxx = w*xRnd/200;
+ int ryy = h*yRnd/200;
+ // were there overflows?
+ if ( rxx < 0 )
+ rxx = w/200*xRnd;
+ if ( ryy < 0 )
+ ryy = h/200*yRnd;
+ int rxx2 = 2*rxx;
+ int ryy2 = 2*ryy;
+ QPointArray a[4];
+ a[0].makeArc( x, y, rxx2, ryy2, 1*16*90, 16*90, xmat );
+ a[1].makeArc( x, y+h-ryy2, rxx2, ryy2, 2*16*90, 16*90, xmat );
+ a[2].makeArc( x+w-rxx2, y+h-ryy2, rxx2, ryy2, 3*16*90, 16*90, xmat );
+ a[3].makeArc( x+w-rxx2, y, rxx2, ryy2, 0*16*90, 16*90, xmat );
+ // ### is there a better way to join QPointArrays?
+ QPointArray aa;
+ aa.resize( a[0].size() + a[1].size() + a[2].size() + a[3].size() );
+ uint j = 0;
+ for ( int k=0; k<4; k++ ) {
+ for ( uint i=0; i<a[k].size(); i++ ) {
+ aa.setPoint( j, a[k].point(i) );
+ j++;
+ }
+ }
+ drawPolyInternal( aa );
+ return;
+ }
+ map( x, y, w, h, &x, &y, &w, &h );
+ }
+ w--;
+ h--;
+ if ( w <= 0 || h <= 0 ) {
+ if ( w == 0 || h == 0 )
+ return;
+ fix_neg_rect( &x, &y, &w, &h );
+ }
+ int rx = (w*xRnd)/200;
+ int ry = (h*yRnd)/200;
+ int rx2 = 2*rx;
+ int ry2 = 2*ry;
+ if ( cbrush.style() != NoBrush ) { // draw filled round rect
+ int dp, ds;
+ if ( cpen.style() == NoPen ) {
+ dp = 0;
+ ds = 1;
+ }
+ else {
+ dp = 1;
+ ds = 0;
+ }
+#define SET_ARC(px, py, w, h, a1, a2) \
+ a->x=px; a->y=py; a->width=w; a->height=h; a->angle1=a1; a->angle2=a2; a++
+ XArc arcs[4];
+ XArc *a = arcs;
+ SET_ARC( x+w-rx2, y, rx2, ry2, 0, 90*64 );
+ SET_ARC( x, y, rx2, ry2, 90*64, 90*64 );
+ SET_ARC( x, y+h-ry2, rx2, ry2, 180*64, 90*64 );
+ SET_ARC( x+w-rx2, y+h-ry2, rx2, ry2, 270*64, 90*64 );
+ XFillArcs( dpy, hd, gc_brush, arcs, 4 );
+#undef SET_ARC
+#define SET_RCT(px, py, w, h) \
+ r->x=px; r->y=py; r->width=w; r->height=h; r++
+ XRectangle rects[3];
+ XRectangle *r = rects;
+ SET_RCT( x+rx, y+dp, w-rx2, ry );
+ SET_RCT( x+dp, y+ry, w+ds, h-ry2 );
+ SET_RCT( x+rx, y+h-ry, w-rx2, ry+ds );
+ XFillRectangles( dpy, hd, gc_brush, rects, 3 );
+#undef SET_RCT
+ }
+ if ( cpen.style() != NoPen ) { // draw outline
+#define SET_ARC(px, py, w, h, a1, a2) \
+ a->x=px; a->y=py; a->width=w; a->height=h; a->angle1=a1; a->angle2=a2; a++
+ XArc arcs[4];
+ XArc *a = arcs;
+ SET_ARC( x+w-rx2, y, rx2, ry2, 0, 90*64 );
+ SET_ARC( x, y, rx2, ry2, 90*64, 90*64 );
+ SET_ARC( x, y+h-ry2, rx2, ry2, 180*64, 90*64 );
+ SET_ARC( x+w-rx2, y+h-ry2, rx2, ry2, 270*64, 90*64 );
+ XDrawArcs( dpy, hd, gc, arcs, 4 );
+#undef SET_ARC
+#define SET_SEG(xp1, yp1, xp2, yp2) \
+ s->x1=xp1; s->y1=yp1; s->x2=xp2; s->y2=yp2; s++
+ XSegment segs[4];
+ XSegment *s = segs;
+ SET_SEG( x+rx, y, x+w-rx, y );
+ SET_SEG( x+rx, y+h, x+w-rx, y+h );
+ SET_SEG( x, y+ry, x, y+h-ry );
+ SET_SEG( x+w, y+ry, x+w, y+h-ry );
+ XDrawSegments( dpy, hd, gc, segs, 4 );
+#undef SET_SET
+ }
+}
+
+/*!
+ Draws an ellipse with center at \a (x + w/2, y + h/2) and size \a
+ (w, h).
+*/
+
+void QPainter::drawEllipse( int x, int y, int w, int h )
+{
+ if ( !isActive() )
+ return;
+ if ( testf(ExtDev|VxF|WxF) ) {
+ if ( testf(ExtDev) ) {
+ QPDevCmdParam param[1];
+ QRect r( x, y, w, h );
+ param[0].rect = &r;
+ if ( !pdev->cmd( QPaintDevice::PdcDrawEllipse, this, param ) ||
+ !hd )
+ return;
+ }
+ if ( txop == TxRotShear ) { // rotate/shear polygon
+ QPointArray a;
+ a.makeArc( x, y, w, h, 0, 360*16, xmat );
+ drawPolyInternal( a );
+ return;
+ }
+ map( x, y, w, h, &x, &y, &w, &h );
+ }
+ if ( w <= 0 || h <= 0 ) {
+ if ( w == 0 || h == 0 )
+ return;
+ fix_neg_rect( &x, &y, &w, &h );
+ }
+ if ( w == 1 && h == 1 ) {
+ XDrawPoint( dpy, hd, (cpen.style() == NoPen)?gc_brush:gc, x, y );
+ return;
+ }
+ w--;
+ h--;
+ if ( cbrush.style() != NoBrush ) { // draw filled ellipse
+ XFillArc( dpy, hd, gc_brush, x, y, w, h, 0, 360*64 );
+ if ( cpen.style() == NoPen ) {
+ XDrawArc( dpy, hd, gc_brush, x, y, w, h, 0, 360*64 );
+ return;
+ }
+ }
+ if ( cpen.style() != NoPen ) // draw outline
+ XDrawArc( dpy, hd, gc, x, y, w, h, 0, 360*64 );
+}
+
+
+/*!
+ Draws an arc defined by the rectangle \a (x, y, w, h), the start
+ angle \a a and the arc length \a alen.
+
+ The angles \a a and \a alen are 1/16th of a degree, i.e. a full
+ circle equals 5760 (16*360). Positive values of \a a and \a alen
+ mean counter-clockwise while negative values mean the clockwise
+ direction. Zero degrees is at the 3 o'clock position.
+
+ Example:
+ \code
+ QPainter p( myWidget );
+ p.drawArc( 10,10, 70,100, 100*16, 160*16 ); // draws a "(" arc
+ \endcode
+
+ \sa drawPie(), drawChord()
+*/
+
+void QPainter::drawArc( int x, int y, int w, int h, int a, int alen )
+{
+ if ( !isActive() )
+ return;
+ if ( testf(ExtDev|VxF|WxF) ) {
+ if ( testf(ExtDev) ) {
+ QPDevCmdParam param[3];
+ QRect r( x, y, w, h );
+ param[0].rect = &r;
+ param[1].ival = a;
+ param[2].ival = alen;
+ if ( !pdev->cmd( QPaintDevice::PdcDrawArc, this, param ) ||
+ !hd )
+ return;
+ }
+ if ( txop == TxRotShear ) { // rotate/shear
+ QPointArray pa;
+ pa.makeArc( x, y, w, h, a, alen, xmat ); // arc polyline
+ drawPolyInternal( pa, FALSE );
+ return;
+ }
+ map( x, y, w, h, &x, &y, &w, &h );
+ }
+ w--;
+ h--;
+ if ( w <= 0 || h <= 0 ) {
+ if ( w == 0 || h == 0 )
+ return;
+ fix_neg_rect( &x, &y, &w, &h );
+ }
+ if ( cpen.style() != NoPen )
+ XDrawArc( dpy, hd, gc, x, y, w, h, a*4, alen*4 );
+}
+
+
+/*!
+ Draws a pie defined by the rectangle \a (x, y, w, h), the start
+ angle \a a and the arc length \a alen.
+
+ The pie is filled with the current brush().
+
+ The angles \a a and \a alen are 1/16th of a degree, i.e. a full
+ circle equals 5760 (16*360). Positive values of \a a and \a alen
+ mean counter-clockwise while negative values mean the clockwise
+ direction. Zero degrees is at the 3 o'clock position.
+
+ \sa drawArc(), drawChord()
+*/
+
+void QPainter::drawPie( int x, int y, int w, int h, int a, int alen )
+{
+ // Make sure "a" is 0..360*16, as otherwise a*4 may overflow 16 bits.
+ if ( a > (360*16) ) {
+ a = a % (360*16);
+ } else if ( a < 0 ) {
+ a = a % (360*16);
+ if ( a < 0 ) a += (360*16);
+ }
+
+ if ( !isActive() )
+ return;
+ if ( testf(ExtDev|VxF|WxF) ) {
+ if ( testf(ExtDev) ) {
+ QPDevCmdParam param[3];
+ QRect r( x, y, w, h );
+ param[0].rect = &r;
+ param[1].ival = a;
+ param[2].ival = alen;
+ if ( !pdev->cmd( QPaintDevice::PdcDrawPie, this, param ) || !hd )
+ return;
+ }
+ if ( txop == TxRotShear ) { // rotate/shear
+ QPointArray pa;
+ pa.makeArc( x, y, w, h, a, alen, xmat ); // arc polyline
+ int n = pa.size();
+ int cx, cy;
+ xmat.map(x+w/2, y+h/2, &cx, &cy);
+ pa.resize( n+2 );
+ pa.setPoint( n, cx, cy ); // add legs
+ pa.setPoint( n+1, pa.at(0) );
+ drawPolyInternal( pa );
+ return;
+ }
+ map( x, y, w, h, &x, &y, &w, &h );
+ }
+ XSetArcMode( dpy, gc_brush, ArcPieSlice );
+ w--;
+ h--;
+ if ( w <= 0 || h <= 0 ) {
+ if ( w == 0 || h == 0 )
+ return;
+ fix_neg_rect( &x, &y, &w, &h );
+ }
+
+ GC g = gc;
+ bool nopen = cpen.style() == NoPen;
+
+ if ( cbrush.style() != NoBrush ) { // draw filled pie
+ XFillArc( dpy, hd, gc_brush, x, y, w, h, a*4, alen*4 );
+ if ( nopen ) {
+ g = gc_brush;
+ nopen = FALSE;
+ }
+ }
+ if ( !nopen ) { // draw pie outline
+ double w2 = 0.5*w; // with, height in ellipsis
+ double h2 = 0.5*h;
+ double xc = (double)x+w2;
+ double yc = (double)y+h2;
+ double ra1 = Q_PI/2880.0*a; // convert a, alen to radians
+ double ra2 = ra1 + Q_PI/2880.0*alen;
+ int xic = qRound(xc);
+ int yic = qRound(yc);
+ XDrawLine( dpy, hd, g, xic, yic,
+ qRound(xc + qcos(ra1)*w2), qRound(yc - qsin(ra1)*h2));
+ XDrawLine( dpy, hd, g, xic, yic,
+ qRound(xc + qcos(ra2)*w2), qRound(yc - qsin(ra2)*h2));
+ XDrawArc( dpy, hd, g, x, y, w, h, a*4, alen*4 );
+ }
+}
+
+
+/*!
+ Draws a chord defined by the rectangle \a (x, y, w, h), the start
+ angle \a a and the arc length \a alen.
+
+ The chord is filled with the current brush().
+
+ The angles \a a and \a alen are 1/16th of a degree, i.e. a full
+ circle equals 5760 (16*360). Positive values of \a a and \a alen
+ mean counter-clockwise while negative values mean the clockwise
+ direction. Zero degrees is at the 3 o'clock position.
+
+ \sa drawArc(), drawPie()
+*/
+
+void QPainter::drawChord( int x, int y, int w, int h, int a, int alen )
+{
+ if ( !isActive() )
+ return;
+ if ( testf(ExtDev|VxF|WxF) ) {
+ if ( testf(ExtDev) ) {
+ QPDevCmdParam param[3];
+ QRect r( x, y, w, h );
+ param[0].rect = &r;
+ param[1].ival = a;
+ param[2].ival = alen;
+ if ( !pdev->cmd(QPaintDevice::PdcDrawChord, this, param) || !hd )
+ return;
+ }
+ if ( txop == TxRotShear ) { // rotate/shear
+ QPointArray pa;
+ pa.makeArc( x, y, w-1, h-1, a, alen, xmat ); // arc polygon
+ int n = pa.size();
+ pa.resize( n+1 );
+ pa.setPoint( n, pa.at(0) ); // connect endpoints
+ drawPolyInternal( pa );
+ return;
+ }
+ map( x, y, w, h, &x, &y, &w, &h );
+ }
+ XSetArcMode( dpy, gc_brush, ArcChord );
+ w--;
+ h--;
+ if ( w <= 0 || h <= 0 ) {
+ if ( w == 0 || h == 0 )
+ return;
+ fix_neg_rect( &x, &y, &w, &h );
+ }
+
+ GC g = gc;
+ bool nopen = cpen.style() == NoPen;
+
+ if ( cbrush.style() != NoBrush ) { // draw filled chord
+ XFillArc( dpy, hd, gc_brush, x, y, w, h, a*4, alen*4 );
+ if ( nopen ) {
+ g = gc_brush;
+ nopen = FALSE;
+ }
+ }
+ if ( !nopen ) { // draw chord outline
+ double w2 = 0.5*w; // with, height in ellipsis
+ double h2 = 0.5*h;
+ double xc = (double)x+w2;
+ double yc = (double)y+h2;
+ double ra1 = Q_PI/2880.0*a; // convert a, alen to radians
+ double ra2 = ra1 + Q_PI/2880.0*alen;
+ XDrawLine( dpy, hd, g,
+ qRound(xc + qcos(ra1)*w2), qRound(yc - qsin(ra1)*h2),
+ qRound(xc + qcos(ra2)*w2), qRound(yc - qsin(ra2)*h2));
+ XDrawArc( dpy, hd, g, x, y, w, h, a*4, alen*4 );
+ }
+ XSetArcMode( dpy, gc_brush, ArcPieSlice );
+}
+
+
+/*!
+ Draws \a nlines separate lines from points defined in \a a,
+ starting at \a a[index] (\a index defaults to 0). If \a nlines is
+ -1 (the default) all points until the end of the array are used
+ (i.e. (a.size()-index)/2 lines are drawn).
+
+ Draws the 1st line from \a a[index] to \a a[index+1]. Draws the
+ 2nd line from \a a[index+2] to \a a[index+3] etc.
+
+ \warning On X11, coordinates that do not fit into 16-bit signed
+ values are truncated. This limitation is expected to go away in
+ Qt 4.
+
+ \sa drawPolyline(), drawPolygon(), QPen
+*/
+
+void QPainter::drawLineSegments( const QPointArray &a, int index, int nlines )
+{
+ if ( nlines < 0 )
+ nlines = a.size()/2 - index/2;
+ if ( index + nlines*2 > (int)a.size() )
+ nlines = (a.size() - index)/2;
+ if ( !isActive() || nlines < 1 || index < 0 )
+ return;
+ QPointArray pa = a;
+ if ( testf(ExtDev|VxF|WxF) ) {
+ if ( testf(ExtDev) ) {
+ if ( 2*nlines != (int)pa.size() ) {
+ pa = QPointArray( nlines*2 );
+ for ( int i=0; i<nlines*2; i++ )
+ pa.setPoint( i, a.point(index+i) );
+ index = 0;
+ }
+ QPDevCmdParam param[1];
+ param[0].ptarr = (QPointArray*)&pa;
+ if ( !pdev->cmd(QPaintDevice::PdcDrawLineSegments, this, param) ||
+ !hd )
+ return;
+ }
+ if ( txop != TxNone ) {
+ pa = xForm( a, index, nlines*2 );
+ if ( pa.size() != a.size() ) {
+ index = 0;
+ nlines = pa.size()/2;
+ }
+ }
+ }
+ if ( cpen.style() != NoPen )
+ XDrawSegments( dpy, hd, gc,
+ (XSegment*)(pa.shortPoints( index, nlines*2 )), nlines );
+}
+
+
+/*!
+ Draws the polyline defined by the \a npoints points in \a a
+ starting at \a a[index]. (\a index defaults to 0.)
+
+ If \a npoints is -1 (the default) all points until the end of the
+ array are used (i.e. a.size()-index-1 line segments are drawn).
+
+ \warning On X11, coordinates that do not fit into 16-bit signed
+ values are truncated. This limitation is expected to go away in
+ Qt 4.
+
+ \sa drawLineSegments(), drawPolygon(), QPen
+*/
+
+void QPainter::drawPolyline( const QPointArray &a, int index, int npoints )
+{
+ if ( npoints < 0 )
+ npoints = a.size() - index;
+ if ( index + npoints > (int)a.size() )
+ npoints = a.size() - index;
+ if ( !isActive() || npoints < 2 || index < 0 )
+ return;
+ QPointArray pa = a;
+ if ( testf(ExtDev|VxF|WxF) ) {
+ if ( testf(ExtDev) ) {
+ if ( npoints != (int)pa.size() ) {
+ pa = QPointArray( npoints );
+ for ( int i=0; i<npoints; i++ )
+ pa.setPoint( i, a.point(index+i) );
+ index = 0;
+ }
+ QPDevCmdParam param[1];
+ param[0].ptarr = (QPointArray*)&pa;
+ if ( !pdev->cmd(QPaintDevice::PdcDrawPolyline, this, param) || !hd )
+ return;
+ }
+ if ( txop != TxNone ) {
+ pa = xForm( pa, index, npoints );
+ if ( pa.size() != a.size() ) {
+ index = 0;
+ npoints = pa.size();
+ }
+ }
+ }
+ if ( cpen.style() != NoPen ) {
+ while(npoints>65535) {
+ XDrawLines( dpy, hd, gc, (XPoint*)(pa.shortPoints( index, 65535 )),
+ 65535, CoordModeOrigin );
+ npoints-=65535;
+ index+=65535;
+ }
+ XDrawLines( dpy, hd, gc, (XPoint*)(pa.shortPoints( index, npoints )),
+ npoints, CoordModeOrigin );
+ }
+}
+
+static int global_polygon_shape = Complex;
+
+/*!
+ Draws the polygon defined by the \a npoints points in \a a
+ starting at \a a[index]. (\a index defaults to 0.)
+
+ If \a npoints is -1 (the default) all points until the end of the
+ array are used (i.e. a.size()-index line segments define the
+ polygon).
+
+ The first point is always connected to the last point.
+
+ The polygon is filled with the current brush(). If \a winding is
+ TRUE, the polygon is filled using the winding fill algorithm. If
+ \a winding is FALSE, the polygon is filled using the even-odd
+ (alternative) fill algorithm.
+
+ \warning On X11, coordinates that do not fit into 16-bit signed
+ values are truncated. This limitation is expected to go away in
+ Qt 4.
+
+ \sa drawLineSegments(), drawPolyline(), QPen
+*/
+
+void QPainter::drawPolygon( const QPointArray &a, bool winding,
+ int index, int npoints )
+{
+ if ( npoints < 0 )
+ npoints = a.size() - index;
+ if ( index + npoints > (int)a.size() )
+ npoints = a.size() - index;
+ if ( !isActive() || npoints < 2 || index < 0 )
+ return;
+ QPointArray pa = a;
+ if ( testf(ExtDev|VxF|WxF) ) {
+ if ( testf(ExtDev) ) {
+ if ( npoints != (int)a.size() ) {
+ pa = QPointArray( npoints );
+ for ( int i=0; i<npoints; i++ )
+ pa.setPoint( i, a.point(index+i) );
+ index = 0;
+ }
+ QPDevCmdParam param[2];
+ param[0].ptarr = (QPointArray*)&pa;
+ param[1].ival = winding;
+ if ( !pdev->cmd(QPaintDevice::PdcDrawPolygon, this, param) || !hd )
+ return;
+ }
+ if ( txop != TxNone ) {
+ pa = xForm( a, index, npoints );
+ if ( pa.size() != a.size() ) {
+ index = 0;
+ npoints = pa.size();
+ }
+ }
+ }
+ if ( winding ) // set to winding fill rule
+ XSetFillRule( dpy, gc_brush, WindingRule );
+
+ if ( pa[index] != pa[index+npoints-1] ){ // close open pointarray
+ pa.detach();
+ pa.resize( index+npoints+1 );
+ pa.setPoint( index+npoints, pa[index] );
+ npoints++;
+ }
+
+ if ( cbrush.style() != NoBrush ) { // draw filled polygon
+ XFillPolygon( dpy, hd, gc_brush,
+ (XPoint*)(pa.shortPoints( index, npoints )),
+ npoints, global_polygon_shape, CoordModeOrigin );
+ }
+ if ( cpen.style() != NoPen ) { // draw outline
+ XDrawLines( dpy, hd, gc, (XPoint*)(pa.shortPoints( index, npoints )),
+ npoints, CoordModeOrigin );
+ }
+ if ( winding ) // set to normal fill rule
+ XSetFillRule( dpy, gc_brush, EvenOddRule );
+}
+
+/*!
+ Draws the convex polygon defined by the \a npoints points in \a pa
+ starting at \a pa[index] (\a index defaults to 0).
+
+ If the supplied polygon is not convex, the results are undefined.
+
+ On some platforms (e.g. X Window), this is faster than
+ drawPolygon().
+
+ \warning On X11, coordinates that do not fit into 16-bit signed
+ values are truncated. This limitation is expected to go away in
+ Qt 4.
+*/
+void QPainter::drawConvexPolygon( const QPointArray &pa,
+ int index, int npoints )
+{
+ global_polygon_shape = Convex;
+ drawPolygon(pa, FALSE, index, npoints);
+ global_polygon_shape = Complex;
+}
+
+
+
+/*!
+ Draws a cubic Bezier curve defined by the control points in \a a,
+ starting at \a a[index] (\a index defaults to 0).
+
+ Control points after \a a[index + 3] are ignored. Nothing happens
+ if there aren't enough control points.
+
+ \warning On X11, coordinates that do not fit into 16-bit signed
+ values are truncated. This limitation is expected to go away in
+ Qt 4.
+*/
+
+void QPainter::drawCubicBezier( const QPointArray &a, int index )
+{
+ if ( !isActive() )
+ return;
+ if ( a.size() - index < 4 ) {
+#if defined(QT_CHECK_RANGE)
+ qWarning( "QPainter::drawCubicBezier: Cubic Bezier needs 4 control "
+ "points" );
+#endif
+ return;
+ }
+ QPointArray pa( a );
+ if ( index != 0 || a.size() > 4 ) {
+ pa = QPointArray( 4 );
+ for ( int i=0; i<4; i++ )
+ pa.setPoint( i, a.point(index+i) );
+ }
+ if ( testf(ExtDev|VxF|WxF) ) {
+ if ( testf(ExtDev) ) {
+ QPDevCmdParam param[1];
+ param[0].ptarr = (QPointArray*)&pa;
+ if ( !pdev->cmd(QPaintDevice::PdcDrawCubicBezier, this, param) ||
+ !hd )
+ return;
+ }
+ if ( txop != TxNone )
+ pa = xForm( pa );
+ }
+ if ( cpen.style() != NoPen ) {
+ pa = pa.cubicBezier();
+ XDrawLines( dpy, hd, gc, (XPoint*)pa.shortPoints(), pa.size(),
+ CoordModeOrigin );
+ }
+}
+
+
+/*!
+ Draws a pixmap at \a (x, y) by copying a part of \a pixmap into
+ the paint device.
+
+ \a (x, y) specifies the top-left point in the paint device that is
+ to be drawn onto. \a (sx, sy) specifies the top-left point in \a
+ pixmap that is to be drawn. The default is (0, 0).
+
+ \a (sw, sh) specifies the size of the pixmap that is to be drawn.
+ The default, (-1, -1), means all the way to the bottom right of
+ the pixmap.
+
+ Currently the mask of the pixmap or it's alpha channel are ignored
+ when painting on a QPrinter.
+
+ \sa bitBlt(), QPixmap::setMask()
+*/
+
+void QPainter::drawPixmap( int x, int y, const QPixmap &pixmap,
+ int sx, int sy, int sw, int sh )
+{
+ if ( !isActive() || pixmap.isNull() )
+ return;
+
+ // right/bottom
+ if ( sw < 0 )
+ sw = pixmap.width() - sx;
+ if ( sh < 0 )
+ sh = pixmap.height() - sy;
+
+ // Sanity-check clipping
+ if ( sx < 0 ) {
+ x -= sx;
+ sw += sx;
+ sx = 0;
+ }
+ if ( sw + sx > pixmap.width() )
+ sw = pixmap.width() - sx;
+ if ( sy < 0 ) {
+ y -= sy;
+ sh += sy;
+ sy = 0;
+ }
+ if ( sh + sy > pixmap.height() )
+ sh = pixmap.height() - sy;
+
+ if ( sw <= 0 || sh <= 0 )
+ return;
+
+ if ( pdev->x11Screen() != pixmap.x11Screen() ) {
+ QPixmap* p = (QPixmap*) &pixmap;
+ p->x11SetScreen( pdev->x11Screen() );
+ }
+
+ QPixmap::x11SetDefaultScreen( pixmap.x11Screen() );
+
+ if ( testf(ExtDev|VxF|WxF) ) {
+ if ( testf(ExtDev) || txop == TxScale || txop == TxRotShear ) {
+ if ( sx != 0 || sy != 0 ||
+ sw != pixmap.width() || sh != pixmap.height() ) {
+ QPixmap tmp( sw, sh, pixmap.depth() );
+ bitBlt( &tmp, 0, 0, &pixmap, sx, sy, sw, sh, CopyROP, TRUE );
+ if ( pixmap.mask() ) {
+ QBitmap mask( sw, sh );
+ bitBlt( &mask, 0, 0, pixmap.mask(), sx, sy, sw, sh,
+ CopyROP, TRUE );
+ tmp.setMask( mask );
+ }
+ drawPixmap( x, y, tmp );
+ return;
+ }
+ if ( testf(ExtDev) ) {
+ QPDevCmdParam param[2];
+ QRect r(x, y, pixmap.width(), pixmap.height());
+ param[0].rect = &r;
+ param[1].pixmap = &pixmap;
+ if ( !pdev->cmd(QPaintDevice::PdcDrawPixmap, this, param) || !hd )
+ return;
+ }
+ if ( txop == TxScale || txop == TxRotShear ) {
+ QWMatrix mat( m11(), m12(),
+ m21(), m22(),
+ dx(), dy() );
+ mat = QPixmap::trueMatrix( mat, sw, sh );
+ QPixmap pm = pixmap.xForm( mat );
+ if ( !pm.mask() && txop == TxRotShear ) {
+ QBitmap bm_clip( sw, sh, 1 );
+ bm_clip.fill( color1 );
+ pm.setMask( bm_clip.xForm(mat) );
+ }
+ map( x, y, &x, &y ); // compute position of pixmap
+ int dx, dy;
+ mat.map( 0, 0, &dx, &dy );
+ uint save_flags = flags;
+ flags = IsActive | (save_flags & ClipOn);
+ drawPixmap( x-dx, y-dy, pm );
+ flags = save_flags;
+ return;
+ }
+ }
+ map( x, y, &x, &y );
+ }
+
+ QBitmap *mask = (QBitmap *)pixmap.mask();
+ bool mono = pixmap.depth() == 1;
+
+ if ( mask && !hasClipping() && pdev != paintEventDevice ) {
+ if ( mono ) { // needs GCs pen color
+ bool selfmask = pixmap.data->selfmask;
+ if ( selfmask ) {
+ XSetFillStyle( dpy, gc, FillStippled );
+ XSetStipple( dpy, gc, pixmap.handle() );
+ } else {
+ XSetFillStyle( dpy, gc, FillOpaqueStippled );
+ XSetStipple( dpy, gc, pixmap.handle() );
+ XSetClipMask( dpy, gc, mask->handle() );
+ XSetClipOrigin( dpy, gc, x-sx, y-sy );
+ }
+ XSetTSOrigin( dpy, gc, x-sx, y-sy );
+ XFillRectangle( dpy, hd, gc, x, y, sw, sh );
+ XSetTSOrigin( dpy, gc, 0, 0 );
+ XSetFillStyle( dpy, gc, FillSolid );
+ if ( !selfmask ) {
+ if ( pdev == paintEventDevice && paintEventClipRegion ) {
+ x11SetClipRegion( dpy, gc, 0, rendhd, *paintEventClipRegion );
+ } else {
+ x11ClearClipRegion(dpy, gc, 0, rendhd);
+ }
+ }
+ } else {
+ bitBlt( pdev, x, y, &pixmap, sx, sy, sw, sh, (RasterOp)rop );
+ }
+ return;
+ }
+
+ QRegion rgn = crgn;
+
+ if ( mask ) { // pixmap has clip mask
+ // Implies that clipping is on, either explicit or implicit
+ // Create a new mask that combines the mask with the clip region
+
+ if ( pdev == paintEventDevice && paintEventClipRegion ) {
+ if ( hasClipping() )
+ rgn = rgn.intersect( *paintEventClipRegion );
+ else
+ rgn = *paintEventClipRegion;
+ }
+
+ QBitmap *comb = new QBitmap( sw, sh );
+ comb->detach();
+ GC cgc = qt_xget_temp_gc( pixmap.x11Screen(), TRUE ); // get temporary mono GC
+ XSetForeground( dpy, cgc, 0 );
+ XFillRectangle( dpy, comb->handle(), cgc, 0, 0, sw, sh );
+ XSetBackground( dpy, cgc, 0 );
+ XSetForeground( dpy, cgc, 1 );
+ int num;
+ XRectangle *rects = (XRectangle *)qt_getClipRects( rgn, num );
+ XSetClipRectangles( dpy, cgc, -x, -y, rects, num, YXBanded );
+ XSetFillStyle( dpy, cgc, FillOpaqueStippled );
+ XSetStipple( dpy, cgc, mask->handle() );
+ XSetTSOrigin( dpy, cgc, -sx, -sy );
+ XFillRectangle( dpy, comb->handle(), cgc, 0, 0, sw, sh );
+ XSetTSOrigin( dpy, cgc, 0, 0 ); // restore cgc
+ XSetFillStyle( dpy, cgc, FillSolid );
+ XSetClipMask( dpy, cgc, None );
+ mask = comb; // it's deleted below
+
+ XSetClipMask( dpy, gc, mask->handle() );
+ XSetClipOrigin( dpy, gc, x, y );
+ }
+
+ if ( mono ) {
+ XSetBackground( dpy, gc, bg_col.pixel(scrn) );
+ XSetFillStyle( dpy, gc, FillOpaqueStippled );
+ XSetStipple( dpy, gc, pixmap.handle() );
+ XSetTSOrigin( dpy, gc, x-sx, y-sy );
+ XFillRectangle( dpy, hd, gc, x, y, sw, sh );
+ XSetTSOrigin( dpy, gc, 0, 0 );
+ XSetFillStyle( dpy, gc, FillSolid );
+ } else {
+#if !defined(QT_NO_XFTFREETYPE) && !defined(QT_NO_XRENDER)
+ Picture pict = rendhd ? XftDrawPicture((XftDraw *) rendhd) : None;
+ QPixmap *alpha = pixmap.data->alphapm;
+
+ if ( pict && pixmap.x11RenderHandle() &&
+ alpha && alpha->x11RenderHandle()) {
+ XRenderComposite(dpy, PictOpOver, pixmap.x11RenderHandle(),
+ alpha->x11RenderHandle(), pict,
+ sx, sy, sx, sy, x, y, sw, sh);
+ } else
+#endif // !QT_NO_XFTFREETYPE && !QT_NO_XRENDER
+ {
+ XCopyArea( dpy, pixmap.handle(), hd, gc, sx, sy, sw, sh, x, y );
+ }
+ }
+
+ if ( mask ) { // restore clipping
+ XSetClipOrigin( dpy, gc, 0, 0 );
+ XSetRegion( dpy, gc, rgn.handle() );
+ delete mask; // delete comb, created above
+ }
+}
+
+
+/* Internal, used by drawTiledPixmap */
+
+static void drawTile( QPainter *p, int x, int y, int w, int h,
+ const QPixmap &pixmap, int xOffset, int yOffset )
+{
+ int yPos, xPos, drawH, drawW, yOff, xOff;
+ yPos = y;
+ yOff = yOffset;
+ while( yPos < y + h ) {
+ drawH = pixmap.height() - yOff; // Cropping first row
+ if ( yPos + drawH > y + h ) // Cropping last row
+ drawH = y + h - yPos;
+ xPos = x;
+ xOff = xOffset;
+ while( xPos < x + w ) {
+ drawW = pixmap.width() - xOff; // Cropping first column
+ if ( xPos + drawW > x + w ) // Cropping last column
+ drawW = x + w - xPos;
+ p->drawPixmap( xPos, yPos, pixmap, xOff, yOff, drawW, drawH );
+ xPos += drawW;
+ xOff = 0;
+ }
+ yPos += drawH;
+ yOff = 0;
+ }
+}
+
+#if 0 // see comment in drawTiledPixmap
+/* Internal, used by drawTiledPixmap */
+
+static void fillTile( QPixmap *tile, const QPixmap &pixmap )
+{
+ bitBlt( tile, 0, 0, &pixmap, 0, 0, -1, -1, Qt::CopyROP, TRUE );
+ int x = pixmap.width();
+ while ( x < tile->width() ) {
+ bitBlt( tile, x,0, tile, 0,0, x,pixmap.height(), Qt::CopyROP, TRUE );
+ x *= 2;
+ }
+ int y = pixmap.height();
+ while ( y < tile->height() ) {
+ bitBlt( tile, 0,y, tile, 0,0, tile->width(),y, Qt::CopyROP, TRUE );
+ y *= 2;
+ }
+}
+#endif
+
+/*!
+ Draws a tiled \a pixmap in the specified rectangle.
+
+ \a (x, y) specifies the top-left point in the paint device that is
+ to be drawn onto; with the width and height given by \a w and \a
+ h. \a (sx, sy) specifies the top-left point in \a pixmap that is
+ to be drawn. The default is (0, 0).
+
+ Calling drawTiledPixmap() is similar to calling drawPixmap()
+ several times to fill (tile) an area with a pixmap, but is
+ potentially much more efficient depending on the underlying window
+ system.
+
+ \sa drawPixmap()
+*/
+
+void QPainter::drawTiledPixmap( int x, int y, int w, int h,
+ const QPixmap &pixmap, int sx, int sy )
+{
+ int sw = pixmap.width();
+ int sh = pixmap.height();
+ if (!sw || !sh )
+ return;
+ if ( sx < 0 )
+ sx = sw - -sx % sw;
+ else
+ sx = sx % sw;
+ if ( sy < 0 )
+ sy = sh - -sy % sh;
+ else
+ sy = sy % sh;
+ /*
+ Requirements for optimizing tiled pixmaps:
+ - not an external device
+ - not scale or rotshear
+ - not mono pixmap
+ - no mask
+ */
+ QBitmap *mask = (QBitmap *)pixmap.mask();
+ if ( !testf(ExtDev) && txop <= TxTranslate && pixmap.depth() > 1 &&
+ mask == 0 ) {
+ if ( txop == TxTranslate )
+ map( x, y, &x, &y );
+
+#if !defined(QT_NO_XFTFREETYPE) && !defined(QT_NO_XRENDER)
+ Picture pict = rendhd ? XftDrawPicture((XftDraw *) rendhd) : None;
+ QPixmap *alpha = pixmap.data->alphapm;
+
+ if (pict && pixmap.x11RenderHandle() && alpha && alpha->x11RenderHandle()) {
+ // this is essentially drawTile() from above, inlined for
+ // the XRenderComposite call
+ int yPos, xPos, drawH, drawW, yOff, xOff;
+ yPos = y;
+ yOff = sy;
+ while( yPos < y + h ) {
+ drawH = pixmap.height() - yOff; // Cropping first row
+ if ( yPos + drawH > y + h ) // Cropping last row
+ drawH = y + h - yPos;
+ xPos = x;
+ xOff = sx;
+ while( xPos < x + w ) {
+ drawW = pixmap.width() - xOff; // Cropping first column
+ if ( xPos + drawW > x + w ) // Cropping last column
+ drawW = x + w - xPos;
+ XRenderComposite(dpy, PictOpOver, pixmap.x11RenderHandle(),
+ alpha->x11RenderHandle(), pict,
+ xOff, yOff, xOff, yOff, xPos, yPos, drawW, drawH);
+ xPos += drawW;
+ xOff = 0;
+ }
+ yPos += drawH;
+ yOff = 0;
+ }
+ return;
+ }
+#endif // !QT_NO_XFTFREETYPE && !QT_NO_XRENDER
+
+ XSetTile( dpy, gc, pixmap.handle() );
+ XSetFillStyle( dpy, gc, FillTiled );
+ XSetTSOrigin( dpy, gc, x-sx, y-sy );
+ XFillRectangle( dpy, hd, gc, x, y, w, h );
+ XSetTSOrigin( dpy, gc, 0, 0 );
+ XSetFillStyle( dpy, gc, FillSolid );
+ return;
+ }
+
+#if 0
+ // maybe there'll be point in this again, but for the time all it
+ // does is make trouble for the postscript code.
+ if ( sw*sh < 8192 && sw*sh < 16*w*h ) {
+ int tw = sw;
+ int th = sh;
+ while( th * tw < 4096 && ( th < h || tw < w ) ) {
+ if ( h/th > w/tw )
+ th *= 2;
+ else
+ tw *= 2;
+ }
+ QPixmap tile( tw, th, pixmap.depth(), QPixmap::NormalOptim );
+ fillTile( &tile, pixmap );
+ if ( mask ) {
+ QBitmap tilemask( tw, th, QPixmap::NormalOptim );
+ fillTile( &tilemask, *mask );
+ tile.setMask( tilemask );
+ }
+ drawTile( this, x, y, w, h, tile, sx, sy );
+ } else {
+ drawTile( this, x, y, w, h, pixmap, sx, sy );
+ }
+#else
+ // for now we'll just output the original and let the postscript
+ // code make what it can of it. qpicture will be unhappy.
+ drawTile( this, x, y, w, h, pixmap, sx, sy );
+#endif
+}
+
+#if 0
+//
+// Generate a string that describes a transformed bitmap. This string is used
+// to insert and find bitmaps in the global pixmap cache.
+//
+
+static QString gen_text_bitmap_key( const QWMatrix &m, const QFont &font,
+ const QString &str, int pos, int len )
+{
+ QString fk = font.key();
+ int sz = 4*2 + len*2 + fk.length()*2 + sizeof(double)*6;
+ QByteArray buf(sz);
+ uchar *p = (uchar *)buf.data();
+ *((double*)p)=m.m11(); p+=sizeof(double);
+ *((double*)p)=m.m12(); p+=sizeof(double);
+ *((double*)p)=m.m21(); p+=sizeof(double);
+ *((double*)p)=m.m22(); p+=sizeof(double);
+ *((double*)p)=m.dx(); p+=sizeof(double);
+ *((double*)p)=m.dy(); p+=sizeof(double);
+ QChar h1( '$' );
+ QChar h2( 'q' );
+ QChar h3( 't' );
+ QChar h4( '$' );
+ *((QChar*)p)=h1; p+=2;
+ *((QChar*)p)=h2; p+=2;
+ *((QChar*)p)=h3; p+=2;
+ *((QChar*)p)=h4; p+=2;
+ memcpy( (char*)p, (char*)(str.unicode()+pos), len*2 ); p += len*2;
+ memcpy( (char*)p, (char*)fk.unicode(), fk.length()*2 ); p += fk.length()*2;
+ return QString( (QChar*)buf.data(), buf.size()/2 );
+}
+
+static QBitmap *get_text_bitmap( const QString &key )
+{
+ return (QBitmap*)QPixmapCache::find( key );
+}
+
+static void ins_text_bitmap( const QString &key, QBitmap *bm )
+{
+ if ( !QPixmapCache::insert(key, bm) ) // cannot insert pixmap
+ delete bm;
+}
+#endif
+
+void qt_draw_transformed_rect( QPainter *p, int x, int y, int w, int h, bool fill )
+{
+ XPoint points[5];
+ int xp = x, yp = y;
+ p->map( xp, yp, &xp, &yp );
+ points[0].x = xp;
+ points[0].y = yp;
+ xp = x + w; yp = y;
+ p->map( xp, yp, &xp, &yp );
+ points[1].x = xp;
+ points[1].y = yp;
+ xp = x + w; yp = y + h;
+ p->map( xp, yp, &xp, &yp );
+ points[2].x = xp;
+ points[2].y = yp;
+ xp = x; yp = y + h;
+ p->map( xp, yp, &xp, &yp );
+ points[3].x = xp;
+ points[3].y = yp;
+ points[4] = points[0];
+
+ if ( fill )
+ XFillPolygon( p->dpy, p->hd, p->gc, points, 4, Convex, CoordModeOrigin );
+ else
+ XDrawLines( p->dpy, p->hd, p->gc, points, 5, CoordModeOrigin );
+}
+
+void qt_draw_background( QPainter *p, int x, int y, int w, int h )
+{
+ if (p->testf(QPainter::ExtDev)) {
+ if (p->pdev->devType() == QInternal::Printer)
+ p->fillRect(x, y, w, h, p->bg_col);
+ return;
+ }
+ XSetForeground( p->dpy, p->gc, p->bg_col.pixel(p->scrn) );
+ qt_draw_transformed_rect( p, x, y, w, h, TRUE);
+ XSetForeground( p->dpy, p->gc, p->cpen.color().pixel(p->scrn) );
+}
+
+/*!
+ Draws at most \a len characters of the string \a str at position
+ \a (x, y).
+
+ \a (x, y) is the base line position. Note that the meaning of \a y
+ is not the same for the two drawText() varieties.
+*/
+void QPainter::drawText( int x, int y, const QString &str, int len, QPainter::TextDirection dir )
+{
+ drawText( x, y, str, 0, len, dir );
+}
+
+/*!
+ Draws at most \a len characters starting at position \a pos from the
+ string \a str to position \a (x, y).
+
+ \a (x, y) is the base line position. Note that the meaning of \a y
+ is not the same for the two drawText() varieties.
+*/
+void QPainter::drawText( int x, int y, const QString &str, int pos, int len, QPainter::TextDirection dir )
+{
+ if ( !isActive() )
+ return;
+ if (len < 0)
+ len = str.length() - pos;
+ if ( len <= 0 || pos >= (int)str.length() ) // empty string
+ return;
+ if ( pos + len > (int)str.length() )
+ len = str.length() - pos;
+
+ if ( testf(DirtyFont) ) {
+ updateFont();
+ }
+
+ if ( testf(ExtDev) && pdev->devType() != QInternal::Printer ) {
+ QPDevCmdParam param[3];
+ QPoint p(x, y);
+ QString string = str.mid( pos, len );
+ param[0].point = &p;
+ param[1].str = &string;
+ param[2].ival = QFont::Latin;
+ if ( !pdev->cmd(QPaintDevice::PdcDrawText2, this, param) || !hd )
+ return;
+ }
+
+ bool simple = (dir == QPainter::Auto) && str.simpleText();
+ // we can't take the complete string here as we would otherwise
+ // get quadratic behaviour when drawing long strings in parts.
+ // we do however need some chars around the part we paint to get arabic shaping correct.
+ // ### maybe possible to remove after cursor restrictions work in QRT
+ int start;
+ int end;
+ if ( simple ) {
+ start = pos;
+ end = pos+len;
+ } else {
+ start = QMAX( 0, pos - 8 );
+ end = QMIN( (int)str.length(), pos + len + 8 );
+ }
+ QConstString cstr( str.unicode() + start, end - start );
+ pos -= start;
+
+ QTextEngine engine( cstr.string(), pfont ? pfont->d : cfont.d );
+ QTextLayout layout( &engine );
+
+ // this is actually what beginLayout does. Inlined here, so we can
+ // avoid the bidi algorithm if we don't need it.
+ engine.itemize( simple ? QTextEngine::NoBidi|QTextEngine::SingleLine : QTextEngine::Full|QTextEngine::SingleLine );
+ engine.currentItem = 0;
+ engine.firstItemInLine = -1;
+
+ if ( dir != Auto ) {
+ int level = dir == RTL ? 1 : 0;
+ for ( int i = engine.items.size(); i >= 0; i-- )
+ engine.items[i].analysis.bidiLevel = level;
+ }
+
+ if ( !simple ) {
+ layout.setBoundary( pos );
+ layout.setBoundary( pos + len );
+ }
+
+ // small hack to force skipping of unneeded items
+ start = 0;
+ while ( engine.items[start].position < pos )
+ ++start;
+ engine.currentItem = start;
+ layout.beginLine( 0xfffffff );
+ end = start;
+ while ( !layout.atEnd() && layout.currentItem().from() < pos + len ) {
+ layout.addCurrentItem();
+ end++;
+ }
+ QFontMetrics fm(fontMetrics());
+ int ascent = fm.ascent(), descent = fm.descent();
+ int left, right;
+ layout.endLine( 0, 0, Qt::SingleLine|Qt::AlignLeft, &ascent, &descent, &left, &right );
+
+ // do _not_ call endLayout() here, as it would clean up the shaped items and we would do shaping another time
+ // for painting.
+
+ int textFlags = 0;
+ if ( cfont.d->underline ) textFlags |= Qt::Underline;
+ if ( cfont.d->overline ) textFlags |= Qt::Overline;
+ if ( cfont.d->strikeOut ) textFlags |= Qt::StrikeOut;
+
+ if ( bg_mode == OpaqueMode )
+ qt_draw_background( this, x, y-ascent, right-left, ascent+descent+1);
+
+ for ( int i = start; i < end; i++ ) {
+ QTextItem ti;
+ ti.item = i;
+ ti.engine = &engine;
+
+ drawTextItem( x, y - ascent, ti, textFlags );
+ }
+ layout.d = 0;
+}
+
+
+/*! \internal
+ Draws the text item \a ti at position \a (x, y ).
+
+ This method ignores the painters background mode and
+ color. drawText and qt_format_text have to do it themselves, as
+ only they know the extents of the complete string.
+
+ It ignores the font set on the painter as the text item has one of its own.
+
+ The underline and strikeout parameters of the text items font are
+ ignored aswell. You'll need to pass in the correct flags to get
+ underlining and strikeout.
+*/
+void QPainter::drawTextItem( int x, int y, const QTextItem &ti, int textFlags )
+{
+ if ( testf(ExtDev) ) {
+ QPDevCmdParam param[2];
+ QPoint p(x, y);
+ param[0].point = &p;
+ param[1].textItem = &ti;
+ bool retval = pdev->cmd(QPaintDevice::PdcDrawTextItem, this, param);
+ if ( !retval || !hd )
+ return;
+ }
+
+ QTextEngine *engine = ti.engine;
+ QScriptItem *si = &engine->items[ti.item];
+
+ engine->shape( ti.item );
+ QFontEngine *fe = si->fontEngine;
+ assert( fe != 0 );
+
+ x += si->x;
+ y += si->y;
+
+ fe->draw( this, x, y, engine, si, textFlags );
+}
+
+#if QT_VERSION >= 0x040000
+#error "remove current position and associated methods"
+#endif
+/*!
+ \obsolete
+ Returns the current position of the pen.
+
+ \sa moveTo()
+ */
+QPoint QPainter::pos() const
+{
+ return curPt;
+}