Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Drag events are called 2x per animation frame #644

Closed
samreid opened this issue Jul 8, 2017 · 28 comments
Closed

Drag events are called 2x per animation frame #644

samreid opened this issue Jul 8, 2017 · 28 comments

Comments

@samreid
Copy link
Member

samreid commented Jul 8, 2017

In phetsims/circuit-construction-kit-dc#74 I noticed that drag events were getting called 2x as often as animation frames. Is that normal? It seems like it has performance implications we may need to address.

For instance, in @pixelzoom's Function Builder, I put a console.log in stepSimulation and MovableObject.drag and saw this output on Mac chrome:
image

Note that one occurrence even had 3 drags between animation frames.

@jonathanolson
Copy link
Contributor

drag events were getting called 2x as often as animation frames. Is that normal?

Pretty much yes. Mouse/touch moves can definitely fire at a faster rate than the frame rate. This is generally nice, as it gives a finer-resolution curve of where the pointer went.

It seems like it has performance implications we may need to address.

Depending on the simulation, it's usually better for maintenance when the view is updated for every user event (as necessary), but definitely for performance-critical parts it can be very helpful to delay expensive operations to the view step. Sometimes this is more convenient anyways (e.g. CanvasNode). Usually would use either the "things change every frame" strategy (e.g. RewardNode, particles, etc.), or the "dirty flag when something changes, change view only if it is dirty" method.

@samreid
Copy link
Member Author

samreid commented Jul 8, 2017

Thanks @jonathanolson, it is good to know. For CCK, I've batched some expensive operations to run in the view step. Not sure if we need to do anything else for this issue, not sure if others are aware of this behavior (though it may be good to know when performance optimizing for iPad2).

@pixelzoom thoughts?

@samreid samreid assigned pixelzoom and unassigned samreid Jul 8, 2017
@pixelzoom
Copy link
Contributor

Mouse/touch moves can definitely fire at a faster rate than the frame rate. This is generally nice, as it gives a finer-resolution curve of where the pointer went.

It may give finer resolution. But since the view is updated only once per frame, what is the benefit of having the sim process more the 1 drag event per frame? E.g. if N drag events occur during a frame, would the user see anything different if the sim received only the last of those drag events?

@pixelzoom pixelzoom assigned jonathanolson and unassigned pixelzoom Jul 10, 2017
@jonathanolson
Copy link
Contributor

E.g. if N drag events occur during a frame, would the user see anything different if the sim received only the last of those drag events?

For simplistic drags, this would be the case. However there are many other cases:

  • "Tracing" feature during Build a Molecule's cut draws the mouse's path. Extra points helpful here.
  • Up/down events can happen, and you want to make sure the pointer is in the correct place when that happens.
  • It lets touch-to-snag work. If you ignore touches in the middle, there's a good chance you will skip over a touch-to-snag object.
  • It makes fuzzing at faster rates possible.

If we're concerned about general performance for this, maybe it would be appropriate to bake in a "lazy" approach to drag handlers, where they ignore certain moves in a frame (and have a step() function that makes them tick). (This approach does have some tricky side-effects, where if the object isn't "dragging" with a pointer, it will probably be receiving a lot more enter/exit events)

@pixelzoom
Copy link
Contributor

pixelzoom commented Jul 10, 2017

For sims where there's no benefit to finer granularity (which I suspect is the majority of sims), it would be unfortunate to have to deal with this is client code in order to hit performance goals. Perhaps a scenery option that allows you to specify whether you want all events or just the "last" event per frame?

@samreid
Copy link
Member Author

samreid commented Jul 10, 2017

Also, my original report was for desktop Chrome--we should check the ratio of input events per frame on iPad2 before optimizing this.

@pixelzoom
Copy link
Contributor

pixelzoom commented Jul 10, 2017

For sims where there's no benefit to finer granularity ...

E.g. for function-builder, testing flagged performance as "OK" on iPad2, and "Poor" on iPad3 (see https://docs.google.com/spreadsheets/d/1crnipxdR5PEn8QBWQKMzPSA6sUMWdaBLRL0j8ZEzvEo/edit#gid=2). In function-builder, "performance" is solely drag performance, and drag handlers are the only performance-critical code. Had I known that multiple drags were being processed per frame, I likely could have improved performance, since this sim has no need for the granularity provided by multiple drag events per frame.

@pixelzoom pixelzoom assigned jonathanolson and unassigned pixelzoom Jul 10, 2017
@jonathanolson
Copy link
Contributor

For sims where there's no benefit to finer granularity

I thought the preference now was for most sims to include touch-snaggability, so there would be some benefit to most sims?

Wouldn't my proposal above (that handles this in the drag handler, so it can choose to ignore unneeded events and not disable it for the entire sim) be preferable?

we should check the ratio of input events per frame on iPad2 before optimizing this.

Agreed. I'll create a quick test that can be browsed without having to enable the console.

"Poor" on iPad3

The iPad 3 I thought had at least as much CPU, but many more pixels, so graphical operations would presumably be the bottleneck there?

jonathanolson added a commit that referenced this issue Jul 10, 2017
@jonathanolson
Copy link
Contributor

Pushed a test that can be loaded by visiting scenery/tests/frames-and-events.html

Every animation frame or move event adds (or changes) the next square. Blue is an animation frame, red is a move event.

Chrome canary example showing my moving my mouse (I get 1 event per frame whem moving):
scenery-events

@jonathanolson
Copy link
Contributor

iPad 2 test (Lave) shows predominantly 2 events per frame.

@jonathanolson
Copy link
Contributor

My normal desktop Chrome (Windows 7, Chrome 59) sends a lot of events:
scenery-events-chrome

OS X chrome sends 2 events per frame.
OS X Safari sends ~4 events per frame.

@pixelzoom
Copy link
Contributor

@jonathanolson asked:

Wouldn't my proposal above (that handles this in the drag handler, so it can choose to ignore unneeded events and not disable it for the entire sim) be preferable?

Yep. That's what I meant by "scenery option that allows you to specify whether you want all events or just the "last" event per frame" -- something baked into common code. To further clarify...

@samreid said:

For CCK, I've batched some expensive operations to run in the view step.

... and that's what I meant by "it would be unfortunate to have to deal with this is client code".

@jonathanolson
Copy link
Contributor

I'll plan to look into modifications we could use for SimpleDragHandler (that's predominantly used for almost all drags).

