IT数码 购物 网址 头条 软件 日历 阅读 图书馆
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
图片批量下载器
↓批量下载图片,美女图库↓
图片自动播放器
↓图片自动播放器↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁
 
   -> 游戏开发 -> 基于Cesium原生方法在项目上的一些使用 -> 正文阅读

[游戏开发]基于Cesium原生方法在项目上的一些使用

刚来公司的项目上的时候,是从离职的前同事手中接过了代码。看别人的代码永远是一种很难受的行为,开立一个新篇章,存储那些关于Cesium的,在项目中“这样设计应该更好”的一些规范和逻辑。多加注释这种个人喜好的代码规范就不在这篇文章的讨论范围内了。

一、新建数据集datasource来管理entity

1.我们该如何存储Entity对象,让我们更方便的找到它

前同事,在项目最喜欢用数组了。每次新增的Entity都是用数组去存储,然后清空的时候,去遍历这个数组,再一个个的去除。

var entityArr = [];        // 存储绘制的实体entity
function removeAllEntity(){
    entityArr.forEach(item){
        viewer.entities.remove(entity)
    }
}

初学Cesium的时候,觉得这也算是个不错的方法。毕竟可以随时去除掉地图上我们想要去除的entity。后来,有一个需求,要求地图上的实体进行聚合的时候,我接触到了Cesium官方的数据集合:new Cesium.DataSource()

并且,在官方的示例中,使用了:new Cesium.CustomDataSource(name)? ?来完成一个聚合的效果。这给了我们什么启发呢?我们可以参照PS里面的图层思想,不需要把所有的entity都绘制在viewer.entities里面,而是绘制在我们自己定义的数据集里面。

var selfDataSource = new Cesium.CustomDataSource('moduleName-whatUse');
viewer.dataSources.add(selfDataSource);
selfDataSource.entities.add({...option实体配置项});

先创建一个datasource,然后将这个集合添加到viewer里面,最后我们就只需要向这个集合添加绘制我们的entity里面了。

这样做的好处在哪里?哪些场景可以应用到?

需要频繁的控制显示隐藏与删除某些entity的时候。比如说,地图上已经绘制好了全国身份的边界线,然后点击某个省份的时候,其他省份的边界线就不要了,只显示点击选中的省份。回退到全国视角的时候,又需要重新把它们绘制出来。

如果是用数组存储这些对象,可能就需要遍历这个数组,然后每一个entity设置为隐藏,或者删除。

但是如果这些entity都绘制在我们自己定义的datasource上,那我们就只需要控制这个datasource的显示与隐藏了。

var arr = [];        // 存了很多entity对象
var datasource = new Cesium.CustomDataSource(name);    // 自己定义的数据集合
viewer.dataSources.add(dataSource);

// 用数组去控制,删除/隐藏
arr.forEach(function (entity) {
    viewer.entities.remove(entity);    // 删除
    entity.show = false;        // 隐藏
})

// 数据集合来控制,删除/隐藏
datasource.show = false;        // 隐藏
datasource.entities.removeAll();    // 删除所有

理论上,我们可以创建很多个数据集来管理我们的entity。

二、使用Cesium自带的属性,来描述一个entity?

?每一个绘制的entity都可以自己设置id和name,但是工作中,根据一个坐标点绘制的entity只用name和id来描述它,有点太鸡肋了。特别是涉及到点击事件的时候。

试想一下这样的一种需求情况:

某个地图上的图标被点击后,需要在UI界面上显示它的一些信息。

这个还好,我们用一个ID来标识它之后,用接口去请求数据,然后渲染更新界面UI。

那如果说,这个需求变动成这样呢?

某个地图上的图标被点击后,需要在UI界面上显示它所在的分类,上级分类,这个坐标点所在的等级。

也不难吧,不就多请求几次接口的问题?但是,我们通过接口请求到地图坐标的时候,我们就已经知道了这个坐标的一些信息。我们是不是要更加利用这些信息?

它就是:properties。Entity的配置项的其中一个,这个并不是一些可视化的图形什么的,更多的用来描述这个Entity。

let entity = viewer.entities.add({
    position,
    billboard,
    properties: {
        'name': Object对象
    }
})

在讲述使用之前,我来说一下前同事如何处理 entity的点击事件。

添加地图点击事件监听 =》判断是否点击到了实体 =》拿到entity的name或者id,然后判断是否包含某个字符串 =》对应的点击事件。

前同事绘制entity的时候,要么是给这个entity取了个特殊点的ID,或者是name用一个下划线来拼接的字符串。很多东西都用下划线来拼接成字符串然后赋值给name?

前同事的点击事件逻辑:

var handle = new Cesium.ScreenSpaceEventHandler(viewer.scene.canvas);
handle.setInputAction(function(event) {
    var pick = viewer.scene.pick(event.position);
    if (Cesium.defined(pick) && (pick.id) && (pick.id._name)){    // 这里还同时判断有没有给name
        if(pick.id._name !== '某个字符串'){
            // 处理逻辑
            // 发起接口请求
        }else if(pick.id._name.indexOf('某个字符串') > -1){
            // 处理逻辑
            // 发起接口请求
        }
    }
},Cesium.ScreenSpaceEventType.LEFT_CLICK)

如果我们在绘制的时候,用properties来存储,那我们只要这样。

let entity = viewer.entities.add({
    position,
    billboard,
    properties: {
        'name': Object对象,
        'address': Object对象
    }
})

handle.setInputAction(function(event) {
    var pick = viewer.scene.pick(event.position);
    if (Cesium.defined(pick) && (pick.id)){
        var ent = pick.id;
        if (ent.properties){    // 设置了properties属性
            if (ent.properties.hasProperty('name')) {         // 拥有name属性
                const name= entity.properties.name.getValue();
                const address= entity.properties.address.getValue();
                // 执行逻辑
            }
        }
    }
},Cesium.ScreenSpaceEventType.LEFT_CLICK)

好处在哪?很简单,你用下划线分割读取一个字符串,你还要知道第几个是什么意思。我直接拿到这个entity的properties,我就知道这个坐标点的其他信息了。

我们通过接口请求过来,循环读取数组来绘制的时候,我们就可以把每一个子项都分别存到对应entity里面。还可以自己控制存哪些。哪种方式优势更大,哪种方式后续的扩展性更好,一看就知道了。

  游戏开发 最新文章
6、英飞凌-AURIX-TC3XX: PWM实验之使用 GT
泛型自动装箱
CubeMax添加Rtthread操作系统 组件STM32F10
python多线程编程:如何优雅地关闭线程
数据类型隐式转换导致的阻塞
WebAPi实现多文件上传,并附带参数
from origin ‘null‘ has been blocked by
UE4 蓝图调用C++函数(附带项目工程)
Unity学习笔记(一)结构体的简单理解与应用
【Memory As a Programming Concept in C a
上一篇文章      下一篇文章      查看所有文章
加:2022-03-15 22:58:26  更:2022-03-15 23:02:30 
 
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁

360图书馆 购物 三丰科技 阅读网 日历 万年历 2025年1日历 -2025/1/16 17:45:18-

图片自动播放器
↓图片自动播放器↓
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
图片批量下载器
↓批量下载图片,美女图库↓
  网站联系: qq:121756557 email:121756557@qq.com  IT数码