Browse Source

Button now listen global focus state (tab key)

master
meriadec 7 years ago
parent
commit
037fdbaa17
No known key found for this signature in database GPG Key ID: 1D2FC2305E2CB399
  1. 106
      src/components/base/Button/index.js

106
src/components/base/Button/index.js

@ -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

Loading…
Cancel
Save