From b62e25d537c9a6cfb8b2004ad898b6bb0635dfd8 Mon Sep 17 00:00:00 2001 From: Ghislain B Date: Fri, 12 May 2023 21:42:37 -0400 Subject: [PATCH] fix(core): set wheel/touch listeners to passive for better perf (#769) - this fixes warnings shown in Chrome and other browser console mentioning that we should consider using `passive` event listeners - also uses a polyfill in case the `passive` option is not supported (for example IE) --- slick.core.js | 28 ++++++++++++++++++++++++++++ slick.interactions.js | 20 ++++++++++---------- 2 files changed, 38 insertions(+), 10 deletions(-) diff --git a/slick.core.js b/slick.core.js index d8101b04..62b748d6 100644 --- a/slick.core.js +++ b/slick.core.js @@ -796,6 +796,32 @@ return Object.entries(obj).length === 0; } + /** + * Check if `passive` option is supported when adding event listener, follows detection provided in MDN: + * https://developer.mozilla.org/en-US/docs/Web/API/EventTarget/addEventListener#safely_detecting_option_support + */ + function passiveSupported() { + let passiveSupported = false; + + try { + const options = { + get passive() { + passiveSupported = true; + return false; + }, + }; + window.addEventListener('test', null, options); + window.removeEventListener('test', null, options); + } catch (err) { + passiveSupported = false; + } + return passiveSupported; + } + + function enablePassiveWhenSupported() { + return passiveSupported() ? { passive: true } : false + } + function noop() { } function offset(el) { @@ -1063,6 +1089,8 @@ "calculateAvailableSpace": calculateAvailableSpace, "createDomElement": createDomElement, "emptyElement": emptyElement, + "passiveSupported": passiveSupported, + "enablePassiveWhenSupported": enablePassiveWhenSupported, "innerSize": innerSize, "isEmptyObject": isEmptyObject, "noop": noop, diff --git a/slick.interactions.js b/slick.interactions.js index 883b2fbc..51be8426 100644 --- a/slick.interactions.js +++ b/slick.interactions.js @@ -12,7 +12,7 @@ * code refs: * https://betterprogramming.pub/perfecting-drag-and-drop-in-pure-vanilla-javascript-a761184b797a * available optional options: - * - containerElement: container DOM element, defaults to "document" + * - containerElement: container DOM element, defaults to "document" * - allowDragFrom: when defined, only allow dragging from an element that matches a specific query selector * - onDragInit: drag initialized callback * - onDragStart: drag started callback @@ -40,7 +40,7 @@ if (containerElement) { containerElement.addEventListener('mousedown', userPressed); - containerElement.addEventListener('touchstart', userPressed); + containerElement.addEventListener('touchstart', userPressed, Slick.Utils.enablePassiveWhenSupported()); } function executeDragCallbackWhenDefined(callback, e, dd) { @@ -52,7 +52,7 @@ function destroy() { if (containerElement) { containerElement.removeEventListener('mousedown', userPressed); - containerElement.removeEventListener('touchstart', userPressed); + containerElement.removeEventListener('touchstart', userPressed, Slick.Utils.enablePassiveWhenSupported()); } } @@ -130,13 +130,13 @@ let { element, onMouseWheel } = options; function destroy() { - element.removeEventListener('wheel', wheelHandler, false); - element.removeEventListener('mousewheel', wheelHandler, false); + element.removeEventListener('wheel', wheelHandler, Slick.Utils.enablePassiveWhenSupported()); + element.removeEventListener('mousewheel', wheelHandler, Slick.Utils.enablePassiveWhenSupported()); } function init() { - element.addEventListener('wheel', wheelHandler, false); - element.addEventListener('mousewheel', wheelHandler, false); + element.addEventListener('wheel', wheelHandler, Slick.Utils.enablePassiveWhenSupported()); + element.addEventListener('mousewheel', wheelHandler, Slick.Utils.enablePassiveWhenSupported()); } // copy over the same event handler code used in jquery.mousewheel @@ -191,7 +191,7 @@ * - onResizeStart: resize start callback * - onResize: resizing callback * - onResizeEnd: resize ended callback - * @param {Object} options + * @param {Object} options * @returns - Resizable instance which includes destroy method * @class Resizable */ @@ -204,7 +204,7 @@ function destroy() { if (resizeableHandleElement && typeof resizeableHandleElement.removeEventListener === 'function') { resizeableHandleElement.removeEventListener('mousedown', resizeStartHandler); - resizeableHandleElement.removeEventListener('touchstart', resizeStartHandler); + resizeableHandleElement.removeEventListener('touchstart', resizeStartHandler, Slick.Utils.enablePassiveWhenSupported()); } } @@ -247,7 +247,7 @@ // add event listeners on the draggable element resizeableHandleElement.addEventListener('mousedown', resizeStartHandler); - resizeableHandleElement.addEventListener('touchstart', resizeStartHandler); + resizeableHandleElement.addEventListener('touchstart', resizeStartHandler, Slick.Utils.enablePassiveWhenSupported()); return { destroy }; }