Determinar quando um processo em shell termina
Quando você executa a função Shell em um procedimento de Visual Basic for Applications (VBA), ela inicia um programa executável de forma assíncrona e retorna o controle ao procedimento. Este programa shelled continua a ser executado independentemente do procedimento até que você o feche.
Se o procedimento precisar aguardar o término do processo shelled, você poderá usar a API do Windows para sondar o status do aplicativo, mas isso não é muito eficiente. Este tópico explica um método mais eficiente.
A API do Windows tem uma funcionalidade integrada que permite que seu aplicativo aguarde até que um processo shelled seja concluído. Para usar essas funções, você precisa ter um identificador para o processo shelled. Para fazer isso, use a função CreateProcess em vez da função Shell para iniciar seu programa em shell.
Criar o processo shelled
Para criar um processo endereçável, use a função CreateProcess para iniciar seu aplicativo shelled. A função CreateProcess fornece ao seu programa o identificador de processo do processo shelled por meio de um de seus parâmetros passados.
Aguarde o fim do processo com shell
Depois de usar a função CreateProcess para obter um identificador de processo, você pode passar esse identificador para a função WaitForSingleObject . Isso faz com que o procedimento VBA suspenda a execução até que o processo shelled termine.
As etapas a seguir são necessárias para criar um procedimento VBA que usa a função CreateProcess para executar o aplicativo Bloco de Notas do Windows. Este código mostra como usar as funções CreateProcess e WaitForSingleObject da API do Windows para aguardar até que um processo shelled termine antes de retomar a execução.
A sintaxe da função CreateProcess é complexa, portanto, no código de exemplo, ela é encapsulada em uma função chamada ExecCmd. ExecCmd usa um parâmetro, a linha de comando do aplicativo a ser executada.
Crie um módulo padrão e cole as seguintes linhas na seção Declarações:
Option Explicit Private Type STARTUPINFO cb As Long lpReserved As String lpDesktop As String lpTitle As String dwX As Long dwY As Long dwXSize As Long dwYSize As Long dwXCountChars As Long dwYCountChars As Long dwFillAttribute As Long dwFlags As Long wShowWindow As Integer cbReserved2 As Integer lpReserved2 As Long hStdInput As Long hStdOutput As Long hStdError As Long End Type Private Type PROCESS_INFORMATION hProcess As Long hThread As Long dwProcessID As Long dwThreadID As Long End Type Private Declare Function WaitForSingleObject Lib "kernel32" (ByVal _ hHandle As Long, ByVal dwMilliseconds As Long) As Long Private Declare Function CreateProcessA Lib "kernel32" (ByVal _ lpApplicationName As Long, ByVal lpCommandLine As String, ByVal _ lpProcessAttributes As Long, ByVal lpThreadAttributes As Long, _ ByVal bInheritHandles As Long, ByVal dwCreationFlags As Long, _ ByVal lpEnvironment As Long, ByVal lpCurrentDirectory As Long, _ lpStartupInfo As STARTUPINFO, lpProcessInformation As _ PROCESS_INFORMATION) As Long Private Declare Function CloseHandle Lib "kernel32" (ByVal _ hObject As Long) As Long Private Const NORMAL_PRIORITY_CLASS = &H20& Private Const INFINITE = -1&
Cole o seguinte código no módulo:
Public Sub ExecCmd(cmdline As String) Dim proc As PROCESS_INFORMATION Dim start As STARTUPINFO Dim ReturnValue As Integer ' Initialize the STARTUPINFO structure: start.cb = Len(start) ' Start the shelled application: ReturnValue = CreateProcessA(0&, cmdline$, 0&, 0&, 1&, _ NORMAL_PRIORITY_CLASS, 0&, 0&, start, proc) ' Wait for the shelled application to finish: Do ReturnValue = WaitForSingleObject(proc.hProcess, 0) DoEvents Loop Until ReturnValue <> 258 ReturnValue = CloseHandle(proc.hProcess) End Sub
Para testar a função, cole o código a seguir na janela Imediato e pressione Enter. Bloco de notas é iniciado. Depois de um momento, feche o Bloco de Notas. A caixa de mensagem é exibida quando o Bloco de Notas fecha.
ExecCmd "NOTEPAD.EXE": MsgBox "Process Finished"
Suporte e comentários
Tem dúvidas ou quer enviar comentários sobre o VBA para Office ou sobre esta documentação? Confira Suporte e comentários sobre o VBA para Office a fim de obter orientação sobre as maneiras pelas quais você pode receber suporte e fornecer comentários.