diff options
Diffstat (limited to 'lib/kwmf/kowmfwrite.cc')
-rw-r--r-- | lib/kwmf/kowmfwrite.cc | 456 |
1 files changed, 456 insertions, 0 deletions
diff --git a/lib/kwmf/kowmfwrite.cc b/lib/kwmf/kowmfwrite.cc new file mode 100644 index 00000000..1c553391 --- /dev/null +++ b/lib/kwmf/kowmfwrite.cc @@ -0,0 +1,456 @@ +/* This file is part of the KDE libraries + * Copyright (c) 2003 thierry lorthiois (lorthioist@wanadoo.fr) + * + * This library is free software; you can redistribute it and/or + * modify it under the terms of the GNU Library General Public + * License version 2 as published by the Free Software Foundation. + * + * 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. +*/ + +#include <math.h> +#include <qfile.h> +#include <qdatastream.h> + +#include <kdebug.h> + +#include "kowmfstruct.h" +#include "kowmfreadprivate.h" +#include "kowmfwrite.h" + +/** + * Private data + */ +class KoWmfWritePrivate +{ +public: + QRect mBBox; // bounding rectangle + int mDpi; // number of point per inch for the default size + int mMaxRecordSize; + + // memory allocation for WMF file + QFile mFileOut; + QDataStream mSt; +}; + + + +KoWmfWrite::KoWmfWrite( const QString& fileName ) { + d = new KoWmfWritePrivate; + + d->mDpi = 1024; + d->mMaxRecordSize = 0; + d->mFileOut.setName( fileName ); +} + +KoWmfWrite::~KoWmfWrite() { + delete d; +} + + +void KoWmfWrite::setDefaultDpi( int dpi ) { + d->mDpi = dpi; +} + + +//----------------------------------------------------------------------------- +// Virtual Painter => create the WMF + +bool KoWmfWrite::begin() { + + if ( !d->mFileOut.open( IO_WriteOnly ) ) + { + kdDebug() << "Cannot open file " << QFile::encodeName(d->mFileOut.name()) << endl; + return false; + } + d->mSt.setDevice( &d->mFileOut ); + d->mSt.setByteOrder( QDataStream::LittleEndian ); + + // reserved placeable and standard header + for ( int i=0 ; i < 10 ; i++ ) { + d->mSt << (Q_UINT32)0; + } + + // initialize the stack of objects + // Pen + d->mSt << (Q_UINT32)8 << (Q_UINT16)0x02FA; + d->mSt << (Q_UINT16)5 << (Q_UINT16)0 << (Q_UINT16)0 << (Q_UINT32)0; + // Brush + d->mSt << (Q_UINT32)7 << (Q_UINT16)0x02FC; + d->mSt << (Q_UINT16)1 << (Q_UINT32)0 << (Q_UINT16)0; + for ( int i=0 ; i < 4 ; i++ ) { + d->mSt << (Q_UINT32)8 << (Q_UINT16)0x02FA << (Q_UINT16)0 << (Q_UINT32)0 << (Q_UINT32)0; + } + d->mMaxRecordSize = 8; + + return true; +} + + +bool KoWmfWrite::end() { + WmfPlaceableHeader pheader = { 0x9AC6CDD7, 0, 0, 0, 0, 0, 0, 0, 0 }; + Q_UINT16 checksum; + + // End of the wmf file + d->mSt << (Q_UINT32)3 << (Q_UINT16)0; + + // adjust header + pheader.left = d->mBBox.left(); + pheader.top = d->mBBox.top(); + pheader.right = d->mBBox.right(); + pheader.bottom = d->mBBox.bottom(); + pheader.inch = d->mDpi; + checksum = KoWmfReadPrivate::calcCheckSum( &pheader ); + + // write headers + d->mFileOut.at( 0 ); + d->mSt << (Q_UINT32)0x9AC6CDD7 << (Q_UINT16)0; + d->mSt << (Q_INT16)d->mBBox.left() << (Q_INT16)d->mBBox.top() << (Q_INT16)d->mBBox.right() << (Q_INT16)d->mBBox.bottom(); + d->mSt << (Q_UINT16)d->mDpi << (Q_UINT32)0 << checksum; + d->mSt << (Q_UINT16)1 << (Q_UINT16)9 << (Q_UINT16)0x300 << (Q_UINT32)(d->mFileOut.size()/2); + d->mSt << (Q_UINT16)6 << (Q_UINT32)d->mMaxRecordSize << (Q_UINT16)0; + + d->mFileOut.close(); + + return true; +} + + +void KoWmfWrite::save() { + d->mSt << (Q_UINT32)3 << (Q_UINT16)0x001E; +} + + +void KoWmfWrite::restore() { + d->mSt << (Q_UINT32)4 << (Q_UINT16)0x0127 << (Q_UINT16)1; +} + + +void KoWmfWrite::setPen( const QPen &pen ) { + int style; + int max = sizeof(koWmfStylePen) / sizeof(Qt::SolidLine); + + // we can't delete an object currently selected + // select another object + d->mSt << (Q_UINT32)4 << (Q_UINT16)0x012D << (Q_UINT16)0; + // delete object + d->mSt << (Q_UINT32)4 << (Q_UINT16)0x01f0 << (Q_UINT16)2; + + for ( style=0 ; style < max ; style++ ) { + if ( koWmfStylePen[ style ] == pen.style() ) break; + } + if ( style == max ) { + // SolidLine + style = 0; + } + d->mSt << (Q_UINT32)8 << (Q_UINT16)0x02FA; + d->mSt << (Q_UINT16)style << (Q_UINT16)pen.width() << (Q_UINT16)0 << (Q_UINT32)winColor( pen.color() ); + + // select object + d->mSt << (Q_UINT32)4 << (Q_UINT16)0x012D << (Q_UINT16)2; +} + + +void KoWmfWrite::setBrush( const QBrush &brush ) { + int style; + int max = sizeof(koWmfStyleBrush) / sizeof(Qt::NoBrush); + + // we can't delete an object currently selected + // select another object + d->mSt << (Q_UINT32)4 << (Q_UINT16)0x012D << (Q_UINT16)1; + // delete object + d->mSt << (Q_UINT32)4 << (Q_UINT16)0x01f0 << (Q_UINT16)3; + + for ( style=0 ; style < max ; style++ ) { + if ( koWmfStyleBrush[ style ] == brush.style() ) break; + } + if ( style == max ) { + // SolidPattern + style = 0; + } + d->mSt << (Q_UINT32)7 << (Q_UINT16)0x02FC; + d->mSt << (Q_UINT16)style << (Q_UINT32)winColor( brush.color() ) << (Q_UINT16)0; + + // select object + d->mSt << (Q_UINT32)4 << (Q_UINT16)0x012D << (Q_UINT16)3; +} + + +void KoWmfWrite::setFont( const QFont & ) { +} + + +void KoWmfWrite::setBackgroundColor( const QColor &c ) { + d->mSt << (Q_UINT32)5 << (Q_UINT16)0x0201 << (Q_UINT32)winColor( c ); +} + + +void KoWmfWrite::setBackgroundMode( Qt::BGMode mode ) { + d->mSt << (Q_UINT32)4 << (Q_UINT16)0x0102; + if ( mode == Qt::TransparentMode ) + d->mSt << (Q_UINT16)1; + else + d->mSt << (Q_UINT16)0; +} + + +void KoWmfWrite::setRasterOp( Qt::RasterOp op ) { + d->mSt << (Q_UINT32)5 << (Q_UINT16)0x0104 << (Q_UINT32)qtRasterToWin32( op ); +} + + +void KoWmfWrite::setWindow( int left, int top, int width, int height ) { + d->mBBox.setRect( left, top, width, height ); + + // windowOrg + d->mSt << (Q_UINT32)5 << (Q_UINT16)0x020B << (Q_UINT16)top << (Q_UINT16)left; + + // windowExt + d->mSt << (Q_UINT32)5 << (Q_UINT16)0x020C << (Q_UINT16)height << (Q_UINT16)width; +} + + +void KoWmfWrite::setClipRegion( const QRegion & ) { + +} + + +void KoWmfWrite::clipping( bool enable ) { + if ( !enable ) { + // clipping region == bounding rectangle + setClipRegion( d->mBBox ); + } +} + + +void KoWmfWrite::moveTo( int left, int top ) { + d->mSt << (Q_UINT32)5 << (Q_UINT16)0x0214 << (Q_UINT16)top << (Q_UINT16)left; +} + + +void KoWmfWrite::lineTo( int left, int top ) { + d->mSt << (Q_UINT32)5 << (Q_UINT16)0x0213 << (Q_UINT16)top << (Q_UINT16)left; +} + + +void KoWmfWrite::drawRect( int left, int top, int width, int height ) { + QRect rec( left, top, width, height ); + + d->mSt << (Q_UINT32)7 << (Q_UINT16)0x041B; + d->mSt << (Q_UINT16)rec.bottom() << (Q_UINT16)rec.right() << (Q_UINT16)rec.top() << (Q_UINT16)rec.left(); +} + + +void KoWmfWrite::drawRoundRect( int left, int top, int width, int height , int roudw, int roudh ) { + int widthCorner, heightCorner; + QRect rec( left, top, width, height ); + + // convert percentage (roundw, roudh) in (widthCorner, heightCorner) + widthCorner = ( roudw * width ) / 100; + heightCorner = ( roudh * height ) / 100; + + d->mSt << (Q_UINT32)9 << (Q_UINT16)0x061C << (Q_UINT16)heightCorner << (Q_UINT16)widthCorner; + d->mSt << (Q_UINT16)rec.bottom() << (Q_UINT16)rec.right() << (Q_UINT16)rec.top() << (Q_UINT16)rec.left(); + + d->mMaxRecordSize = QMAX( d->mMaxRecordSize, 9 ); +} + + +void KoWmfWrite::drawEllipse( int left, int top, int width, int height ) { + QRect rec( left, top, width, height ); + + d->mSt << (Q_UINT32)7 << (Q_UINT16)0x0418; + d->mSt << (Q_UINT16)rec.bottom() << (Q_UINT16)rec.right() << (Q_UINT16)rec.top() << (Q_UINT16)rec.left(); +} + + +void KoWmfWrite::drawArc( int left, int top, int width, int height , int a, int alen ) { + int xCenter, yCenter; + int offXStart, offYStart, offXEnd, offYEnd; + + angleToxy( offXStart, offYStart, offXEnd, offYEnd, a, alen ); + xCenter = left + (width / 2); + yCenter = top + (height / 2); + + d->mSt << (Q_UINT32)11 << (Q_UINT16)0x0817; + d->mSt << (Q_UINT16)(yCenter + offYEnd) << (Q_UINT16)(xCenter + offXEnd); + d->mSt << (Q_UINT16)(yCenter + offYStart) << (Q_UINT16)(xCenter + offXStart); + d->mSt << (Q_UINT16)(top + height) << (Q_UINT16)(left + width); + d->mSt << (Q_UINT16)top << (Q_UINT16)left; + + d->mMaxRecordSize = QMAX( d->mMaxRecordSize, 11 ); +} + + +void KoWmfWrite::drawPie( int left, int top, int width, int height , int a, int alen ) { + int xCenter, yCenter; + int offXStart, offYStart, offXEnd, offYEnd; + + angleToxy( offXStart, offYStart, offXEnd, offYEnd, a, alen ); + xCenter = left + (width / 2); + yCenter = top + (height / 2); + + d->mSt << (Q_UINT32)11 << (Q_UINT16)0x081A; + d->mSt << (Q_UINT16)(yCenter + offYEnd) << (Q_UINT16)(xCenter + offXEnd); + d->mSt << (Q_UINT16)(yCenter + offYStart) << (Q_UINT16)(xCenter + offXStart); + d->mSt << (Q_UINT16)(top + height) << (Q_UINT16)(left + width); + d->mSt << (Q_UINT16)top << (Q_UINT16)left; + + d->mMaxRecordSize = QMAX( d->mMaxRecordSize, 11 ); +} + + +void KoWmfWrite::drawChord( int left, int top, int width, int height , int a, int alen ) { + int xCenter, yCenter; + int offXStart, offYStart, offXEnd, offYEnd; + + angleToxy( offXStart, offYStart, offXEnd, offYEnd, a, alen ); + xCenter = left + (width / 2); + yCenter = top + (height / 2); + + d->mSt << (Q_UINT32)11 << (Q_UINT16)0x0830; + d->mSt << (Q_UINT16)(yCenter + offYEnd) << (Q_UINT16)(xCenter + offXEnd); + d->mSt << (Q_UINT16)(yCenter + offYStart) << (Q_UINT16)(xCenter + offXStart); + d->mSt << (Q_UINT16)(top + height) << (Q_UINT16)(left + width); + d->mSt << (Q_UINT16)top << (Q_UINT16)left; + + d->mMaxRecordSize = QMAX( d->mMaxRecordSize, 11 ); +} + + +void KoWmfWrite::drawPolyline( const QPointArray &pa ) { + int size = 4 + (pa.size() * 2); + + d->mSt << (Q_UINT32)size << (Q_UINT16)0x0325 << (Q_UINT16)pa.size(); + pointArray( pa ); + + d->mMaxRecordSize = QMAX( d->mMaxRecordSize, size ); +} + + +void KoWmfWrite::drawPolygon( const QPointArray &pa, bool ) { + int size = 4 + (pa.size() * 2); + + d->mSt << (Q_UINT32)size << (Q_UINT16)0x0324 << (Q_UINT16)pa.size(); + pointArray( pa ); + + d->mMaxRecordSize = QMAX( d->mMaxRecordSize, size ); +} + + +void KoWmfWrite::drawPolyPolygon( QPtrList<QPointArray>& listPa, bool ) { + + QPointArray *pa; + int sizeArrayPoly = 0; + + for ( pa = listPa.first() ; pa ; pa = listPa.next() ) { + sizeArrayPoly += (pa->size() * 2); + } + int size = 4 + listPa.count() + sizeArrayPoly; + d->mSt << (Q_UINT32)size << (Q_UINT16)0x0538 << (Q_UINT16)listPa.count(); + + // number of point for each Polygon + for ( pa = listPa.first() ; pa ; pa = listPa.next() ) { + d->mSt << (Q_UINT16)pa->size(); + } + + // list of points + for ( pa = listPa.first() ; pa ; pa = listPa.next() ) { + pointArray( *pa ); + } + + d->mMaxRecordSize = QMAX( d->mMaxRecordSize, size ); + +} + + +void KoWmfWrite::drawImage( int , int , const QImage &, int , int , int , int ) { +/* + QImage img; + + img = image; + img.setFormat( "BMP" ); + + QIODevice io = img.ioDevice(); + io.at( 14 ); // skip the BMP header + d->mSt << io.readAll(); +*/ +} + + +void KoWmfWrite::drawText( int , int , int , int , int , const QString& , double ) { +// d->mSt << (Q_UINT32)3 << (Q_UINT16)0x0A32; +} + +//----------------------------------------------------------------------------- +// Utilities and conversion Qt --> Wmf + +void KoWmfWrite::pointArray( const QPointArray &pa ) { + int left, top, i, max; + + for ( i=0, max=pa.size() ; i < max ; i++ ) { + pa.point( i, &left, &top ); + d->mSt << (Q_INT16)left << (Q_INT16)top; + } +} + + +Q_UINT32 KoWmfWrite::winColor( QColor color ) { + Q_UINT32 c; + + c = (color.red() & 0xFF); + c += ( (color.green() & 0xFF) << 8 ); + c += ( (color.blue() & 0xFF) << 16 ); + + return c; +} + + +void KoWmfWrite::angleToxy( int &xStart, int &yStart, int &xEnd, int &yEnd, int a, int alen ) { + double angleStart, angleLength; + + angleStart = ((double)a * 3.14166) / 2880; + angleLength = ((double)alen * 3.14166) / 2880; + + xStart = (int)(cos(angleStart) * 50); + yStart = -(int)(sin(angleStart) * 50); + xEnd = (int)(cos(angleLength) * 50); + yEnd = -(int)(sin(angleLength) * 50); +} + + +Q_UINT16 KoWmfWrite::qtRasterToWin16( Qt::RasterOp op ) const { + int i; + + for ( i=0 ; i < 17 ; i++ ) { + if ( koWmfOpTab16[ i ] == op ) break; + } + + if ( i < 17 ) + return (Q_UINT16)i; + else + return (Q_UINT16)0; +} + + +Q_UINT32 KoWmfWrite::qtRasterToWin32( Qt::RasterOp op ) const { + int i; + + for ( i=0 ; i < 15 ; i++ ) { + if ( koWmfOpTab32[ i ].qtRasterOp == op ) break; + } + + if ( i < 15 ) + return koWmfOpTab32[ i ].winRasterOp; + else + return koWmfOpTab32[ 0 ].winRasterOp; +} + |