mirror of
https://github.com/scratchfoundation/paper.js.git
synced 2025-01-20 22:39:50 -05:00
Improve handling of removeOn*() and filter out duplicate move events.
Closes #913
This commit is contained in:
parent
ee4760afc4
commit
caa93a51ca
1 changed files with 46 additions and 43 deletions
|
@ -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,20 +1096,18 @@ 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) {
|
|
||||||
// Handle mouseenter / leave between items, as well as views.
|
|
||||||
if (item !== overItem) {
|
if (item !== overItem) {
|
||||||
if (overItem)
|
if (overItem)
|
||||||
emitEvent(overItem, 'mouseleave', event, point);
|
emitEvent(overItem, 'mouseleave', event, point);
|
||||||
|
@ -1109,25 +1115,27 @@ new function() { // Injection scope for mouse events on the browser
|
||||||
emitEvent(item, 'mouseenter', event, point);
|
emitEvent(item, 'mouseenter', event, point);
|
||||||
}
|
}
|
||||||
overItem = item;
|
overItem = item;
|
||||||
if (nativeMove && (wasInView ^ inView)) {
|
// Handle mouseenter / leave on the view.
|
||||||
|
if (wasInView ^ inView) {
|
||||||
emitEvent(this, inView ? 'mouseenter' : 'mouseleave', event,
|
emitEvent(this, inView ? 'mouseenter' : 'mouseleave', event,
|
||||||
point);
|
point);
|
||||||
overView = inView ? this : null;
|
overView = inView ? this : null;
|
||||||
handle = true; // To include the leaving move.
|
handle = true; // To include the leaving move.
|
||||||
}
|
}
|
||||||
if (inView || mouse.drag && !lastPoint.equals(point)) {
|
// Now finally handle the mousemove / mousedrag event.
|
||||||
stopped = emitEvents(this, item, moveType, event, point,
|
// mousedrag is allowed to leave the view and still triggers events,
|
||||||
lastPoint);
|
// 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;
|
handle = true;
|
||||||
}
|
}
|
||||||
wasInView = inView;
|
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();
|
||||||
},
|
},
|
||||||
|
|
||||||
|
|
Loading…
Reference in a new issue