Merge pull request #306 from orangemug/feature/accessibility-list-reorder

Keyboard accessible layer options
This commit is contained in:
Orange Mug 2018-06-03 09:57:00 +01:00 committed by GitHub
commit edd09ef585
No known key found for this signature in database
GPG key ID: 4AEE18F83AFDEB23
8 changed files with 245 additions and 57 deletions

View file

@ -7,14 +7,14 @@ templates:
name: "Create artifacts directory" name: "Create artifacts directory"
command: mkdir /tmp/artifacts command: mkdir /tmp/artifacts
- restore_cache: - restore_cache:
key: v1-dependencies-{{ checksum "package.json" }} key: v1-dependencies-{{ arch }}-{{ checksum "package.json" }}
- run: npm install - run: npm install
- save_cache: - save_cache:
paths: paths:
- node_modules - node_modules
key: v1-dependencies-{{ checksum "package.json" }} key: v1-dependencies-{{ arch }}-{{ checksum "package.json" }}
- run: mkdir -p /tmp/artifacts/logs - run: mkdir -p /tmp/artifacts/logs
- run: npm run build - run: npm run build
@ -30,14 +30,14 @@ templates:
name: "Create artifacts directory" name: "Create artifacts directory"
command: mkdir /tmp/artifacts command: mkdir /tmp/artifacts
- restore_cache: - restore_cache:
key: v1-dependencies-{{ checksum "package.json" }} key: v1-dependencies-{{ arch }}-{{ checksum "package.json" }}
- run: npm install - run: npm install
- save_cache: - save_cache:
paths: paths:
- node_modules - node_modules
key: v1-dependencies-{{ checksum "package.json" }} key: v1-dependencies-{{ arch }}-{{ checksum "package.json" }}
- run: mkdir -p /tmp/artifacts/logs - run: mkdir -p /tmp/artifacts/logs
- run: npm run build - run: npm run build
@ -71,7 +71,7 @@ jobs:
dependencies: dependencies:
override: override:
- brew install node@6 - brew install node@6
working_directory: ~/repo-linux-node-v6 working_directory: ~/repo-osx-node-v6
steps: *build-steps steps: *build-steps
build-osx-node-v8: build-osx-node-v8:
macos: macos:
@ -79,7 +79,7 @@ jobs:
dependencies: dependencies:
override: override:
- brew install node@8 - brew install node@8
working_directory: ~/repo-linux-node-v8 working_directory: ~/repo-osx-node-v8
steps: *build-steps steps: *build-steps
build-osx-node-v10: build-osx-node-v10:
macos: macos:
@ -87,7 +87,7 @@ jobs:
dependencies: dependencies:
override: override:
- brew install node@10 - brew install node@10
working_directory: ~/repo-linux-node-v10 working_directory: ~/repo-osx-node-v10
steps: *build-steps steps: *build-steps
workflows: workflows:

25
package-lock.json generated
View file

