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