#include "CCTerrain.h" #include USING_NS_CC; #include "renderer/CCGLProgram.h" #include "renderer/CCGLProgramState.h" #include "renderer/CCRenderer.h" #include "renderer/CCGLProgramStateCache.h" #include "renderer/ccGLStateCache.h" #include "2d/CCCamera.h" #include static const char * vertex_shader = "\ attribute vec4 a_position;\ attribute vec2 a_texCoord;\ attribute vec3 a_normal;\ \n#ifdef GL_ES\n\ varying mediump vec2 v_texCoord;\ varying mediump vec3 v_normal;\ \n#else\n\ varying vec2 v_texCoord;\ varying vec3 v_normal;\ \n#endif\n\ void main()\ {\ gl_Position = CC_MVPMatrix * a_position;\ v_texCoord = a_texCoord;\ v_normal = a_normal;\ }\ "; static const char * fragment_shader_RGB_4_DETAIL ="\n#ifdef GL_ES\n\ precision lowp float;\ \n#endif\n\ uniform vec3 u_color;\ varying vec2 v_texCoord;\ varying vec3 v_normal;\ uniform int u_has_alpha;\ uniform sampler2D u_alphaMap;\ uniform sampler2D u_texture0;\ uniform sampler2D u_texture1;\ uniform sampler2D u_texture2;\ uniform sampler2D u_texture3;\ uniform float u_detailSize[4];\ void main()\ {\ vec3 light_direction = vec3(-1,-1,0);\ float lightFactor = dot(-light_direction,v_normal);\ if(u_has_alpha<=0)\ {\ gl_FragColor = texture2D(u_texture0, v_texCoord)*lightFactor;\ }else\ {\ vec4 blendFactor =texture2D(u_alphaMap,v_texCoord);\ vec4 color = vec4(0,0,0,0);\ color = texture2D(u_texture0, v_texCoord*u_detailSize[0])*blendFactor.r +\ texture2D(u_texture1, v_texCoord*u_detailSize[1])*blendFactor.g + texture2D(u_texture2, v_texCoord*u_detailSize[2])*blendFactor.b;\n\ float grayFactor =dot(blendFactor.rgb, vec3(1, 1, 1));\ color +=texture2D(u_texture3, v_texCoord*u_detailSize[3])*(1.0-grayFactor);\ gl_FragColor = color*lightFactor;\ }\ }"; NS_CC_BEGIN Terrain * Terrain::create(TerrainData ¶meter) { Terrain * terrain = new (std::nothrow)Terrain(); terrain->_terrainData = parameter; terrain->_isCameraViewChanged = true; terrain->_isTerrainModelMatrixChanged = true; //chunksize terrain->_chunkSize = parameter.chunkSize; //heightmap terrain->initHeightMap(parameter.heightMapSrc.c_str()); if(!parameter.alphaMapSrc) { auto textImage = new (std::nothrow)Image(); textImage->initWithImageFile(parameter.detailMaps[0].detailMapSrc); auto texture = new (std::nothrow)Texture2D(); texture->initWithImage(textImage); terrain->textures.push_back(texture); }else { //alpha map auto image = new (std::nothrow)Image(); image->initWithImageFile(parameter.alphaMapSrc); terrain->_alphaMap = new (std::nothrow)Texture2D(); terrain->_alphaMap->initWithImage(image); for(int i =0;i<4;i++) { auto textImage = new (std::nothrow)Image(); textImage->initWithImageFile(parameter.detailMaps[i].detailMapSrc); auto texture = new (std::nothrow)Texture2D(); texture->initWithImage(textImage); terrain->textures.push_back(texture); terrain->detailSize[i] = parameter.detailMaps[i].detailMapSize; } } terrain->init(); terrain->setAnchorPoint(Vec2(0,0)); terrain->autorelease(); return terrain; } bool Terrain::init() { _lodDistance[0]=64; _lodDistance[1]=128; _lodDistance[2]=196; auto shader = GLProgram::createWithByteArrays(vertex_shader,fragment_shader_RGB_4_DETAIL); auto state = GLProgramState::create(shader); setGLProgramState(state); _normalLocation = glGetAttribLocation(this->getGLProgram()->getProgram(),"a_normal"); setDrawWire(false); setIsEnableFrustumCull(true); return true; } void Terrain::draw(cocos2d::Renderer *renderer, const cocos2d::Mat4 &transform, uint32_t flags) { _customCommand.func = CC_CALLBACK_0(Terrain::onDraw, this, transform, flags); renderer->addCommand(&_customCommand); } void Terrain::onDraw(const Mat4 &transform, uint32_t flags) { auto glProgram = getGLProgram(); glProgram->use(); glProgram->setUniformsForBuiltins(transform); glDepthMask(GL_TRUE); GLboolean depthTestCheck; glGetBooleanv(GL_DEPTH_TEST,&depthTestCheck); if(!depthTestCheck) { glEnable(GL_DEPTH_TEST); } GLboolean blendCheck; glGetBooleanv(GL_BLEND,&blendCheck); if(blendCheck) { glDisable(GL_BLEND); } if(!_alphaMap) { glActiveTexture(GL_TEXTURE0); glBindTexture(GL_TEXTURE_2D,textures[0]->getName()); auto texture_location = glGetUniformLocation(glProgram->getProgram(),"u_texture0"); glUniform1i(texture_location,0); auto alpha_location = glGetUniformLocation(glProgram->getProgram(),"u_has_alpha"); glUniform1i(alpha_location,0); }else { for(int i =0;i<_maxDetailMapValue;i++) { glActiveTexture(GL_TEXTURE0+i); glBindTexture(GL_TEXTURE_2D,textures[i]->getName()); glTexParameteri( GL_TEXTURE_2D, GL_TEXTURE_WRAP_S, GL_REPEAT ); glTexParameteri( GL_TEXTURE_2D, GL_TEXTURE_WRAP_T, GL_REPEAT ); glTexParameteri(GL_TEXTURE_2D,GL_TEXTURE_MIN_FILTER,GL_LINEAR); glTexParameteri(GL_TEXTURE_2D,GL_TEXTURE_MAG_FILTER,GL_LINEAR); char str[20]; sprintf(str,"u_texture%d",i); auto texture_location = glGetUniformLocation(glProgram->getProgram(),str); glUniform1i(texture_location,i); sprintf(str,"u_detailSize[%d]",i); auto detailSizeLocation = glGetUniformLocation(glProgram->getProgram(),str); glUniform1f(detailSizeLocation,detailSize[i]); } auto alpha_location = glGetUniformLocation(glProgram->getProgram(),"u_has_alpha"); glUniform1i(alpha_location,1); glActiveTexture(GL_TEXTURE4); glBindTexture(GL_TEXTURE_2D,_alphaMap->getName()); glTexParameteri( GL_TEXTURE_2D, GL_TEXTURE_WRAP_S, GL_REPEAT ); glTexParameteri( GL_TEXTURE_2D, GL_TEXTURE_WRAP_T, GL_REPEAT ); auto alpha_map_location = glGetUniformLocation(glProgram->getProgram(),"u_alphaMap"); glUniform1i(alpha_map_location,4); } auto camera = Camera::getVisitingCamera(); if(memcmp(&_CameraMatrix,&camera->getViewMatrix(),sizeof(Mat4))!=0) { _isCameraViewChanged = true; _CameraMatrix = camera->getViewMatrix(); } auto terrainWorldTransform = getNodeToWorldTransform(); if(memcmp(&_oldTerrrainModelMatrix,&terrainWorldTransform,sizeof(Mat4))!=0) { _isTerrainModelMatrixChanged = true; _oldTerrrainModelMatrix = terrainWorldTransform; } quad->updateAABB(_oldTerrrainModelMatrix); if(_isCameraViewChanged || _isTerrainModelMatrixChanged) { auto camPos = camera->getPosition3D(); //set lod setChunksLOD(camPos); } if(_isCameraViewChanged || _isTerrainModelMatrixChanged) { quad->resetNeedDraw(true);//reset it //camera frustum culling quad->cullByCamera(camera,_oldTerrrainModelMatrix); } quad->draw(); if(_isCameraViewChanged) { _isCameraViewChanged = false; } if(_isTerrainModelMatrixChanged) { _isTerrainModelMatrixChanged = false; } glActiveTexture(GL_TEXTURE0); if(depthTestCheck) { glEnable(GL_DEPTH_TEST); }else { glDisable(GL_DEPTH_TEST); } if(blendCheck) { glEnable(GL_BLEND); } } void Terrain::initHeightMap(const char * heightMap) { _heightMapImage = new Image(); _heightMapImage->initWithImageFile(heightMap); _data = _heightMapImage->getData(); imageWidth =_heightMapImage->getWidth(); imageHeight =_heightMapImage->getHeight(); int chunk_amount_y = imageHeight/_chunkSize.height; int chunk_amount_x = imageWidth/_chunkSize.width; loadVertices(); calculateNormal(); memset(_chunkesArray, 0, sizeof(_chunkesArray)); for(int m =0;m_terrain = this; _chunkesArray[m][n]->_size = _chunkSize; _chunkesArray[m][n]->generate(imageWidth,imageHeight,m,n,_data); } } //calculate the neighbor for(int m =0;m=0) _chunkesArray[m][n]->left = _chunkesArray[m][n-1]; if(n+1right = _chunkesArray[m][n+1]; if(m-1>=0) _chunkesArray[m][n]->back = _chunkesArray[m-1][n]; if(m+1front = _chunkesArray[m+1][n]; } } quad = new QuadTree(0,0,imageWidth,imageHeight,this); } Terrain::Terrain() { _alphaMap = nullptr; } void Terrain::setChunksLOD(Vec3 cameraPos) { int chunk_amount_y = imageHeight/_chunkSize.height; int chunk_amount_x = imageWidth/_chunkSize.width; for(int m=0;m_parent->_worldSpaceAABB; auto center = aabb.getCenter(); float dist = Vec3(center.x,0,center.z).distance(Vec3(cameraPos.x,0,cameraPos.z)); _chunkesArray[m][n]->_currentLod = 3; for(int i =0;i<3;i++) { if(dist<=_lodDistance[i]) { _chunkesArray[m][n]->_currentLod = i; break; } } } } float Terrain::getHeight(float x,float z,Vec3 * normal) { Vec2 pos = Vec2(x,z); //top-left Vec2 tl = Vec2(-1*_terrainData.mapScale*imageWidth/2,-1*_terrainData.mapScale*imageHeight/2); auto result = getNodeToWorldTransform()*Vec4(tl.x,0.0f,tl.y,1.0f); tl = Vec2(result.x,result.z); Vec2 to_tl = pos - tl; //real size Vec2 size = Vec2(imageWidth*_terrainData.mapScale,imageHeight*_terrainData.mapScale); result = getNodeToWorldTransform()*Vec4(size.x,0.0f,size.y,0.0f); size = Vec2(result.x,result.z); float width_ratio = to_tl.x/size.x; float height_ratio = to_tl.y/size.y; float image_x = width_ratio * imageWidth; float image_y = height_ratio * imageHeight; float u =image_x - (int)image_x; float v =image_y - (int)image_y; float i = (int)image_x; float j = (int)image_y; if(image_x>=imageWidth-1 || image_y >=imageHeight-1 || image_x<0 || image_y<0) { return 0; }else { float a = getImageHeight(i,j)*getScaleY(); float b = getImageHeight(i,j+1)*getScaleY(); float c = getImageHeight(i+1,j)*getScaleY(); float d = getImageHeight(i+1,j+1)*getScaleY(); if(normal) { normal->x = c - b; normal->y = 2; normal->z = d - a; normal->normalize(); //(*normal) = (1-u)*(1-v)*getNormal(i,j)+ (1-u)*v*getNormal(i,j+1) + u*(1-v)*getNormal(i+1,j)+ u*v*getNormal(i+1,j+1); } float reuslt = (1-u)*(1-v)*getImageHeight(i,j)*getScaleY() + (1-u)*v*getImageHeight(i,j+1)*getScaleY() + u*(1-v)*getImageHeight(i+1,j)*getScaleY() + u*v*getImageHeight(i+1,j+1)*getScaleY(); return reuslt; } } float Terrain::getHeight(Vec2 pos,Vec3*Normal) { return getHeight(pos.x,pos.y,Normal); } float Terrain::getImageHeight(int pixel_x,int pixel_y) { int byte_stride =1; switch (_heightMapImage->getRenderFormat()) { case Texture2D::PixelFormat::BGRA8888: byte_stride = 4; break; case Texture2D::PixelFormat::RGB888: byte_stride =3; break; case Texture2D::PixelFormat::I8: byte_stride =1; break; default: break; } return _data[(pixel_y*imageWidth+pixel_x)*byte_stride]*1.0/255*_terrainData.mapHeight -0.5*_terrainData.mapHeight; } void Terrain::loadVertices() { m_maxHeight = -999; m_minHeight = 999; for(int i =0;im_maxHeight) m_maxHeight = height; if(height height) { lastRayPosition = rayPos; rayPos += rayStep; height = getHeight(rayPos.x,rayPos.z); } Vec3 startPosition = lastRayPosition; Vec3 endPosition = rayPos; for (int i= 0; i< 32; i++) { // Binary search pass Vec3 middlePoint = (startPosition + endPosition) * 0.5f; if (middlePoint.y < height) endPosition = middlePoint; else startPosition = middlePoint; } Vec3 collisionPoint = (startPosition + endPosition) * 0.5f; return collisionPoint; } void Terrain::setMaxDetailMapAmount(int max_value) { _maxDetailMapValue = max_value; } cocos2d::Vec2 Terrain::convertToTerrainSpace(Vec2 worldSpaceXZ) { Vec2 pos = Vec2(worldSpaceXZ.x,worldSpaceXZ.y); //top-left Vec2 tl = Vec2(-1*_terrainData.mapScale*imageWidth/2,-1*_terrainData.mapScale*imageHeight/2); auto result = getNodeToWorldTransform()*Vec4(tl.x,0.0f,tl.y,1.0f); tl = Vec2(result.x,result.z); Vec2 to_tl = pos - tl; //real size Vec2 size = Vec2(imageWidth*_terrainData.mapScale,imageHeight*_terrainData.mapScale); result = getNodeToWorldTransform()*Vec4(size.x,0.0f,size.y,0.0f); size = Vec2(result.x,result.z); float width_ratio = to_tl.x/size.x; float height_ratio = to_tl.y/size.y; float image_x = width_ratio * imageWidth; float image_y = height_ratio * imageHeight; return Vec2(image_x,image_y); } void Terrain::resetHeightMap(const char * heightMap) { _heightMapImage->release(); vertices.clear(); free(_data); for(int i = 0;i_worldSpaceAABB; } void Terrain::Chunk::finish() { //genearate two VBO ,the first for vertices, we just setup datas once ,won't changed at all //the second vbo for the indices, because we use level of detail technique to each chunk, so we will modified frequently glGenBuffers(2,vbo); //only set for vertices vbo glBindBuffer(GL_ARRAY_BUFFER, vbo[0]); glBufferData(GL_ARRAY_BUFFER, sizeof(TerrainVertexData)*vertices.size(), &vertices[0], GL_STREAM_DRAW); glBindBuffer(GL_ARRAY_BUFFER,0); calculateSlope(); for(int i =0;i<4;i++) { int step = int(powf(2.0f, float(_currentLod))); int indicesAmount =(_terrain->_chunkSize.width/step+1)*(_terrain->_chunkSize.height/step+1)*6+(_terrain->_chunkSize.height/step)*3*2 +(_terrain->_chunkSize.width/step)*3*2; _lod[i].indices.reserve(indicesAmount); } } void Terrain::Chunk::bindAndDraw() { #if (CC_TARGET_PLATFORM == CC_PLATFORM_MAC) || (CC_TARGET_PLATFORM == CC_PLATFORM_WIN32) || (CC_TARGET_PLATFORM == CC_PLATFORM_LINUX) if(_terrain->_isDrawWire) { glPolygonMode(GL_FRONT_AND_BACK,GL_LINE); }else { glPolygonMode(GL_FRONT_AND_BACK,GL_FILL); } #endif glBindBuffer(GL_ARRAY_BUFFER, vbo[0]); if(_terrain->_isCameraViewChanged) { updateVerticesForLOD(); updateIndices(); } glBindBuffer(GL_ELEMENT_ARRAY_BUFFER,vbo[1]); GL::enableVertexAttribs(GL::VERTEX_ATTRIB_FLAG_POSITION | GL::VERTEX_ATTRIB_FLAG_TEX_COORD| GL::VERTEX_ATTRIB_FLAG_NORMAL); unsigned long offset = 0; //position glVertexAttribPointer(GLProgram::VERTEX_ATTRIB_POSITION, 3, GL_FLOAT, GL_FALSE, sizeof(TerrainVertexData), (GLvoid *)offset); offset +=sizeof(Vec3); //texcoord glVertexAttribPointer(GLProgram::VERTEX_ATTRIB_TEX_COORD,2,GL_FLOAT,GL_FALSE,sizeof(TerrainVertexData),(GLvoid *)offset); offset +=sizeof(Tex2F); glEnableVertexAttribArray(_terrain->_normalLocation); //normal glVertexAttribPointer(GLProgram::VERTEX_ATTRIB_NORMAL,3,GL_FLOAT,GL_FALSE,sizeof(TerrainVertexData),(GLvoid *)offset); glDrawElements(GL_TRIANGLES, (GLsizei)_lod[_currentLod].indices.size(), GL_UNSIGNED_SHORT, 0); #if (CC_TARGET_PLATFORM == CC_PLATFORM_MAC) || (CC_TARGET_PLATFORM == CC_PLATFORM_WIN32) || (CC_TARGET_PLATFORM == CC_PLATFORM_LINUX) glPolygonMode(GL_FRONT_AND_BACK,GL_FILL); #endif } void Terrain::Chunk::generate(int imgWidth,int imageHei,int m,int n,const unsigned char * data) { pos_y = m; pos_x = n; for(int i=_size.height*m;i<=_size.height*(m+1);i++) { if(i>=imageHei) break; for(int j=_size.width*n;j<=_size.width*(n+1);j++) { if(j>=imgWidth)break; auto v =_terrain->vertices[i*imgWidth + j]; vertices.push_back (v); } } calculateAABB(); finish(); } Terrain::Chunk::Chunk() { _currentLod = 0; left = nullptr; right = nullptr; back = nullptr; front = nullptr; _oldLod = -1; for(int i =0;i<4;i++) { _neighborOldLOD[i] = -1; } } void Terrain::Chunk::updateIndices() { int currentNeighborLOD[4]; if(left) { currentNeighborLOD[0] = left->_currentLod; }else{currentNeighborLOD[0] = -1;} if(right) { currentNeighborLOD[1] = right->_currentLod; }else{currentNeighborLOD[1] = -1;} if(back) { currentNeighborLOD[2] = back->_currentLod; }else{currentNeighborLOD[2] = -1;} if(front) { currentNeighborLOD[3] = front->_currentLod; }else{currentNeighborLOD[3] = -1;} if(_oldLod == _currentLod &&(memcmp(currentNeighborLOD,_neighborOldLOD,sizeof(currentNeighborLOD))==0) ) { return;// no need to update } memcpy(_neighborOldLOD,currentNeighborLOD,sizeof(currentNeighborLOD)); _oldLod = _currentLod; int gridY = _size.height; int gridX = _size.width; int step = int(powf(2.0f, float(_currentLod))); if((left&&left->_currentLod > _currentLod) ||(right&&right->_currentLod > _currentLod) ||(back&&back->_currentLod > _currentLod) || (front && front->_currentLod > _currentLod)) //need update indices. { //t-junction inner _lod[_currentLod].indices.clear(); for(int i =step;i_currentLod > _currentLod)//left { for(int i =0;i_currentLod > _currentLod) end -=step; if(back&&back->_currentLod > _currentLod) start +=step; for(int i =start;i_currentLod > _currentLod)//LEFT { for(int i =0;i_currentLod > _currentLod) end -=step; if(back&&back->_currentLod > _currentLod) start +=step; for(int i =start;i_currentLod > _currentLod)//front { for(int i =0;i_currentLod > _currentLod)//back { for(int i =0;ipos; for(int i =0;i highest.y) { highest = vertices[i].position; } } auto a = Vec2(lowest.x,lowest.z); auto b = Vec2(highest.x,highest.z); float dist = a.distance(b); slope = (highest.y - lowest.y)/dist; } void Terrain::Chunk::updateVerticesForLOD() { if(_oldLod == _currentLod){ return;} // no need to update vertices vertices_tmp = vertices; int gridY = _size.height; int gridX = _size.width; if(_currentLod>=2 && abs(slope)>1.2) { int step = int(powf(2.0f, float(_currentLod))); for(int i =step;iheight = h; this->width = w; if(width> terrain->_chunkSize.width &&height >terrain->_chunkSize.height) //subdivision { _isTerminal = false; this->tl = new QuadTree(x,y,width/2,height/2,terrain); this->tl->parent = this; this->tr = new QuadTree(x+width/2,y,width/2,height/2,terrain); this->tr->parent = this; this->bl = new QuadTree(x,y+height/2,width/2,height/2,terrain); this->bl->parent = this; this->br = new QuadTree(x+width/2,y+height/2,width/2,height/2,terrain); this->br->parent = this; _aabb.merge(tl->_aabb); _aabb.merge(tr->_aabb); _aabb.merge(bl->_aabb); _aabb.merge(br->_aabb); }else // is terminal Node { int m = pos_y/terrain->_chunkSize.height; int n = pos_x/terrain->_chunkSize.width; _chunk = terrain->_chunkesArray[m][n]; _isTerminal = true; _aabb = _chunk->_aabb; _chunk->_parent = this; } _worldSpaceAABB = _aabb; _worldSpaceAABB.transform(_terrain->getNodeToWorldTransform()); } void Terrain::QuadTree::draw() { if(!_needDraw)return; if(_isTerminal){ this->_chunk->bindAndDraw(); }else { this->tl->draw(); this->tr->draw(); this->br->draw(); this->bl->draw(); } } void Terrain::QuadTree::resetNeedDraw(bool value) { this->_needDraw = value; if(!_isTerminal) { tl->resetNeedDraw(value); tr->resetNeedDraw(value); bl->resetNeedDraw(value); br->resetNeedDraw(value); } } void Terrain::QuadTree::cullByCamera(const Camera * camera,const Mat4 & worldTransform) { if(_terrain->_isTerrainModelMatrixChanged) { _worldSpaceAABB = _aabb; _worldSpaceAABB.transform(worldTransform); } if(!camera->isVisibleInFrustum(&_worldSpaceAABB)) { this->resetNeedDraw(false); }else { if(!_isTerminal){ tl->cullByCamera(camera,worldTransform); tr->cullByCamera(camera,worldTransform); bl->cullByCamera(camera,worldTransform); br->cullByCamera(camera,worldTransform); } } } void Terrain::QuadTree::updateAABB(const Mat4 & worldTransform) { if(_terrain->_isTerrainModelMatrixChanged) { _worldSpaceAABB = _aabb; _worldSpaceAABB.transform(worldTransform); } if(!_isTerminal){ tl->updateAABB(worldTransform); tr->updateAABB(worldTransform); bl->updateAABB(worldTransform); br->updateAABB(worldTransform); } } Terrain::TerrainData::TerrainData(const char * heightMapsrc ,const char * textureSrc,const Size & chunksize,float height,float scale) { this->heightMapSrc = heightMapsrc; this->detailMaps[0].detailMapSrc = textureSrc; this->alphaMapSrc = nullptr; this->chunkSize = chunksize; this->mapHeight = height; this->mapScale = scale; } Terrain::TerrainData::TerrainData(const char * heightMapsrc ,const char * alphamap,const DetailMap& detail1,const DetailMap& detail2,const DetailMap& detail3,const DetailMap& detail4,const Size & chunksize,float height,float scale) { this->heightMapSrc = heightMapsrc; this->alphaMapSrc = const_cast(alphamap); this->detailMaps[0] = detail1; this->detailMaps[1] = detail2; this->detailMaps[2] = detail3; this->detailMaps[3] = detail4; this->chunkSize = chunksize; this->mapHeight = height; this->mapScale = scale; _detailMapAmount = 4; } Terrain::TerrainData::TerrainData(const char* heightMapsrc ,const char * alphamap,const DetailMap& detail1,const DetailMap& detail2,const DetailMap& detail3,const Size & chunksize /*= Size(32,32)*/,float height /*= 2*/,float scale /*= 0.1*/) { this->heightMapSrc = heightMapsrc; this->alphaMapSrc = const_cast(alphamap); this->detailMaps[0] = detail1; this->detailMaps[1] = detail2; this->detailMaps[2] = detail3; this->detailMaps[3] = nullptr; this->chunkSize = chunksize; this->mapHeight = height; this->mapScale = scale; _detailMapAmount = 3; } Terrain::TerrainData::TerrainData() { } Terrain::DetailMap::DetailMap(const char * detailMapPath , float size /*= 35*/) { this->detailMapSrc = detailMapPath; this->detailMapSize = size; } Terrain::DetailMap::DetailMap() { detailMapSrc = ""; detailMapSize = 35; } NS_CC_END