QSkinny 0.8.0
C++/Qt UI toolkit
Loading...
Searching...
No Matches
QskInputContext.cpp
1/******************************************************************************
2 * QSkinny - Copyright (C) The authors
3 * SPDX-License-Identifier: BSD-3-Clause
4 *****************************************************************************/
5
6#include "QskInputContext.h"
7#include "QskInputPanel.h"
8#include "QskInputPanelBox.h"
9
10#include "QskDialog.h"
11#include "QskLinearBox.h"
12#include "QskPopup.h"
13#include "QskQuick.h"
14#include "QskTextPredictor.h"
15#include "QskWindow.h"
16#include "QskPlatform.h"
17#include "QskInputGrabber.h"
18
19#include <qguiapplication.h>
20#include <qmap.h>
21#include <qpointer.h>
22#include <qthread.h>
23
24#include <qpa/qplatforminputcontext.h>
25#include <qpa/qplatformintegration.h>
26
27#if HUNSPELL
28#include "QskHunspellTextPredictor.h"
29#endif
30
31namespace
32{
33 class Popup : public QskPopup
34 {
35 Q_OBJECT
36
37 using Inherited = QskPopup;
38
39 public:
40 Popup()
41 {
42 setPolishOnResize( true );
43 setPolishOnParentResize( true );
44 }
45
46 Q_SIGNALS:
47 void commitRequested();
48
49 protected:
50 void aboutToShow() override
51 {
53
54 if ( popupFlags() & QskPopup::CloseOnPressOutside )
55 {
56 if ( auto inputGrabber = findChild< QskInputGrabber* >() )
57 inputGrabber->installEventFilter( this );
58 }
59 }
60
61 bool eventFilter( QObject* object, QEvent* event ) override
62 {
63 if ( qobject_cast< QskInputGrabber* >( object ) )
64 {
65 if ( event->type() == QEvent::MouseButtonPress )
66 {
67 if ( popupFlags() & QskPopup::CloseOnPressOutside )
68 {
69 Q_EMIT commitRequested();
70 return true;
71 }
72 }
73 }
74
75 return Inherited::eventFilter( object, event );
76 }
77
78 void updateLayout() override
79 {
80 const auto m = margins();
81 const auto item = findChild< const QskInputPanel* >();
82
83 auto r = qskItemGeometry( parentItem() );
84 r -= m;
85 r = qskConstrainedItemRect( item, r );
86 r += m;
87
88 setGeometry( r );
89 }
90 };
91
92 class Panel final : public QskInputPanel
93 {
94 public:
95 Panel( QQuickItem* parent = nullptr )
96 : QskInputPanel( parent )
97 {
98 m_box = new QskInputPanelBox( this );
99
100 connect( m_box, &QskInputPanelBox::keySelected,
101 this, &QskInputPanel::keySelected );
102
103 connect( m_box, &QskInputPanelBox::predictiveTextSelected,
104 this, &QskInputPanel::predictiveTextSelected );
105 }
106
107 void attachItem( QQuickItem* item ) override
108 {
109 m_box->attachInputItem( item );
110 }
111
112 QQuickItem* inputProxy() const override
113 {
114 return m_box->inputProxy();
115 }
116
117 void setPrompt( const QString& prompt ) override
118 {
119 m_box->setInputPrompt( prompt );
120 }
121
122 void setPredictionEnabled( bool on ) override
123 {
124 m_box->setPanelHint( QskInputPanelBox::Prediction, on );
125 }
126
127 void setPrediction( const QStringList& prediction ) override
128 {
129 QskInputPanel::setPrediction( prediction );
130 m_box->setPrediction( prediction );
131 }
132
133 private:
134 QskInputPanelBox* m_box;
135 };
136
137 class Channel
138 {
139 public:
140 // item receiving the input
141 QPointer< QQuickItem > item;
142
143 // panel for inserting the input
144 QPointer< QskInputPanel > panel;
145
146 // popup or window embedding the panel
147 QPointer< Popup > popup;
148 QPointer< QskWindow > window;
149 };
150
151 class ChannelTable
152 {
153 public:
154 inline Channel* currentChannel() const
155 {
156 const auto object = QGuiApplication::focusObject();
157 return channel( qobject_cast< const QQuickItem* >( object ) );
158 }
159
160 inline Channel* channel( const QQuickWindow* window ) const
161 {
162 if ( window )
163 {
164 auto it = m_map.constFind( window );
165 if ( it != m_map.constEnd() )
166 return const_cast< Channel* >( &it.value() );
167 }
168
169 return nullptr;
170 }
171
172 inline Channel* channel( const QskInputPanel* panel ) const
173 {
174 if ( panel )
175 {
176 QMap< const QQuickWindow*, Channel >::const_iterator it;
177
178 for( it = m_map.constBegin(); it != m_map.constEnd(); ++it )
179 {
180 if( it.value().panel == panel )
181 return const_cast< Channel* >( &it.value() );
182 }
183 }
184
185 return nullptr;
186 }
187
188 inline Channel* channel( const QQuickItem* item ) const
189 {
190 if ( item )
191 {
192 // item->window() might already been gone
193 QMap< const QQuickWindow*, Channel >::const_iterator it;
194
195 for( it = m_map.constBegin(); it != m_map.constEnd(); ++it )
196 {
197 if( it.value().item == item )
198 return const_cast< Channel* >( &it.value() );
199 }
200 }
201
202 return nullptr;
203 }
204
205 inline Channel* ancestorChannel( const QQuickItem* item ) const
206 {
207 for ( auto it = m_map.constBegin();
208 it != m_map.constEnd(); ++it )
209 {
210 if ( const auto panel = it.value().panel )
211 {
212 if ( ( item == panel ) || qskIsAncestorOf( panel, item ) )
213 {
214 return const_cast< Channel* >( &it.value() );
215 }
216 }
217 }
218
219 return nullptr;
220 }
221
222 inline Channel* insert( const QQuickWindow* window )
223 {
224 return &m_map[ window ];
225 }
226
227 inline void remove( const QQuickWindow* window )
228 {
229 m_map.remove( window );
230 }
231
232 private:
233 QMap< const QQuickWindow*, Channel > m_map;
234 };
235}
236
237static QPointer< QskInputContext > qskInputContext;
238
239static void qskSendToPlatformContext( QEvent::Type type )
240{
241 const auto integration = qskPlatformIntegration();
242
243 if ( const auto inputContext = integration->inputContext() )
244 {
245 QEvent event( type );
246 QCoreApplication::sendEvent( inputContext, &event );
247 }
248}
249
250static void qskInputContextHook()
251{
252 qAddPostRoutine( [] { delete qskInputContext; } );
253}
254
255Q_COREAPP_STARTUP_FUNCTION( qskInputContextHook )
256
257#if QT_VERSION >= QT_VERSION_CHECK( 6, 2, 0 )
258 static inline QLocale::Territory qskTerritory( const QLocale& locale )
259 { return locale.territory(); }
260#else
261 static inline QLocale::Country qskTerritory( const QLocale& locale )
262 { return locale.country(); }
263#endif
264
265void QskInputContext::setInstance( QskInputContext* inputContext )
266{
267 if ( inputContext != qskInputContext )
268 {
269 const auto oldContext = qskInputContext;
270 qskInputContext = inputContext;
271
272 if ( oldContext && oldContext->parent() == nullptr )
273 {
274 delete oldContext;
275
276 // still needed with > Qt 5.15 ?
277 qskSendToPlatformContext( QEvent::PlatformPanel );
278 }
279 }
280}
281
282QskInputContext* QskInputContext::instance()
283{
284 return qskInputContext;
285}
286
287class QskInputContext::PrivateData
288{
289 public:
290 inline QskInputPanel* createPanel( QskInputContext* context ) const
291 {
292 QskInputPanel* panel = nullptr;
293
294 if ( this->factory )
295 panel = this->factory->createPanel();
296
297 if ( panel == nullptr )
298 panel = new Panel();
299
300 connect( panel, &QskInputPanel::visibleChanged,
301 context, &QskInputContext::activeChanged );
302
303 connect( panel, &QskInputPanel::localeChanged,
304 context, [] { qskSendToPlatformContext( QEvent::LocaleChange ); } );
305
306 connect( panel, &QskInputPanel::inputItemDestroyed,
307 context, [ context, panel ] { context->hideChannel( panel ); } );
308
309 return panel;
310 }
311
312 void closeChannel( Channel* channel )
313 {
314 if ( channel->popup )
315 channel->popup->close(); // deleteOnClose is set
316
317 if ( channel->window )
318 channel->window->close(); // deleteOnClose is set
319 }
320
321 ChannelTable channels;
322 QPointer< QskInputContextFactory > factory;
323};
324
325QskInputContext::QskInputContext()
326 : m_data( new PrivateData() )
327{
328 setObjectName( QStringLiteral( "InputContext" ) );
329}
330
331QskInputContext::~QskInputContext()
332{
333}
334
335void QskInputContext::setFactory( QskInputContextFactory* factory )
336{
337 if ( m_data->factory == factory )
338 return;
339
340 if ( m_data->factory && m_data->factory->parent() == this )
341 delete m_data->factory;
342
343 m_data->factory = factory;
344
345 if ( factory && factory->parent() == nullptr )
346 factory->setParent( this );
347}
348
349QskInputContextFactory* QskInputContext::factory() const
350{
351 return m_data->factory;
352}
353
354std::shared_ptr< QskTextPredictor > QskInputContext::textPredictor( const QLocale& locale )
355{
356 if ( m_data->factory )
357 return m_data->factory->setupPredictor( locale );
358
359 return nullptr;
360}
361
362void QskInputContext::update( const QQuickItem* item, Qt::InputMethodQueries queries )
363{
364 if ( item == nullptr )
365 {
366 // those are coming from QQuickWindow based on focus changes
367 item = qobject_cast< QQuickItem* >( QGuiApplication::focusObject() );
368 }
369
370 auto channel = m_data->channels.channel( item );
371 if ( channel == nullptr )
372 return;
373
374 if ( queries & Qt::ImEnabled )
375 {
376 QInputMethodQueryEvent event( Qt::ImEnabled );
377 QCoreApplication::sendEvent( channel->item, &event );
378
379 if ( !event.value( Qt::ImEnabled ).toBool() )
380 {
381 hidePanel( item );
382 return;
383 }
384 }
385
386 channel->panel->updateInputPanel( queries );
387}
388
389QRectF QskInputContext::panelRect() const
390{
391 /*
392 As we can have more than panel at the same time we
393 better don't return any geometry
394 */
395
396 return QRectF();
397}
398
399void QskInputContext::showPanel( const QQuickItem* item )
400{
401 if ( item == nullptr )
402 return;
403
404 if ( m_data->channels.ancestorChannel( item ) )
405 {
406 // We are inside of an existing panel
407 return;
408 }
409
410 if ( auto channel = m_data->channels.channel( item->window() ) )
411 {
412 if ( channel->item == item )
413 return;
414
415 hidePanel( channel->item );
416 }
417
418 auto panel = m_data->createPanel( this );
419
420 auto channel = m_data->channels.insert( item->window() );
421 channel->item = const_cast< QQuickItem* >( item );
422 channel->panel = panel;
423
424 if ( QskDialog::instance()->policy() == QskDialog::TopLevelWindow )
425 {
426 // The input panel is embedded in a top level window
427
428 auto window = new QskWindow();
429
430 window->setFlags( window->flags() & Qt::Dialog );
431 // window->setModality( Qt::ApplicationModal );
432 window->setAutoLayoutChildren( true );
433#if 0
434 window->setFlags( Qt::Tool | Qt::WindowDoesNotAcceptFocus );
435#endif
436
437 panel->setParentItem( window->contentItem() );
438
439 QSize size = window->sizeConstraint();
440 if ( size.isEmpty() )
441 {
442 // no idea, maybe something based on the screen size
443 size = QSize( 800, 240 );
444 }
445
446 window->resize( size );
447 window->show();
448
449 window->setDeleteOnClose( true );
450
451 channel->window = window;
452 }
453 else
454 {
455 // The input panel is embedded in a popup
456
457 auto popup = new Popup();
458
459 popup->setAutoLayoutChildren( true );
460 popup->setMargins( 5 );
461 popup->setModal( true );
462
463 popup->setPopupFlag( QskPopup::CloseOnPressOutside, true );
464 popup->setPopupFlag( QskPopup::DeleteOnClose, true );
465
466 popup->setParentItem( item->window()->contentItem() );
467 popup->setParent( this );
468
469 channel->popup = popup;
470
471 panel->setParentItem( popup );
472 if ( panel->parent() == nullptr )
473 panel->setParent( popup );
474
475 popup->open();
476
477 connect( popup, &Popup::commitRequested, panel,
478 [panel]() { panel->commitCurrentText( true ); } );
479 }
480
481 panel->attachInputItem( const_cast< QQuickItem* >( item ) );
482}
483
484void QskInputContext::hidePanel( const QQuickItem* item )
485{
486 if ( item == nullptr )
487 return;
488
489 if ( auto channel = m_data->channels.channel( item ) )
490 {
491 m_data->closeChannel( channel );
492 m_data->channels.remove( item->window() );
493 }
494}
495
496void QskInputContext::hideChannel( const QskInputPanel* panel )
497{
498 if ( auto channel = m_data->channels.channel( panel ) )
499 {
500 // channel->item is already dead here
501 m_data->closeChannel( channel );
502 m_data->channels.remove( panel->window() );
503 }
504}
505
506void QskInputContext::setInputPanelVisible( const QQuickItem* item, bool on )
507{
508 // called from inside the controls
509
510 if ( item == nullptr )
511 item = qobject_cast< QQuickItem* >( QGuiApplication::focusObject() );
512
513 if ( item )
514 {
515 if ( on )
516 showPanel( item );
517 else
518 hidePanel( item );
519 }
520}
521
522bool QskInputContext::isInputPanelVisible() const
523{
524 return m_data->channels.currentChannel() != nullptr;
525}
526
527QLocale QskInputContext::locale() const
528{
529 if ( auto channel = m_data->channels.currentChannel() )
530 {
531 if ( channel->panel )
532 return channel->panel->locale();
533 }
534
535 return QLocale();
536}
537
538void QskInputContext::setFocusObject( QObject* )
539{
540}
541
542void QskInputContext::invokeAction(
543 QInputMethod::Action, int cursorPosition )
544{
545 // called from qquicktextinput/qquicktextedit
546 Q_UNUSED( cursorPosition )
547}
548
549void QskInputContext::commitPrediction( bool )
550{
551 /*
552 called, when the input item loses the focus.
553 As it it should be possible to navigate inside of the
554 panel what should we do here ?
555 */
556}
557
558class QskInputContextFactory::PrivateData
559{
560 public:
561 QThread* thread = nullptr;
562 std::shared_ptr< QskTextPredictor > predictor;
563 QLocale predictorLocale;
564};
565
566QskInputContextFactory::QskInputContextFactory( QObject* parent )
567 : QObject( parent )
568 , m_data( new PrivateData() )
569{
570}
571
572QskInputContextFactory::~QskInputContextFactory()
573{
574 if( m_data->thread )
575 {
576 m_data->thread->quit();
577 connect( m_data->thread, &QThread::finished, m_data->thread, &QObject::deleteLater );
578 }
579}
580
581std::shared_ptr< QskTextPredictor > QskInputContextFactory::setupPredictor( const QLocale& locale )
582{
583 if( !m_data->predictor
584 || m_data->predictorLocale.language() != locale.language()
585 || qskTerritory( m_data->predictorLocale ) != qskTerritory( locale ) )
586 {
587 m_data->predictor = std::shared_ptr< QskTextPredictor >( createPredictor( locale ) );
588 m_data->predictorLocale = QLocale( locale.language(), qskTerritory( locale ) );
589
590 if( m_data->predictor )
591 {
592 if( !m_data->thread )
593 {
594 m_data->thread = new QThread();
595 m_data->thread->start();
596 }
597
598 m_data->predictor->moveToThread( m_data->thread );
599 }
600 }
601
602 return m_data->predictor;
603}
604
605QskTextPredictor* QskInputContextFactory::createPredictor( const QLocale& locale )
606{
607#if HUNSPELL
608 return new QskHunspellTextPredictor( locale );
609#else
610 Q_UNUSED( locale )
611#endif
612
613 return nullptr;
614}
615
616QskInputPanel* QskInputContextFactory::createPanel() const
617{
618 return new Panel();
619}
620
621#include "QskInputContext.moc"
622#include "moc_QskInputContext.cpp"
void localeChanged(const QLocale &)
QskMargins margins
Definition QskControl.h:38
QSizeF sizeConstraint
Definition QskControl.h:50
void setPolishOnResize(bool)
Definition QskItem.cpp:476
virtual void aboutToShow()
Definition QskItem.cpp:1099
void setGeometry(qreal x, qreal y, qreal width, qreal height)
Definition QskItem.cpp:332
void aboutToShow() override
Definition QskPopup.cpp:600