207 lines
6.9 KiB
C
207 lines
6.9 KiB
C
/* -*- Mode: C; indent-tabs-mode: nil; c-basic-offset: 8; tab-width: 8 -*-
|
|
*
|
|
* gs-theme-window.c - special toplevel for screensavers
|
|
*
|
|
* Copyright (C) 2005 Ray Strode <rstrode@redhat.com>
|
|
*
|
|
* This program is free software; you can redistribute it and/or
|
|
* modify it under the terms of the GNU Lesser General Public
|
|
* License as published by the Free Software Foundation; either
|
|
* version 2, 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 Lesser General Public License for more
|
|
* details.
|
|
*
|
|
* You should have received a copy of the GNU Lesser General
|
|
* Public License along with this program; if not, write to the
|
|
* Free Software Foundation, Inc., 59 Temple Place - Suite 330,
|
|
* Boston, MA 02111-1307, USA.
|
|
*
|
|
* Originally written by: Ray Strode <rstrode@redhat.com>
|
|
*/
|
|
|
|
#include <errno.h>
|
|
#include <stdlib.h>
|
|
|
|
#include <glib.h>
|
|
#include <glib-object.h>
|
|
#include <glib/gi18n.h>
|
|
|
|
#include <gdk/gdkx.h>
|
|
#include <gtk/gtk.h>
|
|
|
|
#include "gs-theme-window.h"
|
|
|
|
static void gs_theme_window_finalize (GObject *object);
|
|
static void gs_theme_window_real_realize (GtkWidget *widget);
|
|
|
|
static GObjectClass *parent_class = NULL;
|
|
|
|
G_DEFINE_TYPE (GSThemeWindow, gs_theme_window, GTK_TYPE_WINDOW)
|
|
|
|
#define MIN_SIZE 10
|
|
|
|
static void
|
|
gs_theme_window_class_init (GSThemeWindowClass *klass)
|
|
{
|
|
GObjectClass *object_class;
|
|
GtkWidgetClass *widget_class;
|
|
|
|
object_class = G_OBJECT_CLASS (klass);
|
|
widget_class = GTK_WIDGET_CLASS (klass);
|
|
|
|
parent_class = g_type_class_peek_parent (klass);
|
|
|
|
object_class->finalize = gs_theme_window_finalize;
|
|
|
|
widget_class->realize = gs_theme_window_real_realize;
|
|
}
|
|
|
|
static void
|
|
force_no_pixmap_background (GtkWidget *widget)
|
|
{
|
|
static gboolean first_time = TRUE;
|
|
|
|
if (first_time) {
|
|
gtk_rc_parse_string ("\n"
|
|
" style \"gs-theme-engine-style\"\n"
|
|
" {\n"
|
|
" bg_pixmap[NORMAL] = \"<none>\"\n"
|
|
" bg_pixmap[ACTIVE] = \"<none>\"\n"
|
|
" bg_pixmap[PRELIGHT] = \"<none>\"\n"
|
|
" bg_pixmap[SELECTED] = \"<none>\"\n"
|
|
" bg_pixmap[INSENSITIVE] = \"<none>\"\n"
|
|
" bg[NORMAL] = \"#000000\"\n"
|
|
" bg[ACTIVE] = \"#000000\"\n"
|
|
" bg[PRELIGHT] = \"#000000\"\n"
|
|
" bg[SELECTED] = \"#000000\"\n"
|
|
" bg[INSENSITIVE] = \"#000000\"\n"
|
|
" }\n"
|
|
" widget \"gs-window*\" style : highest \"gs-theme-engine-style\"\n"
|
|
"\n");
|
|
first_time = FALSE;
|
|
}
|
|
|
|
gtk_widget_set_name (widget, "gs-window");
|
|
}
|
|
|
|
static void
|
|
gs_theme_window_init (GSThemeWindow *window)
|
|
{
|
|
force_no_pixmap_background (GTK_WIDGET (window));
|
|
}
|
|
|
|
static void
|
|
gs_theme_window_finalize (GObject *object)
|
|
{
|
|
GSThemeWindow *window;
|
|
GObjectClass *parent_class;
|
|
|
|
window = GS_THEME_WINDOW (object);
|
|
|
|
parent_class = G_OBJECT_CLASS (gs_theme_window_parent_class);
|
|
|
|
if (parent_class->finalize != NULL)
|
|
parent_class->finalize (object);
|
|
}
|
|
|
|
static void
|
|
gs_theme_window_real_realize (GtkWidget *widget)
|
|
{
|
|
GdkWindow *window;
|
|
Window remote_xwindow;
|
|
GtkRequisition requisition;
|
|
GtkAllocation allocation;
|
|
const char *preview_xid;
|
|
int x;
|
|
int y;
|
|
int width;
|
|
int height;
|
|
int event_mask;
|
|
|
|
event_mask = 0;
|
|
window = NULL;
|
|
preview_xid = g_getenv ("XSCREENSAVER_WINDOW");
|
|
|
|
if (preview_xid != NULL) {
|
|
char *end;
|
|
|
|
remote_xwindow = (Window) strtoul (preview_xid, &end, 0);
|
|
|
|
if ((remote_xwindow != 0) && (end != NULL) &&
|
|
((*end == ' ') || (*end == '\0')) &&
|
|
((remote_xwindow < G_MAXULONG) || (errno != ERANGE))) {
|
|
window = gdk_window_foreign_new (remote_xwindow);
|
|
|
|
if (window != NULL) {
|
|
/* This is a kludge; we need to set the same
|
|
* flags gs-window-x11.c does, to ensure they
|
|
* don't get unset by gtk_window_map() later.
|
|
*/
|
|
gtk_window_set_decorated (GTK_WINDOW (widget), FALSE);
|
|
|
|
gtk_window_set_skip_taskbar_hint (GTK_WINDOW (widget), TRUE);
|
|
gtk_window_set_skip_pager_hint (GTK_WINDOW (widget), TRUE);
|
|
|
|
gtk_window_set_keep_above (GTK_WINDOW (widget), TRUE);
|
|
|
|
gtk_window_fullscreen (GTK_WINDOW (widget));
|
|
|
|
event_mask = GDK_EXPOSURE_MASK | GDK_STRUCTURE_MASK;
|
|
gtk_widget_set_events (widget, gtk_widget_get_events (widget) | event_mask);
|
|
}
|
|
}
|
|
}
|
|
|
|
if (window == NULL) {
|
|
GtkWidgetClass *parent_class;
|
|
|
|
parent_class = GTK_WIDGET_CLASS (gs_theme_window_parent_class);
|
|
|
|
if (parent_class->realize != NULL)
|
|
parent_class->realize (widget);
|
|
|
|
return;
|
|
}
|
|
|
|
gtk_style_set_background (widget->style,
|
|
window,
|
|
GTK_STATE_NORMAL);
|
|
gdk_window_set_decorations (window, (GdkWMDecoration) 0);
|
|
gdk_window_set_events (window, gdk_window_get_events (window) | event_mask);
|
|
|
|
widget->window = window;
|
|
gdk_window_set_user_data (window, widget);
|
|
GTK_WIDGET_SET_FLAGS (widget, GTK_REALIZED);
|
|
|
|
gdk_window_get_geometry (window, &x, &y, &width, &height, NULL);
|
|
|
|
if (width < MIN_SIZE || height < MIN_SIZE) {
|
|
g_critical ("This window is way too small to use");
|
|
exit (1);
|
|
}
|
|
|
|
gtk_widget_size_request (widget, &requisition);
|
|
allocation.x = x;
|
|
allocation.y = y;
|
|
allocation.width = width;
|
|
allocation.height = height;
|
|
gtk_widget_size_allocate (widget, &allocation);
|
|
gtk_window_resize (GTK_WINDOW (widget), width, height);
|
|
}
|
|
|
|
GtkWidget *
|
|
gs_theme_window_new (void)
|
|
{
|
|
GSThemeWindow *window;
|
|
|
|
window = g_object_new (GS_TYPE_THEME_WINDOW,
|
|
"type", GTK_WINDOW_TOPLEVEL,
|
|
NULL);
|
|
|
|
return GTK_WIDGET (window);
|
|
}
|