|
@ -4,7 +4,6 @@ import React, { PureComponent } from 'react' |
|
|
import Downshift from 'downshift' |
|
|
import Downshift from 'downshift' |
|
|
import styled from 'styled-components' |
|
|
import styled from 'styled-components' |
|
|
import { space } from 'styled-system' |
|
|
import { space } from 'styled-system' |
|
|
import get from 'lodash/get' |
|
|
|
|
|
|
|
|
|
|
|
import type { Element } from 'react' |
|
|
import type { Element } from 'react' |
|
|
|
|
|
|
|
@ -18,18 +17,19 @@ import Text from 'components/base/Text' |
|
|
import Triangles from './Triangles' |
|
|
import Triangles from './Triangles' |
|
|
|
|
|
|
|
|
type Props = { |
|
|
type Props = { |
|
|
|
|
|
fuseOptions?: Object, |
|
|
|
|
|
highlight?: boolean, |
|
|
items: Array<any>, |
|
|
items: Array<any>, |
|
|
value?: Object | null, |
|
|
|
|
|
itemToString?: Function, |
|
|
itemToString?: Function, |
|
|
|
|
|
keyProp?: string, |
|
|
|
|
|
maxHeight?: number, |
|
|
onChange?: Function, |
|
|
onChange?: Function, |
|
|
fuseOptions?: Object, |
|
|
|
|
|
highlight?: boolean, |
|
|
|
|
|
searchable?: boolean, |
|
|
|
|
|
placeholder?: string, |
|
|
placeholder?: string, |
|
|
renderHighlight?: string => Element<*>, |
|
|
renderHighlight?: string => Element<*>, |
|
|
renderSelected?: any => Element<*>, |
|
|
|
|
|
renderItem?: (*) => Element<*>, |
|
|
renderItem?: (*) => Element<*>, |
|
|
keyProp?: string, |
|
|
renderSelected?: any => Element<*>, |
|
|
|
|
|
searchable?: boolean, |
|
|
|
|
|
value?: Object | null, |
|
|
} |
|
|
} |
|
|
|
|
|
|
|
|
const Container = styled(Box).attrs({ relative: true, color: 'steel' })`` |
|
|
const Container = styled(Box).attrs({ relative: true, color: 'steel' })`` |
|
@ -113,13 +113,16 @@ class Select extends PureComponent<Props> { |
|
|
static defaultProps = { |
|
|
static defaultProps = { |
|
|
itemToString: (item: Object) => item && item.name, |
|
|
itemToString: (item: Object) => item && item.name, |
|
|
keyProp: undefined, |
|
|
keyProp: undefined, |
|
|
|
|
|
maxHeight: 300, |
|
|
} |
|
|
} |
|
|
|
|
|
|
|
|
_scrollToSelectedItem = true |
|
|
_scrollToSelectedItem = true |
|
|
|
|
|
_oldHighlightedIndex = 0 |
|
|
_useKeyboard = false |
|
|
_useKeyboard = false |
|
|
|
|
|
_children = {} |
|
|
|
|
|
|
|
|
renderItems = (items: Array<Object>, selectedItem: any, downshiftProps: Object) => { |
|
|
renderItems = (items: Array<Object>, selectedItem: any, downshiftProps: Object) => { |
|
|
const { renderItem, keyProp } = this.props |
|
|
const { renderItem, maxHeight, keyProp } = this.props |
|
|
const { getItemProps, highlightedIndex } = downshiftProps |
|
|
const { getItemProps, highlightedIndex } = downshiftProps |
|
|
|
|
|
|
|
|
const selectedItemIndex = items.indexOf(selectedItem) |
|
|
const selectedItemIndex = items.indexOf(selectedItem) |
|
@ -128,29 +131,36 @@ class Select extends PureComponent<Props> { |
|
|
<Dropdown> |
|
|
<Dropdown> |
|
|
{items.length ? ( |
|
|
{items.length ? ( |
|
|
<GrowScroll |
|
|
<GrowScroll |
|
|
maxHeight={300} |
|
|
maxHeight={maxHeight} |
|
|
onUpdate={scrollbar => { |
|
|
onUpdate={scrollbar => { |
|
|
const { contentEl } = scrollbar |
|
|
const currentHighlighted = this._children[highlightedIndex] |
|
|
const children = get(contentEl, 'children[0].children[0].children', {}) |
|
|
const currentSelectedItem = this._children[selectedItemIndex] |
|
|
|
|
|
|
|
|
const currentHighlighted = children[highlightedIndex] |
|
|
|
|
|
const currentSelectedItem = children[selectedItemIndex] |
|
|
|
|
|
|
|
|
|
|
|
if (this._useKeyboard && currentHighlighted) { |
|
|
if (this._useKeyboard && currentHighlighted) { |
|
|
scrollbar.scrollIntoView(currentHighlighted, { |
|
|
scrollbar.scrollIntoView(currentHighlighted, { |
|
|
alignToTop: false, |
|
|
alignToTop: highlightedIndex < this._oldHighlightedIndex, |
|
|
|
|
|
offsetTop: -1, |
|
|
|
|
|
onlyScrollIfNeeded: true, |
|
|
}) |
|
|
}) |
|
|
} else if (this._scrollToSelectedItem && currentSelectedItem) { |
|
|
} else if (this._scrollToSelectedItem && currentSelectedItem) { |
|
|
scrollbar.scrollIntoView(currentSelectedItem, { |
|
|
window.requestAnimationFrame(() => |
|
|
alignToTop: false, |
|
|
scrollbar.scrollIntoView(currentSelectedItem, { |
|
|
}) |
|
|
offsetTop: -1, |
|
|
|
|
|
}), |
|
|
|
|
|
) |
|
|
|
|
|
|
|
|
this._scrollToSelectedItem = false |
|
|
this._scrollToSelectedItem = false |
|
|
} |
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
this._oldHighlightedIndex = highlightedIndex |
|
|
}} |
|
|
}} |
|
|
> |
|
|
> |
|
|
{items.map((item, i) => ( |
|
|
{items.map((item, i) => ( |
|
|
<ItemWrapper key={keyProp ? item[keyProp] : item.key} {...getItemProps({ item })}> |
|
|
<ItemWrapper |
|
|
|
|
|
key={keyProp ? item[keyProp] : item.key} |
|
|
|
|
|
innerRef={n => (this._children[i] = n)} |
|
|
|
|
|
{...getItemProps({ item })} |
|
|
|
|
|
> |
|
|
<Item highlighted={i === highlightedIndex} horizontal flow={10}> |
|
|
<Item highlighted={i === highlightedIndex} horizontal flow={10}> |
|
|
<Box grow> |
|
|
<Box grow> |
|
|
{renderItem ? ( |
|
|
{renderItem ? ( |
|
|