apeman-react-touchable
Version:
apeman react package for touchable component.
218 lines (186 loc) • 18.4 kB
JavaScript
/**
* 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"]}