@ -5523,6 +5523,11 @@
"readable-stream": "2.3.5" "readable-stream": "2.3.5"
} }
}, },
"focus-group": {
"version": "0.3.1",
"resolved": "https://registry.npmjs.org/focus-group/-/focus-group-0.3.1.tgz",
"integrity": "sha1-4PMu2GsNq91v/OvfiY7LMuR/7c4="
},
"focus-trap": { "focus-trap": {
"version": "2.4.4", "version": "2.4.4",
"resolved": "https://registry.npmjs.org/focus-trap/-/focus-trap-2.4.4.tgz", "resolved": "https://registry.npmjs.org/focus-trap/-/focus-trap-2.4.4.tgz",
@ -8365,6 +8370,11 @@
"resolved": "https://registry.npmjs.org/lodash.capitalize/-/lodash.capitalize-4.2.1.tgz", "resolved": "https://registry.npmjs.org/lodash.capitalize/-/lodash.capitalize-4.2.1.tgz",
"integrity": "sha1-+CbJtOKoUR2E46yinbBeGk87cqk=" "integrity": "sha1-+CbJtOKoUR2E46yinbBeGk87cqk="
}, },
"lodash.clamp": {
"version": "4.0.3",
"resolved": "https://registry.npmjs.org/lodash.clamp/-/lodash.clamp-4.0.3.tgz",
"integrity": "sha1-XCS+3u7vB1NWDcK0y0Zx+Qpt36o="
},
"lodash.clonedeep": { "lodash.clonedeep": {
"version": "4.5.0", "version": "4.5.0",
"resolved": "https://registry.npmjs.org/lodash.clonedeep/-/lodash.clonedeep-4.5.0.tgz", "resolved": "https://registry.npmjs.org/lodash.clonedeep/-/lodash.clonedeep-4.5.0.tgz",
@ -11479,6 +11489,16 @@
"object-assign": "4.1.1" "object-assign": "4.1.1"
} }
}, },
"react-aria-menubutton": {
"version": "5.1.1",
"resolved": "https://registry.npmjs.org/react-aria-menubutton/-/react-aria-menubutton-5.1.1.tgz",
"integrity": "sha512-ceBjPvuqwM2jnRFsfMlpfPdyqQ5cz4STNH7NlKpxStr2uETB/zQ2sHiIUMTuqSuOszU1kgUB2vm3aVn3xdjhcA==",
"requires": {
"focus-group": "0.3.1",
"prop-types": "15.6.1",
"teeny-tap": "0.2.0"
}
},
"react-aria-modal": { "react-aria-modal": {
"version": "2.12.1", "version": "2.12.1",
"resolved": "https://registry.npmjs.org/react-aria-modal/-/react-aria-modal-2.12.1.tgz", "resolved": "https://registry.npmjs.org/react-aria-modal/-/react-aria-modal-2.12.1.tgz",
@ -14298,6 +14318,11 @@
"xtend": "4.0.1" "xtend": "4.0.1"
} }
}, },
"teeny-tap": {
"version": "0.2.0",
"resolved": "https://registry.npmjs.org/teeny-tap/-/teeny-tap-0.2.0.tgz",
"integrity": "sha1-Fn5kUYLQasIi1iuyq2eUenClimg="
},
"text-table": { "text-table": {
"version": "0.2.0", "version": "0.2.0",
"resolved": "https://registry.npmjs.org/text-table/-/text-table-0.2.0.tgz", "resolved": "https://registry.npmjs.org/text-table/-/text-table-0.2.0.tgz",

View file

@ -30,6 +30,7 @@
"github-api": "^3.0.0", "github-api": "^3.0.0",
"jsonlint": "github:josdejong/jsonlint#85a19d7", "jsonlint": "github:josdejong/jsonlint#85a19d7",
"lodash.capitalize": "^4.2.1", "lodash.capitalize": "^4.2.1",
"lodash.clamp": "^4.0.3",
"lodash.clonedeep": "^4.5.0", "lodash.clonedeep": "^4.5.0",
"lodash.isequal": "^4.5.0", "lodash.isequal": "^4.5.0",
"lodash.throttle": "^4.1.1", "lodash.throttle": "^4.1.1",
@ -42,6 +43,7 @@
"prop-types": "^15.6.0", "prop-types": "^15.6.0",
"react": "^16.3.2", "react": "^16.3.2",
"react-addons-pure-render-mixin": "^15.6.2", "react-addons-pure-render-mixin": "^15.6.2",
"react-aria-menubutton": "^5.1.1",
"react-aria-modal": "^2.12.1", "react-aria-modal": "^2.12.1",
"react-autocomplete": "^1.7.2", "react-autocomplete": "^1.7.2",
"react-codemirror2": "^4.2.1", "react-codemirror2": "^4.2.1",

View file

@ -1,5 +1,8 @@
import React from 'react' import React from 'react'
import Mousetrap from 'mousetrap' import Mousetrap from 'mousetrap'
import cloneDeep from 'lodash.clonedeep'
import clamp from 'lodash.clamp'
import {arrayMove} from 'react-sortable-hoc';
import url from 'url' import url from 'url'
import MapboxGlMap from './map/MapboxGlMap' import MapboxGlMap from './map/MapboxGlMap'
@ -168,6 +171,24 @@ export default class App extends React.Component {
}) })
} }
onMoveLayer(move) {
let { oldIndex, newIndex } = move;
let layers = this.state.mapStyle.layers;
oldIndex = clamp(oldIndex, 0, layers.length-1);
newIndex = clamp(newIndex, 0, layers.length-1);
if(oldIndex === newIndex) return;
if (oldIndex === this.state.selectedLayerIndex) {
this.setState({
selectedLayerIndex: newIndex
});
}
layers = layers.slice(0);
layers = arrayMove(layers, oldIndex, newIndex);
this.onLayersChange(layers);
}
onLayersChange(changedLayers) { onLayersChange(changedLayers) {
const changedStyle = { const changedStyle = {
...this.state.mapStyle, ...this.state.mapStyle,
@ -176,6 +197,40 @@ export default class App extends React.Component {
this.onStyleChanged(changedStyle) this.onStyleChanged(changedStyle)
} }
onLayerDestroy(layerId) {
let layers = this.state.mapStyle.layers;
const remainingLayers = layers.slice(0);
const idx = style.indexOfLayer(remainingLayers, layerId)
remainingLayers.splice(idx, 1);
this.onLayersChange(remainingLayers);
}
onLayerCopy(layerId) {
let layers = this.state.mapStyle.layers;
const changedLayers = layers.slice(0)
const idx = style.indexOfLayer(changedLayers, layerId)
const clonedLayer = cloneDeep(changedLayers[idx])
clonedLayer.id = clonedLayer.id + "-copy"
changedLayers.splice(idx, 0, clonedLayer)
this.onLayersChange(changedLayers)
}
onLayerVisibilityToggle(layerId) {
let layers = this.state.mapStyle.layers;
const changedLayers = layers.slice(0)
const idx = style.indexOfLayer(changedLayers, layerId)
const layer = { ...changedLayers[idx] }
const changedLayout = 'layout' in layer ? {...layer.layout} : {}
changedLayout.visibility = changedLayout.visibility === 'none' ? 'visible' : 'none'
layer.layout = changedLayout
changedLayers[idx] = layer
this.onLayersChange(changedLayers)
}
onLayerIdChange(oldId, newId) { onLayerIdChange(oldId, newId) {
const changedLayers = this.state.mapStyle.layers.slice(0) const changedLayers = this.state.mapStyle.layers.slice(0)
const idx = style.indexOfLayer(changedLayers, oldId) const idx = style.indexOfLayer(changedLayers, oldId)
@ -311,6 +366,10 @@ export default class App extends React.Component {
/> />
const layerList = <LayerList const layerList = <LayerList
onMoveLayer={this.onMoveLayer.bind(this)}
onLayerDestroy={this.onLayerDestroy.bind(this)}
onLayerCopy={this.onLayerCopy.bind(this)}
onLayerVisibilityToggle={this.onLayerVisibilityToggle.bind(this)}
onLayersChange={this.onLayersChange.bind(this)} onLayersChange={this.onLayersChange.bind(this)}
onLayerSelect={this.onLayerSelect.bind(this)} onLayerSelect={this.onLayerSelect.bind(this)}
selectedLayerIndex={this.state.selectedLayerIndex} selectedLayerIndex={this.state.selectedLayerIndex}
@ -320,10 +379,17 @@ export default class App extends React.Component {
const layerEditor = selectedLayer ? <LayerEditor const layerEditor = selectedLayer ? <LayerEditor
layer={selectedLayer} layer={selectedLayer}
layerIndex={this.state.selectedLayerIndex}
isFirstLayer={this.state.selectedLayerIndex < 1}
isLastLayer={this.state.selectedLayerIndex === this.state.mapStyle.layers.length-1}
sources={this.state.sources} sources={this.state.sources}
vectorLayers={this.state.vectorLayers} vectorLayers={this.state.vectorLayers}
spec={this.state.spec} spec={this.state.spec}
onMoveLayer={this.onMoveLayer.bind(this)}
onLayerChanged={this.onLayerChanged.bind(this)} onLayerChanged={this.onLayerChanged.bind(this)}
onLayerDestroy={this.onLayerDestroy.bind(this)}
onLayerCopy={this.onLayerCopy.bind(this)}
onLayerVisibilityToggle={this.onLayerVisibilityToggle.bind(this)}
onLayerIdChange={this.onLayerIdChange.bind(this)} onLayerIdChange={this.onLayerIdChange.bind(this)}
/> : null /> : null

View file

@ -1,5 +1,6 @@
import React from 'react' import React from 'react'
import PropTypes from 'prop-types' import PropTypes from 'prop-types'
import { Wrapper, Button, Menu, MenuItem } from 'react-aria-menubutton'
import JSONEditor from './JSONEditor' import JSONEditor from './JSONEditor'
import FilterEditor from '../filter/FilterEditor' import FilterEditor from '../filter/FilterEditor'
@ -13,6 +14,8 @@ import CommentBlock from './CommentBlock'
import LayerSourceBlock from './LayerSourceBlock' import LayerSourceBlock from './LayerSourceBlock'
import LayerSourceLayerBlock from './LayerSourceLayerBlock' import LayerSourceLayerBlock from './LayerSourceLayerBlock'
import MoreVertIcon from 'react-icons/lib/md/more-vert'
import InputBlock from '../inputs/InputBlock' import InputBlock from '../inputs/InputBlock'
import MultiButtonInput from '../inputs/MultiButtonInput' import MultiButtonInput from '../inputs/MultiButtonInput'
@ -45,6 +48,13 @@ export default class LayerEditor extends React.Component {
spec: PropTypes.object.isRequired, spec: PropTypes.object.isRequired,
onLayerChanged: PropTypes.func, onLayerChanged: PropTypes.func,
onLayerIdChange: PropTypes.func, onLayerIdChange: PropTypes.func,
onMoveLayer: PropTypes.func,
onLayerDestroy: PropTypes.func,
onLayerCopy: PropTypes.func,
onLayerVisibilityToggle: PropTypes.func,
isFirstLayer: PropTypes.bool,
isLastLayer: PropTypes.bool,
layerIndex: PropTypes.number,
} }
static defaultProps = { static defaultProps = {
@ -176,6 +186,13 @@ export default class LayerEditor extends React.Component {
} }
} }
moveLayer(offset) {
this.props.onMoveLayer({
oldIndex: this.props.layerIndex,
newIndex: this.props.layerIndex+offset
})
}
render() { render() {
const layerType = this.props.layer.type const layerType = this.props.layer.type
const groups = layoutGroups(layerType).filter(group => { const groups = layoutGroups(layerType).filter(group => {
@ -192,8 +209,73 @@ export default class LayerEditor extends React.Component {
</LayerEditorGroup> </LayerEditorGroup>
}) })
const layout = this.props.layer.layout || {}
const items = {
delete: {
text: "Delete",
handler: () => this.props.onLayerDestroy(this.props.layer.id)
},
duplicate: {
text: "Duplicate",
handler: () => this.props.onLayerCopy(this.props.layer.id)
},
hide: {
text: (layout.visibility === "none") ? "Show" : "Hide",
handler: () => this.props.onLayerVisibilityToggle(this.props.layer.id)
},
moveLayerUp: {
text: "Move layer up",
// Not actually used...
disabled: this.props.isFirstLayer,
handler: () => this.moveLayer(-1)
},
moveLayerDown: {
text: "Move layer down",
// Not actually used...
disabled: this.props.isLastLayer,
handler: () => this.moveLayer(+1)
}
}
function handleSelection(id, event) {
event.stopPropagation;
items[id].handler();
}
return <div className="maputnik-layer-editor" return <div className="maputnik-layer-editor"
> >
<header>
<div className="layer-header">
<h2 className="layer-header__title">
Layer: {this.props.layer.id}
</h2>
<div className="layer-header__info">
<Wrapper
className='more-menu'
onSelection={handleSelection}
closeOnSelection={false}
>
<Button className='more-menu__button'>
<MoreVertIcon className="more-menu__button__svg" />
</Button>
<Menu>
<ul className="more-menu__menu">
{Object.keys(items).map((id, idx) => {
const item = items[id];
return <li key={id}>
<MenuItem value={id} className='more-menu__menu__item'>
{item.text}
</MenuItem>
</li>
})}
</ul>
</Menu>
</Wrapper>
</div>
</div>
</header>
{groups} {groups}
</div> </div>
} }

View file

@ -1,7 +1,6 @@
import React from 'react' import React from 'react'
import PropTypes from 'prop-types' import PropTypes from 'prop-types'
import classnames from 'classnames' import classnames from 'classnames'
import cloneDeep from 'lodash.clonedeep'
import Button from '../Button' import Button from '../Button'
import LayerListGroup from './LayerListGroup' import LayerListGroup from './LayerListGroup'
@ -10,7 +9,7 @@ import AddIcon from 'react-icons/lib/md/add-circle-outline'
import AddModal from '../modals/AddModal' import AddModal from '../modals/AddModal'
import style from '../../libs/style.js' import style from '../../libs/style.js'
import {SortableContainer, SortableHandle, arrayMove} from 'react-sortable-hoc'; import {SortableContainer, SortableHandle} from 'react-sortable-hoc';
const layerListPropTypes = { const layerListPropTypes = {
layers: PropTypes.array.isRequired, layers: PropTypes.array.isRequired,
@ -57,36 +56,6 @@ class LayerListContainer extends React.Component {
} }
} }
onLayerDestroy(layerId) {
const remainingLayers = this.props.layers.slice(0)
const idx = style.indexOfLayer(remainingLayers, layerId)
remainingLayers.splice(idx, 1);
this.props.onLayersChange(remainingLayers)
}
onLayerCopy(layerId) {
const changedLayers = this.props.layers.slice(0)
const idx = style.indexOfLayer(changedLayers, layerId)
const clonedLayer = cloneDeep(changedLayers[idx])
clonedLayer.id = clonedLayer.id + "-copy"
changedLayers.splice(idx, 0, clonedLayer)
this.props.onLayersChange(changedLayers)
}
onLayerVisibilityToggle(layerId) {
const changedLayers = this.props.layers.slice(0)
const idx = style.indexOfLayer(changedLayers, layerId)
const layer = { ...changedLayers[idx] }
const changedLayout = 'layout' in layer ? {...layer.layout} : {}
changedLayout.visibility = changedLayout.visibility === 'none' ? 'visible' : 'none'
layer.layout = changedLayout
changedLayers[idx] = layer
this.props.onLayersChange(changedLayers)
}
toggleModal(modalName) { toggleModal(modalName) {
this.setState({ this.setState({
isOpen: { isOpen: {
@ -186,9 +155,9 @@ class LayerListContainer extends React.Component {
visibility={(layer.layout || {}).visibility} visibility={(layer.layout || {}).visibility}
isSelected={idx === this.props.selectedLayerIndex} isSelected={idx === this.props.selectedLayerIndex}
onLayerSelect={this.props.onLayerSelect} onLayerSelect={this.props.onLayerSelect}
onLayerDestroy={this.onLayerDestroy.bind(this)} onLayerDestroy={this.props.onLayerDestroy.bind(this)}
onLayerCopy={this.onLayerCopy.bind(this)} onLayerCopy={this.props.onLayerCopy.bind(this)}
onLayerVisibilityToggle={this.onLayerVisibilityToggle.bind(this)} onLayerVisibilityToggle={this.props.onLayerVisibilityToggle.bind(this)}
/> />
listItems.push(listItem) listItems.push(listItem)
idx += 1 idx += 1
@ -237,18 +206,10 @@ class LayerListContainer extends React.Component {
export default class LayerList extends React.Component { export default class LayerList extends React.Component {
static propTypes = {...layerListPropTypes} static propTypes = {...layerListPropTypes}
onSortEnd(move) {
const { oldIndex, newIndex } = move
if(oldIndex === newIndex) return
let layers = this.props.layers.slice(0)
layers = arrayMove(layers, oldIndex, newIndex)
this.props.onLayersChange(layers)
}
render() { render() {
return <LayerListContainer return <LayerListContainer
{...this.props} {...this.props}
onSortEnd={this.onSortEnd.bind(this)} onSortEnd={this.props.onMoveLayer.bind(this)}
useDragHandle={true} useDragHandle={true}
/> />
} }

View file

@ -38,7 +38,7 @@ class IconAction extends React.Component {
renderIcon() { renderIcon() {
switch(this.props.action) { switch(this.props.action) {
case 'copy': return <CopyIcon /> case 'duplicate': return <CopyIcon />
case 'show': return <VisibilityIcon /> case 'show': return <VisibilityIcon />
case 'hide': return <VisibilityOffIcon /> case 'hide': return <VisibilityOffIcon />
case 'delete': return <DeleteIcon /> case 'delete': return <DeleteIcon />
@ -46,13 +46,15 @@ class IconAction extends React.Component {
} }
render() { render() {
return <a return <button
tabIndex="-1"
title={this.props.action}
className="maputnik-layer-list-icon-action" className="maputnik-layer-list-icon-action"
data-wd-key={this.props.wdKey} data-wd-key={this.props.wdKey}
onClick={this.props.onClick} onClick={this.props.onClick}
> >
{this.renderIcon()} {this.renderIcon()}
</a> </button>
} }
} }
@ -109,7 +111,7 @@ class LayerListItem extends React.Component {
/> />
<IconAction <IconAction
wdKey={"layer-list-item:"+this.props.layerId+":copy"} wdKey={"layer-list-item:"+this.props.layerId+":copy"}
action={'copy'} action={'duplicate'}
onClick={e => this.props.onLayerCopy(this.props.layerId)} onClick={e => this.props.onLayerCopy(this.props.layerId)}
/> />
<IconAction <IconAction

View file

@ -50,14 +50,25 @@
@include flex-row; @include flex-row;
} }
&-icon-action svg { &-icon-action {
fill: $color-black; display: none;
svg {
fill: $color-black;
}
} }
.maputnik-layer-list-item:hover, .maputnik-layer-list-item:hover,
.maputnik-layer-list-item-selected { .maputnik-layer-list-item-selected {
background-color: lighten($color-black, 2); background-color: lighten($color-black, 2);
.maputnik-layer-list-icon-action {
display: block;
background: initial;
border: none;
padding: 0 2px;
}
.maputnik-layer-list-icon-action svg { .maputnik-layer-list-icon-action svg {
fill: darken($color-lowgray, 0.5); fill: darken($color-lowgray, 0.5);
@ -126,6 +137,7 @@
user-select: none; user-select: none;
padding: $margin-2; padding: $margin-2;
line-height: 20px; line-height: 20px;
border-top: solid 1px #36383e;
@include flex-row; @include flex-row;
@ -168,3 +180,41 @@
color: $color-lowgray; color: $color-lowgray;
} }
} }
.more-menu {
position: relative;
&__menu {
position: absolute;
z-index: 9999;
background: $color-black;
border: solid 1px $color-midgray;
right: 0;
min-width: 120px;
}
&__button__svg {
width: 24px;
height: 24px;
}
&__menu__item {
padding: 4px;
}
}
.layer-header {
display: flex;
padding: 6px;
background: $color-black;
&__title {
flex: 1;
margin: 0;
line-height: 24px;
}
&__info {
min-width: 28px;
}
}