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_starmath.hxx"
26 
27 /*
28  Warning: The SvXMLElementExport helper class creates the beginning and
29  closing tags of xml elements in its constructor and destructor, so theres
30  hidden stuff going on, on occasion the ordering of these classes declarations
31  may be significant
32 */
33 
34 
35 #include <com/sun/star/xml/sax/XErrorHandler.hpp>
36 #include <com/sun/star/xml/sax/XEntityResolver.hpp>
37 #include <com/sun/star/xml/sax/InputSource.hpp>
38 #include <com/sun/star/xml/sax/XDTDHandler.hpp>
39 #include <com/sun/star/xml/sax/XParser.hpp>
40 #include <com/sun/star/io/XActiveDataSource.hpp>
41 #include <com/sun/star/io/XActiveDataControl.hpp>
42 #include <com/sun/star/document/XDocumentProperties.hpp>
43 #include <com/sun/star/document/XDocumentPropertiesSupplier.hpp>
44 #include <com/sun/star/packages/zip/ZipIOException.hpp>
45 #include <com/sun/star/task/XStatusIndicatorFactory.hpp>
46 #include <com/sun/star/beans/PropertyAttribute.hpp>
47 #include <com/sun/star/container/XNameAccess.hpp>
48 #include <com/sun/star/embed/ElementModes.hpp>
49 #include <com/sun/star/uno/Any.h>
50 
51 #include <rtl/math.hxx>
52 #include <sfx2/frame.hxx>
53 #include <sfx2/docfile.hxx>
54 #include <tools/debug.hxx>
55 #include <tools/urlobj.hxx>
56 #include <svtools/sfxecode.hxx>
57 #include <unotools/saveopt.hxx>
58 #include <svl/stritem.hxx>
59 #include <svl/itemprop.hxx>
60 #include <unotools/processfactory.hxx>
61 #include <unotools/streamwrap.hxx>
62 #include <xmloff/xmlnmspe.hxx>
63 #include <xmloff/xmltoken.hxx>
64 #include <xmloff/nmspmap.hxx>
65 #include <xmloff/attrlist.hxx>
66 #include <xmloff/xmluconv.hxx>
67 #include <xmloff/xmlmetai.hxx>
68 #include <osl/mutex.hxx>
69 #include <comphelper/genericpropertyset.hxx>
70 
71 #include <memory>
72 
73 #include "mathmlexport.hxx"
74 #include <starmath.hrc>
75 #include <unomodel.hxx>
76 #include <document.hxx>
77 #include <utility.hxx>
78 #include <config.hxx>
79 
80 using namespace ::com::sun::star::beans;
81 using namespace ::com::sun::star::container;
82 using namespace ::com::sun::star::document;
83 using namespace ::com::sun::star::lang;
84 using namespace ::com::sun::star::uno;
85 using namespace ::com::sun::star;
86 using namespace ::xmloff::token;
87 
88 using ::rtl::OUString;
89 using ::rtl::OUStringBuffer;
90 
91 #define EXPORT_SVC_NAME RTL_CONSTASCII_USTRINGPARAM("com.sun.star.xml.XMLExportFilter")
92 
93 #undef WANTEXCEPT
94 
95 
96 ////////////////////////////////////////////////////////////
97 
Export(SfxMedium & rMedium)98 sal_Bool SmXMLExportWrapper::Export(SfxMedium &rMedium)
99 {
100     sal_Bool bRet=sal_True;
101     uno::Reference<lang::XMultiServiceFactory>
102         xServiceFactory(utl::getProcessServiceFactory());
103     DBG_ASSERT(xServiceFactory.is(),"got no service manager");
104 
105     //Get model
106     uno::Reference< lang::XComponent > xModelComp(xModel, uno::UNO_QUERY );
107 
108     sal_Bool bEmbedded = sal_False;
109     uno::Reference <lang::XUnoTunnel> xTunnel;
110     xTunnel = uno::Reference <lang::XUnoTunnel> (xModel,uno::UNO_QUERY);
111     SmModel *pModel = reinterpret_cast<SmModel *>
112         (xTunnel->getSomething(SmModel::getUnoTunnelId()));
113 
114     SmDocShell *pDocShell = pModel ?
115             static_cast<SmDocShell*>(pModel->GetObjectShell()) : 0;
116     if ( pDocShell &&
117         SFX_CREATE_MODE_EMBEDDED == pDocShell->GetCreateMode() )
118         bEmbedded = sal_True;
119 
120     uno::Reference<task::XStatusIndicator> xStatusIndicator;
121     if (!bEmbedded)
122     {
123         if (pDocShell /*&& pDocShell->GetMedium()*/)
124         {
125             DBG_ASSERT( pDocShell->GetMedium() == &rMedium,
126                     "different SfxMedium found" );
127 
128             SfxItemSet* pSet = rMedium.GetItemSet();
129             if (pSet)
130             {
131                 const SfxUnoAnyItem* pItem = static_cast<const SfxUnoAnyItem*>(
132                     pSet->GetItem(SID_PROGRESS_STATUSBAR_CONTROL) );
133                 if (pItem)
134                     pItem->GetValue() >>= xStatusIndicator;
135             }
136         }
137 
138         // set progress range and start status indicator
139         if (xStatusIndicator.is())
140         {
141             sal_Int32 nProgressRange = bFlat ? 1 : 3;
142             xStatusIndicator->start(String(SmResId(STR_STATSTR_WRITING)),
143                 nProgressRange);
144         }
145     }
146 
147 
148     // create XPropertySet with three properties for status indicator
149     comphelper::PropertyMapEntry aInfoMap[] =
150     {
151         { "UsePrettyPrinting", sizeof("UsePrettyPrinting")-1, 0,
152               &::getBooleanCppuType(),
153               beans::PropertyAttribute::MAYBEVOID, 0},
154         { "BaseURI", sizeof("BaseURI")-1, 0,
155               &::getCppuType( (OUString *)0 ),
156               beans::PropertyAttribute::MAYBEVOID, 0 },
157         { "StreamRelPath", sizeof("StreamRelPath")-1, 0,
158               &::getCppuType( (OUString *)0 ),
159               beans::PropertyAttribute::MAYBEVOID, 0 },
160         { "StreamName", sizeof("StreamName")-1, 0,
161               &::getCppuType( (OUString *)0 ),
162               beans::PropertyAttribute::MAYBEVOID, 0 },
163         { NULL, 0, 0, NULL, 0, 0 }
164     };
165     uno::Reference< beans::XPropertySet > xInfoSet(
166                 comphelper::GenericPropertySet_CreateInstance(
167                             new comphelper::PropertySetInfo( aInfoMap ) ) );
168 
169     SvtSaveOptions aSaveOpt;
170     OUString sUsePrettyPrinting(RTL_CONSTASCII_USTRINGPARAM("UsePrettyPrinting"));
171     sal_Bool bUsePrettyPrinting( bFlat || aSaveOpt.IsPrettyPrinting() );
172     xInfoSet->setPropertyValue( sUsePrettyPrinting, uno::makeAny(bUsePrettyPrinting));
173 
174     // Set base URI
175     OUString sPropName( RTL_CONSTASCII_USTRINGPARAM("BaseURI") );
176     xInfoSet->setPropertyValue( sPropName, makeAny( rMedium.GetBaseURL( true ) ) );
177 
178     sal_Int32 nSteps=0;
179     if (xStatusIndicator.is())
180             xStatusIndicator->setValue(nSteps++);
181     if (!bFlat) //Storage (Package) of Stream
182     {
183         uno::Reference < embed::XStorage > xStg = rMedium.GetOutputStorage();
184         sal_Bool bOASIS = ( SotStorage::GetVersion( xStg ) > SOFFICE_FILEFORMAT_60 );
185 
186         // TODO/LATER: handle the case of embedded links gracefully
187         if ( bEmbedded ) //&& !pStg->IsRoot() )
188         {
189             OUString aName;
190             if ( rMedium.GetItemSet() )
191             {
192                 const SfxStringItem* pDocHierarchItem = static_cast<const SfxStringItem*>(
193                     rMedium.GetItemSet()->GetItem(SID_DOC_HIERARCHICALNAME) );
194                 if ( pDocHierarchItem )
195                     aName = pDocHierarchItem->GetValue();
196             }
197 
198             if ( aName.getLength() )
199             {
200                 sPropName = OUString(RTL_CONSTASCII_USTRINGPARAM("StreamRelPath"));
201                 xInfoSet->setPropertyValue( sPropName, makeAny( aName ) );
202             }
203         }
204 
205         if ( !bEmbedded )
206         {
207             if (xStatusIndicator.is())
208                 xStatusIndicator->setValue(nSteps++);
209 
210             bRet = WriteThroughComponent(
211                     xStg, xModelComp, "meta.xml", xServiceFactory, xInfoSet,
212                     (bOASIS ? "com.sun.star.comp.Math.XMLOasisMetaExporter"
213                             : "com.sun.star.comp.Math.XMLMetaExporter"),
214                     sal_False);
215         }
216         if ( bRet )
217         {
218            if (xStatusIndicator.is())
219                 xStatusIndicator->setValue(nSteps++);
220 
221             bRet = WriteThroughComponent(
222                     xStg, xModelComp, "content.xml", xServiceFactory, xInfoSet,
223                     "com.sun.star.comp.Math.XMLContentExporter");
224         }
225 
226         if ( bRet )
227         {
228             if (xStatusIndicator.is())
229                 xStatusIndicator->setValue(nSteps++);
230 
231             bRet = WriteThroughComponent(
232                     xStg, xModelComp, "settings.xml", xServiceFactory, xInfoSet,
233                     (bOASIS ? "com.sun.star.comp.Math.XMLOasisSettingsExporter"
234                             : "com.sun.star.comp.Math.XMLSettingsExporter") );
235         }
236     }
237     else
238     {
239         SvStream *pStream = rMedium.GetOutStream();
240         uno::Reference<io::XOutputStream> xOut(
241             new utl::OOutputStreamWrapper(*pStream) );
242 
243         if (xStatusIndicator.is())
244             xStatusIndicator->setValue(nSteps++);
245 
246         bRet = WriteThroughComponent(
247             xOut, xModelComp, xServiceFactory, xInfoSet,
248             "com.sun.star.comp.Math.XMLContentExporter");
249     }
250 
251     if (xStatusIndicator.is())
252         xStatusIndicator->end();
253 
254     return bRet;
255 }
256 
257 
258 /// export through an XML exporter component (output stream version)
WriteThroughComponent(Reference<io::XOutputStream> xOutputStream,Reference<XComponent> xComponent,Reference<lang::XMultiServiceFactory> & rFactory,Reference<beans::XPropertySet> & rPropSet,const sal_Char * pComponentName)259 sal_Bool SmXMLExportWrapper::WriteThroughComponent(
260     Reference<io::XOutputStream> xOutputStream,
261     Reference<XComponent> xComponent,
262     Reference<lang::XMultiServiceFactory> & rFactory,
263     Reference<beans::XPropertySet> & rPropSet,
264     const sal_Char* pComponentName )
265 {
266     DBG_ASSERT(xOutputStream.is(), "I really need an output stream!");
267     DBG_ASSERT(xComponent.is(), "Need component!");
268     DBG_ASSERT(NULL != pComponentName, "Need component name!");
269 
270     // get component
271     Reference< io::XActiveDataSource > xSaxWriter(
272         rFactory->createInstance(
273             OUString::createFromAscii("com.sun.star.xml.sax.Writer") ),
274         UNO_QUERY );
275     DBG_ASSERT( xSaxWriter.is(), "can't instantiate XML writer" );
276     if (!xSaxWriter.is())
277         return sal_False;
278 
279     // connect XML writer to output stream
280     xSaxWriter->setOutputStream( xOutputStream );
281 
282     // prepare arguments (prepend doc handler to given arguments)
283     Reference<xml::sax::XDocumentHandler> xDocHandler( xSaxWriter,UNO_QUERY);
284 
285     Sequence<Any> aArgs( 2 );
286     aArgs[0] <<= xDocHandler;
287     aArgs[1] <<= rPropSet;
288 
289     // get filter component
290     Reference< document::XExporter > xExporter(
291         rFactory->createInstanceWithArguments(
292             OUString::createFromAscii(pComponentName), aArgs), UNO_QUERY);
293     DBG_ASSERT( xExporter.is(),
294             "can't instantiate export filter component" );
295     if ( !xExporter.is() )
296         return sal_False;
297 
298 
299     // connect model and filter
300     xExporter->setSourceDocument( xComponent );
301 
302     // filter!
303     Reference < XFilter > xFilter( xExporter, UNO_QUERY );
304     uno::Sequence< PropertyValue > aProps(0);
305     xFilter->filter( aProps );
306 
307     uno::Reference<lang::XUnoTunnel> xFilterTunnel;
308     xFilterTunnel = uno::Reference<lang::XUnoTunnel>
309         ( xFilter, uno::UNO_QUERY );
310     SmXMLExport *pFilter = reinterpret_cast< SmXMLExport * >(
311                 sal::static_int_cast< sal_uIntPtr >(
312                 xFilterTunnel->getSomething( SmXMLExport::getUnoTunnelId() )));
313     return pFilter ? pFilter->GetSuccess() : sal_True;
314 }
315 
316 
317 /// export through an XML exporter component (storage version)
WriteThroughComponent(const Reference<embed::XStorage> & xStorage,Reference<XComponent> xComponent,const sal_Char * pStreamName,Reference<lang::XMultiServiceFactory> & rFactory,Reference<beans::XPropertySet> & rPropSet,const sal_Char * pComponentName,sal_Bool bCompress)318 sal_Bool SmXMLExportWrapper::WriteThroughComponent(
319     const Reference < embed::XStorage >& xStorage,
320     Reference<XComponent> xComponent,
321     const sal_Char* pStreamName,
322     Reference<lang::XMultiServiceFactory> & rFactory,
323     Reference<beans::XPropertySet> & rPropSet,
324     const sal_Char* pComponentName,
325     sal_Bool bCompress
326     )
327 {
328     DBG_ASSERT(xStorage.is(), "Need storage!");
329     DBG_ASSERT(NULL != pStreamName, "Need stream name!");
330 
331     // open stream
332     Reference < io::XStream > xStream;
333     OUString sStreamName = OUString::createFromAscii(pStreamName);
334     try
335     {
336         xStream = xStorage->openStreamElement( sStreamName,
337             embed::ElementModes::READWRITE | embed::ElementModes::TRUNCATE );
338     }
339     catch ( uno::Exception& )
340     {
341         DBG_ERROR( "Can't create output stream in package!" );
342         return sal_False;
343     }
344 
345     String aPropName( String::CreateFromAscii( RTL_CONSTASCII_STRINGPARAM("MediaType") ) );
346     OUString aMime( RTL_CONSTASCII_USTRINGPARAM("text/xml") );
347     uno::Any aAny;
348     aAny <<= aMime;
349 
350     uno::Reference < beans::XPropertySet > xSet( xStream, uno::UNO_QUERY );
351     xSet->setPropertyValue( aPropName, aAny );
352 
353     if ( !bCompress )
354     {
355         aPropName = String::CreateFromAscii( RTL_CONSTASCII_STRINGPARAM("Compressed") );
356         sal_Bool bFalse = sal_False;
357         aAny.setValue( &bFalse, ::getBooleanCppuType() );
358         xSet->setPropertyValue( aPropName, aAny );
359     }
360 
361     // even plain stream must be encrypted in encrypted document
362     OUString aTmpPropName( RTL_CONSTASCII_USTRINGPARAM("UseCommonStoragePasswordEncryption") );
363     sal_Bool bTrue = sal_True;
364     aAny.setValue( &bTrue, ::getBooleanCppuType() );
365     xSet->setPropertyValue( aTmpPropName, aAny );
366 
367     // set Base URL
368     if ( rPropSet.is() )
369     {
370         OUString sPropName( RTL_CONSTASCII_USTRINGPARAM("StreamName") );
371         rPropSet->setPropertyValue( sPropName, makeAny( sStreamName ) );
372     }
373 
374     // write the stuff
375     sal_Bool bRet = WriteThroughComponent( xStream->getOutputStream(), xComponent, rFactory,
376         rPropSet, pComponentName );
377 
378     // stream is closed by SAX parser
379     //if ( bRet )
380     //    xStream->getOutputStream()->closeOutput();
381 
382     return bRet;
383 }
384 
385 ////////////////////////////////////////////////////////////
386 
387 // #110680#
SmXMLExport(const::com::sun::star::uno::Reference<::com::sun::star::lang::XMultiServiceFactory> xServiceFactory,sal_uInt16 nExportFlags)388 SmXMLExport::SmXMLExport(
389     const ::com::sun::star::uno::Reference< ::com::sun::star::lang::XMultiServiceFactory > xServiceFactory,
390     sal_uInt16 nExportFlags)
391 :   SvXMLExport( xServiceFactory, MAP_INCH, XML_MATH, nExportFlags ) ,
392     pTree(0) ,
393     bSuccess(sal_False)
394 {
395 }
396 
getSomething(const uno::Sequence<sal_Int8> & rId)397 sal_Int64 SAL_CALL SmXMLExport::getSomething(
398     const uno::Sequence< sal_Int8 >& rId )
399 throw(uno::RuntimeException)
400 {
401     if ( rId.getLength() == 16 &&
402         0 == rtl_compareMemory( getUnoTunnelId().getConstArray(),
403         rId.getConstArray(), 16 ) )
404     return sal::static_int_cast< sal_Int64 >(reinterpret_cast< sal_uIntPtr >(this));
405 
406     return SvXMLExport::getSomething( rId );
407 }
408 
getUnoTunnelId()409 const uno::Sequence< sal_Int8 > & SmXMLExport::getUnoTunnelId() throw()
410 {
411     static uno::Sequence< sal_Int8 > * pSeq = 0;
412     if ( !pSeq )
413     {
414         osl::Guard< osl::Mutex > aGuard( osl::Mutex::getGlobalMutex() );
415         if ( !pSeq )
416         {
417             static uno::Sequence< sal_Int8 > aSeq( 16 );
418             rtl_createUuid( (sal_uInt8*)aSeq.getArray(), 0, sal_True );
419             pSeq = &aSeq;
420         }
421     }
422     return *pSeq;
423 }
424 
SmXMLExport_getImplementationName()425 OUString SAL_CALL SmXMLExport_getImplementationName() throw()
426 {
427     return OUString( RTL_CONSTASCII_USTRINGPARAM( "com.sun.star.comp.Math.XMLExporter" ) );
428 }
429 
SmXMLExport_getSupportedServiceNames()430 uno::Sequence< OUString > SAL_CALL SmXMLExport_getSupportedServiceNames()
431         throw()
432 {
433     const OUString aServiceName( EXPORT_SVC_NAME );
434     const uno::Sequence< OUString > aSeq( &aServiceName, 1 );
435     return aSeq;
436 }
437 
SmXMLExport_createInstance(const uno::Reference<lang::XMultiServiceFactory> & rSMgr)438 uno::Reference< uno::XInterface > SAL_CALL SmXMLExport_createInstance(
439     const uno::Reference< lang::XMultiServiceFactory > & rSMgr)
440     throw( uno::Exception )
441 {
442     // #110680#
443     // return (cppu::OWeakObject*)new SmXMLExport( EXPORT_ALL );
444     // EXPORT_OASIS is required here allthough there is no differrence between
445     // OOo and OASIS, because without the flag, a transformation to OOo would
446     // be chained in.
447     return (cppu::OWeakObject*)new SmXMLExport( rSMgr, EXPORT_OASIS|EXPORT_ALL );
448 }
449 
450 ////////////////////////////////////////////////////////////
451 
SmXMLExportMetaOOO_getImplementationName()452 OUString SAL_CALL SmXMLExportMetaOOO_getImplementationName() throw()
453 {
454     return OUString( RTL_CONSTASCII_USTRINGPARAM( "com.sun.star.comp.Math.XMLMetaExporter" ) );
455 }
456 
SmXMLExportMetaOOO_getSupportedServiceNames()457 uno::Sequence< OUString > SAL_CALL SmXMLExportMetaOOO_getSupportedServiceNames()
458     throw()
459 {
460     const OUString aServiceName( EXPORT_SVC_NAME );
461     const uno::Sequence< OUString > aSeq( &aServiceName, 1 );
462     return aSeq;
463 }
464 
SmXMLExportMetaOOO_createInstance(const uno::Reference<lang::XMultiServiceFactory> & rSMgr)465 uno::Reference< uno::XInterface > SAL_CALL SmXMLExportMetaOOO_createInstance(
466     const uno::Reference< lang::XMultiServiceFactory > & rSMgr)
467 throw( uno::Exception )
468 {
469     // #110680#
470     // return (cppu::OWeakObject*)new SmXMLExport( EXPORT_META );
471     return (cppu::OWeakObject*)new SmXMLExport( rSMgr, EXPORT_META );
472 }
473 
474 ////////////////////////////////////////////////////////////
475 
SmXMLExportMeta_getImplementationName()476 OUString SAL_CALL SmXMLExportMeta_getImplementationName() throw()
477 {
478     return OUString( RTL_CONSTASCII_USTRINGPARAM( "com.sun.star.comp.Math.XMLOasisMetaExporter" ) );
479 }
480 
SmXMLExportMeta_getSupportedServiceNames()481 uno::Sequence< OUString > SAL_CALL SmXMLExportMeta_getSupportedServiceNames()
482 throw()
483 {
484     const OUString aServiceName( EXPORT_SVC_NAME );
485     const uno::Sequence< OUString > aSeq( &aServiceName, 1 );
486     return aSeq;
487 }
488 
SmXMLExportMeta_createInstance(const uno::Reference<lang::XMultiServiceFactory> & rSMgr)489 uno::Reference< uno::XInterface > SAL_CALL SmXMLExportMeta_createInstance(
490     const uno::Reference< lang::XMultiServiceFactory > & rSMgr)
491 throw( uno::Exception )
492 {
493     // #110680#
494     // return (cppu::OWeakObject*)new SmXMLExport( EXPORT_META );
495     return (cppu::OWeakObject*)new SmXMLExport( rSMgr, EXPORT_OASIS|EXPORT_META );
496 }
497 
498 ////////////////////////////////////////////////////////////
499 
SmXMLExportSettingsOOO_getImplementationName()500 OUString SAL_CALL SmXMLExportSettingsOOO_getImplementationName() throw()
501 {
502     return OUString( RTL_CONSTASCII_USTRINGPARAM( "com.sun.star.comp.Math.XMLSettingsExporter" ) );
503 }
504 
SmXMLExportSettingsOOO_getSupportedServiceNames()505 uno::Sequence< OUString > SAL_CALL SmXMLExportSettingsOOO_getSupportedServiceNames()
506 throw()
507 {
508     const OUString aServiceName( EXPORT_SVC_NAME );
509     const uno::Sequence< OUString > aSeq( &aServiceName, 1 );
510     return aSeq;
511 }
512 
SmXMLExportSettingsOOO_createInstance(const uno::Reference<lang::XMultiServiceFactory> & rSMgr)513 uno::Reference< uno::XInterface > SAL_CALL SmXMLExportSettingsOOO_createInstance(
514     const uno::Reference< lang::XMultiServiceFactory > & rSMgr)
515 throw( uno::Exception )
516 {
517     // #110680#
518     // return (cppu::OWeakObject*)new SmXMLExport( EXPORT_SETTINGS );
519     return (cppu::OWeakObject*)new SmXMLExport( rSMgr, EXPORT_SETTINGS );
520 }
521 
522 ////////////////////////////////////////////////////////////
523 
SmXMLExportSettings_getImplementationName()524 OUString SAL_CALL SmXMLExportSettings_getImplementationName() throw()
525 {
526     return OUString( RTL_CONSTASCII_USTRINGPARAM( "com.sun.star.comp.Math.XMLOasisSettingsExporter" ) );
527 }
528 
SmXMLExportSettings_getSupportedServiceNames()529 uno::Sequence< OUString > SAL_CALL SmXMLExportSettings_getSupportedServiceNames()
530 throw()
531 {
532     const OUString aServiceName( EXPORT_SVC_NAME );
533     const uno::Sequence< OUString > aSeq( &aServiceName, 1 );
534     return aSeq;
535 }
536 
SmXMLExportSettings_createInstance(const uno::Reference<lang::XMultiServiceFactory> & rSMgr)537 uno::Reference< uno::XInterface > SAL_CALL SmXMLExportSettings_createInstance(
538     const uno::Reference< lang::XMultiServiceFactory > & rSMgr)
539 throw( uno::Exception )
540 {
541     // #110680#
542     // return (cppu::OWeakObject*)new SmXMLExport( EXPORT_SETTINGS );
543     return (cppu::OWeakObject*)new SmXMLExport( rSMgr, EXPORT_OASIS|EXPORT_SETTINGS );
544 }
545 
546 ////////////////////////////////////////////////////////////
547 
SmXMLExportContent_getImplementationName()548 OUString SAL_CALL SmXMLExportContent_getImplementationName() throw()
549 {
550     return OUString( RTL_CONSTASCII_USTRINGPARAM( "com.sun.star.comp.Math.XMLContentExporter" ) );
551 }
552 
SmXMLExportContent_getSupportedServiceNames()553 uno::Sequence< OUString > SAL_CALL SmXMLExportContent_getSupportedServiceNames()
554         throw()
555 {
556     const OUString aServiceName( EXPORT_SVC_NAME );
557     const uno::Sequence< OUString > aSeq( &aServiceName, 1 );
558     return aSeq;
559 }
560 
SmXMLExportContent_createInstance(const uno::Reference<lang::XMultiServiceFactory> & rSMgr)561 uno::Reference< uno::XInterface > SAL_CALL SmXMLExportContent_createInstance(
562     const uno::Reference< lang::XMultiServiceFactory > & rSMgr)
563 throw( uno::Exception )
564 {
565     // #110680#
566     // return (cppu::OWeakObject*)new SmXMLExport( EXPORT_CONTENT );
567     // The EXPORT_OASIS flag is only required to avoid that a transformer is
568     // chanied in
569     return (cppu::OWeakObject*)new SmXMLExport( rSMgr, EXPORT_OASIS|EXPORT_CONTENT );
570 }
571 
572 ////////////////////////////////////////////////////////////
573 
574 // XServiceInfo
575 // override empty method from parent class
getImplementationName()576 rtl::OUString SAL_CALL SmXMLExport::getImplementationName()
577 throw(uno::RuntimeException)
578 {
579     OUString aTxt;
580     switch( getExportFlags() )
581     {
582         case EXPORT_META:
583             aTxt = SmXMLExportMeta_getImplementationName();
584             break;
585         case EXPORT_SETTINGS:
586             aTxt = SmXMLExportSettings_getImplementationName();
587             break;
588         case EXPORT_CONTENT:
589             aTxt = SmXMLExportContent_getImplementationName();
590             break;
591         case EXPORT_ALL:
592         default:
593             aTxt = SmXMLExport_getImplementationName();
594             break;
595     }
596     return aTxt;
597 }
598 
exportDoc(enum XMLTokenEnum eClass)599 sal_uInt32 SmXMLExport::exportDoc(enum XMLTokenEnum eClass)
600 {
601     if ( (getExportFlags() & EXPORT_CONTENT) == 0 )
602     {
603         SvXMLExport::exportDoc( eClass );
604     }
605     else
606     {
607         uno::Reference <frame::XModel> xModel = GetModel();
608         uno::Reference <lang::XUnoTunnel> xTunnel;
609         xTunnel = uno::Reference <lang::XUnoTunnel> (xModel,uno::UNO_QUERY);
610         SmModel *pModel = reinterpret_cast<SmModel *>
611             (xTunnel->getSomething(SmModel::getUnoTunnelId()));
612 
613         if (pModel)
614         {
615             SmDocShell *pDocShell =
616                 static_cast<SmDocShell*>(pModel->GetObjectShell());
617             pTree = pDocShell->GetFormulaTree();
618             aText = pDocShell->GetText();
619         }
620 
621         GetDocHandler()->startDocument();
622 
623         /*Add xmlns line*/
624         SvXMLAttributeList &rList = GetAttrList();
625 
626         // make use of a default namespace
627         ResetNamespaceMap();    // Math doesn't need namespaces from xmloff, since it now uses default namespaces (because that is common with current MathML usage in the web)
628         _GetNamespaceMap().Add( OUString::createFromAscii(""), GetXMLToken(XML_N_MATH), XML_NAMESPACE_MATH );
629 
630         rList.AddAttribute(GetNamespaceMap().GetAttrNameByKey(XML_NAMESPACE_MATH_IDX),
631                 GetNamespaceMap().GetNameByKey( XML_NAMESPACE_MATH_IDX));
632 
633         //I think we need something like ImplExportEntities();
634         _ExportContent();
635         GetDocHandler()->endDocument();
636     }
637 
638     bSuccess=sal_True;
639     return 0;
640 }
641 
_ExportContent()642 void SmXMLExport::_ExportContent()
643 {
644     SvXMLElementExport aEquation(*this, XML_NAMESPACE_MATH, XML_MATH, sal_True, sal_True);
645     SvXMLElementExport *pSemantics=0;
646 
647     if (aText.Len())
648     {
649         pSemantics = new SvXMLElementExport(*this, XML_NAMESPACE_MATH,
650             XML_SEMANTICS, sal_True, sal_True);
651     }
652 
653     ExportNodes(pTree, 0);
654 
655     if (aText.Len())
656     {
657         // Convert symbol names
658         uno::Reference <frame::XModel> xModel = GetModel();
659         uno::Reference <lang::XUnoTunnel> xTunnel;
660         xTunnel = uno::Reference <lang::XUnoTunnel> (xModel,uno::UNO_QUERY);
661         SmModel *pModel = reinterpret_cast<SmModel *>
662             (xTunnel->getSomething(SmModel::getUnoTunnelId()));
663         SmDocShell *pDocShell = pModel ?
664             static_cast<SmDocShell*>(pModel->GetObjectShell()) : 0;
665         DBG_ASSERT( pDocShell, "doc shell missing" );
666         if (pDocShell)
667         {
668             SmParser &rParser = pDocShell->GetParser();
669             sal_Bool bVal = rParser.IsExportSymbolNames();
670             rParser.SetExportSymbolNames( sal_True );
671             SmNode *pTmpTree = rParser.Parse( aText );
672             aText = rParser.GetText();
673             delete pTmpTree;
674             rParser.SetExportSymbolNames( bVal );
675         }
676 
677         AddAttribute(XML_NAMESPACE_MATH, XML_ENCODING,
678             OUString(RTL_CONSTASCII_USTRINGPARAM("StarMath 5.0")));
679         SvXMLElementExport aAnnotation(*this, XML_NAMESPACE_MATH,
680             XML_ANNOTATION, sal_True, sal_False);
681         GetDocHandler()->characters(OUString( aText ));
682     }
683     delete pSemantics;
684 }
685 
GetViewSettings(Sequence<PropertyValue> & aProps)686 void SmXMLExport::GetViewSettings( Sequence < PropertyValue >& aProps)
687 {
688     uno::Reference <frame::XModel> xModel = GetModel();
689     if ( !xModel.is() )
690         return;
691 
692     uno::Reference <lang::XUnoTunnel> xTunnel;
693     xTunnel = uno::Reference <lang::XUnoTunnel> (xModel,uno::UNO_QUERY);
694     SmModel *pModel = reinterpret_cast<SmModel *>
695         (xTunnel->getSomething(SmModel::getUnoTunnelId()));
696 
697     if ( !pModel )
698         return;
699 
700     SmDocShell *pDocShell =
701         static_cast<SmDocShell*>(pModel->GetObjectShell());
702     if ( !pDocShell )
703         return;
704 
705     aProps.realloc( 4 );
706     PropertyValue *pValue = aProps.getArray();
707     sal_Int32 nIndex = 0;
708 
709     Rectangle aRect( pDocShell->GetVisArea() );
710 
711     pValue[nIndex].Name = OUString( RTL_CONSTASCII_USTRINGPARAM ( "ViewAreaTop") );
712     pValue[nIndex++].Value <<= aRect.Top();
713 
714     pValue[nIndex].Name = OUString( RTL_CONSTASCII_USTRINGPARAM ( "ViewAreaLeft") );
715     pValue[nIndex++].Value <<= aRect.Left();
716 
717     pValue[nIndex].Name = OUString( RTL_CONSTASCII_USTRINGPARAM ( "ViewAreaWidth") );
718     pValue[nIndex++].Value <<= aRect.GetWidth();
719 
720     pValue[nIndex].Name = OUString( RTL_CONSTASCII_USTRINGPARAM ( "ViewAreaHeight") );
721     pValue[nIndex++].Value <<= aRect.GetHeight();
722 }
723 
GetConfigurationSettings(Sequence<PropertyValue> & rProps)724 void SmXMLExport::GetConfigurationSettings( Sequence < PropertyValue > & rProps)
725 {
726     Reference < XPropertySet > xProps ( GetModel(), UNO_QUERY );
727     if ( xProps.is() )
728     {
729         Reference< XPropertySetInfo > xPropertySetInfo = xProps->getPropertySetInfo();
730         if (xPropertySetInfo.is())
731         {
732             Sequence< Property > aProps = xPropertySetInfo->getProperties();
733             sal_Int32 nCount(aProps.getLength());
734             if (nCount > 0)
735             {
736                 rProps.realloc(nCount);
737                 PropertyValue* pProps = rProps.getArray();
738                 if (pProps)
739                 {
740                     SmConfig *pConfig = SM_MOD()->GetConfig();
741                     const bool bUsedSymbolsOnly = pConfig ? pConfig->IsSaveOnlyUsedSymbols() : false;
742 
743                     const OUString sFormula ( RTL_CONSTASCII_USTRINGPARAM ( "Formula" ) );
744                     const OUString sBasicLibraries ( RTL_CONSTASCII_USTRINGPARAM ( "BasicLibraries" ) );
745                     const OUString sDialogLibraries ( RTL_CONSTASCII_USTRINGPARAM ( "DialogLibraries" ) );
746                     const OUString sRuntimeUID ( RTL_CONSTASCII_USTRINGPARAM ( "RuntimeUID" ) );
747                     for (sal_Int32 i = 0; i < nCount; i++, pProps++)
748                     {
749                         const OUString &rPropName = aProps[i].Name;
750                         if (rPropName != sFormula &&
751                             rPropName != sBasicLibraries &&
752                             rPropName != sDialogLibraries &&
753                             rPropName != sRuntimeUID)
754                         {
755                             pProps->Name = rPropName;
756 
757                             rtl::OUString aActualName( rPropName );
758 
759                             // handle 'save used symbols only'
760                             if (bUsedSymbolsOnly && rPropName.equalsAscii("Symbols"))
761                                 aActualName = OUString( RTL_CONSTASCII_USTRINGPARAM ( "UserDefinedSymbolsInUse" ) );
762 
763                             pProps->Value = xProps->getPropertyValue( aActualName );
764                         }
765                     }
766                 }
767             }
768         }
769     }
770 }
771 
ExportLine(const SmNode * pNode,int nLevel)772 void SmXMLExport::ExportLine(const SmNode *pNode, int nLevel)
773 {
774     ExportExpression(pNode, nLevel);
775 }
776 
ExportBinaryHorizontal(const SmNode * pNode,int nLevel)777 void SmXMLExport::ExportBinaryHorizontal(const SmNode *pNode, int nLevel)
778 {
779     ExportExpression(pNode, nLevel);
780 }
781 
ExportUnaryHorizontal(const SmNode * pNode,int nLevel)782 void SmXMLExport::ExportUnaryHorizontal(const SmNode *pNode, int nLevel)
783 {
784     ExportExpression(pNode, nLevel);
785 }
786 
ExportExpression(const SmNode * pNode,int nLevel)787 void SmXMLExport::ExportExpression(const SmNode *pNode, int nLevel)
788 {
789     SvXMLElementExport *pRow=0;
790     sal_uLong  nSize = pNode->GetNumSubNodes();
791 
792     // #i115443: nodes of type expression always need to be grouped with mrow statement
793     if (nSize > 1 || (pNode && pNode->GetType() == NEXPRESSION))
794         pRow = new SvXMLElementExport(*this, XML_NAMESPACE_MATH, XML_MROW, sal_True, sal_True);
795 
796     //if (nSize)
797     //{
798         for (sal_uInt16 i = 0; i < nSize; i++)
799             if (const SmNode *pTemp = pNode->GetSubNode(i))
800                 ExportNodes(pTemp, nLevel+1);
801     //}
802 #if 0
803     else
804     {
805         //This saves us from situations like "a newline" where the
806         //lack of a term following the newline would otherwise create
807         //a incorrect token like <mtr/>
808         SvXMLElementExport aDummy(*this, XML_NAMESPACE_MATH, XML_MI, sal_True, sal_False);
809         sal_Unicode nArse[2] = {'\n','\0'};
810         GetDocHandler()->characters(nArse);
811     }
812 #endif
813 
814     delete pRow;
815 }
816 
ExportBinaryVertical(const SmNode * pNode,int nLevel)817 void SmXMLExport::ExportBinaryVertical(const SmNode *pNode, int nLevel)
818 {
819     DBG_ASSERT(pNode->GetNumSubNodes()==3,"Bad Fraction");
820     SvXMLElementExport aFraction(*this, XML_NAMESPACE_MATH, XML_MFRAC, sal_True, sal_True);
821     ExportNodes(pNode->GetSubNode(0), nLevel);
822     ExportNodes(pNode->GetSubNode(2), nLevel);
823 }
824 
ExportTable(const SmNode * pNode,int nLevel)825 void SmXMLExport::ExportTable(const SmNode *pNode, int nLevel)
826 {
827     SvXMLElementExport *pTable=0;
828 
829     sal_uInt16 nSize = pNode->GetNumSubNodes();
830 
831     //If the list ends in newline then the last entry has
832     //no subnodes, the newline is superfulous so we just drop
833     //the last node, inclusion would create a bad MathML
834     //table
835     if (nSize >= 1 && pNode->GetSubNode(nSize-1)->GetNumSubNodes() == 0)
836         --nSize;
837 
838     // try to avoid creating a mtable element when the formula consists only
839     // of a single output line
840     if (nLevel || (nSize >1))
841         pTable = new SvXMLElementExport(*this, XML_NAMESPACE_MATH, XML_MTABLE, sal_True, sal_True);
842 
843     for (sal_uInt16 i = 0; i < nSize; i++)
844         if (const SmNode *pTemp = pNode->GetSubNode(i))
845         {
846             SvXMLElementExport *pRow=0;
847             SvXMLElementExport *pCell=0;
848             if (pTable)
849             {
850                 pRow  = new SvXMLElementExport(*this, XML_NAMESPACE_MATH, XML_MTR, sal_True, sal_True);
851                 pCell = new SvXMLElementExport(*this, XML_NAMESPACE_MATH, XML_MTD, sal_True, sal_True);
852             }
853             ExportNodes(pTemp, nLevel+1);
854             delete pCell;
855             delete pRow;
856         }
857 
858     delete pTable;
859 }
860 
ExportMath(const SmNode * pNode,int)861 void SmXMLExport::ExportMath(const SmNode *pNode, int /*nLevel*/)
862 {
863     const SmMathSymbolNode *pTemp = static_cast<const SmMathSymbolNode *>(pNode);
864     SvXMLElementExport aMath(*this, XML_NAMESPACE_MATH, XML_MO, sal_True, sal_False);
865     sal_Unicode nArse[2];
866     nArse[0] = pTemp->GetText().GetChar(0);
867     sal_Unicode cTmp = ConvertMathToMathML( nArse[0] );
868     if (cTmp != 0)
869         nArse[0] = cTmp;
870     DBG_ASSERT(nArse[0] != 0xffff,"Non existant symbol");
871     nArse[1] = 0;
872     GetDocHandler()->characters(nArse);
873 }
874 
ExportText(const SmNode * pNode,int)875 void SmXMLExport::ExportText(const SmNode *pNode, int /*nLevel*/)
876 {
877     SvXMLElementExport *pText;
878     const SmTextNode *pTemp = static_cast<const SmTextNode *>(pNode);
879     switch (pNode->GetToken().eType)
880     {
881         default:
882         case TIDENT:
883         {
884             //Note that we change the fontstyle to italic for strings that
885             //are italic and longer than a single character.
886             sal_Bool bIsItalic = IsItalic( pTemp->GetFont() );
887             if ((pTemp->GetText().Len() > 1) && bIsItalic)
888                 AddAttribute(XML_NAMESPACE_MATH, XML_MATHVARIANT, XML_ITALIC);
889             else if ((pTemp->GetText().Len() == 1) && !bIsItalic)
890                 AddAttribute(XML_NAMESPACE_MATH, XML_MATHVARIANT, XML_NORMAL);
891             pText = new SvXMLElementExport(*this, XML_NAMESPACE_MATH, XML_MI,sal_True,sal_False);
892             break;
893         }
894         case TNUMBER:
895             pText = new SvXMLElementExport(*this, XML_NAMESPACE_MATH, XML_MN,sal_True,sal_False);
896             break;
897         case TTEXT:
898             pText = new SvXMLElementExport(*this, XML_NAMESPACE_MATH, XML_MTEXT,sal_True,sal_False);
899             break;
900         }
901     GetDocHandler()->characters(OUString(pTemp->GetText().GetBuffer()));
902     delete pText;
903 }
904 
ExportBlank(const SmNode *,int)905 void SmXMLExport::ExportBlank(const SmNode * /*pNode*/, int /*nLevel*/)
906 {
907     //!! exports an empty <mi> tag since for example "~_~" is allowed in
908     //!! Math (so it has no sense at all) but must not result in an empty
909     //!! <msub> tag in MathML !!
910 
911     SvXMLElementExport *pText;
912     //const SmBlankNode *pTemp = static_cast<const SmBlankNode *>(pNode);
913 
914     pText = new SvXMLElementExport(*this, XML_NAMESPACE_MATH, XML_MI, sal_True, sal_False);
915 
916     GetDocHandler()->characters( OUString() );
917     delete pText;
918 }
919 
ExportSubSupScript(const SmNode * pNode,int nLevel)920 void SmXMLExport::ExportSubSupScript(const SmNode *pNode, int nLevel)
921 {
922     const SmNode *pSub  = 0;
923     const SmNode *pSup  = 0;
924     const SmNode *pCSub = 0;
925     const SmNode *pCSup = 0;
926     const SmNode *pLSub = 0;
927     const SmNode *pLSup = 0;
928     SvXMLElementExport *pThing = 0, *pThing2 = 0;
929 
930     //if we have prescripts at all then we must use the tensor notation
931 
932     //This is one of those excellent locations where scope is vital to
933     //arrange the construction and destruction of the element helper
934     //classes correctly
935     pLSub = pNode->GetSubNode(LSUB+1);
936     pLSup = pNode->GetSubNode(LSUP+1);
937     if (pLSub || pLSup)
938     {
939         SvXMLElementExport aMultiScripts(*this, XML_NAMESPACE_MATH,
940             XML_MMULTISCRIPTS, sal_True, sal_True);
941 
942 
943         if (NULL != (pCSub = pNode->GetSubNode(CSUB+1))
944             && NULL != (pCSup = pNode->GetSubNode(CSUP+1)))
945         {
946             pThing2 = new SvXMLElementExport(*this, XML_NAMESPACE_MATH,
947                 XML_MUNDEROVER, sal_True,sal_True);
948         }
949         else if (NULL != (pCSub = pNode->GetSubNode(CSUB+1)))
950         {
951             pThing2 = new SvXMLElementExport(*this, XML_NAMESPACE_MATH,
952                 XML_MUNDER, sal_True,sal_True);
953         }
954         else if (NULL != (pCSup = pNode->GetSubNode(CSUP+1)))
955         {
956             pThing2 = new SvXMLElementExport(*this, XML_NAMESPACE_MATH,
957                 XML_MOVER, sal_True,sal_True);
958         }
959 
960         ExportNodes(pNode->GetSubNode(0), nLevel+1);    //Main Term
961 
962         if (pCSub)
963             ExportNodes(pCSub, nLevel+1);
964         if (pCSup)
965             ExportNodes(pCSup, nLevel+1);
966         delete pThing2;
967 
968         pSub = pNode->GetSubNode(RSUB+1);
969         pSup = pNode->GetSubNode(RSUP+1);
970         if (pSub || pSup)
971         {
972             if (pSub)
973                 ExportNodes(pSub, nLevel+1);
974             else
975             {
976                 SvXMLElementExport aNone(*this, XML_NAMESPACE_MATH, XML_NONE,sal_True,sal_True);
977             }
978             if (pSup)
979                 ExportNodes(pSup, nLevel+1);
980             else
981             {
982                 SvXMLElementExport aNone(*this, XML_NAMESPACE_MATH, XML_NONE,sal_True,sal_True);
983             }
984         }
985 
986         //Seperator element between suffix and prefix sub/sup pairs
987         {
988             SvXMLElementExport aPrescripts(*this, XML_NAMESPACE_MATH,
989                 XML_MPRESCRIPTS, sal_True,sal_True);
990         }
991 
992         if (pLSub)
993             ExportNodes(pLSub, nLevel+1);
994         else
995         {
996             SvXMLElementExport aNone(*this, XML_NAMESPACE_MATH, XML_NONE,
997                 sal_True,sal_True);
998 
999         }
1000         if (pLSup)
1001             ExportNodes(pLSup, nLevel+1);
1002         else
1003         {
1004             SvXMLElementExport aNone(*this, XML_NAMESPACE_MATH, XML_NONE,
1005                 sal_True,sal_True);
1006 
1007         }
1008     }
1009     else
1010     {
1011         if (NULL != (pSub = pNode->GetSubNode(RSUB+1)) &&
1012             NULL != (pSup = pNode->GetSubNode(RSUP+1)))
1013         {
1014             pThing = new SvXMLElementExport(*this, XML_NAMESPACE_MATH,
1015                 XML_MSUBSUP, sal_True,sal_True);
1016         }
1017         else if (NULL != (pSub = pNode->GetSubNode(RSUB+1)))
1018         {
1019             pThing = new SvXMLElementExport(*this, XML_NAMESPACE_MATH, XML_MSUB,
1020                 sal_True,sal_True);
1021         }
1022         else if (NULL != (pSup = pNode->GetSubNode(RSUP+1)))
1023         {
1024             pThing = new SvXMLElementExport(*this, XML_NAMESPACE_MATH, XML_MSUP,
1025                 sal_True,sal_True);
1026         }
1027 
1028         if (NULL != (pCSub = pNode->GetSubNode(CSUB+1))
1029             && NULL != (pCSup=pNode->GetSubNode(CSUP+1)))
1030         {
1031             pThing2 = new SvXMLElementExport(*this, XML_NAMESPACE_MATH,
1032                 XML_MUNDEROVER, sal_True,sal_True);
1033         }
1034         else if (NULL != (pCSub = pNode->GetSubNode(CSUB+1)))
1035         {
1036             pThing2 = new SvXMLElementExport(*this, XML_NAMESPACE_MATH,
1037                 XML_MUNDER, sal_True,sal_True);
1038         }
1039         else if (NULL != (pCSup = pNode->GetSubNode(CSUP+1)))
1040         {
1041             pThing2 = new SvXMLElementExport(*this, XML_NAMESPACE_MATH,
1042                 XML_MOVER, sal_True,sal_True);
1043         }
1044         ExportNodes(pNode->GetSubNode(0), nLevel+1);    //Main Term
1045 
1046         if (pCSub)
1047             ExportNodes(pCSub, nLevel+1);
1048         if (pCSup)
1049             ExportNodes(pCSup, nLevel+1);
1050         delete pThing2;
1051 
1052         if (pSub)
1053             ExportNodes(pSub, nLevel+1);
1054         if (pSup)
1055             ExportNodes(pSup, nLevel+1);
1056         delete pThing;
1057     }
1058 }
1059 
ExportBrace(const SmNode * pNode,int nLevel)1060 void SmXMLExport::ExportBrace(const SmNode *pNode, int nLevel)
1061 {
1062     const SmNode *pTemp;
1063     const SmNode *pLeft=pNode->GetSubNode(0);
1064     const SmNode *pRight=pNode->GetSubNode(2);
1065     SvXMLElementExport *pFences=0,*pRow=0;
1066     if ( ((pLeft) && (pLeft->GetToken().eType != TNONE)) &&
1067         ((pRight) && (pRight->GetToken().eType != TNONE)) &&
1068         (pNode->GetScaleMode() == SCALE_HEIGHT))
1069     {
1070         sal_Unicode nArse[2];
1071         nArse[1] = 0;
1072         nArse[0] = static_cast<
1073             const SmMathSymbolNode* >(pLeft)->GetText().GetChar(0);
1074         DBG_ASSERT(nArse[0] != 0xffff,"Non existant symbol");
1075         AddAttribute(XML_NAMESPACE_MATH, XML_OPEN,nArse);
1076         nArse[0] = static_cast<
1077             const SmMathSymbolNode* >(pRight)->GetText().GetChar(0);
1078         DBG_ASSERT(nArse[0] != 0xffff,"Non existant symbol");
1079         AddAttribute(XML_NAMESPACE_MATH, XML_CLOSE,nArse);
1080         pFences = new SvXMLElementExport(*this, XML_NAMESPACE_MATH, XML_MFENCED,
1081             sal_True,sal_True);
1082     }
1083     else if (pLeft && (pLeft->GetToken().eType != TNONE))
1084     {
1085         pRow = new SvXMLElementExport(*this, XML_NAMESPACE_MATH, XML_MROW,
1086             sal_True, sal_True);
1087         if (pNode->GetScaleMode() == SCALE_HEIGHT)
1088             AddAttribute(XML_NAMESPACE_MATH, XML_STRETCHY, XML_TRUE);
1089         else
1090             AddAttribute(XML_NAMESPACE_MATH, XML_STRETCHY, XML_FALSE);
1091         ExportNodes(pLeft, nLevel+1);
1092     }
1093     else
1094         pRow = new SvXMLElementExport(*this, XML_NAMESPACE_MATH, XML_MROW,
1095             sal_True, sal_True);
1096 
1097     if (NULL != (pTemp = pNode->GetSubNode(1)))
1098         ExportNodes(pTemp, nLevel+1);
1099     if (pFences)
1100         delete pFences;
1101     else if (pRight && (pRight->GetToken().eType != TNONE))
1102     {
1103         if (pNode->GetScaleMode() == SCALE_HEIGHT)
1104             AddAttribute(XML_NAMESPACE_MATH, XML_STRETCHY, XML_TRUE);
1105         else
1106             AddAttribute(XML_NAMESPACE_MATH, XML_STRETCHY, XML_FALSE);
1107         ExportNodes(pRight, nLevel+1);
1108     }
1109     delete pRow;
1110 }
1111 
ExportRoot(const SmNode * pNode,int nLevel)1112 void SmXMLExport::ExportRoot(const SmNode *pNode, int nLevel)
1113 {
1114     if (pNode->GetSubNode(0))
1115     {
1116         SvXMLElementExport aRoot(*this, XML_NAMESPACE_MATH, XML_MROOT,sal_True,
1117             sal_True);
1118         ExportNodes(pNode->GetSubNode(2), nLevel+1);
1119         ExportNodes(pNode->GetSubNode(0), nLevel+1);
1120     }
1121     else
1122     {
1123         SvXMLElementExport aSqrt(*this, XML_NAMESPACE_MATH, XML_MSQRT,sal_True,
1124             sal_True);
1125         ExportNodes(pNode->GetSubNode(2), nLevel+1);
1126     }
1127 }
1128 
ExportOperator(const SmNode * pNode,int nLevel)1129 void SmXMLExport::ExportOperator(const SmNode *pNode, int nLevel)
1130 {
1131     /*we need to either use content or font and size attributes
1132      *here*/
1133 #if 0
1134     {
1135     SvXMLElementExport aMath(*this, XML_NAMESPACE_MATH, XML_MO,
1136         sal_True,sal_False);
1137     SmTextNode *pTemp = (SmTextNode *)pNode->GetSubNode(0);
1138     GetDocHandler()->characters(pTemp->GetText());
1139     }
1140 #endif
1141     SvXMLElementExport aRow(*this, XML_NAMESPACE_MATH, XML_MROW,
1142         sal_True, sal_True);
1143     ExportNodes(pNode->GetSubNode(0), nLevel+1);
1144     ExportNodes(pNode->GetSubNode(1), nLevel+1);
1145 }
1146 
ExportAttributes(const SmNode * pNode,int nLevel)1147 void SmXMLExport::ExportAttributes(const SmNode *pNode, int nLevel)
1148 {
1149     SvXMLElementExport *pElement=0;
1150 
1151     if (pNode->GetToken().eType == TUNDERLINE)
1152     {
1153         AddAttribute(XML_NAMESPACE_MATH, XML_ACCENTUNDER,
1154             XML_TRUE);
1155         pElement = new SvXMLElementExport(*this, XML_NAMESPACE_MATH, XML_MUNDER,
1156             sal_True,sal_True);
1157     }
1158     else if (pNode->GetToken().eType != TOVERSTRIKE)
1159     {
1160         AddAttribute(XML_NAMESPACE_MATH, XML_ACCENT,
1161             XML_TRUE);
1162         pElement = new SvXMLElementExport(*this, XML_NAMESPACE_MATH, XML_MOVER,
1163             sal_True,sal_True);
1164     }
1165 
1166     ExportNodes(pNode->GetSubNode(1), nLevel+1);
1167     switch (pNode->GetToken().eType)
1168     {
1169         case TOVERLINE:
1170             {
1171             //proper entity support required
1172             SvXMLElementExport aMath(*this, XML_NAMESPACE_MATH, XML_MO,
1173                 sal_True,sal_True);
1174 #if 0
1175             GetDocHandler()->characters(
1176                 OUString(RTL_CONSTASCII_USTRINGPARAM("&overbar;")));
1177 #else
1178             sal_Unicode nArse[2] = {0xAF,0x00};
1179 #endif
1180             GetDocHandler()->characters(nArse);
1181             }
1182             break;
1183         case TUNDERLINE:
1184             {
1185             //proper entity support required
1186             SvXMLElementExport aMath(*this, XML_NAMESPACE_MATH, XML_MO,
1187                 sal_True,sal_True);
1188 #if 0
1189             GetDocHandler()->characters(
1190                 OUString(RTL_CONSTASCII_USTRINGPARAM("&underbar;")));
1191 #else
1192             sal_Unicode nArse[2] = {0x0332,0x00};
1193 #endif
1194             GetDocHandler()->characters(nArse);
1195             }
1196             break;
1197         case TOVERSTRIKE:
1198             break;
1199         default:
1200             ExportNodes(pNode->GetSubNode(0), nLevel+1);
1201             break;
1202     }
1203     delete pElement;
1204 }
1205 
lcl_HasEffectOnMathvariant(const SmTokenType eType)1206 static bool lcl_HasEffectOnMathvariant( const SmTokenType eType )
1207 {
1208     return  eType == TBOLD || eType == TNBOLD ||
1209             eType == TITALIC || eType == TNBOLD ||
1210             eType == TSANS || eType == TSERIF || eType == TFIXED;
1211 }
1212 
ExportFont(const SmNode * pNode,int nLevel)1213 void SmXMLExport::ExportFont(const SmNode *pNode, int nLevel)
1214 {
1215     SvXMLElementExport *pElement = 0;
1216 
1217     //
1218     // gather the mathvariant attribut relevant data from all
1219     // successively following SmFontNodes...
1220     //
1221     int nBold   = -1;   // for the following variables: -1 = yet undefined; 0 = false; 1 = true;
1222     int nItalic = -1;   // for the following variables: -1 = yet undefined; 0 = false; 1 = true;
1223     int nSansSerifFixed   = -1;
1224     SmTokenType eNodeType = TUNKNOWN;
1225     while (lcl_HasEffectOnMathvariant( (eNodeType = pNode->GetToken().eType) ))
1226     {
1227         switch (eNodeType)
1228         {
1229             case TBOLD      : nBold   = 1; break;
1230             case TNBOLD     : nBold   = 0; break;
1231             case TITALIC    : nItalic = 1; break;
1232             case TNITALIC   : nItalic = 0; break;
1233             case TSANS      : nSansSerifFixed  = 0; break;
1234             case TSERIF     : nSansSerifFixed  = 1; break;
1235             case TFIXED     : nSansSerifFixed  = 2; break;
1236             default:
1237                 DBG_ASSERT( 0, "unexpected case" );
1238         }
1239         // According to the parser every node that is to be evaluated heres
1240         // has a single non-zero subnode at index 1!! Thus we only need to check
1241         // that single node for follow-up nodes that have an effect on the attribute.
1242         if (pNode->GetNumSubNodes() > 1 && pNode->GetSubNode(1) &&
1243             lcl_HasEffectOnMathvariant( pNode->GetSubNode(1)->GetToken().eType))
1244         {
1245             pNode = pNode->GetSubNode(1);
1246         }
1247         else
1248             break;
1249     }
1250 
1251     switch (pNode->GetToken().eType)
1252     {
1253         //wrap a phantom element around everything*/
1254         case TPHANTOM:
1255             pElement = new SvXMLElementExport(*this, XML_NAMESPACE_MATH,
1256                 XML_MPHANTOM, sal_True,sal_True);
1257             break;
1258         case TBLACK:
1259             AddAttribute(XML_NAMESPACE_MATH, XML_COLOR, XML_BLACK);
1260             break;
1261         case TWHITE:
1262             AddAttribute(XML_NAMESPACE_MATH, XML_COLOR, XML_WHITE);
1263             break;
1264         case TRED:
1265             AddAttribute(XML_NAMESPACE_MATH, XML_COLOR, XML_RED);
1266             break;
1267         case TGREEN:
1268             AddAttribute(XML_NAMESPACE_MATH, XML_COLOR, XML_GREEN);
1269             break;
1270         case TBLUE:
1271             AddAttribute(XML_NAMESPACE_MATH, XML_COLOR, XML_BLUE);
1272             break;
1273         case TCYAN:
1274             AddAttribute(XML_NAMESPACE_MATH, XML_COLOR, XML_AQUA);
1275             break;
1276         case TMAGENTA:
1277             AddAttribute(XML_NAMESPACE_MATH, XML_COLOR, XML_FUCHSIA);
1278             break;
1279         case TYELLOW:
1280             AddAttribute(XML_NAMESPACE_MATH, XML_COLOR, XML_YELLOW);
1281             break;
1282         case TSIZE:
1283             {
1284                 const SmFontNode *pFontNode = static_cast<const SmFontNode *>(pNode);
1285                 const Fraction &aFrac = pFontNode->GetSizeParameter();
1286 
1287                 OUStringBuffer sStrBuf;
1288                 switch(pFontNode->GetSizeType())
1289                 {
1290                     case FNTSIZ_MULTIPLY:
1291                         SvXMLUnitConverter::convertDouble(sStrBuf,
1292                             static_cast<double>(aFrac*Fraction(100.00)));
1293                         sStrBuf.append(static_cast<sal_Unicode>('%'));
1294                         break;
1295                     case FNTSIZ_DIVIDE:
1296                         SvXMLUnitConverter::convertDouble(sStrBuf,
1297                             static_cast<double>(Fraction(100.00)/aFrac));
1298                         sStrBuf.append(static_cast<sal_Unicode>('%'));
1299                         break;
1300                     case FNTSIZ_ABSOLUT:
1301                         SvXMLUnitConverter::convertDouble(sStrBuf,
1302                             static_cast<double>(aFrac));
1303                         sStrBuf.append(
1304                             GetXMLToken(XML_UNIT_PT));
1305                         break;
1306                     default:
1307                         {
1308                             //The problem here is that the wheels fall off because
1309                             //font size is stored in 100th's of a mm not pts, and
1310                             //rounding errors take their toll on the original
1311                             //value specified in points.
1312 
1313                             //Must fix StarMath to retain the original pt values
1314                             Fraction aTemp = Sm100th_mmToPts(pFontNode->GetFont().
1315                                 GetSize().Height());
1316 
1317                             if (pFontNode->GetSizeType() == FNTSIZ_MINUS)
1318                                 aTemp-=aFrac;
1319                             else
1320                                 aTemp+=aFrac;
1321 
1322                             double mytest = static_cast<double>(aTemp);
1323 
1324                             mytest = ::rtl::math::round(mytest,1);
1325                             SvXMLUnitConverter::convertDouble(sStrBuf,mytest);
1326                             sStrBuf.append(GetXMLToken(XML_UNIT_PT));
1327                         }
1328                         break;
1329                 }
1330 
1331                 OUString sStr(sStrBuf.makeStringAndClear());
1332                 AddAttribute(XML_NAMESPACE_MATH, XML_MATHSIZE, sStr);
1333             }
1334             break;
1335         case TBOLD:
1336         case TITALIC:
1337         case TNBOLD:
1338         case TNITALIC:
1339         case TFIXED:
1340         case TSANS:
1341         case TSERIF:
1342             {
1343                 // nBold:   -1 = yet undefined; 0 = false; 1 = true;
1344                 // nItalic: -1 = yet undefined; 0 = false; 1 = true;
1345                 // nSansSerifFixed: -1 = undefined; 0 = sans; 1 = serif; 2 = fixed;
1346                 const sal_Char *pText = "normal";
1347                 if (nSansSerifFixed == -1 || nSansSerifFixed == 1)
1348                 {
1349                     pText = "normal";
1350                     if (nBold == 1 && nItalic != 1)
1351                         pText = "bold";
1352                     else if (nBold != 1 && nItalic == 1)
1353                         pText = "italic";
1354                     else if (nBold == 1 && nItalic == 1)
1355                         pText = "bold-italic";
1356                 }
1357                 else if (nSansSerifFixed == 0)
1358                 {
1359                     pText = "sans-serif";
1360                     if (nBold == 1 && nItalic != 1)
1361                         pText = "bold-sans-serif";
1362                     else if (nBold != 1 && nItalic == 1)
1363                         pText = "sans-serif-italic";
1364                     else if (nBold == 1 && nItalic == 1)
1365                         pText = "sans-serif-bold-italic";
1366                 }
1367                 else if (nSansSerifFixed == 2)
1368                     pText = "monospace";    // no modifiers allowed for monospace ...
1369                 else
1370                 {
1371                     DBG_ASSERT( 0, "unexpected case" );
1372                 }
1373                 AddAttribute(XML_NAMESPACE_MATH, XML_MATHVARIANT, A2OU(pText));
1374             }
1375             break;
1376         default:
1377             break;
1378 
1379     }
1380 #if 0
1381     if (pNode->GetNumSubNodes() > 1) //or in the future is a node that
1382                                      //cannot take the currently supported
1383                                      //properties
1384 #endif
1385     //for now we will just always export with a style and not worry about
1386     //anyone else for the moment.
1387     {
1388         //wrap a style around it
1389         SvXMLElementExport aStyle(*this, XML_NAMESPACE_MATH, XML_MSTYLE, sal_True,sal_True);
1390         ExportExpression(pNode, nLevel);
1391     }
1392 #if 0
1393     else
1394         ExportNodes(pNode->GetSubNode(0), nLevel+1);
1395 #endif
1396 
1397     delete pElement;
1398 }
1399 
1400 
ExportVerticalBrace(const SmNode * pNode,int nLevel)1401 void SmXMLExport::ExportVerticalBrace(const SmNode *pNode, int nLevel)
1402 {
1403     //Place the overbrace value OVER a vertical brace and then place that
1404     //expression OVER the overbrace value, If someone can find a
1405     //dedicated term in MathML to handle this overbrace/underbrace concept
1406     //let me know. C.
1407     XMLTokenEnum which;
1408 
1409     switch (pNode->GetToken().eType)
1410     {
1411         case TOVERBRACE:
1412         default:
1413             which = XML_MOVER;
1414             break;
1415         case TUNDERBRACE:
1416             which = XML_MUNDER;
1417             break;
1418     }
1419 
1420     DBG_ASSERT(pNode->GetNumSubNodes()==3,"Bad Vertical Brace");
1421     SvXMLElementExport aOver1(*this, XML_NAMESPACE_MATH,which, sal_True, sal_True);
1422     {//Scoping
1423         // using accents will draw the over-/underbraces too close to the base
1424         // see http://www.w3.org/TR/MathML2/chapter3.html#id.3.4.5.2
1425         // also XML_ACCENT is illegal with XML_MUNDER. Thus no XML_ACCENT attribut here!
1426 //        AddAttribute(XML_NAMESPACE_MATH, XML_ACCENT, XML_sal_True);
1427         SvXMLElementExport aOver2(*this, XML_NAMESPACE_MATH,which, sal_True, sal_True);
1428         ExportNodes(pNode->GetSubNode(0), nLevel);
1429         ExportNodes(pNode->GetSubNode(1), nLevel);
1430     }
1431     ExportNodes(pNode->GetSubNode(2), nLevel);
1432 }
1433 
ExportMatrix(const SmNode * pNode,int nLevel)1434 void SmXMLExport::ExportMatrix(const SmNode *pNode, int nLevel)
1435 {
1436     SvXMLElementExport aTable(*this, XML_NAMESPACE_MATH, XML_MTABLE, sal_True, sal_True);
1437     const SmMatrixNode *pMatrix = static_cast<const SmMatrixNode *>(pNode);
1438     sal_uInt16 i=0;
1439     for (sal_uLong y = 0; y < pMatrix->GetNumRows(); y++)
1440     {
1441         SvXMLElementExport aRow(*this, XML_NAMESPACE_MATH, XML_MTR, sal_True, sal_True);
1442         for (sal_uLong x = 0; x < pMatrix->GetNumCols(); x++)
1443             if (const SmNode *pTemp = pNode->GetSubNode(i++))
1444             {
1445                 SvXMLElementExport aCell(*this, XML_NAMESPACE_MATH, XML_MTD, sal_True, sal_True);
1446                 ExportNodes(pTemp, nLevel+1);
1447             }
1448     }
1449 }
1450 
ExportNodes(const SmNode * pNode,int nLevel)1451 void SmXMLExport::ExportNodes(const SmNode *pNode, int nLevel)
1452 {
1453     if (!pNode)
1454         return;
1455     switch(pNode->GetType())
1456     {
1457         case NTABLE:
1458             ExportTable(pNode, nLevel);
1459             break;
1460         case NALIGN:
1461         case NBRACEBODY:
1462         case NEXPRESSION:
1463             ExportExpression(pNode, nLevel);
1464             break;
1465         case NLINE:
1466             ExportLine(pNode, nLevel);
1467             break;
1468         case NTEXT:
1469             ExportText(pNode, nLevel);
1470             break;
1471         case NSPECIAL: //NSPECIAL requires some sort of Entity preservation in the XML engine.
1472         case NGLYPH_SPECIAL:
1473         case NMATH:
1474             {
1475                 sal_Unicode cTmp = 0;
1476                 const SmTextNode *pTemp = static_cast< const SmTextNode * >(pNode);
1477                 if (pTemp->GetText().Len() > 0)
1478                     cTmp = ConvertMathToMathML( pTemp->GetText().GetChar(0) );
1479                 if (cTmp == 0)
1480                 {
1481                     // no conversion to MathML implemented -> export it as text
1482                     // thus at least it will not vanish into nothing
1483                     ExportText(pNode, nLevel);
1484                 }
1485                 else
1486                 {
1487                     //To fully handle generic MathML we need to implement the full
1488                     //operator dictionary, we will generate MathML with explicit
1489                     //stretchiness for now.
1490                     sal_Int16 nLength = GetAttrList().getLength();
1491                     sal_Bool bAddStretch=sal_True;
1492                     for ( sal_Int16 i = 0; i < nLength; i++ )
1493                     {
1494                         OUString sLocalName;
1495                         sal_uInt16 nPrefix = GetNamespaceMap().GetKeyByAttrName(
1496                             GetAttrList().getNameByIndex(i), &sLocalName );
1497 
1498                         if ( ( XML_NAMESPACE_MATH == nPrefix ) &&
1499                             IsXMLToken(sLocalName, XML_STRETCHY) )
1500                         {
1501                             bAddStretch = sal_False;
1502                             break;
1503                         }
1504                     }
1505                     if (bAddStretch)
1506                     {
1507                         AddAttribute(XML_NAMESPACE_MATH, XML_STRETCHY, XML_FALSE);
1508                     }
1509                     ExportMath(pNode, nLevel);
1510                 }
1511             }
1512             break;
1513         case NPLACE:
1514             ExportMath(pNode, nLevel);
1515             break;
1516         case NBINHOR:
1517             ExportBinaryHorizontal(pNode, nLevel);
1518             break;
1519         case NUNHOR:
1520             ExportUnaryHorizontal(pNode, nLevel);
1521             break;
1522         case NBRACE:
1523             ExportBrace(pNode, nLevel);
1524             break;
1525         case NBINVER:
1526             ExportBinaryVertical(pNode, nLevel);
1527             break;
1528         case NSUBSUP:
1529             ExportSubSupScript(pNode, nLevel);
1530             break;
1531         case NROOT:
1532             ExportRoot(pNode, nLevel);
1533             break;
1534         case NOPER:
1535             ExportOperator(pNode, nLevel);
1536             break;
1537         case NATTRIBUT:
1538             ExportAttributes(pNode, nLevel);
1539             break;
1540         case NFONT:
1541             ExportFont(pNode, nLevel);
1542             break;
1543         case NVERTICAL_BRACE:
1544             ExportVerticalBrace(pNode, nLevel);
1545             break;
1546         case NMATRIX:
1547             ExportMatrix(pNode, nLevel);
1548             break;
1549         case NBLANK:
1550             ExportBlank(pNode, nLevel);
1551             break;
1552        default:
1553             DBG_ASSERT( 0, "Warning: failed to export a node?" );
1554             break;
1555 
1556 #if 0
1557         default:
1558             {
1559             sal_uLong  nSize = pNode->GetNumSubNodes();
1560             for (sal_uLong i = 0; i < nSize; i++)
1561                 if (SmNode *pTemp = pNode->GetSubNode(i))
1562                     ExportNodes(pTemp, nLevel+1);
1563             }
1564             break;
1565 #endif
1566     }
1567 }
1568 
1569 ////////////////////////////////////////////////////////////
1570 
1571