summaryrefslogtreecommitdiffstats
path: root/src/desktop-events.cpp
diff options
context:
space:
mode:
Diffstat (limited to 'src/desktop-events.cpp')
-rw-r--r--src/desktop-events.cpp551
1 files changed, 551 insertions, 0 deletions
diff --git a/src/desktop-events.cpp b/src/desktop-events.cpp
new file mode 100644
index 0000000..7ab2750
--- /dev/null
+++ b/src/desktop-events.cpp
@@ -0,0 +1,551 @@
+// SPDX-License-Identifier: GPL-2.0-or-later
+/**
+ * @file
+ * Event handlers for SPDesktop.
+ */
+/* Author:
+ * Lauris Kaplinski <lauris@kaplinski.com>
+ * Abhishek Sharma
+ *
+ * Copyright (C) 1999-2002 Lauris Kaplinski
+ * Copyright (C) 1999-2010 Others
+ *
+ * Released under GNU GPL v2+, read the file 'COPYING' for more information.
+ */
+
+#include <map>
+#include <string>
+
+#include "desktop-events.h"
+
+#include <gdkmm/display.h>
+#include <gdkmm/seat.h>
+
+#include <gtk/gtk.h>
+
+#include <glibmm/i18n.h>
+
+#include <2geom/line.h>
+#include <2geom/angle.h>
+
+#include "desktop.h"
+#include "document-undo.h"
+#include "document.h"
+#include "message-context.h"
+#include "preferences.h"
+#include "snap.h"
+
+#include "actions/actions-tools.h"
+
+#include "display/control/snap-indicator.h"
+#include "display/control/canvas-item-guideline.h"
+
+#include "object/sp-guide.h"
+#include "object/sp-namedview.h"
+#include "object/sp-root.h"
+
+#include "ui/cursor-utils.h"
+#include "ui/dialog-events.h"
+#include "ui/dialog/guides.h"
+#include "ui/event-debug.h"
+#include "ui/tools/tool-base.h"
+#include "ui/tools/node-tool.h"
+#include "ui/tools/select-tool.h"
+#include "ui/widget/canvas.h"
+
+#include "widgets/desktop-widget.h"
+
+#include "xml/repr.h"
+
+using Inkscape::DocumentUndo;
+
+static void snoop_extended(GdkEvent* event, SPDesktop *desktop);
+static void init_extended();
+
+/* Root canvas item handler */
+
+bool sp_desktop_root_handler(GdkEvent *event, SPDesktop *desktop)
+{
+#ifdef EVENT_DEBUG
+ ui_dump_event(reinterpret_cast<GdkEvent *>(event), "sp_desktop_root_handler");
+#endif
+ static bool watch = false;
+ static bool first = true;
+
+ if ( first ) {
+ Inkscape::Preferences *prefs = Inkscape::Preferences::get();
+ if ( prefs->getBool("/options/useextinput/value", true)
+ && prefs->getBool("/options/switchonextinput/value") ) {
+ watch = true;
+ init_extended();
+ }
+ first = false;
+ }
+ if ( watch ) {
+ snoop_extended(event, desktop);
+ }
+
+ if (auto ec = desktop->event_context) {
+ return (bool)ec->start_root_handler(event);
+ }
+ return false;
+}
+
+
+
+static Geom::Point drag_origin;
+static SPGuideDragType drag_type = SP_DRAG_NONE;
+//static bool reset_drag_origin = false; // when Ctrl is pressed while dragging, this is used to trigger resetting of the
+// // drag origin to that location so that constrained movement is more intuitive
+
+// Min distance from anchor to initiate rotation, measured in screenpixels
+#define tol 40.0
+
+bool sp_dt_guide_event(GdkEvent *event, Inkscape::CanvasItemGuideLine *guide_item, SPGuide *guide)
+{
+#ifdef EVENT_DEBUG
+ ui_dump_event(reinterpret_cast<GdkEvent *>(event), "sp_dt_guide_event");
+#endif
+
+ static bool moved = false;
+ bool ret = false;
+
+ SPDesktop *desktop = guide_item->get_canvas()->get_desktop();
+ if (!desktop) {
+ std::cerr << "sp_dt_guide_event: No desktop!" << std::endl;
+ return false;
+ }
+ // Limit to select tool only.
+ if (!dynamic_cast<Inkscape::UI::Tools::SelectTool *>(desktop->event_context) &&
+ !dynamic_cast<Inkscape::UI::Tools::NodeTool *>(desktop->event_context)) {
+ return false;
+ }
+
+ switch (event->type) {
+ case GDK_2BUTTON_PRESS:
+ if (event->button.button == 1) {
+ drag_type = SP_DRAG_NONE;
+ desktop->event_context->discard_delayed_snap_event();
+ guide_item->ungrab();
+ Inkscape::UI::Dialogs::GuidelinePropertiesDialog::showDialog(guide, desktop);
+ ret = true;
+ }
+ break;
+
+ case GDK_BUTTON_PRESS:
+ if (event->button.button == 1 && !guide->getLocked()) {
+ Geom::Point const event_w(event->button.x, event->button.y);
+ Geom::Point const event_dt(desktop->w2d(event_w));
+
+ // Due to the tolerance allowed when grabbing a guide, event_dt will generally
+ // be close to the guide but not just exactly on it. The drag origin calculated
+ // here must be exactly on the guide line though, otherwise
+ // small errors will occur once we snap, see
+ // https://bugs.launchpad.net/inkscape/+bug/333762
+ drag_origin = Geom::projection(event_dt, Geom::Line(guide->getPoint(), guide->angle()));
+
+ if (event->button.state & GDK_SHIFT_MASK) {
+ // with shift we rotate the guide
+ drag_type = SP_DRAG_ROTATE;
+ } else {
+ if (event->button.state & GDK_CONTROL_MASK) {
+ drag_type = SP_DRAG_MOVE_ORIGIN;
+ } else {
+ drag_type = SP_DRAG_TRANSLATE;
+ }
+ }
+
+ if (drag_type == SP_DRAG_ROTATE || drag_type == SP_DRAG_TRANSLATE) {
+ guide_item->grab((Gdk::BUTTON_RELEASE_MASK |
+ Gdk::BUTTON_PRESS_MASK |
+ Gdk::POINTER_MOTION_MASK ),
+ nullptr);
+ }
+ ret = true;
+ }
+ break;
+
+ case GDK_MOTION_NOTIFY:
+ if (drag_type != SP_DRAG_NONE) {
+ Geom::Point const motion_w(event->motion.x,
+ event->motion.y);
+ Geom::Point motion_dt(desktop->w2d(motion_w));
+
+ sp_event_context_snap_delay_handler(
+ desktop->event_context, (void *) guide_item, (void *) guide, (GdkEventMotion *)event,
+ Inkscape::UI::Tools::DelayedSnapEvent::GUIDE_HANDLER);
+
+ // This is for snapping while dragging existing guidelines. New guidelines,
+ // which are dragged off the ruler, are being snapped in sp_dt_ruler_event
+ SnapManager &m = desktop->namedview->snap_manager;
+ m.setup(desktop, true, guide, nullptr);
+ if (drag_type == SP_DRAG_MOVE_ORIGIN) {
+ // If we snap in guideConstrainedSnap() below, then motion_dt will
+ // be forced to be on the guide. If we don't snap however, then
+ // the origin should still be constrained to the guide. So let's do
+ // that explicitly first:
+ Geom::Line line(guide->getPoint(), guide->angle());
+ Geom::Coord t = line.nearestTime(motion_dt);
+ motion_dt = line.pointAt(t);
+ if (!(event->motion.state & GDK_SHIFT_MASK)) {
+ m.guideConstrainedSnap(motion_dt, *guide);
+ }
+ } else if (!((drag_type == SP_DRAG_ROTATE) && (event->motion.state & GDK_CONTROL_MASK))) {
+ // cannot use shift here to disable snapping, because we already use it for rotating the guide
+ Geom::Point temp;
+ if (drag_type == SP_DRAG_ROTATE) {
+ temp = guide->getPoint();
+ m.guideFreeSnap(motion_dt, temp, true, false);
+ guide->moveto(temp, false);
+ } else {
+ temp = guide->getNormal();
+ m.guideFreeSnap(motion_dt, temp, false, true);
+ guide->set_normal(temp, false);
+ }
+ }
+ m.unSetup();
+
+ switch (drag_type) {
+ case SP_DRAG_TRANSLATE:
+ {
+ guide->moveto(motion_dt, false);
+ break;
+ }
+ case SP_DRAG_ROTATE:
+ {
+ Geom::Point pt = motion_dt - guide->getPoint();
+ Geom::Angle angle(pt);
+ if (event->motion.state & GDK_CONTROL_MASK) {
+ Inkscape::Preferences *prefs = Inkscape::Preferences::get();
+ unsigned const snaps = abs(prefs->getInt("/options/rotationsnapsperpi/value", 12));
+ bool const relative_snaps = prefs->getBool("/options/relativeguiderotationsnap/value", false);
+ if (snaps) {
+ if (relative_snaps) {
+ Geom::Angle orig_angle(guide->getNormal());
+ Geom::Angle snap_angle = angle - orig_angle;
+ double sections = floor(snap_angle.radians0() * snaps / M_PI + .5);
+ angle = (M_PI / snaps) * sections + orig_angle.radians0();
+ } else {
+ double sections = floor(angle.radians0() * snaps / M_PI + .5);
+ angle = (M_PI / snaps) * sections;
+ }
+ }
+ }
+ guide->set_normal(Geom::Point::polar(angle).cw(), false);
+ break;
+ }
+ case SP_DRAG_MOVE_ORIGIN:
+ {
+ guide->moveto(motion_dt, false);
+ break;
+ }
+ case SP_DRAG_NONE:
+ assert(false);
+ break;
+ }
+ moved = true;
+ desktop->set_coordinate_status(motion_dt);
+ desktop->getCanvas()->grab_focus();
+
+ ret = true;
+ }
+ break;
+
+ case GDK_BUTTON_RELEASE:
+ if (drag_type != SP_DRAG_NONE && event->button.button == 1) {
+ desktop->event_context->discard_delayed_snap_event();
+
+ if (moved) {
+ Geom::Point const event_w(event->button.x,
+ event->button.y);
+ Geom::Point event_dt(desktop->w2d(event_w));
+ SnapManager &m = desktop->namedview->snap_manager;
+ m.setup(desktop, true, guide, nullptr);
+ if (drag_type == SP_DRAG_MOVE_ORIGIN) {
+ // If we snap in guideConstrainedSnap() below, then motion_dt will
+ // be forced to be on the guide. If we don't snap however, then
+ // the origin should still be constrained to the guide. So let's
+ // do that explicitly first:
+ Geom::Line line(guide->getPoint(), guide->angle());
+ Geom::Coord t = line.nearestTime(event_dt);
+ event_dt = line.pointAt(t);
+ if (!(event->button.state & GDK_SHIFT_MASK)) {
+ m.guideConstrainedSnap(event_dt, *guide);
+ }
+ } else if (!((drag_type == SP_DRAG_ROTATE) && (event->motion.state & GDK_CONTROL_MASK))) {
+ // cannot use shift here to disable snapping, because we already use it for rotating the guide
+ Geom::Point temp;
+ if (drag_type == SP_DRAG_ROTATE) {
+ temp = guide->getPoint();
+ m.guideFreeSnap(event_dt, temp, true, false);
+ guide->moveto(temp, false);
+ } else {
+ temp = guide->getNormal();
+ m.guideFreeSnap(event_dt, temp, false, true);
+ guide->set_normal(temp, false);
+ }
+ }
+ m.unSetup();
+
+ if (guide_item->get_canvas()->world_point_inside_canvas(event_w)) {
+ switch (drag_type) {
+ case SP_DRAG_TRANSLATE:
+ {
+ guide->moveto(event_dt, true);
+ break;
+ }
+ case SP_DRAG_ROTATE:
+ {
+ Geom::Point pt = event_dt - guide->getPoint();
+ Geom::Angle angle(pt);
+ if (event->motion.state & GDK_CONTROL_MASK) {
+ Inkscape::Preferences *prefs = Inkscape::Preferences::get();
+ unsigned const snaps = abs(prefs->getInt("/options/rotationsnapsperpi/value", 12));
+ bool const relative_snaps = prefs->getBool("/options/relativeguiderotationsnap/value", false);
+ if (snaps) {
+ if (relative_snaps) {
+ Geom::Angle orig_angle(guide->getNormal());
+ Geom::Angle snap_angle = angle - orig_angle;
+ double sections = floor(snap_angle.radians0() * snaps / M_PI + .5);
+ angle = (M_PI / snaps) * sections + orig_angle.radians0();
+ } else {
+ double sections = floor(angle.radians0() * snaps / M_PI + .5);
+ angle = (M_PI / snaps) * sections;
+ }
+ }
+ }
+ guide->set_normal(Geom::Point::polar(angle).cw(), true);
+ break;
+ }
+ case SP_DRAG_MOVE_ORIGIN:
+ {
+ guide->moveto(event_dt, true);
+ break;
+ }
+ case SP_DRAG_NONE:
+ assert(false);
+ break;
+ }
+ DocumentUndo::done(desktop->getDocument(), _("Move guide"), "");
+ } else {
+ /* Undo movement of any attached shapes. */
+ guide->moveto(guide->getPoint(), false);
+ guide->set_normal(guide->getNormal(), false);
+ guide->remove();
+ guide_item = nullptr;
+ desktop->event_context->use_tool_cursor();
+
+ DocumentUndo::done(desktop->getDocument(), _("Delete guide"), "");
+ }
+ moved = false;
+ desktop->set_coordinate_status(event_dt);
+ }
+ drag_type = SP_DRAG_NONE;
+ if (guide_item) {
+ guide_item->ungrab();
+ }
+ ret = true;
+ }
+ break;
+
+ case GDK_ENTER_NOTIFY:
+ {
+ // This is a UX thing. Check if the canvas has focus, so the user knows they can
+ // use hotkeys. See issue: https://gitlab.com/inkscape/inkscape/-/issues/2439
+ if (!guide->getLocked() && desktop->getCanvas()->has_focus()) {
+ guide_item->set_stroke(guide->getHiColor());
+ }
+
+ // set move or rotate cursor
+ Geom::Point const event_w(event->crossing.x, event->crossing.y);
+
+ auto display = desktop->getCanvas()->get_display();
+ auto window = desktop->getCanvas()->get_window();
+
+ Glib::RefPtr<Gdk::Cursor> cursor;
+ if (guide->getLocked()) {
+ cursor = Inkscape::load_svg_cursor(display, window, "select.svg");
+ } else if ((event->crossing.state & GDK_SHIFT_MASK) && (drag_type != SP_DRAG_MOVE_ORIGIN)) {
+ cursor = Inkscape::load_svg_cursor(display, window, "rotate.svg");
+ } else {
+ cursor = Gdk::Cursor::create(display, "grab");
+ }
+ window->set_cursor(cursor);
+
+ char *guide_description = guide->description();
+ desktop->guidesMessageContext()->setF(Inkscape::NORMAL_MESSAGE, _("<b>Guideline</b>: %s"), guide_description);
+ g_free(guide_description);
+ break;
+ }
+
+ case GDK_LEAVE_NOTIFY:
+
+ guide_item->set_stroke(guide->getColor());
+
+ // restore event context's cursor
+ desktop->event_context->use_tool_cursor();
+
+ desktop->guidesMessageContext()->clear();
+ break;
+
+ case GDK_KEY_PRESS:
+ switch (Inkscape::UI::Tools::get_latin_keyval (&event->key)) {
+ case GDK_KEY_Delete:
+ case GDK_KEY_KP_Delete:
+ case GDK_KEY_BackSpace:
+ {
+ SPDocument *doc = guide->document;
+ if (!guide->getLocked()) {
+ guide->remove();
+ guide_item = nullptr;
+ DocumentUndo::done(doc, _("Delete guide"), "");
+ ret = true;
+ desktop->event_context->discard_delayed_snap_event();
+ desktop->event_context->use_tool_cursor();
+ }
+ break;
+ }
+ case GDK_KEY_Shift_L:
+ case GDK_KEY_Shift_R:
+ if (drag_type != SP_DRAG_MOVE_ORIGIN) {
+
+ auto display = desktop->getCanvas()->get_display();
+ auto window = desktop->getCanvas()->get_window();
+
+ auto cursor = Inkscape::load_svg_cursor(display, window, "rotate.svg");
+ window->set_cursor(cursor);
+ ret = true;
+ break;
+ }
+
+ default:
+ // do nothing;
+ break;
+ }
+ break;
+
+ case GDK_KEY_RELEASE:
+ switch (Inkscape::UI::Tools::get_latin_keyval (&event->key)) {
+ case GDK_KEY_Shift_L:
+ case GDK_KEY_Shift_R:
+ {
+ auto display = Gdk::Display::get_default();
+ auto guide_cursor = Gdk::Cursor::create(display, "grab");
+ desktop->getCanvas()->get_window()->set_cursor(guide_cursor);
+ break;
+ }
+ default:
+ // do nothing;
+ break;
+ }
+ break;
+
+ default:
+ break;
+ }
+
+ return ret;
+}
+
+static std::map<std::string, Glib::ustring> toolToUse;
+static std::string lastName;
+static GdkInputSource lastType = GDK_SOURCE_MOUSE;
+
+static void init_extended()
+{
+ Glib::ustring avoidName("pad");
+ auto display = Gdk::Display::get_default();
+ auto seat = display->get_default_seat();
+ auto const devices = seat->get_slaves(Gdk::SEAT_CAPABILITY_ALL);
+
+ if ( !devices.empty() ) {
+ for (auto const &dev : devices) {
+ auto const devName = dev->get_name();
+ auto devSrc = dev->get_source();
+
+ if ( !devName.empty()
+ && (avoidName != devName)
+ && (devSrc != Gdk::SOURCE_MOUSE) ) {
+// g_message("Adding '%s' as [%d]", devName, devSrc);
+
+ // Set the initial tool for the device
+ switch ( devSrc ) {
+ case Gdk::SOURCE_PEN:
+ toolToUse[devName] = "Calligraphic";
+ break;
+ case Gdk::SOURCE_ERASER:
+ toolToUse[devName] = "Eraser";
+ break;
+ case Gdk::SOURCE_CURSOR:
+ toolToUse[devName] = "Select";
+ break;
+ default:
+ ; // do not add
+ }
+// } else if (devName) {
+// g_message("Skippn '%s' as [%d]", devName, devSrc);
+ }
+ }
+ }
+}
+
+
+void snoop_extended(GdkEvent* event, SPDesktop *desktop)
+{
+ GdkDevice *source_device = gdk_event_get_source_device (event);
+ GdkInputSource source = GDK_SOURCE_MOUSE;
+ std::string name;
+
+ if (! source_device) return;
+ switch ( event->type ) {
+ case GDK_MOTION_NOTIFY:
+ case GDK_BUTTON_PRESS:
+ case GDK_2BUTTON_PRESS:
+ case GDK_3BUTTON_PRESS:
+ case GDK_BUTTON_RELEASE:
+ case GDK_SCROLL:
+ case GDK_PROXIMITY_IN:
+ case GDK_PROXIMITY_OUT:
+ // fix to previous code using event->device that did not point to original device that generated the event.
+ source = gdk_device_get_source(source_device);
+ name = gdk_device_get_name(source_device);
+
+ break;
+
+ default:
+ ;
+ }
+
+
+ if (!name.empty()) {
+ if ( lastType != source || lastName != name ) {
+ // The device switched. See if it is one we 'count'
+ //g_message("Changed device %s -> %s", lastName.c_str(), name.c_str());
+ auto it = toolToUse.find(lastName);
+ if (it != toolToUse.end()) {
+ // Save the tool currently selected for next time the input
+ // device shows up.
+ it->second = get_active_tool(desktop);
+ }
+
+ it = toolToUse.find(name);
+ if (it != toolToUse.end() ) {
+ set_active_tool(desktop, it->second);
+ }
+
+ lastName = name;
+ lastType = source;
+ }
+ }
+}
+/*
+ Local Variables:
+ mode:c++
+ c-file-style:"stroustrup"
+ c-file-offsets:((innamespace . 0)(inline-open . 0)(case-label . +))
+ indent-tabs-mode:nil
+ fill-column:99
+ End:
+*/
+// vim: filetype=cpp:expandtab:shiftwidth=4:tabstop=8:softtabstop=4:fileencoding=utf-8:textwidth=99 :
+