QSkinny 0.8.0
C++/Qt UI toolkit
Loading...
Searching...
No Matches
QskShortcutMap.cpp
1/******************************************************************************
2 * QSkinny - Copyright (C) The authors
3 * SPDX-License-Identifier: BSD-3-Clause
4 *****************************************************************************/
5
6#include "QskShortcutMap.h"
7#include "QskMetaInvokable.h"
8#include "QskQuick.h"
9#include "QskInternalMacros.h"
10
11#include <qkeysequence.h>
12#include <qquickitem.h>
13
14QSK_QT_PRIVATE_BEGIN
15#include <QtGui/private/qguiapplication_p.h>
16QSK_QT_PRIVATE_END
17
18#include <map>
19
20static inline QShortcutMap* qskShortcutMap()
21{
22 return qGuiApp ? &QGuiApplicationPrivate::instance()->shortcutMap : nullptr;
23}
24
25class QskShortcutHandler final : public QObject
26{
27 public:
28 QskShortcutHandler();
29
30 int insert( QQuickItem*, const QKeySequence&, bool autoRepeat,
31 const QObject*, const QskMetaInvokable& );
32
33 void remove( int id );
34
35 void setEnabled( int id, bool );
36 void setEnabled( const QKeySequence&, bool on );
37 void setAutoRepeat( int id, bool repeat );
38
39 bool eventFilter( QObject*, QEvent* ) override;
40 bool invoke( QQuickItem*, const QKeySequence& );
41
42 private:
43 void cleanUp( QObject* );
44
45 class InvokeData
46 {
47 public:
48 InvokeData()
49 : item( nullptr )
50 , receiver( nullptr )
51 {
52 }
53
54 void invoke() const
55 {
56 auto that = const_cast< InvokeData* >( this );
57 auto object = const_cast< QObject* >( receiver );
58
59 void* args[] = { nullptr };
60 that->invokable.invoke( object, args, Qt::AutoConnection );
61 }
62
63 QKeySequence sequence;
64 QQuickItem* item;
65 const QObject* receiver;
66 QskMetaInvokable invokable;
67 };
68
69 std::map< int, InvokeData > m_invokeDataMap;
70};
71
72Q_GLOBAL_STATIC( QskShortcutHandler, qskShortcutHandler )
73
74static bool qskContextMatcher( QObject* object, Qt::ShortcutContext context )
75{
76 if ( context == Qt::ApplicationShortcut )
77 return true;
78
79 auto item = qobject_cast< QQuickItem* >( object );
80 if ( item && context == Qt::WindowShortcut )
81 {
82 if ( QskShortcutMap::contextMatcher( item, context ) )
83 {
84 /*
85 Unfortunatley there is no way to know about
86 the contextItem without making it the receiver of
87 the following QShortcutEvent. So we have to install
88 an event handler to process and swallow it in QskShortcutHandler.
89 */
90
91 item->installEventFilter( qskShortcutHandler );
92 return true;
93 }
94 }
95
96 return false;
97}
98
99QskShortcutHandler::QskShortcutHandler()
100{
101 // to process all sort of shortcut events at the same place
102 installEventFilter( this );
103}
104
105int QskShortcutHandler::insert(
106 QQuickItem* item, const QKeySequence& sequence, bool autoRepeat,
107 const QObject* receiver, const QskMetaInvokable& invokable )
108{
109 if ( sequence.isEmpty() )
110 {
111 qDebug() << "QskShortcutMap: invalid shortcut key sequence";
112 return 0;
113 }
114
115 if ( invokable.parameterCount() > 0 )
116 {
117 qDebug() << "QskShortcutMap: invalid slot parameter count";
118 return 0;
119 }
120
121 if ( receiver )
122 {
123 connect( receiver, &QObject::destroyed,
124 this, &QskShortcutHandler::cleanUp, Qt::UniqueConnection );
125 }
126
127 int id = 0;
128
129 auto map = qskShortcutMap();
130
131 if ( item )
132 {
133 if ( item != receiver )
134 {
135 connect( item, &QObject::destroyed,
136 this, &QskShortcutHandler::cleanUp, Qt::UniqueConnection );
137 }
138
139 id = map->addShortcut( item, sequence, Qt::WindowShortcut, qskContextMatcher );
140 }
141 else
142 {
143 id = map->addShortcut( this, sequence, Qt::ApplicationShortcut, qskContextMatcher );
144 }
145
146 auto& data = m_invokeDataMap[ id ];
147
148 data.sequence = sequence;
149 data.item = item;
150 data.receiver = receiver;
151 data.invokable = invokable;
152
153 if ( !autoRepeat )
154 setAutoRepeat( id, false );
155
156 return id;
157}
158
159void QskShortcutHandler::remove( int id )
160{
161 auto it = m_invokeDataMap.find( id );
162 if ( it == m_invokeDataMap.end() )
163 return;
164
165 auto map = qskShortcutMap();
166 map->removeShortcut( id, nullptr );
167
168 const QQuickItem* item = it->second.item;
169 const QObject* receiver = it->second.receiver;
170
171 m_invokeDataMap.erase( it );
172
173 /*
174 Finally let's check if we can disconnect
175 from the destroyed signals
176 */
177 for ( const auto& entry : std::as_const( m_invokeDataMap ) )
178 {
179 if ( item == nullptr && receiver == nullptr )
180 break;
181
182 if ( entry.second.item == item )
183 item = nullptr;
184
185 if ( entry.second.receiver == receiver )
186 receiver = nullptr;
187 }
188
189 if ( item )
190 item->disconnect( this );
191
192 if ( receiver && receiver != item )
193 receiver->disconnect( this );
194}
195
196void QskShortcutHandler::cleanUp( QObject* object )
197{
198 /*
199 When item != receiver we might remain being connected
200 to destroyed signals we are not interested in anymore. TODO ...
201 */
202 auto map = qskShortcutMap();
203
204 for ( auto it = m_invokeDataMap.begin(); it != m_invokeDataMap.end(); )
205 {
206 const auto& data = it->second;
207
208 if ( data.item == object || data.receiver == object )
209 {
210 if ( map )
211 map->removeShortcut( it->first, nullptr );
212
213 it = m_invokeDataMap.erase( it );
214
215 continue;
216 }
217
218 ++it;
219 }
220}
221
222void QskShortcutHandler::setEnabled( const QKeySequence& sequence, bool on )
223{
224 for ( auto it = m_invokeDataMap.begin();
225 it != m_invokeDataMap.end(); ++it )
226 {
227 if ( it->second.sequence == sequence )
228 setEnabled( it->first, on );
229 }
230}
231
232void QskShortcutHandler::setEnabled( int id, bool enabled )
233{
234 auto map = qskShortcutMap();
235 map->setShortcutEnabled( enabled, id, this );
236}
237
238void QskShortcutHandler::setAutoRepeat( int id, bool repeat )
239{
240 auto map = qskShortcutMap();
241 map->setShortcutAutoRepeat( repeat, id, this );
242}
243
244bool QskShortcutHandler::eventFilter( QObject* object, QEvent* event )
245{
246 if ( event->type() != QEvent::Shortcut )
247 return false;
248
249 if ( object != this )
250 object->removeEventFilter( this );
251
252 const QShortcutEvent* se = static_cast< const QShortcutEvent* >( event );
253
254#if 0
255 // do we want to handle this ???
256 if ( se->isAmbiguous() )
257 ....
258#endif
259
260 const auto it = m_invokeDataMap.find( se->shortcutId() );
261 if ( it != m_invokeDataMap.end() )
262 {
263 auto& data = it->second;
264
265 Q_ASSERT( data.item == nullptr || data.item == object );
266
267 data.invoke();
268
269 return true;
270 }
271
272 // seems like someone else is also interested in shortcuts
273 return false;
274}
275
276bool QskShortcutHandler::invoke( QQuickItem* item, const QKeySequence& sequence )
277{
278 bool found = false;
279
280 for ( const auto& entry : std::as_const( m_invokeDataMap ) )
281 {
282 auto& data = entry.second;
283
284 if ( ( data.item == item ) && ( data.sequence == sequence ) )
285 {
286 data.invoke();
287 found = true;
288 }
289 }
290
291 return found;
292}
293
294int QskShortcutMap::addMethod( QQuickItem* item, const QKeySequence& sequence,
295 bool autoRepeat, const QObject* receiver, const char* method )
296{
297 if ( receiver == nullptr )
298 {
299 qDebug() << "QskShortcutMap: bad receiver for shortcut:" << sequence;
300 return 0;
301 }
302
303 return qskShortcutHandler->insert(
304 item, sequence, autoRepeat, receiver, qskMetaMethod( receiver, method ) );
305}
306
307int QskShortcutMap::addFunction( QQuickItem* item, const QKeySequence& sequence,
308 bool autoRepeat, const QObject* receiver, const QskMetaFunction& function )
309{
310 if ( ( receiver == nullptr ) &&
311 ( function.functionType() == QskMetaFunction::MemberFunction ) )
312 {
313 qDebug() << "QskShortcutMap: bad receiver for shortcut:" << sequence;
314 return 0;
315 }
316
317 return qskShortcutHandler->insert(
318 item, sequence, autoRepeat, receiver, function );
319}
320
321bool QskShortcutMap::invokeCallback( const QKeySequence& sequence )
322{
323 QQuickItem* item = nullptr;
324 return qskShortcutHandler->invoke( item, sequence );
325}
326
327bool QskShortcutMap::invokeCallback( QQuickWindow* window, const QKeySequence& sequence )
328{
329 auto item = window ? window->contentItem() : nullptr;
330 return qskShortcutHandler->invoke( item, sequence );
331}
332
333bool QskShortcutMap::invokeCallback( QQuickItem* item, const QKeySequence& sequence )
334{
335 return qskShortcutHandler->invoke( item, sequence );
336}
337
338void QskShortcutMap::setAutoRepeat( int id, bool on )
339{
340 qskShortcutHandler->setAutoRepeat( id, on );
341}
342
343void QskShortcutMap::setEnabled( const QKeySequence& sequence, bool on )
344{
345 qskShortcutHandler->setEnabled( sequence, on );
346}
347
348void QskShortcutMap::setEnabled( int id, bool on )
349{
350 qskShortcutHandler->setEnabled( id, on );
351}
352
353void QskShortcutMap::removeShortcut( int id )
354{
355 qskShortcutHandler->remove( id );
356}
357
358bool QskShortcutMap::contextMatcher(
359 const QQuickItem* item, Qt::ShortcutContext context )
360{
361 if ( context == Qt::ApplicationShortcut )
362 return true;
363
364 if ( item && context == Qt::WindowShortcut )
365 {
366 const auto focusWindow = QGuiApplication::focusWindow();
367
368 const auto window = item->window();
369 if ( window == nullptr || window != focusWindow )
370 {
371 return false;
372 }
373
374 while ( item )
375 {
376 /*
377 We have to find out if the active focus is inside
378 the surronding shortcut scope.
379 */
380 if ( qskIsShortcutScope( item ) )
381 {
382 if ( !item->hasFocus() )
383 return false;
384 }
385
386 item = item->parentItem();
387 }
388
389 return true;
390 }
391
392 return false;
393}