Tutor2 - 渲染几何体数据

From KlayGE
Revision as of 22:54, 1 February 2011 by Gongminmin (Talk | contribs)

Jump to: navigation, search

这一节当中我们将尝试使用三种不同的方法来渲染几何体:分别是使用预设的辅助对象类,从文件读取,以及在程序中使用顶点和索引数组来进行构建。

头文件和框架类的定义:

  1. include <KlayGE/KlayGE.hpp>
  2. include <KlayGE/App3D.hpp>
  3. include <KlayGE/ResLoader.hpp>
  4. include <KlayGE/Context.hpp>
  5. include <KlayGE/CameraController.hpp>
  6. include <KlayGE/Font.hpp>
  7. include <KlayGE/RenderEngine.hpp>
  8. include <KlayGE/RenderFactory.hpp>
  9. include <KlayGE/RenderEffect.hpp>
  10. include <KlayGE/SceneObjectHelper.hpp>
  11. include <KlayGE/Mesh.hpp>
  1. include <vector>
  2. include <sstream>

class TutorFramework : public KlayGE::App3DFramework {

public:

   TutorFramework();
   

protected:

   virtual void InitObjects();
   

private:

   virtual void DoUpdateOverlay();
   virtual KlayGE::uint32_t DoUpdate(KlayGE::uint32_t pass);
   
   // 使用轨迹球控制器来控制相机,浏览场景
   KlayGE::TrackballCameraController tbController_;
   
   KlayGE::FontPtr font_;
   
   // 使用SceneObjectHelper来管理场景中的对象
   KlayGE::SceneObjectHelperPtr renderableBox_;
   KlayGE::SceneObjectHelperPtr renderableFile_;
   KlayGE::SceneObjectHelperPtr renderableMesh_;

};

这里与上一单元有所不同的是,我们新增了一些成员变量来进行场景对象的管理以及相机的控制。除此之外,我们还将从静态网格模型StaticMesh派生得到一个用户类,用于构建来自顶点数据或者文件的几何体对象:

class RenderPolygon : public KlayGE::StaticMesh {

public:

   RenderPolygon(KlayGE::RenderModelPtr model, std::wstring const& name);
   virtual void OnRenderBegin();

};

我们的主函数与上一个例子相比不会有任何变化:

int main(int argc, char** argv) {

   KlayGE::ResLoader::Instance().AddPath("../Samples/media/Common");
   KlayGE::Context::Instance().LoadCfg("KlayGE.cfg");
   TutorFramework app;
   app.Create();
   app.Run();
   return 0;

}

在初始化场景对象时,我们依次构建一个预设的立方体对象,一个从meshml文件中读取的模型,以及一个自定义的模型:

