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; | ||
|  | } |