1. 概述

DEM(地形文件)天然自带三维信息,可以将其转换成gltf模型文件。DEM是栅格数据,可以通过GDAL进行读取;gltf是一种JSON格式,可以采用nlohmann/json进行读写。

2. 详细

直接把代码贴出来:

#include <iostream>
#include <fstream>
#include <iomanip>
#include <nlohmann\json.hpp>
#include "fifo_map.hpp" #include <gdal/gdal_priv.h> using namespace std;
using namespace nlohmann; // A workaround to give to use fifo_map as map, we are just ignoring the 'less' compare
template<class K, class V, class dummy_compare, class A>
using my_workaround_fifo_map = fifo_map<K, V, fifo_map_compare<K>, A>;
using my_json = basic_json<my_workaround_fifo_map>; int main()
{
GDALAllRegister();
CPLSetConfigOption("GDAL_FILENAME_IS_UTF8", "NO"); //支持中文路径 my_json gltf; gltf["asset"] = {
{"generator", "CL"},
{"version", "2.0"}
}; gltf["scene"] = 0;
gltf["scenes"] = {
{{"nodes", {0} }}
}; gltf["nodes"] = {
{{"mesh", 0}}
}; my_json positionJson;
positionJson["POSITION"] = 1;
positionJson["TEXCOORD_0"] = 2; my_json primitivesJson;
primitivesJson = {
{{"attributes", positionJson}, {"indices", 0}, {"material", 0} }
}; gltf["meshes"] = {
{{"primitives", primitivesJson}}
}; my_json pbrJson;
pbrJson["baseColorTexture"]["index"] = 0; gltf["materials"] = {
{{"pbrMetallicRoughness", pbrJson}}
}; size_t pointNum = 0;
size_t binBufNum = 0;
size_t indicesNum = 0; {
string binPath = "D:/Work/WebGLTutorial/Data/new.bin";
ofstream binFile(binPath, std::ios::binary); const char *filePath = "D:/Work/WebGLTutorial/Data/DEM.tif";
GDALDataset* img = (GDALDataset *)GDALOpen(filePath, GA_ReadOnly);
if (!img)
{
printf("Can't Open Image!");
return 0;
}
int bufWidth = img->GetRasterXSize(); //图像宽度
int bufHeight = img->GetRasterYSize(); //图像高度
int bandNum = img->GetRasterCount(); //波段数
if (bandNum != 1)
{
printf("DEM波段数不为1");
return 0;
}
int depth = GDALGetDataTypeSize(img->GetRasterBand(1)->GetRasterDataType()) / 8; //图像深度 //获取地理坐标信息
double padfTransform[6];
if (img->GetGeoTransform(padfTransform) == CE_Failure)
{
printf("获取仿射变换参数失败");
return 0;
} double startX = padfTransform[0];
double dX = padfTransform[1];
double startY = padfTransform[3];
double dY = padfTransform[5]; //申请buf
size_t imgBufNum = (size_t)bufWidth * bufHeight * bandNum;
float *imgBuf = new float[imgBufNum]; //读取
img->RasterIO(GF_Read, 0, 0, bufWidth, bufHeight, imgBuf, bufWidth, bufHeight,
GDT_Float32, bandNum, nullptr, bandNum*depth, bufWidth*bandNum*depth, depth); pointNum = (size_t)bufWidth * bufHeight;
size_t position_texture_num = pointNum * 5;
float *position_texture = new float[position_texture_num]; for (int yi = 0; yi < bufHeight; yi++)
{
for (int xi = 0; xi < bufWidth; xi++)
{
size_t n = (size_t)(bufWidth * 5) * yi + 5 * xi;
position_texture[n] = dX * xi;
position_texture[n+1] = dY * yi;
size_t m = (size_t)(bufWidth * bandNum) * yi + bandNum * xi;
position_texture[n + 2] = imgBuf[m];
position_texture[n + 3] = float(xi) / (bufWidth-1);
position_texture[n + 4] = float(yi) / (bufHeight-1);
}
} //释放
delete[] imgBuf;
imgBuf = nullptr; binFile.write((char*)position_texture, position_texture_num * sizeof(float)); size_t vertexBufNum = position_texture_num * sizeof(float);
binBufNum = binBufNum + vertexBufNum; int mod = vertexBufNum % sizeof(uint16_t);
if (mod != 0)
{
int spaceNum = sizeof(float) - mod;
char *space = new char[spaceNum];
binBufNum = binBufNum + sizeof(char) * spaceNum;
memset(space, 0, sizeof(char) * spaceNum);
binFile.write(space, sizeof(char) * spaceNum);
delete[] space;
space = nullptr;
} indicesNum = (size_t)(bufWidth - 1) * (bufHeight - 1) * 2 * 3;
uint16_t *indices = new uint16_t[indicesNum]; for (int yi = 0; yi < bufHeight-1; yi++)
{
for (int xi = 0; xi < bufWidth-1; xi++)
{
uint16_t m00 = (uint16_t)(bufWidth * yi + xi) ;
uint16_t m01 = (uint16_t)(bufWidth * (yi+1) + xi);
uint16_t m11 = (uint16_t)(bufWidth * (yi + 1) + xi + 1);
uint16_t m10 = (uint16_t)(bufWidth * yi + xi + 1); size_t n = (size_t)(bufWidth - 1) * yi + xi;
indices[n * 6] = m00;
indices[n * 6 + 1] = m01;
indices[n * 6 + 2] = m11;
indices[n * 6 + 3] = m11;
indices[n * 6 + 4] = m10;
indices[n * 6 + 5] = m00;
}
} binFile.write((char*)indices, sizeof(uint16_t) * indicesNum);
binBufNum = binBufNum + sizeof(uint16_t) * indicesNum; delete[] position_texture;
position_texture = nullptr; delete[] indices;
indices = nullptr;
} gltf["textures"] = {
{{"sampler", 0}, {"source", 0}}
}; gltf["images"] = {
{{"uri", "tex.jpg"}}
}; gltf["samplers"] = {
{{"magFilter", 9729}, {"minFilter", 9987}, {"wrapS", 33648}, {"wrapT", 33648}}
}; gltf["buffers"] = {
{{"uri", "new.bin"}, {"byteLength", binBufNum}}
}; my_json indicesBufferJson;
indicesBufferJson["buffer"] = 0;
indicesBufferJson["byteOffset"] = pointNum * 5 * 4;
indicesBufferJson["byteLength"] = indicesNum * 2;
indicesBufferJson["target"] = 34963; my_json positionBufferJson;
positionBufferJson["buffer"] = 0;
positionBufferJson["byteStride"] = sizeof(float) * 5;
positionBufferJson["byteOffset"] = 0;
positionBufferJson["byteLength"] = pointNum * 5 * 4;
positionBufferJson["target"] = 34962; gltf["bufferViews"] = {
indicesBufferJson, positionBufferJson
}; my_json indicesAccessors;
indicesAccessors["bufferView"] = 0;
indicesAccessors["byteOffset"] = 0;
indicesAccessors["componentType"] = 5123;
indicesAccessors["count"] = indicesNum;
indicesAccessors["type"] = "SCALAR";
indicesAccessors["max"] = { 18719 };
indicesAccessors["min"] = { 0 }; my_json positionAccessors;
positionAccessors["bufferView"] = 1;
positionAccessors["byteOffset"] = 0;
positionAccessors["componentType"] = 5126;
positionAccessors["count"] = pointNum;
positionAccessors["type"] = "VEC3";
positionAccessors["max"] = { 770, 0.0, 1261.151611328125 };
positionAccessors["min"] = { 0.0, -2390, 733.5555419921875 }; my_json textureAccessors;
textureAccessors["bufferView"] = 1;
textureAccessors["byteOffset"] = sizeof(float) * 3;
textureAccessors["componentType"] = 5126;
textureAccessors["count"] = pointNum;
textureAccessors["type"] = "VEC2";
textureAccessors["max"] = { 1, 1 };
textureAccessors["min"] = { 0, 0 }; gltf["accessors"] = {
indicesAccessors, positionAccessors, textureAccessors
}; string jsonFile = "D:/Work/WebGLTutorial/Data/new.gltf";
std::ofstream outFile(jsonFile);
outFile << std::setw(4) << gltf << std::endl;
}

