[Tizen] Add screen and client rotation itself function
[platform/core/uifw/dali-adaptor.git] / dali / integration-api / adaptor-framework / scene-holder-impl.cpp
1 /*
2  * Copyright (c) 2020 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/integration-api/adaptor-framework/scene-holder-impl.h>
20
21 // EXTERNAL INCLUDES
22 #include <sys/time.h>
23 #include <dali/public-api/actors/actor.h>
24 #include <dali/public-api/actors/layer.h>
25 #include <dali/integration-api/debug.h>
26 #include <dali/integration-api/events/key-event-integ.h>
27 #include <dali/integration-api/events/touch-event-integ.h>
28 #include <dali/integration-api/events/hover-event-integ.h>
29 #include <dali/integration-api/events/wheel-event-integ.h>
30
31 // INTERNAL INCLUDES
32 #include <dali/internal/adaptor/common/adaptor-impl.h>
33 #include <dali/internal/adaptor/common/lifecycle-observer.h>
34 #include <dali/internal/input/common/key-impl.h>
35 #include <dali/internal/input/common/physical-keyboard-impl.h>
36
37 namespace Dali
38 {
39
40 namespace Internal
41 {
42
43 namespace Adaptor
44 {
45
46 namespace
47 {
48
49 #if defined(DEBUG_ENABLED)
50 Integration::Log::Filter* gTouchEventLogFilter  = Integration::Log::Filter::New(Debug::NoLogging, false, "LOG_ADAPTOR_EVENTS_TOUCH");
51 #endif
52
53 // Copied from x server
54 static uint32_t GetCurrentMilliSeconds(void)
55 {
56   struct timeval tv;
57
58   struct timespec tp;
59   static clockid_t clockid;
60
61   if (!clockid)
62   {
63 #ifdef CLOCK_MONOTONIC_COARSE
64     if (clock_getres(CLOCK_MONOTONIC_COARSE, &tp) == 0 &&
65       (tp.tv_nsec / 1000) <= 1000 && clock_gettime(CLOCK_MONOTONIC_COARSE, &tp) == 0)
66     {
67       clockid = CLOCK_MONOTONIC_COARSE;
68     }
69     else
70 #endif
71     if (clock_gettime(CLOCK_MONOTONIC, &tp) == 0)
72     {
73       clockid = CLOCK_MONOTONIC;
74     }
75     else
76     {
77       clockid = ~0L;
78     }
79   }
80   if (clockid != ~0L && clock_gettime(clockid, &tp) == 0)
81   {
82     return static_cast<uint32_t>( (tp.tv_sec * 1000 ) + (tp.tv_nsec / 1000000L) );
83   }
84
85   gettimeofday(&tv, NULL);
86   return static_cast<uint32_t>( (tv.tv_sec * 1000 ) + (tv.tv_usec / 1000) );
87 }
88
89 } // unnamed namespace
90
91 uint32_t SceneHolder::mSceneHolderCounter = 0;
92
93 class SceneHolder::SceneHolderLifeCycleObserver : public LifeCycleObserver
94 {
95 public:
96
97   SceneHolderLifeCycleObserver(Adaptor*& adaptor)
98   : mAdaptor( adaptor )
99   {
100   };
101
102 private: // Adaptor::LifeCycleObserver interface
103
104   void OnStart() override {};
105   void OnPause() override {};
106   void OnResume() override {};
107   void OnStop() override {};
108   void OnDestroy() override
109   {
110     mAdaptor = nullptr;
111   };
112
113 private:
114   Adaptor*& mAdaptor;
115 };
116
117
118 SceneHolder::SceneHolder()
119 : mLifeCycleObserver( new SceneHolderLifeCycleObserver( mAdaptor ) ),
120   mId( mSceneHolderCounter++ ),
121   mSurface( nullptr ),
122   mAdaptor( nullptr ),
123   mIsBeingDeleted( false ),
124   mAdaptorStarted( false ),
125   mVisible( true )
126 {
127 }
128
129 SceneHolder::~SceneHolder()
130 {
131   if ( mAdaptor )
132   {
133     mAdaptor->RemoveObserver( *mLifeCycleObserver.get() );
134     mAdaptor->RemoveWindow( this );
135
136     mAdaptor = nullptr;
137   }
138
139   if ( mScene )
140   {
141     mScene.Discard();
142   }
143 }
144
145 void SceneHolder::Add( Dali::Actor actor )
146 {
147   if ( mScene )
148   {
149     mScene.Add( actor );
150   }
151 }
152
153 void SceneHolder::Remove( Dali::Actor actor )
154 {
155   if ( mScene )
156   {
157     mScene.Remove( actor );
158   }
159 }
160
161 Dali::Layer SceneHolder::GetRootLayer() const
162 {
163   return mScene ? mScene.GetRootLayer() : Dali::Layer();
164 }
165
166 uint32_t SceneHolder::GetId() const
167 {
168   return mId;
169 }
170
171 std::string SceneHolder::GetName() const
172 {
173   return mName;
174 }
175
176 bool SceneHolder::IsVisible() const
177 {
178   return mVisible;
179 }
180
181 Dali::Integration::Scene SceneHolder::GetScene()
182 {
183   return mScene;
184 }
185
186 void SceneHolder::SetSurface(Dali::RenderSurfaceInterface* surface)
187 {
188   mSurface.reset( surface );
189
190   mScene.SurfaceReplaced();
191
192   SurfaceResized( false );
193
194   unsigned int dpiHorizontal, dpiVertical;
195   dpiHorizontal = dpiVertical = 0;
196
197   mSurface->GetDpi( dpiHorizontal, dpiVertical );
198   mScene.SetDpi( Vector2( static_cast<float>( dpiHorizontal ), static_cast<float>( dpiVertical ) ) );
199
200   mSurface->SetAdaptor( *mAdaptor );
201
202   OnSurfaceSet( surface );
203 }
204
205 void SceneHolder::SurfaceResized( bool forceUpdate )
206 {
207   PositionSize surfacePositionSize = mSurface->GetPositionSize();
208   int orientation = mSurface->GetOrientation();
209   mScene.SurfaceResized( static_cast<float>( surfacePositionSize.width ), static_cast<float>( surfacePositionSize.height ), orientation, forceUpdate );
210 }
211
212 Dali::RenderSurfaceInterface* SceneHolder::GetSurface() const
213 {
214   return mSurface.get();
215 }
216
217 void SceneHolder::SetBackgroundColor( const Vector4& color )
218 {
219   if( mScene )
220   {
221     mScene.SetBackgroundColor( color );
222   }
223 }
224
225 Vector4 SceneHolder::GetBackgroundColor() const
226 {
227   return mScene ? mScene.GetBackgroundColor() : Color::BLACK;
228 }
229
230 void SceneHolder::SetAdaptor(Dali::Adaptor& adaptor)
231 {
232   // Avoid doing this more than once
233   if( mAdaptorStarted )
234   {
235     return;
236   }
237
238   mAdaptorStarted = true;
239
240   // Create the scene
241   PositionSize surfacePositionSize = mSurface->GetPositionSize();
242   mScene = Dali::Integration::Scene::New( Size(static_cast<float>( surfacePositionSize.width ), static_cast<float>( surfacePositionSize.height )) );
243
244   Internal::Adaptor::Adaptor& adaptorImpl = Internal::Adaptor::Adaptor::GetImplementation( adaptor );
245   mAdaptor = &adaptorImpl;
246
247   // Create an observer for the adaptor lifecycle
248   mAdaptor->AddObserver( *mLifeCycleObserver );
249
250   if ( mSurface )
251   {
252     unsigned int dpiHorizontal, dpiVertical;
253     dpiHorizontal = dpiVertical = 0;
254
255     mSurface->GetDpi( dpiHorizontal, dpiVertical );
256     mScene.SetDpi( Vector2( static_cast<float>( dpiHorizontal ), static_cast<float>( dpiVertical ) ) );
257
258     mSurface->SetAdaptor( *mAdaptor );
259   }
260
261   OnAdaptorSet( adaptor );
262 }
263
264 void SceneHolder::Pause()
265 {
266   Reset();
267
268   OnPause();
269 }
270
271 void SceneHolder::Resume()
272 {
273   Reset();
274
275   OnResume();
276 }
277
278 void SceneHolder::FeedTouchPoint( Dali::Integration::Point& point, int timeStamp )
279 {
280   if( timeStamp < 1 )
281   {
282     timeStamp = GetCurrentMilliSeconds();
283   }
284
285   RecalculateTouchPosition( point );
286
287   Integration::TouchEvent touchEvent;
288   Integration::HoverEvent hoverEvent;
289   Integration::TouchEventCombiner::EventDispatchType type = mCombiner.GetNextTouchEvent(point, timeStamp, touchEvent, hoverEvent);
290   if( type != Integration::TouchEventCombiner::DispatchNone )
291   {
292     DALI_LOG_INFO( gTouchEventLogFilter, Debug::General, "%d: Device %d: Button state %d (%.2f, %.2f)\n", timeStamp, point.GetDeviceId(), point.GetState(), point.GetScreenPosition().x, point.GetScreenPosition().y );
293
294     // Signals can be emitted while processing core events, and the scene holder could be deleted in the signal callback.
295     // Keep the handle alive until the core events are processed.
296     Dali::BaseHandle sceneHolder( this );
297
298     // First the touch and/or hover event & related gesture events are queued
299     if( type == Integration::TouchEventCombiner::DispatchTouch || type == Integration::TouchEventCombiner::DispatchBoth )
300     {
301       mScene.QueueEvent( touchEvent );
302     }
303
304     if( type == Integration::TouchEventCombiner::DispatchHover || type == Integration::TouchEventCombiner::DispatchBoth )
305     {
306       mScene.QueueEvent( hoverEvent );
307     }
308
309     // Next the events are processed with a single call into Core
310     mAdaptor->ProcessCoreEvents();
311   }
312 }
313
314 void SceneHolder::FeedWheelEvent( Dali::Integration::WheelEvent& wheelEvent )
315 {
316   // Signals can be emitted while processing core events, and the scene holder could be deleted in the signal callback.
317   // Keep the handle alive until the core events are processed.
318   Dali::BaseHandle sceneHolder( this );
319
320   mScene.QueueEvent( wheelEvent );
321   mAdaptor->ProcessCoreEvents();
322 }
323
324 void SceneHolder::FeedKeyEvent( Dali::Integration::KeyEvent& keyEvent )
325 {
326   Dali::PhysicalKeyboard physicalKeyboard = PhysicalKeyboard::Get();
327   if( physicalKeyboard )
328   {
329     if( ! KeyLookup::IsDeviceButton( keyEvent.keyName.c_str() ) )
330     {
331       GetImplementation( physicalKeyboard ).KeyReceived( keyEvent.time > 1 );
332     }
333   }
334
335   // Signals can be emitted while processing core events, and the scene holder could be deleted in the signal callback.
336   // Keep the handle alive until the core events are processed.
337   Dali::BaseHandle sceneHolder( this );
338
339   // Create send KeyEvent to Core.
340   mScene.QueueEvent( keyEvent );
341   mAdaptor->ProcessCoreEvents();
342 }
343
344 Dali::Integration::SceneHolder SceneHolder::Get( Dali::Actor actor )
345 {
346   SceneHolder* sceneHolderImpl = nullptr;
347
348   if ( Internal::Adaptor::Adaptor::IsAvailable() )
349   {
350     Dali::Internal::Adaptor::Adaptor& adaptor = Internal::Adaptor::Adaptor::GetImplementation( Internal::Adaptor::Adaptor::Get() );
351     sceneHolderImpl = adaptor.GetWindow( actor );
352   }
353
354   return Dali::Integration::SceneHolder( sceneHolderImpl );
355 }
356
357 void SceneHolder::Reset()
358 {
359   mCombiner.Reset();
360
361   // Any touch listeners should be told of the interruption.
362   Integration::TouchEvent event;
363   Integration::Point point;
364   point.SetState( PointState::INTERRUPTED );
365   event.AddPoint( point );
366
367   // First the touch event & related gesture events are queued
368   mScene.QueueEvent( event );
369
370   // Next the events are processed with a single call into Core
371   mAdaptor->ProcessCoreEvents();
372 }
373
374
375 }// Adaptor
376
377 }// Internal
378
379 } // Dali