| // Copyright 2018 The Chromium OS Authors. All rights reserved. |
| // Use of this source code is governed by a BSD-style license that can be |
| // found in the LICENSE file. |
| |
| #include "sommelier.h" // NOLINT(build/include_directory) |
| #include "sommelier-tracing.h" // NOLINT(build/include_directory) |
| |
| #include <assert.h> |
| #include <stdlib.h> |
| #include <string.h> |
| |
| #include "aura-shell-client-protocol.h" // NOLINT(build/include_directory) |
| #include "gtk-shell-server-protocol.h" // NOLINT(build/include_directory) |
| |
| struct sl_host_gtk_shell { |
| struct sl_aura_shell* aura_shell; |
| struct wl_resource* resource; |
| struct zaura_shell* proxy; |
| struct wl_callback* callback; |
| char* startup_id; |
| struct wl_list surfaces; |
| }; |
| |
| struct sl_host_gtk_surface { |
| struct wl_resource* resource; |
| struct zaura_surface* proxy; |
| struct wl_list link; |
| struct sl_aura_shell* aura_shell; |
| }; |
| |
| static void sl_gtk_surface_set_dbus_properties( |
| struct wl_client* client, |
| struct wl_resource* resource, |
| const char* application_id, |
| const char* app_menu_path, |
| const char* menubar_path, |
| const char* window_object_path, |
| const char* application_object_path, |
| const char* unique_bus_name) { |
| struct sl_host_gtk_surface* host = |
| static_cast<sl_host_gtk_surface*>(wl_resource_get_user_data(resource)); |
| |
| zaura_surface_set_application_id(host->proxy, application_id); |
| } // NOLINT(whitespace/indent) |
| |
| static void sl_gtk_surface_set_modal(struct wl_client* client, |
| struct wl_resource* resource) {} |
| |
| static void sl_gtk_surface_unset_modal(struct wl_client* client, |
| struct wl_resource* resource) {} |
| |
| static void sl_gtk_surface_present(struct wl_client* client, |
| struct wl_resource* resource, |
| uint32_t time) {} |
| |
| static const struct gtk_surface1_interface sl_gtk_surface_implementation = { |
| sl_gtk_surface_set_dbus_properties, sl_gtk_surface_set_modal, |
| sl_gtk_surface_unset_modal, sl_gtk_surface_present}; |
| |
| static void sl_destroy_host_gtk_surface(struct wl_resource* resource) { |
| struct sl_host_gtk_surface* host = |
| static_cast<sl_host_gtk_surface*>(wl_resource_get_user_data(resource)); |
| |
| zaura_surface_destroy(host->proxy); |
| wl_list_remove(&host->link); |
| wl_resource_set_user_data(resource, NULL); |
| free(host); |
| } |
| |
| static void sl_gtk_shell_get_gtk_surface(struct wl_client* client, |
| struct wl_resource* resource, |
| uint32_t id, |
| struct wl_resource* surface_resource) { |
| struct sl_host_gtk_shell* host = |
| static_cast<sl_host_gtk_shell*>(wl_resource_get_user_data(resource)); |
| struct sl_host_surface* host_surface = static_cast<sl_host_surface*>( |
| wl_resource_get_user_data(surface_resource)); |
| struct sl_host_gtk_surface* host_gtk_surface = |
| static_cast<sl_host_gtk_surface*>(malloc(sizeof(*host_gtk_surface))); |
| assert(host_gtk_surface); |
| |
| wl_list_insert(&host->surfaces, &host_gtk_surface->link); |
| host_gtk_surface->aura_shell = host->aura_shell; |
| host_gtk_surface->resource = |
| wl_resource_create(client, >k_surface1_interface, 1, id); |
| wl_resource_set_implementation(host_gtk_surface->resource, |
| &sl_gtk_surface_implementation, |
| host_gtk_surface, sl_destroy_host_gtk_surface); |
| host_gtk_surface->proxy = |
| zaura_shell_get_aura_surface(host->proxy, host_surface->proxy); |
| zaura_surface_set_startup_id(host_gtk_surface->proxy, host->startup_id); |
| } |
| |
| static void sl_gtk_shell_set_startup_id(struct wl_client* client, |
| struct wl_resource* resource, |
| const char* startup_id) { |
| struct sl_host_gtk_shell* host = |
| static_cast<sl_host_gtk_shell*>(wl_resource_get_user_data(resource)); |
| struct sl_host_gtk_surface* surface; |
| |
| free(host->startup_id); |
| host->startup_id = startup_id ? strdup(startup_id) : NULL; |
| |
| wl_list_for_each(surface, &host->surfaces, link) |
| zaura_surface_set_startup_id(surface->proxy, host->startup_id); |
| } |
| |
| static void sl_gtk_shell_system_bell(struct wl_client* client, |
| struct wl_resource* resource, |
| struct wl_resource* surface_resource) {} |
| |
| static const struct gtk_shell1_interface sl_gtk_shell_implementation = { |
| sl_gtk_shell_get_gtk_surface, sl_gtk_shell_set_startup_id, |
| sl_gtk_shell_system_bell}; |
| |
| static void sl_destroy_host_gtk_shell(struct wl_resource* resource) { |
| struct sl_host_gtk_shell* host = |
| static_cast<sl_host_gtk_shell*>(wl_resource_get_user_data(resource)); |
| |
| free(host->startup_id); |
| wl_callback_destroy(host->callback); |
| zaura_shell_destroy(host->proxy); |
| wl_resource_set_user_data(resource, NULL); |
| free(host); |
| } |
| |
| static void sl_gtk_shell_callback_done(void* data, |
| struct wl_callback* callback, |
| uint32_t serial) { |
| TRACE_EVENT("shell", "sl_gtk_shell_callback_done"); |
| struct sl_host_gtk_shell* host = |
| static_cast<sl_host_gtk_shell*>(wl_callback_get_user_data(callback)); |
| |
| gtk_shell1_send_capabilities(host->resource, 0); |
| } |
| |
| static const struct wl_callback_listener sl_gtk_shell_callback_listener = { |
| sl_gtk_shell_callback_done}; |
| |
| static void sl_bind_host_gtk_shell(struct wl_client* client, |
| void* data, |
| uint32_t version, |
| uint32_t id) { |
| struct sl_context* ctx = (struct sl_context*)data; |
| struct sl_host_gtk_shell* host = |
| static_cast<sl_host_gtk_shell*>(malloc(sizeof(*host))); |
| assert(host); |
| host->aura_shell = ctx->aura_shell; |
| host->startup_id = NULL; |
| wl_list_init(&host->surfaces); |
| host->resource = wl_resource_create(client, >k_shell1_interface, 1, id); |
| wl_resource_set_implementation(host->resource, &sl_gtk_shell_implementation, |
| host, sl_destroy_host_gtk_shell); |
| host->proxy = static_cast<zaura_shell*>(wl_registry_bind( |
| wl_display_get_registry(ctx->display), ctx->aura_shell->id, |
| &zaura_shell_interface, ctx->aura_shell->version)); |
| zaura_shell_set_user_data(host->proxy, host); |
| |
| host->callback = wl_display_sync(ctx->aura_shell->ctx->display); |
| wl_callback_set_user_data(host->callback, host); |
| wl_callback_add_listener(host->callback, &sl_gtk_shell_callback_listener, |
| host); |
| } |
| |
| struct sl_global* sl_gtk_shell_global_create(struct sl_context* ctx) { |
| return sl_global_create(ctx, >k_shell1_interface, 1, ctx, |
| sl_bind_host_gtk_shell); |
| } |