2011-02-24 04:14:30 +01:00
|
|
|
/* X-Chat
|
|
|
|
|
* Copyright (C) 1998 Peter Zelezny.
|
|
|
|
|
*
|
|
|
|
|
* This program is free software; you can redistribute it and/or modify
|
|
|
|
|
* it under the terms of the GNU General Public License as published by
|
|
|
|
|
* the Free Software Foundation; either version 2 of the License, or
|
|
|
|
|
* (at your option) any later version.
|
|
|
|
|
*
|
|
|
|
|
* This program is distributed in the hope that it will be useful,
|
|
|
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
|
* GNU General Public License for more details.
|
|
|
|
|
*
|
|
|
|
|
* You should have received a copy of the GNU General Public License
|
|
|
|
|
* along with this program; if not, write to the Free Software
|
2012-12-23 11:36:54 -08:00
|
|
|
* Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA
|
2011-02-24 04:14:30 +01:00
|
|
|
*/
|
|
|
|
|
#define _FILE_OFFSET_BITS 64 /* allow selection of large files */
|
|
|
|
|
#include <stdio.h>
|
|
|
|
|
#include <stdlib.h>
|
|
|
|
|
#include <string.h>
|
|
|
|
|
#include <stdarg.h>
|
|
|
|
|
#include <sys/types.h>
|
|
|
|
|
#include <sys/stat.h>
|
|
|
|
|
#include <fcntl.h>
|
2011-12-11 17:34:02 +01:00
|
|
|
|
2011-02-24 04:14:30 +01:00
|
|
|
#include "fe-gtk.h"
|
|
|
|
|
|
2013-03-17 15:11:23 -07:00
|
|
|
#include <gdk/gdkkeysyms.h>
|
2014-01-31 17:20:17 -05:00
|
|
|
#if defined (WIN32) || defined (__APPLE__)
|
|
|
|
|
#include <pango/pangocairo.h>
|
|
|
|
|
#endif
|
2011-02-24 04:14:30 +01:00
|
|
|
|
2017-08-31 09:52:45 -04:00
|
|
|
#ifdef GDK_WINDOWING_X11
|
|
|
|
|
#include <gdk/gdkx.h>
|
|
|
|
|
#endif
|
|
|
|
|
|
2026-01-05 23:12:38 -07:00
|
|
|
#include "../common/zoitechat.h"
|
2011-02-24 04:14:30 +01:00
|
|
|
#include "../common/fe.h"
|
|
|
|
|
#include "../common/util.h"
|
2012-10-22 17:00:48 +02:00
|
|
|
#include "../common/cfgfiles.h"
|
2026-01-05 23:12:38 -07:00
|
|
|
#include "../common/zoitechatc.h"
|
2012-10-30 22:13:40 +01:00
|
|
|
#include "../common/typedef.h"
|
2011-02-24 04:14:30 +01:00
|
|
|
#include "gtkutil.h"
|
|
|
|
|
#include "pixmaps.h"
|
2011-12-11 17:34:02 +01:00
|
|
|
|
2011-02-28 18:59:32 +01:00
|
|
|
#ifdef WIN32
|
2012-07-21 14:26:19 +02:00
|
|
|
#include <io.h>
|
2011-12-11 17:34:02 +01:00
|
|
|
#else
|
|
|
|
|
#include <unistd.h>
|
2011-02-28 18:59:32 +01:00
|
|
|
#endif
|
2011-02-24 04:14:30 +01:00
|
|
|
|
|
|
|
|
/* gtkutil.c, just some gtk wrappers */
|
|
|
|
|
|
|
|
|
|
extern void path_part (char *file, char *path, int pathlen);
|
|
|
|
|
|
|
|
|
|
struct file_req
|
|
|
|
|
{
|
|
|
|
|
GtkWidget *dialog;
|
|
|
|
|
void *userdata;
|
|
|
|
|
filereqcallback callback;
|
|
|
|
|
int flags; /* FRF_* flags */
|
|
|
|
|
};
|
|
|
|
|
|
2026-02-18 00:00:59 -07:00
|
|
|
#if HAVE_GTK3
|
|
|
|
|
static const char *
|
|
|
|
|
gtkutil_menu_custom_icon_from_stock (const char *stock_name)
|
|
|
|
|
{
|
|
|
|
|
static const struct
|
|
|
|
|
{
|
|
|
|
|
const char *stock;
|
|
|
|
|
const char *custom_icon;
|
|
|
|
|
} icon_map[] = {
|
|
|
|
|
{ "gtk-new", "zc-menu-new" },
|
|
|
|
|
{ "gtk-index", "zc-menu-network-list" },
|
|
|
|
|
{ "gtk-revert-to-saved", "zc-menu-load-plugin" },
|
|
|
|
|
{ "gtk-redo", "zc-menu-detach" },
|
|
|
|
|
{ "gtk-close", "zc-menu-close" },
|
|
|
|
|
{ "gtk-quit", "zc-menu-quit" },
|
|
|
|
|
{ "gtk-disconnect", "zc-menu-disconnect" },
|
|
|
|
|
{ "gtk-connect", "zc-menu-connect" },
|
|
|
|
|
{ "gtk-jump-to", "zc-menu-join" },
|
|
|
|
|
{ "gtk-preferences", "zc-menu-preferences" },
|
|
|
|
|
{ "gtk-clear", "zc-menu-clear" },
|
|
|
|
|
{ "gtk-copy", "zc-menu-copy" },
|
|
|
|
|
{ "gtk-delete", "zc-menu-delete" },
|
|
|
|
|
{ "gtk-add", "zc-menu-add" },
|
|
|
|
|
{ "gtk-remove", "zc-menu-remove" },
|
|
|
|
|
{ "gtk-spell-check", "zc-menu-spell-check" },
|
|
|
|
|
{ "gtk-save", "zc-menu-save" },
|
2026-02-18 00:32:42 -07:00
|
|
|
{ "gtk-save-as", "zc-menu-save-as" },
|
2026-02-18 00:00:59 -07:00
|
|
|
{ "gtk-refresh", "zc-menu-refresh" },
|
|
|
|
|
{ "gtk-justify-left", "zc-menu-search" },
|
|
|
|
|
{ "gtk-find", "zc-menu-find" },
|
2026-02-18 00:32:42 -07:00
|
|
|
{ "gtk-go-back", "zc-menu-previous" },
|
|
|
|
|
{ "gtk-go-forward", "zc-menu-next" },
|
2026-02-18 00:00:59 -07:00
|
|
|
{ "gtk-help", "zc-menu-help" },
|
|
|
|
|
{ "gtk-about", "zc-menu-about" },
|
|
|
|
|
};
|
|
|
|
|
size_t i;
|
|
|
|
|
|
|
|
|
|
if (!stock_name)
|
|
|
|
|
return NULL;
|
|
|
|
|
|
|
|
|
|
for (i = 0; i < G_N_ELEMENTS (icon_map); i++)
|
|
|
|
|
{
|
|
|
|
|
if (strcmp (stock_name, icon_map[i].stock) == 0)
|
|
|
|
|
return icon_map[i].custom_icon;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return NULL;
|
|
|
|
|
}
|
2026-02-18 00:52:49 -07:00
|
|
|
|
|
|
|
|
static const char *
|
|
|
|
|
gtkutil_menu_custom_icon_from_icon_name (const char *icon_name)
|
|
|
|
|
{
|
|
|
|
|
static const struct
|
|
|
|
|
{
|
|
|
|
|
const char *icon;
|
|
|
|
|
const char *custom_icon;
|
|
|
|
|
} icon_map[] = {
|
|
|
|
|
{ "document-new", "zc-menu-new" },
|
|
|
|
|
{ "view-list", "zc-menu-network-list" },
|
|
|
|
|
{ "document-open", "zc-menu-load-plugin" },
|
|
|
|
|
{ "edit-redo", "zc-menu-detach" },
|
|
|
|
|
{ "window-close", "zc-menu-close" },
|
|
|
|
|
{ "application-exit", "zc-menu-quit" },
|
|
|
|
|
{ "network-disconnect", "zc-menu-disconnect" },
|
|
|
|
|
{ "network-connect", "zc-menu-connect" },
|
|
|
|
|
{ "go-jump", "zc-menu-join" },
|
|
|
|
|
{ "preferences-system", "zc-menu-preferences" },
|
|
|
|
|
{ "edit-clear", "zc-menu-clear" },
|
|
|
|
|
{ "edit-copy", "zc-menu-copy" },
|
|
|
|
|
{ "edit-delete", "zc-menu-delete" },
|
|
|
|
|
{ "list-add", "zc-menu-add" },
|
|
|
|
|
{ "list-remove", "zc-menu-remove" },
|
|
|
|
|
{ "tools-check-spelling", "zc-menu-spell-check" },
|
|
|
|
|
{ "document-save", "zc-menu-save" },
|
|
|
|
|
{ "document-save-as", "zc-menu-save-as" },
|
|
|
|
|
{ "view-refresh", "zc-menu-refresh" },
|
|
|
|
|
{ "edit-find", "zc-menu-find" },
|
|
|
|
|
{ "go-previous", "zc-menu-previous" },
|
|
|
|
|
{ "go-next", "zc-menu-next" },
|
|
|
|
|
{ "help-browser", "zc-menu-help" },
|
|
|
|
|
{ "help-about", "zc-menu-about" },
|
|
|
|
|
{ "network-workgroup", "zc-menu-chanlist" },
|
|
|
|
|
};
|
|
|
|
|
size_t i;
|
|
|
|
|
|
|
|
|
|
if (!icon_name)
|
|
|
|
|
return NULL;
|
|
|
|
|
|
|
|
|
|
for (i = 0; i < G_N_ELEMENTS (icon_map); i++)
|
|
|
|
|
{
|
|
|
|
|
if (strcmp (icon_name, icon_map[i].icon) == 0)
|
|
|
|
|
return icon_map[i].custom_icon;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return NULL;
|
|
|
|
|
}
|
2026-02-18 00:00:59 -07:00
|
|
|
#endif
|
|
|
|
|
|
2026-02-18 00:16:04 -07:00
|
|
|
#if !HAVE_GTK3
|
|
|
|
|
static const char *
|
|
|
|
|
gtkutil_stock_from_menu_custom_icon (const char *custom_icon)
|
|
|
|
|
{
|
|
|
|
|
static const struct
|
|
|
|
|
{
|
|
|
|
|
const char *custom_icon;
|
|
|
|
|
const char *stock;
|
|
|
|
|
} icon_map[] = {
|
|
|
|
|
{ "zc-menu-new", GTK_STOCK_NEW },
|
|
|
|
|
{ "zc-menu-network-list", GTK_STOCK_INDEX },
|
|
|
|
|
{ "zc-menu-load-plugin", GTK_STOCK_REVERT_TO_SAVED },
|
|
|
|
|
{ "zc-menu-detach", GTK_STOCK_REDO },
|
|
|
|
|
{ "zc-menu-close", GTK_STOCK_CLOSE },
|
|
|
|
|
{ "zc-menu-quit", GTK_STOCK_QUIT },
|
|
|
|
|
{ "zc-menu-disconnect", GTK_STOCK_DISCONNECT },
|
|
|
|
|
{ "zc-menu-connect", GTK_STOCK_CONNECT },
|
|
|
|
|
{ "zc-menu-join", GTK_STOCK_JUMP_TO },
|
|
|
|
|
{ "zc-menu-chanlist", GTK_STOCK_INDEX },
|
|
|
|
|
{ "zc-menu-preferences", GTK_STOCK_PREFERENCES },
|
|
|
|
|
{ "zc-menu-clear", GTK_STOCK_CLEAR },
|
|
|
|
|
{ "zc-menu-copy", GTK_STOCK_COPY },
|
|
|
|
|
{ "zc-menu-delete", GTK_STOCK_DELETE },
|
|
|
|
|
{ "zc-menu-add", GTK_STOCK_ADD },
|
|
|
|
|
{ "zc-menu-remove", GTK_STOCK_REMOVE },
|
|
|
|
|
{ "zc-menu-spell-check", GTK_STOCK_SPELL_CHECK },
|
|
|
|
|
{ "zc-menu-save", GTK_STOCK_SAVE },
|
2026-02-18 00:32:42 -07:00
|
|
|
{ "zc-menu-save-as", GTK_STOCK_SAVE_AS },
|
2026-02-18 00:16:04 -07:00
|
|
|
{ "zc-menu-refresh", GTK_STOCK_REFRESH },
|
|
|
|
|
{ "zc-menu-search", GTK_STOCK_JUSTIFY_LEFT },
|
|
|
|
|
{ "zc-menu-find", GTK_STOCK_FIND },
|
2026-02-18 00:32:42 -07:00
|
|
|
{ "zc-menu-previous", GTK_STOCK_GO_BACK },
|
|
|
|
|
{ "zc-menu-next", GTK_STOCK_GO_FORWARD },
|
2026-02-18 00:16:04 -07:00
|
|
|
{ "zc-menu-help", GTK_STOCK_HELP },
|
|
|
|
|
{ "zc-menu-about", GTK_STOCK_ABOUT },
|
|
|
|
|
};
|
|
|
|
|
size_t i;
|
|
|
|
|
|
|
|
|
|
if (!custom_icon)
|
|
|
|
|
return NULL;
|
|
|
|
|
|
|
|
|
|
for (i = 0; i < G_N_ELEMENTS (icon_map); i++)
|
|
|
|
|
{
|
|
|
|
|
if (strcmp (custom_icon, icon_map[i].custom_icon) == 0)
|
|
|
|
|
return icon_map[i].stock;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return custom_icon;
|
|
|
|
|
}
|
|
|
|
|
#endif
|
|
|
|
|
|
2026-02-18 00:41:55 -07:00
|
|
|
static GdkPixbuf *
|
|
|
|
|
gtkutil_menu_icon_pixbuf_new (const char *icon_name)
|
|
|
|
|
{
|
|
|
|
|
GdkPixbuf *pixbuf = NULL;
|
|
|
|
|
char *resource_path;
|
|
|
|
|
|
|
|
|
|
if (!icon_name || !g_str_has_prefix (icon_name, "zc-menu-"))
|
|
|
|
|
return NULL;
|
|
|
|
|
|
|
|
|
|
resource_path = g_strdup_printf ("/icons/menu/light/%s.svg", icon_name + strlen ("zc-menu-"));
|
|
|
|
|
pixbuf = gdk_pixbuf_new_from_resource_at_scale (resource_path, -1, -1, TRUE, NULL);
|
|
|
|
|
g_free (resource_path);
|
|
|
|
|
|
|
|
|
|
return pixbuf;
|
|
|
|
|
}
|
|
|
|
|
|
2026-01-23 10:32:17 -07:00
|
|
|
#if HAVE_GTK3
|
|
|
|
|
const char *
|
|
|
|
|
gtkutil_icon_name_from_stock (const char *stock_name)
|
|
|
|
|
{
|
2026-01-25 12:53:15 -07:00
|
|
|
static const struct
|
|
|
|
|
{
|
|
|
|
|
const char *stock;
|
|
|
|
|
const char *icon;
|
|
|
|
|
} icon_map[] = {
|
|
|
|
|
{ "gtk-new", "document-new" },
|
|
|
|
|
{ "gtk-open", "document-open" },
|
|
|
|
|
{ "gtk-revert-to-saved", "document-open" },
|
|
|
|
|
{ "gtk-save", "document-save" },
|
|
|
|
|
{ "gtk-save-as", "document-save-as" },
|
|
|
|
|
{ "gtk-add", "list-add" },
|
|
|
|
|
{ "gtk-cancel", "dialog-cancel" },
|
|
|
|
|
{ "gtk-ok", "dialog-ok" },
|
|
|
|
|
{ "gtk-no", "dialog-cancel" },
|
|
|
|
|
{ "gtk-yes", "dialog-ok" },
|
|
|
|
|
{ "gtk-apply", "dialog-apply" },
|
|
|
|
|
{ "gtk-dialog-error", "dialog-error" },
|
|
|
|
|
{ "gtk-copy", "edit-copy" },
|
|
|
|
|
{ "gtk-delete", "edit-delete" },
|
|
|
|
|
{ "gtk-remove", "list-remove" },
|
|
|
|
|
{ "gtk-clear", "edit-clear" },
|
|
|
|
|
{ "gtk-redo", "edit-redo" },
|
|
|
|
|
{ "gtk-find", "edit-find" },
|
|
|
|
|
{ "gtk-justify-left", "edit-find" },
|
|
|
|
|
{ "gtk-refresh", "view-refresh" },
|
|
|
|
|
{ "gtk-go-back", "go-previous" },
|
|
|
|
|
{ "gtk-go-forward", "go-next" },
|
|
|
|
|
{ "gtk-index", "view-list" },
|
|
|
|
|
{ "gtk-jump-to", "go-jump" },
|
|
|
|
|
{ "gtk-media-play", "media-playback-start" },
|
|
|
|
|
{ "gtk-preferences", "preferences-system" },
|
|
|
|
|
{ "gtk-help", "help-browser" },
|
|
|
|
|
{ "gtk-about", "help-about" },
|
|
|
|
|
{ "gtk-close", "window-close" },
|
|
|
|
|
{ "gtk-quit", "application-exit" },
|
|
|
|
|
{ "gtk-connect", "network-connect" },
|
|
|
|
|
{ "gtk-disconnect", "network-disconnect" },
|
|
|
|
|
{ "gtk-network", "network-workgroup" },
|
|
|
|
|
{ "gtk-spell-check", "tools-check-spelling" },
|
|
|
|
|
};
|
|
|
|
|
size_t i;
|
|
|
|
|
|
2026-01-23 10:32:17 -07:00
|
|
|
if (!stock_name)
|
|
|
|
|
return NULL;
|
|
|
|
|
|
2026-01-25 12:53:15 -07:00
|
|
|
for (i = 0; i < G_N_ELEMENTS (icon_map); i++)
|
|
|
|
|
{
|
|
|
|
|
if (strcmp (stock_name, icon_map[i].stock) == 0)
|
|
|
|
|
return icon_map[i].icon;
|
|
|
|
|
}
|
2026-01-23 10:32:17 -07:00
|
|
|
|
|
|
|
|
return stock_name;
|
|
|
|
|
}
|
2026-01-30 07:53:50 -07:00
|
|
|
#endif
|
2026-01-23 11:17:28 -07:00
|
|
|
|
2026-02-17 23:49:59 -07:00
|
|
|
#if HAVE_GTK3
|
|
|
|
|
static const char *
|
|
|
|
|
gtkutil_menu_icon_theme_variant (void)
|
|
|
|
|
{
|
|
|
|
|
GtkSettings *settings;
|
|
|
|
|
gboolean prefer_dark = FALSE;
|
|
|
|
|
char *theme_name = NULL;
|
|
|
|
|
char *theme_name_lower = NULL;
|
|
|
|
|
const char *theme_variant = "light";
|
|
|
|
|
|
|
|
|
|
settings = gtk_settings_get_default ();
|
|
|
|
|
if (settings)
|
|
|
|
|
{
|
|
|
|
|
g_object_get (G_OBJECT (settings), "gtk-application-prefer-dark-theme", &prefer_dark, NULL);
|
|
|
|
|
g_object_get (G_OBJECT (settings), "gtk-theme-name", &theme_name, NULL);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (theme_name)
|
|
|
|
|
theme_name_lower = g_ascii_strdown (theme_name, -1);
|
|
|
|
|
if (prefer_dark || (theme_name_lower && g_strrstr (theme_name_lower, "dark")))
|
|
|
|
|
theme_variant = "dark";
|
|
|
|
|
|
|
|
|
|
g_free (theme_name_lower);
|
|
|
|
|
g_free (theme_name);
|
|
|
|
|
|
|
|
|
|
return theme_variant;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static GtkWidget *
|
|
|
|
|
gtkutil_menu_icon_image_new (const char *icon_name, GtkIconSize size)
|
|
|
|
|
{
|
|
|
|
|
GtkWidget *image = NULL;
|
|
|
|
|
GdkPixbuf *pixbuf = NULL;
|
|
|
|
|
char *resource_path;
|
|
|
|
|
const char *variant;
|
|
|
|
|
|
|
|
|
|
if (!icon_name || !g_str_has_prefix (icon_name, "zc-menu-"))
|
|
|
|
|
return NULL;
|
|
|
|
|
|
|
|
|
|
variant = gtkutil_menu_icon_theme_variant ();
|
|
|
|
|
resource_path = g_strdup_printf ("/icons/menu/%s/%s.svg", variant, icon_name + strlen ("zc-menu-"));
|
|
|
|
|
if (!g_resources_get_info (resource_path, G_RESOURCE_LOOKUP_FLAGS_NONE, NULL, NULL, NULL))
|
|
|
|
|
{
|
|
|
|
|
g_free (resource_path);
|
|
|
|
|
resource_path = g_strdup_printf ("/icons/menu/light/%s.svg", icon_name + strlen ("zc-menu-"));
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
pixbuf = gdk_pixbuf_new_from_resource_at_scale (resource_path, -1, -1, TRUE, NULL);
|
|
|
|
|
if (pixbuf)
|
|
|
|
|
{
|
|
|
|
|
image = gtk_image_new_from_pixbuf (pixbuf);
|
|
|
|
|
g_object_unref (pixbuf);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
g_free (resource_path);
|
|
|
|
|
|
|
|
|
|
if (image)
|
|
|
|
|
{
|
|
|
|
|
GtkIconSize tmp_size;
|
|
|
|
|
gint width;
|
|
|
|
|
gint height;
|
|
|
|
|
|
|
|
|
|
tmp_size = size;
|
|
|
|
|
if (gtk_icon_size_lookup (tmp_size, &width, &height))
|
|
|
|
|
gtk_image_set_pixel_size (GTK_IMAGE (image), MAX (width, height));
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return image;
|
|
|
|
|
}
|
|
|
|
|
#endif
|
|
|
|
|
|
2026-01-30 07:53:50 -07:00
|
|
|
GtkWidget *
|
2026-01-23 11:17:28 -07:00
|
|
|
gtkutil_image_new_from_stock (const char *stock, GtkIconSize size)
|
|
|
|
|
{
|
2026-01-30 07:53:50 -07:00
|
|
|
#if HAVE_GTK3
|
2026-02-17 23:49:59 -07:00
|
|
|
GtkWidget *image;
|
2026-02-18 00:00:59 -07:00
|
|
|
const char *icon_name;
|
|
|
|
|
|
|
|
|
|
icon_name = gtkutil_icon_name_from_stock (stock);
|
2026-02-18 00:05:44 -07:00
|
|
|
if (!icon_name && stock && g_str_has_prefix (stock, "zc-menu-"))
|
|
|
|
|
icon_name = stock;
|
2026-02-18 00:00:59 -07:00
|
|
|
if (size == GTK_ICON_SIZE_MENU)
|
|
|
|
|
{
|
|
|
|
|
const char *menu_icon_name = gtkutil_menu_custom_icon_from_stock (stock);
|
|
|
|
|
|
2026-02-18 00:52:49 -07:00
|
|
|
if (!menu_icon_name)
|
|
|
|
|
menu_icon_name = gtkutil_menu_custom_icon_from_icon_name (icon_name);
|
|
|
|
|
|
2026-02-18 00:00:59 -07:00
|
|
|
if (menu_icon_name)
|
|
|
|
|
icon_name = menu_icon_name;
|
|
|
|
|
}
|
2026-01-23 11:17:28 -07:00
|
|
|
|
2026-02-17 23:49:59 -07:00
|
|
|
image = gtkutil_menu_icon_image_new (icon_name, size);
|
|
|
|
|
if (image)
|
|
|
|
|
return image;
|
|
|
|
|
|
2026-01-23 11:17:28 -07:00
|
|
|
return gtk_image_new_from_icon_name (icon_name, size);
|
2026-01-30 08:21:55 -07:00
|
|
|
#elif !HAVE_GTK3
|
2026-02-18 00:41:55 -07:00
|
|
|
if (stock && g_str_has_prefix (stock, "zc-menu-"))
|
|
|
|
|
{
|
|
|
|
|
GdkPixbuf *pixbuf = gtkutil_menu_icon_pixbuf_new (stock);
|
|
|
|
|
|
|
|
|
|
if (pixbuf)
|
|
|
|
|
{
|
|
|
|
|
GtkWidget *image = gtk_image_new_from_pixbuf (pixbuf);
|
|
|
|
|
g_object_unref (pixbuf);
|
|
|
|
|
return image;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2026-02-18 00:16:04 -07:00
|
|
|
if (stock && g_str_has_prefix (stock, "zc-menu-"))
|
|
|
|
|
stock = gtkutil_stock_from_menu_custom_icon (stock);
|
2026-01-30 07:53:50 -07:00
|
|
|
return gtk_image_new_from_stock (stock, size);
|
|
|
|
|
#endif
|
2026-01-23 11:17:28 -07:00
|
|
|
}
|
2026-01-30 07:53:50 -07:00
|
|
|
|
|
|
|
|
GtkWidget *
|
|
|
|
|
gtkutil_button_new_from_stock (const char *stock, const char *label)
|
|
|
|
|
{
|
|
|
|
|
#if HAVE_GTK3
|
|
|
|
|
GtkWidget *button = label ? gtk_button_new_with_mnemonic (label) : gtk_button_new ();
|
|
|
|
|
|
|
|
|
|
if (stock)
|
|
|
|
|
{
|
|
|
|
|
GtkWidget *image = gtkutil_image_new_from_stock (stock, GTK_ICON_SIZE_BUTTON);
|
|
|
|
|
|
|
|
|
|
if (image)
|
|
|
|
|
{
|
|
|
|
|
gtk_button_set_image (GTK_BUTTON (button), image);
|
|
|
|
|
gtk_button_set_always_show_image (GTK_BUTTON (button), TRUE);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return button;
|
2026-01-30 08:21:55 -07:00
|
|
|
#elif !HAVE_GTK3
|
2026-01-30 07:53:50 -07:00
|
|
|
if (stock)
|
|
|
|
|
return gtk_button_new_from_stock (stock);
|
|
|
|
|
if (label)
|
|
|
|
|
return gtk_button_new_with_mnemonic (label);
|
|
|
|
|
return gtk_button_new ();
|
2026-01-23 10:32:17 -07:00
|
|
|
#endif
|
2026-01-30 07:53:50 -07:00
|
|
|
}
|
2026-01-23 10:32:17 -07:00
|
|
|
|
2026-01-25 11:37:04 -07:00
|
|
|
#if HAVE_GTK3
|
2026-01-31 16:37:15 -07:00
|
|
|
void
|
|
|
|
|
gtkutil_append_font_css (GString *css, const PangoFontDescription *font_desc)
|
|
|
|
|
{
|
|
|
|
|
PangoFontMask mask;
|
|
|
|
|
|
|
|
|
|
if (!font_desc)
|
|
|
|
|
return;
|
|
|
|
|
|
|
|
|
|
mask = pango_font_description_get_set_fields (font_desc);
|
|
|
|
|
|
|
|
|
|
if (mask & PANGO_FONT_MASK_FAMILY)
|
|
|
|
|
{
|
|
|
|
|
const char *family = pango_font_description_get_family (font_desc);
|
|
|
|
|
|
|
|
|
|
if (family && *family)
|
|
|
|
|
g_string_append_printf (css, " font-family: \"%s\";", family);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (mask & PANGO_FONT_MASK_STYLE)
|
|
|
|
|
{
|
|
|
|
|
const char *style = "normal";
|
|
|
|
|
|
|
|
|
|
switch (pango_font_description_get_style (font_desc))
|
|
|
|
|
{
|
|
|
|
|
case PANGO_STYLE_ITALIC:
|
|
|
|
|
style = "italic";
|
|
|
|
|
break;
|
|
|
|
|
case PANGO_STYLE_OBLIQUE:
|
|
|
|
|
style = "oblique";
|
|
|
|
|
break;
|
|
|
|
|
default:
|
|
|
|
|
style = "normal";
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
g_string_append_printf (css, " font-style: %s;", style);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (mask & PANGO_FONT_MASK_VARIANT)
|
|
|
|
|
{
|
|
|
|
|
const char *variant = "normal";
|
|
|
|
|
|
|
|
|
|
if (pango_font_description_get_variant (font_desc) == PANGO_VARIANT_SMALL_CAPS)
|
|
|
|
|
variant = "small-caps";
|
|
|
|
|
|
|
|
|
|
g_string_append_printf (css, " font-variant: %s;", variant);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (mask & PANGO_FONT_MASK_WEIGHT)
|
|
|
|
|
{
|
|
|
|
|
int weight = (int) pango_font_description_get_weight (font_desc);
|
|
|
|
|
|
|
|
|
|
if (weight < 100)
|
|
|
|
|
weight = 100;
|
|
|
|
|
if (weight > 900)
|
|
|
|
|
weight = 900;
|
|
|
|
|
|
|
|
|
|
g_string_append_printf (css, " font-weight: %d;", weight);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (mask & PANGO_FONT_MASK_STRETCH)
|
|
|
|
|
{
|
|
|
|
|
const char *stretch = "normal";
|
|
|
|
|
|
|
|
|
|
switch (pango_font_description_get_stretch (font_desc))
|
|
|
|
|
{
|
|
|
|
|
case PANGO_STRETCH_ULTRA_CONDENSED:
|
|
|
|
|
stretch = "ultra-condensed";
|
|
|
|
|
break;
|
|
|
|
|
case PANGO_STRETCH_EXTRA_CONDENSED:
|
|
|
|
|
stretch = "extra-condensed";
|
|
|
|
|
break;
|
|
|
|
|
case PANGO_STRETCH_CONDENSED:
|
|
|
|
|
stretch = "condensed";
|
|
|
|
|
break;
|
|
|
|
|
case PANGO_STRETCH_SEMI_CONDENSED:
|
|
|
|
|
stretch = "semi-condensed";
|
|
|
|
|
break;
|
|
|
|
|
case PANGO_STRETCH_SEMI_EXPANDED:
|
|
|
|
|
stretch = "semi-expanded";
|
|
|
|
|
break;
|
|
|
|
|
case PANGO_STRETCH_EXPANDED:
|
|
|
|
|
stretch = "expanded";
|
|
|
|
|
break;
|
|
|
|
|
case PANGO_STRETCH_EXTRA_EXPANDED:
|
|
|
|
|
stretch = "extra-expanded";
|
|
|
|
|
break;
|
|
|
|
|
case PANGO_STRETCH_ULTRA_EXPANDED:
|
|
|
|
|
stretch = "ultra-expanded";
|
|
|
|
|
break;
|
|
|
|
|
default:
|
|
|
|
|
stretch = "normal";
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
g_string_append_printf (css, " font-stretch: %s;", stretch);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (mask & PANGO_FONT_MASK_SIZE)
|
|
|
|
|
{
|
|
|
|
|
double size = (double) pango_font_description_get_size (font_desc) / PANGO_SCALE;
|
|
|
|
|
char size_buf[G_ASCII_DTOSTR_BUF_SIZE];
|
|
|
|
|
const char *unit = "pt";
|
|
|
|
|
|
|
|
|
|
if (pango_font_description_get_size_is_absolute (font_desc))
|
|
|
|
|
unit = "px";
|
|
|
|
|
|
|
|
|
|
g_ascii_formatd (size_buf, sizeof (size_buf), "%.2f", size);
|
|
|
|
|
g_string_append_printf (css, " font-size: %s%s;", size_buf, unit);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2026-01-25 11:30:21 -07:00
|
|
|
void
|
2026-01-25 11:37:04 -07:00
|
|
|
gtkutil_apply_palette (GtkWidget *widget, const GdkRGBA *bg, const GdkRGBA *fg,
|
2026-01-25 11:30:21 -07:00
|
|
|
const PangoFontDescription *font_desc)
|
2026-01-25 11:37:04 -07:00
|
|
|
#else
|
|
|
|
|
void
|
|
|
|
|
gtkutil_apply_palette (GtkWidget *widget, const GdkColor *bg, const GdkColor *fg,
|
|
|
|
|
const PangoFontDescription *font_desc)
|
|
|
|
|
#endif
|
2026-01-25 11:30:21 -07:00
|
|
|
{
|
|
|
|
|
if (!widget)
|
|
|
|
|
return;
|
|
|
|
|
|
|
|
|
|
#if HAVE_GTK3
|
|
|
|
|
{
|
|
|
|
|
static const char *class_name = "zoitechat-palette";
|
|
|
|
|
GtkStyleContext *context = gtk_widget_get_style_context (widget);
|
|
|
|
|
GtkCssProvider *provider = g_object_get_data (G_OBJECT (widget),
|
|
|
|
|
"zoitechat-palette-provider");
|
2026-01-25 12:19:44 -07:00
|
|
|
gboolean new_provider = FALSE;
|
2026-01-25 11:30:21 -07:00
|
|
|
GString *css;
|
|
|
|
|
gchar *bg_color = NULL;
|
|
|
|
|
gchar *fg_color = NULL;
|
|
|
|
|
|
|
|
|
|
if (!bg && !fg && !font_desc)
|
|
|
|
|
{
|
|
|
|
|
gtk_style_context_remove_class (context, class_name);
|
|
|
|
|
if (provider)
|
|
|
|
|
{
|
|
|
|
|
gtk_style_context_remove_provider (context, GTK_STYLE_PROVIDER (provider));
|
|
|
|
|
g_object_set_data (G_OBJECT (widget), "zoitechat-palette-provider", NULL);
|
|
|
|
|
}
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (!provider)
|
|
|
|
|
{
|
|
|
|
|
provider = gtk_css_provider_new ();
|
|
|
|
|
g_object_set_data_full (G_OBJECT (widget), "zoitechat-palette-provider",
|
|
|
|
|
provider, g_object_unref);
|
2026-01-25 12:19:44 -07:00
|
|
|
new_provider = TRUE;
|
2026-01-25 11:30:21 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
css = g_string_new (".");
|
|
|
|
|
g_string_append (css, class_name);
|
|
|
|
|
g_string_append (css, " {");
|
|
|
|
|
if (bg)
|
|
|
|
|
{
|
|
|
|
|
bg_color = gdk_rgba_to_string (bg);
|
|
|
|
|
g_string_append_printf (css, " background-color: %s;", bg_color);
|
|
|
|
|
}
|
|
|
|
|
if (fg)
|
|
|
|
|
{
|
|
|
|
|
fg_color = gdk_rgba_to_string (fg);
|
|
|
|
|
g_string_append_printf (css, " color: %s;", fg_color);
|
|
|
|
|
}
|
2026-01-31 16:37:15 -07:00
|
|
|
gtkutil_append_font_css (css, font_desc);
|
2026-01-25 11:30:21 -07:00
|
|
|
g_string_append (css, " }");
|
|
|
|
|
|
|
|
|
|
gtk_css_provider_load_from_data (provider, css->str, -1, NULL);
|
2026-01-25 12:19:44 -07:00
|
|
|
if (new_provider)
|
|
|
|
|
{
|
|
|
|
|
gtk_style_context_add_provider (context, GTK_STYLE_PROVIDER (provider),
|
|
|
|
|
GTK_STYLE_PROVIDER_PRIORITY_APPLICATION);
|
|
|
|
|
}
|
2026-01-25 11:30:21 -07:00
|
|
|
gtk_style_context_add_class (context, class_name);
|
|
|
|
|
|
|
|
|
|
g_string_free (css, TRUE);
|
|
|
|
|
g_free (bg_color);
|
|
|
|
|
g_free (fg_color);
|
|
|
|
|
}
|
|
|
|
|
#else
|
|
|
|
|
gtk_widget_modify_base (widget, GTK_STATE_NORMAL, bg);
|
|
|
|
|
gtk_widget_modify_text (widget, GTK_STATE_NORMAL, fg);
|
|
|
|
|
gtk_widget_modify_font (widget, (PangoFontDescription *) font_desc);
|
|
|
|
|
#endif
|
|
|
|
|
}
|
|
|
|
|
|
2011-02-24 04:14:30 +01:00
|
|
|
static void
|
|
|
|
|
gtkutil_file_req_destroy (GtkWidget * wid, struct file_req *freq)
|
|
|
|
|
{
|
|
|
|
|
freq->callback (freq->userdata, NULL);
|
2014-12-28 06:37:25 -05:00
|
|
|
g_free (freq);
|
2011-02-24 04:14:30 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static void
|
2014-08-20 12:50:08 -07:00
|
|
|
gtkutil_check_file (char *filename, struct file_req *freq)
|
2011-02-24 04:14:30 +01:00
|
|
|
{
|
|
|
|
|
int axs = FALSE;
|
|
|
|
|
|
2026-02-15 15:02:26 -07:00
|
|
|
if (filename == NULL || filename[0] == '\0')
|
|
|
|
|
{
|
|
|
|
|
fe_message (_("No file selected."), FE_MSG_ERROR);
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
2014-08-20 12:50:08 -07:00
|
|
|
GFile *file = g_file_new_for_path (filename);
|
2011-02-24 04:14:30 +01:00
|
|
|
|
|
|
|
|
if (freq->flags & FRF_WRITE)
|
|
|
|
|
{
|
2014-08-20 12:50:08 -07:00
|
|
|
GFile *parent = g_file_get_parent (file);
|
|
|
|
|
|
|
|
|
|
GFileInfo *fi = g_file_query_info (parent, G_FILE_ATTRIBUTE_ACCESS_CAN_WRITE, G_FILE_QUERY_INFO_NONE, NULL, NULL);
|
|
|
|
|
if (fi != NULL)
|
|
|
|
|
{
|
|
|
|
|
if (g_file_info_get_attribute_boolean (fi, G_FILE_ATTRIBUTE_ACCESS_CAN_WRITE))
|
|
|
|
|
{
|
|
|
|
|
axs = TRUE;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
g_object_unref (fi);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
g_object_unref (parent);
|
|
|
|
|
}
|
|
|
|
|
else
|
2011-02-24 04:14:30 +01:00
|
|
|
{
|
2026-01-17 18:31:04 -07:00
|
|
|
GFileInfo *fi = g_file_query_info (file, G_FILE_ATTRIBUTE_ACCESS_CAN_WRITE "," G_FILE_ATTRIBUTE_STANDARD_TYPE, G_FILE_QUERY_INFO_NONE, NULL, NULL);
|
2014-08-20 12:50:08 -07:00
|
|
|
|
|
|
|
|
if (fi != NULL)
|
2011-02-24 04:14:30 +01:00
|
|
|
{
|
2014-08-20 12:50:08 -07:00
|
|
|
if (g_file_info_get_file_type (fi) != G_FILE_TYPE_DIRECTORY || (freq->flags & FRF_CHOOSEFOLDER))
|
|
|
|
|
{
|
2011-02-24 04:14:30 +01:00
|
|
|
axs = TRUE;
|
2014-08-20 12:50:08 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
g_object_unref (fi);
|
2011-02-24 04:14:30 +01:00
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2014-08-20 12:50:08 -07:00
|
|
|
g_object_unref (file);
|
|
|
|
|
|
2011-02-24 04:14:30 +01:00
|
|
|
if (axs)
|
|
|
|
|
{
|
2014-08-20 12:50:08 -07:00
|
|
|
char *filename_utf8 = g_filename_to_utf8 (filename, -1, NULL, NULL, NULL);
|
|
|
|
|
if (filename_utf8 != NULL)
|
2011-02-24 04:14:30 +01:00
|
|
|
{
|
2014-08-20 12:50:08 -07:00
|
|
|
freq->callback (freq->userdata, filename_utf8);
|
|
|
|
|
g_free (filename_utf8);
|
|
|
|
|
}
|
|
|
|
|
else
|
2011-02-24 04:14:30 +01:00
|
|
|
{
|
|
|
|
|
fe_message ("Filename encoding is corrupt.", FE_MSG_ERROR);
|
|
|
|
|
}
|
2014-08-20 12:50:08 -07:00
|
|
|
}
|
|
|
|
|
else
|
2011-02-24 04:14:30 +01:00
|
|
|
{
|
|
|
|
|
if (freq->flags & FRF_WRITE)
|
2014-08-20 12:50:08 -07:00
|
|
|
{
|
2011-02-24 04:14:30 +01:00
|
|
|
fe_message (_("Cannot write to that file."), FE_MSG_ERROR);
|
2014-08-20 12:50:08 -07:00
|
|
|
}
|
2011-02-24 04:14:30 +01:00
|
|
|
else
|
2014-08-20 12:50:08 -07:00
|
|
|
{
|
2011-02-24 04:14:30 +01:00
|
|
|
fe_message (_("Cannot read that file."), FE_MSG_ERROR);
|
2014-08-20 12:50:08 -07:00
|
|
|
}
|
2011-02-24 04:14:30 +01:00
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static void
|
2026-02-16 15:06:59 -07:00
|
|
|
gtkutil_file_req_done_chooser (GtkFileChooser *fs, struct file_req *freq)
|
2011-02-24 04:14:30 +01:00
|
|
|
{
|
|
|
|
|
GSList *files, *cur;
|
|
|
|
|
|
|
|
|
|
if (freq->flags & FRF_MULTIPLE)
|
|
|
|
|
{
|
|
|
|
|
files = cur = gtk_file_chooser_get_filenames (fs);
|
|
|
|
|
while (cur)
|
|
|
|
|
{
|
|
|
|
|
gtkutil_check_file (cur->data, freq);
|
|
|
|
|
g_free (cur->data);
|
|
|
|
|
cur = cur->next;
|
|
|
|
|
}
|
|
|
|
|
if (files)
|
|
|
|
|
g_slist_free (files);
|
2014-08-20 12:50:08 -07:00
|
|
|
}
|
|
|
|
|
else
|
2011-02-24 04:14:30 +01:00
|
|
|
{
|
|
|
|
|
if (freq->flags & FRF_CHOOSEFOLDER)
|
2014-08-20 12:50:08 -07:00
|
|
|
{
|
|
|
|
|
gchar *filename = gtk_file_chooser_get_current_folder (fs);
|
|
|
|
|
gtkutil_check_file (filename, freq);
|
|
|
|
|
g_free (filename);
|
|
|
|
|
}
|
2011-02-24 04:14:30 +01:00
|
|
|
else
|
2014-08-20 12:50:08 -07:00
|
|
|
{
|
|
|
|
|
gchar *filename = gtk_file_chooser_get_filename (fs);
|
2026-02-15 15:02:26 -07:00
|
|
|
if (filename != NULL)
|
|
|
|
|
{
|
|
|
|
|
gtkutil_check_file (filename, freq);
|
|
|
|
|
g_free (filename);
|
|
|
|
|
}
|
2014-08-20 12:50:08 -07:00
|
|
|
}
|
2011-02-24 04:14:30 +01:00
|
|
|
}
|
|
|
|
|
|
2026-02-16 15:06:59 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static void
|
|
|
|
|
gtkutil_file_req_done (GtkWidget * wid, struct file_req *freq)
|
|
|
|
|
{
|
|
|
|
|
gtkutil_file_req_done_chooser (GTK_FILE_CHOOSER (freq->dialog), freq);
|
|
|
|
|
|
2011-02-24 04:14:30 +01:00
|
|
|
/* this should call the "destroy" cb, where we free(freq) */
|
|
|
|
|
gtk_widget_destroy (freq->dialog);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static void
|
|
|
|
|
gtkutil_file_req_response (GtkWidget *dialog, gint res, struct file_req *freq)
|
|
|
|
|
{
|
2026-02-16 18:57:48 -07:00
|
|
|
if (res == GTK_RESPONSE_ACCEPT)
|
2011-02-24 04:14:30 +01:00
|
|
|
{
|
|
|
|
|
gtkutil_file_req_done (dialog, freq);
|
2026-02-16 18:57:48 -07:00
|
|
|
return;
|
2011-02-24 04:14:30 +01:00
|
|
|
}
|
2026-02-16 18:57:48 -07:00
|
|
|
|
|
|
|
|
/* this should call the "destroy" cb, where we free(freq) */
|
|
|
|
|
gtk_widget_destroy (dialog);
|
2011-02-24 04:14:30 +01:00
|
|
|
}
|
|
|
|
|
|
2026-02-16 15:37:59 -07:00
|
|
|
#if defined (WIN32) && HAVE_GTK3
|
|
|
|
|
static gboolean
|
|
|
|
|
gtkutil_native_dialog_unref_idle (gpointer native)
|
|
|
|
|
{
|
|
|
|
|
g_object_unref (native);
|
|
|
|
|
return G_SOURCE_REMOVE;
|
|
|
|
|
}
|
2026-02-16 17:29:25 -07:00
|
|
|
|
|
|
|
|
static void
|
|
|
|
|
gtkutil_native_file_req_response (GtkNativeDialog *dialog, gint res, struct file_req *freq)
|
|
|
|
|
{
|
|
|
|
|
if (res == GTK_RESPONSE_ACCEPT)
|
|
|
|
|
gtkutil_file_req_done_chooser (GTK_FILE_CHOOSER (dialog), freq);
|
|
|
|
|
|
|
|
|
|
/* Match gtk dialog flow by always sending NULL to indicate completion. */
|
|
|
|
|
freq->callback (freq->userdata, NULL);
|
|
|
|
|
g_free (freq);
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
|
* Defer unref until idle to avoid disposing the native chooser while
|
|
|
|
|
* still in the button-release signal stack on Windows.
|
|
|
|
|
*/
|
|
|
|
|
g_idle_add (gtkutil_native_dialog_unref_idle, dialog);
|
|
|
|
|
}
|
2026-02-16 15:37:59 -07:00
|
|
|
#endif
|
|
|
|
|
|
2011-02-24 04:14:30 +01:00
|
|
|
void
|
2022-04-16 18:41:34 -05:00
|
|
|
gtkutil_file_req (GtkWindow *parent, const char *title, void *callback, void *userdata, char *filter, char *extensions,
|
2011-02-24 04:14:30 +01:00
|
|
|
int flags)
|
|
|
|
|
{
|
|
|
|
|
struct file_req *freq;
|
|
|
|
|
GtkWidget *dialog;
|
2012-07-21 21:42:48 +02:00
|
|
|
GtkFileFilter *filefilter;
|
|
|
|
|
char *token;
|
|
|
|
|
char *tokenbuffer;
|
2026-02-15 16:15:25 -07:00
|
|
|
const char *xdir;
|
2026-02-16 15:06:59 -07:00
|
|
|
GtkWindow *effective_parent = parent;
|
|
|
|
|
|
|
|
|
|
extern GtkWidget *parent_window;
|
|
|
|
|
if (effective_parent == NULL && parent_window != NULL)
|
|
|
|
|
effective_parent = GTK_WINDOW (parent_window);
|
|
|
|
|
|
|
|
|
|
xdir = get_xdir ();
|
|
|
|
|
|
|
|
|
|
#if defined (WIN32) && HAVE_GTK3
|
|
|
|
|
{
|
|
|
|
|
GtkFileChooserNative *native = gtk_file_chooser_native_new (
|
|
|
|
|
title,
|
|
|
|
|
effective_parent,
|
|
|
|
|
(flags & FRF_CHOOSEFOLDER) ? GTK_FILE_CHOOSER_ACTION_SELECT_FOLDER :
|
|
|
|
|
((flags & FRF_WRITE) ? GTK_FILE_CHOOSER_ACTION_SAVE : GTK_FILE_CHOOSER_ACTION_OPEN),
|
|
|
|
|
(flags & FRF_WRITE) ? _("_Save") : _("_Open"),
|
|
|
|
|
_("_Cancel"));
|
|
|
|
|
GtkFileChooser *native_chooser = GTK_FILE_CHOOSER (native);
|
|
|
|
|
|
|
|
|
|
if (flags & FRF_MULTIPLE)
|
|
|
|
|
gtk_file_chooser_set_select_multiple (native_chooser, TRUE);
|
|
|
|
|
if (flags & FRF_WRITE && !(flags & FRF_NOASKOVERWRITE))
|
|
|
|
|
gtk_file_chooser_set_do_overwrite_confirmation (native_chooser, TRUE);
|
|
|
|
|
|
|
|
|
|
if ((flags & FRF_EXTENSIONS || flags & FRF_MIMETYPES) && extensions != NULL)
|
|
|
|
|
{
|
|
|
|
|
GtkFileFilter *native_filter = gtk_file_filter_new ();
|
|
|
|
|
char *native_tokenbuffer = g_strdup (extensions);
|
|
|
|
|
char *native_token = strtok (native_tokenbuffer, ";");
|
|
|
|
|
|
|
|
|
|
while (native_token != NULL)
|
|
|
|
|
{
|
|
|
|
|
if (flags & FRF_EXTENSIONS)
|
|
|
|
|
gtk_file_filter_add_pattern (native_filter, native_token);
|
|
|
|
|
else
|
|
|
|
|
gtk_file_filter_add_mime_type (native_filter, native_token);
|
|
|
|
|
native_token = strtok (NULL, ";");
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
g_free (native_tokenbuffer);
|
|
|
|
|
gtk_file_chooser_set_filter (native_chooser, native_filter);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (filter && filter[0] && (flags & FRF_FILTERISINITIAL))
|
|
|
|
|
{
|
|
|
|
|
if (flags & FRF_WRITE)
|
|
|
|
|
{
|
|
|
|
|
char temp[1024];
|
|
|
|
|
path_part (filter, temp, sizeof (temp));
|
|
|
|
|
if (temp[0] && g_file_test (temp, G_FILE_TEST_IS_DIR))
|
|
|
|
|
gtk_file_chooser_set_current_folder (native_chooser, temp);
|
|
|
|
|
else if (xdir && xdir[0] && g_file_test (xdir, G_FILE_TEST_IS_DIR))
|
|
|
|
|
gtk_file_chooser_set_current_folder (native_chooser, xdir);
|
|
|
|
|
gtk_file_chooser_set_current_name (native_chooser, file_part (filter));
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
if (g_file_test (filter, G_FILE_TEST_IS_DIR))
|
|
|
|
|
gtk_file_chooser_set_current_folder (native_chooser, filter);
|
|
|
|
|
else if (xdir && xdir[0] && g_file_test (xdir, G_FILE_TEST_IS_DIR))
|
|
|
|
|
gtk_file_chooser_set_current_folder (native_chooser, xdir);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
else if (!(flags & FRF_RECENTLYUSED))
|
|
|
|
|
{
|
|
|
|
|
if (xdir && xdir[0] && g_file_test (xdir, G_FILE_TEST_IS_DIR))
|
|
|
|
|
gtk_file_chooser_set_current_folder (native_chooser, xdir);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
freq = g_new (struct file_req, 1);
|
|
|
|
|
freq->dialog = NULL;
|
|
|
|
|
freq->flags = flags;
|
|
|
|
|
freq->callback = callback;
|
|
|
|
|
freq->userdata = userdata;
|
|
|
|
|
|
2026-02-16 17:29:25 -07:00
|
|
|
g_signal_connect (native, "response",
|
|
|
|
|
G_CALLBACK (gtkutil_native_file_req_response), freq);
|
|
|
|
|
gtk_native_dialog_show (GTK_NATIVE_DIALOG (native));
|
2026-02-16 15:06:59 -07:00
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
#endif
|
2011-02-24 04:14:30 +01:00
|
|
|
|
|
|
|
|
if (flags & FRF_WRITE)
|
|
|
|
|
{
|
2026-01-30 15:27:01 -07:00
|
|
|
#if HAVE_GTK3
|
|
|
|
|
dialog = gtk_file_chooser_dialog_new (title, NULL,
|
|
|
|
|
GTK_FILE_CHOOSER_ACTION_SAVE,
|
|
|
|
|
_("_Cancel"), GTK_RESPONSE_CANCEL,
|
|
|
|
|
_("_Save"), GTK_RESPONSE_ACCEPT,
|
|
|
|
|
NULL);
|
|
|
|
|
#elif !HAVE_GTK3
|
2011-02-24 04:14:30 +01:00
|
|
|
dialog = gtk_file_chooser_dialog_new (title, NULL,
|
|
|
|
|
GTK_FILE_CHOOSER_ACTION_SAVE,
|
|
|
|
|
GTK_STOCK_CANCEL, GTK_RESPONSE_CANCEL,
|
|
|
|
|
GTK_STOCK_SAVE, GTK_RESPONSE_ACCEPT,
|
|
|
|
|
NULL);
|
2026-01-30 15:27:01 -07:00
|
|
|
#endif
|
2012-07-14 20:46:42 +02:00
|
|
|
|
2011-02-24 04:14:30 +01:00
|
|
|
if (!(flags & FRF_NOASKOVERWRITE))
|
|
|
|
|
gtk_file_chooser_set_do_overwrite_confirmation (GTK_FILE_CHOOSER (dialog), TRUE);
|
|
|
|
|
}
|
|
|
|
|
else
|
2026-01-30 15:27:01 -07:00
|
|
|
#if HAVE_GTK3
|
|
|
|
|
dialog = gtk_file_chooser_dialog_new (title, NULL,
|
|
|
|
|
GTK_FILE_CHOOSER_ACTION_OPEN,
|
|
|
|
|
_("_Cancel"), GTK_RESPONSE_CANCEL,
|
|
|
|
|
_("_Open"), GTK_RESPONSE_ACCEPT,
|
|
|
|
|
NULL);
|
|
|
|
|
#elif !HAVE_GTK3
|
2011-02-24 04:14:30 +01:00
|
|
|
dialog = gtk_file_chooser_dialog_new (title, NULL,
|
|
|
|
|
GTK_FILE_CHOOSER_ACTION_OPEN,
|
|
|
|
|
GTK_STOCK_CANCEL, GTK_RESPONSE_CANCEL,
|
|
|
|
|
GTK_STOCK_OK, GTK_RESPONSE_ACCEPT,
|
|
|
|
|
NULL);
|
2026-01-30 15:27:01 -07:00
|
|
|
#endif
|
2013-08-30 20:19:10 -04:00
|
|
|
|
|
|
|
|
if (filter && filter[0] && (flags & FRF_FILTERISINITIAL))
|
2011-02-24 04:14:30 +01:00
|
|
|
{
|
2013-08-30 20:19:10 -04:00
|
|
|
if (flags & FRF_WRITE)
|
2012-10-22 17:00:48 +02:00
|
|
|
{
|
2013-08-30 20:19:10 -04:00
|
|
|
char temp[1024];
|
|
|
|
|
path_part (filter, temp, sizeof (temp));
|
2026-02-15 16:15:25 -07:00
|
|
|
if (temp[0] && g_file_test (temp, G_FILE_TEST_IS_DIR))
|
|
|
|
|
gtk_file_chooser_set_current_folder (GTK_FILE_CHOOSER (dialog), temp);
|
|
|
|
|
else if (xdir && xdir[0] && g_file_test (xdir, G_FILE_TEST_IS_DIR))
|
|
|
|
|
gtk_file_chooser_set_current_folder (GTK_FILE_CHOOSER (dialog), xdir);
|
2013-08-30 20:19:10 -04:00
|
|
|
gtk_file_chooser_set_current_name (GTK_FILE_CHOOSER (dialog), file_part (filter));
|
2012-10-22 17:00:48 +02:00
|
|
|
}
|
|
|
|
|
else
|
2026-02-15 16:15:25 -07:00
|
|
|
{
|
|
|
|
|
if (g_file_test (filter, G_FILE_TEST_IS_DIR))
|
|
|
|
|
gtk_file_chooser_set_current_folder (GTK_FILE_CHOOSER (dialog), filter);
|
|
|
|
|
else if (xdir && xdir[0] && g_file_test (xdir, G_FILE_TEST_IS_DIR))
|
|
|
|
|
gtk_file_chooser_set_current_folder (GTK_FILE_CHOOSER (dialog), xdir);
|
|
|
|
|
}
|
2011-02-24 04:14:30 +01:00
|
|
|
}
|
2013-08-30 20:19:10 -04:00
|
|
|
else if (!(flags & FRF_RECENTLYUSED))
|
2026-02-15 16:15:25 -07:00
|
|
|
{
|
|
|
|
|
if (xdir && xdir[0] && g_file_test (xdir, G_FILE_TEST_IS_DIR))
|
|
|
|
|
gtk_file_chooser_set_current_folder (GTK_FILE_CHOOSER (dialog), xdir);
|
|
|
|
|
}
|
2013-08-30 20:19:10 -04:00
|
|
|
|
|
|
|
|
if (flags & FRF_MULTIPLE)
|
|
|
|
|
gtk_file_chooser_set_select_multiple (GTK_FILE_CHOOSER (dialog), TRUE);
|
|
|
|
|
if (flags & FRF_CHOOSEFOLDER)
|
|
|
|
|
gtk_file_chooser_set_action (GTK_FILE_CHOOSER (dialog), GTK_FILE_CHOOSER_ACTION_SELECT_FOLDER);
|
2011-02-24 04:14:30 +01:00
|
|
|
|
2013-08-31 15:31:22 -04:00
|
|
|
if ((flags & FRF_EXTENSIONS || flags & FRF_MIMETYPES) && extensions != NULL)
|
2012-07-21 21:42:48 +02:00
|
|
|
{
|
|
|
|
|
filefilter = gtk_file_filter_new ();
|
|
|
|
|
tokenbuffer = g_strdup (extensions);
|
|
|
|
|
token = strtok (tokenbuffer, ";");
|
|
|
|
|
|
|
|
|
|
while (token != NULL)
|
|
|
|
|
{
|
2013-08-31 15:31:22 -04:00
|
|
|
if (flags & FRF_EXTENSIONS)
|
|
|
|
|
gtk_file_filter_add_pattern (filefilter, token);
|
|
|
|
|
else
|
|
|
|
|
gtk_file_filter_add_mime_type (filefilter, token);
|
2012-07-21 21:42:48 +02:00
|
|
|
token = strtok (NULL, ";");
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
g_free (tokenbuffer);
|
|
|
|
|
gtk_file_chooser_set_filter (GTK_FILE_CHOOSER (dialog), filefilter);
|
|
|
|
|
}
|
|
|
|
|
|
2026-02-15 16:15:25 -07:00
|
|
|
if (xdir && xdir[0] && g_file_test (xdir, G_FILE_TEST_IS_DIR))
|
|
|
|
|
{
|
|
|
|
|
GError *shortcut_error = NULL;
|
|
|
|
|
|
|
|
|
|
gtk_file_chooser_add_shortcut_folder (GTK_FILE_CHOOSER (dialog), xdir, &shortcut_error);
|
|
|
|
|
if (shortcut_error)
|
|
|
|
|
g_error_free (shortcut_error);
|
|
|
|
|
}
|
2014-12-28 06:37:25 -05:00
|
|
|
freq = g_new (struct file_req, 1);
|
2011-02-24 04:14:30 +01:00
|
|
|
freq->dialog = dialog;
|
|
|
|
|
freq->flags = flags;
|
|
|
|
|
freq->callback = callback;
|
|
|
|
|
freq->userdata = userdata;
|
|
|
|
|
|
|
|
|
|
g_signal_connect (G_OBJECT (dialog), "response",
|
|
|
|
|
G_CALLBACK (gtkutil_file_req_response), freq);
|
|
|
|
|
g_signal_connect (G_OBJECT (dialog), "destroy",
|
|
|
|
|
G_CALLBACK (gtkutil_file_req_destroy), (gpointer) freq);
|
2022-04-16 18:41:34 -05:00
|
|
|
|
2026-02-16 15:06:59 -07:00
|
|
|
if (effective_parent)
|
|
|
|
|
gtk_window_set_transient_for (GTK_WINDOW (dialog), effective_parent);
|
2022-04-16 18:41:34 -05:00
|
|
|
|
|
|
|
|
if (flags & FRF_MODAL)
|
|
|
|
|
{
|
2026-02-16 15:06:59 -07:00
|
|
|
g_assert (effective_parent);
|
2022-04-16 18:41:34 -05:00
|
|
|
gtk_window_set_modal (GTK_WINDOW (dialog), TRUE);
|
|
|
|
|
}
|
|
|
|
|
|
2011-02-24 04:14:30 +01:00
|
|
|
gtk_widget_show (dialog);
|
|
|
|
|
}
|
|
|
|
|
|
2013-03-17 15:11:23 -07:00
|
|
|
static gboolean
|
|
|
|
|
gtkutil_esc_destroy (GtkWidget * win, GdkEventKey * key, gpointer userdata)
|
|
|
|
|
{
|
2013-04-05 17:33:35 -03:00
|
|
|
GtkWidget *wid;
|
2013-04-05 16:56:57 -03:00
|
|
|
|
2013-04-05 17:33:35 -03:00
|
|
|
/* Destroy the window of detached utils */
|
|
|
|
|
if (!gtk_widget_is_toplevel (win))
|
|
|
|
|
{
|
|
|
|
|
if (gdk_window_get_type_hint (gtk_widget_get_window (win)) == GDK_WINDOW_TYPE_HINT_DIALOG)
|
|
|
|
|
wid = gtk_widget_get_parent (win);
|
|
|
|
|
else
|
|
|
|
|
return FALSE;
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
wid = win;
|
2013-04-05 16:56:57 -03:00
|
|
|
|
2013-09-15 03:21:56 -04:00
|
|
|
if (key->keyval == GDK_KEY_Escape)
|
2013-04-05 17:33:35 -03:00
|
|
|
gtk_widget_destroy (wid);
|
|
|
|
|
|
2013-03-17 15:11:23 -07:00
|
|
|
return FALSE;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
|
gtkutil_destroy_on_esc (GtkWidget *win)
|
|
|
|
|
{
|
|
|
|
|
g_signal_connect (G_OBJECT (win), "key_press_event", G_CALLBACK (gtkutil_esc_destroy), win);
|
|
|
|
|
}
|
|
|
|
|
|
2011-02-24 04:14:30 +01:00
|
|
|
void
|
|
|
|
|
gtkutil_destroy (GtkWidget * igad, GtkWidget * dgad)
|
|
|
|
|
{
|
|
|
|
|
gtk_widget_destroy (dgad);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static void
|
|
|
|
|
gtkutil_get_str_response (GtkDialog *dialog, gint arg1, gpointer entry)
|
|
|
|
|
{
|
|
|
|
|
void (*callback) (int cancel, char *text, void *user_data);
|
|
|
|
|
char *text;
|
|
|
|
|
void *user_data;
|
|
|
|
|
|
|
|
|
|
text = (char *) gtk_entry_get_text (GTK_ENTRY (entry));
|
|
|
|
|
callback = g_object_get_data (G_OBJECT (dialog), "cb");
|
|
|
|
|
user_data = g_object_get_data (G_OBJECT (dialog), "ud");
|
|
|
|
|
|
|
|
|
|
switch (arg1)
|
|
|
|
|
{
|
|
|
|
|
case GTK_RESPONSE_REJECT:
|
|
|
|
|
callback (TRUE, text, user_data);
|
|
|
|
|
gtk_widget_destroy (GTK_WIDGET (dialog));
|
|
|
|
|
break;
|
|
|
|
|
case GTK_RESPONSE_ACCEPT:
|
|
|
|
|
callback (FALSE, text, user_data);
|
|
|
|
|
gtk_widget_destroy (GTK_WIDGET (dialog));
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static void
|
|
|
|
|
gtkutil_str_enter (GtkWidget *entry, GtkWidget *dialog)
|
|
|
|
|
{
|
|
|
|
|
gtk_dialog_response (GTK_DIALOG (dialog), GTK_RESPONSE_ACCEPT);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
|
fe_get_str (char *msg, char *def, void *callback, void *userdata)
|
|
|
|
|
{
|
|
|
|
|
GtkWidget *dialog;
|
|
|
|
|
GtkWidget *entry;
|
|
|
|
|
GtkWidget *hbox;
|
|
|
|
|
GtkWidget *label;
|
2013-07-03 19:48:45 +01:00
|
|
|
extern GtkWidget *parent_window;
|
2011-02-24 04:14:30 +01:00
|
|
|
|
2026-01-30 15:27:01 -07:00
|
|
|
#if HAVE_GTK3
|
|
|
|
|
dialog = gtk_dialog_new_with_buttons (msg, NULL, 0,
|
|
|
|
|
_("_Cancel"), GTK_RESPONSE_REJECT,
|
|
|
|
|
_("_OK"), GTK_RESPONSE_ACCEPT,
|
|
|
|
|
NULL);
|
|
|
|
|
#elif !HAVE_GTK3
|
2011-02-24 04:14:30 +01:00
|
|
|
dialog = gtk_dialog_new_with_buttons (msg, NULL, 0,
|
|
|
|
|
GTK_STOCK_CANCEL, GTK_RESPONSE_REJECT,
|
|
|
|
|
GTK_STOCK_OK, GTK_RESPONSE_ACCEPT,
|
|
|
|
|
NULL);
|
2026-01-30 15:27:01 -07:00
|
|
|
#endif
|
2013-05-12 01:43:27 -04:00
|
|
|
|
2013-07-03 19:48:45 +01:00
|
|
|
gtk_window_set_transient_for (GTK_WINDOW (dialog), GTK_WINDOW (parent_window));
|
2013-05-12 01:43:27 -04:00
|
|
|
gtk_box_set_homogeneous (GTK_BOX (gtk_dialog_get_content_area (GTK_DIALOG (dialog))), TRUE);
|
2012-10-22 22:27:30 +02:00
|
|
|
|
2013-01-19 12:33:16 -08:00
|
|
|
if (userdata == (void *)1) /* nick box is usually on the very bottom, make it centered */
|
2012-10-22 22:27:30 +02:00
|
|
|
{
|
|
|
|
|
gtk_window_set_position (GTK_WINDOW (dialog), GTK_WIN_POS_CENTER);
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
gtk_window_set_position (GTK_WINDOW (dialog), GTK_WIN_POS_MOUSE);
|
|
|
|
|
}
|
|
|
|
|
|
2026-02-05 01:59:15 -07:00
|
|
|
hbox = gtkutil_box_new (GTK_ORIENTATION_HORIZONTAL, TRUE, 0);
|
2011-02-24 04:14:30 +01:00
|
|
|
|
|
|
|
|
g_object_set_data (G_OBJECT (dialog), "cb", callback);
|
|
|
|
|
g_object_set_data (G_OBJECT (dialog), "ud", userdata);
|
|
|
|
|
|
|
|
|
|
entry = gtk_entry_new ();
|
|
|
|
|
g_signal_connect (G_OBJECT (entry), "activate",
|
|
|
|
|
G_CALLBACK (gtkutil_str_enter), dialog);
|
|
|
|
|
gtk_entry_set_text (GTK_ENTRY (entry), def);
|
|
|
|
|
gtk_box_pack_end (GTK_BOX (hbox), entry, 0, 0, 0);
|
|
|
|
|
|
|
|
|
|
label = gtk_label_new (msg);
|
|
|
|
|
gtk_box_pack_end (GTK_BOX (hbox), label, 0, 0, 0);
|
|
|
|
|
|
|
|
|
|
g_signal_connect (G_OBJECT (dialog), "response",
|
|
|
|
|
G_CALLBACK (gtkutil_get_str_response), entry);
|
|
|
|
|
|
2013-05-12 01:43:27 -04:00
|
|
|
gtk_container_add (GTK_CONTAINER (gtk_dialog_get_content_area (GTK_DIALOG (dialog))), hbox);
|
2011-02-24 04:14:30 +01:00
|
|
|
|
|
|
|
|
gtk_widget_show_all (dialog);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static void
|
|
|
|
|
gtkutil_get_number_response (GtkDialog *dialog, gint arg1, gpointer spin)
|
|
|
|
|
{
|
|
|
|
|
void (*callback) (int cancel, int value, void *user_data);
|
|
|
|
|
int num;
|
|
|
|
|
void *user_data;
|
|
|
|
|
|
|
|
|
|
num = gtk_spin_button_get_value_as_int (GTK_SPIN_BUTTON (spin));
|
|
|
|
|
callback = g_object_get_data (G_OBJECT (dialog), "cb");
|
|
|
|
|
user_data = g_object_get_data (G_OBJECT (dialog), "ud");
|
|
|
|
|
|
|
|
|
|
switch (arg1)
|
|
|
|
|
{
|
|
|
|
|
case GTK_RESPONSE_REJECT:
|
|
|
|
|
callback (TRUE, num, user_data);
|
|
|
|
|
gtk_widget_destroy (GTK_WIDGET (dialog));
|
|
|
|
|
break;
|
|
|
|
|
case GTK_RESPONSE_ACCEPT:
|
|
|
|
|
callback (FALSE, num, user_data);
|
|
|
|
|
gtk_widget_destroy (GTK_WIDGET (dialog));
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2013-12-31 17:06:58 -05:00
|
|
|
static void
|
|
|
|
|
gtkutil_get_bool_response (GtkDialog *dialog, gint arg1, gpointer spin)
|
|
|
|
|
{
|
|
|
|
|
void (*callback) (int value, void *user_data);
|
|
|
|
|
void *user_data;
|
|
|
|
|
|
|
|
|
|
callback = g_object_get_data (G_OBJECT (dialog), "cb");
|
|
|
|
|
user_data = g_object_get_data (G_OBJECT (dialog), "ud");
|
|
|
|
|
|
|
|
|
|
switch (arg1)
|
|
|
|
|
{
|
|
|
|
|
case GTK_RESPONSE_REJECT:
|
|
|
|
|
callback (0, user_data);
|
|
|
|
|
gtk_widget_destroy (GTK_WIDGET (dialog));
|
|
|
|
|
break;
|
|
|
|
|
case GTK_RESPONSE_ACCEPT:
|
|
|
|
|
callback (1, user_data);
|
|
|
|
|
gtk_widget_destroy (GTK_WIDGET (dialog));
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2011-02-24 04:14:30 +01:00
|
|
|
void
|
|
|
|
|
fe_get_int (char *msg, int def, void *callback, void *userdata)
|
|
|
|
|
{
|
|
|
|
|
GtkWidget *dialog;
|
|
|
|
|
GtkWidget *spin;
|
|
|
|
|
GtkWidget *hbox;
|
|
|
|
|
GtkWidget *label;
|
|
|
|
|
GtkAdjustment *adj;
|
2013-07-03 19:48:45 +01:00
|
|
|
extern GtkWidget *parent_window;
|
2011-02-24 04:14:30 +01:00
|
|
|
|
2026-01-30 15:27:01 -07:00
|
|
|
#if HAVE_GTK3
|
|
|
|
|
dialog = gtk_dialog_new_with_buttons (msg, NULL, 0,
|
|
|
|
|
_("_Cancel"), GTK_RESPONSE_REJECT,
|
|
|
|
|
_("_OK"), GTK_RESPONSE_ACCEPT,
|
|
|
|
|
NULL);
|
|
|
|
|
#elif !HAVE_GTK3
|
2011-02-24 04:14:30 +01:00
|
|
|
dialog = gtk_dialog_new_with_buttons (msg, NULL, 0,
|
|
|
|
|
GTK_STOCK_CANCEL, GTK_RESPONSE_REJECT,
|
|
|
|
|
GTK_STOCK_OK, GTK_RESPONSE_ACCEPT,
|
|
|
|
|
NULL);
|
2026-01-30 15:27:01 -07:00
|
|
|
#endif
|
2013-05-12 01:43:27 -04:00
|
|
|
gtk_box_set_homogeneous (GTK_BOX (gtk_dialog_get_content_area (GTK_DIALOG (dialog))), TRUE);
|
2011-02-24 04:14:30 +01:00
|
|
|
gtk_window_set_position (GTK_WINDOW (dialog), GTK_WIN_POS_MOUSE);
|
2013-07-03 19:48:45 +01:00
|
|
|
gtk_window_set_transient_for (GTK_WINDOW (dialog), GTK_WINDOW (parent_window));
|
|
|
|
|
|
2026-02-05 01:59:15 -07:00
|
|
|
hbox = gtkutil_box_new (GTK_ORIENTATION_HORIZONTAL, TRUE, 0);
|
2011-02-24 04:14:30 +01:00
|
|
|
|
|
|
|
|
g_object_set_data (G_OBJECT (dialog), "cb", callback);
|
|
|
|
|
g_object_set_data (G_OBJECT (dialog), "ud", userdata);
|
|
|
|
|
|
|
|
|
|
spin = gtk_spin_button_new (NULL, 1, 0);
|
|
|
|
|
adj = gtk_spin_button_get_adjustment ((GtkSpinButton*)spin);
|
2013-05-12 01:43:27 -04:00
|
|
|
gtk_adjustment_set_lower (adj, 0);
|
|
|
|
|
gtk_adjustment_set_upper (adj, 1024);
|
|
|
|
|
gtk_adjustment_set_step_increment (adj, 1);
|
2011-02-24 04:14:30 +01:00
|
|
|
gtk_adjustment_changed (adj);
|
|
|
|
|
gtk_spin_button_set_value ((GtkSpinButton*)spin, def);
|
|
|
|
|
gtk_box_pack_end (GTK_BOX (hbox), spin, 0, 0, 0);
|
|
|
|
|
|
|
|
|
|
label = gtk_label_new (msg);
|
|
|
|
|
gtk_box_pack_end (GTK_BOX (hbox), label, 0, 0, 0);
|
|
|
|
|
|
|
|
|
|
g_signal_connect (G_OBJECT (dialog), "response",
|
|
|
|
|
G_CALLBACK (gtkutil_get_number_response), spin);
|
|
|
|
|
|
2013-05-12 01:43:27 -04:00
|
|
|
gtk_container_add (GTK_CONTAINER (gtk_dialog_get_content_area (GTK_DIALOG (dialog))), hbox);
|
2011-02-24 04:14:30 +01:00
|
|
|
|
|
|
|
|
gtk_widget_show_all (dialog);
|
|
|
|
|
}
|
|
|
|
|
|
2013-12-31 17:06:58 -05:00
|
|
|
void
|
|
|
|
|
fe_get_bool (char *title, char *prompt, void *callback, void *userdata)
|
|
|
|
|
{
|
|
|
|
|
GtkWidget *dialog;
|
|
|
|
|
GtkWidget *prompt_label;
|
|
|
|
|
extern GtkWidget *parent_window;
|
|
|
|
|
|
2026-01-30 15:27:01 -07:00
|
|
|
#if HAVE_GTK3
|
|
|
|
|
dialog = gtk_dialog_new_with_buttons (title, NULL, 0,
|
|
|
|
|
_("_No"), GTK_RESPONSE_REJECT,
|
|
|
|
|
_("_Yes"), GTK_RESPONSE_ACCEPT,
|
|
|
|
|
NULL);
|
|
|
|
|
#elif !HAVE_GTK3
|
2013-12-31 17:06:58 -05:00
|
|
|
dialog = gtk_dialog_new_with_buttons (title, NULL, 0,
|
|
|
|
|
GTK_STOCK_NO, GTK_RESPONSE_REJECT,
|
|
|
|
|
GTK_STOCK_YES, GTK_RESPONSE_ACCEPT,
|
|
|
|
|
NULL);
|
2026-01-30 15:27:01 -07:00
|
|
|
#endif
|
2013-12-31 17:06:58 -05:00
|
|
|
gtk_box_set_homogeneous (GTK_BOX (gtk_dialog_get_content_area (GTK_DIALOG (dialog))), TRUE);
|
|
|
|
|
gtk_window_set_position (GTK_WINDOW (dialog), GTK_WIN_POS_MOUSE);
|
|
|
|
|
gtk_window_set_transient_for (GTK_WINDOW (dialog), GTK_WINDOW (parent_window));
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
g_object_set_data (G_OBJECT (dialog), "cb", callback);
|
|
|
|
|
g_object_set_data (G_OBJECT (dialog), "ud", userdata);
|
|
|
|
|
|
|
|
|
|
prompt_label = gtk_label_new (prompt);
|
|
|
|
|
|
|
|
|
|
g_signal_connect (G_OBJECT (dialog), "response",
|
|
|
|
|
G_CALLBACK (gtkutil_get_bool_response), NULL);
|
|
|
|
|
|
|
|
|
|
gtk_container_add (GTK_CONTAINER (gtk_dialog_get_content_area (GTK_DIALOG (dialog))), prompt_label);
|
|
|
|
|
|
|
|
|
|
gtk_widget_show_all (dialog);
|
|
|
|
|
}
|
|
|
|
|
|
2011-02-24 04:14:30 +01:00
|
|
|
GtkWidget *
|
|
|
|
|
gtkutil_button (GtkWidget *box, char *stock, char *tip, void *callback,
|
|
|
|
|
void *userdata, char *labeltext)
|
|
|
|
|
{
|
|
|
|
|
GtkWidget *wid, *img, *bbox;
|
|
|
|
|
|
|
|
|
|
wid = gtk_button_new ();
|
|
|
|
|
|
|
|
|
|
if (labeltext)
|
|
|
|
|
{
|
|
|
|
|
gtk_button_set_label (GTK_BUTTON (wid), labeltext);
|
2026-01-23 10:52:17 -07:00
|
|
|
img = NULL;
|
|
|
|
|
#if HAVE_GTK3
|
|
|
|
|
if (stock)
|
2026-01-23 11:17:28 -07:00
|
|
|
img = gtkutil_image_new_from_stock (stock, GTK_ICON_SIZE_BUTTON);
|
2026-01-23 10:52:17 -07:00
|
|
|
#endif
|
|
|
|
|
#if !HAVE_GTK3
|
|
|
|
|
if (stock)
|
|
|
|
|
img = gtk_image_new_from_stock (stock, GTK_ICON_SIZE_BUTTON);
|
|
|
|
|
#endif
|
|
|
|
|
if (img)
|
2026-01-31 15:35:43 -07:00
|
|
|
{
|
2026-01-23 10:52:17 -07:00
|
|
|
gtk_button_set_image (GTK_BUTTON (wid), img);
|
2026-01-31 15:35:43 -07:00
|
|
|
#if HAVE_GTK3
|
|
|
|
|
gtk_button_set_always_show_image (GTK_BUTTON (wid), TRUE);
|
|
|
|
|
#endif
|
|
|
|
|
}
|
2011-02-24 04:14:30 +01:00
|
|
|
gtk_button_set_use_underline (GTK_BUTTON (wid), TRUE);
|
|
|
|
|
if (box)
|
|
|
|
|
gtk_container_add (GTK_CONTAINER (box), wid);
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
2026-02-05 01:59:15 -07:00
|
|
|
bbox = gtkutil_box_new (GTK_ORIENTATION_HORIZONTAL, FALSE, 0);
|
2011-02-24 04:14:30 +01:00
|
|
|
gtk_container_add (GTK_CONTAINER (wid), bbox);
|
|
|
|
|
gtk_widget_show (bbox);
|
|
|
|
|
|
2026-01-23 10:52:17 -07:00
|
|
|
img = NULL;
|
|
|
|
|
#if HAVE_GTK3
|
|
|
|
|
if (stock)
|
2026-01-23 11:17:28 -07:00
|
|
|
img = gtkutil_image_new_from_stock (stock, GTK_ICON_SIZE_BUTTON);
|
2026-01-23 10:52:17 -07:00
|
|
|
#endif
|
|
|
|
|
#if !HAVE_GTK3
|
|
|
|
|
if (stock)
|
|
|
|
|
img = gtk_image_new_from_stock (stock, GTK_ICON_SIZE_BUTTON);
|
|
|
|
|
#endif
|
|
|
|
|
if (img)
|
|
|
|
|
{
|
|
|
|
|
gtk_container_add (GTK_CONTAINER (bbox), img);
|
|
|
|
|
gtk_widget_show (img);
|
2026-01-31 15:35:43 -07:00
|
|
|
#if HAVE_GTK3
|
|
|
|
|
gtk_button_set_always_show_image (GTK_BUTTON (wid), TRUE);
|
|
|
|
|
#endif
|
2026-01-23 10:52:17 -07:00
|
|
|
}
|
2011-02-24 04:14:30 +01:00
|
|
|
gtk_box_pack_start (GTK_BOX (box), wid, 0, 0, 0);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
g_signal_connect (G_OBJECT (wid), "clicked",
|
|
|
|
|
G_CALLBACK (callback), userdata);
|
|
|
|
|
gtk_widget_show (wid);
|
|
|
|
|
if (tip)
|
2014-01-18 04:08:32 -05:00
|
|
|
gtk_widget_set_tooltip_text (wid, tip);
|
2011-02-24 04:14:30 +01:00
|
|
|
|
|
|
|
|
return wid;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
|
gtkutil_label_new (char *text, GtkWidget * box)
|
|
|
|
|
{
|
|
|
|
|
GtkWidget *label = gtk_label_new (text);
|
|
|
|
|
gtk_container_add (GTK_CONTAINER (box), label);
|
|
|
|
|
gtk_widget_show (label);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
GtkWidget *
|
|
|
|
|
gtkutil_entry_new (int max, GtkWidget * box, void *callback,
|
|
|
|
|
gpointer userdata)
|
|
|
|
|
{
|
2014-02-08 09:53:56 -05:00
|
|
|
GtkWidget *entry = gtk_entry_new ();
|
|
|
|
|
gtk_entry_set_max_length (GTK_ENTRY (entry), max);
|
2011-02-24 04:14:30 +01:00
|
|
|
gtk_container_add (GTK_CONTAINER (box), entry);
|
|
|
|
|
if (callback)
|
|
|
|
|
g_signal_connect (G_OBJECT (entry), "changed",
|
|
|
|
|
G_CALLBACK (callback), userdata);
|
|
|
|
|
gtk_widget_show (entry);
|
|
|
|
|
return entry;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
|
show_and_unfocus (GtkWidget * wid)
|
|
|
|
|
{
|
2013-05-12 01:43:27 -04:00
|
|
|
gtk_widget_set_can_focus (wid, FALSE);
|
2011-02-24 04:14:30 +01:00
|
|
|
gtk_widget_show (wid);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
|
gtkutil_set_icon (GtkWidget *win)
|
|
|
|
|
{
|
2012-11-03 09:10:38 +01:00
|
|
|
#ifndef WIN32
|
|
|
|
|
/* FIXME: Magically breaks icon rendering in most
|
|
|
|
|
* (sub)windows, but OFC only on Windows. GTK <3
|
|
|
|
|
*/
|
2026-01-05 23:12:38 -07:00
|
|
|
gtk_window_set_icon (GTK_WINDOW (win), pix_zoitechat);
|
2012-11-03 09:10:38 +01:00
|
|
|
#endif
|
2011-02-24 04:14:30 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
extern GtkWidget *parent_window; /* maingui.c */
|
|
|
|
|
|
|
|
|
|
GtkWidget *
|
|
|
|
|
gtkutil_window_new (char *title, char *role, int width, int height, int flags)
|
|
|
|
|
{
|
|
|
|
|
GtkWidget *win;
|
|
|
|
|
|
|
|
|
|
win = gtk_window_new (GTK_WINDOW_TOPLEVEL);
|
|
|
|
|
gtkutil_set_icon (win);
|
|
|
|
|
#ifdef WIN32
|
2026-01-05 23:12:38 -07:00
|
|
|
gtk_window_set_wmclass (GTK_WINDOW (win), "ZoiteChat", "zoitechat");
|
2011-02-24 04:14:30 +01:00
|
|
|
#endif
|
|
|
|
|
gtk_window_set_title (GTK_WINDOW (win), title);
|
|
|
|
|
gtk_window_set_default_size (GTK_WINDOW (win), width, height);
|
|
|
|
|
gtk_window_set_role (GTK_WINDOW (win), role);
|
|
|
|
|
if (flags & 1)
|
|
|
|
|
gtk_window_set_position (GTK_WINDOW (win), GTK_WIN_POS_MOUSE);
|
|
|
|
|
if ((flags & 2) && parent_window)
|
|
|
|
|
{
|
|
|
|
|
gtk_window_set_type_hint (GTK_WINDOW (win), GDK_WINDOW_TYPE_HINT_DIALOG);
|
|
|
|
|
gtk_window_set_transient_for (GTK_WINDOW (win), GTK_WINDOW (parent_window));
|
2013-08-05 04:03:26 -04:00
|
|
|
gtk_window_set_destroy_with_parent (GTK_WINDOW (win), TRUE);
|
2011-02-24 04:14:30 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return win;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* pass NULL as selection to paste to both clipboard & X11 text */
|
|
|
|
|
void
|
|
|
|
|
gtkutil_copy_to_clipboard (GtkWidget *widget, GdkAtom selection,
|
|
|
|
|
const gchar *str)
|
|
|
|
|
{
|
|
|
|
|
GtkWidget *win;
|
|
|
|
|
GtkClipboard *clip, *clip2;
|
|
|
|
|
|
|
|
|
|
win = gtk_widget_get_toplevel (GTK_WIDGET (widget));
|
2013-05-12 01:43:27 -04:00
|
|
|
if (gtk_widget_is_toplevel (win))
|
2011-02-24 04:14:30 +01:00
|
|
|
{
|
|
|
|
|
int len = strlen (str);
|
|
|
|
|
|
|
|
|
|
if (selection)
|
|
|
|
|
{
|
|
|
|
|
clip = gtk_widget_get_clipboard (win, selection);
|
|
|
|
|
gtk_clipboard_set_text (clip, str, len);
|
|
|
|
|
} else
|
|
|
|
|
{
|
|
|
|
|
/* copy to both primary X selection and clipboard */
|
|
|
|
|
clip = gtk_widget_get_clipboard (win, GDK_SELECTION_PRIMARY);
|
|
|
|
|
clip2 = gtk_widget_get_clipboard (win, GDK_SELECTION_CLIPBOARD);
|
|
|
|
|
gtk_clipboard_set_text (clip, str, len);
|
|
|
|
|
gtk_clipboard_set_text (clip2, str, len);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* Treeview util functions */
|
|
|
|
|
|
|
|
|
|
GtkWidget *
|
|
|
|
|
gtkutil_treeview_new (GtkWidget *box, GtkTreeModel *model,
|
|
|
|
|
GtkTreeCellDataFunc mapper, ...)
|
|
|
|
|
{
|
|
|
|
|
GtkWidget *win, *view;
|
|
|
|
|
GtkCellRenderer *renderer = NULL;
|
|
|
|
|
GtkTreeViewColumn *col;
|
|
|
|
|
va_list args;
|
|
|
|
|
int col_id = 0;
|
|
|
|
|
GType type;
|
|
|
|
|
char *title, *attr;
|
|
|
|
|
|
|
|
|
|
win = gtk_scrolled_window_new (0, 0);
|
|
|
|
|
gtk_container_add (GTK_CONTAINER (box), win);
|
2026-02-16 18:40:15 -07:00
|
|
|
if (GTK_IS_BOX (box))
|
|
|
|
|
{
|
|
|
|
|
gtk_box_set_child_packing (GTK_BOX (box), win, TRUE, TRUE, 0,
|
|
|
|
|
GTK_PACK_START);
|
|
|
|
|
}
|
2011-02-24 04:14:30 +01:00
|
|
|
gtk_scrolled_window_set_policy (GTK_SCROLLED_WINDOW (win),
|
2026-02-16 18:40:15 -07:00
|
|
|
GTK_POLICY_AUTOMATIC, GTK_POLICY_AUTOMATIC);
|
|
|
|
|
gtk_widget_set_vexpand (win, TRUE);
|
|
|
|
|
gtk_widget_set_hexpand (win, TRUE);
|
2011-02-24 04:14:30 +01:00
|
|
|
gtk_widget_show (win);
|
|
|
|
|
|
|
|
|
|
view = gtk_tree_view_new_with_model (model);
|
|
|
|
|
/* the view now has a ref on the model, we can unref it */
|
|
|
|
|
g_object_unref (G_OBJECT (model));
|
|
|
|
|
gtk_container_add (GTK_CONTAINER (win), view);
|
|
|
|
|
|
|
|
|
|
va_start (args, mapper);
|
|
|
|
|
for (col_id = va_arg (args, int); col_id != -1; col_id = va_arg (args, int))
|
|
|
|
|
{
|
|
|
|
|
type = gtk_tree_model_get_column_type (model, col_id);
|
|
|
|
|
switch (type)
|
|
|
|
|
{
|
|
|
|
|
case G_TYPE_BOOLEAN:
|
|
|
|
|
renderer = gtk_cell_renderer_toggle_new ();
|
|
|
|
|
attr = "active";
|
|
|
|
|
break;
|
|
|
|
|
case G_TYPE_STRING: /* fall through */
|
|
|
|
|
default:
|
|
|
|
|
renderer = gtk_cell_renderer_text_new ();
|
|
|
|
|
attr = "text";
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
title = va_arg (args, char *);
|
|
|
|
|
if (mapper) /* user-specified function to set renderer attributes */
|
|
|
|
|
{
|
|
|
|
|
col = gtk_tree_view_column_new_with_attributes (title, renderer, NULL);
|
|
|
|
|
gtk_tree_view_column_set_cell_data_func (col, renderer, mapper,
|
|
|
|
|
GINT_TO_POINTER (col_id), NULL);
|
|
|
|
|
} else
|
|
|
|
|
{
|
|
|
|
|
/* just set the typical attribute for this type of renderer */
|
|
|
|
|
col = gtk_tree_view_column_new_with_attributes (title, renderer,
|
|
|
|
|
attr, col_id, NULL);
|
|
|
|
|
}
|
|
|
|
|
gtk_tree_view_append_column (GTK_TREE_VIEW (view), col);
|
2016-03-29 12:19:19 -04:00
|
|
|
if (title == NULL)
|
|
|
|
|
gtk_tree_view_column_set_visible (col, FALSE);
|
2011-02-24 04:14:30 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
va_end (args);
|
|
|
|
|
|
|
|
|
|
return view;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
gboolean
|
|
|
|
|
gtkutil_treemodel_string_to_iter (GtkTreeModel *model, gchar *pathstr, GtkTreeIter *iter_ret)
|
|
|
|
|
{
|
|
|
|
|
GtkTreePath *path = gtk_tree_path_new_from_string (pathstr);
|
|
|
|
|
gboolean success;
|
|
|
|
|
|
|
|
|
|
success = gtk_tree_model_get_iter (model, iter_ret, path);
|
|
|
|
|
gtk_tree_path_free (path);
|
|
|
|
|
return success;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/*gboolean
|
|
|
|
|
gtkutil_treeview_get_selected_iter (GtkTreeView *view, GtkTreeIter *iter_ret)
|
|
|
|
|
{
|
|
|
|
|
GtkTreeModel *store;
|
|
|
|
|
GtkTreeSelection *select;
|
|
|
|
|
|
|
|
|
|
select = gtk_tree_view_get_selection (view);
|
|
|
|
|
return gtk_tree_selection_get_selected (select, &store, iter_ret);
|
|
|
|
|
}*/
|
|
|
|
|
|
|
|
|
|
gboolean
|
|
|
|
|
gtkutil_treeview_get_selected (GtkTreeView *view, GtkTreeIter *iter_ret, ...)
|
|
|
|
|
{
|
|
|
|
|
GtkTreeModel *store;
|
|
|
|
|
GtkTreeSelection *select;
|
|
|
|
|
gboolean has_selected;
|
|
|
|
|
va_list args;
|
|
|
|
|
|
|
|
|
|
select = gtk_tree_view_get_selection (view);
|
|
|
|
|
has_selected = gtk_tree_selection_get_selected (select, &store, iter_ret);
|
|
|
|
|
|
|
|
|
|
if (has_selected) {
|
|
|
|
|
va_start (args, iter_ret);
|
|
|
|
|
gtk_tree_model_get_valist (store, iter_ret, args);
|
|
|
|
|
va_end (args);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return has_selected;
|
|
|
|
|
}
|
|
|
|
|
|
2017-08-31 09:52:45 -04:00
|
|
|
gboolean
|
|
|
|
|
gtkutil_tray_icon_supported (GtkWindow *window)
|
|
|
|
|
{
|
2026-01-23 00:43:30 -07:00
|
|
|
#ifdef GDK_WINDOWING_X11
|
2017-08-31 09:52:45 -04:00
|
|
|
GdkScreen *screen = gtk_window_get_screen (window);
|
|
|
|
|
GdkDisplay *display = gdk_screen_get_display (screen);
|
2026-02-04 21:07:49 -07:00
|
|
|
#if HAVE_GTK3
|
|
|
|
|
if (!GDK_IS_X11_DISPLAY (display))
|
2026-02-01 14:04:56 -07:00
|
|
|
return FALSE;
|
2026-02-01 15:23:36 -07:00
|
|
|
#endif
|
2017-08-31 09:52:45 -04:00
|
|
|
int screen_number = gdk_screen_get_number (screen);
|
|
|
|
|
Display *xdisplay = gdk_x11_display_get_xdisplay (display);
|
|
|
|
|
char *selection_name = g_strdup_printf ("_NET_SYSTEM_TRAY_S%d", screen_number);
|
|
|
|
|
Atom selection_atom = XInternAtom (xdisplay, selection_name, False);
|
|
|
|
|
Window tray_window = None;
|
|
|
|
|
|
|
|
|
|
XGrabServer (xdisplay);
|
|
|
|
|
|
|
|
|
|
tray_window = XGetSelectionOwner (xdisplay, selection_atom);
|
|
|
|
|
|
|
|
|
|
XUngrabServer (xdisplay);
|
|
|
|
|
XFlush (xdisplay);
|
|
|
|
|
g_free (selection_name);
|
|
|
|
|
|
|
|
|
|
return (tray_window != None);
|
2026-01-23 00:43:30 -07:00
|
|
|
#else
|
|
|
|
|
return TRUE;
|
2017-08-31 09:52:45 -04:00
|
|
|
#endif
|
|
|
|
|
}
|
|
|
|
|
|
2014-01-31 17:20:17 -05:00
|
|
|
#if defined (WIN32) || defined (__APPLE__)
|
|
|
|
|
gboolean
|
|
|
|
|
gtkutil_find_font (const char *fontname)
|
|
|
|
|
{
|
|
|
|
|
int i;
|
|
|
|
|
int n_families;
|
|
|
|
|
const char *family_name;
|
|
|
|
|
PangoFontMap *fontmap;
|
|
|
|
|
PangoFontFamily *family;
|
|
|
|
|
PangoFontFamily **families;
|
|
|
|
|
|
|
|
|
|
fontmap = pango_cairo_font_map_get_default ();
|
|
|
|
|
pango_font_map_list_families (fontmap, &families, &n_families);
|
|
|
|
|
|
|
|
|
|
for (i = 0; i < n_families; i++)
|
|
|
|
|
{
|
|
|
|
|
family = families[i];
|
|
|
|
|
family_name = pango_font_family_get_name (family);
|
|
|
|
|
|
|
|
|
|
if (!g_ascii_strcasecmp (family_name, fontname))
|
|
|
|
|
{
|
|
|
|
|
g_free (families);
|
|
|
|
|
return TRUE;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
g_free (families);
|
|
|
|
|
return FALSE;
|
|
|
|
|
}
|
|
|
|
|
#endif
|
2026-02-05 01:59:15 -07:00
|
|
|
|
|
|
|
|
GtkWidget *
|
|
|
|
|
gtkutil_box_new (GtkOrientation orientation, gboolean homogeneous, gint spacing)
|
|
|
|
|
{
|
|
|
|
|
#if HAVE_GTK3
|
|
|
|
|
GtkWidget *box = gtk_box_new (orientation, spacing);
|
|
|
|
|
|
|
|
|
|
gtk_box_set_homogeneous (GTK_BOX (box), homogeneous);
|
|
|
|
|
return box;
|
|
|
|
|
#elif !HAVE_GTK3
|
|
|
|
|
if (orientation == GTK_ORIENTATION_HORIZONTAL)
|
|
|
|
|
return gtk_hbox_new (homogeneous, spacing);
|
|
|
|
|
|
|
|
|
|
return gtk_vbox_new (homogeneous, spacing);
|
|
|
|
|
#endif
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
GtkWidget *
|
|
|
|
|
gtkutil_grid_new (guint rows, guint columns, gboolean homogeneous)
|
|
|
|
|
{
|
|
|
|
|
#if HAVE_GTK3
|
|
|
|
|
GtkWidget *grid = gtk_grid_new ();
|
|
|
|
|
|
|
|
|
|
gtk_grid_set_row_homogeneous (GTK_GRID (grid), homogeneous);
|
|
|
|
|
gtk_grid_set_column_homogeneous (GTK_GRID (grid), homogeneous);
|
|
|
|
|
return grid;
|
|
|
|
|
#elif !HAVE_GTK3
|
|
|
|
|
return gtk_table_new (rows, columns, homogeneous);
|
|
|
|
|
#endif
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
#if HAVE_GTK3
|
|
|
|
|
static GtkAlign
|
|
|
|
|
gtkutil_align_from_options (GtkutilAttachOptions options, GtkAlign default_align)
|
|
|
|
|
{
|
|
|
|
|
if (options & GTKUTIL_ATTACH_FILL)
|
|
|
|
|
return GTK_ALIGN_FILL;
|
|
|
|
|
|
|
|
|
|
return default_align;
|
|
|
|
|
}
|
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
|
static gboolean
|
|
|
|
|
gtkutil_expansion_from_options (GtkutilAttachOptions options, gboolean default_expand)
|
|
|
|
|
{
|
|
|
|
|
if (options & GTKUTIL_ATTACH_EXPAND)
|
|
|
|
|
return TRUE;
|
|
|
|
|
|
|
|
|
|
return default_expand;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
|
gtkutil_grid_attach (GtkWidget *table, GtkWidget *child,
|
|
|
|
|
guint left_attach, guint right_attach,
|
|
|
|
|
guint top_attach, guint bottom_attach,
|
|
|
|
|
GtkutilAttachOptions xoptions, GtkutilAttachOptions yoptions,
|
|
|
|
|
guint xpad, guint ypad)
|
|
|
|
|
{
|
|
|
|
|
#if HAVE_GTK3
|
|
|
|
|
gtk_widget_set_hexpand (child, gtkutil_expansion_from_options (xoptions, FALSE));
|
|
|
|
|
gtk_widget_set_vexpand (child, gtkutil_expansion_from_options (yoptions, FALSE));
|
|
|
|
|
gtk_widget_set_halign (child, gtkutil_align_from_options (xoptions, GTK_ALIGN_CENTER));
|
|
|
|
|
gtk_widget_set_valign (child, gtkutil_align_from_options (yoptions, GTK_ALIGN_CENTER));
|
|
|
|
|
gtk_widget_set_margin_start (child, xpad);
|
|
|
|
|
gtk_widget_set_margin_end (child, xpad);
|
|
|
|
|
gtk_widget_set_margin_top (child, ypad);
|
|
|
|
|
gtk_widget_set_margin_bottom (child, ypad);
|
|
|
|
|
gtk_grid_attach (GTK_GRID (table), child, left_attach, top_attach,
|
|
|
|
|
right_attach - left_attach, bottom_attach - top_attach);
|
|
|
|
|
#elif !HAVE_GTK3
|
|
|
|
|
gtk_table_attach (GTK_TABLE (table), child, left_attach, right_attach,
|
|
|
|
|
top_attach, bottom_attach, xoptions, yoptions, xpad, ypad);
|
|
|
|
|
#endif
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
|
gtkutil_grid_attach_defaults (GtkWidget *table, GtkWidget *child,
|
|
|
|
|
guint left_attach, guint right_attach,
|
|
|
|
|
guint top_attach, guint bottom_attach)
|
|
|
|
|
{
|
|
|
|
|
#if HAVE_GTK3
|
|
|
|
|
gtk_widget_set_hexpand (child, TRUE);
|
|
|
|
|
gtk_widget_set_vexpand (child, TRUE);
|
|
|
|
|
gtk_widget_set_halign (child, GTK_ALIGN_FILL);
|
|
|
|
|
gtk_widget_set_valign (child, GTK_ALIGN_FILL);
|
|
|
|
|
gtk_grid_attach (GTK_GRID (table), child, left_attach, top_attach,
|
|
|
|
|
right_attach - left_attach, bottom_attach - top_attach);
|
|
|
|
|
#elif !HAVE_GTK3
|
|
|
|
|
gtk_table_attach_defaults (GTK_TABLE (table), child, left_attach, right_attach,
|
|
|
|
|
top_attach, bottom_attach);
|
|
|
|
|
#endif
|
|
|
|
|
}
|