apeman-react-select
Version:
apeman react package for select component.
431 lines (373 loc) • 34.4 kB
JavaScript
/**
* apeman react package for select component.
* @class ApSelect
*/
'use strict';
Object.defineProperty(exports, "__esModule", {
value: true
});
exports.ApSelect = undefined;
var _assign = require('babel-runtime/core-js/object/assign');
var _assign2 = _interopRequireDefault(_assign);
var _keys = require('babel-runtime/core-js/object/keys');
var _keys2 = _interopRequireDefault(_keys);
var _react = require('react');
var _react2 = _interopRequireDefault(_react);
var _reactDom = require('react-dom');
var _reactDom2 = _interopRequireDefault(_reactDom);
var _classnames = require('classnames');
var _classnames2 = _interopRequireDefault(_classnames);
var _ap_select_item = require('./ap_select_item');
var _ap_select_item2 = _interopRequireDefault(_ap_select_item);
var _ap_select_label = require('./ap_select_label');
var _ap_select_label2 = _interopRequireDefault(_ap_select_label);
var _numcal = require('numcal');
var _numcal2 = _interopRequireDefault(_numcal);
var _bwindow = require('bwindow');
var _apemanReactMixinLayout = require('apeman-react-mixin-layout');
var _apemanReactTouchable = require('apeman-react-touchable');
function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; }
/** @lends ApSelect */
var ApSelect = _react2.default.createClass({
displayName: 'ApSelect',
// --------------------
// Specs
// --------------------
propTypes: {
/** Options of select */
options: _react.PropTypes.object.isRequired,
/** Option elements to render */
optionElements: _react.PropTypes.object,
/** Name of select element */
name: _react.PropTypes.string,
/** Value of select element */
value: _react.PropTypes.string,
/** Allow multiple select */
multiple: _react.PropTypes.bool,
/** Handler for change event */
onChange: _react.PropTypes.func,
/** Icon to toggle select */
openIcon: _react.PropTypes.string,
/** Placeholder of select element */
placeholder: _react.PropTypes.string
},
mixins: [_apemanReactMixinLayout.ApLayoutMixin],
statics: {},
getInitialState: function getInitialState() {
var s = this;
return {
focused: false,
focusIndex: s.getIndexForValue(s.props.value)
};
},
getDefaultProps: function getDefaultProps() {
return {
optionElements: null,
value: '',
name: null,
multiple: false,
onChange: null,
openIcon: 'ion ion-arrow-down-b',
placeholder: null
};
},
render: function render() {
var s = this;
var state = s.state,
props = s.props,
layouts = s.layouts;
var options = props.options,
optionElements = props.optionElements;
var values = s.getOptionValues();
var hasOption = options && (0, _keys2.default)(options).length > 0;
if (!hasOption) {
return null;
}
var _option = function _option(value) {
return optionElements && optionElements[value] || options[value] || null;
};
return _react2.default.createElement(
'span',
{ className: (0, _classnames2.default)('ap-select-wrap') },
_react2.default.createElement(
'span',
{ className: (0, _classnames2.default)('ap-select-options-list', {
'ap-select-options-list-visible': state.focused
}), ref: function ref(list) {
return s.registerNode(list, 'list');
}
},
_react2.default.createElement(
'ul',
{ className: 'ap-select-options-list-inner',
style: layouts.listInner },
values.map(function (value, i) {
return _react2.default.createElement(
'li',
{ key: value,
value: value,
className: (0, _classnames2.default)('ap-select-options-list-item') },
_react2.default.createElement(
_ap_select_item2.default,
{ onTap: s.handleItemTap,
data: value,
focused: state.focusIndex === i,
label: options[value]
},
_option(value) || null
)
);
})
)
),
_react2.default.createElement(
'select',
{ id: props.id,
name: props.name,
placeholder: props.placeholder,
onChange: props.onChange,
className: (0, _classnames2.default)('ap-select', props.className),
onFocus: function onFocus() {
return s.setFocus(true);
},
style: (0, _assign2.default)({}, props.style),
tabIndex: '-1'
},
values.map(function (value) {
return _react2.default.createElement(
'option',
{ key: value, value: value },
options[value]
);
}),
props.children
),
_react2.default.createElement('input', { type: 'text',
ref: function ref(text) {
return s.registerNode(text, 'text');
},
className: 'ap-select-dummy-text',
onKeyUp: s.handleKeyUp,
onKeyDown: s.handleKeyDown,
onFocus: function onFocus() {
return s.setFocus(true);
},
onBlur: function onBlur() {
return s.setFocus(false);
}
}),
_react2.default.createElement(_ap_select_label2.default, { value: _option(props.value),
placeholder: props.placeholder,
icon: props.openIcon,
onTap: s.handleLabelTap
})
);
},
// --------------------
// Lifecycle
// --------------------
componentWillMount: function componentWillMount() {
var s = this;
s.nodes = {};
},
componentDidMount: function componentDidMount() {
var s = this;
var body = (0, _bwindow.get)('document.body');
body.addEventListener('click', s.handleClickForOutside);
},
componentWillUnmount: function componentWillUnmount() {
var s = this;
var body = (0, _bwindow.get)('document.body');
body.removeEventListener('click', s.handleClickForOutside);
},
// ------------------
// Custom
// ------------------
moveFocusIndex: function moveFocusIndex(i) {
var s = this;
var state = s.state;
var values = s.getOptionValues();
var index = state.focusIndex + i;
var over = index === -1 || index === values.length;
if (over) {
return;
}
s.setState({
focusIndex: index
});
},
enterFocused: function enterFocused(e) {
var s = this;
var state = s.state,
props = s.props;
if (!state.focused) {
return;
}
var values = s.getOptionValues();
var value = values[state.focusIndex];
s.setState({
focused: false,
focusIndex: s.getIndexForValue(value)
});
e.target.value = value;
if (props.onChange) {
props.onChange(e);
}
},
getOptionValues: function getOptionValues() {
var s = this;
var props = s.props;
return (0, _keys2.default)(props.options || {});
},
getIndexForValue: function getIndexForValue(value) {
var s = this;
return s.getOptionValues().indexOf(value);
},
registerNode: function registerNode(elm, name) {
var s = this;
s.nodes[name] = _reactDom2.default.findDOMNode(elm);
},
// --------------------
// Handle
// --------------------
handleLabelTap: function handleLabelTap(e) {
var s = this;
var state = s.state;
var text = s.nodes.text;
var focused = !state.focused;
if (focused) {
s.layout();
text.focus();
} else {
text.blur();
}
s.setState({
focused: focused,
focusIndex: s.getIndexForValue(s.props.value)
});
},
setFocus: function setFocus(focused) {
var s = this;
if (focused === s.state.focused) {
return;
}
if (s._focusAt) {
var fromLastFocusAt = new Date() - s._focusAt;
if (fromLastFocusAt < 500) {
return;
}
}
s._focusAt = new Date();
s.setState({
focused: focused
});
},
handleKeyDown: function handleKeyDown(e) {
var s = this;
var props = s.props;
if (!s.state.focused) {
s.setState({ focused: true });
return;
}
switch (e.keyCode) {
case 38:
// UP
s.moveFocusIndex(-1);
break;
case 40:
// DOWN
s.moveFocusIndex(+1);
break;
case 13:
// Enter
s.enterFocused(e);
break;
case 9:
// Tab
break;
default:
e.preventDefault();
e.stopPropagation();
break;
}
if (props.onKeyDown) {
props.onKeyDown(e);
}
},
handleKeyUp: function handleKeyUp(e) {
var s = this;
var props = s.props;
if (props.onKeyUp) {
props.onKeyUp(e);
}
e.stopPropagation();
},
handleItemTap: function handleItemTap(e) {
var s = this;
var props = s.props;
(0, _assign2.default)(e.target, {
value: e.target.value || e.data || null,
name: e.target.name || props.name
});
if (props.onChange) {
props.onChange(e);
}
s.setState({
focused: false,
focusIndex: s.getIndexForValue(s.props.value)
});
},
handleClickForOutside: function handleClickForOutside(e) {
var s = this;
var node = _reactDom2.default.findDOMNode(s);
if (!node) {
return;
}
var contained = node.contains(e.target);
if (!contained) {
s.outsideDidTap(e);
}
},
outsideDidTap: function outsideDidTap(e) {
var s = this;
s.setFocus(false);
},
// ------------------
// ApLayoutMixin
// ------------------
getInitialLayouts: function getInitialLayouts() {
return {
listInner: {
transform: 'initial'
}
};
},
calcLayouts: function calcLayouts() {
var s = this;
var _window = window,
innerHeight = _window.innerHeight,
innerWidth = _window.innerWidth;
var list = s.nodes.list;
if (!list) {
return {};
}
return {
listInner: s._listInnerLayout(list.getBoundingClientRect(), innerWidth, innerHeight)
};
},
// ------------------
// Private
// ------------------
_listInnerLayout: function _listInnerLayout(rect, boundsWidth, boundsHeight) {
var x = _numcal2.default.min(boundsWidth - rect.right, 0);
var y = _numcal2.default.min(boundsHeight - rect.bottom, 0);
var maxHeight = _numcal2.default.min(boundsHeight, 280) + 'px';
return {
transform: 'translate(' + x + 'px, ' + y + 'px)',
maxHeight: maxHeight
};
}
});
exports.ApSelect = ApSelect;
exports.default = (0, _apemanReactTouchable.withOutside)(ApSelect);
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["ap_select.jsx"],"names":["ApSelect","createClass","propTypes","options","object","isRequired","optionElements","name","string","value","multiple","bool","onChange","func","openIcon","placeholder","mixins","statics","getInitialState","s","focused","focusIndex","getIndexForValue","props","getDefaultProps","render","state","layouts","values","getOptionValues","hasOption","length","_option","list","registerNode","listInner","map","i","handleItemTap","id","className","setFocus","style","children","text","handleKeyUp","handleKeyDown","handleLabelTap","componentWillMount","nodes","componentDidMount","body","addEventListener","handleClickForOutside","componentWillUnmount","removeEventListener","moveFocusIndex","index","over","setState","enterFocused","e","target","indexOf","elm","findDOMNode","layout","focus","blur","_focusAt","fromLastFocusAt","Date","keyCode","preventDefault","stopPropagation","onKeyDown","onKeyUp","data","node","contained","contains","outsideDidTap","getInitialLayouts","transform","calcLayouts","window","innerHeight","innerWidth","_listInnerLayout","getBoundingClientRect","rect","boundsWidth","boundsHeight","x","min","right","y","bottom","maxHeight"],"mappings":"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","file":"ap_select.jsx","sourceRoot":"lib","sourcesContent":["/**\n * apeman react package for select component.\n * @class ApSelect\n */\n\n'use strict'\n\nimport React, { Component, PropTypes as types } from 'react'\nimport ReactDOM from 'react-dom'\nimport classnames from 'classnames'\nimport ApSelectItem from './ap_select_item'\nimport ApSelectLabel from './ap_select_label'\nimport numcal from 'numcal'\nimport { get } from 'bwindow'\nimport { ApLayoutMixin } from 'apeman-react-mixin-layout'\nimport { withOutside } from 'apeman-react-touchable'\n\n/** @lends ApSelect */\nconst ApSelect = React.createClass({\n\n  // --------------------\n  // Specs\n  // --------------------\n\n  propTypes: {\n    /** Options of select */\n    options: types.object.isRequired,\n    /** Option elements to render */\n    optionElements: types.object,\n    /** Name of select element */\n    name: types.string,\n    /** Value of select element */\n    value: types.string,\n    /** Allow multiple select */\n    multiple: types.bool,\n    /** Handler for change event */\n    onChange: types.func,\n    /** Icon to toggle select */\n    openIcon: types.string,\n    /** Placeholder of select element */\n    placeholder: types.string\n  },\n\n  mixins: [\n    ApLayoutMixin\n  ],\n\n  statics: {},\n\n  getInitialState () {\n    const s = this\n    return {\n      focused: false,\n      focusIndex: s.getIndexForValue(s.props.value)\n    }\n  },\n\n  getDefaultProps () {\n    return {\n      optionElements: null,\n      value: '',\n      name: null,\n      multiple: false,\n      onChange: null,\n      openIcon: 'ion ion-arrow-down-b',\n      placeholder: null\n    }\n  },\n\n  render () {\n    const s = this\n    let { state, props, layouts } = s\n\n    let { options, optionElements } = props\n    let values = s.getOptionValues()\n\n    let hasOption = options && Object.keys(options).length > 0\n    if (!hasOption) {\n      return null\n    }\n\n    let _option = (value) => optionElements && optionElements[ value ] || options[ value ] || null\n\n    return (\n      <span className={ classnames('ap-select-wrap') }>\n          <span className={ classnames('ap-select-options-list', {\n            'ap-select-options-list-visible': state.focused\n          }) } ref={ (list) => s.registerNode(list, 'list') }\n          >\n              <ul className='ap-select-options-list-inner'\n                  style={ layouts.listInner }>\n                  { values.map((value, i) =>\n                    <li key={ value }\n                        value={ value }\n                        className={ classnames('ap-select-options-list-item') }>\n                      <ApSelectItem onTap={ s.handleItemTap }\n                                    data={ value }\n                                    focused={ state.focusIndex === i }\n                                    label={ options[ value ] }\n                      >\n                        { _option(value) || null }\n                      </ApSelectItem>\n                    </li>\n                  ) }\n              </ul>\n          </span>\n          <select id={ props.id }\n                  name={ props.name }\n                  placeholder={ props.placeholder }\n                  onChange={ props.onChange }\n                  className={ classnames('ap-select', props.className) }\n                  onFocus={ () => s.setFocus(true) }\n                  style={ Object.assign({}, props.style) }\n                  tabIndex=\"-1\"\n          >\n              { values.map((value) =>\n                <option key={ value } value={ value }>{ options[ value ] }</option>\n              ) }\n            { props.children }\n          </select>\n          <input type='text'\n                 ref={ (text) => s.registerNode(text, 'text') }\n                 className='ap-select-dummy-text'\n                 onKeyUp={ s.handleKeyUp }\n                 onKeyDown={ s.handleKeyDown }\n                 onFocus={ () => s.setFocus(true) }\n                 onBlur={ () => s.setFocus(false) }\n          />\n          <ApSelectLabel value={ _option(props.value) }\n                         placeholder={ props.placeholder }\n                         icon={ props.openIcon }\n                         onTap={ s.handleLabelTap }\n          />\n\n      </span>\n    )\n  },\n\n  // --------------------\n  // Lifecycle\n  // --------------------\n\n  componentWillMount () {\n    const s = this\n    s.nodes = {}\n  },\n\n  componentDidMount () {\n    const s = this\n    let body = get('document.body')\n    body.addEventListener('click', s.handleClickForOutside)\n  },\n\n  componentWillUnmount () {\n    const s = this\n    let body = get('document.body')\n    body.removeEventListener('click', s.handleClickForOutside)\n\n  },\n  // ------------------\n  // Custom\n  // ------------------\n\n  moveFocusIndex (i) {\n    const s = this\n    let { state } = s\n    let values = s.getOptionValues()\n    let index = state.focusIndex + i\n    let over = (index === -1) || (index === values.length)\n    if (over) {\n      return\n    }\n    s.setState({\n      focusIndex: index\n    })\n  },\n\n  enterFocused (e) {\n    const s = this\n    let { state, props } = s\n    if (!state.focused) {\n      return\n    }\n    let values = s.getOptionValues()\n    let value = values[ state.focusIndex ]\n    s.setState({\n      focused: false,\n      focusIndex: s.getIndexForValue(value)\n    })\n    e.target.value = value\n    if (props.onChange) {\n      props.onChange(e)\n    }\n  },\n\n  getOptionValues () {\n    const s = this\n    let { props } = s\n    return Object.keys(props.options || {})\n  },\n\n  getIndexForValue (value) {\n    const s = this\n    return s.getOptionValues().indexOf(value)\n  },\n\n  registerNode (elm, name) {\n    const s = this\n    s.nodes[ name ] = ReactDOM.findDOMNode(elm)\n  },\n\n  // --------------------\n  // Handle\n  // --------------------\n\n  handleLabelTap (e) {\n    const s = this\n    let { state } = s\n\n    let { text } = s.nodes\n\n    let focused = !state.focused\n    if (focused) {\n      s.layout()\n      text.focus()\n    } else {\n      text.blur()\n    }\n    s.setState({\n      focused,\n      focusIndex: s.getIndexForValue(s.props.value)\n    })\n  },\n\n  setFocus (focused) {\n    const s = this\n    if (focused === s.state.focused) {\n      return\n    }\n    if(s._focusAt){\n      const fromLastFocusAt = new Date()  - s._focusAt\n      if(fromLastFocusAt < 500){\n        return\n      }\n    }\n    s._focusAt = new Date()\n    s.setState({\n      focused\n    })\n  },\n\n  handleKeyDown (e) {\n    const s = this\n    let { props } = s\n    if (!s.state.focused) {\n      s.setState({ focused: true })\n      return\n    }\n    switch (e.keyCode) {\n      case 38: // UP\n        s.moveFocusIndex(-1)\n        break\n      case 40: // DOWN\n        s.moveFocusIndex(+1)\n        break\n      case 13: // Enter\n        s.enterFocused(e)\n        break\n      case 9: // Tab\n        break\n      default:\n        e.preventDefault()\n        e.stopPropagation()\n        break\n    }\n    if (props.onKeyDown) {\n      props.onKeyDown(e)\n    }\n  },\n\n  handleKeyUp (e) {\n    const s = this\n    let { props } = s\n    if (props.onKeyUp) {\n      props.onKeyUp(e)\n    }\n    e.stopPropagation()\n  },\n\n  handleItemTap (e) {\n    const s = this\n    let { props } = s\n    Object.assign(e.target, {\n      value: e.target.value || e.data || null,\n      name: e.target.name || props.name\n    })\n    if (props.onChange) {\n      props.onChange(e)\n    }\n    s.setState({\n      focused: false,\n      focusIndex: s.getIndexForValue(s.props.value)\n    })\n  },\n\n  handleClickForOutside (e) {\n    const s = this\n    let node = ReactDOM.findDOMNode(s)\n    if (!node) {\n      return\n    }\n    let contained = node.contains(e.target)\n    if (!contained) {\n      s.outsideDidTap(e)\n    }\n  },\n\n  outsideDidTap (e) {\n    const s = this\n    s.setFocus(false)\n  },\n\n  // ------------------\n  // ApLayoutMixin\n  // ------------------\n\n  getInitialLayouts () {\n    return {\n      listInner: {\n        transform: 'initial'\n      }\n    }\n  },\n\n  calcLayouts () {\n    const s = this\n    let { innerHeight, innerWidth } = window\n    let { list } = s.nodes\n    if (!list) {\n      return {}\n    }\n    return {\n      listInner: s._listInnerLayout(list.getBoundingClientRect(), innerWidth, innerHeight)\n    }\n  },\n\n  // ------------------\n  // Private\n  // ------------------\n\n  _listInnerLayout (rect, boundsWidth, boundsHeight) {\n    let x = numcal.min(boundsWidth - rect.right, 0)\n    let y = numcal.min(boundsHeight - rect.bottom, 0)\n\n    let maxHeight = `${numcal.min(boundsHeight, 280)}px`\n    return {\n      transform: `translate(${x}px, ${y}px)`,\n      maxHeight\n    }\n  }\n})\n\nexport { ApSelect }\nexport default withOutside(ApSelect)\n"]}