OpenGL ----- 球體世界

球體世界_效果圖
#include "GLTools.h"
#include "GLShaderManager.h"
#include "GLFrustum.h"
#include "GLBatch.h"
#include "GLMatrixStack.h"
#include "GLGeometryTransform.h"
#include "StopWatch.h"

#include <math.h>
#include <stdio.h>

#ifdef __APPLE__
#include <glut/glut.h>
#else
#define FREEGLUT_STATIC
#include <GL/glut.h>
#endif

//**4月褥、添加附加隨機球
#define NUM_SPHERES 50
GLFrame spheres[NUM_SPHERES];

GLShaderManager     shaderManager;          // 著色器管理器
GLMatrixStack       modelViewMatrix;        // 模型視圖矩陣
GLMatrixStack       projectionMatrix;       // 投影矩陣
GLFrustum           viewFrustum;            // 視景體
GLGeometryTransform transformPipeline;      // 幾何圖形變換管道

GLTriangleBatch     torusBatch;             // 花托批處理
GLBatch             floorBatch;             // 地板批處理

//**2脂崔、定義公轉球的批處理(公轉自轉)**
GLTriangleBatch     sphereBatch;            //球批處理

//**3、角色幀 照相機角色幀(全局照相機實例)
GLFrame             cameraFrame;

//**5蚌本、添加紋理
//紋理標記數(shù)組
GLuint uiTextures[3];

bool LoadTGATexture(const char *szFileName, GLenum minFilter, GLenum magFilter, GLenum wrapMode)
{

    GLbyte *pBits;
    int nWidth, nHeight, nComponents;
    GLenum eFormat;
    
    //1.讀取紋理數(shù)據(jù)
    pBits = gltReadTGABits(szFileName, &nWidth, &nHeight, &nComponents, &eFormat);
    if(pBits == NULL)
        return false;
    
    //2、設置紋理參數(shù)
    //參數(shù)1:紋理維度
    //參數(shù)2:為S/T坐標設置模式
    //參數(shù)3:wrapMode,環(huán)繞模式
    glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_WRAP_S, wrapMode);
    glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_WRAP_T, wrapMode);
    
    //參數(shù)1:紋理維度
    //參數(shù)2:線性過濾
    //參數(shù)3:wrapMode,環(huán)繞模式
    glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MIN_FILTER, minFilter);
    glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MAG_FILTER, magFilter);
    
    //3.載入紋理
    //參數(shù)1:紋理維度
    //參數(shù)2:mip貼圖層次
    //參數(shù)3:紋理單元存儲的顏色成分(從讀取像素圖是獲得)-將內部參數(shù)nComponents改為了通用壓縮紋理格式GL_COMPRESSED_RGB
    //參數(shù)4:加載紋理寬
    //參數(shù)5:加載紋理高
    //參數(shù)6:加載紋理的深度
    //參數(shù)7:像素數(shù)據(jù)的數(shù)據(jù)類型(GL_UNSIGNED_BYTE斑鼻,每個顏色分量都是一個8位無符號整數(shù))
    //參數(shù)8:指向紋理圖像數(shù)據(jù)的指針
    glTexImage2D(GL_TEXTURE_2D, 0, GL_COMPRESSED_RGB, nWidth, nHeight, 0,
                 eFormat, GL_UNSIGNED_BYTE, pBits);
    
    //使用完畢釋放pBits
    free(pBits);
    
    //只有minFilter 等于以下四種模式页滚,才可以生成Mip貼圖
    //GL_NEAREST_MIPMAP_NEAREST具有非常好的性能,并且閃爍現(xiàn)象非常弱
    //GL_LINEAR_MIPMAP_NEAREST常常用于對游戲進行加速稳吮,它使用了高質量的線性過濾器
    //GL_LINEAR_MIPMAP_LINEAR 和GL_NEAREST_MIPMAP_LINEAR 過濾器在Mip層之間執(zhí)行了一些額外的插值,以消除他們之間的過濾痕跡井濒。
    //GL_LINEAR_MIPMAP_LINEAR 三線性Mip貼圖灶似。紋理過濾的黃金準則,具有最高的精度瑞你。
    if(minFilter == GL_LINEAR_MIPMAP_LINEAR ||
       minFilter == GL_LINEAR_MIPMAP_NEAREST ||
       minFilter == GL_NEAREST_MIPMAP_LINEAR ||
       minFilter == GL_NEAREST_MIPMAP_NEAREST)
    //4.加載Mip,紋理生成所有的Mip層
    //參數(shù):GL_TEXTURE_1D酪惭、GL_TEXTURE_2D、GL_TEXTURE_3D
    glGenerateMipmap(GL_TEXTURE_2D);
    
    
    return true;
}



