/* -*- Mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*- */ /* * This file is part of the LibreOffice project. * * This Source Code Form is subject to the terms of the Mozilla Public * License, v. 2.0. If a copy of the MPL was not distributed with this * file, You can obtain one at http://mozilla.org/MPL/2.0/. * * This file incorporates work covered by the following license notice: * * Licensed to the Apache Software Foundation (ASF) under one or more * contributor license agreements. See the NOTICE file distributed * with this work for additional information regarding copyright * ownership. The ASF licenses this file to you under the Apache * License, Version 2.0 (the "License"); you may not use this file * except in compliance with the License. You may obtain a copy of * the License at http://www.apache.org/licenses/LICENSE-2.0 . */ #include #include #include #include #include #include #include #include #include #include #include #include #include #include "xmloff/xmlnmspe.hxx" #include "xmloff/nmspmap.hxx" #include "xmloff/xmltoken.hxx" #include "xmloff/xmlmetae.hxx" #include "xmloff/DashStyle.hxx" #include "xmloff/GradientStyle.hxx" #include "xmloff/HatchStyle.hxx" #include "xmloff/ImageStyle.hxx" #include "xmloff/MarkerStyle.hxx" #include #include #include #include "svx/xmlgrhlp.hxx" #include "xmlxtexp.hxx" #include #include using namespace com::sun::star; using namespace com::sun::star::container; using namespace com::sun::star::document; using namespace com::sun::star::uno; using namespace com::sun::star::awt; using namespace com::sun::star::lang; using namespace com::sun::star::xml::sax; using namespace ::xmloff::token; using namespace cppu; using com::sun::star::embed::XTransactedObject; class SvxXMLTableEntryExporter { public: explicit SvxXMLTableEntryExporter( SvXMLExport& rExport ) : mrExport( rExport ) {} virtual ~SvxXMLTableEntryExporter(); virtual void exportEntry( const OUString& rStrName, const Any& rValue ) = 0; protected: SvXMLExport& mrExport; }; class SvxXMLColorEntryExporter : public SvxXMLTableEntryExporter { public: explicit SvxXMLColorEntryExporter( SvXMLExport& rExport ); virtual ~SvxXMLColorEntryExporter(); virtual void exportEntry( const OUString& rStrName, const Any& rValue ) override; }; class SvxXMLLineEndEntryExporter : public SvxXMLTableEntryExporter { public: explicit SvxXMLLineEndEntryExporter( SvXMLExport& rExport ); virtual ~SvxXMLLineEndEntryExporter(); virtual void exportEntry( const OUString& rStrName, const Any& rValue ) override; private: XMLMarkerStyleExport maMarkerStyle; }; class SvxXMLDashEntryExporter : public SvxXMLTableEntryExporter { public: explicit SvxXMLDashEntryExporter( SvXMLExport& rExport ); virtual ~SvxXMLDashEntryExporter(); virtual void exportEntry( const OUString& rStrName, const Any& rValue ) override; private: XMLDashStyleExport maDashStyle; }; class SvxXMLHatchEntryExporter : public SvxXMLTableEntryExporter { public: explicit SvxXMLHatchEntryExporter( SvXMLExport& rExport ); virtual ~SvxXMLHatchEntryExporter(); virtual void exportEntry( const OUString& rStrName, const Any& rValue ) override; private: XMLHatchStyleExport maHatchStyle; }; class SvxXMLGradientEntryExporter : public SvxXMLTableEntryExporter { public: explicit SvxXMLGradientEntryExporter( SvXMLExport& rExport ); virtual ~SvxXMLGradientEntryExporter(); virtual void exportEntry( const OUString& rStrName, const Any& rValue ) override; private: XMLGradientStyleExport maGradientStyle; }; class SvxXMLBitmapEntryExporter : public SvxXMLTableEntryExporter { public: explicit SvxXMLBitmapEntryExporter( SvXMLExport& rExport ); virtual ~SvxXMLBitmapEntryExporter(); virtual void exportEntry( const OUString& rStrName, const Any& rValue ) override; private: XMLImageStyle maImageStyle; }; SvxXMLXTableExportComponent::SvxXMLXTableExportComponent( const css::uno::Reference< css::uno::XComponentContext >& rContext, const OUString& rFileName, const uno::Reference & rHandler, const uno::Reference& xTable, uno::Reference& xGrfResolver ) : SvXMLExport(rContext, "", rFileName, rHandler, nullptr, FUNIT_100TH_MM), mxTable( xTable ) { GetNamespaceMap_().Add( GetXMLToken(XML_NP_OOO), GetXMLToken(XML_N_OOO), XML_NAMESPACE_OOO ); GetNamespaceMap_().Add( GetXMLToken(XML_NP_OFFICE), GetXMLToken(XML_N_OFFICE), XML_NAMESPACE_OFFICE ); GetNamespaceMap_().Add( GetXMLToken(XML_NP_DRAW), GetXMLToken(XML_N_DRAW), XML_NAMESPACE_DRAW ); GetNamespaceMap_().Add( GetXMLToken(XML_NP_XLINK), GetXMLToken(XML_N_XLINK), XML_NAMESPACE_XLINK ); GetNamespaceMap_().Add( GetXMLToken(XML_NP_SVG), GetXMLToken(XML_N_SVG), XML_NAMESPACE_SVG ); SetGraphicResolver( xGrfResolver ); setExportFlags( SvXMLExportFlags::NONE ); } SvxXMLXTableExportComponent::~SvxXMLXTableExportComponent() { } static void initializeStreamMetadata( const uno::Reference< uno::XInterface > &xOut ) { uno::Reference< beans::XPropertySet > xProps( xOut, uno::UNO_QUERY ); if( !xProps.is() ) { OSL_FAIL( "Missing stream metadata interface" ); return; } try { xProps->setPropertyValue("MediaType", uno::makeAny( OUString( "text/xml" ) ) ); // use stock encryption xProps->setPropertyValue("UseCommonStoragePasswordEncryption", uno::makeAny( true ) ); } catch ( const uno::Exception & ) { OSL_FAIL( "exception setting stream metadata" ); } } static void createStorageStream( uno::Reference < io::XOutputStream > *xOut, SvXMLGraphicHelper **ppGraphicHelper, const uno::Reference < embed::XStorage >& xSubStorage ) { uno::Reference < io::XStream > xStream; xStream = xSubStorage->openStreamElement( "Content.xml", embed::ElementModes::WRITE ); *ppGraphicHelper = SvXMLGraphicHelper::Create( xSubStorage, GRAPHICHELPER_MODE_WRITE ); initializeStreamMetadata( xStream ); *xOut = xStream->getOutputStream(); } bool SvxXMLXTableExportComponent::save( const OUString& rURL, const uno::Reference& xTable, const uno::Reference& xStorage, OUString *pOptName ) throw (css::uno::RuntimeException, std::exception) { bool bRet = false; SfxMedium* pMedium = nullptr; SvXMLGraphicHelper* pGraphicHelper = nullptr; sal_Int32 eCreate = embed::ElementModes::WRITE | embed::ElementModes::TRUNCATE; INetURLObject aURLObj( rURL ); bool bToStorage = aURLObj.GetProtocol() == INetProtocol::NotValid; // a relative path bool bSaveAsStorage = xTable->getElementType() == cppu::UnoType::get(); if( pOptName ) *pOptName = rURL; try { uno::Reference< uno::XComponentContext> xContext( ::comphelper::getProcessComponentContext() ); uno::Reference< xml::sax::XWriter > xWriter = xml::sax::Writer::create( xContext ); uno::Reference < io::XStream > xStream; uno::Reference < io::XOutputStream > xOut; uno::Reference xSubStorage; uno::Reference< XGraphicObjectResolver > xGrfResolver; uno::Reference xHandler( xWriter, uno::UNO_QUERY ); if( !bToStorage || !xStorage.is() ) { // local URL -> SfxMedium route if( bSaveAsStorage ) xSubStorage = ::comphelper::OStorageHelper::GetStorageFromURL( rURL, eCreate ); else { pMedium = new SfxMedium( rURL, StreamMode::WRITE | StreamMode::TRUNC ); SvStream* pStream = pMedium->GetOutStream(); if( !pStream ) { OSL_FAIL( "no output stream!" ); return false; } xOut = new utl::OOutputStreamWrapper( *pStream ); } } else // save into the xSubStorage { OUString aPath = rURL; if( bSaveAsStorage ) { try { xSubStorage = xStorage->openStorageElement( aPath, eCreate ); } catch (uno::Exception &) { OSL_FAIL( "no output storage!" ); return false; } } else { aPath += ".xml"; try { xStream = xStorage->openStreamElement( aPath, eCreate ); if( !xStream.is() ) return false; initializeStreamMetadata( xStream ); xOut = xStream->getOutputStream(); } catch (uno::Exception &) { OSL_FAIL( "no output stream!" ); return false; } if( pOptName ) *pOptName = aPath; } } if( !xOut.is() && xSubStorage.is() ) createStorageStream( &xOut, &pGraphicHelper, xSubStorage ); if( !xOut.is() ) return false; uno::Reference xMetaSrc( xWriter, uno::UNO_QUERY ); xMetaSrc->setOutputStream( xOut ); if( pGraphicHelper ) xGrfResolver = pGraphicHelper; // Finally do the export const OUString aName; uno::Reference< SvxXMLXTableExportComponent > xExporter( new SvxXMLXTableExportComponent( xContext, aName, xHandler, xTable, xGrfResolver ) ); bRet = xExporter->exportTable(); if( pGraphicHelper ) SvXMLGraphicHelper::Destroy( pGraphicHelper ); if( xSubStorage.is() ) { uno::Reference< XTransactedObject > xTrans( xSubStorage, UNO_QUERY ); if( xTrans.is() ) xTrans->commit(); uno::Reference< XComponent > xComp( xSubStorage, UNO_QUERY ); if( xComp.is() ) xSubStorage->dispose(); } } catch( uno::Exception& ) { bRet = false; } if( pMedium ) { pMedium->Commit(); delete pMedium; } return bRet; } bool SvxXMLXTableExportComponent::exportTable() throw() { bool bRet = false; try { GetDocHandler()->startDocument(); addChaffWhenEncryptedStorage(); // export namespaces sal_uInt16 nPos = GetNamespaceMap().GetFirstKey(); while( USHRT_MAX != nPos ) { GetAttrList().AddAttribute( GetNamespaceMap().GetAttrNameByKey( nPos ), GetNamespaceMap().GetNameByKey( nPos ) ); nPos = GetNamespaceMap().GetNextKey( nPos ); } do { if( !mxTable.is() ) break; char const* pEleName; Type aExportType = mxTable->getElementType(); std::unique_ptr pExporter; if( aExportType == cppu::UnoType::get() ) { pExporter.reset(new SvxXMLColorEntryExporter(*this)); pEleName = "color-table"; } else if( aExportType == cppu::UnoType< drawing::PolyPolygonBezierCoords >::get() ) { pExporter.reset(new SvxXMLLineEndEntryExporter(*this)); pEleName = "marker-table"; } else if( aExportType == cppu::UnoType< drawing::LineDash >::get() ) { pExporter.reset(new SvxXMLDashEntryExporter(*this)); pEleName = "dash-table"; } else if( aExportType == cppu::UnoType< drawing::Hatch >::get() ) { pExporter.reset(new SvxXMLHatchEntryExporter(*this)); pEleName = "hatch-table"; } else if( aExportType == cppu::UnoType< awt::Gradient >::get() ) { pExporter.reset(new SvxXMLGradientEntryExporter(*this)); pEleName = "gradient-table"; } else if( aExportType == cppu::UnoType::get()) { pExporter.reset(new SvxXMLBitmapEntryExporter(*this)); pEleName = "bitmap-table"; } else { OSL_FAIL( "unknown type for export"); break; } SvXMLElementExport aElem( *this, XML_NAMESPACE_OOO, pEleName, true, true ); Sequence< OUString > aNames = mxTable->getElementNames(); const sal_Int32 nCount = aNames.getLength(); const OUString* pNames = aNames.getConstArray(); Any aAny; sal_Int32 nIndex; for( nIndex = 0; nIndex < nCount; nIndex++, pNames++ ) { aAny = mxTable->getByName( *pNames ); pExporter->exportEntry( *pNames, aAny ); } bRet = true; } while(false); GetDocHandler()->endDocument(); } catch( Exception const& ) { bRet = false; } return bRet; } // methods without content: void SvxXMLXTableExportComponent::ExportAutoStyles_() {} void SvxXMLXTableExportComponent::ExportMasterStyles_() {} void SvxXMLXTableExportComponent::ExportContent_() {} SvxXMLTableEntryExporter::~SvxXMLTableEntryExporter() { } SvxXMLColorEntryExporter::SvxXMLColorEntryExporter( SvXMLExport& rExport ) : SvxXMLTableEntryExporter( rExport ) { } SvxXMLColorEntryExporter::~SvxXMLColorEntryExporter() { } void SvxXMLColorEntryExporter::exportEntry( const OUString& rStrName, const Any& rValue ) { mrExport.AddAttribute( XML_NAMESPACE_DRAW, XML_NAME, rStrName ); sal_Int32 nColor = 0; rValue >>= nColor; OUStringBuffer aOut; ::sax::Converter::convertColor( aOut, nColor ); mrExport.AddAttribute( XML_NAMESPACE_DRAW, XML_COLOR, aOut.makeStringAndClear() ); SvXMLElementExport aElem( mrExport, XML_NAMESPACE_DRAW, XML_COLOR, true, true ); } SvxXMLLineEndEntryExporter::SvxXMLLineEndEntryExporter( SvXMLExport& rExport ) : SvxXMLTableEntryExporter( rExport ), maMarkerStyle( rExport ) { } SvxXMLLineEndEntryExporter::~SvxXMLLineEndEntryExporter() { } void SvxXMLLineEndEntryExporter::exportEntry( const OUString& rStrName, const Any& rValue ) { maMarkerStyle.exportXML( rStrName, rValue ); } SvxXMLDashEntryExporter::SvxXMLDashEntryExporter( SvXMLExport& rExport ) : SvxXMLTableEntryExporter( rExport ), maDashStyle( rExport ) { } SvxXMLDashEntryExporter::~SvxXMLDashEntryExporter() { } void SvxXMLDashEntryExporter::exportEntry( const OUString& rStrName, const Any& rValue ) { maDashStyle.exportXML( rStrName, rValue ); } SvxXMLHatchEntryExporter::SvxXMLHatchEntryExporter( SvXMLExport& rExport ) : SvxXMLTableEntryExporter( rExport ), maHatchStyle( rExport ) { } SvxXMLHatchEntryExporter::~SvxXMLHatchEntryExporter() { } void SvxXMLHatchEntryExporter::exportEntry( const OUString& rStrName, const Any& rValue ) { maHatchStyle.exportXML( rStrName, rValue ); } SvxXMLGradientEntryExporter::SvxXMLGradientEntryExporter( SvXMLExport& rExport ) : SvxXMLTableEntryExporter( rExport ), maGradientStyle( rExport ) { } SvxXMLGradientEntryExporter::~SvxXMLGradientEntryExporter() { } void SvxXMLGradientEntryExporter::exportEntry( const OUString& rStrName, const Any& rValue ) { maGradientStyle.exportXML( rStrName, rValue ); } SvxXMLBitmapEntryExporter::SvxXMLBitmapEntryExporter( SvXMLExport& rExport ) : SvxXMLTableEntryExporter( rExport ) { } SvxXMLBitmapEntryExporter::~SvxXMLBitmapEntryExporter() { } void SvxXMLBitmapEntryExporter::exportEntry( const OUString& rStrName, const Any& rValue ) { maImageStyle.exportXML( rStrName, rValue, mrExport ); } /* vim:set shiftwidth=4 softtabstop=4 expandtab: */