react-flashlight
Version:
A flashlight effect for React
205 lines (185 loc) • 27 kB
JavaScript
Object.defineProperty(exports, "__esModule", {
value: true
});
exports["default"] = ReactFlashlight;
var _react = _interopRequireDefault(require("react"));
var _propTypes = _interopRequireDefault(require("prop-types"));
var _resizeObserverPolyfill = _interopRequireDefault(require("resize-observer-polyfill"));
function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { "default": obj }; }
/**
*
* @param {HTMLElement} children The component's children
* @param {bool} showCursor If true, shows the cursor
* @param {number} size the initial size of the light
* @param {object} initialPosition An object {x: value, y: value} defining the initial position
* @param {object} moveTo An object {x: value, y: value} defining the location to where the light will be moved
* @param {number} speed Defines the transition speed of the movement of the light
* @param {bool} contain If true, the light can't move outside of the container
* @param {bool} enableMouse If true, the user can control the light with its mouse
* @param {number} darkness Defines how dark is the "room"
*/
function ReactFlashlight(props) {
var enabled = props.enabled,
children = props.children,
showCursor = props.showCursor,
size = props.size,
initialPosition = props.initialPosition,
moveTo = props.moveTo,
speed = props.speed,
enableMouse = props.enableMouse,
darkness = props.darkness;
var lightStyle = {
position: "absolute",
top: 0,
left: 0,
// To control the size of the light, simply use a percentage on the background creating the effect - init with size
background: "radial-gradient(transparent 0%, rgba(0, 0, 0, " + darkness + ") " + size + "px, rgba(0, 0, 0, " + (darkness + 0.1) + ") 80%)",
transition: "none",
pointerEvents: "none",
willChange: "transform"
};
var elements = _react["default"].Children.map(children, function (child) {
return {
light: _react["default"].useRef(),
container: _react["default"].useRef()
};
});
_react["default"].useEffect(function () {
var last_known_scroll_position = 0;
var ticking = false;
elements.forEach(function (element) {
var container = element.container.current;
container.style.overflow = "hidden";
container.style.position = "relative";
if (!showCursor) container.style.cursor = "none";
}); // Resizes the light
function resizeLights() {
elements.forEach(function (element) {
resizeLight(element);
});
}
function resizeLight(element) {
var light = element.light.current;
var maskSize = window.innerWidth > window.innerHeight ? window.innerWidth : window.innerHeight;
light.style.width = maskSize * 2 + "px";
light.style.height = maskSize * 2 + "px";
light.style.left = initialPosition.x - maskSize + "px";
light.style.top = initialPosition.y - maskSize + "px";
}
function handleMouseMove(e) {
if (!ticking) {
window.requestAnimationFrame(function () {
ticking = false;
elements.forEach(function (element) {
var light = element.light.current;
var container = element.container.current;
var lightStyle = window.getComputedStyle(light, null);
var containerStyle = container.getBoundingClientRect();
light.style.transition = "opacity ease-in-out " + speed + "ms";
light.style.left = e.clientX - containerStyle.left - parseInt(lightStyle.width) / 2 + "px";
light.style.top = e.clientY - containerStyle.top - parseInt(lightStyle.height) / 2 + "px";
});
});
ticking = true;
}
}
function handleScroll(e) {
var increment = window.scrollY - last_known_scroll_position;
last_known_scroll_position = window.scrollY;
if (!ticking) {
window.requestAnimationFrame(function () {
ticking = false;
elements.forEach(function (element) {
var light = element.light.current;
light.style.transition = "opacity ease-in-out " + speed + "ms";
light.style.top = parseInt(light.style.top) + increment + "px";
});
});
ticking = true;
}
}
resizeLights();
var resizeObservers = [children.length];
elements.forEach(function (element, i) {
resizeObservers[i] = new _resizeObserverPolyfill["default"](function () {
return resizeLights();
});
});
if (enableMouse) window.addEventListener("mousemove", handleMouseMove);
if (enableMouse) window.addEventListener('scroll', handleScroll);
window.addEventListener("resize", resizeLights);
elements.forEach(function (element, i) {
resizeObservers[i].observe(element.container.current);
}); // Cleanup
return function () {
if (enableMouse) window.removeEventListener("mousemove", handleMouseMove);
if (enableMouse) window.removeEventListener("scroll", handleScroll);
window.removeEventListener("resize", resizeLights);
elements.forEach(function (element, i) {
resizeObservers[i].disconnect;
});
};
}, []);
/**
* This is executed when moveTo props change
*/
_react["default"].useEffect(function () {
if (moveTo) {
elements.forEach(function (element) {
var light = element.light.current;
light.style.transition = "all ease-in-out " + speed + "ms";
var lightStyle = window.getComputedStyle(light, null);
light.style.left = moveTo.x - parseInt(lightStyle.width) / 2 + "px";
light.style.top = moveTo.y - parseInt(lightStyle.height) / 2 + "px";
});
}
}, [moveTo]);
_react["default"].useEffect(function () {
elements.forEach(function (element) {
var light = element.light.current;
light.style.transition = "opacity ease-in-out " + speed + "ms";
enabled ? light.style.opacity = "1" : light.style.opacity = "0";
});
}, [enabled]);
return _react["default"].Children.map(children, function (child, i) {
return _react["default"].cloneElement(child, {
ref: elements[i].container,
children: _react["default"].createElement(_react["default"].Fragment, null, _react["default"].createElement("div", {
"data-testid": "react-flashlight",
style: lightStyle,
ref: elements[i].light
}), child.props.children)
});
});
}
ReactFlashlight.propTypes = {
enabled: _propTypes["default"].bool,
children: _propTypes["default"].node,
showCursor: _propTypes["default"].bool,
size: _propTypes["default"].number,
initialPosition: _propTypes["default"].shape({
x: _propTypes["default"].number,
y: _propTypes["default"].number
}),
moveTo: _propTypes["default"].object,
speed: _propTypes["default"].number,
contain: _propTypes["default"].bool,
enableMouse: _propTypes["default"].bool,
darkness: _propTypes["default"].number
};
ReactFlashlight.defaultProps = {
enabled: true,
children: _react["default"].createElement("div", null),
showCursor: false,
size: 150,
initialPosition: {
x: 0,
y: 0
},
moveTo: null,
speed: 1000,
enableMouse: true,
darkness: 0.9
};
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../src/ReactFlashlight.js"],"names":["ReactFlashlight","props","enabled","children","showCursor","size","initialPosition","moveTo","speed","enableMouse","darkness","lightStyle","position","top","left","background","transition","pointerEvents","willChange","elements","React","Children","map","child","light","useRef","container","useEffect","last_known_scroll_position","ticking","forEach","element","current","style","overflow","cursor","resizeLights","resizeLight","maskSize","window","innerWidth","innerHeight","width","height","x","y","handleMouseMove","e","requestAnimationFrame","getComputedStyle","containerStyle","getBoundingClientRect","clientX","parseInt","clientY","handleScroll","increment","scrollY","resizeObservers","length","i","ResizeObserver","addEventListener","observe","removeEventListener","disconnect","opacity","cloneElement","ref","propTypes","PropTypes","bool","node","number","shape","object","contain","defaultProps"],"mappings":"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","sourcesContent":["import React from \"react\";\r\nimport PropTypes from \"prop-types\";\r\nimport ResizeObserver from 'resize-observer-polyfill';\r\n\r\n/**\r\n * \r\n * @param {HTMLElement} children The component's children\r\n * @param {bool} showCursor If true, shows the cursor\r\n * @param {number} size the initial size of the light\r\n * @param {object} initialPosition An object {x: value, y: value} defining the initial position\r\n * @param {object} moveTo An object {x: value, y: value} defining the location to where the light will be moved\r\n * @param {number} speed Defines the transition speed of the movement of the light\r\n * @param {bool} contain If true, the light can't move outside of the container\r\n * @param {bool} enableMouse If true, the user can control the light with its mouse\r\n * @param {number} darkness Defines how dark is the \"room\"\r\n */\r\n\r\nexport default function ReactFlashlight(props) {\r\n\r\n  const {\r\n    enabled,\r\n    children,\r\n    showCursor,\r\n    size,\r\n    initialPosition,\r\n    moveTo,\r\n    speed,\r\n    enableMouse,\r\n    darkness,\r\n  } = props; \r\n\r\n  const lightStyle = {\r\n    position: \"absolute\",\r\n    top: 0,\r\n    left: 0,\r\n    // To control the size of the light, simply use a percentage on the background creating the effect - init with size\r\n    background: \"radial-gradient(transparent 0%, rgba(0, 0, 0, \" + darkness + \") \" + size + \"px, rgba(0, 0, 0, \" + (darkness + 0.1) + \") 80%)\",\r\n    transition: \"none\",\r\n    pointerEvents: \"none\",\r\n    willChange: \"transform\"\r\n  }\r\n\r\n  const elements = React.Children.map(children, child => ({\r\n    light: React.useRef(),\r\n    container: React.useRef(),\r\n  }))\r\n\r\n  React.useEffect(()=>{\r\n\r\n    let last_known_scroll_position = 0;\r\n    let ticking = false;\r\n\r\n    elements.forEach(element => {\r\n      const container = element.container.current;\r\n      container.style.overflow = \"hidden\";\r\n      container.style.position = \"relative\";\r\n      if (!showCursor) container.style.cursor = \"none\";\r\n    })\r\n\r\n    // Resizes the light\r\n\r\n    function resizeLights() {\r\n      elements.forEach(element => {\r\n        resizeLight(element);\r\n      })\r\n    }\r\n\r\n    function resizeLight(element) {\r\n      const light = element.light.current;\r\n      const maskSize = window.innerWidth > window.innerHeight ? window.innerWidth : window.innerHeight\r\n\r\n      light.style.width = maskSize * 2 + \"px\";\r\n      light.style.height = maskSize * 2 + \"px\";\r\n  \r\n      light.style.left = initialPosition.x - maskSize + \"px\";\r\n      light.style.top = initialPosition.y - maskSize + \"px\";\r\n    }\r\n\r\n    function handleMouseMove(e) {\r\n      if (!ticking) {\r\n        window.requestAnimationFrame(function() {\r\n          ticking = false;\r\n          elements.forEach(element => {\r\n            const light = element.light.current;\r\n            const container = element.container.current;\r\n            const lightStyle = window.getComputedStyle(light, null);\r\n            const containerStyle = container.getBoundingClientRect();\r\n            light.style.transition = \"opacity ease-in-out \" + speed + \"ms\";\r\n            light.style.left = e.clientX - containerStyle.left - parseInt(lightStyle.width) / 2 + \"px\";\r\n            light.style.top = e.clientY - containerStyle.top - parseInt(lightStyle.height) / 2 + \"px\";\r\n          })\r\n        });\r\n        ticking = true;\r\n      }\r\n    }\r\n\r\n    function handleScroll(e) {\r\n      let increment = window.scrollY - last_known_scroll_position;\r\n      last_known_scroll_position = window.scrollY;\r\n      if (!ticking) {\r\n        window.requestAnimationFrame(function() {\r\n          ticking = false;\r\n          elements.forEach(element => {\r\n            const light = element.light.current;\r\n            light.style.transition = \"opacity ease-in-out \" + speed + \"ms\";\r\n            light.style.top = parseInt(light.style.top) + increment + \"px\";\r\n          })\r\n        });\r\n        ticking = true;\r\n      }\r\n    }\r\n\r\n    resizeLights();\r\n\r\n    const resizeObservers = [children.length];\r\n\r\n    elements.forEach((element, i) => {\r\n      resizeObservers[i] = new ResizeObserver(()=>resizeLights());\r\n    });\r\n\r\n    if (enableMouse) window.addEventListener(\"mousemove\", handleMouseMove);\r\n    if (enableMouse) window.addEventListener('scroll', handleScroll);\r\n    window.addEventListener(\"resize\", resizeLights);\r\n    elements.forEach((element, i) => {\r\n      resizeObservers[i].observe(element.container.current);\r\n    });\r\n\r\n    // Cleanup\r\n    return ()=>{\r\n      if (enableMouse) window.removeEventListener(\"mousemove\", handleMouseMove);\r\n      if (enableMouse) window.removeEventListener(\"scroll\", handleScroll);\r\n      window.removeEventListener(\"resize\", resizeLights)\r\n      elements.forEach((element, i) => {\r\n        resizeObservers[i].disconnect;\r\n      });\r\n    }\r\n\r\n  }, []);\r\n\r\n  /**\r\n   * This is executed when moveTo props change\r\n   */\r\n\r\n  React.useEffect(()=>{\r\n\r\n    if (moveTo) {\r\n      elements.forEach(element => {\r\n        const light = element.light.current;\r\n        light.style.transition = \"all ease-in-out \" + speed + \"ms\";\r\n        const lightStyle = window.getComputedStyle(light, null);\r\n        light.style.left = moveTo.x - parseInt(lightStyle.width) / 2 + \"px\";\r\n        light.style.top = moveTo.y - parseInt(lightStyle.height) / 2 + \"px\";\r\n      })\r\n    }\r\n  }, [moveTo])\r\n\r\n  React.useEffect(()=>{\r\n    elements.forEach(element => {\r\n      const light = element.light.current;\r\n      light.style.transition = \"opacity ease-in-out \" + speed + \"ms\";\r\n      enabled ? light.style.opacity = \"1\" : light.style.opacity = \"0\";\r\n    });\r\n\r\n  }, [enabled])\r\n\r\n  return ( \r\n    React.Children.map(children, (child, i) =>\r\n      React.cloneElement(child,\r\n        {\r\n          ref: elements[i].container,\r\n          children:\r\n            <>\r\n              <div\r\n                data-testid=\"react-flashlight\"\r\n                style={lightStyle}\r\n                ref={elements[i].light}\r\n              />\r\n              {child.props.children}\r\n            </>\r\n        }) \r\n    )\r\n  )\r\n}\r\n\r\nReactFlashlight.propTypes = {\r\n  enabled: PropTypes.bool,\r\n  children: PropTypes.node,\r\n  showCursor: PropTypes.bool,\r\n  size: PropTypes.number,\r\n  initialPosition: PropTypes.shape({\r\n    x: PropTypes.number,\r\n    y: PropTypes.number,\r\n  }),\r\n  moveTo: PropTypes.object,\r\n  speed: PropTypes.number,\r\n  contain: PropTypes.bool,\r\n  enableMouse: PropTypes.bool,\r\n  darkness: PropTypes.number,\r\n}\r\n\r\nReactFlashlight.defaultProps = {\r\n  enabled: true,\r\n  children: <div></div>,\r\n  showCursor: false,\r\n  size: 150,\r\n  initialPosition: {x: 0, y: 0},\r\n  moveTo: null,\r\n  speed: 1000,\r\n  enableMouse: true,\r\n  darkness: 0.9,\r\n} "]}
;