Version 24.2.2.2, tag libreoffice-24.2.2.2
[LibreOffice.git] / svx / source / xml / xmlxtexp.cxx
blob214d976b7d714666439a6999e3125418a1b2990d
1 /* -*- Mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*- */
2 /*
3 * This file is part of the LibreOffice project.
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/.
9 * This file incorporates work covered by the following license notice:
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 .
20 #include <sal/config.h>
22 #include <comphelper/diagnose_ex.hxx>
23 #include <tools/urlobj.hxx>
24 #include <com/sun/star/beans/XPropertySet.hpp>
25 #include <com/sun/star/container/XNameContainer.hpp>
26 #include <com/sun/star/xml/sax/Writer.hpp>
27 #include <com/sun/star/uno/Sequence.hxx>
28 #include <com/sun/star/drawing/PolyPolygonBezierCoords.hpp>
29 #include <com/sun/star/drawing/LineDash.hpp>
30 #include <com/sun/star/awt/Gradient.hpp>
31 #include <com/sun/star/awt/XBitmap.hpp>
32 #include <com/sun/star/drawing/Hatch.hpp>
33 #include <com/sun/star/embed/ElementModes.hpp>
35 #include <sax/tools/converter.hxx>
36 #include <sfx2/docfile.hxx>
37 #include <rtl/ustrbuf.hxx>
38 #include <xmloff/xmlnamespace.hxx>
39 #include <xmloff/namespacemap.hxx>
41 #include <xmloff/xmltoken.hxx>
42 #include <xmloff/xmlmetae.hxx>
43 #include <xmloff/DashStyle.hxx>
44 #include <xmloff/GradientStyle.hxx>
45 #include <xmloff/HatchStyle.hxx>
46 #include <xmloff/ImageStyle.hxx>
47 #include <xmloff/MarkerStyle.hxx>
48 #include <com/sun/star/embed/XTransactedObject.hpp>
49 #include <comphelper/processfactory.hxx>
50 #include <unotools/streamwrap.hxx>
51 #include <svx/xmlgrhlp.hxx>
53 #include <xmlxtexp.hxx>
55 #include <comphelper/storagehelper.hxx>
56 #include <memory>
58 using namespace com::sun::star;
59 using namespace com::sun::star::container;
60 using namespace com::sun::star::document;
61 using namespace com::sun::star::uno;
62 using namespace com::sun::star::awt;
63 using namespace com::sun::star::lang;
64 using namespace com::sun::star::xml::sax;
65 using namespace ::xmloff::token;
66 using namespace cppu;
68 using com::sun::star::embed::XTransactedObject;
70 namespace {
72 class SvxXMLTableEntryExporter
74 public:
75 explicit SvxXMLTableEntryExporter( SvXMLExport& rExport ) : mrExport( rExport ) {}
76 virtual ~SvxXMLTableEntryExporter();
78 virtual void exportEntry( const OUString& rStrName, const Any& rValue ) = 0;
80 protected:
81 SvXMLExport& mrExport;
84 class SvxXMLColorEntryExporter : public SvxXMLTableEntryExporter
86 public:
87 explicit SvxXMLColorEntryExporter( SvXMLExport& rExport );
89 virtual void exportEntry( const OUString& rStrName, const Any& rValue ) override;
92 class SvxXMLLineEndEntryExporter : public SvxXMLTableEntryExporter
94 public:
95 explicit SvxXMLLineEndEntryExporter( SvXMLExport& rExport );
97 virtual void exportEntry( const OUString& rStrName, const Any& rValue ) override;
98 private:
99 XMLMarkerStyleExport maMarkerStyle;
102 class SvxXMLDashEntryExporter : public SvxXMLTableEntryExporter
104 public:
105 explicit SvxXMLDashEntryExporter( SvXMLExport& rExport );
107 virtual void exportEntry( const OUString& rStrName, const Any& rValue ) override;
109 private:
110 XMLDashStyleExport maDashStyle;
113 class SvxXMLHatchEntryExporter : public SvxXMLTableEntryExporter
115 public:
116 explicit SvxXMLHatchEntryExporter( SvXMLExport& rExport );
118 virtual void exportEntry( const OUString& rStrName, const Any& rValue ) override;
119 private:
120 XMLHatchStyleExport maHatchStyle;
123 class SvxXMLGradientEntryExporter : public SvxXMLTableEntryExporter
125 public:
126 explicit SvxXMLGradientEntryExporter( SvXMLExport& rExport );
128 virtual void exportEntry( const OUString& rStrName, const Any& rValue ) override;
129 private:
130 XMLGradientStyleExport maGradientStyle;
133 class SvxXMLBitmapEntryExporter : public SvxXMLTableEntryExporter
135 public:
136 explicit SvxXMLBitmapEntryExporter( SvXMLExport& rExport );
138 virtual void exportEntry( const OUString& rStrName, const Any& rValue ) override;
143 SvxXMLXTableExportComponent::SvxXMLXTableExportComponent(
144 const css::uno::Reference< css::uno::XComponentContext >& rContext,
145 const uno::Reference<xml::sax::XDocumentHandler> & rHandler,
146 const uno::Reference<container::XNameContainer >& xTable,
147 uno::Reference<document::XGraphicStorageHandler> const & xGraphicStorageHandler)
148 : SvXMLExport(rContext, "", /*rFileName*/"", rHandler, nullptr, FieldUnit::MM_100TH, SvXMLExportFlags::NONE),
149 mxTable( xTable )
152 GetNamespaceMap_().Add( GetXMLToken(XML_NP_OOO), GetXMLToken(XML_N_OOO), XML_NAMESPACE_OOO );
153 GetNamespaceMap_().Add( GetXMLToken(XML_NP_OFFICE), GetXMLToken(XML_N_OFFICE), XML_NAMESPACE_OFFICE );
154 GetNamespaceMap_().Add( GetXMLToken(XML_NP_DRAW), GetXMLToken(XML_N_DRAW), XML_NAMESPACE_DRAW );
155 GetNamespaceMap_().Add( GetXMLToken(XML_NP_XLINK), GetXMLToken(XML_N_XLINK), XML_NAMESPACE_XLINK );
156 GetNamespaceMap_().Add( GetXMLToken(XML_NP_SVG), GetXMLToken(XML_N_SVG), XML_NAMESPACE_SVG );
157 GetNamespaceMap_().Add( GetXMLToken(XML_NP_LO_EXT), GetXMLToken(XML_N_LO_EXT), XML_NAMESPACE_LO_EXT);
158 SetGraphicStorageHandler(xGraphicStorageHandler);
161 SvxXMLXTableExportComponent::~SvxXMLXTableExportComponent()
165 static void initializeStreamMetadata( const uno::Reference< uno::XInterface > &xOut )
167 uno::Reference< beans::XPropertySet > xProps( xOut, uno::UNO_QUERY );
168 if( !xProps.is() )
170 OSL_FAIL( "Missing stream metadata interface" );
171 return;
176 xProps->setPropertyValue("MediaType", uno::Any( OUString( "text/xml" ) ) );
178 // use stock encryption
179 xProps->setPropertyValue("UseCommonStoragePasswordEncryption", uno::Any( true ) );
180 } catch ( const uno::Exception & )
182 TOOLS_WARN_EXCEPTION("svx", "exception setting stream metadata");
186 static void createStorageStream( uno::Reference < io::XOutputStream > *xOut,
187 rtl::Reference<SvXMLGraphicHelper>& rxGraphicHelper,
188 const uno::Reference < embed::XStorage >& xSubStorage )
190 uno::Reference < io::XStream > xStream = xSubStorage->openStreamElement(
191 "Content.xml",
192 embed::ElementModes::WRITE );
193 rxGraphicHelper = SvXMLGraphicHelper::Create( xSubStorage, SvXMLGraphicHelperMode::Write );
194 initializeStreamMetadata( xStream );
195 *xOut = xStream->getOutputStream();
198 bool SvxXMLXTableExportComponent::save(
199 const OUString& rURL,
200 const uno::Reference<container::XNameContainer >& xTable,
201 const uno::Reference<embed::XStorage >& xStorage,
202 OUString *pOptName )
204 bool bRet = false;
205 std::unique_ptr<SfxMedium> pMedium;
206 rtl::Reference<SvXMLGraphicHelper> xGraphicHelper;
208 INetURLObject aURLObj( rURL );
209 bool bToStorage = aURLObj.GetProtocol() == INetProtocol::NotValid; // a relative path
210 bool bSaveAsStorage = xTable->getElementType() == cppu::UnoType<awt::XBitmap>::get();
212 if( pOptName )
213 *pOptName = rURL;
217 uno::Reference< uno::XComponentContext> xContext( ::comphelper::getProcessComponentContext() );
219 uno::Reference< xml::sax::XWriter > xWriter = xml::sax::Writer::create( xContext );
221 uno::Reference < io::XStream > xStream;
222 uno::Reference < io::XOutputStream > xOut;
223 uno::Reference<embed::XStorage > xSubStorage;
224 uno::Reference<XGraphicStorageHandler> xGraphicStorageHandler;
225 const sal_Int32 eCreate = embed::ElementModes::WRITE | embed::ElementModes::TRUNCATE;
227 if( !bToStorage || !xStorage.is() )
228 { // local URL -> SfxMedium route
229 if( bSaveAsStorage )
231 // ideally this should use a ZIP_STORAGE_FORMAT_STRING storage
232 // but changing it to that could cause problems loading the
233 // file with an old version of LO that expects to find in the
234 // user profile a PACKAGE_STORAGE_FORMAT_STRING storage
235 xSubStorage = ::comphelper::OStorageHelper::GetStorageFromURL( rURL, eCreate );
237 else
239 pMedium.reset(new SfxMedium( rURL, StreamMode::WRITE | StreamMode::TRUNC ));
241 SvStream* pStream = pMedium->GetOutStream();
242 if( !pStream )
244 OSL_FAIL( "no output stream!" );
245 return false;
248 xOut = new utl::OOutputStreamWrapper( *pStream );
251 else // save into the xSubStorage
253 OUString aPath = rURL;
255 if( bSaveAsStorage )
257 try {
258 xSubStorage = xStorage->openStorageElement( aPath, eCreate );
259 } catch (uno::Exception &) {
260 OSL_FAIL( "no output storage!" );
261 return false;
264 else
266 aPath += ".xml";
267 try {
268 xStream = xStorage->openStreamElement( aPath, eCreate );
269 if( !xStream.is() )
270 return false;
271 initializeStreamMetadata( xStream );
272 xOut = xStream->getOutputStream();
273 } catch (uno::Exception &) {
274 OSL_FAIL( "no output stream!" );
275 return false;
277 if( pOptName )
278 *pOptName = aPath;
282 if( !xOut.is() && xSubStorage.is() )
283 createStorageStream( &xOut, xGraphicHelper, xSubStorage );
284 if( !xOut.is() )
285 return false;
287 xWriter->setOutputStream( xOut );
288 if( xGraphicHelper.is() )
289 xGraphicStorageHandler = xGraphicHelper.get();
291 // Finally do the export
292 rtl::Reference< SvxXMLXTableExportComponent > xExporter( new SvxXMLXTableExportComponent( xContext, xWriter, xTable, xGraphicStorageHandler ) );
293 bRet = xExporter->exportTable();
295 if( xGraphicHelper )
296 xGraphicHelper->dispose();
297 xGraphicHelper.clear();
299 if( xSubStorage.is() )
301 uno::Reference< XTransactedObject > xTrans( xSubStorage, UNO_QUERY );
302 if( xTrans.is() )
303 xTrans->commit();
305 xSubStorage->dispose();
308 catch( uno::Exception& )
310 bRet = false;
313 if( pMedium )
314 pMedium->Commit();
316 return bRet;
319 bool SvxXMLXTableExportComponent::exportTable() noexcept
321 bool bRet = false;
325 GetDocHandler()->startDocument();
327 addChaffWhenEncryptedStorage();
329 // export namespaces
330 sal_uInt16 nPos = GetNamespaceMap().GetFirstKey();
331 while( USHRT_MAX != nPos )
333 GetAttrList().AddAttribute( GetNamespaceMap().GetAttrNameByKey( nPos ), GetNamespaceMap().GetNameByKey( nPos ) );
334 nPos = GetNamespaceMap().GetNextKey( nPos );
339 if( !mxTable.is() )
340 break;
342 char const* pEleName;
343 Type aExportType = mxTable->getElementType();
344 std::unique_ptr<SvxXMLTableEntryExporter> pExporter;
346 if( aExportType == cppu::UnoType<sal_Int32>::get() )
348 pExporter.reset(new SvxXMLColorEntryExporter(*this));
349 pEleName = "color-table";
351 else if( aExportType == cppu::UnoType< drawing::PolyPolygonBezierCoords >::get() )
353 pExporter.reset(new SvxXMLLineEndEntryExporter(*this));
354 pEleName = "marker-table";
356 else if( aExportType == cppu::UnoType< drawing::LineDash >::get() )
358 pExporter.reset(new SvxXMLDashEntryExporter(*this));
359 pEleName = "dash-table";
361 else if( aExportType == cppu::UnoType< drawing::Hatch >::get() )
363 pExporter.reset(new SvxXMLHatchEntryExporter(*this));
364 pEleName = "hatch-table";
366 else if( aExportType == cppu::UnoType< awt::Gradient >::get() )
368 pExporter.reset(new SvxXMLGradientEntryExporter(*this));
369 pEleName = "gradient-table";
371 else if( aExportType == cppu::UnoType<awt::XBitmap>::get())
373 pExporter.reset(new SvxXMLBitmapEntryExporter(*this));
374 pEleName = "bitmap-table";
376 else
378 OSL_FAIL( "unknown type for export");
379 break;
382 SvXMLElementExport aElem( *this, XML_NAMESPACE_OOO, pEleName, true, true );
384 const Sequence< OUString > aNames = mxTable->getElementNames();
385 Any aAny;
387 for( const OUString& rName : aNames )
389 aAny = mxTable->getByName( rName );
390 pExporter->exportEntry( rName, aAny );
393 bRet = true;
395 while(false);
397 GetDocHandler()->endDocument();
399 catch( Exception const& )
401 bRet = false;
404 return bRet;
407 // methods without content:
408 void SvxXMLXTableExportComponent::ExportAutoStyles_() {}
409 void SvxXMLXTableExportComponent::ExportMasterStyles_() {}
410 void SvxXMLXTableExportComponent::ExportContent_() {}
413 SvxXMLTableEntryExporter::~SvxXMLTableEntryExporter()
418 SvxXMLColorEntryExporter::SvxXMLColorEntryExporter( SvXMLExport& rExport )
419 : SvxXMLTableEntryExporter( rExport )
423 void SvxXMLColorEntryExporter::exportEntry( const OUString& rStrName, const Any& rValue )
425 mrExport.AddAttribute( XML_NAMESPACE_DRAW, XML_NAME, rStrName );
427 sal_Int32 nColor = 0;
428 rValue >>= nColor;
430 OUStringBuffer aOut;
431 ::sax::Converter::convertColor( aOut, nColor );
432 mrExport.AddAttribute( XML_NAMESPACE_DRAW, XML_COLOR, aOut.makeStringAndClear() );
434 SvXMLElementExport aElem( mrExport, XML_NAMESPACE_DRAW, XML_COLOR, true, true );
438 SvxXMLLineEndEntryExporter::SvxXMLLineEndEntryExporter( SvXMLExport& rExport )
439 : SvxXMLTableEntryExporter( rExport ), maMarkerStyle( rExport )
443 void SvxXMLLineEndEntryExporter::exportEntry( const OUString& rStrName, const Any& rValue )
445 maMarkerStyle.exportXML( rStrName, rValue );
449 SvxXMLDashEntryExporter::SvxXMLDashEntryExporter( SvXMLExport& rExport )
450 : SvxXMLTableEntryExporter( rExport ), maDashStyle( rExport )
454 void SvxXMLDashEntryExporter::exportEntry( const OUString& rStrName, const Any& rValue )
456 maDashStyle.exportXML( rStrName, rValue );
460 SvxXMLHatchEntryExporter::SvxXMLHatchEntryExporter( SvXMLExport& rExport )
461 : SvxXMLTableEntryExporter( rExport ), maHatchStyle( rExport )
465 void SvxXMLHatchEntryExporter::exportEntry( const OUString& rStrName, const Any& rValue )
467 maHatchStyle.exportXML( rStrName, rValue );
471 SvxXMLGradientEntryExporter::SvxXMLGradientEntryExporter( SvXMLExport& rExport )
472 : SvxXMLTableEntryExporter( rExport ), maGradientStyle( rExport )
476 void SvxXMLGradientEntryExporter::exportEntry( const OUString& rStrName, const Any& rValue )
478 maGradientStyle.exportXML( rStrName, rValue );
482 SvxXMLBitmapEntryExporter::SvxXMLBitmapEntryExporter( SvXMLExport& rExport )
483 : SvxXMLTableEntryExporter( rExport )
487 void SvxXMLBitmapEntryExporter::exportEntry( const OUString& rStrName, const Any& rValue )
489 XMLImageStyle::exportXML(rStrName, rValue, mrExport);
492 /* vim:set shiftwidth=4 softtabstop=4 expandtab: */