// 18.RayPicking #include"VulkanAppFramework.h" #include #include #include #include #include #include #include #include using namespace hgl; using namespace hgl::graph; constexpr uint32_t SCREEN_WIDTH=1280; constexpr uint32_t SCREEN_HEIGHT=720; static float position_data[2][3]= { {100,100,100}, {0,0,0} }; static float lumiance_data[2]={1,1}; static Color4f white_color(1,1,1,1); static Color4f yellow_color(1,1,0,1); class TestApp:public CameraAppFramework { Color4f color; DeviceBuffer *ubo_color=nullptr; private: SceneNode render_root; RenderList * render_list =nullptr; Material * material =nullptr; MaterialInstance * mi_plane_grid =nullptr; MaterialInstance * mi_line =nullptr; Pipeline * pipeline =nullptr; Primitive * ro_plane_grid =nullptr; Primitive * ro_line =nullptr; VBO * vbo_pos =nullptr; Ray ray; private: bool InitMaterialAndPipeline() { mtl::Material3DCreateConfig cfg(device->GetDeviceAttribute(),"VertexLuminance3D",Prim::Lines); cfg.local_to_world=true; //AutoDelete mci=mtl::CreateVertexLuminance3D(&cfg); AutoDelete mci=mtl::LoadMaterialFromFile("Std3D/VertexLum3D",&cfg); material=db->CreateMaterial(mci); if(!material)return(false); db->global_descriptor.Bind(material); mi_plane_grid=db->CreateMaterialInstance(material); if(!mi_plane_grid)return(false); mi_plane_grid->WriteMIData(white_color); mi_line=db->CreateMaterialInstance(material); if(!mi_line)return(false); mi_line->WriteMIData(yellow_color); pipeline=CreatePipeline(material,InlinePipeline::Solid3D,Prim::Lines); if(!pipeline) return(false); return(true); } Renderable *Add(Primitive *r,MaterialInstance *mi) { Renderable *ri=db->CreateRenderable(r,mi,pipeline); if(!ri) { LOG_ERROR(OS_TEXT("Create Renderable failed.")); return(nullptr); } render_root.CreateSubNode(ri); return ri; } bool CreateRenderObject() { using namespace inline_geometry; { struct PlaneGridCreateInfo pgci; pgci.grid_size.width =32; pgci.grid_size.height=32; pgci.sub_count.width =8; pgci.sub_count.height=8; pgci.lum=0.5; pgci.sub_lum=0.75; ro_plane_grid=CreatePlaneGrid(db,material->GetDefaultVIL(),&pgci); } { ro_line=db->CreatePrimitive(2); if(!ro_line)return(false); if(!ro_line->Set(VAN::Position, vbo_pos= db->CreateVBO(VF_V3F,2,position_data )))return(false); if(!ro_line->Set(VAN::Luminance, db->CreateVBO(VF_V1F,2,lumiance_data )))return(false); } return(true); } bool InitScene() { Add(ro_plane_grid,mi_plane_grid); Add(ro_line,mi_line); camera->pos=Vector3f(32,32,32); camera_control->SetTarget(Vector3f(0,0,0)); camera_control->Refresh(); render_root.RefreshMatrix(); render_list->Expend(&render_root); return(true); } public: ~TestApp() { SAFE_CLEAR(render_list); } bool Init() { if(!CameraAppFramework::Init(SCREEN_WIDTH,SCREEN_HEIGHT)) return(false); render_list=new RenderList(device); if(!InitMaterialAndPipeline()) return(false); if(!CreateRenderObject()) return(false); if(!InitScene()) return(false); return(true); } void BuildCommandBuffer(uint32 index) override { const CameraInfo &ci=GetCameraInfo(); const ViewportInfo &vi=GetViewportInfo(); ray.Set(GetMouseCoord(),&ci,&vi); //设置射线查询的屏幕坐标点 const Vector3f pos=ray.ClosestPoint(Vector3f(0,0,0)); //求射线上与点(0,0,0)最近的点的坐标 vbo_pos->Write(&pos,3*sizeof(float)); //更新VBO上这个点的位置 VulkanApplicationFramework::BuildCommandBuffer(index,render_list); } void Resize(int w,int h) override { CameraAppFramework::Resize(w,h); VulkanApplicationFramework::BuildCommandBuffer(render_list); } };//class TestApp:public CameraAppFramework int main(int,char **) { TestApp app; if(!app.Init()) return(-1); while(app.Run()); return 0; }