void SetupRC()
{
    //1.設置清屏顏色到顏色緩存區(qū)
    glClearColor(0.0f, 0.0f, 0.0f, 1.0f);
    
    //2.初始化著色器管理器
    shaderManager.InitializeStockShaders();
    
    //3.開啟深度測試/背面剔除
    glEnable(GL_DEPTH_TEST);
    glEnable(GL_CULL_FACE);

    //4.設置大球球
    gltMakeSphere(torusBatch, 0.4f, 40, 80);
    
    //5.設置小球(公轉自轉)
    gltMakeSphere(sphereBatch, 0.1f, 26, 13);
    
    //6.設置地板頂點數(shù)據(jù)&地板紋理
    GLfloat texSize = 10.0f;
    floorBatch.Begin(GL_TRIANGLE_FAN, 4,1);
    floorBatch.MultiTexCoord2f(0, 0.0f, 0.0f);
    floorBatch.Vertex3f(-20.f, -0.41f, 20.0f);
    
    floorBatch.MultiTexCoord2f(0, texSize, 0.0f);
    floorBatch.Vertex3f(20.0f, -0.41f, 20.f);
    
    floorBatch.MultiTexCoord2f(0, texSize, texSize);
    floorBatch.Vertex3f(20.0f, -0.41f, -20.0f);
    
    floorBatch.MultiTexCoord2f(0, 0.0f, texSize);
    floorBatch.Vertex3f(-20.0f, -0.41f, -20.0f);
    floorBatch.End();
    
    //7.隨機小球球頂點坐標數(shù)據(jù)
    for (int i = 0; i < NUM_SPHERES; i++) {
        
        //y軸不變者甲,X,Z產生隨機值
        GLfloat x = ((GLfloat)((rand() % 400) - 200 ) * 0.1f);
        GLfloat z = ((GLfloat)((rand() % 400) - 200 ) * 0.1f);
        
        //在y方向春感,將球體設置為0.0的位置,這使得它們看起來是飄浮在眼睛的高度
        //對spheres數(shù)組中的每一個頂點虏缸,設置頂點數(shù)據(jù)
        spheres[i].SetOrigin(x, 0.0f, z);
    }
    
    
    //8.命名紋理對象
    glGenTextures(3, uiTextures);
    
    //9.將TGA文件加載為2D紋理鲫懒。
    //參數(shù)1:紋理文件名稱
    //參數(shù)2&參數(shù)3:需要縮小&放大的過濾器
    //參數(shù)4:紋理坐標環(huán)繞模式
    glBindTexture(GL_TEXTURE_2D, uiTextures[0]);
    LoadTGATexture("marble.tga", GL_LINEAR_MIPMAP_LINEAR, GL_LINEAR, GL_REPEAT);
    
    
    glBindTexture(GL_TEXTURE_2D, uiTextures[1]);
    LoadTGATexture("marslike.tga", GL_LINEAR_MIPMAP_LINEAR,
                   GL_LINEAR, GL_CLAMP_TO_EDGE);
    
    
    glBindTexture(GL_TEXTURE_2D, uiTextures[2]);
    LoadTGATexture("moonlike.tga", GL_LINEAR_MIPMAP_LINEAR,
                   GL_LINEAR, GL_CLAMP_TO_EDGE);
    
   
}

