Gamp v0.0.7-36-g24b1eb6
Gamp: Graphics, Audio, Multimedia and Processing
Loading...
Searching...
No Matches
Scene.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_SCENE_HPP_
12#define JAU_GAMP_GRAPH_UI_SCENE_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/vec2i.hpp>
23#include <jau/math/vec3f.hpp>
24#include <jau/math/vec4f.hpp>
25#include <jau/math/vec4f.hpp>
27
28#include <gamp/Gamp.hpp>
34
35namespace gamp::graph::ui {
36
37 using namespace jau::math;
38 using namespace jau::math::util;
39 using namespace jau::math::geom;
40
41 using namespace gamp::wt;
42 using namespace gamp::wt::event;
43
44 using namespace gamp::graph;
45
46 /** @defgroup Gamp_GraphUI Gamp Graph UI
47 * Graph UI Framework
48 *
49 * @{
50 */
51
52 class Scene {
53 private:
54 ShapeRef m_activeShape;
55
56 public:
58 { }
59
60 private:
61 /**
62 * General {@link Shape} visitor
63 * @param s the {@link Shape} to process
64 * @param pmv the {@link PMVMatrix4f} setup from the {@link Scene} down to the {@link Shape}
65 * @return the picked shape signaling end of traversal, otherwise null to continue
66 */
67 typedef jau::function<ShapeRef(ShapeRef& s, PMVMat4f& pmv)> PickVisitorFunc;
68
69 static ShapeRef pickForAllRenderedDesc(Container& cont, PMVMat4f& pmv, const PickVisitorFunc& visitor) {
70 ShapeRef picked = nullptr;
71 ShapeList& shapes = cont.renderedShapes();
72 // synchronized( shapes ) { // tripple-buffering is just almost enough FIXME
73 for(size_t i=shapes.size(); !picked && i-- > 0; ) {
74 ShapeRef& s = shapes[i];
75 pmv.pushMv();
76 s->applyMatToMv(pmv);
77 picked = visitor(s, pmv);
78 if( s->asContainer() ) {
79 ShapeRef childPick = pickForAllRenderedDesc(*s->asContainer(), pmv, visitor);
80 if( childPick ) {
81 picked = childPick; // child picked overrides group parent!
82 }
83 }
84 pmv.popMv();
85 }
86 // }
87 return picked;
88 }
89
90 /**
91 * Attempt to pick a {@link Shape} using the window coordinates and contained {@ling Shape}'s {@link AABBox} {@link Shape#getBounds() bounds}
92 * using a ray-intersection algorithm in Z-axis descending order.
93 * <p>
94 * If {@link Shape} was found the given action is performed.
95 * </p>
96 * <p>
97 * Method performs on current thread and returns after either a shaper is determined to be picked/active or
98 * probing every {@link Shape} w/o result.
99 * </p>
100 * @param pmv a new {@link PMVMatrix4f} which will {@link Scene.PMVMatrixSetup#set(PMVMatrix4f, Recti) be setup},
101 * {@link Shape#applyMatToMv(PMVMatrix4f) shape-transformed} and can be reused by the caller and runnable.
102 * @param ray temporary {@link Ray} storage, passed for reusage
103 * @param glWinX window X coordinate, bottom-left origin
104 * @param glWinY window Y coordinate, bottom-left origin
105 * @param objPos storage for found object position in model-space if found {@link Shape}
106 * @param runnable the action to perform if {@link Shape} was found
107 * @return last picked (inner) Shape if any or null
108 */
109 ShapeRef pickShape(PMVMat4f& pmv, const Vec2i& winPos, Vec3f& objPos, final Shape.Visitor1 visitor) {
110 setupMatrix(pmv);
111
112 final float winZ0 = 0f;
113 final float winZ1 = 0.3f;
114 /**
115 final FloatBuffer winZRB = Buffers.newDirectFloatBuffer(1);
116 gl.glReadPixels( x, y, 1, 1, GL2ES2.GL_DEPTH_COMPONENT, GL.GL_FLOAT, winZRB);
117 winZ1 = winZRB.get(0); // dir
118 */
119 final Recti viewport = getViewport();
120 final int[] shapeIdx = { -1 };
121 return pickForAllRenderedDesc(this, pmv, (final Shape s, final PMVMatrix4f pmv2) -> {
122 shapeIdx[0]++;
123 if( pmv.mapWinToRay(glWinX, glWinY, winZ0, winZ1, viewport, ray) ) {
124 final AABBox sbox = s.getBounds();
125 if( sbox.intersectsRay(ray) ) {
126 if( null == sbox.getRayIntersection(objPos, ray, FloatUtil.EPSILON, true) ) {
127 throw new InternalError("Ray "+ray+", box "+sbox);
128 }
129 if( visitor.visit(s) ) {
130 return s;
131 }
132 }
133 }
134 return null;
135 });
136 }
137 /**
138 * Pick the shape using the event coordinates
139 * @param e original Newt {@link MouseEvent}
140 * @param glWinX in GL window coordinates, origin bottom-left
141 * @param glWinY in GL window coordinates, origin bottom-left
142 */
143 private final Shape dispatchMouseEventPickShape(final MouseEvent e, final int glWinX, final int glWinY) {
144 final Shape shape = pickShape(dispMEPSPMv, dispMEPSRay, glWinX, glWinY, dispMEPSObjPos, (final Shape s) -> {
145 // return s.isInteractive() && ( s.dispatchMouseEvent(e, glWinX, glWinY, dispMEPSObjPos) || true );
146 if( !s.isInteractive() ) {
147 if( DEBUG_PICKING ) {
148 System.err.printf("Pick.X.0: shape %s/%s, [%d, %d]%n", s.getClass().getSimpleName(), s.getName(), glWinX, glWinY);
149 }
150 return false;
151 }
152 if( !s.dispatchMouseEvent(e, glWinX, glWinY, dispMEPSObjPos) ) {
153 if( DEBUG_PICKING ) {
154 System.err.printf("Pick.X.1: shape %s/%s, [%d, %d], %s%n", s.getClass().getSimpleName(), s.getName(), glWinX, glWinY, e);
155 }
156 return false;
157 }
158 if( DEBUG_PICKING ) {
159 System.err.printf("Pick.X.S: shape %s/%s, [%d, %d], %s%n", s.getClass().getSimpleName(), s.getName(), glWinX, glWinY, e);
160 }
161 return true;
162 });
163 if( null != shape ) {
164 if( DEBUG_PICKING ) {
165 System.err.printf("Pick.X: shape %s/%s%n%n", shape.getClass().getSimpleName(), shape.getName());
166 }
167 setActiveShape(shape);
168 return shape;
169 } else {
170 if( DEBUG_PICKING ) {
171 System.err.printf("Pick.X: shape null%n%n");
172 }
173 releaseActiveShape();
174 return null;
175 }
176 }
177 private final PMVMatrix4f dispMEPSPMv = new PMVMatrix4f();
178 private final Ray dispMEPSRay = new Ray();
179 private final Vec3f dispMEPSObjPos = new Vec3f();
180
181 /**
182 * Dispatch event to shape
183 * @param shape target active shape of event
184 * @param e original Newt {@link MouseEvent}
185 * @param glWinX in GL window coordinates, origin bottom-left
186 * @param glWinY in GL window coordinates, origin bottom-left
187 */
188 private final void dispatchMouseEventForShape(final Shape shape, final MouseEvent e, final int glWinX, final int glWinY) {
189 final PMVMatrix4f pmv = new PMVMatrix4f();
190 final Vec3f objPos = new Vec3f();
191 winToShapeCoord(shape, glWinX, glWinY, pmv, objPos, () -> { shape.dispatchMouseEvent(e, glWinX, glWinY, objPos); });
192 if( DEBUG_PICKING ) {
193 System.err.printf("ForShape: shape %s/%s%n%n", shape.getClass().getSimpleName(), shape.getName());
194 }
195 }
196
197 class SBCPointerListener : public PointerListener {
198 private:
199 Scene& scene;
200 Vec2i lastPos;
201 int lastId;
202 bool mouseOver;
203
204 void clear() {
205 lastPos.x = -1; lastPos.y = -1; lastId = -1; mouseOver = false;
206 }
207 const ShapeRef& dispatchPickShape(const PointerEvent& e) {
208 const ShapeRef s = dispatchMouseEventPickShape(e);
209 if( !s ) {
210 clear();
211 }
212 return s;
213 }
214 public:
215 SBCPointerListener(Scene& p): scene(p) { clear(); }
216
217 void pointerPressed(PointerEvent& e) override {
218 if( -1 == lastId || e.pointerId() == lastId ) {
219 lastPos = e.position();
220 lastId = e.pointerId();
221 }
222 // Can't use selected activeShape via mouseOver,
223 // since mouseMove and mousePressed/Drag or mouseClicked may-shall select a different shape
224 // based on draggable. mouseMove simply activates any shape.
225 dispatchPickShape(e);
226 }
227
228 void pointerReleased(PointerEvent& e) override {
229 if( mouseOver && scene.m_activeShape &&
230 scene.m_activeShape.isInteractive() &&
231 // !pinchToZoomGesture.isWithinGesture() &&
232 e.pointerId() == lastId )
233 {
234 scene.dispatchMouseEventForShape(scene.m_activeShape, e);
235 } else {
236 dispatchPickShape(e);
237 }
238 if( !mouseOver && 1 == e.pointerCount() ) {
239 // Release active shape: last pointer has been lifted!
240 scene.releaseActiveShape();
241 clear();
242 }
243 }
244
245 void pointerClicked(PointerEvent& e) override {
246 // Can't use selected activeShape via mouseOver,
247 // since mouseMove and mousePressed/Drag or mouseClicked may-shall select a different shape
248 // based on draggable. mouseMove simply activates any shape.
249 dispatchPickShape(e);
250 }
251
252 void pointerDragged(PointerEvent& e) override {
253 scene.clearToolTip();
254 // drag activeShape, if no gesture-activity, only on 1st pointer
255 if( scene.m_activeShape &&
256 scene.m_activeShape.isInteractive() &&
257 // !pinchToZoomGesture.isWithinGesture() &&
258 e.pointerId() == lastId )
259 {
260 lastPos = e.position();
261 scene.dispatchMouseEventForShape(scene.m_activeShape, e);
262 }
263 }
264
265 void pointerWheelMoved(PointerEvent&) override {
266 scene.clearToolTip();
267 if( mouseOver && scene.m_activeShape &&
268 scene.m_activeShape.isInteractive() &&
269 // !pinchToZoomGesture.isWithinGesture() &&
270 e.pointerId() == lastId ) {
271 scene.dispatchMouseEventForShape(scene.m_activeShape, e);
272 } else {
273 dispatchPickShape(e);
274 }
275 }
276
277 void pointerMoved(PointerEvent&) override {
278 if( -1 == lastId || e.pointerId() == lastId ) {
279 lastPos = e.position();
280 lastId = e.getPointerId(0);
281 }
282 scene.clearToolTip();
283 const ShapeRef& s = dispatchPickShape(e);
284 if( s ) {
285 mouseOver = true;
286 // synchronized( toolTipActive ) {
287 // FIXME toolTipActive.set( s.startToolTip(true /* lookupParents */) );
288 // }
289 }
290 }
291 void pointerExited(PointerEvent&) override {
292 scene.clearToolTip();
293 scene.releaseActiveShape();
294 clear();
295 }
296 };
297
298 class SBCKeyListener : public KeyListener {
299 private:
300 Scene& m_parent;
301
302 public:
303 SBCKeyListener(Scene& p): m_parent(p) { }
304
305 void keyPressed(KeyEvent& e, const KeyboardTracker& kt) override {
306 if( m_parent.m_activeShape && m_parent.activeShape.isInteractive() ) {
307 m_parent.activeShape.dispatchKeyEvent(e, kt);
308 }
309 }
310 void keyReleased(KeyEvent& e, const KeyboardTracker& kt) override {
311 if( m_parent.m_activeShape && m_parent.activeShape.isInteractive() ) {
312 m_parent.activeShape.dispatchKeyEvent(e, kt);
313 }
314 }
315 };
316 typedef std::shared_ptr<SBCKeyListener> SBCKeyListenerRef;
317
318 };
319
320 /**@}*/
321
322} // namespace gamp::graph
323
324#endif /* JAU_GAMP_GRAPH_UI_SCENE_HPP_ */
Container interface of UI Shapes.
Definition Container.hpp:59
virtual ShapeList & renderedShapes() const noexcept=0
Returns added shapes which are rendered and sorted by z-axis in ascending order toward z-near.
constexpr int pointerId(size_t index=0) const noexcept
Return the pointer id for the given index or -1 if index not available.
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.
Class template jau::function is a general-purpose static-polymorphic function wrapper.
value_type x
Definition vec2i.hpp:65
constexpr_cxx20 PMVMatrix4 & popMv() noexcept
Pop the modelview matrix from its stack.
Definition pmvmat4f.hpp:787
constexpr_cxx20 PMVMatrix4 & pushMv() noexcept
Push the modelview matrix to its stack, while preserving its values.
Definition pmvmat4f.hpp:805
@ null
Denotes a func::null_target_t.
std::vector< ShapeRef > ShapeList
Definition Container.hpp:50
std::shared_ptr< Shape > ShapeRef
Definition Container.hpp:49
Vector2I< int > Vec2i
Definition vec2i.hpp:328
RectI< int > Recti
Definition recti.hpp:139
Vector3F< float > Vec3f
Definition vec3f.hpp:436
PMVMatrix4< float > PMVMat4f