tdf#131098 docx export: write fill property of graphic
[LibreOffice.git] / oox / source / helper / storagebase.cxx
blob46466036316f395af8c715579e8276c8fc21e3ea
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 <oox/helper/storagebase.hxx>
22 #include <com/sun/star/io/XStream.hpp>
23 #include <osl/diagnose.h>
24 #include <rtl/ustrbuf.hxx>
25 #include <oox/helper/binaryinputstream.hxx>
26 #include <oox/helper/binaryoutputstream.hxx>
27 #include <utility>
29 namespace oox {
31 using namespace ::com::sun::star::embed;
32 using namespace ::com::sun::star::io;
33 using namespace ::com::sun::star::uno;
35 namespace {
37 void lclSplitFirstElement( OUString& orElement, OUString& orRemainder, const OUString& _aFullName )
39 OUString aFullName = _aFullName;
40 sal_Int32 nSlashPos = aFullName.indexOf( '/' );
42 // strip leading slashes
43 while( nSlashPos == 0 )
45 aFullName = aFullName.copy(1);
46 nSlashPos = aFullName.indexOf( '/' );
49 if( (0 <= nSlashPos) && (nSlashPos < aFullName.getLength()) )
51 orElement = aFullName.copy( 0, nSlashPos );
52 orRemainder = aFullName.copy( nSlashPos + 1 );
54 else
56 orElement = aFullName;
60 } // namespace
62 StorageBase::StorageBase( const Reference< XInputStream >& rxInStream, bool bBaseStreamAccess ) :
63 mxInStream( rxInStream ),
64 mbBaseStreamAccess( bBaseStreamAccess ),
65 mbReadOnly( true )
67 OSL_ENSURE( mxInStream.is(), "StorageBase::StorageBase - missing base input stream" );
70 StorageBase::StorageBase( const Reference< XStream >& rxOutStream, bool bBaseStreamAccess ) :
71 mxOutStream( rxOutStream ),
72 mbBaseStreamAccess( bBaseStreamAccess ),
73 mbReadOnly( false )
75 OSL_ENSURE( mxOutStream.is(), "StorageBase::StorageBase - missing base output stream" );
78 StorageBase::StorageBase( const StorageBase& rParentStorage, OUString aStorageName, bool bReadOnly ) :
79 maParentPath( rParentStorage.getPath() ),
80 maStorageName(std::move( aStorageName )),
81 mbBaseStreamAccess( false ),
82 mbReadOnly( bReadOnly )
86 StorageBase::~StorageBase()
90 bool StorageBase::isStorage() const
92 return implIsStorage();
95 bool StorageBase::isRootStorage() const
97 return implIsStorage() && maStorageName.isEmpty();
100 Reference< XStorage > StorageBase::getXStorage() const
102 return implGetXStorage();
105 OUString StorageBase::getPath() const
107 OUStringBuffer aBuffer( maParentPath );
108 if( !aBuffer.isEmpty() )
109 aBuffer.append( '/' );
110 aBuffer.append( maStorageName );
111 return aBuffer.makeStringAndClear();
114 void StorageBase::getElementNames( ::std::vector< OUString >& orElementNames ) const
116 orElementNames.clear();
117 implGetElementNames( orElementNames );
120 StorageRef StorageBase::openSubStorage( const OUString& rStorageName, bool bCreateMissing )
122 StorageRef xSubStorage;
123 OSL_ENSURE( !bCreateMissing || !mbReadOnly, "StorageBase::openSubStorage - cannot create substorage in read-only mode" );
124 if( !bCreateMissing || !mbReadOnly )
126 OUString aElement, aRemainder;
127 lclSplitFirstElement( aElement, aRemainder, rStorageName );
128 if( !aElement.isEmpty() )
129 xSubStorage = getSubStorage( aElement, bCreateMissing );
130 if( xSubStorage && !aRemainder.isEmpty() )
131 xSubStorage = xSubStorage->openSubStorage( aRemainder, bCreateMissing );
133 return xSubStorage;
136 Reference< XInputStream > StorageBase::openInputStream( const OUString& rStreamName )
138 Reference< XInputStream > xInStream;
139 OUString aElement, aRemainder;
140 lclSplitFirstElement( aElement, aRemainder, rStreamName );
141 if( !aElement.isEmpty() )
143 if( !aRemainder.isEmpty() )
145 StorageRef xSubStorage = getSubStorage( aElement, false );
146 if( xSubStorage )
147 xInStream = xSubStorage->openInputStream( aRemainder );
149 else
151 xInStream = implOpenInputStream( aElement );
154 else if( mbBaseStreamAccess )
156 xInStream = mxInStream;
158 return xInStream;
161 Reference< XOutputStream > StorageBase::openOutputStream( const OUString& rStreamName )
163 Reference< XOutputStream > xOutStream;
164 OSL_ENSURE( !mbReadOnly, "StorageBase::openOutputStream - cannot create output stream in read-only mode" );
165 if( !mbReadOnly )
167 OUString aElement, aRemainder;
168 lclSplitFirstElement( aElement, aRemainder, rStreamName );
169 if( !aElement.isEmpty() )
171 if( !aRemainder.isEmpty() )
173 StorageRef xSubStorage = getSubStorage( aElement, true );
174 if( xSubStorage )
175 xOutStream = xSubStorage->openOutputStream( aRemainder );
177 else
179 xOutStream = implOpenOutputStream( aElement );
182 else if( mbBaseStreamAccess )
184 xOutStream = mxOutStream->getOutputStream();
187 return xOutStream;
190 void StorageBase::copyToStorage( StorageBase& rDestStrg, const OUString& rElementName )
192 OSL_ENSURE( rDestStrg.isStorage() && !rDestStrg.isReadOnly(), "StorageBase::copyToStorage - invalid destination" );
193 OSL_ENSURE( !rElementName.isEmpty(), "StorageBase::copyToStorage - invalid element name" );
194 if( !rDestStrg.isStorage() || rDestStrg.isReadOnly() || rElementName.isEmpty() )
195 return;
197 StorageRef xSubStrg = openSubStorage( rElementName, false );
198 if( xSubStrg )
200 StorageRef xDestSubStrg = rDestStrg.openSubStorage( rElementName, true );
201 if( xDestSubStrg )
202 xSubStrg->copyStorageToStorage( *xDestSubStrg );
204 else
206 Reference< XInputStream > xInStrm = openInputStream( rElementName );
207 if( xInStrm.is() )
209 Reference< XOutputStream > xOutStrm = rDestStrg.openOutputStream( rElementName );
210 if( xOutStrm.is() )
212 BinaryXInputStream aInStrm( xInStrm, true );
213 BinaryXOutputStream aOutStrm( xOutStrm, true );
214 aInStrm.copyToStream( aOutStrm );
220 void StorageBase::copyStorageToStorage( StorageBase& rDestStrg )
222 OSL_ENSURE( rDestStrg.isStorage() && !rDestStrg.isReadOnly(), "StorageBase::copyToStorage - invalid destination" );
223 if( rDestStrg.isStorage() && !rDestStrg.isReadOnly() )
225 ::std::vector< OUString > aElements;
226 getElementNames( aElements );
227 for (auto const& elem : aElements)
228 copyToStorage(rDestStrg, elem);
232 void StorageBase::commit()
234 OSL_ENSURE( !mbReadOnly, "StorageBase::commit - cannot commit in read-only mode" );
235 if( !mbReadOnly )
237 // commit all open substorages
238 maSubStorages.forEachMem( &StorageBase::commit );
239 // commit this storage
240 implCommit();
244 // private --------------------------------------------------------------------
246 StorageRef StorageBase::getSubStorage( const OUString& rElementName, bool bCreateMissing )
248 StorageRef& rxSubStrg = maSubStorages[ rElementName ];
249 if( !rxSubStrg )
250 rxSubStrg = implOpenSubStorage( rElementName, bCreateMissing );
251 return rxSubStrg;
254 } // namespace oox
256 /* vim:set shiftwidth=4 softtabstop=4 expandtab: */