focus-components-v3
Version:
Focus web components to build applications (based on Material Design)
368 lines (282 loc) • 36.3 kB
JavaScript
Object.defineProperty(exports, "__esModule", {
value: true
});
var _extends = Object.assign || function (target) { for (var i = 1; i < arguments.length; i++) { var source = arguments[i]; for (var key in source) { if (Object.prototype.hasOwnProperty.call(source, key)) { target[key] = source[key]; } } } return target; };
var _class;
var _react = require('react');
var _react2 = _interopRequireDefault(_react);
var _reactDom = require('react-dom');
var _reactDom2 = _interopRequireDefault(_reactDom);
var _buttonBackToTop = require('../button-back-to-top');
var _buttonBackToTop2 = _interopRequireDefault(_buttonBackToTop);
var _stickyMenu = require('./sticky-menu');
var _stickyMenu2 = _interopRequireDefault(_stickyMenu);
var _scroll = require('../behaviours/scroll');
var _scroll2 = _interopRequireDefault(_scroll);
var _grid = require('../grid');
var _grid2 = _interopRequireDefault(_grid);
var _column = require('../column');
var _column2 = _interopRequireDefault(_column);
var _debounce = require('lodash/debounce');
var _debounce2 = _interopRequireDefault(_debounce);
var _filter = require('lodash/filter');
var _filter2 = _interopRequireDefault(_filter);
var _first = require('lodash/first');
var _first2 = _interopRequireDefault(_first);
var _last = require('lodash/last');
var _last2 = _interopRequireDefault(_last);
var _xor = require('lodash/xor');
var _xor2 = _interopRequireDefault(_xor);
function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; }
function _defaults(obj, defaults) { var keys = Object.getOwnPropertyNames(defaults); for (var i = 0; i < keys.length; i++) { var key = keys[i]; var value = Object.getOwnPropertyDescriptor(defaults, key); if (value && value.configurable && obj[key] === undefined) { Object.defineProperty(obj, key, value); } } return obj; }
function _objectWithoutProperties(obj, keys) { var target = {}; for (var i in obj) { if (keys.indexOf(i) >= 0) continue; if (!Object.prototype.hasOwnProperty.call(obj, i)) continue; target[i] = obj[i]; } return target; }
function _classCallCheck(instance, Constructor) { if (!(instance instanceof Constructor)) { throw new TypeError("Cannot call a class as a function"); } }
function _possibleConstructorReturn(self, call) { if (!self) { throw new ReferenceError("this hasn't been initialised - super() hasn't been called"); } return call && (typeof call === "object" || typeof call === "function") ? call : self; }
function _inherits(subClass, superClass) { if (typeof superClass !== "function" && superClass !== null) { throw new TypeError("Super expression must either be null or a function, not " + typeof superClass); } subClass.prototype = Object.create(superClass && superClass.prototype, { constructor: { value: subClass, enumerable: false, writable: true, configurable: true } }); if (superClass) Object.setPrototypeOf ? Object.setPrototypeOf(subClass, superClass) : _defaults(subClass, superClass); }
var BackToTopComponent = _buttonBackToTop2.default;
// component default props.
var defaultProps = {
hasMenu: true, //Activate the presence of the sticky navigation component.
hasBackToTop: true, //Activate the presence of BackToTop button
offset: 100, //offset position when affix
scrollDelay: 10 //defaut debounce delay for scroll spy call
};
// component props definition.
var propTypes = {
hasMenu: _react.PropTypes.bool,
hasBackToTop: _react.PropTypes.bool,
offset: _react.PropTypes.number,
scrollDelay: _react.PropTypes.number
};
/**
* ScrollspyContainer component.
*/
var ScrollspyContainer = (0, _scroll2.default)(_class = function (_Component) {
_inherits(ScrollspyContainer, _Component);
function ScrollspyContainer(props) {
_classCallCheck(this, ScrollspyContainer);
var _this = _possibleConstructorReturn(this, _Component.call(this, props));
_this._executeRefreshMenu = function (time) {
_this._timeouts = [];
for (var i = 0; i < time; i++) {
_this._timeouts.push(setTimeout(_this._refreshMenu.bind(_this), i * 1000));
}
};
_this._debounceRefreshMenu = function () {
_this._debouncedRefresh();
};
_this._refreshMenu = function () {
if (!_this.props.hasMenu) {
return;
}
var stickyMenu = _this.refs.stickyMenu;
var clickedId = _this.state.clickedId;
var menus = _this._buildMenuList(); //build the menu list
//TODO remove this check
var affix = stickyMenu ? _this._isMenuAffix() : _this.state.affix; //Calculate menu position (affix or not)
// Check if scroll is at cliked item level
var isAtClickedItem = void 0;
if (clickedId !== undefined) {
var selector = '[data-spy=\'' + clickedId + '\']';
var node = document.querySelector(selector);
var nodePosition = _this.scrollPosition(node);
var positionTop = _this._getElementRealPosition(nodePosition.top);
isAtClickedItem = _this.scrollPosition().top === positionTop;
}
_this.setState({
menuList: menus,
clickedId: isAtClickedItem ? undefined : clickedId,
affix: affix
});
};
_this._buildMenuList = function () {
var hasMenu = _this.props.hasMenu;
if (!hasMenu) {
return [];
}
var detectionOffset = window.screen.height / 5;
var currentScrollPosition = { top: _this.scrollPosition().top, left: _this.scrollPosition().left };
var isAtPageBottom = _this.isAtPageBottom();
//get the menu list (without blocks in popin)
var thisComponentNode = _reactDom2.default.findDOMNode(_this);
var allDataSpy = thisComponentNode.querySelectorAll('[data-spy]');
var popinDataSpy = thisComponentNode.querySelectorAll('[data-focus=\'popin-window\'] [data-spy]');
var selectionList = (0, _xor2.default)(allDataSpy, popinDataSpy);
if (selectionList.length === 0) {
return;
}
var menuList = selectionList.map(function (selection, index) {
var title = selection.querySelector('[data-spy-title]');
var nodeId = selection.getAttribute('data-spy');
return {
index: index,
label: title.innerHTML,
nodeId: nodeId,
scrollTop: _this.scrollPosition(selection).top, // offset of 10 to be safe
isActive: false,
onClick: _this._getMenuItemClickHandler(nodeId)
};
});
var nextTitles = (0, _filter2.default)(menuList, function (n) {
return currentScrollPosition.top + detectionOffset < _this._getElementRealPosition(n.scrollTop);
});
//Calculate current node
//by default, first node is indexed
var currentIndex = menuList[0].index;
if (0 < nextTitles.length) {
//check the first node
var firstNode = (0, _first2.default)(nextTitles);
var index = firstNode.index;
if (0 < index) {
currentIndex = menuList[index - 1].index;
}
} else {
//means that the position is the last title
currentIndex = (0, _last2.default)(menuList).index;
}
var clickedId = _this.state.clickedId;
if (isAtPageBottom && undefined !== clickedId) {
menuList = menuList.map(function (item) {
if (item.nodeId === clickedId) {
item.isActive = true;
}
return item;
});
_this.setState({ clickedId: undefined });
} else {
menuList[currentIndex].isActive = true;
}
return menuList;
};
_this._getElementRealPosition = function (position) {
var sscDomNode = _reactDom2.default.findDOMNode(_this);
var sscPosition = _this.scrollPosition(sscDomNode);
return position - sscPosition.top;
};
_this._isMenuAffix = function () {
var offset = _this.props.offset;
var hasMenu = _this.props.hasMenu;
if (!hasMenu) {
return false;
}
var sscDomNode = _reactDom2.default.findDOMNode(_this);
var currentViewPosition = sscDomNode.getBoundingClientRect();
var containerPaddingTop = _this._getPaddingTopValue();
offset -= containerPaddingTop;
return currentViewPosition.top <= offset;
};
_this._getPaddingTopValue = function () {
var sscDomNode = _reactDom2.default.findDOMNode(_this);
var computedStyles = window.getComputedStyle(sscDomNode, null);
var paddingTop = computedStyles.getPropertyValue('padding-top');
return paddingTop ? parseInt(paddingTop, 0) : 0;
};
var state = {
menuList: [],
affix: false
};
_this.state = state;
return _this;
}
/** @inheritDoc */
ScrollspyContainer.prototype.componentDidMount = function componentDidMount() {
this._scrollCarrier = window;
this._debouncedRefresh = (0, _debounce2.default)(this._refreshMenu, this.props.scrollDelay);
this._scrollCarrier.addEventListener('scroll', this._debounceRefreshMenu);
this._scrollCarrier.addEventListener('resize', this._debounceRefreshMenu);
this._executeRefreshMenu(10);
};
/** @inheritDoc */
ScrollspyContainer.prototype.componentWillUnmount = function componentWillUnmount() {
this._timeouts.map(clearTimeout);
this._scrollCarrier.removeEventListener('scroll', this._debounceRefreshMenu);
this._scrollCarrier.removeEventListener('resize', this._debounceRefreshMenu);
this._debouncedRefresh.cancel();
};
/**
* Refresh screen X times.
* @param {number} time number of execution
*/
/**
* The scroll event handler
* @private
*/
/**
* Build the list of menus.
* @private
* @return {array} the list of menus.
*/
/**
* Calculate the real position of an element, depending on declared offset in props.
* @private
* @param {number} position position
* @return {number} the real position
*/
/**
* Calculate menu position (affix or not)
* @private
* @return {Boolean} true is menu must be affix, else false
*/
/**
* Handle click on item menu function.
* @private
* @param {string} menuId node spyId in DOM to scroll to
* @return {function} function to call
*/
ScrollspyContainer.prototype._getMenuItemClickHandler = function _getMenuItemClickHandler(menuId) {
var _this2 = this;
return function () {
_this2.setState({
clickedId: menuId
}, function () {
_this2._refreshMenu();
_this2._onMenuItemClick(menuId);
});
};
};
/**
* Menu click function. Scroll to the node position.
* @private
* @param {string} menuId node spyId in DOM to scroll to
*/
ScrollspyContainer.prototype._onMenuItemClick = function _onMenuItemClick(menuId) {
var selector = '[data-spy=\'' + menuId + '\']';
var node = document.querySelector(selector);
var nodePosition = this.scrollPosition(node);
var positionTop = this._getElementRealPosition(nodePosition.top);
this.scrollTo(undefined, positionTop);
};
/** @inheritedDoc */
ScrollspyContainer.prototype.render = function render() {
var _props = this.props,
children = _props.children,
hasMenu = _props.hasMenu,
hasBackToTop = _props.hasBackToTop,
offset = _props.offset,
scrollDelay = _props.scrollDelay,
otherProps = _objectWithoutProperties(_props, ['children', 'hasMenu', 'hasBackToTop', 'offset', 'scrollDelay']);
var _state = this.state,
affix = _state.affix,
menuList = _state.menuList;
return _react2.default.createElement(
'div',
_extends({ 'data-focus': 'scrollspy-container' }, otherProps),
hasMenu && _react2.default.createElement(_stickyMenu2.default, { affix: affix, affixOffset: offset, menuList: menuList, ref: 'stickyMenu' }),
_react2.default.createElement(
'div',
{ 'data-focus': 'scrollspy-container-content' },
children
),
hasBackToTop && _react2.default.createElement(BackToTopComponent, null)
);
};
return ScrollspyContainer;
}(_react.Component)) || _class;
//Static props.
ScrollspyContainer.displayName = 'ScrollspyContainer';
ScrollspyContainer.defaultProps = defaultProps;
ScrollspyContainer.propTypes = propTypes;
exports.default = ScrollspyContainer;
module.exports = exports['default'];
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["fr-FR.js"],"names":["BackToTopComponent","defaultProps","hasMenu","hasBackToTop","offset","scrollDelay","propTypes","bool","number","ScrollspyContainer","props","_executeRefreshMenu","_timeouts","i","time","push","setTimeout","_refreshMenu","bind","_debounceRefreshMenu","_debouncedRefresh","stickyMenu","refs","clickedId","state","menus","_buildMenuList","affix","_isMenuAffix","isAtClickedItem","undefined","selector","node","document","querySelector","nodePosition","scrollPosition","positionTop","_getElementRealPosition","top","setState","menuList","detectionOffset","window","screen","height","currentScrollPosition","left","isAtPageBottom","thisComponentNode","findDOMNode","allDataSpy","querySelectorAll","popinDataSpy","selectionList","length","map","selection","index","title","nodeId","getAttribute","label","innerHTML","scrollTop","isActive","onClick","_getMenuItemClickHandler","nextTitles","n","currentIndex","firstNode","item","position","sscDomNode","sscPosition","currentViewPosition","getBoundingClientRect","containerPaddingTop","_getPaddingTopValue","computedStyles","getComputedStyle","paddingTop","getPropertyValue","parseInt","componentDidMount","_scrollCarrier","addEventListener","componentWillUnmount","clearTimeout","removeEventListener","cancel","menuId","_onMenuItemClick","scrollTo","render","children","otherProps","displayName"],"mappings":"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","file":"fr-FR.js","sourcesContent":["import React, {Component, PropTypes} from 'react';\r\nimport ReactDOM from 'react-dom';\r\nimport BackToTop from '../button-back-to-top'\r\nimport StickyMenu from './sticky-menu';\r\nimport Scroll from '../behaviours/scroll';\r\nimport Grid from '../grid';\r\nimport Column from '../column';\r\n\r\nimport debounce from 'lodash/debounce';\r\nimport filter from 'lodash/filter';\r\nimport first from 'lodash/first';\r\nimport last from 'lodash/last';\r\nimport xor from 'lodash/xor';\r\n\r\nconst BackToTopComponent = BackToTop;\r\n\r\n// component default props.\r\nconst defaultProps = {\r\n    hasMenu: true, //Activate the presence of the sticky navigation component.\r\n    hasBackToTop: true, //Activate the presence of BackToTop button\r\n    offset: 100, //offset position when affix\r\n    scrollDelay: 10 //defaut debounce delay for scroll spy call\r\n};\r\n\r\n// component props definition.\r\nconst propTypes = {\r\n    hasMenu: PropTypes.bool,\r\n    hasBackToTop: PropTypes.bool,\r\n    offset: PropTypes.number,\r\n    scrollDelay: PropTypes.number\r\n};\r\n\r\n/**\r\n* ScrollspyContainer component.\r\n*/\r\n@Scroll\r\nclass ScrollspyContainer extends Component {\r\n    constructor(props) {\r\n        super(props);\r\n        const state = {\r\n            menuList: [],\r\n            affix: false\r\n        };\r\n        this.state = state;\r\n    }\r\n\r\n    /** @inheritDoc */\r\n    componentDidMount() {\r\n        this._scrollCarrier = window;\r\n        this._debouncedRefresh = debounce(this._refreshMenu, this.props.scrollDelay);\r\n        this._scrollCarrier.addEventListener('scroll', this._debounceRefreshMenu);\r\n        this._scrollCarrier.addEventListener('resize', this._debounceRefreshMenu);\r\n        this._executeRefreshMenu(10);\r\n    }\r\n\r\n    /** @inheritDoc */\r\n    componentWillUnmount() {\r\n        this._timeouts.map(clearTimeout);\r\n        this._scrollCarrier.removeEventListener('scroll', this._debounceRefreshMenu);\r\n        this._scrollCarrier.removeEventListener('resize', this._debounceRefreshMenu);\r\n        this._debouncedRefresh.cancel();\r\n    }\r\n\r\n    /**\r\n    * Refresh screen X times.\r\n    * @param  {number} time number of execution\r\n    */\r\n    _executeRefreshMenu = time => {\r\n        this._timeouts = [];\r\n        for (let i = 0; i < time; i++) {\r\n            this._timeouts.push(setTimeout(this._refreshMenu.bind(this), i * 1000));\r\n        }\r\n    };\r\n\r\n    _debounceRefreshMenu = () => {\r\n        this._debouncedRefresh();\r\n    };\r\n\r\n    /**\r\n    * The scroll event handler\r\n    * @private\r\n    */\r\n    _refreshMenu = () => {\r\n        if(!this.props.hasMenu) { return; }\r\n        const {stickyMenu} = this.refs;\r\n        const {clickedId} = this.state;\r\n        const menus = this._buildMenuList(); //build the menu list\r\n        //TODO remove this check\r\n        const affix = stickyMenu ? this._isMenuAffix() : this.state.affix; //Calculate menu position (affix or not)\r\n        // Check if scroll is at cliked item level\r\n        let isAtClickedItem;\r\n        if (clickedId !== undefined) {\r\n            const selector = `[data-spy='${clickedId}']`;\r\n            const node = document.querySelector(selector);\r\n            const nodePosition = this.scrollPosition(node);\r\n            const positionTop = this._getElementRealPosition(nodePosition.top);\r\n            isAtClickedItem = this.scrollPosition().top === positionTop;\r\n        }\r\n        this.setState({\r\n            menuList: menus,\r\n            clickedId: isAtClickedItem ? undefined : clickedId,\r\n            affix\r\n        });\r\n    };\r\n\r\n    /**\r\n    * Build the list of menus.\r\n    * @private\r\n    * @return {array} the list of menus.\r\n    */\r\n    _buildMenuList = () => {\r\n        const {hasMenu} = this.props;\r\n        if(!hasMenu) {\r\n            return [];\r\n        }\r\n        const detectionOffset = window.screen.height / 5;\r\n        let currentScrollPosition = {top: this.scrollPosition().top, left: this.scrollPosition().left};\r\n        let isAtPageBottom = this.isAtPageBottom();\r\n\r\n        //get the menu list (without blocks in popin)\r\n        const thisComponentNode = ReactDOM.findDOMNode(this);\r\n        const allDataSpy = thisComponentNode.querySelectorAll('[data-spy]');\r\n        const popinDataSpy = thisComponentNode.querySelectorAll(`[data-focus='popin-window'] [data-spy]`);\r\n        const selectionList = xor(allDataSpy, popinDataSpy);\r\n\r\n        if(selectionList.length === 0) {\r\n            return;\r\n        }\r\n        let menuList = selectionList.map((selection, index) => {\r\n            const title = selection.querySelector('[data-spy-title]');\r\n            const nodeId = selection.getAttribute('data-spy');\r\n            return {\r\n                index: index,\r\n                label: title.innerHTML,\r\n                nodeId: nodeId,\r\n                scrollTop: this.scrollPosition(selection).top, // offset of 10 to be safe\r\n                isActive: false,\r\n                onClick: this._getMenuItemClickHandler(nodeId)\r\n            };\r\n        });\r\n\r\n        const nextTitles = filter(menuList, n => (currentScrollPosition.top + detectionOffset < this._getElementRealPosition(n.scrollTop)));\r\n\r\n        //Calculate current node\r\n        //by default, first node is indexed\r\n        let currentIndex = menuList[0].index;\r\n        if(0 < nextTitles.length) {\r\n            //check the first node\r\n            const firstNode = first(nextTitles);\r\n            const index = firstNode.index;\r\n            if(0 < index) {\r\n                currentIndex = menuList[index - 1].index;\r\n            }\r\n        } else {\r\n            //means that the position is the last title\r\n            currentIndex = last(menuList).index;\r\n        }\r\n        let clickedId = this.state.clickedId;\r\n        if(isAtPageBottom && undefined !== clickedId) {\r\n            menuList = menuList.map(item => {\r\n                if (item.nodeId === clickedId) {\r\n                    item.isActive = true;\r\n                }\r\n                return item;\r\n            });\r\n            this.setState({clickedId: undefined});\r\n        }else {\r\n            menuList[currentIndex].isActive = true;\r\n        }\r\n        return menuList;\r\n    };\r\n\r\n    /**\r\n    * Calculate the real position of an element, depending on declared offset in props.\r\n    * @private\r\n    * @param  {number} position position\r\n    * @return {number} the real position\r\n    */\r\n    _getElementRealPosition = (position) => {\r\n        const sscDomNode = ReactDOM.findDOMNode(this);\r\n        const sscPosition = this.scrollPosition(sscDomNode);\r\n        return position - sscPosition.top;\r\n    };\r\n\r\n    /**\r\n    * Calculate menu position (affix or not)\r\n    * @private\r\n    * @return {Boolean} true is menu must be affix, else false\r\n    */\r\n    _isMenuAffix = () => {\r\n        let {offset} = this.props;\r\n        const {hasMenu} = this.props;\r\n        if(!hasMenu) {\r\n            return false;\r\n        }\r\n        const sscDomNode = ReactDOM.findDOMNode(this);\r\n        const currentViewPosition = sscDomNode.getBoundingClientRect();\r\n        const containerPaddingTop = this._getPaddingTopValue();\r\n        offset -= containerPaddingTop;\r\n        return currentViewPosition.top <= offset;\r\n    };\r\n\r\n    _getPaddingTopValue = () => {\r\n        const sscDomNode = ReactDOM.findDOMNode(this);\r\n        const computedStyles = window.getComputedStyle(sscDomNode, null);\r\n        const paddingTop = computedStyles.getPropertyValue('padding-top');\r\n        return paddingTop ? parseInt(paddingTop, 0) : 0;\r\n    };\r\n\r\n    /**\r\n    * Handle click on item menu function.\r\n    * @private\r\n    * @param  {string} menuId  node spyId in DOM to scroll to\r\n    * @return {function}        function to call\r\n    */\r\n    _getMenuItemClickHandler(menuId) {\r\n        return () => {\r\n            this.setState({\r\n                clickedId: menuId\r\n            }, () => {\r\n                this._refreshMenu();\r\n                this._onMenuItemClick(menuId);\r\n            });\r\n        }\r\n    }\r\n\r\n    /**\r\n    * Menu click function. Scroll to the node position.\r\n    * @private\r\n    * @param  {string} menuId  node spyId in DOM to scroll to\r\n    */\r\n    _onMenuItemClick(menuId) {\r\n        const selector = `[data-spy='${menuId}']`;\r\n        const node = document.querySelector(selector);\r\n        const nodePosition = this.scrollPosition(node);\r\n        const positionTop = this._getElementRealPosition(nodePosition.top);\r\n        this.scrollTo(undefined, positionTop);\r\n    }\r\n\r\n    /** @inheritedDoc */\r\n    render() {\r\n        const {children, hasMenu, hasBackToTop, offset, scrollDelay, ...otherProps} = this.props;\r\n        const {affix, menuList} = this.state;\r\n        return (\r\n            <div data-focus='scrollspy-container' {...otherProps}>\r\n                {hasMenu &&\r\n                    <StickyMenu affix={affix} affixOffset={offset} menuList={menuList} ref='stickyMenu' />\r\n                }\r\n                <div data-focus='scrollspy-container-content'>\r\n                    {children}\r\n                </div>\r\n                {hasBackToTop &&\r\n                    <BackToTopComponent />\r\n                }\r\n            </div>\r\n        );\r\n    }\r\n}\r\n\r\n//Static props.\r\nScrollspyContainer.displayName = 'ScrollspyContainer';\r\nScrollspyContainer.defaultProps = defaultProps;\r\nScrollspyContainer.propTypes = propTypes;\r\n\r\nexport default ScrollspyContainer;\r\n"]}
;