| /* testcombochange.c |
| * Copyright (C) 2004 Red Hat, Inc. |
| * Author: Owen Taylor |
| * |
| * This library is free software; you can redistribute it and/or |
| * modify it under the terms of the GNU Library General Public |
| * License as published by the Free Software Foundation; either |
| * version 2 of the License, or (at your option) any later version. |
| * |
| * This library 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 |
| * Library General Public License for more details. |
| * |
| * You should have received a copy of the GNU Library General Public |
| * License along with this library. If not, see <http://www.gnu.org/licenses/>. |
| */ |
| |
| #include "config.h" |
| #include <gtk/gtk.h> |
| #include <stdarg.h> |
| |
| G_GNUC_BEGIN_IGNORE_DEPRECATIONS |
| |
| GtkWidget *text_view; |
| GtkListStore *model; |
| GArray *contents; |
| |
| static char next_value = 'A'; |
| |
| G_GNUC_PRINTF (1, 2) static void |
| combochange_log (const char *fmt, |
| ...) |
| { |
| GtkTextBuffer *buffer = gtk_text_view_get_buffer (GTK_TEXT_VIEW (text_view)); |
| GtkTextIter iter; |
| va_list vap; |
| char *msg; |
| GString *order_string; |
| GtkTextMark *tmp_mark; |
| int i; |
| |
| va_start (vap, fmt); |
| |
| msg = g_strdup_vprintf (fmt, vap); |
| |
| gtk_text_buffer_get_end_iter (buffer, &iter); |
| gtk_text_buffer_insert (buffer, &iter, msg, -1); |
| |
| order_string = g_string_new ("\n "); |
| for (i = 0; i < contents->len; i++) |
| { |
| if (i) |
| g_string_append_c (order_string, ' '); |
| g_string_append_c (order_string, g_array_index (contents, char, i)); |
| } |
| g_string_append_c (order_string, '\n'); |
| gtk_text_buffer_insert (buffer, &iter, order_string->str, -1); |
| g_string_free (order_string, TRUE); |
| |
| tmp_mark = gtk_text_buffer_create_mark (buffer, NULL, &iter, FALSE); |
| gtk_text_view_scroll_mark_onscreen (GTK_TEXT_VIEW (text_view), tmp_mark); |
| gtk_text_buffer_delete_mark (buffer, tmp_mark); |
| |
| g_free (msg); |
| } |
| |
| static void |
| on_insert (void) |
| { |
| GtkTreeIter iter; |
| |
| int insert_pos; |
| char new_value[2]; |
| |
| new_value[0] = next_value++; |
| new_value[1] = '\0'; |
| |
| if (next_value > 'Z') |
| next_value = 'A'; |
| |
| if (contents->len) |
| insert_pos = g_random_int_range (0, contents->len + 1); |
| else |
| insert_pos = 0; |
| |
| gtk_list_store_insert (model, &iter, insert_pos); |
| gtk_list_store_set (model, &iter, 0, new_value, -1); |
| |
| g_array_insert_val (contents, insert_pos, new_value); |
| |
| combochange_log ("Inserted '%c' at position %d", new_value[0], insert_pos); |
| } |
| |
| static void |
| on_delete (void) |
| { |
| GtkTreeIter iter; |
| |
| int delete_pos; |
| char old_val; |
| |
| if (!contents->len) |
| return; |
| |
| delete_pos = g_random_int_range (0, contents->len); |
| gtk_tree_model_iter_nth_child (GTK_TREE_MODEL (model), &iter, NULL, delete_pos); |
| |
| gtk_list_store_remove (model, &iter); |
| |
| old_val = g_array_index (contents, char, delete_pos); |
| g_array_remove_index (contents, delete_pos); |
| combochange_log ("Deleted '%c' from position %d", old_val, delete_pos); |
| } |
| |
| static void |
| on_reorder (void) |
| { |
| GArray *new_contents; |
| int *shuffle_array; |
| int i; |
| |
| shuffle_array = g_new (int, contents->len); |
| |
| for (i = 0; i < contents->len; i++) |
| shuffle_array[i] = i; |
| |
| for (i = 0; i + 1 < contents->len; i++) |
| { |
| int pos = g_random_int_range (i, contents->len); |
| int tmp; |
| |
| tmp = shuffle_array[i]; |
| shuffle_array[i] = shuffle_array[pos]; |
| shuffle_array[pos] = tmp; |
| } |
| |
| gtk_list_store_reorder (model, shuffle_array); |
| |
| new_contents = g_array_new (FALSE, FALSE, sizeof (char)); |
| for (i = 0; i < contents->len; i++) |
| g_array_append_val (new_contents, |
| g_array_index (contents, char, shuffle_array[i])); |
| g_array_free (contents, TRUE); |
| contents = new_contents; |
| |
| combochange_log ("Reordered array"); |
| |
| g_free (shuffle_array); |
| } |
| |
| static int n_animations = 0; |
| static int timer = 0; |
| |
| static int |
| animation_timer (gpointer data) |
| { |
| switch (g_random_int_range (0, 3)) |
| { |
| case 0: |
| on_insert (); |
| break; |
| case 1: |
| on_delete (); |
| break; |
| case 2: |
| on_reorder (); |
| break; |
| default: |
| g_assert_not_reached (); |
| } |
| |
| n_animations--; |
| return n_animations > 0; |
| } |
| |
| static void |
| on_animate (void) |
| { |
| n_animations += 20; |
| |
| timer = g_timeout_add (1000, (GSourceFunc) animation_timer, NULL); |
| } |
| |
| int |
| main (int argc, char **argv) |
| { |
| GtkWidget *content_area; |
| GtkWidget *window; |
| GtkWidget *scrolled_window; |
| GtkWidget *hbox; |
| GtkWidget *button_vbox; |
| GtkWidget *combo_vbox; |
| GtkWidget *button; |
| GtkWidget *combo; |
| GtkCellRenderer *cell_renderer; |
| |
| gtk_init (); |
| |
| model = gtk_list_store_new (1, G_TYPE_STRING); |
| contents = g_array_new (FALSE, FALSE, sizeof (char)); |
| |
| window = gtk_window_new (); |
| gtk_window_set_title (GTK_WINDOW (window), "ComboBox Change"); |
| |
| content_area = gtk_box_new (GTK_ORIENTATION_VERTICAL, 12); |
| gtk_window_set_child (GTK_WINDOW (window), content_area); |
| |
| hbox = gtk_box_new (GTK_ORIENTATION_HORIZONTAL, 12); |
| gtk_box_append (GTK_BOX (content_area), hbox); |
| |
| combo_vbox = gtk_box_new (GTK_ORIENTATION_VERTICAL, 8); |
| gtk_box_append (GTK_BOX (hbox), combo_vbox); |
| |
| combo = gtk_combo_box_new_with_model (GTK_TREE_MODEL (model)); |
| cell_renderer = gtk_cell_renderer_text_new (); |
| gtk_cell_layout_pack_start (GTK_CELL_LAYOUT (combo), cell_renderer, TRUE); |
| gtk_cell_layout_set_attributes (GTK_CELL_LAYOUT (combo), cell_renderer, |
| "text", 0, NULL); |
| gtk_widget_set_margin_start (combo, 12); |
| gtk_box_append (GTK_BOX (combo_vbox), combo); |
| |
| scrolled_window = gtk_scrolled_window_new (); |
| gtk_widget_set_hexpand (scrolled_window, TRUE); |
| gtk_box_append (GTK_BOX (hbox), scrolled_window); |
| gtk_scrolled_window_set_policy (GTK_SCROLLED_WINDOW (scrolled_window), |
| GTK_POLICY_AUTOMATIC, GTK_POLICY_AUTOMATIC); |
| |
| text_view = gtk_text_view_new (); |
| gtk_text_view_set_editable (GTK_TEXT_VIEW (text_view), FALSE); |
| gtk_text_view_set_cursor_visible (GTK_TEXT_VIEW (text_view), FALSE); |
| |
| gtk_scrolled_window_set_child (GTK_SCROLLED_WINDOW (scrolled_window), text_view); |
| |
| button_vbox = gtk_box_new (GTK_ORIENTATION_VERTICAL, 8); |
| gtk_box_append (GTK_BOX (hbox), button_vbox); |
| |
| gtk_window_set_default_size (GTK_WINDOW (window), 500, 300); |
| |
| button = gtk_button_new_with_label ("Insert"); |
| gtk_box_append (GTK_BOX (button_vbox), button); |
| g_signal_connect (button, "clicked", G_CALLBACK (on_insert), NULL); |
| |
| button = gtk_button_new_with_label ("Delete"); |
| gtk_box_append (GTK_BOX (button_vbox), button); |
| g_signal_connect (button, "clicked", G_CALLBACK (on_delete), NULL); |
| |
| button = gtk_button_new_with_label ("Reorder"); |
| gtk_box_append (GTK_BOX (button_vbox), button); |
| g_signal_connect (button, "clicked", G_CALLBACK (on_reorder), NULL); |
| |
| button = gtk_button_new_with_label ("Animate"); |
| gtk_box_append (GTK_BOX (button_vbox), button); |
| g_signal_connect (button, "clicked", G_CALLBACK (on_animate), NULL); |
| |
| GtkWidget *close_button = gtk_button_new_with_mnemonic ("_Close"); |
| gtk_widget_set_hexpand (close_button, TRUE); |
| gtk_box_append (GTK_BOX (content_area), close_button); |
| |
| gtk_window_present (GTK_WINDOW (window)); |
| |
| GMainLoop *loop = g_main_loop_new (NULL, FALSE); |
| |
| g_signal_connect_swapped (close_button, "clicked", |
| G_CALLBACK (gtk_window_destroy), |
| window); |
| g_signal_connect_swapped (window, "destroy", |
| G_CALLBACK (g_main_loop_quit), |
| loop); |
| |
| g_main_loop_run (loop); |
| |
| g_main_loop_unref (loop); |
| |
| return 0; |
| } |