1 | /* |
---|
2 | orxonox - the future of 3D-vertical-scrollers |
---|
3 | |
---|
4 | Copyright (C) 2004 orx |
---|
5 | |
---|
6 | This program is free software; you can redistribute it and/or modify |
---|
7 | it under the terms of the GNU General Public License as published by |
---|
8 | the Free Software Foundation; either version 2, or (at your option) |
---|
9 | any later version. |
---|
10 | |
---|
11 | ### File Specific: |
---|
12 | main-programmer: Stefan Lienard |
---|
13 | co-programmer: ... |
---|
14 | */ |
---|
15 | |
---|
16 | #include "mapped_water.h" |
---|
17 | #include "util/loading/load_param.h" |
---|
18 | #include "util/loading/factory.h" |
---|
19 | #include "state.h" |
---|
20 | #include "t_animation.h" |
---|
21 | #include <cmath> |
---|
22 | #include "glgui.h" |
---|
23 | #include "shell_command.h" |
---|
24 | #include "script_class.h" |
---|
25 | |
---|
26 | #include "resource_shader.h" |
---|
27 | |
---|
28 | |
---|
29 | ObjectListDefinition(MappedWater); |
---|
30 | CREATE_FACTORY(MappedWater); |
---|
31 | |
---|
32 | SHELL_COMMAND(gui, MappedWater, toggleGui); |
---|
33 | SHELL_COMMAND(output, MappedWater, saveParams); |
---|
34 | |
---|
35 | CREATE_SCRIPTABLE_CLASS(MappedWater, |
---|
36 | addMethod("waterUV", Executor2<MappedWater, lua_State*, float, float>(&MappedWater::fadeWaterUV)) |
---|
37 | ->addMethod("waterFlow", Executor2<MappedWater, lua_State*, float, float>(&MappedWater::fadeWaterFlow)) |
---|
38 | ->addMethod("shineSize", Executor2<MappedWater, lua_State*, float, float>(&MappedWater::fadeShineSize)) |
---|
39 | ->addMethod("shineStrength", Executor2<MappedWater, lua_State*, float, float>(&MappedWater::fadeShineStrength)) |
---|
40 | ->addMethod("reflStrength", Executor2<MappedWater, lua_State*, float, float>(&MappedWater::fadeReflStrength)) |
---|
41 | ->addMethod("refraction", Executor2<MappedWater, lua_State*, float, float>(&MappedWater::fadeRefraction)) |
---|
42 | ->addMethod("waterHeight", Executor2<MappedWater, lua_State*, float, float>(&MappedWater::fadeWaterHeight)) |
---|
43 | ->addMethod("waterColor", Executor4<MappedWater, lua_State*, float, float, float, float>(&MappedWater::fadeWaterColor))); |
---|
44 | |
---|
45 | |
---|
46 | /** |
---|
47 | * @brief constructor |
---|
48 | * @param root xml data |
---|
49 | */ |
---|
50 | MappedWater::MappedWater(const TiXmlElement* root) |
---|
51 | { |
---|
52 | this->registerObject(this, MappedWater::_objectList); |
---|
53 | this->toList(OM_ENVIRON); |
---|
54 | |
---|
55 | cam_uni = NULL; |
---|
56 | light_uni = NULL; |
---|
57 | color_uni = NULL; |
---|
58 | shineSize_uni = NULL; |
---|
59 | shineStrength_uni = NULL; |
---|
60 | reflStrength_uni = NULL; |
---|
61 | refr_uni = NULL; |
---|
62 | |
---|
63 | /// sets start values and parameters |
---|
64 | this->initParams(); |
---|
65 | // now the standard values were loaded, if the values are specified in the .oxw file |
---|
66 | // loadParams will overwrite the standard values with the specified values |
---|
67 | if (root != NULL) |
---|
68 | this->loadParams(root); |
---|
69 | |
---|
70 | /// initialization of the textures |
---|
71 | this->initTextures(); |
---|
72 | |
---|
73 | /// initialization of the shaders |
---|
74 | this->initShaders(); |
---|
75 | |
---|
76 | /// calculation of the 4 verts of the water quad |
---|
77 | this->calcVerts(); |
---|
78 | |
---|
79 | // init gui |
---|
80 | this->box = NULL; |
---|
81 | } |
---|
82 | |
---|
83 | /** |
---|
84 | * @brief deltes shader and the uniform used by the camera |
---|
85 | */ |
---|
86 | MappedWater::~MappedWater() |
---|
87 | { |
---|
88 | if ( cam_uni ) |
---|
89 | delete cam_uni; |
---|
90 | |
---|
91 | if ( color_uni ) |
---|
92 | delete color_uni; |
---|
93 | |
---|
94 | if ( light_uni ) |
---|
95 | delete light_uni; |
---|
96 | |
---|
97 | if ( shineSize_uni ) |
---|
98 | delete shineSize_uni; |
---|
99 | |
---|
100 | if ( shineStrength_uni ) |
---|
101 | delete shineStrength_uni; |
---|
102 | |
---|
103 | if ( reflStrength_uni ) |
---|
104 | delete reflStrength_uni; |
---|
105 | |
---|
106 | if ( refr_uni ) |
---|
107 | delete refr_uni; |
---|
108 | } |
---|
109 | |
---|
110 | /** |
---|
111 | * @brief initialization of loadable parameters, sets start standard values |
---|
112 | */ |
---|
113 | void MappedWater::initParams() |
---|
114 | { |
---|
115 | // those standardvalues will be overwritten if they're also set in the oxw file |
---|
116 | this->setWaterPos(0, 0, 0); |
---|
117 | this->setWaterSize(100, 100); |
---|
118 | this->setWaterUV(9); |
---|
119 | this->setWaterFlow(0.08f); |
---|
120 | this->setLightPos(0, 10, 0); |
---|
121 | this->setWaterAngle(0); |
---|
122 | this->setNormalMapScale(0.25f); |
---|
123 | this->setWaterColor(0.1f, 0.2f, 0.4f); |
---|
124 | this->setShineSize(128); |
---|
125 | this->setShineStrength(0.7f); |
---|
126 | this->setReflStrength(1.0f); |
---|
127 | this->setRefraction(0.009f); |
---|
128 | |
---|
129 | // initialization of the texture coords, speeds etc... |
---|
130 | // normalUV wont change anymore |
---|
131 | this->normalUV = this->waterUV * this->kNormalMapScale; |
---|
132 | // move and move2 are used for the reflection and refraction, the values are changed in tick() |
---|
133 | this->move = 0; |
---|
134 | this->move2 = this->move * this->kNormalMapScale; |
---|
135 | |
---|
136 | // initalize fading bools |
---|
137 | this->bFadeWaterHeight = false; |
---|
138 | this->bFadeWaterUV = false; |
---|
139 | this->bFadeWaterFlow = false; |
---|
140 | this->bFadeShineSize = false; |
---|
141 | this->bFadeShineStrength = false; |
---|
142 | this->bFadeReflStrength = false; |
---|
143 | this->bFadeRefraction = false; |
---|
144 | this->bFadeWaterColor = false; |
---|
145 | } |
---|
146 | |
---|
147 | /** |
---|
148 | * @brief initialization of the textures |
---|
149 | */ |
---|
150 | void MappedWater::initTextures() |
---|
151 | { |
---|
152 | // sets parameters for the textures |
---|
153 | this->textureSize = 512; |
---|
154 | unsigned int channels = 32; |
---|
155 | GLenum type = GL_RGBA; |
---|
156 | |
---|
157 | // set up refleciton texture |
---|
158 | Texture reflTex(GL_TEXTURE_2D, this->textureSize, this->textureSize, channels, type); |
---|
159 | mat.setDiffuseMap(reflTex, 0); |
---|
160 | // set up refraction texture |
---|
161 | Texture refrTex(GL_TEXTURE_2D, this->textureSize, this->textureSize, channels, type); |
---|
162 | mat.setDiffuseMap(refrTex, 1); |
---|
163 | // load normal map |
---|
164 | mat.setDiffuseMap("textures/water_normalmap.bmp", GL_TEXTURE_2D, 2); |
---|
165 | // load dudv map |
---|
166 | mat.setDiffuseMap("textures/water_dudvmap.bmp", GL_TEXTURE_2D, 3); |
---|
167 | |
---|
168 | // sets texture parameters for reflection texture |
---|
169 | glBindTexture(GL_TEXTURE_2D, this->mat.diffuseTextureID(0)); |
---|
170 | glTexParameterf(GL_TEXTURE_2D,GL_TEXTURE_MIN_FILTER,GL_LINEAR); |
---|
171 | glTexParameterf(GL_TEXTURE_2D,GL_TEXTURE_MAG_FILTER,GL_LINEAR); |
---|
172 | glTexParameterf(GL_TEXTURE_2D, GL_TEXTURE_WRAP_S, GL_CLAMP_TO_EDGE); |
---|
173 | glTexParameterf(GL_TEXTURE_2D, GL_TEXTURE_WRAP_T, GL_CLAMP_TO_EDGE); |
---|
174 | // sets texture parameters for refraction texture |
---|
175 | glBindTexture(GL_TEXTURE_2D, this->mat.diffuseTextureID(1)); |
---|
176 | glTexParameterf(GL_TEXTURE_2D,GL_TEXTURE_MIN_FILTER,GL_LINEAR); |
---|
177 | glTexParameterf(GL_TEXTURE_2D,GL_TEXTURE_MAG_FILTER,GL_LINEAR); |
---|
178 | glTexParameterf(GL_TEXTURE_2D, GL_TEXTURE_WRAP_S, GL_CLAMP_TO_EDGE); |
---|
179 | glTexParameterf(GL_TEXTURE_2D, GL_TEXTURE_WRAP_T, GL_CLAMP_TO_EDGE); |
---|
180 | } |
---|
181 | |
---|
182 | /** |
---|
183 | * @brief initialization of the shaders |
---|
184 | */ |
---|
185 | void MappedWater::initShaders() |
---|
186 | { |
---|
187 | if ( !Shader::isSupported() ) |
---|
188 | { |
---|
189 | return; |
---|
190 | } |
---|
191 | |
---|
192 | // load shader files |
---|
193 | shader = ResourceShader( "shaders/mapped_water.vert", "shaders/mapped_water.frag"); |
---|
194 | |
---|
195 | this->shader.activateShader(); |
---|
196 | // Set the variable "reflection" to correspond to the first texture unit |
---|
197 | Shader::Uniform(shader, "reflection").set(0); |
---|
198 | // Set the variable "refraction" to correspond to the second texture unit |
---|
199 | Shader::Uniform(shader, "refraction").set(1); |
---|
200 | // Set the variable "normalMap" to correspond to the third texture unit |
---|
201 | Shader::Uniform(shader, "normalMap").set(2); |
---|
202 | // Set the variable "dudvMap" to correspond to the fourth texture unit |
---|
203 | Shader::Uniform(shader, "dudvMap").set(3); |
---|
204 | // Set the variable "depthMap" to correspond to the fifth texture unit |
---|
205 | Shader::Uniform(shader, "depthMap").set(1); |
---|
206 | // Give the variable "waterColor" a blue color |
---|
207 | color_uni = new Shader::Uniform(shader, "waterColor"); |
---|
208 | color_uni->set(waterColor.x, waterColor.y, waterColor.z, 1.0f); |
---|
209 | // Give the variable "lightPos" our hard coded light position |
---|
210 | light_uni = new Shader::Uniform(shader, "lightPos"); |
---|
211 | light_uni->set(lightPos.x, lightPos.y, lightPos.z, 1.0f); |
---|
212 | // Set the variable "shine" |
---|
213 | shineSize_uni = new Shader::Uniform(shader, "kShine"); |
---|
214 | shineSize_uni->set(this->shineSize); |
---|
215 | // Set the variable "shineStrength" |
---|
216 | shineStrength_uni = new Shader::Uniform(shader, "shineStrength"); |
---|
217 | shineStrength_uni->set(this->shineStrength); |
---|
218 | // Set the variable "reflStrength" |
---|
219 | reflStrength_uni = new Shader::Uniform(shader, "reflStrength"); |
---|
220 | reflStrength_uni->set(this->reflStrength); |
---|
221 | // Set the variable "refraction" |
---|
222 | refr_uni = new Shader::Uniform(shader, "kRefraction"); |
---|
223 | refr_uni->set(this->refraction); |
---|
224 | // uniform for the camera position |
---|
225 | cam_uni = new Shader::Uniform(shader, "cameraPos"); |
---|
226 | |
---|
227 | this->shader.deactivateShader(); |
---|
228 | } |
---|
229 | |
---|
230 | /** |
---|
231 | * @brief calculates the 4 verts of the water quad |
---|
232 | */ |
---|
233 | void MappedWater::calcVerts() |
---|
234 | { |
---|
235 | float deg2radtemp = this->waterAngle / 180 * PI; |
---|
236 | |
---|
237 | this->waterVerts[2] = this->waterVerts[0] + cos(deg2radtemp) * this->xWidth; |
---|
238 | this->waterVerts[3] = this->waterVerts[1] + sin(deg2radtemp) * this->xWidth; |
---|
239 | this->waterVerts[4] = this->waterVerts[0] + cos(deg2radtemp) * this->xWidth - sin(deg2radtemp) * this->zWidth; |
---|
240 | this->waterVerts[5] = this->waterVerts[1] + sin(deg2radtemp) * this->xWidth + cos(deg2radtemp) * this->zWidth; |
---|
241 | this->waterVerts[6] = this->waterVerts[0] - sin(deg2radtemp) * this->zWidth; |
---|
242 | this->waterVerts[7] = this->waterVerts[1] + cos(deg2radtemp) * this->zWidth; |
---|
243 | } |
---|
244 | |
---|
245 | /** |
---|
246 | * @brief resets the waterColor in the Shader |
---|
247 | * @param r new value for red |
---|
248 | * @param g new value for green |
---|
249 | * @param b new value for blue |
---|
250 | */ |
---|
251 | void MappedWater::resetWaterColor(float r, float g, float b) |
---|
252 | { |
---|
253 | if ( !Shader::isSupported() ) |
---|
254 | { |
---|
255 | return; |
---|
256 | } |
---|
257 | |
---|
258 | this->shader.activateShader(); |
---|
259 | this->waterColor = Vector(r, g, b); |
---|
260 | |
---|
261 | // Give the variable "waterColor" a color |
---|
262 | color_uni->set(waterColor.x, waterColor.y, waterColor.z, 1.0f); |
---|
263 | |
---|
264 | this->shader.deactivateShader(); |
---|
265 | } |
---|
266 | |
---|
267 | /** |
---|
268 | * @brief resets the shininess in the Shader |
---|
269 | * @param shine new value for the shininess |
---|
270 | */ |
---|
271 | void MappedWater::resetShineSize(float shine) |
---|
272 | { |
---|
273 | if ( !Shader::isSupported() ) |
---|
274 | { |
---|
275 | return; |
---|
276 | } |
---|
277 | |
---|
278 | this->shader.activateShader(); |
---|
279 | this->shineSize = shine; |
---|
280 | |
---|
281 | // Set the variable "shine" |
---|
282 | shineSize_uni->set(this->shineSize); |
---|
283 | |
---|
284 | this->shader.deactivateShader(); |
---|
285 | } |
---|
286 | |
---|
287 | /** |
---|
288 | * @brief resets the strength of the specular reflection in the Shader |
---|
289 | * @param strength new value for the strength of the specular reflection |
---|
290 | */ |
---|
291 | void MappedWater::resetShineStrength(float strength) |
---|
292 | { |
---|
293 | if ( !Shader::isSupported() ) |
---|
294 | { |
---|
295 | return; |
---|
296 | } |
---|
297 | |
---|
298 | this->shader.activateShader(); |
---|
299 | this->shineStrength = strength; |
---|
300 | |
---|
301 | // Set the variable "shine" |
---|
302 | shineStrength_uni->set(this->shineStrength); |
---|
303 | |
---|
304 | this->shader.deactivateShader(); |
---|
305 | } |
---|
306 | |
---|
307 | /** |
---|
308 | * @brief resets the strength of the reflection in the Shader |
---|
309 | * @param strength new value for the strength of the reflection |
---|
310 | */ |
---|
311 | void MappedWater::resetReflStrength(float strength) |
---|
312 | { |
---|
313 | if ( !Shader::isSupported() ) |
---|
314 | { |
---|
315 | return; |
---|
316 | } |
---|
317 | |
---|
318 | this->shader.activateShader(); |
---|
319 | this->reflStrength = strength; |
---|
320 | |
---|
321 | // Set the variable "shine" |
---|
322 | reflStrength_uni->set(this->reflStrength); |
---|
323 | |
---|
324 | this->shader.deactivateShader(); |
---|
325 | } |
---|
326 | |
---|
327 | /** |
---|
328 | * @brief resets the refraction in the Shader |
---|
329 | * @param refraction new value for the refraction |
---|
330 | */ |
---|
331 | void MappedWater::resetRefraction(float refraction) |
---|
332 | { |
---|
333 | if ( !Shader::isSupported() ) |
---|
334 | { |
---|
335 | return; |
---|
336 | } |
---|
337 | |
---|
338 | this->shader.activateShader(); |
---|
339 | this->refraction = refraction; |
---|
340 | |
---|
341 | // Set the variable "shine" |
---|
342 | refr_uni->set(this->refraction); |
---|
343 | |
---|
344 | this->shader.deactivateShader(); |
---|
345 | } |
---|
346 | |
---|
347 | /** |
---|
348 | * @brief resets the lightPos in the Shader |
---|
349 | * @param x new x value |
---|
350 | * @param y new y value |
---|
351 | * @param z new z value |
---|
352 | */ |
---|
353 | void MappedWater::resetLightPos(float x, float y, float z) |
---|
354 | { |
---|
355 | if ( !Shader::isSupported() ) |
---|
356 | { |
---|
357 | return; |
---|
358 | } |
---|
359 | |
---|
360 | this->shader.activateShader(); |
---|
361 | this->lightPos = Vector(x, y, z); |
---|
362 | |
---|
363 | // Give the variable "lightPos" our hard coded light position |
---|
364 | light_uni->set(lightPos.x, lightPos.y, lightPos.z, 1.0f); |
---|
365 | |
---|
366 | this->shader.deactivateShader(); |
---|
367 | } |
---|
368 | |
---|
369 | /** |
---|
370 | * @brief ends the refraction and saves the graphic buffer into a texture |
---|
371 | * @param root xml data |
---|
372 | */ |
---|
373 | void MappedWater::loadParams(const TiXmlElement* root) |
---|
374 | { |
---|
375 | WorldEntity::loadParams(root); |
---|
376 | |
---|
377 | LoadParam(root, "waterpos", this, MappedWater, setWaterPos); |
---|
378 | LoadParam(root, "watersize", this, MappedWater, setWaterSize); |
---|
379 | LoadParam(root, "wateruv", this, MappedWater, setWaterUV); |
---|
380 | LoadParam(root, "waterflow", this, MappedWater, setWaterFlow); |
---|
381 | LoadParam(root, "lightpos", this, MappedWater, setLightPos); |
---|
382 | LoadParam(root, "waterangle", this, MappedWater, setWaterAngle); |
---|
383 | LoadParam(root, "normalmapscale", this, MappedWater, setNormalMapScale); |
---|
384 | LoadParam(root, "shinesize", this, MappedWater, setShineSize); |
---|
385 | LoadParam(root, "shinestrength", this, MappedWater, setShineStrength); |
---|
386 | LoadParam(root, "reflstrength", this, MappedWater, setReflStrength); |
---|
387 | LoadParam(root, "refraction", this, MappedWater, setRefraction); |
---|
388 | LoadParam(root, "watercolor", this, MappedWater, setWaterColor); |
---|
389 | } |
---|
390 | |
---|
391 | /** |
---|
392 | * @brief prints the xml code of the water params |
---|
393 | */ |
---|
394 | void MappedWater::saveParams() |
---|
395 | { |
---|
396 | // it's not too nice, but it works fine |
---|
397 | PRINT(0)("\nMappedWater XML Code:\n<MappedWater>\n"); |
---|
398 | |
---|
399 | PRINT(0)(" <waterpos>%f, %f, %f</waterpos>\n", this->waterVerts[0], this->waterHeight, this->waterVerts[1]); |
---|
400 | PRINT(0)(" <watersize>%f, %f</watersize>\n", this->xWidth, this->zWidth); |
---|
401 | PRINT(0)(" <wateruv>%f</wateruv>\n", this->waterUV); |
---|
402 | PRINT(0)(" <waterflow>%f</waterflow>\n", this->waterFlow); |
---|
403 | PRINT(0)(" <lightpos>%f, %f, %f</lightpos>\n", this->lightPos.x, this->lightPos.y, this->lightPos.z); |
---|
404 | PRINT(0)(" <waterangle>%f</waterangle>\n", this->waterAngle); |
---|
405 | PRINT(0)(" <normalmapscale>%f</normalmapscale>\n", this->kNormalMapScale); |
---|
406 | PRINT(0)(" <shinesize>%f</shinesize>\n", this->shineSize); |
---|
407 | PRINT(0)(" <shinestrength>%f</shinestrength>\n", this->shineStrength); |
---|
408 | PRINT(0)(" <reflstrength>%f</reflstrength>\n", this->reflStrength); |
---|
409 | PRINT(0)(" <refraction>%f</refraction>\n", this->refraction); |
---|
410 | PRINT(0)(" <watercolor>%f, %f, %f</watercolor>\n", this->waterColor.x, this->waterColor.y, this->waterColor.z); |
---|
411 | |
---|
412 | PRINT(0)("</MappedWater>\n"); |
---|
413 | } |
---|
414 | |
---|
415 | /** |
---|
416 | * @brief starts the slider gui that lets you edit all water parameters |
---|
417 | */ |
---|
418 | void MappedWater::toggleGui() |
---|
419 | { |
---|
420 | if (this->box == NULL) |
---|
421 | { |
---|
422 | this->box = new OrxGui::GLGuiBox(OrxGui::Vertical); |
---|
423 | { |
---|
424 | OrxGui::GLGuiBox* waterColorBox = new OrxGui::GLGuiBox(OrxGui::Horizontal); |
---|
425 | { |
---|
426 | OrxGui::GLGuiText* waterColorText = new OrxGui::GLGuiText(); |
---|
427 | waterColorText->setText("WaterColor"); |
---|
428 | waterColorBox->pack(waterColorText); |
---|
429 | |
---|
430 | OrxGui::GLGuiSlider* waterColorR = new OrxGui::GLGuiSlider(); |
---|
431 | waterColorR->setRange(0, 1.0f); |
---|
432 | waterColorR->setValue(this->waterColor.x); |
---|
433 | waterColorR->setStep(0.1f); |
---|
434 | waterColorR->valueChanged.connect(this, &MappedWater::resetWaterColorR); |
---|
435 | waterColorBox->pack(waterColorR); |
---|
436 | |
---|
437 | OrxGui::GLGuiSlider* waterColorG = new OrxGui::GLGuiSlider(); |
---|
438 | waterColorG->setRange(0, 1.0f); |
---|
439 | waterColorG->setStep(0.1f); |
---|
440 | waterColorG->setValue(this->waterColor.y); |
---|
441 | waterColorG->valueChanged.connect(this, &MappedWater::resetWaterColorG); |
---|
442 | waterColorBox->pack(waterColorG); |
---|
443 | |
---|
444 | OrxGui::GLGuiSlider* waterColorB = new OrxGui::GLGuiSlider(); |
---|
445 | waterColorB->setRange(0, 1.0f); |
---|
446 | waterColorB->setStep(0.1f); |
---|
447 | waterColorB->setValue(this->waterColor.z); |
---|
448 | waterColorB->valueChanged.connect(this, &MappedWater::resetWaterColorB); |
---|
449 | waterColorBox->pack(waterColorB); |
---|
450 | } |
---|
451 | this->box->pack(waterColorBox); |
---|
452 | |
---|
453 | OrxGui::GLGuiBox* waterUVBox = new OrxGui::GLGuiBox(OrxGui::Horizontal); |
---|
454 | { |
---|
455 | OrxGui::GLGuiText* waterUVText = new OrxGui::GLGuiText(); |
---|
456 | waterUVText->setText("WaterUV"); |
---|
457 | waterUVBox->pack(waterUVText); |
---|
458 | |
---|
459 | OrxGui::GLGuiSlider* waterUV = new OrxGui::GLGuiSlider(); |
---|
460 | waterUV->setRange(1, 20); |
---|
461 | waterUV->setValue(this->waterUV); |
---|
462 | waterUV->setStep(1); |
---|
463 | waterUV->valueChanged.connect(this, &MappedWater::setWaterUV); |
---|
464 | waterUVBox->pack(waterUV); |
---|
465 | } |
---|
466 | this->box->pack(waterUVBox); |
---|
467 | |
---|
468 | OrxGui::GLGuiBox* waterFlowBox = new OrxGui::GLGuiBox(OrxGui::Horizontal); |
---|
469 | { |
---|
470 | OrxGui::GLGuiText* waterFlowText = new OrxGui::GLGuiText(); |
---|
471 | waterFlowText->setText("WaterFlow"); |
---|
472 | waterFlowBox->pack(waterFlowText); |
---|
473 | |
---|
474 | OrxGui::GLGuiSlider* waterFlow = new OrxGui::GLGuiSlider(); |
---|
475 | waterFlow->setRange(0.01f, 2); |
---|
476 | waterFlow->setValue(this->waterFlow); |
---|
477 | waterFlow->setStep(0.02f); |
---|
478 | waterFlow->valueChanged.connect(this, &MappedWater::setWaterFlow); |
---|
479 | waterFlowBox->pack(waterFlow); |
---|
480 | } |
---|
481 | this->box->pack(waterFlowBox); |
---|
482 | |
---|
483 | OrxGui::GLGuiBox* shineSizeBox = new OrxGui::GLGuiBox(OrxGui::Horizontal); |
---|
484 | { |
---|
485 | OrxGui::GLGuiText* shineSizeText = new OrxGui::GLGuiText(); |
---|
486 | shineSizeText->setText("ShineSize"); |
---|
487 | shineSizeBox->pack(shineSizeText); |
---|
488 | |
---|
489 | OrxGui::GLGuiSlider* shineSize = new OrxGui::GLGuiSlider(); |
---|
490 | shineSize->setRange(1, 128); |
---|
491 | shineSize->setValue(this->shineSize); |
---|
492 | shineSize->setStep(1); |
---|
493 | shineSize->valueChanged.connect(this, &MappedWater::resetShineSize); |
---|
494 | shineSizeBox->pack(shineSize); |
---|
495 | } |
---|
496 | this->box->pack(shineSizeBox); |
---|
497 | |
---|
498 | OrxGui::GLGuiBox* shineStrengthBox = new OrxGui::GLGuiBox(OrxGui::Horizontal); |
---|
499 | { |
---|
500 | OrxGui::GLGuiText* shineStrengthText = new OrxGui::GLGuiText(); |
---|
501 | shineStrengthText->setText("ShineStrength"); |
---|
502 | shineStrengthBox->pack(shineStrengthText); |
---|
503 | |
---|
504 | OrxGui::GLGuiSlider* shineStrength = new OrxGui::GLGuiSlider(); |
---|
505 | shineStrength->setRange(0, 1); |
---|
506 | shineStrength->setValue(this->shineStrength); |
---|
507 | shineStrength->setStep(0.1f); |
---|
508 | shineStrength->valueChanged.connect(this, &MappedWater::resetShineStrength); |
---|
509 | shineStrengthBox->pack(shineStrength); |
---|
510 | } |
---|
511 | this->box->pack(shineStrengthBox); |
---|
512 | |
---|
513 | OrxGui::GLGuiBox* reflStrengthBox = new OrxGui::GLGuiBox(OrxGui::Horizontal); |
---|
514 | { |
---|
515 | OrxGui::GLGuiText* reflStrengthText = new OrxGui::GLGuiText(); |
---|
516 | reflStrengthText->setText("ReflStrength"); |
---|
517 | reflStrengthBox->pack(reflStrengthText); |
---|
518 | |
---|
519 | OrxGui::GLGuiSlider* reflStrength = new OrxGui::GLGuiSlider(); |
---|
520 | reflStrength->setRange(0, 1); |
---|
521 | reflStrength->setValue(this->reflStrength); |
---|
522 | reflStrength->setStep(0.1f); |
---|
523 | reflStrength->valueChanged.connect(this, &MappedWater::resetReflStrength); |
---|
524 | reflStrengthBox->pack(reflStrength); |
---|
525 | } |
---|
526 | this->box->pack(reflStrengthBox); |
---|
527 | |
---|
528 | OrxGui::GLGuiBox* refractionBox = new OrxGui::GLGuiBox(OrxGui::Horizontal); |
---|
529 | { |
---|
530 | OrxGui::GLGuiText* refractionText = new OrxGui::GLGuiText(); |
---|
531 | refractionText->setText("Refraction"); |
---|
532 | refractionBox->pack(refractionText); |
---|
533 | |
---|
534 | OrxGui::GLGuiSlider* refraction = new OrxGui::GLGuiSlider(); |
---|
535 | refraction->setRange(0.001f, 0.1f); |
---|
536 | refraction->setValue(this->refraction); |
---|
537 | refraction->setStep(0.004f); |
---|
538 | refraction->valueChanged.connect(this, &MappedWater::resetRefraction); |
---|
539 | refractionBox->pack(refraction); |
---|
540 | } |
---|
541 | this->box->pack(refractionBox); |
---|
542 | |
---|
543 | OrxGui::GLGuiBox* lightPosBox = new OrxGui::GLGuiBox(OrxGui::Horizontal); |
---|
544 | { |
---|
545 | OrxGui::GLGuiText* lightPosText = new OrxGui::GLGuiText(); |
---|
546 | lightPosText->setText("LightPos"); |
---|
547 | lightPosBox->pack(lightPosText); |
---|
548 | |
---|
549 | OrxGui::GLGuiSlider* lightPosX = new OrxGui::GLGuiSlider(); |
---|
550 | lightPosX->setRange(-4000, 4000); |
---|
551 | lightPosX->setValue(this->lightPos.x); |
---|
552 | lightPosX->setStep(15); |
---|
553 | lightPosX->valueChanged.connect(this, &MappedWater::resetLightPosX); |
---|
554 | lightPosBox->pack(lightPosX); |
---|
555 | |
---|
556 | OrxGui::GLGuiSlider* lightPosY = new OrxGui::GLGuiSlider(); |
---|
557 | lightPosY->setRange(-4000, 4000); |
---|
558 | lightPosY->setStep(15); |
---|
559 | lightPosY->setValue(this->lightPos.y); |
---|
560 | lightPosY->valueChanged.connect(this, &MappedWater::resetLightPosY); |
---|
561 | lightPosBox->pack(lightPosY); |
---|
562 | |
---|
563 | OrxGui::GLGuiSlider* lightPosZ = new OrxGui::GLGuiSlider(); |
---|
564 | lightPosZ->setRange(-4000, 4000); |
---|
565 | lightPosZ->setStep(15); |
---|
566 | lightPosZ->setValue(this->lightPos.z); |
---|
567 | lightPosZ->valueChanged.connect(this, &MappedWater::resetLightPosZ); |
---|
568 | lightPosBox->pack(lightPosZ); |
---|
569 | } |
---|
570 | this->box->pack(lightPosBox); |
---|
571 | |
---|
572 | OrxGui::GLGuiBox* waterHeightBox = new OrxGui::GLGuiBox(OrxGui::Horizontal); |
---|
573 | { |
---|
574 | OrxGui::GLGuiText* waterHeightText = new OrxGui::GLGuiText(); |
---|
575 | waterHeightText->setText("WaterHeight"); |
---|
576 | waterHeightBox->pack(waterHeightText); |
---|
577 | |
---|
578 | OrxGui::GLGuiSlider* waterHeight = new OrxGui::GLGuiSlider(); |
---|
579 | waterHeight->setRange(100, 370); |
---|
580 | waterHeight->setValue(this->waterHeight); |
---|
581 | waterHeight->setStep(4); |
---|
582 | waterHeight->valueChanged.connect(this, &MappedWater::setWaterHeight); |
---|
583 | waterHeightBox->pack(waterHeight); |
---|
584 | } |
---|
585 | this->box->pack(waterHeightBox); |
---|
586 | } |
---|
587 | |
---|
588 | this->box->showAll(); |
---|
589 | this->box->setAbsCoor2D(300, 40); |
---|
590 | OrxGui::GLGuiHandler::getInstance()->activate(); |
---|
591 | OrxGui::GLGuiHandler::getInstance()->activateCursor(); |
---|
592 | } |
---|
593 | else |
---|
594 | { |
---|
595 | OrxGui::GLGuiHandler::getInstance()->deactivate(); |
---|
596 | OrxGui::GLGuiHandler::getInstance()->deactivateCursor(); |
---|
597 | delete this->box; |
---|
598 | this->box = NULL; |
---|
599 | } |
---|
600 | } |
---|
601 | |
---|
602 | /** |
---|
603 | * @brief activates the water shader and draws a quad with four textures on it |
---|
604 | */ |
---|
605 | void MappedWater::draw() const |
---|
606 | { |
---|
607 | glMatrixMode(GL_MODELVIEW); |
---|
608 | glPushMatrix(); |
---|
609 | |
---|
610 | // don't use glRotate or glTranslate here... the shader won't work anymore |
---|
611 | |
---|
612 | mat.select(); |
---|
613 | if ( Shader::isSupported() ) |
---|
614 | { |
---|
615 | this->shader.activateShader(); |
---|
616 | |
---|
617 | // reset the camera uniform to the current cam position |
---|
618 | Vector pos = State::getCameraNode()->getAbsCoor(); |
---|
619 | cam_uni->set(pos.x, pos.y, pos.z, 1.0f); |
---|
620 | |
---|
621 | glDisable(GL_BLEND); |
---|
622 | |
---|
623 | // TODO change texture coords, so water doesnt look distorted when xWidth != zWidth |
---|
624 | glBegin(GL_QUADS); |
---|
625 | // The back left vertice for the water |
---|
626 | glMultiTexCoord2f(GL_TEXTURE0, 0, waterUV); // Reflection texture |
---|
627 | glMultiTexCoord2f(GL_TEXTURE1, 0, waterUV - move); // Refraction texture |
---|
628 | glMultiTexCoord2f(GL_TEXTURE2, 0, normalUV + move2); // Normal map texture |
---|
629 | glMultiTexCoord2f(GL_TEXTURE3, 0, 0); // DUDV map texture |
---|
630 | glVertex3f(this->waterVerts[0], this->waterHeight, this->waterVerts[1]); |
---|
631 | |
---|
632 | // The front left vertice for the water |
---|
633 | glMultiTexCoord2f(GL_TEXTURE0, 0, 0); // Reflection texture |
---|
634 | glMultiTexCoord2f(GL_TEXTURE1, 0, -move); // Refraction texture |
---|
635 | glMultiTexCoord2f(GL_TEXTURE2, 0, move2); // Normal map texture |
---|
636 | glMultiTexCoord2f(GL_TEXTURE3, 0, 0); // DUDV map texture |
---|
637 | glVertex3f(this->waterVerts[2], this->waterHeight, this->waterVerts[3]); |
---|
638 | |
---|
639 | // The front right vertice for the water |
---|
640 | glMultiTexCoord2f(GL_TEXTURE0, waterUV, 0); // Reflection texture |
---|
641 | glMultiTexCoord2f(GL_TEXTURE1, waterUV, -move); // Refraction texture |
---|
642 | glMultiTexCoord2f(GL_TEXTURE2, normalUV, move2); // Normal map texture |
---|
643 | glMultiTexCoord2f(GL_TEXTURE3, 0, 0); // DUDV map texture |
---|
644 | glVertex3f(this->waterVerts[4], this->waterHeight, this->waterVerts[5]); |
---|
645 | |
---|
646 | // The back right vertice for the water |
---|
647 | glMultiTexCoord2f(GL_TEXTURE0, waterUV, waterUV); // Reflection texture |
---|
648 | glMultiTexCoord2f(GL_TEXTURE1, waterUV, waterUV - move); // Refraction texture |
---|
649 | glMultiTexCoord2f(GL_TEXTURE2, normalUV, normalUV + move2); // Normal map texture |
---|
650 | glMultiTexCoord2f(GL_TEXTURE3, 0, 0); // DUDV map texture |
---|
651 | glVertex3f(this->waterVerts[6], this->waterHeight, this->waterVerts[7]); |
---|
652 | glEnd(); |
---|
653 | |
---|
654 | this->shader.deactivateShader(); |
---|
655 | } |
---|
656 | mat.unselect(); |
---|
657 | |
---|
658 | glPopMatrix(); |
---|
659 | } |
---|
660 | |
---|
661 | /** |
---|
662 | * @brief tick tack, calculates the flow of the water |
---|
663 | */ |
---|
664 | void MappedWater::tick(float dt) |
---|
665 | { |
---|
666 | // makes the water flow |
---|
667 | this->move += this->waterFlow * dt; |
---|
668 | this->move2 = this->move * this->kNormalMapScale; |
---|
669 | |
---|
670 | // fading TODO fix this so it isnt hacky anymore |
---|
671 | if(bFadeWaterUV) |
---|
672 | { |
---|
673 | this->waterUVFader = new tAnimation<MappedWater>(this, &MappedWater::setWaterUV); |
---|
674 | this->waterUVFader->setInfinity(ANIM_INF_CONSTANT); |
---|
675 | |
---|
676 | this->waterUVFader->addKeyFrame(this->waterUV, this->waterUVFadeTime, ANIM_LINEAR); |
---|
677 | this->waterUVFader->addKeyFrame(this->newWaterUV, 0, ANIM_LINEAR); |
---|
678 | |
---|
679 | bFadeWaterUV = false; |
---|
680 | this->waterUVFader->replay(); |
---|
681 | } |
---|
682 | |
---|
683 | if(bFadeWaterFlow) |
---|
684 | { |
---|
685 | this->waterFlowFader = new tAnimation<MappedWater>(this, &MappedWater::setWaterFlow); |
---|
686 | this->waterFlowFader->setInfinity(ANIM_INF_CONSTANT); |
---|
687 | |
---|
688 | this->waterFlowFader->addKeyFrame(this->waterFlow, this->waterFlowFadeTime, ANIM_LINEAR); |
---|
689 | this->waterFlowFader->addKeyFrame(this->newWaterFlow, 0, ANIM_LINEAR); |
---|
690 | |
---|
691 | bFadeWaterFlow = false; |
---|
692 | this->waterFlowFader->replay(); |
---|
693 | } |
---|
694 | |
---|
695 | if(bFadeShineSize) |
---|
696 | { |
---|
697 | this->shineSizeFader = new tAnimation<MappedWater>(this, &MappedWater::resetShineSize); |
---|
698 | this->shineSizeFader->setInfinity(ANIM_INF_CONSTANT); |
---|
699 | |
---|
700 | this->shineSizeFader->addKeyFrame(this->shineSize, this->shineSizeFadeTime, ANIM_LINEAR); |
---|
701 | this->shineSizeFader->addKeyFrame(this->newShineSize, 0, ANIM_LINEAR); |
---|
702 | |
---|
703 | bFadeShineSize = false; |
---|
704 | this->shineSizeFader->replay(); |
---|
705 | } |
---|
706 | |
---|
707 | if(bFadeShineStrength) |
---|
708 | { |
---|
709 | this->shineStrengthFader = new tAnimation<MappedWater>(this, &MappedWater::resetShineStrength); |
---|
710 | this->shineStrengthFader->setInfinity(ANIM_INF_CONSTANT); |
---|
711 | |
---|
712 | this->shineStrengthFader->addKeyFrame(this->shineStrength, this->shineStrengthFadeTime, ANIM_LINEAR); |
---|
713 | this->shineStrengthFader->addKeyFrame(this->newShineStrength, 0, ANIM_LINEAR); |
---|
714 | |
---|
715 | bFadeShineStrength = false; |
---|
716 | this->shineStrengthFader->replay(); |
---|
717 | } |
---|
718 | |
---|
719 | if(bFadeReflStrength) |
---|
720 | { |
---|
721 | this->reflStrengthFader = new tAnimation<MappedWater>(this, &MappedWater::resetReflStrength); |
---|
722 | this->reflStrengthFader->setInfinity(ANIM_INF_CONSTANT); |
---|
723 | |
---|
724 | this->reflStrengthFader->addKeyFrame(this->reflStrength, this->reflStrengthFadeTime, ANIM_LINEAR); |
---|
725 | this->reflStrengthFader->addKeyFrame(this->newReflStrength, 0, ANIM_LINEAR); |
---|
726 | |
---|
727 | bFadeReflStrength = false; |
---|
728 | this->reflStrengthFader->replay(); |
---|
729 | } |
---|
730 | |
---|
731 | if(bFadeRefraction) |
---|
732 | { |
---|
733 | this->refractionFader = new tAnimation<MappedWater>(this, &MappedWater::resetRefraction); |
---|
734 | this->refractionFader->setInfinity(ANIM_INF_CONSTANT); |
---|
735 | |
---|
736 | this->refractionFader->addKeyFrame(this->refraction, this->refractionFadeTime, ANIM_LINEAR); |
---|
737 | this->refractionFader->addKeyFrame(this->newRefraction, 0, ANIM_LINEAR); |
---|
738 | |
---|
739 | bFadeRefraction = false; |
---|
740 | this->refractionFader->replay(); |
---|
741 | } |
---|
742 | |
---|
743 | if(bFadeWaterHeight) |
---|
744 | { |
---|
745 | this->waterHeightFader = new tAnimation<MappedWater>(this, &MappedWater::setWaterHeight); |
---|
746 | this->waterHeightFader->setInfinity(ANIM_INF_CONSTANT); |
---|
747 | |
---|
748 | this->waterHeightFader->addKeyFrame(this->waterHeight, this->waterHeightFadeTime, ANIM_LINEAR); |
---|
749 | this->waterHeightFader->addKeyFrame(this->newWaterHeight, 0, ANIM_LINEAR); |
---|
750 | |
---|
751 | bFadeWaterHeight = false; |
---|
752 | this->waterHeightFader->replay(); |
---|
753 | } |
---|
754 | |
---|
755 | if(bFadeWaterColor) |
---|
756 | { |
---|
757 | this->waterColorRFader = new tAnimation<MappedWater>(this, &MappedWater::resetWaterColorR); |
---|
758 | this->waterColorRFader->setInfinity(ANIM_INF_CONSTANT); |
---|
759 | |
---|
760 | this->waterColorRFader->addKeyFrame(this->waterColor.x, this->waterColorFadeTime, ANIM_LINEAR); |
---|
761 | this->waterColorRFader->addKeyFrame(this->newWaterColor.x, 0, ANIM_LINEAR); |
---|
762 | |
---|
763 | this->waterColorRFader->replay(); |
---|
764 | |
---|
765 | this->waterColorGFader = new tAnimation<MappedWater>(this, &MappedWater::resetWaterColorG); |
---|
766 | this->waterColorGFader->setInfinity(ANIM_INF_CONSTANT); |
---|
767 | |
---|
768 | this->waterColorGFader->addKeyFrame(this->waterColor.y, this->waterColorFadeTime, ANIM_LINEAR); |
---|
769 | this->waterColorGFader->addKeyFrame(this->newWaterColor.y, 0, ANIM_LINEAR); |
---|
770 | |
---|
771 | this->waterColorGFader->replay(); |
---|
772 | |
---|
773 | this->waterColorBFader = new tAnimation<MappedWater>(this, &MappedWater::resetWaterColorB); |
---|
774 | this->waterColorBFader->setInfinity(ANIM_INF_CONSTANT); |
---|
775 | |
---|
776 | this->waterColorBFader->addKeyFrame(this->waterColor.z, this->waterColorFadeTime, ANIM_LINEAR); |
---|
777 | this->waterColorBFader->addKeyFrame(this->newWaterColor.z, 0, ANIM_LINEAR); |
---|
778 | |
---|
779 | this->waterColorBFader->replay(); |
---|
780 | |
---|
781 | bFadeWaterColor = false; |
---|
782 | } |
---|
783 | } |
---|
784 | |
---|
785 | /** |
---|
786 | * @brief prepares everything to render the reflection texutre |
---|
787 | */ |
---|
788 | void MappedWater::activateReflection() |
---|
789 | { |
---|
790 | // To create the reflection texture we just need to set the view port |
---|
791 | // to our texture map size, then render the current scene our camera |
---|
792 | // is looking at to the already allocated texture unit. Since this |
---|
793 | // is a reflection of the top of the water surface we use clipping |
---|
794 | // planes to only render the top of the world as a reflection. If |
---|
795 | // we are below the water we don't flip the reflection but just use |
---|
796 | // the current view of the top as we are seeing through the water. |
---|
797 | // When you look through water at the surface it isn't really reflected, |
---|
798 | // only when looking down from above the water on the surface. |
---|
799 | |
---|
800 | // save viewport matrix and change the viewport size |
---|
801 | glPushAttrib(GL_VIEWPORT_BIT); |
---|
802 | glViewport(0,0, textureSize, textureSize); |
---|
803 | |
---|
804 | glMatrixMode(GL_MODELVIEW); |
---|
805 | glPushMatrix(); |
---|
806 | |
---|
807 | // If our camera is above the water we will render the scene flipped upside down. |
---|
808 | // In order to line up the reflection nicely with the world we have to translate |
---|
809 | // the world to the position of our reflected surface, multiplied by two. |
---|
810 | glEnable(GL_CLIP_PLANE0); |
---|
811 | Vector pos = State::getCameraNode()->getAbsCoor(); |
---|
812 | |
---|
813 | if(pos.y > waterHeight) |
---|
814 | { |
---|
815 | // Translate the world, then flip it upside down |
---|
816 | glTranslatef(0, waterHeight * 2, 0); |
---|
817 | glScalef(1, -1, 1); |
---|
818 | |
---|
819 | // Since the world is updside down we need to change the culling to FRONT |
---|
820 | glCullFace(GL_FRONT); |
---|
821 | |
---|
822 | // Set our plane equation and turn clipping on |
---|
823 | double plane[4] = {0, 1, 0, -waterHeight}; |
---|
824 | glClipPlane(GL_CLIP_PLANE0, plane); |
---|
825 | } |
---|
826 | else |
---|
827 | { |
---|
828 | // If the camera is below the water we don't want to flip the world, |
---|
829 | // but just render it clipped so only the top is drawn. |
---|
830 | double plane[4] = {0, 1, 0, waterHeight}; |
---|
831 | glClipPlane(GL_CLIP_PLANE0, plane); |
---|
832 | } |
---|
833 | } |
---|
834 | |
---|
835 | /** |
---|
836 | * @brief ends the reflection and saves the graphic buffer into a texture |
---|
837 | */ |
---|
838 | void MappedWater::deactivateReflection() |
---|
839 | { |
---|
840 | glDisable(GL_CLIP_PLANE0); |
---|
841 | glCullFace(GL_BACK); |
---|
842 | |
---|
843 | // Create the texture and store it on the video card |
---|
844 | mat.renderToTexture(0, GL_TEXTURE_2D, 0, 0, 0, 0, 0, textureSize, textureSize); |
---|
845 | |
---|
846 | glPopMatrix(); |
---|
847 | glPopAttrib(); |
---|
848 | } |
---|
849 | |
---|
850 | /** |
---|
851 | * @brief prepares everything to render the refraction texutre |
---|
852 | */ |
---|
853 | void MappedWater::activateRefraction() |
---|
854 | { |
---|
855 | // To create the refraction and depth textures we do the same thing |
---|
856 | // we did for the reflection texture, except we don't need to turn |
---|
857 | // the world upside down. We want to find the depth of the water, |
---|
858 | // not the depth of the sky and above water terrain. |
---|
859 | |
---|
860 | // save viewport matrix and change the viewport size |
---|
861 | glPushAttrib(GL_VIEWPORT_BIT); |
---|
862 | glViewport(0,0, textureSize, textureSize); |
---|
863 | |
---|
864 | glMatrixMode(GL_MODELVIEW); |
---|
865 | glPushMatrix(); |
---|
866 | |
---|
867 | // If our camera is above the water we will render only the parts that |
---|
868 | // are under the water. If the camera is below the water we render |
---|
869 | // only the stuff above the water. Like the reflection texture, we |
---|
870 | // incorporate clipping planes to only render what we need. |
---|
871 | |
---|
872 | // If the camera is above water, render the data below the water |
---|
873 | glEnable(GL_CLIP_PLANE0); |
---|
874 | Vector pos = State::getCameraNode()->getAbsCoor(); |
---|
875 | if(pos.y > waterHeight) |
---|
876 | { |
---|
877 | double plane[4] = {0, -1, 0, waterHeight}; |
---|
878 | glClipPlane(GL_CLIP_PLANE0, plane); |
---|
879 | } |
---|
880 | // If the camera is below the water, only render the data above the water |
---|
881 | else |
---|
882 | { |
---|
883 | glCullFace(GL_FRONT); |
---|
884 | double plane[4] = {0, 1, 0, -waterHeight}; |
---|
885 | glClipPlane(GL_CLIP_PLANE0, plane); |
---|
886 | } |
---|
887 | } |
---|
888 | |
---|
889 | /** |
---|
890 | * @brief ends the refraction and saves the graphic buffer into a texture |
---|
891 | */ |
---|
892 | void MappedWater::deactivateRefraction() |
---|
893 | { |
---|
894 | glDisable(GL_CLIP_PLANE0); |
---|
895 | glCullFace(GL_BACK); |
---|
896 | |
---|
897 | // Create the texture and store it on the video card |
---|
898 | mat.renderToTexture(1, GL_TEXTURE_2D, 0, 0, 0, 0, 0, textureSize, textureSize); |
---|
899 | |
---|
900 | glPopMatrix(); |
---|
901 | glPopAttrib(); |
---|
902 | } |
---|