xemu/ui/noc_file_dialog.h

288 lines
9.6 KiB
C

/* noc_file_dialog library
*
* Copyright (c) 2015 Guillaume Chereau <guillaume@noctua-software.com>
*
* 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 <stdlib.h>
#include <string.h>
static char *g_noc_file_dialog_ret = NULL;
#ifdef NOC_FILE_DIALOG_GTK
#include <gtk/gtk.h>
#ifdef GDK_WINDOWING_X11
#include <gdk/gdkx.h>
#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 <windows.h>
#include <commdlg.h>
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 <AppKit/AppKit.h>
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