summaryrefslogtreecommitdiffstats
path: root/karbon/render/vkopainter.cc
diff options
context:
space:
mode:
Diffstat (limited to 'karbon/render/vkopainter.cc')
-rw-r--r--karbon/render/vkopainter.cc943
1 files changed, 943 insertions, 0 deletions
diff --git a/karbon/render/vkopainter.cc b/karbon/render/vkopainter.cc
new file mode 100644
index 00000000..21ab911c
--- /dev/null
+++ b/karbon/render/vkopainter.cc
@@ -0,0 +1,943 @@
+/* This file is part of the KDE project.
+ Copyright (C) 2001, 2002, 2003 The Karbon Developers
+
+ This library is free software; you can redistribute it and/or
+ modify it under the terms of the GNU Library General Public
+ License as published by the Free Software Foundation; either
+ version 2 of the License, or (at your option) any later version.
+
+ This library is distributed in the hope that it will be useful,
+ but WITHOUT ANY WARRANTY; without even the implied warranty of
+ MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
+ Library General Public License for more details.
+
+ You should have received a copy of the GNU Library General Public License
+ along with this library; see the file COPYING.LIB. If not, write to
+ the Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor,
+ * Boston, MA 02110-1301, USA.
+*/
+
+// kopainter/libart wrapper
+
+#include "vkopainter.h"
+#include "vstroke.h"
+#include "vfill.h"
+#include "vcolor.h"
+#include "vpattern.h"
+
+#include <qpaintdevice.h>
+#include <qpixmap.h>
+#include <qpointarray.h>
+#include <qimage.h>
+
+#include "libart_lgpl/art_vpath.h"
+#include <libart_lgpl/art_bpath.h>
+#include <libart_lgpl/art_vpath_bpath.h>
+#include <libart_lgpl/art_svp_vpath.h>
+#include <libart_lgpl/art_svp_vpath_stroke.h>
+#include <libart_lgpl/art_svp.h>
+#include <libart_lgpl/art_svp_ops.h>
+#include <libart_lgpl/art_affine.h>
+#include <libart_lgpl/art_svp_intersect.h>
+#include <libart_lgpl/art_rect_svp.h>
+#include <libart_lgpl/art_pathcode.h>
+#include <libart_lgpl/art_vpath_dash.h>
+#include "art_rgba_affine.h"
+#include "art_render_misc.h"
+#include <libart_lgpl/art_render_svp.h>
+
+#include "art_rgb_svp.h"
+#include "art_render_pattern.h"
+
+#include <X11/Xlib.h>
+
+#include <gdk-pixbuf-xlibrgb.h>
+
+#include <kdebug.h>
+#include <kglobal.h>
+#include <kiconloader.h>
+#include <math.h>
+
+#include <KoPoint.h>
+#include <KoRect.h>
+#include <karbon_factory.h>
+#include <karbon_resourceserver.h>
+
+
+#define INITIAL_ALLOC 300
+#define ALLOC_INCREMENT 100
+
+VKoPainter::VKoPainter( QPaintDevice *target, unsigned int w, unsigned int h, bool bDrawNodes )
+: VPainter( target, w, h ), m_target( target ), m_bDrawNodes( bDrawNodes )
+{
+ //kdDebug(38000) << "w : " << w << endl;
+ //kdDebug(38000) << "h : " << h << endl;
+ m_width = w;//( w > 0 ) ? w : target->width();
+ m_height= h;//( h > 0 ) ? h : target->height();
+ m_buffer = 0L;
+ m_path = 0L;
+ m_index = 0;
+ resize( m_width, m_height );
+ clear();
+ m_clipPaths.setAutoDelete( false );
+
+ m_stroke = 0L;
+ m_fill = 0L;
+ m_fillRule = evenOdd;
+
+ xlib_rgb_init_with_depth( target->x11Display(), XScreenOfDisplay( target->x11Display(),
+ target->x11Screen() ), target->x11Depth() );
+
+ gc = XCreateGC( target->x11Display(), target->handle(), 0, 0 );
+
+ m_zoomFactor = 1;
+}
+
+VKoPainter::VKoPainter( unsigned char *buffer, unsigned int w, unsigned int h, bool bDrawNodes )
+: VPainter( 0L, w, h ), m_buffer( buffer ), m_bDrawNodes( bDrawNodes )
+{
+ //kdDebug(38000) << "w : " << w << endl;
+ //kdDebug(38000) << "h : " << h << endl;
+ m_target = 0L;
+ m_width = w;
+ m_height= h;
+ m_path = 0L;
+ m_index = 0;
+ clear();
+ m_clipPaths.setAutoDelete( false );
+
+ m_stroke = 0L;
+ m_fill = 0L;
+
+ gc = 0L;
+
+ m_zoomFactor = 1;
+}
+
+VKoPainter::~VKoPainter()
+{
+ // If we are in target mode, we created a buffer, else if we used the other ctor
+ // we didn't.
+ if( m_target )
+ art_free( m_buffer );
+
+ delete m_stroke;
+ delete m_fill;
+ if( m_path )
+ art_free( m_path );
+
+ if( gc )
+ XFreeGC( m_target->x11Display(), gc );
+}
+
+void
+VKoPainter::resize( unsigned int w, unsigned int h )
+{
+ if( !m_buffer || w != m_width || h != m_height )
+ {
+ // TODO : realloc?
+ art_free( m_buffer );
+ m_buffer = 0;
+ m_width = w;
+ m_height = h;
+ if ( m_width != 0 && m_height != 0 )
+ m_buffer = art_new( art_u8, m_width * m_height * 4 );
+ clear();
+ }
+}
+
+void
+VKoPainter::begin()
+{
+}
+
+void
+VKoPainter::end()
+{
+ //xlib_draw_rgb_image( m_target->handle(), gc, 0, 0, m_width, m_height,
+ // XLIB_RGB_DITHER_NONE, m_buffer, m_width * 4 );
+ xlib_draw_rgb_32_image( m_target->handle(), gc, 0, 0, m_width, m_height,
+ XLIB_RGB_DITHER_NONE, m_buffer, m_width * 4 );
+ /*xlib_draw_rgb_image( pix.handle(), gc, 0, 0, m_width, m_height,
+ XLIB_RGB_DITHER_NONE, m_buffer, m_width * 3 );
+ bitBlt( m_target, 0, 0, &pix, 0, 0, m_width, m_height );*/
+}
+
+void
+VKoPainter::blit( const KoRect &r )
+{
+ //kdDebug(38000) << "m_width : " << m_width << endl;
+ //kdDebug(38000) << "m_height : " << m_height << endl;
+ int x = KMAX( 0, int( r.x() ) );
+ int y = KMAX( 0, int( r.y() ) );
+ int width = KMIN( m_width, (unsigned int)KMAX( 0, int( r.x() + r.width() ) ) );
+ int height = KMIN( m_height, (unsigned int)KMAX( 0, int( r.y() + r.height() ) ) );
+ xlib_draw_rgb_32_image( m_target->handle(), gc, x, y, width - x, height - y,
+ XLIB_RGB_DITHER_NONE, m_buffer + (x * 4) + (y * m_width * 4), m_width * 4 );
+}
+
+void
+VKoPainter::clear()
+{
+ if( m_buffer )
+ memset( m_buffer, qRgba( 255, 255, 255, 255 ), m_width * m_height * 4 );
+}
+
+void
+VKoPainter::clear( const QColor &c )
+{
+ if( m_buffer )
+ memset( m_buffer, c.rgb(), m_width * m_height * 4 );
+}
+
+void
+VKoPainter::clear( const KoRect &r, const QColor &c )
+{
+ unsigned int color = c.rgb();
+ int x = KMAX( 0, int( r.x() ) );
+ int y = KMAX( 0, int( r.y() ) );
+ int width = KMIN( m_width, (unsigned int)KMAX( 0, int( r.x() + r.width() ) ) );
+ int height = KMIN( m_height, (unsigned int)KMAX( 0, int( r.y() + r.height() ) ) );
+ if( m_buffer )
+ {
+ for( int i = y;i < height;i++)
+ memset( m_buffer + int( x * 4) + int( i * ( m_width * 4 ) ),
+ qRgba( qRed( color ), qGreen( color ), qBlue( color ), 100 ), int( width * 4 ) );
+ }
+}
+
+void
+VKoPainter::setWorldMatrix( const QWMatrix &mat )
+{
+ m_matrix = mat;
+}
+
+void
+VKoPainter::setZoomFactor( double zoomFactor )
+{
+ m_zoomFactor = zoomFactor;
+}
+
+void
+VKoPainter::ensureSpace( unsigned int newindex )
+{
+ if( m_index == 0 )
+ {
+ if( !m_path )
+ m_path = art_new( ArtBpath, INITIAL_ALLOC );
+ m_alloccount = INITIAL_ALLOC;
+ }
+ else if( newindex > m_alloccount )
+ {
+ m_alloccount += ALLOC_INCREMENT;
+ m_path = art_renew( m_path, ArtBpath, m_alloccount );
+ }
+}
+
+void
+VKoPainter::moveTo( const KoPoint &p )
+{
+ ensureSpace( m_index + 1 );
+
+ m_path[ m_index ].code = ART_MOVETO;
+
+ m_path[ m_index ].x3 = p.x() * m_zoomFactor;
+ m_path[ m_index ].y3 = p.y() * m_zoomFactor;
+
+ m_index++;
+}
+
+void
+VKoPainter::lineTo( const KoPoint &p )
+{
+ ensureSpace( m_index + 1 );
+
+ m_path[ m_index ].code = ART_LINETO;
+ m_path[ m_index ].x3 = p.x() * m_zoomFactor;
+ m_path[ m_index ].y3 = p.y() * m_zoomFactor;
+
+ m_index++;
+}
+
+void
+VKoPainter::curveTo( const KoPoint &p1, const KoPoint &p2, const KoPoint &p3 )
+{
+ ensureSpace( m_index + 1 );
+
+ m_path[ m_index ].code = ART_CURVETO;
+ m_path[ m_index ].x1 = p1.x() * m_zoomFactor;
+ m_path[ m_index ].y1 = p1.y() * m_zoomFactor;
+ m_path[ m_index ].x2 = p2.x() * m_zoomFactor;
+ m_path[ m_index ].y2 = p2.y() * m_zoomFactor;
+ m_path[ m_index ].x3 = p3.x() * m_zoomFactor;
+ m_path[ m_index ].y3 = p3.y() * m_zoomFactor;
+
+ m_index++;
+}
+
+void
+VKoPainter::newPath()
+{
+ m_index = 0;
+}
+
+void
+VKoPainter::setFillRule( VFillRule fillRule )
+{
+ m_fillRule = fillRule;
+}
+
+void
+VKoPainter::fillPath()
+{
+ if( m_index == 0 ) return;
+
+ // find begin of last subpath
+ int find = -1;
+ for( int i = m_index - 1; i >= 0; i-- )
+ {
+ if( m_path[i].code == ART_MOVETO_OPEN || m_path[i].code == ART_MOVETO )
+ {
+ find = i;
+ break;
+ }
+ }
+
+ // for now, always close
+ if( find != -1 && ( m_path[ find ].x3 != m_path[ m_index - 1 ].x3 ||
+ m_path[ find ].y3 != m_path[ m_index - 1 ].y3 ) )
+ {
+ ensureSpace( m_index + 1 );
+
+ m_path[ m_index ].code = ART_LINETO;
+ m_path[ m_index ].x3 = m_path[ find ].x3;
+ m_path[ m_index ].y3 = m_path[ find ].y3;
+
+ m_index++;
+ m_path[ m_index ].code = ART_END;
+ }
+ else
+ m_path[ m_index++ ].code = ART_END;
+
+ if( m_fill && m_fill->type() != VFill::none )
+ {
+ ArtVpath *path = art_bez_path_to_vec( m_path , 0.25 );
+ drawVPath( path );
+ }
+
+ m_index--;
+}
+
+void
+VKoPainter::strokePath()
+{
+ if( m_index == 0 ) return;
+
+ if( m_stroke && m_stroke->lineWidth() == 0 )
+ return;
+ if( m_path[ m_index ].code != ART_END)
+ m_path[ m_index ].code = ART_END;
+
+ ArtVpath *path = art_bez_path_to_vec( m_path , 0.25 );
+
+ drawVPath( path );
+}
+
+void
+VKoPainter::setClipPath()
+{
+ ArtVpath *path;
+ path = art_bez_path_to_vec( m_path , 0.25 );
+ m_clipPaths.append( art_svp_from_vpath( path ) );
+ art_free( path );
+}
+
+void
+VKoPainter::resetClipPath()
+{
+ art_svp_free( m_clipPaths.current() );
+ m_clipPaths.remove();
+}
+
+void
+VKoPainter::setPen( const VStroke &stroke )
+{
+ delete m_stroke;
+ m_stroke = new VStroke;
+ *m_stroke = stroke;
+}
+
+void
+VKoPainter::setPen( const QColor &c )
+{
+ delete m_stroke;
+ m_stroke = new VStroke;
+
+ float r = static_cast<float>( c.red() ) / 255.0;
+ float g = static_cast<float>( c.green() ) / 255.0;
+ float b = static_cast<float>( c.blue() ) / 255.0;
+
+ VColor color;
+ color.set( r, g, b );
+ m_stroke->setColor( color );
+}
+
+void
+VKoPainter::setPen( Qt::PenStyle style )
+{
+ if( style == Qt::NoPen )
+ {
+ delete m_stroke;
+ m_stroke = 0L;
+ }
+}
+
+void
+VKoPainter::setBrush( const QColor &c )
+{
+ delete m_fill;
+ m_fill = new VFill;
+
+ float r = static_cast<float>( c.red() ) / 255.0;
+ float g = static_cast<float>( c.green() ) / 255.0;
+ float b = static_cast<float>( c.blue() ) / 255.0;
+
+ VColor color;
+ color.set( r, g, b );
+ m_fill->setColor( color );
+}
+
+void
+VKoPainter::setBrush( Qt::BrushStyle style )
+{
+ if( style == Qt::NoBrush )
+ {
+ delete m_fill;
+ m_fill = 0L;
+ }
+}
+
+void
+VKoPainter::setBrush( const VFill &fill )
+{
+ delete m_fill;
+ m_fill = new VFill;
+ *m_fill = fill;
+}
+
+void
+VKoPainter::save()
+{
+}
+
+void
+VKoPainter::restore()
+{
+}
+
+void
+VKoPainter::setRasterOp( Qt::RasterOp )
+{
+}
+
+void
+VKoPainter::clampToViewport( int &x0, int &y0, int &x1, int &y1 )
+{
+ // clamp to viewport
+ x0 = kMax( x0, 0 );
+ x0 = kMin( x0, int( m_width ) );
+ y0 = kMax( y0, 0 );
+ y0 = kMin( y0, int ( m_height ) );
+ x1 = kMax( x1, 0 );
+ x1 = kMin( x1, int( m_width ) );
+ y1 = kMax( y1, 0 );
+ y1 = kMin( y1, int( m_height ) );
+}
+
+void
+VKoPainter::clampToViewport( const ArtSVP &svp, int &x0, int &y0, int &x1, int &y1 )
+{
+ // get SVP bbox
+ ArtDRect bbox;
+ art_drect_svp( &bbox, &svp );
+ // Remove comments if we really decide for SVP bbox usage
+ //m_bbox = KoRect( bbox.x0, bbox.y0, bbox.x1 - bbox.x0, bbox.y1 - bbox.y0 );
+
+ // clamp to viewport
+ x0 = int( bbox.x0 );
+ x0 = kMax( x0, 0 );
+ x0 = kMin( x0, int( m_width ) );
+ y0 = int( bbox.y0 );
+ y0 = kMax( y0, 0 );
+ y0 = kMin( y0, int ( m_height ) );
+ x1 = int( bbox.x1 ) + 1;
+ x1 = kMax( x1, 0 );
+ x1 = kMin( x1, int( m_width ) );
+ y1 = int( bbox.y1 ) + 1;
+ y1 = kMax( y1, 0 );
+ y1 = kMin( y1, int( m_height ) );
+}
+
+void
+VKoPainter::drawVPath( ArtVpath *vec )
+{
+ ArtSVP *strokeSvp = 0L;
+ ArtSVP *fillSvp = 0L;
+
+ // set up world matrix
+ double affine[6];
+ affine[0] = m_matrix.m11();
+ affine[1] = 0;//m_matrix.m12();
+ affine[2] = 0;//m_matrix.m21();
+ affine[3] = m_matrix.m22();
+ affine[4] = m_matrix.dx();
+ affine[5] = m_matrix.dy();
+ ArtVpath *temp = art_vpath_affine_transform( vec, affine );
+ art_free( vec );
+ vec = temp;
+
+ int af = 0;
+ int as = 0;
+ art_u32 fillColor = 0;
+
+ // filling
+ QColor color;
+ if( m_fill && m_fill->type() != VFill::none )
+ {
+ color = m_fill->color();
+ af = qRound( 255 * m_fill->color().opacity() );
+ fillColor = ( 0 << 24 ) | ( color.blue() << 16 ) | ( color.green() << 8 ) | color.red();
+
+ ArtSvpWriter *swr;
+ ArtSVP *temp;
+ temp = art_svp_from_vpath( vec );
+
+ if( m_fillRule == evenOdd )
+ swr = art_svp_writer_rewind_new( ART_WIND_RULE_ODDEVEN );
+ else
+ swr = art_svp_writer_rewind_new( ART_WIND_RULE_NONZERO );
+
+ art_svp_intersector( temp, swr );
+ fillSvp = art_svp_writer_rewind_reap( swr );
+
+ art_svp_free( temp );
+ }
+
+ art_u32 strokeColor = 0;
+ // stroke
+ if( m_stroke && m_stroke->type() != VStroke::none )
+ {
+ ArtPathStrokeCapType capStyle = ART_PATH_STROKE_CAP_BUTT;
+ ArtPathStrokeJoinType joinStyle = ART_PATH_STROKE_JOIN_MITER;
+ // TODO : non rgb support ?
+
+ color = m_stroke->color();
+ as = qRound( 255 * m_stroke->color().opacity() );
+ strokeColor = ( 0 << 24 ) | ( color.blue() << 16 ) | ( color.green() << 8 ) | color.red();
+
+ double ratio = m_zoomFactor;//sqrt(pow(affine[0], 2) + pow(affine[3], 2)) / sqrt(2);
+ if( m_stroke->dashPattern().array().count() > 0 )
+ {
+ // there are dashes to be rendered
+ ArtVpathDash dash;
+ dash.offset = m_stroke->dashPattern().offset() * ratio;
+ dash.n_dash = m_stroke->dashPattern().array().count();
+ double *dashes = new double[ dash.n_dash ];
+ for( int i = 0; i < dash.n_dash; i++ )
+ dashes[i] = m_stroke->dashPattern().array()[i] * ratio;
+
+ dash.dash = dashes;
+ // get the dashed VPath and use that for the stroke render operation
+ ArtVpath *vec2 = art_vpath_dash( vec, &dash );
+ art_free( vec );
+
+ vec = vec2;
+ delete [] dashes;
+ }
+ // caps translation karbon -> art
+ if( m_stroke->lineCap() == VStroke::capRound )
+ capStyle = ART_PATH_STROKE_CAP_ROUND;
+ else if( m_stroke->lineCap() == VStroke::capSquare )
+ capStyle = ART_PATH_STROKE_CAP_SQUARE;
+
+ // join translation karbon -> art
+ if( m_stroke->lineJoin() == VStroke::joinRound )
+ joinStyle = ART_PATH_STROKE_JOIN_ROUND;
+ else if( m_stroke->lineJoin() == VStroke::joinBevel )
+ joinStyle = ART_PATH_STROKE_JOIN_BEVEL;
+
+ // zoom stroke width;
+ strokeSvp = art_svp_vpath_stroke( vec, joinStyle, capStyle, ratio * m_stroke->lineWidth(), m_stroke->miterLimit(), 0.25 );
+ }
+
+ int x0, y0, x1, y1;
+
+ // render the svp to the buffer
+ if( strokeSvp )
+ {
+ if( m_stroke && m_stroke->type() == VStroke::grad )
+ applyGradient( strokeSvp, false );
+ else if( m_stroke && m_stroke->type() == VStroke::patt )
+ applyPattern( strokeSvp, false );
+ else
+ {
+ clampToViewport( *strokeSvp, x0, y0, x1, y1 );
+ if( x0 != x1 && y0 != y1 )
+ art_rgb_svp_alpha_( strokeSvp, x0, y0, x1, y1, strokeColor, as, m_buffer + x0 * 4 + y0 * m_width * 4, m_width * 4, 0 );
+ }
+ art_svp_free( strokeSvp );
+ }
+
+ if( fillSvp )
+ {
+ if( m_fill && m_fill->type() == VFill::grad )
+ applyGradient( fillSvp, true );
+ else if( m_fill && m_fill->type() == VFill::patt )
+ applyPattern( fillSvp, true );
+ else
+ {
+ clampToViewport( *fillSvp, x0, y0, x1, y1 );
+ if( x0 != x1 && y0 != y1 )
+ art_rgb_svp_alpha_( fillSvp, x0, y0, x1, y1, fillColor, af, m_buffer + x0 * 4 + y0 * m_width * 4, m_width * 4, 0 );
+ }
+ art_svp_free( fillSvp );
+ }
+
+ //delete m_stroke;
+ //m_stroke = 0L;
+ //delete m_fill;
+ //m_fill = 0L;
+
+ art_free( vec );
+}
+
+void
+VKoPainter::applyPattern( ArtSVP *svp, bool fill )
+{
+ if(!svp) {
+ return;
+ }
+
+ int x0, y0, x1, y1;
+ clampToViewport( *svp, x0, y0, x1, y1 );
+
+ ArtRender *render = 0L;
+
+ VPattern pat = fill ? m_fill->pattern() : m_stroke->pattern();
+ if( !pat.isValid() ) {
+ pat.load( KGlobal::iconLoader()->iconPath( "karbon.png", -KIcon::SizeMedium ) ); }
+ if( !pat.isValid() ) {
+ pat = *(dynamic_cast<VPattern *>(KarbonFactory::rServer()->patterns().getFirst() )) ;}
+
+ ArtPattern *pattern = art_new( ArtPattern, 1 );
+
+ double dx = ( pat.vector().x() - pat.origin().x() ) * m_zoomFactor;
+ double dy = ( pat.vector().y() - pat.origin().y() ) * m_zoomFactor;
+
+ pattern->twidth = pat.tileWidth();
+ pattern->theight = pat.tileHeight();
+ pattern->buffer = pat.pixels();
+ pattern->opacity = fill ? short( m_fill->color().opacity() * 255.0 ) : short( m_stroke->color().opacity() * 255.0 );
+ pattern->angle = atan2( -dy, dx );
+
+ if( x0 != x1 && y0 != y1 )
+ {
+ render = art_render_new( x0, y0, x1, y1, m_buffer + 4 * int(x0) + m_width * 4 * int(y0), m_width * 4, 3, 8, ART_ALPHA_PREMUL, 0 );
+ art_render_svp( render, svp );
+ art_render_pattern( render, pattern, ART_FILTER_HYPER );
+ }
+
+ if( render )
+ art_render_invoke( render );
+ art_free( pattern );
+}
+
+void
+VKoPainter::applyGradient( ArtSVP *svp, bool fill )
+{
+ int x0, y0, x1, y1;
+ clampToViewport( *svp, x0, y0, x1, y1 );
+
+ ArtRender *render = 0L;
+
+ VGradient gradient = fill ? m_fill->gradient() : m_stroke->gradient();
+ float opa = fill ? m_fill->color().opacity() : m_stroke->color().opacity();
+
+ if( gradient.type() == VGradient::linear )
+ {
+ ArtGradientLinear *linear = art_new( ArtGradientLinear, 1 );
+
+ // TODO : make variable
+ if( gradient.repeatMethod() == VGradient::none )
+ linear->spread = ART_GRADIENT_PAD;
+ else if( gradient.repeatMethod() == VGradient::repeat )
+ linear->spread = ART_GRADIENT_REPEAT;
+ else if( gradient.repeatMethod() == VGradient::reflect )
+ linear->spread = ART_GRADIENT_REFLECT;
+
+ double _x1 = gradient.origin().x();
+ double _x2 = gradient.vector().x();
+ double _y2 = gradient.origin().y();
+ double _y1 = gradient.vector().y();
+
+ double dx = ( _x2 - _x1 ) * m_zoomFactor;
+ _y1 = m_matrix.m22() * _y1 + m_matrix.dy() / m_zoomFactor;
+ _y2 = m_matrix.m22() * _y2 + m_matrix.dy() / m_zoomFactor;
+ double dy = ( _y1 - _y2 ) * m_zoomFactor;
+ double scale = 1.0 / ( dx * dx + dy * dy );
+
+ linear->a = dx * scale;
+ linear->b = dy * scale;
+ linear->c = -( ( _x1 * m_zoomFactor + m_matrix.dx() ) * linear->a +
+ ( _y2 * m_zoomFactor ) * linear->b );
+
+ // get stop array
+ int offsets = -1;
+ linear->stops = buildStopArray( gradient, offsets );
+ linear->n_stops = offsets;
+
+ if( x0 != x1 && y0 != y1 && offsets >= 0 )
+ {
+ render = art_render_new( x0, y0, x1, y1, m_buffer + 4 * int(x0) + m_width * 4 * int(y0), m_width * 4, 3, 8, ART_ALPHA_PREMUL, 0 );
+ int opacity = int( opa * 255.0 );
+ art_render_svp( render, svp );
+ art_render_mask_solid (render, (opacity << 8) + opacity + (opacity >> 7));
+ art_karbon_render_gradient_linear( render, linear, ART_FILTER_NEAREST );
+ art_render_invoke( render );
+ }
+ art_free( linear->stops );
+ art_free( linear );
+ }
+ else if( gradient.type() == VGradient::radial )
+ {
+ ArtGradientRadial *radial = art_new( ArtGradientRadial, 1 );
+
+ // TODO : make variable
+ if( gradient.repeatMethod() == VGradient::none )
+ radial->spread = ART_GRADIENT_PAD;
+ else if( gradient.repeatMethod() == VGradient::repeat )
+ radial->spread = ART_GRADIENT_REPEAT;
+ else if( gradient.repeatMethod() == VGradient::reflect )
+ radial->spread = ART_GRADIENT_REFLECT;
+
+ radial->affine[0] = m_matrix.m11();
+ radial->affine[1] = m_matrix.m12();
+ radial->affine[2] = m_matrix.m21();
+ radial->affine[3] = m_matrix.m22();
+ radial->affine[4] = m_matrix.dx();
+ radial->affine[5] = m_matrix.dy();
+
+ double cx = gradient.origin().x() * m_zoomFactor;
+ double cy = gradient.origin().y() * m_zoomFactor;
+ double fx = gradient.focalPoint().x() * m_zoomFactor;
+ double fy = gradient.focalPoint().y() * m_zoomFactor;
+ double r = sqrt( pow( gradient.vector().x() - gradient.origin().x(), 2 ) +
+ pow( gradient.vector().y() - gradient.origin().y(), 2 ) );
+ r *= m_zoomFactor;
+
+ radial->fx = (fx - cx) / r;
+ radial->fy = (fy - cy) / r;
+
+ double aff1[6], aff2[6];
+ art_affine_scale( aff1, r, r);
+ art_affine_translate( aff2, cx, cy );
+ art_affine_multiply( aff1, aff1, aff2 );
+ art_affine_multiply( aff1, aff1, radial->affine );
+ art_affine_invert( radial->affine, aff1 );
+
+ // get stop array
+ int offsets = -1;
+ radial->stops = buildStopArray( gradient, offsets );
+ radial->n_stops = offsets;
+
+ if( x0 != x1 && y0 != y1 && offsets >= 0 )
+ {
+ render = art_render_new( x0, y0, x1, y1, m_buffer + 4 * x0 + m_width * 4 * y0, m_width * 4, 3, 8, ART_ALPHA_PREMUL, 0 );
+ int opacity = int( opa * 255.0 );
+ art_render_svp( render, svp );
+ art_render_mask_solid (render, (opacity << 8) + opacity + (opacity >> 7));
+ art_karbon_render_gradient_radial( render, radial, ART_FILTER_NEAREST );
+ art_render_invoke( render );
+ }
+ art_free( radial->stops );
+ art_free( radial );
+ }
+ else if( gradient.type() == VGradient::conic )
+ {
+ ArtGradientConical *conical = art_new( ArtGradientConical, 1 );
+
+ // TODO : make variable
+ if( gradient.repeatMethod() == VGradient::none )
+ conical->spread = ART_GRADIENT_PAD;
+ else if( gradient.repeatMethod() == VGradient::repeat )
+ conical->spread = ART_GRADIENT_REPEAT;
+ else if( gradient.repeatMethod() == VGradient::reflect )
+ conical->spread = ART_GRADIENT_REFLECT;
+
+ double cx = gradient.origin().x() * m_zoomFactor;
+ cx = m_matrix.m11() * cx + m_matrix.dx();
+ double cy = gradient.origin().y() * m_zoomFactor;
+ cy = m_matrix.m22() * cy + m_matrix.dy();
+ double r = sqrt( pow( gradient.vector().x() - gradient.origin().x(), 2 ) +
+ pow( gradient.vector().y() - gradient.origin().y(), 2 ) );
+ r *= m_zoomFactor;
+
+ conical->cx = cx;
+ conical->cy = cy;
+ conical->r = r;
+
+ // get stop array
+ int offsets = -1;
+ conical->stops = buildStopArray( gradient, offsets );
+ conical->n_stops = offsets;
+
+ if( x0 != x1 && y0 != y1 && offsets >= 0 )
+ {
+ render = art_render_new( x0, y0, x1, y1, m_buffer + 4 * x0 + m_width * 4 * y0, m_width * 4, 3, 8, ART_ALPHA_PREMUL, 0 );
+ int opacity = int( opa * 255.0 );
+ art_render_svp( render, svp );
+ art_render_mask_solid (render, (opacity << 8) + opacity + (opacity >> 7));
+ art_karbon_render_gradient_conical( render, conical, ART_FILTER_NEAREST );
+ art_render_invoke( render );
+ }
+ art_free( conical->stops );
+ art_free( conical );
+ }
+}
+
+ArtGradientStop *
+VKoPainter::buildStopArray( VGradient &gradient, int &offsets )
+{
+ // TODO : make this generic
+ QPtrVector<VColorStop> colorStops = gradient.colorStops();
+ offsets = colorStops.count();
+
+ ArtGradientStop *stopArray = art_new( ArtGradientStop, offsets * 2 - 1 );
+
+ for( int offset = 0 ; offset < offsets ; offset++ )
+ {
+ double ramp = colorStops[ offset ]->rampPoint;
+ //double mid = colorStops[ offset ]->midPoint;
+ stopArray[ offset * 2 ].offset = ramp;
+
+ QColor qStopColor = colorStops[ offset ]->color;
+ int r = qRed( qStopColor.rgb() );
+ int g = qGreen( qStopColor.rgb() );
+ int b = qBlue( qStopColor.rgb() );
+ art_u32 rgba = (r << 24) | (g << 16) | (b << 8) | qAlpha(qStopColor.rgb());
+ /* convert from separated to premultiplied alpha */
+ int a = int( colorStops[ offset]->color.opacity() * 255.0 );
+ r = (rgba >> 24) * a + 0x80;
+ r = (r + (r >> 8)) >> 8;
+ g = ((rgba >> 16) & 0xff) * a + 0x80;
+ g = (g + (g >> 8)) >> 8;
+ b = ((rgba >> 8) & 0xff) * a + 0x80;
+ b = (b + (b >> 8)) >> 8;
+ stopArray[ offset * 2 ].color[ 0 ] = ART_PIX_MAX_FROM_8(r);
+ stopArray[ offset * 2 ].color[ 1 ] = ART_PIX_MAX_FROM_8(g);
+ stopArray[ offset * 2 ].color[ 2 ] = ART_PIX_MAX_FROM_8(b);
+ stopArray[ offset * 2 ].color[ 3 ] = ART_PIX_MAX_FROM_8(a);
+
+ if( offset + 1 != offsets )
+ {
+ stopArray[ offset * 2 + 1 ].offset = ramp + ( colorStops[ offset + 1 ]->rampPoint - ramp ) * colorStops[ offset ]->midPoint;
+
+ QColor qStopColor2 = colorStops[ offset + 1 ]->color;
+ rgba = int(r + ((qRed(qStopColor2.rgb()) - r)) * 0.5) << 24 |
+ int(g + ((qGreen(qStopColor2.rgb()) - g)) * 0.5) << 16 |
+ int(b + ((qBlue(qStopColor2.rgb()) - b)) * 0.5) << 8 |
+ qAlpha(qStopColor2.rgb());
+ /* convert from separated to premultiplied alpha */
+ int a = int( colorStops[ offset]->color.opacity() * 255.0 );
+ r = (rgba >> 24) * a + 0x80;
+ r = (r + (r >> 8)) >> 8;
+ g = ((rgba >> 16) & 0xff) * a + 0x80;
+ g = (g + (g >> 8)) >> 8;
+ b = ((rgba >> 8) & 0xff) * a + 0x80;
+ b = (b + (b >> 8)) >> 8;
+ stopArray[ offset * 2 + 1 ].color[ 0 ] = ART_PIX_MAX_FROM_8(r);
+ stopArray[ offset * 2 + 1 ].color[ 1 ] = ART_PIX_MAX_FROM_8(g);
+ stopArray[ offset * 2 + 1 ].color[ 2 ] = ART_PIX_MAX_FROM_8(b);
+ stopArray[ offset * 2 + 1 ].color[ 3 ] = ART_PIX_MAX_FROM_8(a);
+ }
+ }
+
+ offsets = offsets * 2 - 1;
+ return stopArray;
+}
+
+void
+VKoPainter::drawNode( const KoPoint& p, int width )
+{
+ if( !m_bDrawNodes ) return;
+
+ KoPoint _p( m_matrix.map( QPoint( int( p.x() * m_zoomFactor ), int( p.y() * m_zoomFactor ) ) ) );
+ int x1 = int( _p.x() - width );
+ int x2 = int( _p.x() + width );
+ int y1 = int( _p.y() - width );
+ int y2 = int( _p.y() + width );
+
+ clampToViewport( x1, y1, x2, y2 );
+
+ int baseindex = 4 * x1 + ( m_width * 4 * y1 );
+
+ QColor color = m_fill->color();
+ for( int i = 0; i < y2 - y1; i++ )
+ {
+ for( int j = 0; j < x2 - x1; j++ )
+ {
+ m_buffer[ baseindex + 4 * j + ( m_width * 4 * i ) ] = color.red();
+ m_buffer[ baseindex + 4 * j + ( m_width * 4 * i ) + 1 ] = color.green();
+ m_buffer[ baseindex + 4 * j + ( m_width * 4 * i ) + 2 ] = color.blue();
+ m_buffer[ baseindex + 4 * j + ( m_width * 4 * i ) + 3 ] = 0xFF;
+ }
+ }
+}
+
+void
+VKoPainter::drawImage( const QImage &image, const QWMatrix &affine )
+{
+ // set up world matrix
+ double affineresult[6];
+
+ affineresult[0] = affine.m11() * m_matrix.m11() * m_zoomFactor + affine.m12() * m_matrix.m21();
+ affineresult[1] = (affine.m11() * m_matrix.m12() + affine.m12() * m_matrix.m22() ) * m_zoomFactor;
+ affineresult[2] = (affine.m21() * m_matrix.m11() + affine.m22() * m_matrix.m21() ) * m_zoomFactor;
+ affineresult[3] = affine.m22() * m_matrix.m22() * m_zoomFactor + affine.m21() * m_matrix.m12();
+ affineresult[4] = m_matrix.dx() + affine.dx() * m_zoomFactor;
+ affineresult[5] = m_matrix.dy() - affine.dy() * m_zoomFactor;
+
+ //art_affine_scale( affineresult, m_zoomFactor, m_zoomFactor);
+ /*kdDebug(38000) << "affineresult[0] : " << affineresult[0] << endl;
+ kdDebug(38000) << "affineresult[1] : " << affineresult[1] << endl;
+ kdDebug(38000) << "affineresult[2] : " << affineresult[2] << endl;
+ kdDebug(38000) << "affineresult[3] : " << affineresult[3] << endl;
+ kdDebug(38000) << "affineresult[4] : " << affineresult[4] << endl;
+ kdDebug(38000) << "affineresult[5] : " << affineresult[5] << endl;
+ kdDebug(38000) << "m_matrix.dx() : " << m_matrix.dx() << endl;
+ kdDebug(38000) << "affine.dx() : " << affine.dx() << endl;
+ kdDebug(38000) << "image.height() : " << image.height() << endl;*/
+ art_rgba_affine( m_buffer, 0, 0, m_width, m_height, m_width * 4,
+ image.bits(), image.width(), image.height(), image.width() * 4,
+ affineresult, ART_FILTER_NEAREST, 0L );
+}
+
+void
+VKoPainter::drawRect( const KoRect &r )
+{
+ newPath();
+ moveTo( r.topLeft() );
+ lineTo( r.topRight() );
+ lineTo( r.bottomRight() );
+ lineTo( r.bottomLeft() );
+ lineTo( r.topLeft() );
+ fillPath();
+ strokePath();
+}
+
+void
+VKoPainter::drawRect( double x, double y, double w, double h )
+{
+ drawRect( KoRect( x, y, w, h ) );
+}
+