QSkinny 0.8.0
C++/Qt UI toolkit
Loading...
Searching...
No Matches
QskInputPanel.cpp
1/******************************************************************************
2 * QSkinny - Copyright (C) The authors
3 * SPDX-License-Identifier: BSD-3-Clause
4 *****************************************************************************/
5
6#include "QskInputPanel.h"
7#include "QskInputContext.h"
8#include "QskTextPredictor.h"
9
10#include <qpointer.h>
11#include <qtextformat.h>
12
13namespace
14{
15 struct Result
16 {
17 int key = 0;
18
19 QString text;
20 bool isFinal = true;
21 };
22}
23
24static void qskRegisterInputPanel()
25{
26 qRegisterMetaType< Result >( "Result" );
27}
28
29Q_CONSTRUCTOR_FUNCTION( qskRegisterInputPanel )
30
31static inline QQuickItem* qskReceiverItem( const QskInputPanel* panel )
32{
33 if ( auto item = panel->inputProxy() )
34 return item;
35
36 return panel->inputItem();
37}
38
39static inline bool qskUsePrediction( Qt::InputMethodHints hints )
40{
41 constexpr Qt::InputMethodHints mask =
42 Qt::ImhNoPredictiveText | Qt::ImhExclusiveInputMask | Qt::ImhHiddenText;
43
44 return ( hints & mask ) == 0;
45}
46
47static inline void qskSendText(
48 QQuickItem* receiver, const QString& text, bool isFinal )
49{
50 if ( receiver == nullptr )
51 return;
52
53 if ( isFinal )
54 {
55 QInputMethodEvent event;
56
57 /*
58 QQuickTextInput is buggy when receiving empty commit strings.
59 We need to send a wrong replaceLength to work around it.
60 See QTBUG: 68874
61 */
62 if ( text.isEmpty() )
63 event.setCommitString( text, 0, 1 );
64 else
65 event.setCommitString( text );
66
67 QCoreApplication::sendEvent( receiver, &event );
68 }
69 else
70 {
71 QTextCharFormat format;
72 format.setFontUnderline( true );
73
74 const QInputMethodEvent::Attribute attribute(
75 QInputMethodEvent::TextFormat, 0, text.length(), format );
76
77 QInputMethodEvent event( text, { attribute } );
78
79 QCoreApplication::sendEvent( receiver, &event );
80 }
81}
82
83static inline void qskSendReplaceText( QQuickItem* receiver, const QString& text )
84{
85 if ( receiver == nullptr )
86 return;
87
88 QInputMethodEvent::Attribute attribute(
89 QInputMethodEvent::Selection, 0, 32767, QVariant() );
90
91 QInputMethodEvent event( QString(), { attribute } );
92 QCoreApplication::sendEvent( receiver, &event );
93
94 qskSendText( receiver, text, true );
95}
96
97static inline void qskSendKey( QQuickItem* receiver, int key )
98{
99 if ( receiver == nullptr )
100 return;
101
102 QKeyEvent keyPress( QEvent::KeyPress, key, Qt::NoModifier );
103 QCoreApplication::sendEvent( receiver, &keyPress );
104
105 QKeyEvent keyRelease( QEvent::KeyRelease, key, Qt::NoModifier );
106 QCoreApplication::sendEvent( receiver, &keyRelease );
107}
108
109namespace
110{
111 class KeyProcessor : public QObject
112 {
113 Q_OBJECT
114
115 public:
116 QString preedit() const
117 {
118 return m_preedit;
119 }
120
121 void processKey(
122 int key, Qt::InputMethodHints inputHints, QskInputPanel* panel,
123 QskTextPredictor* predictor, int spaceLeft )
124 {
125 // reset:
126 m_currentResult.isFinal = true;
127 m_currentResult.text.clear();
128 m_currentResult.key = 0;
129
130 m_predictor = predictor;
131 m_spaceLeft = spaceLeft;
132
133 // First we have to handle the control keys
134
135 switch ( key )
136 {
137 case Qt::Key_Backspace:
138 case Qt::Key_Muhenkan:
139 {
140 if ( predictor && !m_preedit.isEmpty() )
141 {
142 m_preedit.chop( 1 );
143
144 m_currentResult.text = m_preedit;
145 m_currentResult.isFinal = false;
146
147 Q_EMIT panel->predictionRequested( m_preedit );
148 // Let the input field update right away, otherwise
149 // we'll get weird effects with fast backspace presses:
150 Q_EMIT keyProcessingFinished( m_currentResult );
151 }
152 else
153 {
154 m_currentResult.key = Qt::Key_Backspace;
155 Q_EMIT keyProcessingFinished( m_currentResult );
156 }
157
158 return;
159 }
160 case Qt::Key_Return:
161 {
162 if ( predictor )
163 {
164 if ( !m_preedit.isEmpty() )
165 {
166 if ( spaceLeft )
167 {
168 m_currentResult.text = m_preedit.left( spaceLeft );
169 m_currentResult.isFinal = true;
170 }
171
172 reset();
173 Q_EMIT keyProcessingFinished( m_currentResult );
174 return;
175 }
176 }
177
178 if ( !( inputHints & Qt::ImhMultiLine ) )
179 {
180 m_currentResult.key = Qt::Key_Return;
181 Q_EMIT keyProcessingFinished( m_currentResult );
182 return;
183 }
184
185 break;
186 }
187 case Qt::Key_Space:
188 {
189 if ( predictor )
190 {
191 if ( !m_preedit.isEmpty() && spaceLeft )
192 {
193 m_preedit += keyString( key );
194 m_preedit = m_preedit.left( spaceLeft );
195
196 m_currentResult.text = m_preedit;
197 m_currentResult.isFinal = true;
198
199 reset();
200
201 Q_EMIT keyProcessingFinished( m_currentResult );
202 return;
203 }
204 }
205
206 break;
207 }
208 case Qt::Key_Left:
209 case Qt::Key_Right:
210 case Qt::Key_Escape:
211 case Qt::Key_Cancel:
212 {
213 m_currentResult.key = key;
214 Q_EMIT keyProcessingFinished( m_currentResult );
215 return;
216 }
217 }
218
219 const QString text = keyString( key );
220
221 if ( predictor )
222 {
223 m_preedit += text;
224 Q_EMIT panel->predictionRequested( m_preedit );
225 }
226 else
227 {
228 m_currentResult.text = text;
229 m_currentResult.isFinal = true;
230 Q_EMIT keyProcessingFinished( m_currentResult );
231 }
232 }
233
234 void reset()
235 {
236 m_preedit.clear();
237 }
238
239 void continueProcessingKey( const QStringList& candidates )
240 {
241 if ( m_predictor )
242 {
243 if ( candidates.count() > 0 )
244 {
245 m_currentResult.text = m_preedit;
246 m_currentResult.isFinal = false;
247 }
248 else
249 {
250 m_currentResult.text = m_preedit.left( m_spaceLeft );
251 m_currentResult.isFinal = true;
252
253 m_preedit.clear();
254 }
255 }
256
257 Q_EMIT keyProcessingFinished( m_currentResult );
258 }
259
260 Q_SIGNALS:
261 void keyProcessingFinished( const Result& );
262
263 private:
264 inline QString keyString( int keyCode ) const
265 {
266 // Special case entry codes here, else default to the symbol
267 switch ( keyCode )
268 {
269 case Qt::Key_Shift:
270 case Qt::Key_CapsLock:
271 case Qt::Key_Mode_switch:
272 case Qt::Key_Backspace:
273 case Qt::Key_Muhenkan:
274 return QString();
275
276 case Qt::Key_Return:
277 case Qt::Key_Kanji:
278 return QChar( QChar::CarriageReturn );
279
280 case Qt::Key_Space:
281 return QChar( QChar::Space );
282
283 default:
284 break;
285 }
286
287 return QChar( keyCode );
288 }
289
290 QString m_preedit;
291 int m_spaceLeft = -1;
292 QskTextPredictor* m_predictor = nullptr;
293 Result m_currentResult;
294 };
295}
296
297class QskInputPanel::PrivateData
298{
299 public:
300 PrivateData( QskInputPanel* panel )
301 : panel( panel )
302 {
303 }
304
305 KeyProcessor keyProcessor;
306 QPointer< QQuickItem > inputItem;
307
308 QLocale predictorLocale;
309 std::shared_ptr< QskTextPredictor > predictor;
310 QStringList candidates;
311
312 Qt::InputMethodHints inputHints;
313 bool hasPredictorLocale = false;
314 QskInputPanel* panel;
315
316 void handleKeyProcessingFinished( const Result& result )
317 {
318 switch ( result.key )
319 {
320 case 0:
321 {
322 qskSendText( qskReceiverItem( panel ),
323 result.text, result.isFinal );
324 break;
325 }
326 case Qt::Key_Return:
327 {
328 panel->commitCurrentText( false );
329 qskSendKey( inputItem, result.key );
330 break;
331 }
332 case Qt::Key_Escape:
333 case Qt::Key_Cancel:
334 {
335 qskSendKey( inputItem, result.key );
336 break;
337 }
338 default:
339 {
340 qskSendKey( qskReceiverItem( panel ), result.key );
341 }
342 }
343 }
344};
345
346QskInputPanel::QskInputPanel( QQuickItem* parent )
347 : Inherited( parent )
348 , m_data( new PrivateData( this ) )
349{
350 setAutoLayoutChildren( true );
351 setLayoutAlignmentHint( Qt::AlignHCenter | Qt::AlignBottom );
352
353 initSizePolicy( QskSizePolicy::Ignored, QskSizePolicy::Constrained );
354
355 connect( this, &QskInputPanel::keySelected,
356 this, &QskInputPanel::commitKey );
357
358 connect( this, &QskInputPanel::predictiveTextSelected,
359 this, &QskInputPanel::commitPredictiveText );
360
361 connect( this, &QskControl::localeChanged,
362 this, &QskInputPanel::updateLocale );
363
364 connect( &m_data->keyProcessor, &KeyProcessor::keyProcessingFinished,
365 this, [this]( const Result& result ) { m_data->handleKeyProcessingFinished( result ); } );
366
367 updateLocale( locale() );
368}
369
370QskInputPanel::~QskInputPanel()
371{
372}
373
374void QskInputPanel::attachInputItem( QQuickItem* item )
375{
376 if ( item == m_data->inputItem )
377 return;
378
379 if ( m_data->inputItem )
380 {
381 disconnect( m_data->inputItem, &QObject::destroyed,
382 this, &QskInputPanel::inputItemDestroyed );
383 }
384
385 m_data->inputItem = item;
386
387 if ( item )
388 {
389 if ( m_data->predictor )
390 Q_EMIT predictionReset();
391
392 m_data->keyProcessor.reset();
393 m_data->inputHints = Qt::InputMethodHints();
394
395 attachItem( item );
396
397 Qt::InputMethodQueries queries = Qt::ImQueryAll;
398 queries &= ~Qt::ImEnabled;
399
400 updateInputPanel( queries );
401
402 if ( inputProxy() )
403 {
404 /*
405 Hiding the cursor in item. We use postEvent
406 so that everything on the item is done,
407 when receiving the event.
408 */
409 const QInputMethodEvent::Attribute attribute(
410 QInputMethodEvent::Cursor, 0, 0, QVariant() );
411
412 QCoreApplication::postEvent( item,
413 new QInputMethodEvent( QString(), { attribute } ) );
414 }
415
416 connect( item, &QObject::destroyed,
417 this, &QskInputPanel::inputItemDestroyed,
418 Qt::UniqueConnection );
419 }
420 else
421 {
422 attachItem( nullptr );
423 }
424}
425
426void QskInputPanel::updateInputPanel( Qt::InputMethodQueries queries )
427{
428 if ( m_data->inputItem == nullptr )
429 return;
430
431 QInputMethodQueryEvent event( queries );
432 QCoreApplication::sendEvent( m_data->inputItem, &event );
433
434 if ( queries & Qt::ImHints )
435 {
436 m_data->inputHints = static_cast< Qt::InputMethodHints >(
437 event.value( Qt::ImHints ).toInt() );
438
439 setPredictionEnabled(
440 m_data->predictor && qskUsePrediction( m_data->inputHints ) );
441 }
442
443 if ( queries & Qt::ImPreferredLanguage )
444 {
445 setLocale( event.value( Qt::ImPreferredLanguage ).toLocale() );
446 }
447
448 if ( queries & Qt::ImInputItemClipRectangle )
449 {
450 /*
451 Could be used to move the panel,
452 so that it does not hide the item.
453 */
454 }
455}
456
457void QskInputPanel::updateLocale( const QLocale& locale )
458{
459 if ( !m_data->hasPredictorLocale || locale != m_data->predictorLocale )
460 {
461 m_data->hasPredictorLocale = true;
462 m_data->predictorLocale = locale;
463
464 resetPredictor( locale );
465 m_data->keyProcessor.reset();
466 }
467}
468
469void QskInputPanel::resetPredictor( const QLocale& locale )
470{
471 auto predictor = QskInputContext::instance()->textPredictor( locale );
472
473 if ( predictor == m_data->predictor )
474 return;
475
476 m_data->predictor = predictor;
477
478 if ( predictor )
479 {
480 // text predictor lives in another thread, so these will all be QueuedConnections:
481 connect( this, &QskInputPanel::predictionReset,
482 predictor.get(), &QskTextPredictor::reset );
483 connect( this, &QskInputPanel::predictionRequested,
484 predictor.get(), &QskTextPredictor::request );
485
486 connect( predictor.get(), &QskTextPredictor::predictionChanged,
487 this, &QskInputPanel::updatePrediction );
488 }
489
490 setPredictionEnabled(
491 predictor && qskUsePrediction( m_data->inputHints ) );
492}
493
494void QskInputPanel::commitPredictiveText( int index )
495{
496 QString text;
497
498 if ( m_data->predictor )
499 {
500 text = m_data->candidates.at( index );
501 Q_EMIT predictionReset();
502 }
503
504 m_data->keyProcessor.reset();
505
506 setPrediction( {} );
507
508 qskSendText( qskReceiverItem( this ), text, true );
509}
510
511void QskInputPanel::updatePrediction( const QString& text, const QStringList& candidates )
512{
513 if ( m_data->predictor )
514 {
515 if( m_data->keyProcessor.preedit() != text )
516 {
517 // This must be for another input panel
518 return;
519 }
520
521 setPrediction( candidates );
522 m_data->keyProcessor.continueProcessingKey( candidates );
523 }
524 else
525 {
526 qWarning() << "got prediction update, but no predictor. Something is wrong";
527 }
528}
529
530QQuickItem* QskInputPanel::inputProxy() const
531{
532 return nullptr;
533}
534
535QQuickItem* QskInputPanel::inputItem() const
536{
537 return m_data->inputItem;
538}
539
540void QskInputPanel::setPrompt( const QString& )
541{
542}
543
544void QskInputPanel::setPredictionEnabled( bool )
545{
546}
547
548void QskInputPanel::setPrediction(const QStringList& candidates )
549{
550 m_data->candidates = candidates;
551}
552
553Qt::Alignment QskInputPanel::alignment() const
554{
555 /*
556 When we have an input proxy, we don't care if
557 the input item becomes hidden
558 */
559
560 return inputProxy() ? Qt::AlignVCenter : Qt::AlignBottom;
561}
562
563QStringList QskInputPanel::candidates() const
564{
565 return m_data->candidates;
566}
567
568void QskInputPanel::commitKey( int key )
569{
570 if ( m_data->inputItem == nullptr )
571 return;
572
573 int spaceLeft = -1;
574
575 if ( !( m_data->inputHints & Qt::ImhMultiLine ) )
576 {
577 QInputMethodQueryEvent event1( Qt::ImMaximumTextLength );
578 QCoreApplication::sendEvent( m_data->inputItem, &event1 );
579
580 const int maxChars = event1.value( Qt::ImMaximumTextLength ).toInt();
581 if ( maxChars >= 0 )
582 {
583 QInputMethodQueryEvent event2( Qt::ImSurroundingText );
584 QCoreApplication::sendEvent( qskReceiverItem( this ), &event2 );
585
586 const auto text = event2.value( Qt::ImSurroundingText ).toString();
587 spaceLeft = maxChars - text.length();
588 }
589 }
590
591 QskTextPredictor* predictor = nullptr;
592 if ( qskUsePrediction( m_data->inputHints ) )
593 predictor = m_data->predictor.get(); // ### we could also make the predictor member of keyProcessor a shared ptr?
594
595 m_data->keyProcessor.processKey(
596 key, m_data->inputHints, this, predictor, spaceLeft );
597}
598
599void QskInputPanel::commitCurrentText( bool isFinal )
600{
601 if ( auto proxy = inputProxy() )
602 {
603 // using Qt::ImSurroundingText instead ??
604 const auto value = proxy->property( "text" );
605 if ( value.canConvert< QString >() )
606 qskSendReplaceText( m_data->inputItem, value.toString() );
607 }
608
609 if ( isFinal )
610 commitKey( Qt::Key_Escape );
611}
612
613#include "moc_QskInputPanel.cpp"
614#include "QskInputPanel.moc"
void setLocale(const QLocale &)
void localeChanged(const QLocale &)
QLocale locale
Definition QskControl.h:27