/* * Copyright (c) 2016 Samsung Electronics Co., Ltd All Rights Reserved * * Licensed under the Apache License, Version 2.0 (the License); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an AS IS BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ using System; using System.Collections.Generic; namespace ElmSharp { /// /// EcoreMainloop is a helper class, it provide functions relative Ecore's main loop. /// public static class EcoreMainloop { static readonly Dictionary> _taskMap = new Dictionary>(); static readonly Object _taskLock = new Object(); static int _newTaskId = 0; static Interop.Ecore.EcoreTaskCallback _nativeHandler; static EcoreMainloop() { Interop.Ecore.ecore_init(); Interop.Ecore.ecore_main_loop_glib_integrate(); _nativeHandler = NativeHandler; } /// /// Checks if you are calling this function from the main thread. /// /// True is the calling function is the same thread, false otherwise. public static bool IsMainThread => Interop.Eina.eina_main_loop_is(); /// /// Runs the application main loop. /// public static void Begin() { Interop.Ecore.ecore_main_loop_begin(); } /// /// Quits the main loop once all the events currently on the queue have been processed. /// public static void Quit() { Interop.Ecore.ecore_main_loop_quit(); } /// /// Adds an idler handler. /// /// The action to call when idling public static void Post(Action task) { int id = RegistHandler(() => { task(); return false; }); Interop.Ecore.ecore_idler_add(_nativeHandler, (IntPtr)id); } /// /// Calls callback asynchronously in the main loop. /// /// The action wanted to be called public static void PostAndWakeUp(Action task) { int id = RegistHandler(() => { task(); return false; }); Interop.Ecore.ecore_main_loop_thread_safe_call_async(_nativeHandler, (IntPtr)id); } /// /// Calls callback synchronously in the main loop. /// /// The action wanted to be called public static void Send(Action task) { int id = RegistHandler(() => { task(); return false; }); Interop.Ecore.ecore_main_loop_thread_safe_call_sync(_nativeHandler, (IntPtr)id); } /// /// Creates a timer to call the given function in the given period of time. /// /// The interval in seconds. /// The given function. /// A timer object handler on success, NULL on failure. public static IntPtr AddTimer(double interval, Func handler) { int id = RegistHandler(handler); return Interop.Ecore.ecore_timer_add(interval, _nativeHandler, (IntPtr)id); } /// /// Removes the specified timer from the timer list. /// /// The specified timer handler public static void RemoveTimer(IntPtr id) { int taskId = (int)Interop.Ecore.ecore_timer_del(id); _taskMap.Remove(taskId); } static int RegistHandler(Func task) { int taskId; lock (_taskLock) { taskId = _newTaskId++; } _taskMap[taskId] = task; return taskId; } static bool NativeHandler(IntPtr user_data) { int task_id = (int)user_data; Func userAction = null; if (_taskMap.TryGetValue(task_id, out userAction)) { bool result = false; if (userAction != null) result = userAction(); if (result == false) _taskMap.Remove(task_id); return result; } return false; } } }