Click here to Skip to main content
65,938 articles
CodeProject is changing. Read more.
Articles
(untagged)

Enhanced version of the CreateProcess function

0.00/5 (No votes)
26 Apr 2004 1  
Waits for the child process to exit, minimizes the main window while the child process runs, and automatically restores the window when it exits.

Introduction

In one of my programs, I needed to wait until an external application finished processing some data, and wanted to minimize the main window during that time to prevent the user from doing anything else before the results were available. That's why I wrote CreateProcessEx.

The CreateProcessEx function takes seven parameters but only the first two are required:

  • lpAppName:Pointer to a null-terminated string that specifies the application to execute. The string can specify the full path and file name of the application to execute, or it can specify a partial name.

    If lpAppName is NULL, the application name must be the first white space-delimited token in the lpCmdLine string.

  • lpCmdLine: Pointer to a null-terminated string that specifies the command line to execute. This parameter can be NULL.
  • bAppInCmdLine: Indicates if the lpAppName string must be used as the first white space-delimited token in the lpCmdLine string.
  • bCompletePath: Indicates if the lpAppName string specifies a partial name that must be completed using the path of the calling process.
  • bWaitForProcess: Indicates if the calling process must wait until child process exits.
  • bMinimizeOnWait: Indicates if the main window will be minimized while the child process runs and automatically restored when it exits.
  • hMainWnd: Handle of the main window to be minimized. If hMainWnd is NULL, then AfxGetMainWnd() is used.
DWORD CreateProcessEx ( LPCSTR lpAppPath, LPCSTR lpCmdLine, 
    BOOL bAppInCmdLine, BOOL bCompletePath,
    BOOL bWaitForProcess, BOOL bMinimizeOnWait, HWND hMainWnd ) {

    STARTUPINFO startupInfo;
    PROCESS_INFORMATION    processInformation;    

    char szAppPath    [ _MAX_PATH  ];
    char szCmdLine    [ _MAX_PATH  ];
    char drive        [ _MAX_DRIVE ];
    char dir        [ _MAX_DIR   ];
    char name       [ _MAX_FNAME ];
    char ext        [ _MAX_EXT ];

    szAppPath[ 0 ] = '\0';
    szCmdLine[ 0 ] = '\0';
    
    ZeroMemory( &startupInfo, sizeof( STARTUPINFO ));

    startupInfo.cb = sizeof( STARTUPINFO );

    ZeroMemory( &processInformation, sizeof( PROCESS_INFORMATION ));

    if ( bCompletePath ) {

        GetModuleFileName( GetModuleHandle( NULL ), szAppPath, _MAX_PATH );

        _splitpath( szAppPath, drive, dir, NULL, NULL );
        _splitpath( lpAppPath, NULL, NULL, name, ext );

        _makepath ( szAppPath, drive, dir, name, strlen( ext ) ? ext : ".exe" );
    }
    else strcpy( szAppPath, lpAppPath );

    if ( bAppInCmdLine ) {

        strcpy( szCmdLine, "\"" );
        strcat( szCmdLine, szAppPath );
        strcat( szCmdLine, "\"" );
    }

    if ( lpCmdLine ) {

        // We must use quotation marks around the command line (if any)...


        if ( bAppInCmdLine ) strcat( szCmdLine, " \"" );
                        else strcpy( szCmdLine, "\"" );

        strcat( szCmdLine, lpCmdLine );
        strcat( szCmdLine, "\"" );
    }

    DWORD dwExitCode = -1;
           
    if ( CreateProcess(    bAppInCmdLine ? NULL: szAppPath,    // lpszImageName

                        szCmdLine,                            // lpszCommandLine

                        0,                                    // lpsaProcess

                        0,                                    // lpsaThread

                        FALSE,                                // fInheritHandles

                        DETACHED_PROCESS,                    // fdwCreate

                        0,                                    // lpvEnvironment

                        0,                                    // lpszCurDir

                        &startupInfo,                        // lpsiStartupInfo

                        &processInformation                    // lppiProcInfo

                    )) {

        if ( bWaitForProcess ) {

            if ( bMinimizeOnWait )
                    
                if ( IsWindow( hMainWnd )) ShowWindow( hMainWnd, SW_MINIMIZE );
                #ifdef __AFX_H__
                else AfxGetMainWnd()->ShowWindow( SW_MINIMIZE );
                #endif

            WaitForSingleObject( processInformation.hProcess, INFINITE );

            if ( bMinimizeOnWait )

                if ( IsWindow( hMainWnd )) ShowWindow( hMainWnd, SW_RESTORE );
                #ifdef __AFX_H__
                else AfxGetMainWnd()->ShowWindow( SW_RESTORE );
                #endif

            GetExitCodeProcess( processInformation.hProcess, &dwExitCode );
        }
        else {

            CloseHandle( processInformation.hThread );
            CloseHandle( processInformation.hProcess );

            dwExitCode = 0;
        }
    }

    return dwExitCode;
}

To use the function, simply include the CreateProcessEx.h and CreateProcessEx.cpp files in your project. That's it!

License

This article has no explicit license attached to it but may contain usage terms in the article text or the download files themselves. If in doubt please contact the author via the discussion board below.

A list of licenses authors might use can be found here