meriadec
7 years ago
7 changed files with 171 additions and 49 deletions
@ -0,0 +1,101 @@ |
|||||
|
// @flow
|
||||
|
|
||||
|
import React, { PureComponent } from 'react' |
||||
|
import styled from 'styled-components' |
||||
|
import noop from 'lodash/noop' |
||||
|
import Downshift from 'downshift' |
||||
|
|
||||
|
import Box from 'components/base/Box' |
||||
|
|
||||
|
type ItemType = { |
||||
|
key: string, |
||||
|
label: any, |
||||
|
} |
||||
|
|
||||
|
type Props = { |
||||
|
children: any, |
||||
|
items: Array<ItemType>, |
||||
|
value?: ItemType | null, |
||||
|
onChange?: ItemType => void, |
||||
|
} |
||||
|
|
||||
|
const Trigger = styled(Box)` |
||||
|
outline: none; |
||||
|
cursor: pointer; |
||||
|
` |
||||
|
|
||||
|
const Drop = styled(Box).attrs({ |
||||
|
bg: 'white', |
||||
|
boxShadow: 0, |
||||
|
})` |
||||
|
position: absolute; |
||||
|
top: 100%; |
||||
|
right: 0; |
||||
|
` |
||||
|
|
||||
|
const Item = styled(Box).attrs({ |
||||
|
py: 2, |
||||
|
px: 4, |
||||
|
bg: p => (p.isHighlighted ? 'pearl' : ''), |
||||
|
})` |
||||
|
white-space: nowrap; |
||||
|
` |
||||
|
|
||||
|
function itemToString(item) { |
||||
|
return item ? item.label : '' |
||||
|
} |
||||
|
|
||||
|
class DropDown extends PureComponent<Props> { |
||||
|
static defaultProps = { |
||||
|
value: null, |
||||
|
onChange: noop, |
||||
|
} |
||||
|
|
||||
|
renderItems = (items: Array<ItemType>, selectedItem: ItemType, downshiftProps: Object) => { |
||||
|
const { getItemProps, highlightedIndex } = downshiftProps |
||||
|
|
||||
|
return ( |
||||
|
<Drop> |
||||
|
{items.map((item, i) => ( |
||||
|
<Item isHighlighted={highlightedIndex === i} key={item.key} {...getItemProps({ item })}> |
||||
|
{item.label} |
||||
|
</Item> |
||||
|
))} |
||||
|
</Drop> |
||||
|
) |
||||
|
} |
||||
|
|
||||
|
render() { |
||||
|
const { children, items, value, onChange, ...props } = this.props |
||||
|
return ( |
||||
|
<Downshift |
||||
|
onChange={onChange} |
||||
|
itemToString={itemToString} |
||||
|
selectedItem={value} |
||||
|
render={({ |
||||
|
getButtonProps, |
||||
|
getRootProps, |
||||
|
isOpen, |
||||
|
openMenu, |
||||
|
selectedItem, |
||||
|
...downshiftProps |
||||
|
}) => ( |
||||
|
<Box |
||||
|
{...getRootProps({ refKey: 'innerRef' })} |
||||
|
horizontal |
||||
|
align="center" |
||||
|
relative |
||||
|
{...props} |
||||
|
> |
||||
|
<Trigger {...getButtonProps()} tabIndex={0}> |
||||
|
{children} |
||||
|
</Trigger> |
||||
|
{isOpen && this.renderItems(items, selectedItem, downshiftProps)} |
||||
|
</Box> |
||||
|
)} |
||||
|
/> |
||||
|
) |
||||
|
} |
||||
|
} |
||||
|
|
||||
|
export default DropDown |
Loading…
Reference in new issue