advwater.cpp

来自「这是整套横扫千军3D版游戏的源码」· C++ 代码 · 共 367 行

CPP
367
字号
// DrawWater.cpp: implementation of the CAdvWater class.
//
//////////////////////////////////////////////////////////////////////

#include "StdAfx.h"
#include "AdvWater.h"
#include "Game/Game.h"
#include "Rendering/GL/myGL.h"
#include "Game/Camera.h"
#include "Rendering/GL/VertexArray.h"
#include "Map/ReadMap.h"
#include "LogOutput.h"
#include "Map/BaseGroundDrawer.h"
#include "BaseSky.h"
#include "Rendering/UnitModels/UnitDrawer.h"
#include "Sim/Projectiles/ProjectileHandler.h"
#include "Sim/Misc/FeatureHandler.h"
#include "Lua/LuaCallInHandler.h"

#include "mmgr.h"

//////////////////////////////////////////////////////////////////////
// Construction/Destruction
//////////////////////////////////////////////////////////////////////

extern GLfloat FogLand[]; 

CAdvWater::CAdvWater(bool loadShader)
{
	glGenTextures(1, &reflectTexture);
	unsigned char* scrap=SAFE_NEW unsigned char[512*512*4];

	glBindTexture(GL_TEXTURE_2D, reflectTexture);
	glTexParameteri(GL_TEXTURE_2D,GL_TEXTURE_MAG_FILTER,GL_LINEAR);
	glTexParameteri(GL_TEXTURE_2D,GL_TEXTURE_MIN_FILTER,GL_LINEAR);
	glTexParameteri(GL_TEXTURE_2D,GL_TEXTURE_WRAP_S,GL_CLAMP_TO_EDGE);
	glTexParameteri(GL_TEXTURE_2D,GL_TEXTURE_WRAP_T,GL_CLAMP_TO_EDGE);
	glTexImage2D(GL_TEXTURE_2D,0,GL_RGBA8 ,512, 512, 0,GL_RGBA, GL_UNSIGNED_BYTE, scrap);

	glGenTextures(1, &bumpTexture);
	glBindTexture(GL_TEXTURE_2D, bumpTexture);
	glTexParameteri(GL_TEXTURE_2D,GL_TEXTURE_MAG_FILTER,GL_LINEAR);
	glTexParameteri(GL_TEXTURE_2D,GL_TEXTURE_MIN_FILTER,GL_LINEAR);
	glTexImage2D(GL_TEXTURE_2D,0,GL_RGBA8 ,128, 128, 0,GL_RGBA, GL_UNSIGNED_BYTE, scrap);

	glGenTextures(4, rawBumpTexture);

	for(int y=0;y<64;++y){
		for(int x=0;x<64;++x){
			scrap[(y*64+x)*4+0]=128;
			scrap[(y*64+x)*4+1]=(unsigned char)(sin(y*PI*2.0f/64.0f)*128+128);
			scrap[(y*64+x)*4+2]=0;
			scrap[(y*64+x)*4+3]=255;
		}
	}
	glBindTexture(GL_TEXTURE_2D, rawBumpTexture[0]);
	glTexParameteri(GL_TEXTURE_2D,GL_TEXTURE_MAG_FILTER,GL_LINEAR);
	glTexParameteri(GL_TEXTURE_2D,GL_TEXTURE_MIN_FILTER,GL_LINEAR);
	glTexImage2D(GL_TEXTURE_2D,0,GL_RGBA8 ,64, 64, 0,GL_RGBA, GL_UNSIGNED_BYTE, scrap);
/*
	for(int y=0;y<64;++y){
		for(int x=0;x<64;++x){
			scrap[(y*64+x)*4+0]=(sin(x*PI*2.0f/64.0f))*128+128;
			scrap[(y*64+x)*4+1]=128;
			scrap[(y*64+x)*4+2]=0;
			scrap[(y*64+x)*4+3]=255;
		}
	}
	glBindTexture(GL_TEXTURE_2D, rawBumpTexture[0]);
	glTexParameteri(GL_TEXTURE_2D,GL_TEXTURE_MAG_FILTER,GL_LINEAR);
	glTexParameteri(GL_TEXTURE_2D,GL_TEXTURE_MIN_FILTER,GL_LINEAR);
	glTexImage2D(GL_TEXTURE_2D,0,4 ,64, 64, 0,GL_RGBA, GL_UNSIGNED_BYTE, scrap);

	/*/
	for(int y=0;y<64;++y){
		for(int x=0;x<64;++x){
			float ang=26.5f*PI/180.0f;
			float pos=y*2+x;
			scrap[(y*64+x)*4+0]=(unsigned char)((sin(pos*PI*2.0f/64.0f))*128*sin(ang))+128;
			scrap[(y*64+x)*4+1]=(unsigned char)((sin(pos*PI*2.0f/64.0f))*128*cos(ang))+128;
		}
	}
	glBindTexture(GL_TEXTURE_2D, rawBumpTexture[1]);
	glTexParameteri(GL_TEXTURE_2D,GL_TEXTURE_MAG_FILTER,GL_LINEAR);
	glTexParameteri(GL_TEXTURE_2D,GL_TEXTURE_MIN_FILTER,GL_LINEAR);
	glTexImage2D(GL_TEXTURE_2D,0,GL_RGBA8 ,64, 64, 0,GL_RGBA, GL_UNSIGNED_BYTE, scrap);

	for(int y=0;y<64;++y){
		for(int x=0;x<64;++x){
			float ang=-19*PI/180.0f;
			float pos=3*y-x;
			scrap[(y*64+x)*4+0]=(unsigned char)((sin(pos*PI*2.0f/64.0f))*128*sin(ang))+128;
			scrap[(y*64+x)*4+1]=(unsigned char)((sin(pos*PI*2.0f/64.0f))*128*cos(ang))+128;
		}
	}
	glBindTexture(GL_TEXTURE_2D, rawBumpTexture[2]);
	glTexParameteri(GL_TEXTURE_2D,GL_TEXTURE_MAG_FILTER,GL_LINEAR);
	glTexParameteri(GL_TEXTURE_2D,GL_TEXTURE_MIN_FILTER,GL_LINEAR);
	glTexImage2D(GL_TEXTURE_2D,0,GL_RGBA8 ,64, 64, 0,GL_RGBA, GL_UNSIGNED_BYTE, scrap);
/**/
	delete[] scrap;

	if (loadShader)
		waterFP=LoadFragmentProgram("water.fp");

	waterSurfaceColor = readmap->waterSurfaceColor;
}

