react-interactive-svg
Version:
A react component that helps you make svg shapes interactives
227 lines (185 loc) • 17.5 kB
JavaScript
Object.defineProperty(exports, "__esModule", {
value: true
});
exports["default"] = void 0;
var _react = require("react");
var _propTypes = _interopRequireDefault(require("prop-types"));
function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { "default": obj }; }
function asyncGeneratorStep(gen, resolve, reject, _next, _throw, key, arg) { try { var info = gen[key](arg); var value = info.value; } catch (error) { reject(error); return; } if (info.done) { resolve(value); } else { Promise.resolve(value).then(_next, _throw); } }
function _asyncToGenerator(fn) { return function () { var self = this, args = arguments; return new Promise(function (resolve, reject) { var gen = fn.apply(self, args); function _next(value) { asyncGeneratorStep(gen, resolve, reject, _next, _throw, "next", value); } function _throw(err) { asyncGeneratorStep(gen, resolve, reject, _next, _throw, "throw", err); } _next(undefined); }); }; }
function _slicedToArray(arr, i) { return _arrayWithHoles(arr) || _iterableToArrayLimit(arr, i) || _nonIterableRest(); }
function _nonIterableRest() { throw new TypeError("Invalid attempt to destructure non-iterable instance"); }
function _iterableToArrayLimit(arr, i) { if (!(Symbol.iterator in Object(arr) || Object.prototype.toString.call(arr) === "[object Arguments]")) { return; } var _arr = []; var _n = true; var _d = false; var _e = undefined; try { for (var _i = arr[Symbol.iterator](), _s; !(_n = (_s = _i.next()).done); _n = true) { _arr.push(_s.value); if (i && _arr.length === i) break; } } catch (err) { _d = true; _e = err; } finally { try { if (!_n && _i["return"] != null) _i["return"](); } finally { if (_d) throw _e; } } return _arr; }
function _arrayWithHoles(arr) { if (Array.isArray(arr)) return arr; }
var SvgParser = function SvgParser(props) {
var _useState = (0, _react.useState)([]),
_useState2 = _slicedToArray(_useState, 2),
childNodes = _useState2[0],
setChildNodes = _useState2[1]; // Wait for ref to exist
(0, _react.useEffect)(function () {
// Check theres even a child
if (props.children) {
if (props.children.ref.current) {
var child = props.children.ref.current; // Check that the child node is SVG
if (child.nodeName !== 'svg') {
throw new Error('The child must be an svg element');
} // Get al child nodes
var nodes = child.childNodes;
var nodeIds = []; // set allowed nodes
var allowedNodes = props.allowedShapes; // Insert a unique id to all child nodes
var _iteratorNormalCompletion = true;
var _didIteratorError = false;
var _iteratorError = undefined;
try {
for (var _iterator = nodes.entries()[Symbol.iterator](), _step; !(_iteratorNormalCompletion = (_step = _iterator.next()).done); _iteratorNormalCompletion = true) {
var _step$value = _slicedToArray(_step.value, 2),
i = _step$value[0],
node = _step$value[1];
if (allowedNodes.includes(node.nodeName)) {
var nodeId = "svg-node-".concat(i);
node.id = nodeId;
nodeIds.push(nodeId);
}
}
} catch (err) {
_didIteratorError = true;
_iteratorError = err;
} finally {
try {
if (!_iteratorNormalCompletion && _iterator["return"] != null) {
_iterator["return"]();
}
} finally {
if (_didIteratorError) {
throw _iteratorError;
}
}
}
setChildNodes(nodeIds);
}
}
}, [props.children]); // Listen for nodes to be added
(0, _react.useEffect)(function () {
// onClick Function
var onElementClicked =
/*#__PURE__*/
function () {
var _ref = _asyncToGenerator(
/*#__PURE__*/
regeneratorRuntime.mark(function _callee(e) {
var change;
return regeneratorRuntime.wrap(function _callee$(_context) {
while (1) {
switch (_context.prev = _context.next) {
case 0:
_context.next = 2;
return props.onElementClicked(e);
case 2:
change = _context.sent;
if (change) {
e.target.style.fill = props.activeColor;
}
case 4:
case "end":
return _context.stop();
}
}
}, _callee);
}));
return function onElementClicked(_x) {
return _ref.apply(this, arguments);
};
}(); // On hover in function
var onElementHoverIn = function onElementHoverIn(e) {
e.target.style.stroke = props.hoverBorderColor;
props.onElementHover(e);
}; // On hover out function
var onElementHoverOut = function onElementHoverOut(e) {
e.target.style.stroke = props.defaultHoverBorder;
props.onElementHover(e);
}; // add event listeners
var _iteratorNormalCompletion2 = true;
var _didIteratorError2 = false;
var _iteratorError2 = undefined;
try {
for (var _iterator2 = childNodes[Symbol.iterator](), _step2; !(_iteratorNormalCompletion2 = (_step2 = _iterator2.next()).done); _iteratorNormalCompletion2 = true) {
var element = _step2.value;
var node = document.querySelector("#".concat(element));
node.style.fill = 'transparent';
node.style.cursor = 'pointer'; // Set click listener only if prop is received
if (props.onElementClicked) {
node.addEventListener('click', onElementClicked);
} // set hover listeners only if porp is received
if (props.onElementHover) {
node.addEventListener('mouseover', onElementHoverIn);
node.addEventListener('mouseout', onElementHoverOut);
}
}
} catch (err) {
_didIteratorError2 = true;
_iteratorError2 = err;
} finally {
try {
if (!_iteratorNormalCompletion2 && _iterator2["return"] != null) {
_iterator2["return"]();
}
} finally {
if (_didIteratorError2) {
throw _iteratorError2;
}
}
}
return function () {
// Remove event listeners
var _iteratorNormalCompletion3 = true;
var _didIteratorError3 = false;
var _iteratorError3 = undefined;
try {
for (var _iterator3 = childNodes[Symbol.iterator](), _step3; !(_iteratorNormalCompletion3 = (_step3 = _iterator3.next()).done); _iteratorNormalCompletion3 = true) {
var element = _step3.value;
var node = document.querySelector("#".concat(element)); // Set click listener only if prop is received
if (props.onElementClicked) {
node.addEventListener('click', onElementClicked);
}
if (props.onElementHover) {
node.removeEventListener('mouseover', onElementHoverIn);
node.removeEventListener('mouseout', onElementHoverOut);
}
}
} catch (err) {
_didIteratorError3 = true;
_iteratorError3 = err;
} finally {
try {
if (!_iteratorNormalCompletion3 && _iterator3["return"] != null) {
_iterator3["return"]();
}
} finally {
if (_didIteratorError3) {
throw _iteratorError3;
}
}
}
};
}, [childNodes, props]);
return props.children || null;
};
SvgParser.defaultProps = {
hoverBorderColor: '#009cff82',
defaultHoverBorder: 'black',
activeColor: '#009cff82',
allowedShapes: ['polygon', 'rect', 'circle']
};
SvgParser.propTypes = {
onElementHover: _propTypes["default"].func,
onElementClicked: _propTypes["default"].func,
hoverBorderColor: _propTypes["default"].string,
defaultHoverBorder: _propTypes["default"].string,
activeColor: _propTypes["default"].string,
children: _propTypes["default"].element.isRequired,
allowedShapes: _propTypes["default"].array
};
var _default = SvgParser;
exports["default"] = _default;
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../src/index.js"],"names":[],"mappings":"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","file":"lib.dist.js","sourcesContent":["import { useEffect, useState } from 'react'\r\nimport PropTypes from 'prop-types'\r\n\r\nconst SvgParser = (props) => {\r\n  const [childNodes, setChildNodes] = useState([])\r\n\r\n  // Wait for ref to exist\r\n  useEffect(() => {\r\n    // Check theres even a child\r\n    if (props.children) {\r\n      if (props.children.ref.current) {\r\n        const child = props.children.ref.current\r\n        // Check that the child node is SVG\r\n        if (child.nodeName !== 'svg') {\r\n          throw new Error('The child must be an svg element')\r\n        }\r\n\r\n        // Get al child nodes\r\n        const nodes = child.childNodes\r\n        const nodeIds = []\r\n\r\n        // set allowed nodes\r\n        const allowedNodes = props.allowedShapes\r\n\r\n        // Insert a unique id to all child nodes\r\n        for (const [i, node] of nodes.entries()) {\r\n          if (allowedNodes.includes(node.nodeName)) {\r\n            const nodeId = `svg-node-${i}`\r\n            node.id = nodeId\r\n            nodeIds.push(nodeId)\r\n          }\r\n        }\r\n\r\n        setChildNodes(nodeIds)\r\n      }\r\n    }\r\n  }, [props.children])\r\n\r\n  // Listen for nodes to be added\r\n  useEffect(() => {\r\n    // onClick Function\r\n    const onElementClicked = async (e) => {\r\n      // set background color of element\r\n      const change = await props.onElementClicked(e)\r\n      if (change) {\r\n        e.target.style.fill = props.activeColor\r\n      }\r\n    }\r\n    // On hover in function\r\n    const onElementHoverIn = (e) => {\r\n      e.target.style.stroke = props.hoverBorderColor\r\n      props.onElementHover(e)\r\n    }\r\n    // On hover out function\r\n    const onElementHoverOut = (e) => {\r\n      e.target.style.stroke = props.defaultHoverBorder\r\n      props.onElementHover(e)\r\n    }\r\n    // add event listeners\r\n    for (const element of childNodes) {\r\n      const node = document.querySelector(`#${element}`)\r\n      node.style.fill = 'transparent'\r\n      node.style.cursor = 'pointer'\r\n\r\n      // Set click listener only if prop is received\r\n      if (props.onElementClicked) {\r\n        node.addEventListener('click', onElementClicked)\r\n      }\r\n\r\n      // set hover listeners only if porp is received\r\n      if (props.onElementHover) {\r\n        node.addEventListener('mouseover', onElementHoverIn)\r\n        node.addEventListener('mouseout', onElementHoverOut)\r\n      }\r\n    }\r\n\r\n    return () => {\r\n      // Remove event listeners\r\n      for (const element of childNodes) {\r\n        const node = document.querySelector(`#${element}`)\r\n        // Set click listener only if prop is received\r\n        if (props.onElementClicked) {\r\n          node.addEventListener('click', onElementClicked)\r\n        }\r\n        if (props.onElementHover) {\r\n          node.removeEventListener('mouseover', onElementHoverIn)\r\n          node.removeEventListener('mouseout', onElementHoverOut)\r\n        }\r\n      }\r\n    }\r\n  }, [childNodes, props])\r\n\r\n  return props.children || null\r\n}\r\n\r\nSvgParser.defaultProps = {\r\n  hoverBorderColor: '#009cff82',\r\n  defaultHoverBorder: 'black',\r\n  activeColor: '#009cff82',\r\n  allowedShapes: ['polygon', 'rect', 'circle']\r\n}\r\n\r\nSvgParser.propTypes = {\r\n  onElementHover: PropTypes.func,\r\n  onElementClicked: PropTypes.func,\r\n  hoverBorderColor: PropTypes.string,\r\n  defaultHoverBorder: PropTypes.string,\r\n  activeColor: PropTypes.string,\r\n  children: PropTypes.element.isRequired,\r\n  allowedShapes: PropTypes.array\r\n}\r\n\r\nexport default SvgParser\r\n"]}
;