//刪除紋理
void ShutdownRC(void)
{
    glDeleteTextures(3, uiTextures);
}

// 屏幕更改大小或已初始化
void ChangeSize(int nWidth, int nHeight)
{
    //1.設置視口
    glViewport(0, 0, nWidth, nHeight);
    
    //2.設置投影方式
    viewFrustum.SetPerspective(35.0f, float(nWidth)/float(nHeight), 1.0f, 100.0f);
    
    //3.將投影矩陣加載到投影矩陣堆棧,
    projectionMatrix.LoadMatrix(viewFrustum.GetProjectionMatrix());
    modelViewMatrix.LoadIdentity();
    
    //4.將投影矩陣堆棧和模型視圖矩陣對象設置到管道中
    transformPipeline.SetMatrixStacks(modelViewMatrix, projectionMatrix);
   
    
}

void drawSomething(GLfloat yRot)
{
    //1.定義光源位置&漫反射顏色
    static GLfloat vWhite[] = { 1.0f, 1.0f, 1.0f, 1.0f };
    static GLfloat vLightPos[] = { 0.0f, 3.0f, 0.0f, 1.0f };
    
    //2.繪制懸浮小球球
    glBindTexture(GL_TEXTURE_2D, uiTextures[2]);
    for(int i = 0; i < NUM_SPHERES; i++) {
        modelViewMatrix.PushMatrix();
        modelViewMatrix.MultMatrix(spheres[i]);
        shaderManager.UseStockShader(GLT_SHADER_TEXTURE_POINT_LIGHT_DIFF,
                                     modelViewMatrix.GetMatrix(),
                                     transformPipeline.GetProjectionMatrix(),
                                     vLightPos,
                                     vWhite,
                                     0);
        sphereBatch.Draw();
        modelViewMatrix.PopMatrix();
    }
    
    //3.繪制大球球
    modelViewMatrix.Translate(0.0f, 0.2f, -2.5f);
    modelViewMatrix.PushMatrix();
    modelViewMatrix.Rotate(yRot, 0.0f, 1.0f, 0.0f);
    glBindTexture(GL_TEXTURE_2D, uiTextures[1]);
    shaderManager.UseStockShader(GLT_SHADER_TEXTURE_POINT_LIGHT_DIFF,
                                 modelViewMatrix.GetMatrix(),
                                 transformPipeline.GetProjectionMatrix(),
                                 vLightPos,
                                 vWhite,
                                 0);
    torusBatch.Draw();
    modelViewMatrix.PopMatrix();
    
    //4.繪制公轉小球球(公轉自轉)
    modelViewMatrix.PushMatrix();
    modelViewMatrix.Rotate(yRot * -2.0f, 0.0f, 1.0f, 0.0f);
    modelViewMatrix.Translate(0.8f, 0.0f, 0.0f);
    glBindTexture(GL_TEXTURE_2D, uiTextures[2]);
    shaderManager.UseStockShader(GLT_SHADER_TEXTURE_POINT_LIGHT_DIFF,
                                 modelViewMatrix.GetMatrix(),
                                 transformPipeline.GetProjectionMatrix(),
                                 vLightPos,
                                 vWhite,
                                 0);
    sphereBatch.Draw();
    modelViewMatrix.PopMatrix();
    
}



