|
@ -1,11 +1,12 @@ |
|
|
// @flow
|
|
|
// @flow
|
|
|
|
|
|
|
|
|
import React from 'react' |
|
|
import React, { PureComponent } from 'react' |
|
|
import styled from 'styled-components' |
|
|
import styled from 'styled-components' |
|
|
import { space, fontSize, fontWeight, color } from 'styled-system' |
|
|
import { space, fontSize, fontWeight, color } from 'styled-system' |
|
|
import noop from 'lodash/noop' |
|
|
import noop from 'lodash/noop' |
|
|
import { track } from 'analytics/segment' |
|
|
import { track } from 'analytics/segment' |
|
|
|
|
|
|
|
|
|
|
|
import { isGlobalTabEnabled } from 'config/global-tab' |
|
|
import { darken, lighten, rgba } from 'styles/helpers' |
|
|
import { darken, lighten, rgba } from 'styles/helpers' |
|
|
import fontFamily from 'styles/styled/fontFamily' |
|
|
import fontFamily from 'styles/styled/fontFamily' |
|
|
import { focusedShadowStyle } from 'components/base/Box/Tabbable' |
|
|
import { focusedShadowStyle } from 'components/base/Box/Tabbable' |
|
@ -16,21 +17,28 @@ type Style = any // FIXME |
|
|
|
|
|
|
|
|
const buttonStyles: { [_: string]: Style } = { |
|
|
const buttonStyles: { [_: string]: Style } = { |
|
|
default: { |
|
|
default: { |
|
|
default: noop, |
|
|
default: p => ` |
|
|
|
|
|
box-shadow: ${p.isFocused ? focusedShadowStyle : ''} |
|
|
|
|
|
`,
|
|
|
active: p => ` |
|
|
active: p => ` |
|
|
background: ${rgba(p.theme.colors.fog, 0.3)}; |
|
|
background: ${rgba(p.theme.colors.fog, 0.3)}; |
|
|
`,
|
|
|
`,
|
|
|
hover: p => ` |
|
|
hover: p => ` |
|
|
background: ${rgba(p.theme.colors.fog, 0.2)}; |
|
|
background: ${rgba(p.theme.colors.fog, 0.2)}; |
|
|
`,
|
|
|
`,
|
|
|
focus: () => ` |
|
|
|
|
|
box-shadow: ${focusedShadowStyle}; |
|
|
|
|
|
`,
|
|
|
|
|
|
}, |
|
|
}, |
|
|
primary: { |
|
|
primary: { |
|
|
default: p => ` |
|
|
default: p => ` |
|
|
background: ${p.disabled ? `${p.theme.colors.lightFog} !important` : p.theme.colors.wallet}; |
|
|
background: ${p.disabled ? `${p.theme.colors.lightFog} !important` : p.theme.colors.wallet}; |
|
|
color: ${p.disabled ? p.theme.colors.grey : p.theme.colors.white}; |
|
|
color: ${p.disabled ? p.theme.colors.grey : p.theme.colors.white}; |
|
|
|
|
|
box-shadow: ${ |
|
|
|
|
|
p.isFocused |
|
|
|
|
|
? ` |
|
|
|
|
|
0 0 0 1px ${darken(p.theme.colors.wallet, 0.3)} inset, |
|
|
|
|
|
0 0 0 1px ${rgba(p.theme.colors.wallet, 0.5)}, |
|
|
|
|
|
0 0 0 4px ${rgba(p.theme.colors.wallet, 0.3)};` |
|
|
|
|
|
: '' |
|
|
|
|
|
} |
|
|
`,
|
|
|
`,
|
|
|
hover: p => ` |
|
|
hover: p => ` |
|
|
background: ${lighten(p.theme.colors.wallet, 0.05)}; |
|
|
background: ${lighten(p.theme.colors.wallet, 0.05)}; |
|
@ -38,17 +46,20 @@ const buttonStyles: { [_: string]: Style } = { |
|
|
active: p => ` |
|
|
active: p => ` |
|
|
background: ${darken(p.theme.colors.wallet, 0.1)}; |
|
|
background: ${darken(p.theme.colors.wallet, 0.1)}; |
|
|
`,
|
|
|
`,
|
|
|
focus: p => ` |
|
|
|
|
|
box-shadow: |
|
|
|
|
|
0 0 0 1px ${darken(p.theme.colors.wallet, 0.3)} inset, |
|
|
|
|
|
0 0 0 1px ${rgba(p.theme.colors.wallet, 0.5)}, |
|
|
|
|
|
0 0 0 4px ${rgba(p.theme.colors.wallet, 0.3)}; |
|
|
|
|
|
`,
|
|
|
|
|
|
}, |
|
|
}, |
|
|
danger: { |
|
|
danger: { |
|
|
default: p => ` |
|
|
default: p => ` |
|
|
background: ${p.disabled ? `${p.theme.colors.lightFog} !important` : p.theme.colors.alertRed}; |
|
|
background: ${p.disabled ? `${p.theme.colors.lightFog} !important` : p.theme.colors.alertRed}; |
|
|
color: ${p.disabled ? p.theme.colors.grey : p.theme.colors.white}; |
|
|
color: ${p.disabled ? p.theme.colors.grey : p.theme.colors.white}; |
|
|
|
|
|
box-shadow: ${ |
|
|
|
|
|
p.isFocused |
|
|
|
|
|
? ` |
|
|
|
|
|
0 0 0 1px ${darken(p.theme.colors.alertRed, 0.3)} inset, |
|
|
|
|
|
0 0 0 1px ${rgba(p.theme.colors.alertRed, 0.5)}, |
|
|
|
|
|
0 0 0 4px ${rgba(p.theme.colors.alertRed, 0.3)}; |
|
|
|
|
|
` |
|
|
|
|
|
: '' |
|
|
|
|
|
} |
|
|
`,
|
|
|
`,
|
|
|
hover: p => ` |
|
|
hover: p => ` |
|
|
background: ${lighten(p.theme.colors.alertRed, 0.1)}; |
|
|
background: ${lighten(p.theme.colors.alertRed, 0.1)}; |
|
@ -56,12 +67,6 @@ const buttonStyles: { [_: string]: Style } = { |
|
|
active: p => ` |
|
|
active: p => ` |
|
|
background: ${darken(p.theme.colors.alertRed, 0.1)}; |
|
|
background: ${darken(p.theme.colors.alertRed, 0.1)}; |
|
|
`,
|
|
|
`,
|
|
|
focus: p => ` |
|
|
|
|
|
box-shadow: |
|
|
|
|
|
0 0 0 1px ${darken(p.theme.colors.alertRed, 0.3)} inset, |
|
|
|
|
|
0 0 0 1px ${rgba(p.theme.colors.alertRed, 0.5)}, |
|
|
|
|
|
0 0 0 4px ${rgba(p.theme.colors.alertRed, 0.3)}; |
|
|
|
|
|
`,
|
|
|
|
|
|
}, |
|
|
}, |
|
|
outline: { |
|
|
outline: { |
|
|
default: p => ` |
|
|
default: p => ` |
|
@ -114,18 +119,22 @@ const buttonStyles: { [_: string]: Style } = { |
|
|
|
|
|
|
|
|
function getStyles(props, state) { |
|
|
function getStyles(props, state) { |
|
|
let output = `` |
|
|
let output = `` |
|
|
const defaultStyle = buttonStyles.default[state] |
|
|
let hasModifier = false |
|
|
if (defaultStyle) { |
|
|
|
|
|
output += defaultStyle(props) || '' |
|
|
|
|
|
} |
|
|
|
|
|
for (const s in buttonStyles) { |
|
|
for (const s in buttonStyles) { |
|
|
if (buttonStyles.hasOwnProperty(s) && props[s] === true) { |
|
|
if (buttonStyles.hasOwnProperty(s) && props[s] === true) { |
|
|
const style = buttonStyles[s][state] |
|
|
const style = buttonStyles[s][state] |
|
|
if (style) { |
|
|
if (style) { |
|
|
|
|
|
hasModifier = true |
|
|
output += style(props) |
|
|
output += style(props) |
|
|
} |
|
|
} |
|
|
} |
|
|
} |
|
|
} |
|
|
} |
|
|
|
|
|
if (!hasModifier) { |
|
|
|
|
|
const defaultStyle = buttonStyles.default[state] |
|
|
|
|
|
if (defaultStyle) { |
|
|
|
|
|
output += defaultStyle(props) || '' |
|
|
|
|
|
} |
|
|
|
|
|
} |
|
|
return output |
|
|
return output |
|
|
} |
|
|
} |
|
|
|
|
|
|
|
@ -176,9 +185,38 @@ type Props = { |
|
|
eventProperties?: Object, |
|
|
eventProperties?: Object, |
|
|
} |
|
|
} |
|
|
|
|
|
|
|
|
const Button = (props: Props) => { |
|
|
class Button extends PureComponent< |
|
|
const { disabled } = props |
|
|
Props, |
|
|
const { onClick, children, isLoading, event, eventProperties, ...rest } = props |
|
|
{ |
|
|
|
|
|
isFocused: boolean, |
|
|
|
|
|
}, |
|
|
|
|
|
> { |
|
|
|
|
|
static defaultProps = { |
|
|
|
|
|
onClick: noop, |
|
|
|
|
|
primary: false, |
|
|
|
|
|
small: false, |
|
|
|
|
|
padded: false, |
|
|
|
|
|
danger: false, |
|
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
state = { |
|
|
|
|
|
isFocused: false, |
|
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
handleFocus = () => { |
|
|
|
|
|
if (isGlobalTabEnabled()) { |
|
|
|
|
|
this.setState({ isFocused: true }) |
|
|
|
|
|
} |
|
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
handleBlur = () => { |
|
|
|
|
|
this.setState({ isFocused: false }) |
|
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
render() { |
|
|
|
|
|
const { isFocused } = this.state |
|
|
|
|
|
const { disabled } = this.props |
|
|
|
|
|
const { onClick, children, isLoading, event, eventProperties, ...rest } = this.props |
|
|
const isClickDisabled = disabled || isLoading |
|
|
const isClickDisabled = disabled || isLoading |
|
|
const onClickHandler = e => { |
|
|
const onClickHandler = e => { |
|
|
if (onClick) { |
|
|
if (onClick) { |
|
@ -189,21 +227,17 @@ const Button = (props: Props) => { |
|
|
} |
|
|
} |
|
|
} |
|
|
} |
|
|
return ( |
|
|
return ( |
|
|
<Base {...rest} onClick={isClickDisabled ? undefined : onClickHandler}> |
|
|
<Base |
|
|
|
|
|
{...rest} |
|
|
|
|
|
onClick={isClickDisabled ? undefined : onClickHandler} |
|
|
|
|
|
isFocused={isFocused} |
|
|
|
|
|
onFocus={this.handleFocus} |
|
|
|
|
|
onBlur={this.handleBlur} |
|
|
|
|
|
> |
|
|
{isLoading ? <Spinner size={16} /> : children} |
|
|
{isLoading ? <Spinner size={16} /> : children} |
|
|
</Base> |
|
|
</Base> |
|
|
) |
|
|
) |
|
|
} |
|
|
} |
|
|
|
|
|
|
|
|
Button.defaultProps = { |
|
|
|
|
|
children: undefined, |
|
|
|
|
|
disabled: undefined, |
|
|
|
|
|
icon: undefined, |
|
|
|
|
|
onClick: noop, |
|
|
|
|
|
primary: false, |
|
|
|
|
|
small: false, |
|
|
|
|
|
padded: false, |
|
|
|
|
|
danger: false, |
|
|
|
|
|
} |
|
|
} |
|
|
|
|
|
|
|
|
export default Button |
|
|
export default Button |
|
|