cc205624创建于 2025年9月13日历史提交
#include <stdlib.h>
#include <stdio.h>
#include <string.h>
#include <errno.h>
#include <unistd.h>
#include <sys/wait.h> // for waitpid
#include "osdialog.h"


static const char zenityBin[] = "zenity";


static void string_list_clear(char** list) {
	while (*list) {
		OSDIALOG_FREE(*list);
		*list = NULL;
		list++;
	}
}


static int string_list_exec(const char* path, const char* const* args, char* outBuf, size_t outLen, char* errBuf, size_t errLen) {
	int outStream[2];
	if (outBuf)
		if (pipe(outStream) == -1)
			return -1;

	int errStream[2];
	if (errBuf)
		if (pipe(errStream) == -1)
			return -1;

	// The classic fork-and-exec routine
	pid_t pid = fork();
	if (pid < 0) {
		return -1;
	}
	else if (pid == 0) {
		// child process
		// Route stdout to outStream
		if (outBuf) {
			while ((dup2(outStream[1], STDOUT_FILENO) == -1) && (errno == EINTR)) {}
			close(outStream[0]);
			close(outStream[1]);
		}
		// Route stdout to outStream
		if (errBuf) {
			while ((dup2(errStream[1], STDERR_FILENO) == -1) && (errno == EINTR)) {}
			close(errStream[0]);
			close(errStream[1]);
		}
		// POSIX guarantees that execvp does not modify the args array, so it's safe to remove const with a cast.
		int err = execvp(path, (char* const*) args);
		if (err)
			exit(0);
		// Will never reach
		exit(0);
	}

	// parent process
	// Close the pipe inputs because the parent doesn't need them
	if (outBuf)
		close(outStream[1]);
	if (errBuf)
		close(errStream[1]);
	// Wait for child process to close
	int status = -1;
	int options = 0;
	waitpid(pid, &status, options);
	// Read streams
	if (outBuf) {
		ssize_t count = read(outStream[0], outBuf, outLen - 1);
		if (count < 0)
			count = 0;
		outBuf[count] = '\0';
		close(outStream[0]);
	}
	if (errBuf) {
		ssize_t count = read(errStream[0], errBuf, errLen - 1);
		if (count < 0)
			count = 0;
		errBuf[count] = '\0';
		close(errStream[0]);
	}
	return status;
}


int osdialog_message(osdialog_message_level level, osdialog_message_buttons buttons, const char* message) {
	char* args[32];
	int argIndex = 0;

	args[argIndex++] = osdialog_strdup(zenityBin);
	// The API doesn't provide a title, so just make it blank.
	args[argIndex++] = osdialog_strdup("--title");
	args[argIndex++] = osdialog_strdup("");

	args[argIndex++] = osdialog_strdup("--no-markup");

	args[argIndex++] = osdialog_strdup("--width");
	args[argIndex++] = osdialog_strdup("500");

	if (buttons == OSDIALOG_OK_CANCEL) {
		args[argIndex++] = osdialog_strdup("--question");
		args[argIndex++] = osdialog_strdup("--ok-label");
		args[argIndex++] = osdialog_strdup("OK");
		args[argIndex++] = osdialog_strdup("--cancel-label");
		args[argIndex++] = osdialog_strdup("Cancel");
	}
	else if (buttons == OSDIALOG_YES_NO) {
		args[argIndex++] = osdialog_strdup("--question");
		args[argIndex++] = osdialog_strdup("--ok-label");
		args[argIndex++] = osdialog_strdup("Yes");
		args[argIndex++] = osdialog_strdup("--cancel-label");
		args[argIndex++] = osdialog_strdup("No");
	}
	else if (level == OSDIALOG_INFO) {
		args[argIndex++] = osdialog_strdup("--info");
	}
	else if (level == OSDIALOG_WARNING) {
		args[argIndex++] = osdialog_strdup("--warning");
	}
	else if (level == OSDIALOG_ERROR) {
		args[argIndex++] = osdialog_strdup("--error");
	}

	args[argIndex++] = osdialog_strdup("--text");
	args[argIndex++] = osdialog_strdup(message);

	args[argIndex++] = NULL;
	int ret = string_list_exec(zenityBin, (const char* const*) args, NULL, 0, NULL, 0);
	string_list_clear(args);
	return ret == 0;
}


