summaryrefslogtreecommitdiffstats
path: root/tdegtk/tqtcairopainter.cpp
diff options
context:
space:
mode:
Diffstat (limited to 'tdegtk/tqtcairopainter.cpp')
-rw-r--r--tdegtk/tqtcairopainter.cpp434
1 files changed, 410 insertions, 24 deletions
diff --git a/tdegtk/tqtcairopainter.cpp b/tdegtk/tqtcairopainter.cpp
index 8b38bc7..6bf5b9f 100644
--- a/tdegtk/tqtcairopainter.cpp
+++ b/tdegtk/tqtcairopainter.cpp
@@ -29,6 +29,317 @@
#include "tqpaintdevicemetrics.h"
#undef Qt
+#define CAIRO_PIXEL_OFFSET (0.5)
+
+#define SET_BIT(x, y) (x |= 1 << y)
+#define TEST_BIT(x, y) ((x & (1 << y)) >> y)
+
+cairo_surface_t* TQImageToCairoSurface(TQImage origimg) {
+ cairo_surface_t* ret;
+
+ TQImage img;
+ if (origimg.depth() < 24) {
+ img = origimg.convertDepth(32);
+ }
+ else {
+ img = origimg;
+ }
+
+ cairo_format_t cairo_format;
+ int depth = img.depth();
+ if (depth == 32) {
+ cairo_format = CAIRO_FORMAT_ARGB32;
+ }
+ else if (depth == 24) {
+ cairo_format = CAIRO_FORMAT_RGB24;
+ }
+ else {
+ cairo_format = CAIRO_FORMAT_RGB24;
+ }
+
+ int stride = cairo_format_stride_for_width(cairo_format, img.width());
+ ret = cairo_image_surface_create_for_data(img.bits(), cairo_format, img.width(), img.height(), stride);
+
+ return ret;
+}
+
+TQImage CairoSurfaceToTQImage(cairo_surface_t* surface) {
+ cairo_surface_flush(surface);
+
+ cairo_format_t cairo_format = cairo_image_surface_get_format(surface);
+ int height = cairo_image_surface_get_height(surface);
+ int width = cairo_image_surface_get_width(surface);
+ int depth;
+ if (cairo_format == CAIRO_FORMAT_ARGB32) {
+ depth = 32;
+ }
+ else if (cairo_format == CAIRO_FORMAT_RGB24) {
+ depth = 24;
+ }
+ else {
+ // FIXME
+ // Force Cairo to convert the surface to a format that TQImage can read!
+ printf("[WARNING] Tried to convert a Cairo surface of format %d to a TQImage (NULL image returned!)\n\r", cairo_format); fflush(stdout);
+ return TQImage();
+ }
+
+ return TQImage(cairo_image_surface_get_data(surface), width, height, depth, (TQRgb*)NULL, 0, TQImage::BigEndian);
+}
+
+void TQt3CairoPaintDevice::dualStrokePen() {
+ if (m_bgColorMode == TQt::OpaqueMode) {
+ // Draw background
+ cairo_save(m_painter);
+ updatePen(TRUE);
+ cairo_stroke(m_painter);
+ cairo_restore(m_painter);
+ }
+ // Draw foreground
+ updatePen(FALSE);
+ cairo_stroke(m_painter);
+}
+
+void TQt3CairoPaintDevice::dualStrokeBrush() {
+ if (m_bgColorMode == TQt::OpaqueMode) {
+ // Draw background
+ cairo_save(m_painter);
+ updateBrush(TRUE);
+ cairo_fill(m_painter);
+ cairo_restore(m_painter);
+ }
+ // Draw foreground
+ updateBrush(FALSE);
+ cairo_fill(m_painter);
+}
+
+void TQt3CairoPaintDevice::updatePen(bool backgroundStroke) {
+ if (!m_painter) {
+ return;
+ }
+
+ int ps = m_pen.style();
+
+ double dashes[10]; // custom pen dashes
+ int dash_len = 0; // length of dash list
+ int dash_offset = 0;
+ cairo_line_cap_t cp = CAIRO_LINE_CAP_BUTT;
+ cairo_line_join_t jn = CAIRO_LINE_JOIN_MITER;
+
+ /*
+ We are emulating Windows here. Windows treats m_pen.width() == 1
+ (or 0) as a very special case. The fudge variable unifies this
+ case with the general case.
+ */
+ int dot = m_pen.width(); // width of a dot
+ int fudge = 1;
+ //bool allow_zero_lw = TRUE;
+ bool allow_zero_lw = FALSE;
+ if ( dot <= 1 ) {
+ dot = 3;
+ fudge = 2;
+ }
+
+ switch( ps ) {
+ case TQPainter::NoPen:
+ case TQPainter::SolidLine:
+ break;
+ case TQPainter::DashLine:
+ dashes[0] = fudge * 3 * dot;
+ dashes[1] = fudge * dot;
+ dash_len = 2;
+ allow_zero_lw = FALSE;
+ break;
+ case TQPainter::DotLine:
+ dashes[0] = dot;
+ dashes[1] = dot;
+ dash_len = 2;
+ allow_zero_lw = FALSE;
+ break;
+ case TQPainter::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 TQPainter::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;
+ break;
+ case TQPainter::FineDotLine:
+ dot = 1;
+ dashes[0] = dot;
+ dashes[1] = dot;
+ dash_len = 2;
+ allow_zero_lw = FALSE;
+ }
+ Q_ASSERT( dash_len <= (int) sizeof(dashes) );
+
+ switch ( m_pen.capStyle() ) {
+ case TQPainter::SquareCap:
+ cp = CAIRO_LINE_CAP_SQUARE;
+ break;
+ case TQPainter::RoundCap:
+ cp = CAIRO_LINE_CAP_ROUND;
+ break;
+ case TQPainter::FlatCap:
+ default:
+ cp = CAIRO_LINE_CAP_BUTT;
+ break;
+ }
+
+ switch ( m_pen.joinStyle() ) {
+ case TQPainter::BevelJoin:
+ jn = CAIRO_LINE_JOIN_BEVEL;
+ break;
+ case TQPainter::RoundJoin:
+ jn = CAIRO_LINE_JOIN_ROUND;
+ break;
+ case TQPainter::MiterJoin:
+ default:
+ jn = CAIRO_LINE_JOIN_MITER;
+ break;
+ }
+
+ if (backgroundStroke) {
+ dash_len = 0;
+
+ }
+
+ cairo_set_dash(m_painter, dashes, dash_len, dash_offset);
+ cairo_set_line_cap(m_painter, cp);
+ cairo_set_line_join(m_painter, jn);
+ cairo_set_line_width(m_painter, ((!allow_zero_lw) && (m_pen.width() == 0)) ? 1 : m_pen.width());
+
+ TQRgb color = (backgroundStroke)?m_bgColor.rgb():m_pen.color().rgb();
+ cairo_set_source_rgba(m_painter, tqRed(color), tqGreen(color), tqBlue(color), tqAlpha(color));
+}
+
+void TQt3CairoPaintDevice::updateBrush(bool backgroundStroke) {
+ if (!m_painter) {
+ return;
+ }
+
+ if (backgroundStroke) {
+ TQRgb color = m_bgColor.rgb();
+ cairo_pattern_t* pattern = cairo_pattern_create_rgba(tqRed(color), tqGreen(color), tqBlue(color), tqAlpha(color));
+ cairo_set_source(m_painter, pattern);
+ cairo_pattern_set_extend(cairo_get_source(m_painter), CAIRO_EXTEND_REPEAT);
+ cairo_pattern_destroy(pattern);
+ }
+ else {
+ 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 };
+
+ int bs = m_brush.style();
+
+ const uchar *pat = 0; // pattern
+ int d = 0; // defalt pattern size: d*d
+ if ( bs >= TQBrush::Dense1Pattern && bs <= TQBrush::DiagCrossPattern ) {
+ pat = pat_tbl[ bs-TQBrush::Dense1Pattern ];
+ if ( bs <= TQBrush::Dense7Pattern ) {
+ d = 8;
+ }
+ else if ( bs <= TQBrush::CrossPattern ) {
+ d = 24;
+ }
+ else {
+ d = 16;
+ }
+ }
+
+ if ( (bs == TQBrush::CustomPattern) || pat ) {
+ TQImage brushImage;
+ if ( pat ) {
+ TQRgb color = m_brush.color().rgb();
+ brushImage = TQImage(d, d, 32);
+ int x;
+ int y;
+ int byte = 0;
+ int bit = 7;
+ for (x=0; x<d; x++) {
+ for (y=0; y<d; y++) {
+ brushImage.setPixel(x, y, (TEST_BIT(pat[byte], bit))?color:0x00000000);
+ bit--;
+ if (bit < 0) {
+ bit = 7;
+ byte++;
+ }
+ }
+ }
+ }
+ else {
+ TQPixmap *pm;
+ pm = m_brush.pixmap();
+ brushImage = pm->convertToImage();
+ }
+
+ cairo_surface_t* brushSurface = TQImageToCairoSurface(brushImage);
+ cairo_pattern_t* pattern = cairo_pattern_create_for_surface(brushSurface);
+ cairo_set_source(m_painter, pattern);
+ cairo_pattern_set_extend(cairo_get_source(m_painter), CAIRO_EXTEND_REPEAT);
+ cairo_pattern_destroy(pattern);
+ cairo_surface_destroy(brushSurface);
+ }
+ else {
+ TQRgb color = m_brush.color().rgb();
+ cairo_pattern_t* pattern = cairo_pattern_create_rgba(tqRed(color), tqGreen(color), tqBlue(color), tqAlpha(color));
+ cairo_set_source(m_painter, pattern);
+ cairo_pattern_set_extend(cairo_get_source(m_painter), CAIRO_EXTEND_REPEAT);
+ cairo_pattern_destroy(pattern);
+ }
+ }
+}
+
/*!
\class TQt3CairoPaintDevice tdeqt4painter.h
\brief The TQt3CairoPaintDevice class is a paint device that translates
@@ -44,10 +355,9 @@
*/
TQt3CairoPaintDevice::TQt3CairoPaintDevice( cairo_surface_t *cairosurface )
- : TQPaintDevice( TQInternal::Picture | TQInternal::ExternalDevice )
+ : TQPaintDevice( TQInternal::Picture | TQInternal::ExternalDevice ), m_painter(NULL)
{
m_surface = cairosurface;
- m_painter = cairo_create(m_surface);
}
/*!
@@ -55,7 +365,10 @@ TQt3CairoPaintDevice::TQt3CairoPaintDevice( cairo_surface_t *cairosurface )
*/
TQt3CairoPaintDevice::~TQt3CairoPaintDevice()
{
- cairo_destroy(m_painter);
+ if (m_painter) {
+ cairo_destroy(m_painter);
+ m_painter = NULL;
+ }
}
/*!
@@ -68,13 +381,42 @@ bool TQt3CairoPaintDevice::cmd( int c, TQPainter *pt, TQPDevCmdParam *p )
Q_UNUSED(pt);
unsigned int i;
- int x;
- int y;
+
+ double x;
+ double y;
+ double width;
+ double height;
int index;
int count;
int lineCount;
+ // Convert data types
+ if (p) {
+// if ((c == PdcDrawPoint) || (c == PdcMoveTo) || (c == PdcLineTo) || (c == PdcSetBrushOrigin)) {
+// qt4point1 = QPoint(p[0].point->x(), p[0].point->y());
+// }
+// if (c == PdcDrawLine) {
+// qt4point1 = QPoint(p[0].point->x(), p[0].point->y());
+// qt4point2 = QPoint(p[1].point->x(), p[1].point->y());
+// }
+// if ((c == PdcDrawPolyline) || (c == PdcDrawPolygon) || (c == PdcDrawLineSegments) || (c == PdcDrawCubicBezier)) {
+// TQPointArray qt3parray = *p[0].ptarr;
+// qt4polygon.resize(qt3parray.count());
+// for (i=0;i<qt3parray.count();i++) {
+// qt3parray.point(i, &x, &y );
+// qt4polygon.setPoint(i, x, y);
+// }
+// }
+ if ((c == PdcDrawRect) || (c == PdcDrawRoundRect) || (c == PdcDrawEllipse) || (c == PdcDrawArc) || (c == PdcDrawPie) || (c == PdcDrawChord)) {
+ x = p[0].rect->x();
+ y = p[0].rect->y();
+ width = p[0].rect->width();
+ height = p[0].rect->height();
+ }
+ }
+
+ // Perform drawing operation
switch ( c ) { // exec cmd
case PdcNOP:
break;
@@ -93,9 +435,26 @@ bool TQt3CairoPaintDevice::cmd( int c, TQPainter *pt, TQPDevCmdParam *p )
case PdcDrawLine:
m_qt4painter->drawLine( qt4point1, qt4point2 );
break;
+#endif
case PdcDrawRect:
- m_qt4painter->drawRect( qt4PainterAdjustedRectangle(qt4rect, m_qt4painter) );
+ if (m_painter) {
+ if (m_pen.style() != TQPen::NoPen) {
+ cairo_rectangle(m_painter, x+CAIRO_PIXEL_OFFSET, y+CAIRO_PIXEL_OFFSET, width, height);
+ dualStrokePen();
+ }
+ if (m_brush.style() != TQBrush::NoBrush) {
+ int line_width = m_pen.width();
+ cairo_rectangle(m_painter, x+line_width+CAIRO_PIXEL_OFFSET, y+line_width+CAIRO_PIXEL_OFFSET, width-(line_width*2), height-(line_width*2));
+ dualStrokeBrush();
+ }
+ }
+ else {
+#if defined(QT_CHECK_RANGE)
+ tqWarning( "TQt3CairoPaintDevice::cmd: TQPainter::begin must be called before PdcDrawRect" );
+#endif
+ }
break;
+#if 0
case PdcDrawRoundRect:
m_qt4painter->drawRoundedRect( qt4PainterAdjustedRectangle(qt4rect, m_qt4painter), p[1].ival, p[2].ival );
break;
@@ -157,35 +516,44 @@ bool TQt3CairoPaintDevice::cmd( int c, TQPainter *pt, TQPDevCmdParam *p )
}
break;
#endif
+#endif
case PdcBegin:
- if (m_qt4painter->isActive()) {
- // KEEP THIS DEACTIVATED
-// QPaintDevice* pd = m_qt4painter->device();
-// m_qt4painter->end();
-// m_qt4painter->begin(pd);
+ if (!m_painter) {
+ m_bgColor = TQColor(0,0,0);
+ m_bgColorMode = TQt::TransparentMode;
+ m_painter = cairo_create(m_surface);
+ m_pen = TQPen();
+ m_brush = TQBrush();
}
-#if defined(QT_CHECK_RANGE)
- else {
- tqWarning( "TQt3CairoPaintDevice::cmd: Painter has no paint device available" );
- }
-#endif
break;
case PdcEnd:
- // KEEP THIS DEACTIVATED
-// m_qt4painter->end();
+ if (m_painter) {
+ cairo_destroy(m_painter);
+ m_painter = NULL;
+ }
break;
+#if 0
case PdcSave:
m_qt4painter->save();
break;
case PdcRestore:
m_qt4painter->restore();
break;
+#endif
case PdcSetBkColor:
- m_qt4painter->setBackground( QBrush(QColor(p[0].color->red(), p[0].color->green(), p[0].color->blue())) );
+ if (p) {
+ const TQColor* color = p[0].color;
+ if (color) {
+ m_bgColor = *color;
+ }
+ }
break;
case PdcSetBkMode:
- m_qt4painter->setBackgroundMode( qt4bkmode );
+ if (p) {
+ m_bgColorMode = (TQt::BGMode)p[0].ival;
+ }
break;
+#if 0
case PdcSetROP:
m_qt4painter->setCompositionMode(qt4compositionmode);
break;
@@ -195,13 +563,25 @@ bool TQt3CairoPaintDevice::cmd( int c, TQPainter *pt, TQPDevCmdParam *p )
case PdcSetFont:
m_qt4painter->setFont( qt4font );
break;
+#endif
case PdcSetPen:
- m_qt4painter->setPen( qt4pen );
+ if (p) {
+ const TQPen* pen = p[0].pen;
+ if (pen) {
+ m_pen = *pen;
+ }
+ }
break;
case PdcSetBrush:
- m_qt4painter->setBrush( qt4brush );
+ if (p) {
+ const TQBrush* brush = p[0].brush;
+ if (brush) {
+ m_brush = *brush;
+ }
+ }
break;
#if 0
+#if 0
case PdcSetTabStops:
s >> i_16;
painter->setTabStops( i_16 );
@@ -333,8 +713,11 @@ int TQt3CairoPaintDevice::metric( int m ) const
else if (format == CAIRO_FORMAT_RGB24) {
val = 16777216;
}
+ else if (format == CAIRO_FORMAT_RGB16_565) {
+ val = 65536;
+ }
else {
- val = 16777216;
+ val = 65536;
}
break;
case TQPaintDeviceMetrics::PdmDepth:
@@ -345,8 +728,11 @@ int TQt3CairoPaintDevice::metric( int m ) const
else if (format == CAIRO_FORMAT_RGB24) {
val = 24;
}
+ else if (format == CAIRO_FORMAT_RGB16_565) {
+ val = 16;
+ }
else {
- val = 24;
+ val = 16;
}
break;
default: