308 lines
		
	
	
		
			7.5 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			308 lines
		
	
	
		
			7.5 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
/*
 | 
						|
  Native File Dialog
 | 
						|
 | 
						|
  http://www.frogtoss.com/labs
 | 
						|
*/
 | 
						|
 | 
						|
#include <stdio.h>
 | 
						|
#include <assert.h>
 | 
						|
#include <string.h>
 | 
						|
#include "nfd.h"
 | 
						|
#include "nfd_common.h"
 | 
						|
 | 
						|
#define SIMPLE_EXEC_IMPLEMENTATION
 | 
						|
#include "simple_exec.h"
 | 
						|
 | 
						|
 | 
						|
const char NO_ZENITY_MSG[] = "zenity not installed";
 | 
						|
 | 
						|
 | 
						|
static void AddTypeToFilterName( const char *typebuf, char *filterName, size_t bufsize )
 | 
						|
{
 | 
						|
    size_t len = strlen(filterName);
 | 
						|
    if( len > 0 )
 | 
						|
        strncat( filterName, " *.", bufsize - len - 1 );
 | 
						|
    else
 | 
						|
        strncat( filterName, "--file-filter=*.", bufsize - len - 1 );
 | 
						|
    
 | 
						|
    len = strlen(filterName);
 | 
						|
    strncat( filterName, typebuf, bufsize - len - 1 );
 | 
						|
}
 | 
						|
 | 
						|
static void AddFiltersToCommandArgs(char** commandArgs, int commandArgsLen, const char *filterList )
 | 
						|
{
 | 
						|
    char typebuf[NFD_MAX_STRLEN] = {0};
 | 
						|
    const char *p_filterList = filterList;
 | 
						|
    char *p_typebuf = typebuf;
 | 
						|
    char filterName[NFD_MAX_STRLEN] = {0};
 | 
						|
    int i;
 | 
						|
    
 | 
						|
    if ( !filterList || strlen(filterList) == 0 )
 | 
						|
        return;
 | 
						|
 | 
						|
    while ( 1 )
 | 
						|
    {
 | 
						|
        
 | 
						|
        if ( NFDi_IsFilterSegmentChar(*p_filterList) )
 | 
						|
        {
 | 
						|
            char typebufWildcard[NFD_MAX_STRLEN];
 | 
						|
            /* add another type to the filter */
 | 
						|
            assert( strlen(typebuf) > 0 );
 | 
						|
            assert( strlen(typebuf) < NFD_MAX_STRLEN-1 );
 | 
						|
            
 | 
						|
            snprintf( typebufWildcard, NFD_MAX_STRLEN, "*.%s", typebuf );
 | 
						|
 | 
						|
            AddTypeToFilterName( typebuf, filterName, 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 the dialog */
 | 
						|
 | 
						|
            for(i = 0; commandArgs[i] != NULL && i < commandArgsLen; i++);
 | 
						|
 | 
						|
            commandArgs[i] = strdup(filterName);
 | 
						|
            
 | 
						|
            filterName[0] = '\0';
 | 
						|
 | 
						|
            if ( *p_filterList == '\0' )
 | 
						|
                break;
 | 
						|
        }
 | 
						|
 | 
						|
        if ( !NFDi_IsFilterSegmentChar( *p_filterList ) )
 | 
						|
        {
 | 
						|
            *p_typebuf = *p_filterList;
 | 
						|
            p_typebuf++;
 | 
						|
        }
 | 
						|
 | 
						|
        p_filterList++;
 | 
						|
    }
 | 
						|
    
 | 
						|
    /* always append a wildcard option to the end*/
 | 
						|
    
 | 
						|
    for(i = 0; commandArgs[i] != NULL && i < commandArgsLen; i++);
 | 
						|
 | 
						|
    commandArgs[i] = strdup("--file-filter=*.*");
 | 
						|
}
 | 
						|
 | 
						|
static nfdresult_t ZenityCommon(char** command, int commandLen, const char* defaultPath, const char* filterList, char** stdOut)
 | 
						|
{
 | 
						|
    if(defaultPath != NULL)
 | 
						|
    {
 | 
						|
        char* prefix = "--filename=";
 | 
						|
        int len = strlen(prefix) + strlen(defaultPath) + 1;
 | 
						|
 | 
						|
        char* tmp = (char*) calloc(len, 1);
 | 
						|
        strcat(tmp, prefix);
 | 
						|
        strcat(tmp, defaultPath);
 | 
						|
 | 
						|
        int i;
 | 
						|
        for(i = 0; command[i] != NULL && i < commandLen; i++);
 | 
						|
 | 
						|
        command[i] = tmp;
 | 
						|
    }
 | 
						|
 | 
						|
    AddFiltersToCommandArgs(command, commandLen, filterList);
 | 
						|
 | 
						|
    int byteCount = 0;
 | 
						|
    int exitCode = 0;
 | 
						|
    int processInvokeError = runCommandArray(stdOut, &byteCount, &exitCode, 0, command);
 | 
						|
 | 
						|
    for(int i = 0; command[i] != NULL && i < commandLen; i++)
 | 
						|
        free(command[i]);
 | 
						|
 | 
						|
    nfdresult_t result = NFD_OKAY;
 | 
						|
 | 
						|
    if(processInvokeError == COMMAND_NOT_FOUND)
 | 
						|
    {
 | 
						|
        NFDi_SetError(NO_ZENITY_MSG);
 | 
						|
        result = NFD_ERROR;
 | 
						|
    }
 | 
						|
    else
 | 
						|
    {
 | 
						|
        if(exitCode == 1)
 | 
						|
            result = NFD_CANCEL;
 | 
						|
    }
 | 
						|
 | 
						|
    return result;
 | 
						|
}
 | 
						|
 
 | 
						|
 | 
						|
static nfdresult_t AllocPathSet(char* zenityList, nfdpathset_t *pathSet )
 | 
						|
{
 | 
						|
    assert(zenityList);
 | 
						|
    assert(pathSet);
 | 
						|
    
 | 
						|
    size_t len = strlen(zenityList) + 1;
 | 
						|
    pathSet->buf = NFDi_Malloc(len);
 | 
						|
 | 
						|
    int numEntries = 1;
 | 
						|
 | 
						|
    for(size_t i = 0; i < len; i++)
 | 
						|
    {
 | 
						|
        char ch = zenityList[i];
 | 
						|
 | 
						|
        if(ch == '|')
 | 
						|
        {
 | 
						|
            numEntries++;
 | 
						|
            ch = '\0';
 | 
						|
        }
 | 
						|
 | 
						|
        pathSet->buf[i] = ch;
 | 
						|
    }
 | 
						|
 | 
						|
    pathSet->count = numEntries;
 | 
						|
    assert( pathSet->count > 0 );
 | 
						|
 | 
						|
    pathSet->indices = NFDi_Malloc( sizeof(size_t)*pathSet->count );
 | 
						|
 | 
						|
    int entry = 0;
 | 
						|
    pathSet->indices[0] = 0;
 | 
						|
    for(size_t i = 0; i < len; i++)
 | 
						|
    {
 | 
						|
        char ch = zenityList[i];
 | 
						|
 | 
						|
        if(ch == '|')
 | 
						|
        {
 | 
						|
            entry++;
 | 
						|
            pathSet->indices[entry] = i + 1;
 | 
						|
        }
 | 
						|
    }
 | 
						|
    
 | 
						|
    return NFD_OKAY;
 | 
						|
}
 | 
						|
                                 
 | 
						|
/* public */
 | 
						|
 | 
						|
nfdresult_t NFD_OpenDialog( const char *filterList,
 | 
						|
                            const nfdchar_t *defaultPath,
 | 
						|
                            nfdchar_t **outPath )
 | 
						|
{    
 | 
						|
    int commandLen = 100;
 | 
						|
    char* command[commandLen];
 | 
						|
    memset(command, 0, commandLen * sizeof(char*));
 | 
						|
 | 
						|
    command[0] = strdup("zenity");
 | 
						|
    command[1] = strdup("--file-selection");
 | 
						|
    command[2] = strdup("--title=Open File");
 | 
						|
 | 
						|
    char* stdOut = NULL;
 | 
						|
    nfdresult_t result = ZenityCommon(command, commandLen, defaultPath, filterList, &stdOut);
 | 
						|
            
 | 
						|
    if(stdOut != NULL)
 | 
						|
    {
 | 
						|
        size_t len = strlen(stdOut);
 | 
						|
        *outPath = NFDi_Malloc(len);
 | 
						|
        memcpy(*outPath, stdOut, len);
 | 
						|
        (*outPath)[len-1] = '\0'; // trim out the final \n with a null terminator
 | 
						|
        free(stdOut);
 | 
						|
    }
 | 
						|
    else
 | 
						|
    {
 | 
						|
        *outPath = NULL;
 | 
						|
    }
 | 
						|
 | 
						|
    return result;
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
nfdresult_t NFD_OpenDialogMultiple( const nfdchar_t *filterList,
 | 
						|
                                    const nfdchar_t *defaultPath,
 | 
						|
                                    nfdpathset_t *outPaths )
 | 
						|
{
 | 
						|
    int commandLen = 100;
 | 
						|
    char* command[commandLen];
 | 
						|
    memset(command, 0, commandLen * sizeof(char*));
 | 
						|
 | 
						|
    command[0] = strdup("zenity");
 | 
						|
    command[1] = strdup("--file-selection");
 | 
						|
    command[2] = strdup("--title=Open Files");
 | 
						|
    command[3] = strdup("--multiple");
 | 
						|
 | 
						|
    char* stdOut = NULL;
 | 
						|
    nfdresult_t result = ZenityCommon(command, commandLen, defaultPath, filterList, &stdOut);
 | 
						|
            
 | 
						|
    if(stdOut != NULL)
 | 
						|
    {
 | 
						|
        size_t len = strlen(stdOut);
 | 
						|
        stdOut[len-1] = '\0'; // remove trailing newline
 | 
						|
 | 
						|
        if ( AllocPathSet( stdOut, outPaths ) == NFD_ERROR )
 | 
						|
            result = NFD_ERROR;
 | 
						|
 | 
						|
        free(stdOut);
 | 
						|
    }
 | 
						|
    else
 | 
						|
    {
 | 
						|
        result = NFD_ERROR;
 | 
						|
    }
 | 
						|
 | 
						|
    return result;
 | 
						|
}
 | 
						|
 | 
						|
nfdresult_t NFD_SaveDialog( const nfdchar_t *filterList,
 | 
						|
                            const nfdchar_t *defaultPath,
 | 
						|
                            nfdchar_t **outPath )
 | 
						|
{
 | 
						|
    int commandLen = 100;
 | 
						|
    char* command[commandLen];
 | 
						|
    memset(command, 0, commandLen * sizeof(char*));
 | 
						|
 | 
						|
    command[0] = strdup("zenity");
 | 
						|
    command[1] = strdup("--file-selection");
 | 
						|
    command[2] = strdup("--title=Save File");
 | 
						|
    command[3] = strdup("--save");
 | 
						|
 | 
						|
    char* stdOut = NULL;
 | 
						|
    nfdresult_t result = ZenityCommon(command, commandLen, defaultPath, filterList, &stdOut);
 | 
						|
            
 | 
						|
    if(stdOut != NULL)
 | 
						|
    {
 | 
						|
        size_t len = strlen(stdOut);
 | 
						|
        *outPath = NFDi_Malloc(len);
 | 
						|
        memcpy(*outPath, stdOut, len);
 | 
						|
        (*outPath)[len-1] = '\0'; // trim out the final \n with a null terminator
 | 
						|
        free(stdOut);
 | 
						|
    }
 | 
						|
    else
 | 
						|
    {
 | 
						|
        *outPath = NULL;
 | 
						|
    }
 | 
						|
 | 
						|
    return result;
 | 
						|
}
 | 
						|
 | 
						|
nfdresult_t NFD_PickFolder(const nfdchar_t *defaultPath,
 | 
						|
    nfdchar_t **outPath)
 | 
						|
{
 | 
						|
    int commandLen = 100;
 | 
						|
    char* command[commandLen];
 | 
						|
    memset(command, 0, commandLen * sizeof(char*));
 | 
						|
 | 
						|
    command[0] = strdup("zenity");
 | 
						|
    command[1] = strdup("--file-selection");
 | 
						|
    command[2] = strdup("--directory");
 | 
						|
    command[3] = strdup("--title=Select folder");
 | 
						|
 | 
						|
    char* stdOut = NULL;
 | 
						|
    nfdresult_t result = ZenityCommon(command, commandLen, defaultPath, "", &stdOut);
 | 
						|
            
 | 
						|
    if(stdOut != NULL)
 | 
						|
    {
 | 
						|
        size_t len = strlen(stdOut);
 | 
						|
        *outPath = NFDi_Malloc(len);
 | 
						|
        memcpy(*outPath, stdOut, len);
 | 
						|
        (*outPath)[len-1] = '\0'; // trim out the final \n with a null terminator
 | 
						|
        free(stdOut);
 | 
						|
    }
 | 
						|
    else
 | 
						|
    {
 | 
						|
        *outPath = NULL;
 | 
						|
    }
 | 
						|
 | 
						|
    return result;
 | 
						|
}
 |