2 * Copyright (c) 2023 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/system/tizen-wayland/widget-application-impl-tizen.h>
22 #include <dali/devel-api/events/key-event-devel.h>
23 #include <dali/internal/adaptor/common/adaptor-impl.h>
24 #include <dali/internal/system/common/environment-variables.h>
25 #include <dali/internal/system/tizen-wayland/widget-controller-tizen.h>
26 #include <dali/public-api/adaptor-framework/widget-impl.h>
27 #include <dali/public-api/adaptor-framework/widget.h>
31 #include <widget_base.h>
40 * This Api is called when widget viewer send keyEvent.
41 * In this API, widget framework create a new keyEvent, find the proper widget and send this event.
42 * Finally widget framework receive feedback from widget.
44 #ifdef OVER_TIZEN_VERSION_7
45 bool OnKeyEventCallback(const char* id, screen_connector_event_type_e eventType, int keyCode, const char* keyName, long long cls, long long subcls, const char* identifier, long long timestamp, void* userData)
47 Dali::Internal::Adaptor::WidgetApplicationTizen* application = static_cast<Dali::Internal::Adaptor::WidgetApplicationTizen*>(userData);
49 // Create new key for widget
50 Dali::KeyEvent::State state = Dali::KeyEvent::DOWN;
51 if(eventType == SCREEN_CONNECTOR_EVENT_TYPE_KEY_DOWN)
53 state = Dali::KeyEvent::DOWN;
55 else if(eventType == SCREEN_CONNECTOR_EVENT_TYPE_KEY_UP)
57 state = Dali::KeyEvent::UP;
61 std::string keyEventName = std::string(keyName);
62 Dali::KeyEvent event = Dali::DevelKeyEvent::New(keyEventName, "", "", keyCode, 0, timestamp, state, "", "", Device::Class::NONE, Device::Subclass::NONE);
66 std::string widgetId = std::string(id);
67 widget_base_instance_h instanceHandle = application->GetWidgetInstanceFromWidgetId(widgetId);
70 consumed = application->FeedKeyEvent(instanceHandle, event);
78 int OnInstanceInit(widget_base_instance_h instanceHandle, bundle* content, int w, int h, void* classData)
81 widget_base_context_get_id(instanceHandle, &id);
83 widget_base_class_on_create(instanceHandle, content, w, h);
85 Dali::Internal::Adaptor::WidgetApplicationTizen* application = static_cast<Dali::Internal::Adaptor::WidgetApplicationTizen*>(classData);
88 if(application->GetWidgetCount() == 0)
90 window = application->GetWindow();
91 DALI_LOG_RELEASE_INFO("Widget Instance use default Window(win:%p), so it need to bind widget (%dx%d) (id:%s) \n", window, w, h, std::string(id).c_str());
95 window = Dali::Window::New(PositionSize(0, 0, w, h), "", false);
98 DALI_LOG_RELEASE_INFO("Widget Instance create new Window (win:%p, cnt:%d) (%dx%d) (id:%s )\n", window, application->GetWidgetCount(), w, h, std::string(id).c_str());
102 DALI_LOG_ERROR("This device can't support Multi Widget. it means UI may not be properly drawn.");
103 window = application->GetWindow();
107 Any nativeHandle = window.GetNativeHandle();
109 #ifdef ECORE_WAYLAND2
110 Ecore_Wl2_Window* wlWindow = AnyCast<Ecore_Wl2_Window*>(nativeHandle);
112 Ecore_Wl_Window* wlWindow = AnyCast<Ecore_Wl_Window*>(nativeHandle);
115 widget_base_context_window_bind(instanceHandle, id, wlWindow);
116 window.SetSize(Dali::Window::WindowSize(w, h));
118 Dali::Internal::Adaptor::WidgetApplication::CreateWidgetFunctionPair pair = application->GetWidgetCreatingFunctionPair(std::string(id));
119 Dali::WidgetApplication::CreateWidgetFunction createFunction = pair.second;
121 Dali::Widget widgetInstance = createFunction(pair.first);
123 Dali::Internal::Adaptor::Widget::Impl* widgetImpl = new Dali::Internal::Adaptor::WidgetImplTizen(instanceHandle);
124 Internal::Adaptor::GetImplementation(widgetInstance).SetImpl(widgetImpl);
126 application->AddWidget(instanceHandle, widgetInstance, window, std::string(id));
128 std::string encodedContentString = "";
130 if(bundle_get_count(content))
132 bundle_raw* bundleRaw;
134 bundle_encode(content, &bundleRaw, &len);
135 char* encodedContent = reinterpret_cast<char*>(bundleRaw);
136 encodedContentString = std::string(encodedContent);
140 Internal::Adaptor::GetImplementation(widgetInstance).OnCreate(encodedContentString, window);
142 // connect keyEvent for widget
143 #ifdef OVER_TIZEN_VERSION_7
144 application->ConnectKeyEvent(window);
150 int OnInstanceDestroy(widget_base_instance_h instanceHandle, widget_base_destroy_type_e reason, bundle* content, void* classData)
152 Dali::Internal::Adaptor::WidgetApplicationTizen* application = static_cast<Dali::Internal::Adaptor::WidgetApplicationTizen*>(classData);
154 // Get Dali::Widget instance.
155 Dali::Widget widgetInstance = application->GetWidget(instanceHandle);
157 Dali::Widget::Termination destroyReason = Dali::Widget::Termination::TEMPORARY;
159 if(reason == WIDGET_BASE_DESTROY_TYPE_PERMANENT)
161 destroyReason = Dali::Widget::Termination::PERMANENT;
164 std::string encodedContentString = "";
166 if(bundle_get_count(content))
168 bundle_raw* bundleRaw;
170 bundle_encode(content, &bundleRaw, &len);
171 char* encodedContent = reinterpret_cast<char*>(bundleRaw);
172 encodedContentString = std::string(encodedContent);
176 Internal::Adaptor::GetImplementation(widgetInstance).OnTerminate(encodedContentString, destroyReason);
178 widget_base_class_on_destroy(instanceHandle, reason, content);
180 application->DeleteWidget(instanceHandle);
185 int OnInstancePause(widget_base_instance_h instanceHandle, void* classData)
187 widget_base_class_on_pause(instanceHandle);
189 Dali::Internal::Adaptor::WidgetApplicationTizen* application = static_cast<Dali::Internal::Adaptor::WidgetApplicationTizen*>(classData);
191 // Get Dali::Widget instance.
192 Dali::Widget widgetInstance = application->GetWidget(instanceHandle);
194 Internal::Adaptor::GetImplementation(widgetInstance).OnPause();
199 int OnInstanceResume(widget_base_instance_h instanceHandle, void* classData)
201 widget_base_class_on_resume(instanceHandle);
203 Dali::Internal::Adaptor::WidgetApplicationTizen* application = static_cast<Dali::Internal::Adaptor::WidgetApplicationTizen*>(classData);
205 // Get Dali::Widget instance.
206 Dali::Widget widgetInstance = application->GetWidget(instanceHandle);
208 Internal::Adaptor::GetImplementation(widgetInstance).OnResume();
213 int OnInstanceResize(widget_base_instance_h instanceHandle, int w, int h, void* classData)
215 widget_base_class_on_resize(instanceHandle, w, h);
217 Dali::Internal::Adaptor::WidgetApplicationTizen* application = static_cast<Dali::Internal::Adaptor::WidgetApplicationTizen*>(classData);
219 // Get Dali::Widget instance.
220 Dali::Widget widgetInstance = application->GetWidget(instanceHandle);
221 Dali::Window window = application->GetWindowFromWidget(widgetInstance);
222 window.SetSize(Dali::Window::WindowSize(w, h));
223 Internal::Adaptor::GetImplementation(widgetInstance).OnResize(window);
228 int OnInstanceUpdate(widget_base_instance_h instanceHandle, bundle* content, int force, void* classData)
230 widget_base_class_on_update(instanceHandle, content, force);
232 Dali::Internal::Adaptor::WidgetApplicationTizen* application = static_cast<Dali::Internal::Adaptor::WidgetApplicationTizen*>(classData);
234 // Get Dali::Widget instance.
235 Dali::Widget widgetInstance = application->GetWidget(instanceHandle);
237 std::string encodedContentString = "";
239 if(bundle_get_count(content))
241 bundle_raw* bundleRaw;
243 bundle_encode(content, &bundleRaw, &len);
244 char* encodedContent = reinterpret_cast<char*>(bundleRaw);
245 encodedContentString = std::string(encodedContent);
249 Internal::Adaptor::GetImplementation(widgetInstance).OnUpdate(encodedContentString, force);
254 unsigned int GetEnvWidgetRenderRefreshRate()
256 const char* envVariable = std::getenv(DALI_WIDGET_REFRESH_RATE);
258 return envVariable ? std::atoi(envVariable) : 1u; // Default 60 fps
261 } // anonymous namespace
265 WidgetApplicationPtr WidgetApplicationTizen::New(
268 const std::string& stylesheet,
269 const WindowData& windowData)
271 return new WidgetApplicationTizen(argc, argv, stylesheet, windowData);
274 WidgetApplicationTizen::WidgetApplicationTizen(int* argc, char** argv[], const std::string& stylesheet, const WindowData& windowData)
275 : WidgetApplication(argc, argv, stylesheet, windowData),
276 mConnectedKeyEvent(false),
277 mReceivedKeyEvent(false)
281 WidgetApplicationTizen::~WidgetApplicationTizen()
285 void WidgetApplicationTizen::RegisterWidgetCreatingFunction(const std::string& widgetName, Dali::WidgetApplication::CreateWidgetFunction createFunction)
287 AddWidgetCreatingFunctionPair(CreateWidgetFunctionPair(widgetName, createFunction));
289 // Register widget class to widget framework
290 widget_base_class cls = widget_base_class_get_default();
291 cls.ops.create = OnInstanceInit;
292 cls.ops.destroy = OnInstanceDestroy;
293 cls.ops.pause = OnInstancePause;
294 cls.ops.resume = OnInstanceResume;
295 cls.ops.resize = OnInstanceResize;
296 cls.ops.update = OnInstanceUpdate;
298 widget_base_class_add(cls, widgetName.c_str(), this);
301 void WidgetApplicationTizen::AddWidgetCreatingFunctionPair(CreateWidgetFunctionPair pair)
303 mCreateWidgetFunctionContainer.push_back(pair);
306 WidgetApplicationTizen::CreateWidgetFunctionPair WidgetApplicationTizen::GetWidgetCreatingFunctionPair(const std::string& widgetName)
308 int idx = widgetName.find(":");
309 std::string widgetID = widgetName.substr(idx + 1);
310 for(CreateWidgetFunctionContainer::const_iterator iter = mCreateWidgetFunctionContainer.begin(); iter != mCreateWidgetFunctionContainer.end(); ++iter)
312 if(widgetID.compare((*iter).first) == 0)
318 return CreateWidgetFunctionPair("", NULL);
321 void WidgetApplicationTizen::AddWidget(widget_base_instance_h widgetBaseInstance, Dali::Widget widget, Dali::Window window, const std::string& widgetId)
323 mWidgetInstanceContainer.push_back(WidgetInstancePair(widgetBaseInstance, widget));
324 Internal::Adaptor::GetImplementation(widget).SetInformation(window, widgetId);
327 Dali::Widget WidgetApplicationTizen::GetWidget(widget_base_instance_h widgetBaseInstance) const
329 for(auto&& iter : mWidgetInstanceContainer)
331 if((iter).first == widgetBaseInstance)
333 return (iter).second;
336 return Dali::Widget();
339 void WidgetApplicationTizen::DeleteWidget(widget_base_instance_h widgetBaseInstance)
341 // Delete WidgetInstance
342 auto widgetInstance = std::find_if(mWidgetInstanceContainer.begin(),
343 mWidgetInstanceContainer.end(),
344 [widgetBaseInstance](WidgetInstancePair pair) { return (pair.first == widgetBaseInstance); });
346 if(widgetInstance != mWidgetInstanceContainer.end())
348 mWidgetInstanceContainer.erase(widgetInstance);
352 Dali::Window WidgetApplicationTizen::GetWindowFromWidget(Dali::Widget widgetInstance) const
356 return Internal::Adaptor::GetImplementation(widgetInstance).GetWindow();
359 return Dali::Window();
362 widget_base_instance_h WidgetApplicationTizen::GetWidgetInstanceFromWidgetId(std::string& widgetId) const
364 for(auto&& iter : mWidgetInstanceContainer)
366 if(widgetId == Internal::Adaptor::GetImplementation((iter).second).GetWidgetId())
375 int WidgetApplicationTizen::GetWidgetCount()
377 return mWidgetInstanceContainer.size();
380 void WidgetApplicationTizen::ConnectKeyEvent(Dali::Window window)
382 if(!mConnectedKeyEvent)
384 #ifdef OVER_TIZEN_VERSION_7
385 screen_connector_provider_set_key_event_cb(OnKeyEventCallback, this);
387 mConnectedKeyEvent = true;
389 window.KeyEventSignal().Connect(this, &WidgetApplicationTizen::OnWindowKeyEvent);
392 void WidgetApplicationTizen::OnWindowKeyEvent(const Dali::KeyEvent& event)
394 // If Widget Application consume key event, this api is not called.
395 mReceivedKeyEvent = true;
398 bool WidgetApplicationTizen::FeedKeyEvent(widget_base_instance_h instanceHandle, const Dali::KeyEvent& keyEvent)
400 bool consumed = true;
402 // Check if application consume key event
403 Dali::Widget widgetInstance = GetWidget(instanceHandle);
406 Dali::Window window = GetWindowFromWidget(widgetInstance);
408 // Reset the state of key received
409 mReceivedKeyEvent = false;
411 // Feed the keyEvent to widget window
412 DevelWindow::FeedKeyEvent(window, keyEvent);
414 // if the application is not using a key event, verify that the window in the widget has received a key event.
415 if(Internal::Adaptor::GetImplementation(widgetInstance).IsKeyEventUsing() == false)
417 // if the window has received a key event, widget need to consume its key event
418 consumed = (mReceivedKeyEvent) ? false : true;
425 void WidgetApplicationTizen::OnInit()
427 WidgetApplication::OnInit();
429 Dali::Adaptor::Get().SetRenderRefreshRate(GetEnvWidgetRenderRefreshRate());
432 // factory function, must be implemented
433 namespace WidgetApplicationFactory
436 * Create a new widget application
437 * @param[in] argc A pointer to the number of arguments
438 * @param[in] argv A pointer to the argument list
439 * @param[in] stylesheet The path to user defined theme file
440 * @param[in] windowData The window data
442 Dali::Internal::Adaptor::WidgetApplicationPtr Create(int* argc, char** argv[], const std::string& stylesheet, const WindowData& windowData)
444 return WidgetApplicationTizen::New(argc, argv, stylesheet, windowData);
447 } // namespace WidgetApplicationFactory
449 } // namespace Adaptor
451 } // namespace Internal