457 lines
14 KiB
C++
457 lines
14 KiB
C++
/* -*- Mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*- */
|
|
/*************************************************************************
|
|
*
|
|
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
|
|
*
|
|
* Copyright 2000, 2010 Oracle and/or its affiliates.
|
|
*
|
|
* OpenOffice.org - a multi-platform office productivity suite
|
|
*
|
|
* This file is part of OpenOffice.org.
|
|
*
|
|
* OpenOffice.org is free software: you can redistribute it and/or modify
|
|
* it under the terms of the GNU Lesser General Public License version 3
|
|
* only, as published by the Free Software Foundation.
|
|
*
|
|
* OpenOffice.org 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 Lesser General Public License version 3 for more details
|
|
* (a copy is included in the LICENSE file that accompanied this code).
|
|
*
|
|
* You should have received a copy of the GNU Lesser General Public License
|
|
* version 3 along with OpenOffice.org. If not, see
|
|
* <http://www.openoffice.org/license.html>
|
|
* for a copy of the LGPLv3 License.
|
|
*
|
|
************************************************************************/
|
|
|
|
// MARKER(update_precomp.py): autogen include statement, do not remove
|
|
#include "precompiled_xmloff.hxx"
|
|
|
|
|
|
#include <bordrhdl.hxx>
|
|
#include <xmloff/xmltoken.hxx>
|
|
#include <xmloff/xmluconv.hxx>
|
|
#include <rtl/ustrbuf.hxx>
|
|
#include <com/sun/star/uno/Any.hxx>
|
|
#include <com/sun/star/table/BorderLine2.hpp>
|
|
|
|
using ::rtl::OUString;
|
|
using ::rtl::OUStringBuffer;
|
|
|
|
using namespace ::com::sun::star;
|
|
using namespace ::xmloff::token;
|
|
|
|
const sal_Int16 API_LINE_SOLID = 0;
|
|
const sal_Int16 API_LINE_DOTTED = 1;
|
|
const sal_Int16 API_LINE_DASHED = 2;
|
|
|
|
// copied from svx/boxitem.hxx
|
|
#define DEF_LINE_WIDTH_0 1
|
|
#define DEF_LINE_WIDTH_1 35
|
|
#define DEF_LINE_WIDTH_2 88
|
|
#define DEF_LINE_WIDTH_3 141
|
|
#define DEF_LINE_WIDTH_4 176
|
|
|
|
#define DEF_MAX_LINE_WIDHT DEF_LINE_WIDTH_4
|
|
#define DEF_MAX_LINE_DIST DEF_LINE_WIDTH_2
|
|
|
|
#define DEF_DOUBLE_LINE0_OUT DEF_LINE_WIDTH_0
|
|
#define DEF_DOUBLE_LINE0_IN DEF_LINE_WIDTH_0
|
|
#define DEF_DOUBLE_LINE0_DIST DEF_LINE_WIDTH_1
|
|
|
|
#define DEF_DOUBLE_LINE1_OUT DEF_LINE_WIDTH_1
|
|
#define DEF_DOUBLE_LINE1_IN DEF_LINE_WIDTH_1
|
|
#define DEF_DOUBLE_LINE1_DIST DEF_LINE_WIDTH_1
|
|
|
|
#define DEF_DOUBLE_LINE2_OUT DEF_LINE_WIDTH_2
|
|
#define DEF_DOUBLE_LINE2_IN DEF_LINE_WIDTH_2
|
|
#define DEF_DOUBLE_LINE2_DIST DEF_LINE_WIDTH_2
|
|
|
|
#define DEF_DOUBLE_LINE3_OUT DEF_LINE_WIDTH_2
|
|
#define DEF_DOUBLE_LINE3_IN DEF_LINE_WIDTH_1
|
|
#define DEF_DOUBLE_LINE3_DIST DEF_LINE_WIDTH_2
|
|
|
|
#define DEF_DOUBLE_LINE4_OUT DEF_LINE_WIDTH_1
|
|
#define DEF_DOUBLE_LINE4_IN DEF_LINE_WIDTH_2
|
|
#define DEF_DOUBLE_LINE4_DIST DEF_LINE_WIDTH_1
|
|
|
|
#define DEF_DOUBLE_LINE5_OUT DEF_LINE_WIDTH_3
|
|
#define DEF_DOUBLE_LINE5_IN DEF_LINE_WIDTH_2
|
|
#define DEF_DOUBLE_LINE5_DIST DEF_LINE_WIDTH_2
|
|
|
|
#define DEF_DOUBLE_LINE6_OUT DEF_LINE_WIDTH_2
|
|
#define DEF_DOUBLE_LINE6_IN DEF_LINE_WIDTH_3
|
|
#define DEF_DOUBLE_LINE6_DIST DEF_LINE_WIDTH_2
|
|
|
|
#define DEF_DOUBLE_LINE7_OUT DEF_LINE_WIDTH_0
|
|
#define DEF_DOUBLE_LINE7_IN DEF_LINE_WIDTH_0
|
|
#define DEF_DOUBLE_LINE7_DIST DEF_LINE_WIDTH_2
|
|
|
|
#define DEF_DOUBLE_LINE8_OUT DEF_LINE_WIDTH_1
|
|
#define DEF_DOUBLE_LINE8_IN DEF_LINE_WIDTH_0
|
|
#define DEF_DOUBLE_LINE8_DIST DEF_LINE_WIDTH_2
|
|
|
|
#define DEF_DOUBLE_LINE9_OUT DEF_LINE_WIDTH_2
|
|
#define DEF_DOUBLE_LINE9_IN DEF_LINE_WIDTH_0
|
|
#define DEF_DOUBLE_LINE9_DIST DEF_LINE_WIDTH_2
|
|
|
|
#define DEF_DOUBLE_LINE10_OUT DEF_LINE_WIDTH_3
|
|
#define DEF_DOUBLE_LINE10_IN DEF_LINE_WIDTH_0
|
|
#define DEF_DOUBLE_LINE10_DIST DEF_LINE_WIDTH_2
|
|
|
|
// finished copy
|
|
|
|
#define SVX_XML_BORDER_STYLE_NONE 0
|
|
#define SVX_XML_BORDER_STYLE_SOLID 1
|
|
#define SVX_XML_BORDER_STYLE_DOUBLE 2
|
|
#define SVX_XML_BORDER_STYLE_DASHED 3
|
|
#define SVX_XML_BORDER_STYLE_DOTTED 4
|
|
|
|
#define SVX_XML_BORDER_WIDTH_THIN 0
|
|
#define SVX_XML_BORDER_WIDTH_MIDDLE 1
|
|
#define SVX_XML_BORDER_WIDTH_THICK 2
|
|
|
|
SvXMLEnumMapEntry pXML_BorderStyles[] =
|
|
{
|
|
{ XML_NONE, SVX_XML_BORDER_STYLE_NONE },
|
|
{ XML_HIDDEN, SVX_XML_BORDER_STYLE_NONE },
|
|
{ XML_SOLID, SVX_XML_BORDER_STYLE_SOLID },
|
|
{ XML_DOUBLE, SVX_XML_BORDER_STYLE_DOUBLE },
|
|
{ XML_DOTTED, SVX_XML_BORDER_STYLE_DOTTED },
|
|
{ XML_DASHED, SVX_XML_BORDER_STYLE_DASHED },
|
|
{ XML_GROOVE, SVX_XML_BORDER_STYLE_SOLID },
|
|
{ XML_RIDGE, SVX_XML_BORDER_STYLE_SOLID },
|
|
{ XML_INSET, SVX_XML_BORDER_STYLE_SOLID },
|
|
{ XML_OUTSET, SVX_XML_BORDER_STYLE_SOLID },
|
|
{ XML_TOKEN_INVALID, 0 }
|
|
};
|
|
|
|
SvXMLEnumMapEntry pXML_NamedBorderWidths[] =
|
|
{
|
|
{ XML_THIN, SVX_XML_BORDER_WIDTH_THIN },
|
|
{ XML_MIDDLE, SVX_XML_BORDER_WIDTH_MIDDLE },
|
|
{ XML_THICK, SVX_XML_BORDER_WIDTH_THICK },
|
|
{ XML_TOKEN_INVALID, 0 }
|
|
};
|
|
// mapping tables to map external xml input to intarnal box line widths
|
|
|
|
// Ein Eintrag besteht aus vier USHORTs. Der erste ist die Gesamtbreite,
|
|
// die anderen sind die 3 Einzelbreiten
|
|
|
|
#define SBORDER_ENTRY( n ) \
|
|
DEF_LINE_WIDTH_##n, DEF_LINE_WIDTH_##n, 0, 0
|
|
|
|
#define DBORDER_ENTRY( n ) \
|
|
DEF_DOUBLE_LINE##n##_OUT + DEF_DOUBLE_LINE##n##_IN + \
|
|
DEF_DOUBLE_LINE##n##_DIST, \
|
|
DEF_DOUBLE_LINE##n##_OUT, \
|
|
DEF_DOUBLE_LINE##n##_IN, \
|
|
DEF_DOUBLE_LINE##n##_DIST
|
|
|
|
#define TDBORDER_ENTRY( n ) \
|
|
DEF_DOUBLE_LINE##n##_OUT, \
|
|
DEF_DOUBLE_LINE##n##_OUT, \
|
|
DEF_DOUBLE_LINE##n##_IN, \
|
|
DEF_DOUBLE_LINE##n##_DIST
|
|
|
|
|
|
static sal_uInt16 const aSBorderWidths[] =
|
|
{
|
|
SBORDER_ENTRY( 0 ), SBORDER_ENTRY( 1 ), SBORDER_ENTRY( 2 ),
|
|
SBORDER_ENTRY( 3 ), SBORDER_ENTRY( 4 )
|
|
};
|
|
|
|
static sal_uInt16 const aDBorderWidths[] =
|
|
{
|
|
DBORDER_ENTRY( 0 ),
|
|
DBORDER_ENTRY( 7 ),
|
|
DBORDER_ENTRY( 1 ),
|
|
DBORDER_ENTRY( 8 ),
|
|
DBORDER_ENTRY( 4 ),
|
|
DBORDER_ENTRY( 9 ),
|
|
DBORDER_ENTRY( 3 ),
|
|
DBORDER_ENTRY( 10 ),
|
|
DBORDER_ENTRY( 2 ),
|
|
DBORDER_ENTRY( 6 ),
|
|
DBORDER_ENTRY( 5 )
|
|
};
|
|
|
|
void lcl_frmitems_setXMLBorderStyle( table::BorderLine2 & rBorderLine, sal_uInt16 nStyle )
|
|
{
|
|
sal_Int16 eStyle = API_LINE_SOLID;
|
|
switch ( nStyle )
|
|
{
|
|
case SVX_XML_BORDER_STYLE_DOTTED:
|
|
eStyle = API_LINE_DOTTED;
|
|
break;
|
|
case SVX_XML_BORDER_STYLE_DASHED:
|
|
eStyle = API_LINE_DASHED;
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
rBorderLine.LineStyle = eStyle;
|
|
}
|
|
|
|
void lcl_frmitems_setXMLBorderWidth( table::BorderLine &rBorderLine,
|
|
sal_uInt16 nWidth, sal_Bool bDouble )
|
|
{
|
|
if( bDouble )
|
|
{
|
|
const sal_uInt16 *aWidths = aDBorderWidths;
|
|
sal_uInt16 nSize = sizeof( aDBorderWidths );
|
|
sal_uInt16 i = (nSize / sizeof(sal_uInt16)) - 4;
|
|
while( i>0 &&
|
|
nWidth <= ((aWidths[i] + aWidths[i-4]) / 2) )
|
|
{
|
|
i -= 4;
|
|
}
|
|
|
|
rBorderLine.OuterLineWidth = aWidths[i+1];
|
|
rBorderLine.InnerLineWidth = aWidths[i+2];
|
|
rBorderLine.LineDistance = aWidths[i+3];
|
|
}
|
|
else
|
|
{
|
|
rBorderLine.OuterLineWidth = 0 == nWidth ? DEF_LINE_WIDTH_0 : nWidth;
|
|
rBorderLine.InnerLineWidth = 0;
|
|
rBorderLine.LineDistance = 0;
|
|
|
|
}
|
|
}
|
|
|
|
|
|
///////////////////////////////////////////////////////////////////////////////
|
|
//
|
|
// class XMLEscapementPropHdl
|
|
//
|
|
|
|
XMLBorderWidthHdl::~XMLBorderWidthHdl()
|
|
{
|
|
// nothing to do
|
|
}
|
|
|
|
sal_Bool XMLBorderWidthHdl::importXML( const OUString& rStrImpValue, uno::Any& rValue, const SvXMLUnitConverter& rUnitConverter ) const
|
|
{
|
|
SvXMLTokenEnumerator aTokenEnum( rStrImpValue );
|
|
|
|
sal_Int32 nInWidth, nDistance, nOutWidth;
|
|
|
|
OUString aToken;
|
|
if( !aTokenEnum.getNextToken( aToken ) )
|
|
return sal_False;
|
|
|
|
if( !rUnitConverter.convertMeasure( nInWidth, aToken, 0, 500 ) )
|
|
return sal_False;
|
|
|
|
if( !aTokenEnum.getNextToken( aToken ) )
|
|
return sal_False;
|
|
|
|
if( !rUnitConverter.convertMeasure( nDistance, aToken, 0, 500 ) )
|
|
return sal_False;
|
|
|
|
if( !aTokenEnum.getNextToken( aToken ) )
|
|
return sal_False;
|
|
|
|
if( !rUnitConverter.convertMeasure( nOutWidth, aToken, 0, 500 ) )
|
|
return sal_False;
|
|
|
|
table::BorderLine2 aBorderLine;
|
|
if(!(rValue >>= aBorderLine))
|
|
aBorderLine.Color = 0;
|
|
|
|
aBorderLine.InnerLineWidth = sal::static_int_cast< sal_Int16 >(nInWidth);
|
|
aBorderLine.OuterLineWidth = sal::static_int_cast< sal_Int16 >(nOutWidth);
|
|
aBorderLine.LineDistance = sal::static_int_cast< sal_Int16 >(nDistance);
|
|
|
|
rValue <<= aBorderLine;
|
|
return sal_True;
|
|
}
|
|
|
|
sal_Bool XMLBorderWidthHdl::exportXML( OUString& rStrExpValue, const uno::Any& rValue, const SvXMLUnitConverter& rUnitConverter ) const
|
|
{
|
|
OUStringBuffer aOut;
|
|
|
|
table::BorderLine2 aBorderLine;
|
|
if(!(rValue >>= aBorderLine))
|
|
return sal_False;
|
|
|
|
if( aBorderLine.LineDistance == 0 && aBorderLine.InnerLineWidth == 0)
|
|
return sal_False;
|
|
|
|
rUnitConverter.convertMeasure( aOut, aBorderLine.InnerLineWidth );
|
|
aOut.append( sal_Unicode( ' ' ) );
|
|
rUnitConverter.convertMeasure( aOut, aBorderLine.LineDistance );
|
|
aOut.append( sal_Unicode( ' ' ) );
|
|
rUnitConverter.convertMeasure( aOut, aBorderLine.OuterLineWidth );
|
|
|
|
rStrExpValue = aOut.makeStringAndClear();
|
|
return sal_True;
|
|
}
|
|
|
|
///////////////////////////////////////////////////////////////////////////////
|
|
//
|
|
// class XMLEscapementHeightPropHdl
|
|
//
|
|
|
|
XMLBorderHdl::~XMLBorderHdl()
|
|
{
|
|
// nothing to do
|
|
}
|
|
|
|
sal_Bool XMLBorderHdl::importXML( const OUString& rStrImpValue, uno::Any& rValue, const SvXMLUnitConverter& rUnitConverter ) const
|
|
{
|
|
OUString aToken;
|
|
SvXMLTokenEnumerator aTokens( rStrImpValue );
|
|
|
|
sal_Bool bHasStyle = sal_False;
|
|
sal_Bool bHasWidth = sal_False;
|
|
sal_Bool bHasColor = sal_False;
|
|
|
|
sal_uInt16 nStyle = USHRT_MAX;
|
|
sal_uInt16 nWidth = 0;
|
|
sal_uInt16 nNamedWidth = USHRT_MAX;
|
|
Color aColor;
|
|
|
|
sal_Int32 nTemp;
|
|
while( aTokens.getNextToken( aToken ) && aToken.getLength() != 0 )
|
|
{
|
|
if( !bHasWidth &&
|
|
rUnitConverter.convertEnum( nNamedWidth, aToken,
|
|
pXML_NamedBorderWidths ) )
|
|
{
|
|
bHasWidth = sal_True;
|
|
}
|
|
else if( !bHasStyle &&
|
|
rUnitConverter.convertEnum( nStyle, aToken,
|
|
pXML_BorderStyles ) )
|
|
{
|
|
bHasStyle = sal_True;
|
|
}
|
|
else if( !bHasColor && rUnitConverter.convertColor( aColor, aToken ) )
|
|
{
|
|
bHasColor = sal_True;
|
|
}
|
|
else if( !bHasWidth &&
|
|
rUnitConverter.convertMeasure( nTemp, aToken, 0,
|
|
USHRT_MAX ) )
|
|
{
|
|
nWidth = (sal_uInt16)nTemp;
|
|
bHasWidth = sal_True;
|
|
}
|
|
else
|
|
{
|
|
// missformed
|
|
return sal_False;
|
|
}
|
|
}
|
|
|
|
// if there is no style or a different style than none but no width,
|
|
// then the declaration is not valid.
|
|
if( !bHasStyle || (SVX_XML_BORDER_STYLE_NONE != nStyle && !bHasWidth) )
|
|
return sal_False;
|
|
|
|
table::BorderLine2 aBorderLine;
|
|
if(!(rValue >>= aBorderLine))
|
|
{
|
|
aBorderLine.Color = 0;
|
|
aBorderLine.InnerLineWidth = 0;
|
|
aBorderLine.OuterLineWidth = 0;
|
|
aBorderLine.LineDistance = 0;
|
|
}
|
|
|
|
// first of all, delete an empty line
|
|
sal_Bool bDouble = SVX_XML_BORDER_STYLE_DOUBLE == nStyle;
|
|
if( (bHasStyle && SVX_XML_BORDER_STYLE_NONE == nStyle) ||
|
|
(bHasWidth && USHRT_MAX == nNamedWidth && 0 == nWidth) )
|
|
{
|
|
aBorderLine.InnerLineWidth = 0;
|
|
aBorderLine.OuterLineWidth = 0;
|
|
aBorderLine.LineDistance = 0;
|
|
}
|
|
else if( bHasWidth )
|
|
{
|
|
if( USHRT_MAX != nNamedWidth )
|
|
{
|
|
const sal_uInt16 *aWidths = bDouble ? aDBorderWidths
|
|
: aSBorderWidths;
|
|
sal_uInt16 nNWidth = nNamedWidth * 4;
|
|
aBorderLine.OuterLineWidth = aWidths[nNWidth+1];
|
|
aBorderLine.InnerLineWidth = aWidths[nNWidth+2];
|
|
aBorderLine.LineDistance = aWidths[nNWidth+3];
|
|
}
|
|
else
|
|
{
|
|
lcl_frmitems_setXMLBorderWidth( aBorderLine, nWidth, bDouble );
|
|
lcl_frmitems_setXMLBorderStyle( aBorderLine, nStyle );
|
|
}
|
|
}
|
|
else
|
|
{
|
|
lcl_frmitems_setXMLBorderWidth( aBorderLine, 0, bDouble );
|
|
lcl_frmitems_setXMLBorderStyle( aBorderLine, nStyle );
|
|
}
|
|
|
|
// set color
|
|
if( bHasColor )
|
|
aBorderLine.Color = (sal_Int32)aColor.GetRGBColor();
|
|
|
|
rValue <<= aBorderLine;
|
|
return sal_True;
|
|
}
|
|
|
|
sal_Bool XMLBorderHdl::exportXML( OUString& rStrExpValue, const uno::Any& rValue, const SvXMLUnitConverter& rUnitConverter ) const
|
|
{
|
|
OUStringBuffer aOut;
|
|
|
|
table::BorderLine2 aBorderLine;
|
|
if(!(rValue >>= aBorderLine))
|
|
return sal_False;
|
|
|
|
sal_Int32 nWidth = aBorderLine.OuterLineWidth;
|
|
const sal_uInt16 nDistance = aBorderLine.LineDistance;
|
|
if( 0 != nDistance )
|
|
{
|
|
nWidth += nDistance;
|
|
nWidth += aBorderLine.InnerLineWidth;
|
|
}
|
|
|
|
if( nWidth == 0 )
|
|
{
|
|
aOut.append( GetXMLToken( XML_NONE ) );
|
|
}
|
|
else
|
|
{
|
|
rUnitConverter.convertMeasure( aOut, nWidth );
|
|
|
|
aOut.append( sal_Unicode( ' ' ) );
|
|
|
|
XMLTokenEnum eStyleToken = XML_SOLID;
|
|
switch ( aBorderLine.LineStyle )
|
|
{
|
|
case API_LINE_DASHED:
|
|
eStyleToken = XML_DASHED;
|
|
break;
|
|
case API_LINE_DOTTED:
|
|
eStyleToken = XML_DOTTED;
|
|
break;
|
|
case API_LINE_SOLID:
|
|
default:
|
|
eStyleToken = XML_SOLID;
|
|
}
|
|
aOut.append( GetXMLToken((0 == nDistance) ? eStyleToken : XML_DOUBLE) );
|
|
|
|
aOut.append( sal_Unicode( ' ' ) );
|
|
|
|
rUnitConverter.convertColor( aOut, aBorderLine.Color );
|
|
}
|
|
|
|
rStrExpValue = aOut.makeStringAndClear();
|
|
|
|
return sal_True;
|
|
}
|
|
|
|
/* vim:set shiftwidth=4 softtabstop=4 expandtab: */
|