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

Vue(3.3.4)+three.js(0.161.0)实现3D可视化地图

一.前言


        由于最近在学习three.js,所以观摩了一下掘金,csdn等网站上的有关这部分的内容,刚好看到一个带你入门three.js——从0到1实现一个3d可视化地图 - 掘金 (juejin.cn),再加上我的专业属性是地理相关,可以说是专业对口,但文章已经是三年以前写的,而且没有在框架底下完成,有关three的很多API也发生了更改,所以我的思路是来自该篇文章,我进行了模仿和相应的修改,但是大致没有发生改变,可以说是站在前人的肩膀上。

二.预览


 

三.实现


         首先就是开启一个vue项目,再npm install --save three,再引入一下d3就可以了,配置方面没有什么好配置的,这方面大家应该是没问题的。将代码写在子组件里,再引入到App.vue中展示就可以了。需要注意用到的全国的json数据来自DataV.GeoAtlas地理小工具系列 (aliyun.com)

子组件xx.vue对应代码

<template><div id="container" ref="canvasContainer"></div><div id="tooltip" ref="tooltip"></div>
</template><script setup>import * as THREE from 'three';//OrbitControls 是一个附加组件,必须显式导入import { OrbitControls } from 'three/addons/controls/OrbitControls.js';//墨卡托投影转换可以把我们经纬度坐标转换成我们对应平面的2d坐标,d3里面自带墨卡托投影转换//该引入方式是查阅官网得到的import * as d3 from "https://cdn.jsdelivr.net/npm/d3@7/+esm";import { onMounted, onUnmounted,ref } from 'vue';let canvasContainer = ref(null);let tooltip = ref(null)let scene,camera,renderer,ambientLight,raycaster,mouse;let lastPick = null;//初始化摄像机function initCamera(){camera = new THREE.PerspectiveCamera(75,canvasContainer.value.offsetWidth / canvasContainer.value.offsetHeight, 0.1, 1000);camera.position.set(0,0,120);camera.lookAt(scene.position);}//初始化rendererfunction initRenderer(){renderer = new THREE.WebGLRenderer();renderer.setSize(canvasContainer.value.offsetWidth,canvasContainer.value.offsetHeight)}//初始化灯光function initLight(){ambientLight = new THREE.AmbientLight(0xffffff,20);}//加载json数据function loadJson(){const loader = new THREE.FileLoader();loader.load('src/assets/中华人民共和国.json',(data)=>{const jsondata = JSON.parse(data);generateGeometry(jsondata)console.log(jsondata);})}// 根据JSON数据生成地图几何体function generateGeometry(jsondata){let map = new THREE.Object3D();// 使用d3的地图投影const projection = d3.geoMercator().center([104.0,37.5]).translate([0,0]);// 遍历每个省份,创建几何体jsondata.features.forEach((element)=>{let province = new THREE.Object3D();const coordinates = element.geometry.coordinates;if(Array.isArray(coordinates[0][0][0])){coordinates.forEach((multiPolygon)=>{multiPolygon.forEach((polygon)=>{const shape = new THREE.Shape();const points = [];polygon.forEach((coord,i)=>{const [x,y] = projection(coord);if(i===0) shape.moveTo(x,-y);else shape.lineTo(x,-y);points.push(new THREE.Vector3(x,-y,5));})const lineGeometry = new THREE.BufferGeometry().setFromPoints(points);const lineMaterial = new THREE.LineBasicMaterial({ color: 'white' });const line = new THREE.Line(lineGeometry, lineMaterial);const extrudeSettings = { depth: 10, bevelEnabled: false };const geometry = new THREE.ExtrudeGeometry(shape, extrudeSettings);const material = new THREE.MeshBasicMaterial({ color: '#2defff', transparent: true, opacity: 0.6 });const material1 = new THREE.MeshBasicMaterial({color: '#3480C4',transparent: true,opacity: 0.5,})const mesh = new THREE.Mesh(geometry, [material,material1]);province.properties = element.properties;province.add(mesh);province.add(line);})})}else if(Array.isArray(coordinates[0][0])){coordinates.forEach((polygon)=>{const shape = new THREE.Shape();const points = [];polygon.forEach((coord,i)=>{const [x,y] = projection(coord);if(i===0) shape.moveTo(x,-y);else shape.lineTo(x,-y);points.push(new THREE.Vector3(x,-y,5));})const lineGeometry = new THREE.BufferGeometry().setFromPoints(points);const lineMaterial = new THREE.LineBasicMaterial({ color: 'white' });const line = new THREE.Line(lineGeometry, lineMaterial);const extrudeSettings = { depth: 10, bevelEnabled: false };const geometry = new THREE.ExtrudeGeometry(shape, extrudeSettings);const material = new THREE.MeshBasicMaterial({ color: '#2defff', transparent: true, opacity: 0.6 });const material1 = new THREE.MeshBasicMaterial({color: '#3480C4',transparent: true,opacity: 0.5,})const mesh = new THREE.Mesh(geometry, [material,material1]);province.properties = element.properties;province.add(mesh);province.add(line);})}map.add(province);})scene.add(map);}// 设置光线投射器和鼠标位置,用于检测鼠标悬停对象function setRaycaster(){raycaster = new THREE.Raycaster();mouse = new THREE.Vector2();const onMouseMove = (event) => {mouse.x = (event.clientX / canvasContainer.value.offsetWidth) * 2 - 1mouse.y = -(event.clientY / canvasContainer.value.offsetHeight) * 2 + 1tooltip.value.style.left = event.clientX + 2 + 'px'tooltip.value.style.top = event.clientY + 2 + 'px'}window.addEventListener('mousemove', onMouseMove, false)}// 显示或隐藏工具提示function showTip(){if(lastPick){const properties = lastPick.object.parent.properties;tooltip.value.textContent = properties.name;tooltip.value.style.visibility = 'visible';console.log(tooltip.value.textContent);}else{tooltip.value.style.visibility = 'hidden';}}// 动画循环,用于渲染场景和更新状态function animate() {requestAnimationFrame(animate);raycaster.setFromCamera(mouse,camera);const intersects = raycaster.intersectObjects(scene.children,true);if (lastPick) {lastPick.object.material[0].color.set('#2defff')lastPick.object.material[1].color.set('#3480C4')}lastPick = nulllastPick = intersects.find((item) => item.object.material && item.object.material.length === 2)if (lastPick) {lastPick.object.material[0].color.set(0xff0000)lastPick.object.material[1].color.set(0xff0000)}showTip();renderer.render(scene, camera);}//窗口大小改变时,更新摄像机的宽高比和渲染器的大小function handleResize(){if(camera && renderer && canvasContainer.value){camera.aspect = canvasContainer.value.offsetWidth / canvasContainer.value.offsetHeight;camera.updateProjectionMatrix();renderer.setSize(canvasContainer.value.offsetWidth, canvasContainer.value.offsetHeight);}}// 组件挂载时的初始化逻辑onMounted(()=>{scene = new THREE.Scene();setRaycaster();initLight();scene.add(ambientLight);initCamera();loadJson();initRenderer();canvasContainer.value.appendChild(renderer.domElement);new OrbitControls(camera,canvasContainer.value)animate();window.addEventListener('resize',handleResize)})onUnmounted(()=>{window.removeEventListener('resize',handleResize)})
</script><style>body{margin: 0;padding: 0;overflow: hidden;}#container{/* border: 1px solid black; */width: 100vw;height: 100vh;}#tooltip {position: absolute;z-index: 2;background: white;padding: 10px;border-radius: 5px;visibility: hidden;}
</style>