CAdvWater::~CAdvWater()
{
	glDeleteTextures (1, &reflectTexture);
	glDeleteTextures (1, &bumpTexture);
	glDeleteTextures (4, rawBumpTexture);
	glSafeDeleteProgram( waterFP );
}

void CAdvWater::Draw()
{
	Draw(true);
}

void CAdvWater::Draw(bool useBlending)
{
	if(readmap->minheight>10)
		return;
	float3 dir,zpos;
	float3 base=camera->CalcPixelDir(gu->viewPosX,gu->viewSizeY);
	float3 dv=camera->CalcPixelDir(gu->viewPosX,0)-camera->CalcPixelDir(gu->viewPosX,gu->viewSizeY);
	float3 dh=camera->CalcPixelDir(gu->viewPosX+gu->viewSizeX,0)-camera->CalcPixelDir(gu->viewPosX,0);

	float3 xbase;
	const int numDivs=20;

	base*=numDivs;
	float maxY=-0.1f;
	float yInc=1.0f/numDivs;
	float screenY=1;

	unsigned char col[4];
	col[0]=(unsigned char)(waterSurfaceColor.x*255);
	col[1]=(unsigned char)(waterSurfaceColor.y*255);
	col[2]=(unsigned char)(waterSurfaceColor.z*255);

	glDisable(GL_ALPHA_TEST);
	if (useBlending) {
		glEnable(GL_BLEND);
	} else {
		glDisable(GL_BLEND);
	}
	glDepthMask(0);
	glBindTexture(GL_TEXTURE_2D, reflectTexture);
	glActiveTextureARB(GL_TEXTURE1_ARB);
		glBindTexture(GL_TEXTURE_2D, bumpTexture);
		GLfloat plan[]={0.02f,0,0,0};
		glTexGeni(GL_S,GL_TEXTURE_GEN_MODE,GL_EYE_LINEAR);
		glTexGenfv(GL_S,GL_EYE_PLANE,plan);
		glEnable(GL_TEXTURE_GEN_S);
			
		GLfloat plan2[]={0,0,0.02f,0};
		glTexGeni(GL_T,GL_TEXTURE_GEN_MODE,GL_EYE_LINEAR);
		glTexGenfv(GL_T,GL_EYE_PLANE,plan2);
		glEnable(GL_TEXTURE_GEN_T);
	glActiveTextureARB(GL_TEXTURE0_ARB);

	glBindProgramARB( GL_FRAGMENT_PROGRAM_ARB, waterFP );
	glEnable( GL_FRAGMENT_PROGRAM_ARB );

	float3 forward=camera->forward;
	forward.y=0;
	forward.Normalize();

	glProgramEnvParameter4fARB(GL_FRAGMENT_PROGRAM_ARB,0, forward.z,forward.x,0,0);
	glProgramEnvParameter4fARB(GL_FRAGMENT_PROGRAM_ARB,1, -forward.x,forward.z,0,0);

	CVertexArray* va=GetVertexArray();
	va->Initialize();
	for(int a=0;a<5;++a){
		bool maxReached=false;
		for(int y=0;y<numDivs;++y){
			dir=base;
			dir.Normalize();

			if(dir.y>=maxY){
				maxReached=true;
				break;
			}

			xbase=base;
			for(int x=0;x<numDivs+1;++x){
				dir=xbase+dv;
				dir.Normalize();
				zpos=camera->pos+dir*(camera->pos.y/-dir.y);
				zpos.y=sin(zpos.z*0.1f+gs->frameNum*0.06f)*0.06f+0.05f;
				col[3]=(unsigned char)((0.8f+0.7f*(dir.y))*255);
				va->AddVertexTC(zpos,x*(1.0f/numDivs),screenY-yInc,col);

				dir=xbase;
				dir.Normalize();
				zpos=camera->pos+dir*(camera->pos.y/-dir.y);
				zpos.y=sin(zpos.z*0.1f+gs->frameNum*0.06f)*0.06f+0.05f;
				col[3]=(unsigned char)((0.8f+0.7f*(dir.y))*255);
				va->AddVertexTC(zpos,x*(1.0f/numDivs),screenY,col);

				xbase+=dh;
			}
			va->EndStrip();
			base+=dv;
			screenY-=yInc;
		}
		if(!maxReached)
			break;
		dv*=0.5f;
		maxY*=0.5f;
		yInc*=0.5f;
	}
	va->DrawArrayTC(GL_TRIANGLE_STRIP);

	glDepthMask(1);
	glDisable( GL_FRAGMENT_PROGRAM_ARB );
	glActiveTextureARB(GL_TEXTURE1_ARB);
		glDisable(GL_TEXTURE_GEN_S);
		glDisable(GL_TEXTURE_GEN_T);
	glActiveTextureARB(GL_TEXTURE0_ARB);

	if(!useBlending) // for translucent stuff like water, the default mode is blending and alpha testing enabled
		glEnable(GL_BLEND);
}

