mirror of
https://github.com/fergalmoran/turnstone.git
synced 2025-12-22 09:49:56 +00:00
Ability to include sublocations
This commit is contained in:
@@ -26,11 +26,15 @@ export default function ItemContents(props) {
|
|||||||
index,
|
index,
|
||||||
item,
|
item,
|
||||||
query,
|
query,
|
||||||
searchType = 'startswith'
|
searchType = 'startswith',
|
||||||
|
setSelected,
|
||||||
|
totalItems
|
||||||
} = props
|
} = props
|
||||||
|
|
||||||
const globalMatch = searchType === 'contains'
|
const globalMatch = searchType === 'contains'
|
||||||
|
|
||||||
|
const isNYC = item.name === 'New York City, New York, United States'
|
||||||
|
|
||||||
const matchedText = (includeSeparator) => {
|
const matchedText = (includeSeparator) => {
|
||||||
return (
|
return (
|
||||||
<SplitMatch
|
<SplitMatch
|
||||||
@@ -47,16 +51,73 @@ export default function ItemContents(props) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
const img = () => {
|
const img = () => {
|
||||||
return item.name === 'New York City, New York, United States'
|
return isNYC
|
||||||
? <div><img src={imgNewYork} alt={item.name} /></div>
|
? <div className={styles.imgContainer}><img src={imgNewYork} alt={item.name} /></div>
|
||||||
: void 0
|
: void 0
|
||||||
}
|
}
|
||||||
|
|
||||||
|
const SubLocations = () => {
|
||||||
|
if(totalItems <= 5 && (item.neighbourhoods || item.attractions)) {
|
||||||
|
const neighbourhoods = (totalItems > 1) ? item.neighbourhoods.slice(0,5) : item.neighbourhoods
|
||||||
|
const attractions = (totalItems > 1) ? item.attractions.slice(0,5) : item.attractions
|
||||||
|
|
||||||
|
return (
|
||||||
|
<div className={styles.sublocations}>
|
||||||
|
<SubLocationList title="Neighbourhoods" data={neighbourhoods} />
|
||||||
|
<SubLocationList title="Attractions" data={attractions} />
|
||||||
|
</div>
|
||||||
|
)
|
||||||
|
}
|
||||||
|
return null
|
||||||
|
}
|
||||||
|
|
||||||
|
const SubLocationList = (props) => {
|
||||||
|
const { title, data } = props
|
||||||
|
|
||||||
|
if(!data || !data.length) return null
|
||||||
|
|
||||||
|
return (
|
||||||
|
<div className={styles.sublocationList}>
|
||||||
|
<div className={styles.subLocationTitle}>{title}</div>
|
||||||
|
<>
|
||||||
|
{
|
||||||
|
data.map(
|
||||||
|
(value, index) =>
|
||||||
|
<SubLocation
|
||||||
|
key={`neighbourhood${index}`}
|
||||||
|
value={value}>
|
||||||
|
{value.name.split(',')[0]}
|
||||||
|
</SubLocation>
|
||||||
|
)
|
||||||
|
}
|
||||||
|
</>
|
||||||
|
</div>
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
const SubLocation = (props) => {
|
||||||
|
const { children, value } = props
|
||||||
|
|
||||||
|
const handleClick = (evt, value) => {
|
||||||
|
evt.stopPropagation()
|
||||||
|
setSelected(value, 'name')
|
||||||
|
}
|
||||||
|
|
||||||
|
return (
|
||||||
|
<div>
|
||||||
|
<div className={styles.sublocation} onMouseDown={(evt) => handleClick(evt, value)}>
|
||||||
|
{children}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
const firstItem = () => {
|
const firstItem = () => {
|
||||||
return (
|
return (
|
||||||
<div className={`${styles.container} ${styles.first}`}>
|
<div className={`${styles.container} ${styles.first}`}>
|
||||||
{img()}
|
{img()}
|
||||||
<div>{matchedText(false)}</div>
|
<div className={styles.nameContainer}>{matchedText(false)}</div>
|
||||||
|
{SubLocations()}
|
||||||
</div>
|
</div>
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,4 +1,5 @@
|
|||||||
.first > div {
|
.first > .imgContainer,
|
||||||
|
.first > .nameContainer {
|
||||||
display: inline-block;
|
display: inline-block;
|
||||||
height: 50px;
|
height: 50px;
|
||||||
vertical-align: middle;
|
vertical-align: middle;
|
||||||
@@ -25,4 +26,36 @@
|
|||||||
.match {
|
.match {
|
||||||
font-weight: 600;
|
font-weight: 600;
|
||||||
color: #000;
|
color: #000;
|
||||||
|
}
|
||||||
|
|
||||||
|
.sublocations {
|
||||||
|
display: block;
|
||||||
|
padding-top: 10px;
|
||||||
|
padding-left: 48px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.sublocationList {
|
||||||
|
display: inline-block;
|
||||||
|
margin-right: 40px;
|
||||||
|
font-size: 15px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.subLocationTitle {
|
||||||
|
text-transform: uppercase;
|
||||||
|
font-size: 0.8em;
|
||||||
|
color: #777;
|
||||||
|
margin-bottom: 3px;
|
||||||
|
margin-left: 10px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.sublocation {
|
||||||
|
display:inline-block;
|
||||||
|
padding: 2px 10px 3px;
|
||||||
|
color: rgb(118, 58, 214);
|
||||||
|
}
|
||||||
|
|
||||||
|
.sublocation:hover {
|
||||||
|
color: #fff;
|
||||||
|
background: rgb(118, 58, 214);
|
||||||
|
border-radius: 15px;
|
||||||
}
|
}
|
||||||
@@ -1,6 +1,6 @@
|
|||||||
{
|
{
|
||||||
"name": "turnstone",
|
"name": "turnstone",
|
||||||
"version": "0.4.2",
|
"version": "0.5.0",
|
||||||
"description": "In development. React multi-search autocomplete component.",
|
"description": "In development. React multi-search autocomplete component.",
|
||||||
"author": "Tom Southall",
|
"author": "Tom Southall",
|
||||||
"keywords": [],
|
"keywords": [],
|
||||||
|
|||||||
@@ -51,11 +51,11 @@ export const highlightNext = () => {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
export const setSelected = (index) => {
|
export const setSelected = (i) => {
|
||||||
return {
|
const type = types.SET_SELECTED
|
||||||
type: types.SET_SELECTED,
|
return (typeof i === 'object')
|
||||||
index
|
? { type, item: i }
|
||||||
}
|
: { type, index: i }
|
||||||
}
|
}
|
||||||
|
|
||||||
export const clearSelected = () => {
|
export const clearSelected = () => {
|
||||||
|
|||||||
@@ -61,12 +61,21 @@ describe('Actions', () => {
|
|||||||
})
|
})
|
||||||
})
|
})
|
||||||
|
|
||||||
test('setSelected returns expected action', () => {
|
test('setSelected returns expected actions', () => {
|
||||||
const action = actions.setSelected(0)
|
const item = {value: {name: 'foobar'}, text: 'foobar'}
|
||||||
|
let action = actions.setSelected(0)
|
||||||
|
|
||||||
expect(action).toEqual({
|
expect(action).toEqual({
|
||||||
type: 'SET_SELECTED',
|
type: 'SET_SELECTED',
|
||||||
index: 0
|
index: 0
|
||||||
})
|
})
|
||||||
|
|
||||||
|
action = actions.setSelected(item)
|
||||||
|
|
||||||
|
expect(action).toEqual({
|
||||||
|
type: 'SET_SELECTED',
|
||||||
|
item
|
||||||
|
})
|
||||||
})
|
})
|
||||||
|
|
||||||
test('clearSelected returns expected action', () => {
|
test('clearSelected returns expected action', () => {
|
||||||
|
|||||||
@@ -37,6 +37,14 @@ export default function Item(props) {
|
|||||||
dispatch(setSelected(index))
|
dispatch(setSelected(index))
|
||||||
}
|
}
|
||||||
|
|
||||||
|
const setCustomSelected = (value, displayField) => {
|
||||||
|
dispatch(setSelected({
|
||||||
|
value,
|
||||||
|
displayField,
|
||||||
|
text: value[displayField]
|
||||||
|
}))
|
||||||
|
}
|
||||||
|
|
||||||
const itemContents = (ItemContents)
|
const itemContents = (ItemContents)
|
||||||
? <ItemContents
|
? <ItemContents
|
||||||
appearsInDefaultListbox={item.defaultListbox}
|
appearsInDefaultListbox={item.defaultListbox}
|
||||||
@@ -45,6 +53,7 @@ export default function Item(props) {
|
|||||||
item={item.value}
|
item={item.value}
|
||||||
query={query}
|
query={query}
|
||||||
searchType={item.searchType}
|
searchType={item.searchType}
|
||||||
|
setSelected={setCustomSelected}
|
||||||
totalItems={state.items.length}
|
totalItems={state.items.length}
|
||||||
/>
|
/>
|
||||||
: (state.props.matchText
|
: (state.props.matchText
|
||||||
|
|||||||
@@ -1,5 +1,6 @@
|
|||||||
import * as types from '../actions/actionTypes'
|
import * as types from '../actions/actionTypes'
|
||||||
import undef from '../utils/undef'
|
import undef from '../utils/undef'
|
||||||
|
import isUndefined from '../utils/isUndefined'
|
||||||
|
|
||||||
const highlightedItem = (index, items) => {
|
const highlightedItem = (index, items) => {
|
||||||
if(!items[index]) return undef
|
if(!items[index]) return undef
|
||||||
@@ -8,7 +9,7 @@ const highlightedItem = (index, items) => {
|
|||||||
|
|
||||||
const reducer = (state, action) => {
|
const reducer = (state, action) => {
|
||||||
const newState = (() => {
|
const newState = (() => {
|
||||||
let newState
|
let newState, item
|
||||||
|
|
||||||
switch (action.type) {
|
switch (action.type) {
|
||||||
case types.SET_QUERY:
|
case types.SET_QUERY:
|
||||||
@@ -66,10 +67,8 @@ const reducer = (state, action) => {
|
|||||||
? { highlighted: highlightedItem(state.highlighted.index + 1, state.items) }
|
? { highlighted: highlightedItem(state.highlighted.index + 1, state.items) }
|
||||||
: {}
|
: {}
|
||||||
case types.SET_SELECTED:
|
case types.SET_SELECTED:
|
||||||
return {
|
item = isUndefined(action.index) ? action.item : state.items[action.index]
|
||||||
selected: state.items[action.index],
|
return { selected: item, query: item.text }
|
||||||
query: state.items[action.index].text,
|
|
||||||
}
|
|
||||||
default:
|
default:
|
||||||
throw new Error('Invalid action type passed to reducer')
|
throw new Error('Invalid action type passed to reducer')
|
||||||
}
|
}
|
||||||
|
|||||||
Reference in New Issue
Block a user