Necesito parar un proceso de MSDOS en cuanto pulso una tecla

Iniciado por antecessor, 7 Marzo 2015, 17:45 PM

0 Miembros y 1 Visitante están viendo este tema.

antecessor

Hola.

  Tengo una applicación en Visual Basic 6. El objetivo es escanear una carpeta y determinar qué archivos son copias imagen de ghost, esto es, tienen extensión *.GHO. Una vez localizados, pulso el botón test que arranca la aplicación de MSDOS ghost32.exe. Esta aplicación chequea cada copia imagen para verificar si es correcta. Todo funciona bien.

  El problema viene cuando quiero parar la comprobación de las imágenes porque le va a costar mucho y necesito el ordenador para otra cosa, pues bien, no hay manera. Lo único que consigo es, manteniendo la tecla que yo determine pulsada, y cuando está a punto de terminar la comprobación que está haciendo de la copia imagen actual y va a comenzar la próxima comprobación, se pare el programa, y lo que yo quiero es que se pare, o me pregunte si quiero continuar o estoy seguro de parar en cuanto yo le pulse una tecla, por ejemplo ESC. Yo no tengo demasiada idea de Visual Basic y las partes de programa más complicadas las he cogido de código que he encontrado por ahí y adaptado a mi aplicación:

Resumiendo: necesito parar el proceso de MSDOS -ghost32.exe- mientras se esté realizando la comprobación y pulsando una vez la tecla ESC, y que me pregunte si quiero continuar  o realmente parar -salir del bucle FOR o reiniciar el programa-.

-------------------------------------------------------------------------

Private Sub Form_Load()
   
' intervalo para el timer para la detección de tecla
    Timer1.Interval = 50

   
End Sub

----------------------------------------------------------------------------

Private Sub Timer1_Timer()
   
    If GetKeyState(vbKeyEscape) < 0 Then parar = 1    ' Detecto si se ha pulsado la tecla ESCAPE
     

End Sub

----------------------------------------------------------------------------


Private Sub cmdTest_Click()            'Botón que al pulsarlo hará que se empiecen a comprobar las copias imagen
DoEvents                    'Éste es fundamental, no quitar
lblCopiasRestantes.Visible = True
lblQuedan.Visible = True
lblQuedan = lblCount
RutaArchivo = GHO(n)
'La referencia a Windows Script Host Object Model

errorimagen = ""
a = 0
i = 1

Restantes = narchivos
    For n = 0 To narchivos - 1
       
        DoEvents
        Restantes = narchivos - n
        lblQuedan.Caption = Restantes - 1
        temporal = RutaCorta(lstFoundFiles.List(n))
           
        'Comando = "ghost32.exe -batch -chkimg," & lstFoundFiles.List(n)   'Arranca el ghost configurado para chequear imágenes"
        Comando = "ghost32.exe -batch -chkimg," & temporal 'lstFoundFiles.List(n)   Arranca el ghost configurado para chequear imágenes"
                   
                                                        'En lstFoundFiles está la lista de archivos *.GHO encontrados con ruta completa
        errorimagen = ejecutar_Dos(Comando)          'Si la imagen es incorrecta devuelve un error                  'llamo a la función
       
       
        If parar = 1 Then Exit For       'Si hemos pulsado la tecla ESC "cero" salimos del bucle
       
        If errorimagen <> "" Then                    'Si ghost devuelve un error
           
            If i = 1 Then txtResultado.Text = "                                     Las siguientes copias imagen no son correctas" + Chr$(13) + Chr$(10)
            i = i + 1
            a = a + 1
            txtResultado.Text = txtResultado.Text + Chr$(13) + Chr$(10) 'Línea en blanco
            txtResultado.Text = txtResultado.Text + "          Error " & i - 1 & " : " + lstFoundFiles.List(n) + Chr$(13) + Chr$(10)
        End If
       
               
    Next n
     
    If a = 0 Then txtResultado.Text = "                    Todas las copias imagen son correctas:" + Chr$(13) + Chr$(10)
    CmdLimpiarTexto.Visible = True
    cmdImprimir.Visible = True
   
    End Sub

----------------------------------------------------------------------------------------------------------------------------------


