X-Git-Url: https://git.donarmstrong.com/?a=blobdiff_plain;f=src%2Fxo-misc.c;h=cf63de78c3e2cfc1c8377f53bd7ac2d4b96f6b32;hb=b471a5e1ffc9ad87400f079679fdf261875d513e;hp=b7448459e2ca69a26ed76e1ea6e5fe22f12d424f;hpb=34dbe92edd95fbf7082c3e54f438c76355254fe5;p=xournal.git diff --git a/src/xo-misc.c b/src/xo-misc.c index b744845..cf63de7 100644 --- a/src/xo-misc.c +++ b/src/xo-misc.c @@ -93,15 +93,15 @@ struct Page *new_page_with_bg(struct Background *bg, double width, double height void realloc_cur_path(int n) { if (n <= ui.cur_path_storage_alloc) return; - ui.cur_path_storage_alloc = n+10; - ui.cur_path.coords = g_realloc(ui.cur_path.coords, 2*(n+10)*sizeof(double)); + ui.cur_path_storage_alloc = n+100; + ui.cur_path.coords = g_realloc(ui.cur_path.coords, 2*(n+100)*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)); + ui.cur_widths_storage_alloc = n+100; + ui.cur_widths = g_realloc(ui.cur_widths, (n+100)*sizeof(double)); } // undo utility functions @@ -395,14 +395,18 @@ void fix_xinput_coords(GdkEvent *event) *py = iy + sy; } else { - *px += sx; - *py += sy; + /* with GTK+ 2.16 or earlier, the event comes from the parent gdkwindow + and so needs to be adjusted for scrolling */ + if (gtk_major_version == 2 && gtk_minor_version <= 16) { + *px += sx; + *py += sy; + } /* with GTK+ 2.17, events come improperly translated, and the event's GdkWindow isn't even the same for ButtonDown as for MotionNotify... */ - if (!gtk_check_version(2,17,0)) { // GTK+ 2.17 issues !! + if (gtk_major_version == 2 && gtk_minor_version == 17) { // GTK+ 2.17 issues !! gdk_window_get_position(GTK_WIDGET(canvas)->window, &wx, &wy); - *px -= wx; - *py -= wy; + *px += sx - wx; + *py += sy - wy; } } #endif @@ -410,12 +414,24 @@ void fix_xinput_coords(GdkEvent *event) double get_pressure_multiplier(GdkEvent *event) { + double *axes; double rawpressure; + GdkDevice *device; + + if (event->type == GDK_MOTION_NOTIFY) { + axes = event->motion.axes; + device = event->motion.device; + } + else { + axes = event->button.axes; + device = event->button.device; + } - if (event->button.device == gdk_device_get_core_pointer() - || event->button.device->num_axes <= 2) return 1.0; + if (device == gdk_device_get_core_pointer() + || device->num_axes <= 2) return 1.0; - rawpressure = event->button.axes[2]/(event->button.device->axes[2].max - event->button.device->axes[2].min); + rawpressure = axes[2]/(device->axes[2].max - device->axes[2].min); + if (!finite(rawpressure)) return 1.0; return ((1-rawpressure)*ui.width_minimum_multiplier + rawpressure*ui.width_maximum_multiplier); } @@ -649,7 +665,7 @@ void rescale_bg_pixmaps(void) // in progressive mode we scale only visible pages if (ui.progressive_bg && !is_visible(pg)) continue; - if (pg->bg->type == BG_PIXMAP && pg->bg->canvas_item!=NULL) { // do the rescaling ourselves + if (pg->bg->type == BG_PIXMAP && pg->bg->canvas_item!=NULL) { 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); @@ -671,8 +687,8 @@ void rescale_bg_pixmaps(void) // 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; + if (add_bgpdf_request(pg->bg->file_page_seq, zoom_to_request)) + pg->bg->pixbuf_scale = zoom_to_request; } } } @@ -1100,12 +1116,9 @@ void update_mappings_menu_linkings(void) void update_mappings_menu(void) { gtk_widget_set_sensitive(GET_COMPONENT("optionsButtonMappings"), ui.use_xinput); - gtk_widget_set_sensitive(GET_COMPONENT("optionsDiscardCoreEvents"), ui.use_xinput); gtk_widget_set_sensitive(GET_COMPONENT("optionsPressureSensitive"), ui.use_xinput); gtk_check_menu_item_set_active( GTK_CHECK_MENU_ITEM(GET_COMPONENT("optionsButtonMappings")), ui.use_erasertip); - gtk_check_menu_item_set_active( - GTK_CHECK_MENU_ITEM(GET_COMPONENT("optionsDiscardCoreEvents")), ui.discard_corepointer); gtk_check_menu_item_set_active( GTK_CHECK_MENU_ITEM(GET_COMPONENT("optionsPressureSensitive")), ui.pressure_sensitivity); @@ -1277,6 +1290,8 @@ void update_page_stuff(void) gtk_combo_box_set_active(layerbox, ui.cur_page->nlayers-1-ui.layerno); ui.in_update_page_stuff = FALSE; + gtk_container_forall(GTK_CONTAINER(layerbox), unset_flags, (gpointer)GTK_CAN_FOCUS); + // update the paper-style menu radio buttons if (ui.view_continuous) @@ -1314,7 +1329,7 @@ void update_page_stuff(void) break; default: gtk_check_menu_item_set_active( - GTK_CHECK_MENU_ITEM(GET_COMPONENT("papercolorOther")), TRUE); + GTK_CHECK_MENU_ITEM(GET_COMPONENT("papercolorNA")), TRUE); break; } switch (ui.cur_page->bg->ruling) { @@ -1488,7 +1503,6 @@ void process_color_activate(GtkMenuItem *menuitem, int color_no, guint color_rgb } if (ui.cur_mapping != 0 && !ui.button_switch_mapping) return; // not user-generated - reset_focus(); if (ui.cur_item_type == ITEM_TEXT) recolor_temp_text(color_no, color_rgba); @@ -1531,7 +1545,6 @@ void process_thickness_activate(GtkMenuItem *menuitem, int tool, int val) if (ui.cur_mapping != 0 && !ui.button_switch_mapping) 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(); } @@ -1545,7 +1558,6 @@ void process_thickness_activate(GtkMenuItem *menuitem, int tool, int val) which_mapping = ui.cur_mapping; else which_mapping = 0; if (ui.brushes[which_mapping][tool].thickness_no == val) return; - reset_focus(); end_text(); ui.brushes[which_mapping][tool].thickness_no = val; ui.brushes[which_mapping][tool].thickness = predef_thickness[tool][val]; @@ -1558,16 +1570,18 @@ void process_thickness_activate(GtkMenuItem *menuitem, int tool, int val) update_cursor(); } -void process_papercolor_activate(GtkMenuItem *menuitem, int color) +void process_papercolor_activate(GtkMenuItem *menuitem, int color, guint rgba) { struct Page *pg; GList *pglist; gboolean hasdone; - if (!gtk_check_menu_item_get_active(GTK_CHECK_MENU_ITEM (menuitem))) - return; + if (GTK_OBJECT_TYPE(menuitem) == GTK_TYPE_RADIO_MENU_ITEM) { + if (!gtk_check_menu_item_get_active(GTK_CHECK_MENU_ITEM (menuitem))) + return; + } - if ((ui.cur_page->bg->type != BG_SOLID) || ui.bg_apply_all_pages) + if ((ui.cur_page->bg->type != BG_SOLID) || ui.bg_apply_all_pages || color == COLOR_OTHER) gtk_check_menu_item_set_active( GTK_CHECK_MENU_ITEM(GET_COMPONENT("papercolorNA")), TRUE); @@ -1575,7 +1589,7 @@ void process_papercolor_activate(GtkMenuItem *menuitem, int color) 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) { + if (pg->bg->type == BG_SOLID && pg->bg->color_rgba != rgba) { prepare_new_undo(); if (hasdone) undo->multiop |= MULTIOP_CONT_UNDO; undo->multiop |= MULTIOP_CONT_REDO; @@ -1586,7 +1600,7 @@ void process_papercolor_activate(GtkMenuItem *menuitem, int color) undo->bg->canvas_item = NULL; pg->bg->color_no = color; - pg->bg->color_rgba = predef_bgcolors_rgba[color]; + pg->bg->color_rgba = rgba; update_canvas_bg(pg); } if (!ui.bg_apply_all_pages) break; @@ -1661,13 +1675,13 @@ gboolean ok_to_close(void) } // send the focus back to the appropriate widget + void reset_focus(void) { 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 @@ -1798,7 +1812,8 @@ void resize_journal_items_by(GList *itemlist, double scaling_x, double scaling_y /* NOTE ABOUT BUTTON MAPPINGS: ui.cur_mapping is 0 except while a canvas click event is being processed ... or if ui.button_switch_mapping is - enabled and mappings are switched! */ + enabled and mappings are switched (but even then, canvas should have + a pointer grab from the initial click that switched the mapping) */ void switch_mapping(int m) { @@ -1809,6 +1824,8 @@ void switch_mapping(int m) ui.cur_brush = &(ui.brushes[m][ui.toolno[m]]); if (ui.toolno[m] == TOOL_TEXT) ui.cur_brush = &(ui.brushes[m][TOOL_PEN]); + if (m==0) ui.which_unswitch_button = 0; + update_tool_buttons(); update_color_menu(); update_cursor(); @@ -1821,7 +1838,6 @@ void process_mapping_activate(GtkMenuItem *menuitem, int m, int tool) if (ui.toolno[m] == tool) return; switch_mapping(0); end_text(); - reset_focus(); ui.toolno[m] = tool; if (ui.linked_brush[m] == BRUSH_COPIED) { @@ -2009,7 +2025,6 @@ 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")); @@ -2031,7 +2046,6 @@ void hide_unimplemented(void) void do_fullscreen(gboolean active) { end_text(); - reset_focus(); ui.fullscreen = active; gtk_check_menu_item_set_active( GTK_CHECK_MENU_ITEM(GET_COMPONENT("viewFullscreen")), ui.fullscreen); @@ -2106,17 +2120,59 @@ gboolean fix_extended_events (GtkWidget *widget, GdkEvent *event, } */ -// disable xinput when layer combo box is popped up, to avoid GTK+ 2.17 crash -gboolean combobox_popup_disable_xinput (GtkWidget *widget, GdkEvent *event, - gpointer user_data) +/* When enter is pressed into page spinbox, send focus back to canvas. */ + +gboolean handle_activate_signal(GtkWidget *widget, gpointer user_data) { - gboolean is_shown; - - g_object_get(G_OBJECT(widget), "popup-shown", &is_shown, NULL); - gdk_input_set_extension_events(GTK_WIDGET(canvas)->window, - GDK_POINTER_MOTION_MASK | GDK_BUTTON_MOTION_MASK | GDK_BUTTON_PRESS_MASK | GDK_BUTTON_RELEASE_MASK, - (ui.use_xinput && !is_shown)?GDK_EXTENSION_EVENTS_ALL:GDK_EXTENSION_EVENTS_NONE); + reset_focus(); + return FALSE; +} + +/* recursively unset widget flags */ + +void unset_flags(GtkWidget *w, gpointer flag) +{ + GTK_WIDGET_UNSET_FLAGS(w, (GtkWidgetFlags)flag); + if(GTK_IS_CONTAINER(w)) + gtk_container_forall(GTK_CONTAINER(w), unset_flags, flag); } +/* reset focus when a key or button press event reaches someone, or when the + page-number spin button should relinquish control... */ +gboolean intercept_activate_events(GtkWidget *w, GdkEvent *ev, gpointer data) +{ + if (w == GET_COMPONENT("hbox1")) { + /* the event won't be processed since the hbox1 doesn't know what to do with it, + so we might as well kill it and avoid confusing ourselves when it gets + propagated further ... */ + return TRUE; + } + if (w == GET_COMPONENT("spinPageNo")) { + /* we let the spin button take care of itself, and don't steal its focus, + unless the user presses Esc or Tab (in those cases we intervene) */ + if (ev->type != GDK_KEY_PRESS) return FALSE; + if (ev->key.keyval == GDK_Escape) + gtk_spin_button_set_value(GTK_SPIN_BUTTON(w), ui.pageno+1); // abort + else if (ev->key.keyval != GDK_Tab && ev->key.keyval != GDK_ISO_Left_Tab) + return FALSE; // let the spin button process it + } + + // otherwise, we want to make sure the canvas or text item gets focus back... + reset_focus(); + return FALSE; +} + +void install_focus_hooks(GtkWidget *w, gpointer data) +{ + if (w == NULL) return; + g_signal_connect(w, "key-press-event", G_CALLBACK(intercept_activate_events), data); + g_signal_connect(w, "button-press-event", G_CALLBACK(intercept_activate_events), data); + if (GTK_IS_MENU_ITEM(w)) { + g_signal_connect(w, "activate", G_CALLBACK(intercept_activate_events), data); + install_focus_hooks(gtk_menu_item_get_submenu(GTK_MENU_ITEM(w)), data); + } + if(GTK_IS_CONTAINER(w)) + gtk_container_forall(GTK_CONTAINER(w), install_focus_hooks, data); +}