You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
188 lines
4.3 KiB
188 lines
4.3 KiB
<canvas></canvas>
|
|
<script src="ogc-parser.js"></script>
|
|
<script>
|
|
/* global ogcParser */
|
|
function wait(ms = 0) {
|
|
return new Promise((resolve) => {
|
|
setTimeout(resolve, ms);
|
|
});
|
|
}
|
|
|
|
// # need to draw to 2nd canvas, then shave off non perfect pixels
|
|
|
|
async function main() {
|
|
const ctx = document.querySelector('canvas').getContext('2d');
|
|
ctx.canvas.width = 2048;
|
|
ctx.canvas.height = 2048;
|
|
ctx.fillStyle = '#444';
|
|
ctx.fillRect(0, 0, ctx.canvas.width, ctx.canvas.height);
|
|
ctx.translate(ctx.canvas.width / 2, ctx.canvas.height / 2);
|
|
ctx.scale(ctx.canvas.width / 360, ctx.canvas.height / -180);
|
|
|
|
function setColor(color) {
|
|
ctx.fillStyle = color;
|
|
ctx.strokeStyle = color;
|
|
}
|
|
|
|
const handlers = {
|
|
point,
|
|
lineString,
|
|
polygon,
|
|
multiPoint,
|
|
multiLineString,
|
|
multiPolygon,
|
|
};
|
|
|
|
function point(d) {
|
|
ctx.fillRect(...d.point, 1, 1);
|
|
}
|
|
|
|
function setPathFromPoints(points, backward = false) {
|
|
if (backward) {
|
|
const numPoints = points.length / 2;
|
|
const lastPoint = numPoints - 1;
|
|
ctx.moveTo(...points.slice(lastPoint * 2, lastPoint * 2 + 2));
|
|
for (let i = lastPoint - 1; i >= 0; i -= 2) {
|
|
ctx.lineTo(...points.slice(i * 2, i * 2 + 2));
|
|
}
|
|
} else {
|
|
ctx.moveTo(...points.slice(0, 2));
|
|
for (let i = 2; i < points.length; i += 2) {
|
|
ctx.lineTo(...points.slice(i, i + 2));
|
|
}
|
|
}
|
|
}
|
|
|
|
function stroke(ctx) {
|
|
ctx.save();
|
|
ctx.setTransform(1, 0, 0, 1, 0, 0);
|
|
ctx.stroke();
|
|
ctx.restore();
|
|
}
|
|
|
|
function lineString(d) {
|
|
ctx.beginPath();
|
|
setPathFromPoints(d.points);
|
|
stroke(ctx);
|
|
}
|
|
|
|
function polygon(d) {
|
|
ctx.beginPath();
|
|
d.rings.forEach((ring, ndx) => {
|
|
setPathFromPoints(ring, ndx !== 0);
|
|
ctx.closePath();
|
|
});
|
|
ctx.fill();
|
|
stroke(ctx);
|
|
}
|
|
|
|
function multiPoint(d) {
|
|
for (let i = 0; i < d.points.length; i += 2) {
|
|
ctx.fillRect(...d.points.slice(i, i + 2));
|
|
}
|
|
}
|
|
|
|
function multiLineString(d) {
|
|
d.lineStrings.forEach((lineString) => {
|
|
ctx.beginPath();
|
|
setPathFromPoints(lineString);
|
|
stroke(ctx);
|
|
});
|
|
}
|
|
|
|
function multiPolygon(d) {
|
|
d.polygons.forEach((polygon) => {
|
|
ctx.beginPath();
|
|
polygon.forEach((ring, ndx) => {
|
|
setPathFromPoints(ring, ndx !== 0);
|
|
});
|
|
ctx.fill();
|
|
stroke(ctx);
|
|
});
|
|
}
|
|
|
|
const colors = {};
|
|
|
|
const req = await fetch('level1.json');
|
|
const areas = await req.json();
|
|
const min = [Number.MAX_VALUE, Number.MAX_VALUE];
|
|
const max = [Number.MIN_VALUE, Number.MIN_VALUE];
|
|
console.log('num areas:', areas.length);
|
|
for (let ndx = 0; ndx < areas.length; ++ndx) {
|
|
const area = areas[ndx];
|
|
try {
|
|
const buf = new Uint8Array(base64ToUint8Array(area.geom));
|
|
area.geom = ogcParser.parse(buf);
|
|
} catch (e) {
|
|
console.log('ERROR:', e);
|
|
console.log(JSON.stringify(area, null, 2));
|
|
throw e;
|
|
}
|
|
|
|
if (!colors[area.NAME_0]) {
|
|
colors[area.NAME_0] = rgb(r(), r(), r());
|
|
}
|
|
|
|
const color = colors[area.NAME_0];
|
|
|
|
console.log(ndx, area.NAME_0);
|
|
|
|
area.geom.primitives.forEach((primitive) => {
|
|
const fn = handlers[primitive.type];
|
|
setColor(color);
|
|
fn(primitive);
|
|
});
|
|
|
|
min[0] = Math.min(min[0], area.geom.envelope[0]);
|
|
min[0] = Math.min(min[0], area.geom.envelope[1]);
|
|
min[1] = Math.min(min[1], area.geom.envelope[2]);
|
|
min[1] = Math.min(min[1], area.geom.envelope[3]);
|
|
|
|
max[0] = Math.max(max[0], area.geom.envelope[0]);
|
|
max[0] = Math.max(max[0], area.geom.envelope[1]);
|
|
max[1] = Math.max(max[1], area.geom.envelope[2]);
|
|
max[1] = Math.max(max[1], area.geom.envelope[3]);
|
|
|
|
if (ndx % 100 === 99) {
|
|
await wait();
|
|
}
|
|
}
|
|
|
|
console.log('min', min);
|
|
console.log('max', max);
|
|
}
|
|
|
|
function r(min, max) {
|
|
if (min === undefined) {
|
|
min = 0;
|
|
max = 1;
|
|
} else if (max === undefined){
|
|
max = min;
|
|
min = 0;
|
|
}
|
|
return min + Math.random() * (max - min);
|
|
}
|
|
|
|
function rgb(r, g, b) {
|
|
return `rgb(${r * 255 | 0},${g * 255 | 0},${b * 255 | 0})`;
|
|
}
|
|
|
|
// function hsl(h, s, l) {
|
|
// return `hsl(${h * 360 | 0},${s * 100 | 0}%,${l * 100 | 0}%)`;
|
|
// }
|
|
|
|
function base64ToUint8Array(base64) {
|
|
const raw = window.atob(base64);
|
|
const rawLength = raw.length;
|
|
const array = new Uint8Array(new ArrayBuffer(rawLength));
|
|
|
|
for(let i = 0; i < rawLength; ++i) {
|
|
array[i] = raw.charCodeAt(i);
|
|
}
|
|
|
|
return array;
|
|
}
|
|
|
|
main();
|
|
|
|
</script>
|
|
|