eco2d/code/vendors/sfd.c

262 lines
5.9 KiB
C

/*
* Copyright (c) 2017 rxi
*
* 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.
*/
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <limits.h>
#include "sfd.h"
static const char *last_error;
const char* sfd_get_error(void) {
const char *res = last_error;
last_error = NULL;
return res;
}
static int next_filter(char *dst, const char **p) {
int len;
*p += strspn(*p, "|");
if (**p == '\0') {
return 0;
}
len = strcspn(*p, "|");
memcpy(dst, *p, len);
dst[len] = '\0';
*p += len;
return 1;
}
/******************************************************************************
** Windows
*******************************************************************************/
#ifdef _WIN32
#include <windows.h>
typedef struct {
unsigned long process_id;
void* handle_root;
void* handle_first;
} FindMainWindowInfo;
static int find_main_window_callback(HWND handle, LPARAM lParam) {
FindMainWindowInfo* info = (FindMainWindowInfo*)lParam;
unsigned long process_id = 0;
GetWindowThreadProcessId(handle, &process_id);
if (info->process_id == process_id) {
info->handle_first = handle;
if (GetWindow(handle, GW_OWNER) == 0 && IsWindowVisible(handle)) {
info->handle_root = handle;
return 0;
}
}
return 1;
}
static HWND find_main_window() {
FindMainWindowInfo info = {
.process_id = GetCurrentProcessId()
};
EnumWindows(find_main_window_callback, (LPARAM)&info);
return info.handle_root;
}
static const char* make_filter_str(sfd_Options *opt) {
static char buf[1024];
int n;
buf[0] = '\0';
n = 0;
if (opt->filter) {
const char *p;
char b[32];
const char *name = opt->filter_name ? opt->filter_name : opt->filter;
n += sprintf(buf + n, "%s", name) + 1;
p = opt->filter;
while (next_filter(b, &p)) {
n += sprintf(buf + n, "%s;", b);
}
buf[++n] = '\0';
}
n += sprintf(buf + n, "All Files") + 1;
n += sprintf(buf + n, "*.*");
buf[++n] = '\0';
return buf;
}
static void init_ofn(OPENFILENAME *ofn, sfd_Options *opt) {
static char result_buf[2048];
result_buf[0] = '\0';
memset(ofn, 0, sizeof(*ofn));
ofn->hwndOwner = find_main_window();
ofn->lStructSize = sizeof(*ofn);
ofn->lpstrFilter = make_filter_str(opt);
ofn->nFilterIndex = 1;
ofn->lpstrFile = result_buf;
ofn->Flags = OFN_HIDEREADONLY | OFN_OVERWRITEPROMPT;
ofn->nMaxFile = sizeof(result_buf) - 1;
ofn->lpstrInitialDir = opt->path;
ofn->lpstrTitle = opt->title;
ofn->lpstrDefExt = opt->extension;
}
const char* sfd_open_dialog(sfd_Options *opt) {
int ok;
OPENFILENAME ofn;
last_error = NULL;
init_ofn(&ofn, opt);
ok = GetOpenFileName(&ofn);
return ok ? ofn.lpstrFile : NULL;
}
const char* sfd_save_dialog(sfd_Options *opt) {
int ok;
OPENFILENAME ofn;
last_error = NULL;
init_ofn(&ofn, opt);
ok = GetSaveFileName(&ofn);
return ok ? ofn.lpstrFile : NULL;
}
#endif
/******************************************************************************
** Zenity
*******************************************************************************/
#ifndef _WIN32
static const char* file_dialog(sfd_Options *opt, int save) {
static char result_buf[2048];
char buf[2048];
char *p;
const char *title;
FILE *fp;
int n, len;
last_error = NULL;
fp = popen("zenity --version", "r");
if (fp == NULL || pclose(fp) != 0) {
last_error = "could not open zenity";
return NULL;
}
n = sprintf(buf, "zenity --file-selection");
if (save) {
n += sprintf(buf + n, " --save --confirm-overwrite");
}
if (opt->title) {
title = opt->title;
} else {
title = save ? "Save File" : "Open File";
}
n += sprintf(buf + n, " --title=\"%s\"", title);
if (opt->path && opt->path[0] != '\0') {
n += sprintf(buf + n, " --filename=\"");
p = realpath(opt->path, buf + n);
if (p == NULL) {
last_error = "call to realpath() failed";
return NULL;
}
n += strlen(buf + n);
n += sprintf(buf + n, "/\"");
}
if (opt->filter) {
char b[64];
const char *p;
n += sprintf(buf + n, " --file-filter=\"");
if (opt->filter_name) {
n += sprintf(buf + n, "%s | ", opt->filter_name);
}
p = opt->filter;
while (next_filter(b, &p)) {
n += sprintf(buf + n, "\"%s\" ", b);
}
n += sprintf(buf + n, "\"");
}
n += sprintf(buf + n, " --file-filter=\"All Files | *\"");
fp = popen(buf, "r");
len = fread(result_buf, 1, sizeof(result_buf) - 1, fp);
pclose(fp);
if (len > 0) {
result_buf[len - 1] = '\0';
if (save && opt->extension && !strstr(result_buf, opt->extension)) {
sprintf(&result_buf[len - 1], ".%s", opt->extension);
}
return result_buf;
}
return NULL;
}
const char* sfd_open_dialog(sfd_Options *opt) {
return file_dialog(opt, 0);
}
const char* sfd_save_dialog(sfd_Options *opt) {
return file_dialog(opt, 1);
}
#endif