Improve handling of removeOn*() and filter out duplicate move events.

Closes #913
This commit is contained in:
Jürg Lehni 2016-01-27 09:48:20 +01:00
parent ee4760afc4
commit caa93a51ca

View file

@ -997,6 +997,11 @@ new function() { // Injection scope for mouse events on the browser
// Returns true if event was stopped, false otherwise. // Returns true if event was stopped, false otherwise.
function emitEvents(view, item, type, event, point, prevPoint) { function emitEvents(view, item, type, event, point, prevPoint) {
// Before handling events, process removeOn() calls for cleanup.
// NOTE: As soon as there is one event handler receiving mousedrag
// events, non of the removeOnMove() items will be removed while the
// user is dragging.
view._project.removeOn(type);
// Set called to false, so it will reflect if the following calls to // Set called to false, so it will reflect if the following calls to
// emitEvent() have called a handler. // emitEvent() have called a handler.
called = false; called = false;
@ -1062,20 +1067,23 @@ new function() { // Injection scope for mouse events on the browser
*/ */
_handleEvent: function(type, event, point) { _handleEvent: function(type, event, point) {
var handleItems = this._itemEvents[type], var handleItems = this._itemEvents[type],
project = paper.project,
tool = this._scope.tool, tool = this._scope.tool,
view = this; view = this;
// If it's a native mousemove event but the mouse is down, convert
// it to a mousedrag. function responds(type) {
return view._itemEvents[type] || view.responds(type)
|| tool && tool.responds(type);
}
// If it's a native mousemove event but the mouse is down, and at
// least one of the events responds to mousedrag, convert to it.
// NOTE: emitEvent(), as well as Tool#_handleEvent() fall back to // NOTE: emitEvent(), as well as Tool#_handleEvent() fall back to
// mousemove if the objects don't respond to mousedrag. // mousemove if the objects don't respond to mousedrag.
if (type === 'mousemove' && dragging) if (type === 'mousemove' && dragging && responds('mousedrag'))
type = 'mousedrag'; type = 'mousedrag';
// Before handling events, process removeOn() calls for cleanup.
if (project)
project.removeOn(type);
if (!point) if (!point)
point = this.getEventPoint(event); point = this.getEventPoint(event);
// Run the hit-test on items first, but only if we're required to do // Run the hit-test on items first, but only if we're required to do
// so for this given mouse event, see #_countItemEvent(). // so for this given mouse event, see #_countItemEvent().
var inView = this.getBounds().contains(point), var inView = this.getBounds().contains(point),
@ -1088,46 +1096,46 @@ new function() { // Injection scope for mouse events on the browser
// Keep track if view event should be handled, so we can use it // Keep track if view event should be handled, so we can use it
// to decide if tool._handleEvent() shall be called after. // to decide if tool._handleEvent() shall be called after.
handle = false, handle = false,
stopped = false,
mouse = {}; mouse = {};
// Create a simple lookup object to quickly check for different // Create a simple lookup object to quickly check for different
// mouse event types. // mouse event types.
mouse[type.substr(5)] = true; mouse[type.substr(5)] = true;
// Always first call the view's mouse handlers, as required by // Always first call the view's mouse handlers, as required by
// CanvasView, and then handle the active tool after, if any. // CanvasView, and then handle the active tool after, if any.
// Handle mousemove first, even if this is not actually a mousemove // Handle mousemove first, even if this is not actually a mousemove
// event but the mouse has moved since the last event, but do not // event but the mouse has moved since the last event, but do not
// allow it to stop the other events in that case. // allow it to stop the other events in that case.
var nativeMove = mouse.move || mouse.drag, var moveType = mouse.move || mouse.drag ? type : 'mousemove';
moveType = nativeMove && type || 'mousemove'; // Handle mouseenter / leave between items.
if (moveType) { if (item !== overItem) {
// Handle mouseenter / leave between items, as well as views. if (overItem)
if (item !== overItem) { emitEvent(overItem, 'mouseleave', event, point);
if (overItem) if (item)
emitEvent(overItem, 'mouseleave', event, point); emitEvent(item, 'mouseenter', event, point);
if (item)
emitEvent(item, 'mouseenter', event, point);
}
overItem = item;
if (nativeMove && (wasInView ^ inView)) {
emitEvent(this, inView ? 'mouseenter' : 'mouseleave', event,
point);
overView = inView ? this : null;
handle = true; // To include the leaving move.
}
if (inView || mouse.drag && !lastPoint.equals(point)) {
stopped = emitEvents(this, item, moveType, event, point,
lastPoint);
handle = true;
}
wasInView = inView;
} }
overItem = item;
// Handle mouseenter / leave on the view.
if (wasInView ^ inView) {
emitEvent(this, inView ? 'mouseenter' : 'mouseleave', event,
point);
overView = inView ? this : null;
handle = true; // To include the leaving move.
}
// Now finally handle the mousemove / mousedrag event.
// mousedrag is allowed to leave the view and still triggers events,
// but do not trigger two subsequent even with the same location.
if ((inView || mouse.drag) && !point.equals(lastPoint)) {
emitEvents(this, item, moveType, event, point, lastPoint);
handle = true;
}
wasInView = inView;
// Now handle mousedown / mouseup // Now handle mousedown / mouseup
if (!nativeMove && // We emit mousedown only when in the view, and mouseup regardless,
// We emit mousedown only when in the view, and mouseup // as long as the mousedown event was inside.
// regardless, as long as the mousedown event was inside. if (mouse.down && inView || mouse.up && downPoint) {
(handle = mouse.down && inView || mouse.up && downPoint)) { var stopped = emitEvents(this, item, type, event, point,
stopped = emitEvents(this, item, type, event, point, downPoint); downPoint);
if (mouse.down) { if (mouse.down) {
// See if we're clicking again on the same item, within the // See if we're clicking again on the same item, within the
// double-click time. Firefox uses 300ms as the max time // double-click time. Firefox uses 300ms as the max time
@ -1155,6 +1163,7 @@ new function() { // Injection scope for mouse events on the browser
// events that started outside the view as mousemove events on // events that started outside the view as mousemove events on
// the view (needed to handle touch scrolling correctly). // the view (needed to handle touch scrolling correctly).
wasInView = false; wasInView = false;
handle = true;
} }
lastPoint = point; lastPoint = point;
// Now finally call the tool events, but filter mouse move events // Now finally call the tool events, but filter mouse move events
@ -1171,13 +1180,7 @@ new function() { // Injection scope for mouse events on the browser
// - If this is a mousedown event, and the view or tools respond to // - If this is a mousedown event, and the view or tools respond to
// mouseup. // mouseup.
function responds(type) { if (called && !mouse.move || mouse.down && responds('mouseup'))
return view._itemEvents[type] || view.responds(type)
|| tool && tool.responds(type);
}
if (called && (!nativeMove || responds('mousedrag'))
|| mouse.down && responds('mouseup'))
event.preventDefault(); event.preventDefault();
}, },