Qt
Internal/Contributor docs for the Qt SDK. <b>Note:</b> These are NOT official API docs; those are found <a href='https://doc.qt.io/'>here</a>.
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
8#include "qwaylandwindow_p.h"
10
11#include <QtCore/qloggingcategory.h>
12#include <QtGui/QGuiApplication>
13#include <QtGui/private/qguiapplication_p.h>
14#include <QtGui/private/qhighdpiscaling_p.h>
15#include <QtGui/qpa/qplatformintegration.h>
16#include <QtGui/qevent.h>
17#include <QtGui/qwindow.h>
18#include <QTextCharFormat>
19#include <QList>
20#include <QRectF>
21#include <QLocale>
22
24
25Q_DECLARE_LOGGING_CATEGORY(qLcQpaInputMethods)
26
27namespace QtWaylandClient {
28
29namespace {
30
31const Qt::InputMethodQueries supportedQueries2 = Qt::ImEnabled |
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 m_builder.reset();
56 m_preeditCommit = QString();
57 updateState(Qt::ImQueryAll, QtWayland::zwp_text_input_v2::update_state_reset);
58}
59
61{
63 if (!m_preeditCommit.isEmpty()) {
64
66 event.setCommitString(m_preeditCommit);
67 m_preeditCommit = QString();
68
70 }
71 }
72
73 reset();
74}
75
76const wl_callback_listener QWaylandTextInputv2::callbackListener = {
77 QWaylandTextInputv2::resetCallback
78};
79
80void QWaylandTextInputv2::resetCallback(void *data, wl_callback *, uint32_t)
81{
82 QWaylandTextInputv2 *self = static_cast<QWaylandTextInputv2*>(data);
83
84 if (self->m_resetCallback) {
85 wl_callback_destroy(self->m_resetCallback);
86 self->m_resetCallback = nullptr;
87 }
88}
89
90void QWaylandTextInputv2::updateState(Qt::InputMethodQueries queries, uint32_t flags)
91{
93 return;
94
96 return;
97
98 auto *window = static_cast<QWaylandWindow *>(QGuiApplication::focusWindow()->handle());
99 auto *surface = window->wlSurface();
100 if (!surface || (surface != m_surface))
101 return;
102
103 queries &= supportedQueries2;
104
105 // Surrounding text, cursor and anchor positions are transferred together
106 if ((queries & Qt::ImSurroundingText) || (queries & Qt::ImCursorPosition) || (queries & Qt::ImAnchorPosition))
108
111
112 if ((queries & Qt::ImSurroundingText) || (queries & Qt::ImCursorPosition) || (queries & Qt::ImAnchorPosition)) {
113 QString text = event.value(Qt::ImSurroundingText).toString();
114 int cursor = event.value(Qt::ImCursorPosition).toInt();
115 int anchor = event.value(Qt::ImAnchorPosition).toInt();
116
117 // Make sure text is not too big
118 if (text.toUtf8().size() > 2048) {
119 int c = qAbs(cursor - anchor) <= 512 ? qMin(cursor, anchor) + qAbs(cursor - anchor) / 2: cursor;
120
121 const int offset = c - qBound(0, c, 512 - qMin(text.size() - c, 256));
122 text = text.mid(offset + c - 256, 512);
123 cursor -= offset;
124 anchor -= offset;
125 }
126
128 }
129
130 if (queries & Qt::ImHints) {
131 QWaylandInputMethodContentType contentType = QWaylandInputMethodContentType::convert(static_cast<Qt::InputMethodHints>(event.value(Qt::ImHints).toInt()));
132 set_content_type(contentType.hint, contentType.purpose);
133 }
134
135 if (queries & Qt::ImCursorRectangle) {
136 const QRect &cRect = event.value(Qt::ImCursorRectangle).toRect();
137 const QRect &windowRect = QGuiApplication::inputMethod()->inputItemTransform().mapRect(cRect);
138 const QRect &nativeRect = QHighDpi::toNativePixels(windowRect, QGuiApplication::focusWindow());
139 const QMargins margins = window->clientSideMargins();
140 const QRect &surfaceRect = nativeRect.translated(margins.left(), margins.top());
141 set_cursor_rectangle(surfaceRect.x(), surfaceRect.y(), surfaceRect.width(), surfaceRect.height());
142 }
143
144 if (queries & Qt::ImPreferredLanguage) {
145 const QString &language = event.value(Qt::ImPreferredLanguage).toString();
146 set_preferred_language(language);
147 }
148
149 update_state(m_serial, flags);
150 if (flags != QtWayland::zwp_text_input_v2::update_state_change) {
151 if (m_resetCallback)
152 wl_callback_destroy(m_resetCallback);
153 m_resetCallback = wl_display_sync(m_display->wl_display());
154 wl_callback_add_listener(m_resetCallback, &QWaylandTextInputv2::callbackListener, this);
155 }
156}
157
159{
160 // Not supported yet
161}
162
164{
165 return m_inputPanelVisible;
166}
167
169{
170 return m_keyboardRectangle;
171}
172
174{
175 return m_locale;
176}
177
179{
180 return m_inputDirection;
181}
182
183void QWaylandTextInputv2::zwp_text_input_v2_enter(uint32_t serial, ::wl_surface *surface)
184{
185 m_serial = serial;
186 m_surface = surface;
187
188 updateState(Qt::ImQueryAll, QtWayland::zwp_text_input_v2::update_state_enter);
189}
190
191void QWaylandTextInputv2::zwp_text_input_v2_leave(uint32_t serial, ::wl_surface *surface)
192{
193 m_serial = serial;
194
195 if (m_surface != surface) {
196 qCDebug(qLcQpaInputMethods()) << Q_FUNC_INFO << "Got leave event for surface" << surface << "focused surface" << m_surface;
197 }
198
199 m_surface = nullptr;
200}
201
203{
204 const QList<QByteArray> modifiersMap = QByteArray::fromRawData(static_cast<const char*>(map->data), map->size).split('\0');
205
206 m_modifiersMap.clear();
207
208 for (const QByteArray &modifier : modifiersMap) {
209 if (modifier == "Shift")
210 m_modifiersMap.append(Qt::ShiftModifier);
211 else if (modifier == "Control")
212 m_modifiersMap.append(Qt::ControlModifier);
213 else if (modifier == "Alt")
214 m_modifiersMap.append(Qt::AltModifier);
215 else if (modifier == "Mod1")
216 m_modifiersMap.append(Qt::AltModifier);
217 else if (modifier == "Mod4")
218 m_modifiersMap.append(Qt::MetaModifier);
219 else
220 m_modifiersMap.append(Qt::NoModifier);
221 }
222}
223
224void QWaylandTextInputv2::zwp_text_input_v2_input_panel_state(uint32_t visible, int32_t x, int32_t y, int32_t width, int32_t height)
225{
226 const bool inputPanelVisible = (visible == input_panel_visibility_visible);
227 if (m_inputPanelVisible != inputPanelVisible) {
228 m_inputPanelVisible = inputPanelVisible;
229 QGuiApplicationPrivate::platformIntegration()->inputContext()->emitInputPanelVisibleChanged();
230 }
231 const QRectF keyboardRectangle(x, y, width, height);
232 if (m_keyboardRectangle != keyboardRectangle) {
233 m_keyboardRectangle = keyboardRectangle;
234 QGuiApplicationPrivate::platformIntegration()->inputContext()->emitKeyboardRectChanged();
235 }
236}
237
239{
240 if (m_resetCallback) {
241 qCDebug(qLcQpaInputMethods()) << "discard preedit_string: reset not confirmed";
242 m_builder.reset();
243 return;
244 }
245
247 return;
248
249 QInputMethodEvent *event = m_builder.buildPreedit(text);
250
251 m_builder.reset();
252 m_preeditCommit = commit;
253
255 delete event;
256}
257
259{
260 m_builder.addPreeditStyling(index, length, style);
261}
262
267
269{
270 if (m_resetCallback) {
271 qCDebug(qLcQpaInputMethods()) << "discard commit_string: reset not confirmed";
272 m_builder.reset();
273 return;
274 }
275
277 return;
278
279 QInputMethodEvent *event = m_builder.buildCommit(text);
280
281 m_builder.reset();
282
284 delete event;
285}
286
288{
289 m_builder.setCursorPosition(index, anchor);
290}
291
292void QWaylandTextInputv2::zwp_text_input_v2_delete_surrounding_text(uint32_t before_length, uint32_t after_length)
293{
294 m_builder.setDeleteSurroundingText(before_length, after_length);
295}
296
297void QWaylandTextInputv2::zwp_text_input_v2_keysym(uint32_t time, uint32_t sym, uint32_t state, uint32_t modifiers)
298{
299#if QT_CONFIG(xkbcommon)
300 if (m_resetCallback) {
301 qCDebug(qLcQpaInputMethods()) << "discard keysym: reset not confirmed";
302 return;
303 }
304
306 return;
307
308 Qt::KeyboardModifiers qtModifiers = modifiersToQtModifiers(modifiers);
309
310 QEvent::Type type = state == WL_KEYBOARD_KEY_STATE_PRESSED ? QEvent::KeyPress : QEvent::KeyRelease;
312 int qtkey = QXkbCommon::keysymToQtKey(sym, qtModifiers);
313
315 time, type, qtkey, qtModifiers, text);
316#else
317 Q_UNUSED(time);
318 Q_UNUSED(sym);
321#endif
322}
323
325{
326 if (m_resetCallback) {
327 qCDebug(qLcQpaInputMethods()) << "discard language: reset not confirmed";
328 return;
329 }
330
331 const QLocale locale(language);
332 if (m_locale != locale) {
333 m_locale = locale;
334 QGuiApplicationPrivate::platformIntegration()->inputContext()->emitLocaleChanged();
335 }
336}
337
339{
340 if (m_resetCallback) {
341 qCDebug(qLcQpaInputMethods()) << "discard text_direction: reset not confirmed";
342 return;
343 }
344
345 const Qt::LayoutDirection inputDirection = (direction == text_direction_auto) ? Qt::LayoutDirectionAuto :
346 (direction == text_direction_ltr) ? Qt::LeftToRight :
347 (direction == text_direction_rtl) ? Qt::RightToLeft : Qt::LayoutDirectionAuto;
348 if (m_inputDirection != inputDirection) {
349 m_inputDirection = inputDirection;
350 QGuiApplicationPrivate::platformIntegration()->inputContext()->emitInputDirectionChanged(m_inputDirection);
351 }
352}
353
355{
357
358 m_serial = serial;
359 updateState(Qt::ImQueryAll, QtWayland::zwp_text_input_v2::update_state_full);
360}
361
362Qt::KeyboardModifiers QWaylandTextInputv2::modifiersToQtModifiers(uint32_t modifiers)
363{
364 Qt::KeyboardModifiers ret = Qt::NoModifier;
365 for (int i = 0; i < m_modifiersMap.size(); ++i) {
366 if (modifiers & (1 << i)) {
367 ret |= m_modifiersMap[i];
368 }
369 }
370 return ret;
371}
372
373}
374
\inmodule QtCore
Definition qbytearray.h:57
qsizetype size() const noexcept
Returns the number of bytes in this byte array.
Definition qbytearray.h:494
static QByteArray fromRawData(const char *data, qsizetype size)
Constructs a QByteArray that uses the first size bytes of the data array.
Definition qbytearray.h:409
static bool sendEvent(QObject *receiver, QEvent *event)
Sends event event directly to receiver receiver, using the notify() function.
Type
This enum type defines the valid event types in Qt.
Definition qcoreevent.h:51
@ KeyRelease
Definition qcoreevent.h:65
@ KeyPress
Definition qcoreevent.h:64
static QPlatformIntegration * platformIntegration()
static QObject * focusObject()
Returns the QObject in currently active window that will be final receiver of events tied to focus,...
static QWindow * focusWindow()
Returns the QWindow that receives events tied to focus, such as key events.
static QInputMethod * inputMethod()
returns the input method.
The QInputMethodEvent class provides parameters for input method events.
Definition qevent.h:625
The QInputMethodQueryEvent class provides an event sent by the input context to input objects.
Definition qevent.h:679
qsizetype size() const noexcept
Definition qlist.h:397
void append(parameter_type t)
Definition qlist.h:458
void clear()
Definition qlist.h:434
size_type size() const
Definition qmap.h:267
\inmodule QtCore
Definition qmargins.h:24
constexpr int left() const noexcept
Returns the left margin.
Definition qmargins.h:106
constexpr int top() const noexcept
Returns the top margin.
Definition qmargins.h:109
\inmodule QtCore
Definition qobject.h:103
\inmodule QtCore\reentrant
Definition qrect.h:484
\inmodule QtCore\reentrant
Definition qrect.h:30
constexpr QRect translated(int dx, int dy) const noexcept
Returns a copy of the rectangle that is translated dx along the x axis and dy along the y axis,...
Definition qrect.h:261
\macro QT_RESTRICTED_CAST_FROM_ASCII
Definition qstring.h:129
QString mid(qsizetype position, qsizetype n=-1) const &
Definition qstring.cpp:5300
bool isEmpty() const noexcept
Returns true if the string has no characters; otherwise returns false.
Definition qstring.h:192
qsizetype size() const noexcept
Returns the number of characters in this string.
Definition qstring.h:186
QByteArray toUtf8() const &
Definition qstring.h:634
void addPreeditStyling(uint32_t index, uint32_t length, uint32_t style)
QInputMethodEvent * buildCommit(const QString &text)
QInputMethodEvent * buildPreedit(const QString &text)
void setDeleteSurroundingText(uint32_t beforeLength, uint32_t afterLength)
void setCursorPosition(int32_t index, int32_t anchor)
static int indexToWayland(const QString &text, int length, int base=0)
static bool handleKeyEvent(QWindow *window, QEvent::Type t, int k, Qt::KeyboardModifiers mods, const QString &text=QString(), bool autorep=false, ushort count=1)
static QString lookupStringNoKeysymTransformations(xkb_keysym_t keysym)
static int keysymToQtKey(xkb_keysym_t keysym, Qt::KeyboardModifiers modifiers)
struct wl_display * wl_display() const
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 zwp_text_input_v2_delete_surrounding_text(uint32_t before_length, uint32_t after_length) override
void setCursorInsidePreedit(int cursor) 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
QWaylandTextInputv2(QWaylandDisplay *display, struct ::zwp_text_input_v2 *text_input)
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
EGLImageKHR int int EGLuint64KHR * modifiers
QMap< QString, QString > map
[6]
QString text
QCursor cursor
direction
else opt state
[0]
struct wl_display * display
Definition linuxdmabuf.h:41
T toNativePixels(const T &value, const C *context)
Combined button and popup list for selecting options.
@ ImSurroundingText
@ ImCursorPosition
@ ImPreferredLanguage
@ ImAnchorPosition
@ ImCursorRectangle
@ ImHints
@ ImEnabled
@ ImQueryAll
LayoutDirection
@ LeftToRight
@ LayoutDirectionAuto
@ RightToLeft
@ ShiftModifier
@ ControlModifier
@ MetaModifier
@ NoModifier
@ AltModifier
#define Q_FUNC_INFO
#define qCDebug(category,...)
#define Q_DECLARE_LOGGING_CATEGORY(name)
return ret
constexpr const T & qMin(const T &a, const T &b)
Definition qminmax.h:40
constexpr const T & qBound(const T &min, const T &val, const T &max)
Definition qminmax.h:44
constexpr T qAbs(const T &t)
Definition qnumeric.h:328
GLuint64 GLenum void * handle
GLint GLint GLint GLint GLint x
[0]
GLint GLsizei GLsizei height
GLuint index
[2]
GLenum GLuint GLenum GLsizei length
GLint GLint GLint GLint GLsizei GLsizei GLsizei GLboolean commit
GLint GLsizei GLsizei GLenum GLenum GLsizei void * data
GLint GLsizei width
GLenum type
GLbitfield flags
GLenum GLuint GLintptr offset
GLint y
struct _cl_event * event
const GLubyte * c
#define Q_UNUSED(x)
aWidget window() -> setWindowTitle("New Window Title")
[2]
static QWaylandInputMethodContentType convert(Qt::InputMethodHints hints)