73 for(
size_t i=shapes.size(); !picked && i-- > 0; ) {
77 picked = visitor(s, pmv);
78 if( s->asContainer() ) {
79 ShapeRef childPick = pickForAllRenderedDesc(*s->asContainer(), pmv, visitor);
112 final float winZ0 = 0f;
113 final float winZ1 = 0.3f;
119 final Recti viewport = getViewport();
120 final int[] shapeIdx = { -1 };
121 return pickForAllRenderedDesc(
this, pmv, (
final Shape s,
final PMVMatrix4f pmv2) -> {
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);
129 if( visitor.visit(s) ) {
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) -> {
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);
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);
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);
163 if( null != shape ) {
164 if( DEBUG_PICKING ) {
165 System.err.printf(
"Pick.X: shape %s/%s%n%n", shape.getClass().getSimpleName(), shape.getName());
167 setActiveShape(shape);
170 if( DEBUG_PICKING ) {
171 System.err.printf(
"Pick.X: shape null%n%n");
173 releaseActiveShape();
177 private final PMVMatrix4f dispMEPSPMv =
new PMVMatrix4f();
178 private final Ray dispMEPSRay =
new Ray();
179 private final Vec3f dispMEPSObjPos =
new Vec3f();
188 private final void dispatchMouseEventForShape(
final Shape shape,
final MouseEvent e,
final int glWinX,
final int glWinY) {
189 final PMVMatrix4f pmv =
new PMVMatrix4f();
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());
197 class SBCPointerListener :
public PointerListener {
205 lastPos.
x = -1; lastPos.y = -1; lastId = -1; mouseOver =
false;
207 const ShapeRef& dispatchPickShape(
const PointerEvent& e) {
208 const ShapeRef s = dispatchMouseEventPickShape(e);
215 SBCPointerListener(
Scene& p): scene(p) { clear(); }
217 void pointerPressed(PointerEvent& e)
override {
218 if( -1 == lastId || e.
pointerId() == lastId ) {
225 dispatchPickShape(e);
228 void pointerReleased(PointerEvent& e)
override {
229 if( mouseOver && scene.m_activeShape &&
230 scene.m_activeShape.isInteractive() &&
234 scene.dispatchMouseEventForShape(scene.m_activeShape, e);
236 dispatchPickShape(e);
240 scene.releaseActiveShape();
245 void pointerClicked(PointerEvent& e)
override {
249 dispatchPickShape(e);
252 void pointerDragged(PointerEvent& e)
override {
253 scene.clearToolTip();
255 if( scene.m_activeShape &&
256 scene.m_activeShape.isInteractive() &&
261 scene.dispatchMouseEventForShape(scene.m_activeShape, e);
265 void pointerWheelMoved(PointerEvent&)
override {
266 scene.clearToolTip();
267 if( mouseOver && scene.m_activeShape &&
268 scene.m_activeShape.isInteractive() &&
271 scene.dispatchMouseEventForShape(scene.m_activeShape, e);
273 dispatchPickShape(e);
277 void pointerMoved(PointerEvent&)
override {
278 if( -1 == lastId || e.
pointerId() == lastId ) {
280 lastId = e.getPointerId(0);
282 scene.clearToolTip();
283 const ShapeRef& s = dispatchPickShape(e);
291 void pointerExited(PointerEvent&)
override {
292 scene.clearToolTip();
293 scene.releaseActiveShape();
298 class SBCKeyListener :
public KeyListener {
303 SBCKeyListener(
Scene& p): m_parent(p) { }
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);
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);
316 typedef std::shared_ptr<SBCKeyListener> SBCKeyListenerRef;