fix bug
[apps/agl-service-homescreen.git] / src / hs-apprecover.cpp
1 /*
2  * Copyright (c) 2019 TOYOTA MOTOR CORPORATION
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 #include "hs-apprecover.h"
18 #include "hs-appinfo.h"
19 #include "hs-proxy.h"
20 #include "hs-clientmanager.h"
21
22
23 HS_AppRecover* HS_AppRecover::me = nullptr;
24
25 /**
26  * screen_update event handler
27  *
28  * #### Parameters
29  * - api : the api
30  * - event : received event name
31  * - object : received json object
32  *
33  * #### Return
34  * 0 : event can transfer to others
35  * 1 : event not transfer to others
36  */
37 int on_screen_update_event(afb_api_t api, const char *event, struct json_object *object)
38 {
39     HS_AppRecover::instance()->screenUpdated(object);
40     return 0;
41 }
42
43 /**
44  * get instance
45  *
46  * #### Parameters
47  *  - Nothing
48  *
49  * #### Return
50  * HS_AppRecover instance pointer
51  *
52  */
53 HS_AppRecover* HS_AppRecover::instance(void)
54 {
55     if(me == nullptr)
56         me = new HS_AppRecover();
57
58     return me;
59 }
60
61 /**
62  * HS_AppRecover initialize function
63  *
64  * #### Parameters
65  *  - api : the api serving the request
66  *
67  * #### Return
68  * 0 : init success
69  * 1 : init fail
70  *
71  */
72 int HS_AppRecover::init(afb_api_t api)
73 {
74     setEventHook("windowmanager/screenUpdated", on_screen_update_event);
75     HS_WmProxy wm_proxy;
76     wm_proxy.subscribe(api, HS_WmProxy::Event_ScreenUpdated);
77     return 0;
78 }
79
80 /**
81  * starting recover applications
82  *
83  * #### Parameters
84  *  - api : the api
85  *  - map : recover app map
86  *
87  * #### Return
88  * None
89  *
90  */
91 void HS_AppRecover::startRecovery(afb_api_t api, recover_map &map)
92 {
93     HS_AfmMainProxy afm_proxy;
94     for(auto &key : HS_Config::keys_recover_type) {
95         for(auto &m : map[key]){
96             struct app_recover_info recover_info;
97             recover_info.recover_type = key;
98             recover_info.visibility = m.visibility;
99             m_recover_apps_list[m.appid] = std::move(recover_info);
100             if(key == HS_Config::keys_recover_type[1]) {
101                 m_lastmode_list.insert(m.appid);
102             }
103
104             // recover application
105             auto it = m_recovering_set.find(m.appid);
106             if(it == m_recovering_set.end()) {
107                 m_recovering_set.insert(m.appid);
108                 afm_proxy.start(api,  HS_AppInfo::instance()->getAppProperty(m.appid, _keyId));
109             }
110         }
111     }
112 }
113
114 /**
115  * register started applications
116  *
117  * #### Parameters
118  *  - appid : application id liked "dashboard"
119  *
120  * #### Return
121  * false : not recover app
122  * true : recover app
123  * 
124  */
125 bool HS_AppRecover::registerRecoveredApp(const std::string &appid)
126 {
127     bool ret = false;
128     if(m_recovering_set.empty()) {
129         return ret;
130     }
131
132     AFB_INFO("recover appid=[%s].", appid.c_str());
133     auto it = m_recovering_set.find(appid);
134     if(it != m_recovering_set.end()) {
135         m_recovering_set.erase(appid);
136         auto ip = m_recover_apps_list.find(appid);
137         if(ip != m_recover_apps_list.end()
138         && ip->second.visibility) {
139             HS_ClientManager::instance()->pushEvent("showWindow", nullptr, appid);
140         }
141         ret = true;
142     }
143     return ret;
144 }
145
146 /**
147  * screenUpdated event handler
148  *
149  * #### Parameters
150  *  - obj : screenUpdate object
151  *
152  * #### Return
153  * None
154  * 
155  */
156 void HS_AppRecover::screenUpdated(struct json_object *obj)
157 {
158     std::set<std::string> s_mode;
159     struct json_object *ids_obj;
160     if(json_object_object_get_ex(obj, key_ids.c_str(), &ids_obj)) {
161         if(json_object_get_type(ids_obj) == json_type_array) {
162             int array_len = json_object_array_length(ids_obj);
163             for (int i = 0; i < array_len; ++i) {
164                 struct json_object *j_id = json_object_array_get_idx(ids_obj, i);
165                 std::string appid = json_object_get_string(j_id);
166                 if(!isHomescreenApp(appid)) {
167                     s_mode.insert(appid);
168                 }
169             }
170             if(!s_mode.empty()) {
171                 updateLastmode(s_mode);
172             }
173         }
174         else {
175             AFB_WARNING("screenUpdated list isn't array.");
176         }
177     }
178 }
179
180 /**
181  * update lastmode file
182  *
183  * #### Parameters
184  *  - set : new lastmode set
185  *
186  * #### Return
187  * None
188  * 
189  */
190 void HS_AppRecover::updateLastmode(std::set<std::string> &set)
191 {
192     if(set.size() == m_lastmode_list.size()) {
193         bool is_same = true;
194         for(auto &m : set) {
195             auto it = m_lastmode_list.find(m);
196             if(it == m_lastmode_list.end()) {
197                 is_same = false;
198                 break;
199             }
200         }
201         if(is_same) {   // lastmode aren't changed
202             return;
203         }
204     }
205
206     m_lastmode_list.swap(set);
207     struct json_object *arr_obj = json_object_new_array();
208     for(auto &it : m_lastmode_list) {
209         struct json_object *j_obj = json_object_new_object();
210         json_object_object_add(j_obj, HS_Config::key_appid.c_str(), json_object_new_string(it.c_str()));
211         json_object_object_add(j_obj, HS_Config::key_visibility.c_str(), json_object_new_string("visible"));
212         json_object_array_add(arr_obj, j_obj);
213     }
214
215     auto path = HS_Config::root_dir + "/etc/" + HS_Config::lastmode_json;
216     if(writeJsonFile(path.c_str(), arr_obj) < 0) {
217         AFB_ERROR("write lastmode error.");
218     }
219 }
220
221 /**
222  * judge whether app is Homescreen app
223  *
224  * #### Parameters
225  *  - appid : application id
226  *
227  * #### Return
228  * true : homescreen app
229  * false : not homescreen app
230  *
231  */
232 bool HS_AppRecover::isHomescreenApp(const std::string &appid) const
233 {
234     auto it = m_recover_apps_list.find(appid);
235     if(it != m_recover_apps_list.end()
236     && it->second.recover_type == HS_Config::keys_recover_type[0]) {
237             return true;
238     }
239     return false;
240 }