dyomedea/src/components/map.tsx

163 lines
4.4 KiB
TypeScript

import _, { round } from 'lodash';
import react, { useState, useEffect, useMemo, useRef } from 'react';
import Tile from './tile';
import '../theme/map.css';
export const tileSize = 256;
// cf https://wiki.openstreetmap.org/wiki/Slippy_map_tilenames#ECMAScript_(JavaScript/ActionScript,_etc.)
const lon2tile = (lon: number, zoom: number) => {
const real = ((lon + 180) / 360) * Math.pow(2, zoom);
const floor = Math.floor(real);
return [floor, real - floor];
};
const lat2tile = (lat: number, zoom: number) => {
const real =
((1 -
Math.log(
Math.tan((lat * Math.PI) / 180) + 1 / Math.cos((lat * Math.PI) / 180)
) /
Math.PI) /
2) *
Math.pow(2, zoom);
const floor = Math.floor(real);
return [floor, real - floor];
};
const Map: react.FC = () => {
console.log('Log - Rendering <Map />');
const initialCenter: [number, number] = [43.57029965, 3.94242897];
const initialZoom: number = 17;
const [center, setCenter] = useState(initialCenter);
const [zoom, setZoom] = useState(initialZoom);
const [dimensions, setDimensions] = useState({
height: window.innerHeight,
width: window.innerWidth,
});
const resizeHandler = () => {
setDimensions({
height: window.innerHeight,
width: window.innerWidth,
});
};
const debouncedResizeHandler = useMemo(
() => _.debounce(resizeHandler, 300),
[dimensions, zoom, center]
);
useEffect(() => {
window.addEventListener('resize', debouncedResizeHandler);
}, []);
const nbTilesY = _.ceil(dimensions.height / tileSize) + 4;
const nbTilesX = _.ceil(dimensions.width / tileSize) + 4;
const [tileCenterY, reminderY] = lat2tile(center[0], zoom);
const [tileCenterX, reminderX] = lon2tile(center[1], zoom);
const firstTileY = tileCenterY - _.round(nbTilesY / 2);
const firstTileX = tileCenterX - _.round(nbTilesX / 2);
const locationY = (tileCenterY + reminderY - firstTileY) * tileSize;
const locationX = (tileCenterX + reminderX - firstTileX) * tileSize;
const targetLocationY = dimensions.height / 2;
const targetLocationX = dimensions.width / 2;
const deltaY = -targetLocationY + locationY;
const deltaX = -targetLocationX + locationX;
const [delta, setDelta] = useState({ X: deltaX, Y: deltaY });
const genericHandler = (event: any) => {
console.log('Log - Event: ' + event.type);
};
const initialMouseState = {
down: false,
starting: { x: -1, y: -1 },
};
const mouseState = useRef(initialMouseState);
const mouseDownHandler = (event: any) => {
event.preventDefault();
mouseState.current = {
down: true,
starting: { x: event.pageX, y: event.pageY },
};
genericHandler(event);
};
const mouseUpHandler = (event: any) => {
event.preventDefault();
console.log('Log - Up, now do something ! ');
mouseState.current = initialMouseState;
genericHandler(event);
};
const mouseMoveHandler = (event: any) => {
if (mouseState.current.down) {
event.preventDefault();
console.log('Log - Moving...' + event.pageX);
setDelta({
X: deltaX + mouseState.current.starting.x - event.pageX,
Y: deltaY + mouseState.current.starting.y - event.pageY,
});
genericHandler(event);
}
};
const debouncedMouseMoveHandler = useMemo(
() => _.throttle(mouseMoveHandler, 100),
[]
);
const pointerDownHandler = (event: any) => {
event.preventDefault();
console.log('PointerDown');
};
const pointerUpHandler = (event: any) => {
event.preventDefault();
console.log('PointerUp');
};
const pointerMoveHandler = (event: any) => {
event.preventDefault();
console.log('PointerMove');
};
return (
<div
className='tiles'
onMouseDown={mouseDownHandler}
onMouseMove={debouncedMouseMoveHandler}
onMouseUp={mouseUpHandler}
onTouchStart={genericHandler}
onTouchMove={genericHandler}
onTouchCancel={genericHandler}
onTouchEnd={genericHandler}
>
{_.range(nbTilesY).map((iy) => (
<div key={'y' + iy} className='tilesRow'>
{_.range(nbTilesX).map((ix) => (
<Tile
key={'x' + ix + 'y' + iy}
iy={iy}
ix={ix}
x={firstTileX + ix}
y={firstTileY + iy}
delta={delta}
zoom={zoom}
/>
))}
</div>
))}
</div>
);
};
export default Map;