VirtualBox

source: vbox/trunk/src/VBox/Frontends/VirtualBox/src/manager/details/UIDetailsElement.cpp@ 79923

Last change on this file since 79923 was 79923, checked in by vboxsync, 6 years ago

FE/Qt: bugref:7720: VirtualBox Manager: Details pane: Possibility to edit VM guest OS type on-the-fly.

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
  • Property svn:mergeinfo set to (toggle deleted branches)
    /branches/VBox-3.0/src/VBox/Frontends/VirtualBox/src/selector/graphics/chooser/UIGChooserItemMachine.cpp58652,​70973
    /branches/VBox-3.2/src/VBox/Frontends/VirtualBox/src/selector/graphics/chooser/UIGChooserItemMachine.cpp66309,​66318
    /branches/VBox-4.0/src/VBox/Frontends/VirtualBox/src/selector/graphics/chooser/UIGChooserItemMachine.cpp70873
    /branches/VBox-4.1/src/VBox/Frontends/VirtualBox/src/selector/graphics/chooser/UIGChooserItemMachine.cpp74233
    /branches/VBox-4.2/src/VBox/Frontends/VirtualBox/src/selector/graphics/details/UIGDetailsElement.cpp91503-91504,​91506-91508,​91510,​91514-91515,​91521
    /branches/VBox-4.3/src/VBox/Frontends/VirtualBox/src/selector/graphics/details/UIGDetailsElement.cpp91223
    /branches/VBox-4.3/trunk/src/VBox/Frontends/VirtualBox/src/selector/graphics/details/UIGDetailsElement.cpp91223
    /branches/dsen/gui/src/VBox/Frontends/VirtualBox/src/selector/graphics/chooser/UIGChooserItemMachine.cpp79076-79078,​79089,​79109-79110,​79112-79113,​79127-79130,​79134,​79141,​79151,​79155,​79157-79159,​79193,​79197
    /branches/dsen/gui2/src/VBox/Frontends/VirtualBox/src/selector/graphics/details/UIGDetailsElement.cpp79562-79569,​79572-79573,​79578,​79581-79582,​79590-79591,​79598-79599,​79602-79603,​79605-79606,​79632,​79635,​79637,​79644
    /branches/dsen/gui3/src/VBox/Frontends/VirtualBox/src/selector/graphics/details/UIGDetailsElement.cpp79645-79692
    /trunk/src/VBox/Frontends/VirtualBox/src/selector/graphics/chooser/UIGChooserItemMachine.cpp79225,​79271
