Skip to content

Commit

Permalink
Improved web Camera (#3473)
Browse files Browse the repository at this point in the history
Implemented all methods of CameraRef except SetCamera and implemented zoom CameraProps
  • Loading branch information
VolkerLieber authored May 13, 2024
1 parent a83c7f2 commit 7f99408
Showing 1 changed file with 123 additions and 13 deletions.
136 changes: 123 additions & 13 deletions src/web/components/Camera.tsx
Original file line number Diff line number Diff line change
@@ -1,32 +1,142 @@
import React from 'react';
import { Component, ContextType } from 'react';

import { CameraProps, CameraRef } from '../../components/Camera';
import { Position } from '../../types/Position';
import MapContext from '../MapContext';

class Camera extends React.Component<{
centerCoordinate: [number, number] | null;
}> {
context!: React.ContextType<typeof MapContext>;
function isArray<T>(value: T | ArrayLike<T>): value is ArrayLike<T> {
return (value as ArrayLike<T>).length !== undefined;
}

function buildMapboxGlPadding(
padding?: number | number[],
): number | mapboxgl.PaddingOptions | undefined {
if (padding === undefined) {
// undefined
return undefined;
} else if (!isArray(padding)) {
// padding
return padding;
} else {
// Array
if (padding.length === 0) {
// []
return undefined;
} else if (padding.length < 2) {
// [padding]
return padding[0];
} else if (padding.length < 4) {
// [vertical, horizontal]
return {
left: padding[0],
right: padding[0],
top: padding[1],
bottom: padding[1],
};
} else {
// [top, right, bottom, left]
return {
top: padding[0],
right: padding[1],
bottom: padding[2],
left: padding[3],
};
}
}
}

class Camera
extends Component<
Pick<
CameraProps,
'centerCoordinate' | 'zoomLevel' | 'minZoomLevel' | 'maxZoomLevel'
>
>
implements Omit<CameraRef, 'setCamera'>
{
context!: ContextType<typeof MapContext>;

static contextType = MapContext;
static UserTrackingModes = [];

componentDidMount() {
const { map } = this.context;
const { centerCoordinate } = this.props;
if (map && centerCoordinate) {
map.flyTo({ center: centerCoordinate });
if (!map) {
return;
}

// minZoomLevel
if (this.props.minZoomLevel !== undefined) {
map.setMinZoom(this.props.minZoomLevel);
}

// maxZoomLevel
if (this.props.maxZoomLevel !== undefined) {
map.setMaxZoom(this.props.maxZoomLevel);
}

// zoomLevel
if (this.props.zoomLevel !== undefined) {
map.setZoom(this.props.zoomLevel);
}

// centerCoordinate
if (this.props.centerCoordinate !== undefined) {
map.flyTo({
center: this.props.centerCoordinate.slice(0, 2) as [number, number],
duration: 0,
});
}
}

fitBounds(
northEastCoordinates: [number, number],
southWestCoordinates: [number, number],
padding = 0,
animationDuration = 0.0,
northEastCoordinates: Position,
southWestCoordinates: Position,
padding: number | number[] = 0,
animationDuration = 0,
) {
const { map } = this.context;
if (map) {
map.fitBounds([northEastCoordinates, southWestCoordinates]);
map.fitBounds(
[
northEastCoordinates.slice(0, 2) as [number, number],
southWestCoordinates.slice(0, 2) as [number, number],
],
{
padding: buildMapboxGlPadding(padding),
duration: animationDuration,
},
);
}
}

flyTo(centerCoordinate: Position, animationDuration = 2000) {
const { map } = this.context;
if (map) {
map.flyTo({
center: centerCoordinate.slice(0, 2) as [number, number],
duration: animationDuration,
});
}
}

moveTo(centerCoordinate: Position, animationDuration = 0) {
const { map } = this.context;
if (map) {
map.easeTo({
center: centerCoordinate.slice(0, 2) as [number, number],
duration: animationDuration,
});
}
}

zoomTo(zoomLevel: number, animationDuration = 2000) {
const { map } = this.context;
if (map) {
map.flyTo({
zoom: zoomLevel,
duration: animationDuration,
});
}
}

Expand Down

0 comments on commit 7f99408

Please sign in to comment.