xref: /AOO41X/main/oox/source/drawingml/table/tablecell.cxx (revision f157491532701b70eaebd4566f0a2dc3a9ae8228)
1 /**************************************************************
2  *
3  * Licensed to the Apache Software Foundation (ASF) under one
4  * or more contributor license agreements.  See the NOTICE file
5  * distributed with this work for additional information
6  * regarding copyright ownership.  The ASF licenses this file
7  * to you under the Apache License, Version 2.0 (the
8  * "License"); you may not use this file except in compliance
9  * with the License.  You may obtain a copy of the License at
10  *
11  *   http://www.apache.org/licenses/LICENSE-2.0
12  *
13  * Unless required by applicable law or agreed to in writing,
14  * software distributed under the License is distributed on an
15  * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
16  * KIND, either express or implied.  See the License for the
17  * specific language governing permissions and limitations
18  * under the License.
19  *
20  *************************************************************/
21 
22 
23 
24 #include "oox/drawingml/table/tablecell.hxx"
25 #include "oox/drawingml/table/tableproperties.hxx"
26 #include "oox/drawingml/shapepropertymap.hxx"
27 #include "oox/drawingml/textbody.hxx"
28 #include "oox/core/xmlfilterbase.hxx"
29 #include "oox/helper/propertyset.hxx"
30 #include <com/sun/star/container/XNameContainer.hpp>
31 #include <com/sun/star/beans/XMultiPropertySet.hpp>
32 #include <com/sun/star/table/XTable.hpp>
33 #include <com/sun/star/table/XMergeableCellRange.hpp>
34 #include <com/sun/star/table/BorderLine.hpp>
35 #include <com/sun/star/drawing/LineStyle.hpp>
36 #include <com/sun/star/drawing/TextVerticalAdjust.hpp>
37 #include <com/sun/star/drawing/TextHorizontalAdjust.hpp>
38 #include <com/sun/star/text/XText.hpp>
39 #include <com/sun/star/text/WritingMode.hpp>
40 
41 using rtl::OUString;
42 using namespace ::oox::core;
43 using namespace ::com::sun::star;
44 using namespace ::com::sun::star::uno;
45 using namespace ::com::sun::star::beans;
46 using ::com::sun::star::table::BorderLine;
47 using ::com::sun::star::drawing::LineStyle;
48 
49 namespace oox { namespace drawingml { namespace table {
50 
TableCell()51 TableCell::TableCell()
52 : mnRowSpan ( 1 )
53 , mnGridSpan( 1 )
54 , mbhMerge( sal_False )
55 , mbvMerge( sal_False )
56 , mnMarL( 91440 )
57 , mnMarR( 91440 )
58 , mnMarT( 45720 )
59 , mnMarB( 45720 )
60 , mnVertToken( XML_horz )
61 , mnAnchorToken( XML_t )
62 , mbAnchorCtr( sal_False )
63 , mnHorzOverflowToken( XML_clip )
64 {
65 }
~TableCell()66 TableCell::~TableCell()
67 {
68 }
69 
applyLineAttributes(const::oox::core::XmlFilterBase & rFilterBase,Reference<XPropertySet> & rxPropSet,oox::drawingml::LineProperties & rLineProperties,sal_Int32 nPropId)70 void applyLineAttributes( const ::oox::core::XmlFilterBase& rFilterBase,
71         Reference< XPropertySet >& rxPropSet, oox::drawingml::LineProperties& rLineProperties,
72         sal_Int32 nPropId )
73 {
74     BorderLine aBorderLine( 0, 0, 0, 0 );
75     if( rLineProperties.maLineFill.moFillType.differsFrom( XML_noFill ) )
76     {
77         Color aColor = rLineProperties.maLineFill.getBestSolidColor();
78         aBorderLine.Color = aColor.getColor( rFilterBase.getGraphicHelper() );
79         aBorderLine.OuterLineWidth = static_cast< sal_Int16 >( GetCoordinate( rLineProperties.moLineWidth.get( 0 ) ) / 4 );
80         aBorderLine.InnerLineWidth = static_cast< sal_Int16 >( GetCoordinate( rLineProperties.moLineWidth.get( 0 ) ) / 4 );
81         aBorderLine.LineDistance = 0;
82     }
83 
84     PropertySet aPropSet( rxPropSet );
85     aPropSet.setProperty( nPropId, aBorderLine );
86 }
87 
applyBorder(TableStylePart & rTableStylePart,sal_Int32 nLineType,oox::drawingml::LineProperties & rLineProperties)88 void applyBorder( TableStylePart& rTableStylePart, sal_Int32 nLineType, oox::drawingml::LineProperties& rLineProperties )
89 {
90     std::map < sal_Int32, ::oox::drawingml::LinePropertiesPtr >& rPartLineBorders( rTableStylePart.getLineBorders() );
91     std::map < sal_Int32, ::oox::drawingml::LinePropertiesPtr >::const_iterator aIter( rPartLineBorders.find( nLineType ) );
92     if ( ( aIter != rPartLineBorders.end() ) && aIter->second.get() )
93         rLineProperties.assignUsed( *aIter->second );
94 }
95 
applyTableStylePart(const::oox::core::XmlFilterBase & rFilterBase,const Reference<::com::sun::star::table::XCell> & rxCell,oox::drawingml::FillProperties & rFillProperties,oox::drawingml::LineProperties & rLeftBorder,oox::drawingml::LineProperties & rRightBorder,oox::drawingml::LineProperties & rTopBorder,oox::drawingml::LineProperties & rBottomBorder,oox::drawingml::LineProperties & rTopLeftToBottomRightBorder,oox::drawingml::LineProperties & rBottomLeftToTopRightBorder,TableStylePart & rTableStylePart)96 void applyTableStylePart( const ::oox::core::XmlFilterBase& rFilterBase, const Reference < ::com::sun::star::table::XCell >& rxCell, oox::drawingml::FillProperties& rFillProperties,
97      oox::drawingml::LineProperties& rLeftBorder,
98      oox::drawingml::LineProperties& rRightBorder,
99      oox::drawingml::LineProperties& rTopBorder,
100      oox::drawingml::LineProperties& rBottomBorder,
101      oox::drawingml::LineProperties& rTopLeftToBottomRightBorder,
102      oox::drawingml::LineProperties& rBottomLeftToTopRightBorder,
103     TableStylePart& rTableStylePart )
104 {
105     boost::shared_ptr< ::oox::drawingml::FillProperties >& rPartFillPropertiesPtr( rTableStylePart.getFillProperties() );
106     if ( rPartFillPropertiesPtr.get() )
107         rFillProperties.assignUsed( *rPartFillPropertiesPtr );
108 
109     applyBorder( rTableStylePart, XML_left, rLeftBorder );
110     applyBorder( rTableStylePart, XML_right, rRightBorder );
111     applyBorder( rTableStylePart, XML_top, rTopBorder );
112     applyBorder( rTableStylePart, XML_bottom, rBottomBorder );
113     applyBorder( rTableStylePart, XML_tl2br, rTopLeftToBottomRightBorder );
114     applyBorder( rTableStylePart, XML_tr2bl, rBottomLeftToTopRightBorder );
115 
116     TextCharacterProperties aTextCharProps;
117     aTextCharProps.maLatinFont = rTableStylePart.getLatinFont();
118     aTextCharProps.maAsianFont = rTableStylePart.getAsianFont();
119     aTextCharProps.maComplexFont = rTableStylePart.getComplexFont();
120     aTextCharProps.maSymbolFont = rTableStylePart.getSymbolFont();
121     aTextCharProps.maCharColor = rTableStylePart.getTextColor();
122 
123     PropertySet aPropSet( rxCell );
124     aTextCharProps.pushToPropSet( aPropSet, rFilterBase );
125 }
126 
applyTableCellProperties(const Reference<::com::sun::star::table::XCell> & rxCell,const TableCell & rTableCell)127 void applyTableCellProperties( const Reference < ::com::sun::star::table::XCell >& rxCell, const TableCell& rTableCell )
128 {
129     static const rtl::OUString  sTopBorder( RTL_CONSTASCII_USTRINGPARAM( "TextUpperDistance" ) );
130     static const rtl::OUString  sBottomBorder( RTL_CONSTASCII_USTRINGPARAM( "TextLowerDistance" ) );
131     static const rtl::OUString  sLeftBorder( RTL_CONSTASCII_USTRINGPARAM( "TextLeftDistance" ) );
132     static const rtl::OUString  sRightBorder( RTL_CONSTASCII_USTRINGPARAM( "TextRightDistance" ) );
133     static const rtl::OUString  sVerticalAdjust( RTL_CONSTASCII_USTRINGPARAM( "TextVerticalAdjust" ) );
134 
135     Reference< XPropertySet > xPropSet( rxCell, UNO_QUERY_THROW );
136     xPropSet->setPropertyValue( sTopBorder, Any( static_cast< sal_Int32 >( rTableCell.getTopMargin() / 360 ) ) );
137     xPropSet->setPropertyValue( sRightBorder, Any( static_cast< sal_Int32 >( rTableCell.getRightMargin() / 360 ) ) );
138     xPropSet->setPropertyValue( sLeftBorder, Any( static_cast< sal_Int32 >( rTableCell.getLeftMargin() / 360 ) ) );
139     xPropSet->setPropertyValue( sBottomBorder, Any( static_cast< sal_Int32 >( rTableCell.getBottomMargin() / 360 ) ) );
140 
141     drawing::TextVerticalAdjust eVA;
142     switch( rTableCell.getAnchorToken() )
143     {
144         case XML_ctr:   eVA = drawing::TextVerticalAdjust_CENTER; break;
145         case XML_b:     eVA = drawing::TextVerticalAdjust_BOTTOM; break;
146         case XML_just:
147         case XML_dist:
148         default:
149         case XML_t:     eVA = drawing::TextVerticalAdjust_TOP; break;
150     }
151     xPropSet->setPropertyValue( sVerticalAdjust, Any( eVA ) );
152 }
153 
154 // save char color from tblstyle for combination later
lcl_getCharPropFromTblStylePart(TextCharacterProperties & rDstCharProp,const TableStylePart & rSrcTblStyle)155 void lcl_getCharPropFromTblStylePart(TextCharacterProperties& rDstCharProp, const TableStylePart& rSrcTblStyle)
156 {
157     const Color& clr = const_cast<TableStylePart&>(rSrcTblStyle).getTextColor();
158     if (clr.isUsed())
159         rDstCharProp.maCharColor = clr;
160     // TODO: there may be other similar properties from tblstyle which need combination later
161 }
162 
pushToXCell(const::oox::core::XmlFilterBase & rFilterBase,::oox::drawingml::TextListStylePtr pMasterTextListStyle,const::com::sun::star::uno::Reference<::com::sun::star::table::XCell> & rxCell,const TableProperties & rTableProperties,const TableStyle & rTableStyle,sal_Int32 nColumn,sal_Int32 nMaxColumn,sal_Int32 nRow,sal_Int32 nMaxRow)163 void TableCell::pushToXCell( const ::oox::core::XmlFilterBase& rFilterBase, ::oox::drawingml::TextListStylePtr pMasterTextListStyle,
164     const ::com::sun::star::uno::Reference < ::com::sun::star::table::XCell >& rxCell, const TableProperties& rTableProperties,
165         const TableStyle& rTableStyle, sal_Int32 nColumn, sal_Int32 nMaxColumn, sal_Int32 nRow, sal_Int32 nMaxRow )
166 {
167     TableStyle& rTable( const_cast< TableStyle& >( rTableStyle ) );
168     TableProperties& rProperties( const_cast< TableProperties& >( rTableProperties ) );
169 
170     Reference< text::XText > xText( rxCell, UNO_QUERY_THROW );
171     Reference< text::XTextCursor > xAt = xText->createTextCursor();
172 
173     applyTableCellProperties( rxCell, *this );
174 
175     Reference< XPropertySet > xPropSet( rxCell, UNO_QUERY_THROW );
176     oox::drawingml::FillProperties aFillProperties;
177     oox::drawingml::LineProperties aLinePropertiesLeft;
178     oox::drawingml::LineProperties aLinePropertiesRight;
179     oox::drawingml::LineProperties aLinePropertiesTop;
180     oox::drawingml::LineProperties aLinePropertiesBottom;
181     oox::drawingml::LineProperties aLinePropertiesTopLeftToBottomRight;
182     oox::drawingml::LineProperties aLinePropertiesBottomLeftToTopRight;
183 
184     boost::shared_ptr< ::oox::drawingml::FillProperties >& rBackgroundFillPropertiesPtr( rTable.getBackgroundFillProperties() );
185     if ( rBackgroundFillPropertiesPtr.get() )
186         aFillProperties.assignUsed( *rBackgroundFillPropertiesPtr );
187 
188     applyTableStylePart( rFilterBase, rxCell, aFillProperties,
189         aLinePropertiesLeft,
190         aLinePropertiesRight,
191         aLinePropertiesTop,
192         aLinePropertiesBottom,
193         aLinePropertiesTopLeftToBottomRight,
194         aLinePropertiesBottomLeftToTopRight,
195         rTable.getWholeTbl() );
196 
197     // get char color from tblstyle for combination later
198     TextCharacterProperties aTextCharProps;
199     lcl_getCharPropFromTblStylePart(aTextCharProps, rTable.getWholeTbl());
200 
201     if ( rProperties.isFirstRow() && ( nRow == 0 ) )
202     {
203         applyTableStylePart( rFilterBase, rxCell, aFillProperties,
204             aLinePropertiesLeft,
205             aLinePropertiesRight,
206             aLinePropertiesTop,
207             aLinePropertiesBottom,
208             aLinePropertiesTopLeftToBottomRight,
209             aLinePropertiesBottomLeftToTopRight,
210             rTable.getFirstRow() );
211        lcl_getCharPropFromTblStylePart(aTextCharProps, rTable.getFirstRow());
212     }
213     if ( rProperties.isLastRow() && ( nRow == nMaxRow ) )
214     {
215         applyTableStylePart( rFilterBase, rxCell, aFillProperties,
216             aLinePropertiesLeft,
217             aLinePropertiesRight,
218             aLinePropertiesTop,
219             aLinePropertiesBottom,
220             aLinePropertiesTopLeftToBottomRight,
221             aLinePropertiesBottomLeftToTopRight,
222             rTable.getLastRow() );
223         lcl_getCharPropFromTblStylePart(aTextCharProps, rTable.getLastRow());
224     }
225     if ( rProperties.isFirstCol() && ( nColumn == 0 ) )
226     {
227         applyTableStylePart( rFilterBase, rxCell, aFillProperties,
228             aLinePropertiesLeft,
229             aLinePropertiesRight,
230             aLinePropertiesTop,
231             aLinePropertiesBottom,
232             aLinePropertiesTopLeftToBottomRight,
233             aLinePropertiesBottomLeftToTopRight,
234             rTable.getFirstCol() );
235         lcl_getCharPropFromTblStylePart(aTextCharProps, rTable.getFirstCol());
236     }
237     if ( rProperties.isLastCol() && ( nColumn == nMaxColumn ) )
238     {
239         applyTableStylePart( rFilterBase, rxCell, aFillProperties,
240             aLinePropertiesLeft,
241             aLinePropertiesRight,
242             aLinePropertiesTop,
243             aLinePropertiesBottom,
244             aLinePropertiesTopLeftToBottomRight,
245             aLinePropertiesBottomLeftToTopRight,
246             rTable.getLastCol() );
247         lcl_getCharPropFromTblStylePart(aTextCharProps, rTable.getLastCol());
248     }
249     if ( rProperties.isBandRow() )
250     {
251         if ( ( !rProperties.isFirstRow() || ( nRow != 0 ) ) &&
252             ( !rProperties.isLastRow() || ( nRow != nMaxRow ) ) )
253         {
254             sal_Int32 nBand = nRow;
255             if ( rProperties.isFirstRow() )
256                 nBand++;
257             if ( nBand & 1 )
258             {
259                 applyTableStylePart( rFilterBase, rxCell, aFillProperties,
260                     aLinePropertiesLeft,
261                     aLinePropertiesRight,
262                     aLinePropertiesTop,
263                     aLinePropertiesBottom,
264                     aLinePropertiesTopLeftToBottomRight,
265                     aLinePropertiesBottomLeftToTopRight,
266                     rTable.getBand2H() );
267                 lcl_getCharPropFromTblStylePart(aTextCharProps, rTable.getBand2H());
268             }
269             else
270             {
271                 applyTableStylePart( rFilterBase, rxCell, aFillProperties,
272                     aLinePropertiesLeft,
273                     aLinePropertiesRight,
274                     aLinePropertiesTop,
275                     aLinePropertiesBottom,
276                     aLinePropertiesTopLeftToBottomRight,
277                     aLinePropertiesBottomLeftToTopRight,
278                     rTable.getBand1H() );
279                 lcl_getCharPropFromTblStylePart(aTextCharProps, rTable.getBand1H());
280             }
281         }
282     }
283     if ( ( nRow == 0 ) && ( nColumn == 0 ) )
284     {
285         applyTableStylePart( rFilterBase, rxCell, aFillProperties,
286             aLinePropertiesLeft,
287             aLinePropertiesRight,
288             aLinePropertiesTop,
289             aLinePropertiesBottom,
290             aLinePropertiesTopLeftToBottomRight,
291             aLinePropertiesBottomLeftToTopRight,
292             rTable.getNwCell() );
293         lcl_getCharPropFromTblStylePart(aTextCharProps, rTable.getNwCell());
294     }
295     if ( ( nRow == nMaxRow ) && ( nColumn == 0 ) )
296     {
297         applyTableStylePart( rFilterBase, rxCell, aFillProperties,
298             aLinePropertiesLeft,
299             aLinePropertiesRight,
300             aLinePropertiesTop,
301             aLinePropertiesBottom,
302             aLinePropertiesTopLeftToBottomRight,
303             aLinePropertiesBottomLeftToTopRight,
304             rTable.getSwCell() );
305         lcl_getCharPropFromTblStylePart(aTextCharProps, rTable.getSwCell());
306     }
307     if ( ( nRow == 0 ) && ( nColumn == nMaxColumn ) )
308     {
309         applyTableStylePart( rFilterBase, rxCell, aFillProperties,
310             aLinePropertiesLeft,
311             aLinePropertiesRight,
312             aLinePropertiesTop,
313             aLinePropertiesBottom,
314             aLinePropertiesTopLeftToBottomRight,
315             aLinePropertiesBottomLeftToTopRight,
316             rTable.getNeCell() );
317         lcl_getCharPropFromTblStylePart(aTextCharProps, rTable.getNeCell());
318     }
319     if ( ( nRow == nMaxColumn ) && ( nColumn == nMaxColumn ) )
320     {
321         applyTableStylePart( rFilterBase, rxCell, aFillProperties,
322             aLinePropertiesLeft,
323             aLinePropertiesRight,
324             aLinePropertiesTop,
325             aLinePropertiesBottom,
326             aLinePropertiesTopLeftToBottomRight,
327             aLinePropertiesBottomLeftToTopRight,
328             rTable.getSeCell() );
329         lcl_getCharPropFromTblStylePart(aTextCharProps, rTable.getSeCell());
330     }
331     if ( rProperties.isBandCol() )
332     {
333         if ( ( !rProperties.isFirstCol() || ( nColumn != 0 ) ) &&
334             ( !rProperties.isLastCol() || ( nColumn != nMaxColumn ) ) )
335         {
336             sal_Int32 nBand = nColumn;
337             if ( rProperties.isFirstCol() )
338                 nBand++;
339             if ( nBand & 1 )
340             {
341                 applyTableStylePart( rFilterBase, rxCell, aFillProperties,
342                     aLinePropertiesLeft,
343                     aLinePropertiesRight,
344                     aLinePropertiesTop,
345                     aLinePropertiesBottom,
346                     aLinePropertiesTopLeftToBottomRight,
347                     aLinePropertiesBottomLeftToTopRight,
348                     rTable.getBand2V() );
349                 lcl_getCharPropFromTblStylePart(aTextCharProps, rTable.getBand2V());
350             }
351             else
352             {
353                 applyTableStylePart( rFilterBase, rxCell, aFillProperties,
354                     aLinePropertiesLeft,
355                     aLinePropertiesRight,
356                     aLinePropertiesTop,
357                     aLinePropertiesBottom,
358                     aLinePropertiesTopLeftToBottomRight,
359                     aLinePropertiesBottomLeftToTopRight,
360                     rTable.getBand1V() );
361                 lcl_getCharPropFromTblStylePart(aTextCharProps, rTable.getBand1V());
362             }
363         }
364     }
365 
366     getTextBody()->insertAt( rFilterBase, xText, xAt, aTextCharProps, pMasterTextListStyle );
367 
368     aLinePropertiesLeft.assignUsed( maLinePropertiesLeft );
369     aLinePropertiesRight.assignUsed( maLinePropertiesRight );
370     aLinePropertiesTop.assignUsed( maLinePropertiesTop );
371     aLinePropertiesBottom.assignUsed( maLinePropertiesBottom );
372     aLinePropertiesTopLeftToBottomRight.assignUsed( maLinePropertiesTopLeftToBottomRight );
373     aLinePropertiesBottomLeftToTopRight.assignUsed( maLinePropertiesBottomLeftToTopRight );
374     applyLineAttributes( rFilterBase, xPropSet, aLinePropertiesLeft, PROP_LeftBorder );
375     applyLineAttributes( rFilterBase, xPropSet, aLinePropertiesRight, PROP_RightBorder );
376     applyLineAttributes( rFilterBase, xPropSet, aLinePropertiesTop, PROP_TopBorder );
377     applyLineAttributes( rFilterBase, xPropSet, aLinePropertiesBottom, PROP_BottomBorder );
378     applyLineAttributes( rFilterBase, xPropSet, aLinePropertiesTopLeftToBottomRight, PROP_DiagonalTLBR );
379     applyLineAttributes( rFilterBase, xPropSet, aLinePropertiesBottomLeftToTopRight, PROP_DiagonalBLTR );
380 
381     aFillProperties.assignUsed( maFillProperties );
382     ShapePropertyMap aPropMap( rFilterBase.getModelObjectHelper() );
383     // TODO: phClr?
384     aFillProperties.pushToPropMap( aPropMap, rFilterBase.getGraphicHelper() );
385     PropertySet( xPropSet ).setProperties( aPropMap );
386 
387     if ( getVertToken() == XML_eaVert )
388     {
389         xPropSet->setPropertyValue(::rtl::OUString::createFromAscii( "TextWritingMode" ) , Any(com::sun::star::text::WritingMode_TB_RL) );
390     }
391 }
392 
393 } } }
394