//進行調用以繪制場景
void RenderScene(void)
{
    //1.地板顏色值
    static GLfloat vFloorColor[] = { 1.0f, 1.0f, 0.0f, 0.75f};
    
    //2.基于時間動畫
    static CStopWatch   rotTimer;
    float yRot = rotTimer.GetElapsedSeconds() * 60.0f;
    
    //3.清除顏色緩存區(qū)和深度緩沖區(qū)
    glClear(GL_COLOR_BUFFER_BIT | GL_DEPTH_BUFFER_BIT);
    
    //4.壓入棧(棧頂)
    modelViewMatrix.PushMatrix();
    
    //5.設置觀察者矩陣
    M3DMatrix44f mCamera;
    cameraFrame.GetCameraMatrix(mCamera);
    modelViewMatrix.MultMatrix(mCamera);
    
    //6.壓棧(鏡面)
    modelViewMatrix.PushMatrix();
    
    //7.---添加反光效果---
    //翻轉Y軸
    modelViewMatrix.Scale(1.0f, -1.0f, 1.0f);
    //鏡面世界圍繞Y軸平移一定間距
    modelViewMatrix.Translate(0.0f, 0.8f, 0.0f);
    
    //8.指定順時針為正面
    glFrontFace(GL_CW);
  
    //9.繪制地面以外其他部分(鏡面)
    drawSomething(yRot);
   
    //10.恢復為逆時針為正面
    glFrontFace(GL_CCW);
    
    //11.繪制鏡面,恢復矩陣
    modelViewMatrix.PopMatrix();
    
    //12.開啟混合功能(繪制地板)
    glEnable(GL_BLEND);
    //13. 指定glBlendFunc 顏色混合方程式
    glBlendFunc(GL_SRC_ALPHA, GL_ONE_MINUS_SRC_ALPHA);
    
    //14.綁定地面紋理
    glBindTexture(GL_TEXTURE_2D, uiTextures[0]);
    
    /*15.
     紋理調整著色器(將一個基本色乘以一個取自紋理的單元nTextureUnit的紋理)
     參數(shù)1:GLT_SHADER_TEXTURE_MODULATE
     參數(shù)2:模型視圖投影矩陣
     參數(shù)3:顏色
     參數(shù)4:紋理單元(第0層的紋理單元)
     
     */
    shaderManager.UseStockShader(GLT_SHADER_TEXTURE_MODULATE,
                                 transformPipeline.GetModelViewProjectionMatrix(),
                                 vFloorColor,
                                 0);
    //shaderManager.UseStockShader(GLT_SHADER_TEXTURE_REPLACE,transformPipeline.GetModelViewProjectionMatrix(),0);
    
    //開始繪制
    floorBatch.Draw();
    //取消混合
    glDisable(GL_BLEND);
    
    //16.繪制地面以外其他部分
    drawSomething(yRot);
    
    //17.繪制完刽辙,恢復矩陣
    modelViewMatrix.PopMatrix();
    
    //18.交換緩存區(qū)
    glutSwapBuffers();
    
    //19.提交重新渲染
    glutPostRedisplay();
    
    
    
}


//**3.移動照相機參考幀窥岩,來對方向鍵作出響應
void SpeacialKeys(int key,int x,int y)
{
    
    float linear = 0.1f;
    float angular = float(m3dDegToRad(5.0f));
    
    if (key == GLUT_KEY_UP) {
        
        //MoveForward 平移
        cameraFrame.MoveForward(linear);
    }
    
    if (key == GLUT_KEY_DOWN) {
        cameraFrame.MoveForward(-linear);
    }
    
    if (key == GLUT_KEY_LEFT) {
        //RotateWorld 旋轉
        cameraFrame.RotateWorld(angular, 0.0f, 1.0f, 0.0f);
    }
    
    if (key == GLUT_KEY_RIGHT) {
        cameraFrame.RotateWorld(-angular, 0.0f, 1.0f, 0.0f);
    }
    
    
    
}


int main(int argc, char* argv[])
{
    gltSetWorkingDirectory(argv[0]);
    
    glutInit(&argc, argv);
    glutInitDisplayMode(GLUT_DOUBLE | GLUT_RGB | GLUT_DEPTH);
    glutInitWindowSize(800,600);
    
    glutCreateWindow("OpenGL SphereWorld");
    
    glutReshapeFunc(ChangeSize);
    glutDisplayFunc(RenderScene);
    glutSpecialFunc(SpeacialKeys);
    
    GLenum err = glewInit();
    if (GLEW_OK != err) {
        fprintf(stderr, "GLEW Error: %s\n", glewGetErrorString(err));
        return 1;
    }
    
    
    SetupRC();
    glutMainLoop();
    ShutdownRC();
    return 0;
}