File size: 27.5 KB
Line 
1/* $Id: UIDetailsElement.cpp 79923 2019-07-22 12:19:49Z vboxsync $ */
2/** @file
3 * VBox Qt GUI - UIDetailsElement class implementation.
4 */
5
6/*
7 * Copyright (C) 2012-2019 Oracle Corporation
8 *
9 * This file is part of VirtualBox Open Source Edition (OSE), as
10 * available from http://www.virtualbox.org. This file is free software;
11 * you can redistribute it and/or modify it under the terms of the GNU
12 * General Public License (GPL) as published by the Free Software
13 * Foundation, in version 2 as it comes in the "COPYING" file of the
14 * VirtualBox OSE distribution. VirtualBox OSE is distributed in the
15 * hope that it will be useful, but WITHOUT ANY WARRANTY of any kind.
16 */
17
18/* Qt includes: */
19#include <QGraphicsSceneMouseEvent>
20#include <QGraphicsView>
21#include <QPropertyAnimation>
22#include <QSignalTransition>
23#include <QStateMachine>
24#include <QStyleOptionGraphicsItem>
25
26/* GUI includes: */
27#include "QIDialogContainer.h"
28#include "UIActionPool.h"
29#include "UICommon.h"
30#include "UIConverter.h"
31#include "UIDetailsElement.h"
32#include "UIDetailsSet.h"
33#include "UIDetailsModel.h"
34#include "UIExtraDataManager.h"
35#include "UIGraphicsRotatorButton.h"
36#include "UIGraphicsTextPane.h"
37#include "UIIconPool.h"
38#include "UIMachineAttributeSetter.h"
39#include "UINameAndSystemEditor.h"
40#include "UIVirtualBoxManager.h"
41
42
43/** Known anchor roles. */
44enum AnchorRole
45{
46 AnchorRole_Invalid,
47 AnchorRole_MachineName,
48 AnchorRole_OSType,
49 AnchorRole_Storage,
50};
51
52
53UIDetailsElement::UIDetailsElement(UIDetailsSet *pParent, DetailsElementType enmType, bool fOpened)
54 : UIDetailsItem(pParent)
55 , m_pSet(pParent)
56 , m_enmType(enmType)
57#ifdef VBOX_WS_MAC
58 , m_iDefaultToneStart(145)
59 , m_iDefaultToneFinal(155)
60 , m_iHoverToneStart(115)
61 , m_iHoverToneFinal(125)
62#else
63 , m_iDefaultToneStart(160)
64 , m_iDefaultToneFinal(190)
65 , m_iHoverToneStart(160)
66 , m_iHoverToneFinal(190)
67#endif
68 , m_fHovered(false)
69 , m_fNameHovered(false)
70 , m_pHoveringMachine(0)
71 , m_pHoveringAnimationForward(0)
72 , m_pHoveringAnimationBackward(0)
73 , m_iAnimationDuration(300)
74 , m_iDefaultValue(0)
75 , m_iHoveredValue(255)
76 , m_iAnimatedValue(m_iDefaultValue)
77 , m_pButton(0)
78 , m_fClosed(!fOpened)
79 , m_fAnimationRunning(false)
80 , m_iAdditionalHeight(0)
81 , m_pTextPane(0)
82 , m_iMinimumHeaderWidth(0)
83 , m_iMinimumHeaderHeight(0)
84{
85 /* Prepare element: */
86 prepareElement();
87 /* Prepare button: */
88 prepareButton();
89 /* Prepare text-pane: */
90 prepareTextPane();
91
92 /* Setup size-policy: */
93 setSizePolicy(QSizePolicy::Minimum, QSizePolicy::Fixed);
94
95 /* Add item to the parent: */
96 AssertMsg(parentItem(), ("No parent set for details element!"));
97 parentItem()->addItem(this);
98}
99
100UIDetailsElement::~UIDetailsElement()
101{
102 /* Remove item from the parent: */
103 AssertMsg(parentItem(), ("No parent set for details element!"));
104 parentItem()->removeItem(this);
105}
106
107void UIDetailsElement::setText(const UITextTable &text)
108{
109 /* Pass text to text-pane: */
110 m_pTextPane->setText(text);
111}
112
113UITextTable &UIDetailsElement::text() const
114{
115 /* Retrieve text from text-pane: */
116 return m_pTextPane->text();
117}
118
119void UIDetailsElement::close(bool fAnimated /* = true */)
120{
121 m_pButton->setToggled(false, fAnimated);
122}
123
124void UIDetailsElement::open(bool fAnimated /* = true */)
125{
126 m_pButton->setToggled(true, fAnimated);
127}
128
129void UIDetailsElement::markAnimationFinished()
130{
131 /* Mark animation as non-running: */
132 m_fAnimationRunning = false;
133
134 /* Recursively update size-hint: */
135 updateGeometry();
136 /* Repaint: */
137 update();
138}
139
140void UIDetailsElement::updateAppearance()
141{
142 /* Reset name hover state: */
143 m_fNameHovered = false;
144 updateNameHoverLink();
145
146 /* Update anchor role restrictions: */
147 ConfigurationAccessLevel cal = m_pSet->configurationAccessLevel();
148 m_pTextPane->setAnchorRoleRestricted("#machine_name", cal != ConfigurationAccessLevel_Full);
149 m_pTextPane->setAnchorRoleRestricted("#os_type", cal != ConfigurationAccessLevel_Full);
150 m_pTextPane->setAnchorRoleRestricted("#mount", cal == ConfigurationAccessLevel_Null);
151 m_pTextPane->setAnchorRoleRestricted("#attach", cal != ConfigurationAccessLevel_Full);
152}
153
154int UIDetailsElement::minimumWidthHint() const
155{
156 /* Prepare variables: */
157 int iMargin = data(ElementData_Margin).toInt();
158 int iMinimumWidthHint = 0;
159
160 /* Maximum width: */
161 iMinimumWidthHint = qMax(m_iMinimumHeaderWidth, (int)m_pTextPane->minimumSizeHint().width());
162
163 /* And 4 margins: 2 left and 2 right: */
164 iMinimumWidthHint += 4 * iMargin;
165
166 /* Return result: */
167 return iMinimumWidthHint;
168}
169
170int UIDetailsElement::minimumHeightHint() const
171{
172 return minimumHeightHintForElement(m_fClosed);
173}
174
175void UIDetailsElement::showEvent(QShowEvent *pEvent)
176{
177 /* Call to base-class: */
178 UIDetailsItem::showEvent(pEvent);
179
180 /* Update icon: */
181 updateIcon();
182}
183
184void UIDetailsElement::resizeEvent(QGraphicsSceneResizeEvent*)
185{
186 /* Update layout: */
187 updateLayout();
188}
189
190void UIDetailsElement::hoverMoveEvent(QGraphicsSceneHoverEvent *pEvent)
191{
192 /* Update hover state: */
193 if (!m_fHovered)
194 {
195 m_fHovered = true;
196 emit sigHoverEnter();
197 }
198
199 /* Update name-hover state: */
200 handleHoverEvent(pEvent);
201}
202
203void UIDetailsElement::hoverLeaveEvent(QGraphicsSceneHoverEvent *pEvent)
204{
205 /* Update hover state: */
206 if (m_fHovered)
207 {
208 m_fHovered = false;
209 emit sigHoverLeave();
210 }
211
212 /* Update name-hover state: */
213 handleHoverEvent(pEvent);
214}
215
216void UIDetailsElement::mousePressEvent(QGraphicsSceneMouseEvent *pEvent)
217{
218 /* Only for hovered header: */
219 if (!m_fNameHovered)
220 return;
221
222 /* Process link click: */
223 pEvent->accept();
224 QString strCategory;
225 if (m_enmType >= DetailsElementType_General &&
226 m_enmType < DetailsElementType_Description)
227 strCategory = QString("#%1").arg(gpConverter->toInternalString(m_enmType));
228 else if (m_enmType == DetailsElementType_Description)
229 strCategory = QString("#%1%%mTeDescription").arg(gpConverter->toInternalString(m_enmType));
230 emit sigLinkClicked(strCategory, QString(), machine().GetId());
231}
232
233void UIDetailsElement::mouseDoubleClickEvent(QGraphicsSceneMouseEvent *pEvent)
234{
235 /* Only for left-button: */
236 if (pEvent->button() != Qt::LeftButton)
237 return;
238
239 /* Process left-button double-click: */
240 emit sigToggleElement(m_enmType, isClosed());
241}
242
243void UIDetailsElement::paint(QPainter *pPainter, const QStyleOptionGraphicsItem *pOptions, QWidget *)
244{
245 /* Update button visibility: */
246 updateButtonVisibility();
247
248 /* Paint background: */
249 paintBackground(pPainter, pOptions);
250 /* Paint frame: */
251 paintFrame(pPainter, pOptions);
252 /* Paint element info: */
253 paintElementInfo(pPainter, pOptions);
254}
255
256QString UIDetailsElement::description() const
257{
258 return tr("%1 details", "like 'General details' or 'Storage details'").arg(m_strName);
259}
260
261const CMachine &UIDetailsElement::machine()
262{
263 return m_pSet->machine();
264}
265
266void UIDetailsElement::setName(const QString &strName)
267{
268 /* Cache name: */
269 m_strName = strName;
270 QFontMetrics fm(m_nameFont, model()->paintDevice());
271 m_nameSize = QSize(fm.width(m_strName), fm.height());
272
273 /* Update linked values: */
274 updateMinimumHeaderWidth();
275 updateMinimumHeaderHeight();
276}
277
278void UIDetailsElement::setAdditionalHeight(int iAdditionalHeight)
279{
280 /* Cache new value: */
281 m_iAdditionalHeight = iAdditionalHeight;
282 /* Update layout: */
283 updateLayout();
284 /* Repaint: */
285 update();
286}
287
288QVariant UIDetailsElement::data(int iKey) const
289{
290 /* Provide other members with required data: */
291 switch (iKey)
292 {
293 /* Hints: */
294 case ElementData_Margin: return QApplication::style()->pixelMetric(QStyle::PM_SmallIconSize) / 4;
295 case ElementData_Spacing: return QApplication::style()->pixelMetric(QStyle::PM_SmallIconSize) / 2;
296 /* Default: */
297 default: break;
298 }
299 return QVariant();
300}
301
302void UIDetailsElement::addItem(UIDetailsItem*)
303{
304 AssertMsgFailed(("Details element do NOT support children!"));
305}
306
307void UIDetailsElement::removeItem(UIDetailsItem*)
308{
309 AssertMsgFailed(("Details element do NOT support children!"));
310}
311
312QList<UIDetailsItem*> UIDetailsElement::items(UIDetailsItemType) const
313{
314 AssertMsgFailed(("Details element do NOT support children!"));
315 return QList<UIDetailsItem*>();
316}
317
318bool UIDetailsElement::hasItems(UIDetailsItemType) const
319{
320 AssertMsgFailed(("Details element do NOT support children!"));
321 return false;
322}
323
324void UIDetailsElement::clearItems(UIDetailsItemType)
325{
326 AssertMsgFailed(("Details element do NOT support children!"));
327}
328
329void UIDetailsElement::updateLayout()
330{
331 /* Prepare variables: */
332 QSize size = geometry().size().toSize();
333 int iMargin = data(ElementData_Margin).toInt();
334
335 /* Layout button: */
336 int iButtonWidth = m_buttonSize.width();
337 int iButtonHeight = m_buttonSize.height();
338 int iButtonX = size.width() - 2 * iMargin - iButtonWidth;
339 int iButtonY = iButtonHeight == m_iMinimumHeaderHeight ? iMargin :
340 iMargin + (m_iMinimumHeaderHeight - iButtonHeight) / 2;
341 m_pButton->setPos(iButtonX, iButtonY);
342
343 /* If closed: */
344 if (isClosed())
345 {
346 /* Hide text-pane if still visible: */
347 if (m_pTextPane->isVisible())
348 m_pTextPane->hide();
349 }
350 /* If opened: */
351 else
352 {
353 /* Layout text-pane: */
354 int iTextPaneX = 2 * iMargin;
355 int iTextPaneY = iMargin + m_iMinimumHeaderHeight + 2 * iMargin;
356 m_pTextPane->setPos(iTextPaneX, iTextPaneY);
357 m_pTextPane->resize(size.width() - 4 * iMargin,
358 size.height() - 4 * iMargin - m_iMinimumHeaderHeight);
359 /* Show text-pane if still invisible and animation finished: */
360 if (!m_pTextPane->isVisible() && !isAnimationRunning())
361 m_pTextPane->show();
362 }
363}
364
365int UIDetailsElement::minimumHeightHintForElement(bool fClosed) const
366{
367 /* Prepare variables: */
368 int iMargin = data(ElementData_Margin).toInt();
369 int iMinimumHeightHint = 0;
370
371 /* Two margins: */
372 iMinimumHeightHint += 2 * iMargin;
373
374 /* Header height: */
375 iMinimumHeightHint += m_iMinimumHeaderHeight;
376
377 /* Element is opened? */
378 if (!fClosed)
379 {
380 /* Add text height: */
381 if (!m_pTextPane->isEmpty())
382 iMinimumHeightHint += 2 * iMargin + (int)m_pTextPane->minimumSizeHint().height();
383 }
384
385 /* Additional height during animation: */
386 if (m_fAnimationRunning)
387 iMinimumHeightHint += m_iAdditionalHeight;
388
389 /* Return value: */
390 return iMinimumHeightHint;
391}
392
393void UIDetailsElement::sltHandleWindowRemapped()
394{
395 /* Update icon: */
396 updateIcon();
397}
398
399void UIDetailsElement::sltToggleButtonClicked()
400{
401 emit sigToggleElement(m_enmType, isClosed());
402}
403
404void UIDetailsElement::sltElementToggleStart()
405{
406 /* Mark animation running: */
407 m_fAnimationRunning = true;
408
409 /* Setup animation: */
410 updateAnimationParameters();
411
412 /* Invert toggle-state: */
413 m_fClosed = !m_fClosed;
414}
415
416void UIDetailsElement::sltElementToggleFinish(bool fToggled)
417{
418 /* Update toggle-state: */
419 m_fClosed = !fToggled;
420
421 /* Notify about finishing: */
422 emit sigToggleElementFinished();
423}
424
425void UIDetailsElement::sltHandleAnchorClicked(const QString &strAnchor)
426{
427 /* Compose a map of known anchor roles: */
428 QMap<QString, AnchorRole> roles;
429 roles["#machine_name"] = AnchorRole_MachineName;
430 roles["#os_type"] = AnchorRole_OSType;
431 roles["#mount"] = AnchorRole_Storage;
432 roles["#attach"] = AnchorRole_Storage;
433
434 /* Current anchor role: */
435 const QString strRole = strAnchor.section(',', 0, 0);
436 const QString strData = strAnchor.section(',', 1);
437
438 /* Handle known anchor roles: */
439 const AnchorRole enmRole = roles.value(strRole, AnchorRole_Invalid);
440 switch (enmRole)
441 {
442 case AnchorRole_MachineName:
443 case AnchorRole_OSType:
444 {
445 /* Prepare popup: */
446 QPointer<QIDialogContainer> pPopup = new QIDialogContainer(0, Qt::Popup);
447 if (pPopup)
448 {
449 /* Prepare editor: */
450 UINameAndSystemEditor *pEditor =
451 new UINameAndSystemEditor(pPopup,
452 enmRole == AnchorRole_MachineName /* choose name? */,
453 false /* choose path? */,
454 enmRole == AnchorRole_OSType /* choose type? */);
455 if (pEditor)
456 {
457 switch (enmRole)
458 {
459 case AnchorRole_MachineName: pEditor->setName(strData.section(',', 0, 0)); break;
460 case AnchorRole_OSType: pEditor->setTypeId(strData.section(',', 0, 0)); break;
461 default: break;
462 }
463
464 /* Add to popup: */
465 pPopup->setWidget(pEditor);
466 }
467
468 /* Adjust popup geometry: */
469 pPopup->move(QCursor::pos());
470 pPopup->adjustSize();
471
472 /* Execute popup, change machine name if confirmed: */
473 if (pPopup->exec() == QDialog::Accepted)
474 {
475 switch (enmRole)
476 {
477 case AnchorRole_MachineName: setMachineAttribute(machine(), MachineAttribute_Name, QVariant::fromValue(pEditor->name())); break;
478 case AnchorRole_OSType: setMachineAttribute(machine(), MachineAttribute_OSType, QVariant::fromValue(pEditor->typeId())); break;
479 default: break;
480 }
481 }
482
483 /* Delete popup: */
484 delete pPopup;
485 }
486 break;
487 }
488 case AnchorRole_Storage:
489 {
490 /* Prepare storage-menu: */
491 UIMenu menu;
492 menu.setShowToolTip(true);
493
494 /* Storage-controller name: */
495 QString strControllerName = strData.section(',', 0, 0);
496 /* Storage-slot: */
497 StorageSlot storageSlot = gpConverter->fromString<StorageSlot>(strData.section(',', 1));
498
499 /* Fill storage-menu: */
500 uiCommon().prepareStorageMenu(menu, this, SLOT(sltMountStorageMedium()),
501 machine(), strControllerName, storageSlot);
502
503 /* Exec menu: */
504 menu.exec(QCursor::pos());
505 break;
506 }
507 default:
508 break;
509 }
510}
511
512void UIDetailsElement::sltMountStorageMedium()
513{
514 /* Sender action: */
515 QAction *pAction = qobject_cast<QAction*>(sender());
516 AssertMsgReturnVoid(pAction, ("This slot should only be called by menu action!\n"));
517
518 /* Current mount-target: */
519 const UIMediumTarget target = pAction->data().value<UIMediumTarget>();
520
521 /* Update current machine mount-target: */
522 uiCommon().updateMachineStorage(machine(), target);
523}
524
525void UIDetailsElement::prepareElement()
526{
527 /* Initialization: */
528 m_nameFont = font();
529 m_nameFont.setWeight(QFont::Bold);
530 m_textFont = font();
531
532 /* Update icon: */
533 updateIcon();
534
535 /* Create hovering animation machine: */
536 m_pHoveringMachine = new QStateMachine(this);
537 if (m_pHoveringMachine)
538 {
539 /* Create 'default' state: */
540 QState *pStateDefault = new QState(m_pHoveringMachine);
541 /* Create 'hovered' state: */
542 QState *pStateHovered = new QState(m_pHoveringMachine);
543
544 /* Configure 'default' state: */
545 if (pStateDefault)
546 {
547 /* When we entering default state => we assigning animatedValue to m_iDefaultValue: */
548 pStateDefault->assignProperty(this, "animatedValue", m_iDefaultValue);
549
550 /* Add state transition: */
551 QSignalTransition *pDefaultToHovered = pStateDefault->addTransition(this, SIGNAL(sigHoverEnter()), pStateHovered);
552 if (pDefaultToHovered)
553 {
554 /* Create forward animation: */
555 m_pHoveringAnimationForward = new QPropertyAnimation(this, "animatedValue", this);
556 if (m_pHoveringAnimationForward)
557 {
558 m_pHoveringAnimationForward->setDuration(m_iAnimationDuration);
559 m_pHoveringAnimationForward->setStartValue(m_iDefaultValue);
560 m_pHoveringAnimationForward->setEndValue(m_iHoveredValue);
561
562 /* Add to transition: */
563 pDefaultToHovered->addAnimation(m_pHoveringAnimationForward);
564 }
565 }
566 }
567
568 /* Configure 'hovered' state: */
569 if (pStateHovered)
570 {
571 /* When we entering hovered state => we assigning animatedValue to m_iHoveredValue: */
572 pStateHovered->assignProperty(this, "animatedValue", m_iHoveredValue);
573
574 /* Add state transition: */
575 QSignalTransition *pHoveredToDefault = pStateHovered->addTransition(this, SIGNAL(sigHoverLeave()), pStateDefault);
576 if (pHoveredToDefault)
577 {
578 /* Create backward animation: */
579 m_pHoveringAnimationBackward = new QPropertyAnimation(this, "animatedValue", this);
580 if (m_pHoveringAnimationBackward)
581 {
582 m_pHoveringAnimationBackward->setDuration(m_iAnimationDuration);
583 m_pHoveringAnimationBackward->setStartValue(m_iHoveredValue);
584 m_pHoveringAnimationBackward->setEndValue(m_iDefaultValue);
585
586 /* Add to transition: */
587 pHoveredToDefault->addAnimation(m_pHoveringAnimationBackward);
588 }
589 }
590 }
591
592 /* Initial state is 'default': */
593 m_pHoveringMachine->setInitialState(pStateDefault);
594 /* Start state-machine: */
595 m_pHoveringMachine->start();
596 }
597
598 /* Configure connections: */
599 connect(gpManager, &UIVirtualBoxManager::sigWindowRemapped,
600 this, &UIDetailsElement::sltHandleWindowRemapped);
601 connect(this, SIGNAL(sigToggleElement(DetailsElementType, bool)),
602 model(), SLOT(sltToggleElements(DetailsElementType, bool)));
603 connect(this, SIGNAL(sigLinkClicked(const QString&, const QString&, const QUuid &)),
604 model(), SIGNAL(sigLinkClicked(const QString&, const QString&, const QUuid &)));
605}
606
607void UIDetailsElement::prepareButton()
608{
609 /* Setup toggle-button: */
610 m_pButton = new UIGraphicsRotatorButton(this, "additionalHeight", !m_fClosed, true /* reflected */);
611 m_pButton->setAutoHandleButtonClick(false);
612 connect(m_pButton, SIGNAL(sigButtonClicked()), this, SLOT(sltToggleButtonClicked()));
613 connect(m_pButton, SIGNAL(sigRotationStart()), this, SLOT(sltElementToggleStart()));
614 connect(m_pButton, SIGNAL(sigRotationFinish(bool)), this, SLOT(sltElementToggleFinish(bool)));
615 m_buttonSize = m_pButton->minimumSizeHint().toSize();
616}
617
618void UIDetailsElement::prepareTextPane()
619{
620 /* Create text-pane: */
621 m_pTextPane = new UIGraphicsTextPane(this, model()->paintDevice());
622 connect(m_pTextPane, SIGNAL(sigGeometryChanged()), this, SLOT(sltUpdateGeometry()));
623 connect(m_pTextPane, SIGNAL(sigAnchorClicked(const QString&)), this, SLOT(sltHandleAnchorClicked(const QString&)));
624}
625
626void UIDetailsElement::updateIcon()
627{
628 /* Prepare whole icon first of all: */
629 const QIcon icon = gpConverter->toIcon(elementType());
630
631 /* Cache icon: */
632 if (icon.isNull())
633 {
634 /* No icon provided: */
635 m_pixmapSize = QSize();
636 m_pixmap = QPixmap();
637 }
638 else
639 {
640 /* Determine default icon size: */
641 const int iIconMetric = QApplication::style()->pixelMetric(QStyle::PM_SmallIconSize);
642 m_pixmapSize = QSize(iIconMetric, iIconMetric);
643 /* Acquire the icon of corresponding size (taking top-level widget DPI into account): */
644 m_pixmap = icon.pixmap(gpManager->windowHandle(), m_pixmapSize);
645 }
646
647 /* Update linked values: */
648 updateMinimumHeaderWidth();
649 updateMinimumHeaderHeight();
650}
651
652void UIDetailsElement::handleHoverEvent(QGraphicsSceneHoverEvent *pEvent)
653{
654 /* Not for 'preview' element type: */
655 if (m_enmType == DetailsElementType_Preview)
656 return;
657
658 /* Prepare variables: */
659 int iMargin = data(ElementData_Margin).toInt();
660 int iSpacing = data(ElementData_Spacing).toInt();
661 int iNameHeight = m_nameSize.height();
662 int iElementNameX = 2 * iMargin + m_pixmapSize.width() + iSpacing;
663 int iElementNameY = iNameHeight == m_iMinimumHeaderHeight ?
664 iMargin : iMargin + (m_iMinimumHeaderHeight - iNameHeight) / 2;
665
666 /* Simulate hyperlink hovering: */
667 QPoint point = pEvent->pos().toPoint();
668 bool fNameHovered = QRect(QPoint(iElementNameX, iElementNameY), m_nameSize).contains(point);
669 if ( m_pSet->configurationAccessLevel() != ConfigurationAccessLevel_Null
670 && m_fNameHovered != fNameHovered)
671 {
672 m_fNameHovered = fNameHovered;
673 updateNameHoverLink();
674 }
675}
676
677void UIDetailsElement::updateNameHoverLink()
678{
679 if (m_fNameHovered)
680 UICommon::setCursor(this, Qt::PointingHandCursor);
681 else
682 UICommon::unsetCursor(this);
683 update();
684}
685
686void UIDetailsElement::updateAnimationParameters()
687{
688 /* Recalculate animation parameters: */
689 int iOpenedHeight = minimumHeightHintForElement(false);
690 int iClosedHeight = minimumHeightHintForElement(true);
691 int iAdditionalHeight = iOpenedHeight - iClosedHeight;
692 if (m_fClosed)
693 m_iAdditionalHeight = 0;
694 else
695 m_iAdditionalHeight = iAdditionalHeight;
696 m_pButton->setAnimationRange(0, iAdditionalHeight);
697}
698
699void UIDetailsElement::updateButtonVisibility()
700{
701 if (m_fHovered && !m_pButton->isVisible())
702 m_pButton->show();
703 else if (!m_fHovered && m_pButton->isVisible())
704 m_pButton->hide();
705}
706
707void UIDetailsElement::updateMinimumHeaderWidth()
708{
709 /* Prepare variables: */
710 int iSpacing = data(ElementData_Spacing).toInt();
711
712 /* Update minimum-header-width: */
713 m_iMinimumHeaderWidth = m_pixmapSize.width() +
714 iSpacing + m_nameSize.width() +
715 iSpacing + m_buttonSize.width();
716}
717
718void UIDetailsElement::updateMinimumHeaderHeight()
719{
720 /* Update minimum-header-height: */
721 m_iMinimumHeaderHeight = qMax(m_pixmapSize.height(), m_nameSize.height());
722 m_iMinimumHeaderHeight = qMax(m_iMinimumHeaderHeight, m_buttonSize.height());
723}
724
725void UIDetailsElement::paintBackground(QPainter *pPainter, const QStyleOptionGraphicsItem *pOptions) const
726{
727 /* Save painter: */
728 pPainter->save();
729
730 /* Prepare variables: */
731 const int iMargin = data(ElementData_Margin).toInt();
732 const int iHeadHeight = 2 * iMargin + m_iMinimumHeaderHeight;
733 const QRect optionRect = pOptions->rect;
734 const QRect headRect = QRect(optionRect.topLeft(), QSize(optionRect.width(), iHeadHeight));
735 const QRect fullRect = m_fAnimationRunning
736 ? QRect(optionRect.topLeft(), QSize(optionRect.width(), iHeadHeight + m_iAdditionalHeight))
737 : optionRect;
738
739 /* Acquire palette: */
740 const QPalette pal = palette();
741
742 /* Paint default background: */
743 const QColor defaultColor = pal.color(QPalette::Active, QPalette::Mid);
744 const QColor dcTone1 = defaultColor.lighter(m_iDefaultToneFinal);
745 const QColor dcTone2 = defaultColor.lighter(m_iDefaultToneStart);
746 QLinearGradient gradientDefault(fullRect.topLeft(), fullRect.bottomLeft());
747 gradientDefault.setColorAt(0, dcTone1);
748 gradientDefault.setColorAt(1, dcTone2);
749 pPainter->fillRect(fullRect, gradientDefault);
750
751 /* If element is hovered: */
752 if (m_fHovered)
753 {
754 /* Paint hovered background: */
755 const QColor hoveredColor = pal.color(QPalette::Active, QPalette::Highlight);
756 QColor hcTone1 = hoveredColor.lighter(m_iHoverToneFinal);
757 QColor hcTone2 = hoveredColor.lighter(m_iHoverToneStart);
758 hcTone1.setAlpha(m_iAnimatedValue);
759 hcTone2.setAlpha(m_iAnimatedValue);
760 QLinearGradient gradientHovered(headRect.topLeft(), headRect.bottomLeft());
761 gradientHovered.setColorAt(0, hcTone1);
762 gradientHovered.setColorAt(1, hcTone2);
763 pPainter->fillRect(headRect, gradientHovered);
764 }
765
766 /* Restore painter: */
767 pPainter->restore();
768}
769
770void UIDetailsElement::paintFrame(QPainter *pPainter, const QStyleOptionGraphicsItem *pOptions) const
771{
772 /* Save painter: */
773 pPainter->save();
774
775 /* Prepare variables: */
776 const int iMargin = data(ElementData_Margin).toInt();
777 const int iHeadHeight = 2 * iMargin + m_iMinimumHeaderHeight;
778 const QRect optionRect = pOptions->rect;
779 const QRect rectangle = m_fAnimationRunning
780 ? QRect(optionRect.topLeft(), QSize(optionRect.width(), iHeadHeight + m_iAdditionalHeight))
781 : optionRect;
782
783 /* Paint frame: */
784 const QColor strokeColor = palette().color(QPalette::Active, QPalette::Mid).lighter(m_iDefaultToneStart);
785 QPen pen(strokeColor);
786 pen.setWidth(0);
787 pPainter->setPen(pen);
788 pPainter->drawLine(rectangle.topLeft(), rectangle.topRight());
789 pPainter->drawLine(rectangle.bottomLeft(), rectangle.bottomRight());
790 pPainter->drawLine(rectangle.topLeft(), rectangle.bottomLeft());
791 pPainter->drawLine(rectangle.topRight(), rectangle.bottomRight());
792
793 /* Restore painter: */
794 pPainter->restore();
795}
796
797void UIDetailsElement::paintElementInfo(QPainter *pPainter, const QStyleOptionGraphicsItem *) const
798{
799 /* Initialize some necessary variables: */
800 const int iMargin = data(ElementData_Margin).toInt();
801 const int iSpacing = data(ElementData_Spacing).toInt();
802
803 /* Calculate attributes: */
804 const int iPixmapHeight = m_pixmapSize.height();
805 const int iNameHeight = m_nameSize.height();
806 const int iMaximumHeight = qMax(iPixmapHeight, iNameHeight);
807
808 /* Prepare color: */
809 const QPalette pal = palette();
810 const QColor buttonTextColor = pal.color(QPalette::Active, QPalette::ButtonText);
811 const QColor linkTextColor = pal.color(QPalette::Active, QPalette::Link);
812
813 /* Paint pixmap: */
814 int iElementPixmapX = 2 * iMargin;
815 int iElementPixmapY = iPixmapHeight == iMaximumHeight ?
816 iMargin : iMargin + (iMaximumHeight - iPixmapHeight) / 2;
817 paintPixmap(/* Painter: */
818 pPainter,
819 /* Rectangle to paint in: */
820 QRect(QPoint(iElementPixmapX, iElementPixmapY), m_pixmapSize),
821 /* Pixmap to paint: */
822 m_pixmap);
823
824 /* Paint name: */
825 int iMachineNameX = iElementPixmapX +
826 m_pixmapSize.width() +
827 iSpacing;
828 int iMachineNameY = iNameHeight == iMaximumHeight ?
829 iMargin : iMargin + (iMaximumHeight - iNameHeight) / 2;
830 paintText(/* Painter: */
831 pPainter,
832 /* Rectangle to paint in: */
833 QPoint(iMachineNameX, iMachineNameY),
834 /* Font to paint text: */
835 m_nameFont,
836 /* Paint device: */
837 model()->paintDevice(),
838 /* Text to paint: */
839 m_strName,
840 /* Name hovered? */
841 m_fNameHovered ? linkTextColor : buttonTextColor);
842}
843
844/* static */
845void UIDetailsElement::paintPixmap(QPainter *pPainter, const QRect &rect, const QPixmap &pixmap)
846{
847 pPainter->drawPixmap(rect, pixmap);
848}
849
850/* static */
851void UIDetailsElement::paintText(QPainter *pPainter, QPoint point,
852 const QFont &font, QPaintDevice *pPaintDevice,
853 const QString &strText, const QColor &color)
854{
855 /* Prepare variables: */
856 QFontMetrics fm(font, pPaintDevice);
857 point += QPoint(0, fm.ascent());
858
859 /* Draw text: */
860 pPainter->save();
861 pPainter->setFont(font);
862 pPainter->setPen(color);
863 pPainter->drawText(point, strText);
864 pPainter->restore();
865}
Note: See TracBrowser for help on using the repository browser.

© 2025 Oracle Support Privacy / Do Not Sell My Info Terms of Use Trademark Policy Automated Access Etiquette