注意在用JSON数据生成地图集合体时分两种情况是因为:

不同省份数据数组嵌套的层数不一样,类似于下面这两地

 

四.总结

        共勉,如果对于实现的步骤还有疑惑,可以转至我在前言分享的那篇文章 ,它对于实现步骤更详细,可以结合着看。

 

相关文章:

Vue(3.3.4)+three.js(0.161.0)实现3D可视化地图

一.前言 由于最近在学习three.js,所以观摩了一下掘金&#xff0c;csdn等网站上的有关这部分的内容&#xff0c;刚好看到一个带你入门three.js——从0到1实现一个3d可视化地图 - 掘金 (juejin.cn)&#xff0c;再加上我的专业属性是地理相关&#xff0c;可以说是专业对口&#xf…...

瑞吉苍穹外卖如何拓展?已经经过不同公司多轮面试。项目中会问到哪些问题?以及问题如何解决?

别催了&#xff0c;别催了&#xff0c;先收藏吧。 作者大大正在加班加点完成。 文章会尽快发布&#xff0c;关注收藏&#xff0c;尽请期待。 想要加入并查阅作者的知识库可以联系作者 不要白嫖&#xff0c;通过后&#xff0c;附上关注和收藏截图。 已有众多小伙伴加入 目前…...

动态内存分配

