gnome-control-center/libbackground/preview-file-selection.c
Jody Goldberg 69bc347c0f Forgot to commit when I did the 2.1.0.1 snapshot
2002-10-01  Jody Goldberg <jody@gnome.org>

	* Release 2.1.0.1
2002-10-03 10:17:40 +00:00

302 lines
7.5 KiB
C

/* -*- mode: c; style: linux -*- */
/* preview-file-selection.c
* Copyright (C) 2001 Ximian, Inc.
*
* Written by Rachel Hestilow <hestilow@ximian.com>
*
* 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, 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
* Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA
* 02111-1307, USA.
*/
#ifdef HAVE_CONFIG_H
# include "config.h"
#endif
#include <libgnome/libgnome.h>
#include <gtk/gtkimage.h>
#include <gtk/gtkhbox.h>
#include <gtk/gtkframe.h>
#include <gtk/gtklabel.h>
#include <gtk/gtktreeview.h>
#include <gtk/gtkmain.h>
#include "preview-file-selection.h"
#define SCALE 100
struct _PreviewFileSelectionPrivate
{
GtkWidget *preview;
GtkWidget *label;
};
enum
{
PROP_0,
PROP_DO_PREVIEW
};
static GObjectClass *parent_class;
static void preview_file_selection_set_property (GObject *object, guint arg_id, const GValue *value, GParamSpec *spec);
static void preview_file_selection_get_property (GObject *object, guint arg_id, GValue *value, GParamSpec *spec);
static void
preview_file_selection_finalize (GObject *object)
{
PreviewFileSelection *fsel = PREVIEW_FILE_SELECTION (object);
g_free (fsel->priv);
parent_class->finalize (object);
}
static void
preview_file_selection_class_init (GObjectClass *object_class)
{
parent_class = g_type_class_ref (GTK_TYPE_FILE_SELECTION);
object_class->set_property = preview_file_selection_set_property;
object_class->get_property = preview_file_selection_get_property;
object_class->finalize = preview_file_selection_finalize;
g_object_class_install_property
(object_class, PROP_DO_PREVIEW,
g_param_spec_boolean ("do_preview",
"Preview images",
"Whether to preview images",
TRUE,
G_PARAM_READWRITE |
G_PARAM_CONSTRUCT_ONLY));
}
static void
preview_file_selection_init (GObject *object)
{
PreviewFileSelection *fsel = PREVIEW_FILE_SELECTION (object);
fsel->priv = g_new0 (PreviewFileSelectionPrivate, 1);
}
GType
preview_file_selection_get_type (void)
{
static GType type = 0;
if (!type)
{
GTypeInfo info =
{
sizeof (PreviewFileSelectionClass),
NULL,
NULL,
(GClassInitFunc) preview_file_selection_class_init,
NULL,
NULL,
sizeof (PreviewFileSelection),
0,
(GInstanceInitFunc) preview_file_selection_init,
NULL
};
type = g_type_register_static (GTK_TYPE_FILE_SELECTION,
"PreviewFileSelection",
&info, 0);
}
return type;
}
GtkWidget*
preview_file_selection_new (const gchar *title, gboolean do_preview)
{
return GTK_WIDGET (
g_object_new (PREVIEW_FILE_SELECTION_TYPE,
"title", title,
"do_preview", do_preview,
NULL));
}
GdkPixbuf*
preview_file_selection_intelligent_scale (GdkPixbuf *buf, guint scale)
{
GdkPixbuf *scaled;
int w, h;
guint ow = gdk_pixbuf_get_width (buf);
guint oh = gdk_pixbuf_get_height (buf);
if (ow <= scale && oh <= scale)
scaled = gdk_pixbuf_ref (buf);
else
{
if (ow > oh)
{
w = scale;
h = scale * (((double)oh)/(double)ow);
}
else
{
h = scale;
w = scale * (((double)ow)/(double)ow);
}
scaled = gdk_pixbuf_scale_simple (buf, w, h, GDK_INTERP_BILINEAR);
}
return scaled;
}
static void
preview_file_selection_update (PreviewFileSelection *fsel, gpointer data)
{
GdkPixbuf *buf;
const gchar *filename;
g_return_if_fail (IS_PREVIEW_FILE_SELECTION (fsel));
filename = gtk_file_selection_get_filename (GTK_FILE_SELECTION (fsel));
if (filename && (buf = gdk_pixbuf_new_from_file (filename, NULL)))
{
int w, h;
char *size;
GdkPixbuf *scaled = preview_file_selection_intelligent_scale (buf, SCALE);
gtk_image_set_from_pixbuf (GTK_IMAGE (fsel->priv->preview),
scaled);
g_object_unref (scaled);
w = gdk_pixbuf_get_width (buf);
h = gdk_pixbuf_get_height (buf);
size = g_strdup_printf ("%d x %d", w, h);
gtk_label_set_text (GTK_LABEL (fsel->priv->label), size);
g_free (size);
g_object_unref (buf);
}
else {
gtk_image_set_from_file (GTK_IMAGE (fsel->priv->preview), NULL);
gtk_label_set_text (GTK_LABEL (fsel->priv->label), " ");
}
}
static void
preview_file_selection_add_preview (PreviewFileSelection *fsel)
{
GtkWidget *hbox, *frame, *vbox;
g_return_if_fail (IS_PREVIEW_FILE_SELECTION (fsel));
hbox = GTK_FILE_SELECTION (fsel)->file_list;
do
{
hbox = hbox->parent;
if (!hbox)
{
g_warning (_("Can't find an hbox, using a normal file selection"));
return;
}
} while (!GTK_IS_HBOX (hbox));
frame = gtk_frame_new (_("Preview"));
gtk_widget_set_size_request (frame, SCALE + 10, SCALE + 10);
gtk_widget_show (frame);
gtk_box_pack_end (GTK_BOX (hbox), frame, FALSE, FALSE, 0);
vbox = gtk_vbox_new (FALSE, 2);
gtk_widget_show (vbox);
gtk_container_add (GTK_CONTAINER (frame), vbox);
fsel->priv->preview = gtk_image_new ();
gtk_box_pack_start (GTK_BOX (vbox), fsel->priv->preview, FALSE, FALSE, 0);
gtk_widget_show (fsel->priv->preview);
fsel->priv->label = gtk_label_new ("");
gtk_box_pack_start (GTK_BOX (vbox), fsel->priv->label, FALSE, FALSE, 0);
gtk_widget_show (fsel->priv->label);
g_signal_connect_data (G_OBJECT (gtk_tree_view_get_selection (GTK_TREE_VIEW (GTK_FILE_SELECTION (fsel)->file_list))), "changed", (GCallback) preview_file_selection_update, fsel, NULL, G_CONNECT_AFTER | G_CONNECT_SWAPPED);
preview_file_selection_update (fsel, NULL);
}
static void
preview_file_selection_set_property (GObject *object, guint arg_id, const GValue *value, GParamSpec *spec)
{
PreviewFileSelection *fsel = PREVIEW_FILE_SELECTION (object);
switch (arg_id)
{
case PROP_DO_PREVIEW:
if (g_value_get_boolean (value))
preview_file_selection_add_preview (fsel);
break;
}
}
static void preview_file_selection_get_property (GObject *object, guint arg_id, GValue *value, GParamSpec *spec)
{
PreviewFileSelection *fsel = PREVIEW_FILE_SELECTION (object);
switch (arg_id)
{
case PROP_DO_PREVIEW:
g_value_set_boolean (value, fsel->priv->preview != NULL);
break;
}
}
static void
browse_dialog_ok (GtkWidget *button, GnomeFileEntry *entry)
{
gnome_file_entry_set_filename (entry, gtk_file_selection_get_filename (GTK_FILE_SELECTION (entry->fsw)));
g_signal_emit_by_name (entry, "activate");
gtk_widget_destroy (entry->fsw);
}
static void
browse_dialog_kill (GtkFileSelection *fsel, GnomeFileEntry *entry)
{
entry->fsw = NULL;
}
static void
browse_clicked_cb (GnomeFileEntry *entry, const gchar *title)
{
GtkFileSelection *fs = GTK_FILE_SELECTION (preview_file_selection_new (title, TRUE));
entry->fsw = GTK_WIDGET (fs);
g_signal_connect (fs->ok_button, "clicked",
(GCallback) browse_dialog_ok, entry);
g_signal_connect_swapped (fs->cancel_button, "clicked",
(GCallback) gtk_widget_destroy, fs);
g_signal_connect (fs, "destroy",
(GCallback) browse_dialog_kill, entry);
if (gtk_grab_get_current ())
gtk_grab_add (entry->fsw);
}
void
preview_file_selection_hookup_file_entry (GnomeFileEntry *entry, const gchar *title)
{
g_return_if_fail (GNOME_IS_FILE_ENTRY (entry));
g_return_if_fail (title != NULL);
g_signal_connect (G_OBJECT (entry), "browse_clicked",
(GCallback) browse_clicked_cb, (gpointer) title);
}