/* noc_file_dialog library * * Copyright (c) 2015 Guillaume Chereau * * Permission is hereby granted, free of charge, to any person obtaining a copy * of this software and associated documentation files (the "Software"), to * deal in the Software without restriction, including without limitation the * rights to use, copy, modify, merge, publish, distribute, sublicense, and/or * sell copies of the Software, and to permit persons to whom the Software is * furnished to do so, subject to the following conditions: * * The above copyright notice and this permission notice shall be included in * all copies or substantial portions of the Software. * * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS * IN THE SOFTWARE. */ /* A portable library to create open and save dialogs on linux, osx and * windows. * * The library define a single function : noc_file_dialog_open. * With three different implementations. * * Usage: * * The library does not automatically select the implementation, you need to * define one of those macros before including this file: * * NOC_FILE_DIALOG_GTK * NOC_FILE_DIALOG_WIN32 * NOC_FILE_DIALOG_OSX */ enum { NOC_FILE_DIALOG_OPEN = 1 << 0, // Create an open file dialog. NOC_FILE_DIALOG_SAVE = 1 << 1, // Create a save file dialog. NOC_FILE_DIALOG_DIR = 1 << 2, // Open a directory. NOC_FILE_DIALOG_OVERWRITE_CONFIRMATION = 1 << 3, }; // There is a single function defined. /* flags : union of the NOC_FILE_DIALOG_XXX masks. * filters : a list of strings separated by '\0' of the form: * "name1 reg1 name2 reg2 ..." * The last value is followed by two '\0'. For example, * to filter png and jpeg files, you can use: * "png\0*.png\0jpeg\0*.jpeg\0" * You can also separate patterns with ';': * "jpeg\0*.jpg;*.jpeg\0" * Set to NULL for no filter. * default_path : the default file to use or NULL. * default_name : the default file name to use or NULL. * * The function return a C string. There is no need to free it, as it is * managed by the library. The string is valid until the next call to * no_dialog_open. If the user canceled, the return value is NULL. */ const char *noc_file_dialog_open(int flags, const char *filters, const char *default_path, const char *default_name); #ifdef NOC_FILE_DIALOG_IMPLEMENTATION #include #include static char *g_noc_file_dialog_ret = NULL; #ifdef NOC_FILE_DIALOG_GTK #include #ifdef GDK_WINDOWING_X11 #include #endif const char *noc_file_dialog_open(int flags, const char *filters, const char *default_path, const char *default_name) { GtkWidget *dialog; GtkFileFilter *filter; GtkFileChooser *chooser; GtkFileChooserAction action; gint res; char buf[128], *patterns; action = flags & NOC_FILE_DIALOG_SAVE ? GTK_FILE_CHOOSER_ACTION_SAVE : GTK_FILE_CHOOSER_ACTION_OPEN; if (flags & NOC_FILE_DIALOG_DIR) action = GTK_FILE_CHOOSER_ACTION_SELECT_FOLDER; gtk_init_check(NULL, NULL); dialog = gtk_file_chooser_dialog_new( flags & NOC_FILE_DIALOG_SAVE ? "Save File" : "Open File", NULL, action, "_Cancel", GTK_RESPONSE_CANCEL, flags & NOC_FILE_DIALOG_SAVE ? "_Save" : "_Open", GTK_RESPONSE_ACCEPT, NULL ); chooser = GTK_FILE_CHOOSER(dialog); if (flags & NOC_FILE_DIALOG_OVERWRITE_CONFIRMATION) gtk_file_chooser_set_do_overwrite_confirmation(chooser, TRUE); if (default_path) gtk_file_chooser_set_filename(chooser, default_path); if (default_name) gtk_file_chooser_set_current_name(chooser, default_name); while (filters && *filters) { filter = gtk_file_filter_new(); gtk_file_filter_set_name(filter, filters); filters += strlen(filters) + 1; // Split the filter pattern with ';'. strcpy(buf, filters); buf[strlen(buf)] = '\0'; for (patterns = buf; *patterns; patterns++) if (*patterns == ';') *patterns = '\0'; patterns = buf; while (*patterns) { gtk_file_filter_add_pattern(filter, patterns); patterns += strlen(patterns) + 1; } gtk_file_chooser_add_filter(chooser, filter); filters += strlen(filters) + 1; } gtk_widget_show_all(dialog); #ifdef GDK_WINDOWING_X11 if (GDK_IS_X11_DISPLAY(gdk_display_get_default())) { GdkWindow *window = gtk_widget_get_window(dialog); gdk_window_set_events(window, gdk_window_get_events(window) | GDK_PROPERTY_CHANGE_MASK); gtk_window_present_with_time(GTK_WINDOW(dialog), gdk_x11_get_server_time(window)); } #endif res = gtk_dialog_run(GTK_DIALOG(dialog)); free(g_noc_file_dialog_ret); g_noc_file_dialog_ret = NULL; if (res == GTK_RESPONSE_ACCEPT) g_noc_file_dialog_ret = gtk_file_chooser_get_filename(chooser); gtk_widget_destroy(dialog); while (gtk_events_pending()) gtk_main_iteration(); return g_noc_file_dialog_ret; } #endif #ifdef NOC_FILE_DIALOG_WIN32 #include #include const char *noc_file_dialog_open(int flags, const char *filters, const char *default_path, const char *default_name) { OPENFILENAME ofn; // common dialog box structure char szFile[_MAX_PATH]; // buffer for file name char initialDir[_MAX_PATH]; char drive[_MAX_DRIVE]; char dir[_MAX_DIR]; char fname[_MAX_FNAME]; char ext[_MAX_EXT]; int ret; // init default dir and file name _splitpath_s(default_path, drive, _MAX_DRIVE, dir, _MAX_DIR, fname, _MAX_FNAME, ext, _MAX_EXT ); _makepath_s(initialDir, _MAX_PATH, drive, dir, NULL, NULL); if (default_name) strncpy(szFile, default_name, sizeof(szFile) - 1); else _makepath_s(szFile, _MAX_PATH, NULL, NULL, fname, ext); ZeroMemory(&ofn, sizeof(ofn)); ofn.lStructSize = sizeof(ofn); ofn.lpstrFile = szFile; ofn.nMaxFile = sizeof(szFile); ofn.lpstrFilter = filters; ofn.nFilterIndex = 1; ofn.lpstrFileTitle = NULL; ofn.nMaxFileTitle = 0; ofn.lpstrInitialDir = initialDir; ofn.Flags = OFN_PATHMUSTEXIST | OFN_FILEMUSTEXIST | OFN_NOCHANGEDIR; if (flags & NOC_FILE_DIALOG_OPEN) ret = GetOpenFileName(&ofn); else ret = GetSaveFileName(&ofn); free(g_noc_file_dialog_ret); g_noc_file_dialog_ret = ret ? strdup(szFile) : NULL; return g_noc_file_dialog_ret; } #endif #ifdef NOC_FILE_DIALOG_OSX #include const char *noc_file_dialog_open(int flags, const char *filters, const char *default_path, const char *default_name) { NSURL *url; const char *utf8_path; NSSavePanel *panel; NSOpenPanel *open_panel; NSMutableArray *types_array; NSURL *default_url; char buf[128], *patterns; // XXX: I don't know about memory management with cococa, need to check // if I leak memory here. NSAutoreleasePool *pool = [[NSAutoreleasePool alloc] init]; if (flags & NOC_FILE_DIALOG_OPEN) { panel = open_panel = [NSOpenPanel openPanel]; } else { panel = [NSSavePanel savePanel]; } if (flags & NOC_FILE_DIALOG_DIR) { [open_panel setCanChooseDirectories:YES]; [open_panel setCanChooseFiles:NO]; } if (default_path && (strlen(default_path) > 0)) { default_url = [NSURL fileURLWithPath: [NSString stringWithUTF8String:default_path]]; [panel setDirectoryURL:default_url]; [panel setNameFieldStringValue:default_url.lastPathComponent]; } if (filters) { types_array = [NSMutableArray array]; while (*filters) { filters += strlen(filters) + 1; // skip the name // Split the filter pattern with ';'. strcpy(buf, filters); buf[strlen(buf) + 1] = '\0'; for (patterns = buf; *patterns; patterns++) if (*patterns == ';') *patterns = '\0'; patterns = buf; while (*patterns) { assert(strncmp(patterns, "*.", 2) == 0); patterns += 2; // Skip the "*." [types_array addObject:[NSString stringWithUTF8String: patterns]]; patterns += strlen(patterns) + 1; } filters += strlen(filters) + 1; } [panel setAllowedFileTypes:types_array]; } free(g_noc_file_dialog_ret); g_noc_file_dialog_ret = NULL; if ( [panel runModal] == NSModalResponseOK ) { url = [panel URL]; utf8_path = [[url path] UTF8String]; g_noc_file_dialog_ret = strdup(utf8_path); } [pool release]; return g_noc_file_dialog_ret; } #endif #endif