xref: /aoo4110/main/oox/source/ole/vbaproject.cxx (revision b1cdbd2c)
1*b1cdbd2cSJim Jagielski /**************************************************************
2*b1cdbd2cSJim Jagielski  *
3*b1cdbd2cSJim Jagielski  * Licensed to the Apache Software Foundation (ASF) under one
4*b1cdbd2cSJim Jagielski  * or more contributor license agreements.  See the NOTICE file
5*b1cdbd2cSJim Jagielski  * distributed with this work for additional information
6*b1cdbd2cSJim Jagielski  * regarding copyright ownership.  The ASF licenses this file
7*b1cdbd2cSJim Jagielski  * to you under the Apache License, Version 2.0 (the
8*b1cdbd2cSJim Jagielski  * "License"); you may not use this file except in compliance
9*b1cdbd2cSJim Jagielski  * with the License.  You may obtain a copy of the License at
10*b1cdbd2cSJim Jagielski  *
11*b1cdbd2cSJim Jagielski  *   http://www.apache.org/licenses/LICENSE-2.0
12*b1cdbd2cSJim Jagielski  *
13*b1cdbd2cSJim Jagielski  * Unless required by applicable law or agreed to in writing,
14*b1cdbd2cSJim Jagielski  * software distributed under the License is distributed on an
15*b1cdbd2cSJim Jagielski  * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
16*b1cdbd2cSJim Jagielski  * KIND, either express or implied.  See the License for the
17*b1cdbd2cSJim Jagielski  * specific language governing permissions and limitations
18*b1cdbd2cSJim Jagielski  * under the License.
19*b1cdbd2cSJim Jagielski  *
20*b1cdbd2cSJim Jagielski  *************************************************************/
21*b1cdbd2cSJim Jagielski 
22*b1cdbd2cSJim Jagielski 
23*b1cdbd2cSJim Jagielski 
24*b1cdbd2cSJim Jagielski #include "oox/ole/vbaproject.hxx"
25*b1cdbd2cSJim Jagielski 
26*b1cdbd2cSJim Jagielski #include <com/sun/star/document/XStorageBasedDocument.hpp>
27*b1cdbd2cSJim Jagielski #include <com/sun/star/embed/ElementModes.hpp>
28*b1cdbd2cSJim Jagielski #include <com/sun/star/embed/XTransactedObject.hpp>
29*b1cdbd2cSJim Jagielski #include <com/sun/star/frame/XModel.hpp>
30*b1cdbd2cSJim Jagielski #include <com/sun/star/lang/XMultiServiceFactory.hpp>
31*b1cdbd2cSJim Jagielski #include <com/sun/star/script/ModuleType.hpp>
32*b1cdbd2cSJim Jagielski #include <com/sun/star/script/XLibraryContainer.hpp>
33*b1cdbd2cSJim Jagielski #include <com/sun/star/script/vba/XVBACompatibility.hpp>
34*b1cdbd2cSJim Jagielski #include <com/sun/star/script/vba/XVBAMacroResolver.hpp>
35*b1cdbd2cSJim Jagielski #include <com/sun/star/uno/XComponentContext.hpp>
36*b1cdbd2cSJim Jagielski #include <comphelper/configurationhelper.hxx>
37*b1cdbd2cSJim Jagielski #include <comphelper/string.hxx>
38*b1cdbd2cSJim Jagielski #include <rtl/tencinfo.h>
39*b1cdbd2cSJim Jagielski #include <rtl/ustrbuf.h>
40*b1cdbd2cSJim Jagielski #include "oox/helper/binaryinputstream.hxx"
41*b1cdbd2cSJim Jagielski #include "oox/helper/containerhelper.hxx"
42*b1cdbd2cSJim Jagielski #include "oox/helper/propertyset.hxx"
43*b1cdbd2cSJim Jagielski #include "oox/helper/textinputstream.hxx"
44*b1cdbd2cSJim Jagielski #include "oox/ole/olestorage.hxx"
45*b1cdbd2cSJim Jagielski #include "oox/ole/vbacontrol.hxx"
46*b1cdbd2cSJim Jagielski #include "oox/ole/vbahelper.hxx"
47*b1cdbd2cSJim Jagielski #include "oox/ole/vbainputstream.hxx"
48*b1cdbd2cSJim Jagielski #include "oox/ole/vbamodule.hxx"
49*b1cdbd2cSJim Jagielski 
50*b1cdbd2cSJim Jagielski namespace oox {
51*b1cdbd2cSJim Jagielski namespace ole {
52*b1cdbd2cSJim Jagielski 
53*b1cdbd2cSJim Jagielski // ============================================================================
54*b1cdbd2cSJim Jagielski 
55*b1cdbd2cSJim Jagielski using namespace ::com::sun::star::container;
56*b1cdbd2cSJim Jagielski using namespace ::com::sun::star::document;
57*b1cdbd2cSJim Jagielski using namespace ::com::sun::star::embed;
58*b1cdbd2cSJim Jagielski using namespace ::com::sun::star::frame;
59*b1cdbd2cSJim Jagielski using namespace ::com::sun::star::io;
60*b1cdbd2cSJim Jagielski using namespace ::com::sun::star::lang;
61*b1cdbd2cSJim Jagielski using namespace ::com::sun::star::script;
62*b1cdbd2cSJim Jagielski using namespace ::com::sun::star::script::vba;
63*b1cdbd2cSJim Jagielski using namespace ::com::sun::star::uno;
64*b1cdbd2cSJim Jagielski 
65*b1cdbd2cSJim Jagielski using ::comphelper::ConfigurationHelper;
66*b1cdbd2cSJim Jagielski using ::rtl::OUString;
67*b1cdbd2cSJim Jagielski using ::rtl::OUStringBuffer;
68*b1cdbd2cSJim Jagielski 
69*b1cdbd2cSJim Jagielski // ============================================================================
70*b1cdbd2cSJim Jagielski 
71*b1cdbd2cSJim Jagielski namespace {
72*b1cdbd2cSJim Jagielski 
lclReadConfigItem(const Reference<XInterface> & rxConfigAccess,const OUString & rItemName)73*b1cdbd2cSJim Jagielski bool lclReadConfigItem( const Reference< XInterface >& rxConfigAccess, const OUString& rItemName )
74*b1cdbd2cSJim Jagielski {
75*b1cdbd2cSJim Jagielski     // some applications do not support all configuration items, assume 'false' in this case
76*b1cdbd2cSJim Jagielski     try
77*b1cdbd2cSJim Jagielski     {
78*b1cdbd2cSJim Jagielski         Any aItem = ConfigurationHelper::readRelativeKey( rxConfigAccess, CREATE_OUSTRING( "Filter/Import/VBA" ), rItemName );
79*b1cdbd2cSJim Jagielski         return aItem.has< bool >() && aItem.get< bool >();
80*b1cdbd2cSJim Jagielski     }
81*b1cdbd2cSJim Jagielski     catch( Exception& )
82*b1cdbd2cSJim Jagielski     {
83*b1cdbd2cSJim Jagielski     }
84*b1cdbd2cSJim Jagielski     return false;
85*b1cdbd2cSJim Jagielski }
86*b1cdbd2cSJim Jagielski 
87*b1cdbd2cSJim Jagielski } // namespace
88*b1cdbd2cSJim Jagielski 
89*b1cdbd2cSJim Jagielski // ----------------------------------------------------------------------------
90*b1cdbd2cSJim Jagielski 
VbaFilterConfig(const Reference<XComponentContext> & rxContext,const OUString & rConfigCompName)91*b1cdbd2cSJim Jagielski VbaFilterConfig::VbaFilterConfig( const Reference< XComponentContext >& rxContext, const OUString& rConfigCompName )
92*b1cdbd2cSJim Jagielski {
93*b1cdbd2cSJim Jagielski     OSL_ENSURE( rxContext.is(), "VbaFilterConfig::VbaFilterConfig - missing component context" );
94*b1cdbd2cSJim Jagielski     if( rxContext.is() ) try
95*b1cdbd2cSJim Jagielski     {
96*b1cdbd2cSJim Jagielski         OSL_ENSURE( rConfigCompName.getLength() > 0, "VbaFilterConfig::VbaFilterConfig - invalid configuration component name" );
97*b1cdbd2cSJim Jagielski         OUString aConfigPackage = CREATE_OUSTRING( "org.openoffice.Office." ) + rConfigCompName;
98*b1cdbd2cSJim Jagielski         Reference< XMultiServiceFactory > xFactory( rxContext->getServiceManager(), UNO_QUERY_THROW );
99*b1cdbd2cSJim Jagielski         mxConfigAccess = ConfigurationHelper::openConfig( xFactory, aConfigPackage, ConfigurationHelper::E_READONLY );
100*b1cdbd2cSJim Jagielski     }
101*b1cdbd2cSJim Jagielski     catch( Exception& )
102*b1cdbd2cSJim Jagielski     {
103*b1cdbd2cSJim Jagielski     }
104*b1cdbd2cSJim Jagielski     OSL_ENSURE( mxConfigAccess.is(), "VbaFilterConfig::VbaFilterConfig - cannot open configuration" );
105*b1cdbd2cSJim Jagielski }
106*b1cdbd2cSJim Jagielski 
~VbaFilterConfig()107*b1cdbd2cSJim Jagielski VbaFilterConfig::~VbaFilterConfig()
108*b1cdbd2cSJim Jagielski {
109*b1cdbd2cSJim Jagielski }
110*b1cdbd2cSJim Jagielski 
isImportVba() const111*b1cdbd2cSJim Jagielski bool VbaFilterConfig::isImportVba() const
112*b1cdbd2cSJim Jagielski {
113*b1cdbd2cSJim Jagielski     return lclReadConfigItem( mxConfigAccess, CREATE_OUSTRING( "Load" ) );
114*b1cdbd2cSJim Jagielski }
115*b1cdbd2cSJim Jagielski 
isImportVbaExecutable() const116*b1cdbd2cSJim Jagielski bool VbaFilterConfig::isImportVbaExecutable() const
117*b1cdbd2cSJim Jagielski {
118*b1cdbd2cSJim Jagielski     return lclReadConfigItem( mxConfigAccess, CREATE_OUSTRING( "Executable" ) );
119*b1cdbd2cSJim Jagielski }
120*b1cdbd2cSJim Jagielski 
isExportVba() const121*b1cdbd2cSJim Jagielski bool VbaFilterConfig::isExportVba() const
122*b1cdbd2cSJim Jagielski {
123*b1cdbd2cSJim Jagielski     return lclReadConfigItem( mxConfigAccess, CREATE_OUSTRING( "Save" ) );
124*b1cdbd2cSJim Jagielski }
125*b1cdbd2cSJim Jagielski 
126*b1cdbd2cSJim Jagielski // ============================================================================
127*b1cdbd2cSJim Jagielski 
VbaMacroAttacherBase(const OUString & rMacroName)128*b1cdbd2cSJim Jagielski VbaMacroAttacherBase::VbaMacroAttacherBase( const OUString& rMacroName ) :
129*b1cdbd2cSJim Jagielski     maMacroName( rMacroName )
130*b1cdbd2cSJim Jagielski {
131*b1cdbd2cSJim Jagielski     OSL_ENSURE( maMacroName.getLength() > 0, "VbaMacroAttacherBase::VbaMacroAttacherBase - empty macro name" );
132*b1cdbd2cSJim Jagielski }
133*b1cdbd2cSJim Jagielski 
~VbaMacroAttacherBase()134*b1cdbd2cSJim Jagielski VbaMacroAttacherBase::~VbaMacroAttacherBase()
135*b1cdbd2cSJim Jagielski {
136*b1cdbd2cSJim Jagielski }
137*b1cdbd2cSJim Jagielski 
resolveAndAttachMacro(const Reference<XVBAMacroResolver> & rxResolver)138*b1cdbd2cSJim Jagielski void VbaMacroAttacherBase::resolveAndAttachMacro( const Reference< XVBAMacroResolver >& rxResolver )
139*b1cdbd2cSJim Jagielski {
140*b1cdbd2cSJim Jagielski     try
141*b1cdbd2cSJim Jagielski     {
142*b1cdbd2cSJim Jagielski         attachMacro( rxResolver->resolveVBAMacroToScriptURL( maMacroName ) );
143*b1cdbd2cSJim Jagielski     }
144*b1cdbd2cSJim Jagielski     catch( Exception& )
145*b1cdbd2cSJim Jagielski     {
146*b1cdbd2cSJim Jagielski     }
147*b1cdbd2cSJim Jagielski }
148*b1cdbd2cSJim Jagielski 
149*b1cdbd2cSJim Jagielski // ============================================================================
150*b1cdbd2cSJim Jagielski 
VbaProject(const Reference<XComponentContext> & rxContext,const Reference<XModel> & rxDocModel,const OUString & rConfigCompName)151*b1cdbd2cSJim Jagielski VbaProject::VbaProject( const Reference< XComponentContext >& rxContext,
152*b1cdbd2cSJim Jagielski         const Reference< XModel >& rxDocModel, const OUString& rConfigCompName ) :
153*b1cdbd2cSJim Jagielski     VbaFilterConfig( rxContext, rConfigCompName ),
154*b1cdbd2cSJim Jagielski     mxContext( rxContext ),
155*b1cdbd2cSJim Jagielski     mxDocModel( rxDocModel ),
156*b1cdbd2cSJim Jagielski     maPrjName( CREATE_OUSTRING( "Standard" ) )
157*b1cdbd2cSJim Jagielski {
158*b1cdbd2cSJim Jagielski     OSL_ENSURE( mxContext.is(), "VbaProject::VbaProject - missing component context" );
159*b1cdbd2cSJim Jagielski     OSL_ENSURE( mxDocModel.is(), "VbaProject::VbaProject - missing document model" );
160*b1cdbd2cSJim Jagielski     mxBasicLib = openLibrary( PROP_BasicLibraries, false );
161*b1cdbd2cSJim Jagielski     mxDialogLib = openLibrary( PROP_DialogLibraries, false );
162*b1cdbd2cSJim Jagielski }
163*b1cdbd2cSJim Jagielski 
~VbaProject()164*b1cdbd2cSJim Jagielski VbaProject::~VbaProject()
165*b1cdbd2cSJim Jagielski {
166*b1cdbd2cSJim Jagielski }
167*b1cdbd2cSJim Jagielski 
importVbaProject(StorageBase & rVbaPrjStrg,const GraphicHelper & rGraphicHelper,bool bDefaultColorBgr)168*b1cdbd2cSJim Jagielski void VbaProject::importVbaProject( StorageBase& rVbaPrjStrg, const GraphicHelper& rGraphicHelper, bool bDefaultColorBgr )
169*b1cdbd2cSJim Jagielski {
170*b1cdbd2cSJim Jagielski     if( rVbaPrjStrg.isStorage() )
171*b1cdbd2cSJim Jagielski     {
172*b1cdbd2cSJim Jagielski         // load the code modules and forms
173*b1cdbd2cSJim Jagielski         if( isImportVba() )
174*b1cdbd2cSJim Jagielski             importVba( rVbaPrjStrg, rGraphicHelper, bDefaultColorBgr );
175*b1cdbd2cSJim Jagielski         // copy entire storage into model
176*b1cdbd2cSJim Jagielski         if( isExportVba() )
177*b1cdbd2cSJim Jagielski             copyStorage( rVbaPrjStrg );
178*b1cdbd2cSJim Jagielski     }
179*b1cdbd2cSJim Jagielski }
180*b1cdbd2cSJim Jagielski 
registerMacroAttacher(const VbaMacroAttacherRef & rxAttacher)181*b1cdbd2cSJim Jagielski void VbaProject::registerMacroAttacher( const VbaMacroAttacherRef& rxAttacher )
182*b1cdbd2cSJim Jagielski {
183*b1cdbd2cSJim Jagielski     OSL_ENSURE( rxAttacher.get(), "VbaProject::registerMacroAttacher - unexpected empty reference" );
184*b1cdbd2cSJim Jagielski     maMacroAttachers.push_back( rxAttacher );
185*b1cdbd2cSJim Jagielski }
186*b1cdbd2cSJim Jagielski 
hasModules() const187*b1cdbd2cSJim Jagielski bool VbaProject::hasModules() const
188*b1cdbd2cSJim Jagielski {
189*b1cdbd2cSJim Jagielski     return mxBasicLib.is() && mxBasicLib->hasElements();
190*b1cdbd2cSJim Jagielski }
191*b1cdbd2cSJim Jagielski 
hasModule(const OUString & rModuleName) const192*b1cdbd2cSJim Jagielski bool VbaProject::hasModule( const OUString& rModuleName ) const
193*b1cdbd2cSJim Jagielski {
194*b1cdbd2cSJim Jagielski     return mxBasicLib.is() && mxBasicLib->hasByName( rModuleName );
195*b1cdbd2cSJim Jagielski }
196*b1cdbd2cSJim Jagielski 
hasDialogs() const197*b1cdbd2cSJim Jagielski bool VbaProject::hasDialogs() const
198*b1cdbd2cSJim Jagielski {
199*b1cdbd2cSJim Jagielski     return mxDialogLib.is() && mxDialogLib->hasElements();
200*b1cdbd2cSJim Jagielski }
201*b1cdbd2cSJim Jagielski 
hasDialog(const OUString & rDialogName) const202*b1cdbd2cSJim Jagielski bool VbaProject::hasDialog( const OUString& rDialogName ) const
203*b1cdbd2cSJim Jagielski {
204*b1cdbd2cSJim Jagielski     return mxDialogLib.is() && mxDialogLib->hasByName( rDialogName );
205*b1cdbd2cSJim Jagielski }
206*b1cdbd2cSJim Jagielski 
207*b1cdbd2cSJim Jagielski // protected ------------------------------------------------------------------
208*b1cdbd2cSJim Jagielski 
addDummyModule(const OUString & rName,sal_Int32 nType)209*b1cdbd2cSJim Jagielski void VbaProject::addDummyModule( const OUString& rName, sal_Int32 nType )
210*b1cdbd2cSJim Jagielski {
211*b1cdbd2cSJim Jagielski     OSL_ENSURE( rName.getLength() > 0, "VbaProject::addDummyModule - missing module name" );
212*b1cdbd2cSJim Jagielski     maDummyModules[ rName ] = nType;
213*b1cdbd2cSJim Jagielski }
214*b1cdbd2cSJim Jagielski 
prepareImport()215*b1cdbd2cSJim Jagielski void VbaProject::prepareImport()
216*b1cdbd2cSJim Jagielski {
217*b1cdbd2cSJim Jagielski }
218*b1cdbd2cSJim Jagielski 
finalizeImport()219*b1cdbd2cSJim Jagielski void VbaProject::finalizeImport()
220*b1cdbd2cSJim Jagielski {
221*b1cdbd2cSJim Jagielski }
222*b1cdbd2cSJim Jagielski 
223*b1cdbd2cSJim Jagielski // private --------------------------------------------------------------------
224*b1cdbd2cSJim Jagielski 
getLibraryContainer(sal_Int32 nPropId)225*b1cdbd2cSJim Jagielski Reference< XLibraryContainer > VbaProject::getLibraryContainer( sal_Int32 nPropId )
226*b1cdbd2cSJim Jagielski {
227*b1cdbd2cSJim Jagielski     PropertySet aDocProp( mxDocModel );
228*b1cdbd2cSJim Jagielski     Reference< XLibraryContainer > xLibContainer( aDocProp.getAnyProperty( nPropId ), UNO_QUERY );
229*b1cdbd2cSJim Jagielski     return xLibContainer;
230*b1cdbd2cSJim Jagielski }
231*b1cdbd2cSJim Jagielski 
openLibrary(sal_Int32 nPropId,bool bCreateMissing)232*b1cdbd2cSJim Jagielski Reference< XNameContainer > VbaProject::openLibrary( sal_Int32 nPropId, bool bCreateMissing )
233*b1cdbd2cSJim Jagielski {
234*b1cdbd2cSJim Jagielski     Reference< XNameContainer > xLibrary;
235*b1cdbd2cSJim Jagielski     try
236*b1cdbd2cSJim Jagielski     {
237*b1cdbd2cSJim Jagielski         Reference< XLibraryContainer > xLibContainer( getLibraryContainer( nPropId ), UNO_SET_THROW );
238*b1cdbd2cSJim Jagielski         if( bCreateMissing && !xLibContainer->hasByName( CREATE_OUSTRING( "Standard" ) /*maPrjName*/ ) )
239*b1cdbd2cSJim Jagielski             xLibContainer->createLibrary( CREATE_OUSTRING( "Standard" ) /*maPrjName*/ );
240*b1cdbd2cSJim Jagielski         xLibrary.set( xLibContainer->getByName( CREATE_OUSTRING( "Standard" ) /*maPrjName*/ ), UNO_QUERY_THROW );
241*b1cdbd2cSJim Jagielski     }
242*b1cdbd2cSJim Jagielski     catch( Exception& )
243*b1cdbd2cSJim Jagielski     {
244*b1cdbd2cSJim Jagielski     }
245*b1cdbd2cSJim Jagielski     OSL_ENSURE( !bCreateMissing || xLibrary.is(), "VbaProject::openLibrary - cannot create library" );
246*b1cdbd2cSJim Jagielski     return xLibrary;
247*b1cdbd2cSJim Jagielski }
248*b1cdbd2cSJim Jagielski 
createBasicLibrary()249*b1cdbd2cSJim Jagielski Reference< XNameContainer > VbaProject::createBasicLibrary()
250*b1cdbd2cSJim Jagielski {
251*b1cdbd2cSJim Jagielski     if( !mxBasicLib.is() )
252*b1cdbd2cSJim Jagielski         mxBasicLib = openLibrary( PROP_BasicLibraries, true );
253*b1cdbd2cSJim Jagielski     return mxBasicLib;
254*b1cdbd2cSJim Jagielski }
255*b1cdbd2cSJim Jagielski 
createDialogLibrary()256*b1cdbd2cSJim Jagielski Reference< XNameContainer > VbaProject::createDialogLibrary()
257*b1cdbd2cSJim Jagielski {
258*b1cdbd2cSJim Jagielski     if( !mxDialogLib.is() )
259*b1cdbd2cSJim Jagielski         mxDialogLib = openLibrary( PROP_DialogLibraries, true );
260*b1cdbd2cSJim Jagielski     return mxDialogLib;
261*b1cdbd2cSJim Jagielski }
262*b1cdbd2cSJim Jagielski 
importVba(StorageBase & rVbaPrjStrg,const GraphicHelper & rGraphicHelper,bool bDefaultColorBgr)263*b1cdbd2cSJim Jagielski void VbaProject::importVba( StorageBase& rVbaPrjStrg, const GraphicHelper& rGraphicHelper, bool bDefaultColorBgr )
264*b1cdbd2cSJim Jagielski {
265*b1cdbd2cSJim Jagielski     StorageRef xVbaStrg = rVbaPrjStrg.openSubStorage( CREATE_OUSTRING( "VBA" ), false );
266*b1cdbd2cSJim Jagielski     OSL_ENSURE( xVbaStrg.get(), "VbaProject::importVba - cannot open 'VBA' substorage" );
267*b1cdbd2cSJim Jagielski     if( !xVbaStrg )
268*b1cdbd2cSJim Jagielski         return;
269*b1cdbd2cSJim Jagielski 
270*b1cdbd2cSJim Jagielski     /*  Read the 'VBA/dir' stream which contains general settings of the VBA
271*b1cdbd2cSJim Jagielski         project such as the text encoding used throughout several streams, and
272*b1cdbd2cSJim Jagielski         a list of all code modules.
273*b1cdbd2cSJim Jagielski      */
274*b1cdbd2cSJim Jagielski     BinaryXInputStream aInStrm( xVbaStrg->openInputStream( CREATE_OUSTRING( "dir" ) ), true );
275*b1cdbd2cSJim Jagielski     // VbaInputStream implements decompression
276*b1cdbd2cSJim Jagielski     VbaInputStream aDirStrm( aInStrm );
277*b1cdbd2cSJim Jagielski     OSL_ENSURE( !aDirStrm.isEof(), "VbaProject::importVba - cannot open 'dir' stream" );
278*b1cdbd2cSJim Jagielski     if( aDirStrm.isEof() )
279*b1cdbd2cSJim Jagielski         return;
280*b1cdbd2cSJim Jagielski 
281*b1cdbd2cSJim Jagielski     // virtual call, derived classes may do some preparations
282*b1cdbd2cSJim Jagielski     prepareImport();
283*b1cdbd2cSJim Jagielski 
284*b1cdbd2cSJim Jagielski     // read all records of the directory
285*b1cdbd2cSJim Jagielski     rtl_TextEncoding eTextEnc = RTL_TEXTENCODING_MS_1252;
286*b1cdbd2cSJim Jagielski     sal_uInt16 nModuleCount = 0;
287*b1cdbd2cSJim Jagielski     bool bExecutable = isImportVbaExecutable();
288*b1cdbd2cSJim Jagielski 
289*b1cdbd2cSJim Jagielski     typedef RefMap< OUString, VbaModule > VbaModuleMap;
290*b1cdbd2cSJim Jagielski     VbaModuleMap aModules, aModulesByStrm;
291*b1cdbd2cSJim Jagielski 
292*b1cdbd2cSJim Jagielski     sal_uInt16 nRecId = 0;
293*b1cdbd2cSJim Jagielski     StreamDataSequence aRecData;
294*b1cdbd2cSJim Jagielski     while( VbaHelper::readDirRecord( nRecId, aRecData, aDirStrm ) && (nRecId != VBA_ID_PROJECTEND) )
295*b1cdbd2cSJim Jagielski     {
296*b1cdbd2cSJim Jagielski         // create record stream object from imported record data
297*b1cdbd2cSJim Jagielski         SequenceInputStream aRecStrm( aRecData );
298*b1cdbd2cSJim Jagielski         sal_Int32 nRecSize = aRecData.getLength();
299*b1cdbd2cSJim Jagielski         switch( nRecId )
300*b1cdbd2cSJim Jagielski         {
301*b1cdbd2cSJim Jagielski #define OOX_ENSURE_RECORDSIZE( cond ) OSL_ENSURE( cond, "VbaProject::importVba - invalid record size" )
302*b1cdbd2cSJim Jagielski             case VBA_ID_PROJECTCODEPAGE:
303*b1cdbd2cSJim Jagielski             {
304*b1cdbd2cSJim Jagielski                 OOX_ENSURE_RECORDSIZE( nRecSize == 2 );
305*b1cdbd2cSJim Jagielski                 OSL_ENSURE( aModules.empty(), "VbaProject::importVba - unexpected PROJECTCODEPAGE record" );
306*b1cdbd2cSJim Jagielski                 rtl_TextEncoding eNewTextEnc = rtl_getTextEncodingFromWindowsCodePage( aRecStrm.readuInt16() );
307*b1cdbd2cSJim Jagielski                 OSL_ENSURE( eNewTextEnc != RTL_TEXTENCODING_DONTKNOW, "VbaProject::importVba - unknown text encoding" );
308*b1cdbd2cSJim Jagielski                 if( eNewTextEnc != RTL_TEXTENCODING_DONTKNOW )
309*b1cdbd2cSJim Jagielski                     eTextEnc = eNewTextEnc;
310*b1cdbd2cSJim Jagielski             }
311*b1cdbd2cSJim Jagielski             break;
312*b1cdbd2cSJim Jagielski             case VBA_ID_PROJECTNAME:
313*b1cdbd2cSJim Jagielski             {
314*b1cdbd2cSJim Jagielski                 OUString aPrjName = aRecStrm.readCharArrayUC( nRecSize, eTextEnc );
315*b1cdbd2cSJim Jagielski                 OSL_ENSURE( aPrjName.getLength() > 0, "VbaProject::importVba - invalid project name" );
316*b1cdbd2cSJim Jagielski                 if( aPrjName.getLength() > 0 )
317*b1cdbd2cSJim Jagielski                     maPrjName = aPrjName;
318*b1cdbd2cSJim Jagielski             }
319*b1cdbd2cSJim Jagielski             break;
320*b1cdbd2cSJim Jagielski             case VBA_ID_PROJECTMODULES:
321*b1cdbd2cSJim Jagielski                 OOX_ENSURE_RECORDSIZE( nRecSize == 2 );
322*b1cdbd2cSJim Jagielski                 OSL_ENSURE( aModules.empty(), "VbaProject::importVba - unexpected PROJECTMODULES record" );
323*b1cdbd2cSJim Jagielski                 aRecStrm >> nModuleCount;
324*b1cdbd2cSJim Jagielski             break;
325*b1cdbd2cSJim Jagielski             case VBA_ID_MODULENAME:
326*b1cdbd2cSJim Jagielski             {
327*b1cdbd2cSJim Jagielski                 OUString aName = aRecStrm.readCharArrayUC( nRecSize, eTextEnc );
328*b1cdbd2cSJim Jagielski                 OSL_ENSURE( aName.getLength() > 0, "VbaProject::importVba - invalid module name" );
329*b1cdbd2cSJim Jagielski                 OSL_ENSURE( !aModules.has( aName ), "VbaProject::importVba - multiple modules with the same name" );
330*b1cdbd2cSJim Jagielski                 VbaModuleMap::mapped_type& rxModule = aModules[ aName ];
331*b1cdbd2cSJim Jagielski                 rxModule.reset( new VbaModule( mxContext, mxDocModel, aName, eTextEnc, bExecutable ) );
332*b1cdbd2cSJim Jagielski                 // read all remaining records until the MODULEEND record
333*b1cdbd2cSJim Jagielski                 rxModule->importDirRecords( aDirStrm );
334*b1cdbd2cSJim Jagielski                 OSL_ENSURE( !aModulesByStrm.has( rxModule->getStreamName() ), "VbaProject::importVba - multiple modules with the same stream name" );
335*b1cdbd2cSJim Jagielski                 aModulesByStrm[ rxModule->getStreamName() ] = rxModule;
336*b1cdbd2cSJim Jagielski             }
337*b1cdbd2cSJim Jagielski             break;
338*b1cdbd2cSJim Jagielski #undef OOX_ENSURE_RECORDSIZE
339*b1cdbd2cSJim Jagielski         }
340*b1cdbd2cSJim Jagielski     }
341*b1cdbd2cSJim Jagielski     OSL_ENSURE( nModuleCount == aModules.size(), "VbaProject::importVba - invalid module count" );
342*b1cdbd2cSJim Jagielski 
343*b1cdbd2cSJim Jagielski     /*  The directory does not contain the real type of the modules, it
344*b1cdbd2cSJim Jagielski         distinguishes only between 'procedural' and 'document' (the latter
345*b1cdbd2cSJim Jagielski         includes class and form modules). Now, the exact type of all modules
346*b1cdbd2cSJim Jagielski         will be read from the 'PROJECT' stream. It consists of text lines in
347*b1cdbd2cSJim Jagielski         'key=value' format which list the code modules by type.
348*b1cdbd2cSJim Jagielski 
349*b1cdbd2cSJim Jagielski         -   The line 'document=<modulename>/&HXXXXXXXX' declares document
350*b1cdbd2cSJim Jagielski             modules. These are attached to the Word document (usually called
351*b1cdbd2cSJim Jagielski             'ThisDocument'), the Excel workbook (usually called
352*b1cdbd2cSJim Jagielski             'ThisWorkbook'), or single Excel worksheets or chartsheets (usually
353*b1cdbd2cSJim Jagielski             called 'SheetX' or 'ChartX', X being a decimal number). Of course,
354*b1cdbd2cSJim Jagielski             users may rename all these modules. The slash character separates
355*b1cdbd2cSJim Jagielski             an automation server version number (hexadecimal 'XXXXXXXX') from
356*b1cdbd2cSJim Jagielski             the module name.
357*b1cdbd2cSJim Jagielski         -   The line 'Module=<modulename>' declares common procedural code
358*b1cdbd2cSJim Jagielski             modules.
359*b1cdbd2cSJim Jagielski         -   The line 'Class=<modulename>' declares a class module.
360*b1cdbd2cSJim Jagielski         -   The line 'BaseClass=<modulename>' declares a code module attached
361*b1cdbd2cSJim Jagielski             to a user form with the same name.
362*b1cdbd2cSJim Jagielski      */
363*b1cdbd2cSJim Jagielski     BinaryXInputStream aPrjStrm( rVbaPrjStrg.openInputStream( CREATE_OUSTRING( "PROJECT" ) ), true );
364*b1cdbd2cSJim Jagielski     OSL_ENSURE( !aPrjStrm.isEof(), "VbaProject::importVba - cannot open 'PROJECT' stream" );
365*b1cdbd2cSJim Jagielski     // do not exit if this stream does not exist, but proceed to load the modules below
366*b1cdbd2cSJim Jagielski     if( !aPrjStrm.isEof() )
367*b1cdbd2cSJim Jagielski     {
368*b1cdbd2cSJim Jagielski         TextInputStream aPrjTextStrm( mxContext, aPrjStrm, eTextEnc );
369*b1cdbd2cSJim Jagielski         OUString aKey, aValue;
370*b1cdbd2cSJim Jagielski         bool bExitLoop = false;
371*b1cdbd2cSJim Jagielski         while( !bExitLoop && !aPrjTextStrm.isEof() )
372*b1cdbd2cSJim Jagielski         {
373*b1cdbd2cSJim Jagielski             // read a text line from the stream
374*b1cdbd2cSJim Jagielski             OUString aLine = aPrjTextStrm.readLine().trim();
375*b1cdbd2cSJim Jagielski             sal_Int32 nLineLen = aLine.getLength();
376*b1cdbd2cSJim Jagielski             // exit if a subsection starts (section name is given in brackets)
377*b1cdbd2cSJim Jagielski             bExitLoop = (nLineLen >= 2) && (aLine[ 0 ] == '[') && (aLine[ nLineLen - 1 ] == ']');
378*b1cdbd2cSJim Jagielski             if( !bExitLoop && VbaHelper::extractKeyValue( aKey, aValue, aLine ) )
379*b1cdbd2cSJim Jagielski             {
380*b1cdbd2cSJim Jagielski                 sal_Int32 nType = ModuleType::UNKNOWN;
381*b1cdbd2cSJim Jagielski                 if( aKey.equalsIgnoreAsciiCaseAsciiL( RTL_CONSTASCII_STRINGPARAM( "Document" ) ) )
382*b1cdbd2cSJim Jagielski                 {
383*b1cdbd2cSJim Jagielski                     nType = ModuleType::DOCUMENT;
384*b1cdbd2cSJim Jagielski                     // strip automation server version from module names
385*b1cdbd2cSJim Jagielski                     sal_Int32 nSlashPos = aValue.indexOf( '/' );
386*b1cdbd2cSJim Jagielski                     if( nSlashPos >= 0 )
387*b1cdbd2cSJim Jagielski                         aValue = aValue.copy( 0, nSlashPos );
388*b1cdbd2cSJim Jagielski                 }
389*b1cdbd2cSJim Jagielski                 else if( aKey.equalsIgnoreAsciiCaseAsciiL( RTL_CONSTASCII_STRINGPARAM( "Module" ) ) )
390*b1cdbd2cSJim Jagielski                     nType = ModuleType::NORMAL;
391*b1cdbd2cSJim Jagielski                 else if( aKey.equalsIgnoreAsciiCaseAsciiL( RTL_CONSTASCII_STRINGPARAM( "Class" ) ) )
392*b1cdbd2cSJim Jagielski                     nType = ModuleType::CLASS;
393*b1cdbd2cSJim Jagielski                 else if( aKey.equalsIgnoreAsciiCaseAsciiL( RTL_CONSTASCII_STRINGPARAM( "BaseClass" ) ) )
394*b1cdbd2cSJim Jagielski                     nType = ModuleType::FORM;
395*b1cdbd2cSJim Jagielski 
396*b1cdbd2cSJim Jagielski                 if( (nType != ModuleType::UNKNOWN) && (aValue.getLength() > 0) )
397*b1cdbd2cSJim Jagielski                 {
398*b1cdbd2cSJim Jagielski                     OSL_ENSURE( aModules.has( aValue ), "VbaProject::importVba - module not found" );
399*b1cdbd2cSJim Jagielski                     if( VbaModule* pModule = aModules.get( aValue ).get() )
400*b1cdbd2cSJim Jagielski                         pModule->setType( nType );
401*b1cdbd2cSJim Jagielski                 }
402*b1cdbd2cSJim Jagielski             }
403*b1cdbd2cSJim Jagielski         }
404*b1cdbd2cSJim Jagielski     }
405*b1cdbd2cSJim Jagielski 
406*b1cdbd2cSJim Jagielski     // create empty dummy modules
407*b1cdbd2cSJim Jagielski     VbaModuleMap aDummyModules;
408*b1cdbd2cSJim Jagielski     for( DummyModuleMap::iterator aIt = maDummyModules.begin(), aEnd = maDummyModules.end(); aIt != aEnd; ++aIt )
409*b1cdbd2cSJim Jagielski     {
410*b1cdbd2cSJim Jagielski         OSL_ENSURE( !aModules.has( aIt->first ) && !aDummyModules.has( aIt->first ), "VbaProject::importVba - multiple modules with the same name" );
411*b1cdbd2cSJim Jagielski         VbaModuleMap::mapped_type& rxModule = aDummyModules[ aIt->first ];
412*b1cdbd2cSJim Jagielski         rxModule.reset( new VbaModule( mxContext, mxDocModel, aIt->first, eTextEnc, bExecutable ) );
413*b1cdbd2cSJim Jagielski         rxModule->setType( aIt->second );
414*b1cdbd2cSJim Jagielski     }
415*b1cdbd2cSJim Jagielski 
416*b1cdbd2cSJim Jagielski     /*  Now it is time to load the source code. All modules will be inserted
417*b1cdbd2cSJim Jagielski         into the Basic library of the document specified by the 'maPrjName'
418*b1cdbd2cSJim Jagielski         member. Do not create the Basic library, if there are no modules
419*b1cdbd2cSJim Jagielski         specified. */
420*b1cdbd2cSJim Jagielski     if( !aModules.empty() || !aDummyModules.empty() ) try
421*b1cdbd2cSJim Jagielski     {
422*b1cdbd2cSJim Jagielski         // get the model factory and the basic library
423*b1cdbd2cSJim Jagielski         Reference< XMultiServiceFactory > xModelFactory( mxDocModel, UNO_QUERY_THROW );
424*b1cdbd2cSJim Jagielski         Reference< XNameContainer > xBasicLib( createBasicLibrary(), UNO_SET_THROW );
425*b1cdbd2cSJim Jagielski 
426*b1cdbd2cSJim Jagielski         /*  Set library container to VBA compatibility mode. This will create
427*b1cdbd2cSJim Jagielski             the VBA Globals object and store it in the Basic manager of the
428*b1cdbd2cSJim Jagielski             document. */
429*b1cdbd2cSJim Jagielski         try
430*b1cdbd2cSJim Jagielski         {
431*b1cdbd2cSJim Jagielski             Reference< XVBACompatibility >( getLibraryContainer( PROP_BasicLibraries ), UNO_QUERY_THROW )->setVBACompatibilityMode( sal_True );
432*b1cdbd2cSJim Jagielski         }
433*b1cdbd2cSJim Jagielski         catch( Exception& )
434*b1cdbd2cSJim Jagielski         {
435*b1cdbd2cSJim Jagielski         }
436*b1cdbd2cSJim Jagielski 
437*b1cdbd2cSJim Jagielski         // try to get access to document objects related to code modules
438*b1cdbd2cSJim Jagielski         Reference< XNameAccess > xDocObjectNA;
439*b1cdbd2cSJim Jagielski         try
440*b1cdbd2cSJim Jagielski         {
441*b1cdbd2cSJim Jagielski             xDocObjectNA.set( xModelFactory->createInstance( CREATE_OUSTRING( "ooo.vba.VBAObjectModuleObjectProvider" ) ), UNO_QUERY );
442*b1cdbd2cSJim Jagielski         }
443*b1cdbd2cSJim Jagielski         catch( Exception& )
444*b1cdbd2cSJim Jagielski         {
445*b1cdbd2cSJim Jagielski             // not all documents support this
446*b1cdbd2cSJim Jagielski         }
447*b1cdbd2cSJim Jagielski 
448*b1cdbd2cSJim Jagielski         if( xBasicLib.is() )
449*b1cdbd2cSJim Jagielski         {
450*b1cdbd2cSJim Jagielski             // call Basic source code import for each module, boost::[c]ref enforces pass-by-ref
451*b1cdbd2cSJim Jagielski             aModules.forEachMem( &VbaModule::createAndImportModule,
452*b1cdbd2cSJim Jagielski                 ::boost::ref( *xVbaStrg ), ::boost::cref( xBasicLib ),
453*b1cdbd2cSJim Jagielski                 ::boost::cref( xDocObjectNA ) );
454*b1cdbd2cSJim Jagielski 
455*b1cdbd2cSJim Jagielski             // create empty dummy modules
456*b1cdbd2cSJim Jagielski             aDummyModules.forEachMem( &VbaModule::createEmptyModule,
457*b1cdbd2cSJim Jagielski                 ::boost::cref( xBasicLib ), ::boost::cref( xDocObjectNA ) );
458*b1cdbd2cSJim Jagielski         }
459*b1cdbd2cSJim Jagielski     }
460*b1cdbd2cSJim Jagielski     catch( Exception& )
461*b1cdbd2cSJim Jagielski     {
462*b1cdbd2cSJim Jagielski     }
463*b1cdbd2cSJim Jagielski 
464*b1cdbd2cSJim Jagielski     /*  Load the forms. The file format specification requires that a module
465*b1cdbd2cSJim Jagielski         must exist for every form. We are a bit more tolerant and scan the
466*b1cdbd2cSJim Jagielski         project storage for all form substorages. This may 'repair' broken VBA
467*b1cdbd2cSJim Jagielski         storages that misses to mention a module for an existing form. */
468*b1cdbd2cSJim Jagielski     ::std::vector< OUString > aElements;
469*b1cdbd2cSJim Jagielski     rVbaPrjStrg.getElementNames( aElements );
470*b1cdbd2cSJim Jagielski     for( ::std::vector< OUString >::iterator aIt = aElements.begin(), aEnd = aElements.end(); aIt != aEnd; ++aIt )
471*b1cdbd2cSJim Jagielski     {
472*b1cdbd2cSJim Jagielski         // try to open the element as storage
473*b1cdbd2cSJim Jagielski         if( !aIt->equalsAsciiL( RTL_CONSTASCII_STRINGPARAM( "VBA" ) ) )
474*b1cdbd2cSJim Jagielski         {
475*b1cdbd2cSJim Jagielski             StorageRef xSubStrg = rVbaPrjStrg.openSubStorage( *aIt, false );
476*b1cdbd2cSJim Jagielski             if( xSubStrg.get() ) try
477*b1cdbd2cSJim Jagielski             {
478*b1cdbd2cSJim Jagielski                 // resolve module name from storage name (which equals the module stream name)
479*b1cdbd2cSJim Jagielski                 VbaModule* pModule = aModulesByStrm.get( *aIt ).get();
480*b1cdbd2cSJim Jagielski                 OSL_ENSURE( pModule && (pModule->getType() == ModuleType::FORM),
481*b1cdbd2cSJim Jagielski                     "VbaProject::importVba - form substorage without form module" );
482*b1cdbd2cSJim Jagielski                 OUString aModuleName;
483*b1cdbd2cSJim Jagielski                 if( pModule )
484*b1cdbd2cSJim Jagielski                     aModuleName = pModule->getName();
485*b1cdbd2cSJim Jagielski 
486*b1cdbd2cSJim Jagielski                 // create and import the form
487*b1cdbd2cSJim Jagielski                 Reference< XNameContainer > xDialogLib( createDialogLibrary(), UNO_SET_THROW );
488*b1cdbd2cSJim Jagielski                 VbaUserForm aForm( mxContext, mxDocModel, rGraphicHelper, bDefaultColorBgr );
489*b1cdbd2cSJim Jagielski                 aForm.importForm( xDialogLib, *xSubStrg, aModuleName, eTextEnc );
490*b1cdbd2cSJim Jagielski             }
491*b1cdbd2cSJim Jagielski             catch( Exception& )
492*b1cdbd2cSJim Jagielski             {
493*b1cdbd2cSJim Jagielski             }
494*b1cdbd2cSJim Jagielski         }
495*b1cdbd2cSJim Jagielski     }
496*b1cdbd2cSJim Jagielski 
497*b1cdbd2cSJim Jagielski     // attach macros to registered objects
498*b1cdbd2cSJim Jagielski     attachMacros();
499*b1cdbd2cSJim Jagielski     // virtual call, derived classes may do some more processing
500*b1cdbd2cSJim Jagielski     finalizeImport();
501*b1cdbd2cSJim Jagielski }
502*b1cdbd2cSJim Jagielski 
attachMacros()503*b1cdbd2cSJim Jagielski void VbaProject::attachMacros()
504*b1cdbd2cSJim Jagielski {
505*b1cdbd2cSJim Jagielski     if( !maMacroAttachers.empty() && mxContext.is() ) try
506*b1cdbd2cSJim Jagielski     {
507*b1cdbd2cSJim Jagielski         Reference< XMultiComponentFactory > xFactory( mxContext->getServiceManager(), UNO_SET_THROW );
508*b1cdbd2cSJim Jagielski         Sequence< Any > aArgs( 2 );
509*b1cdbd2cSJim Jagielski         aArgs[ 0 ] <<= mxDocModel;
510*b1cdbd2cSJim Jagielski         aArgs[ 1 ] <<= maPrjName;
511*b1cdbd2cSJim Jagielski         Reference< XVBAMacroResolver > xResolver( xFactory->createInstanceWithArgumentsAndContext(
512*b1cdbd2cSJim Jagielski             CREATE_OUSTRING( "com.sun.star.script.vba.VBAMacroResolver" ), aArgs, mxContext ), UNO_QUERY_THROW );
513*b1cdbd2cSJim Jagielski         maMacroAttachers.forEachMem( &VbaMacroAttacherBase::resolveAndAttachMacro, ::boost::cref( xResolver ) );
514*b1cdbd2cSJim Jagielski     }
515*b1cdbd2cSJim Jagielski     catch( Exception& )
516*b1cdbd2cSJim Jagielski     {
517*b1cdbd2cSJim Jagielski     }
518*b1cdbd2cSJim Jagielski }
519*b1cdbd2cSJim Jagielski 
copyStorage(StorageBase & rVbaPrjStrg)520*b1cdbd2cSJim Jagielski void VbaProject::copyStorage( StorageBase& rVbaPrjStrg )
521*b1cdbd2cSJim Jagielski {
522*b1cdbd2cSJim Jagielski     if( mxContext.is() ) try
523*b1cdbd2cSJim Jagielski     {
524*b1cdbd2cSJim Jagielski         Reference< XStorageBasedDocument > xStorageBasedDoc( mxDocModel, UNO_QUERY_THROW );
525*b1cdbd2cSJim Jagielski         Reference< XStorage > xDocStorage( xStorageBasedDoc->getDocumentStorage(), UNO_QUERY_THROW );
526*b1cdbd2cSJim Jagielski         {
527*b1cdbd2cSJim Jagielski             const sal_Int32 nOpenMode = ElementModes::SEEKABLE | ElementModes::WRITE | ElementModes::TRUNCATE;
528*b1cdbd2cSJim Jagielski             Reference< XStream > xDocStream( xDocStorage->openStreamElement( CREATE_OUSTRING( "_MS_VBA_Macros" ), nOpenMode ), UNO_SET_THROW );
529*b1cdbd2cSJim Jagielski             OleStorage aDestStorage( mxContext, xDocStream, false );
530*b1cdbd2cSJim Jagielski             rVbaPrjStrg.copyStorageToStorage( aDestStorage );
531*b1cdbd2cSJim Jagielski             aDestStorage.commit();
532*b1cdbd2cSJim Jagielski         }
533*b1cdbd2cSJim Jagielski         Reference< XTransactedObject >( xDocStorage, UNO_QUERY_THROW )->commit();
534*b1cdbd2cSJim Jagielski     }
535*b1cdbd2cSJim Jagielski     catch( Exception& )
536*b1cdbd2cSJim Jagielski     {
537*b1cdbd2cSJim Jagielski     }
538*b1cdbd2cSJim Jagielski }
539*b1cdbd2cSJim Jagielski 
540*b1cdbd2cSJim Jagielski // ============================================================================
541*b1cdbd2cSJim Jagielski 
542*b1cdbd2cSJim Jagielski } // namespace ole
543*b1cdbd2cSJim Jagielski } // namespace oox
544