UNPKG

@polygonjs/plugin-mapbox

Version:

Mapbox plugin for the 3D engine https://polygonjs.com

122 lines (106 loc) 4.39 kB
import {Vector2} from 'three'; import {Object3D} from 'three'; import {LineSegments} from 'three'; import {Float32BufferAttribute} from 'three'; import {BufferGeometry} from 'three'; import {CoreGeometry} from '@polygonjs/polygonjs/dist/src/core/geometry/Geometry'; import {ObjectType} from '@polygonjs/polygonjs/dist/src/core/geometry/Constant'; import {BaseSopNodeType} from '@polygonjs/polygonjs/dist/src/engine/nodes/sop/_Base'; import {CoordinatesCollection} from './CoordinatesCollection'; import {ArrayUtils} from '@polygonjs/polygonjs/dist/src/core/ArrayUtils'; import {CoreGeometryBuilderMerge} from '@polygonjs/polygonjs/dist/src/core/geometry/builders/Merge'; import {CoreMapboxString} from './String'; const MULTILINESTRING = 'MultiLineString'; const LINESTRING = 'LineString'; export class FeatureConverter { id: number | undefined; constructor(private node: BaseSopNodeType, private name: string, private features: any[]) {} createObject(): Object3D | undefined { const coordinatesCollections = this._createAllCoordinatesCollections(); const perimeter: number = ArrayUtils.sum(coordinatesCollections.map((f) => f.perimeter())); const sortedFeatures = CoordinatesCollection.sort(coordinatesCollections); const lines = sortedFeatures.map((feature) => { return this._createLine(feature); }); lines.forEach((line) => { const geometry = line.geometry as BufferGeometry; const core_geometry = new CoreGeometry(geometry); core_geometry.addNumericAttrib('perimeter', 1, perimeter); }); const geometries = lines.map((l) => l.geometry) as BufferGeometry[]; const merged_geometry = CoreGeometryBuilderMerge.merge(geometries); if (!merged_geometry) { return; } // pti const core_geometry = new CoreGeometry(merged_geometry); core_geometry.addNumericAttrib('pti', 1, 0); const points = core_geometry.points(); const points_count = points.length; for (let i = 0; i < points.length; i++) { const point = points[i]; const pti = i / (points_count - 1); point.setAttribValue('pti', pti); } const mergedObject = this.node.createObject(merged_geometry, ObjectType.LINE_SEGMENTS); return mergedObject; } _createLine(coordinates_collection: CoordinatesCollection): LineSegments { const points_count = coordinates_collection.coordinates.length; const positions: number[] = []; const indices: number[] = []; for (let i = 0; i < points_count; i++) { const coordinates = coordinates_collection.coordinates[i]; positions.push(coordinates.x); positions.push(0); positions.push(coordinates.y); if (i > 0) { indices.push(i - 1); indices.push(i); } } const geometry = new BufferGeometry(); geometry.setAttribute('position', new Float32BufferAttribute(positions, 3)); geometry.setIndex(indices); const object = this.node.createObject(geometry, ObjectType.LINE_SEGMENTS); const core_geometry = new CoreGeometry(geometry); const id_from_name = CoreMapboxString.toId(this.name) % 10000000; // console.log(this.name, id_from_name) core_geometry.addNumericAttrib('id', 1, this.id); core_geometry.addNumericAttrib('name_id', 1, id_from_name); return object; } private _createAllCoordinatesCollections(): CoordinatesCollection[] { const coordinates_collections: CoordinatesCollection[] = []; this.features.forEach((feature) => { this.id = this.id || feature['id']; const feature_geometry = feature.geometry; if (feature_geometry) { const type = feature_geometry['type']; switch (type) { case MULTILINESTRING: const multi_coordinates = feature_geometry['coordinates']; if (multi_coordinates) { for (let i = 0; i < multi_coordinates.length; i++) { const coordinates = multi_coordinates[i]; coordinates_collections.push(this._create_coordinates(coordinates)); } } break; case LINESTRING: coordinates_collections.push(this._create_coordinates(feature_geometry['coordinates'])); break; default: console.warn(`type ${type} not taken into account`); } } }); return coordinates_collections; } private _create_coordinates(raw_coordinates: [number, number][]): CoordinatesCollection { const vectors = raw_coordinates.map((raw_coordinate) => { return new Vector2(raw_coordinate[0], raw_coordinate[1]); }); return new CoordinatesCollection(vectors); } }