#include #include #include #include static GtkWidget *main_window; static char *filename = NULL; static GtkPageSetup *page_setup = NULL; static GtkPrintSettings *settings = NULL; static gboolean file_changed = FALSE; static GtkTextBuffer *buffer; static GtkWidget *statusbar; static GList *active_prints = NULL; static void update_title (void) { char *basename; char *title; if (filename == NULL) basename = g_strdup ("Untitled"); else basename = g_path_get_basename (filename); title = g_strdup_printf ("Simple Editor with printing - %s", basename); g_free (basename); gtk_window_set_title (GTK_WINDOW (main_window), title); g_free (title); } static void update_statusbar (void) { gchar *msg; gint row, col; GtkTextIter iter; const char *print_str; gtk_statusbar_pop (GTK_STATUSBAR (statusbar), 0); gtk_text_buffer_get_iter_at_mark (buffer, &iter, gtk_text_buffer_get_insert (buffer)); row = gtk_text_iter_get_line (&iter); col = gtk_text_iter_get_line_offset (&iter); print_str = ""; if (active_prints) { GtkPrintOperation *op = active_prints->data; print_str = gtk_print_operation_get_status_string (op); } msg = g_strdup_printf ("%d, %d%s %s", row, col, file_changed?" - Modified":"", print_str); gtk_statusbar_push (GTK_STATUSBAR (statusbar), 0, msg); g_free (msg); } static void update_ui (void) { update_title (); update_statusbar (); } static char * get_text (void) { GtkTextIter start, end; gtk_text_buffer_get_start_iter (buffer, &start); gtk_text_buffer_get_end_iter (buffer, &end); return gtk_text_buffer_get_text (buffer, &start, &end, FALSE); } static void set_text (const char *text, gsize len) { gtk_text_buffer_set_text (buffer, text, len); file_changed = FALSE; update_ui (); } static void do_new (GtkAction *action) { g_free (filename); filename = NULL; set_text ("", 0); } static void load_file (const char *open_filename) { GtkWidget *error_dialog; char *contents; GError *error; gsize len; error_dialog = NULL; error = NULL; if (g_file_get_contents (open_filename, &contents, &len, &error)) { if (g_utf8_validate (contents, len, NULL)) { filename = g_strdup (open_filename); set_text (contents, len); g_free (contents); } else { error_dialog = gtk_message_dialog_new (GTK_WINDOW (main_window), GTK_DIALOG_DESTROY_WITH_PARENT, GTK_MESSAGE_ERROR, GTK_BUTTONS_CLOSE, "Error loading file %s:\n%s", open_filename, "Not valid utf8"); } } else { error_dialog = gtk_message_dialog_new (GTK_WINDOW (main_window), GTK_DIALOG_DESTROY_WITH_PARENT, GTK_MESSAGE_ERROR, GTK_BUTTONS_CLOSE, "Error loading file %s:\n%s", open_filename, error->message); g_error_free (error); } if (error_dialog) { g_signal_connect (error_dialog, "response", G_CALLBACK (gtk_widget_destroy), NULL); gtk_widget_show (error_dialog); } } static void do_open (GtkAction *action) { GtkWidget *dialog; gint response; char *open_filename; dialog = gtk_file_chooser_dialog_new ("Select file", GTK_WINDOW (main_window), GTK_FILE_CHOOSER_ACTION_OPEN, GTK_STOCK_CANCEL, GTK_RESPONSE_CANCEL, GTK_STOCK_OPEN, GTK_RESPONSE_OK, NULL); gtk_dialog_set_default_response (GTK_DIALOG (dialog), GTK_RESPONSE_OK); response = gtk_dialog_run (GTK_DIALOG (dialog)); if (response == GTK_RESPONSE_OK) { open_filename = gtk_file_chooser_get_filename (GTK_FILE_CHOOSER (dialog)); load_file (open_filename); g_free (open_filename); } gtk_widget_destroy (dialog); } static void save_file (const char *save_filename) { char *text = get_text (); GtkWidget *error_dialog; GError *error; error = NULL; if (g_file_set_contents (save_filename, text, -1, &error)) { if (save_filename != filename) { g_free (filename); filename = g_strdup (save_filename); } file_changed = FALSE; update_ui (); } else { error_dialog = gtk_message_dialog_new (GTK_WINDOW (main_window), GTK_DIALOG_DESTROY_WITH_PARENT, GTK_MESSAGE_ERROR, GTK_BUTTONS_CLOSE, "Error saving to file %s:\n%s", filename, error->message); g_signal_connect (error_dialog, "response", G_CALLBACK (gtk_widget_destroy), NULL); gtk_widget_show (error_dialog); g_error_free (error); } } static void do_save_as (GtkAction *action) { GtkWidget *dialog; gint response; char *save_filename; dialog = gtk_file_chooser_dialog_new ("Select file", GTK_WINDOW (main_window), GTK_FILE_CHOOSER_ACTION_SAVE, GTK_STOCK_CANCEL, GTK_RESPONSE_CANCEL, GTK_STOCK_SAVE, GTK_RESPONSE_OK, NULL); gtk_dialog_set_default_response (GTK_DIALOG (dialog), GTK_RESPONSE_OK); response = gtk_dialog_run (GTK_DIALOG (dialog)); if (response == GTK_RESPONSE_OK) { save_filename = gtk_file_chooser_get_filename (GTK_FILE_CHOOSER (dialog)); save_file (save_filename); g_free (save_filename); } gtk_widget_destroy (dialog); } static void do_save (GtkAction *action) { if (filename == NULL) do_save_as (action); else save_file (filename); } typedef struct { char *text; PangoLayout *layout; GList *page_breaks; GtkWidget *font_button; char *font; } PrintData; static void begin_print (GtkPrintOperation *operation, GtkPrintContext *context, PrintData *print_data) { PangoFontDescription *desc; PangoLayoutLine *layout_line; double width, height; double page_height; GList *page_breaks; int num_lines; int line; width = gtk_print_context_get_width (context); height = gtk_print_context_get_height (context); print_data->layout = gtk_print_context_create_pango_layout (context); desc = pango_font_description_from_string (print_data->font); pango_layout_set_font_description (print_data->layout, desc); pango_font_description_free (desc); pango_layout_set_width (print_data->layout, width * PANGO_SCALE); pango_layout_set_text (print_data->layout, print_data->text, -1); num_lines = pango_layout_get_line_count (print_data->layout); page_breaks = NULL; page_height = 0; for (line = 0; line < num_lines; line++) { PangoRectangle ink_rect, logical_rect; double line_height; layout_line = pango_layout_get_line (print_data->layout, line); pango_layout_line_get_extents (layout_line, &ink_rect, &logical_rect); line_height = logical_rect.height / 1024.0; if (page_height + line_height > height) { page_breaks = g_list_prepend (page_breaks, GINT_TO_POINTER (line)); page_height = 0; } page_height += line_height; } page_breaks = g_list_reverse (page_breaks); gtk_print_operation_set_n_pages (operation, g_list_length (page_breaks) + 1); print_data->page_breaks = page_breaks; } static void draw_page (GtkPrintOperation *operation, GtkPrintContext *context, int page_nr, PrintData *print_data) { cairo_t *cr; GList *pagebreak; int start, end, i; PangoLayoutIter *iter; double start_pos; if (page_nr == 0) start = 0; else { pagebreak = g_list_nth (print_data->page_breaks, page_nr - 1); start = GPOINTER_TO_INT (pagebreak->data); } pagebreak = g_list_nth (print_data->page_breaks, page_nr); if (pagebreak == NULL) end = pango_layout_get_line_count (print_data->layout); else end = GPOINTER_TO_INT (pagebreak->data); cr = gtk_print_context_get_cairo_context (context); cairo_set_source_rgb (cr, 0, 0, 0); i = 0; start_pos = 0; iter = pango_layout_get_iter (print_data->layout); do { PangoRectangle logical_rect; PangoLayoutLine *line; int baseline; if (i >= start) { line = pango_layout_iter_get_line (iter); pango_layout_iter_get_line_extents (iter, NULL, &logical_rect); baseline = pango_layout_iter_get_baseline (iter); if (i == start) start_pos = logical_rect.y / 1024.0; cairo_move_to (cr, logical_rect.x / 1024.0, baseline / 1024.0 - start_pos); pango_cairo_show_layout_line (cr, line); } i++; } while (i < end && pango_layout_iter_next_line (iter)); } static void do_page_setup (GtkAction *action) { GtkPageSetup *new_page_setup; if (settings == NULL) settings = gtk_print_settings_new (); new_page_setup = gtk_print_run_page_setup_dialog (GTK_WINDOW (main_window), page_setup, settings); if (page_setup) g_object_unref (page_setup); page_setup = new_page_setup; } static void status_changed_cb (GtkPrintOperation *op, gpointer user_data) { if (gtk_print_operation_is_finished (op)) { active_prints = g_list_remove (active_prints, op); g_object_unref (op); } update_statusbar (); } static GtkWidget * create_custom_widget (GtkPrintOperation *operation, PrintData *data) { GtkWidget *vbox, *hbox, *font, *label; gtk_print_operation_set_custom_tab_label (operation, "Other"); vbox = gtk_vbox_new (FALSE, 0); gtk_container_set_border_width (GTK_CONTAINER (vbox), 12); hbox = gtk_hbox_new (FALSE, 8); gtk_box_pack_start (GTK_BOX (vbox), hbox, FALSE, FALSE, 0); gtk_widget_show (hbox); label = gtk_label_new ("Font:"); gtk_box_pack_start (GTK_BOX (hbox), label, FALSE, FALSE, 0); gtk_widget_show (label); font = gtk_font_button_new_with_font (data->font); gtk_box_pack_start (GTK_BOX (hbox), font, FALSE, FALSE, 0); gtk_widget_show (font); data->font_button = font; return vbox; } static void custom_widget_apply (GtkPrintOperation *operation, GtkWidget *widget, PrintData *data) { const char *selected_font; selected_font = gtk_font_button_get_font_name (GTK_FONT_BUTTON (data->font_button)); g_free (data->font); data->font = g_strdup (selected_font); } typedef struct { GtkPrintOperation *op; GtkPrintOperationPreview *preview; GtkWidget *spin; GtkWidget *area; gint page; PrintData *data; gdouble dpi_x, dpi_y; } PreviewOp; static gboolean preview_expose (GtkWidget *widget, GdkEventExpose *event, gpointer data) { PreviewOp *pop = data; gdk_window_clear (pop->area->window); gtk_print_operation_preview_render_page (pop->preview, pop->page - 1); return TRUE; } static void preview_ready (GtkPrintOperationPreview *preview, GtkPrintContext *context, gpointer data) { PreviewOp *pop = data; gint n_pages; g_object_get (pop->op, "n-pages", &n_pages, NULL); gtk_spin_button_set_range (GTK_SPIN_BUTTON (pop->spin), 1.0, n_pages); g_signal_connect (pop->area, "expose_event", G_CALLBACK (preview_expose), pop); gtk_widget_queue_draw (pop->area); } static void preview_got_page_size (GtkPrintOperationPreview *preview, GtkPrintContext *context, GtkPageSetup *page_setup, gpointer data) { PreviewOp *pop = data; GtkPaperSize *paper_size; double w, h; cairo_t *cr; gdouble dpi_x, dpi_y; paper_size = gtk_page_setup_get_paper_size (page_setup); w = gtk_paper_size_get_width (paper_size, GTK_UNIT_INCH); h = gtk_paper_size_get_height (paper_size, GTK_UNIT_INCH); cr = gdk_cairo_create (pop->area->window); dpi_x = pop->area->allocation.width/w; dpi_y = pop->area->allocation.height/h; if (fabs (dpi_x - pop->dpi_x) > 0.001 || fabs (dpi_y - pop->dpi_y) > 0.001) { gtk_print_context_set_cairo_context (context, cr, dpi_x, dpi_y); pop->dpi_x = dpi_x; pop->dpi_y = dpi_y; } pango_cairo_update_layout (cr, pop->data->layout); cairo_destroy (cr); } static void update_page (GtkSpinButton *widget, gpointer data) { PreviewOp *pop = data; pop->page = gtk_spin_button_get_value_as_int (widget); gtk_widget_queue_draw (pop->area); } static void preview_destroy (GtkWindow *window, PreviewOp *pop) { gtk_print_operation_preview_end_preview (pop->preview); g_object_unref (pop->op); g_free (pop); } static gboolean do_preview (GtkPrintOperation *op, GtkPrintOperationPreview *preview, GtkPrintContext *context, GtkWindow *parent, gpointer data) { GtkPrintSettings *settings; GtkWidget *window, *close, *page, *hbox, *vbox, *da; gdouble width, height; cairo_t *cr; PreviewOp *pop; PrintData *print_data = data; pop = g_new0 (PreviewOp, 1); pop->data = print_data; settings = gtk_print_operation_get_print_settings (op); width = 200; height = 300; window = gtk_window_new (GTK_WINDOW_TOPLEVEL); gtk_window_set_transient_for (GTK_WINDOW (window), GTK_WINDOW (main_window)); vbox = gtk_vbox_new (FALSE, 0); gtk_container_add (GTK_CONTAINER (window), vbox); hbox = gtk_hbox_new (FALSE, 0); gtk_box_pack_start (GTK_BOX (vbox), hbox, FALSE, FALSE, 0); page = gtk_spin_button_new_with_range (1, 100, 1); gtk_box_pack_start (GTK_BOX (hbox), page, FALSE, FALSE, 0); close = gtk_button_new_from_stock (GTK_STOCK_CLOSE); gtk_box_pack_start (GTK_BOX (hbox), close, FALSE, FALSE, 0); da = gtk_drawing_area_new (); gtk_widget_set_size_request (GTK_WIDGET (da), width, height); gtk_box_pack_start (GTK_BOX (vbox), da, TRUE, TRUE, 0); gtk_widget_set_double_buffered (da, FALSE); gtk_widget_realize (da); cr = gdk_cairo_create (da->window); /* TODO: What dpi to use here? This will be used for pagination.. */ gtk_print_context_set_cairo_context (context, cr, 72, 72); cairo_destroy (cr); pop->op = op; pop->preview = preview; pop->spin = page; pop->area = da; pop->page = 1; g_signal_connect (page, "value-changed", G_CALLBACK (update_page), pop); g_signal_connect_swapped (close, "clicked", G_CALLBACK (gtk_widget_destroy), window); g_signal_connect (preview, "ready", G_CALLBACK (preview_ready), pop); g_signal_connect (preview, "got-page-size", G_CALLBACK (preview_got_page_size), pop); g_signal_connect (window, "destroy", G_CALLBACK (preview_destroy), pop); gtk_widget_show_all (window); return TRUE; } static void print_done (GtkPrintOperation *op, GtkPrintOperationResult res, PrintData *print_data) { GError *error = NULL; if (res == GTK_PRINT_OPERATION_RESULT_ERROR) { GtkWidget *error_dialog; gtk_print_operation_get_error (op, &error); error_dialog = gtk_message_dialog_new (GTK_WINDOW (main_window), GTK_DIALOG_DESTROY_WITH_PARENT, GTK_MESSAGE_ERROR, GTK_BUTTONS_CLOSE, "Error printing file:\n%s", error ? error->message : "no details"); g_signal_connect (error_dialog, "response", G_CALLBACK (gtk_widget_destroy), NULL); gtk_widget_show (error_dialog); } else if (res == GTK_PRINT_OPERATION_RESULT_APPLY) { if (settings != NULL) g_object_unref (settings); settings = g_object_ref (gtk_print_operation_get_print_settings (op)); } g_free (print_data->text); g_free (print_data->font); g_free (print_data); if (!gtk_print_operation_is_finished (op)) { g_object_ref (op); active_prints = g_list_append (active_prints, op); update_statusbar (); /* This ref is unref:ed when we get the final state change */ g_signal_connect (op, "status_changed", G_CALLBACK (status_changed_cb), NULL); } } static void do_print (GtkAction *action) { GtkPrintOperation *print; PrintData *print_data; print_data = g_new0 (PrintData, 1); print_data->text = get_text (); print_data->font = g_strdup ("Sans 12"); print = gtk_print_operation_new (); gtk_print_operation_set_track_print_status (print, TRUE); if (settings != NULL) gtk_print_operation_set_print_settings (print, settings); if (page_setup != NULL) gtk_print_operation_set_default_page_setup (print, page_setup); g_signal_connect (print, "begin_print", G_CALLBACK (begin_print), print_data); g_signal_connect (print, "draw_page", G_CALLBACK (draw_page), print_data); g_signal_connect (print, "create_custom_widget", G_CALLBACK (create_custom_widget), print_data); g_signal_connect (print, "custom_widget_apply", G_CALLBACK (custom_widget_apply), print_data); g_signal_connect (print, "preview", G_CALLBACK (do_preview), print_data); g_signal_connect (print, "done", G_CALLBACK (print_done), print_data); gtk_print_operation_set_export_filename (print, "test.pdf"); #if 0 gtk_print_operation_set_allow_async (print, TRUE); #endif gtk_print_operation_run (print, GTK_PRINT_OPERATION_ACTION_PRINT_DIALOG, GTK_WINDOW (main_window), NULL); g_object_unref (print); } static void do_about (GtkAction *action) { const gchar *authors[] = { "Alexander Larsson", NULL }; gtk_show_about_dialog (GTK_WINDOW (main_window), "name", "print test editor", "version", "0.1", "copyright", "(C) Red Hat, Inc", "comments", "Program to demonstrate GTK+ printing.", "authors", authors, NULL); } static void do_quit (GtkAction *action) { gtk_main_quit (); } static GtkActionEntry entries[] = { { "FileMenu", NULL, "_File" }, /* name, stock id, label */ { "HelpMenu", NULL, "_Help" }, /* name, stock id, label */ { "New", GTK_STOCK_NEW, /* name, stock id */ "_New", "N", /* label, accelerator */ "Create a new file", /* tooltip */ G_CALLBACK (do_new) }, { "Open", GTK_STOCK_OPEN, /* name, stock id */ "_Open","O", /* label, accelerator */ "Open a file", /* tooltip */ G_CALLBACK (do_open) }, { "Save", GTK_STOCK_SAVE, /* name, stock id */ "_Save","S", /* label, accelerator */ "Save current file", /* tooltip */ G_CALLBACK (do_save) }, { "SaveAs", GTK_STOCK_SAVE, /* name, stock id */ "Save _As...", NULL, /* label, accelerator */ "Save to a file", /* tooltip */ G_CALLBACK (do_save_as) }, { "Quit", GTK_STOCK_QUIT, /* name, stock id */ "_Quit", "Q", /* label, accelerator */ "Quit", /* tooltip */ G_CALLBACK (do_quit) }, { "About", NULL, /* name, stock id */ "_About", "A", /* label, accelerator */ "About", /* tooltip */ G_CALLBACK (do_about) }, { "PageSetup", NULL, /* name, stock id */ "Page _Setup", NULL, /* label, accelerator */ "Set up the page", /* tooltip */ G_CALLBACK (do_page_setup) }, { "Print", GTK_STOCK_PRINT, /* name, stock id */ NULL, NULL, /* label, accelerator */ "Print the document", /* tooltip */ G_CALLBACK (do_print) }, }; static guint n_entries = G_N_ELEMENTS (entries); static const gchar *ui_info = "" " " " " " " " " " " " " " " " " " " " " " " " " " " " " " " ""; static void buffer_changed_callback (GtkTextBuffer *buffer) { file_changed = TRUE; update_statusbar (); } static void mark_set_callback (GtkTextBuffer *buffer, const GtkTextIter *new_location, GtkTextMark *mark, gpointer data) { update_statusbar (); } static void update_resize_grip (GtkWidget *widget, GdkEventWindowState *event, GtkStatusbar *statusbar) { if (event->changed_mask & (GDK_WINDOW_STATE_MAXIMIZED | GDK_WINDOW_STATE_FULLSCREEN)) { gboolean maximized; maximized = event->new_window_state & (GDK_WINDOW_STATE_MAXIMIZED | GDK_WINDOW_STATE_FULLSCREEN); gtk_statusbar_set_has_resize_grip (statusbar, !maximized); } } static void create_window (void) { GtkWidget *bar; GtkWidget *table; GtkWidget *contents; GtkUIManager *ui; GtkWidget *sw; GtkActionGroup *actions; GError *error; main_window = gtk_window_new (GTK_WINDOW_TOPLEVEL); gtk_window_set_default_size (GTK_WINDOW (main_window), 400, 600); g_signal_connect (main_window, "delete-event", G_CALLBACK (gtk_main_quit), NULL); actions = gtk_action_group_new ("Actions"); gtk_action_group_add_actions (actions, entries, n_entries, NULL); ui = gtk_ui_manager_new (); gtk_ui_manager_insert_action_group (ui, actions, 0); gtk_window_add_accel_group (GTK_WINDOW (main_window), gtk_ui_manager_get_accel_group (ui)); gtk_container_set_border_width (GTK_CONTAINER (main_window), 0); error = NULL; if (!gtk_ui_manager_add_ui_from_string (ui, ui_info, -1, &error)) { g_message ("building menus failed: %s", error->message); g_error_free (error); } table = gtk_table_new (1, 3, FALSE); gtk_container_add (GTK_CONTAINER (main_window), table); bar = gtk_ui_manager_get_widget (ui, "/MenuBar"); gtk_widget_show (bar); gtk_table_attach (GTK_TABLE (table), bar, /* X direction */ /* Y direction */ 0, 1, 0, 1, GTK_EXPAND | GTK_FILL, 0, 0, 0); /* Create document */ sw = gtk_scrolled_window_new (NULL, NULL); gtk_scrolled_window_set_policy (GTK_SCROLLED_WINDOW (sw), GTK_POLICY_AUTOMATIC, GTK_POLICY_AUTOMATIC); gtk_scrolled_window_set_shadow_type (GTK_SCROLLED_WINDOW (sw), GTK_SHADOW_IN); gtk_table_attach (GTK_TABLE (table), sw, /* X direction */ /* Y direction */ 0, 1, 1, 2, GTK_EXPAND | GTK_FILL, GTK_EXPAND | GTK_FILL, 0, 0); contents = gtk_text_view_new (); gtk_widget_grab_focus (contents); gtk_container_add (GTK_CONTAINER (sw), contents); /* Create statusbar */ statusbar = gtk_statusbar_new (); gtk_table_attach (GTK_TABLE (table), statusbar, /* X direction */ /* Y direction */ 0, 1, 2, 3, GTK_EXPAND | GTK_FILL, 0, 0, 0); /* Show text widget info in the statusbar */ buffer = gtk_text_view_get_buffer (GTK_TEXT_VIEW (contents)); g_signal_connect_object (buffer, "changed", G_CALLBACK (buffer_changed_callback), NULL, 0); g_signal_connect_object (buffer, "mark_set", /* cursor moved */ G_CALLBACK (mark_set_callback), NULL, 0); g_signal_connect_object (main_window, "window_state_event", G_CALLBACK (update_resize_grip), statusbar, 0); update_ui (); gtk_widget_show_all (main_window); } int main (int argc, char **argv) { g_set_application_name ("Print editor"); gtk_init (&argc, &argv); create_window (); if (argc == 2) load_file (argv[1]); gtk_main (); return 0; }