Line data Source code
1 : /* -*- Mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*- */
2 : /*
3 : * This file is part of the LibreOffice project.
4 : *
5 : * This Source Code Form is subject to the terms of the Mozilla Public
6 : * License, v. 2.0. If a copy of the MPL was not distributed with this
7 : * file, You can obtain one at http://mozilla.org/MPL/2.0/.
8 : *
9 : * This file incorporates work covered by the following license notice:
10 : *
11 : * Licensed to the Apache Software Foundation (ASF) under one or more
12 : * contributor license agreements. See the NOTICE file distributed
13 : * with this work for additional information regarding copyright
14 : * ownership. The ASF licenses this file to you under the Apache
15 : * License, Version 2.0 (the "License"); you may not use this file
16 : * except in compliance with the License. You may obtain a copy of
17 : * the License at http://www.apache.org/licenses/LICENSE-2.0 .
18 : */
19 :
20 : #include <tools/urlobj.hxx>
21 : #include <com/sun/star/container/XNameContainer.hpp>
22 : #include <com/sun/star/xml/sax/Writer.hpp>
23 : #include <com/sun/star/uno/Sequence.hxx>
24 : #include <com/sun/star/drawing/PolyPolygonBezierCoords.hpp>
25 : #include <com/sun/star/drawing/LineDash.hpp>
26 : #include <com/sun/star/awt/Gradient.hpp>
27 : #include <com/sun/star/drawing/Hatch.hpp>
28 : #include <com/sun/star/io/XActiveDataSource.hpp>
29 : #include <com/sun/star/embed/ElementModes.hpp>
30 :
31 : #include <sax/tools/converter.hxx>
32 : #include <sfx2/docfile.hxx>
33 : #include <rtl/ustrbuf.hxx>
34 : #include "xmloff/xmlnmspe.hxx"
35 : #include "xmloff/nmspmap.hxx"
36 :
37 : #include "xmloff/xmltoken.hxx"
38 : #include "xmloff/xmlmetae.hxx"
39 : #include "xmloff/DashStyle.hxx"
40 : #include "xmloff/GradientStyle.hxx"
41 : #include "xmloff/HatchStyle.hxx"
42 : #include "xmloff/ImageStyle.hxx"
43 : #include "xmloff/MarkerStyle.hxx"
44 : #include <com/sun/star/embed/XTransactedObject.hpp>
45 : #include <comphelper/processfactory.hxx>
46 : #include <unotools/streamwrap.hxx>
47 : #include "svx/xmlgrhlp.hxx"
48 :
49 : #include "xmlxtexp.hxx"
50 :
51 : #include <comphelper/storagehelper.hxx>
52 :
53 : using namespace com::sun::star;
54 : using namespace com::sun::star::container;
55 : using namespace com::sun::star::document;
56 : using namespace com::sun::star::uno;
57 : using namespace com::sun::star::awt;
58 : using namespace com::sun::star::lang;
59 : using namespace com::sun::star::xml::sax;
60 : using namespace ::xmloff::token;
61 : using namespace ::rtl;
62 : using namespace cppu;
63 :
64 : using com::sun::star::embed::XTransactedObject;
65 :
66 : class SvxXMLTableEntryExporter
67 : {
68 : public:
69 0 : SvxXMLTableEntryExporter( SvXMLExport& rExport ) : mrExport( rExport ) {}
70 : virtual ~SvxXMLTableEntryExporter();
71 :
72 : virtual void exportEntry( const OUString& rStrName, const Any& rValue ) = 0;
73 :
74 : protected:
75 : SvXMLExport& mrExport;
76 : };
77 :
78 : class SvxXMLColorEntryExporter : public SvxXMLTableEntryExporter
79 : {
80 : public:
81 : SvxXMLColorEntryExporter( SvXMLExport& rExport );
82 : virtual ~SvxXMLColorEntryExporter();
83 :
84 : virtual void exportEntry( const OUString& rStrName, const Any& rValue ) SAL_OVERRIDE;
85 : };
86 :
87 : class SvxXMLLineEndEntryExporter : public SvxXMLTableEntryExporter
88 : {
89 : public:
90 : SvxXMLLineEndEntryExporter( SvXMLExport& rExport );
91 : virtual ~SvxXMLLineEndEntryExporter();
92 :
93 : virtual void exportEntry( const OUString& rStrName, const Any& rValue ) SAL_OVERRIDE;
94 : private:
95 : XMLMarkerStyleExport maMarkerStyle;
96 : };
97 :
98 : class SvxXMLDashEntryExporter : public SvxXMLTableEntryExporter
99 : {
100 : public:
101 : SvxXMLDashEntryExporter( SvXMLExport& rExport );
102 : virtual ~SvxXMLDashEntryExporter();
103 :
104 : virtual void exportEntry( const OUString& rStrName, const Any& rValue ) SAL_OVERRIDE;
105 :
106 : private:
107 : XMLDashStyleExport maDashStyle;
108 : };
109 :
110 : class SvxXMLHatchEntryExporter : public SvxXMLTableEntryExporter
111 : {
112 : public:
113 : SvxXMLHatchEntryExporter( SvXMLExport& rExport );
114 : virtual ~SvxXMLHatchEntryExporter();
115 :
116 : virtual void exportEntry( const OUString& rStrName, const Any& rValue ) SAL_OVERRIDE;
117 : private:
118 : XMLHatchStyleExport maHatchStyle;
119 : };
120 :
121 : class SvxXMLGradientEntryExporter : public SvxXMLTableEntryExporter
122 : {
123 : public:
124 : SvxXMLGradientEntryExporter( SvXMLExport& rExport );
125 : virtual ~SvxXMLGradientEntryExporter();
126 :
127 : virtual void exportEntry( const OUString& rStrName, const Any& rValue ) SAL_OVERRIDE;
128 : private:
129 : XMLGradientStyleExport maGradientStyle;
130 : };
131 :
132 : class SvxXMLBitmapEntryExporter : public SvxXMLTableEntryExporter
133 : {
134 : public:
135 : SvxXMLBitmapEntryExporter( SvXMLExport& rExport );
136 : virtual ~SvxXMLBitmapEntryExporter();
137 :
138 : virtual void exportEntry( const OUString& rStrName, const Any& rValue ) SAL_OVERRIDE;
139 :
140 : private:
141 : XMLImageStyle maImageStyle;
142 : };
143 :
144 :
145 :
146 : // #110680#
147 0 : SvxXMLXTableExportComponent::SvxXMLXTableExportComponent(
148 : const ::com::sun::star::uno::Reference< ::com::sun::star::uno::XComponentContext > xContext,
149 : const OUString& rFileName,
150 : const uno::Reference<xml::sax::XDocumentHandler> & rHandler,
151 : const uno::Reference<container::XNameContainer >& xTable,
152 : uno::Reference<document::XGraphicObjectResolver >& xGrfResolver )
153 : : SvXMLExport( xContext, "", rFileName, rHandler, NULL, MAP_100TH_MM),
154 0 : mxTable( xTable )
155 : {
156 :
157 0 : _GetNamespaceMap().Add( GetXMLToken(XML_NP_OOO), GetXMLToken(XML_N_OOO), XML_NAMESPACE_OOO );
158 0 : _GetNamespaceMap().Add( GetXMLToken(XML_NP_OFFICE), GetXMLToken(XML_N_OFFICE), XML_NAMESPACE_OFFICE );
159 0 : _GetNamespaceMap().Add( GetXMLToken(XML_NP_DRAW), GetXMLToken(XML_N_DRAW), XML_NAMESPACE_DRAW );
160 0 : _GetNamespaceMap().Add( GetXMLToken(XML_NP_XLINK), GetXMLToken(XML_N_XLINK), XML_NAMESPACE_XLINK );
161 0 : _GetNamespaceMap().Add( GetXMLToken(XML_NP_SVG), GetXMLToken(XML_N_SVG), XML_NAMESPACE_SVG );
162 0 : SetGraphicResolver( xGrfResolver );
163 0 : setExportFlags( 0 );
164 0 : }
165 :
166 0 : SvxXMLXTableExportComponent::~SvxXMLXTableExportComponent()
167 : {
168 0 : }
169 :
170 0 : static void initializeStreamMetadata( const uno::Reference< uno::XInterface > &xOut )
171 : {
172 0 : uno::Reference< beans::XPropertySet > xProps( xOut, uno::UNO_QUERY );
173 0 : if( !xProps.is() )
174 : {
175 : OSL_FAIL( "Missing stream metadata interface" );
176 0 : return;
177 : }
178 :
179 : try
180 : {
181 0 : xProps->setPropertyValue(
182 : OUString( "MediaType" ),
183 0 : uno::makeAny( OUString( "text/xml" ) ) );
184 :
185 : // use stock encryption
186 0 : xProps->setPropertyValue(
187 : OUString( "UseCommonStoragePasswordEncryption" ),
188 0 : uno::makeAny( sal_True ) );
189 0 : } catch ( const uno::Exception & )
190 : {
191 : OSL_FAIL( "exception setting stream metadata" );
192 0 : }
193 : }
194 :
195 0 : static void createStorageStream( uno::Reference < io::XOutputStream > *xOut,
196 : SvXMLGraphicHelper **ppGraphicHelper,
197 : uno::Reference < embed::XStorage > xSubStorage )
198 : {
199 0 : uno::Reference < io::XStream > xStream;
200 0 : xStream = xSubStorage->openStreamElement(
201 : OUString( "Content.xml" ),
202 0 : embed::ElementModes::WRITE );
203 0 : *ppGraphicHelper = SvXMLGraphicHelper::Create( xSubStorage, GRAPHICHELPER_MODE_WRITE );
204 0 : initializeStreamMetadata( xStream );
205 0 : *xOut = xStream->getOutputStream();
206 0 : }
207 :
208 0 : bool SvxXMLXTableExportComponent::save(
209 : const OUString& rURL,
210 : const uno::Reference<container::XNameContainer >& xTable,
211 : const uno::Reference<embed::XStorage >& xStorage,
212 : OUString *pOptName ) throw()
213 : {
214 0 : bool bRet = false;
215 0 : SfxMedium* pMedium = NULL;
216 0 : SvXMLGraphicHelper* pGraphicHelper = NULL;
217 0 : sal_Int32 eCreate = embed::ElementModes::WRITE | embed::ElementModes::TRUNCATE;
218 :
219 0 : INetURLObject aURLObj( rURL );
220 0 : bool bToStorage = aURLObj.GetProtocol() == INET_PROT_NOT_VALID; // a relative path
221 :
222 0 : bool bSaveAsStorage = xTable->getElementType() == ::getCppuType((const OUString*)0);
223 :
224 0 : if( pOptName )
225 0 : *pOptName = rURL;
226 :
227 : try
228 : {
229 0 : uno::Reference< uno::XComponentContext> xContext( ::comphelper::getProcessComponentContext() );
230 :
231 0 : uno::Reference< xml::sax::XWriter > xWriter = xml::sax::Writer::create( xContext );
232 :
233 0 : uno::Reference < io::XStream > xStream;
234 0 : uno::Reference < io::XOutputStream > xOut;
235 0 : uno::Reference<embed::XStorage > xSubStorage;
236 0 : uno::Reference< XGraphicObjectResolver > xGrfResolver;
237 :
238 0 : uno::Reference<xml::sax::XDocumentHandler> xHandler( xWriter, uno::UNO_QUERY );
239 :
240 0 : if( !bToStorage || !xStorage.is() )
241 : { // local URL -> SfxMedium route
242 0 : if( bSaveAsStorage )
243 0 : xSubStorage = ::comphelper::OStorageHelper::GetStorageFromURL( rURL, eCreate );
244 : else
245 : {
246 0 : pMedium = new SfxMedium( rURL, STREAM_WRITE | STREAM_TRUNC );
247 0 : pMedium->IsRemote();
248 :
249 0 : SvStream* pStream = pMedium->GetOutStream();
250 0 : if( !pStream )
251 : {
252 : OSL_FAIL( "no output stream!" );
253 0 : return false;
254 : }
255 :
256 0 : xOut = new utl::OOutputStreamWrapper( *pStream );
257 : }
258 : }
259 : else // save into the xSubStorage
260 : {
261 0 : OUString aPath = rURL;
262 :
263 0 : if( bSaveAsStorage )
264 : {
265 : try {
266 0 : xSubStorage = xStorage->openStorageElement( aPath, eCreate );
267 0 : } catch (uno::Exception &) {
268 : OSL_FAIL( "no output storage!" );
269 0 : return false;
270 : }
271 : }
272 : else
273 : {
274 0 : aPath += OUString( ".xml" );
275 : try {
276 0 : xStream = xStorage->openStreamElement( aPath, eCreate );
277 0 : if( !xStream.is() )
278 0 : return false;
279 0 : initializeStreamMetadata( xStream );
280 0 : xOut = xStream->getOutputStream();
281 0 : } catch (uno::Exception &) {
282 : OSL_FAIL( "no output stream!" );
283 0 : return false;
284 : }
285 0 : if( pOptName )
286 0 : *pOptName = aPath;
287 0 : }
288 : }
289 :
290 0 : if( !xOut.is() && xSubStorage.is() )
291 0 : createStorageStream( &xOut, &pGraphicHelper, xSubStorage );
292 0 : if( !xOut.is() )
293 0 : return false;
294 :
295 0 : uno::Reference<io::XActiveDataSource> xMetaSrc( xWriter, uno::UNO_QUERY );
296 0 : xMetaSrc->setOutputStream( xOut );
297 0 : if( pGraphicHelper )
298 0 : xGrfResolver = pGraphicHelper;
299 :
300 : // Finally do the export
301 0 : const OUString aName;
302 0 : SvxXMLXTableExportComponent aExporter( xContext, aName, xHandler, xTable, xGrfResolver );
303 0 : bRet = aExporter.exportTable();
304 :
305 0 : if( pGraphicHelper )
306 0 : SvXMLGraphicHelper::Destroy( pGraphicHelper );
307 :
308 0 : if( xSubStorage.is() )
309 : {
310 0 : uno::Reference< XTransactedObject > xTrans( xSubStorage, UNO_QUERY );
311 0 : if( xTrans.is() )
312 0 : xTrans->commit();
313 :
314 0 : uno::Reference< XComponent > xComp( xSubStorage, UNO_QUERY );
315 0 : if( xComp.is() )
316 0 : xSubStorage->dispose();
317 0 : }
318 : }
319 0 : catch( uno::Exception& )
320 : {
321 0 : bRet = false;
322 : }
323 :
324 0 : if( pMedium )
325 : {
326 0 : pMedium->Commit();
327 0 : delete pMedium;
328 : }
329 :
330 0 : return bRet;
331 : }
332 :
333 0 : bool SvxXMLXTableExportComponent::exportTable() throw()
334 : {
335 0 : bool bRet = false;
336 :
337 : try
338 : {
339 0 : GetDocHandler()->startDocument();
340 :
341 0 : addChaffWhenEncryptedStorage();
342 :
343 : // export namespaces
344 0 : sal_uInt16 nPos = GetNamespaceMap().GetFirstKey();
345 0 : while( USHRT_MAX != nPos )
346 : {
347 0 : GetAttrList().AddAttribute( GetNamespaceMap().GetAttrNameByKey( nPos ), GetNamespaceMap().GetNameByKey( nPos ) );
348 0 : nPos = GetNamespaceMap().GetNextKey( nPos );
349 : }
350 :
351 : do
352 : {
353 0 : if( !mxTable.is() )
354 0 : break;
355 :
356 : char const* pEleName;
357 0 : Type aExportType = mxTable->getElementType();
358 0 : SvxXMLTableEntryExporter* pExporter = NULL;
359 :
360 0 : if( aExportType == ::getCppuType((const sal_Int32*)0) )
361 : {
362 0 : pExporter = new SvxXMLColorEntryExporter(*this);
363 0 : pEleName = "color-table";
364 : }
365 0 : else if( aExportType == ::getCppuType((const drawing::PolyPolygonBezierCoords*)0) )
366 : {
367 0 : pExporter = new SvxXMLLineEndEntryExporter(*this);
368 0 : pEleName = "marker-table";
369 : }
370 0 : else if( aExportType == ::getCppuType((const drawing::LineDash*)0) )
371 : {
372 0 : pExporter = new SvxXMLDashEntryExporter(*this);
373 0 : pEleName = "dash-table";
374 : }
375 0 : else if( aExportType == ::getCppuType((const drawing::Hatch*)0) )
376 : {
377 0 : pExporter = new SvxXMLHatchEntryExporter(*this);
378 0 : pEleName = "hatch-table";
379 : }
380 0 : else if( aExportType == ::getCppuType((const awt::Gradient*)0))
381 : {
382 0 : pExporter = new SvxXMLGradientEntryExporter(*this);
383 0 : pEleName = "gradient-table";
384 : }
385 0 : else if( aExportType == ::getCppuType((const OUString*)0))
386 : {
387 0 : pExporter = new SvxXMLBitmapEntryExporter(*this);
388 0 : pEleName = "bitmap-table";
389 : }
390 : else
391 : {
392 : OSL_FAIL( "unknown type for export");
393 0 : break;
394 : }
395 :
396 0 : SvXMLElementExport aElem( *this, XML_NAMESPACE_OOO, pEleName, true, true );
397 :
398 0 : Sequence< OUString > aNames = mxTable->getElementNames();
399 0 : const sal_Int32 nCount = aNames.getLength();
400 0 : const OUString* pNames = aNames.getConstArray();
401 0 : Any aAny;
402 :
403 : sal_Int32 nIndex;
404 0 : for( nIndex = 0; nIndex < nCount; nIndex++, pNames++ )
405 : {
406 0 : aAny = mxTable->getByName( *pNames );
407 0 : pExporter->exportEntry( *pNames, aAny );
408 : }
409 0 : delete pExporter;
410 :
411 0 : bRet = true;
412 : }
413 : while(false);
414 :
415 0 : GetDocHandler()->endDocument();
416 : }
417 0 : catch( Exception const& )
418 : {
419 0 : bRet = false;
420 : }
421 :
422 0 : return bRet;
423 : }
424 :
425 : // methods without content:
426 0 : void SvxXMLXTableExportComponent::_ExportAutoStyles() {}
427 0 : void SvxXMLXTableExportComponent::_ExportMasterStyles() {}
428 0 : void SvxXMLXTableExportComponent::_ExportContent() {}
429 :
430 :
431 :
432 0 : SvxXMLTableEntryExporter::~SvxXMLTableEntryExporter()
433 : {
434 0 : }
435 :
436 :
437 :
438 0 : SvxXMLColorEntryExporter::SvxXMLColorEntryExporter( SvXMLExport& rExport )
439 0 : : SvxXMLTableEntryExporter( rExport )
440 : {
441 0 : }
442 :
443 0 : SvxXMLColorEntryExporter::~SvxXMLColorEntryExporter()
444 : {
445 0 : }
446 :
447 0 : void SvxXMLColorEntryExporter::exportEntry( const OUString& rStrName, const Any& rValue )
448 : {
449 0 : mrExport.AddAttribute( XML_NAMESPACE_DRAW, XML_NAME, rStrName );
450 :
451 0 : sal_Int32 nColor = 0;
452 0 : rValue >>= nColor;
453 :
454 0 : OUStringBuffer aOut;
455 0 : ::sax::Converter::convertColor( aOut, nColor );
456 0 : mrExport.AddAttribute( XML_NAMESPACE_DRAW, XML_COLOR, aOut.makeStringAndClear() );
457 :
458 0 : SvXMLElementExport aElem( mrExport, XML_NAMESPACE_DRAW, XML_COLOR, true, true );
459 0 : }
460 :
461 :
462 :
463 0 : SvxXMLLineEndEntryExporter::SvxXMLLineEndEntryExporter( SvXMLExport& rExport )
464 0 : : SvxXMLTableEntryExporter( rExport ), maMarkerStyle( rExport )
465 : {
466 0 : }
467 :
468 0 : SvxXMLLineEndEntryExporter::~SvxXMLLineEndEntryExporter()
469 : {
470 0 : }
471 :
472 0 : void SvxXMLLineEndEntryExporter::exportEntry( const OUString& rStrName, const Any& rValue )
473 : {
474 0 : maMarkerStyle.exportXML( rStrName, rValue );
475 0 : }
476 :
477 :
478 :
479 0 : SvxXMLDashEntryExporter::SvxXMLDashEntryExporter( SvXMLExport& rExport )
480 0 : : SvxXMLTableEntryExporter( rExport ), maDashStyle( rExport )
481 : {
482 0 : }
483 :
484 0 : SvxXMLDashEntryExporter::~SvxXMLDashEntryExporter()
485 : {
486 0 : }
487 :
488 0 : void SvxXMLDashEntryExporter::exportEntry( const OUString& rStrName, const Any& rValue )
489 : {
490 0 : maDashStyle.exportXML( rStrName, rValue );
491 0 : }
492 :
493 :
494 :
495 0 : SvxXMLHatchEntryExporter::SvxXMLHatchEntryExporter( SvXMLExport& rExport )
496 0 : : SvxXMLTableEntryExporter( rExport ), maHatchStyle( rExport )
497 : {
498 0 : }
499 :
500 0 : SvxXMLHatchEntryExporter::~SvxXMLHatchEntryExporter()
501 : {
502 0 : }
503 :
504 0 : void SvxXMLHatchEntryExporter::exportEntry( const OUString& rStrName, const Any& rValue )
505 : {
506 0 : maHatchStyle.exportXML( rStrName, rValue );
507 0 : }
508 :
509 :
510 :
511 0 : SvxXMLGradientEntryExporter::SvxXMLGradientEntryExporter( SvXMLExport& rExport )
512 0 : : SvxXMLTableEntryExporter( rExport ), maGradientStyle( rExport )
513 : {
514 0 : }
515 :
516 0 : SvxXMLGradientEntryExporter::~SvxXMLGradientEntryExporter()
517 : {
518 0 : }
519 :
520 0 : void SvxXMLGradientEntryExporter::exportEntry( const OUString& rStrName, const Any& rValue )
521 : {
522 0 : maGradientStyle.exportXML( rStrName, rValue );
523 0 : }
524 :
525 :
526 :
527 0 : SvxXMLBitmapEntryExporter::SvxXMLBitmapEntryExporter( SvXMLExport& rExport )
528 0 : : SvxXMLTableEntryExporter( rExport )
529 : {
530 0 : }
531 :
532 0 : SvxXMLBitmapEntryExporter::~SvxXMLBitmapEntryExporter()
533 : {
534 0 : }
535 :
536 0 : void SvxXMLBitmapEntryExporter::exportEntry( const OUString& rStrName, const Any& rValue )
537 : {
538 0 : maImageStyle.exportXML( rStrName, rValue, mrExport );
539 0 : }
540 :
541 : /* vim:set shiftwidth=4 softtabstop=4 expandtab: */
|