react-lightning-design-components
Version:
Salesforce Lightning Design System components built with React 16
323 lines (274 loc) • 28.8 kB
JavaScript
'use strict';
Object.defineProperty(exports, "__esModule", {
value: true
});
var _extends2 = require('babel-runtime/helpers/extends');
var _extends3 = _interopRequireDefault(_extends2);
var _objectWithoutProperties2 = require('babel-runtime/helpers/objectWithoutProperties');
var _objectWithoutProperties3 = _interopRequireDefault(_objectWithoutProperties2);
var _getPrototypeOf = require('babel-runtime/core-js/object/get-prototype-of');
var _getPrototypeOf2 = _interopRequireDefault(_getPrototypeOf);
var _classCallCheck2 = require('babel-runtime/helpers/classCallCheck');
var _classCallCheck3 = _interopRequireDefault(_classCallCheck2);
var _createClass2 = require('babel-runtime/helpers/createClass');
var _createClass3 = _interopRequireDefault(_createClass2);
var _possibleConstructorReturn2 = require('babel-runtime/helpers/possibleConstructorReturn');
var _possibleConstructorReturn3 = _interopRequireDefault(_possibleConstructorReturn2);
var _inherits2 = require('babel-runtime/helpers/inherits');
var _inherits3 = _interopRequireDefault(_inherits2);
var _react = require('react');
var _react2 = _interopRequireDefault(_react);
var _reactDom = require('react-dom');
var _reactDom2 = _interopRequireDefault(_reactDom);
var _propTypes = require('prop-types');
var _propTypes2 = _interopRequireDefault(_propTypes);
var _classnames = require('classnames');
var _classnames2 = _interopRequireDefault(_classnames);
var _Button = require('./Button');
var _Button2 = _interopRequireDefault(_Button);
var _DropdownMenu = require('./DropdownMenu');
var _DropdownMenu2 = _interopRequireDefault(_DropdownMenu);
var _util = require('./util');
function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; }
var DropdownButton = function (_Component) {
(0, _inherits3.default)(DropdownButton, _Component);
function DropdownButton(props) {
(0, _classCallCheck3.default)(this, DropdownButton);
var _this = (0, _possibleConstructorReturn3.default)(this, (DropdownButton.__proto__ || (0, _getPrototypeOf2.default)(DropdownButton)).call(this, props));
_this.state = { opened: false };
_this.currentWidth = 0;
(0, _util.registerStyle)('no-hover-popup', [['.slds-dropdown-trigger:hover .slds-dropdown--menu.react-slds-no-hover-popup', '{ visibility: hidden; opacity: 0; }'], ['.slds-dropdown-trigger.react-slds-dropdown-opened .slds-dropdown--menu', '{ visibility: visible !important; opacity: 1 !important; }']]);
_this.triggerRef = _this.triggerRef.bind(_this);
_this.dropdownRef = _this.dropdownRef.bind(_this);
return _this;
}
(0, _createClass3.default)(DropdownButton, [{
key: 'onBlur',
value: function onBlur() {
var _this2 = this;
setTimeout(function () {
if (!_this2.isFocusedInComponent()) {
_this2.setState({ opened: false });
if (_this2.props.onBlur) {
_this2.props.onBlur();
}
}
}, 10);
}
}, {
key: 'onKeyDown',
value: function onKeyDown(e) {
var _this3 = this;
if (e.keyCode === 40) {
// down
e.preventDefault();
e.stopPropagation();
if (!this.state.opened) {
this.setState({ opened: true });
if (this.props.onClick) {
this.props.onClick(e);
}
setTimeout(function () {
_this3.focusToTargetItemEl();
}, 20);
} else {
this.focusToTargetItemEl();
}
} else if (e.keyCode === 27) {
// ESC
e.preventDefault();
e.stopPropagation();
this.setState({ opened: false });
}
}
}, {
key: 'onTriggerClick',
value: function onTriggerClick() {
if (this.props.inheritWidth) {
this.currentWidth = this.getCurrentWidth();
}
var triggerElem = _reactDom2.default.findDOMNode(this.trigger);
if (triggerElem !== document.activeElement) triggerElem.focus();
if (!this.props.hoverPopup) {
this.setState({ opened: !this.state.opened });
}
if (this.props.onClick) {
var _props;
(_props = this.props).onClick.apply(_props, arguments);
}
}
}, {
key: 'onMenuItemClick',
value: function onMenuItemClick() {
var _this4 = this;
if (!this.props.hoverPopup) {
setTimeout(function () {
var triggerElem = _reactDom2.default.findDOMNode(_this4.trigger);
if (triggerElem) triggerElem.focus();
if (triggerElem) _this4.setState({ opened: false });
if (_this4.props.focusOnInput) _this4.props.focusOnInput();
}, 10);
}
if (this.props.onMenuItemClick) {
var _props2;
(_props2 = this.props).onMenuItemClick.apply(_props2, arguments);
}
}
}, {
key: 'onMenuClose',
value: function onMenuClose() {
var triggerElem = _reactDom2.default.findDOMNode(this.trigger);
triggerElem.focus();
this.setState({ opened: false });
}
}, {
key: 'getCurrentWidth',
value: function getCurrentWidth() {
var htmlElemnt = _reactDom2.default.findDOMNode(this) || {};
return htmlElemnt.offsetWidth || 0;
}
}, {
key: 'isFocusedInComponent',
value: function isFocusedInComponent() {
var rootEl = _reactDom2.default.findDOMNode(this);
var targetEl = document.activeElement;
while (targetEl && targetEl !== rootEl) {
targetEl = targetEl.parentNode;
}
return !!targetEl;
}
}, {
key: 'focusToTargetItemEl',
value: function focusToTargetItemEl() {
var dropdownEl = _reactDom2.default.findDOMNode(this.dropdown);
var firstItemEl = dropdownEl.querySelector('.slds-is-selected > .react-slds-menuitem[tabIndex]') || dropdownEl.querySelector('.react-slds-menuitem[tabIndex]');
if (firstItemEl) {
firstItemEl.focus();
}
}
}, {
key: 'triggerRef',
value: function triggerRef(ref) {
this.trigger = ref;
}
}, {
key: 'dropdownRef',
value: function dropdownRef(ref) {
this.dropdown = ref;
}
}, {
key: 'renderButton',
value: function renderButton(_ref) {
var grouped = _ref.grouped,
isFirstInGroup = _ref.isFirstInGroup,
isLastInGroup = _ref.isLastInGroup,
props = (0, _objectWithoutProperties3.default)(_ref, ['grouped', 'isFirstInGroup', 'isLastInGroup']);
var pprops = (0, _extends3.default)({}, props);
delete pprops.onMenuItemClick;
delete pprops.inheritWidth;
delete pprops.focusOnInput;
var button = _react2.default.createElement(_Button2.default, (0, _extends3.default)({}, pprops, {
'aria-haspopup': true,
ref: this.triggerRef,
onClick: this.onTriggerClick.bind(this),
onKeyDown: this.onKeyDown.bind(this),
onBlur: this.onBlur.bind(this)
}));
if (grouped) {
var noneStyle = { display: 'none' };
return _react2.default.createElement(
'div',
{ className: 'slds-button-group' },
isFirstInGroup ? null : _react2.default.createElement('button', { className: 'slds-button', style: noneStyle }),
button,
isLastInGroup ? null : _react2.default.createElement('button', { className: 'slds-button', style: noneStyle })
);
}
return button;
}
}, {
key: 'render',
value: function render() {
var _props3 = this.props,
inheritWidth = _props3.inheritWidth,
className = _props3.className,
listClassName = _props3.listClassName,
menuClassName = _props3.menuClassName,
_props3$menuAlign = _props3.menuAlign,
menuAlign = _props3$menuAlign === undefined ? 'left' : _props3$menuAlign,
menuSize = _props3.menuSize,
nubbinTop = _props3.nubbinTop,
hoverPopup = _props3.hoverPopup,
menuHeader = _props3.menuHeader,
type = _props3.type,
label = _props3.label,
children = _props3.children,
backgroundColor = _props3.backgroundColor,
keyCodesToCloseMenu = _props3.keyCodesToCloseMenu,
props = (0, _objectWithoutProperties3.default)(_props3, ['inheritWidth', 'className', 'listClassName', 'menuClassName', 'menuAlign', 'menuSize', 'nubbinTop', 'hoverPopup', 'menuHeader', 'type', 'label', 'children', 'backgroundColor', 'keyCodesToCloseMenu']);
var icon = this.props.icon;
var dropdownClassNames = (0, _classnames2.default)(className, 'slds-dropdown-trigger', {
'slds-button-space-left': !props.grouped,
'react-slds-dropdown-opened': this.state.opened
});
var iconMore = null;
if (!label && !icon) {
icon = 'down';
}
if (label || type === 'icon-more') {
iconMore = 'down';
}
return _react2.default.createElement(
'div',
{ className: dropdownClassNames, style: { backgroundColor: backgroundColor } },
this.renderButton((0, _extends3.default)({ type: type, label: label, icon: icon, iconMore: iconMore }, props)),
_react2.default.createElement(
_DropdownMenu2.default,
{
className: menuClassName,
listClassName: listClassName,
align: menuAlign,
minWidth: inheritWidth ? this.currentWidth : 0,
header: menuHeader,
size: menuSize,
nubbinTop: nubbinTop,
hoverPopup: hoverPopup,
ref: this.dropdownRef,
onMenuItemClick: this.onMenuItemClick.bind(this),
onMenuClose: this.onMenuClose.bind(this),
onBlur: this.onBlur.bind(this),
keyCodesToCloseMenu: keyCodesToCloseMenu
},
children
)
);
}
}]);
return DropdownButton;
}(_react.Component);
exports.default = DropdownButton;
DropdownButton.propTypes = {
className: _propTypes2.default.string,
listClassName: _propTypes2.default.string,
menuClassName: _propTypes2.default.string,
label: _propTypes2.default.node,
type: _propTypes2.default.string,
icon: _propTypes2.default.string,
menuAlign: _propTypes2.default.oneOf(['left', 'center', 'right']),
menuSize: _propTypes2.default.oneOf(['small', 'medium', 'large']),
menuHeader: _propTypes2.default.string,
nubbinTop: _propTypes2.default.bool,
hoverPopup: _propTypes2.default.bool,
onBlur: _propTypes2.default.func,
onClick: _propTypes2.default.func,
onMenuItemClick: _propTypes2.default.func,
grouped: _propTypes2.default.bool,
isFirstInGroup: _propTypes2.default.bool,
isLastInGroup: _propTypes2.default.bool,
children: _propTypes2.default.node,
inheritWidth: _propTypes2.default.bool,
backgroundColor: _propTypes2.default.string,
focusOnInput: _propTypes2.default.func,
keyCodesToCloseMenu: _propTypes2.default.arrayOf(_propTypes2.default.number)
};
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../src/scripts/DropdownButton.js"],"names":["DropdownButton","props","state","opened","currentWidth","triggerRef","bind","dropdownRef","setTimeout","isFocusedInComponent","setState","onBlur","e","keyCode","preventDefault","stopPropagation","onClick","focusToTargetItemEl","inheritWidth","getCurrentWidth","triggerElem","findDOMNode","trigger","document","activeElement","focus","hoverPopup","focusOnInput","onMenuItemClick","htmlElemnt","offsetWidth","rootEl","targetEl","parentNode","dropdownEl","dropdown","firstItemEl","querySelector","ref","grouped","isFirstInGroup","isLastInGroup","pprops","button","onTriggerClick","onKeyDown","noneStyle","display","className","listClassName","menuClassName","menuAlign","menuSize","nubbinTop","menuHeader","type","label","children","backgroundColor","keyCodesToCloseMenu","icon","dropdownClassNames","iconMore","renderButton","onMenuClose","propTypes","string","node","oneOf","bool","func","arrayOf","number"],"mappings":"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","file":"DropdownButton.js","sourcesContent":["import React, { Component } from 'react';\nimport ReactDOM from 'react-dom';\nimport PropTypes from 'prop-types';\nimport classnames from 'classnames';\nimport Button from './Button';\nimport DropdownMenu from './DropdownMenu';\nimport { registerStyle } from './util';\n\nexport default class DropdownButton extends Component {\n  constructor(props) {\n    super(props);\n    this.state = { opened: false };\n    this.currentWidth = 0;\n    registerStyle('no-hover-popup', [\n      [\n        '.slds-dropdown-trigger:hover .slds-dropdown--menu.react-slds-no-hover-popup',\n        '{ visibility: hidden; opacity: 0; }',\n      ],\n      [\n        '.slds-dropdown-trigger.react-slds-dropdown-opened .slds-dropdown--menu',\n        '{ visibility: visible !important; opacity: 1 !important; }',\n      ],\n    ]);\n\n    this.triggerRef = this.triggerRef.bind(this);\n    this.dropdownRef = this.dropdownRef.bind(this);\n  }\n\n  onBlur() {\n    setTimeout(() => {\n      if (!this.isFocusedInComponent()) {\n        this.setState({ opened: false });\n        if (this.props.onBlur) {\n          this.props.onBlur();\n        }\n      }\n    }, 10);\n  }\n\n  onKeyDown(e) {\n    if (e.keyCode === 40) { // down\n      e.preventDefault();\n      e.stopPropagation();\n      if (!this.state.opened) {\n        this.setState({ opened: true });\n        if (this.props.onClick) {\n          this.props.onClick(e);\n        }\n        setTimeout(() => {\n          this.focusToTargetItemEl();\n        }, 20);\n      } else {\n        this.focusToTargetItemEl();\n      }\n    } else if (e.keyCode === 27) { // ESC\n      e.preventDefault();\n      e.stopPropagation();\n      this.setState({ opened: false });\n    }\n  }\n\n  onTriggerClick(...args) {\n    if (this.props.inheritWidth) {\n      this.currentWidth = this.getCurrentWidth();\n    }\n\n    const triggerElem = ReactDOM.findDOMNode(this.trigger);\n    if (triggerElem !== document.activeElement) triggerElem.focus();\n\n    if (!this.props.hoverPopup) {\n      this.setState({ opened: !this.state.opened });\n    }\n    if (this.props.onClick) {\n      this.props.onClick(...args);\n    }\n  }\n\n  onMenuItemClick(...args) {\n    if (!this.props.hoverPopup) {\n      setTimeout(() => {\n        const triggerElem = ReactDOM.findDOMNode(this.trigger);\n        if (triggerElem) triggerElem.focus();\n        if (triggerElem) this.setState({ opened: false });\n        if (this.props.focusOnInput) this.props.focusOnInput();\n      }, 10);\n    }\n    if (this.props.onMenuItemClick) {\n      this.props.onMenuItemClick(...args);\n    }\n  }\n\n  onMenuClose() {\n    const triggerElem = ReactDOM.findDOMNode(this.trigger);\n    triggerElem.focus();\n    this.setState({ opened: false });\n  }\n\n  getCurrentWidth() {\n    const htmlElemnt = ReactDOM.findDOMNode(this) || {};\n    return htmlElemnt.offsetWidth || 0;\n  }\n\n  isFocusedInComponent() {\n    const rootEl = ReactDOM.findDOMNode(this);\n    let targetEl = document.activeElement;\n    while (targetEl && targetEl !== rootEl) {\n      targetEl = targetEl.parentNode;\n    }\n    return !!targetEl;\n  }\n\n  focusToTargetItemEl() {\n    const dropdownEl = ReactDOM.findDOMNode(this.dropdown);\n    const firstItemEl =\n      dropdownEl.querySelector('.slds-is-selected > .react-slds-menuitem[tabIndex]') ||\n      dropdownEl.querySelector('.react-slds-menuitem[tabIndex]');\n    if (firstItemEl) {\n      firstItemEl.focus();\n    }\n  }\n\n  triggerRef(ref) {\n    this.trigger = ref;\n  }\n\n  dropdownRef(ref) {\n    this.dropdown = ref;\n  }\n\n  renderButton({ grouped, isFirstInGroup, isLastInGroup, ...props }) {\n    const pprops = { ...props };\n    delete pprops.onMenuItemClick;\n    delete pprops.inheritWidth;\n    delete pprops.focusOnInput;\n    const button = (\n      <Button\n        { ...pprops }\n        aria-haspopup\n        ref={this.triggerRef}\n        onClick={ this.onTriggerClick.bind(this) }\n        onKeyDown={ this.onKeyDown.bind(this) }\n        onBlur={ this.onBlur.bind(this) }\n      />\n    );\n\n    if (grouped) {\n      const noneStyle = { display: 'none' };\n      return (\n        <div className='slds-button-group'>\n          { isFirstInGroup ? null : <button className='slds-button' style={ noneStyle }></button> }\n          { button }\n          { isLastInGroup ? null : <button className='slds-button' style={ noneStyle }></button> }\n        </div>\n      );\n    }\n\n    return button;\n  }\n\n  render() {\n    const {\n      inheritWidth,\n      className, listClassName, menuClassName, menuAlign = 'left', menuSize, nubbinTop, hoverPopup,\n      menuHeader, type, label, children, backgroundColor, keyCodesToCloseMenu, ...props,\n    } = this.props;\n    let { icon } = this.props;\n    const dropdownClassNames = classnames(\n      className,\n      'slds-dropdown-trigger',\n      {\n        'slds-button-space-left': !props.grouped,\n        'react-slds-dropdown-opened': this.state.opened,\n      }\n    );\n    let iconMore = null;\n    if (!label && !icon) {\n      icon = 'down';\n    }\n    if (label || type === 'icon-more') {\n      iconMore = 'down';\n    }\n    return (\n      <div className={ dropdownClassNames } style={{ backgroundColor }}>\n        { this.renderButton({ type, label, icon, iconMore, ...props }) }\n        <DropdownMenu\n          className={menuClassName}\n          listClassName={listClassName}\n          align={ menuAlign }\n          minWidth={inheritWidth ? this.currentWidth : 0}\n          header={ menuHeader }\n          size={ menuSize }\n          nubbinTop={ nubbinTop }\n          hoverPopup={ hoverPopup }\n          ref={this.dropdownRef}\n          onMenuItemClick={ this.onMenuItemClick.bind(this) }\n          onMenuClose={ this.onMenuClose.bind(this) }\n          onBlur={ this.onBlur.bind(this) }\n          keyCodesToCloseMenu={ keyCodesToCloseMenu }\n        >\n          { children }\n        </DropdownMenu>\n      </div>\n    );\n  }\n\n}\n\nDropdownButton.propTypes = {\n  className: PropTypes.string,\n  listClassName: PropTypes.string,\n  menuClassName: PropTypes.string,\n  label: PropTypes.node,\n  type: PropTypes.string,\n  icon: PropTypes.string,\n  menuAlign: PropTypes.oneOf(['left', 'center', 'right']),\n  menuSize: PropTypes.oneOf(['small', 'medium', 'large']),\n  menuHeader: PropTypes.string,\n  nubbinTop: PropTypes.bool,\n  hoverPopup: PropTypes.bool,\n  onBlur: PropTypes.func,\n  onClick: PropTypes.func,\n  onMenuItemClick: PropTypes.func,\n  grouped: PropTypes.bool,\n  isFirstInGroup: PropTypes.bool,\n  isLastInGroup: PropTypes.bool,\n  children: PropTypes.node,\n  inheritWidth: PropTypes.bool,\n  backgroundColor: PropTypes.string,\n  focusOnInput: PropTypes.func,\n  keyCodesToCloseMenu: PropTypes.arrayOf(PropTypes.number),\n};\n"]}