6#include "QskInputContext.h"
7#include "QskInputPanel.h"
8#include "QskInputPanelBox.h"
11#include "QskLinearBox.h"
14#include "QskTextPredictor.h"
16#include "QskPlatform.h"
17#include "QskInputGrabber.h"
19#include <qguiapplication.h>
24#include <qpa/qplatforminputcontext.h>
25#include <qpa/qplatformintegration.h>
28#include "QskHunspellTextPredictor.h"
43 setPolishOnParentResize(
true );
47 void commitRequested();
54 if ( popupFlags() & QskPopup::CloseOnPressOutside )
56 if (
auto inputGrabber = findChild< QskInputGrabber* >() )
57 inputGrabber->installEventFilter(
this );
61 bool eventFilter( QObject*
object, QEvent* event )
override
63 if ( qobject_cast< QskInputGrabber* >(
object ) )
65 if ( event->type() == QEvent::MouseButtonPress )
67 if ( popupFlags() & QskPopup::CloseOnPressOutside )
69 Q_EMIT commitRequested();
75 return Inherited::eventFilter(
object, event );
78 void updateLayout()
override
81 const auto item = findChild< const QskInputPanel* >();
83 auto r = qskItemGeometry( parentItem() );
85 r = qskConstrainedItemRect( item, r );
95 Panel( QQuickItem* parent =
nullptr )
100 connect( m_box, &QskInputPanelBox::keySelected,
101 this, &QskInputPanel::keySelected );
103 connect( m_box, &QskInputPanelBox::predictiveTextSelected,
104 this, &QskInputPanel::predictiveTextSelected );
107 void attachItem( QQuickItem* item )
override
109 m_box->attachInputItem( item );
112 QQuickItem* inputProxy()
const override
114 return m_box->inputProxy();
117 void setPrompt(
const QString& prompt )
override
119 m_box->setInputPrompt( prompt );
122 void setPredictionEnabled(
bool on )
override
124 m_box->setPanelHint( QskInputPanelBox::Prediction, on );
127 void setPrediction(
const QStringList& prediction )
override
129 QskInputPanel::setPrediction( prediction );
130 m_box->setPrediction( prediction );
141 QPointer< QQuickItem > item;
144 QPointer< QskInputPanel > panel;
147 QPointer< Popup > popup;
148 QPointer< QskWindow > window;
154 inline Channel* currentChannel()
const
156 const auto object = QGuiApplication::focusObject();
157 return channel( qobject_cast< const QQuickItem* >(
object ) );
160 inline Channel* channel(
const QQuickWindow* window )
const
164 auto it = m_map.constFind( window );
165 if ( it != m_map.constEnd() )
166 return const_cast< Channel*
>( &it.value() );
176 QMap< const QQuickWindow*, Channel >::const_iterator it;
178 for( it = m_map.constBegin(); it != m_map.constEnd(); ++it )
180 if( it.value().panel == panel )
181 return const_cast< Channel*
>( &it.value() );
188 inline Channel* channel(
const QQuickItem* item )
const
193 QMap< const QQuickWindow*, Channel >::const_iterator it;
195 for( it = m_map.constBegin(); it != m_map.constEnd(); ++it )
197 if( it.value().item == item )
198 return const_cast< Channel*
>( &it.value() );
205 inline Channel* ancestorChannel(
const QQuickItem* item )
const
207 for (
auto it = m_map.constBegin();
208 it != m_map.constEnd(); ++it )
210 if (
const auto panel = it.value().panel )
212 if ( ( item == panel ) || qskIsAncestorOf( panel, item ) )
214 return const_cast< Channel*
>( &it.value() );
222 inline Channel* insert(
const QQuickWindow* window )
224 return &m_map[ window ];
227 inline void remove(
const QQuickWindow* window )
229 m_map.remove( window );
233 QMap< const QQuickWindow*, Channel > m_map;
237static QPointer< QskInputContext > qskInputContext;
239static void qskSendToPlatformContext( QEvent::Type type )
241 const auto integration = qskPlatformIntegration();
243 if (
const auto inputContext = integration->inputContext() )
245 QEvent event( type );
246 QCoreApplication::sendEvent( inputContext, &event );
250static void qskInputContextHook()
252 qAddPostRoutine( [] {
delete qskInputContext; } );
255Q_COREAPP_STARTUP_FUNCTION( qskInputContextHook )
257#if QT_VERSION >= QT_VERSION_CHECK( 6, 2, 0 )
258 static inline QLocale::Territory qskTerritory(
const QLocale& locale )
259 {
return locale.territory(); }
261 static inline QLocale::Country qskTerritory(
const QLocale& locale )
262 {
return locale.country(); }
267 if ( inputContext != qskInputContext )
269 const auto oldContext = qskInputContext;
270 qskInputContext = inputContext;
272 if ( oldContext && oldContext->parent() ==
nullptr )
277 qskSendToPlatformContext( QEvent::PlatformPanel );
284 return qskInputContext;
287class QskInputContext::PrivateData
295 panel = this->factory->createPanel();
297 if ( panel ==
nullptr )
300 connect( panel, &QskInputPanel::visibleChanged,
301 context, &QskInputContext::activeChanged );
304 context, [] { qskSendToPlatformContext( QEvent::LocaleChange ); } );
306 connect( panel, &QskInputPanel::inputItemDestroyed,
307 context, [ context, panel ] { context->hideChannel( panel ); } );
312 void closeChannel( Channel* channel )
314 if ( channel->popup )
315 channel->popup->close();
317 if ( channel->window )
318 channel->window->close();
321 ChannelTable channels;
322 QPointer< QskInputContextFactory > factory;
325QskInputContext::QskInputContext()
326 : m_data( new PrivateData() )
328 setObjectName( QStringLiteral(
"InputContext" ) );
331QskInputContext::~QskInputContext()
337 if ( m_data->factory == factory )
340 if ( m_data->factory && m_data->factory->parent() ==
this )
341 delete m_data->factory;
343 m_data->factory = factory;
345 if ( factory && factory->parent() ==
nullptr )
346 factory->setParent(
this );
351 return m_data->factory;
354std::shared_ptr< QskTextPredictor > QskInputContext::textPredictor(
const QLocale& locale )
356 if ( m_data->factory )
357 return m_data->factory->setupPredictor( locale );
362void QskInputContext::update(
const QQuickItem* item, Qt::InputMethodQueries queries )
364 if ( item ==
nullptr )
367 item = qobject_cast< QQuickItem* >( QGuiApplication::focusObject() );
370 auto channel = m_data->channels.channel( item );
371 if ( channel ==
nullptr )
374 if ( queries & Qt::ImEnabled )
376 QInputMethodQueryEvent event( Qt::ImEnabled );
377 QCoreApplication::sendEvent( channel->item, &event );
379 if ( !event.value( Qt::ImEnabled ).toBool() )
386 channel->panel->updateInputPanel( queries );
389QRectF QskInputContext::panelRect()
const
399void QskInputContext::showPanel(
const QQuickItem* item )
401 if ( item ==
nullptr )
404 if ( m_data->channels.ancestorChannel( item ) )
410 if (
auto channel = m_data->channels.channel( item->window() ) )
412 if ( channel->item == item )
415 hidePanel( channel->item );
418 auto panel = m_data->createPanel(
this );
420 auto channel = m_data->channels.insert( item->window() );
421 channel->item =
const_cast< QQuickItem*
>( item );
422 channel->panel = panel;
424 if ( QskDialog::instance()->policy() == QskDialog::TopLevelWindow )
430 window->setFlags( window->flags() & Qt::Dialog );
432 window->setAutoLayoutChildren(
true );
434 window->setFlags( Qt::Tool | Qt::WindowDoesNotAcceptFocus );
437 panel->setParentItem( window->contentItem() );
440 if ( size.isEmpty() )
443 size = QSize( 800, 240 );
446 window->resize( size );
449 window->setDeleteOnClose(
true );
451 channel->window = window;
457 auto popup =
new Popup();
459 popup->setAutoLayoutChildren(
true );
460 popup->setMargins( 5 );
461 popup->setModal(
true );
463 popup->setPopupFlag( QskPopup::CloseOnPressOutside,
true );
464 popup->setPopupFlag( QskPopup::DeleteOnClose,
true );
466 popup->setParentItem( item->window()->contentItem() );
467 popup->setParent(
this );
469 channel->popup = popup;
471 panel->setParentItem( popup );
472 if ( panel->parent() ==
nullptr )
473 panel->setParent( popup );
477 connect( popup, &Popup::commitRequested, panel,
478 [panel]() { panel->commitCurrentText(
true ); } );
481 panel->attachInputItem(
const_cast< QQuickItem*
>( item ) );
484void QskInputContext::hidePanel(
const QQuickItem* item )
486 if ( item ==
nullptr )
489 if (
auto channel = m_data->channels.channel( item ) )
491 m_data->closeChannel( channel );
492 m_data->channels.remove( item->window() );
496void QskInputContext::hideChannel(
const QskInputPanel* panel )
498 if (
auto channel = m_data->channels.channel( panel ) )
501 m_data->closeChannel( channel );
502 m_data->channels.remove( panel->window() );
506void QskInputContext::setInputPanelVisible(
const QQuickItem* item,
bool on )
510 if ( item ==
nullptr )
511 item = qobject_cast< QQuickItem* >( QGuiApplication::focusObject() );
522bool QskInputContext::isInputPanelVisible()
const
524 return m_data->channels.currentChannel() !=
nullptr;
527QLocale QskInputContext::locale()
const
529 if (
auto channel = m_data->channels.currentChannel() )
531 if ( channel->panel )
532 return channel->panel->locale();
538void QskInputContext::setFocusObject( QObject* )
542void QskInputContext::invokeAction(
543 QInputMethod::Action,
int cursorPosition )
546 Q_UNUSED( cursorPosition )
549void QskInputContext::commitPrediction(
bool )
558class QskInputContextFactory::PrivateData
561 QThread* thread =
nullptr;
562 std::shared_ptr< QskTextPredictor > predictor;
563 QLocale predictorLocale;
566QskInputContextFactory::QskInputContextFactory( QObject* parent )
568 , m_data( new PrivateData() )
572QskInputContextFactory::~QskInputContextFactory()
576 m_data->thread->quit();
577 connect( m_data->thread, &QThread::finished, m_data->thread, &QObject::deleteLater );
581std::shared_ptr< QskTextPredictor > QskInputContextFactory::setupPredictor(
const QLocale& locale )
583 if( !m_data->predictor
584 || m_data->predictorLocale.language() != locale.language()
585 || qskTerritory( m_data->predictorLocale ) != qskTerritory( locale ) )
587 m_data->predictor = std::shared_ptr< QskTextPredictor >( createPredictor( locale ) );
588 m_data->predictorLocale = QLocale( locale.language(), qskTerritory( locale ) );
590 if( m_data->predictor )
592 if( !m_data->thread )
594 m_data->thread =
new QThread();
595 m_data->thread->start();
598 m_data->predictor->moveToThread( m_data->thread );
602 return m_data->predictor;
605QskTextPredictor* QskInputContextFactory::createPredictor(
const QLocale& locale )
621#include "QskInputContext.moc"
622#include "moc_QskInputContext.cpp"
void localeChanged(const QLocale &)
void setPolishOnResize(bool)
virtual void aboutToShow()
void setGeometry(qreal x, qreal y, qreal width, qreal height)