2 * Copyright (c) 2024 Samsung Electronics Co., Ltd.
4 * Licensed under the Apache License, Version 2.0 (the "License");
5 * you may not use this file except in compliance with the License.
6 * You may obtain a copy of the License at
8 * http://www.apache.org/licenses/LICENSE-2.0
10 * Unless required by applicable law or agreed to in writing, software
11 * distributed under the License is distributed on an "AS IS" BASIS,
12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13 * See the License for the specific language governing permissions and
14 * limitations under the License.
19 #include <dali/internal/event/events/touch-event-processor.h>
21 #if defined(DEBUG_ENABLED)
26 #include <dali/integration-api/debug.h>
27 #include <dali/integration-api/events/touch-event-integ.h>
28 #include <dali/integration-api/trace.h>
29 #include <dali/internal/event/actors/actor-impl.h>
30 #include <dali/internal/event/actors/layer-impl.h>
31 #include <dali/internal/event/common/scene-impl.h>
32 #include <dali/internal/event/events/hit-test-algorithm-impl.h>
33 #include <dali/internal/event/events/multi-point-event-util.h>
34 #include <dali/internal/event/events/touch-event-impl.h>
35 #include <dali/internal/event/render-tasks/render-task-impl.h>
36 #include <dali/public-api/events/touch-event.h>
37 #include <dali/public-api/math/vector2.h>
38 #include <dali/public-api/signals/callback.h>
40 namespace Dali::Internal
44 DALI_INIT_TRACE_FILTER(gTraceFilter, DALI_TRACE_PERFORMANCE_MARKER, false);
45 #if defined(DEBUG_ENABLED)
46 Debug::Filter* gLogFilter = Debug::Filter::New(Debug::NoLogging, false, "LOG_TOUCH_PROCESSOR");
47 #endif // defined(DEBUG_ENABLED)
50 * Structure for Variables used in the ProcessTouchEvent method.
52 struct ProcessTouchEventVariables
54 ProcessTouchEventVariables(TouchEventProcessor& eventProcessor, bool geometry)
55 : processor(eventProcessor),
60 TouchEventProcessor& processor; ///< A handle to the touch-event-processor.
61 const bool isGeometry; ///< Whether it's a geometry or not.
62 Actor* lastPrimaryHitActor{nullptr}; ///< The last primary hit-actor.
63 Actor* lastConsumedActor{nullptr}; ///< The last consuming actor.
64 TouchEventPtr touchEventImpl; ///< The current touch-event-impl.
65 Dali::TouchEvent touchEventHandle; ///< The handle to the touch-event-impl.
66 RenderTaskPtr currentRenderTask; ///< The current render-task.
67 Dali::Actor consumedActor; ///< The actor that consumed the event.
68 Dali::Actor primaryHitActor; ///< The actor that has been hit by the primary point.
69 Integration::Point* primaryPoint{nullptr}; ///< The primary point of the hit.
70 PointState::Type primaryPointState{PointState::STARTED}; ///< The state of the primary point.
73 const char* TOUCH_POINT_STATE[6] =
83 bool ShouldEmitInterceptTouchEvent(const Actor& actorImpl, const Dali::TouchEvent& event)
85 PointState::Type state = event.GetState(0);
86 return actorImpl.GetInterceptTouchRequired() && (actorImpl.IsHittable() || state == PointState::INTERRUPTED ) && (state != PointState::MOTION || actorImpl.IsDispatchTouchMotion());
89 bool ShouldEmitTouchEvent(const Actor& actorImpl, const Dali::TouchEvent& event)
91 PointState::Type state = event.GetState(0);
92 return actorImpl.GetTouchRequired() && (actorImpl.IsHittable() || state == PointState::INTERRUPTED ) && (state != PointState::MOTION || actorImpl.IsDispatchTouchMotion());
96 Dali::Actor EmitInterceptTouchSignals(Dali::Actor actor, const Dali::TouchEvent& touchEvent)
98 Dali::Actor interceptedActor;
102 Dali::Actor parent = actor.GetParent();
105 // Recursively deliver events to the actor and its parents for intercept touch event.
106 interceptedActor = EmitInterceptTouchSignals(parent, touchEvent);
109 if(!interceptedActor)
111 bool intercepted = false;
112 Actor& actorImpl(GetImplementation(actor));
113 if(ShouldEmitInterceptTouchEvent(actorImpl, touchEvent))
115 DALI_TRACE_SCOPE(gTraceFilter, "DALI_EMIT_INTERCEPT_TOUCH_EVENT_SIGNAL");
116 intercepted = actorImpl.EmitInterceptTouchEventSignal(touchEvent);
119 interceptedActor = Dali::Actor(&actorImpl);
124 return interceptedActor;
129 Dali::Actor EmitGeoInterceptTouchSignals(std::list<Dali::Internal::Actor*>& actorLists, std::list<Dali::Internal::Actor*>& interceptActorList, const Dali::TouchEvent& touchEvent, ActorObserver& lastConsumedActor)
131 interceptActorList.clear();
132 Dali::Actor interceptedActor;
133 for(auto&& actor : actorLists)
135 interceptActorList.push_back(actor);
136 if(ShouldEmitInterceptTouchEvent(*actor, touchEvent))
138 DALI_TRACE_SCOPE(gTraceFilter, "DALI_EMIT_INTERCEPT_TOUCH_EVENT_SIGNAL");
139 if(actor->EmitInterceptTouchEventSignal(touchEvent))
141 interceptedActor = Dali::Actor(actor);
145 // If there is a consumed actor, the intercept is sent only up to the moment before the consumed actor.
146 if(lastConsumedActor.GetActor() == actor)
151 return interceptedActor;
155 * Recursively deliver events to the actor and its parents, until the event is consumed or the stage is reached.
157 Dali::Actor EmitTouchSignals(Dali::Actor actor, const Dali::TouchEvent& touchEvent)
159 Dali::Actor consumedActor;
163 Dali::Actor oldParent(actor.GetParent());
165 Actor& actorImpl(GetImplementation(actor));
167 bool consumed(false);
169 // Only emit the signal if the actor's touch signal has connections (or derived actor implementation requires touch).
170 if(ShouldEmitTouchEvent(actorImpl, touchEvent))
172 DALI_TRACE_SCOPE(gTraceFilter, "DALI_EMIT_TOUCH_EVENT_SIGNAL");
173 consumed = actorImpl.EmitTouchEventSignal(touchEvent);
178 // One of this actor's listeners has consumed the event so set this actor as the consumed actor.
179 consumedActor = Dali::Actor(&actorImpl);
183 // The actor may have been removed/reparented during the signal callbacks.
184 Dali::Actor parent = actor.GetParent();
187 (parent == oldParent))
189 // One of the actor's parents may consumed the event and they should be set as the consumed actor.
190 consumedActor = EmitTouchSignals(parent, touchEvent);
195 return consumedActor;
199 * Recursively deliver events to the actor and its below actor, until the event is consumed or the stage is reached.
201 Dali::Actor EmitGeoTouchSignals(std::list<Dali::Internal::Actor*>& actorLists, const Dali::TouchEvent& touchEvent)
203 Dali::Actor consumedActor;
205 std::list<Dali::Internal::Actor*>::reverse_iterator rIter = actorLists.rbegin();
206 for(; rIter != actorLists.rend(); rIter++)
208 Actor* actorImpl(*rIter);
209 // Only emit the signal if the actor's touch signal has connections (or derived actor implementation requires touch).
210 if(ShouldEmitTouchEvent(*actorImpl, touchEvent))
212 DALI_TRACE_SCOPE(gTraceFilter, "DALI_EMIT_TOUCH_EVENT_SIGNAL");
213 if(actorImpl->EmitTouchEventSignal(touchEvent))
215 // One of this actor's listeners has consumed the event so set this actor as the consumed actor.
216 consumedActor = Dali::Actor(actorImpl);
221 return consumedActor;
224 Dali::Actor AllocAndEmitTouchSignals(unsigned long time, Dali::Actor actor, const Integration::Point& point, RenderTaskPtr renderTask)
226 TouchEventPtr touchEvent(new TouchEvent(time));
227 Dali::TouchEvent touchEventHandle(touchEvent.Get());
229 touchEvent->AddPoint(point);
230 touchEvent->SetRenderTask(Dali::RenderTask(renderTask.Get()));
232 return EmitTouchSignals(actor, touchEventHandle);
235 Dali::Actor GeoAllocAndEmitTouchSignals(std::list<Dali::Internal::Actor*>& actorLists, unsigned long time, const Integration::Point& point, RenderTaskPtr renderTask)
237 TouchEventPtr touchEvent(new TouchEvent(time));
238 Dali::TouchEvent touchEventHandle(touchEvent.Get());
240 touchEvent->AddPoint(point);
241 touchEvent->SetRenderTask(Dali::RenderTask(renderTask.Get()));
243 return EmitGeoTouchSignals(actorLists, touchEventHandle);
247 * Changes the state of the primary point to leave and emits the touch signals
249 Dali::Actor EmitTouchSignals(Actor* actor, RenderTask& renderTask, const TouchEventPtr& originalTouchEvent, PointState::Type state, bool isGeometry)
251 Dali::Actor consumingActor;
255 TouchEventPtr touchEventImpl = TouchEvent::Clone(*originalTouchEvent.Get());
256 touchEventImpl->SetRenderTask(Dali::RenderTask(&renderTask));
258 Integration::Point& primaryPoint = touchEventImpl->GetPoint(0);
260 const Vector2& screenPosition = primaryPoint.GetScreenPosition();
261 Vector2 localPosition;
262 actor->ScreenToLocal(renderTask, localPosition.x, localPosition.y, screenPosition.x, screenPosition.y);
264 primaryPoint.SetLocalPosition(localPosition);
265 primaryPoint.SetHitActor(Dali::Actor(actor));
266 primaryPoint.SetState(state);
270 std::list<Dali::Internal::Actor*> actorLists;
271 actorLists.push_back(actor);
272 consumingActor = EmitGeoTouchSignals(actorLists, Dali::TouchEvent(touchEventImpl.Get()));
276 consumingActor = EmitTouchSignals(Dali::Actor(actor), Dali::TouchEvent(touchEventImpl.Get()));
280 return consumingActor;
284 * @brief Parses the primary touch point by performing a hit-test if necessary
286 * @param[out] hitTestResults The hit test results are put into this variable
287 * @param[in/out] capturingTouchActorObserver The observer for the capturing touch actor member
288 * @param[in] lastRenderTask The last render task member
289 * @param[in] currentPoint The current point information
290 * @param[in] scene The scene that this touch is related to
291 * @param[in] actorLists The list of actors that can be touched, from leaf actor to root.
293 void ParsePrimaryTouchPoint(
294 HitTestAlgorithm::Results& hitTestResults,
295 ActorObserver& capturingTouchActorObserver,
296 ActorObserver& ownTouchActorObserver,
297 const RenderTaskPtr& lastRenderTask,
298 const Integration::Point& currentPoint,
299 const Internal::Scene& scene,
300 std::list<Dali::Internal::Actor*>& actorLists)
302 Actor* capturingTouchActor = capturingTouchActorObserver.GetActor();
304 // We only set the capturing touch actor when the first touch-started actor captures all touch so if it's set, just use it
305 if(capturingTouchActor && lastRenderTask)
307 hitTestResults.actor = Dali::Actor(capturingTouchActor);
308 hitTestResults.renderTask = lastRenderTask;
309 const Vector2& screenPosition = currentPoint.GetScreenPosition();
310 capturingTouchActor->ScreenToLocal(*lastRenderTask, hitTestResults.actorCoordinates.x, hitTestResults.actorCoordinates.y, screenPosition.x, screenPosition.y);
314 Actor* ownTouchActor = ownTouchActorObserver.GetActor();
315 HitTestAlgorithm::HitTest(scene.GetSize(), scene.GetRenderTaskList(), scene.GetLayerList(), currentPoint.GetScreenPosition(), hitTestResults, ownTouchActor, scene.IsGeometryHittestEnabled());
317 if(currentPoint.GetState() == PointState::STARTED && hitTestResults.actor)
319 bool isGeometry = scene.IsGeometryHittestEnabled();
320 // If we've just started touch, then check whether the actor has requested to capture all touch events
321 Actor* hitActor = &GetImplementation(hitTestResults.actor);
322 if(hitActor->CapturesAllTouchAfterStart() || isGeometry)
324 capturingTouchActorObserver.SetActor(hitActor);
326 if(hitActor->IsAllowedOnlyOwnTouch() || isGeometry)
328 ownTouchActorObserver.SetActor(hitActor);
332 actorLists = hitTestResults.actorLists;
338 } // unnamed namespace
340 struct TouchEventProcessor::Impl
343 * Emits an interrupted event.
344 * @param[in/out] localVars The struct of stack variables used by ProcessTouchEvent
345 * @param[in] event The touch event that has occurred
347 static inline void EmitInterruptedEvent(ProcessTouchEventVariables& localVars, const Integration::TouchEvent& event)
349 Dali::Actor consumingActor;
350 Integration::Point currentPoint(event.points[0]);
352 if(localVars.isGeometry)
354 // Since the geometry way only receives touch events from the consumed actor,
355 // it first searches for whether there is a consumed actor and then sends the event
356 Actor* touchConsumedActor(localVars.processor.mLastConsumedActor.GetActor());
357 Actor* touchDownConsumedActor(localVars.processor.mTouchDownConsumedActor.GetActor());
358 Actor* lastPrimaryHitActor(localVars.processor.mLastPrimaryHitActor.GetActor());
359 if(touchConsumedActor)
361 Dali::Actor touchConsumedActorHandle(touchConsumedActor);
362 currentPoint.SetHitActor(touchConsumedActorHandle);
363 std::list<Dali::Internal::Actor*> actorLists;
364 actorLists.push_back(touchConsumedActor);
365 GeoAllocAndEmitTouchSignals(actorLists, event.time, currentPoint, localVars.processor.mLastRenderTask);
367 else if(touchDownConsumedActor)
369 Dali::Actor touchDownConsumedActorHandle(touchDownConsumedActor);
370 currentPoint.SetHitActor(touchDownConsumedActorHandle);
371 std::list<Dali::Internal::Actor*> actorLists;
372 actorLists.push_back(touchDownConsumedActor);
373 GeoAllocAndEmitTouchSignals(actorLists, event.time, currentPoint, localVars.processor.mLastRenderTask);
375 else if(lastPrimaryHitActor)
377 Dali::Actor lastPrimaryHitActorHandle(lastPrimaryHitActor);
378 currentPoint.SetHitActor(lastPrimaryHitActorHandle);
380 GeoAllocAndEmitTouchSignals(localVars.processor.mCandidateActorLists, event.time, currentPoint, localVars.processor.mLastRenderTask);
385 Actor* lastPrimaryHitActor(localVars.processor.mLastPrimaryHitActor.GetActor());
386 if(lastPrimaryHitActor)
388 Dali::Actor lastPrimaryHitActorHandle(lastPrimaryHitActor);
389 currentPoint.SetHitActor(lastPrimaryHitActorHandle);
391 consumingActor = AllocAndEmitTouchSignals(event.time, lastPrimaryHitActorHandle, currentPoint, localVars.processor.mLastRenderTask);
394 // If the last consumed actor was different to the primary hit actor then inform it as well (if it has not already been informed).
395 Actor* lastConsumedActor(localVars.processor.mLastConsumedActor.GetActor());
396 if(lastConsumedActor &&
397 lastConsumedActor != lastPrimaryHitActor &&
398 lastConsumedActor != consumingActor)
400 Dali::Actor lastConsumedActorHandle(lastConsumedActor);
401 currentPoint.SetHitActor(lastConsumedActorHandle);
402 AllocAndEmitTouchSignals(event.time, lastConsumedActorHandle, currentPoint, localVars.processor.mLastRenderTask);
405 // Tell the touch-down consuming actor as well, if required
406 Actor* touchDownConsumedActor(localVars.processor.mTouchDownConsumedActor.GetActor());
407 if(touchDownConsumedActor &&
408 touchDownConsumedActor != lastPrimaryHitActor &&
409 touchDownConsumedActor != lastConsumedActor &&
410 touchDownConsumedActor != consumingActor)
412 Dali::Actor touchDownConsumedActorHandle(touchDownConsumedActor);
414 currentPoint.SetHitActor(touchDownConsumedActorHandle);
415 AllocAndEmitTouchSignals(event.time, touchDownConsumedActorHandle, currentPoint, localVars.processor.mLastRenderTask);
419 localVars.processor.Clear();
420 localVars.processor.mTouchDownConsumedActor.SetActor(nullptr);
422 currentPoint.SetHitActor(Dali::Actor());
424 TouchEventPtr touchEventImpl(new TouchEvent(event.time));
425 Dali::TouchEvent touchEventHandle(touchEventImpl.Get());
427 touchEventImpl->AddPoint(currentPoint);
429 localVars.processor.mScene.EmitTouchedSignal(touchEventHandle);
433 * Recursively deliver events to the actor and its parents, until the event is consumed or the stage is reached.
434 * @param[in/out] localVars The struct of stack variables used by ProcessTouchEvent
435 * @return True if consumed, false otherwise.
437 static inline bool DeliverEventsToActorAndParents(ProcessTouchEventVariables& localVars)
439 bool consumed = false;
440 TouchEventProcessor& processor = localVars.processor;
441 if(localVars.currentRenderTask)
443 if(localVars.isGeometry)
445 Actor* interceptedTouchActor(processor.mInterceptedTouchActor.GetActor());
446 if(interceptedTouchActor)
448 Actor* touchConsumedActor(processor.mLastConsumedActor.GetActor());
449 if(touchConsumedActor) // If there is a consumed actor, send events only to the consumed actor.
451 RenderTask& currentRenderTaskImpl = *localVars.currentRenderTask.Get();
452 localVars.consumedActor = EmitTouchSignals(touchConsumedActor, currentRenderTaskImpl, localVars.touchEventImpl, localVars.primaryPointState, localVars.isGeometry);
454 else // If there is an intercepted actor, send a touch event starting from the intercepted actor.
456 Dali::Actor interceptedTouchActorHandle(interceptedTouchActor);
457 std::list<Dali::Internal::Actor*> interceptActorLists = localVars.processor.mInterceptedActorLists;
458 localVars.consumedActor = EmitGeoTouchSignals(interceptActorLists, localVars.touchEventHandle);
463 Dali::Actor interceptedActor;
464 // Let's find out if there is an intercept actor.
465 interceptedActor = EmitGeoInterceptTouchSignals(processor.mCandidateActorLists, processor.mInterceptedActorLists, localVars.touchEventHandle, processor.mLastConsumedActor);
468 processor.mInterceptedTouchActor.SetActor(&GetImplementation(interceptedActor));
470 // If there is an interception, send an interrupted event to the last consumed actor or to the actor that hit previously.
471 if(processor.mLastConsumedActor.GetActor() &&
472 processor.mLastConsumedActor.GetActor() != interceptedActor &&
473 processor.mLastRenderTask &&
474 processor.mLastPrimaryPointState != PointState::FINISHED)
476 EmitTouchSignals(processor.mLastConsumedActor.GetActor(), *processor.mLastRenderTask.Get(), localVars.touchEventImpl, PointState::INTERRUPTED, localVars.isGeometry);
477 processor.mTouchDownConsumedActor.SetActor(nullptr);
478 processor.mLastConsumedActor.SetActor(nullptr);
480 else if(processor.mLastPrimaryHitActor.GetActor() &&
481 processor.mLastPrimaryHitActor.GetActor() != interceptedActor &&
482 processor.mLastRenderTask &&
483 processor.mLastPrimaryPointState != PointState::FINISHED)
485 std::list<Dali::Internal::Actor*>::reverse_iterator rIter = processor.mCandidateActorLists.rbegin();
486 for(; rIter != processor.mCandidateActorLists.rend(); rIter++)
488 Actor* actorImpl(*rIter);
489 if(actorImpl == interceptedActor)
493 EmitTouchSignals(actorImpl, *processor.mLastRenderTask.Get(), localVars.touchEventImpl, PointState::INTERRUPTED, localVars.isGeometry);
498 Actor* touchConsumedActor(processor.mLastConsumedActor.GetActor());
499 if(touchConsumedActor) // If there is a consumed actor, send events only to the consumed actor.
501 RenderTask& currentRenderTaskImpl = *localVars.currentRenderTask.Get();
502 localVars.consumedActor = EmitTouchSignals(touchConsumedActor, currentRenderTaskImpl, localVars.touchEventImpl, localVars.primaryPointState, localVars.isGeometry);
506 // Let's propagate touch events from the intercepted actor or start propagating touch events from the leaf actor.
507 localVars.consumedActor = EmitGeoTouchSignals(interceptedActor ? processor.mInterceptedActorLists : processor.mCandidateActorLists, localVars.touchEventHandle);
509 if(localVars.consumedActor)
511 // If consumed, the actors who previously received the touch are interrupted, indicating that the touch has been consumed by another actor.
513 if(localVars.primaryPointState != PointState::DOWN)
515 std::list<Dali::Internal::Actor*>::reverse_iterator rIter = std::find(processor.mCandidateActorLists.rbegin(), processor.mCandidateActorLists.rend(), localVars.consumedActor);
516 if(rIter != processor.mCandidateActorLists.rend())
518 for(++rIter; rIter != processor.mCandidateActorLists.rend(); ++rIter)
520 Actor* actorImpl(*rIter);
521 EmitTouchSignals(actorImpl, *processor.mLastRenderTask.Get(), localVars.touchEventImpl, PointState::INTERRUPTED, localVars.isGeometry);
527 std::list<Dali::Internal::Actor*>::iterator iter = std::find(processor.mCandidateActorLists.begin(), processor.mCandidateActorLists.end(), localVars.consumedActor);
528 if(iter != processor.mCandidateActorLists.end())
530 for(++iter; iter != processor.mCandidateActorLists.end(); ++iter)
532 Actor* actorImpl(*iter);
533 EmitTouchSignals(actorImpl, *localVars.currentRenderTask.Get(), localVars.touchEventImpl, PointState::INTERRUPTED, localVars.isGeometry);
542 Actor* interceptedTouchActor(processor.mInterceptedTouchActor.GetActor());
543 if(interceptedTouchActor)
545 Dali::Actor interceptedTouchActorHandle(interceptedTouchActor);
546 localVars.consumedActor = EmitTouchSignals(interceptedTouchActorHandle, localVars.touchEventHandle);
550 // Emit the intercept touch signal
551 Dali::Actor interceptedActor = EmitInterceptTouchSignals(localVars.primaryHitActor, localVars.touchEventHandle);
554 processor.mInterceptedTouchActor.SetActor(&GetImplementation(interceptedActor));
555 // If the child is being touched, INTERRUPTED is sent.
556 if(processor.mLastPrimaryHitActor.GetActor() &&
557 processor.mLastPrimaryHitActor.GetActor() != interceptedActor &&
558 processor.mLastRenderTask &&
559 processor.mLastPrimaryPointState != PointState::FINISHED)
561 EmitTouchSignals(processor.mLastPrimaryHitActor.GetActor(), *processor.mLastRenderTask.Get(), localVars.touchEventImpl, PointState::INTERRUPTED, localVars.isGeometry);
562 processor.mTouchDownConsumedActor.SetActor(nullptr);
564 localVars.consumedActor = EmitTouchSignals(interceptedActor, localVars.touchEventHandle);
568 localVars.consumedActor = EmitTouchSignals(localVars.primaryHitActor, localVars.touchEventHandle);
573 consumed = localVars.consumedActor ? true : false;
575 if(localVars.primaryPointState == PointState::MOTION)
577 DALI_LOG_INFO(gLogFilter,
579 "PrimaryHitActor: (%p) id(%d), name(%s), state(%s), screenPosition(%f, %f), isGeo : %d \n",
580 localVars.primaryHitActor ? reinterpret_cast<void*>(&localVars.primaryHitActor.GetBaseObject()) : NULL,
581 localVars.primaryHitActor ? localVars.primaryHitActor.GetProperty<int32_t>(Dali::Actor::Property::ID) : -1,
582 localVars.primaryHitActor ? localVars.primaryHitActor.GetProperty<std::string>(Dali::Actor::Property::NAME).c_str() : "",
583 TOUCH_POINT_STATE[localVars.primaryPointState],
584 localVars.primaryPoint->GetScreenPosition().x,
585 localVars.primaryPoint->GetScreenPosition().y,
586 localVars.isGeometry);
587 DALI_LOG_INFO(gLogFilter,
589 "ConsumedActor: (%p) id(%d), name(%s), state(%s), isGeo : %d \n",
590 localVars.consumedActor ? reinterpret_cast<void*>(&localVars.consumedActor.GetBaseObject()) : NULL,
591 localVars.consumedActor ? localVars.consumedActor.GetProperty<int32_t>(Dali::Actor::Property::ID) : -1,
592 localVars.consumedActor ? localVars.consumedActor.GetProperty<std::string>(Dali::Actor::Property::NAME).c_str() : "",
593 TOUCH_POINT_STATE[localVars.primaryPointState],
594 localVars.isGeometry);
598 DALI_LOG_RELEASE_INFO("PrimaryHitActor:(%p), id(%d), name(%s), state(%s), screenPosition(%f, %f), isGeo : %d \n",
599 localVars.primaryHitActor ? reinterpret_cast<void*>(&localVars.primaryHitActor.GetBaseObject()) : NULL,
600 localVars.primaryHitActor ? localVars.primaryHitActor.GetProperty<int32_t>(Dali::Actor::Property::ID) : -1,
601 localVars.primaryHitActor ? localVars.primaryHitActor.GetProperty<std::string>(Dali::Actor::Property::NAME).c_str() : "",
602 TOUCH_POINT_STATE[localVars.primaryPointState],
603 localVars.primaryPoint->GetScreenPosition().x,
604 localVars.primaryPoint->GetScreenPosition().y,
605 localVars.isGeometry);
606 DALI_LOG_RELEASE_INFO("ConsumedActor: (%p), id(%d), name(%s), state(%s), isGeo : %d \n",
607 localVars.consumedActor ? reinterpret_cast<void*>(&localVars.consumedActor.GetBaseObject()) : NULL,
608 localVars.consumedActor ? localVars.consumedActor.GetProperty<int32_t>(Dali::Actor::Property::ID) : -1,
609 localVars.consumedActor ? localVars.consumedActor.GetProperty<std::string>(Dali::Actor::Property::NAME).c_str() : "",
610 TOUCH_POINT_STATE[localVars.primaryPointState],
611 localVars.isGeometry);
615 if((localVars.primaryPointState == PointState::DOWN) &&
616 (localVars.touchEventImpl->GetPointCount() == 1) &&
617 (localVars.consumedActor && localVars.consumedActor.GetProperty<bool>(Dali::Actor::Property::CONNECTED_TO_SCENE)))
619 processor.mTouchDownConsumedActor.SetActor(&GetImplementation(localVars.consumedActor));
626 * Deliver Leave event to last hit or consuming actor if required.
627 * @param[in/out] localVars The struct of stack variables used by ProcessTouchEvent
628 * @return True if consumed, false otherwise.
630 static inline bool DeliverLeaveEvent(ProcessTouchEventVariables& localVars)
632 bool consumed = false;
633 TouchEventProcessor& processor(localVars.processor);
634 if((localVars.primaryPointState == PointState::MOTION) || (localVars.primaryPointState == PointState::UP) || (localVars.primaryPointState == PointState::STATIONARY))
636 if(processor.mLastRenderTask)
638 Dali::Actor leaveEventConsumer;
639 RenderTask& lastRenderTaskImpl = *processor.mLastRenderTask.Get();
641 if(localVars.isGeometry)
643 if(localVars.lastPrimaryHitActor)
645 if(!localVars.lastPrimaryHitActor->IsHittable() || !IsActuallySensitive(localVars.lastPrimaryHitActor))
647 // At this point mLastPrimaryHitActor was touchable and sensitive in the previous touch event process but is not in the current one.
648 // An interrupted event is send to allow some actors to go back to their original state (i.e. Button controls)
649 DALI_LOG_RELEASE_INFO("InterruptedActor(Hit): (%p) %s\n", reinterpret_cast<void*>(localVars.lastPrimaryHitActor), localVars.lastPrimaryHitActor->GetName().data());
650 leaveEventConsumer = EmitTouchSignals(localVars.lastPrimaryHitActor, lastRenderTaskImpl, localVars.touchEventImpl, PointState::INTERRUPTED, localVars.isGeometry);
654 consumed = leaveEventConsumer ? true : false;
656 // Check if the motion event has been consumed by another actor's listener. In this case, the previously
657 // consumed actor's listeners may need to be informed (through a leave event).
658 // Further checks here to ensure we do not signal the same actor twice for the same event.
659 if(localVars.lastConsumedActor &&
660 localVars.lastConsumedActor != localVars.lastPrimaryHitActor &&
661 localVars.lastConsumedActor != leaveEventConsumer)
663 if(!localVars.lastConsumedActor->IsHittable() || !IsActuallySensitive(localVars.lastConsumedActor))
665 // At this point mLastConsumedActor was touchable and sensitive in the previous touch event process but is not in the current one.
666 // An interrupted event is send to allow some actors to go back to their original state (i.e. Button controls)
667 DALI_LOG_RELEASE_INFO("InterruptedActor(Consume): (%p) %s\n", reinterpret_cast<void*>(localVars.lastConsumedActor), localVars.lastConsumedActor->GetName().data());
668 EmitTouchSignals(processor.mLastConsumedActor.GetActor(), lastRenderTaskImpl, localVars.touchEventImpl, PointState::INTERRUPTED, localVars.isGeometry);
674 if(localVars.lastPrimaryHitActor &&
675 localVars.lastPrimaryHitActor != localVars.primaryHitActor &&
676 localVars.lastPrimaryHitActor != localVars.consumedActor)
678 if(localVars.lastPrimaryHitActor->IsHittable() && IsActuallySensitive(localVars.lastPrimaryHitActor))
680 if(localVars.lastPrimaryHitActor->GetLeaveRequired())
682 DALI_LOG_RELEASE_INFO("LeaveActor(Hit): (%p) %d %s\n", reinterpret_cast<void*>(localVars.lastPrimaryHitActor), localVars.lastPrimaryHitActor->GetId(), localVars.lastPrimaryHitActor->GetName().data());
683 leaveEventConsumer = EmitTouchSignals(localVars.lastPrimaryHitActor, lastRenderTaskImpl, localVars.touchEventImpl, PointState::LEAVE, localVars.isGeometry);
688 // At this point mLastPrimaryHitActor was touchable and sensitive in the previous touch event process but is not in the current one.
689 // An interrupted event is send to allow some actors to go back to their original state (i.e. Button controls)
690 DALI_LOG_RELEASE_INFO("InterruptedActor(Hit): (%p) %d %s\n", reinterpret_cast<void*>(localVars.lastPrimaryHitActor), localVars.lastPrimaryHitActor->GetId(), localVars.lastPrimaryHitActor->GetName().data());
691 leaveEventConsumer = EmitTouchSignals(localVars.lastPrimaryHitActor, lastRenderTaskImpl, localVars.touchEventImpl, PointState::INTERRUPTED, localVars.isGeometry);
695 consumed |= leaveEventConsumer ? true : false;
697 // Check if the motion event has been consumed by another actor's listener. In this case, the previously
698 // consumed actor's listeners may need to be informed (through a leave event).
699 // Further checks here to ensure we do not signal the same actor twice for the same event.
700 if(localVars.lastConsumedActor &&
701 localVars.lastConsumedActor != localVars.consumedActor &&
702 localVars.lastConsumedActor != localVars.lastPrimaryHitActor &&
703 localVars.lastConsumedActor != localVars.primaryHitActor &&
704 localVars.lastConsumedActor != leaveEventConsumer)
706 if(localVars.lastConsumedActor->IsHittable() && IsActuallySensitive(localVars.lastConsumedActor))
708 if(localVars.lastConsumedActor->GetLeaveRequired())
710 DALI_LOG_RELEASE_INFO("LeaveActor(Consume): (%p) %d %s\n", reinterpret_cast<void*>(localVars.lastConsumedActor), localVars.lastConsumedActor->GetId(), localVars.lastConsumedActor->GetName().data());
711 EmitTouchSignals(localVars.lastConsumedActor, lastRenderTaskImpl, localVars.touchEventImpl, PointState::LEAVE, localVars.isGeometry);
716 // At this point mLastConsumedActor was touchable and sensitive in the previous touch event process but is not in the current one.
717 // An interrupted event is send to allow some actors to go back to their original state (i.e. Button controls)
718 DALI_LOG_RELEASE_INFO("InterruptedActor(Consume): (%p) %d %s\n", reinterpret_cast<void*>(localVars.lastConsumedActor), localVars.lastConsumedActor->GetId(), localVars.lastConsumedActor->GetName().data());
719 EmitTouchSignals(processor.mLastConsumedActor.GetActor(), lastRenderTaskImpl, localVars.touchEventImpl, PointState::INTERRUPTED, localVars.isGeometry);
729 * Update the processor member appropriately by handling the final up event, and setting the last hit/consumed events etc.
730 * @param[in/out] localVars The struct of stack variables used by ProcessTouchEvent
732 static inline void UpdateMembersWithCurrentHitInformation(ProcessTouchEventVariables& localVars)
734 // If our primary point is an Up event, then the primary point (in multi-touch) will change next
735 // time so set our last primary actor to NULL. Do the same to the last consumed actor as well.
737 TouchEventProcessor& processor(localVars.processor);
738 if(localVars.primaryPointState == PointState::UP)
744 // The primaryHitActor may have been removed from the scene so ensure it is still on the scene before setting members.
745 if(localVars.primaryHitActor && GetImplementation(localVars.primaryHitActor).OnScene())
747 processor.mLastPrimaryHitActor.SetActor(&GetImplementation(localVars.primaryHitActor));
749 // Only observe the consumed actor if we have a primaryHitActor (check if it is still on the scene).
750 if(localVars.consumedActor && GetImplementation(localVars.consumedActor).OnScene())
752 processor.mLastConsumedActor.SetActor(&GetImplementation(localVars.consumedActor));
756 if(localVars.isGeometry)
758 if(localVars.lastConsumedActor && !localVars.lastConsumedActor->OnScene())
760 processor.mLastConsumedActor.SetActor(nullptr);
765 processor.mLastConsumedActor.SetActor(nullptr);
769 processor.mLastRenderTask = localVars.currentRenderTask;
770 processor.mLastPrimaryPointState = localVars.primaryPointState;
780 * Deliver an event to the touch-down actor and emit from the scene if required.
781 * @param[in/out] localVars The struct of stack variables used by ProcessTouchEvent
782 * @param[in] event The touch event that has occurred
784 static inline void DeliverEventToTouchDownActorAndScene(ProcessTouchEventVariables& localVars, const Integration::TouchEvent& event)
786 TouchEventProcessor& processor(localVars.processor);
787 if(localVars.touchEventImpl->GetPointCount() == 1) // Only want the first touch and the last release
789 switch(localVars.primaryPointState)
793 Actor* touchDownConsumedActor(processor.mTouchDownConsumedActor.GetActor());
794 if(touchDownConsumedActor &&
795 touchDownConsumedActor != localVars.consumedActor &&
796 touchDownConsumedActor != localVars.lastPrimaryHitActor &&
797 touchDownConsumedActor != localVars.lastConsumedActor)
799 Dali::Actor touchDownConsumedActorHandle(touchDownConsumedActor);
801 Integration::Point currentPoint = localVars.touchEventImpl->GetPoint(0);
802 currentPoint.SetHitActor(touchDownConsumedActorHandle);
803 currentPoint.SetState(PointState::INTERRUPTED);
805 if(localVars.isGeometry)
807 std::list<Dali::Internal::Actor*> actorLists;
808 actorLists.push_back(touchDownConsumedActor);
809 GeoAllocAndEmitTouchSignals(actorLists, event.time, currentPoint, nullptr /* Not Required for this use case */);
813 AllocAndEmitTouchSignals(event.time, touchDownConsumedActorHandle, currentPoint, nullptr /* Not Required for this use case */);
817 processor.mTouchDownConsumedActor.SetActor(nullptr);
818 processor.mInterceptedTouchActor.SetActor(nullptr);
823 case PointState::DOWN:
825 processor.mScene.EmitTouchedSignal(localVars.touchEventHandle);
829 case PointState::MOTION:
830 case PointState::LEAVE:
831 case PointState::STATIONARY:
832 case PointState::INTERRUPTED:
842 TouchEventProcessor::TouchEventProcessor(Scene& scene)
844 mLastPrimaryHitActor(MakeCallback(this, &TouchEventProcessor::OnObservedActorDisconnected)),
845 mLastConsumedActor(),
846 mCapturingTouchActor(),
848 mTouchDownConsumedActor(),
849 mInterceptedTouchActor(),
851 mLastPrimaryPointState(PointState::FINISHED)
853 DALI_LOG_TRACE_METHOD(gLogFilter);
856 TouchEventProcessor::~TouchEventProcessor()
858 DALI_LOG_TRACE_METHOD(gLogFilter);
861 bool TouchEventProcessor::ProcessTouchEvent(const Integration::TouchEvent& event)
863 DALI_LOG_TRACE_METHOD(gLogFilter);
864 DALI_ASSERT_ALWAYS(!event.points.empty() && "Empty TouchEvent sent from Integration\n");
866 PRINT_HIERARCHY(gLogFilter);
868 DALI_TRACE_SCOPE(gTraceFilter, "DALI_PROCESS_TOUCH_EVENT");
870 ProcessTouchEventVariables localVars(*this, mScene.IsGeometryHittestEnabled());
872 // 1) Check if it is an interrupted event - we should inform our last primary hit actor about this
873 // and emit the stage signal as well.
875 if(event.points[0].GetState() == PointState::INTERRUPTED)
877 Impl::EmitInterruptedEvent(localVars, event);
878 return false; // No need for hit testing & already an interrupted event so just return false
882 localVars.touchEventImpl = new TouchEvent(event.time);
883 localVars.touchEventHandle = Dali::TouchEvent(localVars.touchEventImpl.Get());
885 DALI_LOG_INFO(gLogFilter, Debug::Concise, "\n");
886 DALI_LOG_INFO(gLogFilter, Debug::General, "Point(s): %d\n", event.GetPointCount());
888 bool firstPointParsed = false;
889 for(auto&& currentPoint : event.points)
891 HitTestAlgorithm::Results hitTestResults;
892 hitTestResults.point = currentPoint;
893 hitTestResults.eventTime = event.time;
894 if(!firstPointParsed)
896 firstPointParsed = true;
897 ParsePrimaryTouchPoint(hitTestResults, mCapturingTouchActor, mOwnTouchActor, mLastRenderTask, currentPoint, mScene, mCandidateActorLists);
899 // Only set the currentRenderTask for the primary hit actor.
900 localVars.currentRenderTask = hitTestResults.renderTask;
901 localVars.touchEventImpl->SetRenderTask(Dali::RenderTask(localVars.currentRenderTask.Get()));
905 Actor* capturingTouchActor = mCapturingTouchActor.GetActor();
906 if(capturingTouchActor && mLastRenderTask)
908 hitTestResults.actor = Dali::Actor(capturingTouchActor);
909 hitTestResults.renderTask = mLastRenderTask;
910 const Vector2& screenPosition = currentPoint.GetScreenPosition();
911 capturingTouchActor->ScreenToLocal(*mLastRenderTask, hitTestResults.actorCoordinates.x, hitTestResults.actorCoordinates.y, screenPosition.x, screenPosition.y);
915 HitTestAlgorithm::HitTest(mScene.GetSize(), mScene.GetRenderTaskList(), mScene.GetLayerList(), currentPoint.GetScreenPosition(), hitTestResults, nullptr, localVars.isGeometry);
919 Integration::Point newPoint(currentPoint);
920 newPoint.SetHitActor(hitTestResults.actor);
921 newPoint.SetLocalPosition(hitTestResults.actorCoordinates);
923 localVars.touchEventImpl->AddPoint(newPoint);
925 DALI_LOG_INFO(gLogFilter,
927 " State(%s), Screen(%.0f, %.0f), HitActor(%p, %s), Local(%.2f, %.2f)\n",
928 TOUCH_POINT_STATE[currentPoint.GetState()],
929 currentPoint.GetScreenPosition().x,
930 currentPoint.GetScreenPosition().y,
931 (hitTestResults.actor ? reinterpret_cast<void*>(&hitTestResults.actor.GetBaseObject()) : NULL),
932 (hitTestResults.actor ? hitTestResults.actor.GetProperty<std::string>(Dali::Actor::Property::NAME).c_str() : ""),
933 hitTestResults.actorCoordinates.x,
934 hitTestResults.actorCoordinates.y);
937 // 3) Recursively deliver events to the actor and its parents, until the event is consumed or the stage is reached.
938 localVars.primaryPoint = &localVars.touchEventImpl->GetPoint(0);
939 localVars.primaryHitActor = localVars.primaryPoint->GetHitActor();
940 localVars.primaryPointState = localVars.primaryPoint->GetState();
942 bool consumed = Impl::DeliverEventsToActorAndParents(localVars);
944 // 4) Check if the last primary hit actor requires a leave event and if it was different to the current primary
945 // hit actor. Also process the last consumed actor in the same manner.
946 localVars.lastPrimaryHitActor = mInterceptedTouchActor.GetActor() ? mInterceptedTouchActor.GetActor() : mLastPrimaryHitActor.GetActor();
947 localVars.lastConsumedActor = mLastConsumedActor.GetActor();
948 consumed |= Impl::DeliverLeaveEvent(localVars);
950 // 5) Update the processor member appropriately.
951 Impl::UpdateMembersWithCurrentHitInformation(localVars);
953 // 6) Emit an interrupted event to the touch-down actor if it hasn't consumed the up and
954 // emit the stage touched event if required.
955 Impl::DeliverEventToTouchDownActorAndScene(localVars, event);
960 void TouchEventProcessor::OnObservedActorDisconnected(Actor* actor)
962 if(mScene.IsGeometryHittestEnabled() && (actor == mLastConsumedActor.GetActor() || actor == mLastPrimaryHitActor.GetActor()))
964 Dali::Actor actorHandle(actor);
965 Integration::Point point;
966 point.SetState(PointState::INTERRUPTED);
967 point.SetHitActor(actorHandle);
968 if(actor == mLastConsumedActor.GetActor())
970 std::list<Dali::Internal::Actor*> actorLists;
971 actorLists.push_back(mLastConsumedActor.GetActor());
972 GeoAllocAndEmitTouchSignals(actorLists, 0, point, mLastRenderTask);
974 else if(!mLastConsumedActor.GetActor())
976 GeoAllocAndEmitTouchSignals(mCandidateActorLists, 0, point, mLastRenderTask);
978 // Do not set mLastPrimaryHitActor to NULL we may be iterating through its observers
979 mLastConsumedActor.SetActor(nullptr);
980 mLastRenderTask.Reset();
981 mLastPrimaryPointState = PointState::FINISHED;
985 if(actor == mLastPrimaryHitActor.GetActor())
987 Dali::Actor actorHandle(actor);
988 Integration::Point point;
989 point.SetState(PointState::INTERRUPTED);
990 point.SetHitActor(actorHandle);
992 TouchEventPtr touchEventImpl(new TouchEvent);
993 touchEventImpl->AddPoint(point);
994 Dali::TouchEvent touchEventHandle(touchEventImpl.Get());
996 Dali::Actor eventConsumer = EmitTouchSignals(actorHandle, touchEventHandle);
997 if(mLastConsumedActor.GetActor() != eventConsumer)
999 EmitTouchSignals(Dali::Actor(mLastConsumedActor.GetActor()), touchEventHandle);
1002 // Do not set mLastPrimaryHitActor to NULL we may be iterating through its observers
1003 mLastConsumedActor.SetActor(nullptr);
1004 mLastRenderTask.Reset();
1005 mLastPrimaryPointState = PointState::FINISHED;
1010 void TouchEventProcessor::Clear()
1012 mLastPrimaryHitActor.SetActor(nullptr);
1013 mLastConsumedActor.SetActor(nullptr);
1014 mCapturingTouchActor.SetActor(nullptr);
1015 mOwnTouchActor.SetActor(nullptr);
1016 mInterceptedTouchActor.SetActor(nullptr);
1017 mLastRenderTask.Reset();
1018 mLastPrimaryPointState = PointState::FINISHED;
1021 } // namespace Dali::Internal