#include "config.h" #include #include #include #include #include #include #include "demos.h" static GtkTextBuffer *info_buffer; static GtkTextBuffer *source_buffer; static gchar *current_file = NULL; enum { TITLE_COLUMN, FILENAME_COLUMN, FUNC_COLUMN, STYLE_COLUMN, NUM_COLUMNS }; typedef struct _CallbackData CallbackData; struct _CallbackData { GtkTreeModel *model; GtkTreePath *path; }; #ifdef G_OS_WIN32 #undef DEMOCODEDIR static char * get_democodedir (void) { static char *result = NULL; if (result == NULL) { result = g_win32_get_package_installation_directory_of_module (NULL); if (result == NULL) result = "unknown-location"; result = g_strconcat (result, "\\share\\gtk-2.0\\demo", NULL); } return result; } #define DEMOCODEDIR get_democodedir () #endif /** * demo_find_file: * @base: base filename * @err: location to store error, or %NULL. * * Looks for @base first in the current directory, then in the * location GTK+ where it will be installed on make install, * returns the first file found. * * Return value: the filename, if found or %NULL **/ gchar * demo_find_file (const char *base, GError **err) { g_return_val_if_fail (err == NULL || *err == NULL, NULL); if (g_file_test ("gtk-logo-rgb.gif", G_FILE_TEST_EXISTS) && g_file_test (base, G_FILE_TEST_EXISTS)) return g_strdup (base); else { char *filename = g_build_filename (DEMOCODEDIR, base, NULL); if (!g_file_test (filename, G_FILE_TEST_EXISTS)) { g_set_error (err, G_FILE_ERROR, G_FILE_ERROR_NOENT, "Cannot find demo data file \"%s\"", base); g_free (filename); return NULL; } return filename; } } static void window_closed_cb (GtkWidget *window, gpointer data) { CallbackData *cbdata = data; GtkTreeIter iter; PangoStyle style; gtk_tree_model_get_iter (cbdata->model, &iter, cbdata->path); gtk_tree_model_get (GTK_TREE_MODEL (cbdata->model), &iter, STYLE_COLUMN, &style, -1); if (style == PANGO_STYLE_ITALIC) gtk_tree_store_set (GTK_TREE_STORE (cbdata->model), &iter, STYLE_COLUMN, PANGO_STYLE_NORMAL, -1); gtk_tree_path_free (cbdata->path); g_free (cbdata); } gboolean read_line (FILE *stream, GString *str) { int n_read = 0; #ifdef HAVE_FLOCKFILE flockfile (stream); #endif g_string_truncate (str, 0); while (1) { int c; #ifdef HAVE_FLOCKFILE c = getc_unlocked (stream); #else c = getc (stream); #endif if (c == EOF) goto done; else n_read++; switch (c) { case '\r': case '\n': { #ifdef HAVE_FLOCKFILE int next_c = getc_unlocked (stream); #else int next_c = getc (stream); #endif if (!(next_c == EOF || (c == '\r' && next_c == '\n') || (c == '\n' && next_c == '\r'))) ungetc (next_c, stream); goto done; } default: g_string_append_c (str, c); } } done: #ifdef HAVE_FLOCKFILE funlockfile (stream); #endif return n_read > 0; } /* Stupid syntax highlighting. * * No regex was used in the making of this highlighting. * It should only work for simple cases. This is good, as * that's all we should have in the demos. */ /* This code should not be used elsewhere, except perhaps as an example of how * to iterate through a text buffer. */ enum { STATE_NORMAL, STATE_IN_COMMENT }; static gchar *tokens[] = { "/*", "\"", NULL }; static gchar *types[] = { "static", "const ", "void", "gint", " int ", " char ", "gchar ", "gfloat", "float", "double", "gint8", "gint16", "gint32", "guint", "guint8", "guint16", "guint32", "guchar", "glong", "gboolean" , "gshort", "gushort", "gulong", "gdouble", "gldouble", "gpointer", "NULL", "GList", "GSList", "FALSE", "TRUE", "FILE ", "GtkColorSelection ", "GtkWidget ", "GtkButton ", "GdkColor ", "GdkRectangle ", "GdkEventExpose ", "GdkGC ", "GdkPixbufLoader ", "GdkPixbuf ", "GError", "size_t", "GtkAboutDialog ", "GtkAction ", "GtkActionEntry ", "GtkRadioActionEntry ", "GtkIconFactory ", "GtkStockItem ", "GtkIconSet ", "GtkTextBuffer ", "GtkStatusbar ", "GtkTextIter ", "GtkTextMark ", "GdkEventWindowState ", "GtkActionGroup ", "GtkUIManager ", "GtkRadioAction ", "GtkActionClass ", "GtkToggleActionEntry ", "GtkAssistant ", "GtkBuilder ", "GtkSizeGroup ", "GtkTreeModel ", "GtkTreeSelection ", "GdkDisplay ", "GdkScreen ", "GdkWindow ", "GdkEventButton ", "GdkCursor ", "GtkTreeIter ", "GtkTreeViewColumn ", "GdkDisplayManager ", "GtkClipboard ", "GtkIconSize ", "GtkImage ", "GdkDragContext ", "GtkSelectionData ", "GtkDialog ", "GtkMenuItem ", "GtkListStore ", "GtkCellLayout ", "GtkCellRenderer ", "GtkTreePath ", "GtkTreeStore ", "GtkEntry ", "GtkEditable ", "GtkEditableInterface ", "GdkPixmap ", "GdkEventConfigure ", "GdkEventMotion ", "GdkModifierType ", "GtkEntryCompletion ", "GtkToolItem ", "GDir ", "GtkIconView ", "GtkCellRendererText ", "GtkContainer ", "GtkAccelGroup ", "GtkPaned ", "GtkPrintOperation ", "GtkPrintContext ", "cairo_t ", "PangoLayout " "PangoFontDescription ", "PangoRenderer ", "PangoMatrix ", "PangoContext ", "PangoLayout ", "GtkTable ", "GtkToggleButton ", "GString ", "GtkIconSize ", "GtkTreeView ", "GtkTextTag ", "GdkEvent ", "GdkEventKey ", "GtkTextView ", "GdkEventVisibility ", "GdkBitmap ", "GtkTextChildAnchor ", "GArray ", "GtkCellEditable ", "GtkCellRendererToggle ", NULL }; static gchar *control[] = { " if ", " while ", " else", " do ", " for ", "?", ":", "return ", "goto ", NULL }; void parse_chars (gchar *text, gchar **end_ptr, gint *state, gchar **tag, gboolean start) { gint i; gchar *next_token; /* Handle comments first */ if (*state == STATE_IN_COMMENT) { *end_ptr = strstr (text, "*/"); if (*end_ptr) { *end_ptr += 2; *state = STATE_NORMAL; *tag = "comment"; } return; } *tag = NULL; *end_ptr = NULL; /* check for comment */ if (!strncmp (text, "/*", 2)) { *end_ptr = strstr (text, "*/"); if (*end_ptr) *end_ptr += 2; else *state = STATE_IN_COMMENT; *tag = "comment"; return; } /* check for preprocessor defines */ if (*text == '#' && start) { *end_ptr = NULL; *tag = "preprocessor"; return; } /* functions */ if (start && * text != '\t' && *text != ' ' && *text != '{' && *text != '}') { if (strstr (text, "(")) { *end_ptr = strstr (text, "("); *tag = "function"; return; } } /* check for types */ for (i = 0; types[i] != NULL; i++) if (!strncmp (text, types[i], strlen (types[i])) || (start && types[i][0] == ' ' && !strncmp (text, types[i] + 1, strlen (types[i]) - 1))) { *end_ptr = text + strlen (types[i]); *tag = "type"; return; } /* check for control */ for (i = 0; control[i] != NULL; i++) if (!strncmp (text, control[i], strlen (control[i]))) { *end_ptr = text + strlen (control[i]); *tag = "control"; return; } /* check for string */ if (text[0] == '"') { gint maybe_escape = FALSE; *end_ptr = text + 1; *tag = "string"; while (**end_ptr != '\000') { if (**end_ptr == '\"' && !maybe_escape) { *end_ptr += 1; return; } if (**end_ptr == '\\') maybe_escape = TRUE; else maybe_escape = FALSE; *end_ptr += 1; } return; } /* not at the start of a tag. Find the next one. */ for (i = 0; tokens[i] != NULL; i++) { next_token = strstr (text, tokens[i]); if (next_token) { if (*end_ptr) *end_ptr = (*end_ptrmessage); g_error_free (err); return; } file = g_fopen (full_filename, "r"); if (!file) g_warning ("Cannot open %s: %s\n", full_filename, g_strerror (errno)); g_free (full_filename); if (!file) return; gtk_text_buffer_get_iter_at_offset (info_buffer, &start, 0); while (read_line (file, buffer)) { gchar *p = buffer->str; gchar *q; gchar *r; switch (state) { case 0: /* Reading title */ while (*p == '/' || *p == '*' || g_ascii_isspace (*p)) p++; r = p; while (*r != '/' && strlen (r)) r++; if (strlen (r) > 0) p = r + 1; q = p + strlen (p); while (q > p && g_ascii_isspace (*(q - 1))) q--; if (q > p) { int len_chars = g_utf8_pointer_to_offset (p, q); end = start; g_assert (strlen (p) >= q - p); gtk_text_buffer_insert (info_buffer, &end, p, q - p); start = end; gtk_text_iter_backward_chars (&start, len_chars); gtk_text_buffer_apply_tag_by_name (info_buffer, "title", &start, &end); start = end; state++; } break; case 1: /* Reading body of info section */ while (g_ascii_isspace (*p)) p++; if (*p == '*' && *(p + 1) == '/') { gtk_text_buffer_get_iter_at_offset (source_buffer, &start, 0); state++; } else { int len; while (*p == '*' || g_ascii_isspace (*p)) p++; len = strlen (p); while (g_ascii_isspace (*(p + len - 1))) len--; if (len > 0) { if (in_para) gtk_text_buffer_insert (info_buffer, &start, " ", 1); g_assert (strlen (p) >= len); gtk_text_buffer_insert (info_buffer, &start, p, len); in_para = 1; } else { gtk_text_buffer_insert (info_buffer, &start, "\n", 1); in_para = 0; } } break; case 2: /* Skipping blank lines */ while (g_ascii_isspace (*p)) p++; if (*p) { p = buffer->str; state++; /* Fall through */ } else break; case 3: /* Reading program body */ gtk_text_buffer_insert (source_buffer, &start, p, -1); gtk_text_buffer_insert (source_buffer, &start, "\n", 1); break; } } fclose (file); fontify (); g_string_free (buffer, TRUE); } void row_activated_cb (GtkTreeView *tree_view, GtkTreePath *path, GtkTreeViewColumn *column) { GtkTreeIter iter; PangoStyle style; GDoDemoFunc func; GtkWidget *window; GtkTreeModel *model; model = gtk_tree_view_get_model (tree_view); gtk_tree_model_get_iter (model, &iter, path); gtk_tree_model_get (GTK_TREE_MODEL (model), &iter, FUNC_COLUMN, &func, STYLE_COLUMN, &style, -1); if (func) { gtk_tree_store_set (GTK_TREE_STORE (model), &iter, STYLE_COLUMN, (style == PANGO_STYLE_ITALIC ? PANGO_STYLE_NORMAL : PANGO_STYLE_ITALIC), -1); window = (func) (gtk_widget_get_toplevel (GTK_WIDGET (tree_view))); if (window != NULL) { CallbackData *cbdata; cbdata = g_new (CallbackData, 1); cbdata->model = model; cbdata->path = gtk_tree_path_copy (path); g_signal_connect (window, "destroy", G_CALLBACK (window_closed_cb), cbdata); } } } static void selection_cb (GtkTreeSelection *selection, GtkTreeModel *model) { GtkTreeIter iter; GValue value = {0, }; if (! gtk_tree_selection_get_selected (selection, NULL, &iter)) return; gtk_tree_model_get_value (model, &iter, FILENAME_COLUMN, &value); if (g_value_get_string (&value)) load_file (g_value_get_string (&value)); g_value_unset (&value); } static GtkWidget * create_text (GtkTextBuffer **buffer, gboolean is_source) { GtkWidget *scrolled_window; GtkWidget *text_view; PangoFontDescription *font_desc; scrolled_window = gtk_scrolled_window_new (NULL, NULL); gtk_scrolled_window_set_policy (GTK_SCROLLED_WINDOW (scrolled_window), GTK_POLICY_AUTOMATIC, GTK_POLICY_AUTOMATIC); gtk_scrolled_window_set_shadow_type (GTK_SCROLLED_WINDOW (scrolled_window), GTK_SHADOW_IN); text_view = gtk_text_view_new (); *buffer = gtk_text_buffer_new (NULL); gtk_text_view_set_buffer (GTK_TEXT_VIEW (text_view), *buffer); gtk_text_view_set_editable (GTK_TEXT_VIEW (text_view), FALSE); gtk_text_view_set_cursor_visible (GTK_TEXT_VIEW (text_view), FALSE); gtk_container_add (GTK_CONTAINER (scrolled_window), text_view); if (is_source) { font_desc = pango_font_description_from_string ("monospace"); gtk_widget_modify_font (text_view, font_desc); pango_font_description_free (font_desc); gtk_text_view_set_wrap_mode (GTK_TEXT_VIEW (text_view), GTK_WRAP_NONE); } else { /* Make it a bit nicer for text. */ gtk_text_view_set_wrap_mode (GTK_TEXT_VIEW (text_view), GTK_WRAP_WORD); gtk_text_view_set_pixels_above_lines (GTK_TEXT_VIEW (text_view), 2); gtk_text_view_set_pixels_below_lines (GTK_TEXT_VIEW (text_view), 2); } return scrolled_window; } static GtkWidget * create_tree (void) { GtkTreeSelection *selection; GtkCellRenderer *cell; GtkWidget *tree_view; GtkTreeViewColumn *column; GtkTreeStore *model; GtkTreeIter iter; GtkWidget *box, *label, *scrolled_window; Demo *d = testgtk_demos; model = gtk_tree_store_new (NUM_COLUMNS, G_TYPE_STRING, G_TYPE_STRING, G_TYPE_POINTER, G_TYPE_INT); tree_view = gtk_tree_view_new (); gtk_tree_view_set_model (GTK_TREE_VIEW (tree_view), GTK_TREE_MODEL (model)); selection = gtk_tree_view_get_selection (GTK_TREE_VIEW (tree_view)); gtk_tree_selection_set_mode (GTK_TREE_SELECTION (selection), GTK_SELECTION_BROWSE); gtk_widget_set_size_request (tree_view, 200, -1); /* this code only supports 1 level of children. If we * want more we probably have to use a recursing function. */ while (d->title) { Demo *children = d->children; gtk_tree_store_append (GTK_TREE_STORE (model), &iter, NULL); gtk_tree_store_set (GTK_TREE_STORE (model), &iter, TITLE_COLUMN, d->title, FILENAME_COLUMN, d->filename, FUNC_COLUMN, d->func, STYLE_COLUMN, PANGO_STYLE_NORMAL, -1); d++; if (!children) continue; while (children->title) { GtkTreeIter child_iter; gtk_tree_store_append (GTK_TREE_STORE (model), &child_iter, &iter); gtk_tree_store_set (GTK_TREE_STORE (model), &child_iter, TITLE_COLUMN, children->title, FILENAME_COLUMN, children->filename, FUNC_COLUMN, children->func, STYLE_COLUMN, PANGO_STYLE_NORMAL, -1); children++; } } cell = gtk_cell_renderer_text_new (); column = gtk_tree_view_column_new_with_attributes ("Widget (double click for demo)", cell, "text", TITLE_COLUMN, "style", STYLE_COLUMN, NULL); gtk_tree_view_append_column (GTK_TREE_VIEW (tree_view), GTK_TREE_VIEW_COLUMN (column)); gtk_tree_model_get_iter_first (GTK_TREE_MODEL (model), &iter); gtk_tree_selection_select_iter (GTK_TREE_SELECTION (selection), &iter); g_signal_connect (selection, "changed", G_CALLBACK (selection_cb), model); g_signal_connect (tree_view, "row_activated", G_CALLBACK (row_activated_cb), model); gtk_tree_view_collapse_all (GTK_TREE_VIEW (tree_view)); gtk_tree_view_set_headers_visible (GTK_TREE_VIEW (tree_view), FALSE); scrolled_window = gtk_scrolled_window_new (NULL, NULL); gtk_scrolled_window_set_policy (GTK_SCROLLED_WINDOW (scrolled_window), GTK_POLICY_NEVER, GTK_POLICY_AUTOMATIC); gtk_container_add (GTK_CONTAINER (scrolled_window), tree_view); label = gtk_label_new ("Widget (double click for demo)"); box = gtk_notebook_new (); gtk_notebook_append_page (GTK_NOTEBOOK (box), scrolled_window, label); gtk_widget_grab_focus (tree_view); g_object_unref (model); return box; } static void setup_default_icon (void) { GdkPixbuf *pixbuf; char *filename; GError *err; err = NULL; pixbuf = NULL; filename = demo_find_file ("gtk-logo-rgb.gif", &err); if (filename) { pixbuf = gdk_pixbuf_new_from_file (filename, &err); g_free (filename); } /* Ignoring this error (passing NULL instead of &err above) * would probably be reasonable for most apps. We're just * showing off. */ if (err) { GtkWidget *dialog; dialog = gtk_message_dialog_new (NULL, 0, GTK_MESSAGE_ERROR, GTK_BUTTONS_CLOSE, "Failed to read icon file: %s", err->message); g_error_free (err); g_signal_connect (dialog, "response", G_CALLBACK (gtk_widget_destroy), NULL); } if (pixbuf) { GList *list; GdkPixbuf *transparent; /* The gtk-logo-rgb icon has a white background, make it transparent */ transparent = gdk_pixbuf_add_alpha (pixbuf, TRUE, 0xff, 0xff, 0xff); list = NULL; list = g_list_append (list, transparent); gtk_window_set_default_icon_list (list); g_list_free (list); g_object_unref (pixbuf); g_object_unref (transparent); } } int main (int argc, char **argv) { GtkWidget *window; GtkWidget *notebook; GtkWidget *hbox; GtkWidget *tree; GtkTextTag *tag; /* Most code in gtk-demo is intended to be exemplary, but not * these few lines, which are just a hack so gtk-demo will work * in the GTK tree without installing it. */ if (g_file_test ("../../gdk-pixbuf/libpixbufloader-pnm.la", G_FILE_TEST_EXISTS)) { g_setenv ("GDK_PIXBUF_MODULE_FILE", "../../gdk-pixbuf/loaders.cache", TRUE); g_setenv ("GTK_IM_MODULE_FILE", "../../modules/input/immodules.cache", TRUE); } /* -- End of hack -- */ gtk_init (&argc, &argv); setup_default_icon (); window = gtk_window_new (GTK_WINDOW_TOPLEVEL); gtk_window_set_title (GTK_WINDOW (window), "GTK+ Code Demos"); g_signal_connect_after (window, "destroy", G_CALLBACK (gtk_main_quit), NULL); hbox = gtk_hbox_new (FALSE, 0); gtk_container_add (GTK_CONTAINER (window), hbox); tree = create_tree (); gtk_box_pack_start (GTK_BOX (hbox), tree, FALSE, FALSE, 0); notebook = gtk_notebook_new (); gtk_box_pack_start (GTK_BOX (hbox), notebook, TRUE, TRUE, 0); gtk_notebook_append_page (GTK_NOTEBOOK (notebook), create_text (&info_buffer, FALSE), gtk_label_new_with_mnemonic ("_Info")); tag = gtk_text_buffer_create_tag (info_buffer, "title", "font", "Sans 18", NULL); g_object_unref (info_buffer); gtk_notebook_append_page (GTK_NOTEBOOK (notebook), create_text (&source_buffer, TRUE), gtk_label_new_with_mnemonic ("_Source")); tag = gtk_text_buffer_create_tag (source_buffer, "comment", "foreground", "DodgerBlue", NULL); tag = gtk_text_buffer_create_tag (source_buffer, "type", "foreground", "ForestGreen", NULL); tag = gtk_text_buffer_create_tag (source_buffer, "string", "foreground", "RosyBrown", "weight", PANGO_WEIGHT_BOLD, NULL); tag = gtk_text_buffer_create_tag (source_buffer, "control", "foreground", "purple", NULL); tag = gtk_text_buffer_create_tag (source_buffer, "preprocessor", "style", PANGO_STYLE_OBLIQUE, "foreground", "burlywood4", NULL); tag = gtk_text_buffer_create_tag (source_buffer, "function", "weight", PANGO_WEIGHT_BOLD, "foreground", "DarkGoldenrod4", NULL); g_object_unref (source_buffer); gtk_window_set_default_size (GTK_WINDOW (window), 600, 400); gtk_widget_show_all (window); load_file (testgtk_demos[0].filename); gtk_main (); return 0; }