void TutorFramework::InitObjects() {

   font_ = KlayGE::Context::Instance().RenderFactoryInstance().MakeFont("gkai00mp.kfont");
   
   // 首先构建的是预设的辅助几何体,例如三角条带组成的立方体RenderableTriBox,它的输入参数为一个由最小/最大坐标
   // 构建的Box对象,以及立方体的颜色。KlayGE中的所有可视物体(即Renderable的派生类,包括RenderableTriBox等)
   // 都必须对应有RenderTechnique,即渲染这个对象的方法——通常从FX文件中读取并获得
   KlayGE::Box boxRange(KlayGE::float3(-1.0f,-0.25f,-0.25f), KlayGE::float3(-0.5f, 0.25f, 0.25f));
   KlayGE::Color boxColor(1.0f, 0.0f, 0.0f, 1.0f);
   
   // 构建一个场景对象SceneObjectHelper并使用AddToSceneManager将其添加到场景管理器中,从而在窗口中进行渲染
   // SceneObjectHelper的传入参数除了辅助几何体的实例以外,还有一个属性参数,它的取值可以为:
   // SOA_Cullable:这个对象参与裁减。即,当它位于视锥体之外时,它会被自动排除出渲染队列之外,从而降低渲染负担。
   //               如果没有设置这一参数,那么系统将总是渲染这个对象,无论它是否在可见区域之内
   // SOA_Overlay:这个对象的渲染始终位于其它对象之前,即覆盖在默认场景之上
   // SOA_Moveable:这个对象是可以移动的。此时系统在计算它是否位于视锥体内时,会将GetModelMatrix()考虑在结果当中
   // SOA_Unvisible:这个对象是不可见的
   renderableBox_ = KlayGE::MakeSharedPtr<KlayGE::SceneObjectHelper>(
       KlayGE::MakeSharedPtr<KlayGE::RenderableTriBox>(boxRange, boxColor), KlayGE::SceneObject::SOA_Cullable);
   renderableBox_->AddToSceneManager();
   
   // 第二个要构建的几何体,我们选择从.meshml模型文件中读取(这里的teapot.meshml保存在Samples/media/Common目录下)
   // LoadModel()的第一个参数为文件名,第二个参数影响了D3D11下的数据访问策略(在OpenGL下无用处),之后的两个参数指定
   // 模型和网格数据对象实例的构建方法。注意这里的RenderPolygon就是我们之前自定义的StaticMesh派生类
   KlayGE::RenderModelPtr loadedModel = KlayGE::LoadModel("teapot.meshml", KlayGE::EAH_GPU_Read,
       KlayGE::CreateModelFactory<KlayGE::RenderModel>(), KlayGE::CreateMeshFactory<RenderPolygon>())();
   
   // 将模型加入到场景对象中
   renderableFile_ = KlayGE::MakeSharedPtr<KlayGE::SceneObjectHelper>(loadedModel, KlayGE::SceneObject::SOA_Cullable);
   renderableFile_->AddToSceneManager();
   
   // 下一步我们将要自己定义一串顶点数据,以及用户绘制这些顶点所需的图元和索引数据
   // 这里我们将试图通过8个顶点来绘制一个完整的立方体
   std::vector<KlayGE::float3> vertices;
   vertices.push_back(KlayGE::float3(0.5f,-0.25f, 0.25f));
   vertices.push_back(KlayGE::float3(1.0f,-0.25f, 0.25f));
   vertices.push_back(KlayGE::float3(1.0f,-0.25f,-0.25f));
   vertices.push_back(KlayGE::float3(0.5f,-0.25f,-0.25f));
   vertices.push_back(KlayGE::float3(0.5f, 0.25f, 0.25f));
   vertices.push_back(KlayGE::float3(1.0f, 0.25f, 0.25f));
   vertices.push_back(KlayGE::float3(1.0f, 0.25f,-0.25f));
   vertices.push_back(KlayGE::float3(0.5f, 0.25f,-0.25f));
   
   // 首先我们需要构建一个几何体模型RenderModel,它是所有StaticMesh,也就是静态网格的载体。一个RenderModel可以包含
   // 一个或多个网格对象,每个网格对象都有自己的MaterialID(材质ID),RenderTechnique等属性
   KlayGE::RenderModelPtr model = KlayGE::MakeSharedPtr<KlayGE::RenderModel>(L"model");
   
   // 显而易见,要构建一个最简单的立方体,我们至少需要两个网格对象(表达立方体侧面和顶面的图元信息)
   std::vector<KlayGE::StaticMeshPtr> meshes(2);
   
   // 第一个网格对象用于生成立方体的侧面,我们通过索引数组来反复引用指定位置的顶点,构成立方体侧面的三角条带图元
   std::vector<KlayGE::uint16_t> indices1;
   indices1.push_back(0); indices1.push_back(4); indices1.push_back(1); indices1.push_back(5);
   indices1.push_back(2); indices1.push_back(6); indices1.push_back(3); indices1.push_back(7);
   indices1.push_back(0); indices1.push_back(4);
   
   // 创建立方体侧面的网格对象
   meshes[0] = KlayGE::MakeSharedPtr<RenderPolygon>(model, L"side_mesh");
   
   // 将顶点数据的地址和大小传递给网格对象,并指定元素类型,以及D3D11下的数据访问策略
   // 这里的元素类型vertex_element由三个参数组成:
   // 第一个VEU_Position即顶点属性类型,除了顶点坐标之外,还可以为法线VEU_Normal,纹理坐标VEU_TextureCoord等等
   // 第二个参数为索引值,对于纹理坐标属性,它表示该纹理坐标对应的纹理通道
   // 第三个参数表示数据的类型,例如EF_GR32F(float2),EF_BGR32F(float3),EF_ABGR32F(float4)等
   meshes[0]->AddVertexStream(&vertices[0], static_cast<KlayGE::uint32_t>(sizeof(vertices[0]) * vertices.size()),
                         KlayGE::vertex_element(KlayGE::VEU_Position, 0, KlayGE::EF_BGR32F), KlayGE::EAH_GPU_Read);
   
   // 将索引数据的地址和大小传递给网格对象,此外还有索引数据的类型(EF_R16UI表示16位无符号整数)和数据访问策略
   meshes[0]->AddIndexStream(&indices1[0], static_cast<KlayGE::uint32_t>(sizeof(indices1[0]) * indices1.size()),
                             KlayGE::EF_R16UI, KlayGE::EAH_GPU_Read);
   
   // 设置图元的绘制方式,这里我们设置立方体侧面采取三角条带化的方法进行表达
   meshes[0]->GetRenderLayout()->TopologyType(KlayGE::RenderLayout::TT_TriangleStrip);
   
   //第二个网格对象用来构建立方体的顶面和底面,这一次我们会使用三角面的图元绘制方式,以及与之对应的顶点索引数组
   std::vector<KlayGE::uint16_t> indices2;
   indices2.push_back(0); indices2.push_back(1); indices2.push_back(2);
   indices2.push_back(0); indices2.push_back(2); indices2.push_back(3);
   indices2.push_back(7); indices2.push_back(6); indices2.push_back(5);
   indices2.push_back(7); indices2.push_back(5); indices2.push_back(4);
   
   // 构建网格对象并传递顶点数组和索引数组数据
   meshes[1] = KlayGE::MakeSharedPtr<RenderPolygon>(model, L"cap_mesh");
   meshes[1]->AddVertexStream(&vertices[0], static_cast<KlayGE::uint32_t>(sizeof(vertices[0]) * vertices.size()),
                         KlayGE::vertex_element(KlayGE::VEU_Position, 0, KlayGE::EF_BGR32F), KlayGE::EAH_GPU_Read);
   meshes[1]->AddIndexStream(&indices2[0], static_cast<KlayGE::uint32_t>(sizeof(indices2[0]) * indices2.size()),
                             KlayGE::EF_R16UI, KlayGE::EAH_GPU_Read);
   meshes[1]->GetRenderLayout()->TopologyType(KlayGE::RenderLayout::TT_TriangleList);
   
   // 将所有的网格对象传递给RenderModel几何模型
   model->AssignMeshes(meshes.begin(), meshes.end());
   
   // 将几何模型传递给场景对象,加入到场景当中
   renderableMesh_ = KlayGE::MakeSharedPtr<KlayGE::SceneObjectHelper>(model, KlayGE::SceneObject::SOA_Cullable);
   renderableMesh_->AddToSceneManager();
   
   // 为了观察场景中建立的物体,我们需要设置一个合适的观察矩阵和投影矩阵
   this->LookAt(KlayGE::float3(0, 0,-4.0f), KlayGE::float3(0, 0, 0));
   this->Proj(0.1f, 20.0f);
   
   // 我们可以将场景主相机传递给轨迹球控制器,并设置交互浏览时的放缩比例。之后就可以通过鼠标拖动来自由观察场景了
   tbController_.AttachCamera(this->ActiveCamera());
   tbController_.Scalers(0.01f, 0.05f);

}

