X-Git-Url: https://git.donarmstrong.com/?a=blobdiff_plain;f=src%2Fxo-misc.c;h=0d4abc2e01fbec3358b5641ff319d040eb020b29;hb=e648cece76b671b5b0e165acc6d14178e8de1000;hp=eb30aac92da2e4dfe4fbc37c487be07223feb265;hpb=e93f1837951e3c7a09dd224273de3983c8ecc2e6;p=xournal.git diff --git a/src/xo-misc.c b/src/xo-misc.c index eb30aac..0d4abc2 100644 --- a/src/xo-misc.c +++ b/src/xo-misc.c @@ -3,14 +3,19 @@ #endif #include +#include #include #include +#include #include "xournal.h" #include "xo-interface.h" #include "xo-support.h" #include "xo-callbacks.h" #include "xo-misc.h" +#include "xo-file.h" +#include "xo-paint.h" +#include "xo-shapes.h" // some global constants @@ -26,8 +31,8 @@ guint predef_bgcolors_rgba[COLOR_MAX] = // meaningless ones set to white double predef_thickness[NUM_STROKE_TOOLS][THICKNESS_MAX] = { { 0.42, 0.85, 1.41, 2.26, 5.67 }, // pen thicknesses = 0.15, 0.3, 0.5, 0.8, 2 mm - { 2.83, 2.83, 7.08, 14.17, 14.17 }, // eraser thicknesses = 1, 2.5, 5 mm - { 2.83, 2.83, 7.08, 14.17, 14.17 }, // highlighter thicknesses = 1, 2.5, 5 mm + { 2.83, 2.83, 8.50, 19.84, 19.84 }, // eraser thicknesses = 1, 3, 7 mm + { 2.83, 2.83, 8.50, 19.84, 19.84 }, // highlighter thicknesses = 1, 3, 7 mm }; // some manipulation functions @@ -91,6 +96,13 @@ void realloc_cur_path(int n) ui.cur_path.coords = g_realloc(ui.cur_path.coords, 2*(n+10)*sizeof(double)); } +void realloc_cur_widths(int n) +{ + if (n <= ui.cur_widths_storage_alloc) return; + ui.cur_widths_storage_alloc = n+10; + ui.cur_widths = g_realloc(ui.cur_widths, (n+10)*sizeof(double)); +} + // undo utility functions void prepare_new_undo(void) @@ -120,15 +132,22 @@ void clear_redo_stack(void) while (redo!=NULL) { if (redo->type == ITEM_STROKE) { gnome_canvas_points_free(redo->item->path); + if (redo->item->brush.variable_width) g_free(redo->item->widths); g_free(redo->item); /* the strokes are unmapped, so there are no associated canvas items */ } - else if (redo->type == ITEM_ERASURE) { + else if (redo->type == ITEM_TEXT) { + g_free(redo->item->text); + g_free(redo->item->font_name); + g_free(redo->item); + } + else if (redo->type == ITEM_ERASURE || redo->type == ITEM_RECOGNIZER) { for (list = redo->erasurelist; list!=NULL; list=list->next) { erasure = (struct UndoErasureData *)list->data; for (repl = erasure->replacement_items; repl!=NULL; repl=repl->next) { it = (struct Item *)repl->data; gnome_canvas_points_free(it->path); + if (it->brush.variable_width) g_free(it->widths); g_free(it); } g_list_free(erasure->replacement_items); @@ -148,13 +167,19 @@ void clear_redo_stack(void) redo->page->group = NULL; delete_page(redo->page); } - else if (redo->type == ITEM_MOVESEL) { + else if (redo->type == ITEM_MOVESEL || redo->type == ITEM_REPAINTSEL) { + g_list_free(redo->itemlist); g_list_free(redo->auxlist); + } + else if (redo->type == ITEM_RESIZESEL) { g_list_free(redo->itemlist); } else if (redo->type == ITEM_PASTE) { for (list = redo->itemlist; list!=NULL; list=list->next) { it = (struct Item *)list->data; - if (it->type == ITEM_STROKE) gnome_canvas_points_free(it->path); + if (it->type == ITEM_STROKE) { + gnome_canvas_points_free(it->path); + if (it->brush.variable_width) g_free(it->widths); + } g_free(it); } g_list_free(redo->itemlist); @@ -162,6 +187,10 @@ void clear_redo_stack(void) else if (redo->type == ITEM_NEW_LAYER) { g_free(redo->layer); } + else if (redo->type == ITEM_TEXT_EDIT || redo->type == ITEM_TEXT_ATTRIB) { + g_free(redo->str); + if (redo->type == ITEM_TEXT_ATTRIB) g_free(redo->brush); + } u = redo; redo = redo->next; @@ -179,10 +208,15 @@ void clear_undo_stack(void) while (undo!=NULL) { // for strokes, items are already in the journal, so we don't free them // for erasures, we need to free the dead items - if (undo->type == ITEM_ERASURE) { + if (undo->type == ITEM_ERASURE || undo->type == ITEM_RECOGNIZER) { for (list = undo->erasurelist; list!=NULL; list=list->next) { erasure = (struct UndoErasureData *)list->data; - gnome_canvas_points_free(erasure->item->path); + if (erasure->item->type == ITEM_STROKE) { + gnome_canvas_points_free(erasure->item->path); + if (erasure->item->brush.variable_width) g_free(erasure->item->widths); + } + if (erasure->item->type == ITEM_TEXT) + { g_free(erasure->item->text); g_free(erasure->item->font_name); } g_free(erasure->item); g_list_free(erasure->replacement_items); g_free(erasure); @@ -197,7 +231,13 @@ void clear_undo_stack(void) } g_free(undo->bg); } - else if (undo->type == ITEM_MOVESEL || undo->type == ITEM_PASTE) { + else if (undo->type == ITEM_MOVESEL || undo->type == ITEM_REPAINTSEL) { + g_list_free(undo->itemlist); g_list_free(undo->auxlist); + } + else if (undo->type == ITEM_RESIZESEL) { + g_list_free(undo->itemlist); + } + else if (undo->type == ITEM_PASTE) { g_list_free(undo->itemlist); } else if (undo->type == ITEM_DELETE_LAYER) { @@ -208,6 +248,10 @@ void clear_undo_stack(void) undo->page->group = NULL; delete_page(undo->page); } + else if (undo->type == ITEM_TEXT_EDIT || undo->type == ITEM_TEXT_ATTRIB) { + g_free(undo->str); + if (undo->type == ITEM_TEXT_ATTRIB) g_free(undo->brush); + } u = undo; undo = undo->next; @@ -254,6 +298,9 @@ void delete_layer(struct Layer *l) item = (struct Item *)l->items->data; if (item->type == ITEM_STROKE && item->path != NULL) gnome_canvas_points_free(item->path); + if (item->type == ITEM_TEXT) { + g_free(item->font_name); g_free(item->text); + } // don't need to delete the canvas_item, as it's part of the group destroyed below g_free(item); l->items = g_list_delete_link(l->items, l->items); @@ -306,8 +353,8 @@ void fix_xinput_coords(GdkEvent *event) { double *axes, *px, *py, axis_width; GdkDevice *device; - int wx, wy, sx, sy; - + int wx, wy, sx, sy, ix, iy; + if (event->type == GDK_BUTTON_PRESS || event->type == GDK_BUTTON_RELEASE) { axes = event->button.axes; px = &(event->button.x); @@ -321,23 +368,45 @@ void fix_xinput_coords(GdkEvent *event) device = event->motion.device; } else return; // nothing we know how to do - - gdk_window_get_origin(event->any.window, &wx, &wy); + + // use canvas window, not event window (else get GTK+ 2.11 bugs!) + gdk_window_get_origin(GTK_WIDGET(canvas)->window, &wx, &wy); gnome_canvas_get_scroll_offsets(canvas, &sx, &sy); + +#ifdef ENABLE_XINPUT_BUGFIX + // fix broken events with the core pointer's location + if (!finite(axes[0]) || !finite(axes[1]) || (axes[0]==0. && axes[1]==0.)) { + gdk_window_get_pointer(GTK_WIDGET(canvas)->window, &ix, &iy, NULL); + *px = ix + sx; + *py = iy + sy; + } + else { + axis_width = device->axes[0].max - device->axes[0].min; + if (axis_width>EPSILON) + *px = (axes[0]/axis_width)*ui.screen_width + sx - wx; + axis_width = device->axes[1].max - device->axes[1].min; + if (axis_width>EPSILON) + *py = (axes[1]/axis_width)*ui.screen_height + sy - wy; + } +#endif +} + +double get_pressure_multiplier(GdkEvent *event) +{ + double rawpressure; - axis_width = device->axes[0].max - device->axes[0].min; - if (axis_width>EPSILON) - *px = (axes[0]/axis_width)*ui.screen_width + sx - wx; + if (event->button.device == gdk_device_get_core_pointer() + || event->button.device->num_axes <= 2) return 1.0; - axis_width = device->axes[1].max - device->axes[1].min; - if (axis_width>EPSILON) - *py = (axes[1]/axis_width)*ui.screen_height + sy - wy; + rawpressure = event->button.axes[2]/(event->button.device->axes[2].max - event->button.device->axes[2].min); + + return ((1-rawpressure)*ui.width_minimum_multiplier + rawpressure*ui.width_maximum_multiplier); } void update_item_bbox(struct Item *item) { int i; - double *p; + gdouble *p, h, w; if (item->type == ITEM_STROKE) { item->bbox.left = item->bbox.right = item->path->coords[0]; @@ -350,6 +419,12 @@ void update_item_bbox(struct Item *item) if (p[1] > item->bbox.bottom) item->bbox.bottom = p[1]; } } + if (item->type == ITEM_TEXT && item->canvas_item!=NULL) { + h=0.; w=0.; + g_object_get(item->canvas_item, "text_width", &w, "text_height", &h, NULL); + item->bbox.right = item->bbox.left + w; + item->bbox.bottom = item->bbox.top + h; + } } void make_page_clipbox(struct Page *pg) @@ -366,6 +441,47 @@ void make_page_clipbox(struct Page *pg) gnome_canvas_path_def_unref(pg_clip); } +void make_canvas_item_one(GnomeCanvasGroup *group, struct Item *item) +{ + PangoFontDescription *font_desc; + GnomeCanvasPoints points; + int j; + + if (item->type == ITEM_STROKE) { + if (!item->brush.variable_width) + item->canvas_item = gnome_canvas_item_new(group, + gnome_canvas_line_get_type(), "points", item->path, + "cap-style", GDK_CAP_ROUND, "join-style", GDK_JOIN_ROUND, + "fill-color-rgba", item->brush.color_rgba, + "width-units", item->brush.thickness, NULL); + else { + item->canvas_item = gnome_canvas_item_new(group, + gnome_canvas_group_get_type(), NULL); + points.num_points = 2; + points.ref_count = 1; + for (j = 0; j < item->path->num_points-1; j++) { + points.coords = item->path->coords+2*j; + gnome_canvas_item_new((GnomeCanvasGroup *) item->canvas_item, + gnome_canvas_line_get_type(), "points", &points, + "cap-style", GDK_CAP_ROUND, "join-style", GDK_JOIN_ROUND, + "fill-color-rgba", item->brush.color_rgba, + "width-units", item->widths[j], NULL); + } + } + } + if (item->type == ITEM_TEXT) { + font_desc = pango_font_description_from_string(item->font_name); + pango_font_description_set_absolute_size(font_desc, + item->font_size*ui.zoom*PANGO_SCALE); + item->canvas_item = gnome_canvas_item_new(group, + gnome_canvas_text_get_type(), + "x", item->bbox.left, "y", item->bbox.top, "anchor", GTK_ANCHOR_NW, + "font-desc", font_desc, "fill-color-rgba", item->brush.color_rgba, + "text", item->text, NULL); + update_item_bbox(item); + } +} + void make_canvas_items(void) { struct Page *pg; @@ -388,13 +504,8 @@ void make_canvas_items(void) pg->group, gnome_canvas_group_get_type(), NULL); for (itemlist = l->items; itemlist!=NULL; itemlist = itemlist->next) { item = (struct Item *)itemlist->data; - if (item->type == ITEM_STROKE && item->canvas_item == NULL) { - item->canvas_item = gnome_canvas_item_new(l->group, - gnome_canvas_line_get_type(), "points", item->path, - "cap-style", GDK_CAP_ROUND, "join-style", GDK_JOIN_ROUND, - "fill-color-rgba", item->brush.color_rgba, - "width-units", item->brush.thickness, NULL); - } + if (item->canvas_item == NULL) + make_canvas_item_one(l->group, item); } } } @@ -408,6 +519,7 @@ void update_canvas_bg(struct Page *pg) double *pt; double x, y; int w, h; + gboolean is_well_scaled; if (pg->bg->canvas_item != NULL) gtk_object_destroy(GTK_OBJECT(pg->bg->canvas_item)); @@ -492,15 +604,23 @@ void update_canvas_bg(struct Page *pg) if (pg->bg->type == BG_PDF) { if (pg->bg->pixbuf == NULL) return; - pg->bg->canvas_item = gnome_canvas_item_new(pg->group, - gnome_canvas_pixbuf_get_type(), - "pixbuf", pg->bg->pixbuf, - "width", pg->width, "height", pg->height, - "width-set", TRUE, "height-set", TRUE, - NULL); + is_well_scaled = (fabs(pg->bg->pixel_width - pg->width*ui.zoom) < 2. + && fabs(pg->bg->pixel_height - pg->height*ui.zoom) < 2.); + if (is_well_scaled) + pg->bg->canvas_item = gnome_canvas_item_new(pg->group, + gnome_canvas_pixbuf_get_type(), + "pixbuf", pg->bg->pixbuf, + "width-in-pixels", TRUE, "height-in-pixels", TRUE, + NULL); + else + pg->bg->canvas_item = gnome_canvas_item_new(pg->group, + gnome_canvas_pixbuf_get_type(), + "pixbuf", pg->bg->pixbuf, + "width", pg->width, "height", pg->height, + "width-set", TRUE, "height-set", TRUE, + NULL); lower_canvas_item_to(pg->group, pg->bg->canvas_item, NULL); } - } gboolean is_visible(struct Page *pg) @@ -520,13 +640,15 @@ void rescale_bg_pixmaps(void) GList *pglist; struct Page *pg; GdkPixbuf *pix; + gboolean is_well_scaled; + gdouble zoom_to_request; for (pglist = journal.pages; pglist!=NULL; pglist = pglist->next) { pg = (struct Page *)pglist->data; // in progressive mode we scale only visible pages if (ui.progressive_bg && !is_visible(pg)) continue; - if (pg->bg->type == BG_PIXMAP) { // do the rescaling ourselves + if (pg->bg->type == BG_PIXMAP && pg->bg->canvas_item!=NULL) { // do the rescaling ourselves if (ui.antialias_bg) { if (pg->bg->pixbuf_scale == ui.zoom) continue; set_cursor_busy(TRUE); @@ -539,16 +661,30 @@ void rescale_bg_pixmaps(void) set_cursor_busy(FALSE); } else { - pix = GDK_PIXBUF(g_object_get_data(G_OBJECT(pg->bg->canvas_item), "pixbuf")); + g_object_get(G_OBJECT(pg->bg->canvas_item), "pixbuf", &pix, NULL); if (pix!=pg->bg->pixbuf) gnome_canvas_item_set(pg->bg->canvas_item, "pixbuf", pg->bg->pixbuf, NULL); pg->bg->pixbuf_scale = 0; } } - if (pg->bg->type == BG_PDF) { // request an asynchronous update - if (pg->bg->pixbuf_scale == ui.zoom) continue; - add_bgpdf_request(pg->bg->file_page_seq, ui.zoom, FALSE); - pg->bg->pixbuf_scale = ui.zoom; + if (pg->bg->type == BG_PDF) { + // make pixmap scale to correct size if current one is wrong + is_well_scaled = (fabs(pg->bg->pixel_width - pg->width*ui.zoom) < 2. + && fabs(pg->bg->pixel_height - pg->height*ui.zoom) < 2.); + if (pg->bg->canvas_item != NULL && !is_well_scaled) { + g_object_get(pg->bg->canvas_item, "width-in-pixels", &is_well_scaled, NULL); + if (is_well_scaled) + gnome_canvas_item_set(pg->bg->canvas_item, + "width", pg->width, "height", pg->height, + "width-in-pixels", FALSE, "height-in-pixels", FALSE, + "width-set", TRUE, "height-set", TRUE, + NULL); + } + // request an asynchronous update to a better pixmap if needed + zoom_to_request = MIN(ui.zoom, MAX_SAFE_RENDER_DPI/72.0); + if (pg->bg->pixbuf_scale == zoom_to_request) continue; + add_bgpdf_request(pg->bg->file_page_seq, zoom_to_request); + pg->bg->pixbuf_scale = zoom_to_request; } } } @@ -580,11 +716,19 @@ void lower_canvas_item_to(GnomeCanvasGroup *g, GnomeCanvasItem *item, GnomeCanva g->item_list_end = g_list_last(g->item_list); } +void rgb_to_gdkcolor(guint rgba, GdkColor *color) +{ + color->pixel = 0; + color->red = ((rgba>>24)&0xff)*0x101; + color->green = ((rgba>>16)&0xff)*0x101; + color->blue = ((rgba>>8)&0xff)*0x101; +} + // some interface functions void update_thickness_buttons(void) { - if (ui.toolno >= NUM_STROKE_TOOLS) { + if (ui.selection!=NULL || ui.toolno[ui.cur_mapping] >= NUM_STROKE_TOOLS) { gtk_toggle_tool_button_set_active( GTK_TOGGLE_TOOL_BUTTON(GET_COMPONENT("buttonThicknessOther")), TRUE); } else @@ -609,7 +753,8 @@ void update_thickness_buttons(void) void update_color_buttons(void) { - if (ui.toolno != TOOL_PEN && ui.toolno != TOOL_HIGHLIGHTER) { + if (ui.selection!=NULL || (ui.toolno[ui.cur_mapping] != TOOL_PEN + && ui.toolno[ui.cur_mapping] != TOOL_HIGHLIGHTER && ui.toolno[ui.cur_mapping] != TOOL_TEXT)) { gtk_toggle_tool_button_set_active( GTK_TOGGLE_TOOL_BUTTON(GET_COMPONENT("buttonColorOther")), TRUE); } else @@ -666,7 +811,7 @@ void update_color_buttons(void) void update_tool_buttons(void) { - switch(ui.toolno) { + switch(ui.toolno[ui.cur_mapping]) { case TOOL_PEN: gtk_toggle_tool_button_set_active( GTK_TOGGLE_TOOL_BUTTON(GET_COMPONENT("buttonPen")), TRUE); @@ -695,17 +840,26 @@ void update_tool_buttons(void) gtk_toggle_tool_button_set_active( GTK_TOGGLE_TOOL_BUTTON(GET_COMPONENT("buttonVerticalSpace")), TRUE); break; + case TOOL_HAND: + gtk_toggle_tool_button_set_active( + GTK_TOGGLE_TOOL_BUTTON(GET_COMPONENT("buttonHand")), TRUE); + break; } gtk_toggle_tool_button_set_active( - GTK_TOGGLE_TOOL_BUTTON(GET_COMPONENT("buttonRuler")), ui.ruler); + GTK_TOGGLE_TOOL_BUTTON(GET_COMPONENT("buttonRuler")), + ui.toolno[ui.cur_mapping]ruler); + gtk_toggle_tool_button_set_active( + GTK_TOGGLE_TOOL_BUTTON(GET_COMPONENT("buttonReco")), + ui.toolno[ui.cur_mapping]recognizer); + update_thickness_buttons(); update_color_buttons(); } void update_tool_menu(void) { - switch(ui.toolno) { + switch(ui.toolno[0]) { case TOOL_PEN: gtk_check_menu_item_set_active( GTK_CHECK_MENU_ITEM(GET_COMPONENT("toolsPen")), TRUE); @@ -734,23 +888,40 @@ void update_tool_menu(void) gtk_check_menu_item_set_active( GTK_CHECK_MENU_ITEM(GET_COMPONENT("toolsVerticalSpace")), TRUE); break; + case TOOL_HAND: + gtk_check_menu_item_set_active( + GTK_CHECK_MENU_ITEM(GET_COMPONENT("toolsHand")), TRUE); + break; } gtk_check_menu_item_set_active( - GTK_CHECK_MENU_ITEM(GET_COMPONENT("toolsRuler")), ui.ruler); + GTK_CHECK_MENU_ITEM(GET_COMPONENT("toolsRuler")), + ui.toolno[0]ruler); + gtk_toggle_tool_button_set_active( + GTK_TOGGLE_TOOL_BUTTON(GET_COMPONENT("buttonReco")), + ui.toolno[ui.cur_mapping]recognizer); gtk_check_menu_item_set_active( - GTK_CHECK_MENU_ITEM(GET_COMPONENT("toolsRuler")), ui.ruler); + GTK_CHECK_MENU_ITEM(GET_COMPONENT("toolsRuler")), + ui.toolno[0]voffset*ui.zoom); - gnome_canvas_set_pixels_per_unit(canvas, ui.zoom); + ui.cur_page->voffset*ui.zoom); */ + gnome_canvas_get_scroll_offsets(canvas, &cx, &cy); + cy = ui.cur_page->voffset*ui.zoom; + gnome_canvas_scroll_to(canvas, cx, cy); + + if (refresh_all) + gnome_canvas_set_pixels_per_unit(canvas, ui.zoom); + else if (!ui.view_continuous) + gnome_canvas_item_move(GNOME_CANVAS_ITEM(ui.cur_page->group), 0., 0.); } } @@ -915,8 +1201,6 @@ void update_page_stuff(void) GtkSpinButton *spin; struct Page *pg; double vertpos, maxwidth; - struct Layer *layer; - int relscroll; // move the page groups to their rightful locations or hide them if (ui.view_continuous) { @@ -957,12 +1241,12 @@ void update_page_stuff(void) gtk_spin_button_set_range(spin, 1, journal.npages+1); /* npages+1 will be used to create a new page at end */ gtk_spin_button_set_value(spin, ui.pageno+1); - g_snprintf(tmp, 10, " of %d", journal.npages); + g_snprintf(tmp, 10, _(" of %d"), journal.npages); gtk_label_set_text(GTK_LABEL(GET_COMPONENT("labelNumpages")), tmp); layerbox = GTK_COMBO_BOX(GET_COMPONENT("comboLayer")); if (ui.layerbox_length == 0) { - gtk_combo_box_prepend_text(layerbox, "Background"); + gtk_combo_box_prepend_text(layerbox, _("Background")); ui.layerbox_length++; } while (ui.layerbox_length > ui.cur_page->nlayers+1) { @@ -970,7 +1254,7 @@ void update_page_stuff(void) ui.layerbox_length--; } while (ui.layerbox_length < ui.cur_page->nlayers+1) { - g_snprintf(tmp, 10, "Layer %d", ui.layerbox_length++); + g_snprintf(tmp, 10, _("Layer %d"), ui.layerbox_length++); gtk_combo_box_prepend_text(layerbox, tmp); } gtk_combo_box_set_active(layerbox, ui.cur_page->nlayers-1-ui.layerno); @@ -985,7 +1269,7 @@ void update_page_stuff(void) gtk_check_menu_item_set_active( GTK_CHECK_MENU_ITEM(GET_COMPONENT("viewOnePage")), TRUE); - if (ui.cur_page->bg->type == BG_SOLID) { + if (ui.cur_page->bg->type == BG_SOLID && !ui.bg_apply_all_pages) { switch (ui.cur_page->bg->color_no) { case COLOR_WHITE: gtk_check_menu_item_set_active( @@ -1044,7 +1328,7 @@ void update_page_stuff(void) // enable/disable the page/layer menu items and toolbar buttons gtk_widget_set_sensitive(GET_COMPONENT("journalPaperColor"), - ui.cur_page->bg->type == BG_SOLID); + ui.cur_page->bg->type == BG_SOLID || ui.bg_apply_all_pages); gtk_widget_set_sensitive(GET_COMPONENT("journalSetAsDefault"), ui.cur_page->bg->type == BG_SOLID); @@ -1072,6 +1356,7 @@ void update_toolbar_and_menu(void) update_pen_props_menu(); update_eraser_props_menu(); update_highlighter_props_menu(); + update_mappings_menu(); gtk_toggle_tool_button_set_active( GTK_TOGGLE_TOOL_BUTTON(GET_COMPONENT("buttonFullscreen")), ui.fullscreen); @@ -1085,14 +1370,15 @@ void update_file_name(char *filename) if (ui.filename != NULL) g_free(ui.filename); ui.filename = filename; if (filename == NULL) { - gtk_window_set_title(GTK_WINDOW (winMain), "Xournal"); + gtk_window_set_title(GTK_WINDOW (winMain), _("Xournal")); return; } p = g_utf8_strrchr(filename, -1, '/'); if (p == NULL) p = filename; else p = g_utf8_next_char(p); - g_snprintf(tmp, 100, "Xournal - %s", p); + g_snprintf(tmp, 100, _("Xournal - %s"), p); gtk_window_set_title(GTK_WINDOW (winMain), tmp); + new_mru_entry(filename); } void update_undo_redo_enabled(void) @@ -1107,18 +1393,57 @@ void update_copy_paste_enabled(void) { gtk_widget_set_sensitive(GET_COMPONENT("editCut"), ui.selection!=NULL); gtk_widget_set_sensitive(GET_COMPONENT("editCopy"), ui.selection!=NULL); + gtk_widget_set_sensitive(GET_COMPONENT("editPaste"), ui.cur_item_type!=ITEM_TEXT); gtk_widget_set_sensitive(GET_COMPONENT("editDelete"), ui.selection!=NULL); gtk_widget_set_sensitive(GET_COMPONENT("buttonCut"), ui.selection!=NULL); gtk_widget_set_sensitive(GET_COMPONENT("buttonCopy"), ui.selection!=NULL); + gtk_widget_set_sensitive(GET_COMPONENT("buttonPaste"), ui.cur_item_type!=ITEM_TEXT); +} + +void update_mapping_linkings(int toolno) +{ + int i; + + for (i = 1; i<=NUM_BUTTONS; i++) { + if (ui.linked_brush[i] == BRUSH_LINKED) { + if (toolno >= 0 && toolno < NUM_STROKE_TOOLS) + g_memmove(&(ui.brushes[i][toolno]), &(ui.brushes[0][toolno]), sizeof(struct Brush)); + } + if (ui.linked_brush[i] == BRUSH_COPIED && toolno == ui.toolno[i]) { + ui.linked_brush[i] = BRUSH_STATIC; + if (i==1 || i==2) update_mappings_menu_linkings(); + } + } } void set_cur_color(int color) { ui.cur_brush->color_no = color; - if (ui.toolno == TOOL_HIGHLIGHTER) - ui.cur_brush->color_rgba = predef_colors_rgba[color] & HILITER_ALPHA_MASK; + if (ui.toolno[0] == TOOL_HIGHLIGHTER) + ui.cur_brush->color_rgba = predef_colors_rgba[color] & ui.hiliter_alpha_mask; else ui.cur_brush->color_rgba = predef_colors_rgba[color]; + update_mapping_linkings(ui.toolno[0]); +} + +void recolor_temp_text(int color_no, guint color_rgba) +{ + GdkColor gdkcolor; + + if (ui.cur_item_type!=ITEM_TEXT) return; + if (ui.cur_item->text!=NULL && ui.cur_item->brush.color_rgba != color_rgba) { + prepare_new_undo(); + undo->type = ITEM_TEXT_ATTRIB; + undo->item = ui.cur_item; + undo->str = g_strdup(ui.cur_item->font_name); + undo->val_x = ui.cur_item->font_size; + undo->brush = (struct Brush *)g_memdup(&(ui.cur_item->brush), sizeof(struct Brush)); + } + ui.cur_item->brush.color_no = color_no; + ui.cur_item->brush.color_rgba = color_rgba; + rgb_to_gdkcolor(color_rgba, &gdkcolor); + gtk_widget_modify_text(ui.cur_item->widget, GTK_STATE_NORMAL, &gdkcolor); + gtk_widget_grab_focus(ui.cur_item->widget); } void process_color_activate(GtkMenuItem *menuitem, int color) @@ -1131,19 +1456,29 @@ void process_color_activate(GtkMenuItem *menuitem, int color) return; } - if (ui.toolno != TOOL_PEN && ui.toolno != TOOL_HIGHLIGHTER - && ui.toolno != TOOL_TEXT && ui.selection == NULL) { - ui.toolno = TOOL_PEN; - ui.cur_brush = ui.brushes+TOOL_PEN; + if (ui.cur_mapping != 0) return; // not user-generated + reset_focus(); + + if (ui.cur_item_type == ITEM_TEXT) + recolor_temp_text(color, predef_colors_rgba[color]); + + if (ui.selection != NULL) { + recolor_selection(color); + update_color_buttons(); + update_color_menu(); + } + + if (ui.toolno[0] != TOOL_PEN && ui.toolno[0] != TOOL_HIGHLIGHTER + && ui.toolno[0] != TOOL_TEXT) { + if (ui.selection != NULL) return; + end_text(); + ui.toolno[0] = TOOL_PEN; + ui.cur_brush = &(ui.brushes[0][TOOL_PEN]); update_tool_buttons(); update_tool_menu(); } - if (ui.toolno == TOOL_PEN || ui.toolno == TOOL_HIGHLIGHTER) - set_cur_color(color); - - // later add selection tools - + set_cur_color(color); update_color_buttons(); update_color_menu(); update_cursor(); @@ -1159,14 +1494,25 @@ void process_thickness_activate(GtkMenuItem *menuitem, int tool, int val) return; } + if (ui.cur_mapping != 0) return; // not user-generated + + if (ui.selection != NULL && GTK_OBJECT_TYPE(menuitem) != GTK_TYPE_RADIO_MENU_ITEM) { + reset_focus(); + rethicken_selection(val); + update_thickness_buttons(); + } + if (tool >= NUM_STROKE_TOOLS) { update_thickness_buttons(); // undo illegal button selection return; } - - if (ui.brushes[tool].thickness_no == val) return; // avoid loops - ui.brushes[tool].thickness_no = val; - ui.brushes[tool].thickness = predef_thickness[tool][val]; + + if (ui.brushes[0][tool].thickness_no == val) return; + reset_focus(); + end_text(); + ui.brushes[0][tool].thickness_no = val; + ui.brushes[0][tool].thickness = predef_thickness[tool][val]; + update_mapping_linkings(tool); update_thickness_buttons(); if (tool == TOOL_PEN) update_pen_props_menu(); @@ -1177,67 +1523,94 @@ void process_thickness_activate(GtkMenuItem *menuitem, int tool, int val) void process_papercolor_activate(GtkMenuItem *menuitem, int color) { + struct Page *pg; + GList *pglist; + gboolean hasdone; + if (!gtk_check_menu_item_get_active(GTK_CHECK_MENU_ITEM (menuitem))) return; - if (ui.cur_page->bg->type != BG_SOLID) { + if ((ui.cur_page->bg->type != BG_SOLID) || ui.bg_apply_all_pages) gtk_check_menu_item_set_active( GTK_CHECK_MENU_ITEM(GET_COMPONENT("papercolorNA")), TRUE); - return; - } - - if (ui.cur_page->bg->color_no == color) return; - - reset_selection(); - prepare_new_undo(); - undo->type = ITEM_NEW_BG_ONE; - undo->page = ui.cur_page; - undo->bg = (struct Background *)g_memdup(ui.cur_page->bg, sizeof(struct Background)); - undo->bg->canvas_item = NULL; - ui.cur_page->bg->color_no = color; - ui.cur_page->bg->color_rgba = predef_bgcolors_rgba[color]; - update_canvas_bg(ui.cur_page); + pg = ui.cur_page; + hasdone = FALSE; + for (pglist = journal.pages; pglist!=NULL; pglist = pglist->next) { + if (ui.bg_apply_all_pages) pg = (struct Page *)pglist->data; + if (pg->bg->type == BG_SOLID && pg->bg->color_no != color) { + prepare_new_undo(); + if (hasdone) undo->multiop |= MULTIOP_CONT_UNDO; + undo->multiop |= MULTIOP_CONT_REDO; + hasdone = TRUE; + undo->type = ITEM_NEW_BG_ONE; + undo->page = pg; + undo->bg = (struct Background *)g_memdup(pg->bg, sizeof(struct Background)); + undo->bg->canvas_item = NULL; + + pg->bg->color_no = color; + pg->bg->color_rgba = predef_bgcolors_rgba[color]; + update_canvas_bg(pg); + } + if (!ui.bg_apply_all_pages) break; + } + if (hasdone) undo->multiop -= MULTIOP_CONT_REDO; } void process_paperstyle_activate(GtkMenuItem *menuitem, int style) { + struct Page *pg; + GList *pglist; + gboolean hasdone, must_upd; + if (!gtk_check_menu_item_get_active(GTK_CHECK_MENU_ITEM (menuitem))) return; - if (ui.cur_page->bg->type == BG_SOLID && ui.cur_page->bg->ruling == style) - return; + if (ui.bg_apply_all_pages) + gtk_check_menu_item_set_active( + GTK_CHECK_MENU_ITEM(GET_COMPONENT("paperstyleNA")), TRUE); - reset_selection(); - prepare_new_undo(); - undo->type = ITEM_NEW_BG_ONE; - undo->page = ui.cur_page; - undo->bg = (struct Background *)g_memdup(ui.cur_page->bg, sizeof(struct Background)); - undo->bg->canvas_item = NULL; - - if (ui.cur_page->bg->type != BG_SOLID) { - ui.cur_page->bg->type = BG_SOLID; - ui.cur_page->bg->color_no = COLOR_WHITE; - ui.cur_page->bg->color_rgba = predef_bgcolors_rgba[COLOR_WHITE]; - ui.cur_page->bg->filename = NULL; - ui.cur_page->bg->pixbuf = NULL; - update_page_stuff(); + pg = ui.cur_page; + hasdone = FALSE; + must_upd = FALSE; + for (pglist = journal.pages; pglist!=NULL; pglist = pglist->next) { + if (ui.bg_apply_all_pages) pg = (struct Page *)pglist->data; + if (pg->bg->type != BG_SOLID || pg->bg->ruling != style) { + prepare_new_undo(); + undo->type = ITEM_NEW_BG_ONE; + if (hasdone) undo->multiop |= MULTIOP_CONT_UNDO; + undo->multiop |= MULTIOP_CONT_REDO; + hasdone = TRUE; + undo->page = pg; + undo->bg = (struct Background *)g_memdup(pg->bg, sizeof(struct Background)); + undo->bg->canvas_item = NULL; + + if (pg->bg->type != BG_SOLID) { + pg->bg->type = BG_SOLID; + pg->bg->color_no = COLOR_WHITE; + pg->bg->color_rgba = predef_bgcolors_rgba[COLOR_WHITE]; + pg->bg->filename = NULL; + pg->bg->pixbuf = NULL; + must_upd = TRUE; + } + pg->bg->ruling = style; + update_canvas_bg(pg); + } + if (!ui.bg_apply_all_pages) break; } - - ui.cur_page->bg->ruling = style; - update_canvas_bg(ui.cur_page); + if (hasdone) undo->multiop -= MULTIOP_CONT_REDO; + if (must_upd) update_page_stuff(); } gboolean ok_to_close(void) { GtkWidget *dialog; GtkResponseType response; - GList *pagelist; if (ui.saved) return TRUE; dialog = gtk_message_dialog_new(GTK_WINDOW (winMain), GTK_DIALOG_DESTROY_WITH_PARENT, - GTK_MESSAGE_WARNING, GTK_BUTTONS_YES_NO, "Save changes to '%s'?", - (ui.filename!=NULL) ? ui.filename:"Untitled"); + GTK_MESSAGE_WARNING, GTK_BUTTONS_YES_NO, _("Save changes to '%s'?"), + (ui.filename!=NULL) ? ui.filename:_("Untitled")); gtk_dialog_add_button(GTK_DIALOG (dialog), GTK_STOCK_CANCEL, GTK_RESPONSE_CANCEL); response = gtk_dialog_run(GTK_DIALOG (dialog)); gtk_widget_destroy(dialog); @@ -1250,10 +1623,14 @@ gboolean ok_to_close(void) return TRUE; } -// test if we're still busy loading a PDF background file -gboolean page_ops_forbidden(void) +// send the focus back to the appropriate widget +void reset_focus(void) { - return (bgpdf.status != STATUS_NOT_INIT && bgpdf.create_pages); + if (ui.cur_item_type == ITEM_TEXT) + gtk_widget_grab_focus(ui.cur_item->widget); + else + gtk_widget_grab_focus(GTK_WIDGET(canvas)); + reset_recognizer(); } // selection / clipboard stuff @@ -1267,24 +1644,334 @@ void reset_selection(void) g_free(ui.selection); ui.selection = NULL; update_copy_paste_enabled(); + update_color_menu(); + update_thickness_buttons(); + update_color_buttons(); + update_font_button(); + update_cursor(); } -void move_journal_items_by(GList *itemlist, double dx, double dy) +void move_journal_items_by(GList *itemlist, double dx, double dy, + struct Layer *l1, struct Layer *l2, GList *depths) { struct Item *item; + GnomeCanvasItem *refitem; + GList *link; int i; double *pt; while (itemlist!=NULL) { item = (struct Item *)itemlist->data; - if (item->type == ITEM_STROKE) { + if (item->type == ITEM_STROKE) for (pt=item->path->coords, i=0; ipath->num_points; i++, pt+=2) { pt[0] += dx; pt[1] += dy; } + if (item->type == ITEM_STROKE || item->type == ITEM_TEXT || item->type == ITEM_TEMP_TEXT) { item->bbox.left += dx; item->bbox.right += dx; item->bbox.top += dy; item->bbox.bottom += dy; } + if (l1 != l2) { + // find out where to insert + if (depths != NULL) { + if (depths->data == NULL) link = l2->items; + else { + link = g_list_find(l2->items, depths->data); + if (link != NULL) link = link->next; + } + } else link = NULL; + l2->items = g_list_insert_before(l2->items, link, item); + l2->nitems++; + l1->items = g_list_remove(l1->items, item); + l1->nitems--; + } + if (depths != NULL) { // also raise/lower the canvas items + if (item->canvas_item!=NULL) { + if (depths->data == NULL) link = NULL; + else link = g_list_find(l2->items, depths->data); + if (link != NULL) refitem = ((struct Item *)(link->data))->canvas_item; + else refitem = NULL; + lower_canvas_item_to(l2->group, item->canvas_item, refitem); + } + depths = depths->next; + } itemlist = itemlist->next; } } + +void resize_journal_items_by(GList *itemlist, double scaling_x, double scaling_y, + double offset_x, double offset_y) +{ + struct Item *item; + GList *list; + double mean_scaling, temp; + double *pt, *wid; + GnomeCanvasGroup *group; + int i; + + /* geometric mean of x and y scalings = rescaling for stroke widths + and for text font sizes */ + mean_scaling = sqrt(fabs(scaling_x * scaling_y)); + + for (list = itemlist; list != NULL; list = list->next) { + item = (struct Item *)list->data; + if (item->type == ITEM_STROKE) { + item->brush.thickness = item->brush.thickness * mean_scaling; + for (i=0, pt=item->path->coords; ipath->num_points; i++, pt+=2) { + pt[0] = pt[0]*scaling_x + offset_x; + pt[1] = pt[1]*scaling_y + offset_y; + } + if (item->brush.variable_width) + for (i=0, wid=item->widths; ipath->num_points-1; i++, wid++) + *wid = *wid * mean_scaling; + + item->bbox.left = item->bbox.left*scaling_x + offset_x; + item->bbox.right = item->bbox.right*scaling_x + offset_x; + item->bbox.top = item->bbox.top*scaling_y + offset_y; + item->bbox.bottom = item->bbox.bottom*scaling_y + offset_y; + if (item->bbox.left > item->bbox.right) { + temp = item->bbox.left; + item->bbox.left = item->bbox.right; + item->bbox.right = temp; + } + if (item->bbox.top > item->bbox.bottom) { + temp = item->bbox.top; + item->bbox.top = item->bbox.bottom; + item->bbox.bottom = temp; + } + } + if (item->type == ITEM_TEXT) { + /* must scale about NW corner -- all other points of the text box + are font- and zoom-dependent, so scaling about center of text box + couldn't be undone properly. FIXME? */ + item->font_size *= mean_scaling; + item->bbox.left = item->bbox.left*scaling_x + offset_x; + item->bbox.top = item->bbox.top*scaling_y + offset_y; + } + // redraw the item + if (item->canvas_item!=NULL) { + group = (GnomeCanvasGroup *) item->canvas_item->parent; + gtk_object_destroy(GTK_OBJECT(item->canvas_item)); + make_canvas_item_one(group, item); + } + } +} + +// Switch between button mappings + +/* NOTE ABOUT BUTTON MAPPINGS: ui.cur_mapping is 0 except while a canvas + click event is being processed ... */ + +void switch_mapping(int m) +{ + if (ui.cur_mapping == m) return; + + ui.cur_mapping = m; + if (ui.toolno[m] < NUM_STROKE_TOOLS) + ui.cur_brush = &(ui.brushes[m][ui.toolno[m]]); + if (ui.toolno[m] == TOOL_TEXT) + ui.cur_brush = &(ui.brushes[m][TOOL_PEN]); + update_tool_buttons(); + update_color_menu(); + update_cursor(); +} + +void process_mapping_activate(GtkMenuItem *menuitem, int m, int tool) +{ + if (!gtk_check_menu_item_get_active(GTK_CHECK_MENU_ITEM(menuitem))) return; + if (ui.cur_mapping!=0) return; + if (ui.toolno[m] == tool) return; + end_text(); + reset_focus(); + + ui.toolno[m] = tool; + if (ui.linked_brush[m] == BRUSH_COPIED) { + ui.linked_brush[m] = BRUSH_STATIC; + update_mappings_menu_linkings(); + } +} + +// update the ordering of components in the main vbox + +const char *vbox_component_names[VBOX_MAIN_NITEMS]= + {"scrolledwindowMain", "menubar", "toolbarMain", "toolbarPen", "hbox1"}; + +void update_vbox_order(int *order) +{ + int i, j; + GtkWidget *child; + GtkBox *vboxMain = GTK_BOX(GET_COMPONENT("vboxMain")); + gboolean present[VBOX_MAIN_NITEMS]; + + for (i=0; i=VBOX_MAIN_NITEMS) continue; + present[order[i]] = TRUE; + child = GET_COMPONENT(vbox_component_names[order[i]]); + gtk_box_reorder_child(vboxMain, child, j++); + gtk_widget_show(child); + } + for (i=1; ifont_name, ui.cur_item->font_size); + else if (ui.selection!=NULL && ui.selection->items!=NULL && + ui.selection->items->next==NULL && + (it=(struct Item*)ui.selection->items->data)->type == ITEM_TEXT) + str = g_strdup_printf("%s %.1f", it->font_name, it->font_size); + else + str = g_strdup_printf("%s %.1f", ui.font_name, ui.font_size); + return str; +} + +void update_font_button(void) +{ + gchar *str; + + str = make_cur_font_name(); + gtk_font_button_set_font_name(GTK_FONT_BUTTON(GET_COMPONENT("fontButton")), str); + g_free(str); +} + +gboolean can_accel(GtkWidget *widget, guint id, gpointer data) +{ + return GTK_WIDGET_SENSITIVE(widget); +} + +void allow_all_accels(void) +{ + g_signal_connect((gpointer) GET_COMPONENT("fileNew"), + "can-activate-accel", G_CALLBACK(can_accel), NULL); + g_signal_connect((gpointer) GET_COMPONENT("fileOpen"), + "can-activate-accel", G_CALLBACK(can_accel), NULL); + g_signal_connect((gpointer) GET_COMPONENT("fileSave"), + "can-activate-accel", G_CALLBACK(can_accel), NULL); + g_signal_connect((gpointer) GET_COMPONENT("filePrint"), + "can-activate-accel", G_CALLBACK(can_accel), NULL); + g_signal_connect((gpointer) GET_COMPONENT("filePrintPDF"), + "can-activate-accel", G_CALLBACK(can_accel), NULL); + g_signal_connect((gpointer) GET_COMPONENT("fileQuit"), + "can-activate-accel", G_CALLBACK(can_accel), NULL); + g_signal_connect((gpointer) GET_COMPONENT("editUndo"), + "can-activate-accel", G_CALLBACK(can_accel), NULL); + g_signal_connect((gpointer) GET_COMPONENT("editRedo"), + "can-activate-accel", G_CALLBACK(can_accel), NULL); + g_signal_connect((gpointer) GET_COMPONENT("editCut"), + "can-activate-accel", G_CALLBACK(can_accel), NULL); + g_signal_connect((gpointer) GET_COMPONENT("editCopy"), + "can-activate-accel", G_CALLBACK(can_accel), NULL); + g_signal_connect((gpointer) GET_COMPONENT("editPaste"), + "can-activate-accel", G_CALLBACK(can_accel), NULL); + g_signal_connect((gpointer) GET_COMPONENT("editDelete"), + "can-activate-accel", G_CALLBACK(can_accel), NULL); + g_signal_connect((gpointer) GET_COMPONENT("viewFullscreen"), + "can-activate-accel", G_CALLBACK(can_accel), NULL); + g_signal_connect((gpointer) GET_COMPONENT("viewZoomIn"), + "can-activate-accel", G_CALLBACK(can_accel), NULL); + g_signal_connect((gpointer) GET_COMPONENT("viewZoomOut"), + "can-activate-accel", G_CALLBACK(can_accel), NULL); + g_signal_connect((gpointer) GET_COMPONENT("viewNormalSize"), + "can-activate-accel", G_CALLBACK(can_accel), NULL); + g_signal_connect((gpointer) GET_COMPONENT("viewPageWidth"), + "can-activate-accel", G_CALLBACK(can_accel), NULL); + g_signal_connect((gpointer) GET_COMPONENT("viewFirstPage"), + "can-activate-accel", G_CALLBACK(can_accel), NULL); + g_signal_connect((gpointer) GET_COMPONENT("viewPreviousPage"), + "can-activate-accel", G_CALLBACK(can_accel), NULL); + g_signal_connect((gpointer) GET_COMPONENT("viewNextPage"), + "can-activate-accel", G_CALLBACK(can_accel), NULL); + g_signal_connect((gpointer) GET_COMPONENT("viewLastPage"), + "can-activate-accel", G_CALLBACK(can_accel), NULL); + g_signal_connect((gpointer) GET_COMPONENT("toolsPen"), + "can-activate-accel", G_CALLBACK(can_accel), NULL); + g_signal_connect((gpointer) GET_COMPONENT("toolsEraser"), + "can-activate-accel", G_CALLBACK(can_accel), NULL); + g_signal_connect((gpointer) GET_COMPONENT("toolsHighlighter"), + "can-activate-accel", G_CALLBACK(can_accel), NULL); + g_signal_connect((gpointer) GET_COMPONENT("toolsText"), + "can-activate-accel", G_CALLBACK(can_accel), NULL); +/* g_signal_connect((gpointer) GET_COMPONENT("toolsSelectRegion"), + "can-activate-accel", G_CALLBACK(can_accel), NULL); */ + g_signal_connect((gpointer) GET_COMPONENT("toolsSelectRectangle"), + "can-activate-accel", G_CALLBACK(can_accel), NULL); + g_signal_connect((gpointer) GET_COMPONENT("toolsVerticalSpace"), + "can-activate-accel", G_CALLBACK(can_accel), NULL); + g_signal_connect((gpointer) GET_COMPONENT("toolsHand"), + "can-activate-accel", G_CALLBACK(can_accel), NULL); + g_signal_connect((gpointer) GET_COMPONENT("toolsTextFont"), + "can-activate-accel", G_CALLBACK(can_accel), NULL); + g_signal_connect((gpointer) GET_COMPONENT("toolsRuler"), + "can-activate-accel", G_CALLBACK(can_accel), NULL); + g_signal_connect((gpointer) GET_COMPONENT("toolsReco"), + "can-activate-accel", G_CALLBACK(can_accel), NULL); +} + +void add_scroll_bindings(void) +{ + GtkBindingSet *binding_set; + + binding_set = gtk_binding_set_by_class( + G_OBJECT_GET_CLASS(GET_COMPONENT("scrolledwindowMain"))); + gtk_binding_entry_add_signal(binding_set, GDK_Up, 0, + "scroll_child", 2, GTK_TYPE_SCROLL_TYPE, GTK_SCROLL_STEP_BACKWARD, + G_TYPE_BOOLEAN, FALSE); + gtk_binding_entry_add_signal(binding_set, GDK_KP_Up, 0, + "scroll_child", 2, GTK_TYPE_SCROLL_TYPE, GTK_SCROLL_STEP_BACKWARD, + G_TYPE_BOOLEAN, FALSE); + gtk_binding_entry_add_signal(binding_set, GDK_Down, 0, + "scroll_child", 2, GTK_TYPE_SCROLL_TYPE, GTK_SCROLL_STEP_FORWARD, + G_TYPE_BOOLEAN, FALSE); + gtk_binding_entry_add_signal(binding_set, GDK_KP_Down, 0, + "scroll_child", 2, GTK_TYPE_SCROLL_TYPE, GTK_SCROLL_STEP_FORWARD, + G_TYPE_BOOLEAN, FALSE); + gtk_binding_entry_add_signal(binding_set, GDK_Left, 0, + "scroll_child", 2, GTK_TYPE_SCROLL_TYPE, GTK_SCROLL_STEP_BACKWARD, + G_TYPE_BOOLEAN, TRUE); + gtk_binding_entry_add_signal(binding_set, GDK_KP_Left, 0, + "scroll_child", 2, GTK_TYPE_SCROLL_TYPE, GTK_SCROLL_STEP_BACKWARD, + G_TYPE_BOOLEAN, TRUE); + gtk_binding_entry_add_signal(binding_set, GDK_Right, 0, + "scroll_child", 2, GTK_TYPE_SCROLL_TYPE, GTK_SCROLL_STEP_FORWARD, + G_TYPE_BOOLEAN, TRUE); + gtk_binding_entry_add_signal(binding_set, GDK_KP_Right, 0, + "scroll_child", 2, GTK_TYPE_SCROLL_TYPE, GTK_SCROLL_STEP_FORWARD, + G_TYPE_BOOLEAN, TRUE); +} + +gboolean is_event_within_textview(GdkEventButton *event) +{ + double pt[2]; + + if (ui.cur_item_type!=ITEM_TEXT) return FALSE; + get_pointer_coords((GdkEvent *)event, pt); + if (pt[0]bbox.left || pt[0]>ui.cur_item->bbox.right) return FALSE; + if (pt[1]bbox.top || pt[1]>ui.cur_item->bbox.bottom) return FALSE; + return TRUE; +} + +void hide_unimplemented(void) +{ + gtk_widget_hide(GET_COMPONENT("filePrintOptions")); + gtk_widget_hide(GET_COMPONENT("journalFlatten")); + gtk_widget_hide(GET_COMPONENT("papercolorOther")); + gtk_widget_hide(GET_COMPONENT("toolsSelectRegion")); + gtk_widget_hide(GET_COMPONENT("buttonSelectRegion")); + gtk_widget_hide(GET_COMPONENT("button2SelectRegion")); + gtk_widget_hide(GET_COMPONENT("button3SelectRegion")); + gtk_widget_hide(GET_COMPONENT("colorOther")); + gtk_widget_hide(GET_COMPONENT("helpIndex")); + + /* config file only works with glib 2.6 */ + if (glib_minor_version<6) { + gtk_widget_hide(GET_COMPONENT("optionsAutoSavePrefs")); + gtk_widget_hide(GET_COMPONENT("optionsSavePreferences")); + } +}