Ecosyste.ms: Awesome
An open API service indexing awesome lists of open source software.
https://github.com/Alex-Butenko/OpenXML-for-VSTO
Enable OpenXML for VSTO add-ins
https://github.com/Alex-Butenko/OpenXML-for-VSTO
excel excel-addin excel-automation interop msword openxml openxml-excel powerpoint powerpoint-add-in powerpoint-automation vsto word-add-in word-automation
Last synced: 2 months ago
JSON representation
Enable OpenXML for VSTO add-ins
- Host: GitHub
- URL: https://github.com/Alex-Butenko/OpenXML-for-VSTO
- Owner: Alex-Butenko
- License: bsd-3-clause
- Created: 2020-05-13T03:25:05.000Z (over 4 years ago)
- Default Branch: master
- Last Pushed: 2024-07-25T17:03:07.000Z (6 months ago)
- Last Synced: 2024-10-07T00:25:45.132Z (3 months ago)
- Topics: excel, excel-addin, excel-automation, interop, msword, openxml, openxml-excel, powerpoint, powerpoint-add-in, powerpoint-automation, vsto, word-add-in, word-automation
- Language: C#
- Homepage:
- Size: 75.2 KB
- Stars: 7
- Watchers: 1
- Forks: 3
- Open Issues: 0
-
Metadata Files:
- Readme: README.md
- License: LICENSE
Awesome Lists containing this project
- jimsghstars - Alex-Butenko/OpenXML-for-VSTO - Enable OpenXML for VSTO add-ins (C# #)
README
# OpenXML-for-VSTO
OpenXML-for-VSTO provides the ability to process content as OpenXML from VSTO-addins* [Introduction](#introduction)
* [How it works?](#How-it-works)
* [Where to use](#Where-to-se)
* [Where not to use](#Where-not-to-use)
* [Benchmark](#Benchmark)
* [Issues](#Issues)
* [Tips and tricks](#Tips-and-tricks)## Introduction
One of the problems of creating VSTO add-ins for MS Office is the performance of access to a large number of elements. Access is through the boundaries of the AppDomains, with a lot of Reflection. Each call can read/write only one parameter at a time, therefore, for large tables/documents, the add-in have to make millions of such calls. Moreover, these calls are performed in the UI thread, blocking or slowing down user input and worsening the user experience.
Using OpenXML would be a good solution, but unfortunately, the Office API does not provide easy access to OpenXML.
This library provides the ability to get/set an OpenXML object directly from/to VSTO add-in.## How it works
* Copy an object to a file using Clipboard.
* Work with the file as with any other OpenXML file.
* Copy the object back.## When to use
In VSTO add-ins, in operations with thousands of objects, involving styles and formats.## When not to use
* Office Interop except for VSTO add-ins. In this case the library does not give any advantage, as you can replace interop with any OpenXML library all at once.
* Operations with small sets of objects - each copy paste has significant overhead that does not worth it for small sets.
* Operations with values/formulas in Excel - these can be much faster accessed throuch 2d arrays.## Benchmark
### Excel
Benchmark is based on writing and reading cells - 1, 100, 10,000 or 1,000,000 two different ways - using this library (and ClosedXML) and with pure Excel interop. Each individual cell value, background color, font color, font size, italic and bold status and number format. No optimizations.Writing operations:
| | Pure VSTO | Using OpenXML-for-VSTO |
|-----------------|------------|------------------------|
| 1 cell |00:00:00.254| 00:00:00.938 |
| 100 cells |00:00:00.302| 00:00:01.225 |
| 10000 cells |00:00:09.709| 00:00:01.967 |
| 1,000,000 cells |00:18:31.897| 00:01:28.600 |Reading operations:
| | Pure VSTO | Using OpenXML-for-VSTO |
|-----------------|------------|------------------------|
| 1 cell |00:00:00.009| 00:00:00.754 |
| 100 cells |00:00:00.045| 00:00:00.740 |
| 10000 cells |00:00:03.287| 00:00:01.476 |
| 1,000,000 cells |00:05:45.179| 00:00:47.838 |Even though performance for Excel can be improved in some scenarios by using union ranges, ClosedXML algorithms are less than optimal in this scenario as well, and most likely in real scenario performance gain can be much bigger.
### Word
Benchmark is based on writing and reading runs of text - 1, 100, 10,000, 100,000 or 1,000,000 two different ways - using this library and with pure Word interop. Each individual run text, background color, font color, font size, italic and bold status. No optimizations.Writing operations:
| | Pure VSTO | Using OpenXML-for-VSTO |
|-----------------|------------|------------------------|
| 1 run |00:00:00.027| 00:00:00.828 |
| 100 runs |00:00:00.623| 00:00:00.583 |
| 10,000 runs |00:02:10.232| 00:00:01.360 |
| 100,000 runs |00:51:37.007| 00:00:09.061 |
| 1,000,000 runs | | 00:01:36.512 |Reading operations:
| | Pure VSTO | Using OpenXML-for-VSTO |
|-----------------|------------|------------------------|
| 1 run |00:00:00.013| 00:00:00.519 |
| 100 runs |00:00:00.150| 00:00:00.186 |
| 10,000 runs |00:00:16.284| 00:00:01.377 |
| 100,000 runs |00:03:02.905| 00:00:08.515 |
| 1,000,000 runs | | 00:01:23.920 |## Issues
* It uses the clipboard, so after each operation it looses data user potentiall saved there. There are workaronds, like save and restore clipboard for each operation, but these are not part of this library
* Beginning from certain size (around 12 mb) of objects, the library may cause IsolatedStorageException. This exception happens because VSTO appdomain has incorrect Evidence. It is easy to fix by calling this code once before using OpenXML-for-VSTO with big objects:
```c#
System.Security.Policy.Evidence newEvidence = new System.Security.Policy.Evidence();
newEvidence.AddHostEvidence(new System.Security.Policy.Zone(System.Security.SecurityZone.MyComputer));System.AppDomain.CurrentDomain
.GetType()
.GetField("_SecurityIdentity", System.Reflection.BindingFlags.Instance | System.Reflection.BindingFlags.NonPublic)?
.SetValue(System.AppDomain.CurrentDomain, newEvidence);
```## Tips and tricks
You can save a copy of the whole workbook without using this library by using Workbook.SaveCopyAs method.
This does not work with Word documents, but then you can cast Document to [IPersistFile](https://docs.microsoft.com/en-us/windows/win32/api/objidl/nn-objidl-ipersistfile?redirectedfrom=MSDN) and use method Save```c#
using System.Runtime.InteropServices.ComTypes;IPersistFile persistFile = (IPersistFile) doc;
persistFile.Save(, false);
```