?著作權歸作者所有,轉載或內容合作請聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個濱河市扫倡,隨后出現(xiàn)的幾起案子谦秧,更是在濱河造成了極大的恐慌,老刑警劉巖撵溃,帶你破解...
    沈念sama閱讀 218,284評論 6 506
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件疚鲤,死亡現(xiàn)場離奇詭異,居然都是意外死亡缘挑,警方通過查閱死者的電腦和手機集歇,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 93,115評論 3 395
  • 文/潘曉璐 我一進店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來语淘,“玉大人诲宇,你說我怎么就攤上這事』谭” “怎么了姑蓝?”我有些...
    開封第一講書人閱讀 164,614評論 0 354
  • 文/不壞的土叔 我叫張陵,是天一觀的道長吕粗。 經常有香客問我纺荧,道長,這世上最難降的妖魔是什么颅筋? 我笑而不...
    開封第一講書人閱讀 58,671評論 1 293
  • 正文 為了忘掉前任宙暇,我火速辦了婚禮,結果婚禮上议泵,老公的妹妹穿的比我還像新娘占贫。我一直安慰自己,他們只是感情好先口,可當我...
    茶點故事閱讀 67,699評論 6 392
  • 文/花漫 我一把揭開白布型奥。 她就那樣靜靜地躺著,像睡著了一般池充。 火紅的嫁衣襯著肌膚如雪桩引。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 51,562評論 1 305
  • 那天收夸,我揣著相機與錄音坑匠,去河邊找鬼。 笑死卧惜,一個胖子當著我的面吹牛厘灼,可吹牛的內容都是我干的。 我是一名探鬼主播咽瓷,決...
    沈念sama閱讀 40,309評論 3 418
  • 文/蒼蘭香墨 我猛地睜開眼设凹,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了茅姜?” 一聲冷哼從身側響起闪朱,我...
    開封第一講書人閱讀 39,223評論 0 276
  • 序言:老撾萬榮一對情侶失蹤月匣,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后奋姿,有當?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體锄开,經...
    沈念sama閱讀 45,668評論 1 314
  • 正文 獨居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內容為張勛視角 年9月15日...
    茶點故事閱讀 37,859評論 3 336
  • 正文 我和宋清朗相戀三年称诗,在試婚紗的時候發(fā)現(xiàn)自己被綠了萍悴。 大學時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點故事閱讀 39,981評論 1 348
  • 序言:一個原本活蹦亂跳的男人離奇死亡寓免,死狀恐怖癣诱,靈堂內的尸體忽然破棺而出,到底是詐尸還是另有隱情袜香,我是刑警寧澤撕予,帶...
    沈念sama閱讀 35,705評論 5 347
  • 正文 年R本政府宣布,位于F島的核電站蜈首,受9級特大地震影響嗅蔬,放射性物質發(fā)生泄漏。R本人自食惡果不足惜疾就,卻給世界環(huán)境...
    茶點故事閱讀 41,310評論 3 330
  • 文/蒙蒙 一澜术、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧猬腰,春花似錦鸟废、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,904評論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至鼠冕,卻和暖如春添寺,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背懈费。 一陣腳步聲響...
    開封第一講書人閱讀 33,023評論 1 270
  • 我被黑心中介騙來泰國打工计露, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留,地道東北人憎乙。 一個月前我還...
    沈念sama閱讀 48,146評論 3 370
  • 正文 我出身青樓票罐,卻偏偏與公主長得像,于是被迫代替她去往敵國和親泞边。 傳聞我的和親對象是個殘疾皇子该押,可洞房花燭夜當晚...
    茶點故事閱讀 44,933評論 2 355