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 // MARKER(update_precomp.py): autogen include statement, do not remove
25 #include "precompiled_sdext.hxx"
26 
27 #include "pdfiprocessor.hxx"
28 #include "xmlemitter.hxx"
29 #include "pdfihelper.hxx"
30 #include "imagecontainer.hxx"
31 #include "style.hxx"
32 #include "writertreevisiting.hxx"
33 #include "genericelements.hxx"
34 
35 #include <basegfx/polygon/b2dpolypolygontools.hxx>
36 #include <basegfx/range/b2drange.hxx>
37 
38 
39 namespace pdfi
40 {
41 
42 void WriterXmlEmitter::visit( HyperlinkElement& elem, const std::list< Element* >::const_iterator&   )
43 {
44     if( elem.Children.empty() )
45         return;
46 
47     const char* pType = dynamic_cast<DrawElement*>(elem.Children.front()) ? "draw:a" : "text:a";
48 
49     PropertyMap aProps;
50     aProps[ USTR( "xlink:type" ) ] = USTR( "simple" );
51     aProps[ USTR( "xlink:href" ) ] = elem.URI;
52     aProps[ USTR( "office:target-frame-name" ) ] = USTR( "_blank" );
53     aProps[ USTR( "xlink:show" ) ] = USTR( "new" );
54 
55     m_rEmitContext.rEmitter.beginTag( pType, aProps );
56     std::list< Element* >::iterator this_it =  elem.Children.begin();
57     while( this_it !=elem.Children.end() && *this_it != &elem )
58     {
59         (*this_it)->visitedBy( *this, this_it );
60         this_it++;
61     }
62     m_rEmitContext.rEmitter.endTag( pType );
63 }
64 
65 void WriterXmlEmitter::visit( TextElement& elem, const std::list< Element* >::const_iterator&   )
66 {
67     if( ! elem.Text.getLength() )
68         return;
69 
70     PropertyMap aProps;
71     if( elem.StyleId != -1 )
72     {
73         aProps[ rtl::OUString( RTL_CONSTASCII_USTRINGPARAM( "text:style-name" ) ) ] =
74             m_rEmitContext.rStyles.getStyleName( elem.StyleId );
75     }
76 
77     m_rEmitContext.rEmitter.beginTag( "text:span", aProps );
78     m_rEmitContext.rEmitter.write( elem.Text.makeStringAndClear() );
79     std::list< Element* >::iterator this_it =  elem.Children.begin();
80     while( this_it !=elem.Children.end() && *this_it != &elem )
81     {
82         (*this_it)->visitedBy( *this, this_it );
83         this_it++;
84     }
85 
86     m_rEmitContext.rEmitter.endTag( "text:span" );
87 }
88 
89 void WriterXmlEmitter::visit( ParagraphElement& elem, const std::list< Element* >::const_iterator&   )
90 {
91     PropertyMap aProps;
92     if( elem.StyleId != -1 )
93     {
94         aProps[ USTR( "text:style-name" ) ] = m_rEmitContext.rStyles.getStyleName( elem.StyleId );
95     }
96     const char* pTagType = "text:p";
97     if( elem.Type == elem.Headline )
98         pTagType = "text:h";
99     m_rEmitContext.rEmitter.beginTag( pTagType, aProps );
100 
101     std::list< Element* >::iterator this_it =  elem.Children.begin();
102     while( this_it !=elem.Children.end() && *this_it != &elem )
103     {
104         (*this_it)->visitedBy( *this, this_it );
105         this_it++;
106     }
107 
108     m_rEmitContext.rEmitter.endTag( pTagType );
109 }
110 
111 void WriterXmlEmitter::fillFrameProps( DrawElement&       rElem,
112                                        PropertyMap&       rProps,
113                                        const EmitContext& rEmitContext )
114 {
115     double rel_x = rElem.x, rel_y = rElem.y;
116 
117     // find anchor type by recursing though parents
118     Element* pAnchor = rElem.Parent;
119     while( pAnchor &&
120            ! dynamic_cast<ParagraphElement*>(pAnchor) &&
121            ! dynamic_cast<PageElement*>(pAnchor) )
122     {
123         pAnchor = pAnchor->Parent;
124     }
125     if( pAnchor )
126     {
127         if( dynamic_cast<ParagraphElement*>(pAnchor) )
128         {
129             rProps[ USTR( "text:anchor-type" ) ] =
130                 rElem.isCharacter ? USTR( "character" ) : USTR( "paragraph" );
131         }
132         else
133         {
134             PageElement* pPage = dynamic_cast<PageElement*>(pAnchor);
135             rProps[ USTR( "text:anchor-type" ) ] = USTR( "page" );
136             rProps[ USTR( "text:anchor-page-number" ) ] = rtl::OUString::valueOf(pPage->PageNumber);
137         }
138         rel_x -= pAnchor->x;
139         rel_y -= pAnchor->y;
140     }
141 
142     rProps[ USTR( "draw:z-index" ) ] = rtl::OUString::valueOf( rElem.ZOrder );
143     rProps[ USTR( "draw:style-name" )] = rEmitContext.rStyles.getStyleName( rElem.StyleId );
144     rProps[ USTR( "svg:width" ) ]   = convertPixelToUnitString( rElem.w );
145     rProps[ USTR( "svg:height" ) ]  = convertPixelToUnitString( rElem.h );
146 
147     const GraphicsContext& rGC =
148         rEmitContext.rProcessor.getGraphicsContext( rElem.GCId );
149     if( rGC.Transformation.isIdentity() )
150     {
151         if( !rElem.isCharacter )
152         {
153             rProps[ USTR( "svg:x" ) ]       = convertPixelToUnitString( rel_x );
154             rProps[ USTR( "svg:y" ) ]       = convertPixelToUnitString( rel_y );
155         }
156     }
157     else
158     {
159         basegfx::B2DTuple aScale, aTranslation;
160         double fRotate, fShearX;
161 
162         rGC.Transformation.decompose( aScale, aTranslation, fRotate, fShearX );
163 
164         rtl::OUStringBuffer aBuf( 256 );
165 
166         // TODO(F2): general transformation case missing; if implemented, note
167         // that ODF rotation is oriented the other way
168 
169         // build transformation string
170         if( fShearX != 0.0 )
171         {
172             aBuf.appendAscii( "skewX( " );
173             aBuf.append( fShearX );
174             aBuf.appendAscii( " )" );
175         }
176         if( fRotate != 0.0 )
177         {
178             if( aBuf.getLength() > 0 )
179                 aBuf.append( sal_Unicode(' ') );
180             aBuf.appendAscii( "rotate( " );
181             aBuf.append( -fRotate );
182             aBuf.appendAscii( " )" );
183 
184         }
185         if( ! rElem.isCharacter )
186         {
187             if( aBuf.getLength() > 0 )
188                 aBuf.append( sal_Unicode(' ') );
189             aBuf.appendAscii( "translate( " );
190             aBuf.append( convertPixelToUnitString( rel_x ) );
191             aBuf.append( sal_Unicode(' ') );
192             aBuf.append( convertPixelToUnitString( rel_y ) );
193             aBuf.appendAscii( " )" );
194          }
195 
196         rProps[ USTR( "draw:transform" ) ] = aBuf.makeStringAndClear();
197     }
198 }
199 
200 void WriterXmlEmitter::visit( FrameElement& elem, const std::list< Element* >::const_iterator&   )
201 {
202     if( elem.Children.empty() )
203         return;
204 
205     bool bTextBox = (dynamic_cast<ParagraphElement*>(elem.Children.front()) != NULL);
206     PropertyMap aFrameProps;
207     fillFrameProps( elem, aFrameProps, m_rEmitContext );
208     m_rEmitContext.rEmitter.beginTag( "draw:frame", aFrameProps );
209     if( bTextBox )
210         m_rEmitContext.rEmitter.beginTag( "draw:text-box", PropertyMap() );
211 
212     std::list< Element* >::iterator this_it =  elem.Children.begin();
213     while( this_it !=elem.Children.end() && *this_it != &elem )
214     {
215         (*this_it)->visitedBy( *this, this_it );
216         this_it++;
217     }
218 
219     if( bTextBox )
220         m_rEmitContext.rEmitter.endTag( "draw:text-box" );
221     m_rEmitContext.rEmitter.endTag( "draw:frame" );
222 }
223 
224 void WriterXmlEmitter::visit( PolyPolyElement& elem, const std::list< Element* >::const_iterator& )
225 {
226     elem.updateGeometry();
227     /* note:
228      *   aw recommends using 100dth of mm in all respects since the xml import
229      *   (a) is buggy (see issue 37213)
230      *   (b) is optimized for 100dth of mm and does not scale itself then,
231      *       this does not gain us speed but makes for smaller rounding errors since
232      *       the xml importer coordinates are integer based
233      */
234     for (sal_uInt32 i = 0; i< elem.PolyPoly.count(); i++)
235     {
236         basegfx::B2DPolygon b2dPolygon;
237         b2dPolygon =  elem.PolyPoly.getB2DPolygon( i );
238 
239         for ( sal_uInt32 j = 0; j< b2dPolygon.count(); j++ )
240         {
241             basegfx::B2DPoint point;
242             basegfx::B2DPoint nextPoint;
243             point = b2dPolygon.getB2DPoint( j );
244 
245             basegfx::B2DPoint prevPoint;
246             prevPoint = b2dPolygon.getPrevControlPoint( j ) ;
247 
248             point.setX( convPx2mmPrec2( point.getX() )*100.0 );
249             point.setY( convPx2mmPrec2( point.getY() )*100.0 );
250 
251             if ( b2dPolygon.isPrevControlPointUsed( j ) )
252             {
253                 prevPoint.setX( convPx2mmPrec2( prevPoint.getX() )*100.0 );
254                 prevPoint.setY( convPx2mmPrec2( prevPoint.getY() )*100.0 );
255             }
256 
257             if ( b2dPolygon.isNextControlPointUsed( j ) )
258             {
259                 nextPoint = b2dPolygon.getNextControlPoint( j ) ;
260                 nextPoint.setX( convPx2mmPrec2( nextPoint.getX() )*100.0 );
261                 nextPoint.setY( convPx2mmPrec2( nextPoint.getY() )*100.0 );
262             }
263 
264             b2dPolygon.setB2DPoint( j, point );
265 
266             if ( b2dPolygon.isPrevControlPointUsed( j ) )
267                 b2dPolygon.setPrevControlPoint( j , prevPoint ) ;
268 
269             if ( b2dPolygon.isNextControlPointUsed( j ) )
270                 b2dPolygon.setNextControlPoint( j , nextPoint ) ;
271         }
272 
273         elem.PolyPoly.setB2DPolygon( i, b2dPolygon );
274     }
275 
276     PropertyMap aProps;
277     fillFrameProps( elem, aProps, m_rEmitContext );
278     rtl::OUStringBuffer aBuf( 64 );
279     aBuf.appendAscii( "0 0 " );
280     aBuf.append( convPx2mmPrec2(elem.w)*100.0 );
281     aBuf.append( sal_Unicode(' ') );
282     aBuf.append( convPx2mmPrec2(elem.h)*100.0 );
283     aProps[ USTR( "svg:viewBox" ) ] = aBuf.makeStringAndClear();
284     aProps[ USTR( "svg:d" ) ]       = basegfx::tools::exportToSvgD( elem.PolyPoly, true, true, false );
285 
286     m_rEmitContext.rEmitter.beginTag( "draw:path", aProps );
287     m_rEmitContext.rEmitter.endTag( "draw:path" );
288 }
289 
290 void WriterXmlEmitter::visit( ImageElement& elem, const std::list< Element* >::const_iterator& )
291 {
292     PropertyMap aImageProps;
293     m_rEmitContext.rEmitter.beginTag( "draw:image", aImageProps );
294     m_rEmitContext.rEmitter.beginTag( "office:binary-data", PropertyMap() );
295     m_rEmitContext.rImages.writeBase64EncodedStream( elem.Image, m_rEmitContext);
296     m_rEmitContext.rEmitter.endTag( "office:binary-data" );
297     m_rEmitContext.rEmitter.endTag( "draw:image" );
298 }
299 
300 void WriterXmlEmitter::visit( PageElement& elem, const std::list< Element* >::const_iterator&   )
301 {
302     if( m_rEmitContext.xStatusIndicator.is() )
303         m_rEmitContext.xStatusIndicator->setValue( elem.PageNumber );
304 
305     std::list< Element* >::iterator this_it =  elem.Children.begin();
306     while( this_it !=elem.Children.end() && *this_it != &elem )
307     {
308         (*this_it)->visitedBy( *this, this_it );
309         this_it++;
310     }
311 }
312 
313 void WriterXmlEmitter::visit( DocumentElement& elem, const std::list< Element* >::const_iterator&)
314 {
315     m_rEmitContext.rEmitter.beginTag( "office:body", PropertyMap() );
316     m_rEmitContext.rEmitter.beginTag( "office:text", PropertyMap() );
317 
318     for( std::list< Element* >::iterator it = elem.Children.begin(); it != elem.Children.end(); ++it )
319     {
320         PageElement* pPage = dynamic_cast<PageElement*>(*it);
321         if( pPage )
322         {
323             // emit only page anchored objects
324             // currently these are only DrawElement types
325             for( std::list< Element* >::iterator child_it = pPage->Children.begin(); child_it != pPage->Children.end(); ++child_it )
326             {
327                 if( dynamic_cast<DrawElement*>(*child_it) != NULL )
328                     (*child_it)->visitedBy( *this, child_it );
329             }
330         }
331     }
332 
333     // do not emit page anchored objects, they are emitted before
334     // (must precede all pages in writer document) currently these are
335     // only DrawElement types
336     for( std::list< Element* >::iterator it = elem.Children.begin(); it != elem.Children.end(); ++it )
337     {
338         if( dynamic_cast<DrawElement*>(*it) == NULL )
339             (*it)->visitedBy( *this, it );
340     }
341 
342     m_rEmitContext.rEmitter.endTag( "office:text" );
343     m_rEmitContext.rEmitter.endTag( "office:body" );
344 }
345 
346 /////////////////////////////////////////////////////////////////
347 
348 void WriterXmlOptimizer::visit( HyperlinkElement&, const std::list< Element* >::const_iterator& )
349 {
350 }
351 
352 void WriterXmlOptimizer::visit( TextElement&, const std::list< Element* >::const_iterator&)
353 {
354 }
355 
356 void WriterXmlOptimizer::visit( FrameElement& elem, const std::list< Element* >::const_iterator& )
357 {
358     elem.applyToChildren(*this);
359 }
360 
361 void WriterXmlOptimizer::visit( ImageElement&, const std::list< Element* >::const_iterator& )
362 {
363 }
364 
365 void WriterXmlOptimizer::visit( PolyPolyElement& elem, const std::list< Element* >::const_iterator& )
366 {
367     /* note: optimize two consecutive PolyPolyElements that
368      *  have the same path but one of which is a stroke while
369      *     the other is a fill
370      */
371     if( elem.Parent )
372     {
373         // find following PolyPolyElement in parent's children list
374         std::list< Element* >::iterator this_it = elem.Parent->Children.begin();
375         while( this_it != elem.Parent->Children.end() && *this_it != &elem )
376             ++this_it;
377 
378         if( this_it != elem.Parent->Children.end() )
379         {
380             std::list< Element* >::iterator next_it = this_it;
381             if( ++next_it != elem.Parent->Children.end() )
382             {
383                 PolyPolyElement* pNext = dynamic_cast<PolyPolyElement*>(*next_it);
384                 if( pNext && pNext->PolyPoly == elem.PolyPoly )
385                 {
386                     const GraphicsContext& rNextGC =
387                         m_rProcessor.getGraphicsContext( pNext->GCId );
388                     const GraphicsContext& rThisGC =
389                         m_rProcessor.getGraphicsContext( elem.GCId );
390 
391                     if( rThisGC.BlendMode      == rNextGC.BlendMode &&
392                         rThisGC.Flatness       == rNextGC.Flatness &&
393                         rThisGC.Transformation == rNextGC.Transformation &&
394                         rThisGC.Clip           == rNextGC.Clip &&
395                         pNext->Action          == PATH_STROKE &&
396                         (elem.Action == PATH_FILL || elem.Action == PATH_EOFILL) )
397                     {
398                         GraphicsContext aGC = rThisGC;
399                         aGC.LineJoin  = rNextGC.LineJoin;
400                         aGC.LineCap   = rNextGC.LineCap;
401                         aGC.LineWidth = rNextGC.LineWidth;
402                         aGC.MiterLimit= rNextGC.MiterLimit;
403                         aGC.DashArray = rNextGC.DashArray;
404                         aGC.LineColor = rNextGC.LineColor;
405                         elem.GCId = m_rProcessor.getGCId( aGC );
406 
407                         elem.Action |= pNext->Action;
408 
409                         elem.Children.splice( elem.Children.end(), pNext->Children );
410                         elem.Parent->Children.erase( next_it );
411                         delete pNext;
412                     }
413                 }
414             }
415         }
416     }
417 }
418 
419 void WriterXmlOptimizer::visit( ParagraphElement& elem, const std::list< Element* >::const_iterator& rParentIt)
420 {
421     optimizeTextElements( elem );
422 
423     elem.applyToChildren(*this);
424 
425     if( elem.Parent && rParentIt != elem.Parent->Children.end() )
426     {
427         // find if there is a previous paragraph that might be a heading for this one
428         std::list<Element*>::const_iterator prev = rParentIt;
429         ParagraphElement* pPrevPara = NULL;
430         while( prev != elem.Parent->Children.begin() )
431         {
432             --prev;
433             pPrevPara = dynamic_cast< ParagraphElement* >(*prev);
434             if( pPrevPara )
435             {
436                 /* What constitutes a heading ? current hints are:
437                  * - one line only
438                  * - not too far away from this paragraph (two heading height max ?)
439                  * - font larger or bold
440                  * this is of course incomplete
441                  * FIXME: improve hints for heading
442                  */
443                 // check for single line
444                 if( pPrevPara->isSingleLined( m_rProcessor ) )
445                 {
446                     double head_line_height = pPrevPara->getLineHeight( m_rProcessor );
447                     if( pPrevPara->y + pPrevPara->h + 2*head_line_height > elem.y )
448                     {
449                         // check for larger font
450                         if( head_line_height > elem.getLineHeight( m_rProcessor ) )
451                         {
452                             pPrevPara->Type = elem.Headline;
453                         }
454                         else
455                         {
456                             // check whether text of pPrevPara is bold (at least first text element)
457                             // and this para is not bold (dito)
458                             TextElement* pPrevText = pPrevPara->getFirstTextChild();
459                             TextElement* pThisText = elem.getFirstTextChild();
460                             if( pPrevText && pThisText )
461                             {
462                                 const FontAttributes& rPrevFont = m_rProcessor.getFont( pPrevText->FontId );
463                                 const FontAttributes& rThisFont = m_rProcessor.getFont( pThisText->FontId );
464                                 if( rPrevFont.isBold && ! rThisFont.isBold )
465                                     pPrevPara->Type = elem.Headline;
466                             }
467                         }
468                     }
469                 }
470                 break;
471             }
472         }
473     }
474 }
475 
476 void WriterXmlOptimizer::visit( PageElement& elem, const std::list< Element* >::const_iterator& )
477 {
478     if( m_rProcessor.getStatusIndicator().is() )
479         m_rProcessor.getStatusIndicator()->setValue( elem.PageNumber );
480 
481     // resolve hyperlinks
482     elem.resolveHyperlinks();
483 
484     elem.resolveFontStyles( m_rProcessor ); // underlines and such
485 
486     // FIXME: until hyperlinks and font effects are adjusted for
487     // geometrical search handle them before sorting
488     m_rProcessor.sortElements( &elem );
489 
490     // find paragraphs in text
491     ParagraphElement* pCurPara = NULL;
492     std::list< Element* >::iterator page_element, next_page_element;
493     next_page_element = elem.Children.begin();
494     double fCurLineHeight = 0.0; // average height of text items in current para
495     int nCurLineElements = 0; // number of line contributing elements in current para
496     double line_left = elem.w, line_right = 0.0;
497     double column_width = elem.w*0.75; // estimate text width
498     // TODO: guess columns
499     while( next_page_element != elem.Children.end() )
500     {
501         page_element = next_page_element++;
502         ParagraphElement* pPagePara = dynamic_cast<ParagraphElement*>(*page_element);
503         if( pPagePara )
504         {
505             pCurPara = pPagePara;
506             // adjust line height and text items
507             fCurLineHeight = 0.0;
508             nCurLineElements = 0;
509             for( std::list< Element* >::iterator it = pCurPara->Children.begin();
510                  it != pCurPara->Children.end(); ++it )
511             {
512                 TextElement* pTestText = dynamic_cast<TextElement*>(*it);
513                 if( pTestText )
514                 {
515                     fCurLineHeight = (fCurLineHeight*double(nCurLineElements) + pTestText->h)/double(nCurLineElements+1);
516                     nCurLineElements++;
517                 }
518             }
519             continue;
520         }
521 
522         HyperlinkElement* pLink = dynamic_cast<HyperlinkElement*>(*page_element);
523         DrawElement* pDraw = dynamic_cast<DrawElement*>(*page_element);
524         if( ! pDraw && pLink && ! pLink->Children.empty() )
525             pDraw = dynamic_cast<DrawElement*>(pLink->Children.front() );
526         if( pDraw )
527         {
528             // insert small drawing objects as character, else leave them page bound
529 
530             bool bInsertToParagraph = false;
531             // first check if this is either inside the paragraph
532             if( pCurPara && pDraw->y < pCurPara->y + pCurPara->h )
533             {
534                 if( pDraw->h < fCurLineHeight * 1.5 )
535                 {
536                     bInsertToParagraph = true;
537                     fCurLineHeight = (fCurLineHeight*double(nCurLineElements) + pDraw->h)/double(nCurLineElements+1);
538                     nCurLineElements++;
539                     // mark draw element as character
540                     pDraw->isCharacter = true;
541                 }
542             }
543             // or perhaps the draw element begins a new paragraph
544             else if( next_page_element != elem.Children.end() )
545             {
546                 TextElement* pText = dynamic_cast<TextElement*>(*next_page_element);
547                 if( ! pText )
548                 {
549                     ParagraphElement* pPara = dynamic_cast<ParagraphElement*>(*next_page_element);
550                     if( pPara && ! pPara->Children.empty() )
551                         pText = dynamic_cast<TextElement*>(pPara->Children.front());
552                 }
553                 if( pText && // check there is a text
554                     pDraw->h < pText->h*1.5 && // and it is approx the same height
555                     // and either upper or lower edge of pDraw is inside text's vertical range
556                     ( ( pDraw->y >= pText->y && pDraw->y <= pText->y+pText->h ) ||
557                       ( pDraw->y+pDraw->h >= pText->y && pDraw->y+pDraw->h <= pText->y+pText->h )
558                       )
559                     )
560                 {
561                     bInsertToParagraph = true;
562                     fCurLineHeight = pDraw->h;
563                     nCurLineElements = 1;
564                     line_left = pDraw->x;
565                     line_right = pDraw->x + pDraw->w;
566                     // begin a new paragraph
567                     pCurPara = NULL;
568                     // mark draw element as character
569                     pDraw->isCharacter = true;
570                 }
571             }
572 
573             if( ! bInsertToParagraph )
574             {
575                 pCurPara = NULL;
576                 continue;
577             }
578         }
579 
580         TextElement* pText = dynamic_cast<TextElement*>(*page_element);
581         if( ! pText && pLink && ! pLink->Children.empty() )
582             pText = dynamic_cast<TextElement*>(pLink->Children.front());
583         if( pText )
584         {
585             Element* pGeo = pLink ? static_cast<Element*>(pLink) :
586                                     static_cast<Element*>(pText);
587             if( pCurPara )
588             {
589                 // there was already a text element, check for a new paragraph
590                 if( nCurLineElements > 0 )
591                 {
592                     // if the new text is significantly distant from the paragraph
593                     // begin a new paragraph
594                     if( pGeo->y > pCurPara->y+pCurPara->h + fCurLineHeight*0.5 )
595                         pCurPara = NULL; // insert new paragraph
596                     else if( pGeo->y > (pCurPara->y+pCurPara->h - fCurLineHeight*0.05) )
597                     {
598                         // new paragraph if either the last line of the paragraph
599                         // was significantly shorter than the paragraph as a whole
600                         if( (line_right - line_left) < pCurPara->w*0.75 )
601                             pCurPara = NULL;
602                         // or the last line was significantly smaller than the column width
603                         else if( (line_right - line_left) < column_width*0.75 )
604                             pCurPara = NULL;
605                     }
606                 }
607             }
608             // update line height/width
609             if( pCurPara )
610             {
611                 fCurLineHeight = (fCurLineHeight*double(nCurLineElements) + pGeo->h)/double(nCurLineElements+1);
612                 nCurLineElements++;
613                 if( pGeo->x < line_left )
614                     line_left = pGeo->x;
615                 if( pGeo->x+pGeo->w > line_right )
616                     line_right = pGeo->x+pGeo->w;
617             }
618             else
619             {
620                 fCurLineHeight = pGeo->h;
621                 nCurLineElements = 1;
622                 line_left = pGeo->x;
623                 line_right = pGeo->x + pGeo->w;
624             }
625         }
626 
627         // move element to current paragraph
628         if( ! pCurPara ) // new paragraph, insert one
629         {
630             pCurPara = m_rProcessor.getElementFactory()->createParagraphElement( NULL );
631             // set parent
632             pCurPara->Parent = &elem;
633             //insert new paragraph before current element
634             page_element = elem.Children.insert( page_element, pCurPara );
635             // forward iterator to current element again
636             ++ page_element;
637             // update next_element which is now invalid
638             next_page_element = page_element;
639             ++ next_page_element;
640         }
641         Element* pCurEle = *page_element;
642         pCurEle->setParent( page_element, pCurPara );
643         OSL_ENSURE( !pText || pCurEle == pText || pCurEle == pLink, "paragraph child list in disorder" );
644         if( pText || pDraw )
645             pCurPara->updateGeometryWith( pCurEle );
646     }
647 
648     // process children
649     elem.applyToChildren(*this);
650 
651     // find possible header and footer
652     checkHeaderAndFooter( elem );
653 }
654 
655 void WriterXmlOptimizer::checkHeaderAndFooter( PageElement& rElem )
656 {
657     /* indicators for a header:
658      *  - single line paragrah at top of page (  inside 15% page height)
659      *  - at least linheight above the next paragr   aph
660      *
661      *  indicators for a footer likewise:
662      *  - single line paragraph at bottom of page (inside 15% page height)
663      *  - at least lineheight below the previous paragraph
664      */
665 
666     // detect header
667     // Note: the following assumes that the pages' chiuldren have been
668     // sorted geometrically
669     std::list< Element* >::iterator it = rElem.Children.begin();
670     while( it != rElem.Children.end() )
671     {
672         ParagraphElement* pPara = dynamic_cast<ParagraphElement*>(*it);
673         if( pPara )
674         {
675             if( pPara->y+pPara->h < rElem.h*0.15 && pPara->isSingleLined( m_rProcessor ) )
676             {
677                 std::list< Element* >::iterator next_it = it;
678                 ParagraphElement* pNextPara = NULL;
679                 while( ++next_it != rElem.Children.end() && pNextPara == NULL )
680                 {
681                     pNextPara = dynamic_cast<ParagraphElement*>(*next_it);
682                 }
683                 if( pNextPara && pNextPara->y > pPara->y+pPara->h*2 )
684                 {
685                     rElem.HeaderElement = pPara;
686                     pPara->Parent = NULL;
687                     rElem.Children.remove( pPara );
688                 }
689             }
690             break;
691         }
692         ++it;
693     }
694 
695     // detect footer
696     std::list< Element* >::reverse_iterator rit = rElem.Children.rbegin();
697     while( rit != rElem.Children.rend() )
698     {
699         ParagraphElement* pPara = dynamic_cast<ParagraphElement*>(*rit);
700         if( pPara )
701         {
702             if( pPara->y > rElem.h*0.85 && pPara->isSingleLined( m_rProcessor ) )
703             {
704                 std::list< Element* >::reverse_iterator next_it = rit;
705                 ParagraphElement* pNextPara = NULL;
706                 while( ++next_it != rElem.Children.rend() && pNextPara == NULL )
707                 {
708                     pNextPara = dynamic_cast<ParagraphElement*>(*next_it);
709                 }
710                 if( pNextPara && pNextPara->y < pPara->y-pPara->h*2 )
711                 {
712                     rElem.FooterElement = pPara;
713                     pPara->Parent = NULL;
714                     rElem.Children.remove( pPara );
715                 }
716             }
717             break;
718         }
719         ++rit;
720     }
721 }
722 
723 void WriterXmlOptimizer::optimizeTextElements(Element& rParent)
724 {
725     if( rParent.Children.empty() ) // this should not happen
726     {
727         OSL_ENSURE( 0, "empty paragraph optimized" );
728         return;
729     }
730 
731     // concatenate child elements with same font id
732     std::list< Element* >::iterator next = rParent.Children.begin();
733     std::list< Element* >::iterator it = next++;
734     FrameElement* pFrame = dynamic_cast<FrameElement*>(rParent.Parent);
735     bool bRotatedFrame = false;
736     if( pFrame )
737     {
738         const GraphicsContext& rFrameGC = m_rProcessor.getGraphicsContext( pFrame->GCId );
739         if( rFrameGC.isRotatedOrSkewed() )
740             bRotatedFrame = true;
741     }
742     while( next != rParent.Children.end() )
743     {
744         bool bConcat = false;
745         TextElement* pCur = dynamic_cast<TextElement*>(*it);
746         if( pCur )
747         {
748             TextElement* pNext = dynamic_cast<TextElement*>(*next);
749             if( pNext )
750             {
751                 const GraphicsContext& rCurGC = m_rProcessor.getGraphicsContext( pCur->GCId );
752                 const GraphicsContext& rNextGC = m_rProcessor.getGraphicsContext( pNext->GCId );
753 
754                 // line and space optimization; works only in strictly horizontal mode
755 
756                 if( !bRotatedFrame
757                     && ! rCurGC.isRotatedOrSkewed()
758                     && ! rNextGC.isRotatedOrSkewed()
759                     && pNext->Text.charAt( 0 ) != sal_Unicode(' ')
760                     && pCur->Text.getLength() >  0
761                     && pCur->Text.charAt( pCur->Text.getLength()-1 ) != sal_Unicode(' ')
762                     )
763                 {
764                     // check for new line in paragraph
765                     if( pNext->y > pCur->y+pCur->h )
766                     {
767                         // new line begins
768                         // check whether a space would should be inserted or a hyphen removed
769                         sal_Unicode aLastCode = pCur->Text.charAt( pCur->Text.getLength()-1 );
770                         if( aLastCode == '-'
771                             || aLastCode == 0x2010
772                             || (aLastCode >= 0x2012 && aLastCode <= 0x2015)
773                             || aLastCode == 0xff0d
774                         )
775                         {
776                             // cut a hyphen
777                             pCur->Text.setLength( pCur->Text.getLength()-1 );
778                         }
779                         // append a space unless there is a non breaking hyphen
780                         else if( aLastCode != 0x2011 )
781                         {
782                             pCur->Text.append( sal_Unicode( ' ' ) );
783                         }
784                     }
785                     else // we're continuing the same line
786                     {
787                         // check whether a space would should be inserted
788                         // check for a small horizontal offset
789                         if( pCur->x + pCur->w + pNext->h*0.15 < pNext->x )
790                         {
791                             pCur->Text.append( sal_Unicode(' ') );
792                         }
793                     }
794                 }
795                 // concatenate consecutive text elements unless there is a
796                 // font or text color or matrix change, leave a new span in that case
797                 if( pCur->FontId == pNext->FontId &&
798                     rCurGC.FillColor.Red == rNextGC.FillColor.Red &&
799                     rCurGC.FillColor.Green == rNextGC.FillColor.Green &&
800                     rCurGC.FillColor.Blue == rNextGC.FillColor.Blue &&
801                     rCurGC.FillColor.Alpha == rNextGC.FillColor.Alpha &&
802                     rCurGC.Transformation == rNextGC.Transformation
803                     )
804                 {
805                     pCur->updateGeometryWith( pNext );
806                     // append text to current element
807                     pCur->Text.append( pNext->Text.getStr(), pNext->Text.getLength() );
808                     // append eventual children to current element
809                     // and clear children (else the children just
810                     // appended to pCur would be destroyed)
811                     pCur->Children.splice( pCur->Children.end(), pNext->Children );
812                     // get rid of the now useless element
813                     rParent.Children.erase( next );
814                     delete pNext;
815                     bConcat = true;
816                 }
817             }
818         }
819         else if( dynamic_cast<HyperlinkElement*>(*it) )
820             optimizeTextElements( **it );
821         if( bConcat )
822         {
823             next = it;
824             ++next;
825         }
826         else
827         {
828             ++it;
829             ++next;
830         }
831     }
832 }
833 
834 void WriterXmlOptimizer::visit( DocumentElement& elem, const std::list< Element* >::const_iterator&)
835 {
836     elem.applyToChildren(*this);
837 }
838 
839 //////////////////////////////////////////////////////////////////////////////////
840 
841 
842 void WriterXmlFinalizer::visit( PolyPolyElement& elem, const std::list< Element* >::const_iterator& )
843 {
844     // xxx TODO copied from DrawElement
845     const GraphicsContext& rGC = m_rProcessor.getGraphicsContext(elem.GCId );
846     PropertyMap aProps;
847     aProps[ USTR( "style:family" ) ] = USTR( "graphic" );
848 
849     PropertyMap aGCProps;
850 
851     // TODO(F3): proper dash emulation
852     if( elem.Action & PATH_STROKE )
853     {
854         aGCProps[ USTR("draw:stroke") ] = rGC.DashArray.empty() ? USTR("solid") : USTR("dash");
855         aGCProps[ USTR("svg:stroke-color") ] = getColorString( rGC.LineColor );
856         if( rGC.LineWidth != 0.0 )
857         {
858             ::basegfx::B2DVector aVec(rGC.LineWidth,0);
859             aVec *= rGC.Transformation;
860 
861             aVec.setX ( convPx2mmPrec2( aVec.getX() )*100.0 );
862             aVec.setY ( convPx2mmPrec2( aVec.getY() )*100.0 );
863 
864             aGCProps[ USTR("svg:stroke-width") ] = rtl::OUString::valueOf( aVec.getLength() );
865         }
866     }
867     else
868     {
869         aGCProps[ USTR("draw:stroke") ] = USTR("none");
870     }
871 
872     // TODO(F1): check whether stuff could be emulated by gradient/bitmap/hatch
873     if( elem.Action & (PATH_FILL | PATH_EOFILL) )
874     {
875         aGCProps[ USTR("draw:fill") ]   = USTR("solid");
876         aGCProps[ USTR("draw:fill-color") ] = getColorString( rGC.FillColor );
877     }
878     else
879     {
880         aGCProps[ USTR("draw:fill") ] = USTR("none");
881     }
882 
883     StyleContainer::Style aStyle( "style:style", aProps );
884     StyleContainer::Style aSubStyle( "style:graphic-properties", aGCProps );
885     aStyle.SubStyles.push_back( &aSubStyle );
886 
887     elem.StyleId = m_rStyleContainer.getStyleId( aStyle );
888 }
889 
890 void WriterXmlFinalizer::visit( HyperlinkElement&, const std::list< Element* >::const_iterator& )
891 {
892 }
893 
894 void WriterXmlFinalizer::visit( TextElement& elem, const std::list< Element* >::const_iterator& )
895 {
896     const FontAttributes& rFont = m_rProcessor.getFont( elem.FontId );
897     PropertyMap aProps;
898     aProps[ USTR( "style:family" ) ] = USTR( "text" );
899 
900     PropertyMap aFontProps;
901 
902     // family name
903     aFontProps[ USTR( "fo:font-family" ) ] = rFont.familyName;
904     // bold
905     if( rFont.isBold )
906     {
907         aFontProps[ USTR( "fo:font-weight" ) ]         = USTR( "bold" );
908         aFontProps[ USTR( "fo:font-weight-asian" ) ]   = USTR( "bold" );
909         aFontProps[ USTR( "fo:font-weight-complex" ) ] = USTR( "bold" );
910     }
911     // italic
912     if( rFont.isItalic )
913     {
914         aFontProps[ USTR( "fo:font-style" ) ]         = USTR( "italic" );
915         aFontProps[ USTR( "fo:font-style-asian" ) ]   = USTR( "italic" );
916         aFontProps[ USTR( "fo:font-style-complex" ) ] = USTR( "italic" );
917     }
918     // underline
919     if( rFont.isUnderline )
920     {
921         aFontProps[ USTR( "style:text-underline-style" ) ]  = USTR( "solid" );
922         aFontProps[ USTR( "style:text-underline-width" ) ]  = USTR( "auto" );
923         aFontProps[ USTR( "style:text-underline-color" ) ]  = USTR( "font-color" );
924     }
925     // outline
926     if( rFont.isOutline )
927     {
928         aFontProps[ USTR( "style:text-outline" ) ]  = USTR( "true" );
929     }
930     // size
931     rtl::OUStringBuffer aBuf( 32 );
932     aBuf.append( rFont.size*72/PDFI_OUTDEV_RESOLUTION );
933     aBuf.appendAscii( "pt" );
934     rtl::OUString aFSize = aBuf.makeStringAndClear();
935     aFontProps[ USTR( "fo:font-size" ) ]            = aFSize;
936     aFontProps[ USTR( "style:font-size-asian" ) ]   = aFSize;
937     aFontProps[ USTR( "style:font-size-complex" ) ] = aFSize;
938     // color
939     const GraphicsContext& rGC = m_rProcessor.getGraphicsContext( elem.GCId );
940     aFontProps[ USTR( "fo:color" ) ]                 =  getColorString( rFont.isOutline ? rGC.LineColor : rGC.FillColor );
941 
942     StyleContainer::Style aStyle( "style:style", aProps );
943     StyleContainer::Style aSubStyle( "style:text-properties", aFontProps );
944     aStyle.SubStyles.push_back( &aSubStyle );
945     elem.StyleId = m_rStyleContainer.getStyleId( aStyle );
946 }
947 
948 void WriterXmlFinalizer::visit( ParagraphElement& elem, const std::list< Element* >::const_iterator& rParentIt )
949 {
950     PropertyMap aParaProps;
951 
952     if( elem.Parent )
953     {
954         // check for center alignement
955         // criterion: paragraph is small relative to parent and distributed around its center
956         double p_x = elem.Parent->x;
957         double p_y = elem.Parent->y;
958         double p_w = elem.Parent->w;
959         double p_h = elem.Parent->h;
960 
961         PageElement* pPage = dynamic_cast<PageElement*>(elem.Parent);
962         if( pPage )
963         {
964             p_x += pPage->LeftMargin;
965             p_y += pPage->TopMargin;
966             p_w -= pPage->LeftMargin+pPage->RightMargin;
967             p_h -= pPage->TopMargin+pPage->BottomMargin;
968         }
969         bool bIsCenter = false;
970         if( elem.w < ( p_w/2) )
971         {
972             double delta = elem.w/4;
973             // allow very small paragraphs to deviate a little more
974             // relative to parent's center
975             if( elem.w <  p_w/8 )
976                 delta = elem.w;
977             if( fabs( elem.x+elem.w/2 - ( p_x+ p_w/2) ) <  delta ||
978                 (pPage && fabs( elem.x+elem.w/2 - (pPage->x + pPage->w/2) ) <  delta) )
979             {
980                 bIsCenter = true;
981                 aParaProps[ USTR( "fo:text-align" ) ] = USTR( "center" );
982             }
983         }
984         if( ! bIsCenter && elem.x > p_x + p_w/10 )
985         {
986             // indent
987             rtl::OUStringBuffer aBuf( 32 );
988             aBuf.append( convPx2mm( elem.x - p_x ) );
989             aBuf.appendAscii( "mm" );
990             aParaProps[ USTR( "fo:margin-left" ) ] = aBuf.makeStringAndClear();
991         }
992 
993         // check whether to leave some space to next paragraph
994         // find wether there is a next paragraph
995         std::list< Element* >::const_iterator it = rParentIt;
996         const ParagraphElement* pNextPara = NULL;
997         while( ++it != elem.Parent->Children.end() && ! pNextPara )
998             pNextPara = dynamic_cast< const ParagraphElement* >(*it);
999         if( pNextPara )
1000         {
1001             if( pNextPara->y - (elem.y+elem.h) > convmm2Px( 10 ) )
1002             {
1003                 rtl::OUStringBuffer aBuf( 32 );
1004                 aBuf.append( convPx2mm( pNextPara->y - (elem.y+elem.h) ) );
1005                 aBuf.appendAscii( "mm" );
1006                 aParaProps[ USTR( "fo:margin-bottom" ) ] = aBuf.makeStringAndClear();
1007             }
1008         }
1009     }
1010 
1011     if( ! aParaProps.empty() )
1012     {
1013         PropertyMap aProps;
1014         aProps[ USTR( "style:family" ) ] = USTR( "paragraph" );
1015         StyleContainer::Style aStyle( "style:style", aProps );
1016         StyleContainer::Style aSubStyle( "style:paragraph-properties", aParaProps );
1017         aStyle.SubStyles.push_back( &aSubStyle );
1018         elem.StyleId = m_rStyleContainer.getStyleId( aStyle );
1019     }
1020 
1021     elem.applyToChildren(*this);
1022 }
1023 
1024 void WriterXmlFinalizer::visit( FrameElement& elem, const std::list< Element* >::const_iterator&)
1025 {
1026     PropertyMap aProps;
1027     aProps[ USTR( "style:family" ) ] = USTR( "graphic" );
1028 
1029     PropertyMap aGCProps;
1030 
1031     aGCProps[ USTR("draw:stroke") ] = USTR("none");
1032     aGCProps[ USTR("draw:fill") ] = USTR("none");
1033 
1034     StyleContainer::Style aStyle( "style:style", aProps );
1035     StyleContainer::Style aSubStyle( "style:graphic-properties", aGCProps );
1036     aStyle.SubStyles.push_back( &aSubStyle );
1037 
1038     elem.StyleId = m_rStyleContainer.getStyleId( aStyle );
1039     elem.applyToChildren(*this);
1040 }
1041 
1042 void WriterXmlFinalizer::visit( ImageElement&, const std::list< Element* >::const_iterator& )
1043 {
1044 }
1045 
1046 void WriterXmlFinalizer::setFirstOnPage( ParagraphElement&    rElem,
1047                                          StyleContainer&      rStyles,
1048                                          const rtl::OUString& rMasterPageName )
1049 {
1050     PropertyMap aProps;
1051     if( rElem.StyleId != -1 )
1052     {
1053         const PropertyMap* pProps = rStyles.getProperties( rElem.StyleId );
1054         if( pProps )
1055             aProps = *pProps;
1056     }
1057 
1058     aProps[ USTR( "style:family" ) ] = USTR( "paragraph" );
1059     aProps[ USTR( "style:master-page-name" ) ] = rMasterPageName;
1060 
1061     if( rElem.StyleId != -1 )
1062         rElem.StyleId = rStyles.setProperties( rElem.StyleId, aProps );
1063     else
1064     {
1065         StyleContainer::Style aStyle( "style:style", aProps );
1066         rElem.StyleId = rStyles.getStyleId( aStyle );
1067     }
1068 }
1069 
1070 void WriterXmlFinalizer::visit( PageElement& elem, const std::list< Element* >::const_iterator& )
1071 {
1072     if( m_rProcessor.getStatusIndicator().is() )
1073         m_rProcessor.getStatusIndicator()->setValue( elem.PageNumber );
1074 
1075     // transform from pixel to mm
1076     double page_width = convPx2mm( elem.w ), page_height = convPx2mm( elem.h );
1077 
1078     // calculate page margins out of the relevant children (paragraphs)
1079     elem.TopMargin = elem.h, elem.BottomMargin = 0, elem.LeftMargin = elem.w, elem.RightMargin = 0;
1080     // first element should be a paragraphy
1081     ParagraphElement* pFirstPara = NULL;
1082     for( std::list< Element* >::const_iterator it = elem.Children.begin(); it != elem.Children.end(); ++it )
1083     {
1084         if( dynamic_cast<ParagraphElement*>( *it ) )
1085         {
1086             if( (*it)->x < elem.LeftMargin )
1087                 elem.LeftMargin = (*it)->x;
1088             if( (*it)->y < elem.TopMargin )
1089                 elem.TopMargin = (*it)->y;
1090             if( (*it)->x + (*it)->w > elem.w - elem.RightMargin )
1091                 elem.RightMargin = elem.w - ((*it)->x + (*it)->w);
1092             if( (*it)->y + (*it)->h > elem.h - elem.BottomMargin )
1093                 elem.BottomMargin = elem.h - ((*it)->y + (*it)->h);
1094             if( ! pFirstPara )
1095                 pFirstPara = dynamic_cast<ParagraphElement*>( *it );
1096         }
1097     }
1098     if( elem.HeaderElement && elem.HeaderElement->y < elem.TopMargin )
1099         elem.TopMargin = elem.HeaderElement->y;
1100     if( elem.FooterElement && elem.FooterElement->y+elem.FooterElement->h > elem.h - elem.BottomMargin )
1101         elem.BottomMargin = elem.h - (elem.FooterElement->y + elem.FooterElement->h);
1102 
1103     // transform margins to mm
1104     double left_margin     = convPx2mm( elem.LeftMargin );
1105     double right_margin    = convPx2mm( elem.RightMargin );
1106     double top_margin      = convPx2mm( elem.TopMargin );
1107     double bottom_margin   = convPx2mm( elem.BottomMargin );
1108     if( ! pFirstPara )
1109     {
1110         // use default page margins
1111         left_margin     = 10;
1112         right_margin    = 10;
1113         top_margin      = 10;
1114         bottom_margin   = 10;
1115     }
1116 
1117     // round left/top margin to nearest mm
1118     left_margin     = rtl_math_round( left_margin, 0, rtl_math_RoundingMode_Floor );
1119     top_margin      = rtl_math_round( top_margin, 0, rtl_math_RoundingMode_Floor );
1120     // round (fuzzy) right/bottom margin to nearest cm
1121     right_margin    = rtl_math_round( right_margin, right_margin >= 10 ? -1 : 0, rtl_math_RoundingMode_Floor );
1122     bottom_margin   = rtl_math_round( bottom_margin, bottom_margin >= 10 ? -1 : 0, rtl_math_RoundingMode_Floor );
1123 
1124     // set reasonable default in case of way too large margins
1125     // e.g. no paragraph case
1126     if( left_margin > page_width/2.0 - 10 )
1127         left_margin = 10;
1128     if( right_margin > page_width/2.0 - 10 )
1129         right_margin = 10;
1130     if( top_margin > page_height/2.0 - 10 )
1131         top_margin = 10;
1132     if( bottom_margin > page_height/2.0 - 10 )
1133         bottom_margin = 10;
1134 
1135     // catch the weird cases
1136     if( left_margin < 0 )
1137         left_margin = 0;
1138     if( right_margin < 0 )
1139         right_margin = 0;
1140     if( top_margin < 0 )
1141         top_margin = 0;
1142     if( bottom_margin < 0 )
1143         bottom_margin = 0;
1144 
1145     // widely differing margins are unlikely to be correct
1146     if( right_margin > left_margin*1.5 )
1147         right_margin = left_margin;
1148 
1149     elem.LeftMargin      = convmm2Px( left_margin );
1150     elem.RightMargin     = convmm2Px( right_margin );
1151     elem.TopMargin       = convmm2Px( top_margin );
1152     elem.BottomMargin    = convmm2Px( bottom_margin );
1153 
1154     // get styles for paragraphs
1155     PropertyMap aPageProps;
1156     PropertyMap aPageLayoutProps;
1157     rtl::OUStringBuffer aBuf( 64 );
1158     aPageLayoutProps[ USTR( "fo:page-width" ) ]     = unitMMString( page_width );
1159     aPageLayoutProps[ USTR( "fo:page-height" ) ]    = unitMMString( page_height );
1160     aPageLayoutProps[ USTR( "style:print-orientation" ) ]
1161         = elem.w < elem.h ? USTR( "portrait" ) : USTR( "landscape" );
1162     aPageLayoutProps[ USTR( "fo:margin-top" ) ]     = unitMMString( top_margin );
1163     aPageLayoutProps[ USTR( "fo:margin-bottom" ) ]  = unitMMString( bottom_margin );
1164     aPageLayoutProps[ USTR( "fo:margin-left" ) ]    = unitMMString( left_margin );
1165     aPageLayoutProps[ USTR( "fo:margin-right" ) ]   = unitMMString( right_margin );
1166     aPageLayoutProps[ USTR( "style:writing-mode" ) ]= USTR( "lr-tb" );
1167 
1168     StyleContainer::Style aStyle( "style:page-layout", aPageProps);
1169     StyleContainer::Style aSubStyle( "style:page-layout-properties", aPageLayoutProps);
1170     aStyle.SubStyles.push_back(&aSubStyle);
1171     sal_Int32 nPageStyle = m_rStyleContainer.impl_getStyleId( aStyle, false );
1172 
1173     // create master page
1174     rtl::OUString aMasterPageLayoutName = m_rStyleContainer.getStyleName( nPageStyle );
1175     aPageProps[ USTR( "style:page-layout-name" ) ] = aMasterPageLayoutName;
1176     StyleContainer::Style aMPStyle( "style:master-page", aPageProps );
1177     StyleContainer::Style aHeaderStyle( "style:header", PropertyMap() );
1178     StyleContainer::Style aFooterStyle( "style:footer", PropertyMap() );
1179     if( elem.HeaderElement )
1180     {
1181         elem.HeaderElement->visitedBy( *this, std::list<Element*>::iterator() );
1182         aHeaderStyle.ContainedElement = elem.HeaderElement;
1183         aMPStyle.SubStyles.push_back( &aHeaderStyle );
1184     }
1185     if( elem.FooterElement )
1186     {
1187         elem.FooterElement->visitedBy( *this, std::list<Element*>::iterator() );
1188         aFooterStyle.ContainedElement = elem.FooterElement;
1189         aMPStyle.SubStyles.push_back( &aFooterStyle );
1190     }
1191     elem.StyleId = m_rStyleContainer.impl_getStyleId( aMPStyle,false );
1192 
1193 
1194     rtl::OUString aMasterPageName = m_rStyleContainer.getStyleName( elem.StyleId );
1195 
1196     // create styles for children
1197     elem.applyToChildren(*this);
1198 
1199     // no paragraph or other elements before the first paragraph
1200     if( ! pFirstPara )
1201     {
1202         pFirstPara = m_rProcessor.getElementFactory()->createParagraphElement( NULL );
1203         pFirstPara->Parent = &elem;
1204         elem.Children.push_front( pFirstPara );
1205     }
1206     setFirstOnPage(*pFirstPara, m_rStyleContainer, aMasterPageName);
1207 }
1208 
1209 void WriterXmlFinalizer::visit( DocumentElement& elem, const std::list< Element* >::const_iterator& )
1210 {
1211     elem.applyToChildren(*this);
1212 }
1213 
1214 }
1215