void CAdvWater::UpdateWater(CGame* game)
{
	if (readmap->minheight > 10 || readmap->voidWater)
		return;

	glViewport(0,0,128,128);

	glClearColor(0.0f,0.0f,0.0f,1);
	glClear(GL_COLOR_BUFFER_BIT | GL_DEPTH_BUFFER_BIT);
	glColorMask(1,1,1,0);

	glMatrixMode(GL_PROJECTION);
	glLoadIdentity();
	glOrtho(0,1,0,1,-1,1);
	glMatrixMode(GL_MODELVIEW);
	glLoadIdentity();
	glEnable(GL_TEXTURE_2D);
	glBindTexture(GL_TEXTURE_2D, rawBumpTexture[0]);
	glEnable(GL_BLEND);
	glBlendFunc(GL_ONE,GL_ONE);

	glColor3f(1,1,1);
	glBegin(GL_QUADS);
/*	glColor3f(0.49f,0.49f,0.49f);
	glTexCoord2f(0,0+gs->frameNum*0.01f);glVertex3f(0,0,0);
	glTexCoord2f(0,2+gs->frameNum*0.01f);glVertex3f(0,1,0);
	glTexCoord2f(1,2+gs->frameNum*0.01f);glVertex3f(1,1,0);
	glTexCoord2f(1,0+gs->frameNum*0.01f);glVertex3f(1,0,0);
/*
	glColor3f(0.32f,0.32f,0.32f);
	glTexCoord2f(0,0-gs->frameNum*0.037f);glVertex3f(0,0,0);
	glTexCoord2f(0,2-gs->frameNum*0.037f);glVertex3f(0,1,0);
	glTexCoord2f(1,2-gs->frameNum*0.037f);glVertex3f(1,1,0);
	glTexCoord2f(1,0-gs->frameNum*0.037f);glVertex3f(1,0,0);
	glEnd();

	glBindTexture(GL_TEXTURE_2D, rawBumpTexture[0]);

	glColor3f(1,1,1);
	glBegin(GL_QUADS);
	glColor3f(0.45f,0.45f,0.45f);
	glTexCoord2f(0+gs->frameNum*0.02f,0);glVertex3f(0,0,0);
	glTexCoord2f(1+gs->frameNum*0.02f,0);glVertex3f(0,1,0);
	glTexCoord2f(1+gs->frameNum*0.02f,1);glVertex3f(1,1,0);
	glTexCoord2f(0+gs->frameNum*0.02f,1);glVertex3f(1,0,0);

	glColor3f(0.29f,0.29f,0.29f);
	glTexCoord2f(0-gs->frameNum*0.021f,0);glVertex3f(0,0,0);
	glTexCoord2f(1-gs->frameNum*0.021f,0);glVertex3f(0,1,0);
	glTexCoord2f(1-gs->frameNum*0.021f,1);glVertex3f(1,1,0);
	glTexCoord2f(0-gs->frameNum*0.021f,1);glVertex3f(1,0,0);
	glEnd();
/*/
	glColor3f(0.2f,0.2f,0.2f);
	glTexCoord2f(0,0+gs->frameNum*0.0046f);glVertex3f(0,0,0);
	glTexCoord2f(0,2+gs->frameNum*0.0046f);glVertex3f(0,1,0);
	glTexCoord2f(2,2+gs->frameNum*0.0046f);glVertex3f(1,1,0);
	glTexCoord2f(2,0+gs->frameNum*0.0046f);glVertex3f(1,0,0);

	glColor3f(0.2f,0.2f,0.2f);
	glTexCoord2f(0,0+gs->frameNum*0.0026f);glVertex3f(0,0,0);
	glTexCoord2f(0,4+gs->frameNum*0.0026f);glVertex3f(0,1,0);
	glTexCoord2f(2,4+gs->frameNum*0.0026f);glVertex3f(1,1,0);
	glTexCoord2f(2,0+gs->frameNum*0.0026f);glVertex3f(1,0,0);

	glTexCoord2f(0,0+gs->frameNum*0.0012f);glVertex3f(0,0,0);
	glTexCoord2f(0,8+gs->frameNum*0.0012f);glVertex3f(0,1,0);
	glTexCoord2f(2,8+gs->frameNum*0.0012f);glVertex3f(1,1,0);
	glTexCoord2f(2,0+gs->frameNum*0.0012f);glVertex3f(1,0,0);
	glEnd();

	glBindTexture(GL_TEXTURE_2D, rawBumpTexture[1]);
	glBegin(GL_QUADS);
	glColor3f(0.2f,0.2f,0.2f);
	glTexCoord2f(0,0+gs->frameNum*0.0036f);glVertex3f(0,0,0);
	glTexCoord2f(0,1+gs->frameNum*0.0036f);glVertex3f(0,1,0);
	glTexCoord2f(1,1+gs->frameNum*0.0036f);glVertex3f(1,1,0);
	glTexCoord2f(1,0+gs->frameNum*0.0036f);glVertex3f(1,0,0);
	glEnd();

	glBindTexture(GL_TEXTURE_2D, rawBumpTexture[2]);
	glBegin(GL_QUADS);
	glColor3f(0.2f,0.2f,0.2f);
	glTexCoord2f(0,0+gs->frameNum*0.0082f);glVertex3f(0,0,0);
	glTexCoord2f(0,1+gs->frameNum*0.0082f);glVertex3f(0,1,0);
	glTexCoord2f(1,1+gs->frameNum*0.0082f);glVertex3f(1,1,0);
	glTexCoord2f(1,0+gs->frameNum*0.0082f);glVertex3f(1,0,0);
	glEnd();
/**/
	glBindTexture(GL_TEXTURE_2D, bumpTexture);
	glCopyTexSubImage2D(GL_TEXTURE_2D,0,0,0,0,0,128,128);

	CCamera *realCam = camera;
	camera = new CCamera(*realCam);

	camera->up.x=0;
	camera->up.y=1;
	camera->up.z=0;
	camera->forward.y*=-1;
	camera->pos.y*=-1;
	camera->Update(false);

	glViewport(0,0,512,512);

	glClearColor(0.2f,0.4f,0.2f,1);
	glClear(GL_COLOR_BUFFER_BIT | GL_DEPTH_BUFFER_BIT);

	game->SetDrawMode(CGame::reflectionDraw);

	sky->Draw();

	glEnable(GL_CLIP_PLANE2);
	double plane[4]={0,1,0,0};
	glClipPlane(GL_CLIP_PLANE2 ,plane);
	drawReflection=true;

	readmap->GetGroundDrawer()->Draw(true);
	unitDrawer->Draw(true);
	featureHandler->Draw();
	ph->Draw(true);
	luaCallIns.DrawWorldReflection();

	game->SetDrawMode(CGame::normalDraw);

	drawReflection=false;
	glDisable(GL_CLIP_PLANE2);

	glColorMask(1,1,1,1);
	glBindTexture(GL_TEXTURE_2D, reflectTexture);
	glCopyTexSubImage2D(GL_TEXTURE_2D,0,0,0,0,0,512,512);

	glViewport(gu->viewPosX,0,gu->viewSizeX,gu->viewSizeY);
	glClearColor(FogLand[0],FogLand[1],FogLand[2],1);

	delete camera;
	camera = realCam;
	camera->Update(false);
}

⌨️ 快捷键说明

复制代码Ctrl + C
搜索代码Ctrl + F
全屏模式F11
增大字号Ctrl + =
减小字号Ctrl + -
显示快捷键?