Gamp v0.0.7-36-g24b1eb6
Gamp: Graphics, Audio, Multimedia and Processing
Loading...
Searching...
No Matches
Primitives01.cpp
Go to the documentation of this file.
1/*
2 * Author: Sven Gothel <sgothel@jausoft.com>
3 * Copyright Gothel Software e.K.
4 *
5 * SPDX-License-Identifier: MIT
6 *
7 * This Source Code Form is subject to the terms of the MIT License
8 * If a copy of the MIT was not distributed with this file,
9 * you can obtain one at https://opensource.org/license/mit/.
10 */
11
12#include <cstdio>
13#include <cmath>
14#include <memory>
15
16#include <jau/basic_types.hpp>
17#include <jau/darray.hpp>
18#include <jau/file_util.hpp>
19#include <jau/float_math.hpp>
20#include <jau/float_types.hpp>
21#include <jau/fraction_type.hpp>
24#include <jau/math/vec3f.hpp>
25#include <jau/math/vec4f.hpp>
26#include <jau/math/vec4f.hpp>
28
29#include <gamp/Gamp.hpp>
34
36
37using namespace jau::math;
38using namespace jau::math::util;
39using namespace jau::math::geom;
40
41using namespace gamp::wt;
42using namespace gamp::wt::event;
43
44using namespace gamp::render::gl::glsl;
45using namespace gamp::render::gl::data;
46
47struct PMVMat4fUniform {
48 PMVMat4f m;
50
52 : m( PMVMat4f::INVERSE_PROJECTION | PMVMat4f::INVERSE_MODELVIEW | PMVMat4f::INVERSE_TRANSPOSED_MODELVIEW ),
53 u( GLUniformSyncMatrices4f::create("mgl_PMVMatrix", m.getSyncPMvMviMvit()) ) // P, Mv, Mvi and Mvit
54 {}
55};
56
57class Shape {
58 private:
59 GLenum m_type;
60 ShaderState& m_st;
61 PMVMat4fUniform& m_pmvMat;
63
64 Vec3f m_position;
65 Quat4f m_rotation;
66 GLUniformVec4fRef m_uColor;
67
68 public:
69 Shape(GLenum type, ShaderState &st, PMVMat4fUniform& pmvMatU)
70 : m_type(type), m_st(st), m_pmvMat(pmvMatU), m_array( GLFloatArrayDataServer::createGLSLInterleaved(2*3, false, 4, GL_STATIC_DRAW) )
71 {
72 m_array->addGLSLSubArray("mgl_Vertex", 3, GL_ARRAY_BUFFER);
73 m_array->addGLSLSubArray("mgl_Normal", 3, GL_ARRAY_BUFFER);
74 m_st.ownAttribute(m_array, true);
75
76 m_uColor = GLUniformVec4f::create("mgl_StaticColor", Vec4f(1, 0, 0, 1));
77 m_st.ownUniform(m_uColor, true);
78 }
79
80 constexpr const Vec3f& position() const noexcept { return m_position; }
81 constexpr Vec3f& position() noexcept { return m_position; }
82
83 constexpr const Quat4f& rotation() const noexcept { return m_rotation; }
84 constexpr Quat4f& rotation() noexcept { return m_rotation; }
85
86 constexpr const GLFloatArrayDataServerRef& vertices() const noexcept { return m_array; }
87 constexpr GLFloatArrayDataServerRef& vertices() noexcept { return m_array; }
88
89 const Vec4f& color() const noexcept { return m_uColor->vec4f(); }
90 void setColor(const Vec4f& c) noexcept { m_uColor->vec4f()=c; }
91
92 void seal(GL&gl, bool seal) {
93 m_array->seal(gl, seal);
94 }
95 void draw(GL &gl) {
96 m_pmvMat.m.pushMv();
97 m_pmvMat.m.translateMv(m_position); // identity + translate, scaled
98 // Rotate shape around its scaled center
99 m_pmvMat.m.rotateMv(m_rotation);
100 m_st.pushUniform(gl, m_pmvMat.u); // automatic sync + update of Mvi + Mvit
101
102 m_st.pushUniform(gl, m_uColor);
103
104 m_array->enableBuffer(gl, true);
105 ::glDrawArrays(m_type, 0, m_array->elemCount());
106 m_array->enableBuffer(gl, false);
107 m_pmvMat.m.popMv();
108 }
109
110};
111
113 private:
114 constexpr static jau::math::Vec3f lightPos = jau::math::Vec3f(0.0f, 5.0f, 10.0f);
115 constexpr static float half = 0.5f;
116 constexpr static float qter = 0.25f;
117 constexpr static float zNear= 1.0f;
118 constexpr static float zFar =100.0f;
119
120 ShaderState m_st;
121 Recti m_viewport;
122 bool m_initialized;
123 bool m_animating = true;
125 PMVMat4fUniform m_pmvMat;
126 Vec4f mgl_ColorStatic = Vec4f(0, 0, 0, 1);
127 Shape m_shape1, m_shape2;
128
129 public:
132 m_initialized(false),
133 m_shape1(GL_TRIANGLE_STRIP, m_st, m_pmvMat), m_shape2(GL_LINE_STRIP, m_st, m_pmvMat) { }
134
135 Recti& viewport() noexcept { return m_viewport; }
136 const Recti& viewport() const noexcept { return m_viewport; }
137
138 PMVMat4f& pmv() noexcept { return m_pmvMat.m; }
139 const PMVMat4f& pmv() const noexcept { return m_pmvMat.m; }
140 bool animating() const noexcept { return m_animating; }
141 bool& animating() noexcept { return m_animating; }
142
143 bool init(const WindowRef& win, const jau::fraction_timespec& when) override {
144 jau::fprintf_td(when.to_ms(), stdout, "RL::init: %s\n", toString().c_str());
145 m_tlast = when;
146
147 GL& gl = GL::downcast(win->renderContext());
148 ShaderCodeRef vp0 = ShaderCode::create(gl, GL_VERTEX_SHADER, "demos/glsl",
149 "demos/glsl/bin", "SingleLight0");
150 ShaderCodeRef fp0 = ShaderCode::create(gl, GL_FRAGMENT_SHADER, "demos/glsl",
151 "demos/glsl/bin", "SingleLight0");
152 if( !vp0 || !fp0 ) {
153 jau::fprintf_td(when.to_ms(), stdout, "ERROR %s:%d: %s\n", E_FILE_LINE, toString().c_str());
154 win->dispose(when);
155 return false;
156 }
157 {
158 std::string custom = "#define MAX_TEXTURE_UNITS 0\n";
159 size_t vsPos = vp0->defaultShaderCustomization(gl, true, true);
160 size_t fsPos = fp0->defaultShaderCustomization(gl, true, true);
161 vp0->insertShaderSource(0, vsPos, custom);
162 fp0->insertShaderSource(0, fsPos, custom);
163 }
164
166 if( !sp0->add(gl, vp0, true) || !sp0->add(gl, fp0, true) ) {
167 jau::fprintf_td(when.to_ms(), stdout, "ERROR %s:%d: %s\n", E_FILE_LINE, toString().c_str());
168 sp0->destroy(gl);
169 win->dispose(when);
170 return false;
171 }
172 m_st.attachShaderProgram(gl, sp0, true);
173
174 // setup mgl_PMVMatrix
175 m_pmvMat.m.getP().loadIdentity();
176 m_pmvMat.m.getMv().loadIdentity();
177 m_st.ownUniform(m_pmvMat.u, true);
178
179 GLUniformVec3fRef lightU = GLUniformVec3f::create("mgl_LightPos", lightPos);
180 m_st.ownUniform(lightU, true);
181
182 m_st.pushAllUniforms(gl);
183
184 const Vec3f frontNormal(0, 0, 1);
185 const Vec3f backNormal(0, 0, -1);
186 {
187 GLFloatArrayDataServerRef& v = m_shape1.vertices();
188 float dz = 0.001f;
189 v->put3f(-half, half, dz); v->put3f(frontNormal);
190 v->put3f( half, half, dz); v->put3f(frontNormal);
191 v->put3f(-half, -half, dz); v->put3f(frontNormal);
192 v->put3f( half, -half, dz); v->put3f(frontNormal);
193
194 dz = -0.001f;
195 v->put3f(-half, half, dz); v->put3f(backNormal);
196 v->put3f( half, half, dz); v->put3f(backNormal);
197 v->put3f(-half, -half, dz); v->put3f(backNormal);
198 v->put3f( half, -half, dz); v->put3f(backNormal);
199 m_shape1.seal(gl, true);
200 m_shape1.setColor(Vec4f(0, 1, 0, 1));
201 m_shape1.position().x = 1.5f;
202 }
203 {
204 const float lineWidth = 1/10.0f;
205 const float tw = 1.0f;
206 const float th = 1.0f;
207
208 float lwh = lineWidth/2.0f;
209
210 float twh = tw/2.0f;
211 float thh = th/2.0f;
212
213 float ctrX = 0, ctrY = 0;
214 float ctrZ = 0;
215
216 GLFloatArrayDataServerRef& v = m_shape2.vertices();
217 // CCW
218 v->put3f(ctrX-lwh, ctrY+thh, ctrZ); v->put3f(frontNormal); // vert: left-top
219 v->put3f(ctrX-lwh, ctrY+lwh, ctrZ); v->put3f(frontNormal);
220 v->put3f(ctrX-twh, ctrY+lwh, ctrZ); v->put3f(frontNormal); // horz: left-top
221 v->put3f(ctrX-twh, ctrY-lwh, ctrZ); v->put3f(frontNormal); // horz: left-bottom
222 v->put3f(ctrX-lwh, ctrY-lwh, ctrZ); v->put3f(frontNormal);
223 v->put3f(ctrX-lwh, ctrY-thh, ctrZ); v->put3f(frontNormal); // vert: left-bottom
224 v->put3f(ctrX+lwh, ctrY-thh, ctrZ); v->put3f(frontNormal); // vert: right-bottom
225 v->put3f(ctrX+lwh, ctrY-lwh, ctrZ); v->put3f(frontNormal);
226 v->put3f(ctrX+twh, ctrY-lwh, ctrZ); v->put3f(frontNormal); // horz: right-bottom
227 v->put3f(ctrX+twh, ctrY+lwh, ctrZ); v->put3f(frontNormal); // horz: right-top
228 v->put3f(ctrX+lwh, ctrY+lwh, ctrZ); v->put3f(frontNormal);
229 v->put3f(ctrX+lwh, ctrY+thh, ctrZ); v->put3f(frontNormal); // vert: right-top
230 v->put3f(ctrX-lwh, ctrY+thh, ctrZ); v->put3f(frontNormal); // vert: left-top
231 m_shape2.seal(gl, true);
232 m_shape2.setColor(Vec4f(0, 0, 1, 1));
233 m_shape2.position().x = -1.5f;
234 }
235
236 ::glClearColor(1.0f, 1.0f, 1.0f, 0.0f);
237 ::glEnable(GL_DEPTH_TEST);
238
239 m_initialized = sp0->inUse();
240 if( !m_initialized ) {
241 jau::fprintf_td(when.to_ms(), stdout, "ERROR %s:%d: %s\n", E_FILE_LINE, toString().c_str());
242 m_st.destroy(gl);
243 win->dispose(when);
244 }
245 return m_initialized;
246 }
247
248 void dispose(const WindowRef& win, const jau::fraction_timespec& when) override {
249 jau::fprintf_td(when.to_ms(), stdout, "RL::dispose: %s\n", toString().c_str());
250 m_st.destroy(GL::downcast(win->renderContext()));
251 m_initialized = false;
252 }
253
254 void reshape(const WindowRef& win, const jau::math::Recti& viewport, const jau::fraction_timespec& when) override {
255 GL& gl = GL::downcast(win->renderContext());
256 jau::fprintf_td(when.to_ms(), stdout, "RL::reshape: %s\n", toString().c_str());
257 m_viewport = viewport;
258
259 m_pmvMat.m.getP().loadIdentity();
260 const float aspect = 1.0f;
261 const float fovy_deg=45.0f;
262 const float aspect2 = ( (float) m_viewport.width() / (float) m_viewport.height() ) / aspect;
263 m_pmvMat.m.perspectiveP(jau::adeg_to_rad(fovy_deg), aspect2, zNear, zFar);
264 m_st.useProgram(gl, true);
265 m_st.pushUniform(gl, m_pmvMat.u); // automatic sync + update of Mvi + Mvit
266 m_st.useProgram(gl, true);
267 }
268
269 void display(const WindowRef& win, const jau::fraction_timespec& when) override {
270 // jau::fprintf_td(when.to_ms(), stdout, "RL::display: %s, %s\n", toString().c_str(), win->toString().c_str());
271 if( !m_initialized ) {
272 return;
273 }
274 GL& gl = GL::downcast(win->renderContext());
275 ::glClear(GL_COLOR_BUFFER_BIT | GL_DEPTH_BUFFER_BIT);
276
277 m_st.useProgram(gl, true);
278 m_pmvMat.m.getMv().loadIdentity();
279 m_pmvMat.m.translateMv(0, 0, -5);
280
281 if( animating() ) {
282 constexpr double angle_per_sec = 30;
283 const float rad = (float) ( (when - m_tlast).to_double() * angle_per_sec );
284 m_shape1.rotation().rotateByAngleY(jau::adeg_to_rad( rad ));
285 m_shape2.rotation().rotateByAngleY(jau::adeg_to_rad( -rad ));
286 }
287 m_shape1.draw(gl);
288 m_shape2.draw(gl);
289
290 m_st.useProgram(gl, false);
291
292 m_tlast = when;
293 }
294
295 std::string toStringImpl() const noexcept override { return "Primitives01"; }
296};
297
298class Example : public Primitives01 {
299 private:
300 class MyKeyListener : public KeyListener {
301 private:
302 Primitives01& m_parent;
303 public:
304 MyKeyListener(Primitives01& p) : m_parent(p) {}
305
306 void keyPressed(KeyEvent& e, const KeyboardTracker& kt) override {
307 jau::fprintf_td(e.when().to_ms(), stdout, "KeyPressed: %s; keys %zu\n", e.toString().c_str(), kt.pressedKeyCodes().bitCount());
308 if( e.keySym() == VKeyCode::VK_ESCAPE ) {
309 WindowRef win = e.source().lock();
310 if( win ) {
311 win->dispose(e.when());
312 }
313 } else if( e.keySym() == VKeyCode::VK_PAUSE || e.keySym() == VKeyCode::VK_P ) {
314 m_parent.animating() = !m_parent.animating();
315 } else if( e.keySym() == VKeyCode::VK_W ) {
316 WindowRef win = e.source().lock();
317 jau::fprintf_td(e.when().to_ms(), stdout, "Source: %s\n", win ? win->toString().c_str() : "null");
318 }
319 }
320 void keyReleased(KeyEvent& e, const KeyboardTracker& kt) override {
321 jau::fprintf_td(e.when().to_ms(), stdout, "KeyRelease: %s; keys %zu\n", e.toString().c_str(), kt.pressedKeyCodes().bitCount());
322 }
323 };
324 typedef std::shared_ptr<MyKeyListener> MyKeyListenerRef;
325 MyKeyListenerRef m_kl;
326
327 public:
329 : Primitives01(),
330 m_kl(std::make_shared<MyKeyListener>(*this)) { }
331
332 bool init(const WindowRef& win, const jau::fraction_timespec& when) override {
333 if( !Primitives01::init(win, when) ) {
334 return false;
335 }
336 win->addKeyListener(m_kl);
337 return true;
338 }
339 void dispose(const WindowRef& win, const jau::fraction_timespec& when) override {
340 win->removeKeyListener(m_kl);
341 Primitives01::dispose(win, when);
342 }
343};
344
345int main(int argc, char *argv[]) // NOLINT(bugprone-exception-escape)
346{
348
349 return launch("Primitives01.cpp",
351 std::make_shared<Example>(), argc, argv);
352}
int launch(std::string_view sfile, const GLLaunchProps &props, const RenderListenerRef &demo, int argc, char *argv[])
int main(int argc, char *argv[])
#define E_FILE_LINE
bool init(const WindowRef &win, const jau::fraction_timespec &when) override
Called by the drawable immediately after the render context is initialized.
void dispose(const WindowRef &win, const jau::fraction_timespec &when) override
Notifies the listener to perform the release of all renderer resources per context,...
void reshape(const WindowRef &win, const jau::math::Recti &viewport, const jau::fraction_timespec &when) override
Called by the drawable during the first repaint after the component has been resized.
PMVMat4f & pmv() noexcept
std::string toStringImpl() const noexcept override
const PMVMat4f & pmv() const noexcept
bool animating() const noexcept
void display(const WindowRef &win, const jau::fraction_timespec &when) override
Called by the drawable to initiate rendering by the client.
const Recti & viewport() const noexcept
bool init(const WindowRef &win, const jau::fraction_timespec &when) override
Called by the drawable immediately after the render context is initialized.
bool & animating() noexcept
Recti & viewport() noexcept
void dispose(const WindowRef &win, const jau::fraction_timespec &when) override
Notifies the listener to perform the release of all renderer resources per context,...
void setColor(const Vec4f &c) noexcept
constexpr const GLFloatArrayDataServerRef & vertices() const noexcept
constexpr Vec3f & position() noexcept
void draw(GL &gl)
constexpr const Quat4f & rotation() const noexcept
Shape(GLenum type, ShaderState &st, PMVMat4fUniform &pmvMatU)
constexpr const Vec3f & position() const noexcept
constexpr GLFloatArrayDataServerRef & vertices() noexcept
const Vec4f & color() const noexcept
void seal(GL &gl, bool seal)
constexpr Quat4f & rotation() noexcept
static GLContext & downcast(RenderContext *rc)
Downcast dereferenced given RenderContext* to GLContext&, throws exception if signature doesn't match...
Specifies the OpenGL profile.
Definition GLContext.hpp:41
static constexpr std::string_view GLES2
The embedded OpenGL profile ES 2.x, with x >= 0.
Definition GLContext.hpp:64
static std::shared_ptr< GLUniformVec3f > create(const string_t &name, const jau::math::Vec3f &v)
static std::shared_ptr< GLUniformVec4f > create(const string_t &name, const jau::math::Vec4f &v)
size_t defaultShaderCustomization(const GL &gl, bool preludeVersion, bool addDefaultPrecision)
Default customization of this shader source code.
static ShaderCodeRef create(GLenum type, size_t count, const source_list_t &sources) noexcept
size_t insertShaderSource(size_t shaderIdx, stringview_t tag, size_t fromIndex, stringview_t data) noexcept
Adds data after the line containing tag.
void destroy(GL &gl) noexcept
Detaches all shader codes and deletes the program.
static ShaderProgramRef create() noexcept
bool add(const ShaderCodeRef &shaderCode) noexcept
Adds a new shader to this program.
constexpr bool inUse() const noexcept
ShaderState allows to sharing data between shader programs, while updating the attribute and uniform ...
constexpr RenderListener(Private) noexcept
Private ctor for shared_ptr<RenderListener> instance method w/o public ctor.
Definition Window.hpp:60
std::string toString() const noexcept
Definition Window.hpp:112
const gamp::render::RenderContext * renderContext() const noexcept
Definition Surface.hpp:129
size_t removeKeyListener(const KeyListenerRef &l)
Definition Window.hpp:311
void dispose(const jau::fraction_timespec &when) noexcept override
Definition Window.hpp:355
std::string toString() const noexcept
Definition gamp_wt.cpp:145
void addKeyListener(const KeyListenerRef &l)
Definition Window.hpp:310
std::string toString() const noexcept
Definition KeyEvent.hpp:855
constexpr VKeyCode keySym() const noexcept
Returns the virtual key symbol reflecting the current keyboard layout.
Definition KeyEvent.hpp:798
virtual const PressedKeyCodes & pressedKeyCodes() const noexcept=0
constexpr const WindowWeakPtr & source() const noexcept
Definition Event.hpp:85
constexpr const jau::fraction_timespec & when() const noexcept
Definition Event.hpp:84
size_t bitCount() const noexcept
Definition bitfield.hpp:176
constexpr T adeg_to_rad(const T arc_degree) noexcept
Converts arc-degree to radians.
GLArrayDataServer< float > GLFloatArrayDataServer
GLArrayDataServerRef< float > GLFloatArrayDataServerRef
std::shared_ptr< GLUniformData > GLUniformDataRef
std::shared_ptr< GLUniformVec4f > GLUniformVec4fRef
std::shared_ptr< GLUniformVec3f > GLUniformVec3fRef
std::shared_ptr< ShaderProgram > ShaderProgramRef
std::shared_ptr< ShaderCode > ShaderCodeRef
@ verbose
Verbose operations (debugging).
std::shared_ptr< Window > WindowRef
Definition Event.hpp:36
Vector4F< float > Vec4f
Definition vec4f.hpp:375
Quaternion< float > Quat4f
RectI< int > Recti
Definition recti.hpp:139
Vector3F< float > Vec3f
Definition vec3f.hpp:436
PMVMatrix4< float > PMVMat4f
int fprintf_td(const uint64_t elapsed_ms, FILE *stream, const char *format,...) noexcept
Convenient fprintf() invocation, prepending the given elapsed_ms timestamp.
Definition debug.cpp:270
STL namespace.
GLUniformDataRef u
Timespec structure using int64_t for its components in analogy to struct timespec_t on 64-bit platfor...
constexpr uint64_t to_ms() const noexcept
Returns time in milliseconds.