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

Using passive event handlers #35

Closed
ernest-okot opened this issue Jun 20, 2017 · 15 comments
Closed

Using passive event handlers #35

ernest-okot opened this issue Jun 20, 2017 · 15 comments

Comments

@ernest-okot
Copy link

Related to d3-selection#113, how can we use passive event handlers for drag events?

@jrencz
Copy link

jrencz commented Jul 12, 2017

@ernest-okot listeners added via drag.on will accept 3rd argument as you may see in drag.js@L147

The problem is that drag calls selection.on internally: namely: inside the drag function.

Until we get a better API for what's there inside drag function you may be interested in the following patch:

DON'T USE THIS TECHNIQUE. IT'S HERE FOR A REFERENCE. SEE COMMENTS BELOW

const clone = selection.sort(); 

// Not sure if it's the best way to clone a selection. I found no other.
// Cloning to prevent exposing patched `on` to the rest of the codebase

clone.on = (() => {
  const originalSelectionOn = selection.on.bind(selection);
  return (name, fn, options = {passive: true}) =>
    originalSelectionOn(name, fn, options);
})();

clone.call(
  drag()
    .on('start', () => {}, {passive: true})
    .on('drag', () => {}, {passive: true})
    .on('drag', () => {}, {passive: true})
);

This way I got rid of all [Violation] warnings Chrome Canary showed me: both those that I could pass the 3rd argument (drag.on) and those what drag calls internally.

@mbostock
Copy link
Member

Passive listeners are not appropriate for the drag behavior as it must cancel scrolling.

@jrencz
Copy link

jrencz commented Jul 12, 2017

yeah, I noticed that, especially on mobile, it broke some things.

@mbostock do all handlers drag set up needs to be active or just some of them?

@mbostock
Copy link
Member

You can see the table of events in the README. My guess is that the Violation warnings occur when you have a non-passive event listener that does not call event.preventDefault. Actually, it does occur to me that it might be possible to make the mousedown and touchstart event listeners passive, since due to the convoluted way in which these events need to be handled, the default behavior on these events is not prevented even though mousemove and touchmove are subsequently prevented.

@jrencz
Copy link

jrencz commented Jul 12, 2017

So, as I understand, you will think about making some of the internally attached handlers passive, but it's not as easy as saying "this one can be passive and that one not".

Then my patch snippet definitely does more harm than good. I'll leave a remark in the original comment in case someone finds it later.

@mbostock
Copy link
Member

It seems that the Violation appears in the touchstart and touchmove event listeners in Chrome on macOS, even though those event listeners are never invoked (because it’s not a touch device). That seems unfortunate, since it implies that you get a warning even if your event listeners are necessarily non-passive. It might be possible to make the touchstart listener passive, but it’s not possible to make touchmove passive as we need to call event.preventDefault.

Also, per #9, I’d rather always call event.preventDefault. It’s just I can’t because it breaks capturing of mouseup outside the window and it breaks click emulation on touch devices.

@jrencz
Copy link

jrencz commented Jul 12, 2017

Thanks for the briefing

@mbostock
Copy link
Member

mbostock commented Jul 12, 2017

This might be fixable with Pointer Events (#25) but browser support is still around 60% (behind a flag Firefox, and missing from Safari and iOS).

@jrencz
Copy link

jrencz commented Jul 12, 2017

Have you already considered using pointer events when they're available with a fallback to mouse events? I'm just thinking out loud, I'm not aware of if it's worth maintaining that or how complex may it become

@mbostock
Copy link
Member

This is already some of the most complex and issue-prone code in D3, so I am wary of adding another code path for a subset of browsers.

Reading a little more about passive event listeners, it seems that you can have a passive event listener that cancels scroll by using the touch-action style property. So perhaps we can make them passive after all?

@jrencz
Copy link

jrencz commented Jul 12, 2017

Too bad it's again Safari that does not respect that

@mbostock
Copy link
Member

Okay. I think I have a good fix for the warning: don’t register touch event listeners on devices that don’t support touch.

mbostock added a commit that referenced this issue Jul 12, 2017
This avoids a [Violation] warning in Chrome (#35).
mbostock added a commit to d3/d3-zoom that referenced this issue Jul 12, 2017
This avoids a [Violation] warning in Chrome.

Related #99 #103 d3/d3-drag#35.
@jrencz
Copy link

jrencz commented Jul 12, 2017

What about devices like MS Surface which may support touch in general but user has other options?

generally you will have 'ontouchstart' in window === true on Surface but it's not certain that the user will ever use touch. The current mean of interaction should also be detected (more or less like this):

let currentMeanOfInteractionIsTouch = false;

window.addEventListener('touchstart', () => {
  currentMeanOfInteractionIsTouch = true;
  // attach other touch event listeners
}, {passive: true})

window.addEventListener('mousedown', () => {
  currentMeanOfInteractionIsTouch = false;
  // remove touch event listeners but the one above, attach mouse
}, {passive: true})

I just can't tell up front if attaching events on demand won't be either too late or too expensive. How do you think?

@Herst
Copy link
Contributor

Herst commented Jul 28, 2017

@jrencz On convertible devices like the Surface you could also try to detect whether it is in tablet mode, e.g. see the discussion at https://stackoverflow.com/questions/32493093, but neither the solutions from there nor your solution belong inside d3 IMHO since it's too use-case specific.

On a side note, I came across this and related bug tracker entries while looking for solutions for issues I was having with pan and drag-drop with d3 in Microsoft Edge and IE11 even in demos from @mbostock. There I learned touch-action with which I was able to solve the problems by simply disallowing any touch action except for pinch-zoom inside the SVG, apparently the preventDefault method isn't working in these browsers.

@jrencz
Copy link

jrencz commented Jul 28, 2017

@Herst I do own Surface and I barely use tablet mode. Yet I tend to touch the screen a lot in desktop mode. It serves a purpose of lightweight task home laptop, not a tablet at all in my household. I'd bet I'm not alone in this mode of usage.

So detecting the mode (btw: how can it be done in JS?) is not the best choice IMO

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
None yet
Development

No branches or pull requests

4 participants