目录 存在动态内存分配的原因动态内存函数mallocfreecallocrealloc 常见的动态内存错误C/C程序的内存开辟柔性数组柔性数组的特点柔性数组的使用柔性数组的优势 存在动态内存分配的原因 内存开辟方式,例如&#xff1a; int val 20;在栈空间上开辟四个字节 char arr[10] { 0 …...

【C语言】常见的动态内存管理错误

前言 上一篇介绍了C语言中 动态内存管理函数&#xff0c;本片讲解的是 在我们使用动态内存管理时 常见的错误&#xff0c;一起来看看吧~ 欢迎关注个人主页&#xff1a;逸狼 创造不易&#xff0c;可以点点赞吗~ 如有错误&#xff0c;欢迎指出~ 目录 1.对NULL指针的解引⽤操作 错…...

数据结构之二叉树的精讲

&#x1d649;&#x1d65e;&#x1d658;&#x1d65a;!!&#x1f44f;&#x1f3fb;‧✧̣̥̇‧✦&#x1f44f;&#x1f3fb;‧✧̣̥̇‧✦ &#x1f44f;&#x1f3fb;‧✧̣̥̇:Solitary_walk ⸝⋆ ━━━┓ - 个性标签 - &#xff1a;来于“云”的“羽球人”。…...

ETL是什么

一、ETL概念 ETL&#xff0c;是英文Extract-Transform-Load的缩写&#xff0c;用来描述将数据从来源端经过抽取&#xff08;extract&#xff09;、转换&#xff08;transform&#xff09;、加载&#xff08;load&#xff09;至目的端的过程。ETL一词较常用在数据仓库&#xff…...

华为配置WLAN高密业务示例

配置WLAN高密业务示例 组网图形 图1 配置高密WLAN环境网络部署组网图 业务需求组网需求数据规划配置思路配置注意事项操作步骤配置文件 业务需求 体育场由于需要接入用户数量很大&#xff0c;AP间部署距离较小&#xff0c;因此AP间的干扰较大&#xff0c;可能导致用户上网网…...

C++——类和对象(1)

1. 类 我们之前提及过C语言是面向过程的语言&#xff0c;其解决问题的方式是关注问题过程&#xff0c;然后逐步解决。而C是面向对象编程&#xff0c;聚焦于对象&#xff0c;依靠多个对象之间的交互关系解决问题。而类这个概念的引入则是面向对象的最深刻体现。 1.1 C中的结构体…...

vue+element ui上传图片到七牛云服务器

本来打算做一个全部都是前端完成的资源上传到七牛云的demo&#xff0c;但是需要获取token&#xff0c;经历了九九八十一难&#xff0c;最终还是选择放弃&#xff0c;token从后端获取&#xff08;springboot&#xff09;。如果你们有前端直接能解决的麻烦记得私我哦&#xff01;…...

学不动系列-git-hooks和husky+lintstage

git-hooks 为了保证提交的代码符合规范&#xff0c;可以在上传代码时进行校验。常用husky来协助进行代码提交时的eslint校验。husky是基于git-hooks来实现&#xff0c;在使用husky之前&#xff0c;我们先来研究一下git-hooks。 构建git-hooks测试项目 需要使用git-hooks就需…...

