KWin
Loading...
Searching...
No Matches
inputmethod_v1.cpp
Go to the documentation of this file.
1/*
2 SPDX-FileCopyrightText: 2020 Aleix Pol Gonzalez <aleixpol@kde.org>
3
4 SPDX-License-Identifier: LGPL-2.1-only OR LGPL-3.0-only OR LicenseRef-KDE-Accepted-LGPL
5*/
6
7#include "inputmethod_v1.h"
8#include "display.h"
9#include "keyboard.h"
10#include "keyboard_p.h"
11#include "output.h"
12#include "seat.h"
13#include "surface.h"
14#include "utils/common.h"
15#include "utils/ramfile.h"
16
17#include <QHash>
18
19#include <unistd.h>
20
21#include "qwayland-server-input-method-unstable-v1.h"
22#include "qwayland-server-text-input-unstable-v1.h"
23#include "qwayland-server-wayland.h"
24
25namespace KWin
26{
27static int s_version = 1;
28
29class InputKeyboardV1InterfacePrivate : public QtWaylandServer::wl_keyboard
30{
31public:
35};
36
37InputMethodGrabV1::InputMethodGrabV1(QObject *parent)
38 : QObject(parent)
39 , d(new InputKeyboardV1InterfacePrivate)
40{
41}
42
46
47void InputMethodGrabV1::sendKeymap(const QByteArray &keymap)
48{
49 RamFile keymapFile("kwin-xkb-input-method-grab-keymap", keymap.constData(), keymap.size() + 1); // include QByteArray null terminator
50
51 const auto resources = d->resourceMap();
52 for (auto r : resources) {
53 d->send_keymap(r->handle, QtWaylandServer::wl_keyboard::keymap_format::keymap_format_xkb_v1, keymapFile.fd(), keymapFile.size());
54 }
55}
56
57void InputMethodGrabV1::sendKey(quint32 serial, quint32 timestamp, quint32 key, KeyboardKeyState state)
58{
59 const auto resources = d->resourceMap();
60 for (auto r : resources) {
61 d->send_key(r->handle, serial, timestamp, key, quint32(state));
62 }
63}
64
65void InputMethodGrabV1::sendModifiers(quint32 serial, quint32 depressed, quint32 latched, quint32 locked, quint32 group)
66{
67 const auto resources = d->resourceMap();
68 for (auto r : resources) {
69 d->send_modifiers(r->handle, serial, depressed, latched, locked, group);
70 }
71}
72
73class InputMethodContextV1InterfacePrivate : public QtWaylandServer::zwp_input_method_context_v1
74{
75public:
77 : zwp_input_method_context_v1()
78 , q(q)
79 {
80 }
81
85
86 void zwp_input_method_context_v1_commit_string(Resource *, uint32_t serial, const QString &text) override
87 {
88 Q_EMIT q->commitString(serial, text);
89 }
90 void zwp_input_method_context_v1_preedit_string(Resource *, uint32_t serial, const QString &text, const QString &commit) override
91 {
92 Q_EMIT q->preeditString(serial, text, commit);
93 }
94
95 void zwp_input_method_context_v1_preedit_styling(Resource *, uint32_t index, uint32_t length, uint32_t style) override
96 {
97 Q_EMIT q->preeditStyling(index, length, style);
98 }
99 void zwp_input_method_context_v1_preedit_cursor(Resource *, int32_t index) override
100 {
101 Q_EMIT q->preeditCursor(index);
102 }
103 void zwp_input_method_context_v1_delete_surrounding_text(Resource *, int32_t index, uint32_t length) override
104 {
105 Q_EMIT q->deleteSurroundingText(index, length);
106 }
107 void zwp_input_method_context_v1_cursor_position(Resource *, int32_t index, int32_t anchor) override
108 {
109 Q_EMIT q->cursorPosition(index, anchor);
110 }
111 void zwp_input_method_context_v1_modifiers_map(Resource *, wl_array *map) override
112 {
113 const auto mods = QByteArray(static_cast<const char *>(map->data), map->size);
114
115 Q_EMIT q->modifiersMap(mods);
116 }
117 void zwp_input_method_context_v1_keysym(Resource *, uint32_t serial, uint32_t time, uint32_t sym, uint32_t state, uint32_t modifiers) override
118 {
119 Q_EMIT q->keysym(serial, time, sym, state == WL_KEYBOARD_KEY_STATE_PRESSED, modifiers);
120 }
121 void zwp_input_method_context_v1_grab_keyboard(Resource *resource, uint32_t id) override
122 {
124 m_keyboardGrab->d->add(resource->client(), id, 1);
126 }
127 void zwp_input_method_context_v1_key(Resource *, uint32_t serial, uint32_t time, uint32_t key, uint32_t state) override
128 {
129 Q_EMIT q->key(serial, time, key, state == WL_KEYBOARD_KEY_STATE_PRESSED);
130 }
132 uint32_t serial,
133 uint32_t mods_depressed,
134 uint32_t mods_latched,
135 uint32_t mods_locked,
136 uint32_t group) override
137 {
138 Q_EMIT q->modifiers(serial, mods_depressed, mods_latched, mods_locked, group);
139 }
140 void zwp_input_method_context_v1_language(Resource *, uint32_t serial, const QString &language) override
141 {
142 Q_EMIT q->language(serial, language);
143 }
144 void zwp_input_method_context_v1_text_direction(Resource *, uint32_t serial, uint32_t direction) override
145 {
146 Qt::LayoutDirection qtDirection;
147 switch (direction) {
148 case ZWP_TEXT_INPUT_V1_TEXT_DIRECTION_LTR:
149 qtDirection = Qt::LeftToRight;
150 break;
151 case ZWP_TEXT_INPUT_V1_TEXT_DIRECTION_RTL:
152 qtDirection = Qt::RightToLeft;
153 break;
154 case ZWP_TEXT_INPUT_V1_TEXT_DIRECTION_AUTO:
155 qtDirection = Qt::LayoutDirectionAuto;
156 break;
157 default:
158 Q_UNREACHABLE();
159 break;
160 }
161 Q_EMIT q->textDirection(serial, qtDirection);
162 }
163
164 void zwp_input_method_context_v1_destroy(Resource *resource) override
165 {
166 wl_resource_destroy(resource->handle);
167 }
168
170 std::unique_ptr<InputMethodGrabV1> m_keyboardGrab;
171};
172
173InputMethodContextV1Interface::InputMethodContextV1Interface(InputMethodV1Interface *parent)
174 : QObject(parent)
176{
177}
178
180
182{
183 for (auto r : d->resourceMap()) {
184 d->send_commit_state(r->handle, serial);
185 }
186}
187
189{
190 quint32 contentHint = QtWaylandServer::zwp_text_input_v1::content_hint_none;
191 quint32 contentPurpose;
192
193 if (hint.testFlag(TextInputContentHint::AutoCapitalization)) {
194 contentHint |= QtWaylandServer::zwp_text_input_v1::content_hint_auto_capitalization;
195 }
196 if (hint.testFlag(TextInputContentHint::AutoCorrection)) {
197 contentHint |= QtWaylandServer::zwp_text_input_v1::content_hint_auto_correction;
198 }
199 if (hint.testFlag(TextInputContentHint::AutoCompletion)) {
200 contentHint |= QtWaylandServer::zwp_text_input_v1::content_hint_auto_completion;
201 }
202 if (hint.testFlag(TextInputContentHint::LowerCase)) {
203 contentHint |= QtWaylandServer::zwp_text_input_v1::content_hint_lowercase;
204 }
205 if (hint.testFlag(TextInputContentHint::UpperCase)) {
206 contentHint |= QtWaylandServer::zwp_text_input_v1::content_hint_uppercase;
207 }
208 if (hint.testFlag(TextInputContentHint::TitleCase)) {
209 contentHint |= QtWaylandServer::zwp_text_input_v1::content_hint_titlecase;
210 }
211 if (hint.testFlag(TextInputContentHint::HiddenText)) {
212 contentHint |= QtWaylandServer::zwp_text_input_v1::content_hint_hidden_text;
213 }
214 if (hint.testFlag(TextInputContentHint::SensitiveData)) {
215 contentHint |= QtWaylandServer::zwp_text_input_v1::content_hint_lowercase;
216 }
217 if (hint.testFlag(TextInputContentHint::Latin)) {
218 contentHint |= QtWaylandServer::zwp_text_input_v1::content_hint_latin;
219 }
220 if (hint.testFlag(TextInputContentHint::MultiLine)) {
221 contentHint |= QtWaylandServer::zwp_text_input_v1::content_hint_multiline;
222 }
223 if (hint.testFlag(TextInputContentHint::None)) {
224 contentHint |= QtWaylandServer::zwp_text_input_v1::content_hint_none;
225 }
226
227 switch (purpose) {
229 contentPurpose = QtWaylandServer::zwp_text_input_v1::content_purpose_alpha;
230 break;
232 contentPurpose = QtWaylandServer::zwp_text_input_v1::content_purpose_digits;
233 break;
235 contentPurpose = QtWaylandServer::zwp_text_input_v1::content_purpose_number;
236 break;
238 contentPurpose = QtWaylandServer::zwp_text_input_v1::content_purpose_phone;
239 break;
241 contentPurpose = QtWaylandServer::zwp_text_input_v1::content_purpose_url;
242 break;
244 contentPurpose = QtWaylandServer::zwp_text_input_v1::content_purpose_email;
245 break;
247 contentPurpose = QtWaylandServer::zwp_text_input_v1::content_purpose_name;
248 break;
250 contentPurpose = QtWaylandServer::zwp_text_input_v1::content_purpose_password;
251 break;
253 contentPurpose = QtWaylandServer::zwp_text_input_v1::content_purpose_date;
254 break;
256 contentPurpose = QtWaylandServer::zwp_text_input_v1::content_purpose_time;
257 break;
259 contentPurpose = QtWaylandServer::zwp_text_input_v1::content_purpose_datetime;
260 break;
262 contentPurpose = QtWaylandServer::zwp_text_input_v1::content_purpose_terminal;
263 break;
265 default:
266 contentPurpose = QtWaylandServer::zwp_text_input_v1::content_purpose_normal;
267 }
268
269 for (auto r : d->resourceMap()) {
270 d->send_content_type(r->handle, contentHint, contentPurpose);
271 }
272}
273
274void InputMethodContextV1Interface::sendInvokeAction(uint32_t button, uint32_t index)
275{
276 for (auto r : d->resourceMap()) {
277 d->send_invoke_action(r->handle, button, index);
278 }
279}
280
282{
283 for (auto r : d->resourceMap()) {
284 d->send_preferred_language(r->handle, language);
285 }
286}
287
289{
290 for (auto r : d->resourceMap()) {
291 d->send_reset(r->handle);
292 }
293}
294
295void InputMethodContextV1Interface::sendSurroundingText(const QString &text, uint32_t cursor, uint32_t anchor)
296{
297 for (auto r : d->resourceMap()) {
298 d->send_surrounding_text(r->handle, text, cursor, anchor);
299 }
300}
301
303{
304 return d->m_keyboardGrab.get();
305}
306
307class InputPanelSurfaceV1InterfacePrivate : public QtWaylandServer::zwp_input_panel_surface_v1
308{
310
311public:
313 : zwp_input_panel_surface_v1()
314 , q(q)
316 {
317 }
318
320 {
321 Q_EMIT q->overlayPanel();
322 }
323
324 void zwp_input_panel_surface_v1_set_toplevel(Resource *, struct ::wl_resource *output, uint32_t position) override
325 {
327 }
328
330 {
331 delete q;
332 }
333
335 QPointer<SurfaceInterface> surface;
336};
337
338InputPanelSurfaceV1Interface::InputPanelSurfaceV1Interface(SurfaceInterface *surface, quint32 id, QObject *parent)
339 : QObject(parent)
340 , d(new InputPanelSurfaceV1InterfacePrivate(surface, id, this))
341{
342}
343
348
350{
351 static SurfaceRole role(QByteArrayLiteral("input_panel_surface_v1"));
352 return &role;
353}
354
355class InputPanelV1InterfacePrivate : public QtWaylandServer::zwp_input_panel_v1
356{
357public:
359 : zwp_input_panel_v1(*d, s_version)
360 , q(q)
361 {
362 }
363
364 void zwp_input_panel_v1_get_input_panel_surface(Resource *resource, uint32_t id, struct ::wl_resource *surfaceResource) override
365 {
366 auto surface = SurfaceInterface::get(surfaceResource);
367
368 if (const SurfaceRole *role = surface->role()) {
370 wl_resource_post_error(resource->handle, 0, "the surface already has a role assigned %s", role->name().constData());
371 return;
372 }
373 } else {
374 surface->setRole(InputPanelSurfaceV1Interface::role());
375 }
376
377 auto interface = new InputPanelSurfaceV1Interface(surface, id, nullptr);
378 interface->d->init(resource->client(), id, resource->version());
379
380 Q_EMIT q->inputPanelSurfaceAdded(interface);
381 }
382
384};
385
387 : QObject(parent)
388 , d(new InputPanelV1InterfacePrivate(this, display))
389{
390}
391
393
395{
396 return d->surface;
397}
398
399class InputMethodV1InterfacePrivate : public QtWaylandServer::zwp_input_method_v1
400{
401public:
403 : zwp_input_method_v1(*d, s_version)
404 , q(q)
405 , m_display(d)
406 {
407 }
408
409 void zwp_input_method_v1_bind_resource(Resource *resource) override
410 {
411 if (!m_context) {
412 return;
413 }
414
415 auto addedResource = m_context->d->add(resource->client(), resource->version());
416 send_activate(resource->handle, addedResource->handle);
417 }
418
419 std::unique_ptr<InputMethodContextV1Interface> m_context;
422};
423
425 : QObject(parent)
426 , d(new InputMethodV1InterfacePrivate(d, this))
427{
428}
429
431
433{
434 if (d->m_context) {
435 return;
436 }
437
438 d->m_context.reset(new InputMethodContextV1Interface(this));
439
440 for (auto resource : d->resourceMap()) {
441 auto connection = d->m_context->d->add(resource->client(), resource->version());
442 d->send_activate(resource->handle, connection->handle);
443 }
444}
445
447{
448 if (!d->m_context) {
449 return;
450 }
451
452 for (auto resource : d->resourceMap()) {
453 auto connection = d->m_context->d->resourceMap().value(resource->client());
454 if (connection) {
455 d->send_deactivate(resource->handle, connection->handle);
456 }
457 }
458 d->m_context.reset();
459}
460
462{
463 return d->m_context.get();
464}
465
466}
467
468#include "moc_inputmethod_v1.cpp"
Class holding the Wayland server display loop.
Definition display.h:34
void language(quint32 serial, const QString &language)
void deleteSurroundingText(qint32 index, quint32 length)
InputMethodGrabV1 * keyboardGrab() const
void sendSurroundingText(const QString &text, quint32 cursor, quint32 anchor)
void modifiersMap(const QByteArray &map)
void sendInvokeAction(quint32 button, quint32 index)
void preeditString(quint32 serial, const QString &text, const QString &commit)
void key(quint32 serial, quint32 time, quint32 key, bool pressed)
void modifiers(quint32 serial, quint32 mods_depressed, quint32 mods_latched, quint32 mods_locked, quint32 group)
void cursorPosition(qint32 index, qint32 anchor)
void sendContentType(KWin::TextInputContentHints hint, KWin::TextInputContentPurpose purpose)
void keysym(quint32 serial, quint32 time, quint32 sym, bool pressed, quint32 modifiers)
void keyboardGrabRequested(InputMethodGrabV1 *keyboardGrab)
void sendPreferredLanguage(const QString &language)
void textDirection(quint32 serial, Qt::LayoutDirection direction)
void preeditStyling(quint32 index, quint32 length, quint32 style)
void commitString(quint32 serial, const QString &text)
void zwp_input_method_context_v1_preedit_styling(Resource *, uint32_t index, uint32_t length, uint32_t style) override
InputMethodContextV1Interface *const q
void zwp_input_method_context_v1_grab_keyboard(Resource *resource, uint32_t id) override
void zwp_input_method_context_v1_delete_surrounding_text(Resource *, int32_t index, uint32_t length) override
void zwp_input_method_context_v1_modifiers_map(Resource *, wl_array *map) override
void zwp_input_method_context_v1_preedit_string(Resource *, uint32_t serial, const QString &text, const QString &commit) override
InputMethodContextV1InterfacePrivate(InputMethodContextV1Interface *q)
void zwp_input_method_context_v1_key(Resource *, uint32_t serial, uint32_t time, uint32_t key, uint32_t state) override
void zwp_input_method_context_v1_text_direction(Resource *, uint32_t serial, uint32_t direction) override
void zwp_input_method_context_v1_modifiers(Resource *, uint32_t serial, uint32_t mods_depressed, uint32_t mods_latched, uint32_t mods_locked, uint32_t group) override
void zwp_input_method_context_v1_commit_string(Resource *, uint32_t serial, const QString &text) override
void zwp_input_method_context_v1_language(Resource *, uint32_t serial, const QString &language) override
void zwp_input_method_context_v1_cursor_position(Resource *, int32_t index, int32_t anchor) override
void zwp_input_method_context_v1_destroy(Resource *resource) override
void zwp_input_method_context_v1_keysym(Resource *, uint32_t serial, uint32_t time, uint32_t sym, uint32_t state, uint32_t modifiers) override
std::unique_ptr< InputMethodGrabV1 > m_keyboardGrab
void zwp_input_method_context_v1_preedit_cursor(Resource *, int32_t index) override
void sendKey(quint32 serial, quint32 timestamp, quint32 key, KeyboardKeyState state)
void sendModifiers(quint32 serial, quint32 depressed, quint32 latched, quint32 locked, quint32 group)
void sendKeymap(const QByteArray &content)
InputMethodContextV1Interface * context() const
InputMethodV1Interface(Display *d, QObject *parent)
std::unique_ptr< InputMethodContextV1Interface > m_context
InputMethodV1Interface *const q
InputMethodV1InterfacePrivate(Display *d, InputMethodV1Interface *q)
void zwp_input_method_v1_bind_resource(Resource *resource) override
SurfaceInterface * surface() const
void topLevel(OutputInterface *output, Position position)
void zwp_input_panel_surface_v1_set_toplevel(Resource *, struct ::wl_resource *output, uint32_t position) override
void zwp_input_panel_surface_v1_destroy_resource(Resource *) override
InputPanelSurfaceV1InterfacePrivate(SurfaceInterface *surface, quint32 id, InputPanelSurfaceV1Interface *q)
InputPanelSurfaceV1Interface *const q
void zwp_input_panel_surface_v1_set_overlay_panel(Resource *) override
InputPanelV1Interface(Display *display, QObject *parent)
void inputPanelSurfaceAdded(InputPanelSurfaceV1Interface *surface)
void zwp_input_panel_v1_get_input_panel_surface(Resource *resource, uint32_t id, struct ::wl_resource *surfaceResource) override
InputPanelV1InterfacePrivate(InputPanelV1Interface *q, Display *d)
InputPanelV1Interface *const q
static OutputInterface * get(wl_resource *native)
Definition output.cpp:320
Creates a file in memory.
Definition ramfile.h:47
int fd() const
Definition ramfile.cpp:150
int size() const
Definition ramfile.cpp:159
Resource representing a wl_surface.
Definition surface.h:80
static SurfaceInterface * get(wl_resource *native)
Definition surface.cpp:819
KWIN_EXPORT xcb_connection_t * connection()
Definition xcb.h:19
TextInputContentPurpose
Definition textinput.h:72
KeyboardKeyState
Definition seat.h:82