UNPKG

apeman-react-touchable

Version:
218 lines (186 loc) 18.4 kB
/** * Higher order component with touching feature * @function withTouch * @param {function} Component - A component constructor * @returns {function} - Wrapped component */ 'use strict'; Object.defineProperty(exports, "__esModule", { value: true }); var _toConsumableArray2 = require('babel-runtime/helpers/toConsumableArray'); var _toConsumableArray3 = _interopRequireDefault(_toConsumableArray2); var _assign = require('babel-runtime/core-js/object/assign'); var _assign2 = _interopRequireDefault(_assign); var _getIterator2 = require('babel-runtime/core-js/get-iterator'); var _getIterator3 = _interopRequireDefault(_getIterator2); var _keys = require('babel-runtime/core-js/object/keys'); var _keys2 = _interopRequireDefault(_keys); var _breact = require('breact'); var _react = require('react'); var _react2 = _interopRequireDefault(_react); var _reactDom = require('react-dom'); var _reactDom2 = _interopRequireDefault(_reactDom); var _asobj = require('asobj'); var _enums = require('./enums'); var _new_hammer = require('./helpers/new_hammer'); var _new_hammer2 = _interopRequireDefault(_new_hammer); function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; } var propTypes = { /** Handler for "tap" event */ onTap: _react.PropTypes.func, /** Handler for "doubletap" event */ onDoubleTap: _react.PropTypes.func, /** Handler for "pan" event */ onPan: _react.PropTypes.func, /** Handler for "panstart" event */ onPanStart: _react.PropTypes.func, /** Handler for "panmove" event */ onPanMove: _react.PropTypes.func, /** Handler for "panend" event */ onPanEnd: _react.PropTypes.func, /** Handler for "pancancel" event */ onPanCancel: _react.PropTypes.func, /** Handler for "swipe" event */ onSwipe: _react.PropTypes.func, /** Handler for "press" event */ onPress: _react.PropTypes.func, /** Handler for "pressup" event */ onPressUp: _react.PropTypes.func, /** Handler for "pinch" event */ onPinch: _react.PropTypes.func, /** Handler for "pinchstart" event */ onPinchStart: _react.PropTypes.func, /** Handler for "pinchmove" event */ onPinchMove: _react.PropTypes.func, /** Handler for "pinchend" event */ onPinchEnd: _react.PropTypes.func, /** Handler for "pinchcancel" event */ onPinchCancel: _react.PropTypes.func, /** Handler for "rotate" event */ onRotate: _react.PropTypes.func, /** Handler for "rotatestart" event */ onRotateStart: _react.PropTypes.func, /** Handler for "rotatemove" event */ onRotateMove: _react.PropTypes.func, /** Handler for "rotateend" event */ onRotateEnd: _react.PropTypes.func, /** Handler for "rotatecancel" event */ onRotateCancel: _react.PropTypes.func, /** Options for touch */ touchOptions: _react.PropTypes.object, /** Pixcel for pan threshold */ panThreshold: _react.PropTypes.number, /** Direction for pan */ panDirection: _react.PropTypes.oneOf(_enums.directions), /** Direction for swipe */ swipeDirection: _react.PropTypes.oneOf(_enums.directions) }; function createTouchHammer(node, props, getData) { if (!node) { return; } var hammer = (0, _new_hammer2.default)(node); if (props.touchOptions) { hammer.set(props.touchOptions); } (0, _keys2.default)(propTypes).filter(function (key) { return !!props[key]; }).forEach(function (key) { if (/^on/.test(key)) { (function () { var event = key.replace(/^on/, '').toLowerCase(); var handler = props[key]; hammer.on(event, function (e) { e.data = getData(); // Set touching data. handler(e); }); })(); } if (/Direction$/.test(key)) { var gesture = key.replace(/Direction$/, ''); hammer.get(gesture).set({ direction: hammer.Hammer['DIRECTION_' + props[key]] }); } if (/Threshold$/.test(key)) { var threshold = key.replace(/Threshold$/, ''); hammer.get(threshold).set({ threshold: props[key] }); } }); return hammer; } function destroyTouchHammer(hammer) { if (!hammer) { return; } hammer.stop(); hammer.destroy(); } function supportsTouch(props) { var touchableKey = (0, _keys2.default)(propTypes); var _iteratorNormalCompletion = true; var _didIteratorError = false; var _iteratorError = undefined; try { for (var _iterator = (0, _getIterator3.default)((0, _keys2.default)(props)), _step; !(_iteratorNormalCompletion = (_step = _iterator.next()).done); _iteratorNormalCompletion = true) { var key = _step.value; var hit = !!~touchableKey.indexOf(key); if (hit) { return true; } } } catch (err) { _didIteratorError = true; _iteratorError = err; } finally { try { if (!_iteratorNormalCompletion && _iterator.return) { _iterator.return(); } } finally { if (_didIteratorError) { throw _iteratorError; } } } return false; } /** @lends withTouch */ function withTouch(Component) { var config = arguments.length <= 1 || arguments[1] === undefined ? {} : arguments[1]; var getTouchData = config.getTouchData; var spec = (0, _assign2.default)({ displayName: 'withTouch', propTypes: propTypes, render: function render() { var s = this; var props = s.props; var wrappedProps = (0, _asobj.clone)(props, { without: (0, _keys2.default)(propTypes || {}) }); wrappedProps.hasTouch = !!props.onTap; var children = _react2.default.Children.toArray(props.children); return _react2.default.createElement.apply(_react2.default, [Component, wrappedProps].concat((0, _toConsumableArray3.default)(children))); }, componentDidMount: function componentDidMount() { var s = this; var touchable = supportsTouch(s.props); var getData = function getData() { return s.getTouchData ? s.getTouchData() : undefined; }; if (touchable) { s.touchHammer = createTouchHammer(_reactDom2.default.findDOMNode(s), s.props, getData); } }, componentWillUnmount: function componentWillUnmount() { var s = this; var hammer = s.touchHammer; if (hammer) { destroyTouchHammer(hammer); } delete s.touchHammer; } }, { getTouchData: getTouchData }); return (0, _breact.wrap)(Component, spec); } exports.default = withTouch; //# sourceMappingURL=data:application/json;base64,{"version":3,"sources":["with_touch.jsx"],"names":["propTypes","onTap","func","onDoubleTap","onPan","onPanStart","onPanMove","onPanEnd","onPanCancel","onSwipe","onPress","onPressUp","onPinch","onPinchStart","onPinchMove","onPinchEnd","onPinchCancel","onRotate","onRotateStart","onRotateMove","onRotateEnd","onRotateCancel","touchOptions","object","panThreshold","number","panDirection","oneOf","swipeDirection","createTouchHammer","node","props","getData","hammer","set","filter","key","forEach","test","event","replace","toLowerCase","handler","on","e","data","gesture","get","direction","Hammer","threshold","destroyTouchHammer","stop","destroy","supportsTouch","touchableKey","hit","indexOf","withTouch","Component","config","getTouchData","spec","displayName","render","s","wrappedProps","without","hasTouch","children","Children","toArray","createElement","componentDidMount","touchable","undefined","touchHammer","findDOMNode","componentWillUnmount"],"mappings":"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","file":"with_touch.jsx","sourceRoot":"lib","sourcesContent":["/**\n * Higher order component with touching feature\n * @function withTouch\n * @param {function} Component - A component constructor\n * @returns {function} - Wrapped component\n */\n'use strict'\n\nimport {wrap} from 'breact'\nimport React, {PropTypes as types} from 'react'\nimport ReactDOM from 'react-dom'\nimport {clone} from 'asobj'\nimport {directions} from './enums'\nimport newHammer from './helpers/new_hammer'\n\nconst propTypes = {\n  /** Handler for \"tap\" event */\n  onTap: types.func,\n  /** Handler for \"doubletap\" event */\n  onDoubleTap: types.func,\n  /** Handler for \"pan\" event */\n  onPan: types.func,\n  /** Handler for \"panstart\" event */\n  onPanStart: types.func,\n  /** Handler for \"panmove\" event */\n  onPanMove: types.func,\n  /** Handler for \"panend\" event */\n  onPanEnd: types.func,\n  /** Handler for \"pancancel\" event */\n  onPanCancel: types.func,\n  /** Handler for \"swipe\" event */\n  onSwipe: types.func,\n  /** Handler for \"press\" event */\n  onPress: types.func,\n  /** Handler for \"pressup\" event */\n  onPressUp: types.func,\n  /** Handler for \"pinch\" event */\n  onPinch: types.func,\n  /** Handler for \"pinchstart\" event */\n  onPinchStart: types.func,\n  /** Handler for \"pinchmove\" event */\n  onPinchMove: types.func,\n  /** Handler for \"pinchend\" event */\n  onPinchEnd: types.func,\n  /** Handler for \"pinchcancel\" event */\n  onPinchCancel: types.func,\n  /** Handler for \"rotate\" event */\n  onRotate: types.func,\n  /** Handler for \"rotatestart\" event */\n  onRotateStart: types.func,\n  /** Handler for \"rotatemove\" event */\n  onRotateMove: types.func,\n  /** Handler for \"rotateend\" event */\n  onRotateEnd: types.func,\n  /** Handler for \"rotatecancel\" event */\n  onRotateCancel: types.func,\n  /** Options for touch */\n  touchOptions: types.object,\n  /** Pixcel for pan threshold */\n  panThreshold: types.number,\n  /** Direction for pan */\n  panDirection: types.oneOf(directions),\n  /** Direction for swipe */\n  swipeDirection: types.oneOf(directions)\n}\n\nfunction createTouchHammer (node, props, getData) {\n  if (!node) {\n    return\n  }\n  let hammer = newHammer(node)\n  if (props.touchOptions) {\n    hammer.set(props.touchOptions)\n  }\n  Object.keys(propTypes)\n    .filter((key) => !!props[ key ])\n    .forEach((key) => {\n      if (/^on/.test(key)) {\n        let event = key.replace(/^on/, '').toLowerCase()\n        let handler = props[ key ]\n        hammer.on(event, (e) => {\n          e.data = getData() // Set touching data.\n          handler(e)\n        })\n      }\n      if (/Direction$/.test(key)) {\n        let gesture = key.replace(/Direction$/, '')\n        hammer.get(gesture).set({ direction: hammer.Hammer[ 'DIRECTION_' + props[ key ] ] })\n      }\n      if (/Threshold$/.test(key)) {\n        let threshold = key.replace(/Threshold$/, '')\n        hammer.get(threshold).set({ threshold: props[ key ] })\n      }\n    })\n  return hammer\n}\n\nfunction destroyTouchHammer (hammer) {\n  if (!hammer) {\n    return\n  }\n  hammer.stop()\n  hammer.destroy()\n}\n\nfunction supportsTouch (props) {\n  let touchableKey = Object.keys(propTypes)\n  for (let key of Object.keys(props)) {\n    let hit = !!~touchableKey.indexOf(key)\n    if (hit) {\n      return true\n    }\n  }\n  return false\n}\n\n/** @lends withTouch */\nfunction withTouch (Component, config = {}) {\n  let { getTouchData } = config\n  let spec = Object.assign({\n    displayName: 'withTouch',\n\n    propTypes,\n\n    render () {\n      const s = this\n      let { props } = s\n      let wrappedProps = clone(props, {\n        without: Object.keys(propTypes || {})\n      })\n      wrappedProps.hasTouch = !!(props.onTap)\n      let children = React.Children.toArray(props.children)\n      return React.createElement(Component, wrappedProps, ...children)\n    },\n\n    componentDidMount () {\n      const s = this\n      let touchable = supportsTouch(s.props)\n      let getData = () => s.getTouchData ? s.getTouchData() : undefined\n      if (touchable) {\n        s.touchHammer = createTouchHammer(ReactDOM.findDOMNode(s), s.props, getData)\n      }\n    },\n\n    componentWillUnmount () {\n      const s = this\n      let hammer = s.touchHammer\n      if (hammer) {\n        destroyTouchHammer(hammer)\n      }\n      delete s.touchHammer\n    }\n  }, { getTouchData })\n  return wrap(Component, spec)\n}\n\nexport default withTouch\n"]}