Merge branch 'tizen' into devel/new_mesh
[platform/core/uifw/dali-adaptor.git] / adaptors / x11 / window-impl-x.cpp
1 /*
2  * Copyright (c) 2014 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 "window-impl.h"
20
21 // EXTERNAL HEADERS
22 #include <Ecore.h>
23 #include <Ecore_X.h>
24
25 #include <dali/integration-api/core.h>
26 #include <dali/integration-api/system-overlay.h>
27 #include <dali/public-api/render-tasks/render-task.h>
28 #include <dali/public-api/render-tasks/render-task-list.h>
29
30 // INTERNAL HEADERS
31 #include <window-render-surface.h>
32 #include <drag-and-drop-detector-impl.h>
33 #include <indicator-impl.h>
34 #include <window-visibility-observer.h>
35 #include <orientation.h>
36 #include <orientation-impl.h>
37
38 namespace
39 {
40 const float INDICATOR_ANIMATION_DURATION( 0.18f ); // 180 milli seconds
41 const float INDICATOR_SHOW_Y_POSITION( 0.0f );
42 const float INDICATOR_HIDE_Y_POSITION( -52.0f );
43 }
44
45 namespace Dali
46 {
47 namespace Internal
48 {
49 namespace Adaptor
50 {
51 #if defined(DEBUG_ENABLED)
52 Debug::Filter* gWindowLogFilter = Debug::Filter::New(Debug::Concise, false, "LOG_WINDOW");
53 #endif
54
55 /**
56  * TODO: Abstract Window class out and move this into a window implementation for Ecore
57  */
58 struct Window::EventHandler
59 {
60   /**
61    * Constructor
62    * @param[in]  window  A pointer to the window class.
63    */
64   EventHandler( Window* window )
65   : mWindow( window ),
66     mWindowPropertyHandler( ecore_event_handler_add( ECORE_X_EVENT_WINDOW_PROPERTY,  EcoreEventWindowPropertyChanged, this ) ),
67     mClientMessagehandler( ecore_event_handler_add( ECORE_X_EVENT_CLIENT_MESSAGE,  EcoreEventClientMessage, this ) ),
68     mEcoreWindow( 0 )
69   {
70     // store ecore window handle
71     ECore::WindowRenderSurface* x11Window( dynamic_cast< ECore::WindowRenderSurface * >( mWindow->mSurface ) );
72     if( x11Window )
73     {
74       mEcoreWindow = x11Window->GetXWindow();
75     }
76     DALI_ASSERT_ALWAYS( mEcoreWindow != 0 && "There is no ecore x window");
77
78 #ifndef DALI_PROFILE_UBUNTU
79     // set property on window to get deiconify approve client message
80     unsigned int tmp = 1;
81     ecore_x_window_prop_card32_set(mEcoreWindow,
82                              ECORE_X_ATOM_E_DEICONIFY_APPROVE,
83                              &tmp, 1);
84 #endif // DALI_PROFILE_UBUNTU
85
86     ecore_x_input_multi_select( mEcoreWindow );
87   }
88
89   /**
90    * Destructor
91    */
92   ~EventHandler()
93   {
94     if ( mWindowPropertyHandler )
95     {
96       ecore_event_handler_del( mWindowPropertyHandler );
97     }
98     if ( mClientMessagehandler )
99     {
100       ecore_event_handler_del( mClientMessagehandler );
101     }
102   }
103
104   // Static methods
105
106   /// Called when the window properties are changed.
107   static Eina_Bool EcoreEventWindowPropertyChanged( void* data, int type, void* event )
108   {
109     Ecore_X_Event_Window_Property* propertyChangedEvent( (Ecore_X_Event_Window_Property*)event );
110     EventHandler* handler( (EventHandler*)data );
111     Eina_Bool handled( ECORE_CALLBACK_PASS_ON );
112
113     if ( handler && handler->mWindow )
114     {
115       WindowVisibilityObserver* observer( handler->mWindow->mAdaptor );
116       if ( observer && ( propertyChangedEvent->win == handler->mEcoreWindow ) )
117       {
118         Ecore_X_Window_State_Hint state( ecore_x_icccm_state_get( propertyChangedEvent->win ) );
119
120         switch ( state )
121         {
122           case ECORE_X_WINDOW_STATE_HINT_WITHDRAWN:
123           {
124             // Window was hidden.
125             observer->OnWindowHidden();
126             DALI_LOG_INFO( gWindowLogFilter, Debug::General, "Window (%d) Withdrawn\n", handler->mEcoreWindow );
127             handled = ECORE_CALLBACK_DONE;
128           }
129           break;
130
131           case ECORE_X_WINDOW_STATE_HINT_ICONIC:
132           {
133             // Window was iconified (minimised).
134             observer->OnWindowHidden();
135             DALI_LOG_INFO( gWindowLogFilter, Debug::General, "Window (%d) Iconfied\n", handler->mEcoreWindow );
136             handled = ECORE_CALLBACK_DONE;
137           }
138           break;
139
140           case ECORE_X_WINDOW_STATE_HINT_NORMAL:
141           {
142             // Window was shown.
143             observer->OnWindowShown();
144             DALI_LOG_INFO( gWindowLogFilter, Debug::General, "Window (%d) Shown\n", handler->mEcoreWindow );
145             handled = ECORE_CALLBACK_DONE;
146           }
147           break;
148
149           default:
150             // Ignore
151             break;
152         }
153       }
154     }
155
156     return handled;
157   }
158
159   /// Called when the window properties are changed.
160   static Eina_Bool EcoreEventClientMessage( void* data, int type, void* event )
161   {
162     Eina_Bool handled( ECORE_CALLBACK_PASS_ON );
163 #ifndef DALI_PROFILE_UBUNTU
164     Ecore_X_Event_Client_Message* clientMessageEvent( (Ecore_X_Event_Client_Message*)event );
165     EventHandler* handler( (EventHandler*)data );
166
167     if (clientMessageEvent->message_type == ECORE_X_ATOM_E_DEICONIFY_APPROVE)
168     {
169       ECore::WindowRenderSurface* x11Window( dynamic_cast< ECore::WindowRenderSurface * >( handler->mWindow->mSurface ) );
170       WindowVisibilityObserver* observer( handler->mWindow->mAdaptor );
171
172       if ( observer && ( (unsigned int)clientMessageEvent->data.l[0] == handler->mEcoreWindow ) )
173       {
174         if (clientMessageEvent->data.l[1] == 0) //wm sends request message using value 0
175         {
176           observer->OnWindowShown();
177
178           // request to approve the deiconify. render-surface should send proper event after real rendering
179           if(x11Window)
180           {
181             x11Window->RequestToApproveDeiconify();
182           }
183
184           handled = ECORE_CALLBACK_DONE;
185         }
186       }
187     }
188 #endif // DALI_PROFILE_UBUNTU
189
190     return handled;
191   }
192
193   // Data
194   Window* mWindow;
195   Ecore_Event_Handler* mWindowPropertyHandler;
196   Ecore_Event_Handler* mClientMessagehandler;
197   Ecore_X_Window mEcoreWindow;
198 };
199
200
201 Window* Window::New(const PositionSize& posSize, const std::string& name, bool isTransparent)
202 {
203   Window* window = new Window();
204   window->mIsTransparent = isTransparent;
205   window->Initialize(posSize, name);
206   return window;
207 }
208
209 void Window::SetAdaptor(Dali::Adaptor& adaptor)
210 {
211   DALI_ASSERT_ALWAYS( !mStarted && "Adaptor already started" );
212   mStarted = true;
213
214   // Only create one overlay per window
215   Internal::Adaptor::Adaptor& adaptorImpl = Internal::Adaptor::Adaptor::GetImplementation(adaptor);
216   Integration::Core& core = adaptorImpl.GetCore();
217   mOverlay = &core.GetSystemOverlay();
218
219   Dali::RenderTaskList taskList = mOverlay->GetOverlayRenderTasks();
220   taskList.CreateTask();
221
222   mAdaptor = &adaptorImpl;
223   mAdaptor->AddObserver( *this );
224
225   // Can only create the detector when we know the Core has been instantiated.
226   mDragAndDropDetector = DragAndDropDetector::New();
227   mAdaptor->SetDragAndDropDetector( &GetImplementation( mDragAndDropDetector ) );
228
229   if( mOrientation )
230   {
231     mOrientation->SetAdaptor(adaptor);
232   }
233
234   if( mIndicator != NULL )
235   {
236     mIndicator->SetAdaptor(mAdaptor);
237   }
238 }
239
240 RenderSurface* Window::GetSurface()
241 {
242   return mSurface;
243 }
244
245 void Window::ShowIndicator( Dali::Window::IndicatorVisibleMode visibleMode )
246 {
247   DALI_LOG_TRACE_METHOD_FMT( gWindowLogFilter, "visible : %d\n", visibleMode );
248   DALI_ASSERT_DEBUG(mOverlay);
249
250   ECore::WindowRenderSurface* x11Window = dynamic_cast< ECore::WindowRenderSurface * >( mSurface );
251   DALI_ASSERT_DEBUG(x11Window);
252   Ecore_X_Window xWinId = x11Window->GetXWindow();
253
254   mIndicatorVisible = visibleMode;
255
256   if ( mIndicatorVisible == Dali::Window::VISIBLE )
257   {
258     // when the indicator is visible, set proper mode for indicator server according to bg mode
259     if ( mIndicatorOpacityMode == Dali::Window::OPAQUE )
260     {
261       ecore_x_e_illume_indicator_opacity_set(xWinId, ECORE_X_ILLUME_INDICATOR_OPAQUE);
262     }
263     else if ( mIndicatorOpacityMode == Dali::Window::TRANSLUCENT )
264     {
265       ecore_x_e_illume_indicator_opacity_set(xWinId, ECORE_X_ILLUME_INDICATOR_TRANSLUCENT);
266     }
267 #if defined(DALI_PROFILE_MOBILE)
268     else if ( mIndicatorOpacityMode == Dali::Window::TRANSPARENT )
269     {
270       ecore_x_e_illume_indicator_opacity_set(xWinId, ECORE_X_ILLUME_INDICATOR_OPAQUE);
271     }
272 #endif
273   }
274   else
275   {
276     // when the indicator is not visible, set TRANSPARENT mode for indicator server
277     ecore_x_e_illume_indicator_opacity_set(xWinId, ECORE_X_ILLUME_INDICATOR_TRANSPARENT); // it means hidden indicator
278   }
279
280   DoShowIndicator( mIndicatorOrientation );
281 }
282
283 void Window::RotateIndicator(Dali::Window::WindowOrientation orientation)
284 {
285   DALI_LOG_TRACE_METHOD_FMT( gWindowLogFilter, "Orientation: %d\n", orientation );
286
287   DoRotateIndicator( orientation );
288 }
289
290 void Window::SetIndicatorBgOpacity( Dali::Window::IndicatorBgOpacity opacityMode )
291 {
292   mIndicatorOpacityMode = opacityMode;
293
294   if( mIndicator != NULL )
295   {
296     mIndicator->SetOpacityMode( opacityMode );
297   }
298 }
299
300 void Window::SetClass(std::string name, std::string klass)
301 {
302   // Get render surface's x11 window
303   if( mSurface )
304   {
305     ECore::WindowRenderSurface* x11Window = dynamic_cast< ECore::WindowRenderSurface * >( mSurface );
306     if( x11Window )
307     {
308       ecore_x_icccm_name_class_set( x11Window->GetXWindow(), name.c_str(), klass.c_str() );
309     }
310   }
311 }
312
313 Window::Window()
314 : mSurface(NULL),
315   mIndicatorVisible(Dali::Window::INVISIBLE),
316   mIndicatorIsShown(false),
317   mShowRotatedIndicatorOnClose(false),
318   mStarted(false),
319   mIsTransparent(false),
320   mWMRotationAppSet(false),
321   mIndicator(NULL),
322   mIndicatorOrientation(Dali::Window::PORTRAIT),
323   mNextIndicatorOrientation(Dali::Window::PORTRAIT),
324   mIndicatorOpacityMode(Dali::Window::OPAQUE),
325   mOverlay(NULL),
326   mAdaptor(NULL),
327   mEventHandler(NULL),
328   mPreferredOrientation(Dali::Window::PORTRAIT)
329 {
330 }
331
332 Window::~Window()
333 {
334   delete mEventHandler;
335
336   if( mIndicator )
337   {
338     mOverlay->Remove( mIndicator->GetActor() );
339     Dali::RenderTaskList taskList = mOverlay->GetOverlayRenderTasks();
340     Dali::RenderTask indicatorTask = taskList.GetTask(0);
341     mOverlay->GetOverlayRenderTasks().RemoveTask(indicatorTask);
342     mIndicator->Close();
343     delete mIndicator;
344   }
345
346   if( mAdaptor )
347   {
348     mAdaptor->RemoveObserver( *this );
349     mAdaptor->SetDragAndDropDetector( NULL );
350     mAdaptor = NULL;
351   }
352
353   delete mSurface;
354 }
355
356 void Window::Initialize(const PositionSize& windowPosition, const std::string& name)
357 {
358   // create an X11 window by default
359   Any surface;
360   ECore::WindowRenderSurface* windowSurface = new ECore::WindowRenderSurface( windowPosition, surface, name, mIsTransparent );
361   windowSurface->Map();
362
363   mSurface = windowSurface;
364
365   mOrientation = Orientation::New(this);
366
367   // create event handler for X11 window
368   mEventHandler = new EventHandler( this );
369 }
370
371 void Window::DoShowIndicator( Dali::Window::WindowOrientation lastOrientation )
372 {
373   if( mIndicator == NULL )
374   {
375     if( mIndicatorVisible != Dali::Window::INVISIBLE )
376     {
377       mIndicator = new Indicator( mAdaptor, mIndicatorOrientation, this );
378       mIndicator->SetOpacityMode( mIndicatorOpacityMode );
379       Dali::Actor actor = mIndicator->GetActor();
380       SetIndicatorActorRotation();
381       mOverlay->Add(actor);
382     }
383     // else don't create a hidden indicator
384   }
385   else // Already have indicator
386   {
387     if( mIndicatorVisible == Dali::Window::VISIBLE )
388     {
389       // If we are resuming, and rotation has changed,
390       if( mIndicatorIsShown == false && mIndicatorOrientation != mNextIndicatorOrientation )
391       {
392         // then close current indicator and open new one
393         mShowRotatedIndicatorOnClose = true;
394         mIndicator->Close(); // May synchronously call IndicatorClosed() callback & 1 level of recursion
395         // Don't show actor - will contain indicator for old orientation.
396       }
397     }
398   }
399
400   // set indicator visible mode
401   if( mIndicator != NULL )
402   {
403     mIndicator->SetVisible( mIndicatorVisible );
404   }
405
406   bool show = (mIndicatorVisible != Dali::Window::INVISIBLE );
407   SetIndicatorProperties( show, lastOrientation );
408   mIndicatorIsShown = show;
409 }
410
411 void Window::DoRotateIndicator( Dali::Window::WindowOrientation orientation )
412 {
413   if( mIndicatorIsShown )
414   {
415     mShowRotatedIndicatorOnClose = true;
416     mNextIndicatorOrientation = orientation;
417     mIndicator->Close(); // May synchronously call IndicatorClosed() callback
418   }
419   else
420   {
421     // Save orientation for when the indicator is next shown
422     mShowRotatedIndicatorOnClose = false;
423     mNextIndicatorOrientation = orientation;
424   }
425 }
426
427 void Window::SetIndicatorProperties( bool isShow, Dali::Window::WindowOrientation lastOrientation )
428 {
429   ECore::WindowRenderSurface* x11Window = dynamic_cast< ECore::WindowRenderSurface * >( mSurface );
430   if( x11Window )
431   {
432     Ecore_X_Window win = x11Window->GetXWindow();
433
434     int show_state = static_cast<int>( isShow );
435     ecore_x_window_prop_property_set( win,
436                                       ECORE_X_ATOM_E_ILLUME_INDICATOR_STATE,
437                                       ECORE_X_ATOM_CARDINAL, 32, &show_state, 1);
438
439     if ( isShow )
440     {
441       ecore_x_e_illume_indicator_state_set(win, ECORE_X_ILLUME_INDICATOR_STATE_ON);
442     }
443     else
444     {
445       ecore_x_e_illume_indicator_state_set(win, ECORE_X_ILLUME_INDICATOR_STATE_OFF);
446     }
447   }
448 }
449
450 void Window::IndicatorTypeChanged(Indicator::Type type)
451 {
452   ECore::WindowRenderSurface* x11Window = dynamic_cast< ECore::WindowRenderSurface * >( mSurface );
453   if( x11Window )
454   {
455 #ifndef DALI_PROFILE_UBUNTU
456     Ecore_X_Window win = x11Window->GetXWindow();
457     switch(type)
458     {
459       case Indicator::INDICATOR_TYPE_1:
460         ecore_x_e_illume_indicator_type_set( win, ECORE_X_ILLUME_INDICATOR_TYPE_1 );
461         break;
462
463       case Indicator::INDICATOR_TYPE_2:
464         ecore_x_e_illume_indicator_type_set( win, ECORE_X_ILLUME_INDICATOR_TYPE_2 );
465         break;
466
467       case Indicator::INDICATOR_TYPE_UNKNOWN:
468       default:
469         break;
470     }
471 #endif // DALI_PROFILE_UBUNTU
472   }
473 }
474
475 void Window::IndicatorClosed( Indicator* indicator )
476 {
477   DALI_LOG_TRACE_METHOD( gWindowLogFilter );
478
479   if( mShowRotatedIndicatorOnClose )
480   {
481     Dali::Window::WindowOrientation currentOrientation = mIndicatorOrientation;
482     mIndicator->Open(mNextIndicatorOrientation);
483     mIndicatorOrientation = mNextIndicatorOrientation;
484     SetIndicatorActorRotation();
485     DoShowIndicator(currentOrientation);
486   }
487 }
488
489 void Window::IndicatorVisibilityChanged(bool isVisible)
490 {
491   mIndicatorVisibilityChangedSignal.Emit(isVisible);
492 }
493
494 void Window::SetIndicatorActorRotation()
495 {
496   DALI_LOG_TRACE_METHOD( gWindowLogFilter );
497   DALI_ASSERT_DEBUG( mIndicator != NULL );
498
499   Dali::Actor actor = mIndicator->GetActor();
500   switch( mIndicatorOrientation )
501   {
502     case Dali::Window::PORTRAIT:
503       actor.SetParentOrigin( ParentOrigin::TOP_CENTER );
504       actor.SetAnchorPoint(  AnchorPoint::TOP_CENTER );
505       actor.SetOrientation( Degree(0), Vector3::ZAXIS );
506       break;
507     case Dali::Window::PORTRAIT_INVERSE:
508       actor.SetParentOrigin( ParentOrigin::BOTTOM_CENTER );
509       actor.SetAnchorPoint(  AnchorPoint::TOP_CENTER );
510       actor.SetOrientation( Degree(180), Vector3::ZAXIS );
511       break;
512     case Dali::Window::LANDSCAPE:
513       actor.SetParentOrigin( ParentOrigin::CENTER_LEFT );
514       actor.SetAnchorPoint(  AnchorPoint::TOP_CENTER );
515       actor.SetOrientation( Degree(270), Vector3::ZAXIS );
516       break;
517     case Dali::Window::LANDSCAPE_INVERSE:
518       actor.SetParentOrigin( ParentOrigin::CENTER_RIGHT );
519       actor.SetAnchorPoint(  AnchorPoint::TOP_CENTER );
520       actor.SetOrientation( Degree(90), Vector3::ZAXIS );
521       break;
522   }
523 }
524
525 void Window::Raise()
526 {
527   ECore::WindowRenderSurface* x11Window = dynamic_cast< ECore::WindowRenderSurface * >( mSurface );
528   if( x11Window )
529   {
530     Ecore_X_Window win = x11Window->GetXWindow();
531     ecore_x_window_raise(win);
532   }
533 }
534
535 void Window::Lower()
536 {
537   ECore::WindowRenderSurface* x11Window = dynamic_cast< ECore::WindowRenderSurface * >( mSurface );
538   if( x11Window )
539   {
540     Ecore_X_Window win = x11Window->GetXWindow();
541     ecore_x_window_lower(win);
542   }
543 }
544
545 void Window::Activate()
546 {
547   ECore::WindowRenderSurface* x11Window = dynamic_cast< ECore::WindowRenderSurface * >( mSurface );
548   if( x11Window )
549   {
550     Ecore_X_Window win = x11Window->GetXWindow();
551     ecore_x_netwm_client_active_request(ecore_x_window_root_get(win), win, 1 /* request type, 1:application, 2:pager */, 0);
552   }
553 }
554
555 Dali::DragAndDropDetector Window::GetDragAndDropDetector() const
556 {
557   return mDragAndDropDetector;
558 }
559
560 Dali::Any Window::GetNativeHandle() const
561 {
562   if(mEventHandler)
563   {
564     return mEventHandler->mEcoreWindow;
565   }
566   else
567   {
568     return Dali::Any();
569   }
570 }
571
572 void Window::OnStart()
573 {
574   ShowIndicator( mIndicatorVisible );
575 }
576
577 void Window::OnPause()
578 {
579 }
580
581 void Window::OnResume()
582 {
583   // resume indicator status
584   if( mIndicator != NULL )
585   {
586     // Restore own indicator opacity
587     // Send opacity mode to indicator service when app resumed
588     mIndicator->SetOpacityMode( mIndicatorOpacityMode );
589   }
590 }
591
592 void Window::OnStop()
593 {
594   if( mIndicator )
595   {
596     mIndicator->Close();
597   }
598
599   delete mIndicator;
600   mIndicator = NULL;
601 }
602
603 void Window::OnDestroy()
604 {
605   mAdaptor = NULL;
606 }
607
608 void Window::AddAvailableOrientation(Dali::Window::WindowOrientation orientation)
609 {
610   bool found = false;
611
612   for( std::size_t i=0; i<mAvailableOrientations.size(); i++ )
613   {
614     if(mAvailableOrientations[i] == orientation)
615     {
616       found = true;
617       break;
618     }
619   }
620
621   if( ! found )
622   {
623     mAvailableOrientations.push_back(orientation);
624     SetAvailableOrientations( mAvailableOrientations );
625   }
626 }
627
628 void Window::RemoveAvailableOrientation(Dali::Window::WindowOrientation orientation)
629 {
630   for( std::vector<Dali::Window::WindowOrientation>::iterator iter = mAvailableOrientations.begin();
631        iter != mAvailableOrientations.end(); ++iter )
632   {
633     if( *iter == orientation )
634     {
635       mAvailableOrientations.erase( iter );
636       break;
637     }
638   }
639   SetAvailableOrientations( mAvailableOrientations );
640 }
641
642 void Window::SetAvailableOrientations(const std::vector<Dali::Window::WindowOrientation>& orientations)
643 {
644   DALI_ASSERT_ALWAYS( mAvailableOrientations.size() <= 4 && "Incorrect number of available orientations" );
645
646   mAvailableOrientations = orientations;
647   ECore::WindowRenderSurface* x11Window = dynamic_cast< ECore::WindowRenderSurface * >( mSurface );
648   if( x11Window )
649   {
650 #ifndef DALI_PROFILE_UBUNTU
651     Ecore_X_Window ecoreWindow = x11Window->GetXWindow();
652     if( ! mWMRotationAppSet )
653     {
654       mWMRotationAppSet = true;
655       ecore_x_e_window_rotation_app_set(ecoreWindow, EINA_TRUE);
656     }
657
658     int rotations[4];
659     for( std::size_t i=0; i<mAvailableOrientations.size(); i++ )
660     {
661       rotations[i] = static_cast<int>(mAvailableOrientations[i]);
662     }
663     ecore_x_e_window_rotation_available_rotations_set(ecoreWindow, rotations, mAvailableOrientations.size() );
664 #endif // DALI_PROFILE_UBUNTU
665   }
666 }
667
668 const std::vector<Dali::Window::WindowOrientation>& Window::GetAvailableOrientations()
669 {
670   return mAvailableOrientations;
671 }
672
673 void Window::SetPreferredOrientation(Dali::Window::WindowOrientation orientation)
674 {
675   mPreferredOrientation = orientation;
676
677   ECore::WindowRenderSurface* x11Window = dynamic_cast< ECore::WindowRenderSurface * >( mSurface );
678   if( x11Window )
679   {
680 #ifndef DALI_PROFILE_UBUNTU
681     Ecore_X_Window ecoreWindow = x11Window->GetXWindow();
682
683     if( ! mWMRotationAppSet )
684     {
685       mWMRotationAppSet = true;
686       ecore_x_e_window_rotation_app_set(ecoreWindow, EINA_TRUE);
687     }
688
689     ecore_x_e_window_rotation_preferred_rotation_set(ecoreWindow, orientation);
690 #endif // DALI_PROFILE_UBUNTU
691   }
692 }
693
694 Dali::Window::WindowOrientation Window::GetPreferredOrientation()
695 {
696   return mPreferredOrientation;
697 }
698
699 void Window::RotationDone( int orientation, int width, int height )
700 {
701   // Tell window manager we're done
702   ECore::WindowRenderSurface* x11Window = dynamic_cast< ECore::WindowRenderSurface * >( mSurface );
703   if( x11Window )
704   {
705 #ifndef DALI_PROFILE_UBUNTU
706     Ecore_X_Window ecoreWindow = x11Window->GetXWindow();
707     Ecore_X_Window root = ecore_x_window_root_get(ecoreWindow);
708
709     /**
710      * send rotation done message to wm, even if window is already rotated.
711      * that's why wm must be wait for comming rotation done message
712      * after sending rotation request.
713      */
714     ecore_x_e_window_rotation_change_done_send(root, ecoreWindow, orientation, width, height);
715
716     /**
717      * set rotate window property
718      */
719     int angles[2] = { orientation, orientation };
720     ecore_x_window_prop_property_set( ecoreWindow,
721                                      ECORE_X_ATOM_E_ILLUME_ROTATE_WINDOW_ANGLE,
722                                      ECORE_X_ATOM_CARDINAL, 32, &angles, 2 );
723 #endif // DALI_PROFILE_UBUNTU
724   }
725 }
726
727
728 } // Adaptor
729 } // Internal
730 } // Dali