K8S相关小技巧《四》

需求&#xff1a; 我作为Kubernetes的集群管理员&#xff0c;前一段时间有收到一个需求&#xff0c;需要我创建一个受限访问的用户kubeconfig&#xff0c;提供给跳板机的某用户。 该kubeconfig需要在非Kubernetes节点的某跳板机上由指定的非root用户使用&#xff0c;该用户仅能…...

Delphi 报错 Type androidx.collection.ArraySet is defined multiple times

Delphi 11 建立一个新的 Multi-Device Application 编译成app的时候报错 报错信息 [PAClient Error] Error: E7688 Unable to execute "E:\Program\Java\jdk1.8.0_301\bin\java.exe" -cp "e:\program\embarcadero\studio\22.0\bin\Android\r8-3.3.28.jar"…...

Post请求中文乱码问题

url*************************************这里填写自己请求的网址 response requests.post(url, datajson.dumps(body),headersheader) r response.text print 打印乱码长这样&#xff1a; data:{“code”:0,“data”:{“end”:false,“message”:“{\n “ˆ—¡A”: [“…...

LeetCode -- 79.单词搜索

1. 问题描述 给定一个 m x n 二维字符网格 board 和一个字符串单词 word 。如果 word 存在于网格中&#xff0c;返回 true &#xff1b;否则&#xff0c;返回 false 。 单词必须按照字母顺序&#xff0c;通过相邻的单元格内的字母构成&#xff0c;其中“相邻”单元格是那些水…...

单元测试、集成测试、系统测试有什么不同?

单元测试、集成测试和系统测试是软件测试开发中不可或缺的部分。 单元测试&#xff1a; 范围&#xff1a;单元测试是对软件中最小的可测试单元的测试&#xff0c;通常是函数、方法或类。 目的&#xff1a;它的目标是验证每个单独的单元是否按照预期工作&#xff0c;以增加代码…...

数据迁移DTS | 云上MySQL 数据库迁移至达梦数据库

引入 云上 MySQL 数据库 —> 向达梦国产化数据库迁移 下载&安装 达梦客户端工具 DM->可参考之前国产化专栏达梦文章 创建模式 在客户端分别依次执行以下命令脚本&#xff08;这里没有通过客户端管理工具去创建达梦数据库的模式&#xff0c;当然也可以通过图形化界…...

Linux进程管理:(二)进程调度原语

文章说明&#xff1a; Linux内核版本&#xff1a;5.0 架构&#xff1a;ARM64 参考资料及图片来源&#xff1a;《奔跑吧Linux内核》 Linux 5.0内核源码注释仓库地址&#xff1a; zhangzihengya/LinuxSourceCode_v5.0_study (github.com) 进程调度的概念比较简单&#xff0c…...

Compose 介绍

Compose 介绍 Android Compose 是 Google 官方推出的用于构建原生 Android UI 的现代工具包。它使用 Kotlin 语言编写&#xff0c;可以帮助开发人员更轻松、更快速地创建精美、响应式和高性能的 Android 应用。 Compose 的优势 声明式 UI&#xff1a; Compose 使用声明式 UI…...

5分钟搞定Python中函数的参数

函数的灵活性非常高&#xff0c;除了常规定义的位置参数以外&#xff0c;还支持默认参数、关键字参数、以及可变参数 ... 这样以来&#xff0c;不但能应对各种复杂的情况&#xff0c;甚至还可以简化调用者的代码。 位置参数 在调用函数时&#xff0c;一般会根据函数定义的参数…...

Gitlab: 私有化部署

目录 1. 说明 2. 资源要求 3. 安装 4. 配置实践 4.1 服务器 4.2 人员与项目 4.2 部署准备 4.2.1 访问变量及用户账号设置 4.2.2 Runner设置 4.2.3 要点 5. 应用项目 CI/CD 6. 参考 1. 说明 gitlab是一个强大且免费的代码管理/部署工具&#xff0c;能统一集成代码仓…...

Face Analysis WebUI快速部署:Docker Compose一键拉起+模型自动下载

