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