1
0
mirror of https://github.com/kremalicious/metamask-extension.git synced 2024-12-23 09:52:26 +01:00
Mark Stacey 46157cb476
Remove redundant PropTypes (#8126)
Specifying a PropType of either type `node` or type
`arrayOf(PropTypes.node)` is redundant, because an array of nodes is
itself a node.
2020-02-27 10:31:59 -04:00

287 lines
6.1 KiB
JavaScript

import React, { Component } from 'react'
import PropTypes from 'prop-types'
let index = 0
let extraSheet
const insertRule = (css) => {
if (!extraSheet) {
// First time, create an extra stylesheet for adding rules
extraSheet = document.createElement('style')
document.getElementsByTagName('head')[0].appendChild(extraSheet)
// Keep reference to actual StyleSheet object (`styleSheet` for IE < 9)
extraSheet = extraSheet.sheet || extraSheet.styleSheet
}
const index = (extraSheet.cssRules || extraSheet.rules).length
extraSheet.insertRule(css, index)
return extraSheet
}
const insertKeyframesRule = (keyframes) => {
// random name
const name = 'anim_' + (++index) + (+new Date())
let css = '@' + 'keyframes ' + name + ' {'
for (const key in keyframes) {
css += key + ' {'
for (const property in keyframes[key]) {
const part = ':' + keyframes[key][property] + ';'
css += property + part
}
css += '}'
}
css += '}'
insertRule(css)
return name
}
const animation = {
show: {
animationDuration: '0.3s',
animationTimingFunction: 'ease-out',
},
hide: {
animationDuration: '0.3s',
animationTimingFunction: 'ease-out',
},
showContentAnimation: insertKeyframesRule({
'0%': {
opacity: 0,
},
'100%': {
opacity: 1,
},
}),
hideContentAnimation: insertKeyframesRule({
'0%': {
opacity: 1,
},
'100%': {
opacity: 0,
},
}),
showBackdropAnimation: insertKeyframesRule({
'0%': {
opacity: 0,
},
'100%': {
opacity: 0.9,
},
}),
hideBackdropAnimation: insertKeyframesRule({
'0%': {
opacity: 0.9,
},
'100%': {
opacity: 0,
},
}),
}
const endEvents = ['transitionend', 'animationend']
function addEventListener (node, eventName, eventListener) {
node.addEventListener(eventName, eventListener, false)
}
function removeEventListener (node, eventName, eventListener) {
node.removeEventListener(eventName, eventListener, false)
}
const removeEndEventListener = (node, eventListener) => {
if (endEvents.length === 0) {
return
}
endEvents.forEach(function (endEvent) {
removeEventListener(node, endEvent, eventListener)
})
}
const addEndEventListener = (node, eventListener) => {
if (endEvents.length === 0) {
// If CSS transitions are not supported, trigger an "end animation"
// event immediately.
window.setTimeout(eventListener, 0)
return
}
endEvents.forEach(function (endEvent) {
addEventListener(node, endEvent, eventListener)
})
}
class FadeModal extends Component {
content = null
static propTypes = {
backdrop: PropTypes.bool,
backdropStyle: PropTypes.object,
closeOnClick: PropTypes.bool,
contentStyle: PropTypes.object,
keyboard: PropTypes.bool,
modalStyle: PropTypes.object,
onShow: PropTypes.func,
onHide: PropTypes.func,
children: PropTypes.node,
}
static defaultProps = {
onShow: function () {},
onHide: function () {},
keyboard: true,
backdrop: true,
closeOnClick: true,
modalStyle: {},
backdropStyle: {},
contentStyle: {},
children: [],
}
state = {
willHide: true,
hidden: true,
}
addTransitionListener = (node, handle) => {
if (node) {
const endListener = function (e) {
if (e && e.target !== node) {
return
}
removeEndEventListener(node, endListener)
handle()
}
addEndEventListener(node, endListener)
}
}
handleBackdropClick = () => {
if (this.props.closeOnClick) {
this.hide()
}
}
hasHidden = () => {
return this.state.hidden
}
render () {
if (this.state.hidden) {
return null
}
const { willHide } = this.state
const { modalStyle } = this.props
const backdropStyle = Object.assign({}, {
animationName: willHide ? animation.hideBackdropAnimation : animation.showBackdropAnimation,
animationTimingFunction: (willHide ? animation.hide : animation.show).animationTimingFunction,
}, this.props.backdropStyle)
const contentStyle = Object.assign({}, {
animationDuration: (willHide ? animation.hide : animation.show).animationDuration,
animationName: willHide ? animation.hideContentAnimation : animation.showContentAnimation,
animationTimingFunction: (willHide ? animation.hide : animation.show).animationTimingFunction,
}, this.props.contentStyle)
const backdrop = this.props.backdrop
? (
<div
className="backdrop"
style={backdropStyle}
onClick={this.props.closeOnClick
? this.handleBackdropClick
: null}
/>
) : undefined
if (willHide) {
this.addTransitionListener(this.content, this.leave)
}
return (
<span>
<div className="modal" style={modalStyle}>
<div
className="content"
ref={(el) => (this.content = el)}
tabIndex="-1"
style={contentStyle}
>
{this.props.children}
</div>
</div>
{backdrop}
</span>
)
}
leave = () => {
this.setState({
hidden: true,
})
this.props.onHide(this.state.hideSource)
}
enter = () => {
this.props.onShow()
}
show = () => {
if (!this.state.hidden) {
return
}
this.setState({
willHide: false,
hidden: false,
})
setTimeout(function () {
this.addTransitionListener(this.content, this.enter)
}.bind(this), 0)
}
hide = () => {
if (this.hasHidden()) {
return
}
this.setState({
willHide: true,
})
}
listenKeyboard = (event) => {
if (typeof this.props.keyboard === 'function') {
this.props.keyboard(event)
} else {
this.closeOnEsc(event)
}
}
closeOnEsc = (event) => {
if (this.props.keyboard &&
(event.key === 'Escape' ||
event.keyCode === 27)) {
this.hide()
}
}
UNSAFE_componentDidMount = () => {
window.addEventListener('keydown', this.listenKeyboard, true)
}
UNSAFE_componentWillUnmount = () => {
window.removeEventListener('keydown', this.listenKeyboard, true)
}
}
export default FadeModal