当前位置: 首页 > news >正文

three.js通过GEO数据生成3D地图

GeoJSON是一种对地理数据结构进行编码的格式

地图下载 https://datav.aliyun.com/portal/school/atlas/area_selector

生成3D地图

在这里插入图片描述

<script setup>
import * as THREE from 'three';
import { OrbitControls } from 'three/addons/controls/OrbitControls.js';
import * as d3 from 'd3';// 场景
const scene = new THREE.Scene();// 相机
const camera = new THREE.PerspectiveCamera(75, window.innerWidth / window.innerHeight, 0.1, 1000);
camera.position.set(0, 0, 10);// 辅助线
const axesHelper = new THREE.AxesHelper(5); // 长度
scene.add(axesHelper);// 初始化渲染器
const renderer = new THREE.WebGLRenderer({antialias: true // 锯齿模糊
});
// renderer.setClearColor(0x444444, 1); // 设置背景颜色
renderer.setSize(window.innerWidth, window.innerHeight); //设置three.js渲染区域的尺寸(像素px)
document.body.appendChild(renderer.domElement);
// renderer.render(scene, camera);// 相机围绕目标进行轨道运动; 注意OrbitControls会影响 camera.lookAt(x,y,z)失效,需设置controls.target.set(x,y,z)才能生效
const controls = new OrbitControls(camera, renderer.domElement);
// 可设置控制器阻尼,让控制器更有真实效果,必须在你的动画循环里调用.update()
controls.enableDamping = true// 浏览器动画
const clock = new THREE.Clock()
function render(time) {controls.update() // 阻尼控制器更新renderer.render(scene, camera);requestAnimationFrame(render); // 请求动画帧
}
render()const loader = new THREE.FileLoader(); // 加载文件
loader.load('./100000_full.json', (data) => {const jsonData = JSON.parse(data); // 解析json文件operationData(jsonData)
})const map = new THREE.Object3D()
const operationData = (json) => {// console.log(json);const features = json.features; // 数据features.forEach((feature) => {// 创建省份的物体const province = new THREE.Object3D()province.name = feature.properties.name// 获取经纬度坐标const coordinates = feature.geometry.coordinates// console.log(feature.geometry);if (feature.geometry.type === 'Polygon') { // 多边形coordinates.forEach((item) => {const mesh = createMesh(item) // 创建多边形mesh.name = feature.properties.nameprovince.add(mesh) // 添加到省份})} else if (feature.geometry.type === 'MultiPolygon') { // 多多边形coordinates.forEach((item) => {item.forEach((item) => {const mesh = createMesh(item) // 创建多边形mesh.name = feature.properties.nameprovince.add(mesh) // 添加到省份})})}map.add(province) // 添加到地图})// console.log(map);scene.add(map)
}const projection = d3.geoMercator().center([116.4074, 39.9042]).translate([0, 0, 0]) // 以北京为中心,转换后的坐标(本身是一个球, 转换为平面)
const createMesh = (polygon) => {// console.log(polygon);// 根据经纬度创建平面const shape = new THREE.Shape();polygon.forEach((row, i) => {// console.log(row, projection(row));const [longitude, latitude] = projection(row) // 转换后的坐标if (i === 0) {shape.moveTo(longitude, -latitude) // 移动到第一个点} else {shape.lineTo(longitude, -latitude) // 连接到第二个点}})// 根据形状挤出几何体const geometry = new THREE.ExtrudeGeometry(shape, {depth: 5, // 挤出的深度})const color = new THREE.Color(Math.random() * 0xffffff) // 随机颜色const material = new THREE.MeshBasicMaterial({color, // 颜色transparent: true, // 透明opacity: 0.5, // 透明度})return new THREE.Mesh(geometry, material)
}// 监听画面变化,更新渲染画面
window.addEventListener('resize', () => {// 更新摄像头camera.aspect = window.innerWidth / window.innerHeight;// 更新摄像机的投影矩阵camera.updateProjectionMatrix();// 更新渲染器renderer.setSize(window.innerWidth, window.innerHeight);// 设置渲染器的像素比renderer.setPixelRatio(window.devicePixelRatio)
})
</script>

选中高亮效果

