2015-01-15 15:54:12 +08:00
|
|
|
/****************************************************************************
|
|
|
|
Copyright (c) 2014 Chukong Technologies Inc.
|
|
|
|
|
|
|
|
http://www.cocos2d-x.org
|
|
|
|
|
|
|
|
Permission is hereby granted, free of charge, to any person obtaining a copy
|
|
|
|
of this software and associated documentation files (the "Software"), to deal
|
|
|
|
in the Software without restriction, including without limitation the rights
|
|
|
|
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
|
|
|
copies of the Software, and to permit persons to whom the Software is
|
|
|
|
furnished to do so, subject to the following conditions:
|
|
|
|
|
|
|
|
The above copyright notice and this permission notice shall be included in
|
|
|
|
all copies or substantial portions of the Software.
|
|
|
|
|
|
|
|
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
|
|
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
|
|
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
|
|
|
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|
|
|
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
|
|
|
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
|
|
|
|
THE SOFTWARE.
|
|
|
|
****************************************************************************/
|
2015-01-15 11:45:06 +08:00
|
|
|
#ifndef CC_TERRAIN_H
|
|
|
|
#define CC_TERRAIN_H
|
2015-01-15 15:54:12 +08:00
|
|
|
|
2015-01-15 11:45:06 +08:00
|
|
|
#include "2d/CCNode.h"
|
|
|
|
#include "renderer/CCTexture2D.h"
|
|
|
|
#include "renderer/CCCustomCommand.h"
|
|
|
|
#include "3d/CCAABB.h"
|
|
|
|
#include "2d/CCCamera.h"
|
|
|
|
#include <vector>
|
|
|
|
NS_CC_BEGIN
|
2015-01-15 15:54:12 +08:00
|
|
|
/*
|
|
|
|
*the maximum amount of the chunkes
|
|
|
|
**/
|
2015-01-15 11:45:06 +08:00
|
|
|
#define MAX_CHUNKES 256
|
|
|
|
|
|
|
|
/*
|
2015-01-15 15:54:12 +08:00
|
|
|
*Terrain
|
|
|
|
*use to render outdoor or large scene via heightMap
|
2015-01-15 11:45:06 +08:00
|
|
|
**/
|
|
|
|
class CC_DLL Terrain :public Node{
|
|
|
|
public:
|
2015-01-15 15:54:12 +08:00
|
|
|
|
|
|
|
/*
|
|
|
|
*DetailMap
|
|
|
|
*this struct maintain a detail map data ,including source file ,detail size.
|
|
|
|
*the DetailMap can use for terrain splatting
|
|
|
|
**/
|
2015-01-15 11:45:06 +08:00
|
|
|
struct CC_DLL DetailMap{
|
2015-01-15 15:54:12 +08:00
|
|
|
/*Constructors*/
|
2015-01-15 11:45:06 +08:00
|
|
|
DetailMap();
|
|
|
|
DetailMap(const char * detailMapSrc , float size = 35);
|
2015-01-15 15:54:12 +08:00
|
|
|
/*detail Image source file path*/
|
2015-01-15 11:45:06 +08:00
|
|
|
std::string detailMapSrc;
|
2015-01-15 15:54:12 +08:00
|
|
|
/*detailMapSize determine how many tiles that Terrain represent*/
|
2015-01-15 11:45:06 +08:00
|
|
|
float detailMapSize;
|
|
|
|
};
|
2015-01-15 15:54:12 +08:00
|
|
|
|
|
|
|
/*
|
|
|
|
*TerrainData
|
|
|
|
*This TerrainData struct warp all parameter that Terrain need to create
|
|
|
|
*/
|
2015-01-15 11:45:06 +08:00
|
|
|
struct CC_DLL TerrainData
|
|
|
|
{
|
2015-01-15 15:54:12 +08:00
|
|
|
/*Constructors*/
|
2015-01-15 11:45:06 +08:00
|
|
|
TerrainData();
|
|
|
|
TerrainData(const char* heightMapsrc ,const char * textureSrc,const Size & chunksize = Size(32,32),float mapHeight = 2,float mapScale = 0.1);
|
|
|
|
TerrainData(const char* heightMapsrc ,const char * alphamap,const DetailMap& detail1,const DetailMap& detail2,const DetailMap& detail3,const DetailMap& detail4,const Size & chunksize = Size(32,32),float mapHeight = 2,float mapScale = 0.1);
|
2015-01-15 15:54:12 +08:00
|
|
|
|
|
|
|
/*
|
|
|
|
*deterimine the chunk size,chunk is the minimal subdivision of the Terrain
|
|
|
|
*/
|
2015-01-15 11:45:06 +08:00
|
|
|
Size chunkSize;
|
2015-01-15 15:54:12 +08:00
|
|
|
/*height Map source path*/
|
2015-01-15 11:45:06 +08:00
|
|
|
std::string heightMapSrc;
|
2015-01-15 15:54:12 +08:00
|
|
|
/*the source path of the alpha map*/
|
2015-01-15 11:45:06 +08:00
|
|
|
char* alphaMapSrc;
|
2015-01-15 15:54:12 +08:00
|
|
|
/*detail maps*/
|
2015-01-15 11:45:06 +08:00
|
|
|
DetailMap detailMaps[4];
|
2015-01-15 15:54:12 +08:00
|
|
|
/*terrain Maximum height*/
|
2015-01-15 11:45:06 +08:00
|
|
|
float mapHeight;
|
2015-01-15 15:54:12 +08:00
|
|
|
/*terrain scale factor*/
|
2015-01-15 11:45:06 +08:00
|
|
|
float mapScale;
|
|
|
|
};
|
|
|
|
private:
|
2015-01-15 15:54:12 +08:00
|
|
|
|
|
|
|
/*
|
|
|
|
*terrain vertices internal data format
|
|
|
|
**/
|
2015-01-15 11:45:06 +08:00
|
|
|
struct TerrainVertexData
|
|
|
|
{
|
2015-01-15 15:54:12 +08:00
|
|
|
/*constructor*/
|
2015-01-15 11:45:06 +08:00
|
|
|
TerrainVertexData(){};
|
|
|
|
TerrainVertexData(Vec3 v1 ,Tex2F v2)
|
|
|
|
{
|
|
|
|
position = v1;
|
|
|
|
texcoord = v2;
|
|
|
|
};
|
2015-01-15 15:54:12 +08:00
|
|
|
/*the vertex's attributes*/
|
2015-01-15 11:45:06 +08:00
|
|
|
cocos2d::Vec3 position;
|
|
|
|
cocos2d::Tex2F texcoord;
|
|
|
|
cocos2d::Vec3 normal;
|
|
|
|
};
|
2015-01-15 15:54:12 +08:00
|
|
|
|
|
|
|
/*
|
|
|
|
*the terminal node of quad, use to subdivision terrain mesh and LOD
|
|
|
|
**/
|
2015-01-15 11:45:06 +08:00
|
|
|
struct Chunk
|
|
|
|
{
|
2015-01-15 15:54:12 +08:00
|
|
|
/*Constructor*/
|
2015-01-15 11:45:06 +08:00
|
|
|
Chunk();
|
2015-01-15 15:54:12 +08:00
|
|
|
/*vertices*/
|
2015-01-15 11:45:06 +08:00
|
|
|
std::vector<TerrainVertexData> vertices;
|
2015-01-15 15:54:12 +08:00
|
|
|
/*LOD indices*/
|
2015-01-15 11:45:06 +08:00
|
|
|
struct LOD{
|
2015-01-15 15:54:12 +08:00
|
|
|
std::vector<GLushort> indices;
|
2015-01-15 11:45:06 +08:00
|
|
|
};
|
|
|
|
GLuint vbo[2];
|
2015-01-15 15:54:12 +08:00
|
|
|
/*we now have four levels of detail*/
|
2015-01-15 11:45:06 +08:00
|
|
|
LOD _lod[4];
|
2015-01-15 15:54:12 +08:00
|
|
|
/*AABB in local space*/
|
2015-01-15 11:45:06 +08:00
|
|
|
AABB _aabb;
|
2015-01-15 15:54:12 +08:00
|
|
|
/*setup Chunk data*/
|
2015-01-15 11:45:06 +08:00
|
|
|
void generate(int map_width,int map_height,int m,int n,const unsigned char * data);
|
2015-01-15 15:54:12 +08:00
|
|
|
/*calculateAABB*/
|
2015-01-15 11:45:06 +08:00
|
|
|
void calculateAABB();
|
2015-01-15 15:54:12 +08:00
|
|
|
/*internal use draw function*/
|
2015-01-15 11:45:06 +08:00
|
|
|
void bindAndDraw();
|
2015-01-15 15:54:12 +08:00
|
|
|
/*finish opengl setup*/
|
2015-01-15 11:45:06 +08:00
|
|
|
void finish();
|
2015-01-15 15:54:12 +08:00
|
|
|
/*use linear-sample vertices for LOD mesh*/
|
2015-01-15 11:45:06 +08:00
|
|
|
void updateVerticesForLOD();
|
2015-01-15 15:54:12 +08:00
|
|
|
/*calculate the average slope of the chunk*/
|
2015-01-15 11:45:06 +08:00
|
|
|
void calculateSlope();
|
2015-01-15 15:54:12 +08:00
|
|
|
/*updateIndices for every frame*/
|
2015-01-15 11:45:06 +08:00
|
|
|
void updateIndices();
|
2015-01-15 15:54:12 +08:00
|
|
|
|
|
|
|
/*current LOD of the chunk*/
|
|
|
|
int _currentLod;
|
|
|
|
/*the left,right,front,back neighbors*/
|
2015-01-15 11:45:06 +08:00
|
|
|
Chunk * left;
|
|
|
|
Chunk * right;
|
|
|
|
Chunk * front;
|
|
|
|
Chunk * back;
|
2015-01-15 15:54:12 +08:00
|
|
|
|
|
|
|
//the position
|
2015-01-15 11:45:06 +08:00
|
|
|
int pos_x;
|
|
|
|
int pos_y;
|
2015-01-15 15:54:12 +08:00
|
|
|
/*parent terrain*/
|
2015-01-15 11:45:06 +08:00
|
|
|
Terrain * _terrain;
|
2015-01-15 15:54:12 +08:00
|
|
|
/*chunk size*/
|
2015-01-15 11:45:06 +08:00
|
|
|
Size _size;
|
2015-01-15 15:54:12 +08:00
|
|
|
/*chunk's estimated slope*/
|
2015-01-15 11:45:06 +08:00
|
|
|
float slope;
|
2015-01-15 15:54:12 +08:00
|
|
|
std::vector<TerrainVertexData> vertices_tmp;
|
2015-01-15 11:45:06 +08:00
|
|
|
};
|
2015-01-15 15:54:12 +08:00
|
|
|
|
|
|
|
/*
|
|
|
|
*QuadTree
|
|
|
|
*use to frustum culling and set LOD
|
|
|
|
**/
|
2015-01-15 11:45:06 +08:00
|
|
|
struct QuadTree
|
|
|
|
{
|
|
|
|
QuadTree(int x,int y,int width,int height,Terrain * terrain);
|
|
|
|
void draw();
|
|
|
|
void resetNeedDraw(bool value);
|
|
|
|
void cullByCamera(const Camera * camera,const Mat4 & worldTransform);
|
|
|
|
QuadTree * tl;
|
|
|
|
QuadTree * tr;
|
|
|
|
QuadTree * bl;
|
|
|
|
QuadTree * br;
|
|
|
|
bool _isTerminal;
|
|
|
|
Chunk * _chunk;
|
|
|
|
int pos_x;
|
|
|
|
int pos_y;
|
|
|
|
int height;
|
|
|
|
int width;
|
|
|
|
QuadTree * parent;
|
|
|
|
AABB _aabb;
|
|
|
|
bool _needDraw;
|
|
|
|
};
|
2015-01-15 15:54:12 +08:00
|
|
|
|
|
|
|
|
2015-01-15 11:45:06 +08:00
|
|
|
friend QuadTree;
|
|
|
|
friend Chunk;
|
|
|
|
public:
|
2015-01-15 15:54:12 +08:00
|
|
|
/*init function*/
|
2015-01-15 11:45:06 +08:00
|
|
|
bool init();
|
|
|
|
void initHeightMap(const char* heightMap);
|
2015-01-15 15:54:12 +08:00
|
|
|
/*create entry*/
|
2015-01-15 11:45:06 +08:00
|
|
|
static Terrain * create(TerrainData ¶meter);
|
2015-01-15 15:54:12 +08:00
|
|
|
/*get specified position's height mapping to the terrain*/
|
2015-01-15 11:45:06 +08:00
|
|
|
float getHeight(float x,float y, float z);
|
|
|
|
float getHeight(Vec3 pos);
|
2015-01-15 15:54:12 +08:00
|
|
|
|
|
|
|
/*get height from the raw height map*/
|
2015-01-15 11:45:06 +08:00
|
|
|
float getImageHeight(int pixel_x,int pixel_y);
|
2015-01-15 15:54:12 +08:00
|
|
|
/*Debug Use only, show the wireline instead of the surface. only support desktop platform*/
|
2015-01-15 11:45:06 +08:00
|
|
|
void setDrawWire(bool bool_value);
|
2015-01-15 15:54:12 +08:00
|
|
|
/*Set threshold distance of each LOD level,must equal or gereater than the chunk size*/
|
2015-01-15 11:45:06 +08:00
|
|
|
void setLODDistance(float lod_1,float lod_2,float lod_3);
|
2015-01-15 15:54:12 +08:00
|
|
|
/*Switch frustumCulling Flag*/
|
2015-01-15 11:45:06 +08:00
|
|
|
void setIsEnableFrustumCull(bool bool_value);
|
2015-01-15 15:54:12 +08:00
|
|
|
// Overrides, internal use only
|
|
|
|
virtual void draw(cocos2d::Renderer* renderer, const cocos2d::Mat4 &transform, uint32_t flags) override;
|
2015-01-15 11:45:06 +08:00
|
|
|
private:
|
|
|
|
Terrain();
|
|
|
|
void onDraw(const Mat4 &transform, uint32_t flags);
|
2015-01-15 15:54:12 +08:00
|
|
|
//set each chunk's LOD
|
2015-01-15 11:45:06 +08:00
|
|
|
void setChunksLOD(Vec3 cameraPos);
|
2015-01-15 15:54:12 +08:00
|
|
|
//load vertices for whole height map
|
2015-01-15 11:45:06 +08:00
|
|
|
void loadVertices();
|
2015-01-15 15:54:12 +08:00
|
|
|
//calculate Normal Line for each Vertex
|
2015-01-15 11:45:06 +08:00
|
|
|
void calculateNormal();
|
|
|
|
private:
|
|
|
|
TerrainData _terrainData;
|
|
|
|
bool _isDrawWire;
|
|
|
|
unsigned char * _data;
|
|
|
|
float _lodDistance[3];
|
|
|
|
std::vector<Texture2D *>textures;
|
|
|
|
Texture2D * _alphaMap;
|
|
|
|
CustomCommand _customCommand;
|
|
|
|
GLuint vbo[2];
|
|
|
|
QuadTree * quad;
|
|
|
|
int detailSize[4];
|
|
|
|
Chunk * _chunkesArray[256][256];
|
|
|
|
std::vector<TerrainVertexData> vertices;
|
|
|
|
std::vector<GLushort > indices;
|
|
|
|
int imageWidth;
|
|
|
|
int imageHeight;
|
|
|
|
Size _chunkSize;
|
|
|
|
bool _isEnableFrustumCull;
|
|
|
|
};
|
|
|
|
NS_CC_END
|
|
|
|
#endif
|