void osdialog_message_async(osdialog_message_level level, osdialog_message_buttons buttons, const char* message, osdialog_message_callback* cb, void* user) {
	// Fake async placeholder
	int result = osdialog_message(level, buttons, message);
	if (cb) {
		cb(result, user);
	}
}


char* osdialog_prompt(osdialog_message_level level, const char* message, const char* text) {
	(void) level;
	char* args[32];
	int argIndex = 0;

	args[argIndex++] = osdialog_strdup(zenityBin);
	args[argIndex++] = osdialog_strdup("--title");
	args[argIndex++] = osdialog_strdup("");
	args[argIndex++] = osdialog_strdup("--entry");
	args[argIndex++] = osdialog_strdup("--text");
	args[argIndex++] = osdialog_strdup(message ? message : "");
	args[argIndex++] = osdialog_strdup("--entry-text");
	args[argIndex++] = osdialog_strdup(text ? text : "");
	// Unfortunately the level is ignored

	args[argIndex++] = NULL;
	char outBuf[4096 + 1];
	int ret = string_list_exec(zenityBin, (const char* const*) args, outBuf, sizeof(outBuf), NULL, 0);
	string_list_clear(args);
	if (ret != 0)
		return NULL;

	// Remove trailing newline
	size_t outLen = strlen(outBuf);
	if (outLen > 0)
		outBuf[outLen - 1] = '\0';
	return osdialog_strdup(outBuf);
}


void osdialog_prompt_async(osdialog_message_level level, const char* message, const char* text, osdialog_prompt_callback* cb, void* user) {
	// Fake async placeholder
	char* result = osdialog_prompt(level, message, text);
	if (cb) {
		cb(result, user);
	}
	else {
		OSDIALOG_FREE(result);
	}
}


static int supports_confirm_overwrite() {
	char* args[32];
	int argIndex = 0;

	args[argIndex++] = osdialog_strdup(zenityBin);
	args[argIndex++] = osdialog_strdup("--help-file-selection");
	args[argIndex++] = NULL;

	char outBuf[1 << 12];
	int ret = string_list_exec(zenityBin, (const char* const*) args, outBuf, sizeof(outBuf), NULL, 0);
	string_list_clear(args);

	if (ret != 0)
		return 0;

	return strstr(outBuf, "--confirm-overwrite") != NULL;
}


static int supports_confirm_overwrite_cached = -1;


