2017-01-12 18:27:44 +01:00
|
|
|
import React from 'react'
|
2017-11-06 16:32:04 +01:00
|
|
|
import PropTypes from 'prop-types'
|
2017-01-13 15:55:49 +01:00
|
|
|
import { saveAs } from 'file-saver'
|
2017-01-12 18:27:44 +01:00
|
|
|
|
2018-05-17 11:44:54 +02:00
|
|
|
import * as styleSpec from '@mapbox/mapbox-gl-style-spec/style-spec'
|
2017-01-12 18:27:44 +01:00
|
|
|
import InputBlock from '../inputs/InputBlock'
|
|
|
|
import StringInput from '../inputs/StringInput'
|
2017-01-16 15:07:41 +01:00
|
|
|
import CheckboxInput from '../inputs/CheckboxInput'
|
2017-01-12 18:27:44 +01:00
|
|
|
import Button from '../Button'
|
|
|
|
import Modal from './Modal'
|
|
|
|
import MdFileDownload from 'react-icons/lib/md/file-download'
|
2017-10-16 02:59:06 +02:00
|
|
|
import TiClipboard from 'react-icons/lib/ti/clipboard'
|
2017-01-16 15:07:41 +01:00
|
|
|
import style from '../../libs/style.js'
|
2017-01-12 18:27:44 +01:00
|
|
|
import GitHub from 'github-api'
|
2017-10-16 02:59:06 +02:00
|
|
|
import { CopyToClipboard } from 'react-copy-to-clipboard'
|
2017-01-12 18:27:44 +01:00
|
|
|
|
|
|
|
|
|
|
|
class Gist extends React.Component {
|
|
|
|
static propTypes = {
|
2017-11-06 16:32:04 +01:00
|
|
|
mapStyle: PropTypes.object.isRequired,
|
|
|
|
onStyleChanged: PropTypes.func.isRequired,
|
2017-01-12 18:27:44 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
constructor(props) {
|
|
|
|
super(props);
|
2017-01-16 15:07:41 +01:00
|
|
|
this.state = {
|
|
|
|
preview: false,
|
2018-02-02 18:21:59 +01:00
|
|
|
public: false,
|
2017-01-16 15:07:41 +01:00
|
|
|
saving: false,
|
|
|
|
latestGist: null,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
componentWillReceiveProps(nextProps) {
|
|
|
|
this.setState({
|
|
|
|
...this.state,
|
2017-01-16 20:07:21 +01:00
|
|
|
preview: !!(nextProps.mapStyle.metadata || {})['maputnik:openmaptiles_access_token']
|
2017-01-16 15:07:41 +01:00
|
|
|
})
|
2017-01-12 18:27:44 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
onSave() {
|
|
|
|
this.setState({
|
2017-01-16 15:07:41 +01:00
|
|
|
...this.state,
|
2017-01-12 18:27:44 +01:00
|
|
|
saving: true
|
|
|
|
});
|
2018-02-01 20:54:44 +01:00
|
|
|
|
|
|
|
const preview = this.state.preview;
|
|
|
|
|
|
|
|
const mapboxToken = (this.props.mapStyle.metadata || {})['maputnik:mapbox_access_token'];
|
2017-01-16 15:07:41 +01:00
|
|
|
|
|
|
|
const mapStyleStr = preview ?
|
2017-04-10 14:29:57 +02:00
|
|
|
styleSpec.format(stripAccessTokens(style.replaceAccessToken(this.props.mapStyle))) :
|
|
|
|
styleSpec.format(stripAccessTokens(this.props.mapStyle));
|
2017-01-12 18:27:44 +01:00
|
|
|
const styleTitle = this.props.mapStyle.name || 'Style';
|
|
|
|
const htmlStr = `
|
|
|
|
<!DOCTYPE html>
|
|
|
|
<html>
|
|
|
|
<head>
|
|
|
|
<meta charset="UTF-8">
|
|
|
|
<meta name="viewport" content="width=device-width, initial-scale=1">
|
|
|
|
<title>`+styleTitle+` Preview</title>
|
2018-02-02 19:03:37 +01:00
|
|
|
<link rel="stylesheet" type="text/css" href="https://api.mapbox.com/mapbox-gl-js/v0.44.0/mapbox-gl.css" />
|
|
|
|
<script src="https://api.mapbox.com/mapbox-gl-js/v0.44.0/mapbox-gl.js"></script>
|
2017-01-12 18:27:44 +01:00
|
|
|
<style>
|
|
|
|
body { margin:0; padding:0; }
|
|
|
|
#map { position:absolute; top:0; bottom:0; width:100%; }
|
|
|
|
</style>
|
|
|
|
</head>
|
|
|
|
<body>
|
|
|
|
<div id='map'></div>
|
|
|
|
<script>
|
2018-02-01 20:54:44 +01:00
|
|
|
mapboxgl.accessToken = '${mapboxToken}';
|
2017-01-12 18:27:44 +01:00
|
|
|
var map = new mapboxgl.Map({
|
|
|
|
container: 'map',
|
|
|
|
style: 'style.json',
|
|
|
|
attributionControl: true,
|
|
|
|
hash: true
|
|
|
|
});
|
|
|
|
map.addControl(new mapboxgl.NavigationControl());
|
|
|
|
</script>
|
|
|
|
</body>
|
|
|
|
</html>
|
|
|
|
`
|
2017-01-16 15:07:41 +01:00
|
|
|
const files = {
|
|
|
|
"style.json": {
|
|
|
|
content: mapStyleStr
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if(preview) {
|
|
|
|
files["index.html"] = {
|
|
|
|
content: htmlStr
|
|
|
|
}
|
|
|
|
}
|
2017-01-12 18:27:44 +01:00
|
|
|
const gh = new GitHub();
|
|
|
|
let gist = gh.getGist(); // not a gist yet
|
|
|
|
gist.create({
|
2018-02-02 18:21:59 +01:00
|
|
|
public: this.state.public,
|
2017-01-16 15:07:41 +01:00
|
|
|
description: styleTitle,
|
|
|
|
files: files
|
2017-01-12 18:27:44 +01:00
|
|
|
}).then(function({data}) {
|
|
|
|
return gist.read();
|
|
|
|
}).then(function({data}) {
|
|
|
|
this.setState({
|
2017-01-16 15:07:41 +01:00
|
|
|
...this.state,
|
|
|
|
latestGist: data,
|
|
|
|
saving: false,
|
2017-01-12 18:27:44 +01:00
|
|
|
});
|
|
|
|
}.bind(this));
|
|
|
|
}
|
|
|
|
|
2017-01-16 15:07:41 +01:00
|
|
|
onPreviewChange(value) {
|
|
|
|
this.setState({
|
|
|
|
...this.state,
|
|
|
|
preview: value
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
2018-02-02 18:21:59 +01:00
|
|
|
onPublicChange(value) {
|
|
|
|
this.setState({
|
|
|
|
...this.state,
|
|
|
|
public: value
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
2017-01-16 15:07:41 +01:00
|
|
|
changeMetadataProperty(property, value) {
|
|
|
|
const changedStyle = {
|
|
|
|
...this.props.mapStyle,
|
|
|
|
metadata: {
|
|
|
|
...this.props.mapStyle.metadata,
|
|
|
|
[property]: value
|
|
|
|
}
|
|
|
|
}
|
|
|
|
this.props.onStyleChanged(changedStyle)
|
|
|
|
}
|
|
|
|
|
|
|
|
renderPreviewLink() {
|
|
|
|
const gist = this.state.latestGist;
|
|
|
|
const user = gist.user || 'anonymous';
|
|
|
|
const preview = !!gist.files['index.html'];
|
|
|
|
if(preview) {
|
2017-11-08 09:51:24 +01:00
|
|
|
return <span><a target="_blank" rel="noopener noreferrer" href={"https://bl.ocks.org/"+user+"/"+gist.id}>Preview</a>,{' '}</span>
|
2017-01-16 15:07:41 +01:00
|
|
|
}
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
2017-01-12 18:27:44 +01:00
|
|
|
renderLatestGist() {
|
|
|
|
const gist = this.state.latestGist;
|
|
|
|
const saving = this.state.saving;
|
2017-01-16 15:07:41 +01:00
|
|
|
if(saving) {
|
|
|
|
return <p>Saving...</p>
|
|
|
|
} else if(gist) {
|
2017-01-12 18:27:44 +01:00
|
|
|
const user = gist.user || 'anonymous';
|
2017-10-16 02:59:06 +02:00
|
|
|
const rawGistLink = "https://gist.githubusercontent.com/" + user + "/" + gist.id + "/raw/" + gist.history[0].version + "/style.json"
|
|
|
|
const maputnikStyleLink = "https://maputnik.github.io/editor/?style=" + rawGistLink
|
|
|
|
return <div className="maputnik-render-gist">
|
|
|
|
<p>
|
|
|
|
Latest saved gist:{' '}
|
|
|
|
{this.renderPreviewLink(this)}
|
2017-11-08 09:51:24 +01:00
|
|
|
<a target="_blank" rel="noopener noreferrer" href={"https://gist.github.com/" + user + "/" + gist.id}>Source</a>
|
2017-10-16 02:59:06 +02:00
|
|
|
</p>
|
|
|
|
<p>
|
|
|
|
<CopyToClipboard text={maputnikStyleLink}>
|
|
|
|
<span>Share this style: <Button><TiClipboard size={18} /></Button></span>
|
|
|
|
</CopyToClipboard>
|
|
|
|
<StringInput value={maputnikStyleLink} />
|
|
|
|
</p>
|
|
|
|
</div>
|
2017-01-12 18:27:44 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
render() {
|
2017-01-16 15:07:41 +01:00
|
|
|
return <div className="maputnik-export-gist">
|
2017-01-12 18:27:44 +01:00
|
|
|
<Button onClick={this.onSave.bind(this)}>
|
|
|
|
<MdFileDownload />
|
|
|
|
Save to Gist (anonymous)
|
|
|
|
</Button>
|
2018-02-02 18:21:59 +01:00
|
|
|
<div className="maputnik-modal-sub-section">
|
|
|
|
<CheckboxInput
|
|
|
|
value={this.state.public}
|
|
|
|
name='gist-style-public'
|
|
|
|
onChange={this.onPublicChange.bind(this)}
|
|
|
|
/>
|
|
|
|
<span> Public gist</span>
|
|
|
|
</div>
|
|
|
|
<div className="maputnik-modal-sub-section">
|
|
|
|
<CheckboxInput
|
|
|
|
value={this.state.preview}
|
|
|
|
name='gist-style-preview'
|
|
|
|
onChange={this.onPreviewChange.bind(this)}
|
|
|
|
/>
|
|
|
|
<span> Include preview</span>
|
|
|
|
</div>
|
2017-01-16 15:07:41 +01:00
|
|
|
{this.state.preview ?
|
|
|
|
<div>
|
|
|
|
<InputBlock
|
|
|
|
label={"OpenMapTiles Access Token: "}>
|
|
|
|
<StringInput
|
2017-01-16 20:07:21 +01:00
|
|
|
value={(this.props.mapStyle.metadata || {})['maputnik:openmaptiles_access_token']}
|
2017-01-16 15:07:41 +01:00
|
|
|
onChange={this.changeMetadataProperty.bind(this, "maputnik:openmaptiles_access_token")}/>
|
|
|
|
</InputBlock>
|
2018-02-01 20:54:44 +01:00
|
|
|
<InputBlock
|
|
|
|
label={"Mapbox Access Token: "}>
|
|
|
|
<StringInput
|
|
|
|
value={(this.props.mapStyle.metadata || {})['maputnik:mapbox_access_token']}
|
|
|
|
onChange={this.changeMetadataProperty.bind(this, "maputnik:mapbox_access_token")}/>
|
|
|
|
</InputBlock>
|
2017-11-08 09:51:24 +01:00
|
|
|
<a target="_blank" rel="noopener noreferrer" href="https://openmaptiles.com/hosting/">Get your free access token</a>
|
2017-01-16 15:07:41 +01:00
|
|
|
</div>
|
|
|
|
: null}
|
2017-01-12 18:27:44 +01:00
|
|
|
{this.renderLatestGist()}
|
|
|
|
</div>
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-01-13 15:55:49 +01:00
|
|
|
function stripAccessTokens(mapStyle) {
|
|
|
|
const changedMetadata = { ...mapStyle.metadata }
|
|
|
|
delete changedMetadata['maputnik:mapbox_access_token']
|
|
|
|
delete changedMetadata['maputnik:openmaptiles_access_token']
|
|
|
|
return {
|
|
|
|
...mapStyle,
|
|
|
|
metadata: changedMetadata
|
|
|
|
}
|
|
|
|
}
|
2017-01-12 18:27:44 +01:00
|
|
|
|
|
|
|
class ExportModal extends React.Component {
|
|
|
|
static propTypes = {
|
2017-11-06 16:32:04 +01:00
|
|
|
mapStyle: PropTypes.object.isRequired,
|
|
|
|
onStyleChanged: PropTypes.func.isRequired,
|
|
|
|
isOpen: PropTypes.bool.isRequired,
|
|
|
|
onOpenToggle: PropTypes.func.isRequired,
|
2017-01-12 18:27:44 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
constructor(props) {
|
|
|
|
super(props);
|
|
|
|
}
|
|
|
|
|
2017-01-13 15:55:49 +01:00
|
|
|
downloadStyle() {
|
2017-04-10 14:29:57 +02:00
|
|
|
const blob = new Blob([styleSpec.format(stripAccessTokens(this.props.mapStyle))], {type: "application/json;charset=utf-8"});
|
2017-01-13 15:55:49 +01:00
|
|
|
saveAs(blob, this.props.mapStyle.id + ".json");
|
2017-01-12 18:27:44 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
render() {
|
|
|
|
return <Modal
|
2018-01-05 18:45:55 +01:00
|
|
|
data-wd-key="export-modal"
|
2017-01-12 18:27:44 +01:00
|
|
|
isOpen={this.props.isOpen}
|
|
|
|
onOpenToggle={this.props.onOpenToggle}
|
|
|
|
title={'Export Style'}
|
|
|
|
>
|
|
|
|
|
|
|
|
<div className="maputnik-modal-section">
|
|
|
|
<h4>Download Style</h4>
|
|
|
|
<p>
|
|
|
|
Download a JSON style to your computer.
|
|
|
|
</p>
|
2017-01-13 15:55:49 +01:00
|
|
|
<Button onClick={this.downloadStyle.bind(this)}>
|
2017-01-12 18:27:44 +01:00
|
|
|
<MdFileDownload />
|
|
|
|
Download
|
|
|
|
</Button>
|
|
|
|
</div>
|
|
|
|
|
2018-05-08 16:21:14 +02:00
|
|
|
<div className="maputnik-modal-section hide">
|
2017-01-12 18:27:44 +01:00
|
|
|
<h4>Save style</h4>
|
2017-01-16 15:07:41 +01:00
|
|
|
<Gist mapStyle={this.props.mapStyle} onStyleChanged={this.props.onStyleChanged}/>
|
2017-01-12 18:27:44 +01:00
|
|
|
</div>
|
|
|
|
</Modal>
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
export default ExportModal
|