| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780 | /*  Native File Dialog  http://www.frogtoss.com/labs */#define _CRTDBG_MAP_ALLOC  #include <stdlib.h>  #include <crtdbg.h>  /* only locally define UNICODE in this compilation unit */#ifndef UNICODE#define UNICODE#endif#ifdef __MINGW32__// Explicitly setting NTDDI version, this is necessary for the MinGW compiler#define NTDDI_VERSION NTDDI_VISTA#define _WIN32_WINNT _WIN32_WINNT_VISTA#endif#include <wchar.h>#include <stdio.h>#include <assert.h>#include <windows.h>#include <shobjidl.h>#include "nfd_common.h"// allocs the space in outPath -- call free()static void CopyWCharToNFDChar( const wchar_t *inStr, nfdchar_t **outStr ){    int inStrCharacterCount = static_cast<int>(wcslen(inStr));     int bytesNeeded = WideCharToMultiByte( CP_UTF8, 0,                                           inStr, inStrCharacterCount,                                           NULL, 0, NULL, NULL );        assert( bytesNeeded );    bytesNeeded += 1;    *outStr = (nfdchar_t*)NFDi_Malloc( bytesNeeded );    if ( !*outStr )        return;    int bytesWritten = WideCharToMultiByte( CP_UTF8, 0,                                            inStr, -1,                                            *outStr, bytesNeeded,                                            NULL, NULL );    assert( bytesWritten ); _NFD_UNUSED( bytesWritten );}/* includes NULL terminator byte in return */static size_t GetUTF8ByteCountForWChar( const wchar_t *str ){    size_t bytesNeeded = WideCharToMultiByte( CP_UTF8, 0,                                              str, -1,                                              NULL, 0, NULL, NULL );    assert( bytesNeeded );    return bytesNeeded+1;}// write to outPtr -- no free() necessary.  No memory stomp tests are done -- they must be done// before entering this function.static int CopyWCharToExistingNFDCharBuffer( const wchar_t *inStr, nfdchar_t *outPtr ){    int inStrCharacterCount = static_cast<int>(wcslen(inStr));    int bytesNeeded = static_cast<int>(GetUTF8ByteCountForWChar( inStr ));    /* invocation copies null term */    int bytesWritten = WideCharToMultiByte( CP_UTF8, 0,                                            inStr, -1,                                            outPtr, bytesNeeded,                                            NULL, 0 );    assert( bytesWritten );    return bytesWritten;}// allocs the space in outStr -- call free()static void CopyNFDCharToWChar( const nfdchar_t *inStr, wchar_t **outStr ){    int inStrByteCount = static_cast<int>(strlen(inStr));    int charsNeeded = MultiByteToWideChar(CP_UTF8, 0,                                          inStr, inStrByteCount,                                          NULL, 0 );        assert( charsNeeded );    assert( !*outStr );    charsNeeded += 1; // terminator        *outStr = (wchar_t*)NFDi_Malloc( charsNeeded * sizeof(wchar_t) );        if ( !*outStr )        return;            int ret = MultiByteToWideChar(CP_UTF8, 0,                                  inStr, inStrByteCount,                                  *outStr, charsNeeded);    (*outStr)[charsNeeded-1] = '\0';#ifdef _DEBUG    int inStrCharacterCount = static_cast<int>(NFDi_UTF8_Strlen(inStr));    assert( ret == inStrCharacterCount );#else    _NFD_UNUSED(ret);#endif}/* ext is in format "jpg", no wildcards or separators */static int AppendExtensionToSpecBuf( const char *ext, char *specBuf, size_t specBufLen ){    const char SEP[] = ";";    assert( specBufLen > strlen(ext)+3 );        if ( strlen(specBuf) > 0 )    {        strncat( specBuf, SEP, specBufLen - strlen(specBuf) - 1 );        specBufLen += strlen(SEP);    }    char extWildcard[NFD_MAX_STRLEN];    int bytesWritten = sprintf_s( extWildcard, NFD_MAX_STRLEN, "*.%s", ext );    assert( bytesWritten == strlen(ext)+2 );        strncat( specBuf, extWildcard, specBufLen - strlen(specBuf) - 1 );    return NFD_OKAY;}static nfdresult_t AddFiltersToDialog( ::IFileDialog *fileOpenDialog, const char *filterList ){    const wchar_t EMPTY_WSTR[] = L"";    const wchar_t WILDCARD[] = L"*.*";    if ( !filterList || strlen(filterList) == 0 )        return NFD_OKAY;    // Count rows to alloc    UINT filterCount = 1; /* guaranteed to have one filter on a correct, non-empty parse */    const char *p_filterList;    for ( p_filterList = filterList; *p_filterList; ++p_filterList )    {        if ( *p_filterList == ';' )            ++filterCount;    }        assert(filterCount);    if ( !filterCount )    {        NFDi_SetError("Error parsing filters.");        return NFD_ERROR;    }    /* filterCount plus 1 because we hardcode the *.* wildcard after the while loop */    COMDLG_FILTERSPEC *specList = (COMDLG_FILTERSPEC*)NFDi_Malloc( sizeof(COMDLG_FILTERSPEC) * ((size_t)filterCount + 1) );    if ( !specList )    {        return NFD_ERROR;    }    for (UINT i = 0; i < filterCount+1; ++i )    {        specList[i].pszName = NULL;        specList[i].pszSpec = NULL;    }    size_t specIdx = 0;    p_filterList = filterList;    char typebuf[NFD_MAX_STRLEN] = {0};  /* one per comma or semicolon */    char *p_typebuf = typebuf;    char filterName[NFD_MAX_STRLEN] = {0};    char specbuf[NFD_MAX_STRLEN] = {0}; /* one per semicolon */    while ( 1 )     {        if ( NFDi_IsFilterSegmentChar(*p_filterList) )        {            /* append a type to the specbuf (pending filter) */            AppendExtensionToSpecBuf( typebuf, specbuf, NFD_MAX_STRLEN );                        p_typebuf = typebuf;            memset( typebuf, 0, sizeof(char)*NFD_MAX_STRLEN );        }        if ( *p_filterList == ';' || *p_filterList == '\0' )        {            /* end of filter -- add it to specList */                                            CopyNFDCharToWChar( specbuf, (wchar_t**)&specList[specIdx].pszName );            CopyNFDCharToWChar( specbuf, (wchar_t**)&specList[specIdx].pszSpec );                                    memset( specbuf, 0, sizeof(char)*NFD_MAX_STRLEN );            ++specIdx;            if ( specIdx == filterCount )                break;        }        if ( !NFDi_IsFilterSegmentChar( *p_filterList ))        {            *p_typebuf = *p_filterList;            ++p_typebuf;        }        ++p_filterList;    }    /* Add wildcard */    specList[specIdx].pszSpec = WILDCARD;    specList[specIdx].pszName = WILDCARD;        fileOpenDialog->SetFileTypes( filterCount+1, specList );    /* free speclist */    for ( size_t i = 0; i < filterCount; ++i )    {        NFDi_Free( (void*)specList[i].pszSpec );    }    NFDi_Free( specList );        return NFD_OKAY;}static nfdresult_t AllocPathSet( IShellItemArray *shellItems, nfdpathset_t *pathSet ){    const char ERRORMSG[] = "Error allocating pathset.";    assert(shellItems);    assert(pathSet);        // How many items in shellItems?    DWORD numShellItems;    HRESULT result = shellItems->GetCount(&numShellItems);    if ( !SUCCEEDED(result) )    {        NFDi_SetError(ERRORMSG);        return NFD_ERROR;    }    pathSet->count = static_cast<size_t>(numShellItems);    assert( pathSet->count > 0 );    pathSet->indices = (size_t*)NFDi_Malloc( sizeof(size_t)*pathSet->count );    if ( !pathSet->indices )    {        return NFD_ERROR;    }    /* count the total bytes needed for buf */    size_t bufSize = 0;    for ( DWORD i = 0; i < numShellItems; ++i )    {        ::IShellItem *shellItem;        result = shellItems->GetItemAt(i, &shellItem);        if ( !SUCCEEDED(result) )        {            NFDi_SetError(ERRORMSG);            return NFD_ERROR;        }        // Confirm SFGAO_FILESYSTEM is true for this shellitem, or ignore it.        SFGAOF attribs;        result = shellItem->GetAttributes( SFGAO_FILESYSTEM, &attribs );        if ( !SUCCEEDED(result) )        {            NFDi_SetError(ERRORMSG);            return NFD_ERROR;        }        if ( !(attribs & SFGAO_FILESYSTEM) )            continue;        LPWSTR name;        shellItem->GetDisplayName(SIGDN_FILESYSPATH, &name);        // Calculate length of name with UTF-8 encoding        bufSize += GetUTF8ByteCountForWChar( name );                CoTaskMemFree(name);    }    assert(bufSize);    pathSet->buf = (nfdchar_t*)NFDi_Malloc( sizeof(nfdchar_t) * bufSize );    memset( pathSet->buf, 0, sizeof(nfdchar_t) * bufSize );    /* fill buf */    nfdchar_t *p_buf = pathSet->buf;    for (DWORD i = 0; i < numShellItems; ++i )    {        ::IShellItem *shellItem;        result = shellItems->GetItemAt(i, &shellItem);        if ( !SUCCEEDED(result) )        {            NFDi_SetError(ERRORMSG);            return NFD_ERROR;        }        // Confirm SFGAO_FILESYSTEM is true for this shellitem, or ignore it.        SFGAOF attribs;        result = shellItem->GetAttributes( SFGAO_FILESYSTEM, &attribs );        if ( !SUCCEEDED(result) )        {            NFDi_SetError(ERRORMSG);            return NFD_ERROR;        }        if ( !(attribs & SFGAO_FILESYSTEM) )            continue;        LPWSTR name;        shellItem->GetDisplayName(SIGDN_FILESYSPATH, &name);        int bytesWritten = CopyWCharToExistingNFDCharBuffer(name, p_buf);        CoTaskMemFree(name);        ptrdiff_t index = p_buf - pathSet->buf;        assert( index >= 0 );        pathSet->indices[i] = static_cast<size_t>(index);                p_buf += bytesWritten;     }         return NFD_OKAY;}static nfdresult_t SetDefaultPath( IFileDialog *dialog, const char *defaultPath ){    if ( !defaultPath || strlen(defaultPath) == 0 )        return NFD_OKAY;    wchar_t *defaultPathW = {0};    CopyNFDCharToWChar( defaultPath, &defaultPathW );    IShellItem *folder;    HRESULT result = SHCreateItemFromParsingName( defaultPathW, NULL, IID_PPV_ARGS(&folder) );    // Valid non results.    if ( result == HRESULT_FROM_WIN32(ERROR_FILE_NOT_FOUND) || result == HRESULT_FROM_WIN32(ERROR_INVALID_DRIVE) )    {        NFDi_Free( defaultPathW );        return NFD_OKAY;    }    if ( !SUCCEEDED(result) )    {        NFDi_SetError("Error creating ShellItem");        NFDi_Free( defaultPathW );        return NFD_ERROR;    }        // Could also call SetDefaultFolder(), but this guarantees defaultPath -- more consistency across API.    dialog->SetFolder( folder );    NFDi_Free( defaultPathW );    folder->Release();        return NFD_OKAY;}/* public */nfdresult_t NFD_OpenDialog( const nfdchar_t *filterList,                            const nfdchar_t *defaultPath,                            nfdchar_t **outPath ){    nfdresult_t nfdResult = NFD_ERROR;        // Init COM library.    HRESULT coResult = ::CoInitializeEx(NULL,                                        ::COINIT_APARTMENTTHREADED |                                        ::COINIT_DISABLE_OLE1DDE );    ::IFileOpenDialog *fileOpenDialog(NULL);    if ( !SUCCEEDED(coResult))    {        fileOpenDialog = NULL;        NFDi_SetError("Could not initialize COM.");        goto end;    }    // Create dialog    HRESULT result = ::CoCreateInstance(::CLSID_FileOpenDialog, NULL,                                        CLSCTX_ALL, ::IID_IFileOpenDialog,                                        reinterpret_cast<void**>(&fileOpenDialog) );                                    if ( !SUCCEEDED(result) )    {        NFDi_SetError("Could not create dialog.");        goto end;    }    // Build the filter list    if ( !AddFiltersToDialog( fileOpenDialog, filterList ) )    {        goto end;    }    // Set the default path    if ( !SetDefaultPath( fileOpenDialog, defaultPath ) )    {        goto end;    }        // Show the dialog.    result = fileOpenDialog->Show(NULL);    if ( SUCCEEDED(result) )    {        // Get the file name        ::IShellItem *shellItem(NULL);        result = fileOpenDialog->GetResult(&shellItem);        if ( !SUCCEEDED(result) )        {            NFDi_SetError("Could not get shell item from dialog.");            goto end;        }        wchar_t *filePath(NULL);        result = shellItem->GetDisplayName(::SIGDN_FILESYSPATH, &filePath);        if ( !SUCCEEDED(result) )        {            NFDi_SetError("Could not get file path for selected.");            shellItem->Release();            goto end;        }        CopyWCharToNFDChar( filePath, outPath );        CoTaskMemFree(filePath);        if ( !*outPath )        {            /* error is malloc-based, error message would be redundant */            shellItem->Release();            goto end;        }        nfdResult = NFD_OKAY;        shellItem->Release();    }    else if (result == HRESULT_FROM_WIN32(ERROR_CANCELLED) )    {        nfdResult = NFD_CANCEL;    }    else    {        NFDi_SetError("File dialog box show failed.");        nfdResult = NFD_ERROR;    }end:    if (fileOpenDialog)        fileOpenDialog->Release();    if (SUCCEEDED(coResult))        ::CoUninitialize();        return nfdResult;}nfdresult_t NFD_OpenDialogMultiple( const nfdchar_t *filterList,                                    const nfdchar_t *defaultPath,                                    nfdpathset_t *outPaths ){    nfdresult_t nfdResult = NFD_ERROR;        // Init COM library.    HRESULT coResult = ::CoInitializeEx(NULL,                                        ::COINIT_APARTMENTTHREADED |                                        ::COINIT_DISABLE_OLE1DDE );    if ( !SUCCEEDED(coResult))    {        NFDi_SetError("Could not initialize COM.");        return NFD_ERROR;    }    ::IFileOpenDialog *fileOpenDialog(NULL);    // Create dialog    HRESULT result = ::CoCreateInstance(::CLSID_FileOpenDialog, NULL,                                        CLSCTX_ALL, ::IID_IFileOpenDialog,                                        reinterpret_cast<void**>(&fileOpenDialog) );                                    if ( !SUCCEEDED(result) )    {        fileOpenDialog = NULL;        NFDi_SetError("Could not create dialog.");        goto end;    }    // Build the filter list    if ( !AddFiltersToDialog( fileOpenDialog, filterList ) )    {        goto end;    }    // Set the default path    if ( !SetDefaultPath( fileOpenDialog, defaultPath ) )    {        goto end;    }    // Set a flag for multiple options    DWORD dwFlags;    result = fileOpenDialog->GetOptions(&dwFlags);    if ( !SUCCEEDED(result) )    {        NFDi_SetError("Could not get options.");        goto end;    }    result = fileOpenDialog->SetOptions(dwFlags | FOS_ALLOWMULTISELECT);    if ( !SUCCEEDED(result) )    {        NFDi_SetError("Could not set options.");        goto end;    }     // Show the dialog.    result = fileOpenDialog->Show(NULL);    if ( SUCCEEDED(result) )    {        IShellItemArray *shellItems;        result = fileOpenDialog->GetResults( &shellItems );        if ( !SUCCEEDED(result) )        {            NFDi_SetError("Could not get shell items.");            goto end;        }                if ( AllocPathSet( shellItems, outPaths ) == NFD_ERROR )        {            shellItems->Release();            goto end;        }        shellItems->Release();        nfdResult = NFD_OKAY;    }    else if (result == HRESULT_FROM_WIN32(ERROR_CANCELLED) )    {        nfdResult = NFD_CANCEL;    }    else    {        NFDi_SetError("File dialog box show failed.");        nfdResult = NFD_ERROR;    }end:    if ( fileOpenDialog )        fileOpenDialog->Release();    if (SUCCEEDED(coResult))        ::CoUninitialize();        return nfdResult;}nfdresult_t NFD_SaveDialog( const nfdchar_t *filterList,                            const nfdchar_t *defaultPath,                            nfdchar_t **outPath ){    nfdresult_t nfdResult = NFD_ERROR;        // Init COM library.    HRESULT coResult = ::CoInitializeEx(NULL,                                        ::COINIT_APARTMENTTHREADED |                                        ::COINIT_DISABLE_OLE1DDE );    if ( !SUCCEEDED(coResult))    {        NFDi_SetError("Could not initialize COM.");        return NFD_ERROR;    }    ::IFileSaveDialog *fileSaveDialog(NULL);    // Create dialog    HRESULT result = ::CoCreateInstance(::CLSID_FileSaveDialog, NULL,                                        CLSCTX_ALL, ::IID_IFileSaveDialog,                                        reinterpret_cast<void**>(&fileSaveDialog) );    if ( !SUCCEEDED(result) )    {        fileSaveDialog = NULL;        NFDi_SetError("Could not create dialog.");        goto end;    }    // Build the filter list    if ( !AddFiltersToDialog( fileSaveDialog, filterList ) )    {        goto end;    }    // Set the default path    if ( !SetDefaultPath( fileSaveDialog, defaultPath ) )    {        goto end;    }    // Show the dialog.    result = fileSaveDialog->Show(NULL);    if ( SUCCEEDED(result) )    {        // Get the file name        ::IShellItem *shellItem;        result = fileSaveDialog->GetResult(&shellItem);        if ( !SUCCEEDED(result) )        {            NFDi_SetError("Could not get shell item from dialog.");            goto end;        }        wchar_t *filePath(NULL);        result = shellItem->GetDisplayName(::SIGDN_FILESYSPATH, &filePath);        if ( !SUCCEEDED(result) )        {            shellItem->Release();            NFDi_SetError("Could not get file path for selected.");            goto end;        }        CopyWCharToNFDChar( filePath, outPath );        CoTaskMemFree(filePath);        if ( !*outPath )        {            /* error is malloc-based, error message would be redundant */            shellItem->Release();            goto end;        }        nfdResult = NFD_OKAY;        shellItem->Release();    }    else if (result == HRESULT_FROM_WIN32(ERROR_CANCELLED) )    {        nfdResult = NFD_CANCEL;    }    else    {        NFDi_SetError("File dialog box show failed.");        nfdResult = NFD_ERROR;    }    end:    if ( fileSaveDialog )        fileSaveDialog->Release();    if (SUCCEEDED(coResult))        ::CoUninitialize();            return nfdResult;}class AutoCoInit{public:    AutoCoInit()    {        mResult = ::CoInitializeEx(NULL,            ::COINIT_APARTMENTTHREADED |            ::COINIT_DISABLE_OLE1DDE);    }    ~AutoCoInit()    {        if (SUCCEEDED(mResult))        {            ::CoUninitialize();        }    }    HRESULT Result() const { return mResult; }private:    HRESULT mResult;};// VS2010 hasn't got a copy of CComPtr - this was first added in the 2003 SDK, so we make our own small CComPtr insteadtemplate<class T>class ComPtr{public:    ComPtr() : mPtr(NULL) { }    ~ComPtr()    {        if (mPtr)        {            mPtr->Release();        }    }    T* Ptr() const { return mPtr; }    T** operator&() { return &mPtr; }    T* operator->() const { return mPtr; }private:    // Don't allow copy or assignment    ComPtr(const ComPtr&);    ComPtr& operator = (const ComPtr&) const;    T* mPtr;};nfdresult_t NFD_PickFolder(const nfdchar_t *defaultPath,    nfdchar_t **outPath){    // Init COM    AutoCoInit autoCoInit;    if (!SUCCEEDED(autoCoInit.Result()))    {        NFDi_SetError("CoInitializeEx failed.");        return NFD_ERROR;    }    // Create the file dialog COM object    ComPtr<IFileDialog> pFileDialog;    if (!SUCCEEDED(CoCreateInstance(CLSID_FileOpenDialog,                                    NULL,                                    CLSCTX_ALL,                                    IID_PPV_ARGS(&pFileDialog))))    {        NFDi_SetError("CoCreateInstance for CLSID_FileOpenDialog failed.");        return NFD_ERROR;    }    // Set the default path    if (SetDefaultPath(pFileDialog.Ptr(), defaultPath) != NFD_OKAY)    {        NFDi_SetError("SetDefaultPath failed.");        return NFD_ERROR;    }    // Get the dialogs options    DWORD dwOptions = 0;    if (!SUCCEEDED(pFileDialog->GetOptions(&dwOptions)))    {        NFDi_SetError("GetOptions for IFileDialog failed.");        return NFD_ERROR;    }    // Add in FOS_PICKFOLDERS which hides files and only allows selection of folders    if (!SUCCEEDED(pFileDialog->SetOptions(dwOptions | FOS_PICKFOLDERS)))    {        NFDi_SetError("SetOptions for IFileDialog failed.");        return NFD_ERROR;    }    // Show the dialog to the user    const HRESULT result = pFileDialog->Show(NULL);    if (result == HRESULT_FROM_WIN32(ERROR_CANCELLED))    {        return NFD_CANCEL;    }    else if (!SUCCEEDED(result))    {        NFDi_SetError("Show for IFileDialog failed.");        return NFD_ERROR;    }    // Get the shell item result    ComPtr<IShellItem> pShellItem;    if (!SUCCEEDED(pFileDialog->GetResult(&pShellItem)))    {        NFDi_SetError("Could not get shell item from dialog.");        return NFD_ERROR;    }    // Finally get the path    wchar_t *path = NULL;    if (!SUCCEEDED(pShellItem->GetDisplayName(SIGDN_DESKTOPABSOLUTEPARSING, &path)))    {        NFDi_SetError("GetDisplayName for IShellItem failed.");        return NFD_ERROR;    }    // Convert string    CopyWCharToNFDChar(path, outPath);    CoTaskMemFree(path);    if (!*outPath)    {        // error is malloc-based, error message would be redundant        return NFD_ERROR;    }    return NFD_OKAY;}
 |