1cda1292bcc7322cd72202d593290ea821c3ecd3
[src/agl-compositor.git] / src / desktop.c
1 /*
2  * Copyright © 2019 Collabora, Ltd.
3  *
4  * Permission is hereby granted, free of charge, to any person obtaining
5  * a copy of this software and associated documentation files (the
6  * "Software"), to deal in the Software without restriction, including
7  * without limitation the rights to use, copy, modify, merge, publish,
8  * distribute, sublicense, and/or sell copies of the Software, and to
9  * permit persons to whom the Software is furnished to do so, subject to
10  * the following conditions:
11  *
12  * The above copyright notice and this permission notice (including the
13  * next paragraph) shall be included in all copies or substantial
14  * portions of the Software.
15  *
16  * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
17  * EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
18  * MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
19  * NONINFRINGEMENT.  IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS
20  * BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN
21  * ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN
22  * CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
23  * SOFTWARE.
24  */
25
26 #include <assert.h>
27 #include "ivi-compositor.h"
28 #include "policy.h"
29
30 #include "shared/helpers.h"
31 #include <libweston/libweston.h>
32 #include <libweston-desktop/libweston-desktop.h>
33
34 #include "agl-shell-desktop-server-protocol.h"
35
36 static void
37 desktop_advertise_app(struct wl_listener *listener, void *data)
38 {
39         struct ivi_surface *surface;
40
41         surface = wl_container_of(listener, surface, listener_advertise_app);
42
43         agl_shell_desktop_advertise_application_id(surface->ivi, surface);
44 }
45
46 static void
47 desktop_ping_timeout(struct weston_desktop_client *dclient, void *userdata)
48 {
49         /* not supported */
50 }
51
52 static void
53 desktop_pong(struct weston_desktop_client *dclient, void *userdata)
54 {
55         /* not supported */
56 }
57
58 struct weston_output *
59 get_default_output(struct weston_compositor *compositor)
60 {
61         if (wl_list_empty(&compositor->output_list))
62                 return NULL;
63
64         return container_of(compositor->output_list.next,
65                         struct weston_output, link);
66 }
67
68 struct weston_output *
69 get_focused_output(struct weston_compositor *compositor)
70 {
71         struct weston_seat *seat;
72         struct weston_output *output = NULL;
73
74         wl_list_for_each(seat, &compositor->seat_list, link) {
75                 struct weston_touch *touch = weston_seat_get_touch(seat);
76                 struct weston_pointer *pointer = weston_seat_get_pointer(seat);
77                 struct weston_keyboard *keyboard =
78                         weston_seat_get_keyboard(seat);
79
80                 if (touch && touch->focus)
81                         output = touch->focus->output;
82                 else if (pointer && pointer->focus)
83                         output = pointer->focus->output;
84                 else if (keyboard && keyboard->focus)
85                         output = keyboard->focus->output;
86
87                 if (output)
88                         break;
89         }
90
91         return output;
92 }
93
94 void
95 ivi_shell_activate_surface(struct ivi_surface *ivi_surf,
96                           struct ivi_shell_seat *ivi_seat,
97                           uint32_t flags)
98 {
99        struct weston_desktop_surface *dsurface = ivi_surf->dsurface;
100        struct weston_surface *surface =
101                weston_desktop_surface_get_surface(dsurface);
102
103        weston_view_activate_input(ivi_surf->view, ivi_seat->seat, flags);
104
105        if (ivi_seat->focused_surface) {
106                struct ivi_surface *current_focus =
107                        get_ivi_shell_surface(ivi_seat->focused_surface);
108                struct weston_desktop_surface *dsurface_focus;
109                assert(current_focus);
110
111                dsurface_focus = current_focus->dsurface;
112                if (--current_focus->focus_count == 0)
113                        weston_desktop_surface_set_activated(dsurface_focus, false);
114        }
115
116        ivi_seat->focused_surface = surface;
117        if (ivi_surf->focus_count++ == 0)
118                weston_desktop_surface_set_activated(dsurface, true);
119 }
120
121
122 static void
123 desktop_surface_added_configure(struct ivi_surface *surface,
124                                 struct ivi_output *ivi_output)
125 {
126         enum ivi_surface_role role = IVI_SURFACE_ROLE_NONE;
127         struct weston_desktop_surface *dsurface = surface->dsurface;
128
129         ivi_check_pending_surface_desktop(surface, &role);
130         if ((role != IVI_SURFACE_ROLE_DESKTOP &&
131              role != IVI_SURFACE_ROLE_FULLSCREEN &&
132              role != IVI_SURFACE_ROLE_REMOTE) ||
133              role == IVI_SURFACE_ROLE_NONE)
134                 return;
135
136         if (role == IVI_SURFACE_ROLE_FULLSCREEN) {
137                 struct ivi_output *bg_output =
138                         ivi_layout_find_bg_output(surface->ivi);
139                 assert(bg_output);
140                 weston_desktop_surface_set_fullscreen(dsurface, true);
141                 weston_desktop_surface_set_size(dsurface,
142                                                 bg_output->output->width,
143                                                 bg_output->output->height);
144                 return;
145         }
146
147         weston_desktop_surface_set_maximized(dsurface, true);
148         weston_desktop_surface_set_size(dsurface,
149                                         ivi_output->area.width,
150                                         ivi_output->area.height);
151 }
152
153
154 static void
155 desktop_surface_added(struct weston_desktop_surface *dsurface, void *userdata)
156 {
157         struct ivi_compositor *ivi = userdata;
158         struct weston_desktop_client *dclient;
159         struct wl_client *client;
160         struct ivi_surface *surface;
161         struct ivi_output *active_output = NULL;
162         struct weston_output *output = NULL;
163         const char *app_id = NULL;
164
165         dclient = weston_desktop_surface_get_client(dsurface);
166         client = weston_desktop_client_get_client(dclient);
167
168         if (ivi->shell_client.resource &&
169             ivi->shell_client.status == BOUND_FAILED) {
170                 wl_client_post_implementation_error(client,
171                                        "agl_shell has already been bound. "
172                                        "Check out bound_fail event");
173                 return;
174         }
175
176         surface = zalloc(sizeof *surface);
177         if (!surface) {
178                 wl_client_post_no_memory(client);
179                 return;
180         }
181
182         surface->view = weston_desktop_surface_create_view(dsurface);
183         if (!surface->view) {
184                 free(surface);
185                 wl_client_post_no_memory(client);
186                 return;
187         }
188
189         surface->ivi = ivi;
190         surface->dsurface = dsurface;
191         surface->role = IVI_SURFACE_ROLE_NONE;
192         surface->mapped = false;
193         surface->advertised_on_launch = false;
194         surface->checked_pending = false;
195         wl_list_init(&surface->link);
196
197         wl_signal_init(&surface->signal_advertise_app);
198
199         surface->listener_advertise_app.notify = desktop_advertise_app;
200         wl_signal_add(&surface->signal_advertise_app,
201                       &surface->listener_advertise_app);
202
203         weston_desktop_surface_set_user_data(dsurface, surface);
204
205         if (ivi->policy && ivi->policy->api.surface_create &&
206             !ivi->policy->api.surface_create(surface, ivi)) {
207                 wl_client_post_no_memory(client);
208                 return;
209         }
210
211
212         app_id = weston_desktop_surface_get_app_id(dsurface);
213
214         if ((active_output = ivi_layout_find_with_app_id(app_id, ivi)))
215                 ivi_set_pending_desktop_surface_remote(active_output, app_id);
216
217         /* reset any caps to make sure we apply the new caps */
218         ivi_seat_reset_caps_sent(ivi);
219
220         output =  get_focused_output(ivi->compositor);
221         if (!output)
222                 output = get_default_output(ivi->compositor);
223
224         if (output && ivi->shell_client.ready) {
225                 struct ivi_output *ivi_output = to_ivi_output(output);
226                 if (active_output)
227                         desktop_surface_added_configure(surface, active_output);
228                 else
229                         desktop_surface_added_configure(surface, ivi_output);
230         }
231         /*
232          * We delay creating "normal" desktop surfaces until later, to
233          * give the shell-client an oppurtunity to set the surface as a
234          * background/panel.
235          * Also delay the creation in order to have a valid app_id
236          * which will be used to set the proper role.
237          */
238         weston_log("Added surface %p, app_id %s to pending list\n",
239                         surface, app_id);
240         wl_list_insert(&ivi->pending_surfaces, &surface->link);
241
242 }
243
244 static bool
245 desktop_surface_check_last_remote_surfaces(struct ivi_compositor *ivi, enum ivi_surface_role role)
246 {
247         int count = 0;
248         struct ivi_surface *surf;
249
250         wl_list_for_each(surf, &ivi->surfaces, link)
251                 if (surf->role == role)
252                         count++;
253
254         return (count == 1);
255 }
256
257 static void
258 desktop_surface_removed(struct weston_desktop_surface *dsurface, void *userdata)
259 {
260         struct ivi_surface *surface =
261                 weston_desktop_surface_get_user_data(dsurface);
262         struct weston_surface *wsurface =
263                 weston_desktop_surface_get_surface(dsurface);
264         const char *app_id = NULL;
265         struct weston_seat *wseat = NULL;
266         struct ivi_shell_seat *ivi_seat = NULL;
267         struct ivi_output *output = NULL;
268
269         /* we might not have a valid ivi_surface if _added failed due to
270          * protocol errors */
271         if (!surface)
272                 return;
273
274         wseat = get_ivi_shell_weston_first_seat(surface->ivi);
275         if (wseat)
276                 ivi_seat = get_ivi_shell_seat(wseat);
277
278         output = ivi_layout_get_output_from_surface(surface);
279
280         wl_list_remove(&surface->listener_advertise_app.link);
281         surface->listener_advertise_app.notify = NULL;
282
283         app_id = weston_desktop_surface_get_app_id(dsurface);
284
285         /* special corner-case, pending_surfaces which are never activated or
286          * being assigned an output might land here so just remove the surface;
287          *
288          * the DESKTOP role can happen here as well, because we can fall-back 
289          * to that when we try to determine the role type. Application that
290          * do not set the app_id will be land here, when destroyed */
291         if (output == NULL && (surface->role == IVI_SURFACE_ROLE_NONE ||
292                                surface->role == IVI_SURFACE_ROLE_DESKTOP))
293                 goto skip_output_asignment;
294
295         assert(output != NULL);
296
297         /* resize the active surface to the original size */
298         if (surface->role == IVI_SURFACE_ROLE_SPLIT_H ||
299             surface->role == IVI_SURFACE_ROLE_SPLIT_V) {
300                 if (output && output->active) {
301                         ivi_layout_desktop_resize(output->active, output->area_saved);
302                 }
303                 /* restore the area back so we can re-use it again if needed */
304                 output->area = output->area_saved;
305         }
306
307         /* reset the active surface as well */
308         if (output && output->active && output->active == surface) {
309                 output->active->view->is_mapped = false;
310                 output->active->view->surface->is_mapped = false;
311
312                 weston_layer_entry_remove(&output->active->view->layer_link);
313                 output->active = NULL;
314         }
315
316         /* clear out focused_surface to avoid a stale focused_surface. the
317          * client shell is responsible for keeping track and switch back to the
318          * last active surface so we don't get do anything at removal, just
319          * reset it */
320         if (ivi_seat && ivi_seat->focused_surface == wsurface)
321                 ivi_seat->focused_surface = NULL;
322
323         if (surface->role == IVI_SURFACE_ROLE_REMOTE &&
324             output->type == OUTPUT_REMOTE)
325                 ivi_destroy_waltham_destroy(surface);
326
327         /* check if there's a last 'remote' surface and insert a black
328          * surface view if there's no background set for that output
329          */
330         if (desktop_surface_check_last_remote_surfaces(output->ivi, IVI_SURFACE_ROLE_REMOTE) ||
331              desktop_surface_check_last_remote_surfaces(output->ivi, IVI_SURFACE_ROLE_DESKTOP))
332                 if (!output->background)
333                         insert_black_surface(output);
334
335
336         if (weston_surface_is_mapped(wsurface)) {
337                 weston_desktop_surface_unlink_view(surface->view);
338                 weston_view_destroy(surface->view);
339         }
340
341         /* invalidate agl-shell surfaces so we can re-use them when
342          * binding again */
343         if (surface->role == IVI_SURFACE_ROLE_PANEL) {
344                 switch (surface->panel.edge) {
345                 case AGL_SHELL_EDGE_TOP:
346                         output->top = NULL;
347                         break;
348                 case AGL_SHELL_EDGE_BOTTOM:
349                         output->bottom = NULL;
350                         break;
351                 case AGL_SHELL_EDGE_LEFT:
352                         output->left = NULL;
353                         break;
354                 case AGL_SHELL_EDGE_RIGHT:
355                         output->right = NULL;
356                         break;
357                 default:
358                         assert(!"Invalid edge detected\n");
359                 }
360         } else if (surface->role == IVI_SURFACE_ROLE_BACKGROUND) {
361                 output->background = NULL;
362         }
363
364 skip_output_asignment:
365         weston_log("Removed surface %p, app_id %s, role %s\n", surface,
366                         app_id, ivi_layout_get_surface_role_name(surface));
367
368         if (app_id && output)
369                 shell_advertise_app_state(output->ivi, app_id,
370                                           NULL, AGL_SHELL_DESKTOP_APP_STATE_DESTROYED);
371
372         wl_list_remove(&surface->link);
373
374         free(surface);
375 }
376
377 static void
378 desktop_committed(struct weston_desktop_surface *dsurface, 
379                   int32_t sx, int32_t sy, void *userdata)
380 {
381         struct ivi_compositor *ivi = userdata;
382         struct ivi_surface *surface =
383                 weston_desktop_surface_get_user_data(dsurface);
384         struct ivi_policy *policy = surface->ivi->policy;
385
386         if (policy && policy->api.surface_commited &&
387             !policy->api.surface_commited(surface, surface->ivi))
388                 return;
389
390         if (ivi->shell_client.ready && !surface->checked_pending) {
391                 const char * app_id =   weston_desktop_surface_get_app_id(dsurface);
392                 weston_log("Checking pending surface %p, app_id %s\n", surface,
393                         app_id);
394                 wl_list_remove(&surface->link);
395                 wl_list_init(&surface->link);
396                 ivi_check_pending_desktop_surface(surface);
397                 surface->checked_pending = true;
398         }
399
400         if (!surface->advertised_on_launch &&
401             !wl_list_empty(&surface->ivi->desktop_clients))
402                 wl_signal_emit(&surface->signal_advertise_app, surface);
403
404         /* this repaint schedule is needed to allow resizing to work with the
405          * help of the hidden layer:
406          *
407          * 1. add the view in the hidden layer and send out correct dimensions
408          * 2. clients changes its dimensions
409          * 3. client commits with the new dimensions
410          *
411          * For desktop and fullscreen, desktop_surface_added() sends the
412          * dimensions from the beginning so applications no need to resize, but
413          * if that weren't the case we still need this in.
414          */
415         weston_compositor_schedule_repaint(surface->ivi->compositor);
416
417         switch (surface->role) {
418         case IVI_SURFACE_ROLE_DESKTOP:
419         case IVI_SURFACE_ROLE_REMOTE:
420                 ivi_layout_desktop_committed(surface);
421                 break;
422         case IVI_SURFACE_ROLE_POPUP:
423                 ivi_layout_popup_committed(surface);
424                 break;
425         case IVI_SURFACE_ROLE_FULLSCREEN:
426                 ivi_layout_fullscreen_committed(surface);
427                 break;
428         case IVI_SURFACE_ROLE_SPLIT_H:
429         case IVI_SURFACE_ROLE_SPLIT_V:
430                 ivi_layout_split_committed(surface);
431                 break;
432         case IVI_SURFACE_ROLE_NONE:
433         case IVI_SURFACE_ROLE_BACKGROUND:
434         case IVI_SURFACE_ROLE_PANEL:
435         default: /* fall through */
436                 break;
437         }
438 }
439
440 static void
441 desktop_show_window_menu(struct weston_desktop_surface *dsurface,
442                          struct weston_seat *seat, int32_t x, int32_t y,
443                          void *userdata)
444 {
445         /* not supported */
446 }
447
448 static void
449 desktop_set_parent(struct weston_desktop_surface *dsurface,
450                    struct weston_desktop_surface *parent, void *userdata)
451 {
452         /* not supported */
453 }
454
455 static void
456 desktop_move(struct weston_desktop_surface *dsurface,
457              struct weston_seat *seat, uint32_t serial, void *userdata)
458 {
459         /* not supported */
460 }
461
462 static void
463 desktop_resize(struct weston_desktop_surface *dsurface,
464                struct weston_seat *seat, uint32_t serial,
465                enum weston_desktop_surface_edge edges, void *user_data)
466 {
467         /* not supported */
468 }
469
470 static void
471 desktop_fullscreen_requested(struct weston_desktop_surface *dsurface,
472                              bool fullscreen, struct weston_output *output,
473                              void *userdata)
474 {
475         /* not supported */
476 }
477
478 static void
479 desktop_maximized_requested(struct weston_desktop_surface *dsurface,
480                             bool maximized, void *userdata)
481 {
482         /* not supported */
483 }
484
485 static void
486 desktop_minimized_requested(struct weston_desktop_surface *dsurface,
487                             void *userdata)
488 {
489         /* not supported */
490 }
491
492 static void
493 desktop_set_xwayland_position(struct weston_desktop_surface *dsurface,
494                               int32_t x, int32_t y, void *userdata)
495 {
496         /* not supported */
497 }
498
499 static const struct weston_desktop_api desktop_api = {
500         .struct_size = sizeof desktop_api,
501         .ping_timeout = desktop_ping_timeout,
502         .pong = desktop_pong,
503         .surface_added = desktop_surface_added,
504         .surface_removed = desktop_surface_removed,
505         .committed = desktop_committed,
506         .show_window_menu = desktop_show_window_menu,
507         .set_parent = desktop_set_parent,
508         .move = desktop_move,
509         .resize = desktop_resize,
510         .fullscreen_requested = desktop_fullscreen_requested,
511         .maximized_requested = desktop_maximized_requested,
512         .minimized_requested = desktop_minimized_requested,
513         .set_xwayland_position = desktop_set_xwayland_position,
514 };
515
516 static void
517 ivi_shell_destroy(struct wl_listener *listener, void *data)
518 {
519         struct ivi_compositor *ivi = container_of(listener,
520                                 struct ivi_compositor, destroy_listener);
521
522         ivi_shell_finalize(ivi);
523         ivi_compositor_destroy_pending_surfaces(ivi);
524
525         weston_desktop_destroy(ivi->desktop);
526         wl_list_remove(&listener->link);
527 }
528
529 int
530 ivi_desktop_init(struct ivi_compositor *ivi)
531 {
532         ivi->desktop = weston_desktop_create(ivi->compositor, &desktop_api, ivi);
533         if (!ivi->desktop) {
534                 weston_log("Failed to create desktop globals");
535                 return -1;
536         }
537
538         if (!weston_compositor_add_destroy_listener_once(ivi->compositor,
539                         &ivi->destroy_listener, ivi_shell_destroy)) {
540                 return -1;
541         }
542
543         return 0;
544 }