summaryrefslogtreecommitdiffstats
path: root/lib/kotext/KoBorder.cpp
diff options
context:
space:
mode:
Diffstat (limited to 'lib/kotext/KoBorder.cpp')
-rw-r--r--lib/kotext/KoBorder.cpp318
1 files changed, 318 insertions, 0 deletions
diff --git a/lib/kotext/KoBorder.cpp b/lib/kotext/KoBorder.cpp
new file mode 100644
index 00000000..01cfbea0
--- /dev/null
+++ b/lib/kotext/KoBorder.cpp
@@ -0,0 +1,318 @@
+/* This file is part of the KDE project
+ Copyright (C) 2000, 2001 Thomas Zander <[email protected]>
+
+ 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.
+*/
+
+#include "KoBorder.h"
+#include <qdom.h>
+#include <kdebug.h>
+#include "KoZoomHandler.h"
+#include "KoTextFormat.h"
+#include "KoRichText.h" // for KoTextFormat
+#include "KoTextParag.h"
+#include <float.h>
+
+static const struct BorderStyle {
+ QPen::PenStyle penStyle;
+ QCString oasisName;
+ QCString uiStringStyle;
+} s_borderStyles[] = {
+ { QPen::SolidLine, "solid", "_________" }, // SOLID
+ { QPen::DashLine, "dashed", "___ ___ __" }, // DASH
+ { QPen::DotLine, "dotted", "_ _ _ _ _ _" }, // DOT
+ { QPen::DashDotLine, "dot-dash", "___ _ ___ _" }, // DASH_DOT
+ { QPen::DashDotDotLine, "dot-dot-dash", "___ _ _ ___" }, // DASH_DOT_DOT
+ { QPen::SolidLine, "double", "===========" } // DOUBLE_LINE
+};
+
+KoBorder::KoBorder()
+ : color(), m_style( SOLID )
+{
+ setPenWidth( 1 );
+}
+
+KoBorder::KoBorder( const QColor & c, BorderStyle s, double width )
+ : color( c ), m_style( s )
+{
+ setPenWidth( width );
+}
+
+bool KoBorder::operator==( const KoBorder _brd ) const {
+ return ( m_style == _brd.m_style && color == _brd.color && ptPenWidth == _brd.ptPenWidth );
+}
+
+bool KoBorder::operator!=( const KoBorder _brd ) const {
+ return ( m_style != _brd.m_style || color != _brd.color || ptPenWidth != _brd.ptPenWidth );
+}
+
+void KoBorder::setStyle(BorderStyle _style)
+{
+ m_style = _style;
+ setPenWidth( ptPenWidth );
+}
+
+void KoBorder::setPenWidth(double _w)
+{
+ ptPenWidth = _w;
+ if ( m_style == KoBorder::DOUBLE_LINE && _w > 0 )
+ ptWidth = 2 * ptPenWidth + 1;
+ else
+ ptWidth = _w;
+}
+
+QPen KoBorder::borderPen( const KoBorder & _brd, int width, QColor defaultColor )
+{
+ QPen pen( _brd.color, width );
+ if ( !_brd.color.isValid() )
+ pen.setColor( defaultColor );
+
+ pen.setStyle( s_borderStyles[ _brd.m_style ].penStyle );
+
+ return pen;
+}
+
+// KOffice-1.3 file format (deprecated)
+KoBorder KoBorder::loadBorder( const QDomElement & elem )
+{
+ KoBorder bd;
+ if ( elem.hasAttribute("red") )
+ {
+ int r = elem.attribute("red").toInt();
+ int g = elem.attribute("green").toInt();
+ int b = elem.attribute("blue").toInt();
+ bd.color.setRgb( r, g, b );
+ }
+ bd.m_style = static_cast<BorderStyle>( elem.attribute("style").toInt() );
+ bd.setPenWidth( elem.attribute("width").toDouble() );
+ return bd;
+}
+
+void KoBorder::loadFoBorder( const QString& border )
+{
+ //string like "0.088cm solid #800000"
+
+ if (border.isEmpty() || border=="none" || border=="hidden") { // in fact no border
+ setPenWidth( 0 );
+ return;
+ }
+
+ // ## isn't it faster to use QStringList::split than parse it 3 times?
+ QString _width = border.section(' ', 0, 0);
+ QCString _style = border.section(' ', 1, 1).latin1();
+ QString _color = border.section(' ', 2, 2);
+
+ //TODO: let the user choose a more precise border width (in the current unit)
+ double const penWidth = KoUnit::parseValue( _width, 1.0 );
+ //kdDebug() << "penWidth:" << penWidth << endl;
+ if ( penWidth < 1.5 )
+ setPenWidth( 1.0 );
+ else if ( penWidth < 2.5 )
+ setPenWidth( 2.0 );
+ else if ( penWidth < 3.5 )
+ setPenWidth( 3.0 );
+ else if ( penWidth < 4.5 )
+ setPenWidth( 4.0 );
+ else if ( penWidth < 5.5 )
+ setPenWidth( 5.0 );
+ else if ( penWidth < 6.5 )
+ setPenWidth( 6.0 );
+ else if ( penWidth < 7.5 )
+ setPenWidth( 7.0 );
+ else if ( penWidth < 8.5 )
+ setPenWidth( 8.0 );
+ else if ( penWidth < 9.5 )
+ setPenWidth( 9.0 );
+ else
+ setPenWidth( 10.0 );
+
+ m_style = SOLID;
+ for ( uint i = 0; i < sizeof( s_borderStyles ) / sizeof *s_borderStyles; ++i ) {
+ if ( _style == s_borderStyles[i].oasisName )
+ m_style = static_cast<BorderStyle>( i );
+ }
+
+ if ( _color.isEmpty() )
+ color = QColor();
+ else
+ color.setNamedColor( _color );
+}
+
+QString KoBorder::saveFoBorder() const
+{
+ if ( QABS( ptPenWidth ) < 1E-10 ) // i.e. ptPenWidth == 0
+ return "none";
+ //string like "2pt solid #800000"
+ QString str = QString::number( ptPenWidth, 'g', DBL_DIG );
+ str += "pt ";
+ str += s_borderStyles[ m_style ].oasisName;
+ if ( color.isValid() ) {
+ str += ' ';
+ str += color.name();
+ }
+ return str;
+}
+
+// KOffice-1.3 file format (deprecated)
+void KoBorder::save( QDomElement & elem ) const
+{
+ if (color.isValid()) {
+ elem.setAttribute("red", color.red());
+ elem.setAttribute("green", color.green());
+ elem.setAttribute("blue", color.blue());
+ }
+ elem.setAttribute("style", static_cast<int>( m_style ));
+ elem.setAttribute("width", ptPenWidth);
+}
+
+KoBorder::BorderStyle KoBorder::getStyle( const QString &style )
+{
+ for ( uint i = 0; i < sizeof( s_borderStyles ) / sizeof *s_borderStyles; ++i ) {
+ if ( style == s_borderStyles[i].uiStringStyle.data() )
+ return static_cast<BorderStyle>( i );
+ }
+ // default
+ return KoBorder::SOLID;
+}
+
+QString KoBorder::getStyle( const BorderStyle &style )
+{
+ return s_borderStyles[style].uiStringStyle;
+}
+
+int KoBorder::zoomWidthX( double ptWidth, KoZoomHandler * zoomHandler, int minborder )
+{
+ // If a border was set, then zoom it and apply a minimum of 1, so that it's always visible.
+ // If no border was set, apply minborder ( 0 for paragraphs, 1 for frames )
+ return ptWidth > 0 ? QMAX( 1, zoomHandler->zoomItX( ptWidth ) /*applies qRound*/ ) : minborder;
+}
+
+int KoBorder::zoomWidthY( double ptWidth, KoZoomHandler * zoomHandler, int minborder )
+{
+ // If a border was set, then zoom it and apply a minimum of 1, so that it's always visible.
+ // If no border was set, apply minborder ( 0 for paragraphs, 1 for frames )
+ return ptWidth > 0 ? QMAX( 1, zoomHandler->zoomItY( ptWidth ) /*applies qRound*/ ) : minborder;
+}
+
+void KoBorder::drawBorders( QPainter& painter, KoZoomHandler * zoomHandler, const QRect& rect, const KoBorder& leftBorder, const KoBorder& rightBorder, const KoBorder& topBorder, const KoBorder& bottomBorder, int minborder, const QPen& defaultPen, bool drawTopBorder /* = true */, bool drawBottomBorder /* = true */)
+{
+ int topBorderWidth = zoomWidthY( topBorder.width(), zoomHandler, minborder );
+ int bottomBorderWidth = zoomWidthY( bottomBorder.width(), zoomHandler, minborder );
+ int leftBorderWidth = zoomWidthX( leftBorder.width(), zoomHandler, minborder );
+ int rightBorderWidth = zoomWidthX( rightBorder.width(), zoomHandler, minborder );
+
+ int topBorderPenWidth = zoomWidthY( topBorder.penWidth(), zoomHandler, minborder );
+ int bottomBorderPenWidth = zoomWidthY( bottomBorder.penWidth(), zoomHandler, minborder );
+ int leftBorderPenWidth = zoomWidthX( leftBorder.penWidth(), zoomHandler, minborder );
+ int rightBorderPenWidth = zoomWidthX( rightBorder.penWidth(), zoomHandler, minborder );
+
+ // Wide pen don't draw the last pixel, so add one to the bottom and right coords
+ int lastPixelAdj = 1;
+
+ //kdDebug(32500) << "KoBorder::drawBorders widths: top=" << topBorderWidth << " bottom=" << bottomBorderWidth
+ // << " left=" << leftBorderWidth << " right=" << rightBorderWidth << endl;
+
+ //kdDebug(32500) << " penWidths: top=" << topBorderPenWidth << " bottom=" << bottomBorderPenWidth
+ // << " left=" << leftBorderPenWidth << " right=" << rightBorderPenWidth << endl;
+
+ QColor defaultColor = KoTextFormat::defaultTextColor( &painter );
+
+ if ( topBorderWidth > 0 && drawTopBorder )
+ {
+ if ( topBorder.penWidth() > 0 )
+ painter.setPen( KoBorder::borderPen( topBorder, topBorderPenWidth, defaultColor ) );
+ else
+ painter.setPen( defaultPen );
+ int y = rect.top() - topBorderWidth + topBorderPenWidth/2;
+ if ( topBorder.m_style==KoBorder::DOUBLE_LINE)
+ {
+ painter.drawLine( rect.left()-leftBorderWidth, y, rect.right()+2*(rightBorderPenWidth+lastPixelAdj), y );
+ y += topBorderPenWidth + 1;
+ painter.drawLine( rect.left()-leftBorderPenWidth, y, rect.right()+rightBorderPenWidth+lastPixelAdj, y );
+ }
+ else
+ {
+ painter.drawLine( rect.left()-leftBorderWidth, y, rect.right()+rightBorderWidth+lastPixelAdj, y );
+ }
+ }
+ if ( bottomBorderWidth > 0 && drawBottomBorder )
+ {
+ if ( bottomBorder.penWidth() > 0 )
+ painter.setPen( KoBorder::borderPen( bottomBorder, bottomBorderPenWidth, defaultColor ) );
+ else
+ painter.setPen( defaultPen );
+ //kdDebug(32500) << "bottomBorderWidth=" << bottomBorderWidth << " bottomBorderWidth/2=" << (int)bottomBorderWidth/2 << endl;
+ int y = rect.bottom() + bottomBorderPenWidth/2 + 1;
+ //kdDebug(32500) << " -> bottom=" << rect.bottom() << " y=" << y << endl;
+ if ( bottomBorder.m_style==KoBorder::DOUBLE_LINE)
+ {
+ painter.drawLine( rect.left()-leftBorderPenWidth, y, rect.right()+rightBorderPenWidth+lastPixelAdj, y );
+ y += bottomBorderPenWidth + 1;
+ painter.drawLine( rect.left()-leftBorderWidth, y, rect.right()+2*(rightBorderPenWidth+lastPixelAdj), y );
+ }
+ else
+ {
+ painter.drawLine( rect.left()-leftBorderWidth, y, rect.right()+rightBorderWidth+lastPixelAdj, y );
+ }
+ }
+ if ( leftBorderWidth > 0 )
+ {
+ if ( leftBorder.penWidth() > 0 )
+ painter.setPen( KoBorder::borderPen( leftBorder, leftBorderPenWidth, defaultColor ) );
+ else
+ painter.setPen( defaultPen );
+ int x = rect.left() - leftBorderWidth + leftBorderPenWidth/2;
+ if ( leftBorder.m_style==KoBorder::DOUBLE_LINE)
+ {
+ painter.drawLine( x, rect.top()-topBorderWidth, x, rect.bottom()+2*(bottomBorderPenWidth+lastPixelAdj) );
+ x += leftBorderPenWidth + 1;
+ painter.drawLine( x, rect.top()-topBorderPenWidth, x, rect.bottom()+bottomBorderPenWidth+lastPixelAdj );
+ }
+ else
+ {
+ int yTop = rect.top() - topBorderWidth;
+ int yBottom = rect.bottom() + bottomBorderWidth;
+ /*kdDebug(32500) << " pen=" << painter.pen() << " rect=" << rect << " topBorderWidth=" << topBorderWidth
+ << " painting from " << x << "," << yTop
+ << " to " << x << "," << yBottom << endl;*/
+ painter.drawLine( x, yTop, x, yBottom+lastPixelAdj );
+ }
+ }
+ if ( rightBorderWidth > 0 )
+ {
+ if ( rightBorder.penWidth() > 0 )
+ painter.setPen( KoBorder::borderPen( rightBorder, rightBorderPenWidth, defaultColor ) );
+ else
+ painter.setPen( defaultPen );
+ int x = rect.right() + rightBorderPenWidth/2 + 1;
+ if ( rightBorder.m_style==KoBorder::DOUBLE_LINE)
+ {
+ painter.drawLine( x, rect.top()-topBorderPenWidth, x, rect.bottom()+bottomBorderPenWidth+lastPixelAdj );
+ x += rightBorderPenWidth + 1;
+ painter.drawLine( x, rect.top()-topBorderWidth, x, rect.bottom()+2*(bottomBorderPenWidth+lastPixelAdj) );
+
+ }
+ else
+ {
+ int yTop = rect.top()-topBorderWidth;
+ int yBottom = rect.bottom()+bottomBorderWidth+lastPixelAdj;
+ /*kdDebug(32500) << " pen=" << painter.pen() << " rect=" << rect << " topBorderWidth=" << topBorderWidth
+ << " painting from " << x << "," << yTop
+ << " to " << x << "," << yBottom << endl;*/
+ painter.drawLine( x, yTop, x, yBottom );
+ }
+ }
+}