VBA
The VBA interoperability project is a joint Novell & Sun incubator project whose aim is to deliver a way to run Excel and Word macros natively in imported documents.
- Noel Power (lead)
- Andreas Bregas (co-lead)
Visual Basic for Applications (VBA) is an implementation of Microsoft's Visual Basic which is built into all Microsoft Office applications.
Latest news
- CWS SRC680 npower6 has been integrated into m211
- Update Wiki Page with latest info
- Novell and Sun to work together on the VBA incubator project
- I am delighted that Andreas Bregas now joins me to co-lead the vba incubator project
Terminology
- helperapi - The compatibility API generously donated by Sun that was part of their migration toolkit
- oovbaapi - the ooo-build vba interoperability solution originally implemented by Novell + object implementations from the helperapi. Note: Ooo-build contains the latest VBA model and patches which allows you to run macros from your Microsoft Office documents. This model is under heavy development by Novell and others and the changes are continuously ported to openoffice. It is expected that many macros won't run for now.
Dashboard
- CWS SRC680 npower8 (nearly complete) (contains)
* some helperapi object porting * changes to include new multi-inheritance style idl and new style service definitions * removal of idl for compatibilty constants[1]
- CWS SRC680 npower7 integrated
- CWS SRC680 npower6 integrated
[1] these will be generated using the mechanism from the helperapi (basically the idl is generated from metadata files, the idl files are not preserved (or needed) but the types database generated from the idl files
Task queue and status (in order of priority)
Feature | Description | Issue # | Planned cws | State | Milestone integrated | |
autogenerate constants in oovbaapi | port helperapi constant solution to oovbaapi module | Issue 75551 | CWS SRC680 npower8 | done | n/a | |
ooo-build api sync | synchronise ooo-build api changes with upstream | Issue 77189 | CWS SRC680 npower8 | in progress | n/a | |
array of array indexing | in vba array of array indexing is very common, e.g. myArray(1)(2) | n/a | n/a | n/a | n/a | |
toolbox control event support | these controls support a loose binding where the presence of a macro event handler of the form controlname_xxx e.g. controlname_click results in the appropriate macro getting called when the associated action takes place | n/a | n/a | n/a | n/a |
Completed Features
Feature | Description | Issue # | Planned cws | State | Milestone integrated | |
vba constants | adjust basic runtime to be 'conditionally' aware of the vba constants defined in oovbaapi module | Issue 68897 | CWS SRC680 npower7 | complete | n/a | |
support dim of vba object | adjust basic runtime to be 'conditionally' allow a vba compatibilty api object to be used like Dim r1 as Range | Issue 68897 | CWS SRC680 npower7 | complete | n/a | |
compatible erase array | in vba compatibility mode erase does not behave as expected | Issue 70380 | CWS SRC680 npower7 | complete | n/a | |
null variant behaviour | in vba compatibility mode operations on null variant objects behave differently | Issue 71827 | CWS SRC680 npower7 | complete | n/a | |
'Like' | new 'Like' operator for basic | Issue 73830 | CWS SRC680 npower7 | complete | n/a | |
Module scope variable | in vba module scope variables lifetime is tied to the document containing the macro | Issue 76819 | CWS SRC680 npower7 | complete | n/a | |
support simple range short cut references | in vba the evaluate function is used for resolving many things, one of theses is short cut range references e.g. '[a1:b2]' | Issue 68898 | CWS SRC680 npower7 | complete | n/a | |
fix accent handling | in vba mode accents in modules or macros are not handled | Issue 76818 | CWS SRC680 npower7 | complete | n/a | |
fix object comparison | in vba mode there is a problem comparing objects with default paramaters (when the default is not specified) | Issue 76820 | CWS SRC680 npower7 | complete | n/a | |
class module import | when class modules are imported the correct 'option' should be set | Issue 76822 | CWS SRC680 npower7 | complete | n/a | |
migrate ooo-build code | suck in the mountain of idl and implementation code from ooo-build | Issue 74096 | CWS SRC680 npower6 | complete | m211 | |
import vba code | conditionally import (uncommented) vba code and set appropriate module options | Issue 64570 | CWS SRC680 npower6 | complete | m211 |
Broader scope tasks that need more investigation (and splitting into smaller tasks) (to be added)
create list of items that may need promoting from vba compatibility to Standard ooo-basic behaviour e.g. like Issue 70380
How to enable the VBA model
The VBA model is enabled by default in the current ooo-build. It's enough to get, build, install, and be able to run it.
If you are using upstream OpenOffice.org and wish to play with this feature then check out and build the CWS SRC680 npower6 which contains the initial code drop for the oovbaapi
How to disable the VBA model
Depending on how your version of OpenOffice.org delivers the VBA feature to you, there are a number of ways to disable it.
If you have an Openoffice.org built by a linux distro (or a windows one from Novell) that has the vba interop feature then all you need do is navigate to the 'Tools.Options.Load/Save.VBAProperties' page and uncheck the 'Load Excelbasic code to edit' option.
If you are using an own build (incorporating) CWS SRC680 npower6 the above method will always work. But if you built the vba interop feature as an office extension then you can of course disable or delete the vbaapi.oxt extension using the Extension Manager.
How can I help you?
We need help in a couple of areas, porting objects from the helperapi, understanding the VBA usage patterns, implementation and testing; see below.
Porting Objects from the HelperApi
What is the helperapi?
Have a look at the porting notes
Have a look at a porting example
list of objects that need porting
list of methods (to be ported from helperapi) to existing objects
Understanding the VBA usage patterns
VBA model is huge and we need help in understanding the VBA usage patterns for prioritizing as we progress. It's important to receive as many Microsoft Office documents with macros as possible for analysis. We are interested in documents from developers and regular people too.
Dumping the VBA from your MS Office documents
libgsf
If you're worried about the sheet data you can use test-msvba from the libgsf. Download the latest test-msvba.c file and compile it with the following command:
gcc `pkg-config --libs --cflags libgsf-1` -o test-msvba test-msvba.c
Note: you need to have libgsf-devel package installed.
Now, you can dump the VBA from your documents with the following command:
find -name '*.xls' -exec sh -c 'test-msvba {} >> ~/excel-vba-stream-for-noel.bas' \;
pole
Or you can use POLE with GUI. POLE is a portable C++ library to access structured storage. It is designed to be compatible with Microsoft structured storage, also sometimes known as OLE Compound Document.
Exporting modules from the Microsoft Excel
Open your document and edit associated macros. You can see list of modules on the left side of your screen. Use the right mouse button and select Export from the context menu. The whole module will be saved as one huge *.bas.
Where to send
If you are going to send pure VBA streams, send them directly to [Noel dot Power at Novell dot com].
Implementation
The desired sequence
Add an extension to the IDL
The IDL lives in oovbaapi/org/openoffice/*, the IDL syntax can easily be gleaned from a few other IDL files. Two things are worth bearing in mind:
- it's important that the argument names are correct, so that the MsgBox title:="hello" syntax works correctly.
- for optional arguments, or variant types you need to use the any type
Having edited the IDL extension you need to:
- run: build && deliver in oovbaapi (perhaps remove types.db first),
Add the extension to the implementation
The impl. code lives in sc/source/ui/vba/ - create the right header and source files. Add the method (with the correct prototype) to the header - often it's easiest to cut/paste that from the compiled C++ headers in sc/unxlngixxx/inc/org/openoffice/vba/
Then implement the method itself in the .cxx file. Run build debug=true
Running it
You did use [linkoo] right ? - in which case - just re-run soffice.bin (killall -9 soffice.bin first to ensure there is no lurking factory process).
Now of course - you want to add a new test to exercise it to ooo-build/test/macro/*. And of course loop back to the beginning. Once you've got something you want to share - a simple cvs diff -u | tee /tmp/foo.diff in ooo-build/scratch will yield a patch you can mail to Noel / the ooo-build mailing list to get your code folded in. NB. it's important you sign the JCA first.
Testing
It's extremely important to test the work which is already done. You can use test cases from the ooo-build/test/macro directory, you can extend them or you can write your own test cases.
It's a good idea to write test cases directly in the Microsoft Excel?
Accessing the VBA API from other languages
The VBA API is implemented by an UNO service, the service is accessible via a singleton. As such it is possible to use the API from any language that has an UNO binding. The main entry points to the API are certain global variables { Application, ActiveWorkbook and ActiveSheet } or methods { WorkSheets, WorkBooks } exported by the service. Using these globals as a starting point the rest of the api can be accessed.
NOTE: Beaware the idl is not stable, (nor won't be for the forseeable future be 'published') it WILL change, so if you are using a language that has a binary dependency on the interface (like c++ & java) then consider yourselves warned.
Python
vba = uno.getComponentContext().getByName("/singletons/org.openoffice.vba.theGlobals")
sheet = vba.ActiveSheet
book = vba.ActiveWorkbook
Java
import org.openoffice.vba.*;
"
// getComponentCtxFromSomewhere is a fictional
// routine representing acquisition of the component context
// from e.g. remote ServiceManage if the code is a remote client
// or from the XScriptContext if this were a java ScriptingFramework script
XComponentContext ctx = getComponentCtxFromSomewhere();
Object oGlobs = ctx.getValueByName(
"/singletons/org.openoffice.vba.theGlobals");
XGlobals xGlobs = AnyConverter.toObject(
new Type(XGlobals.class), oGlobs);
XWorkBook xWBook = xGlobs.getActiveWorkbook();
XWorkSheet xSheet = xGlobs.getActiveWorkSheet();
OO Basic
Basic allows access to the global variables automatically just like VBA does, there is no need to create any services or access any singletons to use the API
msgbox "Name of the current document is " & ActiveWorkbook.Path
Visual Basic
Visual Basic, VBA, VBScript and other COM-enabled languages supports accessing OpenOffice VBA API via COM-UNO bridge
Set oServiceManager = CreateObject("com.sun.star.ServiceManager")
Set oDesktop = oServiceManager.createInstance("com.sun.star.frame.Desktop")
Dim aNoArgs()
Set oCalcDoc = oDesktop.loadComponentFromURL("private:factory/scalc", "_blank", 0, aNoArgs)
Set v = oServiceManager.DefaultContext.getValueByName("/singletons/org.openoffice.vba.theGlobals")
Set ooApp = v.Application
MsgBox ooApp.Version
Set cell = ooApp.ActiveCell
cell.Value = "Hello!"
Set r = ooApp.Range("B1", "B1")
r.Value = "Hi1"
CHM Help file generated from oovbaapi IDL: http://community.i-rs.ru/index.php?action=dlattach;topic=6096.0;attach=4431