Программно извлекать код макроса (VBA) из документов Word 2007 - PullRequest
15 голосов
/ 08 сентября 2008

Можно ли извлечь весь код VBA из документа Word 2007 "docm" с помощью API?

Я нашел, как вставить код VBA во время выполнения и как удалить весь код VBA, но не извлекать действительный код в поток или строку, которые я могу сохранить (и вставить в другие документы в будущем).

Любые советы или ресурсы будут оценены.

Редактировать : спасибо всем, ответ Aardvark был именно тем, что я искал. Я преобразовал его код в C # и смог вызвать его из библиотеки классов с помощью Visual Studio 2008.

using Microsoft.Office.Interop.Word;
using Microsoft.Vbe.Interop;

...

public List<string> GetMacrosFromDoc()
{
    Document doc = GetWordDoc(@"C:\Temp\test.docm");

    List<string> macros = new List<string>();

    VBProject prj;
    CodeModule code;
    string composedFile;

    prj = doc.VBProject;
    foreach (VBComponent comp in prj.VBComponents)
    {
        code = comp.CodeModule;

        // Put the name of the code module at the top
        composedFile = comp.Name + Environment.NewLine;

        // Loop through the (1-indexed) lines
        for (int i = 0; i < code.CountOfLines; i++)
        {
            composedFile += code.get_Lines(i + 1, 1) + Environment.NewLine;
        }

        // Add the macro to the list
        macros.Add(composedFile);
    }

    CloseDoc(doc);

    return macros;
}

Ответы [ 2 ]

27 голосов
/ 08 сентября 2008

Вы можете экспортировать код в файлы и затем прочитать их обратно.

Я использовал приведенный ниже код, чтобы помочь мне сохранить некоторые макросы Excel под контролем исходного кода (используя Subversion & TortoiseSVN). Он в основном экспортирует весь код в текстовые файлы каждый раз, когда я сохраняю его с открытым редактором VBA. Я поместил текстовые файлы в Subversion, чтобы я мог делать различия. Вы должны быть в состоянии приспособить / украсть часть этого для работы в Word.

Проверка реестра в CanAccessVBOM () соответствует «Доверительному доступу к проекту Visual Basic» в параметре безопасности.

Sub ExportCode()

    If Not CanAccessVBOM Then Exit Sub ' Exit if access to VB object model is not allowed
    If (ThisWorkbook.VBProject.VBE.ActiveWindow Is Nothing) Then
        Exit Sub ' Exit if VBA window is not open
    End If
    Dim comp As VBComponent
    Dim codeFolder As String

    codeFolder = CombinePaths(GetWorkbookPath, "Code")
    On Error Resume Next
    MkDir codeFolder
    On Error GoTo 0
    Dim FileName As String

    For Each comp In ThisWorkbook.VBProject.VBComponents
        Select Case comp.Type
            Case vbext_ct_ClassModule
                FileName = CombinePaths(codeFolder, comp.Name & ".cls")
                DeleteFile FileName
                comp.Export FileName
            Case vbext_ct_StdModule
                FileName = CombinePaths(codeFolder, comp.Name & ".bas")
                DeleteFile FileName
                comp.Export FileName
            Case vbext_ct_MSForm
                FileName = CombinePaths(codeFolder, comp.Name & ".frm")
                DeleteFile FileName
                comp.Export FileName
            Case vbext_ct_Document
                FileName = CombinePaths(codeFolder, comp.Name & ".cls")
                DeleteFile FileName
                comp.Export FileName
        End Select
    Next

End Sub
Function CanAccessVBOM() As Boolean
    ' Check resgistry to see if we can access the VB object model
    Dim wsh As Object
    Dim str1 As String
    Dim AccessVBOM As Long

    Set wsh = CreateObject("WScript.Shell")
    str1 = "HKEY_CURRENT_USER\Software\Microsoft\Office\" & _
        Application.Version & "\Excel\Security\AccessVBOM"
    On Error Resume Next
    AccessVBOM = wsh.RegRead(str1)
    Set wsh = Nothing
    CanAccessVBOM = (AccessVBOM = 1)
End Function


Sub DeleteFile(FileName As String)
    On Error Resume Next
    Kill FileName
End Sub

Function GetWorkbookPath() As String
    Dim fullName As String
    Dim wrkbookName As String
    Dim pos As Long

    wrkbookName = ThisWorkbook.Name
    fullName = ThisWorkbook.fullName

    pos = InStr(1, fullName, wrkbookName, vbTextCompare)

    GetWorkbookPath = Left$(fullName, pos - 1)
End Function

Function CombinePaths(ByVal Path1 As String, ByVal Path2 As String) As String
    If Not EndsWith(Path1, "\") Then
        Path1 = Path1 & "\"
    End If
    CombinePaths = Path1 & Path2
End Function

Function EndsWith(ByVal InString As String, ByVal TestString As String) As Boolean
    EndsWith = (Right$(InString, Len(TestString)) = TestString)
End Function
9 голосов
/ 08 сентября 2008

Вам нужно будет добавить ссылку на Microsoft Visual Basic для расширений приложений 5.3 (или любой другой имеющейся версии). У меня есть VBA SDK и тому подобное на моей коробке - так что это может быть не совсем то, что поставляется с офисом.

Также вам необходимо разрешить доступ к объектной модели VBA - см. «Центр управления безопасностью» в настройках Word. Это в дополнение ко всем другим параметрам безопасности макросов, которые предоставляет Office.

Этот пример извлекает код из текущего документа, в котором он находится - он сам является макросом VBA (и отображает сам себя и любой другой код). Существует также коллекция Application.vbe.VBProjects для доступа к другим документам. Хотя я никогда этого не делал, я предполагаю, что внешнее приложение может открывать файлы, используя эту коллекцию VBProjects. Безопасность - это смешно, поэтому может быть сложно.

Мне также интересно, какой сейчас формат файла docm - XML, как docx? Это был бы лучший подход?

Sub GetCode()

    Dim prj As VBProject
    Dim comp As VBComponent
    Dim code As CodeModule
    Dim composedFile As String
    Dim i As Integer

    Set prj = ThisDocument.VBProject
        For Each comp In prj.VBComponents
            Set code = comp.CodeModule

            composedFile = comp.Name & vbNewLine

            For i = 1 To code.CountOfLines
                composedFile = composedFile & code.Lines(i, 1) & vbNewLine
            Next

            MsgBox composedFile
        Next

End Sub
...