Qt
Internal/Contributor docs for the Qt SDK. Note: These are NOT official API docs; those are found at https://doc.qt.io/
Loading...
Searching...
No Matches
qwaylandtextinputv2.cpp
Go to the documentation of this file.
1// Copyright (C) 2016 Klarälvdalens Datakonsult AB, a KDAB Group company, info@kdab.com
2// SPDX-License-Identifier: LicenseRef-Qt-Commercial OR LGPL-3.0-only OR GPL-2.0-only OR GPL-3.0-only
3// Qt-Security score:significant reason:default
4
5#include <qpa/qplatforminputcontext.h>
6
8
11#include "qwaylandinputmethodeventbuilder_p.h"
12
13#include <QtCore/qloggingcategory.h>
14#include <QtGui/QGuiApplication>
15#include <QtGui/private/qguiapplication_p.h>
16#include <QtGui/private/qhighdpiscaling_p.h>
17#include <QtGui/qpa/qplatformintegration.h>
18#include <QtGui/qevent.h>
19#include <QtGui/qwindow.h>
20#include <QTextCharFormat>
21#include <QList>
22#include <QRectF>
23#include <QLocale>
24
26
27namespace QtWaylandClient {
28
29namespace {
30
31const Qt::InputMethodQueries supportedQueries2 = Qt::ImEnabled |
32 Qt::ImSurroundingText |
33 Qt::ImCursorPosition |
34 Qt::ImAnchorPosition |
35 Qt::ImHints |
36 Qt::ImCursorRectangle |
37 Qt::ImPreferredLanguage;
38}
39
40QWaylandTextInputv2::QWaylandTextInputv2(QWaylandDisplay *display, struct ::zwp_text_input_v2 *text_input)
41 : QtWayland::zwp_text_input_v2(text_input)
42 , m_display(display)
43{
44}
45
47{
48 if (m_resetCallback)
49 wl_callback_destroy(m_resetCallback);
50 destroy();
51}
52
54{
55 qCDebug(qLcQpaInputMethods()) << Q_FUNC_INFO;
56 show_input_panel();
57}
59{
60 qCDebug(qLcQpaInputMethods()) << Q_FUNC_INFO;
61 hide_input_panel();
62}
63void QWaylandTextInputv2::enableSurface(::wl_surface *surface)
64{
65 qCDebug(qLcQpaInputMethods()) << Q_FUNC_INFO << surface;
66 enable(surface);
67}
68void QWaylandTextInputv2::disableSurface(::wl_surface *surface)
69{
70 qCDebug(qLcQpaInputMethods()) << Q_FUNC_INFO << surface;
71 disable(surface);
72}
73
75{
76 qCDebug(qLcQpaInputMethods()) << Q_FUNC_INFO;
77 m_builder.reset();
78 m_preeditCommit = QString();
79 updateState(Qt::ImQueryAll, QtWayland::zwp_text_input_v2::update_state_reset);
80}
81
83{
84 qCDebug(qLcQpaInputMethods()) << Q_FUNC_INFO;
85 if (QObject *o = QGuiApplication::focusObject()) {
86 if (!m_preeditCommit.isEmpty()) {
87
88 QInputMethodEvent event;
89 event.setCommitString(m_preeditCommit);
90 m_preeditCommit = QString();
91
92 QCoreApplication::sendEvent(o, &event);
93 }
94 }
95
96 reset();
97}
98
99const wl_callback_listener QWaylandTextInputv2::callbackListener = {
101};
102
103void QWaylandTextInputv2::resetCallback(void *data, wl_callback *, uint32_t)
104{
105 QWaylandTextInputv2 *self = static_cast<QWaylandTextInputv2*>(data);
106
107 if (self->m_resetCallback) {
108 wl_callback_destroy(self->m_resetCallback);
109 self->m_resetCallback = nullptr;
110 }
111}
112
113void QWaylandTextInputv2::updateState(Qt::InputMethodQueries queries, uint32_t flags)
114{
115 qCDebug(qLcQpaInputMethods()) << Q_FUNC_INFO << queries << flags;
116 if (!QGuiApplication::focusObject())
117 return;
118
119 if (!QGuiApplication::focusWindow() || !QGuiApplication::focusWindow()->handle())
120 return;
121
122 auto *window = static_cast<QWaylandWindow *>(QGuiApplication::focusWindow()->handle());
123 auto *surface = window->wlSurface();
124 if (!surface || (surface != m_surface))
125 return;
126
127 queries &= supportedQueries2;
128
129 // Surrounding text, cursor and anchor positions are transferred together
130 if ((queries & Qt::ImSurroundingText) || (queries & Qt::ImCursorPosition) || (queries & Qt::ImAnchorPosition))
131 queries |= Qt::ImSurroundingText | Qt::ImCursorPosition | Qt::ImAnchorPosition;
132
133 QInputMethodQueryEvent event(queries);
134 QCoreApplication::sendEvent(QGuiApplication::focusObject(), &event);
135
136 if ((queries & Qt::ImSurroundingText) || (queries & Qt::ImCursorPosition) || (queries & Qt::ImAnchorPosition)) {
137 QString text = event.value(Qt::ImSurroundingText).toString();
138 int cursor = event.value(Qt::ImCursorPosition).toInt();
139 int anchor = event.value(Qt::ImAnchorPosition).toInt();
140
141 // Make sure text is not too big
142 if (text.toUtf8().size() > 2048) {
143 int c = qAbs(cursor - anchor) <= 512 ? qMin(cursor, anchor) + qAbs(cursor - anchor) / 2: cursor;
144
145 const int offset = c - qBound(0, c, 512 - qMin(text.size() - c, 256));
146 text = text.mid(offset + c - 256, 512);
147 cursor -= offset;
148 anchor -= offset;
149 }
150
151 set_surrounding_text(text, QWaylandInputMethodEventBuilder::indexToWayland(text, cursor), QWaylandInputMethodEventBuilder::indexToWayland(text, anchor));
152 }
153
154 if (queries & Qt::ImHints) {
155 QWaylandInputMethodContentType contentType = QWaylandInputMethodContentType::convert(static_cast<Qt::InputMethodHints>(event.value(Qt::ImHints).toInt()));
156 set_content_type(contentType.hint, contentType.purpose);
157 }
158
159 if (queries & Qt::ImCursorRectangle) {
160 const QRect &cRect = event.value(Qt::ImCursorRectangle).toRect();
161 const QRect &windowRect = QGuiApplication::inputMethod()->inputItemTransform().mapRect(cRect);
162 const QRect &nativeRect = QHighDpi::toNativePixels(windowRect, QGuiApplication::focusWindow());
163 const QMargins margins = window->clientSideMargins();
164 const QRect &surfaceRect = nativeRect.translated(margins.left(), margins.top());
165 set_cursor_rectangle(surfaceRect.x(), surfaceRect.y(), surfaceRect.width(), surfaceRect.height());
166 }
167
168 if (queries & Qt::ImPreferredLanguage) {
169 const QString &language = event.value(Qt::ImPreferredLanguage).toString();
170 set_preferred_language(language);
171 }
172
173 update_state(m_serial, flags);
174 if (flags != QtWayland::zwp_text_input_v2::update_state_change) {
175 if (m_resetCallback)
176 wl_callback_destroy(m_resetCallback);
177 m_resetCallback = wl_display_sync(m_display->wl_display());
178 wl_callback_add_listener(m_resetCallback, &QWaylandTextInputv2::callbackListener, this);
179 }
180}
181
183{
184 // Not supported yet
185}
186
188{
189 return m_inputPanelVisible;
190}
191
193{
194 return m_keyboardRectangle;
195}
196
198{
199 return m_locale;
200}
201
203{
204 return m_inputDirection;
205}
206
207void QWaylandTextInputv2::zwp_text_input_v2_enter(uint32_t serial, ::wl_surface *surface)
208{
209 qCDebug(qLcQpaInputMethods()) << Q_FUNC_INFO << serial << surface;
210 m_serial = serial;
211 m_surface = surface;
212
213 updateState(Qt::ImQueryAll, QtWayland::zwp_text_input_v2::update_state_enter);
214}
215
216void QWaylandTextInputv2::zwp_text_input_v2_leave(uint32_t serial, ::wl_surface *surface)
217{
218 qCDebug(qLcQpaInputMethods()) << Q_FUNC_INFO << serial << surface;
219 m_serial = serial;
220
221 if (m_surface != surface) {
222 qCDebug(qLcQpaInputMethods()) << Q_FUNC_INFO << "Got leave event for surface" << surface << "focused surface" << m_surface;
223 }
224
225 m_surface = nullptr;
226}
227
229{
230 const QList<QByteArray> modifiersMap = QByteArray::fromRawData(static_cast<const char*>(map->data), map->size).split('\0');
231
232 m_modifiersMap.clear();
233
234 for (const QByteArray &modifier : modifiersMap) {
235 if (modifier == "Shift")
236 m_modifiersMap.append(Qt::ShiftModifier);
237 else if (modifier == "Control")
238 m_modifiersMap.append(Qt::ControlModifier);
239 else if (modifier == "Alt")
240 m_modifiersMap.append(Qt::AltModifier);
241 else if (modifier == "Mod1")
242 m_modifiersMap.append(Qt::AltModifier);
243 else if (modifier == "Mod4")
244 m_modifiersMap.append(Qt::MetaModifier);
245 else
246 m_modifiersMap.append(Qt::NoModifier);
247 }
248}
249
250void QWaylandTextInputv2::zwp_text_input_v2_input_panel_state(uint32_t visible, int32_t x, int32_t y, int32_t width, int32_t height)
251{
252 const bool inputPanelVisible = (visible == input_panel_visibility_visible);
253 if (m_inputPanelVisible != inputPanelVisible) {
254 m_inputPanelVisible = inputPanelVisible;
255 QGuiApplicationPrivate::platformIntegration()->inputContext()->emitInputPanelVisibleChanged();
256 }
257 const QRectF keyboardRectangle(x, y, width, height);
258 if (m_keyboardRectangle != keyboardRectangle) {
259 m_keyboardRectangle = keyboardRectangle;
260 QGuiApplicationPrivate::platformIntegration()->inputContext()->emitKeyboardRectChanged();
261 }
262}
263
264void QWaylandTextInputv2::zwp_text_input_v2_preedit_string(const QString &text, const QString &commit)
265{
266 if (m_resetCallback) {
267 qCDebug(qLcQpaInputMethods()) << "discard preedit_string: reset not confirmed";
268 m_builder.reset();
269 return;
270 }
271
272 if (!QGuiApplication::focusObject())
273 return;
274
275 QInputMethodEvent *event = m_builder.buildPreedit(text);
276
277 m_builder.reset();
278 m_preeditCommit = commit;
279
280 QCoreApplication::sendEvent(QGuiApplication::focusObject(), event);
281 delete event;
282}
283
284void QWaylandTextInputv2::zwp_text_input_v2_preedit_styling(uint32_t index, uint32_t length, uint32_t style)
285{
286 m_builder.addPreeditStyling(index, length, style);
287}
288
290{
291 m_builder.setPreeditCursor(index);
292}
293
295{
296 if (m_resetCallback) {
297 qCDebug(qLcQpaInputMethods()) << "discard commit_string: reset not confirmed";
298 m_builder.reset();
299 return;
300 }
301
302 if (!QGuiApplication::focusObject())
303 return;
304
305 QInputMethodEvent *event = m_builder.buildCommit(text);
306
307 m_builder.reset();
308
309 QCoreApplication::sendEvent(QGuiApplication::focusObject(), event);
310 delete event;
311}
312
313void QWaylandTextInputv2::zwp_text_input_v2_cursor_position(int32_t index, int32_t anchor)
314{
315 m_builder.setCursorPosition(index, anchor);
316}
317
318void QWaylandTextInputv2::zwp_text_input_v2_delete_surrounding_text(uint32_t before_length, uint32_t after_length)
319{
320 m_builder.setDeleteSurroundingText(before_length, after_length);
321}
322
323void QWaylandTextInputv2::zwp_text_input_v2_keysym(uint32_t time, uint32_t sym, uint32_t state, uint32_t modifiers)
324{
325#if QT_CONFIG(xkbcommon)
326 if (m_resetCallback) {
327 qCDebug(qLcQpaInputMethods()) << "discard keysym: reset not confirmed";
328 return;
329 }
330
331 if (!QGuiApplication::focusWindow())
332 return;
333
334 Qt::KeyboardModifiers qtModifiers = modifiersToQtModifiers(modifiers);
335
336 QEvent::Type type = state == WL_KEYBOARD_KEY_STATE_PRESSED ? QEvent::KeyPress : QEvent::KeyRelease;
337 QString text = QXkbCommon::lookupStringNoKeysymTransformations(sym);
338 int qtkey = QXkbCommon::keysymToQtKey(sym, qtModifiers);
339
340 QWindowSystemInterface::handleKeyEvent(QGuiApplication::focusWindow(),
341 time, type, qtkey, qtModifiers, text);
342#else
343 Q_UNUSED(time);
344 Q_UNUSED(sym);
345 Q_UNUSED(state);
346 Q_UNUSED(modifiers);
347#endif
348}
349
351{
352 if (m_resetCallback) {
353 qCDebug(qLcQpaInputMethods()) << "discard language: reset not confirmed";
354 return;
355 }
356
357 const QLocale locale(language);
358 if (m_locale != locale) {
359 m_locale = locale;
360 QGuiApplicationPrivate::platformIntegration()->inputContext()->emitLocaleChanged();
361 }
362}
363
365{
366 if (m_resetCallback) {
367 qCDebug(qLcQpaInputMethods()) << "discard text_direction: reset not confirmed";
368 return;
369 }
370
371 const Qt::LayoutDirection inputDirection = (direction == text_direction_auto) ? Qt::LayoutDirectionAuto :
372 (direction == text_direction_ltr) ? Qt::LeftToRight :
373 (direction == text_direction_rtl) ? Qt::RightToLeft : Qt::LayoutDirectionAuto;
374 if (m_inputDirection != inputDirection) {
375 m_inputDirection = inputDirection;
376 QGuiApplicationPrivate::platformIntegration()->inputContext()->emitInputDirectionChanged(m_inputDirection);
377 }
378}
379
381{
382 Q_UNUSED(flags);
383
384 m_serial = serial;
385 updateState(Qt::ImQueryAll, QtWayland::zwp_text_input_v2::update_state_full);
386}
387
388Qt::KeyboardModifiers QWaylandTextInputv2::modifiersToQtModifiers(uint32_t modifiers)
389{
390 Qt::KeyboardModifiers ret = Qt::NoModifier;
391 for (int i = 0; i < m_modifiersMap.size(); ++i) {
392 if (modifiers & (1 << i)) {
393 ret |= m_modifiersMap[i];
394 }
395 }
396 return ret;
397}
398
399}
400
401QT_END_NAMESPACE
void zwp_text_input_v2_keysym(uint32_t time, uint32_t sym, uint32_t state, uint32_t modifiers) override
void zwp_text_input_v2_preedit_styling(uint32_t index, uint32_t length, uint32_t style) override
void updateState(Qt::InputMethodQueries queries, uint32_t flags) override
void enableSurface(::wl_surface *surface) override
void zwp_text_input_v2_delete_surrounding_text(uint32_t before_length, uint32_t after_length) override
void setCursorInsidePreedit(int cursor) override
void disableSurface(::wl_surface *surface) override
void zwp_text_input_v2_preedit_string(const QString &text, const QString &commit) override
void zwp_text_input_v2_input_method_changed(uint32_t serial, uint32_t flags) override
void zwp_text_input_v2_language(const QString &language) override
Qt::LayoutDirection inputDirection() const override
void zwp_text_input_v2_enter(uint32_t serial, struct ::wl_surface *surface) override
void zwp_text_input_v2_text_direction(uint32_t direction) override
void zwp_text_input_v2_modifiers_map(wl_array *map) override
void zwp_text_input_v2_commit_string(const QString &text) override
void zwp_text_input_v2_cursor_position(int32_t index, int32_t anchor) override
void zwp_text_input_v2_input_panel_state(uint32_t state, int32_t x, int32_t y, int32_t width, int32_t height) override
void zwp_text_input_v2_preedit_cursor(int32_t index) override
void zwp_text_input_v2_leave(uint32_t serial, struct ::wl_surface *surface) override
Combined button and popup list for selecting options.