Extract the text_model_manager interface from input_method

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).
dev
Philipp Brüschweiler 13 years ago committed by Kristian Høgsberg
parent f97f379dfc
commit f25602bdc0
  1. 9
      clients/editor.c
  2. 4
      protocol/text.xml
  3. 86
      src/text-backend.c

@ -40,7 +40,7 @@ struct text_entry {
}; };
struct editor { struct editor {
struct input_method *input_method; struct text_model_manager *text_model_manager;
struct display *display; struct display *display;
struct window *window; struct window *window;
struct widget *widget; struct widget *widget;
@ -86,7 +86,7 @@ text_entry_create(struct editor *editor, const char *text)
entry->widget = editor->widget; entry->widget = editor->widget;
entry->text = strdup(text); entry->text = strdup(text);
entry->active = 0; 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); text_model_add_listener(entry->model, &text_model_listener, entry);
return entry; return entry;
@ -267,8 +267,9 @@ global_handler(struct wl_display *display, uint32_t id,
{ {
struct editor *editor = data; struct editor *editor = data;
if (!strcmp(interface, "input_method")) { if (!strcmp(interface, "text_model_manager")) {
editor->input_method = wl_display_bind(display, id, &input_method_interface); editor->text_model_manager = wl_display_bind(display, id,
&text_model_manager_interface);
} }
} }

@ -36,12 +36,14 @@
<event name="locale"/> <event name="locale"/>
</interface> </interface>
<interface name="input_method" version="1"> <interface name="text_model_manager" version="1">
<request name="create_text_model"> <request name="create_text_model">
<arg name="id" type="new_id" interface="text_model"/> <arg name="id" type="new_id" interface="text_model"/>
<arg name="surface" type="object" interface="wl_surface"/> <arg name="surface" type="object" interface="wl_surface"/>
</request> </request>
</interface>
<interface name="input_method" version="1">
<request name="commit_string"> <request name="commit_string">
<arg name="text" type="string"/> <arg name="text" type="string"/>
<arg name="index" type="uint"/> <arg name="index" type="uint"/>

@ -36,12 +36,13 @@ struct text_model {
}; };
struct input_method { struct input_method {
struct wl_object base; struct wl_resource *input_method_binding;
struct weston_compositor *ec; struct wl_global *input_method_global;
struct wl_global *global; struct wl_global *text_model_manager_global;
struct wl_listener destroy_listener; struct wl_listener destroy_listener;
struct wl_list models;
struct weston_compositor *ec;
struct wl_list models;
struct text_model *active_model; struct text_model *active_model;
}; };
@ -144,10 +145,10 @@ struct text_model_interface text_model_implementation = {
text_model_set_content_type text_model_set_content_type
}; };
static void input_method_create_text_model(struct wl_client *client, static void text_model_manager_create_text_model(struct wl_client *client,
struct wl_resource *resource, struct wl_resource *resource,
uint32_t id, uint32_t id,
struct wl_resource *surface) struct wl_resource *surface)
{ {
struct input_method *input_method = resource->data; struct input_method *input_method = resource->data;
struct text_model *text_model; 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); 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 static void
input_method_commit_string(struct wl_client *client, input_method_commit_string(struct wl_client *client,
struct wl_resource *resource, struct wl_resource *resource,
@ -182,19 +202,41 @@ input_method_commit_string(struct wl_client *client,
} }
} }
struct input_method_interface input_method_implementation = { static const struct input_method_interface input_method_implementation = {
input_method_create_text_model,
input_method_commit_string 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 static void
bind_input_method(struct wl_client *client, bind_input_method(struct wl_client *client,
void *data, void *data,
uint32_t version, uint32_t version,
uint32_t id) uint32_t id)
{ {
wl_client_add_object(client, &input_method_interface, struct input_method *input_method = data;
&input_method_implementation, id, 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 static void
@ -204,7 +246,9 @@ input_method_notifier_destroy(struct wl_listener *listener, void *data)
container_of(listener, struct input_method, destroy_listener); container_of(listener, struct input_method, destroy_listener);
wl_display_remove_global(input_method->ec->wl_display, 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); free(input_method);
} }
@ -215,18 +259,20 @@ input_method_create(struct weston_compositor *ec)
input_method = calloc(1, sizeof *input_method); 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->ec = ec;
input_method->active_model = NULL; input_method->active_model = NULL;
wl_list_init(&input_method->models); wl_list_init(&input_method->models);
input_method->global = wl_display_add_global(ec->wl_display, input_method->input_method_global =
&input_method_interface, wl_display_add_global(ec->wl_display,
input_method, &input_method_interface,
bind_input_method); 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; input_method->destroy_listener.notify = input_method_notifier_destroy;
wl_signal_add(&ec->destroy_signal, &input_method->destroy_listener); wl_signal_add(&ec->destroy_signal, &input_method->destroy_listener);

Loading…
Cancel
Save