2 * GTK - The GIMP Toolkit
3 * Copyright (C) 1999 Red Hat, Inc.
4 * Copyright (C) 2002 Anders Carlsson <andersca@gnu.org>
5 * Copyright (C) 2003 Matthias Clasen <mclasen@redhat.com>
6 * Copyright (C) 2005 Carlos Garnacho Parro <carlosg@gnome.org>
10 * This library is free software; you can redistribute it and/or
11 * modify it under the terms of the GNU Lesser General Public
12 * License as published by the Free Software Foundation; either
13 * version 2 of the License, or (at your option) any later version.
15 * This library is distributed in the hope that it will be useful,
16 * but WITHOUT ANY WARRANTY; without even the implied warranty of
17 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
18 * Lesser General Public License for more details.
20 * You should have received a copy of the GNU Lesser General Public
21 * License along with this library; if not, write to the
22 * Free Software Foundation, Inc., 59 Temple Place - Suite 330,
23 * Boston, MA 02111-1307, USA.
27 * SECTION:gtkassistant
28 * @Short_description: A widget used to guide users through multi-step operations
29 * @Title: GtkAssistant
31 * A #GtkAssistant is a widget used to represent a generally complex
32 * operation splitted in several steps, guiding the user through its pages
33 * and controlling the page flow to collect the necessary data.
35 * <refsect2 id="GtkAssistant-BUILDER-UI">
36 * <title>GtkAssistant as GtkBuildable</title>
38 * The GtkAssistant implementation of the GtkBuildable interface exposes the
39 * @action_area as internal children with the name "action_area".
41 * To add pages to an assistant in GtkBuilder, simply add it as a
42 * <child> to the GtkAssistant object, and set its child properties
52 #include "gtkassistant.h"
54 #include "gtkaccessible.h"
55 #include "gtkbutton.h"
60 #include "gtksizegroup.h"
61 #include "gtksizerequest.h"
65 #include "gtkprivate.h"
66 #include "gtkbuildable.h"
69 #define HEADER_SPACING 12
70 #define ACTION_AREA_SPACING 12
72 typedef struct _GtkAssistantPage GtkAssistantPage;
74 struct _GtkAssistantPage
77 GtkAssistantPageType type;
79 guint complete_set : 1;
82 GdkPixbuf *header_image;
83 GdkPixbuf *sidebar_image;
86 struct _GtkAssistantPrivate
95 GtkWidget *header_image;
96 GtkWidget *sidebar_image;
98 GtkWidget *action_area;
102 GtkAssistantPage *current_page;
104 GSList *visited_pages;
106 GtkSizeGroup *size_group;
108 GtkAssistantPageFunc forward_function;
109 gpointer forward_function_data;
110 GDestroyNotify forward_data_destroy;
115 static void gtk_assistant_class_init (GtkAssistantClass *class);
116 static void gtk_assistant_init (GtkAssistant *assistant);
117 static void gtk_assistant_destroy (GtkObject *object);
118 static void gtk_assistant_style_set (GtkWidget *widget,
119 GtkStyle *old_style);
120 static void gtk_assistant_size_request (GtkWidget *widget,
121 GtkRequisition *requisition);
122 static void gtk_assistant_size_allocate (GtkWidget *widget,
123 GtkAllocation *allocation);
124 static void gtk_assistant_map (GtkWidget *widget);
125 static void gtk_assistant_unmap (GtkWidget *widget);
126 static gboolean gtk_assistant_delete_event (GtkWidget *widget,
128 static gboolean gtk_assistant_expose (GtkWidget *widget,
129 GdkEventExpose *event);
130 static gboolean gtk_assistant_focus (GtkWidget *widget,
131 GtkDirectionType direction);
132 static void gtk_assistant_add (GtkContainer *container,
134 static void gtk_assistant_remove (GtkContainer *container,
136 static void gtk_assistant_forall (GtkContainer *container,
137 gboolean include_internals,
138 GtkCallback callback,
139 gpointer callback_data);
140 static void gtk_assistant_set_child_property (GtkContainer *container,
145 static void gtk_assistant_get_child_property (GtkContainer *container,
151 static AtkObject *gtk_assistant_get_accessible (GtkWidget *widget);
153 static void gtk_assistant_buildable_interface_init (GtkBuildableIface *iface);
154 static GObject *gtk_assistant_buildable_get_internal_child (GtkBuildable *buildable,
156 const gchar *childname);
157 static gboolean gtk_assistant_buildable_custom_tag_start (GtkBuildable *buildable,
160 const gchar *tagname,
161 GMarkupParser *parser,
163 static void gtk_assistant_buildable_custom_finished (GtkBuildable *buildable,
166 const gchar *tagname,
169 static GList* find_page (GtkAssistant *assistant,
175 CHILD_PROP_PAGE_TYPE,
176 CHILD_PROP_PAGE_TITLE,
177 CHILD_PROP_PAGE_HEADER_IMAGE,
178 CHILD_PROP_PAGE_SIDEBAR_IMAGE,
179 CHILD_PROP_PAGE_COMPLETE
191 static guint signals [LAST_SIGNAL] = { 0 };
194 G_DEFINE_TYPE_WITH_CODE (GtkAssistant, gtk_assistant, GTK_TYPE_WINDOW,
195 G_IMPLEMENT_INTERFACE (GTK_TYPE_BUILDABLE,
196 gtk_assistant_buildable_interface_init))
200 gtk_assistant_class_init (GtkAssistantClass *class)
202 GObjectClass *gobject_class;
203 GtkObjectClass *object_class;
204 GtkWidgetClass *widget_class;
205 GtkContainerClass *container_class;
207 gobject_class = (GObjectClass *) class;
208 object_class = (GtkObjectClass *) class;
209 widget_class = (GtkWidgetClass *) class;
210 container_class = (GtkContainerClass *) class;
212 object_class->destroy = gtk_assistant_destroy;
214 widget_class->style_set = gtk_assistant_style_set;
215 widget_class->size_request = gtk_assistant_size_request;
216 widget_class->size_allocate = gtk_assistant_size_allocate;
217 widget_class->map = gtk_assistant_map;
218 widget_class->unmap = gtk_assistant_unmap;
219 widget_class->delete_event = gtk_assistant_delete_event;
220 widget_class->expose_event = gtk_assistant_expose;
221 widget_class->focus = gtk_assistant_focus;
222 widget_class->get_accessible = gtk_assistant_get_accessible;
224 container_class->add = gtk_assistant_add;
225 container_class->remove = gtk_assistant_remove;
226 container_class->forall = gtk_assistant_forall;
227 container_class->set_child_property = gtk_assistant_set_child_property;
228 container_class->get_child_property = gtk_assistant_get_child_property;
231 * GtkAssistant::cancel:
232 * @assistant: the #GtkAssistant
234 * The ::cancel signal is emitted when then the cancel button is clicked.
239 g_signal_new (I_("cancel"),
240 G_TYPE_FROM_CLASS (gobject_class),
242 G_STRUCT_OFFSET (GtkAssistantClass, cancel),
244 g_cclosure_marshal_VOID__VOID,
248 * GtkAssistant::prepare:
249 * @assistant: the #GtkAssistant
250 * @page: the current page
252 * The ::prepare signal is emitted when a new page is set as the assistant's
253 * current page, before making the new page visible. A handler for this signal
254 * can do any preparation which are necessary before showing @page.
259 g_signal_new (I_("prepare"),
260 G_TYPE_FROM_CLASS (gobject_class),
262 G_STRUCT_OFFSET (GtkAssistantClass, prepare),
264 g_cclosure_marshal_VOID__OBJECT,
265 G_TYPE_NONE, 1, GTK_TYPE_WIDGET);
268 * GtkAssistant::apply:
269 * @assistant: the #GtkAssistant
271 * The ::apply signal is emitted when the apply button is clicked. The default
272 * behavior of the #GtkAssistant is to switch to the page after the current
273 * page, unless the current page is the last one.
275 * A handler for the ::apply signal should carry out the actions for which
276 * the wizard has collected data. If the action takes a long time to complete,
277 * you might consider putting a page of type %GTK_ASSISTANT_PAGE_PROGRESS
278 * after the confirmation page and handle this operation within the
279 * #GtkAssistant::prepare signal of the progress page.
284 g_signal_new (I_("apply"),
285 G_TYPE_FROM_CLASS (gobject_class),
287 G_STRUCT_OFFSET (GtkAssistantClass, apply),
289 g_cclosure_marshal_VOID__VOID,
293 * GtkAssistant::close:
294 * @assistant: the #GtkAssistant
296 * The ::close signal is emitted either when the close button of
297 * a summary page is clicked, or when the apply button in the last
298 * page in the flow (of type %GTK_ASSISTANT_PAGE_CONFIRM) is clicked.
303 g_signal_new (I_("close"),
304 G_TYPE_FROM_CLASS (gobject_class),
306 G_STRUCT_OFFSET (GtkAssistantClass, close),
308 g_cclosure_marshal_VOID__VOID,
311 gtk_widget_class_install_style_property (widget_class,
312 g_param_spec_int ("header-padding",
313 P_("Header Padding"),
314 P_("Number of pixels around the header."),
318 GTK_PARAM_READABLE));
319 gtk_widget_class_install_style_property (widget_class,
320 g_param_spec_int ("content-padding",
321 P_("Content Padding"),
322 P_("Number of pixels around the content pages."),
326 GTK_PARAM_READABLE));
329 * GtkAssistant:page-type:
331 * The type of the assistant page.
335 gtk_container_class_install_child_property (container_class,
336 CHILD_PROP_PAGE_TYPE,
337 g_param_spec_enum ("page-type",
339 P_("The type of the assistant page"),
340 GTK_TYPE_ASSISTANT_PAGE_TYPE,
341 GTK_ASSISTANT_PAGE_CONTENT,
342 GTK_PARAM_READWRITE));
345 * GtkAssistant:title:
347 * The title that is displayed in the page header.
349 * If title and header-image are both %NULL, no header is displayed.
353 gtk_container_class_install_child_property (container_class,
354 CHILD_PROP_PAGE_TITLE,
355 g_param_spec_string ("title",
357 P_("The title of the assistant page"),
359 GTK_PARAM_READWRITE));
362 * GtkAssistant:header-image:
364 * The image that is displayed next to the title in the page header.
366 * If title and header-image are both %NULL, no header is displayed.
370 gtk_container_class_install_child_property (container_class,
371 CHILD_PROP_PAGE_HEADER_IMAGE,
372 g_param_spec_object ("header-image",
374 P_("Header image for the assistant page"),
376 GTK_PARAM_READWRITE));
379 * GtkAssistant:header-image:
381 * The image that is displayed next to the page.
383 * Set this to %NULL to make the sidebar disappear.
387 gtk_container_class_install_child_property (container_class,
388 CHILD_PROP_PAGE_SIDEBAR_IMAGE,
389 g_param_spec_object ("sidebar-image",
391 P_("Sidebar image for the assistant page"),
393 GTK_PARAM_READWRITE));
395 * GtkAssistant:complete:
397 * Setting the "complete" child property to %TRUE marks a page as complete
398 * (i.e.: all the required fields are filled out). GTK+ uses this information
399 * to control the sensitivity of the navigation buttons.
403 gtk_container_class_install_child_property (container_class,
404 CHILD_PROP_PAGE_COMPLETE,
405 g_param_spec_boolean ("complete",
407 P_("Whether all required fields on the page have been filled out"),
411 g_type_class_add_private (gobject_class, sizeof (GtkAssistantPrivate));
415 default_forward_function (gint current_page, gpointer data)
417 GtkAssistant *assistant;
418 GtkAssistantPrivate *priv;
419 GtkAssistantPage *page_info;
422 assistant = GTK_ASSISTANT (data);
423 priv = assistant->priv;
425 page_node = g_list_nth (priv->pages, ++current_page);
430 page_info = (GtkAssistantPage *) page_node->data;
432 while (page_node && !gtk_widget_get_visible (page_info->page))
434 page_node = page_node->next;
438 page_info = (GtkAssistantPage *) page_node->data;
445 compute_last_button_state (GtkAssistant *assistant)
447 GtkAssistantPrivate *priv = assistant->priv;
448 GtkAssistantPage *page_info, *current_page_info;
449 gint count, page_num, n_pages;
452 page_num = gtk_assistant_get_current_page (assistant);
453 n_pages = gtk_assistant_get_n_pages (assistant);
454 current_page_info = page_info = g_list_nth_data (priv->pages, page_num);
456 while (page_num >= 0 && page_num < n_pages &&
457 page_info->type == GTK_ASSISTANT_PAGE_CONTENT &&
458 (count == 0 || page_info->complete) &&
461 page_num = (priv->forward_function) (page_num, priv->forward_function_data);
462 page_info = g_list_nth_data (priv->pages, page_num);
467 /* make the last button visible if we can skip multiple
468 * pages and end on a confirmation or summary page
470 if (count > 1 && page_info &&
471 (page_info->type == GTK_ASSISTANT_PAGE_CONFIRM ||
472 page_info->type == GTK_ASSISTANT_PAGE_SUMMARY))
474 gtk_widget_show (priv->last);
475 gtk_widget_set_sensitive (priv->last,
476 current_page_info->complete);
479 gtk_widget_hide (priv->last);
483 compute_progress_state (GtkAssistant *assistant)
485 GtkAssistantPrivate *priv = assistant->priv;
486 gint page_num, n_pages;
488 n_pages = gtk_assistant_get_n_pages (assistant);
489 page_num = gtk_assistant_get_current_page (assistant);
491 page_num = (priv->forward_function) (page_num, priv->forward_function_data);
493 if (page_num >= 0 && page_num < n_pages)
494 gtk_widget_show (priv->forward);
496 gtk_widget_hide (priv->forward);
500 set_assistant_header_image (GtkAssistant *assistant)
502 GtkAssistantPrivate *priv = assistant->priv;
504 gtk_image_set_from_pixbuf (GTK_IMAGE (priv->header_image),
505 priv->current_page->header_image);
509 set_assistant_sidebar_image (GtkAssistant *assistant)
511 GtkAssistantPrivate *priv = assistant->priv;
513 gtk_image_set_from_pixbuf (GTK_IMAGE (priv->sidebar_image),
514 priv->current_page->sidebar_image);
516 if (priv->current_page->sidebar_image)
517 gtk_widget_show (priv->sidebar_image);
519 gtk_widget_hide (priv->sidebar_image);
523 set_assistant_buttons_state (GtkAssistant *assistant)
525 GtkAssistantPrivate *priv = assistant->priv;
527 if (!priv->current_page)
530 switch (priv->current_page->type)
532 case GTK_ASSISTANT_PAGE_INTRO:
533 gtk_widget_set_sensitive (priv->cancel, TRUE);
534 gtk_widget_set_sensitive (priv->forward, priv->current_page->complete);
535 gtk_widget_grab_default (priv->forward);
536 gtk_widget_show (priv->forward);
537 gtk_widget_hide (priv->back);
538 gtk_widget_hide (priv->apply);
539 gtk_widget_hide (priv->close);
540 compute_last_button_state (assistant);
542 case GTK_ASSISTANT_PAGE_CONFIRM:
543 gtk_widget_set_sensitive (priv->cancel, TRUE);
544 gtk_widget_set_sensitive (priv->back, TRUE);
545 gtk_widget_set_sensitive (priv->apply, priv->current_page->complete);
546 gtk_widget_grab_default (priv->apply);
547 gtk_widget_show (priv->back);
548 gtk_widget_show (priv->apply);
549 gtk_widget_hide (priv->forward);
550 gtk_widget_hide (priv->close);
551 gtk_widget_hide (priv->last);
553 case GTK_ASSISTANT_PAGE_CONTENT:
554 gtk_widget_set_sensitive (priv->cancel, TRUE);
555 gtk_widget_set_sensitive (priv->back, TRUE);
556 gtk_widget_set_sensitive (priv->forward, priv->current_page->complete);
557 gtk_widget_grab_default (priv->forward);
558 gtk_widget_show (priv->back);
559 gtk_widget_show (priv->forward);
560 gtk_widget_hide (priv->apply);
561 gtk_widget_hide (priv->close);
562 compute_last_button_state (assistant);
564 case GTK_ASSISTANT_PAGE_SUMMARY:
565 gtk_widget_set_sensitive (priv->close, priv->current_page->complete);
566 gtk_widget_grab_default (priv->close);
567 gtk_widget_show (priv->close);
568 gtk_widget_hide (priv->back);
569 gtk_widget_hide (priv->forward);
570 gtk_widget_hide (priv->apply);
571 gtk_widget_hide (priv->last);
573 case GTK_ASSISTANT_PAGE_PROGRESS:
574 gtk_widget_set_sensitive (priv->cancel, priv->current_page->complete);
575 gtk_widget_set_sensitive (priv->back, priv->current_page->complete);
576 gtk_widget_set_sensitive (priv->forward, priv->current_page->complete);
577 gtk_widget_grab_default (priv->forward);
578 gtk_widget_show (priv->back);
579 gtk_widget_hide (priv->apply);
580 gtk_widget_hide (priv->close);
581 gtk_widget_hide (priv->last);
582 compute_progress_state (assistant);
585 g_assert_not_reached ();
589 gtk_widget_hide (priv->cancel);
590 else if (priv->current_page->type == GTK_ASSISTANT_PAGE_SUMMARY)
591 gtk_widget_hide (priv->cancel);
593 gtk_widget_show (priv->cancel);
595 /* this is quite general, we don't want to
596 * go back if it's the first page */
597 if (!priv->visited_pages)
598 gtk_widget_hide (priv->back);
602 set_current_page (GtkAssistant *assistant,
603 GtkAssistantPage *page)
605 GtkAssistantPrivate *priv = assistant->priv;
606 GtkAssistantPage *old_page;
608 if (priv->current_page &&
609 gtk_widget_is_drawable (priv->current_page->page))
610 old_page = priv->current_page;
614 priv->current_page = page;
616 set_assistant_buttons_state (assistant);
617 set_assistant_header_image (assistant);
618 set_assistant_sidebar_image (assistant);
620 g_signal_emit (assistant, signals [PREPARE], 0, priv->current_page->page);
622 if (gtk_widget_get_visible (priv->current_page->page) && gtk_widget_get_mapped (GTK_WIDGET (assistant)))
624 gtk_widget_set_child_visible (priv->current_page->page, TRUE);
625 gtk_widget_map (priv->current_page->page);
626 gtk_widget_map (priv->current_page->title);
629 if (old_page && gtk_widget_get_mapped (old_page->page))
631 gtk_widget_set_child_visible (old_page->page, FALSE);
632 gtk_widget_unmap (old_page->page);
633 gtk_widget_unmap (old_page->title);
636 if (!gtk_widget_child_focus (priv->current_page->page, GTK_DIR_TAB_FORWARD))
638 GtkWidget *button[6];
641 /* find the best button to focus */
642 button[0] = priv->apply;
643 button[1] = priv->close;
644 button[2] = priv->forward;
645 button[3] = priv->back;
646 button[4] = priv->cancel;
647 button[5] = priv->last;
648 for (i = 0; i < 6; i++)
650 if (gtk_widget_get_visible (button[i]) && gtk_widget_get_sensitive (button[i]))
652 gtk_widget_grab_focus (button[i]);
658 gtk_widget_queue_resize (GTK_WIDGET (assistant));
662 compute_next_step (GtkAssistant *assistant)
664 GtkAssistantPrivate *priv = assistant->priv;
665 GtkAssistantPage *page_info;
666 gint current_page, n_pages, next_page;
668 current_page = gtk_assistant_get_current_page (assistant);
669 page_info = priv->current_page;
670 n_pages = gtk_assistant_get_n_pages (assistant);
672 next_page = (priv->forward_function) (current_page,
673 priv->forward_function_data);
675 if (next_page >= 0 && next_page < n_pages)
677 priv->visited_pages = g_slist_prepend (priv->visited_pages, page_info);
678 set_current_page (assistant, g_list_nth_data (priv->pages, next_page));
687 on_assistant_close (GtkWidget *widget,
688 GtkAssistant *assistant)
690 g_signal_emit (assistant, signals [CLOSE], 0, NULL);
694 on_assistant_apply (GtkWidget *widget,
695 GtkAssistant *assistant)
699 g_signal_emit (assistant, signals [APPLY], 0);
701 success = compute_next_step (assistant);
703 /* if the assistant hasn't switched to another page, just emit
704 * the CLOSE signal, it't the last page in the assistant flow
707 g_signal_emit (assistant, signals [CLOSE], 0);
711 on_assistant_forward (GtkWidget *widget,
712 GtkAssistant *assistant)
714 if (!compute_next_step (assistant))
715 g_critical ("Page flow is broken, you may want to end it with a page of "
716 "type GTK_ASSISTANT_PAGE_CONFIRM or GTK_ASSISTANT_PAGE_SUMMARY");
720 on_assistant_back (GtkWidget *widget,
721 GtkAssistant *assistant)
723 GtkAssistantPrivate *priv = assistant->priv;
724 GtkAssistantPage *page_info;
727 /* skip the progress pages when going back */
730 page_node = priv->visited_pages;
732 g_return_if_fail (page_node != NULL);
734 priv->visited_pages = priv->visited_pages->next;
735 page_info = (GtkAssistantPage *) page_node->data;
736 g_slist_free_1 (page_node);
738 while (page_info->type == GTK_ASSISTANT_PAGE_PROGRESS ||
739 !gtk_widget_get_visible (page_info->page));
741 set_current_page (assistant, page_info);
745 on_assistant_cancel (GtkWidget *widget,
746 GtkAssistant *assistant)
748 g_signal_emit (assistant, signals [CANCEL], 0, NULL);
752 on_assistant_last (GtkWidget *widget,
753 GtkAssistant *assistant)
755 GtkAssistantPrivate *priv = assistant->priv;
757 while (priv->current_page->type == GTK_ASSISTANT_PAGE_CONTENT &&
758 priv->current_page->complete)
759 compute_next_step (assistant);
763 alternative_button_order (GtkAssistant *assistant)
765 GtkSettings *settings;
769 screen = gtk_widget_get_screen (GTK_WIDGET (assistant));
770 settings = gtk_settings_get_for_screen (screen);
772 g_object_get (settings,
773 "gtk-alternative-button-order", &result,
779 gtk_assistant_init (GtkAssistant *assistant)
781 GtkAssistantPrivate *priv;
783 assistant->priv = G_TYPE_INSTANCE_GET_PRIVATE (assistant,
785 GtkAssistantPrivate);
786 priv = assistant->priv;
788 gtk_container_set_reallocate_redraws (GTK_CONTAINER (assistant), TRUE);
789 gtk_container_set_border_width (GTK_CONTAINER (assistant), 12);
791 gtk_widget_push_composite_child ();
794 priv->header_image = gtk_image_new ();
795 gtk_misc_set_alignment (GTK_MISC (priv->header_image), 1., 0.5);
796 gtk_widget_set_parent (priv->header_image, GTK_WIDGET (assistant));
797 gtk_widget_show (priv->header_image);
800 priv->sidebar_image = gtk_image_new ();
801 gtk_misc_set_alignment (GTK_MISC (priv->sidebar_image), 0., 0.);
802 gtk_widget_set_parent (priv->sidebar_image, GTK_WIDGET (assistant));
803 gtk_widget_show (priv->sidebar_image);
806 priv->action_area = gtk_hbox_new (FALSE, 6);
808 priv->close = gtk_button_new_from_stock (GTK_STOCK_CLOSE);
809 priv->apply = gtk_button_new_from_stock (GTK_STOCK_APPLY);
810 priv->forward = gtk_button_new_from_stock (GTK_STOCK_GO_FORWARD);
811 priv->back = gtk_button_new_from_stock (GTK_STOCK_GO_BACK);
812 priv->cancel = gtk_button_new_from_stock (GTK_STOCK_CANCEL);
813 priv->last = gtk_button_new_from_stock (GTK_STOCK_GOTO_LAST);
814 gtk_widget_set_can_default (priv->close, TRUE);
815 gtk_widget_set_can_default (priv->apply, TRUE);
816 gtk_widget_set_can_default (priv->forward, TRUE);
818 priv->size_group = gtk_size_group_new (GTK_SIZE_GROUP_HORIZONTAL);
819 gtk_size_group_add_widget (priv->size_group, priv->close);
820 gtk_size_group_add_widget (priv->size_group, priv->apply);
821 gtk_size_group_add_widget (priv->size_group, priv->forward);
822 gtk_size_group_add_widget (priv->size_group, priv->back);
823 gtk_size_group_add_widget (priv->size_group, priv->cancel);
824 gtk_size_group_add_widget (priv->size_group, priv->last);
826 if (!alternative_button_order (assistant))
828 gtk_box_pack_end (GTK_BOX (priv->action_area), priv->apply, FALSE, FALSE, 0);
829 gtk_box_pack_end (GTK_BOX (priv->action_area), priv->forward, FALSE, FALSE, 0);
830 gtk_box_pack_end (GTK_BOX (priv->action_area), priv->back, FALSE, FALSE, 0);
831 gtk_box_pack_end (GTK_BOX (priv->action_area), priv->last, FALSE, FALSE, 0);
832 gtk_box_pack_end (GTK_BOX (priv->action_area), priv->cancel, FALSE, FALSE, 0);
833 gtk_box_pack_end (GTK_BOX (priv->action_area), priv->close, FALSE, FALSE, 0);
837 gtk_box_pack_end (GTK_BOX (priv->action_area), priv->close, FALSE, FALSE, 0);
838 gtk_box_pack_end (GTK_BOX (priv->action_area), priv->cancel, FALSE, FALSE, 0);
839 gtk_box_pack_end (GTK_BOX (priv->action_area), priv->apply, FALSE, FALSE, 0);
840 gtk_box_pack_end (GTK_BOX (priv->action_area), priv->forward, FALSE, FALSE, 0);
841 gtk_box_pack_end (GTK_BOX (priv->action_area), priv->back, FALSE, FALSE, 0);
842 gtk_box_pack_end (GTK_BOX (priv->action_area), priv->last, FALSE, FALSE, 0);
845 gtk_widget_set_parent (priv->action_area, GTK_WIDGET (assistant));
846 gtk_widget_show (priv->forward);
847 gtk_widget_show (priv->back);
848 gtk_widget_show (priv->cancel);
849 gtk_widget_show (priv->action_area);
851 gtk_widget_pop_composite_child ();
854 priv->current_page = NULL;
855 priv->visited_pages = NULL;
857 priv->forward_function = default_forward_function;
858 priv->forward_function_data = assistant;
859 priv->forward_data_destroy = NULL;
861 g_signal_connect (G_OBJECT (priv->close), "clicked",
862 G_CALLBACK (on_assistant_close), assistant);
863 g_signal_connect (G_OBJECT (priv->apply), "clicked",
864 G_CALLBACK (on_assistant_apply), assistant);
865 g_signal_connect (G_OBJECT (priv->forward), "clicked",
866 G_CALLBACK (on_assistant_forward), assistant);
867 g_signal_connect (G_OBJECT (priv->back), "clicked",
868 G_CALLBACK (on_assistant_back), assistant);
869 g_signal_connect (G_OBJECT (priv->cancel), "clicked",
870 G_CALLBACK (on_assistant_cancel), assistant);
871 g_signal_connect (G_OBJECT (priv->last), "clicked",
872 G_CALLBACK (on_assistant_last), assistant);
876 gtk_assistant_set_child_property (GtkContainer *container,
884 case CHILD_PROP_PAGE_TYPE:
885 gtk_assistant_set_page_type (GTK_ASSISTANT (container), child,
886 g_value_get_enum (value));
888 case CHILD_PROP_PAGE_TITLE:
889 gtk_assistant_set_page_title (GTK_ASSISTANT (container), child,
890 g_value_get_string (value));
892 case CHILD_PROP_PAGE_HEADER_IMAGE:
893 gtk_assistant_set_page_header_image (GTK_ASSISTANT (container), child,
894 g_value_get_object (value));
896 case CHILD_PROP_PAGE_SIDEBAR_IMAGE:
897 gtk_assistant_set_page_side_image (GTK_ASSISTANT (container), child,
898 g_value_get_object (value));
900 case CHILD_PROP_PAGE_COMPLETE:
901 gtk_assistant_set_page_complete (GTK_ASSISTANT (container), child,
902 g_value_get_boolean (value));
905 GTK_CONTAINER_WARN_INVALID_CHILD_PROPERTY_ID (container, property_id, pspec);
911 gtk_assistant_get_child_property (GtkContainer *container,
919 case CHILD_PROP_PAGE_TYPE:
920 g_value_set_enum (value,
921 gtk_assistant_get_page_type (GTK_ASSISTANT (container), child));
923 case CHILD_PROP_PAGE_TITLE:
924 g_value_set_string (value,
925 gtk_assistant_get_page_title (GTK_ASSISTANT (container), child));
927 case CHILD_PROP_PAGE_HEADER_IMAGE:
928 g_value_set_object (value,
929 gtk_assistant_get_page_header_image (GTK_ASSISTANT (container), child));
931 case CHILD_PROP_PAGE_SIDEBAR_IMAGE:
932 g_value_set_object (value,
933 gtk_assistant_get_page_side_image (GTK_ASSISTANT (container), child));
935 case CHILD_PROP_PAGE_COMPLETE:
936 g_value_set_boolean (value,
937 gtk_assistant_get_page_complete (GTK_ASSISTANT (container), child));
940 GTK_CONTAINER_WARN_INVALID_CHILD_PROPERTY_ID (container, property_id, pspec);
946 on_page_notify_visibility (GtkWidget *widget,
950 GtkAssistant *assistant = GTK_ASSISTANT (data);
952 /* update buttons state, flow may have changed */
953 if (gtk_widget_get_mapped (GTK_WIDGET (assistant)))
954 set_assistant_buttons_state (assistant);
958 remove_page (GtkAssistant *assistant,
961 GtkAssistantPrivate *priv = assistant->priv;
962 GtkAssistantPage *page_info;
965 page_info = element->data;
967 /* If this is the current page, we need to switch away. */
968 if (page_info == priv->current_page)
970 if (!compute_next_step (assistant))
972 /* The best we can do at this point is probably to pick the first
975 page_node = priv->pages;
977 while (page_node && !gtk_widget_get_visible (((GtkAssistantPage *) page_node->data)->page))
978 page_node = page_node->next;
980 if (page_node == element)
981 page_node = page_node->next;
984 priv->current_page = page_node->data;
986 priv->current_page = NULL;
990 priv->pages = g_list_remove_link (priv->pages, element);
991 priv->visited_pages = g_slist_remove_all (priv->visited_pages, page_info);
993 g_signal_handlers_disconnect_by_func (page_info->page, on_page_notify_visibility, assistant);
994 gtk_widget_unparent (page_info->page);
996 if (page_info->header_image)
997 g_object_unref (page_info->header_image);
999 if (page_info->sidebar_image)
1000 g_object_unref (page_info->sidebar_image);
1002 gtk_widget_destroy (page_info->title);
1003 g_slice_free (GtkAssistantPage, page_info);
1004 g_list_free_1 (element);
1008 gtk_assistant_destroy (GtkObject *object)
1010 GtkAssistant *assistant = GTK_ASSISTANT (object);
1011 GtkAssistantPrivate *priv = assistant->priv;
1013 if (priv->header_image)
1015 gtk_widget_destroy (priv->header_image);
1016 priv->header_image = NULL;
1019 if (priv->sidebar_image)
1021 gtk_widget_destroy (priv->sidebar_image);
1022 priv->sidebar_image = NULL;
1025 if (priv->action_area)
1027 gtk_widget_destroy (priv->action_area);
1028 priv->action_area = NULL;
1031 if (priv->size_group)
1033 g_object_unref (priv->size_group);
1034 priv->size_group = NULL;
1037 if (priv->forward_function)
1039 if (priv->forward_function_data &&
1040 priv->forward_data_destroy)
1041 priv->forward_data_destroy (priv->forward_function_data);
1043 priv->forward_function = NULL;
1044 priv->forward_function_data = NULL;
1045 priv->forward_data_destroy = NULL;
1048 if (priv->visited_pages)
1050 g_slist_free (priv->visited_pages);
1051 priv->visited_pages = NULL;
1054 /* We set current to NULL so that the remove code doesn't try
1055 * to do anything funny */
1056 priv->current_page = NULL;
1059 remove_page (GTK_ASSISTANT (object), priv->pages);
1061 GTK_OBJECT_CLASS (gtk_assistant_parent_class)->destroy (object);
1065 find_page (GtkAssistant *assistant,
1068 GtkAssistantPrivate *priv = assistant->priv;
1069 GList *child = priv->pages;
1073 GtkAssistantPage *page_info = child->data;
1074 if (page_info->page == page)
1077 child = child->next;
1084 set_title_colors (GtkWidget *assistant,
1085 GtkWidget *title_label)
1089 gtk_widget_ensure_style (assistant);
1090 style = gtk_widget_get_style (assistant);
1092 /* change colors schema, for making the header text visible */
1093 gtk_widget_modify_bg (title_label, GTK_STATE_NORMAL, &style->bg[GTK_STATE_SELECTED]);
1094 gtk_widget_modify_fg (title_label, GTK_STATE_NORMAL, &style->fg[GTK_STATE_SELECTED]);
1098 set_title_font (GtkWidget *assistant,
1099 GtkWidget *title_label)
1101 PangoFontDescription *desc;
1104 desc = pango_font_description_new ();
1105 size = pango_font_description_get_size (gtk_widget_get_style (assistant)->font_desc);
1107 pango_font_description_set_weight (desc, PANGO_WEIGHT_ULTRABOLD);
1108 pango_font_description_set_size (desc, size * PANGO_SCALE_XX_LARGE);
1110 gtk_widget_modify_font (title_label, desc);
1111 pango_font_description_free (desc);
1115 gtk_assistant_style_set (GtkWidget *widget,
1116 GtkStyle *old_style)
1118 GtkAssistant *assistant = GTK_ASSISTANT (widget);
1119 GtkAssistantPrivate *priv = assistant->priv;
1126 GtkAssistantPage *page = list->data;
1128 set_title_colors (widget, page->title);
1129 set_title_font (widget, page->title);
1136 gtk_assistant_size_request (GtkWidget *widget,
1137 GtkRequisition *requisition)
1139 GtkAssistant *assistant = GTK_ASSISTANT (widget);
1140 GtkAssistantPrivate *priv = assistant->priv;
1141 GtkRequisition child_requisition;
1142 gint header_padding, content_padding;
1143 gint width, height, header_width, header_height;
1147 gtk_widget_style_get (widget,
1148 "header-padding", &header_padding,
1149 "content-padding", &content_padding,
1152 header_width = header_height = 0;
1157 GtkAssistantPage *page = list->data;
1160 gtk_size_request_get_size (GTK_SIZE_REQUEST (page->page),
1161 &child_requisition, NULL);
1162 width = MAX (width, child_requisition.width);
1163 height = MAX (height, child_requisition.height);
1165 gtk_size_request_get_size (GTK_SIZE_REQUEST (page->title),
1166 &child_requisition, NULL);
1167 w = child_requisition.width;
1168 h = child_requisition.height;
1170 if (page->header_image)
1172 w += gdk_pixbuf_get_width (page->header_image) + HEADER_SPACING;
1173 h = MAX (h, gdk_pixbuf_get_height (page->header_image));
1176 header_width = MAX (header_width, w);
1177 header_height = MAX (header_height, h);
1182 gtk_size_request_get_size (GTK_SIZE_REQUEST (priv->sidebar_image),
1183 &child_requisition, NULL);
1184 width += child_requisition.width;
1185 height = MAX (height, child_requisition.height);
1187 gtk_widget_set_size_request (priv->header_image, header_width, header_height);
1188 gtk_size_request_get_size (GTK_SIZE_REQUEST (priv->header_image),
1189 &child_requisition, NULL);
1190 width = MAX (width, header_width) + 2 * header_padding;
1191 height += header_height + 2 * header_padding;
1193 gtk_size_request_get_size (GTK_SIZE_REQUEST (priv->action_area),
1194 &child_requisition, NULL);
1195 width = MAX (width, child_requisition.width);
1196 height += child_requisition.height + ACTION_AREA_SPACING;
1198 border_width = gtk_container_get_border_width (GTK_CONTAINER (widget));
1199 width += border_width * 2 + content_padding * 2;
1200 height += border_width * 2 + content_padding * 2;
1202 requisition->width = width;
1203 requisition->height = height;
1208 gtk_assistant_size_allocate (GtkWidget *widget,
1209 GtkAllocation *allocation)
1211 GtkAssistant *assistant = GTK_ASSISTANT (widget);
1212 GtkAssistantPrivate *priv = assistant->priv;
1213 GtkRequisition header_requisition, action_requisition, sidebar_requisition;
1214 GtkAllocation child_allocation, header_allocation;
1215 GtkAllocation action_area_allocation, header_image_allocation;
1216 gint header_padding, content_padding;
1221 rtl = (gtk_widget_get_direction (widget) == GTK_TEXT_DIR_RTL);
1222 pages = priv->pages;
1224 gtk_widget_style_get (widget,
1225 "header-padding", &header_padding,
1226 "content-padding", &content_padding,
1229 gtk_widget_set_allocation (widget, allocation);
1230 border_width = gtk_container_get_border_width (GTK_CONTAINER (widget));
1233 gtk_size_request_get_size (GTK_SIZE_REQUEST (priv->header_image),
1234 &header_requisition, NULL);
1236 header_allocation.x = border_width + header_padding;
1237 header_allocation.y = border_width + header_padding;
1238 header_allocation.width = allocation->width - 2 * border_width - 2 * header_padding;
1239 header_allocation.height = header_requisition.height;
1241 gtk_widget_size_allocate (priv->header_image, &header_allocation);
1244 gtk_size_request_get_size (GTK_SIZE_REQUEST (priv->action_area),
1245 &action_requisition, NULL);
1247 child_allocation.x = border_width;
1248 child_allocation.y = allocation->height - border_width - action_requisition.height;
1249 child_allocation.width = allocation->width - 2 * border_width;
1250 child_allocation.height = action_requisition.height;
1252 gtk_widget_size_allocate (priv->action_area, &child_allocation);
1254 gtk_widget_get_allocation (priv->header_image, &header_image_allocation);
1255 gtk_widget_get_allocation (priv->action_area, &action_area_allocation);
1258 gtk_size_request_get_size (GTK_SIZE_REQUEST (priv->sidebar_image),
1259 &sidebar_requisition, NULL);
1262 child_allocation.x = allocation->width - border_width - sidebar_requisition.width;
1264 child_allocation.x = border_width;
1266 child_allocation.y = border_width + header_image_allocation.height + 2 * header_padding;
1267 child_allocation.width = sidebar_requisition.width;
1268 child_allocation.height = allocation->height - 2 * border_width -
1269 header_image_allocation.height - 2 * header_padding - action_area_allocation.height;
1271 gtk_widget_size_allocate (priv->sidebar_image, &child_allocation);
1274 child_allocation.x = border_width + content_padding;
1275 child_allocation.y = border_width +
1276 header_image_allocation.height + 2 * header_padding + content_padding;
1277 child_allocation.width = allocation->width - 2 * border_width - 2 * content_padding;
1278 child_allocation.height = allocation->height - 2 * border_width -
1279 header_image_allocation.height - 2 * header_padding - ACTION_AREA_SPACING - action_area_allocation.height - 2 * content_padding;
1281 if (gtk_widget_get_visible (priv->sidebar_image))
1283 GtkAllocation sidebar_image_allocation;
1285 gtk_widget_get_allocation (priv->sidebar_image, &sidebar_image_allocation);
1288 child_allocation.x += sidebar_image_allocation.width;
1290 child_allocation.width -= sidebar_image_allocation.width;
1295 GtkAssistantPage *page = pages->data;
1297 gtk_widget_size_allocate (page->page, &child_allocation);
1298 gtk_widget_size_allocate (page->title, &header_allocation);
1299 pages = pages->next;
1304 gtk_assistant_map (GtkWidget *widget)
1306 GtkAssistant *assistant = GTK_ASSISTANT (widget);
1307 GtkAssistantPrivate *priv = assistant->priv;
1309 GtkAssistantPage *page;
1311 gtk_widget_set_mapped (widget, TRUE);
1313 gtk_widget_map (priv->header_image);
1314 gtk_widget_map (priv->action_area);
1316 if (gtk_widget_get_visible (priv->sidebar_image) &&
1317 !gtk_widget_get_mapped (priv->sidebar_image))
1318 gtk_widget_map (priv->sidebar_image);
1320 /* if there's no default page, pick the first one */
1322 if (!priv->current_page)
1324 page_node = priv->pages;
1326 while (page_node && !gtk_widget_get_visible (((GtkAssistantPage *) page_node->data)->page))
1327 page_node = page_node->next;
1330 page = page_node->data;
1334 gtk_widget_get_visible (page->page) &&
1335 !gtk_widget_get_mapped (page->page))
1336 set_current_page (assistant, page);
1338 GTK_WIDGET_CLASS (gtk_assistant_parent_class)->map (widget);
1342 gtk_assistant_unmap (GtkWidget *widget)
1344 GtkAssistant *assistant = GTK_ASSISTANT (widget);
1345 GtkAssistantPrivate *priv = assistant->priv;
1347 gtk_widget_set_mapped (widget, FALSE);
1349 gtk_widget_unmap (priv->header_image);
1350 gtk_widget_unmap (priv->action_area);
1352 if (gtk_widget_is_drawable (priv->sidebar_image))
1353 gtk_widget_unmap (priv->sidebar_image);
1355 if (priv->current_page &&
1356 gtk_widget_is_drawable (priv->current_page->page))
1357 gtk_widget_unmap (priv->current_page->page);
1359 g_slist_free (priv->visited_pages);
1360 priv->visited_pages = NULL;
1361 priv->current_page = NULL;
1363 GTK_WIDGET_CLASS (gtk_assistant_parent_class)->unmap (widget);
1367 gtk_assistant_delete_event (GtkWidget *widget,
1370 GtkAssistant *assistant = GTK_ASSISTANT (widget);
1371 GtkAssistantPrivate *priv = assistant->priv;
1373 /* Do not allow cancelling in the middle of a progress page */
1374 if (priv->current_page &&
1375 (priv->current_page->type != GTK_ASSISTANT_PAGE_PROGRESS ||
1376 priv->current_page->complete))
1377 g_signal_emit (widget, signals [CANCEL], 0, NULL);
1383 assistant_paint_colored_box (GtkWidget *widget)
1385 GtkAssistant *assistant = GTK_ASSISTANT (widget);
1386 GtkAssistantPrivate *priv = assistant->priv;
1387 GtkAllocation allocation, action_area_allocation, header_image_allocation;
1389 gint border_width, header_padding, content_padding;
1391 gint content_x, content_width;
1394 cr = gdk_cairo_create (gtk_widget_get_window (widget));
1395 rtl = (gtk_widget_get_direction (widget) == GTK_TEXT_DIR_RTL);
1396 border_width = gtk_container_get_border_width (GTK_CONTAINER (widget));
1398 gtk_widget_style_get (widget,
1399 "header-padding", &header_padding,
1400 "content-padding", &content_padding,
1403 style = gtk_widget_get_style (widget);
1404 gtk_widget_get_allocation (widget, &allocation);
1405 gtk_widget_get_allocation (priv->action_area, &action_area_allocation);
1406 gtk_widget_get_allocation (priv->header_image, &header_image_allocation);
1409 gdk_cairo_set_source_color (cr, &style->bg[GTK_STATE_SELECTED]);
1410 cairo_rectangle (cr,
1413 allocation.width - 2 * border_width,
1414 allocation.height - action_area_allocation.height - 2 * border_width - ACTION_AREA_SPACING);
1418 content_x = content_padding + border_width;
1419 content_width = allocation.width - 2 * content_padding - 2 * border_width;
1421 if (gtk_widget_get_visible (priv->sidebar_image))
1423 GtkAllocation sidebar_image_allocation;
1425 gtk_widget_get_allocation (priv->sidebar_image, &sidebar_image_allocation);
1428 content_x += sidebar_image_allocation.width;
1429 content_width -= sidebar_image_allocation.width;
1432 gdk_cairo_set_source_color (cr, &style->bg[GTK_STATE_NORMAL]);
1434 cairo_rectangle (cr,
1436 header_image_allocation.height + content_padding + 2 * header_padding + border_width,
1438 allocation.height - 2 * border_width - action_area_allocation.height -
1439 header_image_allocation.height - 2 * content_padding - 2 * header_padding - ACTION_AREA_SPACING);
1446 gtk_assistant_expose (GtkWidget *widget,
1447 GdkEventExpose *event)
1449 GtkAssistant *assistant = GTK_ASSISTANT (widget);
1450 GtkAssistantPrivate *priv = assistant->priv;
1451 GtkContainer *container;
1453 if (gtk_widget_is_drawable (widget))
1455 container = GTK_CONTAINER (widget);
1457 assistant_paint_colored_box (widget);
1459 gtk_container_propagate_expose (container, priv->header_image, event);
1460 gtk_container_propagate_expose (container, priv->sidebar_image, event);
1461 gtk_container_propagate_expose (container, priv->action_area, event);
1463 if (priv->current_page)
1465 gtk_container_propagate_expose (container, priv->current_page->page, event);
1466 gtk_container_propagate_expose (container, priv->current_page->title, event);
1474 gtk_assistant_focus (GtkWidget *widget,
1475 GtkDirectionType direction)
1477 GtkAssistantPrivate *priv;
1478 GtkContainer *container;
1480 container = GTK_CONTAINER (widget);
1481 priv = GTK_ASSISTANT (widget)->priv;
1483 /* we only have to care about 2 widgets, action area and the current page */
1484 if (gtk_container_get_focus_child (container) == priv->action_area)
1486 if (!gtk_widget_child_focus (priv->action_area, direction) &&
1487 (priv->current_page == NULL ||
1488 !gtk_widget_child_focus (priv->current_page->page, direction)))
1490 /* if we're leaving the action area and the current page hasn't
1491 any focusable widget, clear focus and go back to the action area */
1492 gtk_container_set_focus_child (GTK_CONTAINER (priv->action_area), NULL);
1493 gtk_widget_child_focus (priv->action_area, direction);
1498 if ((priv->current_page == NULL ||
1499 !gtk_widget_child_focus (priv->current_page->page, direction)) &&
1500 !gtk_widget_child_focus (priv->action_area, direction))
1502 /* if we're leaving the current page and there isn't nothing focusable
1503 in the action area, try to clear focus and go back to the page */
1504 gtk_window_set_focus (GTK_WINDOW (widget), NULL);
1505 if (priv->current_page != NULL)
1506 gtk_widget_child_focus (priv->current_page->page, direction);
1514 gtk_assistant_add (GtkContainer *container,
1517 gtk_assistant_append_page (GTK_ASSISTANT (container), page);
1521 gtk_assistant_remove (GtkContainer *container,
1524 GtkAssistant *assistant = (GtkAssistant*) container;
1527 element = find_page (assistant, page);
1531 remove_page (assistant, element);
1532 gtk_widget_queue_resize ((GtkWidget *) container);
1537 gtk_assistant_forall (GtkContainer *container,
1538 gboolean include_internals,
1539 GtkCallback callback,
1540 gpointer callback_data)
1542 GtkAssistant *assistant = (GtkAssistant*) container;
1543 GtkAssistantPrivate *priv = assistant->priv;
1546 if (include_internals)
1548 (*callback) (priv->header_image, callback_data);
1549 (*callback) (priv->sidebar_image, callback_data);
1550 (*callback) (priv->action_area, callback_data);
1553 pages = priv->pages;
1557 GtkAssistantPage *page = (GtkAssistantPage *) pages->data;
1559 (*callback) (page->page, callback_data);
1561 if (include_internals)
1562 (*callback) (page->title, callback_data);
1564 pages = pages->next;
1569 * gtk_assistant_new:
1571 * Creates a new #GtkAssistant.
1573 * Return value: a newly created #GtkAssistant
1578 gtk_assistant_new (void)
1580 GtkWidget *assistant;
1582 assistant = g_object_new (GTK_TYPE_ASSISTANT, NULL);
1588 * gtk_assistant_get_current_page:
1589 * @assistant: a #GtkAssistant
1591 * Returns the page number of the current page
1593 * Return value: The index (starting from 0) of the current page in
1594 * the @assistant, if the @assistant has no pages, -1 will be returned
1599 gtk_assistant_get_current_page (GtkAssistant *assistant)
1601 GtkAssistantPrivate *priv;
1603 g_return_val_if_fail (GTK_IS_ASSISTANT (assistant), -1);
1605 priv = assistant->priv;
1607 if (!priv->pages || !priv->current_page)
1610 return g_list_index (priv->pages, priv->current_page);
1614 * gtk_assistant_set_current_page:
1615 * @assistant: a #GtkAssistant
1616 * @page_num: index of the page to switch to, starting from 0.
1617 * If negative, the last page will be used. If greater
1618 * than the number of pages in the @assistant, nothing
1621 * Switches the page to @page_num. Note that this will only be necessary
1622 * in custom buttons, as the @assistant flow can be set with
1623 * gtk_assistant_set_forward_page_func().
1628 gtk_assistant_set_current_page (GtkAssistant *assistant,
1631 GtkAssistantPrivate *priv;
1632 GtkAssistantPage *page;
1634 g_return_if_fail (GTK_IS_ASSISTANT (assistant));
1636 priv = assistant->priv;
1639 page = (GtkAssistantPage *) g_list_nth_data (priv->pages, page_num);
1641 page = (GtkAssistantPage *) g_list_last (priv->pages)->data;
1643 g_return_if_fail (page != NULL);
1645 if (priv->current_page == page)
1648 /* only add the page to the visited list if the
1649 * assistant is mapped, if not, just use it as an
1650 * initial page setting, for the cases where the
1651 * initial page is != to 0
1653 if (gtk_widget_get_mapped (GTK_WIDGET (assistant)))
1654 priv->visited_pages = g_slist_prepend (priv->visited_pages,
1655 priv->current_page);
1657 set_current_page (assistant, page);
1661 * gtk_assistant_get_n_pages:
1662 * @assistant: a #GtkAssistant
1664 * Returns the number of pages in the @assistant
1666 * Return value: The number of pages in the @assistant.
1671 gtk_assistant_get_n_pages (GtkAssistant *assistant)
1673 GtkAssistantPrivate *priv;
1675 g_return_val_if_fail (GTK_IS_ASSISTANT (assistant), 0);
1677 priv = assistant->priv;
1679 return g_list_length (priv->pages);
1683 * gtk_assistant_get_nth_page:
1684 * @assistant: a #GtkAssistant
1685 * @page_num: The index of a page in the @assistant, or -1 to get the last page;
1687 * Returns the child widget contained in page number @page_num.
1689 * Return value: (transfer none): The child widget, or %NULL
1690 * if @page_num is out of bounds.
1695 gtk_assistant_get_nth_page (GtkAssistant *assistant,
1698 GtkAssistantPrivate *priv;
1699 GtkAssistantPage *page;
1702 g_return_val_if_fail (GTK_IS_ASSISTANT (assistant), NULL);
1703 g_return_val_if_fail (page_num >= -1, NULL);
1705 priv = assistant->priv;
1708 elem = g_list_last (priv->pages);
1710 elem = g_list_nth (priv->pages, page_num);
1715 page = (GtkAssistantPage *) elem->data;
1721 * gtk_assistant_prepend_page:
1722 * @assistant: a #GtkAssistant
1723 * @page: a #GtkWidget
1725 * Prepends a page to the @assistant.
1727 * Return value: the index (starting at 0) of the inserted page
1732 gtk_assistant_prepend_page (GtkAssistant *assistant,
1735 g_return_val_if_fail (GTK_IS_ASSISTANT (assistant), 0);
1736 g_return_val_if_fail (GTK_IS_WIDGET (page), 0);
1738 return gtk_assistant_insert_page (assistant, page, 0);
1742 * gtk_assistant_append_page:
1743 * @assistant: a #GtkAssistant
1744 * @page: a #GtkWidget
1746 * Appends a page to the @assistant.
1748 * Return value: the index (starting at 0) of the inserted page
1753 gtk_assistant_append_page (GtkAssistant *assistant,
1756 g_return_val_if_fail (GTK_IS_ASSISTANT (assistant), 0);
1757 g_return_val_if_fail (GTK_IS_WIDGET (page), 0);
1759 return gtk_assistant_insert_page (assistant, page, -1);
1763 * gtk_assistant_insert_page:
1764 * @assistant: a #GtkAssistant
1765 * @page: a #GtkWidget
1766 * @position: the index (starting at 0) at which to insert the page,
1767 * or -1 to append the page to the @assistant
1769 * Inserts a page in the @assistant at a given position.
1771 * Return value: the index (starting from 0) of the inserted page
1776 gtk_assistant_insert_page (GtkAssistant *assistant,
1780 GtkAssistantPrivate *priv;
1781 GtkAssistantPage *page_info;
1784 g_return_val_if_fail (GTK_IS_ASSISTANT (assistant), 0);
1785 g_return_val_if_fail (GTK_IS_WIDGET (page), 0);
1786 g_return_val_if_fail (gtk_widget_get_parent (page) == NULL, 0);
1787 g_return_val_if_fail (!gtk_widget_is_toplevel (page), 0);
1789 priv = assistant->priv;
1791 page_info = g_slice_new0 (GtkAssistantPage);
1792 page_info->page = page;
1793 page_info->title = gtk_label_new (NULL);
1795 g_signal_connect (G_OBJECT (page), "notify::visible",
1796 G_CALLBACK (on_page_notify_visibility), assistant);
1798 gtk_misc_set_alignment (GTK_MISC (page_info->title), 0.,0.5);
1799 set_title_colors (GTK_WIDGET (assistant), page_info->title);
1800 set_title_font (GTK_WIDGET (assistant), page_info->title);
1801 gtk_widget_show (page_info->title);
1803 n_pages = g_list_length (priv->pages);
1805 if (position < 0 || position > n_pages)
1808 priv->pages = g_list_insert (priv->pages, page_info, position);
1810 gtk_widget_set_child_visible (page_info->page, FALSE);
1811 gtk_widget_set_parent (page_info->page, GTK_WIDGET (assistant));
1812 gtk_widget_set_parent (page_info->title, GTK_WIDGET (assistant));
1814 if (gtk_widget_get_realized (GTK_WIDGET (assistant)))
1816 gtk_widget_realize (page_info->page);
1817 gtk_widget_realize (page_info->title);
1820 gtk_widget_queue_resize (GTK_WIDGET (assistant));
1826 * gtk_assistant_set_forward_page_func:
1827 * @assistant: a #GtkAssistant
1828 * @page_func: (allow-none): the #GtkAssistantPageFunc, or %NULL to use the default one
1829 * @data: user data for @page_func
1830 * @destroy: destroy notifier for @data
1832 * Sets the page forwarding function to be @page_func, this function will
1833 * be used to determine what will be the next page when the user presses
1834 * the forward button. Setting @page_func to %NULL will make the assistant
1835 * to use the default forward function, which just goes to the next visible
1841 gtk_assistant_set_forward_page_func (GtkAssistant *assistant,
1842 GtkAssistantPageFunc page_func,
1844 GDestroyNotify destroy)
1846 GtkAssistantPrivate *priv;
1848 g_return_if_fail (GTK_IS_ASSISTANT (assistant));
1850 priv = assistant->priv;
1852 if (priv->forward_data_destroy &&
1853 priv->forward_function_data)
1854 (*priv->forward_data_destroy) (priv->forward_function_data);
1858 priv->forward_function = page_func;
1859 priv->forward_function_data = data;
1860 priv->forward_data_destroy = destroy;
1864 priv->forward_function = default_forward_function;
1865 priv->forward_function_data = assistant;
1866 priv->forward_data_destroy = NULL;
1869 /* Page flow has possibly changed, so the
1870 buttons state might need to change too */
1871 set_assistant_buttons_state (assistant);
1875 * gtk_assistant_add_action_widget:
1876 * @assistant: a #GtkAssistant
1877 * @child: a #GtkWidget
1879 * Adds a widget to the action area of a #GtkAssistant.
1884 gtk_assistant_add_action_widget (GtkAssistant *assistant,
1887 GtkAssistantPrivate *priv;
1889 g_return_if_fail (GTK_IS_ASSISTANT (assistant));
1890 g_return_if_fail (GTK_IS_WIDGET (child));
1892 priv = assistant->priv;
1894 if (GTK_IS_BUTTON (child))
1895 gtk_size_group_add_widget (priv->size_group, child);
1897 gtk_box_pack_end (GTK_BOX (priv->action_area), child, FALSE, FALSE, 0);
1901 * gtk_assistant_remove_action_widget:
1902 * @assistant: a #GtkAssistant
1903 * @child: a #GtkWidget
1905 * Removes a widget from the action area of a #GtkAssistant.
1910 gtk_assistant_remove_action_widget (GtkAssistant *assistant,
1913 GtkAssistantPrivate *priv;
1915 g_return_if_fail (GTK_IS_ASSISTANT (assistant));
1916 g_return_if_fail (GTK_IS_WIDGET (child));
1918 priv = assistant->priv;
1920 if (GTK_IS_BUTTON (child))
1921 gtk_size_group_remove_widget (priv->size_group, child);
1923 gtk_container_remove (GTK_CONTAINER (priv->action_area), child);
1927 * gtk_assistant_set_page_title:
1928 * @assistant: a #GtkAssistant
1929 * @page: a page of @assistant
1930 * @title: the new title for @page
1932 * Sets a title for @page. The title is displayed in the header
1933 * area of the assistant when @page is the current page.
1938 gtk_assistant_set_page_title (GtkAssistant *assistant,
1942 GtkAssistantPage *page_info;
1945 g_return_if_fail (GTK_IS_ASSISTANT (assistant));
1946 g_return_if_fail (GTK_IS_WIDGET (page));
1948 child = find_page (assistant, page);
1950 g_return_if_fail (child != NULL);
1952 page_info = (GtkAssistantPage*) child->data;
1954 gtk_label_set_text ((GtkLabel*) page_info->title, title);
1955 gtk_widget_queue_resize (GTK_WIDGET (assistant));
1956 gtk_widget_child_notify (page, "title");
1960 * gtk_assistant_get_page_title:
1961 * @assistant: a #GtkAssistant
1962 * @page: a page of @assistant
1964 * Gets the title for @page.
1966 * Return value: the title for @page.
1970 G_CONST_RETURN gchar*
1971 gtk_assistant_get_page_title (GtkAssistant *assistant,
1974 GtkAssistantPage *page_info;
1977 g_return_val_if_fail (GTK_IS_ASSISTANT (assistant), NULL);
1978 g_return_val_if_fail (GTK_IS_WIDGET (page), NULL);
1980 child = find_page (assistant, page);
1982 g_return_val_if_fail (child != NULL, NULL);
1984 page_info = (GtkAssistantPage*) child->data;
1986 return gtk_label_get_text ((GtkLabel*) page_info->title);
1990 * gtk_assistant_set_page_type:
1991 * @assistant: a #GtkAssistant
1992 * @page: a page of @assistant
1993 * @type: the new type for @page
1995 * Sets the page type for @page. The page type determines the page
1996 * behavior in the @assistant.
2001 gtk_assistant_set_page_type (GtkAssistant *assistant,
2003 GtkAssistantPageType type)
2005 GtkAssistantPrivate *priv;
2006 GtkAssistantPage *page_info;
2009 g_return_if_fail (GTK_IS_ASSISTANT (assistant));
2010 g_return_if_fail (GTK_IS_WIDGET (page));
2012 priv = assistant->priv;
2013 child = find_page (assistant, page);
2015 g_return_if_fail (child != NULL);
2017 page_info = (GtkAssistantPage*) child->data;
2019 if (type != page_info->type)
2021 page_info->type = type;
2023 /* backwards compatibility to the era before fixing bug 604289 */
2024 if (type == GTK_ASSISTANT_PAGE_SUMMARY && !page_info->complete_set)
2026 gtk_assistant_set_page_complete (assistant, page, TRUE);
2027 page_info->complete_set = FALSE;
2030 /* Always set buttons state, a change in a future page
2031 might change current page buttons */
2032 set_assistant_buttons_state (assistant);
2034 gtk_widget_child_notify (page, "page-type");
2039 * gtk_assistant_get_page_type:
2040 * @assistant: a #GtkAssistant
2041 * @page: a page of @assistant
2043 * Gets the page type of @page.
2045 * Return value: the page type of @page.
2049 GtkAssistantPageType
2050 gtk_assistant_get_page_type (GtkAssistant *assistant,
2053 GtkAssistantPage *page_info;
2056 g_return_val_if_fail (GTK_IS_ASSISTANT (assistant), GTK_ASSISTANT_PAGE_CONTENT);
2057 g_return_val_if_fail (GTK_IS_WIDGET (page), GTK_ASSISTANT_PAGE_CONTENT);
2059 child = find_page (assistant, page);
2061 g_return_val_if_fail (child != NULL, GTK_ASSISTANT_PAGE_CONTENT);
2063 page_info = (GtkAssistantPage*) child->data;
2065 return page_info->type;
2069 * gtk_assistant_set_page_header_image:
2070 * @assistant: a #GtkAssistant
2071 * @page: a page of @assistant
2072 * @pixbuf: (allow-none): the new header image @page
2074 * Sets a header image for @page. This image is displayed in the header
2075 * area of the assistant when @page is the current page.
2080 gtk_assistant_set_page_header_image (GtkAssistant *assistant,
2084 GtkAssistantPrivate *priv;
2085 GtkAssistantPage *page_info;
2088 g_return_if_fail (GTK_IS_ASSISTANT (assistant));
2089 g_return_if_fail (GTK_IS_WIDGET (page));
2090 g_return_if_fail (pixbuf == NULL || GDK_IS_PIXBUF (pixbuf));
2092 priv = assistant->priv;
2093 child = find_page (assistant, page);
2095 g_return_if_fail (child != NULL);
2097 page_info = (GtkAssistantPage*) child->data;
2099 if (pixbuf != page_info->header_image)
2101 if (page_info->header_image)
2103 g_object_unref (page_info->header_image);
2104 page_info->header_image = NULL;
2108 page_info->header_image = g_object_ref (pixbuf);
2110 if (page_info == priv->current_page)
2111 set_assistant_header_image (assistant);
2113 gtk_widget_child_notify (page, "header-image");
2118 * gtk_assistant_get_page_header_image:
2119 * @assistant: a #GtkAssistant
2120 * @page: a page of @assistant
2122 * Gets the header image for @page.
2124 * Return value: (transfer none): the header image for @page, or %NULL
2125 * if there's no header image for the page.
2130 gtk_assistant_get_page_header_image (GtkAssistant *assistant,
2133 GtkAssistantPage *page_info;
2136 g_return_val_if_fail (GTK_IS_ASSISTANT (assistant), NULL);
2137 g_return_val_if_fail (GTK_IS_WIDGET (page), NULL);
2139 child = find_page (assistant, page);
2141 g_return_val_if_fail (child != NULL, NULL);
2143 page_info = (GtkAssistantPage*) child->data;
2145 return page_info->header_image;
2149 * gtk_assistant_set_page_side_image:
2150 * @assistant: a #GtkAssistant
2151 * @page: a page of @assistant
2152 * @pixbuf: (allow-none): the new header image @page
2154 * Sets a header image for @page. This image is displayed in the side
2155 * area of the assistant when @page is the current page.
2160 gtk_assistant_set_page_side_image (GtkAssistant *assistant,
2164 GtkAssistantPrivate *priv;
2165 GtkAssistantPage *page_info;
2168 g_return_if_fail (GTK_IS_ASSISTANT (assistant));
2169 g_return_if_fail (GTK_IS_WIDGET (page));
2170 g_return_if_fail (pixbuf == NULL || GDK_IS_PIXBUF (pixbuf));
2172 priv = assistant->priv;
2173 child = find_page (assistant, page);
2175 g_return_if_fail (child != NULL);
2177 page_info = (GtkAssistantPage*) child->data;
2179 if (pixbuf != page_info->sidebar_image)
2181 if (page_info->sidebar_image)
2183 g_object_unref (page_info->sidebar_image);
2184 page_info->sidebar_image = NULL;
2188 page_info->sidebar_image = g_object_ref (pixbuf);
2190 if (page_info == priv->current_page)
2191 set_assistant_sidebar_image (assistant);
2193 gtk_widget_child_notify (page, "sidebar-image");
2198 * gtk_assistant_get_page_side_image:
2199 * @assistant: a #GtkAssistant
2200 * @page: a page of @assistant
2202 * Gets the header image for @page.
2204 * Return value: (transfer none): the side image for @page, or %NULL
2205 * if there's no side image for the page.
2210 gtk_assistant_get_page_side_image (GtkAssistant *assistant,
2213 GtkAssistantPage *page_info;
2216 g_return_val_if_fail (GTK_IS_ASSISTANT (assistant), NULL);
2217 g_return_val_if_fail (GTK_IS_WIDGET (page), NULL);
2219 child = find_page (assistant, page);
2221 g_return_val_if_fail (child != NULL, NULL);
2223 page_info = (GtkAssistantPage*) child->data;
2225 return page_info->sidebar_image;
2229 * gtk_assistant_set_page_complete:
2230 * @assistant: a #GtkAssistant
2231 * @page: a page of @assistant
2232 * @complete: the completeness status of the page
2234 * Sets whether @page contents are complete. This will make
2235 * @assistant update the buttons state to be able to continue the task.
2240 gtk_assistant_set_page_complete (GtkAssistant *assistant,
2244 GtkAssistantPrivate *priv;
2245 GtkAssistantPage *page_info;
2248 g_return_if_fail (GTK_IS_ASSISTANT (assistant));
2249 g_return_if_fail (GTK_IS_WIDGET (page));
2251 priv = assistant->priv;
2252 child = find_page (assistant, page);
2254 g_return_if_fail (child != NULL);
2256 page_info = (GtkAssistantPage*) child->data;
2258 if (complete != page_info->complete)
2260 page_info->complete = complete;
2261 page_info->complete_set = TRUE;
2263 /* Always set buttons state, a change in a future page
2264 might change current page buttons */
2265 set_assistant_buttons_state (assistant);
2267 gtk_widget_child_notify (page, "complete");
2272 * gtk_assistant_get_page_complete:
2273 * @assistant: a #GtkAssistant
2274 * @page: a page of @assistant
2276 * Gets whether @page is complete.
2278 * Return value: %TRUE if @page is complete.
2283 gtk_assistant_get_page_complete (GtkAssistant *assistant,
2286 GtkAssistantPage *page_info;
2289 g_return_val_if_fail (GTK_IS_ASSISTANT (assistant), FALSE);
2290 g_return_val_if_fail (GTK_IS_WIDGET (page), FALSE);
2292 child = find_page (assistant, page);
2294 g_return_val_if_fail (child != NULL, FALSE);
2296 page_info = (GtkAssistantPage*) child->data;
2298 return page_info->complete;
2302 * gtk_assistant_update_buttons_state:
2303 * @assistant: a #GtkAssistant
2305 * Forces @assistant to recompute the buttons state.
2307 * GTK+ automatically takes care of this in most situations,
2308 * e.g. when the user goes to a different page, or when the
2309 * visibility or completeness of a page changes.
2311 * One situation where it can be necessary to call this
2312 * function is when changing a value on the current page
2313 * affects the future page flow of the assistant.
2318 gtk_assistant_update_buttons_state (GtkAssistant *assistant)
2320 g_return_if_fail (GTK_IS_ASSISTANT (assistant));
2322 set_assistant_buttons_state (assistant);
2326 * gtk_assistant_commit:
2327 * @assistant: a #GtkAssistant
2329 * Erases the visited page history so the back button is not
2330 * shown on the current page, and removes the cancel button
2331 * from subsequent pages.
2333 * Use this when the information provided up to the current
2334 * page is hereafter deemed permanent and cannot be modified
2335 * or undone. For example, showing a progress page to track
2336 * a long-running, unreversible operation after the user has
2337 * clicked apply on a confirmation page.
2342 gtk_assistant_commit (GtkAssistant *assistant)
2344 g_return_if_fail (GTK_IS_ASSISTANT (assistant));
2346 g_slist_free (assistant->priv->visited_pages);
2347 assistant->priv->visited_pages = NULL;
2349 assistant->priv->committed = TRUE;
2351 set_assistant_buttons_state (assistant);
2356 /* accessible implementation */
2359 gtk_assistant_accessible_get_n_children (AtkObject *accessible)
2361 GtkAssistant *assistant;
2364 widget = gtk_accessible_get_widget (GTK_ACCESSIBLE (accessible));
2369 assistant = GTK_ASSISTANT (widget);
2371 return g_list_length (assistant->priv->pages) + 1;
2376 gtk_assistant_accessible_ref_child (AtkObject *accessible,
2379 GtkAssistant *assistant;
2380 GtkAssistantPrivate *priv;
2381 GtkWidget *widget, *child;
2386 widget = gtk_accessible_get_widget (GTK_ACCESSIBLE (accessible));
2390 assistant = GTK_ASSISTANT (widget);
2391 priv = assistant->priv;
2392 n_pages = g_list_length (priv->pages);
2396 else if (index < n_pages)
2398 GtkAssistantPage *page = g_list_nth_data (priv->pages, index);
2401 title = gtk_assistant_get_page_title (assistant, child);
2403 else if (index == n_pages)
2405 child = priv->action_area;
2411 obj = gtk_widget_get_accessible (child);
2414 atk_object_set_name (obj, title);
2416 return g_object_ref (obj);
2420 gtk_assistant_accessible_class_init (AtkObjectClass *class)
2422 class->get_n_children = gtk_assistant_accessible_get_n_children;
2423 class->ref_child = gtk_assistant_accessible_ref_child;
2427 gtk_assistant_accessible_get_type (void)
2429 static GType type = 0;
2434 * Figure out the size of the class and instance
2435 * we are deriving from
2437 AtkObjectFactory *factory;
2440 GType derived_atk_type;
2442 derived_type = g_type_parent (GTK_TYPE_ASSISTANT);
2443 factory = atk_registry_get_factory (atk_get_default_registry (),
2445 derived_atk_type = atk_object_factory_get_accessible_type (factory);
2446 g_type_query (derived_atk_type, &query);
2448 type = g_type_register_static_simple (derived_atk_type,
2449 I_("GtkAssistantAccessible"),
2451 (GClassInitFunc) gtk_assistant_accessible_class_init,
2452 query.instance_size,
2460 gtk_assistant_accessible_new (GObject *obj)
2462 AtkObject *accessible;
2464 g_return_val_if_fail (GTK_IS_ASSISTANT (obj), NULL);
2466 accessible = g_object_new (gtk_assistant_accessible_get_type (), NULL);
2467 atk_object_initialize (accessible, obj);
2473 gtk_assistant_accessible_factory_get_accessible_type (void)
2475 return gtk_assistant_accessible_get_type ();
2479 gtk_assistant_accessible_factory_create_accessible (GObject *obj)
2481 return gtk_assistant_accessible_new (obj);
2485 gtk_assistant_accessible_factory_class_init (AtkObjectFactoryClass *class)
2487 class->create_accessible = gtk_assistant_accessible_factory_create_accessible;
2488 class->get_accessible_type = gtk_assistant_accessible_factory_get_accessible_type;
2492 gtk_assistant_accessible_factory_get_type (void)
2494 static GType type = 0;
2498 type = g_type_register_static_simple (ATK_TYPE_OBJECT_FACTORY,
2499 I_("GtkAssistantAccessibleFactory"),
2500 sizeof (AtkObjectFactoryClass),
2501 (GClassInitFunc) gtk_assistant_accessible_factory_class_init,
2502 sizeof (AtkObjectFactory),
2510 gtk_assistant_get_accessible (GtkWidget *widget)
2512 static gboolean first_time = TRUE;
2516 AtkObjectFactory *factory;
2517 AtkRegistry *registry;
2519 GType derived_atk_type;
2522 * Figure out whether accessibility is enabled by looking at the
2523 * type of the accessible object which would be created for
2524 * the parent type of GtkAssistant.
2526 derived_type = g_type_parent (GTK_TYPE_ASSISTANT);
2528 registry = atk_get_default_registry ();
2529 factory = atk_registry_get_factory (registry,
2531 derived_atk_type = atk_object_factory_get_accessible_type (factory);
2532 if (g_type_is_a (derived_atk_type, GTK_TYPE_ACCESSIBLE))
2534 atk_registry_set_factory_type (registry,
2536 gtk_assistant_accessible_factory_get_type ());
2541 return GTK_WIDGET_CLASS (gtk_assistant_parent_class)->get_accessible (widget);
2545 static GtkBuildableIface *parent_buildable_iface;
2548 gtk_assistant_buildable_interface_init (GtkBuildableIface *iface)
2550 parent_buildable_iface = g_type_interface_peek_parent (iface);
2551 iface->get_internal_child = gtk_assistant_buildable_get_internal_child;
2552 iface->custom_tag_start = gtk_assistant_buildable_custom_tag_start;
2553 iface->custom_finished = gtk_assistant_buildable_custom_finished;
2557 gtk_assistant_buildable_get_internal_child (GtkBuildable *buildable,
2558 GtkBuilder *builder,
2559 const gchar *childname)
2561 if (strcmp (childname, "action_area") == 0)
2562 return G_OBJECT (GTK_ASSISTANT (buildable)->priv->action_area);
2564 return parent_buildable_iface->get_internal_child (buildable,
2570 gtk_assistant_buildable_custom_tag_start (GtkBuildable *buildable,
2571 GtkBuilder *builder,
2573 const gchar *tagname,
2574 GMarkupParser *parser,
2577 return parent_buildable_iface->custom_tag_start (buildable, builder, child,
2578 tagname, parser, data);
2582 gtk_assistant_buildable_custom_finished (GtkBuildable *buildable,
2583 GtkBuilder *builder,
2585 const gchar *tagname,
2588 parent_buildable_iface->custom_finished (buildable, builder, child,
2589 tagname, user_data);