Face Analysis WebUI快速部署&#xff1a;Docker Compose一键拉起模型自动下载 1. 引言&#xff1a;智能人脸分析&#xff0c;触手可及 你是否曾经想过&#xff0c;在自己的电脑上搭建一个专业级的人脸分析系统&#xff1f;不需要复杂的配置&#xff0c;不需要手动下载模型&a…...

C++ STL常用容器1——string容器

写在前面&#xff1a;⭐如果本篇博文对你有帮助&#xff0c;那就关注 点赞 收藏一下吧&#xff01; 目录 1.string基本概念 2.string构造函数 3.string容器 4.string字符串拼接 5.string查找和替换 6.string字符串比较 7.string字符存取 8.string插入和删除 9.strin…...

PMD大数据处理终极指南:如何高效分析TB级代码仓库的10个技巧

PMD大数据处理终极指南&#xff1a;如何高效分析TB级代码仓库的10个技巧 【免费下载链接】pmd An extensible multilanguage static code analyzer. 项目地址: https://gitcode.com/gh_mirrors/pm/pmd PMD作为一款可扩展的多语言静态代码分析工具&#xff0c;能够帮助开…...

Windows本地AI新玩法:Docker Compose一键部署Ollama与Open WebUI,小白也能玩转私有大模型

1. 为什么要在Windows上部署本地大模型&#xff1f; 最近两年AI技术发展迅猛&#xff0c;各种大语言模型层出不穷。但很多朋友可能都有这样的困扰&#xff1a;每次想用AI都得联网&#xff0c;还得担心隐私问题。其实现在完全可以在自己的Windows电脑上搭建一个私有大模型&#…...

PP-DocLayoutV3与JavaScript交互:实现浏览器内文档实时预览与分析

PP-DocLayoutV3与JavaScript交互&#xff1a;实现浏览器内文档实时预览与分析 你有没有遇到过这样的场景&#xff1f;用户上传了一份几十页的PDF报告&#xff0c;你需要在网页上快速预览内容&#xff0c;并且自动识别出里面的标题、段落、表格和图片位置。传统做法是让用户下载…...

unity urp材质球大全

Unityurp>PBRMaterialBundleVol1-1资源-CSDN下载 Unityurp>PBRMaterialBundleVol1-2资源-CSDN下载...

13_主流低代码平台深度对比:简道云、宜搭、LowCodeEngine技术选型

主流低代码平台深度对比&#xff1a;简道云、宜搭、LowCodeEngine技术选型 摘要&#xff1a;市场上低代码平台众多&#xff0c;如何选择适合自身业务需求的平台&#xff1f;本文深度对比简道云、钉钉宜搭、阿里LowCodeEngine三大主流低代码平台&#xff0c;从架构设计、产品定位…...

第5章,[标签 Win32] :GDI 的基本图形

专栏导航 上一篇&#xff1a;第5章&#xff0c;[标签 Win32] &#xff1a;GDI 函数调用 回到目录 下一篇&#xff1a;第5章&#xff0c;[标签 Win32] &#xff1a;GDI 的其他方面的分类 本节前言 对于本节所讲解的知识&#xff0c;有可能&#xff0c;你会需要时不时地参考…...

面试官:聊聊Redis中RDBAOF持久化原理!

Redis 中数据的持久化前言我们知道 Redis 是内存数据库&#xff0c;所有操作都在内存上完成。内存的话&#xff0c;服务器断电&#xff0c;内存上面的数据就会丢失了。这个问题显然是需要解决的。Redis 中引入了持久化来避免数据的丢失&#xff0c;主要有两种持久化的方式 RDB …...

澜起科技年营收55亿:净利22亿 上海融迎及一致行动人套现超10亿

雷递网 雷建平 4月14日澜起科技股份有限公司&#xff08;简称&#xff1a;“澜起科技”&#xff0c;公司代码&#xff1a;688008&#xff09;日前发布2025年的财报。财报显示&#xff0c;澜起科技2025年营收为54.56亿元&#xff0c;较上年同期的36.39亿元增长49.94%。澜起科技称…...