char* osdialog_file(osdialog_file_action action, const char* dir, const char* filename, const osdialog_filters* filters) {
	char* args[32];
	int argIndex = 0;

	args[argIndex++] = osdialog_strdup(zenityBin);
	args[argIndex++] = osdialog_strdup("--title");
	args[argIndex++] = osdialog_strdup("");
	args[argIndex++] = osdialog_strdup("--file-selection");
	if (action == OSDIALOG_OPEN) {
		// This is the default
	}
	else if (action == OSDIALOG_OPEN_DIR) {
		args[argIndex++] = osdialog_strdup("--directory");
	}
	else if (action == OSDIALOG_SAVE) {
		args[argIndex++] = osdialog_strdup("--save");

		// --confirm-overwrite was (accidentally?) removed in Zenity 3.91.0, so using this flag causes Zenity to fail.
		if (supports_confirm_overwrite_cached < 0) {
			supports_confirm_overwrite_cached = supports_confirm_overwrite();
		}
		if (supports_confirm_overwrite_cached) {
			args[argIndex++] = osdialog_strdup("--confirm-overwrite");
		}
	}

	if (dir || filename) {
		args[argIndex++] = osdialog_strdup("--filename");
		char buf[4096];
		if (dir) {
			// Hack: If filename is not given, zenity will open in the parent dir. To avoid this, use a nonsense default filename.
			if (filename && filename[0])
				snprintf(buf, sizeof(buf), "%s/%s", dir, filename);
			else
				snprintf(buf, sizeof(buf), "%s/?", dir);
		}
		else {
			snprintf(buf, sizeof(buf), "%s", filename);
		}
		args[argIndex++] = osdialog_strdup(buf);
	}

	for (const osdialog_filters* filter = filters; filter; filter = filter->next) {
		args[argIndex++] = osdialog_strdup("--file-filter");

		// Set pattern name
		char patternBuf[1024];
		char* patternPtr = patternBuf;
		const char* patternEnd = patternBuf + sizeof(patternBuf);
		int len = snprintf(patternPtr, patternEnd - patternPtr, "%s |", filter->name);
		if (len < 0)
			continue;
		patternPtr += len;

		// Append pattern
		for (const osdialog_filter_patterns* pattern = filter->patterns; pattern; pattern = pattern->next) {
			if (patternPtr >= patternEnd)
				break;
			int len = snprintf(patternPtr, patternEnd - patternPtr, " *.%s", pattern->pattern);
			if (len < 0)
				continue;
			patternPtr += len;
		}
		args[argIndex++] = osdialog_strdup(patternBuf);
	}

	args[argIndex++] = NULL;
	char outBuf[4096 + 1];
	int ret = string_list_exec(zenityBin, (const char* const*) args, outBuf, sizeof(outBuf), NULL, 0);
	string_list_clear(args);
	if (ret != 0)
		return NULL;

	// Remove trailing newline
	size_t outLen = strlen(outBuf);
	if (outLen > 0)
		outBuf[outLen - 1] = '\0';
	return osdialog_strdup(outBuf);
}


void osdialog_file_async(osdialog_file_action action, const char* path, const char* filename, const osdialog_filters* filters, osdialog_file_callback* cb, void* user) {
	// Fake async placeholder
	char* result = osdialog_file(action, path, filename, filters);
	if (cb) {
		cb(result, user);
	}
	else {
		OSDIALOG_FREE(result);
	}
}


int osdialog_color_picker(osdialog_color* color, int opacity) {
	char* args[32];
	int argIndex = 0;

	args[argIndex++] = osdialog_strdup(zenityBin);
	args[argIndex++] = osdialog_strdup("--title");
	args[argIndex++] = osdialog_strdup("");
	args[argIndex++] = osdialog_strdup("--color-selection");

	if (!opacity) {
		color->a = 255;
	}

	// Convert osdialog_color to string
	char buf[128];
	snprintf(buf, sizeof(buf), "rgba(%d,%d,%d,%f)", color->r, color->g, color->b, color->a / 255.f);
	args[argIndex++] = osdialog_strdup("--color");
	args[argIndex++] = osdialog_strdup(buf);

	args[argIndex++] = NULL;
	int ret = string_list_exec(zenityBin, (const char* const*) args, buf, sizeof(buf), NULL, 0);
	string_list_clear(args);
	if (ret != 0)
		return 0;

	// Convert string to osdialog_color
	int r = 0, g = 0, b = 0;
	float a = 1.f;
	if (buf[3] == 'a') {
		sscanf(buf, "rgba(%d,%d,%d,%f)", &r, &g, &b, &a);
	}
	else {
		sscanf(buf, "rgb(%d,%d,%d)", &r, &g, &b);
	}
	color->r = r;
	color->g = g;
	color->b = b;
	color->a = (int) (a * 255.f);

	if (!opacity) {
		color->a = 255;
	}

	return 1;
}


void osdialog_color_picker_async(osdialog_color color, int opacity, osdialog_color_picker_callback* cb, void* user) {
	// Fake async placeholder
	int result = osdialog_color_picker(&color, opacity);
	if (cb) {
		cb(result, color, user);
	}
}