996 lines
32 KiB
TypeScript
996 lines
32 KiB
TypeScript
import maplibregl from "maplibre-gl";
|
|
import polylabel from "polylabel";
|
|
import { BACKGROUND_LAYER_OPTIONS, BackgroundLayerVisibility } from "@/uhm/lib/map/styles/backgroundLayers";
|
|
import { Feature, FeatureCollection, Geometry } from "@/uhm/lib/editor/state/useEditorState";
|
|
import {
|
|
FEATURE_STATE_SOURCE_IDS,
|
|
PATH_ARROW_ICON_ID,
|
|
RASTER_BASE_LAYER_ID,
|
|
RASTER_BASE_SOURCE_ID,
|
|
PATH_ARROW_SOURCE_ID
|
|
} from "@/uhm/lib/map/constants";
|
|
import { PATH_RENDER_BY_TYPE } from "@/uhm/lib/map/styles/style";
|
|
import { getBackgroundRasterSourceSpecification } from "@/uhm/api/tiles";
|
|
import { newId } from "@/uhm/lib/utils/id";
|
|
import { normalizeGeoTypeKey } from "@/uhm/lib/map/geo/geoTypeMap";
|
|
import type { EntityLabelCandidate } from "@/uhm/types/geo";
|
|
|
|
type Coordinate = [number, number];
|
|
type PolygonCoordinates = Coordinate[][];
|
|
type FeatureLabelInfo = {
|
|
entityId: string;
|
|
label: string;
|
|
timeEnd: number | null;
|
|
};
|
|
|
|
export function applyBackgroundLayerVisibility(
|
|
map: maplibregl.Map,
|
|
visibility: BackgroundLayerVisibility
|
|
) {
|
|
syncRasterBaseVisibility(map, visibility[RASTER_BASE_LAYER_ID]);
|
|
|
|
for (const layer of BACKGROUND_LAYER_OPTIONS) {
|
|
if (layer.id === RASTER_BASE_LAYER_ID) continue;
|
|
const nextVisibility = visibility[layer.id] ? "visible" : "none";
|
|
|
|
if (map.getLayer(layer.id)) {
|
|
map.setLayoutProperty(layer.id, "visibility", nextVisibility);
|
|
}
|
|
|
|
const groupedLayerIds = getBackgroundGroupLayerIds(map, layer.id);
|
|
for (const groupedLayerId of groupedLayerIds) {
|
|
if (!map.getLayer(groupedLayerId)) continue;
|
|
map.setLayoutProperty(groupedLayerId, "visibility", nextVisibility);
|
|
}
|
|
}
|
|
}
|
|
|
|
export function syncRasterBaseVisibility(map: maplibregl.Map, shouldShow: boolean) {
|
|
if (shouldShow) {
|
|
void ensureRasterBaseLayer(map).catch((error) => {
|
|
console.error("Failed to load proxied raster background.", error);
|
|
removeRasterBaseLayer(map);
|
|
});
|
|
return;
|
|
}
|
|
removeRasterBaseLayer(map);
|
|
}
|
|
|
|
export async function ensureRasterBaseLayer(map: maplibregl.Map) {
|
|
if (!map.getSource(RASTER_BASE_SOURCE_ID)) {
|
|
const source = await createRasterBaseSource();
|
|
if (map.getSource(RASTER_BASE_SOURCE_ID)) {
|
|
// Another caller already added the source while we were waiting.
|
|
} else {
|
|
map.addSource(RASTER_BASE_SOURCE_ID, source);
|
|
}
|
|
}
|
|
|
|
const beforeId = getRasterBaseInsertBeforeLayerId(map);
|
|
if (!map.getLayer(RASTER_BASE_LAYER_ID)) {
|
|
map.addLayer(createRasterBaseLayer(), beforeId);
|
|
} else if (beforeId && beforeId !== RASTER_BASE_LAYER_ID) {
|
|
map.moveLayer(RASTER_BASE_LAYER_ID, beforeId);
|
|
}
|
|
|
|
map.setLayoutProperty(RASTER_BASE_LAYER_ID, "visibility", "visible");
|
|
}
|
|
|
|
export function removeRasterBaseLayer(map: maplibregl.Map) {
|
|
if (map.getLayer(RASTER_BASE_LAYER_ID)) {
|
|
map.removeLayer(RASTER_BASE_LAYER_ID);
|
|
}
|
|
|
|
if (map.getSource(RASTER_BASE_SOURCE_ID)) {
|
|
map.removeSource(RASTER_BASE_SOURCE_ID);
|
|
}
|
|
}
|
|
|
|
export function createRasterBaseSource() {
|
|
return getBackgroundRasterSourceSpecification();
|
|
}
|
|
|
|
export function createRasterBaseLayer() {
|
|
return {
|
|
id: RASTER_BASE_LAYER_ID,
|
|
type: "raster" as const,
|
|
source: RASTER_BASE_SOURCE_ID,
|
|
paint: {
|
|
"raster-opacity": 0.92,
|
|
"raster-resampling": "linear" as const,
|
|
},
|
|
};
|
|
}
|
|
|
|
function getRasterBaseInsertBeforeLayerId(map: maplibregl.Map): string | undefined {
|
|
const style = map.getStyle();
|
|
const layers = style?.layers || [];
|
|
|
|
return layers.find((layer) => {
|
|
return layer.id !== "background" && layer.id !== RASTER_BASE_LAYER_ID;
|
|
})?.id;
|
|
}
|
|
|
|
function getBackgroundGroupLayerIds(
|
|
map: maplibregl.Map,
|
|
groupId: string
|
|
): string[] {
|
|
const style = map.getStyle();
|
|
if (!style?.layers?.length) return [];
|
|
|
|
return style.layers
|
|
.filter((layer) => {
|
|
const metadata = (layer as { metadata?: Record<string, unknown> }).metadata;
|
|
return metadata?.uhmBackgroundGroupId === groupId;
|
|
})
|
|
.map((layer) => layer.id);
|
|
}
|
|
|
|
export function getSelectableLayers(map: maplibregl.Map): string[] {
|
|
const selectableSources = ["countries", "places", PATH_ARROW_SOURCE_ID];
|
|
const style = map.getStyle();
|
|
if (!style || !style.layers) return [];
|
|
|
|
return style.layers
|
|
.filter((layer) => "source" in layer && selectableSources.includes(layer.source as string))
|
|
.map((layer) => layer.id);
|
|
}
|
|
|
|
export function filterDraftByBinding(
|
|
fc: FeatureCollection,
|
|
selectedFeatureIds: (string | number)[],
|
|
highlightFeatures?: FeatureCollection | null
|
|
): FeatureCollection {
|
|
const selectedIds = new Set(selectedFeatureIds.map(String));
|
|
if (highlightFeatures?.features) {
|
|
for (const f of highlightFeatures.features) {
|
|
if (f.properties?.id != null) selectedIds.add(String(f.properties.id));
|
|
}
|
|
}
|
|
|
|
const childIds = new Set<string>();
|
|
for (const feature of fc.features) {
|
|
for (const id of normalizeBindingIds(feature.properties.binding)) {
|
|
childIds.add(id);
|
|
}
|
|
}
|
|
|
|
if (selectedIds.size === 0) {
|
|
return { ...fc, features: fc.features.filter((f) => !childIds.has(String(f.properties.id))) };
|
|
}
|
|
|
|
const selectedChildren = new Set<string>();
|
|
for (const feature of fc.features) {
|
|
if (selectedIds.has(String(feature.properties.id))) {
|
|
for (const id of normalizeBindingIds(feature.properties.binding)) {
|
|
selectedChildren.add(id);
|
|
}
|
|
}
|
|
}
|
|
|
|
return {
|
|
...fc,
|
|
features: fc.features.filter((feature) => {
|
|
const featureId = String(feature.properties.id);
|
|
if (selectedIds.has(featureId)) return true;
|
|
if (selectedChildren.has(featureId)) return true;
|
|
return !childIds.has(featureId);
|
|
}),
|
|
};
|
|
}
|
|
|
|
export function filterDraftByGeometryVisibility(
|
|
fc: FeatureCollection,
|
|
visibility: Record<string, boolean> | null | undefined
|
|
): FeatureCollection {
|
|
if (!visibility) return fc;
|
|
|
|
return {
|
|
...fc,
|
|
features: fc.features.filter((feature) => {
|
|
const id = String(feature.properties.id);
|
|
// Kiểm tra ẩn theo ID cụ thể (ưu tiên cao nhất)
|
|
if (visibility[id] === false) return false;
|
|
|
|
const key = getFeatureSemanticType(feature);
|
|
if (!key) return true;
|
|
// Kiểm tra ẩn theo loại (semantic type)
|
|
return visibility[key] !== false;
|
|
}),
|
|
};
|
|
}
|
|
|
|
export function normalizeBindingIds(rawBinding: unknown): string[] {
|
|
if (!Array.isArray(rawBinding)) return [];
|
|
const deduped: string[] = [];
|
|
const seen = new Set<string>();
|
|
for (const rawId of rawBinding) {
|
|
if (typeof rawId !== "string" && typeof rawId !== "number") continue;
|
|
const id = String(rawId).trim();
|
|
if (!id || seen.has(id)) continue;
|
|
seen.add(id);
|
|
deduped.push(id);
|
|
}
|
|
return deduped;
|
|
}
|
|
|
|
export function splitDraftFeatures(fc: FeatureCollection) {
|
|
const polygons = {
|
|
type: "FeatureCollection",
|
|
features: fc.features.filter((f) =>
|
|
f.geometry.type !== "Point" && f.geometry.type !== "MultiPoint"
|
|
),
|
|
} as FeatureCollection;
|
|
|
|
const points = {
|
|
type: "FeatureCollection",
|
|
features: fc.features.filter((f) =>
|
|
f.geometry.type === "Point" || f.geometry.type === "MultiPoint"
|
|
),
|
|
} as FeatureCollection;
|
|
|
|
return { polygons, points };
|
|
}
|
|
|
|
export function decoratePointFeaturesWithLabels(
|
|
fc: FeatureCollection,
|
|
labelContext: FeatureCollection = fc,
|
|
timelineYear?: number | null
|
|
): FeatureCollection {
|
|
const getLabel = createFeatureLabelResolver(labelContext, timelineYear);
|
|
return {
|
|
...fc,
|
|
features: fc.features.map((feature) => ({
|
|
...feature,
|
|
properties: {
|
|
...feature.properties,
|
|
point_label: getLabel(feature),
|
|
},
|
|
})),
|
|
};
|
|
}
|
|
|
|
export function decorateLineFeaturesWithLabels(
|
|
fc: FeatureCollection,
|
|
labelContext: FeatureCollection = fc,
|
|
timelineYear?: number | null
|
|
): FeatureCollection {
|
|
const getLabel = createFeatureLabelResolver(labelContext, timelineYear);
|
|
return {
|
|
...fc,
|
|
features: fc.features.map((feature) => ({
|
|
...feature,
|
|
properties: {
|
|
...feature.properties,
|
|
line_label: isLineGeometry(feature.geometry) ? getLabel(feature) : null,
|
|
},
|
|
})),
|
|
};
|
|
}
|
|
|
|
export function buildPolygonLabelFeatureCollection(
|
|
fc: FeatureCollection,
|
|
labelContext: FeatureCollection = fc,
|
|
timelineYear?: number | null
|
|
): FeatureCollection {
|
|
const getLabel = createFeatureLabelResolver(labelContext, timelineYear);
|
|
const features: Feature[] = [];
|
|
|
|
for (const feature of fc.features) {
|
|
const label = getLabel(feature);
|
|
if (!label) continue;
|
|
|
|
const labelPoint = getPolygonLabelPoint(feature.geometry);
|
|
if (!labelPoint) continue;
|
|
|
|
features.push({
|
|
type: "Feature",
|
|
properties: {
|
|
...feature.properties,
|
|
id: `${feature.properties.id}:polygon-label`,
|
|
polygon_label: label,
|
|
},
|
|
geometry: {
|
|
type: "Point",
|
|
coordinates: labelPoint,
|
|
},
|
|
});
|
|
}
|
|
|
|
return { type: "FeatureCollection", features };
|
|
}
|
|
|
|
export function setSelectedFeatureState(
|
|
map: maplibregl.Map,
|
|
id: string | number | null,
|
|
selected: boolean
|
|
) {
|
|
if (id === null) return;
|
|
for (const sourceId of FEATURE_STATE_SOURCE_IDS) {
|
|
if (!map.getSource(sourceId)) continue;
|
|
map.setFeatureState({ source: sourceId, id }, { selected });
|
|
}
|
|
}
|
|
|
|
export function fitMapToFeatureCollection(
|
|
map: maplibregl.Map,
|
|
fc: FeatureCollection,
|
|
padding?: number | maplibregl.PaddingOptions,
|
|
options?: {
|
|
duration?: number;
|
|
maxZoom?: number;
|
|
pointZoom?: number;
|
|
}
|
|
): boolean {
|
|
const bbox = getFeatureCollectionBBox(fc);
|
|
if (!bbox) return false;
|
|
|
|
const resolvedPadding = typeof padding === "number" || padding ? padding : 58;
|
|
const duration = options?.duration ?? 0;
|
|
const maxZoom = options?.maxZoom ?? 7;
|
|
const pointZoom = options?.pointZoom ?? 6;
|
|
|
|
const lngSpan = Math.abs(bbox.maxLng - bbox.minLng);
|
|
const latSpan = Math.abs(bbox.maxLat - bbox.minLat);
|
|
if (lngSpan < 0.000001 && latSpan < 0.000001) {
|
|
map.easeTo({
|
|
center: [bbox.minLng, bbox.minLat],
|
|
zoom: pointZoom,
|
|
padding: resolvedPadding,
|
|
duration,
|
|
});
|
|
return true;
|
|
}
|
|
|
|
map.fitBounds(
|
|
[
|
|
[bbox.minLng, bbox.minLat],
|
|
[bbox.maxLng, bbox.maxLat],
|
|
],
|
|
{
|
|
padding: resolvedPadding,
|
|
maxZoom,
|
|
duration,
|
|
}
|
|
);
|
|
return true;
|
|
}
|
|
|
|
export function getFeatureCollectionBBox(
|
|
fc: FeatureCollection
|
|
): { minLng: number; minLat: number; maxLng: number; maxLat: number } | null {
|
|
const points = fc.features.flatMap((feature) => collectCoordinatePairs(feature.geometry.coordinates));
|
|
if (!points.length) return null;
|
|
|
|
let minLng = Number.POSITIVE_INFINITY;
|
|
let minLat = Number.POSITIVE_INFINITY;
|
|
let maxLng = Number.NEGATIVE_INFINITY;
|
|
let maxLat = Number.NEGATIVE_INFINITY;
|
|
|
|
for (const [lng, lat] of points) {
|
|
minLng = Math.min(minLng, lng);
|
|
minLat = Math.min(minLat, lat);
|
|
maxLng = Math.max(maxLng, lng);
|
|
maxLat = Math.max(maxLat, lat);
|
|
}
|
|
|
|
return { minLng, minLat, maxLng, maxLat };
|
|
}
|
|
|
|
export function collectCoordinatePairs(value: unknown): Array<[number, number]> {
|
|
if (!Array.isArray(value)) return [];
|
|
if (
|
|
value.length >= 2 &&
|
|
typeof value[0] === "number" &&
|
|
typeof value[1] === "number" &&
|
|
Number.isFinite(value[0]) &&
|
|
Number.isFinite(value[1])
|
|
) {
|
|
return [[value[0], value[1]]];
|
|
}
|
|
return value.flatMap((item) => collectCoordinatePairs(item));
|
|
}
|
|
|
|
export function buildPathArrowFeatureCollection(fc: FeatureCollection): FeatureCollection {
|
|
const features: Feature[] = [];
|
|
|
|
for (const feature of fc.features) {
|
|
if (!isPathFeature(feature)) continue;
|
|
|
|
const coordinateGroups = getLineCoordinateGroups(feature.geometry);
|
|
for (const coordinates of coordinateGroups) {
|
|
const geometry = buildPathArrowGeometry(coordinates);
|
|
if (!geometry) continue;
|
|
features.push({
|
|
type: "Feature",
|
|
properties: { ...feature.properties },
|
|
geometry,
|
|
});
|
|
}
|
|
}
|
|
|
|
return {
|
|
type: "FeatureCollection",
|
|
features,
|
|
};
|
|
}
|
|
|
|
export function isPathFeature(feature: Feature): boolean {
|
|
const featureType = getFeatureSemanticType(feature);
|
|
return Boolean(featureType && PATH_RENDER_BY_TYPE[featureType]);
|
|
}
|
|
|
|
export function getFeatureSemanticType(feature: Feature): string | null {
|
|
const value = feature.properties.type || feature.properties.entity_type_id || null;
|
|
return normalizeGeoTypeKey(value);
|
|
}
|
|
|
|
export function buildPathArrowGeometry(coords: [number, number][]): Geometry | null {
|
|
const sourceCoords = removeDuplicatePathCoords(coords);
|
|
if (sourceCoords.length < 2) return null;
|
|
|
|
const origin = sourceCoords[0];
|
|
const originLatRad = toRadians(origin[1]);
|
|
const cosOriginLat = Math.max(Math.cos(originLatRad), 0.000001);
|
|
const projected = sourceCoords.map((coord) => projectLngLat(coord, origin, cosOriginLat));
|
|
const measured = buildMeasuredPath(projected);
|
|
const totalLength = measured[measured.length - 1]?.distance || 0;
|
|
if (totalLength <= 0) return null;
|
|
|
|
const headLength = clampNumber(totalLength * 0.24, totalLength * 0.12, totalLength * 0.45);
|
|
const bodyEndDistance = Math.max(totalLength - headLength, totalLength * 0.35);
|
|
const bodyPoints = measured
|
|
.filter((point) => point.distance < bodyEndDistance)
|
|
.map(({ x, y, distance }) => ({ x, y, distance }));
|
|
bodyPoints.push(pointAtDistance(measured, bodyEndDistance));
|
|
|
|
if (bodyPoints.length < 2) return null;
|
|
|
|
const tailWidth = clampNumber(totalLength * 0.02, 5, 40000);
|
|
const shoulderWidth = clampNumber(totalLength * 0.1, 10, 100000);
|
|
const headWidth = shoulderWidth * 2.0;
|
|
|
|
const leftBody: ProjectedPoint[] = [];
|
|
const rightBody: ProjectedPoint[] = [];
|
|
|
|
for (let i = 0; i < bodyPoints.length; i += 1) {
|
|
const point = bodyPoints[i];
|
|
const normal = normalAt(bodyPoints, i);
|
|
const progress = bodyEndDistance > 0
|
|
? Math.pow(clampNumber(point.distance / bodyEndDistance, 0, 1), 0.9)
|
|
: 0;
|
|
const width = tailWidth + (shoulderWidth - tailWidth) * progress;
|
|
const half = width / 2;
|
|
leftBody.push({
|
|
x: point.x + normal.x * half,
|
|
y: point.y + normal.y * half,
|
|
});
|
|
rightBody.push({
|
|
x: point.x - normal.x * half,
|
|
y: point.y - normal.y * half,
|
|
});
|
|
}
|
|
|
|
const base = bodyPoints[bodyPoints.length - 1];
|
|
const tip = pointAtDistance(measured, totalLength);
|
|
const headNormal = normalFromSegment(base, tip) || normalAt(bodyPoints, bodyPoints.length - 1);
|
|
const headHalf = headWidth / 2;
|
|
const headBaseLeft = {
|
|
x: base.x + headNormal.x * headHalf,
|
|
y: base.y + headNormal.y * headHalf,
|
|
};
|
|
const headBaseRight = {
|
|
x: base.x - headNormal.x * headHalf,
|
|
y: base.y - headNormal.y * headHalf,
|
|
};
|
|
|
|
const ring = [
|
|
...leftBody,
|
|
headBaseLeft,
|
|
{ x: tip.x, y: tip.y },
|
|
headBaseRight,
|
|
...rightBody.reverse(),
|
|
leftBody[0],
|
|
].map((point) => unprojectLngLat(point, origin, cosOriginLat));
|
|
|
|
if (ring.length < 4) return null;
|
|
return {
|
|
type: "Polygon",
|
|
coordinates: [ring],
|
|
};
|
|
}
|
|
|
|
export type ProjectedPoint = {
|
|
x: number;
|
|
y: number;
|
|
};
|
|
|
|
export type MeasuredPoint = ProjectedPoint & {
|
|
distance: number;
|
|
};
|
|
|
|
export function removeDuplicatePathCoords(coords: [number, number][]): [number, number][] {
|
|
const result: [number, number][] = [];
|
|
for (const coord of coords) {
|
|
const last = result[result.length - 1];
|
|
if (last && last[0] === coord[0] && last[1] === coord[1]) continue;
|
|
result.push(coord);
|
|
}
|
|
return result;
|
|
}
|
|
|
|
export function projectLngLat(
|
|
coord: [number, number],
|
|
origin: [number, number],
|
|
cosOriginLat: number
|
|
): ProjectedPoint {
|
|
const earthRadiusMeters = 6371008.8;
|
|
return {
|
|
x: toRadians(coord[0] - origin[0]) * earthRadiusMeters * cosOriginLat,
|
|
y: toRadians(coord[1] - origin[1]) * earthRadiusMeters,
|
|
};
|
|
}
|
|
|
|
export function unprojectLngLat(
|
|
point: ProjectedPoint,
|
|
origin: [number, number],
|
|
cosOriginLat: number
|
|
): [number, number] {
|
|
const earthRadiusMeters = 6371008.8;
|
|
return [
|
|
origin[0] + toDegrees(point.x / (earthRadiusMeters * cosOriginLat)),
|
|
origin[1] + toDegrees(point.y / earthRadiusMeters),
|
|
];
|
|
}
|
|
|
|
export function buildMeasuredPath(points: ProjectedPoint[]): MeasuredPoint[] {
|
|
let distance = 0;
|
|
return points.map((point, index) => {
|
|
if (index > 0) {
|
|
distance += distanceProjected(points[index - 1], point);
|
|
}
|
|
return {
|
|
...point,
|
|
distance,
|
|
};
|
|
});
|
|
}
|
|
|
|
export function pointAtDistance(points: MeasuredPoint[], targetDistance: number): MeasuredPoint {
|
|
if (targetDistance <= 0) return points[0];
|
|
for (let i = 1; i < points.length; i += 1) {
|
|
const prev = points[i - 1];
|
|
const next = points[i];
|
|
if (targetDistance > next.distance) continue;
|
|
const segmentLength = next.distance - prev.distance;
|
|
const t = segmentLength > 0 ? (targetDistance - prev.distance) / segmentLength : 0;
|
|
return {
|
|
x: prev.x + (next.x - prev.x) * t,
|
|
y: prev.y + (next.y - prev.y) * t,
|
|
distance: targetDistance,
|
|
};
|
|
}
|
|
return points[points.length - 1];
|
|
}
|
|
|
|
export function normalAt(points: ProjectedPoint[], index: number): ProjectedPoint {
|
|
const prev = points[Math.max(0, index - 1)];
|
|
const next = points[Math.min(points.length - 1, index + 1)];
|
|
return normalFromSegment(prev, next) || { x: 0, y: 1 };
|
|
}
|
|
|
|
export function normalFromSegment(a: ProjectedPoint, b: ProjectedPoint): ProjectedPoint | null {
|
|
const dx = b.x - a.x;
|
|
const dy = b.y - a.y;
|
|
const length = Math.hypot(dx, dy);
|
|
if (length <= 0) return null;
|
|
return {
|
|
x: -dy / length,
|
|
y: dx / length,
|
|
};
|
|
}
|
|
|
|
export function distanceProjected(a: ProjectedPoint, b: ProjectedPoint): number {
|
|
return Math.hypot(b.x - a.x, b.y - a.y);
|
|
}
|
|
|
|
export function toRadians(value: number): number {
|
|
return (value * Math.PI) / 180;
|
|
}
|
|
|
|
export function toDegrees(value: number): number {
|
|
return (value * 180) / Math.PI;
|
|
}
|
|
|
|
export function ensurePathArrowIcon(map: maplibregl.Map): boolean {
|
|
if (map.hasImage(PATH_ARROW_ICON_ID)) return true;
|
|
const imageData = createPathArrowImageData();
|
|
if (!imageData) return false;
|
|
map.addImage(PATH_ARROW_ICON_ID, imageData, { pixelRatio: 2 });
|
|
return true;
|
|
}
|
|
|
|
export function createPathArrowImageData(): ImageData | null {
|
|
const size = 56;
|
|
if (typeof document === "undefined") return null;
|
|
const canvas = document.createElement("canvas");
|
|
canvas.width = size;
|
|
canvas.height = size;
|
|
const ctx = canvas.getContext("2d");
|
|
if (!ctx) return null;
|
|
|
|
ctx.clearRect(0, 0, size, size);
|
|
|
|
ctx.strokeStyle = "#0f172a";
|
|
ctx.fillStyle = "#38bdf8";
|
|
ctx.lineWidth = 4;
|
|
ctx.lineJoin = "round";
|
|
ctx.lineCap = "round";
|
|
|
|
ctx.beginPath();
|
|
ctx.moveTo(8, 16);
|
|
ctx.lineTo(28, 16);
|
|
ctx.lineTo(28, 10);
|
|
ctx.lineTo(46, 28);
|
|
ctx.lineTo(28, 46);
|
|
ctx.lineTo(28, 40);
|
|
ctx.lineTo(8, 40);
|
|
ctx.closePath();
|
|
ctx.fill();
|
|
ctx.stroke();
|
|
|
|
return ctx.getImageData(0, 0, size, size);
|
|
}
|
|
|
|
export function buildTypeMatchExpression(
|
|
valueByType: Record<string, string | number | boolean>,
|
|
fallback: string | number | boolean
|
|
): maplibregl.ExpressionSpecification {
|
|
const expression: unknown[] = ["match", getFeatureTypeExpression()];
|
|
|
|
for (const [typeId, value] of Object.entries(valueByType)) {
|
|
expression.push(typeId, value);
|
|
}
|
|
|
|
expression.push(fallback);
|
|
return expression as maplibregl.ExpressionSpecification;
|
|
}
|
|
|
|
export function getFeatureTypeExpression(): maplibregl.ExpressionSpecification {
|
|
return [
|
|
"coalesce",
|
|
["get", "type"],
|
|
["get", "entity_type_id"],
|
|
"",
|
|
] as maplibregl.ExpressionSpecification;
|
|
}
|
|
|
|
export function roundZoom(value: number): number {
|
|
return Math.round(value * 10) / 10;
|
|
}
|
|
|
|
function createFeatureLabelResolver(
|
|
fc: FeatureCollection,
|
|
timelineYear?: number | null
|
|
): (feature: Feature) => string | null {
|
|
const directLabelsByFeatureId = new Map<string, FeatureLabelInfo>();
|
|
const inheritedLabelsByChildId = new Map<string, FeatureLabelInfo | null>();
|
|
|
|
for (const feature of fc.features) {
|
|
const labelInfo = getSingleEntityFeatureLabelInfo(feature, timelineYear);
|
|
if (!labelInfo) continue;
|
|
directLabelsByFeatureId.set(String(feature.properties.id), labelInfo);
|
|
}
|
|
|
|
for (const feature of fc.features) {
|
|
const parentLabel = directLabelsByFeatureId.get(String(feature.properties.id));
|
|
const featureId = String(feature.properties.id);
|
|
const bindingIds = normalizeBindingIds(feature.properties.binding);
|
|
|
|
if (parentLabel) {
|
|
for (const childId of bindingIds) {
|
|
mergeInheritedFeatureLabel(inheritedLabelsByChildId, childId, parentLabel);
|
|
}
|
|
}
|
|
|
|
for (const parentId of bindingIds) {
|
|
const linkedParentLabel = directLabelsByFeatureId.get(parentId);
|
|
if (linkedParentLabel) {
|
|
mergeInheritedFeatureLabel(inheritedLabelsByChildId, featureId, linkedParentLabel);
|
|
}
|
|
}
|
|
}
|
|
|
|
return (feature) => {
|
|
const featureId = String(feature.properties.id);
|
|
const directEntityIds = getFeatureEntityIds(feature);
|
|
if (directEntityIds.length > 0) {
|
|
return directLabelsByFeatureId.get(featureId)?.label || null;
|
|
}
|
|
|
|
return inheritedLabelsByChildId.get(featureId)?.label || null;
|
|
};
|
|
}
|
|
|
|
function mergeInheritedFeatureLabel(
|
|
labelsByFeatureId: Map<string, FeatureLabelInfo | null>,
|
|
targetFeatureId: string,
|
|
labelInfo: FeatureLabelInfo
|
|
) {
|
|
const current = labelsByFeatureId.get(targetFeatureId);
|
|
if (current === undefined) {
|
|
labelsByFeatureId.set(targetFeatureId, labelInfo);
|
|
} else if (current && current.entityId === labelInfo.entityId) {
|
|
labelsByFeatureId.set(targetFeatureId, current);
|
|
} else {
|
|
labelsByFeatureId.set(targetFeatureId, null);
|
|
}
|
|
}
|
|
|
|
function getSingleEntityFeatureLabelInfo(
|
|
feature: Feature,
|
|
timelineYear?: number | null
|
|
): FeatureLabelInfo | null {
|
|
const candidates = getFeatureEntityLabelCandidates(feature);
|
|
if (candidates.length > 0) {
|
|
const timelineCandidate = getLatestTimelineEntityCandidate(candidates, timelineYear);
|
|
if (!timelineCandidate) return null;
|
|
return {
|
|
entityId: timelineCandidate.id,
|
|
label: timelineCandidate.name,
|
|
timeEnd: normalizeLabelYear(timelineCandidate.time_end),
|
|
};
|
|
}
|
|
|
|
const entityIds = getFeatureEntityIds(feature);
|
|
if (entityIds.length !== 1) return null;
|
|
|
|
const label = getSingleEntityName(feature);
|
|
if (!label) return null;
|
|
|
|
return { entityId: entityIds[0], label, timeEnd: null };
|
|
}
|
|
|
|
function getLatestTimelineEntityCandidate(
|
|
candidates: EntityLabelCandidate[],
|
|
timelineYear?: number | null
|
|
): EntityLabelCandidate | null {
|
|
if (!candidates.length) return null;
|
|
|
|
const activeCandidates = candidates.filter((candidate) =>
|
|
isEntityCandidateVisibleAtYear(candidate, timelineYear)
|
|
);
|
|
if (!activeCandidates.length) return null;
|
|
|
|
return activeCandidates.sort(compareEntityLabelCandidates)[0] || null;
|
|
}
|
|
|
|
function getFeatureEntityLabelCandidates(feature: Feature): EntityLabelCandidate[] {
|
|
const rawCandidates = feature.properties.entity_label_candidates;
|
|
if (!Array.isArray(rawCandidates)) return [];
|
|
|
|
const byId = new Map<string, EntityLabelCandidate>();
|
|
for (const raw of rawCandidates) {
|
|
if (!raw || typeof raw !== "object") continue;
|
|
const candidate = raw as EntityLabelCandidate;
|
|
const id = String(candidate.id || "").trim();
|
|
const name = String(candidate.name || "").trim();
|
|
if (!id || !name) continue;
|
|
byId.set(id, {
|
|
id,
|
|
name,
|
|
time_start: normalizeLabelYear(candidate.time_start),
|
|
time_end: normalizeLabelYear(candidate.time_end),
|
|
});
|
|
}
|
|
|
|
return Array.from(byId.values());
|
|
}
|
|
|
|
function isEntityCandidateVisibleAtYear(
|
|
candidate: EntityLabelCandidate,
|
|
timelineYear?: number | null
|
|
): boolean {
|
|
if (typeof timelineYear !== "number" || !Number.isFinite(timelineYear)) return true;
|
|
|
|
const start = normalizeLabelYear(candidate.time_start);
|
|
const end = normalizeLabelYear(candidate.time_end);
|
|
if (start != null && timelineYear < start) return false;
|
|
if (end != null && timelineYear > end) return false;
|
|
return true;
|
|
}
|
|
|
|
function compareEntityLabelCandidates(a: EntityLabelCandidate, b: EntityLabelCandidate): number {
|
|
const endA = normalizeLabelYear(a.time_end);
|
|
const endB = normalizeLabelYear(b.time_end);
|
|
const endScoreA = endA == null ? Number.NEGATIVE_INFINITY : endA;
|
|
const endScoreB = endB == null ? Number.NEGATIVE_INFINITY : endB;
|
|
if (endScoreA !== endScoreB) return endScoreB - endScoreA;
|
|
|
|
const startA = normalizeLabelYear(a.time_start);
|
|
const startB = normalizeLabelYear(b.time_start);
|
|
const startScoreA = startA == null ? Number.NEGATIVE_INFINITY : startA;
|
|
const startScoreB = startB == null ? Number.NEGATIVE_INFINITY : startB;
|
|
if (startScoreA !== startScoreB) return startScoreB - startScoreA;
|
|
|
|
return a.name.localeCompare(b.name);
|
|
}
|
|
|
|
function normalizeLabelYear(value: unknown): number | null {
|
|
return typeof value === "number" && Number.isFinite(value) ? value : null;
|
|
}
|
|
|
|
function getFeatureEntityIds(feature: Feature): string[] {
|
|
const rawEntityIds: unknown[] = Array.isArray(feature.properties.entity_ids)
|
|
? feature.properties.entity_ids
|
|
: (typeof feature.properties.entity_id === "string" || typeof feature.properties.entity_id === "number"
|
|
? [feature.properties.entity_id]
|
|
: []);
|
|
|
|
return Array.from(new Set(
|
|
rawEntityIds
|
|
.filter((id): id is string | number => typeof id === "string" || typeof id === "number")
|
|
.map((id) => String(id).trim())
|
|
.filter((id) => id.length > 0)
|
|
));
|
|
}
|
|
|
|
function getSingleEntityName(feature: Feature): string | null {
|
|
const directName = typeof feature.properties.entity_name === "string"
|
|
? feature.properties.entity_name.trim()
|
|
: "";
|
|
if (directName.length > 0) return directName;
|
|
|
|
const names = Array.isArray(feature.properties.entity_names)
|
|
? Array.from(new Set(
|
|
feature.properties.entity_names
|
|
.filter((name): name is string => typeof name === "string")
|
|
.map((name) => name.trim())
|
|
.filter((name) => name.length > 0)
|
|
))
|
|
: [];
|
|
|
|
return names.length === 1 ? names[0] : null;
|
|
}
|
|
|
|
function isLineGeometry(geometry: Geometry): boolean {
|
|
return geometry.type === "LineString" || geometry.type === "MultiLineString";
|
|
}
|
|
|
|
function getLineCoordinateGroups(geometry: Geometry): Coordinate[][] {
|
|
if (geometry.type === "LineString") return [geometry.coordinates];
|
|
if (geometry.type === "MultiLineString") return geometry.coordinates;
|
|
return [];
|
|
}
|
|
|
|
function getPolygonLabelPoint(geometry: Geometry): Coordinate | null {
|
|
if (geometry.type === "Polygon") {
|
|
return getPolygonLabelCandidate(geometry.coordinates)?.point || null;
|
|
}
|
|
|
|
if (geometry.type === "MultiPolygon") {
|
|
let best: { point: Coordinate; distance: number } | null = null;
|
|
for (const polygon of geometry.coordinates) {
|
|
const candidate = getPolygonLabelCandidate(polygon);
|
|
if (!candidate) continue;
|
|
if (!best || candidate.distance > best.distance) {
|
|
best = candidate;
|
|
}
|
|
}
|
|
return best?.point || null;
|
|
}
|
|
|
|
return null;
|
|
}
|
|
|
|
function getPolygonLabelCandidate(polygon: PolygonCoordinates): { point: Coordinate; distance: number } | null {
|
|
const outerRing = polygon[0];
|
|
if (!outerRing || outerRing.length < 3) return null;
|
|
|
|
const bbox = getRingBbox(outerRing);
|
|
if (!bbox) return null;
|
|
|
|
const width = bbox.maxX - bbox.minX;
|
|
const height = bbox.maxY - bbox.minY;
|
|
if (width <= 0 || height <= 0) {
|
|
const fallback: Coordinate = [bbox.minX, bbox.minY];
|
|
return { point: fallback, distance: 0 };
|
|
}
|
|
|
|
const precision = Math.max(Math.max(width, height) / 100, 0.0001);
|
|
const result = polylabel(polygon, precision);
|
|
const x = result[0];
|
|
const y = result[1];
|
|
|
|
if (!Number.isFinite(x) || !Number.isFinite(y)) {
|
|
return { point: [bbox.minX + width / 2, bbox.minY + height / 2], distance: 0 };
|
|
}
|
|
|
|
return { point: [x, y], distance: Number.isFinite(result.distance) ? result.distance : 0 };
|
|
}
|
|
|
|
function getRingBbox(ring: Coordinate[]): { minX: number; minY: number; maxX: number; maxY: number } | null {
|
|
if (!ring.length) return null;
|
|
|
|
let minX = Number.POSITIVE_INFINITY;
|
|
let minY = Number.POSITIVE_INFINITY;
|
|
let maxX = Number.NEGATIVE_INFINITY;
|
|
let maxY = Number.NEGATIVE_INFINITY;
|
|
|
|
for (const [x, y] of ring) {
|
|
minX = Math.min(minX, x);
|
|
minY = Math.min(minY, y);
|
|
maxX = Math.max(maxX, x);
|
|
maxY = Math.max(maxY, y);
|
|
}
|
|
|
|
if (!Number.isFinite(minX) || !Number.isFinite(minY) || !Number.isFinite(maxX) || !Number.isFinite(maxY)) {
|
|
return null;
|
|
}
|
|
|
|
return { minX, minY, maxX, maxY };
|
|
}
|
|
|
|
export function buildClientFeatureId(): string {
|
|
return newId();
|
|
}
|
|
|
|
export function clampNumber(value: number, min: number, max: number): number {
|
|
if (value < min) return min;
|
|
if (value > max) return max;
|
|
return value;
|
|
}
|
|
|
|
export function hashStringToColor(str: string): string {
|
|
let hash = 0;
|
|
for (let i = 0; i < str.length; i++) {
|
|
hash = str.charCodeAt(i) + ((hash << 5) - hash);
|
|
}
|
|
const hue = Math.abs(hash) % 360;
|
|
return `hsl(${hue}, 70%, 50%)`;
|
|
}
|
|
|
|
export function decorateFeaturesWithEntityColors(fc: FeatureCollection): FeatureCollection {
|
|
return {
|
|
...fc,
|
|
features: fc.features.map((feature) => {
|
|
const geomType = feature.geometry?.type;
|
|
if (geomType === "Point" || geomType === "MultiPoint") {
|
|
// Point - giữ nguyên màu của preset/icon
|
|
return feature;
|
|
}
|
|
|
|
if (geomType === "LineString" || geomType === "MultiLineString") {
|
|
const entityIds = getFeatureEntityIds(feature);
|
|
if (entityIds.length > 0) {
|
|
const sortedCombined = [...entityIds].sort().join("+");
|
|
return {
|
|
...feature,
|
|
properties: {
|
|
...feature.properties,
|
|
entity_color: hashStringToColor(sortedCombined),
|
|
},
|
|
};
|
|
}
|
|
return feature;
|
|
}
|
|
|
|
if (geomType === "Polygon" || geomType === "MultiPolygon") {
|
|
const geoId = String(feature.properties?.id || "");
|
|
if (geoId) {
|
|
return {
|
|
...feature,
|
|
properties: {
|
|
...feature.properties,
|
|
entity_color: hashStringToColor(geoId),
|
|
},
|
|
};
|
|
}
|
|
return feature;
|
|
}
|
|
|
|
return feature;
|
|
}),
|
|
};
|
|
}
|