index.html
<!DOCTYPE html>
<html>
<head>
<style type="text/css">
html, body {
margin: 0px;
padding: 0px;
width: 100%;
height: 100%;
}
#map {
width:980px;
height: 500px;
}
.SvgOverlay {
position: relative;
width: 900px;
height: 600px;
}
.SvgOverlay svg {
position: absolute;
top: -4000px;
left: -4000px;
width: 8000px;
height: 8000px;
}
</style>
</head>
<body>
<div id="map"></div>
<script src="//maps.googleapis.com/maps/api/js?sensor=false"></script>
<script src="//unpkg.com/d3@4.12.2/build/d3.min.js"></script>
<script>
d3.json('gasStation.geojson', main);
function main(pointjson) {
const map = new google.maps.Map(document.getElementById('map'), {
zoom: 11,
mapTypeId: google.maps.MapTypeId.ROADMAP,
center: new google.maps.LatLng(36.322356, 139.013057),
});
const overlay = new google.maps.OverlayView();
overlay.onAdd = function () {
const layer = d3.select(this.getPanes().overlayLayer).append("div").attr("class", "SvgOverlay");
const svg = layer.append("svg");
const svgOverlay = svg.append("g").attr("class", "AdminDivisions");
const pointLayer = svgOverlay.append("g");
const voronoiLayer = svgOverlay.append("g");
const markerOverlay = this;
const overlayProjection = markerOverlay.getProjection();
const googleMapProjection = coordinates => {
const googleCoordinates = new google.maps.LatLng(coordinates[1], coordinates[0]);
const pixelCoordinates = overlayProjection.fromLatLngToDivPixel(googleCoordinates);
return [pixelCoordinates.x + 4000, pixelCoordinates.y + 4000];
}
overlay.draw = function () {
const width = svg.node().clientWidth;
const height = svg.node().clientHeight;
const pointdata = pointjson.features;
const positions = [];
pointdata.forEach(d => {
positions.push(googleMapProjection(d.geometry.coordinates));
});
const updatePoint = pointLayer.selectAll(".point").data(positions)
const enterPoint = updatePoint.enter()
.append("circle")
.attr("class", "point")
.attr("r", 2);
const point = updatePoint.merge(enterPoint)
.attr("transform", d => `translate(${d[0]}, ${d[1]})` )
const voronoi = d3.voronoi()
.extent([[-1, -1],[width+1, height+1]]);
const polygons = voronoi(positions).polygons();
voronoiLayer.selectAll(".cell").remove();
voronoiLayer.selectAll(".cell").data(polygons)
.enter()
.append("path")
.attr("class", "cell")
.attr("fill", "none")
.attr("stroke", "red")
.attr("d", d => {
if(!d) return null
return "M" + d.filter( df => df != null ).join("L") + "Z"
})
};
};
overlay.setMap(map);
};
</script>
</body>
</html>