/*
* Copyright (C) 2011-2013 Red Hat, Inc.
*
* 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, see .
*
* Author: Chris Lumens
*/
#include "BaseWindow.h"
#include "SpokeWindow.h"
#include "intl.h"
#include
/**
* SECTION: SpokeWindow
* @title: AnacondaSpokeWindow
* @short_description: Window for displaying single spokes
*
* A #AnacondaSpokeWindow is a top-level window that displays a single spoke
* on the entire screen. Examples include the keyboard and language
* configuration screens off the first hub.
*
* The window consists of two areas:
*
* - A navigation area in the top of the screen, inherited from #AnacondaBaseWindow
* and augmented with a button in the upper left corner.
*
* - An action area in the rest of the screen, taking up a majority of the
* space. This is where widgets will be added and the user will do things.
*/
enum {
PROP_BUTTON_LABEL = 1
};
#define DEFAULT_BUTTON_LABEL _("_Done")
enum {
SIGNAL_BUTTON_CLICKED,
LAST_SIGNAL
};
static guint window_signals[LAST_SIGNAL] = { 0 };
struct _AnacondaSpokeWindowPrivate {
GtkWidget *button;
};
G_DEFINE_TYPE(AnacondaSpokeWindow, anaconda_spoke_window, ANACONDA_TYPE_BASE_WINDOW)
static void anaconda_spoke_window_get_property(GObject *object, guint prop_id, GValue *value, GParamSpec *pspec);
static void anaconda_spoke_window_set_property(GObject *object, guint prop_id, const GValue *value, GParamSpec *pspec);
static void anaconda_spoke_window_realize(GtkWidget *widget, gpointer user_data);
static void anaconda_spoke_window_button_clicked(GtkButton *button,
AnacondaSpokeWindow *win);
static void anaconda_spoke_window_class_init(AnacondaSpokeWindowClass *klass) {
GObjectClass *object_class = G_OBJECT_CLASS(klass);
object_class->set_property = anaconda_spoke_window_set_property;
object_class->get_property = anaconda_spoke_window_get_property;
klass->button_clicked = NULL;
/**
* AnacondaSpokeWindow:button-label:
*
* The :button-label string is the text used to label the button displayed
* in the upper lefthand of the window. By default, this button says Done,
* but it could be changed to anything appropriate.
*
* Since: 1.0
*/
g_object_class_install_property(object_class,
PROP_BUTTON_LABEL,
g_param_spec_string("button-label",
P_("Button Label"),
P_("Label to appear on the upper left button"),
DEFAULT_BUTTON_LABEL,
G_PARAM_READWRITE));
/**
* AnacondaSpokeWindow::button-clicked:
* @window: the window that received the signal
*
* Emitted when the button in the upper left corner has been activated
* (pressed and released). This is commonly the button that takes the user
* back to the hub, but could do other things. Note that we do not want
* to trap people in spokes, so there should always be a way back to the
* hub via this signal, even if it involves canceling some operation or
* resetting things.
*
* Since: 1.0
*/
window_signals[SIGNAL_BUTTON_CLICKED] = g_signal_new("button-clicked",
G_TYPE_FROM_CLASS(object_class),
G_SIGNAL_RUN_FIRST | G_SIGNAL_ACTION,
G_STRUCT_OFFSET(AnacondaSpokeWindowClass, button_clicked),
NULL, NULL,
g_cclosure_marshal_VOID__VOID,
G_TYPE_NONE, 0);
g_type_class_add_private(object_class, sizeof(AnacondaSpokeWindowPrivate));
}
/**
* anaconda_spoke_window_new:
*
* Creates a new #AnacondaSpokeWindow, which is a window designed for
* displaying a single spoke, such as the keyboard or network configuration
* screens.
*
* Returns: A new #AnacondaSpokeWindow.
*/
GtkWidget *anaconda_spoke_window_new() {
return g_object_new(ANACONDA_TYPE_SPOKE_WINDOW, NULL);
}
static void anaconda_spoke_window_init(AnacondaSpokeWindow *win) {
GtkWidget *nav_area;
win->priv = G_TYPE_INSTANCE_GET_PRIVATE(win,
ANACONDA_TYPE_SPOKE_WINDOW,
AnacondaSpokeWindowPrivate);
g_signal_connect(win, "map", G_CALLBACK(anaconda_spoke_window_realize), NULL);
/* Set some default properties. */
gtk_window_set_modal(GTK_WINDOW(win), TRUE);
/* Create the button. */
win->priv->button = gtk_button_new_with_mnemonic(DEFAULT_BUTTON_LABEL);
gtk_widget_set_halign(win->priv->button, GTK_ALIGN_START);
gtk_widget_set_vexpand(win->priv->button, FALSE);
gtk_widget_set_valign(win->priv->button, GTK_ALIGN_END);
gtk_widget_set_margin_bottom(win->priv->button, 6);
/* Hook up some signals for that button. The signal handlers here will
* just raise our own custom signals for the whole window.
*/
g_signal_connect(win->priv->button, "clicked",
G_CALLBACK(anaconda_spoke_window_button_clicked), win);
/* And then put the button into the navigation area. */
nav_area = anaconda_base_window_get_nav_area(ANACONDA_BASE_WINDOW(win));
gtk_grid_attach(GTK_GRID(nav_area), win->priv->button, 0, 1, 1, 2);
}
static void anaconda_spoke_window_realize(GtkWidget *widget, gpointer user_data) {
GtkAccelGroup *accel_group;
GError *error;
GdkPixbuf *pixbuf;
cairo_pattern_t *pattern;
cairo_surface_t *surface;
cairo_t *cr;
AnacondaSpokeWindow *window = ANACONDA_SPOKE_WINDOW(widget);
/* Set the background gradient in the header. If we fail to load the
* background for any reason, just print an error message and display the
* header without an image.
*/
error = NULL;
pixbuf = gdk_pixbuf_new_from_file("/usr/share/anaconda/pixmaps/anaconda_spoke_header.png", &error);
if (!pixbuf) {
fprintf(stderr, "could not load header background: %s\n", error->message);
g_error_free(error);
} else {
GtkWidget *nav_box = anaconda_base_window_get_nav_area_background_window(ANACONDA_BASE_WINDOW(window));
gtk_widget_set_size_request(nav_box, -1, gdk_pixbuf_get_height (pixbuf));
surface = gdk_window_create_similar_surface(gtk_widget_get_window(nav_box), CAIRO_CONTENT_COLOR_ALPHA,
gdk_pixbuf_get_width(pixbuf), gdk_pixbuf_get_height(pixbuf));
cr = cairo_create(surface);
gdk_cairo_set_source_pixbuf(cr, pixbuf, 0, 0);
cairo_paint(cr);
cairo_destroy(cr);
pattern = cairo_pattern_create_for_surface(surface);
cairo_surface_destroy(surface);
cairo_pattern_set_extend(pattern, CAIRO_EXTEND_REPEAT);
gdk_window_set_background_pattern(gtk_widget_get_window(nav_box), pattern);
}
/* Pressing F12 should send you back to the hub, similar to how the old UI worked. */
accel_group = gtk_accel_group_new();
gtk_window_add_accel_group(GTK_WINDOW(window), accel_group);
gtk_widget_add_accelerator(window->priv->button,
"clicked",
accel_group,
GDK_KEY_F12,
0,
0);
}
static void anaconda_spoke_window_button_clicked(GtkButton *button,
AnacondaSpokeWindow *win) {
g_signal_emit(win, window_signals[SIGNAL_BUTTON_CLICKED], 0);
}
static void anaconda_spoke_window_get_property(GObject *object, guint prop_id, GValue *value, GParamSpec *pspec) {
AnacondaSpokeWindow *widget = ANACONDA_SPOKE_WINDOW(object);
AnacondaSpokeWindowPrivate *priv = widget->priv;
switch(prop_id) {
case PROP_BUTTON_LABEL:
g_value_set_string (value, gtk_button_get_label(GTK_BUTTON(priv->button)));
break;
}
}
static void anaconda_spoke_window_set_property(GObject *object, guint prop_id, const GValue *value, GParamSpec *pspec) {
AnacondaSpokeWindow *widget = ANACONDA_SPOKE_WINDOW(object);
AnacondaSpokeWindowPrivate *priv = widget->priv;
switch(prop_id) {
case PROP_BUTTON_LABEL:
gtk_button_set_label(GTK_BUTTON(priv->button), g_value_get_string(value));
gtk_button_set_use_underline(GTK_BUTTON(priv->button), TRUE);
break;
}
}