Qt
Internal/Contributor docs for the Qt SDK. Note: These are NOT official API docs; those are found at https://doc.qt.io/
Loading...
Searching...
No Matches
deviceskin.cpp
Go to the documentation of this file.
1// Copyright (C) 2016 The Qt Company Ltd.
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 "deviceskin_p.h"
5
6#include <QtWidgets/qapplication.h>
7
8#include <QtGui/qbitmap.h>
9#include <QtGui/qevent.h>
10#include <QtGui/qimage.h>
11#include <QtGui/qpainter.h>
12#include <QtGui/qpixmap.h>
13
14#include <QtCore/qdebug.h>
15#include <QtCore/qdir.h>
16#include <QtCore/qfile.h>
17#include <QtCore/qfileinfo.h>
18#include <QtCore/qnamespace.h>
19#include <QtCore/qregularexpression.h>
20#include <QtCore/qtextstream.h>
21#include <QtCore/qtimer.h>
22
23#ifdef TEST_SKIN
24# include <QtWidgets/qmainwindow.h>
25# include <QtWidgets/qboxlayout.h>
26# include <QtWidgets/qdialog.h>
27# include <QtWidgets/qdialogbuttonbox.h>
28#endif
29
31
32using namespace Qt::StringLiterals;
33
34namespace {
35 enum { joydistance = 10, key_repeat_period = 50, key_repeat_delay = 500 };
36 enum { debugDeviceSkin = 0 };
37}
38
39static void parseRect(const QString &value, QRect *rect) {
40 const auto l = QStringView{value}.split(QLatin1Char(' '));
41 rect->setRect(l[0].toInt(), l[1].toInt(), l[2].toInt(), l[3].toInt());
42}
43
44static QString msgImageNotLoaded(const QString &f) {
45 return DeviceSkin::tr("The image file '%1' could not be loaded.").arg(f);
46}
47
48// ------------ DeviceSkinButtonArea
49QDebug &operator<<(QDebug &str, const DeviceSkinButtonArea &a)
50{
51
52 str << "Area: " << a.name << " keyCode=" << a.keyCode << " area=" << a.area
53 << " text=" << a.text << " activeWhenClosed=" << a.activeWhenClosed;
54 return str;
55}
56
57// ------------ DeviceSkinParameters
58
59QDebug operator<<(QDebug str, const DeviceSkinParameters &p)
60{
61 str << "Images " << p.skinImageUpFileName << ','
62 << p.skinImageDownFileName<< ',' << p.skinImageClosedFileName
63 << ',' << p.skinCursorFileName <<"\nScreen: " << p.screenRect
64 << " back: " << p.backScreenRect << " closed: " << p.closedScreenRect
65 << " cursor: " << p.cursorHot << " Prefix: " << p.prefix
66 << " Joystick: " << p.joystick << " MouseHover" << p.hasMouseHover;
67 const int numAreas = p.buttonAreas.size();
68 for (int i = 0; i < numAreas; i++)
69 str << p.buttonAreas[i];
70 return str;
71}
72
74{
75 return backScreenRect.isNull() ? closedScreenRect .size(): backScreenRect.size();
76}
77
79{
80 return secondaryScreenSize() != QSize(0, 0);
81}
82
83bool DeviceSkinParameters::read(const QString &skinDirectory, ReadMode rm, QString *errorMessage)
84{
85 // Figure out the name. remove ending '/' if present
86 QString skinFile = skinDirectory;
87 if (skinFile.endsWith(QLatin1Char('/')))
88 skinFile.truncate(skinFile.size() - 1);
89
90 QFileInfo fi(skinFile);
91 QString fn;
92 if ( fi.isDir() ) {
93 prefix = skinFile;
94 prefix += QLatin1Char('/');
95 fn = prefix;
96 fn += fi.baseName();
97 fn += ".skin"_L1;
98 } else if (fi.isFile()){
99 fn = skinFile;
100 prefix = fi.path();
101 prefix += QLatin1Char('/');
102 } else {
103 *errorMessage = DeviceSkin::tr("The skin directory '%1' does not contain a configuration file.").arg(skinDirectory);
104 return false;
105 }
106 QFile f(fn);
107 if (!f.open(QIODevice::ReadOnly )) {
108 *errorMessage = DeviceSkin::tr("The skin configuration file '%1' could not be opened.").arg(fn);
109 return false;
110 }
111 QTextStream ts(&f);
112 const bool rc = read(ts, rm, errorMessage);
113 if (!rc)
114 *errorMessage = DeviceSkin::tr("The skin configuration file '%1' could not be read: %2")
115 .arg(fn, *errorMessage);
116 return rc;
117}
118bool DeviceSkinParameters::read(QTextStream &ts, ReadMode rm, QString *errorMessage)
119{
120 QStringList closedAreas;
121 QStringList toggleAreas;
122 QStringList toggleActiveAreas;
123 int nareas = 0;
124 screenDepth = 0;
125 QString mark;
126 ts >> mark;
127 hasMouseHover = true; // historical default
128 if (mark == "[SkinFile]"_L1) {
129 const QString UpKey = "Up"_L1;
130 const QString DownKey = "Down"_L1;
131 const QString ClosedKey = "Closed"_L1;
132 const QString ClosedAreasKey = "ClosedAreas"_L1;
133 const QString ScreenKey = "Screen"_L1;
134 const QString ScreenDepthKey = "ScreenDepth"_L1;
135 const QString BackScreenKey = "BackScreen"_L1;
136 const QString ClosedScreenKey = "ClosedScreen"_L1;
137 const QString CursorKey = "Cursor"_L1;
138 const QString AreasKey = "Areas"_L1;
139 const QString ToggleAreasKey = "ToggleAreas"_L1;
140 const QString ToggleActiveAreasKey = "ToggleActiveAreas"_L1;
141 const QString HasMouseHoverKey = "HasMouseHover"_L1;
142 // New
143 while (!nareas) {
144 QString line = ts.readLine();
145 if ( line.isNull() )
146 break;
147 if (!line.isEmpty() && line.at(0) != u'#') {
148 int eq = line.indexOf(QLatin1Char('='));
149 if ( eq >= 0 ) {
150 const QString key = line.left(eq);
151 eq++;
152 while (eq<line.size()-1 && line[eq].isSpace())
153 eq++;
154 const QString value = line.mid(eq);
155 if ( key == UpKey ) {
156 skinImageUpFileName = value;
157 } else if ( key == DownKey ) {
158 skinImageDownFileName = value;
159 } else if ( key == ClosedKey ) {
160 skinImageClosedFileName = value;
161 } else if ( key == ClosedAreasKey ) {
162 closedAreas = value.split(QLatin1Char(' '));
163 } else if ( key == ScreenKey ) {
164 parseRect( value, &screenRect);
165 } else if ( key == ScreenDepthKey ) {
166 screenDepth = value.toInt();
167 } else if ( key == BackScreenKey ) {
168 parseRect(value, &backScreenRect);
169 } else if ( key == ClosedScreenKey ) {
170 parseRect( value, &closedScreenRect );
171 } else if ( key == CursorKey ) {
172 QStringList l = value.split(QLatin1Char(' '));
173 skinCursorFileName = l[0];
174 cursorHot = QPoint(l[1].toInt(),l[2].toInt());
175 } else if ( key == AreasKey ) {
176 nareas = value.toInt();
177 } else if ( key == ToggleAreasKey ) {
178 toggleAreas = value.split(QLatin1Char(' '));
179 } else if ( key == ToggleActiveAreasKey ) {
180 toggleActiveAreas = value.split(QLatin1Char(' '));
181 } else if ( key == HasMouseHoverKey ) {
182 hasMouseHover = value == "true"_L1 || value == "1"_L1;
183 }
184 } else {
185 *errorMessage = DeviceSkin::tr("Syntax error: %1").arg(line);
186 return false;
187 }
188 }
189 }
190 } else {
191 // Old
192 skinImageUpFileName = mark;
193 QString s;
194 int x,y,w,h,na;
195 ts >> s >> x >> y >> w >> h >> na;
196 skinImageDownFileName = s;
197 screenRect.setRect(x, y, w, h);
198 nareas = na;
199 }
200 // Done for short mode
201 if (rm == ReadSizeOnly)
202 return true;
203 // verify skin files exist
204 skinImageUpFileName.insert(0, prefix);
205 if (!QFile(skinImageUpFileName).exists()) {
206 *errorMessage = DeviceSkin::tr("The skin \"up\" image file '%1' does not exist.").arg(skinImageUpFileName);
207 return false;
208 }
209 if (!skinImageUp.load(skinImageUpFileName)) {
210 *errorMessage = msgImageNotLoaded(skinImageUpFileName);
211 return false;
212 }
213
214 skinImageDownFileName.insert(0, prefix);
215 if (!QFile(skinImageDownFileName).exists()) {
216 *errorMessage = DeviceSkin::tr("The skin \"down\" image file '%1' does not exist.").arg(skinImageDownFileName);
217 return false;
218 }
219 if (!skinImageDown.load(skinImageDownFileName)) {
220 *errorMessage = msgImageNotLoaded(skinImageDownFileName);
221 return false;
222 }
223
224 if (!skinImageClosedFileName.isEmpty()) {
225 skinImageClosedFileName.insert(0, prefix);
226 if (!QFile(skinImageClosedFileName).exists()) {
227 *errorMessage = DeviceSkin::tr("The skin \"closed\" image file '%1' does not exist.").arg(skinImageClosedFileName);
228 return false;
229 }
230 if (!skinImageClosed.load(skinImageClosedFileName)) {
231 *errorMessage = msgImageNotLoaded(skinImageClosedFileName);
232 return false;
233 }
234 }
235
236 if (!skinCursorFileName.isEmpty()) {
237 skinCursorFileName.insert(0, prefix);
238 if (!QFile(skinCursorFileName).exists()) {
239 *errorMessage = DeviceSkin::tr("The skin cursor image file '%1' does not exist.").arg(skinCursorFileName);
240 return false;
241 }
242 if (!skinCursor.load(skinCursorFileName)) {
243 *errorMessage = msgImageNotLoaded(skinCursorFileName);
244 return false;
245 }
246 }
247
248 // read areas
249 if (!nareas)
250 return true;
251 buttonAreas.reserve(nareas);
252
253 int i = 0;
254 ts.readLine(); // eol
255 joystick = -1;
256 const QString Joystick = "Joystick"_L1;
257 const QRegularExpression splitRe("[ \t][ \t]*"_L1);
258 Q_ASSERT(splitRe.isValid());
259 while (i < nareas && !ts.atEnd() ) {
260 buttonAreas.push_back(DeviceSkinButtonArea());
261 DeviceSkinButtonArea &area = buttonAreas.back();
262 const QString line = ts.readLine();
263 if ( !line.isEmpty() && line[0] != QLatin1Char('#') ) {
264 const QStringList tok = line.split(splitRe);
265 if ( tok.size()<6 ) {
266 *errorMessage = DeviceSkin::tr("Syntax error in area definition: %1").arg(line);
267 return false;
268 } else {
269 area.name = tok[0];
270 QString k = tok[1];
271 if ( k.left(2).toLower() == "0x"_L1) {
272 area.keyCode = k.mid(2).toInt(0,16);
273 } else {
274 area.keyCode = k.toInt();
275 }
276
277 int p=0;
278 for (int j=2; j < tok.size() - 1; ) {
279 const int x = tok[j++].toInt();
280 const int y = tok[j++].toInt();
281 area.area.putPoints(p++,1,x,y);
282 }
283
284 const QChar doubleQuote = QLatin1Char('"');
285 if ( area.name[0] == doubleQuote && area.name.endsWith(doubleQuote)) {
286 area.name.truncate(area.name.size() - 1);
287 area.name.remove(0, 1);
288 }
289 if ( area.name.size() == 1 )
290 area.text = area.name;
291 if ( area.name == Joystick)
292 joystick = i;
293 area.activeWhenClosed = closedAreas.contains(area.name)
294 || area.keyCode == Qt::Key_Flip; // must be to work
295 area.toggleArea = toggleAreas.contains(area.name);
296 area.toggleActiveArea = toggleActiveAreas.contains(area.name);
297 if (area.toggleArea)
298 toggleAreaList += i;
299 i++;
300 }
301 }
302 }
303 if (i != nareas) {
304 qWarning() << DeviceSkin::tr("Mismatch in number of areas, expected %1, got %2.")
305 .arg(nareas).arg(i);
306 }
307 if (debugDeviceSkin)
308 qDebug() << *this;
309 return true;
310}
311
312// --------- CursorWindow declaration
313
314namespace qvfb_internal {
315
316class CursorWindow : public QWidget
317{
318public:
319 explicit CursorWindow(const QImage &cursor, QPoint hot, QWidget *sk);
320
321 void setView(QWidget*);
323 bool handleMouseEvent(QEvent *ev);
324
325protected:
326 bool event(QEvent *) override;
327 bool eventFilter(QObject*, QEvent *) override;
328
329private:
330 QWidget *mouseRecipient;
331 QWidget *m_view;
332 QWidget *skin;
333 QPoint hotspot;
334};
335}
336
337// --------- Skin
338
339DeviceSkin::DeviceSkin(const DeviceSkinParameters &parameters, QWidget *p ) :
340 QWidget(p),
341 m_parameters(parameters),
342 buttonRegions(parameters.buttonAreas.size(), QRegion()),
343 parent(p),
344 t_skinkey(new QTimer(this)),
345 t_parentmove(new QTimer(this))
346{
347 Q_ASSERT(p);
348 setMouseTracking(true);
349 setAttribute(Qt::WA_NoSystemBackground);
350
351 setZoom(1.0);
352 connect(t_skinkey, &QTimer::timeout, this, &DeviceSkin::skinKeyRepeat );
353 t_parentmove->setSingleShot( true );
354 connect(t_parentmove, &QTimer::timeout, this, &DeviceSkin::moveParent );
355}
356
357void DeviceSkin::skinKeyRepeat()
358{
359 if ( m_view ) {
360 const DeviceSkinButtonArea &area = m_parameters.buttonAreas[buttonIndex];
361 emit skinKeyReleaseEvent(area.keyCode,area.text, true);
362 emit skinKeyPressEvent(area.keyCode, area.text, true);
363 t_skinkey->start(key_repeat_period);
364 }
365}
366
367void DeviceSkin::calcRegions()
368{
369 const int numAreas = m_parameters.buttonAreas.size();
370 for (int i=0; i<numAreas; i++) {
371 QPolygon xa(m_parameters.buttonAreas[i].area.size());
372 int n = m_parameters.buttonAreas[i].area.size();
373 for (int p = 0; p < n; p++) {
374 xa.setPoint(p,transform.map(m_parameters.buttonAreas[i].area[p]));
375 }
376 if (n == 2) {
377 buttonRegions[i] = QRegion(xa.boundingRect());
378 } else {
379 buttonRegions[i] = QRegion(xa);
380 }
381 }
382}
383
384void DeviceSkin::loadImages()
385{
386 QImage iup = m_parameters.skinImageUp;
387 QImage idown = m_parameters.skinImageDown;
388
389 QImage iclosed;
390 const bool hasClosedImage = !m_parameters.skinImageClosed.isNull();
391
392 if (hasClosedImage)
393 iclosed = m_parameters.skinImageClosed;
394 QImage icurs;
395 const bool hasCursorImage = !m_parameters.skinCursor.isNull();
396 if (hasCursorImage)
397 icurs = m_parameters.skinCursor;
398
399 if (!transform.isIdentity()) {
400 iup = iup.transformed(transform, Qt::SmoothTransformation);
401 idown = idown.transformed(transform, Qt::SmoothTransformation);
402 if (hasClosedImage)
403 iclosed = iclosed.transformed(transform, Qt::SmoothTransformation);
404 if (hasCursorImage)
405 icurs = icurs.transformed(transform, Qt::SmoothTransformation);
406 }
407 const Qt::ImageConversionFlags conv = Qt::ThresholdAlphaDither|Qt::AvoidDither;
408 skinImageUp = QPixmap::fromImage(iup);
409 skinImageDown = QPixmap::fromImage(idown, conv);
410 if (hasClosedImage)
411 skinImageClosed = QPixmap::fromImage(iclosed, conv);
412 if (hasCursorImage)
413 skinCursor = QPixmap::fromImage(icurs, conv);
414
415 setFixedSize( skinImageUp.size() );
416 if (!skinImageUp.mask())
417 skinImageUp.setMask(skinImageUp.createHeuristicMask());
418 if (!skinImageClosed.mask())
419 skinImageClosed.setMask(skinImageClosed.createHeuristicMask());
420
421 QWidget* parent = parentWidget();
422 parent->setMask( skinImageUp.mask() );
423 parent->setFixedSize( skinImageUp.size() );
424
425 delete cursorw;
426 cursorw = 0;
427 if (hasCursorImage) {
428 cursorw = new qvfb_internal::CursorWindow(m_parameters.skinCursor, m_parameters.cursorHot, this);
429 if (m_view)
430 cursorw->setView(m_view);
431 }
432}
433
435{
436 delete cursorw;
437}
438
439void DeviceSkin::setTransform(const QTransform &wm)
440{
441 transform = QImage::trueMatrix(wm,m_parameters.skinImageUp.width(),m_parameters.skinImageUp.height());
442 calcRegions();
443 loadImages();
444 if ( m_view ) {
445 QPoint p = transform.map(QPolygon(m_parameters.screenRect)).boundingRect().topLeft();
446 m_view->move(p);
447 }
448 updateSecondaryScreen();
449}
450
451void DeviceSkin::setZoom( double z )
452{
453 setTransform(QTransform().scale(z,z));
454}
455
456void DeviceSkin::updateSecondaryScreen()
457{
458 if (!m_secondaryView)
459 return;
460 if (flipped_open) {
461 if (m_parameters.backScreenRect.isNull()) {
462 m_secondaryView->hide();
463 } else {
464 m_secondaryView->move(transform.map(QPolygon(m_parameters.backScreenRect)).boundingRect().topLeft());
465 m_secondaryView->show();
466 }
467 } else {
468 if (m_parameters.closedScreenRect.isNull()) {
469 m_secondaryView->hide();
470 } else {
471 m_secondaryView->move(transform.map(QPolygon(m_parameters.closedScreenRect)).boundingRect().topLeft());
472 m_secondaryView->show();
473 }
474 }
475}
476
477void DeviceSkin::setView( QWidget *v )
478{
479 m_view = v;
480 m_view->setFocus();
481 m_view->move(transform.map(QPolygon(m_parameters.screenRect)).boundingRect().topLeft());
482 if ( cursorw )
483 cursorw->setView(v);
484}
485
486void DeviceSkin::setSecondaryView( QWidget *v )
487{
488 m_secondaryView = v;
489 updateSecondaryScreen();
490}
491
492void DeviceSkin::paintEvent( QPaintEvent *)
493{
494 QPainter p( this );
495 if ( flipped_open ) {
496 p.drawPixmap(0, 0, skinImageUp);
497 } else {
498 p.drawPixmap(0, 0, skinImageClosed);
499 }
500 QList<int> toDraw;
501 if ( buttonPressed == true ) {
502 toDraw += buttonIndex;
503 }
504 for (int toggle : std::as_const(m_parameters.toggleAreaList)) {
505 const DeviceSkinButtonArea &ba = m_parameters.buttonAreas[toggle];
506 if (flipped_open || ba.activeWhenClosed) {
507 if (ba.toggleArea && ba.toggleActiveArea)
508 toDraw += toggle;
509 }
510 }
511 for (int button : std::as_const(toDraw)) {
512 const DeviceSkinButtonArea &ba = m_parameters.buttonAreas[button];
513 const QRect r = buttonRegions[button].boundingRect();
514 if ( ba.area.size() > 2 )
515 p.setClipRegion(buttonRegions[button]);
516 p.drawPixmap( r.topLeft(), skinImageDown, r);
517 }
518}
519
520void DeviceSkin::mousePressEvent( QMouseEvent *e )
521{
522 if (e->button() == Qt::RightButton) {
523 emit popupMenu();
524 } else {
525 buttonPressed = false;
526
527 onjoyrelease = -1;
528 const int numAreas = m_parameters.buttonAreas.size();
529 for (int i = 0; i < numAreas ; i++) {
530 const DeviceSkinButtonArea &ba = m_parameters.buttonAreas[i];
531 if ( buttonRegions[i].contains( e->position().toPoint() ) ) {
532 if ( flipped_open || ba.activeWhenClosed ) {
533 if ( m_parameters.joystick == i ) {
534 joydown = true;
535 } else {
536 if ( joydown )
537 onjoyrelease = i;
538 else
539 startPress(i);
540 break;
541 if (debugDeviceSkin)// Debug message to be sure we are clicking the right areas
542 qDebug()<< m_parameters.buttonAreas[i].name << " clicked";
543 }
544 }
545 }
546 }
547 clickPos = e->position().toPoint();
548// This is handy for finding the areas to define rectangles for new skins
549 if (debugDeviceSkin)
550 qDebug()<< "Clicked in " << e->position().toPoint().x() << ',' << e->position().toPoint().y();
551 clickPos = e->position().toPoint();
552 }
553}
554
555void DeviceSkin::flip(bool open)
556{
557 if ( flipped_open == open )
558 return;
559 if ( open ) {
560 parent->setMask(skinImageUp.mask());
561 emit skinKeyReleaseEvent(Qt::Key(Qt::Key_Flip), QString(), false);
562 } else {
563 parent->setMask(skinImageClosed.mask());
564 emit skinKeyPressEvent(Qt::Key(Qt::Key_Flip), QString(), false);
565 }
566 flipped_open = open;
567 updateSecondaryScreen();
568 repaint();
569}
570
571void DeviceSkin::startPress(int i)
572{
573 buttonPressed = true;
574 buttonIndex = i;
575 if (m_view) {
576 const DeviceSkinButtonArea &ba = m_parameters.buttonAreas[buttonIndex];
577 if (ba.keyCode == Qt::Key_Flip) {
578 flip(!flipped_open);
579 } else if (ba.toggleArea) {
580 bool active = !ba.toggleActiveArea;
581 const_cast<DeviceSkinButtonArea &>(ba).toggleActiveArea = active;
582 if (active)
583 emit skinKeyPressEvent(ba.keyCode, ba.text, false);
584 else
585 emit skinKeyReleaseEvent(ba.keyCode, ba.text, false);
586 } else {
587 emit skinKeyPressEvent(ba.keyCode, ba.text, false);
588 t_skinkey->start(key_repeat_delay);
589 }
590 repaint(buttonRegions[buttonIndex].boundingRect());
591 }
592}
593
594void DeviceSkin::endPress()
595{
596 const DeviceSkinButtonArea &ba = m_parameters.buttonAreas[buttonIndex];
597 if (m_view && ba.keyCode != Qt::Key_Flip && !ba.toggleArea )
598 emit skinKeyReleaseEvent(ba.keyCode, ba.text, false);
599 t_skinkey->stop();
600 buttonPressed = false;
601 repaint( buttonRegions[buttonIndex].boundingRect() );
602}
603
604void DeviceSkin::mouseMoveEvent( QMouseEvent *e )
605{
606 if ( e->buttons() & Qt::LeftButton ) {
607 const int joystick = m_parameters.joystick;
608 QPoint newpos = e->globalPosition().toPoint() - clickPos;
609 if (joydown) {
610 int k1=0, k2=0;
611 if (newpos.x() < -joydistance) {
612 k1 = joystick+1;
613 } else if (newpos.x() > +joydistance) {
614 k1 = joystick+3;
615 }
616 if (newpos.y() < -joydistance) {
617 k2 = joystick+2;
618 } else if (newpos.y() > +joydistance) {
619 k2 = joystick+4;
620 }
621 if (k1 || k2) {
622 if (!buttonPressed) {
623 onjoyrelease = -1;
624 if (k1 && k2) {
625 startPress(k2);
626 endPress();
627 }
628 startPress(k1 ? k1 : k2);
629 }
630 } else if (buttonPressed) {
631 endPress();
632 }
633 } else if (buttonPressed == false) {
634 parentpos = newpos;
635 if (!t_parentmove->isActive())
636 t_parentmove->start(50);
637 }
638 }
639 if ( cursorw )
640 cursorw->setPos(e->globalPosition().toPoint());
641}
642
644{
645 parent->move( parentpos );
646}
647
648void DeviceSkin::mouseReleaseEvent( QMouseEvent * )
649{
650 if ( buttonPressed )
651 endPress();
652 if ( joydown ) {
653 joydown = false;
654 if (onjoyrelease >= 0) {
655 startPress(onjoyrelease);
656 endPress();
657 }
658 }
659}
660
662{
663 return !skinCursor.isNull();
664}
665
666// ------------------ CursorWindow implementation
667
668namespace qvfb_internal {
669
670bool CursorWindow::eventFilter( QObject *, QEvent *ev)
671{
672 handleMouseEvent(ev);
673 return false;
674}
675
676bool CursorWindow::event( QEvent *ev )
677{
678 if (handleMouseEvent(ev))
679 return true;
680 return QWidget::event(ev);
681}
682
684{
685 bool handledEvent = false;
686 static int inhere=0;
687 if ( !inhere ) {
688 inhere++;
689 if (m_view) {
690 if (ev->type() >= QEvent::MouseButtonPress && ev->type() <= QEvent::MouseMove) {
691 QMouseEvent *e = (QMouseEvent*)ev;
692 QPoint gp = e->globalPosition().toPoint();
693 QPoint vp = m_view->mapFromGlobal(gp);
694 QPoint sp = skin->mapFromGlobal(gp);
695 if (e->type() == QEvent::MouseButtonPress || e->type() == QEvent::MouseButtonDblClick) {
696 if (m_view->rect().contains(vp))
697 mouseRecipient = m_view;
698 else if (skin->parentWidget()->geometry().contains(gp))
699 mouseRecipient = skin;
700 else
701 mouseRecipient = 0;
702 }
703 if (mouseRecipient) {
704 setPos(gp);
705 QMouseEvent me(e->type(),mouseRecipient==skin ? sp : vp,gp,e->button(),e->buttons(),e->modifiers());
706 QApplication::sendEvent(mouseRecipient, &me);
707 } else if (!skin->parentWidget()->geometry().contains(gp)) {
708 hide();
709 } else {
710 setPos(gp);
711 }
712 if (e->type() == QEvent::MouseButtonRelease)
713 mouseRecipient = 0;
714 handledEvent = true;
715 }
716 }
717 inhere--;
718 }
719 return handledEvent;
720}
721
722void CursorWindow::setView(QWidget* v)
723{
724 if ( m_view ) {
725 m_view->removeEventFilter(this);
726 m_view->removeEventFilter(this);
727 }
728 m_view = v;
729 m_view->installEventFilter(this);
730 m_view->installEventFilter(this);
731 mouseRecipient = 0;
732}
733
734CursorWindow::CursorWindow(const QImage &img, QPoint hot, QWidget* sk)
735 : QWidget(0),
736 m_view(0),
737 skin(sk),
738 hotspot(hot)
739{
740 setWindowFlags( Qt::FramelessWindowHint );
741 mouseRecipient = 0;
742 setMouseTracking(true);
743#ifndef QT_NO_CURSOR
744 setCursor(Qt::BlankCursor);
745#endif
746 QPixmap p;
747 p = QPixmap::fromImage(img);
748 if (!p.mask()) {
749 QBitmap bm = img.hasAlphaChannel() ? QBitmap::fromImage(img.createAlphaMask())
750 : QBitmap::fromImage(img.createHeuristicMask());
751 p.setMask(bm);
752 }
753 QPalette palette;
754 palette.setBrush(backgroundRole(), QBrush(p));
755 setPalette(palette);
756 setFixedSize( p.size() );
757 if ( !p.mask().isNull() )
758 setMask(p.mask());
759}
760
761void CursorWindow::setPos(QPoint p)
762{
763 move(p-hotspot);
764 show();
765 raise();
766}
767}
768
769#ifdef TEST_SKIN
770
771int main(int argc,char *argv[])
772{
773 if (argc < 1)
774 return 1;
775 const QString skinFile = QString::fromUtf8(argv[1]);
776 QApplication app(argc,argv);
777 QMainWindow mw;
778
779 DeviceSkinParameters params;
780 QString errorMessage;
781 if (!params.read(skinFile, DeviceSkinParameters::ReadAll, &errorMessage)) {
782 qWarning() << errorMessage;
783 return 1;
784 }
785 DeviceSkin ds(params, &mw);
786 // View Dialog
787 QDialog *dialog = new QDialog();
788 QHBoxLayout *dialogLayout = new QHBoxLayout();
789 dialog->setLayout(dialogLayout);
790 QDialogButtonBox *dialogButtonBox = new QDialogButtonBox(QDialogButtonBox::Ok|QDialogButtonBox::Cancel);
791 QObject::connect(dialogButtonBox, &QDialogButtonBox::rejected, dialog, &QDialog::reject);
792 QObject::connect(dialogButtonBox, &QDialogButtonBox::accepted, dialog, &QDialog::accept);
793 dialogLayout->addWidget(dialogButtonBox);
794 dialog->setFixedSize(params.screenSize());
795 dialog->setParent(&ds, Qt::SubWindow);
796 dialog->setAutoFillBackground(true);
797 ds.setView(dialog);
798
799 QObject::connect(&ds, &DeviceSkin::popupMenu, &mw, &QWidget::close);
800 QObject::connect(&ds, &DeviceSkin::skinKeyPressEvent, &mw, &QWidget::close);
801 mw.show();
802 return app.exec();
803}
804
805#endif
806
807QT_END_NAMESPACE
void mouseReleaseEvent(QMouseEvent *) override
This event handler, for event event, can be reimplemented in a subclass to receive mouse release even...
void mousePressEvent(QMouseEvent *e) override
This event handler, for event event, can be reimplemented in a subclass to receive mouse press events...
~DeviceSkin() override
void paintEvent(QPaintEvent *) override
This event handler can be reimplemented in a subclass to receive paint events passed in event.
void mouseMoveEvent(QMouseEvent *e) override
This event handler, for event event, can be reimplemented in a subclass to receive mouse move events ...
void setZoom(double)
void setTransform(const QTransform &)
bool hasCursor() const
void moveParent()
bool eventFilter(QObject *, QEvent *) override
Filters events if this object has been installed as an event filter for the watched object.
bool handleMouseEvent(QEvent *ev)
CursorWindow(const QImage &cursor, QPoint hot, QWidget *sk)
bool event(QEvent *) override
This virtual function receives events to an object and should return true if the event e was recogniz...
QDebug operator<<(QDebug str, const DeviceSkinParameters &p)
QDebug & operator<<(QDebug &str, const DeviceSkinButtonArea &a)
static void parseRect(const QString &value, QRect *rect)
static QString msgImageNotLoaded(const QString &f)
Combined button and popup list for selecting options.
bool hasSecondaryScreen() const
QSize secondaryScreenSize() const