1 | /* $Id: UIMachineView.cpp 106061 2024-09-16 14:03:52Z vboxsync $ */
|
---|
2 | /** @file
|
---|
3 | * VBox Qt GUI - UIMachineView class implementation.
|
---|
4 | */
|
---|
5 |
|
---|
6 | /*
|
---|
7 | * Copyright (C) 2010-2024 Oracle and/or its affiliates.
|
---|
8 | *
|
---|
9 | * This file is part of VirtualBox base platform packages, as
|
---|
10 | * available from https://www.virtualbox.org.
|
---|
11 | *
|
---|
12 | * This program is free software; you can redistribute it and/or
|
---|
13 | * modify it under the terms of the GNU General Public License
|
---|
14 | * as published by the Free Software Foundation, in version 3 of the
|
---|
15 | * License.
|
---|
16 | *
|
---|
17 | * This program is distributed in the hope that it will be useful, but
|
---|
18 | * WITHOUT ANY WARRANTY; without even the implied warranty of
|
---|
19 | * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
---|
20 | * General Public License for more details.
|
---|
21 | *
|
---|
22 | * You should have received a copy of the GNU General Public License
|
---|
23 | * along with this program; if not, see <https://www.gnu.org/licenses>.
|
---|
24 | *
|
---|
25 | * SPDX-License-Identifier: GPL-3.0-only
|
---|
26 | */
|
---|
27 |
|
---|
28 | /* Qt includes: */
|
---|
29 | #include <QAbstractNativeEventFilter>
|
---|
30 | #include <QAccessibleWidget>
|
---|
31 | #include <QApplication>
|
---|
32 | #include <QBitmap>
|
---|
33 | #include <QCoreApplication>
|
---|
34 | #include <QMainWindow>
|
---|
35 | #include <QPainter>
|
---|
36 | #include <QScrollBar>
|
---|
37 | #include <QTimer>
|
---|
38 | #include <QWindowStateChangeEvent>
|
---|
39 |
|
---|
40 | /* GUI includes: */
|
---|
41 | #include "UICommon.h"
|
---|
42 | #include "UIActionPoolRuntime.h"
|
---|
43 | #include "UIDesktopWidgetWatchdog.h"
|
---|
44 | #include "UIExtraDataManager.h"
|
---|
45 | #include "UIFrameBuffer.h"
|
---|
46 | #include "UIKeyboardHandler.h"
|
---|
47 | #include "UILoggingDefs.h"
|
---|
48 | #include "UIMachine.h"
|
---|
49 | #include "UIMessageCenter.h"
|
---|
50 | #include "UIMachineLogic.h"
|
---|
51 | #include "UIMachineWindow.h"
|
---|
52 | #include "UIMachineViewNormal.h"
|
---|
53 | #include "UIMachineViewFullscreen.h"
|
---|
54 | #include "UIMachineViewSeamless.h"
|
---|
55 | #include "UIMachineViewScale.h"
|
---|
56 | #include "UIMouseHandler.h"
|
---|
57 | #include "UINotificationCenter.h"
|
---|
58 | #include "UITranslationEventListener.h"
|
---|
59 | #ifdef VBOX_WS_MAC
|
---|
60 | # include "UICocoaApplication.h"
|
---|
61 | # include "DarwinKeyboard.h"
|
---|
62 | # include "DockIconPreview.h"
|
---|
63 | #endif
|
---|
64 | #ifdef VBOX_WITH_DRAG_AND_DROP
|
---|
65 | # include "UIDnDHandler.h"
|
---|
66 | #endif
|
---|
67 |
|
---|
68 | /* VirtualBox interface declarations: */
|
---|
69 | #include <VBox/com/VirtualBox.h>
|
---|
70 |
|
---|
71 | /* COM includes: */
|
---|
72 | #include "CGraphicsAdapter.h"
|
---|
73 | #include "CSession.h"
|
---|
74 | #include "CFramebuffer.h"
|
---|
75 | #ifdef VBOX_WITH_DRAG_AND_DROP
|
---|
76 | # include "CDnDSource.h"
|
---|
77 | # include "CDnDTarget.h"
|
---|
78 | # include "CGuest.h"
|
---|
79 | # include "CGuestDnDSource.h"
|
---|
80 | # include "CGuestDnDTarget.h"
|
---|
81 | #endif
|
---|
82 |
|
---|
83 | /* Other VBox includes: */
|
---|
84 | #include <VBox/VBoxOGL.h>
|
---|
85 | #include <VBoxVideo.h>
|
---|
86 |
|
---|
87 | /* External includes: */
|
---|
88 | #include <math.h>
|
---|
89 | #ifdef VBOX_WITH_DRAG_AND_DROP
|
---|
90 | # include <new> /* For bad_alloc. */
|
---|
91 | #endif
|
---|
92 | #ifdef VBOX_WS_MAC
|
---|
93 | # include <Carbon/Carbon.h>
|
---|
94 | #endif
|
---|
95 | #ifdef VBOX_WS_NIX
|
---|
96 | # include <xcb/xcb.h>
|
---|
97 | #endif
|
---|
98 |
|
---|
99 | #ifdef DEBUG_andy
|
---|
100 | /* Macro for debugging drag and drop actions which usually would
|
---|
101 | * go to Main's logging group. */
|
---|
102 | # define DNDDEBUG(x) LogFlowFunc(x)
|
---|
103 | #else
|
---|
104 | # define DNDDEBUG(x)
|
---|
105 | #endif
|
---|
106 |
|
---|
107 |
|
---|
108 | /** QAccessibleWidget extension used as an accessibility interface for Machine-view. */
|
---|
109 | class UIAccessibilityInterfaceForUIMachineView : public QAccessibleWidget
|
---|
110 | {
|
---|
111 | public:
|
---|
112 |
|
---|
113 | /** Returns an accessibility interface for passed @a strClassname and @a pObject. */
|
---|
114 | static QAccessibleInterface *pFactory(const QString &strClassname, QObject *pObject)
|
---|
115 | {
|
---|
116 | /* Creating Machine-view accessibility interface: */
|
---|
117 | if (pObject && strClassname == QLatin1String("UIMachineView"))
|
---|
118 | return new UIAccessibilityInterfaceForUIMachineView(qobject_cast<QWidget*>(pObject));
|
---|
119 |
|
---|
120 | /* Null by default: */
|
---|
121 | return 0;
|
---|
122 | }
|
---|
123 |
|
---|
124 | /** Constructs an accessibility interface passing @a pWidget to the base-class. */
|
---|
125 | UIAccessibilityInterfaceForUIMachineView(QWidget *pWidget)
|
---|
126 | : QAccessibleWidget(pWidget, QAccessible::Canvas)
|
---|
127 | {}
|
---|
128 |
|
---|
129 | /** Returns the number of children. */
|
---|
130 | virtual int childCount() const RT_OVERRIDE
|
---|
131 | {
|
---|
132 | /* Make sure view still alive: */
|
---|
133 | AssertPtrReturn(view(), 0);
|
---|
134 |
|
---|
135 | /* Zero by default: */
|
---|
136 | return 0;
|
---|
137 | }
|
---|
138 |
|
---|
139 | /** Returns the child with the passed @a iIndex. */
|
---|
140 | virtual QAccessibleInterface *child(int iIndex) const RT_OVERRIDE
|
---|
141 | {
|
---|
142 | /* Make sure view still alive: */
|
---|
143 | AssertPtrReturn(view(), 0);
|
---|
144 | /* Make sure index is valid: */
|
---|
145 | AssertReturn(iIndex >= 0 && iIndex < childCount(), 0);
|
---|
146 |
|
---|
147 | /* Null by default: */
|
---|
148 | return 0;
|
---|
149 | }
|
---|
150 |
|
---|
151 | /** Returns the index of passed @a pChild. */
|
---|
152 | virtual int indexOfChild(const QAccessibleInterface *pChild) const RT_OVERRIDE
|
---|
153 | {
|
---|
154 | /* Make sure view still alive: */
|
---|
155 | AssertPtrReturn(view(), -1);
|
---|
156 | /* Make sure child is valid: */
|
---|
157 | AssertReturn(pChild, -1);
|
---|
158 |
|
---|
159 | /* -1 by default: */
|
---|
160 | return -1;;
|
---|
161 | }
|
---|
162 |
|
---|
163 | /** Returns a text for the passed @a enmTextRole. */
|
---|
164 | virtual QString text(QAccessible::Text enmTextRole) const RT_OVERRIDE
|
---|
165 | {
|
---|
166 | /* Make sure view still alive: */
|
---|
167 | AssertPtrReturn(view(), QString());
|
---|
168 |
|
---|
169 | /* Gather suitable text: */
|
---|
170 | Q_UNUSED(enmTextRole);
|
---|
171 | QString strText = view()->toolTip();
|
---|
172 | if (strText.isEmpty())
|
---|
173 | strText = view()->whatsThis();
|
---|
174 | return strText;
|
---|
175 | }
|
---|
176 |
|
---|
177 | private:
|
---|
178 |
|
---|
179 | /** Returns corresponding Machine-view. */
|
---|
180 | UIMachineView *view() const { return qobject_cast<UIMachineView*>(widget()); }
|
---|
181 | };
|
---|
182 |
|
---|
183 |
|
---|
184 | /** QAbstractNativeEventFilter extension
|
---|
185 | * allowing to pre-process native platform events. */
|
---|
186 | class UINativeEventFilter : public QAbstractNativeEventFilter
|
---|
187 | {
|
---|
188 | public:
|
---|
189 |
|
---|
190 | /** Constructs native event filter storing @a pParent to redirect events to. */
|
---|
191 | UINativeEventFilter(UIMachineView *pParent)
|
---|
192 | : m_pParent(pParent)
|
---|
193 | {}
|
---|
194 |
|
---|
195 | /** Redirects all the native events to parent. */
|
---|
196 | bool nativeEventFilter(const QByteArray &eventType, void *pMessage, qintptr*) RT_OVERRIDE RT_FINAL
|
---|
197 | {
|
---|
198 | return m_pParent->nativeEventPreprocessor(eventType, pMessage);
|
---|
199 | }
|
---|
200 |
|
---|
201 | private:
|
---|
202 |
|
---|
203 | /** Holds the passed parent reference. */
|
---|
204 | UIMachineView *m_pParent;
|
---|
205 | };
|
---|
206 |
|
---|
207 |
|
---|
208 | /* static */
|
---|
209 | UIMachineView* UIMachineView::create(UIMachineWindow *pMachineWindow, ulong uScreenId, UIVisualStateType visualStateType)
|
---|
210 | {
|
---|
211 | UIMachineView *pMachineView = 0;
|
---|
212 | switch (visualStateType)
|
---|
213 | {
|
---|
214 | case UIVisualStateType_Normal:
|
---|
215 | pMachineView = new UIMachineViewNormal(pMachineWindow, uScreenId);
|
---|
216 | break;
|
---|
217 | case UIVisualStateType_Fullscreen:
|
---|
218 | pMachineView = new UIMachineViewFullscreen(pMachineWindow, uScreenId);
|
---|
219 | break;
|
---|
220 | case UIVisualStateType_Seamless:
|
---|
221 | pMachineView = new UIMachineViewSeamless(pMachineWindow, uScreenId);
|
---|
222 | break;
|
---|
223 | case UIVisualStateType_Scale:
|
---|
224 | pMachineView = new UIMachineViewScale(pMachineWindow, uScreenId);
|
---|
225 | break;
|
---|
226 | default:
|
---|
227 | break;
|
---|
228 | }
|
---|
229 |
|
---|
230 | /* Load machine-view settings: */
|
---|
231 | pMachineView->loadMachineViewSettings();
|
---|
232 |
|
---|
233 | /* Prepare viewport: */
|
---|
234 | pMachineView->prepareViewport();
|
---|
235 |
|
---|
236 | /* Prepare frame-buffer: */
|
---|
237 | pMachineView->prepareFrameBuffer();
|
---|
238 |
|
---|
239 | /* Prepare common things: */
|
---|
240 | pMachineView->prepareCommon();
|
---|
241 |
|
---|
242 | #ifdef VBOX_WITH_DRAG_AND_DROP
|
---|
243 | /* Prepare DnD: */
|
---|
244 | /* rc ignored */ pMachineView->prepareDnd();
|
---|
245 | #endif
|
---|
246 |
|
---|
247 | /* Prepare event-filters: */
|
---|
248 | pMachineView->prepareFilters();
|
---|
249 |
|
---|
250 | /* Prepare connections: */
|
---|
251 | pMachineView->prepareConnections();
|
---|
252 |
|
---|
253 | /* Prepare console connections: */
|
---|
254 | pMachineView->prepareConsoleConnections();
|
---|
255 |
|
---|
256 | /* Initialization: */
|
---|
257 | pMachineView->sltMachineStateChanged();
|
---|
258 | /** @todo Can we move the call to sltAdditionsStateChanged() from the
|
---|
259 | * subclass constructors here too? It is called for Normal and Seamless,
|
---|
260 | * but not for Fullscreen and Scale. However for Scale it is a no op.,
|
---|
261 | * so it would not hurt. Would it hurt for fullscreen? */
|
---|
262 |
|
---|
263 | /* Set a preliminary maximum size: */
|
---|
264 | pMachineView->setMaximumGuestSize();
|
---|
265 |
|
---|
266 | /* Resend the last resize hint finally: */
|
---|
267 | pMachineView->resendSizeHint();
|
---|
268 |
|
---|
269 | /* Return the created view: */
|
---|
270 | return pMachineView;
|
---|
271 | }
|
---|
272 |
|
---|
273 | /* static */
|
---|
274 | void UIMachineView::destroy(UIMachineView *pMachineView)
|
---|
275 | {
|
---|
276 | if (!pMachineView)
|
---|
277 | return;
|
---|
278 |
|
---|
279 | #ifdef VBOX_WITH_DRAG_AND_DROP
|
---|
280 | /* Cleanup DnD: */
|
---|
281 | pMachineView->cleanupDnd();
|
---|
282 | #endif
|
---|
283 |
|
---|
284 | /* Cleanup frame-buffer: */
|
---|
285 | pMachineView->cleanupFrameBuffer();
|
---|
286 |
|
---|
287 | /* Cleanup native filters: */
|
---|
288 | pMachineView->cleanupNativeFilters();
|
---|
289 |
|
---|
290 | delete pMachineView;
|
---|
291 | }
|
---|
292 |
|
---|
293 | void UIMachineView::applyMachineViewScaleFactor()
|
---|
294 | {
|
---|
295 | /* Sanity check: */
|
---|
296 | if (!frameBuffer())
|
---|
297 | return;
|
---|
298 |
|
---|
299 | /* Acquire selected scale-factor: */
|
---|
300 | double dScaleFactor = gEDataManager->scaleFactor(uiCommon().managedVMUuid(), m_uScreenId);
|
---|
301 |
|
---|
302 | /* Take the device-pixel-ratio into account: */
|
---|
303 | frameBuffer()->setDevicePixelRatio(UIDesktopWidgetWatchdog::devicePixelRatio(machineWindow()));
|
---|
304 | frameBuffer()->setDevicePixelRatioActual(UIDesktopWidgetWatchdog::devicePixelRatioActual(machineWindow()));
|
---|
305 | const double dDevicePixelRatioActual = frameBuffer()->devicePixelRatioActual();
|
---|
306 | const bool fUseUnscaledHiDPIOutput = dScaleFactor != dDevicePixelRatioActual;
|
---|
307 | dScaleFactor = fUseUnscaledHiDPIOutput ? dScaleFactor : 1.0;
|
---|
308 |
|
---|
309 | /* Assign frame-buffer with new values: */
|
---|
310 | frameBuffer()->setScaleFactor(dScaleFactor);
|
---|
311 | frameBuffer()->setUseUnscaledHiDPIOutput(fUseUnscaledHiDPIOutput);
|
---|
312 |
|
---|
313 | /* Propagate the scale-factor related attributes to 3D service if necessary: */
|
---|
314 | bool fAccelerate3DEnabled = false;
|
---|
315 | uimachine()->acquireWhetherAccelerate3DEnabled(fAccelerate3DEnabled);
|
---|
316 | if (fAccelerate3DEnabled)
|
---|
317 | {
|
---|
318 | double dScaleFactorFor3D = dScaleFactor;
|
---|
319 | #if defined(VBOX_WS_WIN) || defined(VBOX_WS_NIX)
|
---|
320 | // WORKAROUND:
|
---|
321 | // On Windows and Linux opposing to macOS it's only Qt which can auto scale up,
|
---|
322 | // not 3D overlay itself, so for auto scale-up mode we have to take that into account.
|
---|
323 | if (!fUseUnscaledHiDPIOutput)
|
---|
324 | dScaleFactorFor3D *= dDevicePixelRatioActual;
|
---|
325 | #endif /* VBOX_WS_WIN || VBOX_WS_NIX */
|
---|
326 | uimachine()->notifyScaleFactorChange(m_uScreenId,
|
---|
327 | (uint32_t)(dScaleFactorFor3D * VBOX_OGL_SCALE_FACTOR_MULTIPLIER),
|
---|
328 | (uint32_t)(dScaleFactorFor3D * VBOX_OGL_SCALE_FACTOR_MULTIPLIER));
|
---|
329 | uimachine()->notifyHiDPIOutputPolicyChange(fUseUnscaledHiDPIOutput);
|
---|
330 | }
|
---|
331 |
|
---|
332 | /* Perform frame-buffer rescaling: */
|
---|
333 | frameBuffer()->performRescale();
|
---|
334 |
|
---|
335 | /* Update console's display viewport and 3D overlay: */
|
---|
336 | updateViewport();
|
---|
337 | }
|
---|
338 |
|
---|
339 | UIMachine *UIMachineView::uimachine() const
|
---|
340 | {
|
---|
341 | return machineWindow()->uimachine();
|
---|
342 | }
|
---|
343 |
|
---|
344 | UIMachineLogic *UIMachineView::machineLogic() const
|
---|
345 | {
|
---|
346 | return machineWindow()->machineLogic();
|
---|
347 | }
|
---|
348 |
|
---|
349 | UIFrameBuffer *UIMachineView::frameBuffer() const
|
---|
350 | {
|
---|
351 | return uimachine()->frameBuffer(m_uScreenId);
|
---|
352 | }
|
---|
353 |
|
---|
354 | int UIMachineView::contentsWidth() const
|
---|
355 | {
|
---|
356 | return frameBuffer()->width();
|
---|
357 | }
|
---|
358 |
|
---|
359 | int UIMachineView::contentsHeight() const
|
---|
360 | {
|
---|
361 | return frameBuffer()->height();
|
---|
362 | }
|
---|
363 |
|
---|
364 | int UIMachineView::contentsX() const
|
---|
365 | {
|
---|
366 | return horizontalScrollBar()->value();
|
---|
367 | }
|
---|
368 |
|
---|
369 | int UIMachineView::contentsY() const
|
---|
370 | {
|
---|
371 | return verticalScrollBar()->value();
|
---|
372 | }
|
---|
373 |
|
---|
374 | int UIMachineView::visibleWidth() const
|
---|
375 | {
|
---|
376 | return horizontalScrollBar()->pageStep();
|
---|
377 | }
|
---|
378 |
|
---|
379 | int UIMachineView::visibleHeight() const
|
---|
380 | {
|
---|
381 | return verticalScrollBar()->pageStep();
|
---|
382 | }
|
---|
383 |
|
---|
384 | QPoint UIMachineView::viewportToContents(const QPoint &viewportPoint) const
|
---|
385 | {
|
---|
386 | /* Get physical contents shifts of scroll-bars: */
|
---|
387 | int iContentsX = contentsX();
|
---|
388 | int iContentsY = contentsY();
|
---|
389 |
|
---|
390 | /* Take the device-pixel-ratio into account: */
|
---|
391 | const double dDevicePixelRatioFormal = frameBuffer()->devicePixelRatio();
|
---|
392 | const double dDevicePixelRatioActual = frameBuffer()->devicePixelRatioActual();
|
---|
393 | if (!frameBuffer()->useUnscaledHiDPIOutput())
|
---|
394 | {
|
---|
395 | iContentsX *= dDevicePixelRatioActual;
|
---|
396 | iContentsY *= dDevicePixelRatioActual;
|
---|
397 | }
|
---|
398 | iContentsX /= dDevicePixelRatioFormal;
|
---|
399 | iContentsY /= dDevicePixelRatioFormal;
|
---|
400 |
|
---|
401 | /* Return point shifted according scroll-bars: */
|
---|
402 | return QPoint(viewportPoint.x() + iContentsX, viewportPoint.y() + iContentsY);
|
---|
403 | }
|
---|
404 |
|
---|
405 | void UIMachineView::scrollBy(int iDx, int iDy)
|
---|
406 | {
|
---|
407 | horizontalScrollBar()->setValue(horizontalScrollBar()->value() + iDx);
|
---|
408 | verticalScrollBar()->setValue(verticalScrollBar()->value() + iDy);
|
---|
409 | }
|
---|
410 |
|
---|
411 | UIVisualStateType UIMachineView::visualStateType() const
|
---|
412 | {
|
---|
413 | return machineLogic()->visualStateType();
|
---|
414 | }
|
---|
415 |
|
---|
416 | double UIMachineView::aspectRatio() const
|
---|
417 | {
|
---|
418 | return frameBuffer() ? (double)(frameBuffer()->width()) / frameBuffer()->height() : 0;
|
---|
419 | }
|
---|
420 |
|
---|
421 | void UIMachineView::setMaximumGuestSize(const QSize &minimumSizeHint /* = QSize() */)
|
---|
422 | {
|
---|
423 | QSize maxSize;
|
---|
424 | switch (m_enmMaximumGuestScreenSizePolicy)
|
---|
425 | {
|
---|
426 | case MaximumGuestScreenSizePolicy_Fixed:
|
---|
427 | maxSize = m_fixedMaxGuestSize;
|
---|
428 | break;
|
---|
429 | case MaximumGuestScreenSizePolicy_Automatic:
|
---|
430 | maxSize = calculateMaxGuestSize().expandedTo(minimumSizeHint);
|
---|
431 | break;
|
---|
432 | case MaximumGuestScreenSizePolicy_Any:
|
---|
433 | /* (0, 0) means any of course. */
|
---|
434 | maxSize = QSize(0, 0);
|
---|
435 | }
|
---|
436 | ASMAtomicWriteU64(&m_u64MaximumGuestSize,
|
---|
437 | RT_MAKE_U64(maxSize.height(), maxSize.width()));
|
---|
438 | }
|
---|
439 |
|
---|
440 | QSize UIMachineView::maximumGuestSize()
|
---|
441 | {
|
---|
442 | uint64_t u64Size = ASMAtomicReadU64(&m_u64MaximumGuestSize);
|
---|
443 | return QSize(int(RT_HI_U32(u64Size)), int(RT_LO_U32(u64Size)));
|
---|
444 | }
|
---|
445 |
|
---|
446 | void UIMachineView::updateViewport()
|
---|
447 | {
|
---|
448 | uimachine()->viewportChanged(screenId(), contentsX(), contentsY(), visibleWidth(), visibleHeight());
|
---|
449 | }
|
---|
450 |
|
---|
451 | #ifdef VBOX_WITH_DRAG_AND_DROP
|
---|
452 | int UIMachineView::dragCheckPending()
|
---|
453 | {
|
---|
454 | int rc;
|
---|
455 |
|
---|
456 | if (!dragAndDropIsActive())
|
---|
457 | rc = VERR_ACCESS_DENIED;
|
---|
458 | # ifdef VBOX_WITH_DRAG_AND_DROP_GH
|
---|
459 | else if (!m_fIsDraggingFromGuest)
|
---|
460 | {
|
---|
461 | /// @todo Add guest->guest DnD functionality here by getting
|
---|
462 | // the source of guest B (when copying from B to A).
|
---|
463 | rc = m_pDnDHandler->dragCheckPending(screenId());
|
---|
464 | if (RT_SUCCESS(rc))
|
---|
465 | m_fIsDraggingFromGuest = true;
|
---|
466 | }
|
---|
467 | else /* Already dragging, so report success. */
|
---|
468 | rc = VINF_SUCCESS;
|
---|
469 | # else
|
---|
470 | rc = VERR_NOT_SUPPORTED;
|
---|
471 | # endif
|
---|
472 |
|
---|
473 | DNDDEBUG(("DnD: dragCheckPending ended with rc=%Rrc\n", rc));
|
---|
474 | return rc;
|
---|
475 | }
|
---|
476 |
|
---|
477 | int UIMachineView::dragStart()
|
---|
478 | {
|
---|
479 | int rc;
|
---|
480 |
|
---|
481 | if (!dragAndDropIsActive())
|
---|
482 | rc = VERR_ACCESS_DENIED;
|
---|
483 | # ifdef VBOX_WITH_DRAG_AND_DROP_GH
|
---|
484 | else if (!m_fIsDraggingFromGuest)
|
---|
485 | rc = VERR_WRONG_ORDER;
|
---|
486 | else
|
---|
487 | {
|
---|
488 | /// @todo Add guest->guest DnD functionality here by getting
|
---|
489 | // the source of guest B (when copying from B to A).
|
---|
490 | rc = m_pDnDHandler->dragStart(screenId());
|
---|
491 |
|
---|
492 | m_fIsDraggingFromGuest = false;
|
---|
493 | }
|
---|
494 | # else
|
---|
495 | rc = VERR_NOT_SUPPORTED;
|
---|
496 | # endif
|
---|
497 |
|
---|
498 | DNDDEBUG(("DnD: dragStart ended with rc=%Rrc\n", rc));
|
---|
499 | return rc;
|
---|
500 | }
|
---|
501 |
|
---|
502 | int UIMachineView::dragStop()
|
---|
503 | {
|
---|
504 | int rc;
|
---|
505 |
|
---|
506 | if (!dragAndDropIsActive())
|
---|
507 | rc = VERR_ACCESS_DENIED;
|
---|
508 | # ifdef VBOX_WITH_DRAG_AND_DROP_GH
|
---|
509 | else if (!m_fIsDraggingFromGuest)
|
---|
510 | rc = VERR_WRONG_ORDER;
|
---|
511 | else
|
---|
512 | rc = m_pDnDHandler->dragStop(screenId());
|
---|
513 | # else
|
---|
514 | rc = VERR_NOT_SUPPORTED;
|
---|
515 | # endif
|
---|
516 |
|
---|
517 | DNDDEBUG(("DnD: dragStop ended with rc=%Rrc\n", rc));
|
---|
518 | return rc;
|
---|
519 | }
|
---|
520 | #endif /* VBOX_WITH_DRAG_AND_DROP */
|
---|
521 |
|
---|
522 | bool UIMachineView::nativeEventPreprocessor(const QByteArray &eventType, void *pMessage)
|
---|
523 | {
|
---|
524 | /* Check if some event should be filtered out.
|
---|
525 | * Returning @c true means filtering-out,
|
---|
526 | * Returning @c false means passing event to Qt. */
|
---|
527 |
|
---|
528 | # if defined(VBOX_WS_MAC)
|
---|
529 |
|
---|
530 | /* Make sure it's generic NSEvent: */
|
---|
531 | if (eventType != "mac_generic_NSEvent")
|
---|
532 | return false;
|
---|
533 | EventRef event = static_cast<EventRef>(darwinCocoaToCarbonEvent(pMessage));
|
---|
534 |
|
---|
535 | switch (::GetEventClass(event))
|
---|
536 | {
|
---|
537 | // Keep in mind that this stuff should not be enabled while we are still using
|
---|
538 | // own native keyboard filter installed through cocoa API, to be reworked.
|
---|
539 | // S.a. registerForNativeEvents call in UIKeyboardHandler implementation.
|
---|
540 | #if 0
|
---|
541 | /* Watch for keyboard-events: */
|
---|
542 | case kEventClassKeyboard:
|
---|
543 | {
|
---|
544 | switch (::GetEventKind(event))
|
---|
545 | {
|
---|
546 | /* Watch for key-events: */
|
---|
547 | case kEventRawKeyDown:
|
---|
548 | case kEventRawKeyRepeat:
|
---|
549 | case kEventRawKeyUp:
|
---|
550 | case kEventRawKeyModifiersChanged:
|
---|
551 | {
|
---|
552 | /* Delegate key-event handling to the keyboard-handler: */
|
---|
553 | return machineLogic()->keyboardHandler()->nativeEventFilter(pMessage, screenId());
|
---|
554 | }
|
---|
555 | default:
|
---|
556 | break;
|
---|
557 | }
|
---|
558 | break;
|
---|
559 | }
|
---|
560 | #endif
|
---|
561 | /* Watch for mouse-events: */
|
---|
562 | case kEventClassMouse:
|
---|
563 | {
|
---|
564 | switch (::GetEventKind(event))
|
---|
565 | {
|
---|
566 | /* Watch for button-events: */
|
---|
567 | case kEventMouseDown:
|
---|
568 | case kEventMouseUp:
|
---|
569 | {
|
---|
570 | /* Delegate button-event handling to the mouse-handler: */
|
---|
571 | return machineLogic()->mouseHandler()->nativeEventFilter(pMessage, screenId());
|
---|
572 | }
|
---|
573 | default:
|
---|
574 | break;
|
---|
575 | }
|
---|
576 | break;
|
---|
577 | }
|
---|
578 | default:
|
---|
579 | break;
|
---|
580 | }
|
---|
581 |
|
---|
582 | # elif defined(VBOX_WS_WIN)
|
---|
583 |
|
---|
584 | /* Make sure it's generic MSG event: */
|
---|
585 | if (eventType != "windows_generic_MSG")
|
---|
586 | return false;
|
---|
587 | MSG *pEvent = static_cast<MSG*>(pMessage);
|
---|
588 |
|
---|
589 | switch (pEvent->message)
|
---|
590 | {
|
---|
591 | /* Watch for key-events: */
|
---|
592 | case WM_KEYDOWN:
|
---|
593 | case WM_SYSKEYDOWN:
|
---|
594 | case WM_KEYUP:
|
---|
595 | case WM_SYSKEYUP:
|
---|
596 | {
|
---|
597 | // WORKAROUND:
|
---|
598 | // There is an issue in the Windows Qt5 event processing sequence
|
---|
599 | // causing QAbstractNativeEventFilter to receive Windows native events
|
---|
600 | // coming not just to the top-level window but to actual target as well.
|
---|
601 | // They are calling one - "global event" and another one - "context event".
|
---|
602 | // That way native events are always duplicated with almost no possibility
|
---|
603 | // to distinguish copies except the fact that synthetic event always have
|
---|
604 | // time set to 0 (actually that field was not initialized at all, we had
|
---|
605 | // fixed that in our private Qt tool). We should skip such events instantly.
|
---|
606 | if (pEvent->time == 0)
|
---|
607 | return false;
|
---|
608 |
|
---|
609 | /* Delegate key-event handling to the keyboard-handler: */
|
---|
610 | return machineLogic()->keyboardHandler()->nativeEventFilter(pMessage, screenId());
|
---|
611 | }
|
---|
612 | default:
|
---|
613 | break;
|
---|
614 | }
|
---|
615 |
|
---|
616 | # elif defined(VBOX_WS_NIX)
|
---|
617 |
|
---|
618 | if (uiCommon().X11ServerAvailable())
|
---|
619 | {
|
---|
620 | /* Make sure it's generic XCB event: */
|
---|
621 | if (eventType != "xcb_generic_event_t")
|
---|
622 | return false;
|
---|
623 | xcb_generic_event_t *pEvent = static_cast<xcb_generic_event_t*>(pMessage);
|
---|
624 |
|
---|
625 | switch (pEvent->response_type & ~0x80)
|
---|
626 | {
|
---|
627 | /* Watch for key-events: */
|
---|
628 | case XCB_KEY_PRESS:
|
---|
629 | case XCB_KEY_RELEASE:
|
---|
630 | {
|
---|
631 | /* Delegate key-event handling to the keyboard-handler: */
|
---|
632 | return machineLogic()->keyboardHandler()->nativeEventFilter(pMessage, screenId());
|
---|
633 | }
|
---|
634 | /* Watch for button-events: */
|
---|
635 | case XCB_BUTTON_PRESS:
|
---|
636 | case XCB_BUTTON_RELEASE:
|
---|
637 | {
|
---|
638 | /* Delegate button-event handling to the mouse-handler: */
|
---|
639 | return machineLogic()->mouseHandler()->nativeEventFilter(pMessage, screenId());
|
---|
640 | }
|
---|
641 | default:
|
---|
642 | break;
|
---|
643 | }
|
---|
644 | }
|
---|
645 |
|
---|
646 | # else
|
---|
647 |
|
---|
648 | # warning "port me!"
|
---|
649 |
|
---|
650 | # endif
|
---|
651 |
|
---|
652 | /* Filter nothing by default: */
|
---|
653 | return false;
|
---|
654 | }
|
---|
655 |
|
---|
656 | #ifdef VBOX_WS_MAC
|
---|
657 | CGImageRef UIMachineView::vmContentImage()
|
---|
658 | {
|
---|
659 | /* Use pause-image if exists: */
|
---|
660 | if (!pausePixmap().isNull())
|
---|
661 | return darwinToCGImageRef(&pausePixmap());
|
---|
662 |
|
---|
663 | /* Create the image ref out of the frame-buffer: */
|
---|
664 | return frameBuffertoCGImageRef(frameBuffer());
|
---|
665 | }
|
---|
666 | #endif /* VBOX_WS_MAC */
|
---|
667 |
|
---|
668 | void UIMachineView::sltHandleNotifyChange(int iWidth, int iHeight)
|
---|
669 | {
|
---|
670 | /* Sanity check: */
|
---|
671 | if (!frameBuffer())
|
---|
672 | return;
|
---|
673 |
|
---|
674 | LogRel2(("GUI: UIMachineView::sltHandleNotifyChange: Screen=%d, Size=%dx%d\n",
|
---|
675 | (unsigned long)m_uScreenId, iWidth, iHeight));
|
---|
676 |
|
---|
677 | /* Some situations require frame-buffer resize-events to be ignored at all,
|
---|
678 | * leaving machine-window, machine-view and frame-buffer sizes preserved: */
|
---|
679 | if (uimachine()->isGuestResizeIgnored())
|
---|
680 | return;
|
---|
681 |
|
---|
682 | /* In some situations especially in some VM states, guest-screen is not drawable: */
|
---|
683 | if (uimachine()->isGuestScreenUnDrawable())
|
---|
684 | return;
|
---|
685 |
|
---|
686 | /* Get old frame-buffer size: */
|
---|
687 | const QSize frameBufferSizeOld = QSize(frameBuffer()->width(),
|
---|
688 | frameBuffer()->height());
|
---|
689 |
|
---|
690 | /* Perform frame-buffer mode-change: */
|
---|
691 | frameBuffer()->handleNotifyChange(iWidth, iHeight);
|
---|
692 |
|
---|
693 | /* Get new frame-buffer size: */
|
---|
694 | const QSize frameBufferSizeNew = QSize(frameBuffer()->width(),
|
---|
695 | frameBuffer()->height());
|
---|
696 |
|
---|
697 | /* For 'scale' mode: */
|
---|
698 | if (visualStateType() == UIVisualStateType_Scale)
|
---|
699 | {
|
---|
700 | /* Assign new frame-buffer logical-size: */
|
---|
701 | QSize scaledSize = size();
|
---|
702 | const double dDevicePixelRatioFormal = frameBuffer()->devicePixelRatio();
|
---|
703 | const double dDevicePixelRatioActual = frameBuffer()->devicePixelRatioActual();
|
---|
704 | scaledSize *= dDevicePixelRatioFormal;
|
---|
705 | if (!frameBuffer()->useUnscaledHiDPIOutput())
|
---|
706 | scaledSize /= dDevicePixelRatioActual;
|
---|
707 | frameBuffer()->setScaledSize(scaledSize);
|
---|
708 |
|
---|
709 | /* Forget the last full-screen size: */
|
---|
710 | uimachine()->setLastFullScreenSize(screenId(), QSize(-1, -1));
|
---|
711 | }
|
---|
712 | /* For other than 'scale' mode: */
|
---|
713 | else
|
---|
714 | {
|
---|
715 | /* Adjust maximum-size restriction for machine-view: */
|
---|
716 | setMaximumSize(sizeHint());
|
---|
717 |
|
---|
718 | /* Disable the resize hint override hack and forget the last full-screen size: */
|
---|
719 | m_sizeHintOverride = QSize(-1, -1);
|
---|
720 | if (visualStateType() == UIVisualStateType_Normal)
|
---|
721 | uimachine()->setLastFullScreenSize(screenId(), QSize(-1, -1));
|
---|
722 |
|
---|
723 | /* Force machine-window update own layout: */
|
---|
724 | QCoreApplication::sendPostedEvents(0, QEvent::LayoutRequest);
|
---|
725 |
|
---|
726 | /* Update machine-view sliders: */
|
---|
727 | updateSliders();
|
---|
728 |
|
---|
729 | /* By some reason Win host forgets to update machine-window central-widget
|
---|
730 | * after main-layout was updated, let's do it for all the hosts: */
|
---|
731 | machineWindow()->centralWidget()->update();
|
---|
732 |
|
---|
733 | /* Normalize 'normal' machine-window geometry if necessary: */
|
---|
734 | if (visualStateType() == UIVisualStateType_Normal &&
|
---|
735 | frameBufferSizeNew != frameBufferSizeOld)
|
---|
736 | machineWindow()->normalizeGeometry(true /* adjust position */, machineWindow()->shouldResizeToGuestDisplay());
|
---|
737 | }
|
---|
738 |
|
---|
739 | /* Perform frame-buffer rescaling: */
|
---|
740 | frameBuffer()->performRescale();
|
---|
741 |
|
---|
742 | #ifdef VBOX_WS_MAC
|
---|
743 | /* Update MacOS X dock icon size: */
|
---|
744 | machineLogic()->updateDockIconSize(screenId(), frameBufferSizeNew.width(), frameBufferSizeNew.height());
|
---|
745 | #endif /* VBOX_WS_MAC */
|
---|
746 |
|
---|
747 | /* Notify frame-buffer resize: */
|
---|
748 | emit sigFrameBufferResize();
|
---|
749 |
|
---|
750 | /* Ask for just required guest display update (it will also update
|
---|
751 | * the viewport through IFramebuffer::NotifyUpdate): */
|
---|
752 | uimachine()->invalidateAndUpdateScreen(m_uScreenId);
|
---|
753 |
|
---|
754 | /* Acquire graphics controller type: */
|
---|
755 | KGraphicsControllerType enmType = KGraphicsControllerType_Null;
|
---|
756 | uimachine()->acquireGraphicsControllerType(enmType);
|
---|
757 |
|
---|
758 | /* If we are in normal or scaled mode and if GA are active,
|
---|
759 | * remember the guest-screen size to be able to restore it when necessary: */
|
---|
760 | /* As machines with Linux/Solaris and VMSVGA are not able to tell us
|
---|
761 | * whether a resize was due to the system or user interaction we currently
|
---|
762 | * do not store hints for these systems except when we explicitly send them
|
---|
763 | * ourselves. Windows guests should use VBoxVGA controllers, not VMSVGA. */
|
---|
764 | if ( !isFullscreenOrSeamless()
|
---|
765 | && uimachine()->isGuestSupportsGraphics()
|
---|
766 | && (enmType != KGraphicsControllerType_VMSVGA))
|
---|
767 | setStoredGuestScreenSizeHint(frameBufferSizeNew);
|
---|
768 |
|
---|
769 | LogRel2(("GUI: UIMachineView::sltHandleNotifyChange: Complete for Screen=%d, Size=%dx%d\n",
|
---|
770 | (unsigned long)m_uScreenId, frameBufferSizeNew.width(), frameBufferSizeNew.height()));
|
---|
771 | }
|
---|
772 |
|
---|
773 | void UIMachineView::sltHandleNotifyUpdate(int iX, int iY, int iWidth, int iHeight)
|
---|
774 | {
|
---|
775 | /* Sanity check: */
|
---|
776 | if (!frameBuffer())
|
---|
777 | return;
|
---|
778 |
|
---|
779 | /* Prepare corresponding viewport part: */
|
---|
780 | QRect rect(iX, iY, iWidth, iHeight);
|
---|
781 |
|
---|
782 | /* Take the scaling into account: */
|
---|
783 | const double dScaleFactor = frameBuffer()->scaleFactor();
|
---|
784 | const QSize scaledSize = frameBuffer()->scaledSize();
|
---|
785 | if (scaledSize.isValid())
|
---|
786 | {
|
---|
787 | /* Calculate corresponding scale-factors: */
|
---|
788 | const double xScaleFactor = visualStateType() == UIVisualStateType_Scale ?
|
---|
789 | (double)scaledSize.width() / frameBuffer()->width() : dScaleFactor;
|
---|
790 | const double yScaleFactor = visualStateType() == UIVisualStateType_Scale ?
|
---|
791 | (double)scaledSize.height() / frameBuffer()->height() : dScaleFactor;
|
---|
792 | /* Adjust corresponding viewport part: */
|
---|
793 | rect.moveTo((int)floor((double)rect.x() * xScaleFactor) - 1,
|
---|
794 | (int)floor((double)rect.y() * yScaleFactor) - 1);
|
---|
795 | rect.setSize(QSize((int)ceil((double)rect.width() * xScaleFactor) + 2,
|
---|
796 | (int)ceil((double)rect.height() * yScaleFactor) + 2));
|
---|
797 | }
|
---|
798 |
|
---|
799 | /* Shift has to be scaled by the device-pixel-ratio
|
---|
800 | * but not scaled by the scale-factor. */
|
---|
801 | rect.translate(-contentsX(), -contentsY());
|
---|
802 |
|
---|
803 | /* Take the device-pixel-ratio into account: */
|
---|
804 | const double dDevicePixelRatioFormal = frameBuffer()->devicePixelRatio();
|
---|
805 | const double dDevicePixelRatioActual = frameBuffer()->devicePixelRatioActual();
|
---|
806 | if (!frameBuffer()->useUnscaledHiDPIOutput() && dDevicePixelRatioActual != 1.0)
|
---|
807 | {
|
---|
808 | rect.moveTo((int)floor((double)rect.x() * dDevicePixelRatioActual) - 1,
|
---|
809 | (int)floor((double)rect.y() * dDevicePixelRatioActual) - 1);
|
---|
810 | rect.setSize(QSize((int)ceil((double)rect.width() * dDevicePixelRatioActual) + 2,
|
---|
811 | (int)ceil((double)rect.height() * dDevicePixelRatioActual) + 2));
|
---|
812 | }
|
---|
813 | if (dDevicePixelRatioFormal != 1.0)
|
---|
814 | {
|
---|
815 | rect.moveTo((int)floor((double)rect.x() / dDevicePixelRatioFormal) - 1,
|
---|
816 | (int)floor((double)rect.y() / dDevicePixelRatioFormal) - 1);
|
---|
817 | rect.setSize(QSize((int)ceil((double)rect.width() / dDevicePixelRatioFormal) + 2,
|
---|
818 | (int)ceil((double)rect.height() / dDevicePixelRatioFormal) + 2));
|
---|
819 | }
|
---|
820 |
|
---|
821 | /* Limit the resulting part by the viewport rectangle: */
|
---|
822 | rect &= viewport()->rect();
|
---|
823 |
|
---|
824 | /* Update corresponding viewport part: */
|
---|
825 | viewport()->update(rect);
|
---|
826 | }
|
---|
827 |
|
---|
828 | void UIMachineView::sltHandleSetVisibleRegion(QRegion region)
|
---|
829 | {
|
---|
830 | /* Used only in seamless-mode. */
|
---|
831 | Q_UNUSED(region);
|
---|
832 | }
|
---|
833 |
|
---|
834 | void UIMachineView::sltPerformGuestResize(const QSize &toSize)
|
---|
835 | {
|
---|
836 | /* There is a couple of things to keep in mind:
|
---|
837 | *
|
---|
838 | * First of all, passed size can be invalid (or even not sane one, where one of values equal to zero). Usually that happens
|
---|
839 | * if this function being invoked with default argument for example by some slot. In such case we get the available size for
|
---|
840 | * the guest-screen we have. We assume here that centralWidget() contains this view only and gives it all available space.
|
---|
841 | * In all other cases we have a valid non-zero size which should be handled as usual.
|
---|
842 | *
|
---|
843 | * Besides that, passed size or size taken from centralWidget() is _not_ absolute one, it's in widget's coordinate system
|
---|
844 | * which can and will be be transformed by scale-factor when appropriate, so before passing this size to a guest it has to
|
---|
845 | * be scaled backward. This is the key aspect in which internal resize differs from resize initiated from the outside. */
|
---|
846 |
|
---|
847 | /* Make sure we have valid size to work with: */
|
---|
848 | QSize size( toSize.isValid() && toSize.width() > 0 && toSize.height() > 0
|
---|
849 | ? toSize : machineWindow()->centralWidget()->size());
|
---|
850 | AssertMsgReturnVoid(size.isValid() && size.width() > 0 && size.height() > 0,
|
---|
851 | ("Size should be valid (%dx%d)!\n", size.width(), size.height()));
|
---|
852 |
|
---|
853 | /* Take the scale-factor(s) into account: */
|
---|
854 | size = scaledBackward(size);
|
---|
855 |
|
---|
856 | /* Update current window size limitations: */
|
---|
857 | setMaximumGuestSize(size);
|
---|
858 |
|
---|
859 | /* Record the hint to extra data, needed for guests using VMSVGA:
|
---|
860 | * This should be done before the actual hint is sent in case the guest overrides it.
|
---|
861 | * Do not send a hint if nothing has changed to prevent the guest being notified about its own changes. */
|
---|
862 | if ( !isFullscreenOrSeamless()
|
---|
863 | && uimachine()->isGuestSupportsGraphics()
|
---|
864 | && ( (int)frameBuffer()->width() != size.width()
|
---|
865 | || (int)frameBuffer()->height() != size.height()
|
---|
866 | || uimachine()->isScreenVisible(screenId()) != uimachine()->isScreenVisibleHostDesires(screenId())))
|
---|
867 | setStoredGuestScreenSizeHint(size);
|
---|
868 |
|
---|
869 | /* If auto-mount of guest-screens (auto-pilot) enabled: */
|
---|
870 | if (gEDataManager->autoMountGuestScreensEnabled(uiCommon().managedVMUuid()))
|
---|
871 | {
|
---|
872 | /* If host and guest have same opinion about guest-screen visibility: */
|
---|
873 | if (uimachine()->isScreenVisible(screenId()) == uimachine()->isScreenVisibleHostDesires(screenId()))
|
---|
874 | {
|
---|
875 | /* Do not send a hint if nothing has changed to prevent the guest being notified about its own changes: */
|
---|
876 | if ((int)frameBuffer()->width() != size.width() || (int)frameBuffer()->height() != size.height())
|
---|
877 | {
|
---|
878 | LogRel(("GUI: UIMachineView::sltPerformGuestResize: Auto-pilot resizing screen %d as %dx%d\n",
|
---|
879 | (int)screenId(), size.width(), size.height()));
|
---|
880 | uimachine()->setVideoModeHint(screenId(),
|
---|
881 | uimachine()->isScreenVisible(screenId()),
|
---|
882 | false /* change origin? */,
|
---|
883 | 0 /* origin x */, 0 /* origin y */,
|
---|
884 | size.width(), size.height(),
|
---|
885 | 0 /* bits per pixel */,
|
---|
886 | true /* notify? */);
|
---|
887 | }
|
---|
888 | }
|
---|
889 | else
|
---|
890 | {
|
---|
891 | /* If host desires to have guest-screen enabled and guest-screen is disabled, retrying: */
|
---|
892 | if (uimachine()->isScreenVisibleHostDesires(screenId()))
|
---|
893 | {
|
---|
894 | /* Send enabling size-hint to the guest: */
|
---|
895 | LogRel(("GUI: UIMachineView::sltPerformGuestResize: Auto-pilot enabling guest-screen %d\n", (int)screenId()));
|
---|
896 | uimachine()->setVideoModeHint(screenId(),
|
---|
897 | true /* enabled? */,
|
---|
898 | false /* change origin? */,
|
---|
899 | 0 /* origin x */, 0 /* origin y */,
|
---|
900 | size.width(), size.height(),
|
---|
901 | 0 /* bits per pixel */,
|
---|
902 | true /* notify? */);
|
---|
903 | }
|
---|
904 | /* If host desires to have guest-screen disabled and guest-screen is enabled, retrying: */
|
---|
905 | else
|
---|
906 | {
|
---|
907 | /* Send disabling size-hint to the guest: */
|
---|
908 | LogRel(("GUI: UIMachineView::sltPerformGuestResize: Auto-pilot disabling guest-screen %d\n", (int)screenId()));
|
---|
909 | uimachine()->setVideoModeHint(screenId(),
|
---|
910 | false /* enabled? */,
|
---|
911 | false /* change origin? */,
|
---|
912 | 0 /* origin x */, 0 /* origin y */,
|
---|
913 | 0 /* width */, 0 /* height */,
|
---|
914 | 0 /* bits per pixel */,
|
---|
915 | true /* notify? */);
|
---|
916 | }
|
---|
917 | }
|
---|
918 | }
|
---|
919 | /* If auto-mount of guest-screens (auto-pilot) disabled: */
|
---|
920 | else
|
---|
921 | {
|
---|
922 | /* Should we send a hint? */
|
---|
923 | bool fSendHint = true;
|
---|
924 | /* Do not send a hint if nothing has changed to prevent the guest being notified about its own changes: */
|
---|
925 | if (fSendHint && (int)frameBuffer()->width() == size.width() && (int)frameBuffer()->height() == size.height())
|
---|
926 | {
|
---|
927 | LogRel(("GUI: UIMachineView::sltPerformGuestResize: Omitting to send size-hint %dx%d to guest-screen %d "
|
---|
928 | "because frame-buffer is already of the same size.\n", size.width(), size.height(), (int)screenId()));
|
---|
929 | fSendHint = false;
|
---|
930 | }
|
---|
931 | /* Do not send a hint if GA supports graphics and we have sent that hint already: */
|
---|
932 | if (fSendHint && uimachine()->isGuestSupportsGraphics() && m_lastSizeHint == size)
|
---|
933 | {
|
---|
934 | LogRel(("GUI: UIMachineView::sltPerformGuestResize: Omitting to send size-hint %dx%d to guest-screen %d "
|
---|
935 | "because this hint was previously sent.\n", size.width(), size.height(), (int)screenId()));
|
---|
936 | fSendHint = false;
|
---|
937 | }
|
---|
938 | if (fSendHint)
|
---|
939 | {
|
---|
940 | LogRel(("GUI: UIMachineView::sltPerformGuestResize: Sending guest size-hint to screen %d as %dx%d\n",
|
---|
941 | (int)screenId(), size.width(), size.height()));
|
---|
942 | uimachine()->setVideoModeHint(screenId(),
|
---|
943 | uimachine()->isScreenVisible(screenId()),
|
---|
944 | false /* change origin? */,
|
---|
945 | 0 /* origin x */, 0 /* origin y */,
|
---|
946 | size.width(), size.height(),
|
---|
947 | 0 /* bits per pixel */,
|
---|
948 | true /* notify? */);
|
---|
949 | m_lastSizeHint = size;
|
---|
950 | }
|
---|
951 | }
|
---|
952 | }
|
---|
953 |
|
---|
954 | void UIMachineView::sltHandleActionTriggerViewScreenToggle(int iScreen, bool fEnabled)
|
---|
955 | {
|
---|
956 | /* Skip unrelated guest-screen index: */
|
---|
957 | if (iScreen != (int)screenId())
|
---|
958 | return;
|
---|
959 |
|
---|
960 | /* Acquire current resolution: */
|
---|
961 | ulong uWidth = 0, uHeight = 0, uDummy = 0;
|
---|
962 | long iDummy = 0;
|
---|
963 | KGuestMonitorStatus enmDummy = KGuestMonitorStatus_Disabled;
|
---|
964 | const bool fSuccess = uimachine()->acquireGuestScreenParameters(screenId(), uWidth, uHeight,
|
---|
965 | uDummy, iDummy, iDummy, enmDummy);
|
---|
966 | if (!fSuccess)
|
---|
967 | return;
|
---|
968 |
|
---|
969 | /* Update desirable screen status: */
|
---|
970 | uimachine()->setScreenVisibleHostDesires(screenId(), fEnabled);
|
---|
971 |
|
---|
972 | /* Send enabling size-hint: */
|
---|
973 | if (fEnabled)
|
---|
974 | {
|
---|
975 | /* Defaults: */
|
---|
976 | if (!uWidth)
|
---|
977 | uWidth = 800;
|
---|
978 | if (!uHeight)
|
---|
979 | uHeight = 600;
|
---|
980 |
|
---|
981 | /* Update current window size limitations: */
|
---|
982 | setMaximumGuestSize(QSize(uWidth, uHeight));
|
---|
983 |
|
---|
984 | /* Record the hint to extra data, needed for guests using VMSVGA:
|
---|
985 | * This should be done before the actual hint is sent in case the guest overrides it.
|
---|
986 | * Do not send a hint if nothing has changed to prevent the guest being notified about its own changes. */
|
---|
987 | if ( !isFullscreenOrSeamless()
|
---|
988 | && uimachine()->isGuestSupportsGraphics()
|
---|
989 | && ( frameBuffer()->width() != uWidth
|
---|
990 | || frameBuffer()->height() != uHeight
|
---|
991 | || uimachine()->isScreenVisible(screenId()) != uimachine()->isScreenVisibleHostDesires(screenId())))
|
---|
992 | setStoredGuestScreenSizeHint(QSize(uWidth, uHeight));
|
---|
993 |
|
---|
994 | /* Send enabling size-hint to the guest: */
|
---|
995 | LogRel(("GUI: UIMachineView::sltHandleActionTriggerViewScreenToggle: Enabling guest-screen %d\n", (int)screenId()));
|
---|
996 | uimachine()->setVideoModeHint(screenId(),
|
---|
997 | true /* enabled? */,
|
---|
998 | false /* change origin? */,
|
---|
999 | 0 /* origin x */, 0 /* origin y */,
|
---|
1000 | uWidth, uHeight,
|
---|
1001 | 0 /* bits per pixel */,
|
---|
1002 | true /* notify? */);
|
---|
1003 | }
|
---|
1004 | else
|
---|
1005 | {
|
---|
1006 | /* Send disabling size-hint to the guest: */
|
---|
1007 | LogRel(("GUI: UIMachineView::sltHandleActionTriggerViewScreenToggle: Disabling guest-screen %d\n", (int)screenId()));
|
---|
1008 | uimachine()->setVideoModeHint(screenId(),
|
---|
1009 | false /* enabled? */,
|
---|
1010 | false /* change origin? */,
|
---|
1011 | 0 /* origin x */, 0 /* origin y */,
|
---|
1012 | 0 /* width */, 0 /* height */,
|
---|
1013 | 0 /* bits per pixel */,
|
---|
1014 | true /* notify? */);
|
---|
1015 | }
|
---|
1016 | }
|
---|
1017 |
|
---|
1018 | void UIMachineView::sltHandleActionTriggerViewScreenResize(int iScreen, const QSize &size)
|
---|
1019 | {
|
---|
1020 | /* Skip unrelated guest-screen index: */
|
---|
1021 | if (iScreen != (int)m_uScreenId)
|
---|
1022 | return;
|
---|
1023 |
|
---|
1024 | /* Make sure we have valid size to work with: */
|
---|
1025 | AssertMsgReturnVoid(size.isValid() && size.width() > 0 && size.height() > 0,
|
---|
1026 | ("Size should be valid (%dx%d)!\n", size.width(), size.height()));
|
---|
1027 |
|
---|
1028 | /* Update current window size limitations: */
|
---|
1029 | setMaximumGuestSize(size);
|
---|
1030 |
|
---|
1031 | /* Record the hint to extra data, needed for guests using VMSVGA:
|
---|
1032 | * This should be done before the actual hint is sent in case the guest overrides it.
|
---|
1033 | * Do not send a hint if nothing has changed to prevent the guest being notified about its own changes. */
|
---|
1034 | if ( !isFullscreenOrSeamless()
|
---|
1035 | && uimachine()->isGuestSupportsGraphics()
|
---|
1036 | && ( (int)frameBuffer()->width() != size.width()
|
---|
1037 | || (int)frameBuffer()->height() != size.height()
|
---|
1038 | || uimachine()->isScreenVisible(screenId()) != uimachine()->isScreenVisibleHostDesires(screenId())))
|
---|
1039 | setStoredGuestScreenSizeHint(size);
|
---|
1040 |
|
---|
1041 | /* Send enabling size-hint to the guest: */
|
---|
1042 | LogRel(("GUI: UIMachineView::sltHandleActionTriggerViewScreenResize: Resizing guest-screen %d\n", (int)screenId()));
|
---|
1043 | uimachine()->setVideoModeHint(screenId(),
|
---|
1044 | true /* enabled? */,
|
---|
1045 | false /* change origin? */,
|
---|
1046 | 0 /* origin x */, 0 /* origin y */,
|
---|
1047 | size.width(), size.height(),
|
---|
1048 | 0 /* bits per pixel */,
|
---|
1049 | true /* notify? */);
|
---|
1050 | }
|
---|
1051 |
|
---|
1052 | void UIMachineView::sltDesktopResized()
|
---|
1053 | {
|
---|
1054 | setMaximumGuestSize();
|
---|
1055 | }
|
---|
1056 |
|
---|
1057 | void UIMachineView::sltHandleScaleFactorChange(const QUuid &uMachineID)
|
---|
1058 | {
|
---|
1059 | /* Skip unrelated machine IDs: */
|
---|
1060 | if (uMachineID != uiCommon().managedVMUuid())
|
---|
1061 | return;
|
---|
1062 |
|
---|
1063 | /* Acquire selected scale-factor: */
|
---|
1064 | double dScaleFactor = gEDataManager->scaleFactor(uiCommon().managedVMUuid(), m_uScreenId);
|
---|
1065 |
|
---|
1066 | /* Take the device-pixel-ratio into account: */
|
---|
1067 | const double dDevicePixelRatioActual = frameBuffer()->devicePixelRatioActual();
|
---|
1068 | const bool fUseUnscaledHiDPIOutput = dScaleFactor != dDevicePixelRatioActual;
|
---|
1069 | dScaleFactor = fUseUnscaledHiDPIOutput ? dScaleFactor : 1.0;
|
---|
1070 |
|
---|
1071 | /* Assign frame-buffer with new values: */
|
---|
1072 | frameBuffer()->setScaleFactor(dScaleFactor);
|
---|
1073 | frameBuffer()->setUseUnscaledHiDPIOutput(fUseUnscaledHiDPIOutput);
|
---|
1074 |
|
---|
1075 | /* Propagate the scale-factor related attributes to 3D service if necessary: */
|
---|
1076 | bool fAccelerate3DEnabled = false;
|
---|
1077 | uimachine()->acquireWhetherAccelerate3DEnabled(fAccelerate3DEnabled);
|
---|
1078 | if (fAccelerate3DEnabled)
|
---|
1079 | {
|
---|
1080 | double dScaleFactorFor3D = dScaleFactor;
|
---|
1081 | #if defined(VBOX_WS_WIN) || defined(VBOX_WS_NIX)
|
---|
1082 | // WORKAROUND:
|
---|
1083 | // On Windows and Linux opposing to macOS it's only Qt which can auto scale up,
|
---|
1084 | // not 3D overlay itself, so for auto scale-up mode we have to take that into account.
|
---|
1085 | if (!fUseUnscaledHiDPIOutput)
|
---|
1086 | dScaleFactorFor3D *= frameBuffer()->devicePixelRatioActual();
|
---|
1087 | #endif /* VBOX_WS_WIN || VBOX_WS_NIX */
|
---|
1088 | uimachine()->notifyScaleFactorChange(m_uScreenId,
|
---|
1089 | (uint32_t)(dScaleFactorFor3D * VBOX_OGL_SCALE_FACTOR_MULTIPLIER),
|
---|
1090 | (uint32_t)(dScaleFactorFor3D * VBOX_OGL_SCALE_FACTOR_MULTIPLIER));
|
---|
1091 | uimachine()->notifyHiDPIOutputPolicyChange(fUseUnscaledHiDPIOutput);
|
---|
1092 | }
|
---|
1093 |
|
---|
1094 | /* Handle scale attributes change: */
|
---|
1095 | handleScaleChange();
|
---|
1096 | /* Adjust guest-screen size: */
|
---|
1097 | adjustGuestScreenSize();
|
---|
1098 |
|
---|
1099 | /* Update scaled pause pixmap, if necessary: */
|
---|
1100 | updateScaledPausePixmap();
|
---|
1101 | viewport()->update();
|
---|
1102 |
|
---|
1103 | /* Update console's display viewport and 3D overlay: */
|
---|
1104 | updateViewport();
|
---|
1105 | }
|
---|
1106 |
|
---|
1107 | void UIMachineView::sltHandleScalingOptimizationChange(const QUuid &uMachineID)
|
---|
1108 | {
|
---|
1109 | /* Skip unrelated machine IDs: */
|
---|
1110 | if (uMachineID != uiCommon().managedVMUuid())
|
---|
1111 | return;
|
---|
1112 |
|
---|
1113 | /* Take the scaling-optimization type into account: */
|
---|
1114 | frameBuffer()->setScalingOptimizationType(gEDataManager->scalingOptimizationType(uiCommon().managedVMUuid()));
|
---|
1115 |
|
---|
1116 | /* Update viewport: */
|
---|
1117 | viewport()->update();
|
---|
1118 | }
|
---|
1119 |
|
---|
1120 | void UIMachineView::sltMachineStateChanged()
|
---|
1121 | {
|
---|
1122 | /* Get machine state: */
|
---|
1123 | KMachineState state = uimachine()->machineState();
|
---|
1124 | switch (state)
|
---|
1125 | {
|
---|
1126 | case KMachineState_Paused:
|
---|
1127 | case KMachineState_TeleportingPausedVM:
|
---|
1128 | {
|
---|
1129 | if ( frameBuffer()
|
---|
1130 | && ( state != KMachineState_TeleportingPausedVM
|
---|
1131 | || m_previousState != KMachineState_Teleporting))
|
---|
1132 | {
|
---|
1133 | // WORKAROUND:
|
---|
1134 | // We can't take pause pixmap if actual state is Saving, this produces
|
---|
1135 | // a lock and GUI will be frozen until SaveState call is complete...
|
---|
1136 | KMachineState enmActualState = KMachineState_Null;
|
---|
1137 | uimachine()->acquireLiveMachineState(enmActualState);
|
---|
1138 | if (enmActualState != KMachineState_Saving)
|
---|
1139 | {
|
---|
1140 | /* Take live pause-pixmap: */
|
---|
1141 | takePausePixmapLive();
|
---|
1142 | /* Fully repaint to pick up pause-pixmap: */
|
---|
1143 | viewport()->update();
|
---|
1144 | }
|
---|
1145 | }
|
---|
1146 | break;
|
---|
1147 | }
|
---|
1148 | case KMachineState_Restoring:
|
---|
1149 | {
|
---|
1150 | /* Only works with the primary screen currently. */
|
---|
1151 | if (screenId() == 0)
|
---|
1152 | {
|
---|
1153 | /* Take snapshot pause-pixmap: */
|
---|
1154 | takePausePixmapSnapshot();
|
---|
1155 | /* Fully repaint to pick up pause-pixmap: */
|
---|
1156 | viewport()->update();
|
---|
1157 | }
|
---|
1158 | break;
|
---|
1159 | }
|
---|
1160 | case KMachineState_Running:
|
---|
1161 | {
|
---|
1162 | if (m_previousState == KMachineState_Paused ||
|
---|
1163 | m_previousState == KMachineState_TeleportingPausedVM ||
|
---|
1164 | m_previousState == KMachineState_Restoring)
|
---|
1165 | {
|
---|
1166 | if (frameBuffer())
|
---|
1167 | {
|
---|
1168 | /* Reset pause-pixmap: */
|
---|
1169 | resetPausePixmap();
|
---|
1170 | /* Ask for full guest display update (it will also update
|
---|
1171 | * the viewport through IFramebuffer::NotifyUpdate): */
|
---|
1172 | uimachine()->invalidateAndUpdate();
|
---|
1173 | }
|
---|
1174 | }
|
---|
1175 | /* Reapply machine-view scale-factor: */
|
---|
1176 | applyMachineViewScaleFactor();
|
---|
1177 | break;
|
---|
1178 | }
|
---|
1179 | default:
|
---|
1180 | break;
|
---|
1181 | }
|
---|
1182 |
|
---|
1183 | m_previousState = state;
|
---|
1184 | }
|
---|
1185 |
|
---|
1186 | void UIMachineView::sltMousePointerShapeChange()
|
---|
1187 | {
|
---|
1188 | /* Fetch the shape and the mask: */
|
---|
1189 | QPixmap pixmapShape = uimachine()->cursorShapePixmap();
|
---|
1190 | QPixmap pixmapMask = uimachine()->cursorMaskPixmap();
|
---|
1191 | const QPoint hotspot = uimachine()->cursorHotspot();
|
---|
1192 | uint uXHot = hotspot.x();
|
---|
1193 | uint uYHot = hotspot.y();
|
---|
1194 |
|
---|
1195 | /* If there is no mask: */
|
---|
1196 | if (pixmapMask.isNull())
|
---|
1197 | {
|
---|
1198 | /* Scale the shape pixmap and
|
---|
1199 | * compose the cursor on the basis of shape only: */
|
---|
1200 | updateMousePointerPixmapScaling(pixmapShape, uXHot, uYHot);
|
---|
1201 | m_cursor = QCursor(pixmapShape, uXHot, uYHot);
|
---|
1202 | }
|
---|
1203 | /* Otherwise: */
|
---|
1204 | else
|
---|
1205 | {
|
---|
1206 | /* Scale the shape and the mask pixmaps and
|
---|
1207 | * compose the cursor on the basis of shape and mask both: */
|
---|
1208 | updateMousePointerPixmapScaling(pixmapShape, uXHot, uYHot);
|
---|
1209 | /// @todo updateMousePointerPixmapScaling(pixmapMask, uXHot, uYHot);
|
---|
1210 | m_cursor = QCursor(QBitmap::fromPixmap(pixmapShape), QBitmap::fromPixmap(pixmapMask), uXHot, uYHot);
|
---|
1211 | }
|
---|
1212 |
|
---|
1213 | /* Let the listeners know: */
|
---|
1214 | emit sigMousePointerShapeChange();
|
---|
1215 | }
|
---|
1216 |
|
---|
1217 | void UIMachineView::sltDetachCOM()
|
---|
1218 | {
|
---|
1219 | #ifdef VBOX_WITH_DRAG_AND_DROP
|
---|
1220 | /* Cleanup DnD: */
|
---|
1221 | cleanupDnd();
|
---|
1222 | #endif
|
---|
1223 | }
|
---|
1224 |
|
---|
1225 | void UIMachineView::sltRetranslateUI()
|
---|
1226 | {
|
---|
1227 | setWhatsThis(tr("Holds the graphical canvas containing guest screen contents."));
|
---|
1228 | }
|
---|
1229 |
|
---|
1230 | UIMachineView::UIMachineView(UIMachineWindow *pMachineWindow, ulong uScreenId)
|
---|
1231 | : QAbstractScrollArea(pMachineWindow->centralWidget())
|
---|
1232 | , m_pMachineWindow(pMachineWindow)
|
---|
1233 | , m_uScreenId(uScreenId)
|
---|
1234 | , m_previousState(KMachineState_Null)
|
---|
1235 | , m_iHostScreenNumber(0)
|
---|
1236 | , m_enmMaximumGuestScreenSizePolicy(MaximumGuestScreenSizePolicy_Automatic)
|
---|
1237 | , m_u64MaximumGuestSize(0)
|
---|
1238 | #ifdef VBOX_WITH_DRAG_AND_DROP_GH
|
---|
1239 | , m_fIsDraggingFromGuest(false)
|
---|
1240 | #endif
|
---|
1241 | , m_pNativeEventFilter(0)
|
---|
1242 | {
|
---|
1243 | /* Install Machine-view accessibility interface factory: */
|
---|
1244 | QAccessible::installFactory(UIAccessibilityInterfaceForUIMachineView::pFactory);
|
---|
1245 | }
|
---|
1246 |
|
---|
1247 | void UIMachineView::loadMachineViewSettings()
|
---|
1248 | {
|
---|
1249 | /* Global settings: */
|
---|
1250 | {
|
---|
1251 | /* Remember the maximum guest size policy for
|
---|
1252 | * telling the guest about video modes we like: */
|
---|
1253 | m_enmMaximumGuestScreenSizePolicy = gEDataManager->maxGuestResolutionPolicy();
|
---|
1254 | if (m_enmMaximumGuestScreenSizePolicy == MaximumGuestScreenSizePolicy_Fixed)
|
---|
1255 | m_fixedMaxGuestSize = gEDataManager->maxGuestResolutionForPolicyFixed();
|
---|
1256 | }
|
---|
1257 | }
|
---|
1258 |
|
---|
1259 | void UIMachineView::prepareViewport()
|
---|
1260 | {
|
---|
1261 | /* Prepare viewport: */
|
---|
1262 | AssertPtrReturnVoid(viewport());
|
---|
1263 | {
|
---|
1264 | /* Enable manual painting: */
|
---|
1265 | viewport()->setAttribute(Qt::WA_OpaquePaintEvent);
|
---|
1266 | /* Enable multi-touch support: */
|
---|
1267 | viewport()->setAttribute(Qt::WA_AcceptTouchEvents);
|
---|
1268 | }
|
---|
1269 | }
|
---|
1270 |
|
---|
1271 | void UIMachineView::prepareFrameBuffer()
|
---|
1272 | {
|
---|
1273 | /* Make sure frame-buffer exists: */
|
---|
1274 | if (!frameBuffer())
|
---|
1275 | return;
|
---|
1276 |
|
---|
1277 | /* If frame-buffer NOT yet initialized: */
|
---|
1278 | if (!frameBuffer()->isInitialized())
|
---|
1279 | {
|
---|
1280 | LogRelFlow(("GUI: UIMachineView::prepareFrameBuffer: Start EMT callbacks accepting for screen: %d\n", screenId()));
|
---|
1281 | /* Initialize for this view: */
|
---|
1282 | frameBuffer()->init(this);
|
---|
1283 | /* Apply machine-view scale-factor: */
|
---|
1284 | applyMachineViewScaleFactor();
|
---|
1285 | }
|
---|
1286 | /* Otherwise it must be unused yet: */
|
---|
1287 | else
|
---|
1288 | {
|
---|
1289 | LogRelFlow(("GUI: UIMachineView::prepareFrameBuffer: Restart EMT callbacks accepting for screen: %d\n", screenId()));
|
---|
1290 | /* Assign it's view: */
|
---|
1291 | frameBuffer()->setView(this);
|
---|
1292 | /* Mark frame-buffer as used again: */
|
---|
1293 | frameBuffer()->setMarkAsUnused(false);
|
---|
1294 | }
|
---|
1295 |
|
---|
1296 | /* Make sure frame-buffer was prepared: */
|
---|
1297 | AssertPtrReturnVoid(frameBuffer());
|
---|
1298 |
|
---|
1299 | /* Reattach to IDisplay: */
|
---|
1300 | frameBuffer()->detach();
|
---|
1301 | frameBuffer()->attach();
|
---|
1302 |
|
---|
1303 | /* Calculate frame-buffer size: */
|
---|
1304 | QSize size;
|
---|
1305 | {
|
---|
1306 | /* Acquire actual machine state to be sure: */
|
---|
1307 | KMachineState enmActualState = KMachineState_Null;
|
---|
1308 | uimachine()->acquireLiveMachineState(enmActualState);
|
---|
1309 |
|
---|
1310 | #ifdef VBOX_WS_NIX
|
---|
1311 | // WORKAROUND:
|
---|
1312 | // No idea why this was required for X11 before.
|
---|
1313 | // Currently I don't see a reason why I should keep it.
|
---|
1314 | # if 0
|
---|
1315 | if (enmActualState == KMachineState_Saved || enmActualState == KMachineState_AbortedSaved)
|
---|
1316 | size = storedGuestScreenSizeHint();
|
---|
1317 | # endif
|
---|
1318 | #endif /* VBOX_WS_NIX */
|
---|
1319 |
|
---|
1320 | /* If there is a preview image saved, we will resize the framebuffer to the size of that image: */
|
---|
1321 | if (enmActualState == KMachineState_Saved || enmActualState == KMachineState_AbortedSaved)
|
---|
1322 | {
|
---|
1323 | ulong uWidth = 0, uHeight = 0;
|
---|
1324 | QVector<KBitmapFormat> formats;
|
---|
1325 | uimachine()->acquireSavedScreenshotInfo(m_uScreenId, uWidth, uHeight, formats);
|
---|
1326 | if (formats.size() > 0)
|
---|
1327 | {
|
---|
1328 | /* Init with the screenshot size: */
|
---|
1329 | size = QSize(uWidth, uHeight);
|
---|
1330 | /* Try to get the real guest dimensions from the save-state: */
|
---|
1331 | long iDummy = 0;
|
---|
1332 | ulong uGuestWidth = 0, uGuestHeight = 0;
|
---|
1333 | bool fDummy = true;
|
---|
1334 | uimachine()->acquireSavedGuestScreenInfo(m_uScreenId,
|
---|
1335 | iDummy, iDummy,
|
---|
1336 | uGuestWidth, uGuestHeight, fDummy);
|
---|
1337 | if (uGuestWidth > 0 && uGuestHeight > 0)
|
---|
1338 | size = QSize(uGuestWidth, uGuestHeight);
|
---|
1339 | }
|
---|
1340 | }
|
---|
1341 |
|
---|
1342 | /* If we have a valid size, resize/rescale the frame-buffer. */
|
---|
1343 | if (size.width() > 0 && size.height() > 0)
|
---|
1344 | {
|
---|
1345 | frameBuffer()->performResize(size.width(), size.height());
|
---|
1346 | frameBuffer()->performRescale();
|
---|
1347 | }
|
---|
1348 | }
|
---|
1349 | }
|
---|
1350 |
|
---|
1351 | void UIMachineView::prepareCommon()
|
---|
1352 | {
|
---|
1353 | /* Prepare view frame: */
|
---|
1354 | setFrameStyle(QFrame::NoFrame);
|
---|
1355 |
|
---|
1356 | /* Setup palette: */
|
---|
1357 | QPalette palette(viewport()->palette());
|
---|
1358 | palette.setColor(viewport()->backgroundRole(), Qt::black);
|
---|
1359 | viewport()->setPalette(palette);
|
---|
1360 |
|
---|
1361 | /* Setup focus policy: */
|
---|
1362 | setFocusPolicy(Qt::WheelFocus);
|
---|
1363 | }
|
---|
1364 |
|
---|
1365 | #ifdef VBOX_WITH_DRAG_AND_DROP
|
---|
1366 | int UIMachineView::prepareDnd(void)
|
---|
1367 | {
|
---|
1368 | /* Enable drag & drop: */
|
---|
1369 | setAcceptDrops(true);
|
---|
1370 |
|
---|
1371 | int vrc;
|
---|
1372 |
|
---|
1373 | /* Create the drag and drop handler instance: */
|
---|
1374 | m_pDnDHandler = new UIDnDHandler(uimachine(), this /* pParent */);
|
---|
1375 | if (m_pDnDHandler)
|
---|
1376 | {
|
---|
1377 | vrc = m_pDnDHandler->init();
|
---|
1378 | }
|
---|
1379 | else
|
---|
1380 | vrc = VERR_NO_MEMORY;
|
---|
1381 |
|
---|
1382 | if (RT_FAILURE(vrc))
|
---|
1383 | LogRel(("DnD: Initialization failed with %Rrc\n", vrc));
|
---|
1384 | return vrc;
|
---|
1385 | }
|
---|
1386 | #endif /* VBOX_WITH_DRAG_AND_DROP */
|
---|
1387 |
|
---|
1388 | void UIMachineView::prepareFilters()
|
---|
1389 | {
|
---|
1390 | /* Enable MouseMove events: */
|
---|
1391 | viewport()->setMouseTracking(true);
|
---|
1392 |
|
---|
1393 | /* We have to watch for own events too: */
|
---|
1394 | installEventFilter(this);
|
---|
1395 |
|
---|
1396 | /* QScrollView does the below on its own, but let's
|
---|
1397 | * do it anyway for the case it will not do it in the future: */
|
---|
1398 | viewport()->installEventFilter(this);
|
---|
1399 |
|
---|
1400 | /* We want to be notified on some parent's events: */
|
---|
1401 | machineWindow()->installEventFilter(this);
|
---|
1402 | }
|
---|
1403 |
|
---|
1404 | void UIMachineView::prepareConnections()
|
---|
1405 | {
|
---|
1406 | /* UICommon connections: */
|
---|
1407 | connect(&uiCommon(), &UICommon::sigAskToDetachCOM, this, &UIMachineView::sltDetachCOM);
|
---|
1408 |
|
---|
1409 | /* Desktop resolution change (e.g. monitor hotplug): */
|
---|
1410 | connect(gpDesktop, &UIDesktopWidgetWatchdog::sigHostScreenResized,
|
---|
1411 | this, &UIMachineView::sltDesktopResized);
|
---|
1412 |
|
---|
1413 | /* Scale-factor change: */
|
---|
1414 | connect(gEDataManager, &UIExtraDataManager::sigScaleFactorChange,
|
---|
1415 | this, &UIMachineView::sltHandleScaleFactorChange);
|
---|
1416 | /* Scaling-optimization change: */
|
---|
1417 | connect(gEDataManager, &UIExtraDataManager::sigScalingOptimizationTypeChange,
|
---|
1418 | this, &UIMachineView::sltHandleScalingOptimizationChange);
|
---|
1419 |
|
---|
1420 | /* Action-pool connections: */
|
---|
1421 | UIActionPoolRuntime *pActionPoolRuntime = qobject_cast<UIActionPoolRuntime*>(actionPool());
|
---|
1422 | if (pActionPoolRuntime)
|
---|
1423 | {
|
---|
1424 | connect(pActionPoolRuntime, &UIActionPoolRuntime::sigNotifyAboutTriggeringViewScreenToggle,
|
---|
1425 | this, &UIMachineView::sltHandleActionTriggerViewScreenToggle);
|
---|
1426 | connect(pActionPoolRuntime, &UIActionPoolRuntime::sigNotifyAboutTriggeringViewScreenResize,
|
---|
1427 | this, &UIMachineView::sltHandleActionTriggerViewScreenResize);
|
---|
1428 | }
|
---|
1429 |
|
---|
1430 | /* Translate initially: */
|
---|
1431 | connect(&translationEventListener(), &UITranslationEventListener::sigRetranslateUI,
|
---|
1432 | this, &UIMachineView::sltRetranslateUI);
|
---|
1433 | sltRetranslateUI();
|
---|
1434 | }
|
---|
1435 |
|
---|
1436 | void UIMachineView::prepareConsoleConnections()
|
---|
1437 | {
|
---|
1438 | /* Machine state-change updater: */
|
---|
1439 | connect(uimachine(), &UIMachine::sigMachineStateChange, this, &UIMachineView::sltMachineStateChanged);
|
---|
1440 | /* Mouse pointer shape updater: */
|
---|
1441 | connect(uimachine(), &UIMachine::sigMousePointerShapeChange, this, &UIMachineView::sltMousePointerShapeChange);
|
---|
1442 | }
|
---|
1443 |
|
---|
1444 | #ifdef VBOX_WITH_DRAG_AND_DROP
|
---|
1445 | void UIMachineView::cleanupDnd()
|
---|
1446 | {
|
---|
1447 | delete m_pDnDHandler;
|
---|
1448 | m_pDnDHandler = 0;
|
---|
1449 | }
|
---|
1450 | #endif /* VBOX_WITH_DRAG_AND_DROP */
|
---|
1451 |
|
---|
1452 | void UIMachineView::cleanupFrameBuffer()
|
---|
1453 | {
|
---|
1454 | /* Make sure framebuffer still present: */
|
---|
1455 | if (!frameBuffer())
|
---|
1456 | return;
|
---|
1457 |
|
---|
1458 | /* Mark framebuffer as unused: */
|
---|
1459 | LogRelFlow(("GUI: UIMachineView::cleanupFrameBuffer: Stop EMT callbacks accepting for screen: %d\n", screenId()));
|
---|
1460 | frameBuffer()->setMarkAsUnused(true);
|
---|
1461 |
|
---|
1462 | /* Process pending framebuffer events: */
|
---|
1463 | QApplication::sendPostedEvents(this, QEvent::MetaCall);
|
---|
1464 |
|
---|
1465 | /* Temporarily detach the framebuffer from IDisplay before detaching
|
---|
1466 | * from view in order to respect the thread synchonisation logic (see UIFrameBuffer.h).
|
---|
1467 | * Note: VBOX_WITH_CROGL additionally requires us to call DetachFramebuffer
|
---|
1468 | * to ensure 3D gets notified of view being destroyed... */
|
---|
1469 | frameBuffer()->detach();
|
---|
1470 |
|
---|
1471 | /* Detach framebuffer from view: */
|
---|
1472 | frameBuffer()->setView(0);
|
---|
1473 | }
|
---|
1474 |
|
---|
1475 | void UIMachineView::cleanupNativeFilters()
|
---|
1476 | {
|
---|
1477 | /* If native event filter exists: */
|
---|
1478 | if (m_pNativeEventFilter)
|
---|
1479 | {
|
---|
1480 | /* Uninstall/destroy existing native event filter: */
|
---|
1481 | qApp->removeNativeEventFilter(m_pNativeEventFilter);
|
---|
1482 | delete m_pNativeEventFilter;
|
---|
1483 | m_pNativeEventFilter = 0;
|
---|
1484 | }
|
---|
1485 | }
|
---|
1486 |
|
---|
1487 | UIActionPool* UIMachineView::actionPool() const
|
---|
1488 | {
|
---|
1489 | return machineWindow()->actionPool();
|
---|
1490 | }
|
---|
1491 |
|
---|
1492 | QSize UIMachineView::sizeHint() const
|
---|
1493 | {
|
---|
1494 | /* Make sure frame-buffer exists: */
|
---|
1495 | QSize size;
|
---|
1496 | if (!frameBuffer())
|
---|
1497 | size = QSize(640, 480);
|
---|
1498 | else
|
---|
1499 | {
|
---|
1500 | // WORKAROUND:
|
---|
1501 | // Temporarily restrict the size to prevent a brief resize to the frame-buffer dimensions when
|
---|
1502 | // we exit full-screen. This is only applied if the frame-buffer is at full-screen dimensions
|
---|
1503 | // and until the first machine view resize.
|
---|
1504 | /* Get the frame-buffer dimensions: */
|
---|
1505 | QSize frameBufferSize(frameBuffer()->width(), frameBuffer()->height());
|
---|
1506 | /* Take the scale-factor(s) into account: */
|
---|
1507 | frameBufferSize = scaledForward(frameBufferSize);
|
---|
1508 | /* Check against the last full-screen size: */
|
---|
1509 | if (frameBufferSize == uimachine()->lastFullScreenSize(screenId()) && m_sizeHintOverride.isValid())
|
---|
1510 | return m_sizeHintOverride;
|
---|
1511 |
|
---|
1512 | /* Get frame-buffer size-hint: */
|
---|
1513 | size = QSize(frameBuffer()->width(), frameBuffer()->height());
|
---|
1514 | /* Take the scale-factor(s) into account: */
|
---|
1515 | size = scaledForward(size);
|
---|
1516 |
|
---|
1517 | #ifdef VBOX_WITH_DEBUGGER_GUI
|
---|
1518 | /// @todo Fix all DEBUGGER stuff!
|
---|
1519 | // WORKAROUND:
|
---|
1520 | // Really ugly workaround for the resizing to 9x1
|
---|
1521 | // done by DevVGA if provoked before power on.
|
---|
1522 | if (size.width() < 16 || size.height() < 16)
|
---|
1523 | if (uiCommon().shouldStartPaused() || uiCommon().isDebuggerAutoShowEnabled())
|
---|
1524 | size = QSize(640, 480);
|
---|
1525 | #endif /* VBOX_WITH_DEBUGGER_GUI */
|
---|
1526 | }
|
---|
1527 |
|
---|
1528 | /* Return the resulting size-hint: */
|
---|
1529 | return QSize(size.width() + frameWidth() * 2, size.height() + frameWidth() * 2);
|
---|
1530 | }
|
---|
1531 |
|
---|
1532 | QSize UIMachineView::storedGuestScreenSizeHint() const
|
---|
1533 | {
|
---|
1534 | /* Load guest-screen size-hint: */
|
---|
1535 | QSize sizeHint = gEDataManager->lastGuestScreenSizeHint(m_uScreenId, uiCommon().managedVMUuid());
|
---|
1536 |
|
---|
1537 | /* Invent the default if necessary: */
|
---|
1538 | if (!sizeHint.isValid())
|
---|
1539 | sizeHint = QSize(800, 600);
|
---|
1540 |
|
---|
1541 | /* Take the scale-factor(s) into account: */
|
---|
1542 | sizeHint = scaledForward(sizeHint);
|
---|
1543 |
|
---|
1544 | /* Return size-hint: */
|
---|
1545 | LogRel2(("GUI: UIMachineView::storedGuestScreenSizeHint: Acquired guest-screen size-hint for screen %d as %dx%d\n",
|
---|
1546 | (int)screenId(), sizeHint.width(), sizeHint.height()));
|
---|
1547 | return sizeHint;
|
---|
1548 | }
|
---|
1549 |
|
---|
1550 | void UIMachineView::setStoredGuestScreenSizeHint(const QSize &sizeHint)
|
---|
1551 | {
|
---|
1552 | /* Save guest-screen size-hint: */
|
---|
1553 | LogRel2(("GUI: UIMachineView::setStoredGuestScreenSizeHint: Storing guest-screen size-hint for screen %d as %dx%d\n",
|
---|
1554 | (int)screenId(), sizeHint.width(), sizeHint.height()));
|
---|
1555 | gEDataManager->setLastGuestScreenSizeHint(m_uScreenId, sizeHint, uiCommon().managedVMUuid());
|
---|
1556 | }
|
---|
1557 |
|
---|
1558 | QSize UIMachineView::requestedGuestScreenSizeHint() const
|
---|
1559 | {
|
---|
1560 | /* Acquire last guest-screen size-hint set, if any: */
|
---|
1561 | bool fDummy = false;
|
---|
1562 | long iDummy = 0;
|
---|
1563 | ulong uWidth = 0, uHeight = 0, uDummy = 0;
|
---|
1564 | uimachine()->acquireVideoModeHint(screenId(), fDummy, fDummy,
|
---|
1565 | iDummy, iDummy, uWidth, uHeight,
|
---|
1566 | uDummy);
|
---|
1567 |
|
---|
1568 | /* Acquire effective frame-buffer size otherwise: */
|
---|
1569 | if (uWidth == 0 || uHeight == 0)
|
---|
1570 | {
|
---|
1571 | uWidth = frameBuffer()->width();
|
---|
1572 | uHeight = frameBuffer()->height();
|
---|
1573 | }
|
---|
1574 |
|
---|
1575 | /* Return result: */
|
---|
1576 | return QSize((int)uWidth, (int)uHeight);
|
---|
1577 | }
|
---|
1578 |
|
---|
1579 | bool UIMachineView::guestScreenVisibilityStatus() const
|
---|
1580 | {
|
---|
1581 | /* Always 'true' for primary guest-screen: */
|
---|
1582 | if (m_uScreenId == 0)
|
---|
1583 | return true;
|
---|
1584 |
|
---|
1585 | /* Actual value for other guest-screens: */
|
---|
1586 | return gEDataManager->lastGuestScreenVisibilityStatus(m_uScreenId, uiCommon().managedVMUuid());
|
---|
1587 | }
|
---|
1588 |
|
---|
1589 | void UIMachineView::handleScaleChange()
|
---|
1590 | {
|
---|
1591 | LogRel(("GUI: UIMachineView::handleScaleChange: Screen=%d\n",
|
---|
1592 | (unsigned long)m_uScreenId));
|
---|
1593 |
|
---|
1594 | /* If machine-window is visible: */
|
---|
1595 | if (uimachine()->isScreenVisible(m_uScreenId))
|
---|
1596 | {
|
---|
1597 | /* For 'scale' mode: */
|
---|
1598 | if (visualStateType() == UIVisualStateType_Scale)
|
---|
1599 | {
|
---|
1600 | /* Assign new frame-buffer logical-size: */
|
---|
1601 | QSize scaledSize = size();
|
---|
1602 | const double dDevicePixelRatioFormal = frameBuffer()->devicePixelRatio();
|
---|
1603 | const double dDevicePixelRatioActual = frameBuffer()->devicePixelRatioActual();
|
---|
1604 | scaledSize *= dDevicePixelRatioFormal;
|
---|
1605 | if (!frameBuffer()->useUnscaledHiDPIOutput())
|
---|
1606 | scaledSize /= dDevicePixelRatioActual;
|
---|
1607 | frameBuffer()->setScaledSize(scaledSize);
|
---|
1608 | }
|
---|
1609 | /* For other than 'scale' mode: */
|
---|
1610 | else
|
---|
1611 | {
|
---|
1612 | /* Adjust maximum-size restriction for machine-view: */
|
---|
1613 | setMaximumSize(sizeHint());
|
---|
1614 |
|
---|
1615 | /* Force machine-window update own layout: */
|
---|
1616 | QCoreApplication::sendPostedEvents(0, QEvent::LayoutRequest);
|
---|
1617 |
|
---|
1618 | /* Update machine-view sliders: */
|
---|
1619 | updateSliders();
|
---|
1620 |
|
---|
1621 | /* By some reason Win host forgets to update machine-window central-widget
|
---|
1622 | * after main-layout was updated, let's do it for all the hosts: */
|
---|
1623 | machineWindow()->centralWidget()->update();
|
---|
1624 |
|
---|
1625 | /* Normalize 'normal' machine-window geometry: */
|
---|
1626 | if (visualStateType() == UIVisualStateType_Normal)
|
---|
1627 | machineWindow()->normalizeGeometry(true /* adjust position */, machineWindow()->shouldResizeToGuestDisplay());
|
---|
1628 | }
|
---|
1629 |
|
---|
1630 | /* Perform frame-buffer rescaling: */
|
---|
1631 | frameBuffer()->performRescale();
|
---|
1632 | }
|
---|
1633 |
|
---|
1634 | LogRelFlow(("GUI: UIMachineView::handleScaleChange: Complete for Screen=%d\n",
|
---|
1635 | (unsigned long)m_uScreenId));
|
---|
1636 | }
|
---|
1637 |
|
---|
1638 | void UIMachineView::resetPausePixmap()
|
---|
1639 | {
|
---|
1640 | /* Reset pixmap(s): */
|
---|
1641 | m_pausePixmap = QPixmap();
|
---|
1642 | m_pausePixmapScaled = QPixmap();
|
---|
1643 | }
|
---|
1644 |
|
---|
1645 | void UIMachineView::takePausePixmapLive()
|
---|
1646 | {
|
---|
1647 | /* Prepare a screen-shot: */
|
---|
1648 | QImage screenShot = QImage(frameBuffer()->width(), frameBuffer()->height(), QImage::Format_RGB32);
|
---|
1649 | /* Which will be a 'black image' by default. */
|
---|
1650 | screenShot.fill(0);
|
---|
1651 |
|
---|
1652 | /* Acquire screen-shot image: */
|
---|
1653 | uimachine()->acquireScreenShot(screenId(), screenShot.width(), screenShot.height(), KBitmapFormat_BGR0, screenShot.bits());
|
---|
1654 |
|
---|
1655 | /* Take the device-pixel-ratio into account: */
|
---|
1656 | const double dDevicePixelRatioActual = frameBuffer()->devicePixelRatioActual();
|
---|
1657 | if (!frameBuffer()->useUnscaledHiDPIOutput() && dDevicePixelRatioActual != 1.0)
|
---|
1658 | screenShot = screenShot.scaled(screenShot.size() * dDevicePixelRatioActual,
|
---|
1659 | Qt::IgnoreAspectRatio, Qt::SmoothTransformation);
|
---|
1660 |
|
---|
1661 | /* Dim screen-shot if it is Ok: */
|
---|
1662 | if (!screenShot.isNull())
|
---|
1663 | dimImage(screenShot);
|
---|
1664 |
|
---|
1665 | /* Finally copy the screen-shot to pause-pixmap: */
|
---|
1666 | m_pausePixmap = QPixmap::fromImage(screenShot);
|
---|
1667 |
|
---|
1668 | /* Take the device-pixel-ratio into account: */
|
---|
1669 | m_pausePixmap.setDevicePixelRatio(frameBuffer()->devicePixelRatio());
|
---|
1670 |
|
---|
1671 | /* Update scaled pause pixmap: */
|
---|
1672 | updateScaledPausePixmap();
|
---|
1673 | }
|
---|
1674 |
|
---|
1675 | void UIMachineView::takePausePixmapSnapshot()
|
---|
1676 | {
|
---|
1677 | /* Acquire the screen-data from the saved-state: */
|
---|
1678 | ulong uDummy = 0;
|
---|
1679 | QVector<BYTE> screenData;
|
---|
1680 | uimachine()->acquireSavedScreenshot(m_uScreenId, KBitmapFormat_PNG, uDummy, uDummy, screenData);
|
---|
1681 | if (screenData.isEmpty())
|
---|
1682 | return;
|
---|
1683 |
|
---|
1684 | /* Acquire the screen-data properties from the saved-state: */
|
---|
1685 | long iDummy = 0;
|
---|
1686 | ulong uGuestWidth = 0, uGuestHeight = 0;
|
---|
1687 | bool fDummy = true;
|
---|
1688 | uimachine()->acquireSavedGuestScreenInfo(m_uScreenId, iDummy, iDummy, uGuestWidth, uGuestHeight, fDummy);
|
---|
1689 |
|
---|
1690 | /* Calculate effective size: */
|
---|
1691 | QSize effectiveSize = uGuestWidth > 0 ? QSize(uGuestWidth, uGuestHeight) : storedGuestScreenSizeHint();
|
---|
1692 |
|
---|
1693 | /* Take the device-pixel-ratio into account: */
|
---|
1694 | const double dDevicePixelRatioActual = frameBuffer()->devicePixelRatioActual();
|
---|
1695 | if (!frameBuffer()->useUnscaledHiDPIOutput() && dDevicePixelRatioActual != 1.0)
|
---|
1696 | effectiveSize *= dDevicePixelRatioActual;
|
---|
1697 |
|
---|
1698 | /* Create a screen-shot on the basis of the screen-data we have in saved-state: */
|
---|
1699 | QImage screenShot = QImage::fromData(screenData.data(), screenData.size(), "PNG").scaled(effectiveSize);
|
---|
1700 |
|
---|
1701 | /* Dim screen-shot if it is Ok: */
|
---|
1702 | if (!screenShot.isNull())
|
---|
1703 | dimImage(screenShot);
|
---|
1704 |
|
---|
1705 | /* Finally copy the screen-shot to pause-pixmap: */
|
---|
1706 | m_pausePixmap = QPixmap::fromImage(screenShot);
|
---|
1707 |
|
---|
1708 | /* Take the device-pixel-ratio into account: */
|
---|
1709 | m_pausePixmap.setDevicePixelRatio(frameBuffer()->devicePixelRatio());
|
---|
1710 |
|
---|
1711 | /* Update scaled pause pixmap: */
|
---|
1712 | updateScaledPausePixmap();
|
---|
1713 | }
|
---|
1714 |
|
---|
1715 | void UIMachineView::updateScaledPausePixmap()
|
---|
1716 | {
|
---|
1717 | /* Make sure pause pixmap is not null: */
|
---|
1718 | if (pausePixmap().isNull())
|
---|
1719 | return;
|
---|
1720 |
|
---|
1721 | /* Make sure scaled-size is not null: */
|
---|
1722 | QSize scaledSize = frameBuffer()->scaledSize();
|
---|
1723 | if (!scaledSize.isValid())
|
---|
1724 | return;
|
---|
1725 |
|
---|
1726 | /* Take the device-pixel-ratio into account: */
|
---|
1727 | const double dDevicePixelRatioActual = frameBuffer()->devicePixelRatioActual();
|
---|
1728 | if (!frameBuffer()->useUnscaledHiDPIOutput() && dDevicePixelRatioActual != 1.0)
|
---|
1729 | scaledSize *= dDevicePixelRatioActual;
|
---|
1730 |
|
---|
1731 | /* Update pause pixmap finally: */
|
---|
1732 | m_pausePixmapScaled = pausePixmap().scaled(scaledSize, Qt::IgnoreAspectRatio, Qt::SmoothTransformation);
|
---|
1733 |
|
---|
1734 | /* Take the device-pixel-ratio into account: */
|
---|
1735 | m_pausePixmapScaled.setDevicePixelRatio(frameBuffer()->devicePixelRatio());
|
---|
1736 | }
|
---|
1737 |
|
---|
1738 | void UIMachineView::updateSliders()
|
---|
1739 | {
|
---|
1740 | /* Make sure framebuffer still present: */
|
---|
1741 | if (!frameBuffer())
|
---|
1742 | return;
|
---|
1743 |
|
---|
1744 | /* Get current viewport size: */
|
---|
1745 | QSize curViewportSize = viewport()->size();
|
---|
1746 | /* Get maximum viewport size: */
|
---|
1747 | const QSize maxViewportSize = maximumViewportSize();
|
---|
1748 | /* Get current frame-buffer size: */
|
---|
1749 | QSize frameBufferSize = QSize(frameBuffer()->width(), frameBuffer()->height());
|
---|
1750 |
|
---|
1751 | /* Take the scale-factor(s) into account: */
|
---|
1752 | frameBufferSize = scaledForward(frameBufferSize);
|
---|
1753 |
|
---|
1754 | /* If maximum viewport size can cover whole frame-buffer => no scroll-bars required: */
|
---|
1755 | if (maxViewportSize.expandedTo(frameBufferSize) == maxViewportSize)
|
---|
1756 | curViewportSize = maxViewportSize;
|
---|
1757 |
|
---|
1758 | /* What length we want scroll-bars of? */
|
---|
1759 | int xRange = frameBufferSize.width() - curViewportSize.width();
|
---|
1760 | int yRange = frameBufferSize.height() - curViewportSize.height();
|
---|
1761 |
|
---|
1762 | /* Take the device-pixel-ratio into account: */
|
---|
1763 | const double dDevicePixelRatioFormal = frameBuffer()->devicePixelRatio();
|
---|
1764 | const double dDevicePixelRatioActual = frameBuffer()->devicePixelRatioActual();
|
---|
1765 | xRange *= dDevicePixelRatioFormal;
|
---|
1766 | yRange *= dDevicePixelRatioFormal;
|
---|
1767 | if (!frameBuffer()->useUnscaledHiDPIOutput())
|
---|
1768 | {
|
---|
1769 | xRange /= dDevicePixelRatioActual;
|
---|
1770 | yRange /= dDevicePixelRatioActual;
|
---|
1771 | }
|
---|
1772 |
|
---|
1773 | /* Configure scroll-bars: */
|
---|
1774 | horizontalScrollBar()->setRange(0, xRange);
|
---|
1775 | verticalScrollBar()->setRange(0, yRange);
|
---|
1776 | horizontalScrollBar()->setPageStep(curViewportSize.width());
|
---|
1777 | verticalScrollBar()->setPageStep(curViewportSize.height());
|
---|
1778 | }
|
---|
1779 |
|
---|
1780 | void UIMachineView::dimImage(QImage &img)
|
---|
1781 | {
|
---|
1782 | for (int y = 0; y < img.height(); ++ y)
|
---|
1783 | {
|
---|
1784 | if (y % 2)
|
---|
1785 | {
|
---|
1786 | if (img.depth() == 32)
|
---|
1787 | {
|
---|
1788 | for (int x = 0; x < img.width(); ++ x)
|
---|
1789 | {
|
---|
1790 | int gray = qGray(img.pixel (x, y)) / 2;
|
---|
1791 | img.setPixel(x, y, qRgb (gray, gray, gray));
|
---|
1792 | }
|
---|
1793 | }
|
---|
1794 | else
|
---|
1795 | {
|
---|
1796 | ::memset(img.scanLine (y), 0, img.bytesPerLine());
|
---|
1797 | }
|
---|
1798 | }
|
---|
1799 | else
|
---|
1800 | {
|
---|
1801 | if (img.depth() == 32)
|
---|
1802 | {
|
---|
1803 | for (int x = 0; x < img.width(); ++ x)
|
---|
1804 | {
|
---|
1805 | int gray = (2 * qGray (img.pixel (x, y))) / 3;
|
---|
1806 | img.setPixel(x, y, qRgb (gray, gray, gray));
|
---|
1807 | }
|
---|
1808 | }
|
---|
1809 | }
|
---|
1810 | }
|
---|
1811 | }
|
---|
1812 |
|
---|
1813 | void UIMachineView::scrollContentsBy(int dx, int dy)
|
---|
1814 | {
|
---|
1815 | /* Call to base-class: */
|
---|
1816 | QAbstractScrollArea::scrollContentsBy(dx, dy);
|
---|
1817 |
|
---|
1818 | /* Update console's display viewport and 3D overlay: */
|
---|
1819 | updateViewport();
|
---|
1820 | }
|
---|
1821 |
|
---|
1822 | #ifdef VBOX_WS_MAC
|
---|
1823 | void UIMachineView::updateDockIcon()
|
---|
1824 | {
|
---|
1825 | machineLogic()->updateDockIcon();
|
---|
1826 | }
|
---|
1827 |
|
---|
1828 | CGImageRef UIMachineView::frameBuffertoCGImageRef(UIFrameBuffer *pFrameBuffer)
|
---|
1829 | {
|
---|
1830 | CGImageRef ir = 0;
|
---|
1831 | CGColorSpaceRef cs = CGColorSpaceCreateDeviceRGB();
|
---|
1832 | if (cs)
|
---|
1833 | {
|
---|
1834 | /* Create the image copy of the framebuffer */
|
---|
1835 | CGDataProviderRef dp = CGDataProviderCreateWithData(pFrameBuffer, pFrameBuffer->address(), pFrameBuffer->bitsPerPixel() / 8 * pFrameBuffer->width() * pFrameBuffer->height(), NULL);
|
---|
1836 | if (dp)
|
---|
1837 | {
|
---|
1838 | ir = CGImageCreate(pFrameBuffer->width(), pFrameBuffer->height(), 8, 32, pFrameBuffer->bytesPerLine(), cs,
|
---|
1839 | kCGImageAlphaNoneSkipFirst | kCGBitmapByteOrder32Host, dp, 0, false,
|
---|
1840 | kCGRenderingIntentDefault);
|
---|
1841 | CGDataProviderRelease(dp);
|
---|
1842 | }
|
---|
1843 | CGColorSpaceRelease(cs);
|
---|
1844 | }
|
---|
1845 | return ir;
|
---|
1846 | }
|
---|
1847 | #endif /* VBOX_WS_MAC */
|
---|
1848 |
|
---|
1849 | bool UIMachineView::isFullscreenOrSeamless() const
|
---|
1850 | {
|
---|
1851 | return visualStateType() == UIVisualStateType_Fullscreen
|
---|
1852 | || visualStateType() == UIVisualStateType_Seamless;
|
---|
1853 | }
|
---|
1854 |
|
---|
1855 | bool UIMachineView::eventFilter(QObject *pWatched, QEvent *pEvent)
|
---|
1856 | {
|
---|
1857 | if (pWatched == viewport())
|
---|
1858 | {
|
---|
1859 | switch (pEvent->type())
|
---|
1860 | {
|
---|
1861 | case QEvent::Resize:
|
---|
1862 | {
|
---|
1863 | /* Notify framebuffer about viewport resize: */
|
---|
1864 | QResizeEvent *pResizeEvent = static_cast<QResizeEvent*>(pEvent);
|
---|
1865 | if (frameBuffer())
|
---|
1866 | frameBuffer()->viewportResized(pResizeEvent);
|
---|
1867 | /* Update console's display viewport and 3D overlay: */
|
---|
1868 | updateViewport();
|
---|
1869 | break;
|
---|
1870 | }
|
---|
1871 | default:
|
---|
1872 | break;
|
---|
1873 | }
|
---|
1874 | }
|
---|
1875 |
|
---|
1876 | if (pWatched == this)
|
---|
1877 | {
|
---|
1878 | switch (pEvent->type())
|
---|
1879 | {
|
---|
1880 | case QEvent::Move:
|
---|
1881 | {
|
---|
1882 | /* Update console's display viewport and 3D overlay: */
|
---|
1883 | updateViewport();
|
---|
1884 | break;
|
---|
1885 | }
|
---|
1886 | default:
|
---|
1887 | break;
|
---|
1888 | }
|
---|
1889 | }
|
---|
1890 |
|
---|
1891 | if (pWatched == machineWindow())
|
---|
1892 | {
|
---|
1893 | switch (pEvent->type())
|
---|
1894 | {
|
---|
1895 | case QEvent::WindowStateChange:
|
---|
1896 | {
|
---|
1897 | /* During minimizing and state restoring machineWindow() gets
|
---|
1898 | * the focus which belongs to console view window, so returning it properly. */
|
---|
1899 | QWindowStateChangeEvent *pWindowEvent = static_cast<QWindowStateChangeEvent*>(pEvent);
|
---|
1900 | if (pWindowEvent->oldState() & Qt::WindowMinimized)
|
---|
1901 | {
|
---|
1902 | if (QApplication::focusWidget())
|
---|
1903 | {
|
---|
1904 | QApplication::focusWidget()->clearFocus();
|
---|
1905 | qApp->processEvents();
|
---|
1906 | }
|
---|
1907 | QTimer::singleShot(0, this, SLOT(setFocus()));
|
---|
1908 | }
|
---|
1909 | break;
|
---|
1910 | }
|
---|
1911 | case QEvent::Move:
|
---|
1912 | {
|
---|
1913 | /* Get current host-screen number: */
|
---|
1914 | const int iCurrentHostScreenNumber = UIDesktopWidgetWatchdog::screenNumber(this);
|
---|
1915 | if (m_iHostScreenNumber != iCurrentHostScreenNumber)
|
---|
1916 | {
|
---|
1917 | /* Recache current host screen: */
|
---|
1918 | m_iHostScreenNumber = iCurrentHostScreenNumber;
|
---|
1919 | /* Reapply machine-view scale-factor if necessary: */
|
---|
1920 | applyMachineViewScaleFactor();
|
---|
1921 | /* For 'normal'/'scaled' visual state type: */
|
---|
1922 | if ( visualStateType() == UIVisualStateType_Normal
|
---|
1923 | || visualStateType() == UIVisualStateType_Scale)
|
---|
1924 | {
|
---|
1925 | /* Make sure action-pool is of 'runtime' type: */
|
---|
1926 | UIActionPoolRuntime *pActionPool = actionPool() && actionPool()->toRuntime() ? actionPool()->toRuntime() : 0;
|
---|
1927 | AssertPtr(pActionPool);
|
---|
1928 | if (pActionPool)
|
---|
1929 | {
|
---|
1930 | /* Inform action-pool about current guest-to-host screen mapping: */
|
---|
1931 | QMap<int, int> screenMap = pActionPool->hostScreenForGuestScreenMap();
|
---|
1932 | screenMap[m_uScreenId] = m_iHostScreenNumber;
|
---|
1933 | pActionPool->setHostScreenForGuestScreenMap(screenMap);
|
---|
1934 | }
|
---|
1935 | }
|
---|
1936 | }
|
---|
1937 | break;
|
---|
1938 | }
|
---|
1939 | default:
|
---|
1940 | break;
|
---|
1941 | }
|
---|
1942 | }
|
---|
1943 |
|
---|
1944 | return QAbstractScrollArea::eventFilter(pWatched, pEvent);
|
---|
1945 | }
|
---|
1946 |
|
---|
1947 | void UIMachineView::resizeEvent(QResizeEvent *pEvent)
|
---|
1948 | {
|
---|
1949 | updateSliders();
|
---|
1950 | return QAbstractScrollArea::resizeEvent(pEvent);
|
---|
1951 | }
|
---|
1952 |
|
---|
1953 | void UIMachineView::moveEvent(QMoveEvent *pEvent)
|
---|
1954 | {
|
---|
1955 | return QAbstractScrollArea::moveEvent(pEvent);
|
---|
1956 | }
|
---|
1957 |
|
---|
1958 | void UIMachineView::paintEvent(QPaintEvent *pPaintEvent)
|
---|
1959 | {
|
---|
1960 | /* Use pause-image if exists: */
|
---|
1961 | if (!pausePixmap().isNull())
|
---|
1962 | {
|
---|
1963 | /* Create viewport painter: */
|
---|
1964 | QPainter painter(viewport());
|
---|
1965 | /* Avoid painting more than necessary: */
|
---|
1966 | painter.setClipRect(pPaintEvent->rect());
|
---|
1967 | /* Can be NULL when the event arrive during COM cleanup: */
|
---|
1968 | UIFrameBuffer *pFramebuffer = frameBuffer();
|
---|
1969 | /* Take the scale-factor into account: */
|
---|
1970 | if ( pFramebuffer
|
---|
1971 | ? pFramebuffer->scaleFactor() == 1.0 && !pFramebuffer->scaledSize().isValid()
|
---|
1972 | : pausePixmapScaled().isNull())
|
---|
1973 | painter.drawPixmap(viewport()->rect().topLeft(), pausePixmap());
|
---|
1974 | else
|
---|
1975 | painter.drawPixmap(viewport()->rect().topLeft(), pausePixmapScaled());
|
---|
1976 | #ifdef VBOX_WS_MAC
|
---|
1977 | /* Update the dock icon: */
|
---|
1978 | updateDockIcon();
|
---|
1979 | #endif /* VBOX_WS_MAC */
|
---|
1980 | return;
|
---|
1981 | }
|
---|
1982 |
|
---|
1983 | /* Delegate the paint function to the UIFrameBuffer interface: */
|
---|
1984 | if (frameBuffer())
|
---|
1985 | frameBuffer()->handlePaintEvent(pPaintEvent);
|
---|
1986 | #ifdef VBOX_WS_MAC
|
---|
1987 | /* Update the dock icon if we are in the running state: */
|
---|
1988 | if (uimachine()->isRunning())
|
---|
1989 | updateDockIcon();
|
---|
1990 | #endif /* VBOX_WS_MAC */
|
---|
1991 | }
|
---|
1992 |
|
---|
1993 | void UIMachineView::focusInEvent(QFocusEvent *pEvent)
|
---|
1994 | {
|
---|
1995 | /* Call to base-class: */
|
---|
1996 | QAbstractScrollArea::focusInEvent(pEvent);
|
---|
1997 |
|
---|
1998 | /* If native event filter isn't exists: */
|
---|
1999 | if (!m_pNativeEventFilter)
|
---|
2000 | {
|
---|
2001 | /* Create/install new native event filter: */
|
---|
2002 | m_pNativeEventFilter = new UINativeEventFilter(this);
|
---|
2003 | qApp->installNativeEventFilter(m_pNativeEventFilter);
|
---|
2004 | }
|
---|
2005 | }
|
---|
2006 |
|
---|
2007 | void UIMachineView::focusOutEvent(QFocusEvent *pEvent)
|
---|
2008 | {
|
---|
2009 | /* If native event filter exists: */
|
---|
2010 | if (m_pNativeEventFilter)
|
---|
2011 | {
|
---|
2012 | /* Uninstall/destroy existing native event filter: */
|
---|
2013 | qApp->removeNativeEventFilter(m_pNativeEventFilter);
|
---|
2014 | delete m_pNativeEventFilter;
|
---|
2015 | m_pNativeEventFilter = 0;
|
---|
2016 | }
|
---|
2017 |
|
---|
2018 | /* Call to base-class: */
|
---|
2019 | QAbstractScrollArea::focusOutEvent(pEvent);
|
---|
2020 | }
|
---|
2021 | #ifdef VBOX_WS_NIX
|
---|
2022 | void UIMachineView::keyPressEvent(QKeyEvent *pEvent)
|
---|
2023 | {
|
---|
2024 | if (uiCommon().displayServerType() == VBGHDISPLAYSERVERTYPE_PURE_WAYLAND)
|
---|
2025 | uimachine()->putScancode(pEvent->nativeScanCode() - 8);
|
---|
2026 | QAbstractScrollArea::keyPressEvent(pEvent);
|
---|
2027 | }
|
---|
2028 |
|
---|
2029 | void UIMachineView::keyReleaseEvent(QKeyEvent *pEvent)
|
---|
2030 | {
|
---|
2031 | if (uiCommon().displayServerType() == VBGHDISPLAYSERVERTYPE_PURE_WAYLAND)
|
---|
2032 | uimachine()->putScancode((pEvent->nativeScanCode() - 8) | 0x80);
|
---|
2033 | QAbstractScrollArea::keyReleaseEvent(pEvent);
|
---|
2034 | }
|
---|
2035 | #endif
|
---|
2036 | #ifdef VBOX_WITH_DRAG_AND_DROP
|
---|
2037 |
|
---|
2038 | bool UIMachineView::dragAndDropCanAccept() const
|
---|
2039 | {
|
---|
2040 | bool fAccept = m_pDnDHandler;
|
---|
2041 | # ifdef VBOX_WITH_DRAG_AND_DROP_GH
|
---|
2042 | if (fAccept)
|
---|
2043 | fAccept = !m_fIsDraggingFromGuest;
|
---|
2044 | # endif
|
---|
2045 | if (fAccept)
|
---|
2046 | {
|
---|
2047 | KDnDMode enmDnDMode = KDnDMode_Disabled;
|
---|
2048 | uimachine()->acquireDnDMode(enmDnDMode);
|
---|
2049 | fAccept = enmDnDMode != KDnDMode_Disabled;
|
---|
2050 | }
|
---|
2051 | return fAccept;
|
---|
2052 | }
|
---|
2053 |
|
---|
2054 | bool UIMachineView::dragAndDropIsActive() const
|
---|
2055 | {
|
---|
2056 | bool fActive = m_pDnDHandler;
|
---|
2057 | if (fActive)
|
---|
2058 | {
|
---|
2059 | KDnDMode enmDnDMode = KDnDMode_Disabled;
|
---|
2060 | uimachine()->acquireDnDMode(enmDnDMode);
|
---|
2061 | fActive = enmDnDMode != KDnDMode_Disabled;
|
---|
2062 | }
|
---|
2063 | return fActive;
|
---|
2064 | }
|
---|
2065 |
|
---|
2066 | void UIMachineView::dragEnterEvent(QDragEnterEvent *pEvent)
|
---|
2067 | {
|
---|
2068 | AssertPtrReturnVoid(pEvent);
|
---|
2069 |
|
---|
2070 | int rc = dragAndDropCanAccept() ? VINF_SUCCESS : VERR_ACCESS_DENIED;
|
---|
2071 | if (RT_SUCCESS(rc))
|
---|
2072 | {
|
---|
2073 | /* Get mouse-pointer location. */
|
---|
2074 | const QPoint &cpnt = viewportToContents(pEvent->position().toPoint());
|
---|
2075 |
|
---|
2076 | /* Ask the target for starting a DnD event. */
|
---|
2077 | Qt::DropAction result = m_pDnDHandler->dragEnter(screenId(),
|
---|
2078 | frameBuffer()->convertHostXTo(cpnt.x()),
|
---|
2079 | frameBuffer()->convertHostYTo(cpnt.y()),
|
---|
2080 | pEvent->proposedAction(),
|
---|
2081 | pEvent->possibleActions(),
|
---|
2082 | pEvent->mimeData());
|
---|
2083 |
|
---|
2084 | /* Set the DnD action returned by the guest. */
|
---|
2085 | pEvent->setDropAction(result);
|
---|
2086 | pEvent->accept();
|
---|
2087 | }
|
---|
2088 |
|
---|
2089 | DNDDEBUG(("DnD: dragEnterEvent ended with rc=%Rrc\n", rc));
|
---|
2090 | }
|
---|
2091 |
|
---|
2092 | void UIMachineView::dragMoveEvent(QDragMoveEvent *pEvent)
|
---|
2093 | {
|
---|
2094 | AssertPtrReturnVoid(pEvent);
|
---|
2095 |
|
---|
2096 | int rc = dragAndDropCanAccept() ? VINF_SUCCESS : VERR_ACCESS_DENIED;
|
---|
2097 | if (RT_SUCCESS(rc))
|
---|
2098 | {
|
---|
2099 | /* Get mouse-pointer location. */
|
---|
2100 | const QPoint &cpnt = viewportToContents(pEvent->position().toPoint());
|
---|
2101 |
|
---|
2102 | /* Ask the guest for moving the drop cursor. */
|
---|
2103 | Qt::DropAction result = m_pDnDHandler->dragMove(screenId(),
|
---|
2104 | frameBuffer()->convertHostXTo(cpnt.x()),
|
---|
2105 | frameBuffer()->convertHostYTo(cpnt.y()),
|
---|
2106 | pEvent->proposedAction(),
|
---|
2107 | pEvent->possibleActions(),
|
---|
2108 | pEvent->mimeData());
|
---|
2109 |
|
---|
2110 | /* Set the DnD action returned by the guest. */
|
---|
2111 | pEvent->setDropAction(result);
|
---|
2112 | pEvent->accept();
|
---|
2113 | }
|
---|
2114 |
|
---|
2115 | DNDDEBUG(("DnD: dragMoveEvent ended with rc=%Rrc\n", rc));
|
---|
2116 | }
|
---|
2117 |
|
---|
2118 | void UIMachineView::dragLeaveEvent(QDragLeaveEvent *pEvent)
|
---|
2119 | {
|
---|
2120 | AssertPtrReturnVoid(pEvent);
|
---|
2121 |
|
---|
2122 | int rc = dragAndDropCanAccept() ? VINF_SUCCESS : VERR_ACCESS_DENIED;
|
---|
2123 | if (RT_SUCCESS(rc))
|
---|
2124 | {
|
---|
2125 | m_pDnDHandler->dragLeave(screenId());
|
---|
2126 |
|
---|
2127 | pEvent->accept();
|
---|
2128 | }
|
---|
2129 |
|
---|
2130 | DNDDEBUG(("DnD: dragLeaveEvent ended with rc=%Rrc\n", rc));
|
---|
2131 | }
|
---|
2132 |
|
---|
2133 | void UIMachineView::dropEvent(QDropEvent *pEvent)
|
---|
2134 | {
|
---|
2135 | AssertPtrReturnVoid(pEvent);
|
---|
2136 |
|
---|
2137 | int rc = dragAndDropCanAccept() ? VINF_SUCCESS : VERR_ACCESS_DENIED;
|
---|
2138 | if (RT_SUCCESS(rc))
|
---|
2139 | {
|
---|
2140 | /* Get mouse-pointer location. */
|
---|
2141 | const QPoint &cpnt = viewportToContents(pEvent->position().toPoint());
|
---|
2142 |
|
---|
2143 | /* Ask the guest for dropping data. */
|
---|
2144 | Qt::DropAction result = m_pDnDHandler->dragDrop(screenId(),
|
---|
2145 | frameBuffer()->convertHostXTo(cpnt.x()),
|
---|
2146 | frameBuffer()->convertHostYTo(cpnt.y()),
|
---|
2147 | pEvent->proposedAction(),
|
---|
2148 | pEvent->possibleActions(),
|
---|
2149 | pEvent->mimeData());
|
---|
2150 |
|
---|
2151 | /* Set the DnD action returned by the guest. */
|
---|
2152 | pEvent->setDropAction(result);
|
---|
2153 | pEvent->accept();
|
---|
2154 | }
|
---|
2155 |
|
---|
2156 | DNDDEBUG(("DnD: dropEvent ended with rc=%Rrc\n", rc));
|
---|
2157 | }
|
---|
2158 |
|
---|
2159 | #endif /* VBOX_WITH_DRAG_AND_DROP */
|
---|
2160 |
|
---|
2161 | QSize UIMachineView::scaledForward(QSize size) const
|
---|
2162 | {
|
---|
2163 | /* Take the scale-factor into account: */
|
---|
2164 | const double dScaleFactor = frameBuffer()->scaleFactor();
|
---|
2165 | if (dScaleFactor != 1.0)
|
---|
2166 | size = QSize((int)(size.width() * dScaleFactor), (int)(size.height() * dScaleFactor));
|
---|
2167 |
|
---|
2168 | /* Take the device-pixel-ratio into account: */
|
---|
2169 | const double dDevicePixelRatioFormal = frameBuffer()->devicePixelRatio();
|
---|
2170 | const double dDevicePixelRatioActual = frameBuffer()->devicePixelRatioActual();
|
---|
2171 | if (!frameBuffer()->useUnscaledHiDPIOutput())
|
---|
2172 | size = QSize(size.width() * dDevicePixelRatioActual, size.height() * dDevicePixelRatioActual);
|
---|
2173 | size = QSize(size.width() / dDevicePixelRatioFormal, size.height() / dDevicePixelRatioFormal);
|
---|
2174 |
|
---|
2175 | /* Return result: */
|
---|
2176 | return size;
|
---|
2177 | }
|
---|
2178 |
|
---|
2179 | QSize UIMachineView::scaledBackward(QSize size) const
|
---|
2180 | {
|
---|
2181 | /* Take the device-pixel-ratio into account: */
|
---|
2182 | const double dDevicePixelRatioFormal = frameBuffer()->devicePixelRatio();
|
---|
2183 | const double dDevicePixelRatioActual = frameBuffer()->devicePixelRatioActual();
|
---|
2184 | size = QSize(size.width() * dDevicePixelRatioFormal, size.height() * dDevicePixelRatioFormal);
|
---|
2185 | if (!frameBuffer()->useUnscaledHiDPIOutput())
|
---|
2186 | size = QSize(size.width() / dDevicePixelRatioActual, size.height() / dDevicePixelRatioActual);
|
---|
2187 |
|
---|
2188 | /* Take the scale-factor into account: */
|
---|
2189 | const double dScaleFactor = frameBuffer()->scaleFactor();
|
---|
2190 | if (dScaleFactor != 1.0)
|
---|
2191 | size = QSize((int)(size.width() / dScaleFactor), (int)(size.height() / dScaleFactor));
|
---|
2192 |
|
---|
2193 | /* Return result: */
|
---|
2194 | return size;
|
---|
2195 | }
|
---|
2196 |
|
---|
2197 | void UIMachineView::updateMousePointerPixmapScaling(QPixmap &pixmap, uint &uXHot, uint &uYHot)
|
---|
2198 | {
|
---|
2199 | #if defined(VBOX_WS_MAC)
|
---|
2200 |
|
---|
2201 | /* Take into account scale-factor if necessary: */
|
---|
2202 | const double dScaleFactor = frameBuffer()->scaleFactor();
|
---|
2203 | //printf("Scale-factor: %f\n", dScaleFactor);
|
---|
2204 | if (dScaleFactor > 1.0)
|
---|
2205 | {
|
---|
2206 | /* Scale the pixmap up: */
|
---|
2207 | pixmap = pixmap.scaled(pixmap.width() * dScaleFactor, pixmap.height() * dScaleFactor,
|
---|
2208 | Qt::IgnoreAspectRatio, Qt::SmoothTransformation);
|
---|
2209 | uXHot *= dScaleFactor;
|
---|
2210 | uYHot *= dScaleFactor;
|
---|
2211 | }
|
---|
2212 |
|
---|
2213 | /* Take into account device-pixel-ratio if necessary: */
|
---|
2214 | const double dDevicePixelRatio = frameBuffer()->devicePixelRatio();
|
---|
2215 | const bool fUseUnscaledHiDPIOutput = frameBuffer()->useUnscaledHiDPIOutput();
|
---|
2216 | //printf("Device-pixel-ratio: %f, Unscaled HiDPI Output: %d\n",
|
---|
2217 | // dDevicePixelRatio, fUseUnscaledHiDPIOutput);
|
---|
2218 | if (dDevicePixelRatio > 1.0 && fUseUnscaledHiDPIOutput)
|
---|
2219 | {
|
---|
2220 | /* Scale the pixmap down: */
|
---|
2221 | pixmap.setDevicePixelRatio(dDevicePixelRatio);
|
---|
2222 | uXHot /= dDevicePixelRatio;
|
---|
2223 | uYHot /= dDevicePixelRatio;
|
---|
2224 | }
|
---|
2225 |
|
---|
2226 | #elif defined(VBOX_WS_WIN) || defined(VBOX_WS_NIX)
|
---|
2227 |
|
---|
2228 | /* We want to scale the pixmap just once, so let's prepare cumulative multiplier: */
|
---|
2229 | double dScaleMultiplier = 1.0;
|
---|
2230 |
|
---|
2231 | /* Take into account scale-factor if necessary: */
|
---|
2232 | const double dScaleFactor = frameBuffer()->scaleFactor();
|
---|
2233 | //printf("Scale-factor: %f\n", dScaleFactor);
|
---|
2234 | if (dScaleFactor > 1.0)
|
---|
2235 | dScaleMultiplier *= dScaleFactor;
|
---|
2236 |
|
---|
2237 | /* Take into account device-pixel-ratio if necessary: */
|
---|
2238 | # ifdef VBOX_WS_WIN
|
---|
2239 | const double dDevicePixelRatio = frameBuffer()->devicePixelRatio();
|
---|
2240 | # endif
|
---|
2241 | const double dDevicePixelRatioActual = frameBuffer()->devicePixelRatioActual();
|
---|
2242 | const bool fUseUnscaledHiDPIOutput = frameBuffer()->useUnscaledHiDPIOutput();
|
---|
2243 | //printf("Device-pixel-ratio/actual: %f/%f, Unscaled HiDPI Output: %d\n",
|
---|
2244 | // dDevicePixelRatio, dDevicePixelRatioActual, fUseUnscaledHiDPIOutput);
|
---|
2245 | if (dDevicePixelRatioActual > 1.0 && !fUseUnscaledHiDPIOutput)
|
---|
2246 | dScaleMultiplier *= dDevicePixelRatioActual;
|
---|
2247 |
|
---|
2248 | /* If scale multiplier was set: */
|
---|
2249 | if (dScaleMultiplier > 1.0)
|
---|
2250 | {
|
---|
2251 | /* Scale the pixmap up: */
|
---|
2252 | pixmap = pixmap.scaled(pixmap.width() * dScaleMultiplier, pixmap.height() * dScaleMultiplier,
|
---|
2253 | Qt::IgnoreAspectRatio, Qt::SmoothTransformation);
|
---|
2254 | uXHot *= dScaleMultiplier;
|
---|
2255 | uYHot *= dScaleMultiplier;
|
---|
2256 | }
|
---|
2257 |
|
---|
2258 | # ifdef VBOX_WS_WIN
|
---|
2259 | /* If device pixel ratio was set: */
|
---|
2260 | if (dDevicePixelRatio > 1.0)
|
---|
2261 | {
|
---|
2262 | /* Scale the pixmap down: */
|
---|
2263 | pixmap.setDevicePixelRatio(dDevicePixelRatio);
|
---|
2264 | uXHot /= dDevicePixelRatio;
|
---|
2265 | uYHot /= dDevicePixelRatio;
|
---|
2266 | }
|
---|
2267 | # endif
|
---|
2268 |
|
---|
2269 | #else
|
---|
2270 |
|
---|
2271 | Q_UNUSED(pixmap);
|
---|
2272 | Q_UNUSED(uXHot);
|
---|
2273 | Q_UNUSED(uYHot);
|
---|
2274 |
|
---|
2275 | #endif
|
---|
2276 | }
|
---|