1.这里使用的DEM是tif格式的图像,使用GDAL读取。由于显示模型文件不需要大坐标,所以没有把DEM的起始XY坐标值算进去。同时附带了一张纹理贴图,正好覆盖整个DEM的范围。

2.转换的的原理非常简单,就是将DEM的每个网格绘制成两个三角形,通过顶点索引进行绘制。gltf具体的规范可以参看github上的教程,网上还有相关的中文翻译

3.原生的nlohmann/json组件写出来的JSON格式是根据字符串顺序排序不是根据插入顺序排序的,查阅的时候不方便。所以这里使用了nlohmann::fifo_map容器专门化对象类型。

3. 结果

转换出来的结果用OSG显示如下:

4. 参考

[1] github上的gltf教程

[2] gltf教程中文翻译

[3] nlohmann/json关于保留插入顺序的讨论

DEM转换为gltf的更多相关文章

  1. 3D模型预处理(格式转换:obj转换为gltf)

    在cesium中导入模型需要的是gltf或glb格式的文件,cesium官方提供了obj转gltf文件的工具,一个obj2gltf的库,地址为https://github.com/Analytical ...

  2. WebGL简易教程(十五):加载gltf模型

    目录 1. 概述 2. 实例 2.1. 数据 2.2. 程序 2.2.1. 文件读取 2.2.2. glTF格式解析 2.2.3. 初始化顶点缓冲区 2.2.4. 其他 3. 结果 4. 参考 5. ...

  3. 【Ceisum】Max转GLTF

    参考资料:https://blog.csdn.net/u011394175/article/details/78919281 1.在3DsMax中加入COLLADA插件:COLLADA-MAX-PC_ ...

  4. glTF格式初步了解

    glTF格式初步了解 近期看到Qt 3D的进展.偶然了解到了一种新的格式:glTF格式.这样的格式据说比现有的3D格式更加符合OpenGL应用的须要.这引起了我的好奇.于是我在Qt 3D的外部链接中找 ...

  5. 3D性能优化 | 说一说glTF文件压缩

    引言 最近做T级互动,需要使用到3D模型.相信大家和我一样,在开始着手的时候,一定会有这么些问题: 1.如何选择3D模型的导出格式 2.如何对模型文件进行优化 3.在大流量的项目中兼容性怎么样 让我们 ...

  6. Cesium中导入三维模型方法(dae到glft/bgltf) 【转】

    http://blog.csdn.net/l491453302/article/details/46766909 目录(?)[+] Cesium中目前支持gltf和bgltf两种格式.“gltf是kh ...

  7. cesium js学习一加载三维模型【转】

    http://blog.csdn.net/tangyajun_168/article/details/50936698 最近项目中用到室外三维模型与室内三维地图交互,室外三维模型的加载我们采用了ces ...

  8. Cesium学习笔记(五):3D 模型 (http://blog.csdn.net/umgsoil/article/details/74572877)

    Cesium支持3D模型,包括关键帧动画,皮肤的改变还有单个节点的选择等,Cesium还提供了了一个基于网络的工具,将COLLADA模型转换为glTF,方便和优化模型添加 还记得我们在实体添加的时候添 ...

  9. Cesium中导入三维模型方法(dae到glft/bgltf)[转]

    Cesium中导入三维模型方法(dae到glft/bgltf) Cesium中目前支持gltf和bgltf两种格式.“gltf是khronos组织(起草OpenGL标准的那家)定义的一种交换格式,用于 ...

随机推荐

  1. RabbitMQ入门(三)订阅模式

      在之前的文章RabbitMQ入门(二)工作队列中,我们创建了一个工作队列.工作队列背后的假设是每一项任务都被准确地传送至一个worker.在本文中,我们将会做一些不同的事情--我们将会把一个消息发 ...

  2. 链接拼接的方法(用于解决同一个脚本返回两种不同的url链接的问题)

    实例一: 上图所示 爬虫返回的链接有一部分带有http前缀,有一部分没有,且也不知道具体哪些链接会出现没有前缀的情况 后面如果通过返回链接进行再次访问,那么肯定会出现报错的问题 思路: 判断 返回值内 ...

  3. PowerCat DNS 隧道通信

    powercat 也是一套基于 DNS 通信协议的工具.Powercat的dns的通信是基于dnscat设计的(其服务端就是dnscat).在使用dnscat时需要进行下载和编译. dnscat服务端 ...

  4. AOP编程实践总结

    AOP编程实践总结 AOP概述 AOP(Aspect-Oriented Programming,面向方面编程)是OOP(Object-Oriented Programing,面向对象编程)的补充和完善 ...

  5. markdown常用语法使用笔记+使用技巧(持续更新......)

    参考引用内容: 简书教程 一 基本语法 1. 标题 语法: 在想要设置为标题的文字前面加#来表示,一个#是一级标题,二个#是二级标题,以此类推.支持六级标题. 注:标准语法一般在#后跟个空格再写文字 ...

  6. 【故障公告】再次遭遇SQL语句执行超时引发网站首页访问故障

    非常抱歉,昨天 18:40~19:10 再次遭遇上次遇到的 SQL 语句执行超时引发的网站首页访问故障,由此您带来麻烦,请您谅解. 上次故障详见故障公告,上次排查下来以为是 SQL Server 参数 ...

  7. 深入理解Java虚拟机内存模型

    前言 本文中部分内容引用至<深入理解Java虚拟机:JVM高级特性与最佳实践(第2版)>第12章,如果有兴趣可自行深入阅读,文末放有书籍PDF版本连接. 一.物理机中的并发 物理机遇到的并 ...

  8. C++ 对TXT 的串并行读写

    任务说明:有36篇文档,现在要读入,并统计词频,字典长度25,希望能够比较串并行读写操作的时间差距. 串行读入并统计词频 // LoadDocsInUbuntu.cpp // #include < ...

  9. Java支付宝PC网站支付功能开发(详细教程)

    一.前言 本案例使用的是Java实现的.使用支付宝的沙盒环境示例.发布需要换成正式环境.这里就不作详细说明了 本代码适合用来做参考,不要直接复制去使用. 没有账号的需要去平台注册一个: 登录支付宝开发 ...

  10. content-type常见类型辨析(以ajax与springmvc前后端交互为例)

    博客搬家: content-type常见类型辨析(以ajax与springmvc前后端交互为例) 在http报文的首部中,有一个字段Content-type,表示请求体(entity body)中的数 ...