Extract the text_model_manager interface from input_method
authorPhilipp Brüschweiler <blei42@gmail.com>
Wed, 11 Jul 2012 20:25:31 +0000 (22:25 +0200)
committerKristian Høgsberg <krh@bitplanet.net>
Sun, 22 Jul 2012 16:06:10 +0000 (12:06 -0400)
This is necessary because all clients need a way to create
text_models, but only one client at a time can be bound to
the input_method global (else we don't know to whom we are
supposed to send events).

clients/editor.c
protocol/text.xml
src/text-backend.c

index 9498d53..19f7073 100644 (file)
@@ -40,7 +40,7 @@ struct text_entry {
 };
 
 struct editor {
-       struct input_method *input_method;
+       struct text_model_manager *text_model_manager;
        struct display *display;
        struct window *window;
        struct widget *widget;
@@ -86,7 +86,7 @@ text_entry_create(struct editor *editor, const char *text)
        entry->widget = editor->widget;
        entry->text = strdup(text);
        entry->active = 0;
-       entry->model = input_method_create_text_model(editor->input_method, surface);
+       entry->model = text_model_manager_create_text_model(editor->text_model_manager, surface);
        text_model_add_listener(entry->model, &text_model_listener, entry);
 
        return entry;
@@ -267,8 +267,9 @@ global_handler(struct wl_display *display, uint32_t id,
 {
        struct editor *editor = data;
 
-       if (!strcmp(interface, "input_method")) {
-               editor->input_method = wl_display_bind(display, id, &input_method_interface);
+       if (!strcmp(interface, "text_model_manager")) {
+               editor->text_model_manager = wl_display_bind(display, id,
+                                                            &text_model_manager_interface);
        }
 }
 
index d22156c..a14277a 100644 (file)
     <event name="locale"/>      
   </interface>
 
-  <interface name="input_method" version="1">
+  <interface name="text_model_manager" version="1">
     <request name="create_text_model">
       <arg name="id" type="new_id" interface="text_model"/>
       <arg name="surface" type="object" interface="wl_surface"/>
     </request>
+  </interface>
 
+  <interface name="input_method" version="1">
     <request name="commit_string">
       <arg name="text" type="string"/>
       <arg name="index" type="uint"/>
index 1eb49f1..353a983 100644 (file)
@@ -36,12 +36,13 @@ struct text_model {
 };
 
 struct input_method {
-       struct wl_object base;
-       struct weston_compositor *ec;
-       struct wl_global *global;
+       struct wl_resource *input_method_binding;
+       struct wl_global *input_method_global;
+       struct wl_global *text_model_manager_global;
        struct wl_listener destroy_listener;
-       struct wl_list models;
 
+       struct weston_compositor *ec;
+       struct wl_list models;
        struct text_model *active_model;
 };
 
@@ -144,10 +145,10 @@ struct text_model_interface text_model_implementation = {
        text_model_set_content_type
 };
 
-static void input_method_create_text_model(struct wl_client *client,
-                                          struct wl_resource *resource,
-                                          uint32_t id,
-                                          struct wl_resource *surface)
+static void text_model_manager_create_text_model(struct wl_client *client,
+                                                struct wl_resource *resource,
+                                                uint32_t id,
+                                                struct wl_resource *surface)
 {
        struct input_method *input_method = resource->data;
        struct text_model *text_model;
@@ -169,6 +170,25 @@ static void input_method_create_text_model(struct wl_client *client,
        wl_list_insert(&input_method->models, &text_model->link);
 };
 
+static const struct text_model_manager_interface text_model_manager_implementation = {
+       text_model_manager_create_text_model
+};
+
+static void
+bind_text_model_manager(struct wl_client *client,
+                       void *data,
+                       uint32_t version,
+                       uint32_t id)
+{
+       struct input_method *input_method = data;
+
+       /* No checking for duplicate binding necessary.
+        * No events have to be sent, so we don't need the return value. */
+       wl_client_add_object(client, &text_model_manager_interface,
+                            &text_model_manager_implementation,
+                            id, input_method);
+}
+
 static void
 input_method_commit_string(struct wl_client *client,
                           struct wl_resource *resource,
@@ -182,19 +202,41 @@ input_method_commit_string(struct wl_client *client,
        }
 }
 
-struct input_method_interface input_method_implementation = {
-       input_method_create_text_model,
+static const struct input_method_interface input_method_implementation = {
        input_method_commit_string
 };
 
 static void
+unbind_input_method(struct wl_resource *resource)
+{
+       struct input_method *input_method = resource->data;
+
+       input_method->input_method_binding = NULL;
+       free(resource);
+}
+
+static void
 bind_input_method(struct wl_client *client,
                  void *data,
                  uint32_t version,
                  uint32_t id)
 {
-       wl_client_add_object(client, &input_method_interface,
-                            &input_method_implementation, id, data);
+       struct input_method *input_method = data;
+       struct wl_resource *resource;
+
+       resource = wl_client_add_object(client, &input_method_interface,
+                                       &input_method_implementation,
+                                       id, input_method);
+
+       if (input_method->input_method_binding == NULL) {
+               resource->destroy = unbind_input_method;
+               input_method->input_method_binding = resource;
+               return;
+       }
+
+       wl_resource_post_error(resource, WL_DISPLAY_ERROR_INVALID_OBJECT,
+                              "interface object already bound");
+       wl_resource_destroy(resource);
 }
 
 static void
@@ -204,7 +246,9 @@ input_method_notifier_destroy(struct wl_listener *listener, void *data)
                container_of(listener, struct input_method, destroy_listener);
 
        wl_display_remove_global(input_method->ec->wl_display,
-                                input_method->global);
+                                input_method->input_method_global);
+       wl_display_remove_global(input_method->ec->wl_display,
+                                input_method->text_model_manager_global);
        free(input_method);
 }
 
@@ -215,18 +259,20 @@ input_method_create(struct weston_compositor *ec)
 
        input_method = calloc(1, sizeof *input_method);
 
-       input_method->base.interface = &input_method_interface;
-       input_method->base.implementation =
-               (void(**)(void)) &input_method_implementation;
        input_method->ec = ec;
        input_method->active_model = NULL;
 
        wl_list_init(&input_method->models);
 
-       input_method->global = wl_display_add_global(ec->wl_display,
-                                                    &input_method_interface,
-                                                    input_method,
-                                                    bind_input_method);
+       input_method->input_method_global =
+               wl_display_add_global(ec->wl_display,
+                                     &input_method_interface,
+                                     input_method, bind_input_method);
+
+       input_method->text_model_manager_global =
+               wl_display_add_global(ec->wl_display,
+                                     &text_model_manager_interface,
+                                     input_method, bind_text_model_manager);
 
        input_method->destroy_listener.notify = input_method_notifier_destroy;
        wl_signal_add(&ec->destroy_signal, &input_method->destroy_listener);