let lastPicker = null; // 上一次选中的物体
window.addEventListener('click', (e) => {// 获取鼠标的位置const mouse = new THREE.Vector2();mouse.x = (e.clientX / window.innerWidth) * 2 - 1; // 横坐标mouse.y = -(e.clientY / window.innerHeight) * 2 + 1; // 纵坐标// 获取鼠标点击的位置const raycaster = new THREE.Raycaster(); // 射线raycaster.setFromCamera(mouse, camera); // 设置射线const intersects = raycaster.intersectObjects(map.children); // 获取射线与物体的交点if (intersects.length > 0) {// console.log(intersects[0].object.name); // 输出交点的物体名称if (lastPicker) { // 上一次选中的物体lastPicker.material.color.copy(lastPicker.material.oldColor) // 恢复原来的颜色}lastPicker = intersects[0].object; // 上一次选中的物体lastPicker.material.oldColor = lastPicker.material.color.clone() // 保存原来的颜色lastPicker.material.color.set(0xffffff) // 选中的物体颜色(此时设置的白色)} else {if (lastPicker) { // 上一次选中的物体lastPicker.material.color.copy(lastPicker.material.oldColor) // 恢复原来的颜色}}
})

生成地图3D线

在这里插入图片描述

// 根据经纬度画线
const createLine = (polygon) => {const lineGeometry = new THREE.BufferGeometry(); // 缓冲区const pointsArray = []; // 点polygon.forEach((row) => {const [longitude, latitude] = projection(row) // 转换后的坐标pointsArray.push(new THREE.Vector3(longitude, -latitude, 10)); // 点})lineGeometry.setFromPoints(pointsArray); // 设置点const color = new THREE.Color(Math.random() * 0xffffff) // 随机颜色 // 线const lineMaterial = new THREE.LineBasicMaterial({color, // 颜色}); // 材质return new THREE.Line(lineGeometry, lineMaterial); // 线
}

这里需要改

    if (feature.geometry.type === 'Polygon') { // 多边形coordinates.forEach((item) => {const mesh = createMesh(item) // 创建多边形mesh.name = feature.properties.nameprovince.add(mesh) // 添加到省份const line = createLine(item) // 创建线province.add(line) // 添加到省份})} else if (feature.geometry.type === 'MultiPolygon') { // 多多边形coordinates.forEach((item) => {item.forEach((item) => {const mesh = createMesh(item) // 创建多边形mesh.name = feature.properties.nameprovince.add(mesh) // 添加到省份const line = createLine(item) // 创建线province.add(line) // 添加到省份})})}

相关文章:

  • 2025年5月HCIP题库(带解析)
  • 基于计算机视觉的试卷答题区表格识别与提取技术
  • js var a=如果ForRemove=true,是“normal“,否则为“bold“
  • 网页版部署MySQL + Qwen3-0.5B + Flask + Dify 工作流部署指南
  • 自定义SpringBoot Starter-笔记
  • 当K8S容器没有bash时高阶排查手段
  • Github上如何准确地搜索开源项目
  • (二)毛子整洁架构(CQRS/Dapper/DomianEvent Handler)
  • 8.软考高项(信息系统项目管理师)-沟通管理
  • 作为主动唤醒的节点,ECU上电如何请求通讯
  • String、StringBuilder、StringBuffer的区别
  • 翻转二叉树(简单)
  • 使用原生javascript手动实现一个可选链运算符
  • 牛客——暴力、技巧、字符与数组的使用(强强联合、字符数量)
  • 【工具】解析URL获取实际图片地址下载原始FFHQ图像
  • C++:实现线程池
  • VMware中虚拟机和主机的SSH远程连接
  • langchain使用推理模型如DeepSeek,删除回答中的推理过程<think></think>
  • 数据库实验10 函数存储
  • vitepress 复杂环境引入 mermaid
  • 太空摄影的发展
  • 驱逐行动再加码?特朗普或向利比亚和卢旺达遣送非法移民
  • 中国难以承受高关税压力?外交部:任何外部冲击都改变不了中国经济基本面
  • 马上评|独生子女奖励不能“私了”,政府诚信是第一诚信
  • 中国医药科技出版社回应发布“男性患子宫肌瘤”论文:正在核查
  • 为什么所有动物里,只有人类幼崽发育得这么慢?