【Cesium开发实战】电子围栏功能的实现,可自定义高度

Cesium有很多很强大的功能,可以在地球上实现很多炫酷的3D效果。今天给大家分享一个可自定义高度的电子围栏。

1.话不多说,先展示。

电子围栏

2.设计思路

点击绘制,在地图上可绘制多个点位,双击结束绘制,可对电子围栏起名称以及自定义电子围栏高度,并以列表形式展示,可同时绘制多个电子围栏,点击列表可飞行至对应的电子围栏,也可删除对应的电子围栏。

3.具体代码

<template><div class="page"><el-button @click="draw">绘制</el-button><el-table :data="dataList" border @row-click="rowClick"><el-table-column prop="name" label="名称" align="center" /><el-table-column prop="action" label="操作" align="center"><template #default="scope"><el-button link type="primary" size="small" @click="delEntity(scope.row, scope.$index)"><el-icon :size="16"><ele-Delete /> </el-icon></el-button></template></el-table-column></el-table></div><el-dialog v-model="dialogFormVisible" title="配置" width="500" :close-on-press-escape="false" :close-on-click-modal="false" :show-close="false"><el-form ref="formRef" :model="form" label-width="auto" :rules="rules"><el-form-item label="电子围栏名称" prop="title"><el-input v-model="form.title" placeholder="请输入" /></el-form-item><el-form-item label="电子围栏高度"><el-input-number :min="0" v-model="form.height" placeholder="请输入" /></el-form-item></el-form><template #footer><div class="dialog-footer"><el-button type="primary" @click="submitForm(formRef)"> 确定 </el-button></div></template></el-dialog>
</template><script setup lang="ts">
import { Cesium } from '/@/utils/cesium';
import { onMounted, onUnmounted, reactive, ref } from 'vue';
import { debounce } from 'lodash';const props = defineProps(['viewer']);const dialogFormVisible = ref(false);var handler: any = null;const formRef = ref();const rules = {title: { required: true, message: '请输入电子围栏名称', trigger: 'blur' },
};// 视点名称
const form = reactive({title: '',height: 100,
});var flyOption = {offset: {heading: Cesium.Math.toRadians(1.0114629015290062),pitch: Cesium.Math.toRadians(-23.53661660731824),roll: Cesium.Math.toRadians(0.00324596311071617),},
};
/*** 添加电子围栏*/
var addElectronicFence = (name: string, positions: any) => {let material = getCustomMaterialWall({image: '/src/assets/cesium/wall.png',freely: 'vertical',direction: '+',count: 2,color: Cesium.Color.RED,duration: 2000,});var geometry = props.viewer.entities.add({id: new Date().getTime(),name: name,wall: {positions: Cesium.Cartesian3.fromDegreesArrayHeights(positions),material: material,},});entities.push(geometry);dataList.push({id: geometry.id,name: geometry.name,positions: positions,});
};var getCustomMaterialWall = (options: any) => {if (props.viewer && options && options.image) {return _initWallCustomMaterialProperty(options);}
};// 动态初始化材质线
var _initWallCustomMaterialProperty = (options: any) => {var Color = Cesium.Color,defaultValue = Cesium.defaultValue,defined = Cesium.defined,defineProperties = Object.defineProperties,Event = Cesium.Event,createPropertyDescriptor = Cesium.createPropertyDescriptor,Property = Cesium.Property,Material = Cesium.Material,MaterialType = options.MaterialType || 'wallType' + parseInt(Math.random() * 1000);function WallLinkCustomMaterialProperty(options) {options = defaultValue(options, defaultValue.EMPTY_OBJECT);this._definitionChanged = new Event();this._color = undefined;this._colorSubscription = undefined;this.color = options.color || Color.BLUE;this.duration = options.duration || 3000;this._time = new Date().getTime();}defineProperties(WallLinkCustomMaterialProperty.prototype, {isvarant: {get: function () {return false;},},definitionChanged: {get: function () {return this._definitionChanged;},},color: createPropertyDescriptor('color'),});WallLinkCustomMaterialProperty.prototype.getType = function (time) {return MaterialType;};WallLinkCustomMaterialProperty.prototype.getValue = function (time, result) {if (!defined(result)) {result = {};}result.color = Property.getValueOrClonedDefault(this._color, time, Color.WHITE, result.color);result.image = options.image;result.time = ((new Date().getTime() - this._time) % this.duration) / this.duration;return result;};WallLinkCustomMaterialProperty.prototype.equals = function (other) {return this === other || (other instanceof WallLinkCustomMaterialProperty && Property.equals(this._color, other._color));};//动态墙Material._materialCache.addMaterial(MaterialType, {fabric: {type: MaterialType,uniforms: {color: new Color(1.0, 0.0, 0.0, 0.5),image: options.image,time: 0,},source: _getDirectionWallShader({get: true,count: options.count,freely: options.freely,direction: options.direction,}),},translucent: function (material) {return true;},});return new WallLinkCustomMaterialProperty(options);
};var _getDirectionWallShader = (options: any) => {if (options && options.get) {var materail ='czm_material czm_getMaterial(czm_materialInput materialInput)\n\{\n\czm_material material = czm_getDefaultMaterial(materialInput);\n\vec2 st = materialInput.st;\n\\n\ ';if (options.freely == 'vertical') {//(由下到上)materail += 'vec4 colorImage = texture2D(image, vec2(fract(float(' + options.count + ')*st.t ' + options.direction + ' time), fract(st.s)));\n\ ';} else {//(逆时针)materail += 'vec4 colorImage = texture2D(image, vec2(fract(float(' + options.count + ')*st.s ' + options.direction + ' time), fract(st.t)));\n\ ';}//泛光materail +='vec4 fragColor;\n\fragColor.rgb = (colorImage.rgb+color.rgb) / 1.0;\n\fragColor = czm_gammaCorrect(fragColor);\n\ ';materail +=' material.diffuse = colorImage.rgb;\n\material.alpha = colorImage.a;\n\material.emission = fragColor.rgb;\n\\n\return material;\n\}\n\';return materail;}
};const drawing = ref(false);// 电子围栏列表数据
const dataList: any = reactive([]);// 电子围栏实体列表
var entities: any = [];// 电子围栏point实体列表
var pointEntities: any = [];// 当前选中实体索引
var geometryIndex: any = null;/*** 点击绘制按钮*/
const draw = () => {drawing.value = true;
};/*** 点击表格一行*/
const rowClick = (row: any, column: any, event: Event) => {if (column && column.property) {let index = dataList.findIndex((v: any) => v.id === row.id);if (index !== -1) {geometryIndex = index;props.viewer.flyTo(entities[index], flyOption);}}
};/*** 删除已绘制的图形*/
const delEntity = (item: any, index: number) => {props.viewer.entities.remove(entities[index]);entities.splice(index, 1);dataList.splice(index, 1);
};/*** 点击确定绘制*/
const submitForm = async (formEl: any) => {const valid = await formEl.validate();if (valid) {var pointList: any = [];pointEntities.forEach((entity: any) => {var { longitude, latitude } = getPosition(entity);pointList.push(longitude, latitude, form.height);props.viewer.entities.remove(entity);});var { longitude, latitude } = getPosition(pointEntities[0]);pointList.push(longitude, latitude, form.height);pointEntities = [];addElectronicFence(form.title, pointList);dialogFormVisible.value = false;formEl.resetFields();}
};/*** 添加鼠标事件*/
const addClickHandler = () => {handler = new Cesium.ScreenSpaceEventHandler(props.viewer.scene.canvas);// 单击绘制handler.setInputAction((event: any) => {leftClickHandler(event);}, Cesium.ScreenSpaceEventType.LEFT_CLICK);// 双击结束point绘制,开始围栏绘制handler.setInputAction((event: any) => {if (drawing.value) {drawing.value = false;}if (pointEntities.length) {dialogFormVisible.value = true;}}, Cesium.ScreenSpaceEventType.LEFT_DOUBLE_CLICK);
};/*** 单击左键*/
const leftClickHandler = debounce((event: any) => {if (drawing.value) {let cartesian = props.viewer.scene.pickPosition(event.position);let cartographic = Cesium.Cartographic.fromCartesian(cartesian);let coordinate = {longitude: Cesium.Math.toDegrees(cartographic.longitude),latitude: Cesium.Math.toDegrees(cartographic.latitude),height: cartographic.height + 1,};var pointGeometry = new Cesium.Entity({name: 'Sample Label',position: Cesium.Cartesian3.fromDegrees(coordinate.longitude, coordinate.latitude, coordinate.height),point: {// 实体是一个点pixelSize: 10, // 点的大小color: Cesium.Color.RED, // 点的颜色},});props.viewer.entities.add(pointGeometry);pointEntities.push(pointGeometry);}
}, 100);/*** 获取实体的经纬度*/
const getPosition = (entity: any) => {// 获取点实体的位置var position = entity.position.getValue(Cesium.JulianDate.now());// 转换Cartesian3到经纬度var cartographic = Cesium.Cartographic.fromCartesian(position, props.viewer.scene.globe.ellipsoid);var longitude = Cesium.Math.toDegrees(cartographic.longitude);var latitude = Cesium.Math.toDegrees(cartographic.latitude);return {longitude,latitude,};
};onMounted(() => {addClickHandler();
});onUnmounted(() => {entities.forEach((entity: any) => {props.viewer.entities.remove(entity);});handler.removeInputAction(Cesium.ScreenSpaceEventType.LEFT_CLICK);handler.removeInputAction(Cesium.ScreenSpaceEventType.LEFT_DOUBLE_CLICK);
});
</script><style scoped>
.page {position: absolute;right: 10px;top: 10px;color: #fff;background: #fff;padding: 10px;border-radius: 5px;width: 300px;
}
</style>

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/bicheng/42910.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

债券一级市场、二级市场介绍

债券市场可以分为一级市场和二级市场&#xff0c;它们分别在债券发行和交易中扮演着不同的角色。 债券一级市场 一级市场是指债券的初级发行市场。在这个市场上&#xff0c;债券发行人&#xff08;如政府、企业等&#xff09;通过中介机构&#xff08;如银行、证券公司等&…

昇思25天学习打卡营第14天|Pix2Pix实现图像转换

Pix2Pix是基于条件生成对抗网络&#xff08;cGAN, Condition Generative Adversarial Networks &#xff09;实现的一种深度学习图像转换模型&#xff0c;该模型是由Phillip Isola等作者在2017年CVPR上提出的&#xff0c;可以实现语义/标签到真实图片、灰度图到彩色图、航空图到…

leetcode--二叉搜索子树的最大键值和

leetcode地址&#xff1a;二叉搜索子树的最大键值和 给你一棵以 root 为根的 二叉树 &#xff0c;请你返回 任意 二叉搜索子树的最大键值和。 二叉搜索树的定义如下&#xff1a; 任意节点的左子树中的键值都 小于 此节点的键值。 任意节点的右子树中的键值都 大于 此节点的键值…

IOC、DI<4> Unity

IOC&#xff08;&#xff09;&#xff1a;控制反转&#xff0c;把程序上层对下层的依赖&#xff0c;转移到第三方的容器来装配 是程序设计的目标&#xff0c;实现方式包含了依赖注入和依赖查找&#xff08;.net里面只有依赖注入&#xff09; DI&#xff1a;依赖注入&#xff0c…

基于jeecgboot-vue3的Flowable流程-集成仿钉钉流程(一)一些样式的调整使用

因为这个项目license问题无法开源&#xff0c;更多技术支持与服务请加入我的知识星球。 1、比如下面的发起人双击后出现的界面不正常&#xff0c; 看它的样式主要是这个里面的margin-left应该太小了&#xff0c; [data-v-45b533d5] .el-tabs__content { margin-top: 50px;mar…

多商户酒店预定SAAS系统(APP):云端革新引领行业未来

在数字化转型的浪潮中&#xff0c;酒店行业正经历着前所未有的变革。多商户酒店预定SAAS系统&#xff08;APP&#xff09;作为一种创新的解决方案&#xff0c;为酒店提供了一个集中化、高效、灵活的在线预订和管理平台。 SAAS系统的核心价值 SAAS系统通过云端服务&#xff0c…

7.1作业6

uart4.h #ifndef __UART4_H__ #define __UART4_H__ #include "stm32mp1xx_gpio.h" #include "stm32mp1xx_rcc.h" #include "stm32mp1xx_uart.h" //rcc/gpio/uart4初始化 void hal_uart4_init(); //发送一个字符 void hal_put_char(const char s…

在C++中暂停程序的几种方法

在开发C应用程序时&#xff0c;有时需要让程序暂停一段时间。例如&#xff0c;你可能需要等待某个操作完成&#xff0c;或是为用户提供某种延迟。在C中&#xff0c;有多种方法可以实现这一需求。 1. 使用 std::this_thread::sleep_for (C11及以上版本) std::this_thread::sle…

【JavaScript脚本宇宙】XML和HTML解析库大揭秘:快速选出最适合你的工具

解析XML和HTML文档&#xff1a;六大常用库功能对比 前言 在现代的Web开发中&#xff0c;处理XML和HTML文档是一个常见的任务。为了更加高效地解析和操作这些文档&#xff0c;开发人员通常会使用各种库和工具。本文将介绍几个流行的XML和HTML解析库&#xff0c;分别对其功能、…

wzoi.cc基数排序AC代码(含链接)

题目描述: 基数排序是一种并不基于关键字间比较和移动操作的排序算法。基数排序是一种借助多关键字排序的思想对单逻辑关键字进行排序的方法。 通过对每一个关键字分别依次进行排序&#xff0c;可以令整个关键字序列得到完整的排序。 在本题中&#xff0c;读入一串16位&…

react native中依赖@react-native-clipboard/clipboard库实现文本复制以及在app中获取复制的文本内容

react native中依赖react-native-clipboard/clipboard库实现文本复制以及在app中获取复制的文本内容 第三方库 第三方库 react-native-clipboard 我的项目react native0.72 我使用react-native-clipboard/clipboard1.13.2 npm install --save react-native-clipboard/clipboa…

留学Essay写作能够顺利拿“A”吗?

对于留学生来说Essay难度不可谓不大&#xff0c;很多同学刚到国外留学的时候都会被Essay难倒一大片&#xff0c;可见Essay写作的难度&#xff0c;当然了&#xff0c;不乏有留学生会选择Essay写作&#xff0c;这其实也是无可厚非的。那么Essay写作想拿高分需要注意哪些呢&#x…

香橙派编译linux内核支持ebpf和虚拟WIFI

前言 上一篇文章香橙派5plus上跑云手机方案一 redroid(带硬件加速)中说了怎么运行redroid&#xff0c;这篇补一下怎么修改参数编译内核。 补充 上篇文章有个内容需要补充一下&#xff1a;更新完内核需要用下面的命令防止内核被apt更新&#xff0c;不然后面使用apt update又回…

浅析C++引用

浅析C引用"&" ​ C中引入了一个新的语言特性——引用(&)&#xff0c;它表示某一对象的别名&#xff0c;对象与该对象的引用都是指向统一地址。那么我们就来看看关于引用的一些知识点吧&#x1f9d0; 特性 引用在定义时必须初始化一个变量可以有多个引用引…

【Tools】了解人工通用智能 (AGI):未来的智能体

什么是人工通用智能 (AGI)&#xff1f; 人工通用智能&#xff08;Artificial General Intelligence&#xff0c;AGI&#xff09;是指一种能够理解、学习和应用知识&#xff0c;具有像人类一样广泛和通用的认知能力的智能系统。与专门处理特定任务的人工智能&#xff08;AI&…

如何设计一个高可扩展的分布式架构?

如何设计一个高可扩展的分布式架构&#xff1f; 大家好&#xff0c;我是微赚淘客系统3.0的小编&#xff0c;也是冬天不穿秋裤&#xff0c;天冷也要风度的程序猿&#xff01; 1. 引言&#xff1a;分布式架构的重要性 随着互联网应用的发展&#xff0c;单一服务器往往难以满足…

【Python】一文向您详细介绍 np.inner()

【Python】一文向您详细介绍 np.inner() 下滑即可查看博客内容 &#x1f308; 欢迎莅临我的个人主页 &#x1f448;这里是我静心耕耘深度学习领域、真诚分享知识与智慧的小天地&#xff01;&#x1f387; &#x1f393; 博主简介&#xff1a;985高校的普通本硕&#xff0c;曾…

算法数据结构必备篇章2

文章目录 36. (必备)二叉树高频题目上37. (必备)二叉树高频题目下38. (必备)常见经典递归过程解析39. (必备)嵌套类问题的递归解题套路40. (必备)N皇后问题(含位运算求解)41. (必备)最大公约数, 同余原理42. (必备)对数器打表找规律的技巧43. (必备)根据数据量猜解法的技巧-天字…

6、Redis系统-数据结构-02-链表

二、List&#xff08;列表&#xff09; 1、List 数据结构的必要性 List 是一种有序的数据结构&#xff0c;可以按顺序存储多个字符串。它的主要特点如下&#xff1a; 有序性&#xff1a;List 中的元素是有序的&#xff0c;可以通过索引访问。双向操作&#xff1a;List 支持从…

JavaWeb系列二十二: 线程数据共享和安全(ThreadLocal)

韩顺平-线程数据共享和安全ThreadLocal 什么是ThreadLocal?ThreadLocal环境搭建ThreadLocal快速入门ThreadLocal源码阅读threadLocal.set()源码threadLocal.get()源码 什么是ThreadLocal? ThreadLocal的作用: 可以实现在同一个线程数据共享, 从而解决多线程数据安全问题.Thr…