Function ejecutar_Dos(Comando As String) As String   'Función para ejecutar ghost32.exe
    Dim oShell As WshShell
    Dim oExec As WshExec
    Dim ret As String
       
    Set oShell = New WshShell
   
       
    ' ejecutar el comando
    Set oExec = oShell.Exec("%comspec% /c " & Comando)
    ret = oExec.StdOut.ReadAll()
       
    ' retornar la salida y devolverla a la función
    ejecutar_Dos = ret ' Replace(ret, Chr(10), vbNewLine)
       
    DoEvents
    Me.SetFocus
End Function



Agredecería alguna ayuda al respecto. Sasludos.

okik

#1
Hola
A mi me pasó lo mismo hace tiempo...


Este código es una plantilla, un ejemplo. Analízalo (es muy sencillo) y lo aplicas a tu código.

Necisitas un PictureBox, un CommandButton y  un control Timer.

Lo que hace es emular un proceso sin fin o muy largo For/next o Do Loop. Si se pulsa escape en el evento Timer detecta la tecla ESC y le da valor Verdadero a una variable que he llamado PauseProc. Al ser verdadera en el proceso ejecuta una procedimiento llamado Esperar que emula un reloj en segundos (sin timer), y el proceso se pausa. Si se vuelve a pulsar el CommandButton continúa el proceso, que no estaba parado, si no pausado por el proceso Espera. Entonces el Timer continua (para detectar la tecla ESC) y para el proceso Espera y entonces Tu proceso continua.

Puedes crear si quieres  una variable StopProc como boleano y si es Verdadero paras el proceso definitvamente.

Espero que te sirva:

Código (vb) [Seleccionar]
Private Declare Function GetKeyState Lib "user32" (ByVal nVirtKey As Long) As Integer
Dim PauseProc As Boolean
Dim Contador As Long
Private Sub Command1_Click()
'comienza el proceso
Timer1.Enabled = True 'Permite detectar la tecla
PauseProc = False 'Permite reaundar si se había pausado (false= no parar)
SimularProceso 'SIMULA EL PROCESO
End Sub
Private Sub Form_Load()
Contador = 1
   Picture1.AutoRedraw = True
End Sub
Private Sub Form_Unload(Cancel As Integer)
   End 'Evita que no se pueda cerrar
End Sub

Private Sub Timer1_Timer()
Dim msgQuest As Variant
If GetKeyState(vbKeyEscape) < 0 Then PauseProc = True 'para el proceso

If PauseProc = True Then 'Si se ha pulsado ESCAPE
       'Muestra la pregunta
       msgQuest = MsgBox("Desea pausar el proceso", vbYesNo, "Question")
       'Si es no continua el proceso
       If msgQuest = vbNo Then
           PauseProc = False 'Continuar proceso
           Else ' si es sí para el timer (para invalidar ESC)
           Timer1.Enabled = False
       End If
Else 'Si no se ha pulsado nada
   PauseProc = False 'No se pare el proceso
   Timer1.Enabled = True 'Atento si se pulsa ESC
End If
End Sub
'////////////////////////////////////////
'SIMULA UN PROCESO FOR/NEXT O DO/LOOP
'////////////////////////////////////////
Public Function SimularProceso()
For i = Contador To 3 ^ 14
Contador = i 'Memoriza el último estado
   Picture1.Cls
   Picture1.Print i
   DoEvents
'Si se ha pausado pausa durante 3600 segundos (un hora)
If PauseProc = True Then Espera (3600)
Next i
End Function

'////////////////////////////////////////
'Espera un determinado tiempo en segundos
'////////////////////////////////////////
Private Sub Espera(Segundos As Single)
 Dim ComienzoSeg As Single
 Dim FinSeg As Single
 ComienzoSeg = Timer
 FinSeg = ComienzoSeg + Segundos
       If PauseProc = False Then Exit Sub ' Si pulsas continuar deja de esperar
 Do While FinSeg > Timer
     DoEvents
     If ComienzoSeg > Timer Then
         FinSeg = FinSeg - 24 * 60 * 60
     End If
 Loop
End Sub



Esto es una forma muy rebuscada. Pero si quieres en lugar de usar For/next  o Do /Loop metes el proceso en el un Timer y cuando te interese lo estableces a False y cuando quieras reaundar a True. Es más sencillo.