2 * Copyright (c) 2015 Samsung Electronics Co., Ltd All Rights Reserved
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.
17 #include "runtime/browser/native_window.h"
19 #include <Ecore_Wayland.h>
21 #include <EWebKit_internal.h>
24 #include "common/arraysize.h"
25 #include "common/logger.h"
30 const char* kEdjePath = "/usr/share/edje/xwalk/xwalk_tizen.edj";
31 const char* kWinowRotationEventKey = "wm,rotation,changed";
32 const char* kWinowFocusedEventKey = "focused";
33 const char* kWinowUnfocusedEventKey = "unfocused";
35 const int kPortraitNaturalAngle[] = {
36 0, // PORTRAIT_PRIMARY
37 180, // PORTRAIT_SECONDARY
38 270, // LANDSCAPE_PRIMARY
39 90, // LANDSCAPE_SECONDARY
43 const int kLandscapeNaturalAngle[] = {
44 270, // PORTRAIT_PRIMARY
45 90, // PORTRAIT_SECONDARY
46 0, // LANDSCAPE_PRIMARY
47 180, // LANDSCAPE_SECONDARY
52 NativeWindow::ScreenOrientation NativeAngleToOrientation(
53 int angle, NativeWindow::ScreenOrientation natural_orientation) {
55 natural_orientation == NativeWindow::ScreenOrientation::PORTRAIT_PRIMARY ?
56 kPortraitNaturalAngle :
57 kLandscapeNaturalAngle;
58 unsigned index = ARRAYSIZE(convert_table) - 1;
59 for (unsigned i = 0; i < ARRAYSIZE(convert_table); ++i) {
60 if (convert_table[i] == angle) {
65 return static_cast<NativeWindow::ScreenOrientation>(index);
71 NativeWindow::NativeWindow()
73 window_type_(Type::NORMAL),
81 NativeWindow::~NativeWindow() {
83 evas_object_del(window_);
86 void NativeWindow::Initialize() {
88 LOGGER(DEBUG) << "already initialized";
92 window_ = CreateWindowInternal();
93 elm_win_conformant_set(window_, EINA_TRUE);
95 ecore_wl_screen_size_get(&w, &h);
96 evas_object_resize(window_, w, h);
97 elm_win_size_base_set(window_, w, h);
98 elm_win_autodel_set(window_, EINA_TRUE);
99 evas_object_smart_callback_add(window_, "delete,request",
100 DidDeleteRequested, this);
101 evas_object_smart_callback_add(window_, "profile,changed",
102 DidProfileChanged, this);
104 #define EVAS_SIZE_EXPAND_FILL(obj) \
105 evas_object_size_hint_weight_set(obj, EVAS_HINT_EXPAND, EVAS_HINT_EXPAND); \
106 evas_object_size_hint_align_set(obj, EVAS_HINT_FILL, EVAS_HINT_FILL);
109 Evas_Object* bg = evas_object_rectangle_add(evas_object_evas_get(window_));
110 evas_object_color_set(bg, 0, 0, 0, 255);
111 EVAS_SIZE_EXPAND_FILL(bg);
112 elm_win_resize_object_add(window_, bg);
113 evas_object_render_op_set(bg, EVAS_RENDER_BLEND);
114 evas_object_show(bg);
117 Evas_Object* conformant = elm_conformant_add(window_);
118 EVAS_SIZE_EXPAND_FILL(conformant);
119 elm_win_resize_object_add(window_, conformant);
120 evas_object_show(conformant);
123 Evas_Object* top_layout = elm_layout_add(conformant);
124 elm_layout_file_set(top_layout, kEdjePath, "web-application");
125 EVAS_SIZE_EXPAND_FILL(top_layout);
126 elm_object_content_set(conformant, top_layout);
127 evas_object_show(top_layout);
130 Evas_Object* focus = elm_bg_add(top_layout);
131 evas_object_size_hint_align_set(focus, EVAS_HINT_FILL, EVAS_HINT_FILL);
132 evas_object_size_hint_weight_set(focus, EVAS_HINT_EXPAND, EVAS_HINT_EXPAND);
133 elm_object_focus_allow_set(focus, EINA_TRUE);
134 elm_object_part_content_set(top_layout, "elm.swallow.content", focus);
135 EVAS_SIZE_EXPAND_FILL(focus);
136 elm_access_object_unregister(focus);
137 evas_object_show(focus);
141 auto focus_callback = [](void* user_data,
144 NativeWindow* window = static_cast<NativeWindow*>(user_data);
145 window->DidFocusChanged(true);
147 auto unfocus_callback = [](void* user_data,
150 NativeWindow* window = static_cast<NativeWindow*>(user_data);
151 window->DidFocusChanged(false);
154 evas_object_smart_callback_add(focus,
155 kWinowFocusedEventKey,
158 evas_object_smart_callback_add(focus,
159 kWinowUnfocusedEventKey,
164 auto rotation_callback = [](void* user_data,
167 NativeWindow* window = static_cast<NativeWindow*>(user_data);
168 int degree = elm_win_rotation_get(obj);
169 window->DidRotation(degree);
171 evas_object_smart_callback_add(window_,
172 kWinowRotationEventKey,
177 natural_orientation_ = ScreenOrientation::LANDSCAPE_PRIMARY;
179 natural_orientation_ = ScreenOrientation::PORTRAIT_PRIMARY;
185 void NativeWindow::DidDeleteRequested(void* /*data*/,
186 Evas_Object* /*obj*/, void* /*event_info*/) {
187 LOGGER(DEBUG) << "didDeleteRequested";
191 void NativeWindow::DidProfileChanged(void* /*data*/,
192 Evas_Object* /*obj*/, void* /*event_info*/) {
193 LOGGER(DEBUG) << "didProfileChanged";
196 Evas_Object* NativeWindow::evas_object() const {
200 void NativeWindow::SetContent(Evas_Object* content) {
202 // If any object was already set as a content object in the same part,
203 // the previous object will be deleted automatically with this call.
204 // If the content is NULL, this call will just delete the previous object.
205 // If the If you wish to preserve it,
206 // issue elm_object_part_content_unset() on it first.
208 evas_object_show(content);
210 elm_object_part_content_unset(focus_, "elm.swallow.content"));
211 elm_object_part_content_set(focus_, "elm.swallow.content", content);
212 elm_object_focus_set(focus_, EINA_TRUE);
215 // attached webview was resized by evas_norender API
216 evas_norender(evas_object_evas_get(window_));
219 void NativeWindow::DidRotation(int degree) {
221 auto it = handler_table_.begin();
222 for ( ; it != handler_table_.end(); ++it) {
227 void NativeWindow::DidFocusChanged(bool got) {
228 if (content_ != NULL) {
229 ewk_view_focus_set(content_, got ? EINA_TRUE : EINA_FALSE);
233 int NativeWindow::AddRotationHandler(RotationHandler handler) {
234 int id = handler_id_++;
235 handler_table_[id] = handler;
239 void NativeWindow::RemoveRotationHandler(int id) {
240 handler_table_.erase(id);
243 NativeWindow::ScreenOrientation NativeWindow::orientation() const {
244 return NativeAngleToOrientation(rotation_, natural_orientation_);
247 void NativeWindow::SetRotationLock(int degree) {
249 rotation_ = degree % 360;
250 elm_win_wm_rotation_preferred_rotation_set(window_, rotation_);
253 void NativeWindow::SetRotationLock(ScreenOrientation orientation) {
254 auto& convert_table =
255 natural_orientation_ == ScreenOrientation::PORTRAIT_PRIMARY ?
256 kPortraitNaturalAngle :
257 kLandscapeNaturalAngle;
258 SetRotationLock(convert_table[static_cast<int>(orientation)]);
262 void NativeWindow::SetAutoRotation() {
263 elm_win_wm_rotation_preferred_rotation_set(window_, -1);
264 if (elm_win_wm_rotation_supported_get(window_)) {
265 const int rotation[4] = {0, 90, 180, 270};
266 elm_win_wm_rotation_available_rotations_set(window_, rotation, 4);
268 rotation_ = elm_win_rotation_get(window_);
271 void NativeWindow::Show() {
272 evas_object_show(window_);
275 void NativeWindow::Active() {
276 elm_win_activate(window_);
279 void NativeWindow::InActive() {
280 elm_win_lower(window_);
283 void NativeWindow::FullScreen(bool enable) {
284 elm_win_indicator_opacity_set(window_,
285 enable ? ELM_WIN_INDICATOR_TRANSPARENT : ELM_WIN_INDICATOR_OPAQUE);
288 #ifdef MANUAL_ROTATE_FEATURE_SUPPORT
289 void NativeWindow::EnableManualRotation(bool enable) {
290 LOGGER(DEBUG) << "set manual rotation : " << (enable ? "enabled" : "disabled");
291 elm_win_wm_rotation_manual_rotation_done_set(window_, enable ? EINA_TRUE : EINA_FALSE);
294 void NativeWindow::ManualRotationDone() {
295 if (EINA_TRUE == elm_win_wm_rotation_manual_rotation_done_get(window_)) {
296 elm_win_wm_rotation_manual_rotation_done(window_);
299 #endif // MANUAL_ROTATE_FEATURE_SUPPORT
301 } // namespace runtime