Merge "If the touch is intercepted, future events will only be called on the parent...
[platform/core/uifw/dali-core.git] / dali / internal / event / events / touch-event-processor.cpp
1 /*
2  * Copyright (c) 2023 Samsung Electronics Co., Ltd.
3  *
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
7  *
8  * http://www.apache.org/licenses/LICENSE-2.0
9  *
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.
15  *
16  */
17
18 // CLASS HEADER
19 #include <dali/internal/event/events/touch-event-processor.h>
20
21 #if defined(DEBUG_ENABLED)
22 #include <sstream>
23 #endif
24
25 // INTERNAL INCLUDES
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>
39
40 namespace Dali
41 {
42 namespace Internal
43 {
44 namespace
45 {
46 DALI_INIT_TRACE_FILTER(gTraceFilter, DALI_TRACE_PERFORMANCE_MARKER, false);
47 #if defined(DEBUG_ENABLED)
48 Debug::Filter* gLogFilter = Debug::Filter::New(Debug::NoLogging, false, "LOG_TOUCH_PROCESSOR");
49
50 const char* TOUCH_POINT_STATE[6] =
51   {
52     "DOWN",
53     "UP",
54     "MOTION",
55     "LEAVE",
56     "STATIONARY",
57     "INTERRUPTED",
58 };
59
60 #endif // defined(DEBUG_ENABLED)
61
62 Dali::Actor EmitInterceptTouchSignals(Dali::Actor actor, const Dali::TouchEvent& touchEvent)
63 {
64   Dali::Actor interceptedActor;
65
66   if(actor)
67   {
68     Dali::Actor parent = actor.GetParent();
69     if(parent)
70     {
71       // Recursively deliver events to the actor and its parents for intercept touch event.
72       interceptedActor = EmitInterceptTouchSignals(parent, touchEvent);
73     }
74
75     if(!interceptedActor)
76     {
77       bool   intercepted = false;
78       Actor& actorImpl(GetImplementation(actor));
79       if(actorImpl.GetInterceptTouchRequired())
80       {
81         DALI_TRACE_SCOPE(gTraceFilter, "DALI_EMIT_INTERCEPT_TOUCH_EVENT_SIGNAL");
82         intercepted = actorImpl.EmitInterceptTouchEventSignal(touchEvent);
83         if(intercepted)
84         {
85           interceptedActor = Dali::Actor(&actorImpl);
86         }
87       }
88     }
89   }
90
91   return interceptedActor;
92 }
93
94 /**
95  *  Recursively deliver events to the actor and its parents, until the event is consumed or the stage is reached.
96  */
97 Dali::Actor EmitTouchSignals(Dali::Actor actor, const Dali::TouchEvent& touchEvent)
98 {
99   Dali::Actor consumedActor;
100
101   if(actor)
102   {
103     Dali::Actor oldParent(actor.GetParent());
104
105     Actor& actorImpl(GetImplementation(actor));
106
107     bool consumed(false);
108
109     // Only emit the signal if the actor's touch signal has connections (or derived actor implementation requires touch).
110     if(actorImpl.GetTouchRequired())
111     {
112       DALI_TRACE_SCOPE(gTraceFilter, "DALI_EMIT_TOUCH_EVENT_SIGNAL");
113       consumed = actorImpl.EmitTouchEventSignal(touchEvent);
114     }
115
116     if(consumed)
117     {
118       // One of this actor's listeners has consumed the event so set this actor as the consumed actor.
119       consumedActor = Dali::Actor(&actorImpl);
120     }
121     else
122     {
123       // The actor may have been removed/reparented during the signal callbacks.
124       Dali::Actor parent = actor.GetParent();
125
126       if(parent &&
127          (parent == oldParent))
128       {
129         // One of the actor's parents may consumed the event and they should be set as the consumed actor.
130         consumedActor = EmitTouchSignals(parent, touchEvent);
131       }
132     }
133   }
134
135   return consumedActor;
136 }
137
138 Dali::Actor AllocAndEmitTouchSignals(unsigned long time, Dali::Actor actor, const Integration::Point& point)
139 {
140   TouchEventPtr    touchEvent(new TouchEvent(time));
141   Dali::TouchEvent touchEventHandle(touchEvent.Get());
142
143   touchEvent->AddPoint(point);
144
145   return EmitTouchSignals(actor, touchEventHandle);
146 }
147
148 /**
149  * Changes the state of the primary point to leave and emits the touch signals
150  */
151 Dali::Actor EmitTouchSignals(Actor* actor, RenderTask& renderTask, const TouchEventPtr& originalTouchEvent, PointState::Type state)
152 {
153   Dali::Actor consumingActor;
154
155   if(actor)
156   {
157     TouchEventPtr touchEventImpl = TouchEvent::Clone(*originalTouchEvent.Get());
158
159     Integration::Point& primaryPoint = touchEventImpl->GetPoint(0);
160
161     const Vector2& screenPosition = primaryPoint.GetScreenPosition();
162     Vector2        localPosition;
163     actor->ScreenToLocal(renderTask, localPosition.x, localPosition.y, screenPosition.x, screenPosition.y);
164
165     primaryPoint.SetLocalPosition(localPosition);
166     primaryPoint.SetHitActor(Dali::Actor(actor));
167     primaryPoint.SetState(state);
168
169     consumingActor = EmitTouchSignals(Dali::Actor(actor), Dali::TouchEvent(touchEventImpl.Get()));
170   }
171
172   return consumingActor;
173 }
174
175 /**
176  * @brief Parses the primary touch point by performing a hit-test if necessary
177  *
178  * @param[out] hitTestResults The hit test results are put into this variable
179  * @param[in/out] capturingTouchActorObserver The observer for the capturing touch actor member
180  * @param[in] lastRenderTask The last render task member
181  * @param[in] currentPoint The current point information
182  * @param[in] scene The scene that this touch is related to
183  */
184 void ParsePrimaryTouchPoint(
185   HitTestAlgorithm::Results& hitTestResults,
186   ActorObserver&             capturingTouchActorObserver,
187   ActorObserver&             ownTouchActorObserver,
188   const RenderTaskPtr&       lastRenderTask,
189   const Integration::Point&  currentPoint,
190   const Internal::Scene&     scene)
191 {
192   Actor* capturingTouchActor = capturingTouchActorObserver.GetActor();
193
194   // We only set the capturing touch actor when the first touch-started actor captures all touch so if it's set, just use it
195   if(capturingTouchActor && lastRenderTask)
196   {
197     hitTestResults.actor          = Dali::Actor(capturingTouchActor);
198     hitTestResults.renderTask     = lastRenderTask;
199     const Vector2& screenPosition = currentPoint.GetScreenPosition();
200     capturingTouchActor->ScreenToLocal(*lastRenderTask, hitTestResults.actorCoordinates.x, hitTestResults.actorCoordinates.y, screenPosition.x, screenPosition.y);
201   }
202   else
203   {
204     Actor* ownTouchActor = ownTouchActorObserver.GetActor();
205     HitTestAlgorithm::HitTest(scene.GetSize(), scene.GetRenderTaskList(), scene.GetLayerList(), currentPoint.GetScreenPosition(), hitTestResults, ownTouchActor);
206
207     if(currentPoint.GetState() == PointState::STARTED && hitTestResults.actor)
208     {
209       // If we've just started touch, then check whether the actor has requested to capture all touch events
210       Actor* hitActor = &GetImplementation(hitTestResults.actor);
211       if(hitActor->CapturesAllTouchAfterStart())
212       {
213         capturingTouchActorObserver.SetActor(hitActor);
214       }
215       if(hitActor->IsAllowedOnlyOwnTouch())
216       {
217         ownTouchActorObserver.SetActor(hitActor);
218       }
219     }
220   }
221 }
222
223 } // unnamed namespace
224
225 TouchEventProcessor::TouchEventProcessor(Scene& scene)
226 : mScene(scene),
227   mLastPrimaryHitActor(MakeCallback(this, &TouchEventProcessor::OnObservedActorDisconnected)),
228   mLastConsumedActor(),
229   mCapturingTouchActor(),
230   mOwnTouchActor(),
231   mTouchDownConsumedActor(),
232   mInterceptedTouchActor(),
233   mLastRenderTask(),
234   mLastPrimaryPointState(PointState::FINISHED)
235 {
236   DALI_LOG_TRACE_METHOD(gLogFilter);
237 }
238
239 TouchEventProcessor::~TouchEventProcessor()
240 {
241   DALI_LOG_TRACE_METHOD(gLogFilter);
242 }
243
244 void TouchEventProcessor::Clear()
245 {
246   mLastPrimaryHitActor.SetActor(nullptr);
247   mLastConsumedActor.SetActor(nullptr);
248   mCapturingTouchActor.SetActor(nullptr);
249   mOwnTouchActor.SetActor(nullptr);
250   mInterceptedTouchActor.SetActor(nullptr);
251   mLastRenderTask.Reset();
252   mLastPrimaryPointState = PointState::FINISHED;
253 }
254
255 bool TouchEventProcessor::ProcessTouchEvent(const Integration::TouchEvent& event)
256 {
257   DALI_LOG_TRACE_METHOD(gLogFilter);
258   DALI_ASSERT_ALWAYS(!event.points.empty() && "Empty TouchEvent sent from Integration\n");
259
260   PRINT_HIERARCHY(gLogFilter);
261
262   DALI_TRACE_SCOPE(gTraceFilter, "DALI_PROCESS_TOUCH_EVENT");
263
264   // 1) Check if it is an interrupted event - we should inform our last primary hit actor about this
265   //    and emit the stage signal as well.
266
267   if(event.points[0].GetState() == PointState::INTERRUPTED)
268   {
269     Dali::Actor        consumingActor;
270     Integration::Point currentPoint(event.points[0]);
271
272     Actor* lastPrimaryHitActor(mLastPrimaryHitActor.GetActor());
273     if(lastPrimaryHitActor)
274     {
275       Dali::Actor lastPrimaryHitActorHandle(lastPrimaryHitActor);
276       currentPoint.SetHitActor(lastPrimaryHitActorHandle);
277
278       consumingActor = AllocAndEmitTouchSignals(event.time, lastPrimaryHitActorHandle, currentPoint);
279     }
280
281     // If the last consumed actor was different to the primary hit actor then inform it as well (if it has not already been informed).
282     Actor* lastConsumedActor(mLastConsumedActor.GetActor());
283     if(lastConsumedActor &&
284        lastConsumedActor != lastPrimaryHitActor &&
285        lastConsumedActor != consumingActor)
286     {
287       Dali::Actor lastConsumedActorHandle(lastConsumedActor);
288       currentPoint.SetHitActor(lastConsumedActorHandle);
289       AllocAndEmitTouchSignals(event.time, lastConsumedActorHandle, currentPoint);
290     }
291
292     // Tell the touch-down consuming actor as well, if required
293     Actor* touchDownConsumedActor(mTouchDownConsumedActor.GetActor());
294     if(touchDownConsumedActor &&
295        touchDownConsumedActor != lastPrimaryHitActor &&
296        touchDownConsumedActor != lastConsumedActor &&
297        touchDownConsumedActor != consumingActor)
298     {
299       Dali::Actor touchDownConsumedActorHandle(touchDownConsumedActor);
300
301       currentPoint.SetHitActor(touchDownConsumedActorHandle);
302       AllocAndEmitTouchSignals(event.time, touchDownConsumedActorHandle, currentPoint);
303     }
304
305     Clear();
306     mTouchDownConsumedActor.SetActor(nullptr);
307
308     currentPoint.SetHitActor(Dali::Actor());
309
310     TouchEventPtr    touchEventImpl(new TouchEvent(event.time));
311     Dali::TouchEvent touchEventHandle(touchEventImpl.Get());
312
313     touchEventImpl->AddPoint(currentPoint);
314
315     mScene.EmitTouchedSignal(touchEventHandle);
316     return false; // No need for hit testing & already an interrupted event so just return false
317   }
318
319   // 2) Hit Testing.
320   TouchEventPtr    touchEventImpl(new TouchEvent(event.time));
321   Dali::TouchEvent touchEventHandle(touchEventImpl.Get());
322
323   DALI_LOG_INFO(gLogFilter, Debug::Concise, "\n");
324   DALI_LOG_INFO(gLogFilter, Debug::General, "Point(s): %d\n", event.GetPointCount());
325
326   RenderTaskPtr currentRenderTask;
327   bool          firstPointParsed = false;
328
329   for(auto&& currentPoint : event.points)
330   {
331     HitTestAlgorithm::Results hitTestResults;
332     hitTestResults.point     = currentPoint;
333     hitTestResults.eventTime = event.time;
334     if(!firstPointParsed)
335     {
336       firstPointParsed = true;
337       ParsePrimaryTouchPoint(hitTestResults, mCapturingTouchActor, mOwnTouchActor, mLastRenderTask, currentPoint, mScene);
338
339       // Only set the currentRenderTask for the primary hit actor.
340       currentRenderTask = hitTestResults.renderTask;
341     }
342     else
343     {
344       HitTestAlgorithm::HitTest(mScene.GetSize(), mScene.GetRenderTaskList(), mScene.GetLayerList(), currentPoint.GetScreenPosition(), hitTestResults);
345     }
346
347     Integration::Point newPoint(currentPoint);
348     newPoint.SetHitActor(hitTestResults.actor);
349     newPoint.SetLocalPosition(hitTestResults.actorCoordinates);
350
351     touchEventImpl->AddPoint(newPoint);
352
353     DALI_LOG_INFO(gLogFilter, Debug::General, "  State(%s), Screen(%.0f, %.0f), HitActor(%p, %s), Local(%.2f, %.2f)\n", TOUCH_POINT_STATE[currentPoint.GetState()], currentPoint.GetScreenPosition().x, currentPoint.GetScreenPosition().y, (hitTestResults.actor ? reinterpret_cast<void*>(&hitTestResults.actor.GetBaseObject()) : NULL), (hitTestResults.actor ? hitTestResults.actor.GetProperty<std::string>(Dali::Actor::Property::NAME).c_str() : ""), hitTestResults.actorCoordinates.x, hitTestResults.actorCoordinates.y);
354   }
355
356   // 3) Recursively deliver events to the actor and its parents, until the event is consumed or the stage is reached.
357
358   bool consumed = false;
359
360   // Emit the touch signal
361   Dali::Actor consumedActor;
362
363   Integration::Point& primaryPoint      = touchEventImpl->GetPoint(0);
364   Dali::Actor         primaryHitActor   = primaryPoint.GetHitActor();
365   PointState::Type    primaryPointState = primaryPoint.GetState();
366
367   if(currentRenderTask)
368   {
369     Actor* interceptedTouchActor(mInterceptedTouchActor.GetActor());
370     if(interceptedTouchActor)
371     {
372       Dali::Actor interceptedTouchActorHandle(interceptedTouchActor);
373       consumedActor = EmitTouchSignals(interceptedTouchActorHandle, touchEventHandle);
374     }
375     else
376     {
377       // Emit the intercept touch signal
378       Dali::Actor interceptedActor = EmitInterceptTouchSignals(primaryHitActor, touchEventHandle);
379       if(interceptedActor)
380       {
381         mInterceptedTouchActor.SetActor(&GetImplementation(interceptedActor));
382         // If the child is being touched, INTERRUPTED is sent.
383         if(mLastPrimaryHitActor.GetActor() &&
384           mLastPrimaryHitActor.GetActor() != interceptedActor &&
385           mLastRenderTask &&
386           mLastPrimaryPointState != PointState::FINISHED)
387         {
388           EmitTouchSignals(mLastPrimaryHitActor.GetActor(), *mLastRenderTask.Get(), touchEventImpl, PointState::INTERRUPTED);
389           mTouchDownConsumedActor.SetActor(nullptr);
390         }
391         consumedActor = EmitTouchSignals(interceptedActor, touchEventHandle);
392       }
393       else
394       {
395         consumedActor = EmitTouchSignals(primaryHitActor, touchEventHandle);
396       }
397     }
398     consumed = consumedActor ? true : false;
399   }
400
401   DALI_LOG_INFO(gLogFilter, Debug::Concise, "PrimaryHitActor:     (%p) %s\n", primaryHitActor ? reinterpret_cast<void*>(&primaryHitActor.GetBaseObject()) : NULL, primaryHitActor ? primaryHitActor.GetProperty<std::string>(Dali::Actor::Property::NAME).c_str() : "");
402   DALI_LOG_INFO(gLogFilter, Debug::Concise, "ConsumedActor:       (%p) %s\n", consumedActor ? reinterpret_cast<void*>(&consumedActor.GetBaseObject()) : NULL, consumedActor ? consumedActor.GetProperty<std::string>(Dali::Actor::Property::NAME).c_str() : "");
403
404   if((primaryPointState == PointState::DOWN) &&
405      (touchEventImpl->GetPointCount() == 1) &&
406      (consumedActor && consumedActor.GetProperty<bool>(Dali::Actor::Property::CONNECTED_TO_SCENE)))
407   {
408     mTouchDownConsumedActor.SetActor(&GetImplementation(consumedActor));
409   }
410
411   // 4) Check if the last primary hit actor requires a leave event and if it was different to the current primary
412   //    hit actor.  Also process the last consumed actor in the same manner.
413   Actor* lastPrimaryHitActor(nullptr);
414   if(mInterceptedTouchActor.GetActor())
415   {
416     lastPrimaryHitActor = mInterceptedTouchActor.GetActor();
417   }
418   else
419   {
420     lastPrimaryHitActor = mLastPrimaryHitActor.GetActor();
421   }
422   Actor* lastConsumedActor(mLastConsumedActor.GetActor());
423   if((primaryPointState == PointState::MOTION) || (primaryPointState == PointState::UP) || (primaryPointState == PointState::STATIONARY))
424   {
425     if(mLastRenderTask)
426     {
427       Dali::Actor leaveEventConsumer;
428       RenderTask& lastRenderTaskImpl = *mLastRenderTask.Get();
429
430       if(lastPrimaryHitActor &&
431          lastPrimaryHitActor != primaryHitActor &&
432          lastPrimaryHitActor != consumedActor)
433       {
434         if(lastPrimaryHitActor->IsHittable() && IsActuallySensitive(lastPrimaryHitActor))
435         {
436           if(lastPrimaryHitActor->GetLeaveRequired())
437           {
438             DALI_LOG_INFO(gLogFilter, Debug::Concise, "LeaveActor(Hit):     (%p) %s\n", reinterpret_cast<void*>(lastPrimaryHitActor), lastPrimaryHitActor->GetName().data());
439             leaveEventConsumer = EmitTouchSignals(lastPrimaryHitActor, lastRenderTaskImpl, touchEventImpl, PointState::LEAVE);
440           }
441         }
442         else
443         {
444           // At this point mLastPrimaryHitActor was touchable and sensitive in the previous touch event process but is not in the current one.
445           // An interrupted event is send to allow some actors to go back to their original state (i.e. Button controls)
446           DALI_LOG_INFO(gLogFilter, Debug::Concise, "InterruptedActor(Hit):     (%p) %s\n", reinterpret_cast<void*>(lastPrimaryHitActor), lastPrimaryHitActor->GetName().data());
447           leaveEventConsumer = EmitTouchSignals(lastPrimaryHitActor, lastRenderTaskImpl, touchEventImpl, PointState::INTERRUPTED);
448         }
449       }
450
451       consumed |= leaveEventConsumer ? true : false;
452
453       // Check if the motion event has been consumed by another actor's listener.  In this case, the previously
454       // consumed actor's listeners may need to be informed (through a leave event).
455       // Further checks here to ensure we do not signal the same actor twice for the same event.
456       if(lastConsumedActor &&
457          lastConsumedActor != consumedActor &&
458          lastConsumedActor != lastPrimaryHitActor &&
459          lastConsumedActor != primaryHitActor &&
460          lastConsumedActor != leaveEventConsumer)
461       {
462         if(lastConsumedActor->IsHittable() && IsActuallySensitive(lastConsumedActor))
463         {
464           if(lastConsumedActor->GetLeaveRequired())
465           {
466             DALI_LOG_INFO(gLogFilter, Debug::Concise, "LeaveActor(Consume): (%p) %s\n", reinterpret_cast<void*>(lastConsumedActor), lastConsumedActor->GetName().data());
467             EmitTouchSignals(lastConsumedActor, lastRenderTaskImpl, touchEventImpl, PointState::LEAVE);
468           }
469         }
470         else
471         {
472           // At this point mLastConsumedActor was touchable and sensitive in the previous touch event process but is not in the current one.
473           // An interrupted event is send to allow some actors to go back to their original state (i.e. Button controls)
474           DALI_LOG_INFO(gLogFilter, Debug::Concise, "InterruptedActor(Consume):     (%p) %s\n", reinterpret_cast<void*>(lastConsumedActor), lastConsumedActor->GetName().data());
475           EmitTouchSignals(mLastConsumedActor.GetActor(), lastRenderTaskImpl, touchEventImpl, PointState::INTERRUPTED);
476         }
477       }
478     }
479   }
480
481   // 5) If our primary point is an Up event, then the primary point (in multi-touch) will change next
482   //    time so set our last primary actor to NULL.  Do the same to the last consumed actor as well.
483   if(primaryPointState == PointState::UP)
484   {
485     Clear();
486   }
487   else
488   {
489     // The primaryHitActor may have been removed from the scene so ensure it is still on the scene before setting members.
490     if(primaryHitActor && GetImplementation(primaryHitActor).OnScene())
491     {
492       mLastPrimaryHitActor.SetActor(&GetImplementation(primaryHitActor));
493
494       // Only observe the consumed actor if we have a primaryHitActor (check if it is still on the scene).
495       if(consumedActor && GetImplementation(consumedActor).OnScene())
496       {
497         mLastConsumedActor.SetActor(&GetImplementation(consumedActor));
498       }
499       else
500       {
501         mLastConsumedActor.SetActor(nullptr);
502       }
503
504       mLastRenderTask        = currentRenderTask;
505       mLastPrimaryPointState = primaryPointState;
506     }
507     else
508     {
509       Clear();
510     }
511   }
512
513   // 6) Emit an interrupted event to the touch-down actor if it hasn't consumed the up and
514   //    emit the stage touched event if required.
515
516   if(touchEventImpl->GetPointCount() == 1) // Only want the first touch and the last release
517   {
518     switch(primaryPointState)
519     {
520       case PointState::UP:
521       {
522         Actor* touchDownConsumedActor(mTouchDownConsumedActor.GetActor());
523         if(touchDownConsumedActor &&
524            touchDownConsumedActor != consumedActor &&
525            touchDownConsumedActor != lastPrimaryHitActor &&
526            touchDownConsumedActor != lastConsumedActor)
527         {
528           Dali::Actor touchDownConsumedActorHandle(touchDownConsumedActor);
529
530           Integration::Point currentPoint = touchEventImpl->GetPoint(0);
531           currentPoint.SetHitActor(touchDownConsumedActorHandle);
532           currentPoint.SetState(PointState::INTERRUPTED);
533
534           AllocAndEmitTouchSignals(event.time, touchDownConsumedActorHandle, currentPoint);
535         }
536
537         mTouchDownConsumedActor.SetActor(nullptr);
538         mInterceptedTouchActor.SetActor(nullptr);
539
540         DALI_FALLTHROUGH;
541       }
542
543       case PointState::DOWN:
544       {
545         mScene.EmitTouchedSignal(touchEventHandle);
546         break;
547       }
548
549       case PointState::MOTION:
550       case PointState::LEAVE:
551       case PointState::STATIONARY:
552       case PointState::INTERRUPTED:
553       {
554         // Ignore
555         break;
556       }
557     }
558   }
559
560   return consumed;
561 }
562
563 void TouchEventProcessor::OnObservedActorDisconnected(Actor* actor)
564 {
565   if(actor == mLastPrimaryHitActor.GetActor())
566   {
567     Dali::Actor actorHandle(actor);
568
569     Integration::Point point;
570     point.SetState(PointState::INTERRUPTED);
571     point.SetHitActor(actorHandle);
572
573     TouchEventPtr touchEventImpl(new TouchEvent);
574     touchEventImpl->AddPoint(point);
575     Dali::TouchEvent touchEventHandle(touchEventImpl.Get());
576
577     Dali::Actor eventConsumer = EmitTouchSignals(actorHandle, touchEventHandle);
578
579     if(mLastConsumedActor.GetActor() != eventConsumer)
580     {
581       EmitTouchSignals(Dali::Actor(mLastConsumedActor.GetActor()), touchEventHandle);
582     }
583
584     // Do not set mLastPrimaryHitActor to NULL we may be iterating through its observers
585
586     mLastConsumedActor.SetActor(nullptr);
587     mLastRenderTask.Reset();
588     mLastPrimaryPointState = PointState::FINISHED;
589   }
590 }
591
592 } // namespace Internal
593
594 } // namespace Dali