If that option is provided, presumably it would be required to step() the drag handler?

@jonathanolson
Copy link
Contributor

jonathanolson commented Jul 10, 2017

Also, note that the above tests weren't under heavy load. If that's the case, it's possible that it would fall back to 1 event per frame.

@pixelzoom
Copy link
Contributor

If that option is provided, presumably it would be required to step() the drag handler?

I'm not clear on this. Can you provide a hypothetical example?

@jonathanolson
Copy link
Contributor

I'm not clear on this. Can you provide a hypothetical example?

this.inputListener = new SimpleDragHandler( { ... } );
node.addInputListener( inputListener );
// ...
step: function( dt ) {
  this.inputListener.step(); // or another function name, since it has no arg?
}

@samreid
Copy link
Member Author

samreid commented Jul 12, 2017

I'm not clear on this. Can you provide a hypothetical example?

If I understand correctly, this is because scenery would have no way of knowing which input event is the last one so would need an explicit step call so it could activate the last event.

@jonathanolson
Copy link
Contributor

If I understand correctly, this is because scenery would have no way of knowing which input event is the last one so would need an explicit step call so it could activate the last event.

Correct. There may be potential to have a listener subtype in Joist that hooks this into the sim step, so that manually stepping isn't done.

@jonathanolson
Copy link
Contributor

Tagging for developer meeting, as the proposed way of handling this could use wider discussion.

@samreid
Copy link
Member Author

samreid commented Jul 12, 2017

There may be potential to have a listener subtype in Joist that hooks this into the sim step

That sounds like it is worth investigation.

@pixelzoom
Copy link
Contributor

My vote is to provide this feature for DragListener (next gen) and punt on SimpleDragHandler.

@samreid
Copy link
Member Author

samreid commented Jul 13, 2017

I agree it would be good feature for next gen DragListener so that it has an option to run one per frame or browser-default per frame.

@jonathanolson
Copy link
Contributor

Will be investigated as part of #131

@jonathanolson
Copy link
Contributor

Implemented basic support for this above, where you can pass collapseDragEvents:true and then call step() on the listener.

If we want to figure out better ways of setting up stepping of listeners (globally, hooking in, etc.), that should be discussed here.

Tested with interaction with the following playground code:

var rect1 = new scenery.Rectangle( 0, 0, 100, 50, { fill: 'red' } );
var rect2 = new scenery.Rectangle( 0, 0, 100, 50, { fill: 'blue', y: 150 } );

var listener1 = new scenery.DragListener( {
  translateNode: true,
  drag: () => console.log( 'drag1' )
} );
var listener2 = new scenery.DragListener( {
  translateNode: true,
  collapseDragEvents: true,
  drag: () => console.log( 'drag2' )
} );
rect1.addInputListener( listener1 );
rect2.addInputListener( listener2 );

scene.addChild( rect1 );
scene.addChild( rect2 );

display.initializeEvents();
( function step() {
  requestAnimationFrame( step );

  listener2.step();

  console.log( 'update' );

  display.updateDisplay();
} )();

With the above code, on MacOS Chrome I was getting 2 drags in-between steps for the red rectangle (1), and 1 drag in-between steps for the blue rectangle (2) with the new options set.

@samreid and @pixelzoom, should we discuss fancier hooks to stepping the listener, and/or does the above change work for you?

@pixelzoom
Copy link
Contributor

Wow, fun reviewing comments from 2+ years ago! I had no recollection whatsoever of this issue. Who is this @pixelzoom guy? :)

The collapseDragEvents option you've added seems sufficient to me. I can't identify a need to get fancier.

@pixelzoom pixelzoom removed their assignment Dec 20, 2019
@samreid
Copy link
Member Author

samreid commented Dec 23, 2019

Can you add documentation of when it is appropriate to specify that option?

@jonathanolson
Copy link
Contributor

Added documentation, good to close?

@samreid
Copy link
Member Author

samreid commented Jun 11, 2020

Looks good, thanks! Closing.

@samreid samreid closed this as completed Jun 11, 2020
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Projects
None yet
Development

No branches or pull requests

3 participants