// Copyright 2015-2017 Parity Technologies (UK) Ltd. // This file is part of Parity. // Parity is free software: you can redistribute it and/or modify // it under the terms of the GNU General Public License as published by // the Free Software Foundation, either version 3 of the License, or // (at your option) any later version. // Parity is distributed in the hope that it will be useful, // but WITHOUT ANY WARRANTY; without even the implied warranty of // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the // GNU General Public License for more details. // You should have received a copy of the GNU General Public License // along with Parity. If not, see . import { isEqual } from 'lodash'; import React, { Component, PropTypes } from 'react'; import ReactDOM from 'react-dom'; import { Toolbar, ToolbarGroup } from 'material-ui/Toolbar'; import { nodeOrStringProptype } from '~/util/proptypes'; import styles from './actionbar.css'; export default class Actionbar extends Component { buttons = {}; buttonsTooltip = {}; static propTypes = { title: nodeOrStringProptype(), buttons: PropTypes.array, children: PropTypes.node, className: PropTypes.string }; static defaultProps = { buttons: [] }; state = { buttons: [] }; componentWillMount () { this.setButtons(this.props); window.addEventListener('resize', this.checkButtonsTooltip); } componentWillUnmount () { window.removeEventListener('resize', this.checkButtonsTooltip); } componentWillReceiveProps (nextProps) { if (!isEqual(this.props.buttons, nextProps.buttons)) { this.setButtons(nextProps); } } render () { const { children, className } = this.props; const classes = `${styles.actionbar} ${className}`; return ( { this.renderTitle() } { this.renderButtons() } { children } ); } renderButtons () { const { buttons } = this.state; if (buttons.length === 0) { return null; } return ( { buttons } ); } renderTitle () { const { title } = this.props; return (

{ title }

); } checkButtonsTooltip = () => { const buttonsTooltip = Object.keys(this.buttons) .reduce((buttonsTooltip, index) => { buttonsTooltip[index] = this.checkButtonTooltip(this.buttons[index]); return buttonsTooltip; }, {}); if (isEqual(buttonsTooltip, this.buttonsTooltip)) { return; } this.buttonsTooltip = buttonsTooltip; this.setButtons(this.props); } checkButtonTooltip = (button) => { const { icon, text } = button; const iconBoundings = icon.getBoundingClientRect(); const textBoundings = text.getBoundingClientRect(); // Visible if the bottom of the text is above the bottom of the // button (text is v-aligned on top) const isTextVisible = textBoundings.top + textBoundings.height < iconBoundings.top + iconBoundings.height; return !isTextVisible; } /** * Return the icon and text nodes of a Button * (and SVG/IMG for the icon next to a span node) */ getIconAndTextNodes (element) { if (!element || !element.children || element.children.length === 0) { return null; } const children = Array.slice(element.children); const text = children.find((child) => child.nodeName.toLowerCase() === 'span'); const icon = children.find((child) => { const nodeName = child.nodeName.toLowerCase(); return nodeName === 'svg' || nodeName === 'img'; }); if (icon && text) { return { icon, text }; } const result = children .map((child) => { return this.getIconAndTextNodes(child); }) .filter((result) => result); return result.length > 0 ? result[0] : null; } /** * Add tooltip to all Buttons */ patchButton (element, extraProps) { if (element.type.displayName !== 'Button') { if (!element.props.children) { return element; } const children = this.patchButton(element.props.children); return React.cloneElement(element, {}, children); } return React.cloneElement(element, extraProps); } setButtons (props) { const buttons = props.buttons .filter((button) => button) .map((button, index) => { const ref = this.setButtonRef.bind(this, index); const showTooltip = this.buttonsTooltip[index]; return this.patchButton(button, { tooltip: showTooltip, ref }); }); this.setState({ buttons }); } setButtonRef = (index, element) => { const node = ReactDOM.findDOMNode(element); const iconAndText = this.getIconAndTextNodes(node); if (!iconAndText) { return; } if (!this.buttons[index]) { this.buttonsTooltip[index] = this.checkButtonTooltip(iconAndText); this.setButtons(this.props); } this.buttons[index] = iconAndText; }; }