覆盖显示的文字,以及主框架的更新函数与上一个例子相比没有太大的区别:

void TutorFramework::DoUpdateOverlay() {

   std::wostringstream stream;
   stream.precision(2);
   stream << std::fixed << this->FPS() << " FPS";
   font_->RenderText(0, 0, KlayGE::Color(1, 1, 0, 1), L"几何体绘制例子", 16);
   font_->RenderText(0, 18, KlayGE::Color(1, 1, 0, 1), stream.str(), 16);

}

KlayGE::uint32_t TutorFramework::DoUpdate(KlayGE::uint32_t pass) {

   KlayGE::RenderEngine& re = KlayGE::Context::Instance().RenderFactoryInstance().RenderEngineInstance();
   re.CurFrameBuffer()->Clear(KlayGE::FrameBuffer::CBM_Color | KlayGE::FrameBuffer::CBM_Depth,
                              KlayGE::Color(0.2f, 0.4f, 0.6f, 1), 1.0f, 0);
   
   return KlayGE::App3DFramework::URV_Need_Flush | KlayGE::App3DFramework::URV_Finished;

}

我们已经注意到,上面构建的几何体都是使用自定义的RenderPolygon作为网格数据的承载者的(预设辅助对象除外), 在自定义类的构造函数中,我们需要设置一个RenderTechnique给网格对象,以保证它被正确地渲染:

RenderPolygon::RenderPolygon(KlayGE::RenderModelPtr model, std::wstring const& name)

KlayGE::StaticMesh(model, name)

{

   // 这里我们直接取得RenderFactory对象并通过它来读取一个fxml效果文件。这里的RenderableHelper.fxml位于KlayGE
   // 目录的media/RenderFX子目录下,这个子目录已经被记录在ResLoader的资源路径当中
   KlayGE::RenderFactory& rf = KlayGE::Context::Instance().RenderFactoryInstance();
   KlayGE::RenderEffectPtr effect = rf.LoadEffect("RenderableHelper.fxml");
   
   // 设置网格所用的渲染方法为TriangleTec,这个方法记录在fxml文件的<technique>元素里
   SetRenderTechnique(effect->TechniqueByName("TriangleTec"));
   
   // 获取记录在fxml文件<parameter>元素中的color参数,并设置颜色参数值
   *(effect->ParameterByName("color")) = KlayGE::float4(1.0f, 0.0f, 0.0f, 1.0f);

}

OnRenderBegin()将在每帧的更新过程中被自动调用,我们可以在其中不断更新着色器效果的参数,或者对网格的属性进行改变:

void RenderPolygon::OnRenderBegin() {

   KlayGE::App3DFramework const& app = KlayGE::Context::Instance().AppInstance();
   
   // 通过阅读RenderableHelper.fxml中的内容可以发现,它需要获取实时的view-projection矩阵并参与顶点着色器的运算,
   // 因此这里我们通过设置matViewProj参数来实现这一要求
   KlayGE::float4x4 view_proj = app.ActiveCamera().ViewMatrix() * app.ActiveCamera().ProjMatrix();
   *(GetRenderTechnique()->Effect().ParameterByName("matViewProj")) = view_proj;

}