VirtualBox

source: vbox/trunk/src/VBox/Frontends/VirtualBox/src/runtime/UISession.cpp@ 29455

Last change on this file since 29455 was 29455, checked in by vboxsync, 15 years ago

FE/Qt: New running VM core: UISession setPause() fix.

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 47.2 KB
Line 
1/* $Id: UISession.cpp 29455 2010-05-14 09:19:26Z vboxsync $ */
2/** @file
3 *
4 * VBox frontends: Qt GUI ("VirtualBox"):
5 * UISession stuff implementation
6 */
7
8/*
9 * Copyright (C) 2010 Oracle Corporation
10 *
11 * This file is part of VirtualBox Open Source Edition (OSE), as
12 * available from http://www.virtualbox.org. This file is free software;
13 * you can redistribute it and/or modify it under the terms of the GNU
14 * General Public License (GPL) as published by the Free Software
15 * Foundation, in version 2 as it comes in the "COPYING" file of the
16 * VirtualBox OSE distribution. VirtualBox OSE is distributed in the
17 * hope that it will be useful, but WITHOUT ANY WARRANTY of any kind.
18 */
19
20/* Global inclues */
21#include <QApplication>
22#include <QWidget>
23#include <QTimer>
24
25/* Local includes */
26#include "UISession.h"
27#include "UIMachine.h"
28#include "UIActionsPool.h"
29#include "UIMachineLogic.h"
30#include "UIMachineWindow.h"
31#include "UIMachineMenuBar.h"
32#include "VBoxProblemReporter.h"
33#include "UIFirstRunWzd.h"
34#ifdef VBOX_WITH_VIDEOHWACCEL
35# include "VBoxFBOverlay.h"
36# include "UIFrameBuffer.h"
37#endif
38
39#ifdef Q_WS_X11
40# include <QX11Info>
41# include <X11/Xlib.h>
42# include <X11/Xutil.h>
43# ifndef VBOX_WITHOUT_XCURSOR
44# include <X11/Xcursor/Xcursor.h>
45# endif
46#endif
47
48#if defined (Q_WS_MAC)
49# include "VBoxUtils.h"
50#endif
51
52/* Guest mouse pointer shape change event: */
53class UIMousePointerShapeChangeEvent : public QEvent
54{
55public:
56
57 UIMousePointerShapeChangeEvent(bool bIsVisible, bool bIsAlpha, uint uXHot, uint uYHot, uint uWidth, uint uHeight, const uchar *pShape)
58 : QEvent((QEvent::Type)UIConsoleEventType_MousePointerShapeChange)
59 , m_bIsVisible(bIsVisible), m_bIsAlpha(bIsAlpha), m_uXHot(uXHot), m_uYHot(uYHot), m_uWidth(uWidth), m_uHeight(uHeight), m_pData(0)
60 {
61 uint dataSize = ((((m_uWidth + 7) / 8 * m_uHeight) + 3) & ~3) + m_uWidth * 4 * m_uHeight;
62 if (pShape)
63 {
64 m_pData = new uchar[dataSize];
65 memcpy((void*)m_pData, (void*)pShape, dataSize);
66 }
67 }
68
69 virtual ~UIMousePointerShapeChangeEvent()
70 {
71 if (m_pData) delete[] m_pData;
72 }
73
74 bool isVisible() const { return m_bIsVisible; }
75 bool hasAlpha() const { return m_bIsAlpha; }
76 uint xHot() const { return m_uXHot; }
77 uint yHot() const { return m_uYHot; }
78 uint width() const { return m_uWidth; }
79 uint height() const { return m_uHeight; }
80 const uchar *shapeData() const { return m_pData; }
81
82private:
83
84 bool m_bIsVisible, m_bIsAlpha;
85 uint m_uXHot, m_uYHot, m_uWidth, m_uHeight;
86 const uchar *m_pData;
87};
88
89/* Guest mouse absolute positioning capability change event: */
90class UIMouseCapabilityChangeEvent : public QEvent
91{
92public:
93
94 UIMouseCapabilityChangeEvent(bool bSupportsAbsolute, bool bSupportsRelative, bool bNeedsHostCursor)
95 : QEvent((QEvent::Type)UIConsoleEventType_MouseCapabilityChange)
96 , m_bSupportsAbsolute(bSupportsAbsolute), m_bSupportsRelative(bSupportsRelative), m_bNeedsHostCursor(bNeedsHostCursor) {}
97
98 bool supportsAbsolute() const { return m_bSupportsAbsolute; }
99 bool supportsRelative() const { return m_bSupportsRelative; }
100 bool needsHostCursor() const { return m_bNeedsHostCursor; }
101
102private:
103
104 bool m_bSupportsAbsolute;
105 bool m_bSupportsRelative;
106 bool m_bNeedsHostCursor;
107};
108
109/* Keyboard LEDs change event: */
110class UIKeyboardLedsChangeEvent : public QEvent
111{
112public:
113
114 UIKeyboardLedsChangeEvent(bool bNumLock, bool bCapsLock, bool bScrollLock)
115 : QEvent((QEvent::Type)UIConsoleEventType_KeyboardLedsChange)
116 , m_bNumLock(bNumLock), m_bCapsLock(bCapsLock), m_bScrollLock(bScrollLock) {}
117
118 bool numLock() const { return m_bNumLock; }
119 bool capsLock() const { return m_bCapsLock; }
120 bool scrollLock() const { return m_bScrollLock; }
121
122private:
123
124 bool m_bNumLock;
125 bool m_bCapsLock;
126 bool m_bScrollLock;
127};
128
129/* Machine state change event: */
130class UIStateChangeEvent : public QEvent
131{
132public:
133
134 UIStateChangeEvent(KMachineState machineState)
135 : QEvent((QEvent::Type)UIConsoleEventType_StateChange)
136 , m_machineState(machineState) {}
137
138 KMachineState machineState() const { return m_machineState; }
139
140private:
141
142 KMachineState m_machineState;
143};
144
145/* Guest Additions state change event: */
146class UIAdditionsStateChangeEvent : public QEvent
147{
148public:
149
150 UIAdditionsStateChangeEvent()
151 : QEvent((QEvent::Type)UIConsoleEventType_AdditionsStateChange) {}
152};
153
154/* Network adapter change event: */
155class UINetworkAdapterChangeEvent : public QEvent
156{
157public:
158
159 UINetworkAdapterChangeEvent(const CNetworkAdapter &networkAdapter)
160 : QEvent((QEvent::Type)UIConsoleEventType_NetworkAdapterChange)
161 , m_networkAdapter(networkAdapter) {}
162
163 const CNetworkAdapter& networkAdapter() { return m_networkAdapter; }
164
165private:
166
167 const CNetworkAdapter m_networkAdapter;
168};
169
170/* Serial port change event: */
171/* Not used:
172class UISerialPortChangeEvent : public QEvent
173{
174public:
175
176 UISerialPortChangeEvent(const CSerialPort &serialPort)
177 : QEvent((QEvent::Type)UIConsoleEventType_SerialPortChange)
178 , m_serialPort(serialPort) {}
179
180 const CSerialPort& serialPort() { return m_serialPort; }
181
182private:
183
184 const CSerialPort m_serialPort;
185};
186*/
187
188/* Parallel port change event: */
189/* Not used:
190class UIParallelPortChangeEvent : public QEvent
191{
192public:
193
194 UIParallelPortChangeEvent(const CParallelPort &parallelPort)
195 : QEvent((QEvent::Type)UIConsoleEventType_ParallelPortChange)
196 , m_parallelPort(parallelPort) {}
197
198 const CParallelPort& parallelPort() { return m_parallelPort; }
199
200private:
201
202 const CParallelPort m_parallelPort;
203};
204*/
205
206/* Storage controller change event: */
207/* Not used:
208class UIStorageControllerChangeEvent : public QEvent
209{
210public:
211
212 UIStorageControllerChangeEvent()
213 : QEvent((QEvent::Type)UIConsoleEventType_StorageControllerChange) {}
214};
215*/
216
217/* Storage medium change event: */
218class UIMediumChangeEvent : public QEvent
219{
220public:
221
222 UIMediumChangeEvent(const CMediumAttachment &mediumAttachment)
223 : QEvent((QEvent::Type)UIConsoleEventType_MediumChange)
224 , m_mediumAttachment(mediumAttachment) {}
225 const CMediumAttachment& mediumAttachment() { return m_mediumAttachment; }
226
227private:
228
229 const CMediumAttachment m_mediumAttachment;
230};
231
232/* CPU change event: */
233/* Not used:
234class UICPUChangeEvent : public QEvent
235{
236public:
237
238 UICPUChangeEvent(ulong uCPU, bool bRemove)
239 : QEvent((QEvent::Type)UIConsoleEventType_CPUChange)
240 , m_uCPU(uCPU), m_bRemove(bRemove) {}
241
242 ulong cpu() const { return m_uCPU; }
243 bool remove() const { return m_bRemove; }
244
245private:
246
247 ulong m_uCPU;
248 bool m_bRemove;
249};
250*/
251
252/* VRDP server change event: */
253/* Not used:
254class UIVRDPServerChangeEvent : public QEvent
255{
256public:
257
258 UIVRDPServerChangeEvent()
259 : QEvent((QEvent::Type)UIConsoleEventType_VRDPServerChange) {}
260};
261*/
262
263/* Remote display info change event: */
264/* Not used:
265class UIRemoteDisplayInfoChangeEvent : public QEvent
266{
267public:
268
269 UIRemoteDisplayInfoChangeEvent()
270 : QEvent((QEvent::Type)UIConsoleEventType_RemoteDisplayInfoChange) {}
271};
272*/
273
274/* USB controller change event: */
275class UIUSBControllerChangeEvent : public QEvent
276{
277public:
278
279 UIUSBControllerChangeEvent()
280 : QEvent((QEvent::Type)UIConsoleEventType_USBControllerChange) {}
281};
282
283/* USB device state change event: */
284class UIUSBDeviceUIStateChangeEvent : public QEvent
285{
286public:
287
288 UIUSBDeviceUIStateChangeEvent(const CUSBDevice &device, bool bAttached, const CVirtualBoxErrorInfo &error)
289 : QEvent((QEvent::Type)UIConsoleEventType_USBDeviceStateChange)
290 , m_device(device), m_bAttached(bAttached), m_error(error) {}
291
292 const CUSBDevice& device() const { return m_device; }
293 bool attached() const { return m_bAttached; }
294 const CVirtualBoxErrorInfo& error() const { return m_error; }
295
296private:
297
298 const CUSBDevice m_device;
299 bool m_bAttached;
300 const CVirtualBoxErrorInfo m_error;
301};
302
303/* Shared folder change event: */
304class UISharedFolderChangeEvent : public QEvent
305{
306public:
307
308 UISharedFolderChangeEvent()
309 : QEvent((QEvent::Type)UIConsoleEventType_SharedFolderChange) {}
310};
311
312/* VM Runtime error event: */
313class UIRuntimeErrorEvent : public QEvent
314{
315public:
316
317 UIRuntimeErrorEvent(bool bFatal, const QString &strErrorID, const QString &strMessage)
318 : QEvent((QEvent::Type)UIConsoleEventType_RuntimeError)
319 , m_bFatal(bFatal), m_strErrorID(strErrorID), m_strMessage(strMessage) {}
320
321 bool fatal() const { return m_bFatal; }
322 QString errorID() const { return m_strErrorID; }
323 QString message() const { return m_strMessage; }
324
325private:
326
327 bool m_bFatal;
328 QString m_strErrorID;
329 QString m_strMessage;
330};
331
332/* Can show window event: */
333/* Not used:
334class UICanUIShowWindowEvent : public QEvent
335{
336public:
337
338 UICanUIShowWindowEvent()
339 : QEvent((QEvent::Type)UIConsoleEventType_CanShowWindow) {}
340};
341*/
342
343/* Show window event: */
344#ifdef Q_WS_MAC
345class UIShowWindowEvent : public QEvent
346{
347public:
348
349 UIShowWindowEvent()
350 : QEvent((QEvent::Type)UIConsoleEventType_ShowWindow) {}
351};
352#endif /* Q_WS_MAC */
353
354class UIConsoleCallback : VBOX_SCRIPTABLE_IMPL(IConsoleCallback)
355{
356public:
357
358 UIConsoleCallback(UISession *pEventHandler)
359 : m_pEventHandler(pEventHandler)
360#if defined (Q_WS_WIN)
361 , m_iRefCount(0)
362#endif
363 {
364 }
365
366 virtual ~UIConsoleCallback()
367 {
368 }
369
370 NS_DECL_ISUPPORTS
371
372#if defined (Q_WS_WIN)
373 STDMETHOD_(ULONG, AddRef)()
374 {
375 return ::InterlockedIncrement(&m_iRefCount);
376 }
377 STDMETHOD_(ULONG, Release)()
378 {
379 long iCount = ::InterlockedDecrement(&m_iRefCount);
380 if (iCount == 0)
381 delete this;
382 return iCount;
383 }
384#endif
385
386 VBOX_SCRIPTABLE_DISPATCH_IMPL(IConsoleCallback)
387
388 STDMETHOD(OnMousePointerShapeChange)(BOOL bIsVisible, BOOL bAlpha, ULONG uXHot, ULONG uYHot, ULONG uWidth, ULONG uHeight, BYTE *pShape)
389 {
390 QApplication::postEvent(m_pEventHandler, new UIMousePointerShapeChangeEvent(bIsVisible, bAlpha, uXHot, uYHot, uWidth, uHeight, pShape));
391 return S_OK;
392 }
393
394 STDMETHOD(OnMouseCapabilityChange)(BOOL bSupportsAbsolute, BOOL bSupportsRelative, BOOL bNeedHostCursor)
395 {
396 QApplication::postEvent(m_pEventHandler, new UIMouseCapabilityChangeEvent(bSupportsAbsolute, bSupportsRelative, bNeedHostCursor));
397 return S_OK;
398 }
399
400 STDMETHOD(OnKeyboardLedsChange)(BOOL bNumLock, BOOL bCapsLock, BOOL bScrollLock)
401 {
402 QApplication::postEvent(m_pEventHandler, new UIKeyboardLedsChangeEvent(bNumLock, bCapsLock, bScrollLock));
403 return S_OK;
404 }
405
406 STDMETHOD(OnStateChange)(MachineState_T machineState)
407 {
408 QApplication::postEvent(m_pEventHandler, new UIStateChangeEvent((KMachineState)machineState));
409 return S_OK;
410 }
411
412 STDMETHOD(OnAdditionsStateChange)()
413 {
414 QApplication::postEvent(m_pEventHandler, new UIAdditionsStateChangeEvent);
415 return S_OK;
416 }
417
418 STDMETHOD(OnNetworkAdapterChange)(INetworkAdapter *pNetworkAdapter)
419 {
420 QApplication::postEvent(m_pEventHandler, new UINetworkAdapterChangeEvent(CNetworkAdapter(pNetworkAdapter)));
421 return S_OK;
422 }
423
424 STDMETHOD(OnSerialPortChange)(ISerialPort * /* pSerialPort */)
425 {
426 /* Not used: QApplication::postEvent(m_pEventHandler, new UISerialPortChangeEvent(CSerialPort(pSerialPort))); */
427 return VBOX_E_DONT_CALL_AGAIN;
428 }
429
430 STDMETHOD(OnParallelPortChange)(IParallelPort * /* pParallelPort */)
431 {
432 /* Not used: QApplication::postEvent(m_pEventHandler, new UIParallelPortChangeEvent(CParallelPort(pParallelPort))); */
433 return VBOX_E_DONT_CALL_AGAIN;
434 }
435
436 STDMETHOD(OnStorageControllerChange)()
437 {
438 /* Not used: QApplication::postEvent(m_pEventHandler, new UIStorageControllerChangeEvent); */
439 return VBOX_E_DONT_CALL_AGAIN;
440 }
441
442 STDMETHOD(OnMediumChange)(IMediumAttachment *pMediumAttachment)
443 {
444 QApplication::postEvent(m_pEventHandler, new UIMediumChangeEvent(CMediumAttachment(pMediumAttachment)));
445 return S_OK;
446 }
447
448 STDMETHOD(OnCPUChange)(ULONG /* uCPU */, BOOL /* bRemove */)
449 {
450 /* Not used: QApplication::postEvent(m_pEventHandler, new UICPUChangeEvent(uCPU, bRemove)); */
451 return VBOX_E_DONT_CALL_AGAIN;
452 }
453
454 STDMETHOD(OnVRDPServerChange)()
455 {
456 /* Not used: QApplication::postEvent(m_pEventHandler, new UIVRDPServerChangeEvent); */
457 return VBOX_E_DONT_CALL_AGAIN;
458 }
459
460 STDMETHOD(OnRemoteDisplayInfoChange)()
461 {
462 /* Not used: QApplication::postEvent(m_pEventHandler, new UIRemoteDisplayInfoChangeEvent); */
463 return VBOX_E_DONT_CALL_AGAIN;
464 }
465
466 STDMETHOD(OnUSBControllerChange)()
467 {
468 QApplication::postEvent(m_pEventHandler, new UIUSBControllerChangeEvent);
469 return S_OK;
470 }
471
472 STDMETHOD(OnUSBDeviceStateChange)(IUSBDevice *pDevice, BOOL bAttached, IVirtualBoxErrorInfo *pError)
473 {
474 QApplication::postEvent(m_pEventHandler, new UIUSBDeviceUIStateChangeEvent(CUSBDevice(pDevice), bAttached, CVirtualBoxErrorInfo(pError)));
475 return S_OK;
476 }
477
478 STDMETHOD(OnSharedFolderChange)(Scope_T scope)
479 {
480 NOREF(scope);
481 QApplication::postEvent(m_pEventHandler, new UISharedFolderChangeEvent);
482 return S_OK;
483 }
484
485 STDMETHOD(OnRuntimeError)(BOOL bFatal, IN_BSTR strId, IN_BSTR strMessage)
486 {
487 QApplication::postEvent(m_pEventHandler, new UIRuntimeErrorEvent(bFatal, QString::fromUtf16(strId), QString::fromUtf16(strMessage)));
488 return S_OK;
489 }
490
491 STDMETHOD(OnCanShowWindow)(BOOL *pbCanShow)
492 {
493 if (!pbCanShow)
494 return E_POINTER;
495
496 *pbCanShow = TRUE;
497 return S_OK;
498 }
499
500 STDMETHOD(OnShowWindow)(ULONG64 *puWinId)
501 {
502 if (!puWinId)
503 return E_POINTER;
504
505#ifdef Q_WS_MAC
506 /* Let's try the simple approach first - grab the focus.
507 * Getting a window out of the dock (minimized or whatever it's called)
508 * needs to be done on the GUI thread, so post it a note: */
509 *puWinId = 0;
510 if (!m_pEventHandler)
511 return S_OK;
512
513 if (::darwinSetFrontMostProcess())
514 QApplication::postEvent(m_pEventHandler, new UIShowWindowEvent);
515 else
516 {
517 /* It failed for some reason, send the other process our PSN so it can try.
518 * (This is just a precaution should Mac OS X start imposing the same sensible
519 * focus stealing restrictions that other window managers implement). */
520 *puWinId = ::darwinGetCurrentProcessId();
521 }
522#else /* Q_WS_MAC */
523 /* Return the ID of the top-level console window. */
524 *puWinId = (ULONG64)m_pEventHandler->winId();
525#endif /* !Q_WS_MAC */
526
527 return S_OK;
528 }
529
530private:
531
532 UISession *m_pEventHandler;
533
534#if defined (Q_WS_WIN)
535 long m_iRefCount;
536#endif
537};
538
539#if !defined (Q_WS_WIN)
540NS_DECL_CLASSINFO(UIConsoleCallback)
541NS_IMPL_THREADSAFE_ISUPPORTS1_CI(UIConsoleCallback, IConsoleCallback)
542#endif
543
544UISession::UISession(UIMachine *pMachine, CSession &sessionReference)
545 : QObject(pMachine)
546 /* Base variables: */
547 , m_pMachine(pMachine)
548 , m_session(sessionReference)
549 , m_callback(CConsoleCallback(new UIConsoleCallback(this)))
550 /* Common variables: */
551 , m_pMenuPool(0)
552#ifdef VBOX_WITH_VIDEOHWACCEL
553 , m_FrameBufferVector(sessionReference.GetMachine().GetMonitorCount())
554#endif
555 , m_machineState(KMachineState_Null)
556#if defined(Q_WS_WIN)
557 , m_alphaCursor(0)
558#endif
559 /* Common flags: */
560 , m_fIsFirstTimeStarted(false)
561 , m_fIsIgnoreRutimeMediumsChanging(false)
562 , m_fIsGuestResizeIgnored(false)
563 , m_fIsSeamlessModeRequested(false)
564 /* Guest additions flags: */
565 , m_fIsGuestAdditionsActive(false)
566 , m_fIsGuestSupportsGraphics(false)
567 , m_fIsGuestSupportsSeamless(false)
568 /* Mouse flags: */
569 , m_fNumLock(false)
570 , m_fCapsLock(false)
571 , m_fScrollLock(false)
572 , m_uNumLockAdaptionCnt(2)
573 , m_uCapsLockAdaptionCnt(2)
574 /* Mouse flags: */
575 , m_fIsMouseSupportsAbsolute(false)
576 , m_fIsMouseSupportsRelative(false)
577 , m_fIsMouseHostCursorNeeded(false)
578 , m_fIsMouseCaptured(false)
579 , m_fIsMouseIntegrated(true)
580 , m_fIsValidPointerShapePresent(false)
581 , m_fIsHidingHostPointer(true)
582{
583 /* Register console callback: */
584 session().GetConsole().RegisterCallback(m_callback);
585
586 /* Prepare main menu: */
587 prepareMenuPool();
588
589 /* Load uisession settings: */
590 loadSessionSettings();
591}
592
593UISession::~UISession()
594{
595 /* Save uisession settings: */
596 saveSessionSettings();
597
598 /* Cleanup main menu: */
599 cleanupMenuPool();
600
601 /* Unregister console callback: */
602 session().GetConsole().UnregisterCallback(m_callback);
603
604#if defined(Q_WS_WIN)
605 /* Destroy alpha cursor: */
606 if (m_alphaCursor)
607 DestroyIcon(m_alphaCursor);
608#endif
609
610#ifdef VBOX_WITH_VIDEOHWACCEL
611 for (int i = m_FrameBufferVector.size() - 1; i >= 0; --i)
612 {
613 UIFrameBuffer *pFb = m_FrameBufferVector[i];
614 if (pFb)
615 {
616 /* Warn framebuffer about its no more necessary: */
617 pFb->setDeleted(true);
618 /* Detach framebuffer from Display: */
619 CDisplay display = session().GetConsole().GetDisplay();
620 display.SetFramebuffer(i, CFramebuffer(NULL));
621 /* Release the reference: */
622 pFb->Release();
623 }
624 }
625#endif
626}
627
628void UISession::powerUp()
629{
630 /* Do nothing if we had started already: */
631 if (isRunning() || isPaused())
632 return;
633
634 /* Prepare powerup: */
635 preparePowerUp();
636
637 /* Get current machine/console: */
638 CMachine machine = session().GetMachine();
639 CConsole console = session().GetConsole();
640
641 /* Power UP machine: */
642 CProgress progress = vboxGlobal().isStartPausedEnabled() || vboxGlobal().isDebuggerAutoShowEnabled() ?
643 console.PowerUpPaused() : console.PowerUp();
644
645 /* Check for immediate failure: */
646 if (!console.isOk())
647 {
648 if (vboxGlobal().showStartVMErrors())
649 vboxProblem().cannotStartMachine(console);
650 QTimer::singleShot(0, this, SLOT(sltCloseVirtualSession()));
651 return;
652 }
653
654 /* Guard progressbar warnings from auto-closing: */
655 if (uimachine()->machineLogic())
656 uimachine()->machineLogic()->setPreventAutoClose(true);
657
658 /* Show "Starting/Restoring" progress dialog: */
659 if (isSaved())
660 vboxProblem().showModalProgressDialog(progress, machine.GetName(), mainMachineWindow(), 0);
661 else
662 vboxProblem().showModalProgressDialog(progress, machine.GetName(), mainMachineWindow());
663
664 /* Allow further auto-closing: */
665 if (uimachine()->machineLogic())
666 uimachine()->machineLogic()->setPreventAutoClose(false);
667
668 /* Check for a progress failure: */
669 if (progress.GetResultCode() != 0)
670 {
671 if (vboxGlobal().showStartVMErrors())
672 vboxProblem().cannotStartMachine(progress);
673 QTimer::singleShot(0, this, SLOT(sltCloseVirtualSession()));
674 return;
675 }
676
677 /* Check if we missed a really quick termination after successful startup, and process it if we did: */
678 if (isTurnedOff())
679 {
680 QTimer::singleShot(0, this, SLOT(sltCloseVirtualSession()));
681 return;
682 }
683
684 /* Check if the required virtualization features are active. We get this
685 * info only when the session is active. */
686 bool fIs64BitsGuest = vboxGlobal().virtualBox().GetGuestOSType(console.GetGuest().GetOSTypeId()).GetIs64Bit();
687 bool fRecommendVirtEx = vboxGlobal().virtualBox().GetGuestOSType(console.GetGuest().GetOSTypeId()).GetRecommendedVirtEx();
688 AssertMsg(!fIs64BitsGuest || fRecommendVirtEx, ("Virtualization support missed for 64bit guest!\n"));
689 bool fIsVirtEnabled = console.GetDebugger().GetHWVirtExEnabled();
690 if (fRecommendVirtEx && !fIsVirtEnabled)
691 {
692 bool fShouldWeClose;
693
694 bool fVTxAMDVSupported = vboxGlobal().virtualBox().GetHost().GetProcessorFeature(KProcessorFeature_HWVirtEx);
695
696 QApplication::processEvents();
697 setPause(true);
698
699 if (fIs64BitsGuest)
700 fShouldWeClose = vboxProblem().warnAboutVirtNotEnabled64BitsGuest(fVTxAMDVSupported);
701 else
702 fShouldWeClose = vboxProblem().warnAboutVirtNotEnabledGuestRequired(fVTxAMDVSupported);
703
704 if (fShouldWeClose)
705 {
706 /* At this point the console is powered up. So we have to close
707 * this session again. */
708 CProgress progress = console.PowerDown();
709 if (console.isOk())
710 {
711 /* Guard progressbar warnings from auto-closing: */
712 if (uimachine()->machineLogic())
713 uimachine()->machineLogic()->setPreventAutoClose(true);
714 /* Show the power down progress dialog */
715 vboxProblem().showModalProgressDialog(progress, machine.GetName(), mainMachineWindow());
716 if (progress.GetResultCode() != 0)
717 vboxProblem().cannotStopMachine(progress);
718 /* Allow further auto-closing: */
719 if (uimachine()->machineLogic())
720 uimachine()->machineLogic()->setPreventAutoClose(false);
721 }
722 else
723 vboxProblem().cannotStopMachine(console);
724 /* Now signal the destruction of the rest. */
725 QTimer::singleShot(0, this, SLOT(sltCloseVirtualSession()));
726 return;
727 }
728 else
729 setPause(false);
730 }
731
732#if 0 // TODO: Rework debugger logic!
733# ifdef VBOX_WITH_DEBUGGER_GUI
734 /* Open the debugger in "full screen" mode requested by the user. */
735 else if (vboxGlobal().isDebuggerAutoShowEnabled())
736 {
737 /* console in upper left corner of the desktop. */
738 QRect rct (0, 0, 0, 0);
739 QDesktopWidget *desktop = QApplication::desktop();
740 if (desktop)
741 rct = desktop->availableGeometry(pos());
742 move (QPoint (rct.x(), rct.y()));
743
744 if (vboxGlobal().isDebuggerAutoShowStatisticsEnabled())
745 sltShowDebugStatistics();
746 if (vboxGlobal().isDebuggerAutoShowCommandLineEnabled())
747 sltShowDebugCommandLine();
748
749 if (!vboxGlobal().isStartPausedEnabled())
750 machineWindowWrapper()->machineView()->pause (false);
751 }
752# endif
753#endif
754
755 /* Warn listeners about machine was started: */
756 emit sigMachineStarted();
757}
758
759UIActionsPool* UISession::actionsPool() const
760{
761 return m_pMachine->actionsPool();
762}
763
764QWidget* UISession::mainMachineWindow() const
765{
766 return uimachine()->machineLogic()->mainMachineWindow()->machineWindow();
767}
768
769UIMachineLogic* UISession::machineLogic() const
770{
771 return uimachine()->machineLogic();
772}
773
774QMenu* UISession::newMenu(UIMainMenuType fOptions /* = UIMainMenuType_ALL */)
775{
776 /* Create new menu: */
777 QMenu *pMenu = m_pMenuPool->createMenu(actionsPool(), fOptions);
778
779 /* Re-init menu pool for the case menu were recreated: */
780 reinitMenuPool();
781
782 /* Return newly created menu: */
783 return pMenu;
784}
785
786QMenuBar* UISession::newMenuBar(UIMainMenuType fOptions /* = UIMainMenuType_ALL */)
787{
788 /* Create new menubar: */
789 QMenuBar *pMenuBar = m_pMenuPool->createMenuBar(actionsPool(), fOptions);
790
791 /* Re-init menu pool for the case menu were recreated: */
792 reinitMenuPool();
793
794 /* Return newly created menubar: */
795 return pMenuBar;
796}
797
798bool UISession::setPause(bool fOn)
799{
800 /* Commenting it out as isPaused() could reflect
801 * quite obsolete state due to synchronization: */
802 //if (isPaused() == fOn)
803 // return true;
804
805 CConsole console = session().GetConsole();
806
807 if (fOn)
808 console.Pause();
809 else
810 console.Resume();
811
812 bool ok = console.isOk();
813 if (!ok)
814 {
815 if (fOn)
816 vboxProblem().cannotPauseMachine(console);
817 else
818 vboxProblem().cannotResumeMachine(console);
819 }
820
821 return ok;
822}
823
824void UISession::sltInstallGuestAdditionsFrom(const QString &strSource)
825{
826 CMachine machine = session().GetMachine();
827 CVirtualBox vbox = vboxGlobal().virtualBox();
828 QString strUuid;
829
830 CMedium image = vbox.FindDVDImage(strSource);
831 if (image.isNull())
832 {
833 image = vbox.OpenDVDImage(strSource, strUuid);
834 if (vbox.isOk())
835 strUuid = image.GetId();
836 }
837 else
838 strUuid = image.GetId();
839
840 if (!vbox.isOk())
841 {
842 vboxProblem().cannotOpenMedium(0, vbox, VBoxDefs::MediumType_DVD, strSource);
843 return;
844 }
845
846 AssertMsg(!strUuid.isNull(), ("Guest Additions image UUID should be valid!\n"));
847
848 QString strCntName;
849 LONG iCntPort = -1, iCntDevice = -1;
850 /* Searching for the first suitable slot */
851 {
852 CStorageControllerVector controllers = machine.GetStorageControllers();
853 int i = 0;
854 while (i < controllers.size() && strCntName.isNull())
855 {
856 CStorageController controller = controllers[i];
857 CMediumAttachmentVector attachments = machine.GetMediumAttachmentsOfController(controller.GetName());
858 int j = 0;
859 while (j < attachments.size() && strCntName.isNull())
860 {
861 CMediumAttachment attachment = attachments[j];
862 if (attachment.GetType() == KDeviceType_DVD)
863 {
864 strCntName = controller.GetName();
865 iCntPort = attachment.GetPort();
866 iCntDevice = attachment.GetDevice();
867 }
868 ++ j;
869 }
870 ++ i;
871 }
872 }
873
874 if (!strCntName.isNull())
875 {
876 bool fIsMounted = false;
877
878 /* Mount medium to the predefined port/device */
879 machine.MountMedium(strCntName, iCntPort, iCntDevice, strUuid, false /* force */);
880 if (machine.isOk())
881 fIsMounted = true;
882 else
883 {
884 /* Ask for force mounting */
885 if (vboxProblem().cannotRemountMedium(0, machine, VBoxMedium(image, VBoxDefs::MediumType_DVD),
886 true /* mount? */, true /* retry? */) == QIMessageBox::Ok)
887 {
888 /* Force mount medium to the predefined port/device */
889 machine.MountMedium(strCntName, iCntPort, iCntDevice, strUuid, true /* force */);
890 if (machine.isOk())
891 fIsMounted = true;
892 else
893 vboxProblem().cannotRemountMedium(0, machine, VBoxMedium(image, VBoxDefs::MediumType_DVD),
894 true /* mount? */, false /* retry? */);
895 }
896 }
897 }
898 else
899 vboxProblem().cannotMountGuestAdditions(machine.GetName());
900}
901
902void UISession::sltCloseVirtualSession()
903{
904 m_pMachine->closeVirtualMachine();
905}
906
907bool UISession::event(QEvent *pEvent)
908{
909 switch (pEvent->type())
910 {
911 case UIConsoleEventType_MousePointerShapeChange:
912 {
913 /* Convert to mouse shape change event: */
914 UIMousePointerShapeChangeEvent *pConsoleEvent = static_cast<UIMousePointerShapeChangeEvent*>(pEvent);
915
916 /* In case of shape data is present: */
917 if (pConsoleEvent->shapeData())
918 {
919 /* We are ignoring visibility flag: */
920 m_fIsHidingHostPointer = false;
921
922 /* And updating current cursor shape: */
923 setPointerShape(pConsoleEvent->shapeData(), pConsoleEvent->hasAlpha(),
924 pConsoleEvent->xHot(), pConsoleEvent->yHot(),
925 pConsoleEvent->width(), pConsoleEvent->height());
926 }
927 /* In case of shape data is NOT present: */
928 else
929 {
930 /* Remember if we should hide the cursor: */
931 m_fIsHidingHostPointer = !pConsoleEvent->isVisible();
932 }
933
934 /* Notify listeners about mouse capability changed: */
935 emit sigMousePointerShapeChange();
936
937 /* Accept event: */
938 pEvent->accept();
939 return true;
940 }
941
942 case UIConsoleEventType_MouseCapabilityChange:
943 {
944 /* Convert to mouse capability event: */
945 UIMouseCapabilityChangeEvent *pConsoleEvent = static_cast<UIMouseCapabilityChangeEvent*>(pEvent);
946
947 /* Check if something had changed: */
948 if (m_fIsMouseSupportsAbsolute != pConsoleEvent->supportsAbsolute() ||
949 m_fIsMouseSupportsRelative != pConsoleEvent->supportsRelative() ||
950 m_fIsMouseHostCursorNeeded != pConsoleEvent->needsHostCursor())
951 {
952 /* Store new data: */
953 m_fIsMouseSupportsAbsolute = pConsoleEvent->supportsAbsolute();
954 m_fIsMouseSupportsRelative = pConsoleEvent->supportsRelative();
955 m_fIsMouseHostCursorNeeded = pConsoleEvent->needsHostCursor();
956
957 /* Notify listeners about mouse capability changed: */
958 emit sigMouseCapabilityChange();
959 }
960
961 /* Accept event: */
962 pEvent->accept();
963 return true;
964 }
965
966 case UIConsoleEventType_KeyboardLedsChange:
967 {
968 /* Convert to keyboard LEDs change event: */
969 UIKeyboardLedsChangeEvent *pConsoleEvent = static_cast<UIKeyboardLedsChangeEvent*>(pEvent);
970
971 /* Check if something had changed: */
972 if (m_fNumLock != pConsoleEvent->numLock() ||
973 m_fCapsLock != pConsoleEvent->capsLock() ||
974 m_fScrollLock != pConsoleEvent->scrollLock())
975 {
976 /* Store new num lock data: */
977 if (m_fNumLock != pConsoleEvent->numLock())
978 {
979 m_fNumLock = pConsoleEvent->numLock();
980 m_uNumLockAdaptionCnt = 2;
981 }
982
983 /* Store new caps lock data: */
984 if (m_fCapsLock != pConsoleEvent->capsLock())
985 {
986 m_fCapsLock = pConsoleEvent->capsLock();
987 m_uCapsLockAdaptionCnt = 2;
988 }
989
990 /* Store new scroll lock data: */
991 if (m_fScrollLock != pConsoleEvent->scrollLock())
992 {
993 m_fScrollLock = pConsoleEvent->scrollLock();
994 }
995
996 /* Notify listeners about mouse capability changed: */
997 emit sigKeyboardLedsChange();
998 }
999
1000 /* Accept event: */
1001 pEvent->accept();
1002 return true;
1003 }
1004
1005 case UIConsoleEventType_StateChange:
1006 {
1007 /* Convert to machine state event: */
1008 UIStateChangeEvent *pConsoleEvent = static_cast<UIStateChangeEvent*>(pEvent);
1009
1010 /* Check if something had changed: */
1011 if (m_machineState != pConsoleEvent->machineState())
1012 {
1013 /* Store new data: */
1014 m_machineState = pConsoleEvent->machineState();
1015
1016 /* Notify listeners about machine state changed: */
1017 emit sigMachineStateChange();
1018 }
1019
1020 /* Accept event: */
1021 pEvent->accept();
1022 return true;
1023 }
1024
1025 case UIConsoleEventType_AdditionsStateChange:
1026 {
1027 /* Get our guest: */
1028 CGuest guest = session().GetConsole().GetGuest();
1029
1030 /* Variable flags: */
1031 bool fIsGuestAdditionsActive = guest.GetAdditionsActive();
1032 bool fIsGuestSupportsGraphics = guest.GetSupportsGraphics();
1033 bool fIsGuestSupportsSeamless = guest.GetSupportsSeamless();
1034
1035 /* Check if something had changed: */
1036 if (m_fIsGuestAdditionsActive != fIsGuestAdditionsActive ||
1037 m_fIsGuestSupportsGraphics != fIsGuestSupportsGraphics ||
1038 m_fIsGuestSupportsSeamless != fIsGuestSupportsSeamless)
1039 {
1040 /* Store new data: */
1041 m_fIsGuestAdditionsActive = fIsGuestAdditionsActive;
1042 m_fIsGuestSupportsGraphics = fIsGuestSupportsGraphics;
1043 m_fIsGuestSupportsSeamless = fIsGuestSupportsSeamless;
1044
1045 /* Notify listeners about guest additions state changed: */
1046 emit sigAdditionsStateChange();
1047 }
1048
1049 /* Accept event: */
1050 pEvent->accept();
1051 return true;
1052 }
1053
1054 case UIConsoleEventType_NetworkAdapterChange:
1055 {
1056 UINetworkAdapterChangeEvent *pConsoleEvent = static_cast<UINetworkAdapterChangeEvent*>(pEvent);
1057 emit sigNetworkAdapterChange(pConsoleEvent->networkAdapter());
1058 return true;
1059 }
1060
1061 /* Not used:
1062 case UIConsoleEventType_SerialPortChange:
1063 {
1064 UISerialPortChangeEvent *pConsoleEvent = static_cast<UISerialPortChangeEvent*>(pEvent);
1065 emit sigSerialPortChange(pConsoleEvent->serialPort());
1066 return true;
1067 }
1068 */
1069
1070 /* Not used:
1071 case UIConsoleEventType_ParallelPortChange:
1072 {
1073 UIParallelPortChangeEvent *pConsoleEvent = static_cast<UIParallelPortChangeEvent*>(pEvent);
1074 emit sigParallelPortChange(pConsoleEvent->parallelPort());
1075 return true;
1076 }
1077 */
1078
1079 /* Not used:
1080 case UIConsoleEventType_StorageControllerChange:
1081 {
1082 emit sigStorageControllerChange();
1083 return true;
1084 }
1085 */
1086
1087 case UIConsoleEventType_MediumChange:
1088 {
1089 UIMediumChangeEvent *pConsoleEvent = static_cast<UIMediumChangeEvent*>(pEvent);
1090 emit sigMediumChange(pConsoleEvent->mediumAttachment());
1091 return true;
1092 }
1093
1094 /* Not used:
1095 case UIConsoleEventType_CPUChange:
1096 {
1097 UICPUChangeEvent *pConsoleEvent = static_cast<UICPUChangeEvent*>(pEvent);
1098 emit sigCPUChange(pConsoleEvent->cpu(), pConsoleEvent->remove());
1099 return true;
1100 }
1101 */
1102
1103 /* Not used:
1104 case UIConsoleEventType_VRDPServerChange:
1105 {
1106 emit sigVRDPServerChange();
1107 return true;
1108 }
1109 */
1110
1111 /* Not used:
1112 case UIConsoleEventType_RemoteDisplayInfoChange:
1113 {
1114 emit sigRemoteDisplayInfoChange();
1115 return true;
1116 }
1117 */
1118
1119 case UIConsoleEventType_USBControllerChange:
1120 {
1121 emit sigUSBControllerChange();
1122 return true;
1123 }
1124
1125 case UIConsoleEventType_USBDeviceStateChange:
1126 {
1127 UIUSBDeviceUIStateChangeEvent *pConsoleEvent = static_cast<UIUSBDeviceUIStateChangeEvent*>(pEvent);
1128 emit sigUSBDeviceStateChange(pConsoleEvent->device(), pConsoleEvent->attached(), pConsoleEvent->error());
1129 return true;
1130 }
1131
1132 case UIConsoleEventType_SharedFolderChange:
1133 {
1134 emit sigSharedFolderChange();
1135 return true;
1136 }
1137
1138 case UIConsoleEventType_RuntimeError:
1139 {
1140 UIRuntimeErrorEvent *pConsoleEvent = static_cast<UIRuntimeErrorEvent*>(pEvent);
1141 emit sigRuntimeError(pConsoleEvent->fatal(), pConsoleEvent->errorID(), pConsoleEvent->message());
1142 return true;
1143 }
1144
1145#ifdef Q_WS_MAC
1146 case UIConsoleEventType_ShowWindow:
1147 {
1148 emit sigShowWindows();
1149 /* Accept event: */
1150 pEvent->accept();
1151 return true;
1152 }
1153#endif
1154
1155 default:
1156 break;
1157 }
1158 return QObject::event(pEvent);
1159}
1160
1161void UISession::prepareMenuPool()
1162{
1163 m_pMenuPool = new UIMachineMenuBar;
1164}
1165
1166void UISession::loadSessionSettings()
1167{
1168 /* Get uisession machine: */
1169 CMachine machine = session().GetConsole().GetMachine();
1170
1171 /* Availability settings: */
1172 {
1173 /* VRDP Stuff: */
1174 CVRDPServer vrdpServer = machine.GetVRDPServer();
1175 if (vrdpServer.isNull())
1176 {
1177 /* Hide VRDP Action: */
1178 uimachine()->actionsPool()->action(UIActionIndex_Toggle_VRDP)->setVisible(false);
1179 }
1180 }
1181
1182 /* Load extra-data settings: */
1183 {
1184 /* Temporary: */
1185 QString strSettings;
1186
1187 /* Is there shoul be First RUN Wizard? */
1188 strSettings = machine.GetExtraData(VBoxDefs::GUI_FirstRun);
1189 if (strSettings == "yes")
1190 m_fIsFirstTimeStarted = true;
1191
1192 /* Ignore mediums mounted at runtime? */
1193 strSettings = machine.GetExtraData(VBoxDefs::GUI_SaveMountedAtRuntime);
1194 if (strSettings == "no")
1195 m_fIsIgnoreRutimeMediumsChanging = true;
1196
1197 /* Should guest autoresize? */
1198 strSettings = machine.GetExtraData(VBoxDefs::GUI_AutoresizeGuest);
1199 QAction *pGuestAutoresizeSwitch = uimachine()->actionsPool()->action(UIActionIndex_Toggle_GuestAutoresize);
1200 pGuestAutoresizeSwitch->blockSignals(true);
1201 pGuestAutoresizeSwitch->setChecked(strSettings != "off");
1202 pGuestAutoresizeSwitch->blockSignals(false);
1203 }
1204}
1205
1206void UISession::saveSessionSettings()
1207{
1208 /* Get uisession machine: */
1209 CMachine machine = session().GetConsole().GetMachine();
1210
1211 /* Save extra-data settings: */
1212 {
1213 /* Disable First RUN Wizard for the since now: */
1214 machine.SetExtraData(VBoxDefs::GUI_FirstRun, QString());
1215
1216 /* Remember if guest should autoresize: */
1217 machine.SetExtraData(VBoxDefs::GUI_AutoresizeGuest,
1218 uimachine()->actionsPool()->action(UIActionIndex_Toggle_GuestAutoresize)->isChecked() ?
1219 QString() : "off");
1220
1221 // TODO: Move to fullscreen/seamless logic:
1222 //machine.SetExtraData(VBoxDefs::GUI_MiniToolBarAutoHide, mMiniToolBar->isAutoHide() ? "on" : "off");
1223 }
1224}
1225
1226void UISession::cleanupMenuPool()
1227{
1228 delete m_pMenuPool;
1229 m_pMenuPool = 0;
1230}
1231
1232WId UISession::winId() const
1233{
1234 return mainMachineWindow()->winId();
1235}
1236
1237void UISession::setPointerShape(const uchar *pShapeData, bool fHasAlpha,
1238 uint uXHot, uint uYHot, uint uWidth, uint uHeight)
1239{
1240 AssertMsg(pShapeData, ("Shape data must not be NULL!\n"));
1241
1242 m_fIsValidPointerShapePresent = false;
1243 const uchar *srcAndMaskPtr = pShapeData;
1244 uint andMaskSize = (uWidth + 7) / 8 * uHeight;
1245 const uchar *srcShapePtr = pShapeData + ((andMaskSize + 3) & ~3);
1246 uint srcShapePtrScan = uWidth * 4;
1247
1248#if defined (Q_WS_WIN)
1249
1250 BITMAPV5HEADER bi;
1251 HBITMAP hBitmap;
1252 void *lpBits;
1253
1254 ::ZeroMemory(&bi, sizeof (BITMAPV5HEADER));
1255 bi.bV5Size = sizeof(BITMAPV5HEADER);
1256 bi.bV5Width = uWidth;
1257 bi.bV5Height = - (LONG)uHeight;
1258 bi.bV5Planes = 1;
1259 bi.bV5BitCount = 32;
1260 bi.bV5Compression = BI_BITFIELDS;
1261 bi.bV5RedMask = 0x00FF0000;
1262 bi.bV5GreenMask = 0x0000FF00;
1263 bi.bV5BlueMask = 0x000000FF;
1264 if (fHasAlpha)
1265 bi.bV5AlphaMask = 0xFF000000;
1266 else
1267 bi.bV5AlphaMask = 0;
1268
1269 HDC hdc = GetDC(NULL);
1270
1271 /* Create the DIB section with an alpha channel: */
1272 hBitmap = CreateDIBSection(hdc, (BITMAPINFO *)&bi, DIB_RGB_COLORS, (void **)&lpBits, NULL, (DWORD) 0);
1273
1274 ReleaseDC(NULL, hdc);
1275
1276 HBITMAP hMonoBitmap = NULL;
1277 if (fHasAlpha)
1278 {
1279 /* Create an empty mask bitmap: */
1280 hMonoBitmap = CreateBitmap(uWidth, uHeight, 1, 1, NULL);
1281 }
1282 else
1283 {
1284 /* Word aligned AND mask. Will be allocated and created if necessary. */
1285 uint8_t *pu8AndMaskWordAligned = NULL;
1286
1287 /* Width in bytes of the original AND mask scan line. */
1288 uint32_t cbAndMaskScan = (uWidth + 7) / 8;
1289
1290 if (cbAndMaskScan & 1)
1291 {
1292 /* Original AND mask is not word aligned. */
1293
1294 /* Allocate memory for aligned AND mask. */
1295 pu8AndMaskWordAligned = (uint8_t *)RTMemTmpAllocZ((cbAndMaskScan + 1) * uHeight);
1296
1297 Assert(pu8AndMaskWordAligned);
1298
1299 if (pu8AndMaskWordAligned)
1300 {
1301 /* According to MSDN the padding bits must be 0.
1302 * Compute the bit mask to set padding bits to 0 in the last byte of original AND mask. */
1303 uint32_t u32PaddingBits = cbAndMaskScan * 8 - uWidth;
1304 Assert(u32PaddingBits < 8);
1305 uint8_t u8LastBytesPaddingMask = (uint8_t)(0xFF << u32PaddingBits);
1306
1307 Log(("u8LastBytesPaddingMask = %02X, aligned w = %d, width = %d, cbAndMaskScan = %d\n",
1308 u8LastBytesPaddingMask, (cbAndMaskScan + 1) * 8, uWidth, cbAndMaskScan));
1309
1310 uint8_t *src = (uint8_t *)srcAndMaskPtr;
1311 uint8_t *dst = pu8AndMaskWordAligned;
1312
1313 unsigned i;
1314 for (i = 0; i < uHeight; i++)
1315 {
1316 memcpy(dst, src, cbAndMaskScan);
1317
1318 dst[cbAndMaskScan - 1] &= u8LastBytesPaddingMask;
1319
1320 src += cbAndMaskScan;
1321 dst += cbAndMaskScan + 1;
1322 }
1323 }
1324 }
1325
1326 /* Create the AND mask bitmap: */
1327 hMonoBitmap = ::CreateBitmap(uWidth, uHeight, 1, 1,
1328 pu8AndMaskWordAligned? pu8AndMaskWordAligned: srcAndMaskPtr);
1329
1330 if (pu8AndMaskWordAligned)
1331 {
1332 RTMemTmpFree(pu8AndMaskWordAligned);
1333 }
1334 }
1335
1336 Assert(hBitmap);
1337 Assert(hMonoBitmap);
1338 if (hBitmap && hMonoBitmap)
1339 {
1340 DWORD *dstShapePtr = (DWORD *) lpBits;
1341
1342 for (uint y = 0; y < uHeight; y ++)
1343 {
1344 memcpy(dstShapePtr, srcShapePtr, srcShapePtrScan);
1345 srcShapePtr += srcShapePtrScan;
1346 dstShapePtr += uWidth;
1347 }
1348
1349 ICONINFO ii;
1350 ii.fIcon = FALSE;
1351 ii.xHotspot = uXHot;
1352 ii.yHotspot = uYHot;
1353 ii.hbmMask = hMonoBitmap;
1354 ii.hbmColor = hBitmap;
1355
1356 HCURSOR hAlphaCursor = CreateIconIndirect(&ii);
1357 Assert(hAlphaCursor);
1358 if (hAlphaCursor)
1359 {
1360 /* Set the new cursor: */
1361 m_cursor = QCursor(hAlphaCursor);
1362 if (m_alphaCursor)
1363 DestroyIcon(m_alphaCursor);
1364 m_alphaCursor = hAlphaCursor;
1365 m_fIsValidPointerShapePresent = true;
1366 }
1367 }
1368
1369 if (hMonoBitmap)
1370 DeleteObject(hMonoBitmap);
1371 if (hBitmap)
1372 DeleteObject(hBitmap);
1373
1374#elif defined (Q_WS_X11) && !defined (VBOX_WITHOUT_XCURSOR)
1375
1376 XcursorImage *img = XcursorImageCreate(uWidth, uHeight);
1377 Assert(img);
1378 if (img)
1379 {
1380 img->xhot = uXHot;
1381 img->yhot = uYHot;
1382
1383 XcursorPixel *dstShapePtr = img->pixels;
1384
1385 for (uint y = 0; y < uHeight; y ++)
1386 {
1387 memcpy (dstShapePtr, srcShapePtr, srcShapePtrScan);
1388
1389 if (!fHasAlpha)
1390 {
1391 /* Convert AND mask to the alpha channel: */
1392 uchar byte = 0;
1393 for (uint x = 0; x < uWidth; x ++)
1394 {
1395 if (!(x % 8))
1396 byte = *(srcAndMaskPtr ++);
1397 else
1398 byte <<= 1;
1399
1400 if (byte & 0x80)
1401 {
1402 /* Linux doesn't support inverted pixels (XOR ops,
1403 * to be exact) in cursor shapes, so we detect such
1404 * pixels and always replace them with black ones to
1405 * make them visible at least over light colors */
1406 if (dstShapePtr [x] & 0x00FFFFFF)
1407 dstShapePtr [x] = 0xFF000000;
1408 else
1409 dstShapePtr [x] = 0x00000000;
1410 }
1411 else
1412 dstShapePtr [x] |= 0xFF000000;
1413 }
1414 }
1415
1416 srcShapePtr += srcShapePtrScan;
1417 dstShapePtr += uWidth;
1418 }
1419
1420 /* Set the new cursor: */
1421 m_cursor = QCursor(XcursorImageLoadCursor(QX11Info::display(), img));
1422 m_fIsValidPointerShapePresent = true;
1423
1424 XcursorImageDestroy(img);
1425 }
1426
1427#elif defined(Q_WS_MAC)
1428
1429 /* Create a ARGB image out of the shape data. */
1430 QImage image (uWidth, uHeight, QImage::Format_ARGB32);
1431 const uint8_t* pbSrcMask = static_cast<const uint8_t*> (srcAndMaskPtr);
1432 unsigned cbSrcMaskLine = RT_ALIGN (uWidth, 8) / 8;
1433 for (unsigned int y = 0; y < uHeight; ++y)
1434 {
1435 for (unsigned int x = 0; x < uWidth; ++x)
1436 {
1437 unsigned int color = ((unsigned int*)srcShapePtr)[y*uWidth+x];
1438 /* If the alpha channel isn't in the shape data, we have to
1439 * create them from the and-mask. This is a bit field where 1
1440 * represent transparency & 0 opaque respectively. */
1441 if (!fHasAlpha)
1442 {
1443 if (!(pbSrcMask[x / 8] & (1 << (7 - (x % 8)))))
1444 color |= 0xff000000;
1445 else
1446 {
1447 /* This isn't quite right, but it's the best we can do I think... */
1448 if (color & 0x00ffffff)
1449 color = 0xff000000;
1450 else
1451 color = 0x00000000;
1452 }
1453 }
1454 image.setPixel (x, y, color);
1455 }
1456 /* Move one scanline forward. */
1457 pbSrcMask += cbSrcMaskLine;
1458 }
1459
1460 /* Set the new cursor: */
1461 m_cursor = QCursor(QPixmap::fromImage(image), uXHot, uYHot);
1462 m_fIsValidPointerShapePresent = true;
1463 NOREF(srcShapePtrScan);
1464
1465#else
1466
1467# warning "port me"
1468
1469#endif
1470}
1471
1472void UISession::reinitMenuPool()
1473{
1474 /* Get uisession machine: */
1475 const CMachine &machine = session().GetConsole().GetMachine();
1476
1477 /* Availability settings: */
1478 {
1479 /* USB Stuff: */
1480 const CUSBController &usbController = machine.GetUSBController();
1481 if ( usbController.isNull()
1482 || !usbController.GetEnabled()
1483 || !usbController.GetProxyAvailable())
1484 {
1485 /* Hide USB menu if controller is NULL or no USB proxy available: */
1486 uimachine()->actionsPool()->action(UIActionIndex_Menu_USBDevices)->setVisible(false);
1487 }
1488 else
1489 {
1490 /* Enable/Disable USB menu depending on USB controller: */
1491 uimachine()->actionsPool()->action(UIActionIndex_Menu_USBDevices)->setEnabled(usbController.GetEnabled());
1492 }
1493 }
1494
1495 /* Prepare some initial settings: */
1496 {
1497 /* Initialize CD/FD menus: */
1498 int iDevicesCountCD = 0;
1499 int iDevicesCountFD = 0;
1500 const CMediumAttachmentVector &attachments = machine.GetMediumAttachments();
1501 foreach (const CMediumAttachment &attachment, attachments)
1502 {
1503 if (attachment.GetType() == KDeviceType_DVD)
1504 ++ iDevicesCountCD;
1505 if (attachment.GetType() == KDeviceType_Floppy)
1506 ++ iDevicesCountFD;
1507 }
1508 QAction *pOpticalDevicesMenu = uimachine()->actionsPool()->action(UIActionIndex_Menu_OpticalDevices);
1509 QAction *pFloppyDevicesMenu = uimachine()->actionsPool()->action(UIActionIndex_Menu_FloppyDevices);
1510 pOpticalDevicesMenu->setData(iDevicesCountCD);
1511 pOpticalDevicesMenu->setVisible(iDevicesCountCD);
1512 pFloppyDevicesMenu->setData(iDevicesCountFD);
1513 pFloppyDevicesMenu->setVisible(iDevicesCountFD);
1514 }
1515}
1516
1517void UISession::preparePowerUp()
1518{
1519#ifdef VBOX_WITH_UPDATE_REQUEST
1520 /* Check for updates if necessary: */
1521 vboxGlobal().showUpdateDialog(false /* force request? */);
1522#endif
1523
1524 /* Notify user about mouse&keyboard auto-capturing: */
1525 if (vboxGlobal().settings().autoCapture())
1526 vboxProblem().remindAboutAutoCapture();
1527
1528 /* Shows first run wizard if necessary: */
1529 if (isFirstTimeStarted())
1530 {
1531 UIFirstRunWzd wzd(mainMachineWindow(), session().GetMachine());
1532 wzd.exec();
1533 }
1534}
1535
1536#ifdef VBOX_WITH_VIDEOHWACCEL
1537UIFrameBuffer* UISession::frameBuffer(ulong screenId) const
1538{
1539 Assert(screenId < (ulong)m_FrameBufferVector.size());
1540 return m_FrameBufferVector.value((int)screenId, NULL);
1541}
1542
1543int UISession::setFrameBuffer(ulong screenId, UIFrameBuffer* pFrameBuffer)
1544{
1545 Assert(screenId < (ulong)m_FrameBufferVector.size());
1546 if (screenId < (ulong)m_FrameBufferVector.size())
1547 {
1548 m_FrameBufferVector[(int)screenId] = pFrameBuffer;
1549 return VINF_SUCCESS;
1550 }
1551 return VERR_INVALID_PARAMETER;
1552}
1553#endif
Note: See TracBrowser for help on using the repository browser.

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