]> git.tdb.fi Git - libs/gl.git/blob - source/object.cpp
Throw if trying to render an Object without a Technique
[libs/gl.git] / source / object.cpp
1 #include <msp/datafile/collection.h>
2 #include <msp/strings/format.h>
3 #include "material.h"
4 #include "mesh.h"
5 #include "object.h"
6 #include "objectinstance.h"
7 #include "program.h"
8 #include "programdata.h"
9 #include "renderer.h"
10 #include "technique.h"
11 #include "texturing.h"
12
13 using namespace std;
14
15 namespace Msp {
16 namespace GL {
17
18 Object::Object():
19         meshes(1)
20 { }
21
22 Object::Object(const Mesh *m, const Technique *t)
23 {
24         set_mesh(m);
25         set_technique(t);
26 }
27
28 // Avoid synthesizing ~RefPtr in files including object.h
29 Object::~Object()
30 { }
31
32 void Object::set_mesh(unsigned i, const Mesh *m)
33 {
34         if(i>meshes.size())
35                 throw out_of_range("Object::set_mesh");
36
37         if(i==meshes.size())
38                 meshes.push_back(m);
39         else
40                 meshes[i] = m;
41         meshes[i].keep();
42 }
43
44 const Mesh *Object::get_mesh(unsigned i) const
45 {
46         if(i>=meshes.size())
47                 return 0;
48
49         return meshes[i].get();
50 }
51
52 void Object::set_technique(const Technique *t)
53 {
54         technique = t;
55         technique.keep();
56 }
57
58 void Object::render(const Tag &tag) const
59 {
60         const RenderPass *pass = get_pass(tag);
61         if(!pass)
62                 return;
63
64         Bind bind_shader(pass->get_shader_program());
65         if(pass->get_shader_data())
66                 pass->get_shader_data()->apply();
67         Bind bind_material(pass->get_material());
68         Bind bind_texturing(pass->get_texturing());
69
70         meshes.front()->draw();
71 }
72
73 void Object::render(Renderer &renderer, const Tag &tag) const
74 {
75         const RenderPass *pass = get_pass(tag);
76         if(!pass)
77                 return;
78
79         Renderer::Push push(renderer);
80         pass->apply(renderer);
81
82         setup_render(renderer, tag);
83         meshes.front()->draw(renderer);
84         finish_render(renderer, tag);
85 }
86
87 void Object::render(Renderer &renderer, const ObjectInstance &inst, const Tag &tag) const
88 {
89         const RenderPass *pass = get_pass(tag);
90         if(!pass)
91                 return;
92
93         Renderer::Push push(renderer);
94         pass->apply(renderer);
95
96         setup_render(renderer, tag);
97         inst.setup_render(renderer, tag);
98         unsigned lod = min<unsigned>(inst.get_level_of_detail(renderer), meshes.size()-1);
99         meshes[lod]->draw(renderer);
100         inst.finish_render(renderer, tag);
101         finish_render(renderer, tag);
102 }
103
104 const RenderPass *Object::get_pass(const Tag &tag) const
105 {
106         if(!technique)
107                 throw logic_error("!technique");
108         if(!technique->has_pass(tag))
109                 return 0;
110         return &technique->get_pass(tag);
111 }
112
113
114 Object::Loader::Loader(Object &o):
115         DataFile::CollectionObjectLoader<Object>(o, 0)
116 {
117         init();
118 }
119
120 Object::Loader::Loader(Object &o, Collection &c):
121         DataFile::CollectionObjectLoader<Object>(o, &c)
122 {
123         init();
124 }
125
126 void Object::Loader::init()
127 {
128         add("mesh",     &Loader::mesh_inline);
129         add("mesh",     &Loader::mesh_inline_lod);
130         add("mesh",     &Loader::mesh);
131         add("mesh",     &Loader::mesh_lod);
132         add("technique", &Loader::technique_inline);
133         add("technique", &Loader::technique);
134
135         // Deprecated alias, will be removed
136         add("lod_mesh", &Loader::mesh_lod);
137 }
138
139 void Object::Loader::mesh_inline()
140 {
141         RefPtr<Mesh> msh = new Mesh;
142         load_sub(*msh);
143         obj.meshes.front() = msh;
144 }
145
146 void Object::Loader::mesh_inline_lod(unsigned l)
147 {
148         if(l>obj.meshes.size())
149                 throw out_of_range("Object::Loader::mesh_inline_lod");
150
151         RefPtr<Mesh> msh = new Mesh;
152         load_sub(*msh);
153         if(l==obj.meshes.size())
154                 obj.meshes.push_back(msh);
155         else
156                 obj.meshes[l] = msh;
157 }
158
159 void Object::Loader::mesh(const std::string &n)
160 {
161         obj.set_mesh(&get_collection().get<Mesh>(n));
162 }
163
164 void Object::Loader::mesh_lod(unsigned l, const string &n)
165 {
166         obj.set_mesh(l, &get_collection().get<Mesh>(n));
167 }
168
169 void Object::Loader::technique_inline()
170 {
171         RefPtr<Technique> tech = new Technique;
172         if(coll)
173                 load_sub(*tech, get_collection());
174         else
175                 load_sub(*tech);
176         obj.technique = tech;
177 }
178
179 void Object::Loader::technique(const std::string &n)
180 {
181         obj.set_technique(&get_collection().get<Technique>(n));
182 }
183
184 } // namespace GL
185 } // namespace Msp