Gamp v0.0.7-36-g24b1eb6
Gamp: Graphics, Audio, Multimedia and Processing
Loading...
Searching...
No Matches
Shape.hpp
Go to the documentation of this file.
1/*
2 * Author: Sven Gothel <sgothel@jausoft.com> (C++, Java) and Rami Santina (Java)
3 * Copyright Gothel Software e.K. and the authors
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#ifndef JAU_GAMP_GRAPH_UI_SHAPE_HPP_
12#define JAU_GAMP_GRAPH_UI_SHAPE_HPP_
13
14#include <jau/basic_types.hpp>
15#include <jau/darray.hpp>
16#include <jau/file_util.hpp>
17#include <jau/float_math.hpp>
18#include <jau/float_types.hpp>
19#include <jau/fraction_type.hpp>
22#include <jau/math/vec3f.hpp>
23#include <jau/math/vec4f.hpp>
24#include <jau/math/vec4f.hpp>
26
27#include <gamp/Gamp.hpp>
32
33namespace gamp::graph::ui {
34
35 using namespace jau::math;
36 using namespace jau::math::util;
37 using namespace jau::math::geom;
38
39 using namespace gamp::wt;
40 using namespace gamp::wt::event;
41
42 using namespace gamp::graph;
43
44 /** \addtogroup Gamp_GraphUI
45 *
46 * @{
47 */
48
49 class Shape {
50 public:
51 /**
52 * Visitor1 method
53 * @param s the {@link Shape} to process
54 * @return true to signal operation complete and to stop traversal, otherwise false
55 */
57
58 /**
59 * Visitor2 method
60 * @param s the {@link Shape} to process
61 * @param pmv the {@link PMVMatrix4f} setup from the {@link Scene} down to the {@link Shape}
62 * @return true to signal operation complete and to stop traversal, otherwise false
63 */
65
66 /**
67 * {@link Shape} move listener
68 * @param s the moved shape
69 * @param origin original position, relative object coordinates to the associated {@link Shape}
70 * @param dest new position, relative object coordinates to the associated {@link Shape}
71 * @param e original Newt {@link MouseEvent}
72 */
73 typedef jau::function<void(ShapeRef& s, const Vec3f& origin, const Vec3f& dest, const PointerEvent& e)> MoveEventCallback;
74
75 /**
76 * {@link Shape} pointer listener, e.g. for {@link Shape#onClicked(PointerListener)}
77 * @param s the associated {@link Shape} for this event
78 * @param pos relative object coordinates to the associated {@link Shape}
79 * @param e original Newt {@link MouseEvent}
80 */
81 typedef jau::function<void(ShapeRef& s, const Vec3f& pos, const PointerEvent& e)> PointerEventCallback;
82
83 /**
84 * General {@link Shape} listener action
85 */
87
88 /**
89 * {@link Shape} draw listener action returning a boolean value
90 * <p>
91 * If {@link #run(Shape, GL2ES2, RegionRenderer)} returns {@code true},
92 * the listener will be removed at {@link Shape#draw(GL2ES2, RegionRenderer)}
93 * otherwise kept calling.
94 * </p>
95 *
96 * Return {@code true} to remove this {@link DrawListener} at {@link Shape#draw(GL2ES2, RegionRenderer)},
97 * otherwise it is being kept and called.
98 * @param shape The shape
99 * @param gl the current {@link GL2ES2} object
100 * @param renderer the {@link RegionRenderer}
101 */
102 // FIXME typedef jau::function<void(ShapeRef& s, RenderContext& rc, GraphRenderer& renderer)> DrawEventCallback;
103
104 private:
105 OutlineShape m_outlines;
106
107 Vec3f m_position;
108 Quat4f m_rotation;
109 Vec3f m_rotPivot;
110 Vec3f m_scale = Vec3f(1, 1, 1);
111 float m_zOffset;
112 Vec4f m_color = Vec4f(0, 0, 0, 1);
113
114 Mat4f iMat;
115 Mat4f tmpMat;
116 bool iMatIdent = true;
117 bool iMatDirty = false;
118
119 public:
121 : m_outlines()
122 { }
123
124 virtual Container* asContainer() const noexcept { return nullptr; }
125
126 constexpr const Vec3f& position() const noexcept { return m_position; }
127 constexpr Vec3f& position() noexcept { iMatDirty=true; return m_position; }
128
129 constexpr const float& zOffset() const noexcept { return m_zOffset; }
130 constexpr float& zOffset() noexcept { iMatDirty=true; return m_zOffset; }
131
132 constexpr const Quat4f& rotation() const noexcept { return m_rotation; }
133 constexpr Quat4f& rotation() noexcept { iMatDirty=true; return m_rotation; }
134
135 constexpr const Vec3f& rotationPivot() const noexcept { return m_rotPivot; }
136 constexpr Vec3f& rotationPivot() noexcept { iMatDirty=true; return m_rotPivot; }
137
138 constexpr const Vec3f& scale() const noexcept { return m_scale; }
139 constexpr Vec3f& scale() noexcept { iMatDirty=true; return m_scale; }
140
141 constexpr const OutlineShape& outlines() const noexcept { return m_outlines; }
142 constexpr OutlineShape& outlines() noexcept { return m_outlines; }
143
144 constexpr const Vec4f& color() const noexcept { return m_color; }
145 constexpr void setColor(const Vec4f& c) noexcept { m_color=c; }
146
147 /**
148 * Applies the internal {@link Matrix4f} to the given {@link PMVMatrix4f#getMv() modelview matrix},
149 * i.e. {@code pmv.mulMv( getMat() )}.
150 * <p>
151 * Calls {@link #updateMat()} if dirty.
152 * </p>
153 * In case {@link #isMatIdentity()} is {@code true}, implementation is a no-operation.
154 * </p>
155 * @param pmv the matrix
156 * @see #isMatIdentity()
157 * @see #updateMat()
158 * @see #getMat()
159 * @see PMVMatrix4f#mulMv(Matrix4f)
160 */
161 void applyMatToMv(PMVMat4f& pmvMat) noexcept {
162 if( iMatDirty ) {
163 updateMat();
164 }
165 if( !iMatIdent ) {
166 pmvMat.mulMv(iMat);
167 }
168 }
169
170 private:
171 void updateMat() noexcept {
172 bool hasPos = !m_position.is_zero();
173 bool hasScale = m_scale != Vec3f::one;
174 bool hasRotate = !m_rotation.isIdentity();
175 bool hasRotPivot = false; // null != rotPivot;
176 const Vec3f& ctr = m_outlines.bounds().center();
177 bool sameScaleRotatePivot = hasScale && hasRotate && ( !hasRotPivot || m_rotPivot == ctr );
178
179 if( sameScaleRotatePivot ) {
180 iMatIdent = false;
181 iMat.setToTranslation(m_position); // identity + translate, scaled
182 // Scale shape from its center position and rotate around its center
183 iMat.translate(Vec3f(ctr).mul(m_scale)); // add-back center, scaled
184 iMat.rotate(m_rotation);
185 iMat.scale(m_scale);
186 iMat.translate(-ctr); // move to center
187 } else if( hasRotate || hasScale ) {
188 iMatIdent = false;
189 iMat.setToTranslation(m_position); // identity + translate, scaled
190 if( hasRotate ) {
191 if( hasRotPivot ) {
192 // Rotate shape around its scaled pivot
193 iMat.translate(Vec3f(m_rotPivot).mul(m_scale)); // pivot back from rot-pivot, scaled
194 iMat.rotate(m_rotation);
195 iMat.translate(Vec3f(-m_rotPivot).mul(m_scale)); // pivot to rot-pivot, scaled
196 } else {
197 // Rotate shape around its scaled center
198 iMat.translate(Vec3f(ctr).mul(m_scale)); // pivot back from center-pivot, scaled
199 iMat.rotate(m_rotation);
200 iMat.translate(Vec3f(-ctr).mul(m_scale)); // pivot to center-pivot, scaled
201 }
202 }
203 if( hasScale ) {
204 // Scale shape from its center position
205 iMat.translate(Vec3f(ctr).mul(m_scale)); // add-back center, scaled
206 iMat.scale(m_scale);
207 iMat.translate(Vec3f(-ctr).mul(m_scale)); // move to center
208 }
209 } else if( hasPos ) {
210 iMatIdent = false;
211 iMat.setToTranslation(m_position); // identity + translate, scaled
212
213 } else {
214 iMatIdent = true;
215 iMat.loadIdentity();
216 }
217 iMatDirty = false;
218 }
219
220 };
221
223 private:
224 GearsES2& m_parent;
225 jau::math::Vec2i preWinPos;
226 jau::math::Vec3f startPos;
227 GearsObjectES2* m_picked = nullptr;
228 bool m_dragInit = true;
229
230 bool mapWinToObj(const GearsObjectES2& shape, const jau::math::Vec2i& winPos, jau::math::Vec3f& viewPos) noexcept {
231 // OK global: m_parent.m_pmvMatrix, m_parent.m_viewport
232 const jau::math::Vec3f& ctr = shape.objBounds().center();
233 if( m_parent.pmvMatrix().mapObjToWin(ctr, m_parent.viewport(), viewPos) ) {
234 const float winZ = viewPos.z;
235 return m_parent.pmvMatrix().mapWinToObj((float)winPos.x, (float)winPos.y, winZ, m_parent.viewport(), viewPos);
236 }
237 return false;
238 }
239 bool mapWinToObjRay(const jau::math::Vec2i& pos, jau::math::Ray3f& ray, const Mat4f& mPmvi) noexcept {
240 // OK global: m_parent.m_pmvMatrix, m_parent.m_viewport
241 constexpr float winZ0 = 0.0f;
242 constexpr float winZ1 = 0.3f;
243 return Mat4f::mapWinToAnyRay((float)pos.x, (float)pos.y, winZ0, winZ1, mPmvi, m_parent.viewport(), ray);
244 }
245
246 bool pick(const PointerEvent& e, const WindowRef&, GearsObjectES2& shape) noexcept {
247 // While being processed fast w/o matrix traversal of shapes,
248 // we still need to use the cached PMvi matrix for win->obj ray for accuracy.
249 // A win->view ray fails in certain angles in edge cases!
250 jau::math::Vec3f objPos;
251 jau::math::Ray3f objRay;
252 const jau::math::Vec2i& winPos = e.position();
253 if( !mapWinToObjRay(winPos, objRay, shape.matPMvi()) ) {
254 return false;
255 }
256 const jau::math::geom::AABBox3f& objBox = shape.objBounds();
257
258 if( !objBox.intersectsRay(objRay) ) {
259 return false;
260 }
261 if( !objBox.getRayIntersection(objPos, objRay, std::numeric_limits<float>::epsilon(), /*assumeIntersection=*/true) ) {
262 printf("obj getRayIntersection failed\n");
263 return false;
264 }
265 preWinPos = winPos;
266 printf("XXX pick: mouse %s -> %s\n", winPos.toString().c_str(), objPos.toString().c_str());
267 printf("XXX pick: %s\n", shape.toString().c_str());
268 return true;
269 }
270
271 bool navigate(const PointerEvent& e, const WindowRef& win, GearsObjectES2& shape) noexcept {
272 jau::math::Vec3f objPos;
273 const jau::math::Vec2i& winPos = e.position();
274 if( !mapWinToObj(shape, winPos, objPos) ) {
275 return false;
276 }
277 if( e.isControlDown() ) {
278 if( m_dragInit ) {
279 m_dragInit = false;
280 startPos = objPos;
281 } else {
283 jau::math::Vec3f diffPos = ( objPos - startPos ).mul(flip);
284 m_parent.pan() += diffPos;
285 }
286 } else {
287 const jau::math::Vec2i& sdim = win->surfaceSize();
288 const float thetaY = 360.0f * ((float)(winPos.x - preWinPos.x) / (float)sdim.x);
289 const float thetaX = 360.0f * ((float)(preWinPos.y - winPos.y) / (float)sdim.y);
290 m_parent.rotEuler().x += jau::adeg_to_rad(thetaX);
291 m_parent.rotEuler().y += jau::adeg_to_rad(thetaY);
292 m_dragInit = true;
293 }
294 preWinPos = winPos;
295 // printf("XXX navi: mouse %s -> %s\n", winPos.toString().c_str(), objPos.toString().c_str());
296 return true;
297 }
298
299 PointerShapeAction pickAction, navigateAction;
300
301 public:
302 MyPointerListener(GearsES2& p): m_parent(p) {
303 pickAction = jau::bind_member(this, &MyPointerListener::pick);
304 navigateAction = jau::bind_member(this, &MyPointerListener::navigate);
305 }
306 void pointerPressed(PointerEvent& e) override {
307 if( e.pointerCount() == 1 ) {
308 WindowRef win = e.source().lock();
309 if( !win ) {
310 return;
311 }
312 GearsObjectES2* new_pick = m_parent.findPick(pickAction, e, win); // no matrix traversal
313 if( m_picked ) {
314 m_picked->picked() = false;
315 }
316 m_picked = new_pick;
317 if( m_picked ) {
318 m_picked->picked() = true;
319 }
320 }
321 }
322 void pointerDragged(PointerEvent& e) override {
323 if( m_picked ) {
324 WindowRef win = e.source().lock();
325 if( !win ) {
326 return;
327 }
328 if( !m_parent.dispatchForShape(*m_picked, navigateAction, e, win) ) { // matrix traversal
329 if( m_picked ) {
330 m_picked->picked() = false;
331 }
332 m_picked = nullptr;
333 printf("XXX shape: lost\n");
334 }
335 }
336 }
337 void pointerWheelMoved(PointerEvent& e) override {
338 const jau::math::Vec3f& rot = e.rotation();
339 if( e.isControlDown() ) {
340 // alternative zoom
341 float incr = e.isShiftDown() ? rot.x : rot.y * 0.5f;
342 m_parent.pan().z += incr;
343 } else {
344 // panning
345 m_parent.pan().x -= rot.x; // positive -> left
346 m_parent.pan().y += rot.y; // positive -> up
347 }
348 }
350 if( m_picked ) {
351 m_picked->picked() = false;
352 printf("XXX shape: released\n");
353 }
354 m_picked = nullptr;
355 m_dragInit = true;
356 }
357 };
358 typedef std::shared_ptr<MyPointerListener> MyPointerListenerRef;
359
360 /**@}*/
361
362} // namespace gamp::graph
363
364#endif /* JAU_GAMP_GRAPH_UI_SHAPE_HPP_ */
jau::function< bool(const PointerEvent &e, const WindowRef &win, GearsObjectES2 &shape)> PointerShapeAction
Definition GearsES2.hpp:49
GearsES2.
Definition GearsES2.hpp:378
GearsObjectES2.
Definition GearsES2.hpp:55
std::string toString() const noexcept
Definition GearsES2.hpp:371
const jau::math::geom::AABBox3f & objBounds() const noexcept
Definition GearsES2.hpp:367
const jau::math::Mat4f & matPMvi() const noexcept
Definition GearsES2.hpp:369
A Generic shape objects which is defined by a list of Outlines.
const AABBox3f & bounds() const noexcept
Container interface of UI Shapes.
Definition Container.hpp:59
void pointerReleased(PointerEvent &) override
Definition Shape.hpp:349
void pointerPressed(PointerEvent &e) override
Definition Shape.hpp:306
void pointerWheelMoved(PointerEvent &e) override
Traditional event name originally produced by a PointerType::mouse pointer type.
Definition Shape.hpp:337
void pointerDragged(PointerEvent &e) override
Definition Shape.hpp:322
constexpr void setColor(const Vec4f &c) noexcept
Definition Shape.hpp:145
jau::function< void(ShapeRef &s, const Vec3f &pos, const PointerEvent &e)> PointerEventCallback
Shape pointer listener, e.g.
Definition Shape.hpp:81
constexpr const Vec3f & position() const noexcept
Definition Shape.hpp:126
jau::function< bool(ShapeRef &s, PMVMat4f &pmv)> Visitor2Func
Visitor2 method.
Definition Shape.hpp:64
constexpr Quat4f & rotation() noexcept
Definition Shape.hpp:133
constexpr const Quat4f & rotation() const noexcept
Definition Shape.hpp:132
constexpr const Vec3f & scale() const noexcept
Definition Shape.hpp:138
constexpr Vec3f & rotationPivot() noexcept
Definition Shape.hpp:136
constexpr const float & zOffset() const noexcept
Definition Shape.hpp:129
virtual Container * asContainer() const noexcept
Definition Shape.hpp:124
constexpr const Vec4f & color() const noexcept
Definition Shape.hpp:144
jau::function< void(ShapeRef &s, const Vec3f &origin, const Vec3f &dest, const PointerEvent &e)> MoveEventCallback
Shape move listener
Definition Shape.hpp:73
jau::function< bool(ShapeRef &s)> Visitor1Func
Visitor1 method.
Definition Shape.hpp:56
constexpr const OutlineShape & outlines() const noexcept
Definition Shape.hpp:141
constexpr OutlineShape & outlines() noexcept
Definition Shape.hpp:142
constexpr Vec3f & scale() noexcept
Definition Shape.hpp:139
constexpr Vec3f & position() noexcept
Definition Shape.hpp:127
constexpr float & zOffset() noexcept
Definition Shape.hpp:130
constexpr const Vec3f & rotationPivot() const noexcept
Definition Shape.hpp:135
jau::function< void(ShapeRef &s)> ShapeEventCallback
General Shape listener action.
Definition Shape.hpp:86
void applyMatToMv(PMVMat4f &pmvMat) noexcept
Applies the internal Matrix4f to the given modelview matrix, i.e.
Definition Shape.hpp:161
constexpr bool isShiftDown() const noexcept
Returns true if modifier() contains InputModifier::shift.
Definition Event.hpp:285
constexpr bool isControlDown() const noexcept
Returns true if modifier() contains InputModifier::ctrl.
Definition Event.hpp:277
Pointer event of type PointerType.
constexpr const jau::math::Vec2i & position(size_t index=0) const noexcept
Returns position of given pointer-index in pixel units.
constexpr size_t pointerCount() const noexcept
See details for multiple-pointer events.
constexpr const jau::math::Vec3f & rotation() const noexcept
Returns a 3-component float array filled with the values of the rotational axis in the following orde...
Listener for PointerEvent.
constexpr const WindowWeakPtr & source() const noexcept
Definition Event.hpp:85
Class template jau::function is a general-purpose static-polymorphic function wrapper.
constexpr Matrix4 & translate(const value_type x, const value_type y, const value_type z) noexcept
Translate this matrix, i.e.
Definition mat4f.hpp:1445
constexpr_cxx26 Matrix4 & rotate(const value_type ang_rad, const value_type x, const value_type y, const value_type z) noexcept
Rotate this matrix about give axis and angle in radians, i.e.
Definition mat4f.hpp:1414
constexpr Matrix4 & scale(const value_type x, const value_type y, const value_type z) noexcept
Scale this matrix, i.e.
Definition mat4f.hpp:1467
static bool mapWinToAnyRay(const value_type winx, const value_type winy, const value_type winz0, const value_type winz1, const Matrix4 &invAny, const Recti &viewport, Ray3 &ray) noexcept
Definition mat4f.hpp:1915
constexpr Matrix4 & setToTranslation(const value_type x, const value_type y, const value_type z) noexcept
Set this matrix to translation.
Definition mat4f.hpp:913
constexpr bool isIdentity() const noexcept
Returns true if this quaternion has identity.
value_type x
Definition vec2i.hpp:65
std::string toString() const noexcept
Definition vec2i.hpp:211
value_type y
Definition vec2i.hpp:66
value_type x
Definition vec3f.hpp:69
static constexpr const value_type one
Definition vec3f.hpp:67
constexpr bool is_zero() const noexcept
Definition vec3f.hpp:241
value_type y
Definition vec3f.hpp:70
std::string toString() const noexcept
Definition vec3f.hpp:239
value_type z
Definition vec3f.hpp:71
Axis Aligned Bounding Box.
Definition aabbox3f.hpp:43
constexpr bool intersectsRay(const Ray3f &r) const noexcept
Check if Ray intersects this bounding box.
Definition aabbox3f.hpp:533
bool getRayIntersection(Vec3f &result, const Ray3f &ray, const float epsilon, const bool assumeIntersection) const noexcept
Return intersection of a Ray with this bounding box, or false if none exist.
Definition aabbox3f.hpp:578
constexpr const Point3f & center() const noexcept
Returns computed center of this aabbox3f of low() and high().
Definition aabbox3f.hpp:105
constexpr T adeg_to_rad(const T arc_degree) noexcept
Converts arc-degree to radians.
jau::function< R(A...)> bind_member(C1 *base, R(C0::*mfunc)(A...)) noexcept
Bind given class instance and non-void member function to an anonymous function using func_member_tar...
std::shared_ptr< MyPointerListener > MyPointerListenerRef
Definition Shape.hpp:358
std::shared_ptr< Shape > ShapeRef
Definition Container.hpp:49
std::shared_ptr< Window > WindowRef
Definition Event.hpp:36
Matrix4< float > Mat4f
Definition mat4f.hpp:1973
Vector2I< int > Vec2i
Definition vec2i.hpp:328
Ray3F< float > Ray3f
Definition vec3f.hpp:502
Vector4F< float > Vec4f
Definition vec4f.hpp:375
constexpr jau::math::Vec3f getEulerAngleOrientation(const jau::math::Vec3f &eulerRotation) noexcept
Returns an orientation vector for given eurler X/Y/Z angles in radians.
Quaternion< float > Quat4f
Vector3F< float > Vec3f
Definition vec3f.hpp:436
PMVMatrix4< float > PMVMat4f
int printf(const char *format,...)
Operating Systems predefined macros.