VirtualBox

source: vbox/trunk/src/VBox/Frontends/VirtualBox/src/VBoxConsoleView.cpp@ 20243

Last change on this file since 20243 was 20243, checked in by vboxsync, 16 years ago

View HW Accel: guest ddraw command definitions, moved opengl framebuffer impl to a separate file

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 140.2 KB
Line 
1/** @file
2 *
3 * VBox frontends: Qt GUI ("VirtualBox"):
4 * VBoxConsoleView class implementation
5 */
6
7/*
8 * Copyright (C) 22006-2007 Sun Microsystems, Inc.
9 *
10 * This file is part of VirtualBox Open Source Edition (OSE), as
11 * available from http://www.virtualbox.org. This file is free software;
12 * you can redistribute it and/or modify it under the terms of the GNU
13 * General Public License (GPL) as published by the Free Software
14 * Foundation, in version 2 as it comes in the "COPYING" file of the
15 * VirtualBox OSE distribution. VirtualBox OSE is distributed in the
16 * hope that it will be useful, but WITHOUT ANY WARRANTY of any kind.
17 *
18 * Please contact Sun Microsystems, Inc., 4150 Network Circle, Santa
19 * Clara, CA 95054 USA or visit http://www.sun.com if you need
20 * additional information or have any questions.
21 */
22
23#include <VBox/VBoxVideo.h>
24
25#include "VBoxConsoleView.h"
26#include "VBoxConsoleWnd.h"
27#include "VBoxUtils.h"
28
29#include "VBoxFrameBuffer.h"
30#include "VBoxGlobal.h"
31#include "VBoxProblemReporter.h"
32
33#ifdef Q_WS_PM
34#include "QIHotKeyEdit.h"
35#endif
36
37/* Qt includes */
38#include <QMenuBar>
39#include <QDesktopWidget>
40#include <QTimer>
41#include <QStatusBar>
42#include <QPainter>
43#include <QBitmap>
44
45#ifdef Q_WS_WIN
46// VBox/cdefs.h defines these:
47#undef LOWORD
48#undef HIWORD
49#undef LOBYTE
50#undef HIBYTE
51#include <windows.h>
52#endif
53
54#ifdef Q_WS_X11
55#include <QX11Info>
56// We need to capture some X11 events directly which
57// requires the XEvent structure to be defined. However,
58// including the Xlib header file will cause some nasty
59// conflicts with Qt. Therefore we use the following hack
60// to redefine those conflicting identifiers.
61#define XK_XKB_KEYS
62#define XK_MISCELLANY
63#include <X11/Xlib.h>
64#include <X11/Xutil.h>
65#include <X11/XKBlib.h>
66#include <X11/keysym.h>
67#ifdef KeyPress
68const int XFocusOut = FocusOut;
69const int XFocusIn = FocusIn;
70const int XKeyPress = KeyPress;
71const int XKeyRelease = KeyRelease;
72#undef KeyRelease
73#undef KeyPress
74#undef FocusOut
75#undef FocusIn
76#endif
77#include "XKeyboard.h"
78#ifndef VBOX_WITHOUT_XCURSOR
79# include <X11/Xcursor/Xcursor.h>
80#endif
81#endif // Q_WS_X11
82
83#if defined (Q_WS_MAC)
84# include "VBoxDockIconPreview.h"
85# include "DarwinKeyboard.h"
86# ifdef QT_MAC_USE_COCOA
87# include "darwin/VBoxCocoaApplication.h"
88# elif defined(VBOX_WITH_HACKED_QT)
89# include "QIApplication.h"
90# endif
91# include <Carbon/Carbon.h>
92# include <VBox/err.h>
93#endif /* defined (Q_WS_MAC) */
94
95#if defined (Q_WS_WIN32)
96
97static HHOOK gKbdHook = NULL;
98static VBoxConsoleView *gView = 0;
99
100LRESULT CALLBACK VBoxConsoleView::lowLevelKeyboardProc (int nCode,
101 WPARAM wParam, LPARAM lParam)
102{
103 Assert (gView);
104 if (gView && nCode == HC_ACTION &&
105 gView->winLowKeyboardEvent (wParam, *(KBDLLHOOKSTRUCT *) lParam))
106 return 1;
107
108 return CallNextHookEx (NULL, nCode, wParam, lParam);
109}
110
111#endif
112
113#if defined (Q_WS_MAC)
114# if defined (QT_MAC_USE_COCOA)
115/**
116 * Event handler callback for Mac OS X, Cocoa variant.
117 *
118 * (Registered with and called from VBoxCocoaApplication.)
119 *
120 * @returns true if the event should be dropped, false if it should be passed
121 * along.
122 * @param pvCocoaEvent The Cocoa event object.
123 * @param pvCarbonEvent The Carbon event object reference.
124 * @param pvUser The user argument.
125 */
126/* static */
127bool VBoxConsoleView::darwinEventHandlerProc (const void *pvCocoaEvent,
128 const void *pvCarbonEvent,
129 void *pvUser)
130{
131 VBoxConsoleView *view = (VBoxConsoleView *)pvUser;
132 EventRef inEvent = (EventRef)pvCarbonEvent;
133 UInt32 eventClass = ::GetEventClass (inEvent);
134
135#if 0
136 /* For debugging events. */
137 if (eventClass != 'cute')
138 ::VBoxCocoaApplication_printEvent ("view: ", pvCocoaEvent);
139#endif
140
141 /*
142 * Not sure but this seems an triggered event if the spotlight searchbar is
143 * displayed. So flag that the host key isn't pressed alone.
144 */
145 if ( eventClass == 'cgs '
146 && view->mIsHostkeyPressed
147 && ::GetEventKind (inEvent) == 0x15)
148 view->mIsHostkeyAlone = false;
149
150 /*
151 * All keyboard class events needs to be handled.
152 */
153 if (eventClass == kEventClassKeyboard)
154 {
155 if (view->darwinKeyboardEvent (pvCocoaEvent, inEvent))
156 return true;
157 }
158 /*
159 * Command-H and Command-Q aren't properly disabled yet, and it's still
160 * possible to use the left command key to invoke them when the keyboard
161 * is captured. We discard the events these if the keyboard is captured
162 * as a half measure to prevent unexpected behaviour. However, we don't
163 * get any key down/up events, so these combinations are dead to the guest...
164 */
165 else if (eventClass == kEventClassCommand)
166 {
167 if (view->mKbdCaptured)
168 return true;
169 }
170 /* Pass the event along. */
171 return false;
172}
173
174# elif !defined (VBOX_WITH_HACKED_QT)
175/**
176 * Event handler callback for Mac OS X.
177 */
178/* static */
179pascal OSStatus VBoxConsoleView::darwinEventHandlerProc (EventHandlerCallRef inHandlerCallRef,
180 EventRef inEvent, void *inUserData)
181{
182 VBoxConsoleView *view = static_cast<VBoxConsoleView *> (inUserData);
183 UInt32 eventClass = ::GetEventClass (inEvent);
184
185 /* For debugging events */
186 /*
187 if (eventClass != 'cute')
188 ::darwinDebugPrintEvent ("view: ", inEvent);
189 */
190
191 /* Not sure but this seems an triggered event if the spotlight searchbar is
192 * displayed. So flag that the host key isn't pressed alone. */
193 if ( eventClass == 'cgs '
194 && view->mIsHostkeyPressed
195 && ::GetEventKind (inEvent) == 0x15)
196 view->mIsHostkeyAlone = false;
197
198 if (eventClass == kEventClassKeyboard)
199 {
200 if (view->darwinKeyboardEvent (NULL, inEvent))
201 return 0;
202 }
203 /*
204 * Command-H and Command-Q aren't properly disabled yet, and it's still
205 * possible to use the left command key to invoke them when the keyboard
206 * is captured. We discard the events these if the keyboard is captured
207 * as a half measure to prevent unexpected behaviour. However, we don't
208 * get any key down/up events, so these combinations are dead to the guest...
209 */
210 else if (eventClass == kEventClassCommand)
211 {
212 if (view->mKbdCaptured)
213 return 0;
214 }
215 return ::CallNextEventHandler (inHandlerCallRef, inEvent);
216}
217
218# else /* VBOX_WITH_HACKED_QT */
219/**
220 * Event handler callback for Mac OS X.
221 */
222/* static */
223bool VBoxConsoleView::macEventFilter (EventRef inEvent, void *inUserData)
224{
225 VBoxConsoleView *view = static_cast<VBoxConsoleView *> (inUserData);
226 UInt32 eventClass = ::GetEventClass (inEvent);
227 UInt32 eventKind = ::GetEventKind (inEvent);
228
229 /* For debugging events */
230 /*
231 if (!(eventClass == 'cute'))
232 ::darwinDebugPrintEvent ("view: ", inEvent);
233 */
234
235 /* Not sure but this seems an triggered event if the spotlight searchbar is
236 * displayed. So flag that the host key isn't pressed alone. */
237 if (eventClass == 'cgs ' && eventKind == 0x15 &&
238 view->mIsHostkeyPressed)
239 view->mIsHostkeyAlone = false;
240
241 if (eventClass == kEventClassKeyboard)
242 {
243 if (view->darwinKeyboardEvent (NULL, inEvent))
244 return true;
245 }
246 return false;
247}
248# endif /* VBOX_WITH_HACKED_QT */
249
250#endif /* Q_WS_MAC */
251
252/** Guest mouse pointer shape change event. */
253class MousePointerChangeEvent : public QEvent
254{
255public:
256 MousePointerChangeEvent (bool visible, bool alpha, uint xhot, uint yhot,
257 uint width, uint height,
258 const uchar *shape) :
259 QEvent ((QEvent::Type) VBoxDefs::MousePointerChangeEventType),
260 vis (visible), alph (alpha), xh (xhot), yh (yhot), w (width), h (height),
261 data (NULL)
262 {
263 // make a copy of shape
264 uint dataSize = ((((width + 7) / 8 * height) + 3) & ~3) + width * 4 * height;
265
266 if (shape) {
267 data = new uchar [dataSize];
268 memcpy ((void *) data, (void *) shape, dataSize);
269 }
270 }
271 ~MousePointerChangeEvent()
272 {
273 if (data) delete[] data;
274 }
275 bool isVisible() const { return vis; }
276 bool hasAlpha() const { return alph; }
277 uint xHot() const { return xh; }
278 uint yHot() const { return yh; }
279 uint width() const { return w; }
280 uint height() const { return h; }
281 const uchar *shapeData() const { return data; }
282private:
283 bool vis, alph;
284 uint xh, yh, w, h;
285 const uchar *data;
286};
287
288/** Guest mouse absolute positioning capability change event. */
289class MouseCapabilityEvent : public QEvent
290{
291public:
292 MouseCapabilityEvent (bool supportsAbsolute, bool needsHostCursor) :
293 QEvent ((QEvent::Type) VBoxDefs::MouseCapabilityEventType),
294 can_abs (supportsAbsolute),
295 needs_host_cursor (needsHostCursor) {}
296 bool supportsAbsolute() const { return can_abs; }
297 bool needsHostCursor() const { return needs_host_cursor; }
298private:
299 bool can_abs;
300 bool needs_host_cursor;
301};
302
303/** Machine state change. */
304class StateChangeEvent : public QEvent
305{
306public:
307 StateChangeEvent (KMachineState state) :
308 QEvent ((QEvent::Type) VBoxDefs::MachineStateChangeEventType),
309 s (state) {}
310 KMachineState machineState() const { return s; }
311private:
312 KMachineState s;
313};
314
315/** Guest Additions property changes. */
316class GuestAdditionsEvent : public QEvent
317{
318public:
319 GuestAdditionsEvent (const QString &aOsTypeId,
320 const QString &aAddVersion,
321 bool aAddActive,
322 bool aSupportsSeamless,
323 bool aSupportsGraphics) :
324 QEvent ((QEvent::Type) VBoxDefs::AdditionsStateChangeEventType),
325 mOsTypeId (aOsTypeId), mAddVersion (aAddVersion),
326 mAddActive (aAddActive), mSupportsSeamless (aSupportsSeamless),
327 mSupportsGraphics (aSupportsGraphics) {}
328 const QString &osTypeId() const { return mOsTypeId; }
329 const QString &additionVersion() const { return mAddVersion; }
330 bool additionActive() const { return mAddActive; }
331 bool supportsSeamless() const { return mSupportsSeamless; }
332 bool supportsGraphics() const { return mSupportsGraphics; }
333private:
334 QString mOsTypeId;
335 QString mAddVersion;
336 bool mAddActive;
337 bool mSupportsSeamless;
338 bool mSupportsGraphics;
339};
340
341/** DVD/Floppy drive change event */
342class MediaDriveChangeEvent : public QEvent
343{
344public:
345 MediaDriveChangeEvent (VBoxDefs::MediaType aType)
346 : QEvent ((QEvent::Type) VBoxDefs::MediaDriveChangeEventType)
347 , mType (aType) {}
348 VBoxDefs::MediaType type() const { return mType; }
349private:
350 VBoxDefs::MediaType mType;
351};
352
353/** Menu activation event */
354class ActivateMenuEvent : public QEvent
355{
356public:
357 ActivateMenuEvent (QAction *aData) :
358 QEvent ((QEvent::Type) VBoxDefs::ActivateMenuEventType),
359 mAction (aData) {}
360 QAction *action() const { return mAction; }
361private:
362 QAction *mAction;
363};
364
365/** VM Runtime error event */
366class RuntimeErrorEvent : public QEvent
367{
368public:
369 RuntimeErrorEvent (bool aFatal, const QString &aErrorID,
370 const QString &aMessage) :
371 QEvent ((QEvent::Type) VBoxDefs::RuntimeErrorEventType),
372 mFatal (aFatal), mErrorID (aErrorID), mMessage (aMessage) {}
373 bool fatal() const { return mFatal; }
374 QString errorID() const { return mErrorID; }
375 QString message() const { return mMessage; }
376private:
377 bool mFatal;
378 QString mErrorID;
379 QString mMessage;
380};
381
382/** Modifier key change event */
383class ModifierKeyChangeEvent : public QEvent
384{
385public:
386 ModifierKeyChangeEvent (bool fNumLock, bool fCapsLock, bool fScrollLock) :
387 QEvent ((QEvent::Type) VBoxDefs::ModifierKeyChangeEventType),
388 mNumLock (fNumLock), mCapsLock (fCapsLock), mScrollLock (fScrollLock) {}
389 bool numLock() const { return mNumLock; }
390 bool capsLock() const { return mCapsLock; }
391 bool scrollLock() const { return mScrollLock; }
392private:
393 bool mNumLock, mCapsLock, mScrollLock;
394};
395
396/** Network adapter change event */
397class NetworkAdapterChangeEvent : public QEvent
398{
399public:
400 NetworkAdapterChangeEvent (INetworkAdapter *aAdapter) :
401 QEvent ((QEvent::Type) VBoxDefs::NetworkAdapterChangeEventType),
402 mAdapter (aAdapter) {}
403 INetworkAdapter* networkAdapter() { return mAdapter; }
404private:
405 INetworkAdapter *mAdapter;
406};
407
408/** USB controller state change event */
409class USBControllerStateChangeEvent : public QEvent
410{
411public:
412 USBControllerStateChangeEvent()
413 : QEvent ((QEvent::Type) VBoxDefs::USBCtlStateChangeEventType) {}
414};
415
416/** USB device state change event */
417class USBDeviceStateChangeEvent : public QEvent
418{
419public:
420 USBDeviceStateChangeEvent (const CUSBDevice &aDevice, bool aAttached,
421 const CVirtualBoxErrorInfo &aError) :
422 QEvent ((QEvent::Type) VBoxDefs::USBDeviceStateChangeEventType),
423 mDevice (aDevice), mAttached (aAttached), mError (aError) {}
424 CUSBDevice device() const { return mDevice; }
425 bool attached() const { return mAttached; }
426 CVirtualBoxErrorInfo error() const { return mError; }
427private:
428 CUSBDevice mDevice;
429 bool mAttached;
430 CVirtualBoxErrorInfo mError;
431};
432
433//
434// VBoxConsoleCallback class
435/////////////////////////////////////////////////////////////////////////////
436
437class VBoxConsoleCallback : VBOX_SCRIPTABLE_IMPL(IConsoleCallback)
438{
439public:
440
441 VBoxConsoleCallback (VBoxConsoleView *v) {
442#if defined (Q_WS_WIN)
443 mRefCnt = 0;
444#endif
445 mView = v;
446 }
447
448 virtual ~VBoxConsoleCallback() {}
449
450 NS_DECL_ISUPPORTS
451
452#if defined (Q_WS_WIN)
453 STDMETHOD_(ULONG, AddRef)() {
454 return ::InterlockedIncrement (&mRefCnt);
455 }
456 STDMETHOD_(ULONG, Release)()
457 {
458 long cnt = ::InterlockedDecrement (&mRefCnt);
459 if (cnt == 0)
460 delete this;
461 return cnt;
462 }
463 STDMETHOD(QueryInterface) (REFIID riid , void **ppObj)
464 {
465 if (riid == IID_IUnknown) {
466 *ppObj = this;
467 AddRef();
468 return S_OK;
469 }
470 if (riid == IID_IConsoleCallback) {
471 *ppObj = this;
472 AddRef();
473 return S_OK;
474 }
475 *ppObj = NULL;
476 return E_NOINTERFACE;
477 }
478#endif
479
480 STDMETHOD(OnMousePointerShapeChange) (BOOL visible, BOOL alpha,
481 ULONG xhot, ULONG yhot,
482 ULONG width, ULONG height,
483 BYTE *shape)
484 {
485 QApplication::postEvent (mView,
486 new MousePointerChangeEvent (visible, alpha,
487 xhot, yhot,
488 width, height, shape));
489 return S_OK;
490 }
491
492 STDMETHOD(OnMouseCapabilityChange)(BOOL supportsAbsolute, BOOL needsHostCursor)
493 {
494 QApplication::postEvent (mView,
495 new MouseCapabilityEvent (supportsAbsolute,
496 needsHostCursor));
497 return S_OK;
498 }
499
500 STDMETHOD(OnKeyboardLedsChange)(BOOL fNumLock, BOOL fCapsLock, BOOL fScrollLock)
501 {
502 QApplication::postEvent (mView,
503 new ModifierKeyChangeEvent (fNumLock, fCapsLock,
504 fScrollLock));
505 return S_OK;
506 }
507
508 STDMETHOD(OnStateChange)(MachineState_T machineState)
509 {
510 LogFlowFunc (("machineState=%d\n", machineState));
511 QApplication::postEvent (mView,
512 new StateChangeEvent ((KMachineState) machineState));
513 return S_OK;
514 }
515
516 STDMETHOD(OnAdditionsStateChange)()
517 {
518 CGuest guest = mView->console().GetGuest();
519 LogFlowFunc (("ver=%s, active=%d\n",
520 guest.GetAdditionsVersion().toLatin1().constData(),
521 guest.GetAdditionsActive()));
522 QApplication::postEvent (mView,
523 new GuestAdditionsEvent (
524 guest.GetOSTypeId(),
525 guest.GetAdditionsVersion(),
526 guest.GetAdditionsActive(),
527 guest.GetSupportsSeamless(),
528 guest.GetSupportsGraphics()));
529 return S_OK;
530 }
531
532 STDMETHOD(OnDVDDriveChange)()
533 {
534 LogFlowFunc (("DVD Drive changed\n"));
535 QApplication::postEvent (mView,
536 new MediaDriveChangeEvent (VBoxDefs::MediaType_DVD));
537 return S_OK;
538 }
539
540 STDMETHOD(OnFloppyDriveChange)()
541 {
542 LogFlowFunc (("Floppy Drive changed\n"));
543 QApplication::postEvent (mView,
544 new MediaDriveChangeEvent (VBoxDefs::MediaType_Floppy));
545 return S_OK;
546 }
547
548 STDMETHOD(OnNetworkAdapterChange) (INetworkAdapter *aNetworkAdapter)
549 {
550 QApplication::postEvent (mView,
551 new NetworkAdapterChangeEvent (aNetworkAdapter));
552 return S_OK;
553 }
554
555 STDMETHOD(OnStorageControllerChange) ()
556 {
557 /* @todo */
558 //QApplication::postEvent (mView,
559 // new StorageControllerChangeEvent ());
560 return S_OK;
561 }
562
563 STDMETHOD(OnSerialPortChange) (ISerialPort *aSerialPort)
564 {
565 NOREF(aSerialPort);
566 return S_OK;
567 }
568
569 STDMETHOD(OnParallelPortChange) (IParallelPort *aParallelPort)
570 {
571 NOREF(aParallelPort);
572 return S_OK;
573 }
574
575 STDMETHOD(OnVRDPServerChange)()
576 {
577 return S_OK;
578 }
579
580 STDMETHOD(OnUSBControllerChange)()
581 {
582 QApplication::postEvent (mView,
583 new USBControllerStateChangeEvent());
584 return S_OK;
585 }
586
587 STDMETHOD(OnUSBDeviceStateChange)(IUSBDevice *aDevice, BOOL aAttached,
588 IVirtualBoxErrorInfo *aError)
589 {
590 QApplication::postEvent (mView,
591 new USBDeviceStateChangeEvent (
592 CUSBDevice (aDevice),
593 bool (aAttached),
594 CVirtualBoxErrorInfo (aError)));
595 return S_OK;
596 }
597
598 STDMETHOD(OnSharedFolderChange) (Scope_T aScope)
599 {
600 NOREF(aScope);
601 QApplication::postEvent (mView,
602 new QEvent ((QEvent::Type)
603 VBoxDefs::SharedFolderChangeEventType));
604 return S_OK;
605 }
606
607 STDMETHOD(OnRuntimeError)(BOOL fatal, IN_BSTR id, IN_BSTR message)
608 {
609 QApplication::postEvent (mView,
610 new RuntimeErrorEvent (!!fatal,
611 QString::fromUtf16 (id),
612 QString::fromUtf16 (message)));
613 return S_OK;
614 }
615
616 STDMETHOD(OnCanShowWindow) (BOOL *canShow)
617 {
618 if (!canShow)
619 return E_POINTER;
620
621 /* as long as there is VBoxConsoleView (which creates/destroys us), it
622 * can be shown */
623 *canShow = TRUE;
624 return S_OK;
625 }
626
627 STDMETHOD(OnShowWindow) (ULONG64 *winId)
628 {
629 if (!winId)
630 return E_POINTER;
631
632#if defined (Q_WS_MAC)
633 /*
634 * Let's try the simple approach first - grab the focus.
635 * Getting a window out of the dock (minimized or whatever it's called)
636 * needs to be done on the GUI thread, so post it a note.
637 */
638 *winId = 0;
639 if (!mView)
640 return S_OK;
641
642 ProcessSerialNumber psn = { 0, kCurrentProcess };
643 OSErr rc = ::SetFrontProcess (&psn);
644 if (!rc)
645 QApplication::postEvent (mView, new QEvent ((QEvent::Type)VBoxDefs::ShowWindowEventType));
646 else
647 {
648 /*
649 * It failed for some reason, send the other process our PSN so it can try.
650 * (This is just a precaution should Mac OS X start imposing the same sensible
651 * focus stealing restrictions that other window managers implement.)
652 */
653 AssertMsgFailed(("SetFrontProcess -> %#x\n", rc));
654 if (::GetCurrentProcess (&psn))
655 *winId = RT_MAKE_U64 (psn.lowLongOfPSN, psn.highLongOfPSN);
656 }
657
658#else
659 /* Return the ID of the top-level console window. */
660 *winId = (ULONG64) mView->window()->winId();
661#endif
662
663 return S_OK;
664 }
665
666protected:
667
668 VBoxConsoleView *mView;
669
670#if defined (Q_WS_WIN)
671private:
672 long mRefCnt;
673#endif
674};
675
676#if !defined (Q_WS_WIN)
677NS_DECL_CLASSINFO (VBoxConsoleCallback)
678NS_IMPL_THREADSAFE_ISUPPORTS1_CI (VBoxConsoleCallback, IConsoleCallback)
679#endif
680
681class VBoxViewport: public QWidget
682{
683public:
684 VBoxViewport (QWidget *aParent)
685 : QWidget (aParent)
686 {
687 /* No need for background drawing */
688 setAttribute (Qt::WA_OpaquePaintEvent);
689 }
690 virtual QPaintEngine * paintEngine() const
691 {
692 if (testAttribute (Qt::WA_PaintOnScreen))
693 return NULL;
694 else
695 return QWidget::paintEngine();
696 }
697};
698
699//
700// VBoxConsoleView class
701/////////////////////////////////////////////////////////////////////////////
702
703/** @class VBoxConsoleView
704 *
705 * The VBoxConsoleView class is a widget that implements a console
706 * for the running virtual machine.
707 */
708
709VBoxConsoleView::VBoxConsoleView (VBoxConsoleWnd *mainWnd,
710 const CConsole &console,
711 VBoxDefs::RenderMode rm,
712 QWidget *parent)
713 : QAbstractScrollArea (parent)
714 , mMainWnd (mainWnd)
715 , mConsole (console)
716 , gs (vboxGlobal().settings())
717 , mAttached (false)
718 , mKbdCaptured (false)
719 , mMouseCaptured (false)
720 , mMouseAbsolute (false)
721 , mMouseIntegration (true)
722 , mDisableAutoCapture (false)
723 , mIsHostkeyPressed (false)
724 , mIsHostkeyAlone (false)
725 , mIgnoreMainwndResize (true)
726 , mAutoresizeGuest (false)
727 , mIgnoreFrameBufferResize (false)
728 , mDoResize (false)
729 , mGuestSupportsGraphics (false)
730 , mNumLock (false)
731 , mScrollLock (false)
732 , mCapsLock (false)
733 , muNumLockAdaptionCnt (2)
734 , muCapsLockAdaptionCnt (2)
735 , mode (rm)
736#if defined(Q_WS_WIN)
737 , mAlphaCursor (NULL)
738#endif
739#if defined(Q_WS_MAC)
740# if !defined (VBOX_WITH_HACKED_QT) && !defined (QT_MAC_USE_COCOA)
741 , mDarwinEventHandlerRef (NULL)
742# endif
743 , mDarwinKeyModifiers (0)
744 , mKeyboardGrabbed (false)
745 , mDockIconEnabled (true)
746#endif
747 , mDesktopGeo (DesktopGeo_Invalid)
748 , mPassCAD (false)
749 , mHideHostPointer (false)
750{
751 Assert (!mConsole.isNull() &&
752 !mConsole.GetDisplay().isNull() &&
753 !mConsole.GetKeyboard().isNull() &&
754 !mConsole.GetMouse().isNull());
755
756#ifdef Q_WS_MAC
757 /* Overlay logo for the dock icon */
758 //mVirtualBoxLogo = ::darwinToCGImageRef ("VirtualBox_cube_42px.png");
759 QString osTypeId = mConsole.GetGuest().GetOSTypeId();
760 mDockIconPreview = new VBoxDockIconPreview (mMainWnd, vboxGlobal().vmGuestOSTypeIcon (osTypeId));
761
762# ifdef QT_MAC_USE_COCOA
763 /** @todo Carbon -> Cocoa */
764# else /* !QT_MAC_USE_COCOA */
765 /* Install the event handler which will proceed external window handling */
766 EventHandlerUPP eventHandler = ::NewEventHandlerUPP (::darwinOverlayWindowHandler);
767 EventTypeSpec eventTypes[] =
768 {
769 { kEventClassVBox, kEventVBoxShowWindow },
770 { kEventClassVBox, kEventVBoxHideWindow },
771 { kEventClassVBox, kEventVBoxMoveWindow },
772 { kEventClassVBox, kEventVBoxResizeWindow },
773 { kEventClassVBox, kEventVBoxDisposeWindow },
774 { kEventClassVBox, kEventVBoxUpdateDock }
775 };
776
777 mDarwinWindowOverlayHandlerRef = NULL;
778 ::InstallApplicationEventHandler (eventHandler, RT_ELEMENTS (eventTypes), &eventTypes[0],
779 this, &mDarwinWindowOverlayHandlerRef);
780 ::DisposeEventHandlerUPP (eventHandler);
781# endif /* !QT_MAC_USE_COCOA */
782#endif /* QT_WS_MAC */
783
784 /* No frame around the view */
785 setFrameStyle (QFrame::NoFrame);
786
787#ifndef VBOX_WITH_VIDEOHWACCEL
788 VBoxViewport *pViewport = new VBoxViewport (this);
789#else
790// /* TODO: temporary always use VBoxGLWidget for debugging */
791 VBoxGLWidget *pViewport = new VBoxGLWidget (this);
792#endif
793 setViewport (pViewport);
794// pViewport->vboxDoInit();
795
796 /* enable MouseMove events */
797 viewport()->setMouseTracking (true);
798
799 /*
800 * QScrollView does the below on its own, but let's do it anyway
801 * for the case it will not do it in the future.
802 */
803 viewport()->installEventFilter (this);
804
805 /* to fix some focus issues */
806 mMainWnd->menuBar()->installEventFilter (this);
807
808 /* we want to be notified on some parent's events */
809 mMainWnd->installEventFilter (this);
810
811#ifdef Q_WS_X11
812 /* initialize the X keyboard subsystem */
813 initXKeyboard (QX11Info::display());
814#endif
815
816 ::memset (mPressedKeys, 0, sizeof (mPressedKeys));
817
818 /* setup rendering */
819
820 CDisplay display = mConsole.GetDisplay();
821 Assert (!display.isNull());
822
823 mFrameBuf = NULL;
824
825 LogFlowFunc (("Rendering mode: %d\n", mode));
826
827 switch (mode)
828 {
829#if defined (VBOX_GUI_USE_QGL)
830 case VBoxDefs::QGLMode:
831 mFrameBuf = new VBoxQGLFrameBuffer (this);
832 break;
833#endif
834#if defined (VBOX_GUI_USE_QIMAGE)
835 case VBoxDefs::QImageMode:
836 mFrameBuf = new VBoxQImageFrameBuffer (this);
837 break;
838#endif
839#if defined (VBOX_GUI_USE_SDL)
840 case VBoxDefs::SDLMode:
841 /* Indicate that we are doing all
842 * drawing stuff ourself */
843 pViewport->setAttribute (Qt::WA_PaintOnScreen);
844# ifdef Q_WS_X11
845 /* This is somehow necessary to prevent strange X11 warnings on
846 * i386 and segfaults on x86_64. */
847 XFlush(QX11Info::display());
848# endif
849 mFrameBuf = new VBoxSDLFrameBuffer (this);
850 /*
851 * disable scrollbars because we cannot correctly draw in a
852 * scrolled window using SDL
853 */
854 horizontalScrollBar()->setEnabled (false);
855 verticalScrollBar()->setEnabled (false);
856 break;
857#endif
858#if defined (VBOX_GUI_USE_DDRAW)
859 case VBoxDefs::DDRAWMode:
860 mFrameBuf = new VBoxDDRAWFrameBuffer (this);
861 break;
862#endif
863#if defined (VBOX_GUI_USE_QUARTZ2D)
864 case VBoxDefs::Quartz2DMode:
865 /* Indicate that we are doing all
866 * drawing stuff ourself */
867 pViewport->setAttribute (Qt::WA_PaintOnScreen);
868 mFrameBuf = new VBoxQuartz2DFrameBuffer (this);
869 break;
870#endif
871 default:
872 AssertReleaseMsgFailed (("Render mode must be valid: %d\n", mode));
873 LogRel (("Invalid render mode: %d\n", mode));
874 qApp->exit (1);
875 break;
876 }
877
878#if defined (VBOX_GUI_USE_DDRAW)
879 if (!mFrameBuf || mFrameBuf->address() == NULL)
880 {
881 if (mFrameBuf)
882 delete mFrameBuf;
883 mode = VBoxDefs::QImageMode;
884 mFrameBuf = new VBoxQImageFrameBuffer (this);
885 }
886#endif
887
888 if (mFrameBuf)
889 {
890 mFrameBuf->AddRef();
891 display.SetFramebuffer (VBOX_VIDEO_PRIMARY_SCREEN, CFramebuffer (mFrameBuf));
892 }
893
894 /* setup the callback */
895 mCallback = CConsoleCallback (new VBoxConsoleCallback (this));
896 mConsole.RegisterCallback (mCallback);
897 AssertWrapperOk (mConsole);
898
899 QPalette palette (viewport()->palette());
900 palette.setColor (viewport()->backgroundRole(), Qt::black);
901 viewport()->setPalette (palette);
902
903 setSizePolicy (QSizePolicy (QSizePolicy::Maximum, QSizePolicy::Maximum));
904 setMaximumSize (sizeHint());
905
906 setFocusPolicy (Qt::WheelFocus);
907
908 /* Remember the desktop geometry and register for geometry change
909 events for telling the guest about video modes we like. */
910
911 QString desktopGeometry = vboxGlobal().settings()
912 .publicProperty ("GUI/MaxGuestResolution");
913 if ((desktopGeometry == QString::null) ||
914 (desktopGeometry == "auto"))
915 setDesktopGeometry (DesktopGeo_Automatic, 0, 0);
916 else if (desktopGeometry == "any")
917 setDesktopGeometry (DesktopGeo_Any, 0, 0);
918 else
919 {
920 int width = desktopGeometry.section (',', 0, 0).toInt();
921 int height = desktopGeometry.section (',', 1, 1).toInt();
922 setDesktopGeometry (DesktopGeo_Fixed, width, height);
923 }
924 connect (QApplication::desktop(), SIGNAL (resized (int)),
925 this, SLOT (doResizeDesktop (int)));
926
927 QString passCAD = mConsole.GetMachine().GetExtraData (VBoxDefs::GUI_PassCAD);
928 if (!passCAD.isEmpty() &&
929 ((passCAD != "false") || (passCAD != "no"))
930 )
931 mPassCAD = true;
932
933#if defined (Q_WS_WIN)
934 gView = this;
935#endif
936
937#if defined (Q_WS_PM)
938 bool ok = VBoxHlpInstallKbdHook (0, winId(), UM_PREACCEL_CHAR);
939 Assert (ok);
940 NOREF (ok);
941#endif
942}
943
944VBoxConsoleView::~VBoxConsoleView()
945{
946#if defined (Q_WS_PM)
947 bool ok = VBoxHlpUninstallKbdHook (0, winId(), UM_PREACCEL_CHAR);
948 Assert (ok);
949 NOREF (ok);
950#endif
951
952#if defined (Q_WS_WIN)
953 if (gKbdHook)
954 UnhookWindowsHookEx (gKbdHook);
955 gView = 0;
956 if (mAlphaCursor)
957 DestroyIcon (mAlphaCursor);
958#endif
959
960 if (mFrameBuf)
961 {
962 /* detach our framebuffer from Display */
963 CDisplay display = mConsole.GetDisplay();
964 Assert (!display.isNull());
965 display.SetFramebuffer (VBOX_VIDEO_PRIMARY_SCREEN, CFramebuffer(NULL));
966 /* release the reference */
967 mFrameBuf->Release();
968 mFrameBuf = NULL;
969 }
970
971 mConsole.UnregisterCallback (mCallback);
972
973#if defined (Q_WS_MAC)
974# if !defined (QT_MAC_USE_COCOA)
975 if (mDarwinWindowOverlayHandlerRef)
976 {
977 ::RemoveEventHandler (mDarwinWindowOverlayHandlerRef);
978 mDarwinWindowOverlayHandlerRef = NULL;
979 }
980# endif
981 delete mDockIconPreview;
982 mDockIconPreview = NULL;
983#endif
984}
985
986//
987// Public members
988/////////////////////////////////////////////////////////////////////////////
989
990QSize VBoxConsoleView::sizeHint() const
991{
992#ifdef VBOX_WITH_DEBUGGER /** @todo figure out a more proper fix. */
993 /* HACK ALERT! Really ugly workaround for the resizing to 9x1 done
994 * by DevVGA if provoked before power on. */
995 QSize fb(mFrameBuf->width(), mFrameBuf->height());
996 if ( ( fb.width() < 16
997 || fb.height() < 16)
998 && ( vboxGlobal().isStartPausedEnabled()
999 || vboxGlobal().isDebuggerAutoShowEnabled()) )
1000 fb = QSize(640, 480);
1001 return QSize (fb.width() + frameWidth() * 2,
1002 fb.height() + frameWidth() * 2);
1003#else
1004 return QSize (mFrameBuf->width() + frameWidth() * 2,
1005 mFrameBuf->height() + frameWidth() * 2);
1006#endif
1007}
1008
1009/**
1010 * Attaches this console view to the managed virtual machine.
1011 *
1012 * @note This method is not really necessary these days -- the only place where
1013 * it gets called is VBoxConsole::openView(), right after powering the
1014 * VM up. We leave it as is just in case attaching/detaching will become
1015 * necessary some day (there are useful attached checks everywhere in the
1016 * code).
1017 */
1018void VBoxConsoleView::attach()
1019{
1020 if (!mAttached)
1021 {
1022 mAttached = true;
1023 }
1024}
1025
1026/**
1027 * Detaches this console view from the VM. Must be called to indicate
1028 * that the virtual machine managed by this instance will be no more valid
1029 * after this call.
1030 *
1031 * @note This method is not really necessary these days -- the only place where
1032 * it gets called is VBoxConsole::closeView(), when the VM is powered
1033 * down, before deleting VBoxConsoleView. We leave it as is just in case
1034 * attaching/detaching will become necessary some day (there are useful
1035 * attached checks everywhere in the code).
1036 */
1037void VBoxConsoleView::detach()
1038{
1039 if (mAttached)
1040 {
1041 /* reuse the focus event handler to uncapture everything */
1042 focusEvent (false);
1043 mAttached = false;
1044 }
1045}
1046
1047/**
1048 * Resizes the toplevel widget to fit the console view w/o scrollbars.
1049 * If adjustPosition is true and such resize is not possible (because the
1050 * console view size is lagrer then the available screen space) the toplevel
1051 * widget is resized and moved to become as large as possible while staying
1052 * fully visible.
1053 */
1054void VBoxConsoleView::normalizeGeometry (bool adjustPosition /* = false */)
1055{
1056 /* Make no normalizeGeometry in case we are in manual resize
1057 * mode or main window is maximized */
1058 if (mMainWnd->isWindowMaximized() || mMainWnd->isWindowFullScreen())
1059 return;
1060
1061 QWidget *tlw = window();
1062
1063 /* calculate client window offsets */
1064 QRect fr = tlw->frameGeometry();
1065 QRect r = tlw->geometry();
1066 int dl = r.left() - fr.left();
1067 int dt = r.top() - fr.top();
1068 int dr = fr.right() - r.right();
1069 int db = fr.bottom() - r.bottom();
1070
1071 /* get the best size w/o scroll bars */
1072 QSize s = tlw->sizeHint();
1073
1074 /* resize the frame to fit the contents */
1075 s -= tlw->size();
1076 fr.setRight (fr.right() + s.width());
1077 fr.setBottom (fr.bottom() + s.height());
1078
1079 if (adjustPosition)
1080 {
1081 QRegion ar;
1082 QDesktopWidget *dwt = QApplication::desktop();
1083 if (dwt->isVirtualDesktop())
1084 /* Compose complex available region */
1085 for (int i = 0; i < dwt->numScreens(); ++ i)
1086 ar += dwt->availableGeometry (i);
1087 else
1088 /* Get just a simple available rectangle */
1089 ar = dwt->availableGeometry (tlw->pos());
1090
1091 fr = VBoxGlobal::normalizeGeometry (
1092 fr, ar, mode != VBoxDefs::SDLMode /* canResize */);
1093 }
1094
1095#if 0
1096 /* center the frame on the desktop */
1097 fr.moveCenter (ar.center());
1098#endif
1099
1100 /* finally, set the frame geometry */
1101 tlw->setGeometry (fr.left() + dl, fr.top() + dt,
1102 fr.width() - dl - dr, fr.height() - dt - db);
1103}
1104
1105/**
1106 * Pauses or resumes the VM execution.
1107 */
1108bool VBoxConsoleView::pause (bool on)
1109{
1110 /* QAction::setOn() emits the toggled() signal, so avoid recursion when
1111 * QAction::setOn() is called from VBoxConsoleWnd::updateMachineState() */
1112 if (isPaused() == on)
1113 return true;
1114
1115 if (on)
1116 mConsole.Pause();
1117 else
1118 mConsole.Resume();
1119
1120 bool ok = mConsole.isOk();
1121 if (!ok)
1122 {
1123 if (on)
1124 vboxProblem().cannotPauseMachine (mConsole);
1125 else
1126 vboxProblem().cannotResumeMachine (mConsole);
1127 }
1128
1129 return ok;
1130}
1131
1132/**
1133 * Temporarily disables the mouse integration (or enables it back).
1134 */
1135void VBoxConsoleView::setMouseIntegrationEnabled (bool enabled)
1136{
1137 if (mMouseIntegration == enabled)
1138 return;
1139
1140 if (mMouseAbsolute)
1141 captureMouse (!enabled, false);
1142
1143 /* Hiding host cursor in case we are entering mouse integration
1144 * mode until it's shape is set to the guest cursor shape in
1145 * OnMousePointerShapeChange event handler.
1146 *
1147 * This is necessary to avoid double-cursor issue when both the
1148 * guest and the host cursors are displayed in one place one-above-one.
1149 *
1150 * This is a workaround because the correct decision is to notify
1151 * the Guest Additions about we are entering the mouse integration
1152 * mode. The GuestOS should hide it's cursor to allow using of
1153 * host cursor for the guest's manipulation.
1154 *
1155 * This notification is not possible right now due to there is
1156 * no the required API. */
1157 if (enabled)
1158 viewport()->setCursor (QCursor (Qt::BlankCursor));
1159
1160 mMouseIntegration = enabled;
1161
1162 emitMouseStateChanged();
1163}
1164
1165void VBoxConsoleView::setAutoresizeGuest (bool on)
1166{
1167 if (mAutoresizeGuest != on)
1168 {
1169 mAutoresizeGuest = on;
1170
1171 maybeRestrictMinimumSize();
1172
1173 if (mGuestSupportsGraphics && mAutoresizeGuest)
1174 doResizeHint();
1175 }
1176}
1177
1178/**
1179 * This method is called by VBoxConsoleWnd after it does everything necessary
1180 * on its side to go to or from fullscreen, but before it is shown.
1181 */
1182void VBoxConsoleView::onFullscreenChange (bool /* on */)
1183{
1184 /* Nothing to do here so far */
1185}
1186
1187/**
1188 * Notify the console scroll-view about the console-window is opened.
1189 */
1190void VBoxConsoleView::onViewOpened()
1191{
1192 /* Variable mIgnoreMainwndResize was initially "true" to ignore QT
1193 * initial resize event in case of auto-resize feature is on.
1194 * Currently, initial resize event is already processed, so we set
1195 * mIgnoreMainwndResize to "false" to process all further resize
1196 * events as user-initiated window resize events. */
1197 mIgnoreMainwndResize = false;
1198}
1199
1200//
1201// Protected Events
1202/////////////////////////////////////////////////////////////////////////////
1203
1204bool VBoxConsoleView::event (QEvent *e)
1205{
1206 if (mAttached)
1207 {
1208 switch (e->type())
1209 {
1210 case QEvent::FocusIn:
1211 {
1212 if (isRunning())
1213 focusEvent (true);
1214 break;
1215 }
1216 case QEvent::FocusOut:
1217 {
1218 if (isRunning())
1219 focusEvent (false);
1220 else
1221 {
1222 /* release the host key and all other pressed keys too even
1223 * when paused (otherwise, we will get stuck keys in the
1224 * guest when doing sendChangedKeyStates() on resume because
1225 * key presses were already recorded in mPressedKeys but key
1226 * releases will most likely not reach us but the new focus
1227 * window instead). */
1228 releaseAllPressedKeys (true /* aReleaseHostKey */);
1229 }
1230 break;
1231 }
1232
1233 case VBoxDefs::ResizeEventType:
1234 {
1235 bool oldIgnoreMainwndResize = mIgnoreMainwndResize;
1236 mIgnoreMainwndResize = true;
1237
1238 VBoxResizeEvent *re = (VBoxResizeEvent *) e;
1239 LogFlow (("VBoxDefs::ResizeEventType: %d x %d x %d bpp\n",
1240 re->width(), re->height(), re->bitsPerPixel()));
1241
1242 /* do frame buffer dependent resize */
1243#if defined (Q_WS_X11) && (QT_VERSION >= 0x040309) && (QT_VERSION < 0x040401)
1244 /* restoreOverrideCursor() is broken in Qt 4.4.0 if WA_PaintOnScreen
1245 * widgets are present. This is the case on linux with SDL. As
1246 * workaround we save/restore the arrow cursor manually. See
1247 * http://trolltech.com/developer/task-tracker/index_html?id=206165&method=entry
1248 * for details. */
1249 QCursor cursor;
1250 if (!mHideHostPointer)
1251 cursor = viewport()->cursor();
1252 else
1253 cursor = QCursor (Qt::BlankCursor);
1254 mFrameBuf->resizeEvent (re);
1255 viewport()->setCursor (cursor);
1256#else
1257 mFrameBuf->resizeEvent (re);
1258 if (!mHideHostPointer)
1259 viewport()->unsetCursor();
1260 else
1261 viewport()->setCursor (QCursor (Qt::BlankCursor));
1262#endif
1263
1264 /* This event appears in case of guest video was changed
1265 * for somehow even without video resolution change.
1266 * In this last case the host VM window will not be resized
1267 * according this event and the host mouse cursor which was
1268 * unset to default here will not be hidden in capture state.
1269 * So it is necessary to perform updateMouseClipping() for
1270 * the guest resize event if the mouse cursor was captured. */
1271 if (mMouseCaptured)
1272 updateMouseClipping();
1273
1274 /* apply maximum size restriction */
1275 setMaximumSize (sizeHint());
1276
1277 maybeRestrictMinimumSize();
1278
1279 /* resize the guest canvas */
1280 if (!mIgnoreFrameBufferResize)
1281 resize (re->width(), re->height());
1282 updateSliders();
1283 /* Let our toplevel widget calculate its sizeHint properly. */
1284#ifdef Q_WS_X11
1285 /* We use processEvents rather than sendPostedEvents & set the
1286 * time out value to max cause on X11 otherwise the layout
1287 * isn't calculated correctly. Dosn't find the bug in Qt, but
1288 * this could be triggered through the async nature of the X11
1289 * window event system. */
1290 QCoreApplication::processEvents (QEventLoop::AllEvents, INT_MAX);
1291#else /* Q_WS_X11 */
1292 QCoreApplication::sendPostedEvents (0, QEvent::LayoutRequest);
1293#endif /* Q_WS_X11 */
1294
1295 if (!mIgnoreFrameBufferResize)
1296 normalizeGeometry (true /* adjustPosition */);
1297
1298 /* report to the VM thread that we finished resizing */
1299 mConsole.GetDisplay().ResizeCompleted (0);
1300
1301 mIgnoreMainwndResize = oldIgnoreMainwndResize;
1302
1303 /* update geometry after entering fullscreen | seamless */
1304 if (mMainWnd->isTrueFullscreen() || mMainWnd->isTrueSeamless())
1305 updateGeometry();
1306
1307 /* make sure that all posted signals are processed */
1308 qApp->processEvents();
1309
1310 /* emit a signal about guest was resized */
1311 emit resizeHintDone();
1312
1313 /* We also recalculate the desktop geometry if this is determined
1314 * automatically. In fact, we only need this on the first resize,
1315 * but it is done every time to keep the code simpler. */
1316 calculateDesktopGeometry();
1317
1318 /* Enable frame-buffer resize watching. */
1319 if (mIgnoreFrameBufferResize)
1320 {
1321 mIgnoreFrameBufferResize = false;
1322 doResizeHint (mNormalSize);
1323 }
1324
1325 return true;
1326 }
1327
1328 /* See VBox[QImage|SDL]FrameBuffer::NotifyUpdate(). */
1329 case VBoxDefs::RepaintEventType:
1330 {
1331 VBoxRepaintEvent *re = (VBoxRepaintEvent *) e;
1332 viewport()->repaint (re->x() - contentsX(),
1333 re->y() - contentsY(),
1334 re->width(), re->height());
1335 /* mConsole.GetDisplay().UpdateCompleted(); - the event was acked already */
1336 return true;
1337 }
1338
1339 case VBoxDefs::SetRegionEventType:
1340 {
1341 VBoxSetRegionEvent *sre = (VBoxSetRegionEvent*) e;
1342 if (mMainWnd->isTrueSeamless() &&
1343 sre->region() != mLastVisibleRegion)
1344 {
1345 mLastVisibleRegion = sre->region();
1346 mMainWnd->setMask (sre->region());
1347 }
1348 else if (!mLastVisibleRegion.isEmpty() &&
1349 !mMainWnd->isTrueSeamless())
1350 mLastVisibleRegion = QRegion();
1351 return true;
1352 }
1353
1354 case VBoxDefs::MousePointerChangeEventType:
1355 {
1356 MousePointerChangeEvent *me = (MousePointerChangeEvent *) e;
1357 /* change cursor shape only when mouse integration is
1358 * supported (change mouse shape type event may arrive after
1359 * mouse capability change that disables integration */
1360 if (mMouseAbsolute)
1361 setPointerShape (me);
1362 else
1363 /* Note: actually we should still remember the requested
1364 * cursor shape. If we can't do that, at least remember
1365 * the requested visiblilty. */
1366 mHideHostPointer = !me->isVisible();
1367 return true;
1368 }
1369 case VBoxDefs::MouseCapabilityEventType:
1370 {
1371 MouseCapabilityEvent *me = (MouseCapabilityEvent *) e;
1372 if (mMouseAbsolute != me->supportsAbsolute())
1373 {
1374 mMouseAbsolute = me->supportsAbsolute();
1375 /* correct the mouse capture state and reset the cursor
1376 * to the default shape if necessary */
1377 if (mMouseAbsolute)
1378 {
1379 CMouse mouse = mConsole.GetMouse();
1380 mouse.PutMouseEventAbsolute (-1, -1, 0, 0);
1381 captureMouse (false, false);
1382 }
1383 else
1384 viewport()->unsetCursor();
1385 emitMouseStateChanged();
1386 vboxProblem().remindAboutMouseIntegration (mMouseAbsolute);
1387 }
1388 if (me->needsHostCursor())
1389 mMainWnd->setMouseIntegrationLocked (false);
1390 return true;
1391 }
1392
1393 case VBoxDefs::ModifierKeyChangeEventType:
1394 {
1395 ModifierKeyChangeEvent *me = (ModifierKeyChangeEvent* )e;
1396 if (me->numLock() != mNumLock)
1397 muNumLockAdaptionCnt = 2;
1398 if (me->capsLock() != mCapsLock)
1399 muCapsLockAdaptionCnt = 2;
1400 mNumLock = me->numLock();
1401 mCapsLock = me->capsLock();
1402 mScrollLock = me->scrollLock();
1403 return true;
1404 }
1405
1406 case VBoxDefs::MachineStateChangeEventType:
1407 {
1408 StateChangeEvent *me = (StateChangeEvent *) e;
1409 LogFlowFunc (("MachineStateChangeEventType: state=%d\n",
1410 me->machineState()));
1411 onStateChange (me->machineState());
1412 emit machineStateChanged (me->machineState());
1413 return true;
1414 }
1415
1416 case VBoxDefs::AdditionsStateChangeEventType:
1417 {
1418 GuestAdditionsEvent *ge = (GuestAdditionsEvent *) e;
1419 LogFlowFunc (("AdditionsStateChangeEventType\n"));
1420
1421 /* Always send a size hint if we are in fullscreen or seamless
1422 * when the graphics capability is enabled, in case the host
1423 * resolution has changed since the VM was last run. */
1424#if 0
1425 if (!mDoResize && !mGuestSupportsGraphics &&
1426 ge->supportsGraphics() &&
1427 (mMainWnd->isTrueSeamless() || mMainWnd->isTrueFullscreen()))
1428 mDoResize = true;
1429#endif
1430
1431 mGuestSupportsGraphics = ge->supportsGraphics();
1432
1433 maybeRestrictMinimumSize();
1434
1435#if 0
1436 /* This will only be acted upon if mDoResize is true. */
1437 doResizeHint();
1438#endif
1439
1440 emit additionsStateChanged (ge->additionVersion(),
1441 ge->additionActive(),
1442 ge->supportsSeamless(),
1443 ge->supportsGraphics());
1444 return true;
1445 }
1446
1447 case VBoxDefs::MediaDriveChangeEventType:
1448 {
1449 MediaDriveChangeEvent *mce = (MediaDriveChangeEvent *) e;
1450 LogFlowFunc (("MediaChangeEvent\n"));
1451
1452 emit mediaDriveChanged (mce->type());
1453 return true;
1454 }
1455
1456 case VBoxDefs::ActivateMenuEventType:
1457 {
1458 ActivateMenuEvent *ame = (ActivateMenuEvent *) e;
1459 ame->action()->trigger();
1460
1461 /*
1462 * The main window and its children can be destroyed at this
1463 * point (if, for example, the activated menu item closes the
1464 * main window). Detect this situation to prevent calls to
1465 * destroyed widgets.
1466 */
1467 QWidgetList list = QApplication::topLevelWidgets();
1468 bool destroyed = list.indexOf (mMainWnd) < 0;
1469 if (!destroyed && mMainWnd->statusBar())
1470 mMainWnd->statusBar()->clearMessage();
1471
1472 return true;
1473 }
1474
1475 case VBoxDefs::NetworkAdapterChangeEventType:
1476 {
1477 /* no specific adapter information stored in this
1478 * event is currently used */
1479 emit networkStateChange();
1480 return true;
1481 }
1482
1483 case VBoxDefs::USBCtlStateChangeEventType:
1484 {
1485 emit usbStateChange();
1486 return true;
1487 }
1488
1489 case VBoxDefs::USBDeviceStateChangeEventType:
1490 {
1491 USBDeviceStateChangeEvent *ue = (USBDeviceStateChangeEvent *)e;
1492
1493 bool success = ue->error().isNull();
1494
1495 if (!success)
1496 {
1497 if (ue->attached())
1498 vboxProblem().cannotAttachUSBDevice (
1499 mConsole,
1500 vboxGlobal().details (ue->device()), ue->error());
1501 else
1502 vboxProblem().cannotDetachUSBDevice (
1503 mConsole,
1504 vboxGlobal().details (ue->device()), ue->error());
1505 }
1506
1507 emit usbStateChange();
1508
1509 return true;
1510 }
1511
1512 case VBoxDefs::SharedFolderChangeEventType:
1513 {
1514 emit sharedFoldersChanged();
1515 return true;
1516 }
1517
1518 case VBoxDefs::RuntimeErrorEventType:
1519 {
1520 RuntimeErrorEvent *ee = (RuntimeErrorEvent *) e;
1521 vboxProblem().showRuntimeError (mConsole, ee->fatal(),
1522 ee->errorID(), ee->message());
1523 return true;
1524 }
1525
1526 case QEvent::KeyPress:
1527 case QEvent::KeyRelease:
1528 {
1529 QKeyEvent *ke = (QKeyEvent *) e;
1530
1531#ifdef Q_WS_PM
1532 /// @todo temporary solution to send Alt+Tab and friends to
1533 // the guest. The proper solution is to write a keyboard
1534 // driver that will steal these combos from the host (it's
1535 // impossible to do so using hooks on OS/2).
1536
1537 if (mIsHostkeyPressed)
1538 {
1539 bool pressed = e->type() == QEvent::KeyPress;
1540 CKeyboard keyboard = mConsole.GetKeyboard();
1541
1542 /* whether the host key is Shift so that it will modify
1543 * the hot key values? Note that we don't distinguish
1544 * between left and right shift here (too much hassle) */
1545 const bool kShift = (gs.hostKey() == VK_SHIFT ||
1546 gs.hostKey() == VK_LSHIFT) &&
1547 (ke->state() & Qt::ShiftModifier);
1548 /* define hot keys according to the Shift state */
1549 const int kAltTab = kShift ? Qt::Key_Exclam : Qt::Key_1;
1550 const int kAltShiftTab = kShift ? Qt::Key_At : Qt::Key_2;
1551 const int kCtrlEsc = kShift ? Qt::Key_AsciiTilde : Qt::Key_QuoteLeft;
1552
1553 /* Simulate Alt+Tab on Host+1 and Alt+Shift+Tab on Host+2 */
1554 if (ke->key() == kAltTab || ke->key() == kAltShiftTab)
1555 {
1556 if (pressed)
1557 {
1558 /* Send the Alt press to the guest */
1559 if (!(mPressedKeysCopy [0x38] & IsKeyPressed))
1560 {
1561 /* store the press in *Copy to have it automatically
1562 * released when the Host key is released */
1563 mPressedKeysCopy [0x38] |= IsKeyPressed;
1564 keyboard.PutScancode (0x38);
1565 }
1566
1567 /* Make sure Shift is pressed if it's Key_2 and released
1568 * if it's Key_1 */
1569 if (ke->key() == kAltTab &&
1570 (mPressedKeysCopy [0x2A] & IsKeyPressed))
1571 {
1572 mPressedKeysCopy [0x2A] &= ~IsKeyPressed;
1573 keyboard.PutScancode (0xAA);
1574 }
1575 else
1576 if (ke->key() == kAltShiftTab &&
1577 !(mPressedKeysCopy [0x2A] & IsKeyPressed))
1578 {
1579 mPressedKeysCopy [0x2A] |= IsKeyPressed;
1580 keyboard.PutScancode (0x2A);
1581 }
1582 }
1583
1584 keyboard.PutScancode (pressed ? 0x0F : 0x8F);
1585
1586 ke->accept();
1587 return true;
1588 }
1589
1590 /* Simulate Ctrl+Esc on Host+Tilde */
1591 if (ke->key() == kCtrlEsc)
1592 {
1593 /* Send the Ctrl press to the guest */
1594 if (pressed && !(mPressedKeysCopy [0x1d] & IsKeyPressed))
1595 {
1596 /* store the press in *Copy to have it automatically
1597 * released when the Host key is released */
1598 mPressedKeysCopy [0x1d] |= IsKeyPressed;
1599 keyboard.PutScancode (0x1d);
1600 }
1601
1602 keyboard.PutScancode (pressed ? 0x01 : 0x81);
1603
1604 ke->accept();
1605 return true;
1606 }
1607 }
1608
1609 /* fall through to normal processing */
1610
1611#endif /* Q_WS_PM */
1612
1613 if (mIsHostkeyPressed && e->type() == QEvent::KeyPress)
1614 {
1615 if (ke->key() >= Qt::Key_F1 && ke->key() <= Qt::Key_F12)
1616 {
1617 QVector <LONG> combo (6);
1618 combo [0] = 0x1d; /* Ctrl down */
1619 combo [1] = 0x38; /* Alt down */
1620 combo [4] = 0xb8; /* Alt up */
1621 combo [5] = 0x9d; /* Ctrl up */
1622 if (ke->key() >= Qt::Key_F1 && ke->key() <= Qt::Key_F10)
1623 {
1624 combo [2] = 0x3b + (ke->key() - Qt::Key_F1); /* F1-F10 down */
1625 combo [3] = 0xbb + (ke->key() - Qt::Key_F1); /* F1-F10 up */
1626 }
1627 /* some scan slice */
1628 else if (ke->key() >= Qt::Key_F11 && ke->key() <= Qt::Key_F12)
1629 {
1630 combo [2] = 0x57 + (ke->key() - Qt::Key_F11); /* F11-F12 down */
1631 combo [3] = 0xd7 + (ke->key() - Qt::Key_F11); /* F11-F12 up */
1632 }
1633 else
1634 Assert (0);
1635
1636 CKeyboard keyboard = mConsole.GetKeyboard();
1637 keyboard.PutScancodes (combo);
1638 }
1639 else if (ke->key() == Qt::Key_Home)
1640 {
1641 /* Activate the main menu */
1642 if (mMainWnd->isTrueSeamless() || mMainWnd->isTrueFullscreen())
1643 mMainWnd->popupMainMenu (mMouseCaptured);
1644 else
1645 {
1646 /* In Qt4 it is not enough to just set the focus to
1647 * menu-bar. So to get the menu-bar we have to send
1648 * Qt::Key_Alt press/release events directly. */
1649 QKeyEvent e1 (QEvent::KeyPress, Qt::Key_Alt,
1650 Qt::NoModifier);
1651 QKeyEvent e2 (QEvent::KeyRelease, Qt::Key_Alt,
1652 Qt::NoModifier);
1653 QApplication::sendEvent (mMainWnd->menuBar(), &e1);
1654 QApplication::sendEvent (mMainWnd->menuBar(), &e2);
1655 }
1656 }
1657 else
1658 {
1659 /* process hot keys not processed in keyEvent()
1660 * (as in case of non-alphanumeric keys) */
1661 processHotKey (QKeySequence (ke->key()),
1662 mMainWnd->menuBar()->actions());
1663 }
1664 }
1665 else if (!mIsHostkeyPressed && e->type() == QEvent::KeyRelease)
1666 {
1667 /* Show a possible warning on key release which seems to
1668 * be more expected by the end user */
1669
1670 if (isPaused())
1671 {
1672 /* if the reminder is disabled we pass the event to
1673 * Qt to enable normal keyboard functionality
1674 * (for example, menu access with Alt+Letter) */
1675 if (!vboxProblem().remindAboutPausedVMInput())
1676 break;
1677 }
1678 }
1679
1680 ke->accept();
1681 return true;
1682 }
1683
1684#ifdef Q_WS_MAC
1685 /* posted OnShowWindow */
1686 case VBoxDefs::ShowWindowEventType:
1687 {
1688 /*
1689 * Dunno what Qt3 thinks a window that has minimized to the dock
1690 * should be - it is not hidden, neither is it minimized. OTOH it is
1691 * marked shown and visible, but not activated. This latter isn't of
1692 * much help though, since at this point nothing is marked activated.
1693 * I might have overlooked something, but I'm buggered what if I know
1694 * what. So, I'll just always show & activate the stupid window to
1695 * make it get out of the dock when the user wishes to show a VM.
1696 */
1697 window()->show();
1698 window()->activateWindow();
1699 return true;
1700 }
1701#endif
1702#ifdef VBOX_WITH_VIDEOHWACCEL
1703 case VBoxDefs::VHWACommandProcessType:
1704 {
1705 VBoxVHWACommandProcessEvent *cmde = (VBoxVHWACommandProcessEvent *)e;
1706 mFrameBuf->doProcessVHWACommand(cmde->command());
1707 return true;
1708 }
1709#endif
1710 default:
1711 break;
1712 }
1713 }
1714
1715 return QAbstractScrollArea::event (e);
1716}
1717
1718bool VBoxConsoleView::eventFilter (QObject *watched, QEvent *e)
1719{
1720 if (mAttached && watched == viewport())
1721 {
1722 switch (e->type())
1723 {
1724 case QEvent::MouseMove:
1725 case QEvent::MouseButtonPress:
1726 case QEvent::MouseButtonDblClick:
1727 case QEvent::MouseButtonRelease:
1728 {
1729 QMouseEvent *me = (QMouseEvent *) e;
1730 if (mouseEvent (me->type(), me->pos(), me->globalPos(),
1731 me->buttons(), me->modifiers(),
1732 0, Qt::Horizontal))
1733 return true; /* stop further event handling */
1734 break;
1735 }
1736 case QEvent::Wheel:
1737 {
1738 QWheelEvent *we = (QWheelEvent *) e;
1739 if (mouseEvent (we->type(), we->pos(), we->globalPos(),
1740 we->buttons(), we->modifiers(),
1741 we->delta(), we->orientation()))
1742 return true; /* stop further event handling */
1743 break;
1744 }
1745#ifdef Q_WS_MAC
1746 case QEvent::Leave:
1747 {
1748 /* Enable mouse event compression if we leave the VM view. This
1749 is necessary for having smooth resizing of the VM/other
1750 windows. */
1751 setMouseCoalescingEnabled (true);
1752 break;
1753 }
1754 case QEvent::Enter:
1755 {
1756 /* Disable mouse event compression if we enter the VM view. So
1757 all mouse events are registered in the VM. Only do this if
1758 the keyboard/mouse is grabbed (this is when we have a valid
1759 event handler). */
1760 setMouseCoalescingEnabled (false);
1761 break;
1762 }
1763#endif /* Q_WS_MAC */
1764 case QEvent::Resize:
1765 {
1766 if (mMouseCaptured)
1767 updateMouseClipping();
1768 break;
1769 }
1770 default:
1771 break;
1772 }
1773 }
1774 else if (watched == mMainWnd)
1775 {
1776 switch (e->type())
1777 {
1778#if defined (Q_WS_WIN32)
1779#if defined (VBOX_GUI_USE_DDRAW)
1780 case QEvent::Move:
1781 {
1782 /*
1783 * notification from our parent that it has moved. We need this
1784 * in order to possibly adjust the direct screen blitting.
1785 */
1786 if (mFrameBuf)
1787 mFrameBuf->moveEvent ((QMoveEvent *) e);
1788 break;
1789 }
1790#endif
1791 /*
1792 * install/uninstall low-level kbd hook on every
1793 * activation/deactivation to:
1794 * a) avoid excess hook calls when we're not active and
1795 * b) be always in front of any other possible hooks
1796 */
1797 case QEvent::WindowActivate:
1798 {
1799 gKbdHook = SetWindowsHookEx (WH_KEYBOARD_LL, lowLevelKeyboardProc,
1800 GetModuleHandle (NULL), 0);
1801 AssertMsg (gKbdHook, ("SetWindowsHookEx(): err=%d", GetLastError()));
1802 break;
1803 }
1804 case QEvent::WindowDeactivate:
1805 {
1806 if (gKbdHook)
1807 {
1808 UnhookWindowsHookEx (gKbdHook);
1809 gKbdHook = NULL;
1810 }
1811 break;
1812 }
1813#endif /* defined (Q_WS_WIN32) */
1814#if defined (Q_WS_MAC)
1815 /*
1816 * Install/remove the keyboard event handler.
1817 */
1818 case QEvent::WindowActivate:
1819 darwinGrabKeyboardEvents (true);
1820 break;
1821 case QEvent::WindowDeactivate:
1822 darwinGrabKeyboardEvents (false);
1823 break;
1824#endif /* defined (Q_WS_MAC) */
1825 case QEvent::Resize:
1826 {
1827 /* Set the "guest needs to resize" hint. This hint is acted upon
1828 * when (and only when) the autoresize property is "true". */
1829 mDoResize = mGuestSupportsGraphics || mMainWnd->isTrueFullscreen();
1830 if (!mIgnoreMainwndResize &&
1831 mGuestSupportsGraphics && mAutoresizeGuest)
1832 QTimer::singleShot (300, this, SLOT (doResizeHint()));
1833 break;
1834 }
1835 case QEvent::WindowStateChange:
1836 {
1837 /* During minimizing and state restoring mMainWnd gets the focus
1838 * which belongs to console view window, so returning it properly. */
1839 QWindowStateChangeEvent *ev = static_cast <QWindowStateChangeEvent*> (e);
1840 if (ev->oldState() & Qt::WindowMinimized)
1841 {
1842 if (QApplication::focusWidget())
1843 {
1844 QApplication::focusWidget()->clearFocus();
1845 qApp->processEvents();
1846 }
1847 QTimer::singleShot (0, this, SLOT (setFocus()));
1848 }
1849 break;
1850 }
1851
1852 default:
1853 break;
1854 }
1855 }
1856 else if (watched == mMainWnd->menuBar())
1857 {
1858 /*
1859 * sometimes when we press ESC in the menu it brings the
1860 * focus away (Qt bug?) causing no widget to have a focus,
1861 * or holds the focus itself, instead of returning the focus
1862 * to the console window. here we fix this.
1863 */
1864 switch (e->type())
1865 {
1866 case QEvent::FocusOut:
1867 {
1868 if (qApp->focusWidget() == 0)
1869 setFocus();
1870 break;
1871 }
1872 case QEvent::KeyPress:
1873 {
1874 QKeyEvent *ke = (QKeyEvent *) e;
1875 if (ke->key() == Qt::Key_Escape && (ke->modifiers() == Qt::NoModifier))
1876 if (mMainWnd->menuBar()->hasFocus())
1877 setFocus();
1878 break;
1879 }
1880 default:
1881 break;
1882 }
1883 }
1884
1885 return QAbstractScrollArea::eventFilter (watched, e);
1886}
1887
1888#if defined(Q_WS_WIN32)
1889
1890/**
1891 * Low-level keyboard event handler,
1892 * @return
1893 * true to indicate that the message is processed and false otherwise
1894 */
1895bool VBoxConsoleView::winLowKeyboardEvent (UINT msg, const KBDLLHOOKSTRUCT &event)
1896{
1897#if 0
1898 LogFlow (("### vkCode=%08X, scanCode=%08X, flags=%08X, dwExtraInfo=%08X (mKbdCaptured=%d)\n",
1899 event.vkCode, event.scanCode, event.flags, event.dwExtraInfo, mKbdCaptured));
1900 char buf [256];
1901 sprintf (buf, "### vkCode=%08X, scanCode=%08X, flags=%08X, dwExtraInfo=%08X",
1902 event.vkCode, event.scanCode, event.flags, event.dwExtraInfo);
1903 mMainWnd->statusBar()->message (buf);
1904#endif
1905
1906 /* Sometimes it happens that Win inserts additional events on some key
1907 * press/release. For example, it prepends ALT_GR in German layout with
1908 * the VK_LCONTROL vkey with curious 0x21D scan code (seems to be necessary
1909 * to specially treat ALT_GR to enter additional chars to regular apps).
1910 * These events are definitely unwanted in VM, so filter them out. */
1911 if (hasFocus() && (event.scanCode & ~0xFF))
1912 return true;
1913
1914 if (!mKbdCaptured)
1915 return false;
1916
1917 /* it's possible that a key has been pressed while the keyboard was not
1918 * captured, but is being released under the capture. Detect this situation
1919 * and return false to let Windows process the message normally and update
1920 * its key state table (to avoid the stuck key effect). */
1921 uint8_t what_pressed = (event.flags & 0x01) && (event.vkCode != VK_RSHIFT)
1922 ? IsExtKeyPressed
1923 : IsKeyPressed;
1924 if ((event.flags & 0x80) /* released */ &&
1925 ((event.vkCode == gs.hostKey() && !hostkey_in_capture) ||
1926 (mPressedKeys [event.scanCode] & (IsKbdCaptured | what_pressed)) == what_pressed))
1927 return false;
1928
1929 MSG message;
1930 message.hwnd = winId();
1931 message.message = msg;
1932 message.wParam = event.vkCode;
1933 message.lParam =
1934 1 |
1935 (event.scanCode & 0xFF) << 16 |
1936 (event.flags & 0xFF) << 24;
1937
1938 /* Windows sets here the extended bit when the Right Shift key is pressed,
1939 * which is totally wrong. Undo it. */
1940 if (event.vkCode == VK_RSHIFT)
1941 message.lParam &= ~0x1000000;
1942
1943 /* we suppose here that this hook is always called on the main GUI thread */
1944 long dummyResult;
1945 return winEvent (&message, &dummyResult);
1946}
1947
1948/**
1949 * Get Win32 messages before they are passed to Qt. This allows us to get
1950 * the keyboard events directly and bypass the harmful Qt translation. A
1951 * return value of @c true indicates to Qt that the event has been handled.
1952 */
1953bool VBoxConsoleView::winEvent (MSG *aMsg, long* /* aResult */)
1954{
1955 if (!mAttached || ! (
1956 aMsg->message == WM_KEYDOWN || aMsg->message == WM_SYSKEYDOWN ||
1957 aMsg->message == WM_KEYUP || aMsg->message == WM_SYSKEYUP
1958 ))
1959 return false;
1960
1961 /* check for the special flag possibly set at the end of this function */
1962 if (aMsg->lParam & (0x1 << 25))
1963 {
1964 aMsg->lParam &= ~(0x1 << 25);
1965 return false;
1966 }
1967
1968#if 0
1969 char buf [256];
1970 sprintf (buf, "WM_%04X: vk=%04X rep=%05d scan=%02X ext=%01d rzv=%01X ctx=%01d prev=%01d tran=%01d",
1971 aMsg->message, aMsg->wParam,
1972 (aMsg->lParam & 0xFFFF),
1973 ((aMsg->lParam >> 16) & 0xFF),
1974 ((aMsg->lParam >> 24) & 0x1),
1975 ((aMsg->lParam >> 25) & 0xF),
1976 ((aMsg->lParam >> 29) & 0x1),
1977 ((aMsg->lParam >> 30) & 0x1),
1978 ((aMsg->lParam >> 31) & 0x1));
1979 mMainWnd->statusBar()->message (buf);
1980 LogFlow (("%s\n", buf));
1981#endif
1982
1983 int scan = (aMsg->lParam >> 16) & 0x7F;
1984 /* scancodes 0x80 and 0x00 are ignored */
1985 if (!scan)
1986 return true;
1987
1988 int vkey = aMsg->wParam;
1989
1990 /* When one of the SHIFT keys is held and one of the cursor movement
1991 * keys is pressed, Windows duplicates SHIFT press/release messages,
1992 * but with the virtual key code set to 0xFF. These virtual keys are also
1993 * sent in some other situations (Pause, PrtScn, etc.). Ignore such
1994 * messages. */
1995 if (vkey == 0xFF)
1996 return true;
1997
1998 int flags = 0;
1999 if (aMsg->lParam & 0x1000000)
2000 flags |= KeyExtended;
2001 if (!(aMsg->lParam & 0x80000000))
2002 flags |= KeyPressed;
2003
2004 switch (vkey)
2005 {
2006 case VK_SHIFT:
2007 case VK_CONTROL:
2008 case VK_MENU:
2009 {
2010 /* overcome stupid Win32 modifier key generalization */
2011 int keyscan = scan;
2012 if (flags & KeyExtended)
2013 keyscan |= 0xE000;
2014 switch (keyscan)
2015 {
2016 case 0x002A: vkey = VK_LSHIFT; break;
2017 case 0x0036: vkey = VK_RSHIFT; break;
2018 case 0x001D: vkey = VK_LCONTROL; break;
2019 case 0xE01D: vkey = VK_RCONTROL; break;
2020 case 0x0038: vkey = VK_LMENU; break;
2021 case 0xE038: vkey = VK_RMENU; break;
2022 }
2023 break;
2024 }
2025 case VK_NUMLOCK:
2026 /* Win32 sets the extended bit for the NumLock key. Reset it. */
2027 flags &= ~KeyExtended;
2028 break;
2029 case VK_SNAPSHOT:
2030 flags |= KeyPrint;
2031 break;
2032 case VK_PAUSE:
2033 flags |= KeyPause;
2034 break;
2035 }
2036
2037 bool result = keyEvent (vkey, scan, flags);
2038 if (!result && mKbdCaptured)
2039 {
2040 /* keyEvent() returned that it didn't process the message, but since the
2041 * keyboard is captured, we don't want to pass it to Windows. We just want
2042 * to let Qt process the message (to handle non-alphanumeric <HOST>+key
2043 * shortcuts for example). So send it direcltly to the window with the
2044 * special flag in the reserved area of lParam (to avoid recursion). */
2045 ::SendMessage (aMsg->hwnd, aMsg->message,
2046 aMsg->wParam, aMsg->lParam | (0x1 << 25));
2047 return true;
2048 }
2049
2050 /* These special keys have to be handled by Windows as well to update the
2051 * internal modifier state and to enable/disable the keyboard LED */
2052 if (vkey == VK_NUMLOCK || vkey == VK_CAPITAL)
2053 return false;
2054
2055 return result;
2056}
2057
2058#elif defined (Q_WS_PM)
2059
2060/**
2061 * Get PM messages before they are passed to Qt. This allows us to get
2062 * the keyboard events directly and bypass the harmful Qt translation. A
2063 * return value of @c true indicates to Qt that the event has been handled.
2064 */
2065bool VBoxConsoleView::pmEvent (QMSG *aMsg)
2066{
2067 if (!mAttached)
2068 return false;
2069
2070 if (aMsg->msg == UM_PREACCEL_CHAR)
2071 {
2072 /* we are inside the input hook */
2073
2074 /* let the message go through the normal system pipeline */
2075 if (!mKbdCaptured)
2076 return false;
2077 }
2078
2079 if (aMsg->msg != WM_CHAR &&
2080 aMsg->msg != UM_PREACCEL_CHAR)
2081 return false;
2082
2083 /* check for the special flag possibly set at the end of this function */
2084 if (SHORT2FROMMP (aMsg->mp2) & 0x8000)
2085 {
2086 aMsg->mp2 = MPFROM2SHORT (SHORT1FROMMP (aMsg->mp2),
2087 SHORT2FROMMP (aMsg->mp2) & ~0x8000);
2088 return false;
2089 }
2090
2091#if 0
2092 {
2093 char buf [256];
2094 sprintf (buf, "*** %s: f=%04X rep=%03d scan=%02X ch=%04X vk=%04X",
2095 (aMsg->msg == WM_CHAR ? "WM_CHAR" : "UM_PREACCEL_CHAR"),
2096 SHORT1FROMMP (aMsg->mp1), CHAR3FROMMP (aMsg->mp1),
2097 CHAR4FROMMP (aMsg->mp1), SHORT1FROMMP (aMsg->mp2),
2098 SHORT2FROMMP (aMsg->mp2));
2099 mMainWnd->statusBar()->message (buf);
2100 LogFlow (("%s\n", buf));
2101 }
2102#endif
2103
2104 USHORT ch = SHORT1FROMMP (aMsg->mp2);
2105 USHORT f = SHORT1FROMMP (aMsg->mp1);
2106
2107 int scan = (unsigned int) CHAR4FROMMP (aMsg->mp1);
2108 if (!scan || scan > 0x7F)
2109 return true;
2110
2111 int vkey = QIHotKeyEdit::virtualKey (aMsg);
2112
2113 int flags = 0;
2114
2115 if ((ch & 0xFF) == 0xE0)
2116 {
2117 flags |= KeyExtended;
2118 scan = ch >> 8;
2119 }
2120 else if (scan == 0x5C && (ch & 0xFF) == '/')
2121 {
2122 /* this is the '/' key on the keypad */
2123 scan = 0x35;
2124 flags |= KeyExtended;
2125 }
2126 else
2127 {
2128 /* For some keys, the scan code passed in QMSG is a pseudo scan
2129 * code. We replace it with a real hardware scan code, according to
2130 * http://www.computer-engineering.org/ps2keyboard/scancodes1.html.
2131 * Also detect Pause and PrtScn and set flags. */
2132 switch (vkey)
2133 {
2134 case VK_ENTER: scan = 0x1C; flags |= KeyExtended; break;
2135 case VK_CTRL: scan = 0x1D; flags |= KeyExtended; break;
2136 case VK_ALTGRAF: scan = 0x38; flags |= KeyExtended; break;
2137 case VK_LWIN: scan = 0x5B; flags |= KeyExtended; break;
2138 case VK_RWIN: scan = 0x5C; flags |= KeyExtended; break;
2139 case VK_WINMENU: scan = 0x5D; flags |= KeyExtended; break;
2140 case VK_FORWARD: scan = 0x69; flags |= KeyExtended; break;
2141 case VK_BACKWARD: scan = 0x6A; flags |= KeyExtended; break;
2142#if 0
2143 /// @todo this would send 0xE0 0x46 0xE0 0xC6. It's not fully
2144 // clear what is more correct
2145 case VK_BREAK: scan = 0x46; flags |= KeyExtended; break;
2146#else
2147 case VK_BREAK: scan = 0; flags |= KeyPause; break;
2148#endif
2149 case VK_PAUSE: scan = 0; flags |= KeyPause; break;
2150 case VK_PRINTSCRN: scan = 0; flags |= KeyPrint; break;
2151 default:;
2152 }
2153 }
2154
2155 if (!(f & KC_KEYUP))
2156 flags |= KeyPressed;
2157
2158 bool result = keyEvent (vkey, scan, flags);
2159 if (!result && mKbdCaptured)
2160 {
2161 /* keyEvent() returned that it didn't process the message, but since the
2162 * keyboard is captured, we don't want to pass it to PM. We just want
2163 * to let Qt process the message (to handle non-alphanumeric <HOST>+key
2164 * shortcuts for example). So send it direcltly to the window with the
2165 * special flag in the reserved area of lParam (to avoid recursion). */
2166 ::WinSendMsg (aMsg->hwnd, WM_CHAR,
2167 aMsg->mp1,
2168 MPFROM2SHORT (SHORT1FROMMP (aMsg->mp2),
2169 SHORT2FROMMP (aMsg->mp2) | 0x8000));
2170 return true;
2171 }
2172 return result;
2173}
2174
2175#elif defined(Q_WS_X11)
2176
2177/**
2178 * This function is a "predicate" for XCheckIfEvent(). It will check
2179 * the XEvent passed to it to see if it is a keypress event matching
2180 * the keyrelease event in @a pvArg.
2181 * @returns True if the event matches, False otherwise
2182 * @param pEvent the event to compare, taken from the event queue
2183 * @param pvArg the keyrelease event we would like to compare against
2184 */
2185static Bool VBoxConsoleViewCompEvent(Display *, XEvent *pEvent,
2186 XPointer pvArg)
2187{
2188 XEvent *pKeyEvent = (XEvent *) pvArg;
2189 if ((pEvent->type == XKeyPress) &&
2190 (pEvent->xkey.keycode == pKeyEvent->xkey.keycode))
2191 return True;
2192 else
2193 return False;
2194}
2195
2196/**
2197 * This routine gets X11 events before they are processed by Qt. This is
2198 * used for our platform specific keyboard implementation. A return value
2199 * of TRUE indicates that the event has been processed by us.
2200 */
2201bool VBoxConsoleView::x11Event (XEvent *event)
2202{
2203 switch (event->type)
2204 {
2205 /* We have to handle XFocusOut right here as this event is not passed
2206 * to VBoxConsoleView::event(). Handling this event is important for
2207 * releasing the keyboard before the screen saver gets active. */
2208 case XFocusOut:
2209 case XFocusIn:
2210 if (isRunning())
2211 focusEvent (event->type == XFocusIn);
2212 return false;
2213 case XKeyPress:
2214 case XKeyRelease:
2215 if (mAttached)
2216 break;
2217 /* else fall through */
2218 default:
2219 return false; /* pass the event to Qt */
2220 }
2221
2222 /* Translate the keycode to a PC scan code. */
2223 unsigned scan = handleXKeyEvent (event);
2224
2225#if 0
2226 char buf [256];
2227 sprintf (buf, "pr=%d kc=%08X st=%08X extended=%s scan=%04X",
2228 event->type == XKeyPress ? 1 : 0, event->xkey.keycode,
2229 event->xkey.state, scan >> 8 ? "true" : "false", scan & 0x7F);
2230 mMainWnd->statusBar()->message (buf);
2231 LogFlow (("### %s\n", buf));
2232#endif
2233
2234 // scancodes 0x00 (no valid translation) and 0x80 are ignored
2235 if (!scan & 0x7F)
2236 return true;
2237
2238 /* Fix for http://www.virtualbox.org/ticket/1296:
2239 * when X11 sends events for repeated keys, it always inserts an
2240 * XKeyRelease before the XKeyPress. */
2241 XEvent returnEvent;
2242 if ((event->type == XKeyRelease) &&
2243 (XCheckIfEvent(event->xkey.display, &returnEvent,
2244 VBoxConsoleViewCompEvent, (XPointer) event) == True)) {
2245 XPutBackEvent(event->xkey.display, &returnEvent);
2246 /* Discard it, don't pass it to Qt. */
2247 return true;
2248 }
2249
2250 KeySym ks = ::XKeycodeToKeysym (event->xkey.display, event->xkey.keycode, 0);
2251
2252 int flags = 0;
2253 if (scan >> 8)
2254 flags |= KeyExtended;
2255 if (event->type == XKeyPress)
2256 flags |= KeyPressed;
2257
2258 /* Remove the extended flag */
2259 scan &= 0x7F;
2260
2261 switch (ks)
2262 {
2263 case XK_Print:
2264 flags |= KeyPrint;
2265 break;
2266 case XK_Pause:
2267 flags |= KeyPause;
2268 break;
2269 }
2270
2271 return keyEvent (ks, scan, flags);
2272}
2273
2274#elif defined (Q_WS_MAC)
2275
2276/**
2277 * Invoked by VBoxConsoleView::darwinEventHandlerProc / VBoxConsoleView::macEventFilter when
2278 * it receives a raw keyboard event.
2279 *
2280 * @param pvCocoaEvent The Cocoa keyboard event. Can be NULL in some configs.
2281 * @param inEvent The keyboard event.
2282 *
2283 * @return true if the key was processed, false if it wasn't processed and should be passed on.
2284 */
2285bool VBoxConsoleView::darwinKeyboardEvent (const void *pvCocoaEvent, EventRef inEvent)
2286{
2287 bool ret = false;
2288 UInt32 EventKind = ::GetEventKind (inEvent);
2289 if (EventKind != kEventRawKeyModifiersChanged)
2290 {
2291 /* convert keycode to set 1 scan code. */
2292 UInt32 keyCode = ~0U;
2293 ::GetEventParameter (inEvent, kEventParamKeyCode, typeUInt32, NULL, sizeof (keyCode), NULL, &keyCode);
2294 unsigned scanCode = ::DarwinKeycodeToSet1Scancode (keyCode);
2295 if (scanCode)
2296 {
2297 /* calc flags. */
2298 int flags = 0;
2299 if (EventKind != kEventRawKeyUp)
2300 flags |= KeyPressed;
2301 if (scanCode & VBOXKEY_EXTENDED)
2302 flags |= KeyExtended;
2303 /** @todo KeyPause, KeyPrint. */
2304 scanCode &= VBOXKEY_SCANCODE_MASK;
2305
2306 /* get the unicode string (if present). */
2307 AssertCompileSize (wchar_t, 2);
2308 AssertCompileSize (UniChar, 2);
2309 ByteCount cbWritten = 0;
2310 wchar_t ucs[8];
2311 if (::GetEventParameter (inEvent, kEventParamKeyUnicodes, typeUnicodeText, NULL,
2312 sizeof (ucs), &cbWritten, &ucs[0]) != 0)
2313 cbWritten = 0;
2314 ucs[cbWritten / sizeof(wchar_t)] = 0; /* The api doesn't terminate it. */
2315
2316 ret = keyEvent (keyCode, scanCode, flags, ucs[0] ? ucs : NULL);
2317 }
2318 }
2319 else
2320 {
2321 /* May contain multiple modifier changes, kind of annoying. */
2322 UInt32 newMask = 0;
2323 ::GetEventParameter (inEvent, kEventParamKeyModifiers, typeUInt32, NULL,
2324 sizeof (newMask), NULL, &newMask);
2325 newMask = ::DarwinAdjustModifierMask (newMask, pvCocoaEvent);
2326 UInt32 changed = newMask ^ mDarwinKeyModifiers;
2327 if (changed)
2328 {
2329 for (UInt32 bit = 0; bit < 32; bit++)
2330 {
2331 if (!(changed & (1 << bit)))
2332 continue;
2333 unsigned scanCode = ::DarwinModifierMaskToSet1Scancode (1 << bit);
2334 if (!scanCode)
2335 continue;
2336 unsigned keyCode = ::DarwinModifierMaskToDarwinKeycode (1 << bit);
2337 Assert (keyCode);
2338
2339 if (!(scanCode & VBOXKEY_LOCK))
2340 {
2341 unsigned flags = (newMask & (1 << bit)) ? KeyPressed : 0;
2342 if (scanCode & VBOXKEY_EXTENDED)
2343 flags |= KeyExtended;
2344 scanCode &= VBOXKEY_SCANCODE_MASK;
2345 ret |= keyEvent (keyCode, scanCode & 0xff, flags);
2346 }
2347 else
2348 {
2349 unsigned flags = 0;
2350 if (scanCode & VBOXKEY_EXTENDED)
2351 flags |= KeyExtended;
2352 scanCode &= VBOXKEY_SCANCODE_MASK;
2353 keyEvent (keyCode, scanCode, flags | KeyPressed);
2354 keyEvent (keyCode, scanCode, flags);
2355 }
2356 }
2357 }
2358
2359 mDarwinKeyModifiers = newMask;
2360
2361 /* Always return true here because we'll otherwise getting a Qt event
2362 we don't want and that will only cause the Pause warning to pop up. */
2363 ret = true;
2364 }
2365
2366 return ret;
2367}
2368
2369
2370/**
2371 * Installs or removes the keyboard event handler.
2372 *
2373 * @param fGrab True if we're to grab the events, false if we're not to.
2374 */
2375void VBoxConsoleView::darwinGrabKeyboardEvents (bool fGrab)
2376{
2377 mKeyboardGrabbed = fGrab;
2378 if (fGrab)
2379 {
2380 /* Disable mouse and keyboard event compression/delaying to make sure
2381 we *really* get all of the events. */
2382 ::CGSetLocalEventsSuppressionInterval (0.0);
2383 setMouseCoalescingEnabled (false);
2384
2385 /* Register the event callback/hook and grab the keyboard. */
2386# ifdef QT_MAC_USE_COCOA
2387 ::VBoxCocoaApplication_setCallback (UINT32_MAX, /** @todo fix mask */
2388 VBoxConsoleView::darwinEventHandlerProc, this);
2389
2390# elif !defined (VBOX_WITH_HACKED_QT)
2391 EventTypeSpec eventTypes[6];
2392 eventTypes[0].eventClass = kEventClassKeyboard;
2393 eventTypes[0].eventKind = kEventRawKeyDown;
2394 eventTypes[1].eventClass = kEventClassKeyboard;
2395 eventTypes[1].eventKind = kEventRawKeyUp;
2396 eventTypes[2].eventClass = kEventClassKeyboard;
2397 eventTypes[2].eventKind = kEventRawKeyRepeat;
2398 eventTypes[3].eventClass = kEventClassKeyboard;
2399 eventTypes[3].eventKind = kEventRawKeyModifiersChanged;
2400 /* For ignorning Command-H and Command-Q which aren't affected by the
2401 * global hotkey stuff (doesn't work well): */
2402 eventTypes[4].eventClass = kEventClassCommand;
2403 eventTypes[4].eventKind = kEventCommandProcess;
2404 eventTypes[5].eventClass = kEventClassCommand;
2405 eventTypes[5].eventKind = kEventCommandUpdateStatus;
2406
2407 EventHandlerUPP eventHandler = ::NewEventHandlerUPP (VBoxConsoleView::darwinEventHandlerProc);
2408
2409 mDarwinEventHandlerRef = NULL;
2410 ::InstallApplicationEventHandler (eventHandler, RT_ELEMENTS (eventTypes), &eventTypes[0],
2411 this, &mDarwinEventHandlerRef);
2412 ::DisposeEventHandlerUPP (eventHandler);
2413
2414# else /* VBOX_WITH_HACKED_QT */
2415 ((QIApplication *)qApp)->setEventFilter (VBoxConsoleView::macEventFilter, this);
2416# endif /* VBOX_WITH_HACKED_QT */
2417
2418 ::DarwinGrabKeyboard (false);
2419 }
2420 else
2421 {
2422 ::DarwinReleaseKeyboard();
2423# ifdef QT_MAC_USE_COCOA
2424 ::VBoxCocoaApplication_unsetCallback (UINT32_MAX, /** @todo fix mask */
2425 VBoxConsoleView::darwinEventHandlerProc, this);
2426# elif !defined(VBOX_WITH_HACKED_QT)
2427 if (mDarwinEventHandlerRef)
2428 {
2429 ::RemoveEventHandler (mDarwinEventHandlerRef);
2430 mDarwinEventHandlerRef = NULL;
2431 }
2432# else /* VBOX_WITH_HACKED_QT */
2433 ((QIApplication *)qApp)->setEventFilter (NULL, NULL);
2434# endif /* VBOX_WITH_HACKED_QT */
2435 }
2436}
2437
2438#endif // defined (Q_WS_MAC)
2439
2440//
2441// Private members
2442/////////////////////////////////////////////////////////////////////////////
2443
2444/**
2445 * Called on every focus change and also to forcibly capture/uncapture the
2446 * input in situations similar to gaining or losing focus.
2447 *
2448 * @param aHasFocus true if the window got focus and false otherwise.
2449 * @param aReleaseHostKey true to release the host key (used only when
2450 * @a aHasFocus is false.
2451 */
2452void VBoxConsoleView::focusEvent (bool aHasFocus,
2453 bool aReleaseHostKey /* = true */)
2454{
2455 if (aHasFocus)
2456 {
2457#ifdef RT_OS_WINDOWS
2458 if ( !mDisableAutoCapture && gs.autoCapture()
2459 && GetAncestor (winId(), GA_ROOT) == GetForegroundWindow())
2460#else
2461 if (!mDisableAutoCapture && gs.autoCapture())
2462#endif /* RT_OS_WINDOWS */
2463 {
2464 captureKbd (true);
2465/// @todo (dmik)
2466// the below is for the mouse auto-capture. disabled for now. in order to
2467// properly support it, we need to know when *all* mouse buttons are
2468// released after we got focus, and grab the mouse only after then.
2469// btw, the similar would be good the for keyboard auto-capture, too.
2470// if (!(mMouseAbsolute && mMouseIntegration))
2471// captureMouse (true);
2472 }
2473
2474 /* reset the single-time disable capture flag */
2475 if (mDisableAutoCapture)
2476 mDisableAutoCapture = false;
2477 }
2478 else
2479 {
2480 captureMouse (false);
2481 captureKbd (false, false);
2482 releaseAllPressedKeys (aReleaseHostKey);
2483 }
2484}
2485
2486/**
2487 * Synchronize the views of the host and the guest to the modifier keys.
2488 * This function will add up to 6 additional keycodes to codes.
2489 *
2490 * @param codes pointer to keycodes which are sent to the keyboard
2491 * @param count pointer to the keycodes counter
2492 */
2493void VBoxConsoleView::fixModifierState (LONG *codes, uint *count)
2494{
2495#if defined(Q_WS_X11)
2496
2497 Window wDummy1, wDummy2;
2498 int iDummy3, iDummy4, iDummy5, iDummy6;
2499 unsigned uMask;
2500 unsigned uKeyMaskNum = 0, uKeyMaskCaps = 0, uKeyMaskScroll = 0;
2501
2502 uKeyMaskCaps = LockMask;
2503 XModifierKeymap* map = XGetModifierMapping(QX11Info::display());
2504 KeyCode keyCodeNum = XKeysymToKeycode(QX11Info::display(), XK_Num_Lock);
2505 KeyCode keyCodeScroll = XKeysymToKeycode(QX11Info::display(), XK_Scroll_Lock);
2506
2507 for (int i = 0; i < 8; i++)
2508 {
2509 if ( keyCodeNum != NoSymbol
2510 && map->modifiermap[map->max_keypermod * i] == keyCodeNum)
2511 uKeyMaskNum = 1 << i;
2512 else if ( keyCodeScroll != NoSymbol
2513 && map->modifiermap[map->max_keypermod * i] == keyCodeScroll)
2514 uKeyMaskScroll = 1 << i;
2515 }
2516 XQueryPointer(QX11Info::display(), DefaultRootWindow(QX11Info::display()), &wDummy1, &wDummy2,
2517 &iDummy3, &iDummy4, &iDummy5, &iDummy6, &uMask);
2518 XFreeModifiermap(map);
2519
2520 if (muNumLockAdaptionCnt && (mNumLock ^ !!(uMask & uKeyMaskNum)))
2521 {
2522 muNumLockAdaptionCnt--;
2523 codes[(*count)++] = 0x45;
2524 codes[(*count)++] = 0x45 | 0x80;
2525 }
2526 if (muCapsLockAdaptionCnt && (mCapsLock ^ !!(uMask & uKeyMaskCaps)))
2527 {
2528 muCapsLockAdaptionCnt--;
2529 codes[(*count)++] = 0x3a;
2530 codes[(*count)++] = 0x3a | 0x80;
2531 }
2532
2533#elif defined(Q_WS_WIN32)
2534
2535 if (muNumLockAdaptionCnt && (mNumLock ^ !!(GetKeyState(VK_NUMLOCK))))
2536 {
2537 muNumLockAdaptionCnt--;
2538 codes[(*count)++] = 0x45;
2539 codes[(*count)++] = 0x45 | 0x80;
2540 }
2541 if (muCapsLockAdaptionCnt && (mCapsLock ^ !!(GetKeyState(VK_CAPITAL))))
2542 {
2543 muCapsLockAdaptionCnt--;
2544 codes[(*count)++] = 0x3a;
2545 codes[(*count)++] = 0x3a | 0x80;
2546 }
2547
2548#elif defined (Q_WS_MAC)
2549
2550 /* if (muNumLockAdaptionCnt) ... - NumLock isn't implemented by Mac OS X so ignore it. */
2551 if (muCapsLockAdaptionCnt && (mCapsLock ^ !!(::GetCurrentEventKeyModifiers() & alphaLock)))
2552 {
2553 muCapsLockAdaptionCnt--;
2554 codes[(*count)++] = 0x3a;
2555 codes[(*count)++] = 0x3a | 0x80;
2556 }
2557
2558#else
2559
2560//#warning Adapt VBoxConsoleView::fixModifierState
2561
2562#endif
2563
2564
2565}
2566
2567/**
2568 * Called on enter/exit seamless/fullscreen mode.
2569 */
2570void VBoxConsoleView::toggleFSMode (const QSize &aSize)
2571{
2572 if ((mGuestSupportsGraphics && mAutoresizeGuest) ||
2573 mMainWnd->isTrueSeamless() ||
2574 mMainWnd->isTrueFullscreen())
2575 {
2576 QSize newSize;
2577 if (aSize.isValid())
2578 {
2579 mNormalSize = aSize;
2580 newSize = maximumSize();
2581 }
2582 else
2583 newSize = mNormalSize;
2584 doResizeHint (newSize);
2585 }
2586
2587 /* Reactivate the console window to preserve the focus position.
2588 * Else focus will move to the mini-tool-bar. */
2589 activateWindow();
2590}
2591
2592/**
2593 * Get the current available desktop geometry for the console/framebuffer
2594 *
2595 * @returns the geometry. An empty rectangle means unrestricted.
2596 */
2597QRect VBoxConsoleView::desktopGeometry()
2598{
2599 QRect rc;
2600 switch (mDesktopGeo)
2601 {
2602 case DesktopGeo_Fixed:
2603 case DesktopGeo_Automatic:
2604 rc = QRect (0, 0,
2605 RT_MAX (mDesktopGeometry.width(), mLastSizeHint.width()),
2606 RT_MAX (mDesktopGeometry.height(), mLastSizeHint.height()));
2607 break;
2608 case DesktopGeo_Any:
2609 rc = QRect (0, 0, 0, 0);
2610 break;
2611 default:
2612 AssertMsgFailed (("Bad geometry type %d\n", mDesktopGeo));
2613 }
2614 return rc;
2615}
2616
2617QRegion VBoxConsoleView::lastVisibleRegion() const
2618{
2619 return mLastVisibleRegion;
2620}
2621
2622bool VBoxConsoleView::isAutoresizeGuestActive()
2623{
2624 return mGuestSupportsGraphics && mAutoresizeGuest;
2625}
2626
2627/**
2628 * Called on every key press and release (while in focus).
2629 *
2630 * @param aKey virtual scan code (virtual key on Win32 and KeySym on X11)
2631 * @param aScan hardware scan code
2632 * @param aFlags flags, a combination of Key* constants
2633 * @param aUniKey Unicode translation of the key. Optional.
2634 *
2635 * @return true to consume the event and false to pass it to Qt
2636 */
2637bool VBoxConsoleView::keyEvent (int aKey, uint8_t aScan, int aFlags,
2638 wchar_t *aUniKey/* = NULL*/)
2639{
2640#if 0
2641 {
2642 char buf [256];
2643 sprintf (buf, "aKey=%08X aScan=%02X aFlags=%08X",
2644 aKey, aScan, aFlags);
2645 mMainWnd->statusBar()->message (buf);
2646 }
2647#endif
2648
2649 const bool isHostKey = aKey == gs.hostKey();
2650
2651 LONG buf [16];
2652 LONG *codes = buf;
2653 uint count = 0;
2654 uint8_t whatPressed = 0;
2655
2656 if (!isHostKey && !mIsHostkeyPressed)
2657 {
2658 if (aFlags & KeyPrint)
2659 {
2660 static LONG PrintMake[] = { 0xE0, 0x2A, 0xE0, 0x37 };
2661 static LONG PrintBreak[] = { 0xE0, 0xB7, 0xE0, 0xAA };
2662 if (aFlags & KeyPressed)
2663 {
2664 codes = PrintMake;
2665 count = SIZEOF_ARRAY (PrintMake);
2666 }
2667 else
2668 {
2669 codes = PrintBreak;
2670 count = SIZEOF_ARRAY (PrintBreak);
2671 }
2672 }
2673 else if (aFlags & KeyPause)
2674 {
2675 if (aFlags & KeyPressed)
2676 {
2677 static LONG Pause[] = { 0xE1, 0x1D, 0x45, 0xE1, 0x9D, 0xC5 };
2678 codes = Pause;
2679 count = SIZEOF_ARRAY (Pause);
2680 }
2681 else
2682 {
2683 /* Pause shall not produce a break code */
2684 return true;
2685 }
2686 }
2687 else
2688 {
2689 if (aFlags & KeyPressed)
2690 {
2691 /* Check if the guest has the same view on the modifier keys (NumLock,
2692 * CapsLock, ScrollLock) as the X server. If not, send KeyPress events
2693 * to synchronize the state. */
2694 fixModifierState (codes, &count);
2695 }
2696
2697 /* Check if it's C-A-D and GUI/PassCAD is not true */
2698 if (!mPassCAD &&
2699 aScan == 0x53 /* Del */ &&
2700 ((mPressedKeys [0x38] & IsKeyPressed) /* Alt */ ||
2701 (mPressedKeys [0x38] & IsExtKeyPressed)) &&
2702 ((mPressedKeys [0x1d] & IsKeyPressed) /* Ctrl */ ||
2703 (mPressedKeys [0x1d] & IsExtKeyPressed)))
2704 {
2705 /* Use the C-A-D combination as a last resort to get the
2706 * keyboard and mouse back to the host when the user forgets
2707 * the Host Key. Note that it's always possible to send C-A-D
2708 * to the guest using the Host+Del combination. BTW, it would
2709 * be preferrable to completely ignore C-A-D in guests, but
2710 * that's not possible because we cannot predict what other
2711 * keys will be pressed next when one of C, A, D is held. */
2712
2713 if (isRunning() && mKbdCaptured)
2714 {
2715 captureKbd (false);
2716 if (!(mMouseAbsolute && mMouseIntegration))
2717 captureMouse (false);
2718 }
2719
2720 return true;
2721 }
2722
2723 /* process the scancode and update the table of pressed keys */
2724 whatPressed = IsKeyPressed;
2725
2726 if (aFlags & KeyExtended)
2727 {
2728 codes [count++] = 0xE0;
2729 whatPressed = IsExtKeyPressed;
2730 }
2731
2732 if (aFlags & KeyPressed)
2733 {
2734 codes [count++] = aScan;
2735 mPressedKeys [aScan] |= whatPressed;
2736 }
2737 else
2738 {
2739 /* if we haven't got this key's press message, we ignore its
2740 * release */
2741 if (!(mPressedKeys [aScan] & whatPressed))
2742 return true;
2743 codes [count++] = aScan | 0x80;
2744 mPressedKeys [aScan] &= ~whatPressed;
2745 }
2746
2747 if (mKbdCaptured)
2748 mPressedKeys [aScan] |= IsKbdCaptured;
2749 else
2750 mPressedKeys [aScan] &= ~IsKbdCaptured;
2751 }
2752 }
2753 else
2754 {
2755 /* currently this is used in winLowKeyboardEvent() only */
2756 hostkey_in_capture = mKbdCaptured;
2757 }
2758
2759 bool emitSignal = false;
2760 int hotkey = 0;
2761
2762 /* process the host key */
2763 if (aFlags & KeyPressed)
2764 {
2765 if (isHostKey)
2766 {
2767 if (!mIsHostkeyPressed)
2768 {
2769 mIsHostkeyPressed = mIsHostkeyAlone = true;
2770 if (isRunning())
2771 saveKeyStates();
2772 emitSignal = true;
2773 }
2774 }
2775 else
2776 {
2777 if (mIsHostkeyPressed)
2778 {
2779 if (mIsHostkeyAlone)
2780 {
2781 hotkey = aKey;
2782 mIsHostkeyAlone = false;
2783 }
2784 }
2785 }
2786 }
2787 else
2788 {
2789 if (isHostKey)
2790 {
2791 if (mIsHostkeyPressed)
2792 {
2793 mIsHostkeyPressed = false;
2794
2795 if (mIsHostkeyAlone)
2796 {
2797 if (isPaused())
2798 {
2799 vboxProblem().remindAboutPausedVMInput();
2800 }
2801 else
2802 if (isRunning())
2803 {
2804 bool captured = mKbdCaptured;
2805 bool ok = true;
2806 if (!captured)
2807 {
2808 /* temporarily disable auto capture that will take
2809 * place after this dialog is dismissed because
2810 * the capture state is to be defined by the
2811 * dialog result itself */
2812 mDisableAutoCapture = true;
2813 bool autoConfirmed = false;
2814 ok = vboxProblem().confirmInputCapture (&autoConfirmed);
2815 if (autoConfirmed)
2816 mDisableAutoCapture = false;
2817 /* otherwise, the disable flag will be reset in
2818 * the next console view's foucs in event (since
2819 * may happen asynchronously on some platforms,
2820 * after we return from this code) */
2821 }
2822
2823 if (ok)
2824 {
2825 captureKbd (!captured, false);
2826 if (!(mMouseAbsolute && mMouseIntegration))
2827 {
2828#ifdef Q_WS_X11
2829 /* make sure that pending FocusOut events from the
2830 * previous message box are handled, otherwise the
2831 * mouse is immediately ungrabbed. */
2832 qApp->processEvents();
2833#endif
2834 captureMouse (mKbdCaptured);
2835 }
2836 }
2837 }
2838 }
2839
2840 if (isRunning())
2841 sendChangedKeyStates();
2842
2843 emitSignal = true;
2844 }
2845 }
2846 else
2847 {
2848 if (mIsHostkeyPressed)
2849 mIsHostkeyAlone = false;
2850 }
2851 }
2852
2853 /* emit the keyboard state change signal */
2854 if (emitSignal)
2855 emitKeyboardStateChanged();
2856
2857 /* Process Host+<key> shortcuts. currently, <key> is limited to
2858 * alphanumeric chars. Other Host+<key> combinations are handled in
2859 * event(). */
2860 if (hotkey)
2861 {
2862 bool processed = false;
2863#if defined (Q_WS_WIN32)
2864 NOREF(aUniKey);
2865 int n = GetKeyboardLayoutList (0, NULL);
2866 Assert (n);
2867 HKL *list = new HKL [n];
2868 GetKeyboardLayoutList (n, list);
2869 for (int i = 0; i < n && !processed; i++)
2870 {
2871 wchar_t ch;
2872 static BYTE keys [256] = {0};
2873 if (!ToUnicodeEx (hotkey, 0, keys, &ch, 1, 0, list [i]) == 1)
2874 ch = 0;
2875 if (ch)
2876 processed = processHotKey (QKeySequence (Qt::UNICODE_ACCEL +
2877 QChar (ch).toUpper().unicode()),
2878 mMainWnd->menuBar()->actions());
2879 }
2880 delete[] list;
2881#elif defined (Q_WS_X11)
2882 NOREF(aUniKey);
2883 Display *display = QX11Info::display();
2884 int keysyms_per_keycode = getKeysymsPerKeycode();
2885 KeyCode kc = XKeysymToKeycode (display, aKey);
2886 // iterate over the first level (not shifted) keysyms in every group
2887 for (int i = 0; i < keysyms_per_keycode && !processed; i += 2)
2888 {
2889 KeySym ks = XKeycodeToKeysym (display, kc, i);
2890 char ch = 0;
2891 if (!XkbTranslateKeySym (display, &ks, 0, &ch, 1, NULL) == 1)
2892 ch = 0;
2893 if (ch)
2894 {
2895 QChar c = QString::fromLocal8Bit (&ch, 1) [0];
2896 processed = processHotKey (QKeySequence (Qt::UNICODE_ACCEL +
2897 c.toUpper().unicode()),
2898 mMainWnd->menuBar()->actions());
2899 }
2900 }
2901#elif defined (Q_WS_MAC)
2902 if (aUniKey && aUniKey [0] && !aUniKey [1])
2903 processed = processHotKey (QKeySequence (Qt::UNICODE_ACCEL +
2904 QChar (aUniKey [0]).toUpper().unicode()),
2905 mMainWnd->menuBar()->actions());
2906
2907 /* Don't consider the hot key as pressed since the guest never saw
2908 * it. (probably a generic thing) */
2909 mPressedKeys [aScan] &= ~whatPressed;
2910#endif
2911
2912 /* grab the key from Qt if processed, or pass it to Qt otherwise
2913 * in order to process non-alphanumeric keys in event(), after they are
2914 * converted to Qt virtual keys. */
2915 return processed;
2916 }
2917
2918 /* no more to do, if the host key is in action or the VM is paused */
2919 if (mIsHostkeyPressed || isHostKey || isPaused())
2920 {
2921 /* grab the key from Qt and from VM if it's a host key,
2922 * otherwise just pass it to Qt */
2923 return isHostKey;
2924 }
2925
2926 CKeyboard keyboard = mConsole.GetKeyboard();
2927 Assert (!keyboard.isNull());
2928
2929#if defined (Q_WS_WIN32)
2930 /* send pending WM_PAINT events */
2931 ::UpdateWindow (viewport()->winId());
2932#endif
2933
2934#if 0
2935 {
2936 char buf [256];
2937 sprintf (buf, "*** SCANS: ");
2938 for (uint i = 0; i < count; ++ i)
2939 sprintf (buf + strlen (buf), "%02X ", codes [i]);
2940 mMainWnd->statusBar()->message (buf);
2941 LogFlow (("%s\n", buf));
2942 }
2943#endif
2944
2945 std::vector <LONG> scancodes(codes, &codes[count]);
2946 keyboard.PutScancodes (QVector<LONG>::fromStdVector(scancodes));
2947
2948 /* grab the key from Qt */
2949 return true;
2950}
2951
2952/**
2953 * Called on every mouse/wheel move and button press/release.
2954 *
2955 * @return true to consume the event and false to pass it to Qt
2956 */
2957bool VBoxConsoleView::mouseEvent (int aType, const QPoint &aPos, const QPoint &aGlobalPos,
2958 Qt::MouseButtons aButtons, Qt::KeyboardModifiers aModifiers,
2959 int aWheelDelta, Qt::Orientation aWheelDir)
2960{
2961#if 0
2962 char buf [256];
2963 sprintf (buf,
2964 "MOUSE: type=%03d x=%03d y=%03d btn=%03d btns=%08X mod=%08X "
2965 "wdelta=%03d wdir=%03d",
2966 aType, aPos.x(), aPos.y(), aButtons, aModifiers,
2967 aWheelDelta, aWheelDir);
2968 mMainWnd->statusBar()->message (buf);
2969#else
2970 Q_UNUSED (aModifiers);
2971#endif
2972
2973 int state = 0;
2974 if (aButtons & Qt::LeftButton)
2975 state |= KMouseButtonState_LeftButton;
2976 if (aButtons & Qt::RightButton)
2977 state |= KMouseButtonState_RightButton;
2978 if (aButtons & Qt::MidButton)
2979 state |= KMouseButtonState_MiddleButton;
2980
2981#ifdef Q_WS_MAC
2982 /* Simulate the right click on
2983 * Host+Left Mouse */
2984 if (mIsHostkeyPressed &&
2985 mIsHostkeyAlone &&
2986 state == KMouseButtonState_LeftButton)
2987 state = KMouseButtonState_RightButton;
2988#endif /* Q_WS_MAC */
2989
2990 int wheel = 0;
2991 if (aWheelDir == Qt::Vertical)
2992 {
2993 /* the absolute value of wheel delta is 120 units per every wheel
2994 * move; positive deltas correspond to counterclockwize rotations
2995 * (usually up), negative -- to clockwize (usually down). */
2996 wheel = - (aWheelDelta / 120);
2997 }
2998
2999 if (mMouseCaptured)
3000 {
3001#ifdef Q_WS_WIN32
3002 /* send pending WM_PAINT events */
3003 ::UpdateWindow (viewport()->winId());
3004#endif
3005
3006 CMouse mouse = mConsole.GetMouse();
3007 mouse.PutMouseEvent (aGlobalPos.x() - mLastPos.x(),
3008 aGlobalPos.y() - mLastPos.y(),
3009 wheel, state);
3010
3011#if defined (Q_WS_MAC)
3012 /*
3013 * Keep the mouse from leaving the widget.
3014 *
3015 * This is a bit tricky to get right because if it escapes we won't necessarily
3016 * get mouse events any longer and can warp it back. So, we keep safety zone
3017 * of up to 300 pixels around the borders of the widget to prevent this from
3018 * happening. Also, the mouse is warped back to the center of the widget.
3019 *
3020 * (Note, aPos seems to be unreliable, it caused endless recursion here at one points...)
3021 * (Note, synergy and other remote clients might not like this cursor warping.)
3022 */
3023 QRect rect = viewport()->visibleRegion().boundingRect();
3024 QPoint pw = viewport()->mapToGlobal (viewport()->pos());
3025 rect.translate (pw.x(), pw.y());
3026
3027 QRect dpRect = QApplication::desktop()->screenGeometry (viewport());
3028 if (rect.intersects (dpRect))
3029 rect = rect.intersect (dpRect);
3030
3031 int wsafe = rect.width() / 6;
3032 rect.setWidth (rect.width() - wsafe * 2);
3033 rect.setLeft (rect.left() + wsafe);
3034
3035 int hsafe = rect.height() / 6;
3036 rect.setWidth (rect.height() - hsafe * 2);
3037 rect.setTop (rect.top() + hsafe);
3038
3039 if (rect.contains (aGlobalPos, true))
3040 mLastPos = aGlobalPos;
3041 else
3042 {
3043 mLastPos = rect.center();
3044 QCursor::setPos (mLastPos);
3045 }
3046
3047#else /* !Q_WS_MAC */
3048
3049 /* "jerk" the mouse by bringing it to the opposite side
3050 * to simulate the endless moving */
3051
3052#ifdef Q_WS_WIN32
3053 int we = viewport()->width() - 1;
3054 int he = viewport()->height() - 1;
3055 QPoint p = aPos;
3056 if (aPos.x() == 0)
3057 p.setX (we - 1);
3058 else if (aPos.x() == we)
3059 p.setX (1);
3060 if (aPos.y() == 0 )
3061 p.setY (he - 1);
3062 else if (aPos.y() == he)
3063 p.setY (1);
3064
3065 if (p != aPos)
3066 {
3067 mLastPos = viewport()->mapToGlobal (p);
3068 QCursor::setPos (mLastPos);
3069 }
3070 else
3071 {
3072 mLastPos = aGlobalPos;
3073 }
3074#else
3075 int we = QApplication::desktop()->width() - 1;
3076 int he = QApplication::desktop()->height() - 1;
3077 QPoint p = aGlobalPos;
3078 if (aGlobalPos.x() == 0)
3079 p.setX (we - 1);
3080 else if (aGlobalPos.x() == we)
3081 p.setX( 1 );
3082 if (aGlobalPos.y() == 0)
3083 p.setY (he - 1);
3084 else if (aGlobalPos.y() == he)
3085 p.setY (1);
3086
3087 if (p != aGlobalPos)
3088 {
3089 mLastPos = p;
3090 QCursor::setPos (mLastPos);
3091 }
3092 else
3093 {
3094 mLastPos = aGlobalPos;
3095 }
3096#endif
3097#endif /* !Q_WS_MAC */
3098 return true; /* stop further event handling */
3099 }
3100 else /* !mMouseCaptured */
3101 {
3102 if (mMainWnd->isTrueFullscreen())
3103 {
3104 if (mode != VBoxDefs::SDLMode)
3105 {
3106 /* try to automatically scroll the guest canvas if the
3107 * mouse is on the screen border */
3108 /// @todo (r=dmik) better use a timer for autoscroll
3109 QRect scrGeo = QApplication::desktop()->screenGeometry (this);
3110 int dx = 0, dy = 0;
3111 if (scrGeo.width() < contentsWidth())
3112 {
3113 if (scrGeo.left() == aGlobalPos.x()) dx = -1;
3114 if (scrGeo.right() == aGlobalPos.x()) dx = +1;
3115 }
3116 if (scrGeo.height() < contentsHeight())
3117 {
3118 if (scrGeo.top() == aGlobalPos.y()) dy = -1;
3119 if (scrGeo.bottom() == aGlobalPos.y()) dy = +1;
3120 }
3121 if (dx || dy)
3122 scrollBy (dx, dy);
3123 }
3124 }
3125
3126 if (mMouseAbsolute && mMouseIntegration)
3127 {
3128 int cw = contentsWidth(), ch = contentsHeight();
3129 int vw = visibleWidth(), vh = visibleHeight();
3130
3131 if (mode != VBoxDefs::SDLMode)
3132 {
3133 /* try to automatically scroll the guest canvas if the
3134 * mouse goes outside its visible part */
3135
3136 int dx = 0;
3137 if (aPos.x() > vw) dx = aPos.x() - vw;
3138 else if (aPos.x() < 0) dx = aPos.x();
3139 int dy = 0;
3140 if (aPos.y() > vh) dy = aPos.y() - vh;
3141 else if (aPos.y() < 0) dy = aPos.y();
3142 if (dx != 0 || dy != 0) scrollBy (dx, dy);
3143 }
3144
3145 QPoint cpnt = viewportToContents (aPos);
3146 if (cpnt.x() < 0) cpnt.setX (0);
3147 else if (cpnt.x() >= cw) cpnt.setX (cw - 1);
3148 if (cpnt.y() < 0) cpnt.setY (0);
3149 else if (cpnt.y() >= ch) cpnt.setY (ch - 1);
3150
3151 CMouse mouse = mConsole.GetMouse();
3152 mouse.PutMouseEventAbsolute (cpnt.x() + 1, cpnt.y() + 1,
3153 wheel, state);
3154 return true; /* stop further event handling */
3155 }
3156 else
3157 {
3158 if (hasFocus() &&
3159 (aType == QEvent::MouseButtonRelease &&
3160 aButtons == Qt::NoButton))
3161 {
3162 if (isPaused())
3163 {
3164 vboxProblem().remindAboutPausedVMInput();
3165 }
3166 else if (isRunning())
3167 {
3168 /* temporarily disable auto capture that will take
3169 * place after this dialog is dismissed because
3170 * the capture state is to be defined by the
3171 * dialog result itself */
3172 mDisableAutoCapture = true;
3173 bool autoConfirmed = false;
3174 bool ok = vboxProblem().confirmInputCapture (&autoConfirmed);
3175 if (autoConfirmed)
3176 mDisableAutoCapture = false;
3177 /* otherwise, the disable flag will be reset in
3178 * the next console view's foucs in event (since
3179 * may happen asynchronously on some platforms,
3180 * after we return from this code) */
3181
3182 if (ok)
3183 {
3184#ifdef Q_WS_X11
3185 /* make sure that pending FocusOut events from the
3186 * previous message box are handled, otherwise the
3187 * mouse is immediately ungrabbed again */
3188 qApp->processEvents();
3189#endif
3190 captureKbd (true);
3191 captureMouse (true);
3192 }
3193 }
3194 }
3195 }
3196 }
3197
3198 return false;
3199}
3200
3201void VBoxConsoleView::onStateChange (KMachineState state)
3202{
3203 switch (state)
3204 {
3205 case KMachineState_Paused:
3206 {
3207 if (mode != VBoxDefs::TimerMode && mFrameBuf)
3208 {
3209 /*
3210 * Take a screen snapshot. Note that TakeScreenShot() always
3211 * needs a 32bpp image
3212 */
3213 QImage shot = QImage (mFrameBuf->width(), mFrameBuf->height(), QImage::Format_RGB32);
3214 CDisplay dsp = mConsole.GetDisplay();
3215 dsp.TakeScreenShot (shot.bits(), shot.width(), shot.height());
3216 /*
3217 * TakeScreenShot() may fail if, e.g. the Paused notification
3218 * was delivered after the machine execution was resumed. It's
3219 * not fatal.
3220 */
3221 if (dsp.isOk())
3222 {
3223 dimImage (shot);
3224 mPausedShot = QPixmap::fromImage (shot);
3225 /* fully repaint to pick up mPausedShot */
3226 repaint();
3227 }
3228 }
3229 /* fall through */
3230 }
3231 case KMachineState_Stuck:
3232 {
3233 /* reuse the focus event handler to uncapture everything */
3234 if (hasFocus())
3235 focusEvent (false /* aHasFocus*/, false /* aReleaseHostKey */);
3236 break;
3237 }
3238 case KMachineState_Running:
3239 {
3240 if (mLastState == KMachineState_Paused)
3241 {
3242 if (mode != VBoxDefs::TimerMode && mFrameBuf)
3243 {
3244 /* reset the pixmap to free memory */
3245 mPausedShot = QPixmap ();
3246 /*
3247 * ask for full guest display update (it will also update
3248 * the viewport through IFramebuffer::NotifyUpdate)
3249 */
3250 CDisplay dsp = mConsole.GetDisplay();
3251 dsp.InvalidateAndUpdate();
3252 }
3253 }
3254 /* reuse the focus event handler to capture input */
3255 if (hasFocus())
3256 focusEvent (true /* aHasFocus */);
3257 break;
3258 }
3259 default:
3260 break;
3261 }
3262
3263 mLastState = state;
3264}
3265
3266void VBoxConsoleView::doRefresh()
3267{
3268 viewport()->repaint();
3269}
3270
3271void VBoxConsoleView::resizeEvent (QResizeEvent *)
3272{
3273 updateSliders();
3274#if defined(Q_WS_MAC) && !defined(QT_MAC_USE_COCOA)
3275 QRect r = viewport()->geometry();
3276// printf ("qt resize: %d %d %d %d\n", r.x(), r.y(), r.width(), r.height());
3277 PostBoundsChanged (r);
3278#endif /* Q_WS_MAC */
3279}
3280
3281void VBoxConsoleView::moveEvent (QMoveEvent *)
3282{
3283#if defined(Q_WS_MAC) && !defined(QT_MAC_USE_COCOA)
3284 QRect r = viewport()->geometry();
3285// printf ("qt resize: %d %d %d %d\n", r.x(), r.y(), r.width(), r.height());
3286 PostBoundsChanged (r);
3287#endif /* Q_WS_MAC */
3288}
3289
3290void VBoxConsoleView::paintEvent (QPaintEvent *pe)
3291{
3292 if (mPausedShot.isNull())
3293 {
3294 /* delegate the paint function to the VBoxFrameBuffer interface */
3295 if (mFrameBuf)
3296 mFrameBuf->paintEvent (pe);
3297#ifdef Q_WS_MAC
3298 /* Update the dock icon if we are in the running state */
3299 if (isRunning())
3300 updateDockIcon();
3301#endif
3302 return;
3303 }
3304
3305#ifdef VBOX_GUI_USE_QUARTZ2D
3306 if (mode == VBoxDefs::Quartz2DMode && mFrameBuf)
3307 {
3308 mFrameBuf->paintEvent (pe);
3309 updateDockIcon();
3310 }
3311 else
3312#endif
3313 {
3314 /* we have a snapshot for the paused state */
3315 QRect r = pe->rect().intersect (viewport()->rect());
3316 /* We have to disable paint on screen if we are using the regular painter */
3317 bool paintOnScreen = viewport()->testAttribute (Qt::WA_PaintOnScreen);
3318 viewport()->setAttribute (Qt::WA_PaintOnScreen, false);
3319 QPainter pnt (viewport());
3320 pnt.drawPixmap (r.x(), r.y(), mPausedShot,
3321 r.x() + contentsX(), r.y() + contentsY(),
3322 r.width(), r.height());
3323 /* Restore the attribute to its previous state */
3324 viewport()->setAttribute (Qt::WA_PaintOnScreen, paintOnScreen);
3325#ifdef Q_WS_MAC
3326 updateDockIcon();
3327#endif
3328 }
3329
3330}
3331
3332/**
3333 * Captures the keyboard. When captured, no keyboard input reaches the host
3334 * system (including most system combinations like Alt-Tab).
3335 *
3336 * @param aCapture true to capture, false to uncapture.
3337 * @param aEmitSignal Whether to emit keyboardStateChanged() or not.
3338 */
3339void VBoxConsoleView::captureKbd (bool aCapture, bool aEmitSignal /* = true */)
3340{
3341 AssertMsg (mAttached, ("Console must be attached"));
3342
3343 if (mKbdCaptured == aCapture)
3344 return;
3345
3346 /* On Win32, keyboard grabbing is ineffective, a low-level keyboard hook is
3347 * used instead. On X11, we use XGrabKey instead of XGrabKeyboard (called
3348 * by QWidget::grabKeyboard()) because the latter causes problems under
3349 * metacity 2.16 (in particular, due to a bug, a window cannot be moved
3350 * using the mouse if it is currently grabing the keyboard). On Mac OS X,
3351 * we use the Qt methods + disabling global hot keys + watching modifiers
3352 * (for right/left separation). */
3353#if defined (Q_WS_WIN32)
3354 /**/
3355#elif defined (Q_WS_X11)
3356 if (aCapture)
3357 XGrabKey (QX11Info::display(), AnyKey, AnyModifier,
3358 window()->winId(), False,
3359 GrabModeAsync, GrabModeAsync);
3360 else
3361 XUngrabKey (QX11Info::display(), AnyKey, AnyModifier,
3362 window()->winId());
3363#elif defined (Q_WS_MAC)
3364 if (aCapture)
3365 {
3366 ::DarwinDisableGlobalHotKeys (true);
3367 grabKeyboard();
3368 }
3369 else
3370 {
3371 ::DarwinDisableGlobalHotKeys (false);
3372 releaseKeyboard();
3373 }
3374#else
3375 if (aCapture)
3376 grabKeyboard();
3377 else
3378 releaseKeyboard();
3379#endif
3380
3381 mKbdCaptured = aCapture;
3382
3383 if (aEmitSignal)
3384 emitKeyboardStateChanged();
3385}
3386
3387/**
3388 * Captures the host mouse pointer. When captured, the mouse pointer is
3389 * unavailable to the host applications.
3390 *
3391 * @param aCapture true to capture, false to uncapture.
3392 * @param aEmitSignal Whether to emit mouseStateChanged() or not.
3393 */
3394void VBoxConsoleView::captureMouse (bool aCapture, bool aEmitSignal /* = true */)
3395{
3396 AssertMsg (mAttached, ("Console must be attached"));
3397
3398 if (mMouseCaptured == aCapture)
3399 return;
3400
3401 if (aCapture)
3402 {
3403 /* memorize the host position where the cursor was captured */
3404 mCapturedPos = QCursor::pos();
3405#ifdef Q_WS_WIN32
3406 viewport()->setCursor (QCursor (Qt::BlankCursor));
3407 /* move the mouse to the center of the visible area */
3408 QCursor::setPos (mapToGlobal (visibleRegion().boundingRect().center()));
3409 mLastPos = QCursor::pos();
3410#elif defined (Q_WS_MAC)
3411 /* move the mouse to the center of the visible area */
3412 mLastPos = mapToGlobal (visibleRegion().boundingRect().center());
3413 QCursor::setPos (mLastPos);
3414 /* grab all mouse events. */
3415 viewport()->grabMouse();
3416#else
3417 viewport()->grabMouse();
3418 mLastPos = QCursor::pos();
3419#endif
3420 }
3421 else
3422 {
3423#ifndef Q_WS_WIN32
3424 viewport()->releaseMouse();
3425#endif
3426 /* release mouse buttons */
3427 CMouse mouse = mConsole.GetMouse();
3428 mouse.PutMouseEvent (0, 0, 0, 0);
3429 }
3430
3431 mMouseCaptured = aCapture;
3432
3433 updateMouseClipping();
3434
3435 if (aEmitSignal)
3436 emitMouseStateChanged();
3437}
3438
3439/**
3440 * Searches for a menu item with a given hot key (shortcut). If the item
3441 * is found, activates it and returns true. Otherwise returns false.
3442 */
3443bool VBoxConsoleView::processHotKey (const QKeySequence &aKey,
3444 const QList <QAction*> &aData)
3445{
3446 foreach (QAction *pAction, aData)
3447 {
3448 if (QMenu *menu = pAction->menu())
3449 {
3450 /* Process recursively for each sub-menu */
3451 if (processHotKey (aKey, menu->actions()))
3452 return true;
3453 }
3454 else
3455 {
3456 QString hotkey = VBoxGlobal::extractKeyFromActionText (pAction->text());
3457 if (pAction->isEnabled() && !hotkey.isEmpty())
3458 {
3459 if (aKey.matches (QKeySequence (hotkey)) == QKeySequence::ExactMatch)
3460 {
3461 /* We asynchronously post a special event instead of calling
3462 * pAction->trigger() directly, to let key presses and
3463 * releases be processed correctly by Qt first.
3464 * Note: we assume that nobody will delete the menu item
3465 * corresponding to the key sequence, so that the pointer to
3466 * menu data posted along with the event will remain valid in
3467 * the event handler, at least until the main window is closed. */
3468 QApplication::postEvent (this, new ActivateMenuEvent (pAction));
3469 return true;
3470 }
3471 }
3472 }
3473 }
3474
3475 return false;
3476}
3477
3478/**
3479 * Send the KEY BREAK code to the VM for all currently pressed keys.
3480 *
3481 * @param aReleaseHostKey @c true to set the host key state to unpressed.
3482 */
3483void VBoxConsoleView::releaseAllPressedKeys (bool aReleaseHostKey /* = true*/)
3484{
3485 AssertMsg (mAttached, ("Console must be attached"));
3486
3487 CKeyboard keyboard = mConsole.GetKeyboard();
3488 bool fSentRESEND = false;
3489
3490 /* send a dummy scan code (RESEND) to prevent the guest OS from recognizing
3491 * a single key click (for ex., Alt) and performing an unwanted action
3492 * (for ex., activating the menu) when we release all pressed keys below.
3493 * Note, that it's just a guess that sending RESEND will give the desired
3494 * effect :), but at least it works with NT and W2k guests. */
3495
3496 /// @todo Sending 0xFE is responsible for the warning
3497 //
3498 // ``atkbd.c: Spurious NAK on isa0060/serio0. Some program might
3499 // be trying access hardware directly''
3500 //
3501 // on Linux guests (#1944). It might also be responsible for #1949. Don't
3502 // send this command unless we really have to release any key modifier.
3503 // --frank
3504
3505 for (uint i = 0; i < SIZEOF_ARRAY (mPressedKeys); i++)
3506 {
3507 if (mPressedKeys [i] & IsKeyPressed)
3508 {
3509 if (!fSentRESEND)
3510 {
3511 keyboard.PutScancode (0xFE);
3512 fSentRESEND = true;
3513 }
3514 keyboard.PutScancode (i | 0x80);
3515 }
3516 else if (mPressedKeys [i] & IsExtKeyPressed)
3517 {
3518 if (!fSentRESEND)
3519 {
3520 keyboard.PutScancode (0xFE);
3521 fSentRESEND = true;
3522 }
3523 QVector <LONG> codes (2);
3524 codes[0] = 0xE0;
3525 codes[1] = i | 0x80;
3526 keyboard.PutScancodes (codes);
3527 }
3528 mPressedKeys [i] = 0;
3529 }
3530
3531 if (aReleaseHostKey)
3532 mIsHostkeyPressed = false;
3533
3534#ifdef Q_WS_MAC
3535 /* clear most of the modifiers. */
3536 mDarwinKeyModifiers &=
3537 alphaLock | kEventKeyModifierNumLockMask |
3538 (aReleaseHostKey ? 0 : ::DarwinKeyCodeToDarwinModifierMask (gs.hostKey()));
3539#endif
3540
3541 emitKeyboardStateChanged();
3542}
3543
3544void VBoxConsoleView::saveKeyStates()
3545{
3546 ::memcpy (mPressedKeysCopy, mPressedKeys, sizeof (mPressedKeys));
3547}
3548
3549void VBoxConsoleView::sendChangedKeyStates()
3550{
3551 AssertMsg (mAttached, ("Console must be attached"));
3552
3553 QVector <LONG> codes (2);
3554 CKeyboard keyboard = mConsole.GetKeyboard();
3555 for (uint i = 0; i < SIZEOF_ARRAY (mPressedKeys); ++ i)
3556 {
3557 uint8_t os = mPressedKeysCopy [i];
3558 uint8_t ns = mPressedKeys [i];
3559 if ((os & IsKeyPressed) != (ns & IsKeyPressed))
3560 {
3561 codes [0] = i;
3562 if (!(ns & IsKeyPressed))
3563 codes[0] |= 0x80;
3564 keyboard.PutScancode (codes[0]);
3565 }
3566 else if ((os & IsExtKeyPressed) != (ns & IsExtKeyPressed))
3567 {
3568 codes [0] = 0xE0;
3569 codes [1] = i;
3570 if (!(ns & IsExtKeyPressed))
3571 codes [1] |= 0x80;
3572 keyboard.PutScancodes (codes);
3573 }
3574 }
3575}
3576
3577void VBoxConsoleView::updateMouseClipping()
3578{
3579 AssertMsg (mAttached, ("Console must be attached"));
3580
3581 if (mMouseCaptured)
3582 {
3583 viewport()->setCursor (QCursor (Qt::BlankCursor));
3584#ifdef Q_WS_WIN32
3585 QRect r = viewport()->rect();
3586 r.moveTopLeft (viewport()->mapToGlobal (QPoint (0, 0)));
3587 RECT rect = { r.left(), r.top(), r.right() + 1, r.bottom() + 1 };
3588 ::ClipCursor (&rect);
3589#endif
3590 }
3591 else
3592 {
3593#ifdef Q_WS_WIN32
3594 ::ClipCursor (NULL);
3595#endif
3596 /* return the cursor to where it was when we captured it and show it */
3597 QCursor::setPos (mCapturedPos);
3598 viewport()->unsetCursor();
3599 }
3600}
3601
3602void VBoxConsoleView::setPointerShape (MousePointerChangeEvent *me)
3603{
3604 if (me->shapeData() != NULL)
3605 {
3606 bool ok = false;
3607
3608 const uchar *srcAndMaskPtr = me->shapeData();
3609 uint andMaskSize = (me->width() + 7) / 8 * me->height();
3610 const uchar *srcShapePtr = me->shapeData() + ((andMaskSize + 3) & ~3);
3611 uint srcShapePtrScan = me->width() * 4;
3612
3613#if defined (Q_WS_WIN)
3614
3615 BITMAPV5HEADER bi;
3616 HBITMAP hBitmap;
3617 void *lpBits;
3618
3619 ::ZeroMemory (&bi, sizeof (BITMAPV5HEADER));
3620 bi.bV5Size = sizeof (BITMAPV5HEADER);
3621 bi.bV5Width = me->width();
3622 bi.bV5Height = - (LONG) me->height();
3623 bi.bV5Planes = 1;
3624 bi.bV5BitCount = 32;
3625 bi.bV5Compression = BI_BITFIELDS;
3626 // specifiy a supported 32 BPP alpha format for Windows XP
3627 bi.bV5RedMask = 0x00FF0000;
3628 bi.bV5GreenMask = 0x0000FF00;
3629 bi.bV5BlueMask = 0x000000FF;
3630 if (me->hasAlpha())
3631 bi.bV5AlphaMask = 0xFF000000;
3632 else
3633 bi.bV5AlphaMask = 0;
3634
3635 HDC hdc = GetDC (NULL);
3636
3637 // create the DIB section with an alpha channel
3638 hBitmap = CreateDIBSection (hdc, (BITMAPINFO *) &bi, DIB_RGB_COLORS,
3639 (void **) &lpBits, NULL, (DWORD) 0);
3640
3641 ReleaseDC (NULL, hdc);
3642
3643 HBITMAP hMonoBitmap = NULL;
3644 if (me->hasAlpha())
3645 {
3646 // create an empty mask bitmap
3647 hMonoBitmap = CreateBitmap (me->width(), me->height(), 1, 1, NULL);
3648 }
3649 else
3650 {
3651 /* Word aligned AND mask. Will be allocated and created if necessary. */
3652 uint8_t *pu8AndMaskWordAligned = NULL;
3653
3654 /* Width in bytes of the original AND mask scan line. */
3655 uint32_t cbAndMaskScan = (me->width() + 7) / 8;
3656
3657 if (cbAndMaskScan & 1)
3658 {
3659 /* Original AND mask is not word aligned. */
3660
3661 /* Allocate memory for aligned AND mask. */
3662 pu8AndMaskWordAligned = (uint8_t *)RTMemTmpAllocZ ((cbAndMaskScan + 1) * me->height());
3663
3664 Assert(pu8AndMaskWordAligned);
3665
3666 if (pu8AndMaskWordAligned)
3667 {
3668 /* According to MSDN the padding bits must be 0.
3669 * Compute the bit mask to set padding bits to 0 in the last byte of original AND mask.
3670 */
3671 uint32_t u32PaddingBits = cbAndMaskScan * 8 - me->width();
3672 Assert(u32PaddingBits < 8);
3673 uint8_t u8LastBytesPaddingMask = (uint8_t)(0xFF << u32PaddingBits);
3674
3675 Log(("u8LastBytesPaddingMask = %02X, aligned w = %d, width = %d, cbAndMaskScan = %d\n",
3676 u8LastBytesPaddingMask, (cbAndMaskScan + 1) * 8, me->width(), cbAndMaskScan));
3677
3678 uint8_t *src = (uint8_t *)srcAndMaskPtr;
3679 uint8_t *dst = pu8AndMaskWordAligned;
3680
3681 unsigned i;
3682 for (i = 0; i < me->height(); i++)
3683 {
3684 memcpy (dst, src, cbAndMaskScan);
3685
3686 dst[cbAndMaskScan - 1] &= u8LastBytesPaddingMask;
3687
3688 src += cbAndMaskScan;
3689 dst += cbAndMaskScan + 1;
3690 }
3691 }
3692 }
3693
3694 /* create the AND mask bitmap */
3695 hMonoBitmap = ::CreateBitmap (me->width(), me->height(), 1, 1,
3696 pu8AndMaskWordAligned? pu8AndMaskWordAligned: srcAndMaskPtr);
3697
3698 if (pu8AndMaskWordAligned)
3699 {
3700 RTMemTmpFree (pu8AndMaskWordAligned);
3701 }
3702 }
3703
3704 Assert (hBitmap);
3705 Assert (hMonoBitmap);
3706 if (hBitmap && hMonoBitmap)
3707 {
3708 DWORD *dstShapePtr = (DWORD *) lpBits;
3709
3710 for (uint y = 0; y < me->height(); y ++)
3711 {
3712 memcpy (dstShapePtr, srcShapePtr, srcShapePtrScan);
3713 srcShapePtr += srcShapePtrScan;
3714 dstShapePtr += me->width();
3715 }
3716
3717 ICONINFO ii;
3718 ii.fIcon = FALSE;
3719 ii.xHotspot = me->xHot();
3720 ii.yHotspot = me->yHot();
3721 ii.hbmMask = hMonoBitmap;
3722 ii.hbmColor = hBitmap;
3723
3724 HCURSOR hAlphaCursor = CreateIconIndirect (&ii);
3725 Assert (hAlphaCursor);
3726 if (hAlphaCursor)
3727 {
3728 viewport()->setCursor (QCursor (hAlphaCursor));
3729 ok = true;
3730 if (mAlphaCursor)
3731 DestroyIcon (mAlphaCursor);
3732 mAlphaCursor = hAlphaCursor;
3733 }
3734 }
3735
3736 if (hMonoBitmap)
3737 DeleteObject (hMonoBitmap);
3738 if (hBitmap)
3739 DeleteObject (hBitmap);
3740
3741#elif defined (Q_WS_X11) && !defined (VBOX_WITHOUT_XCURSOR)
3742
3743 XcursorImage *img = XcursorImageCreate (me->width(), me->height());
3744 Assert (img);
3745 if (img)
3746 {
3747 img->xhot = me->xHot();
3748 img->yhot = me->yHot();
3749
3750 XcursorPixel *dstShapePtr = img->pixels;
3751
3752 for (uint y = 0; y < me->height(); y ++)
3753 {
3754 memcpy (dstShapePtr, srcShapePtr, srcShapePtrScan);
3755
3756 if (!me->hasAlpha())
3757 {
3758 /* convert AND mask to the alpha channel */
3759 uchar byte = 0;
3760 for (uint x = 0; x < me->width(); x ++)
3761 {
3762 if (!(x % 8))
3763 byte = *(srcAndMaskPtr ++);
3764 else
3765 byte <<= 1;
3766
3767 if (byte & 0x80)
3768 {
3769 /* Linux doesn't support inverted pixels (XOR ops,
3770 * to be exact) in cursor shapes, so we detect such
3771 * pixels and always replace them with black ones to
3772 * make them visible at least over light colors */
3773 if (dstShapePtr [x] & 0x00FFFFFF)
3774 dstShapePtr [x] = 0xFF000000;
3775 else
3776 dstShapePtr [x] = 0x00000000;
3777 }
3778 else
3779 dstShapePtr [x] |= 0xFF000000;
3780 }
3781 }
3782
3783 srcShapePtr += srcShapePtrScan;
3784 dstShapePtr += me->width();
3785 }
3786
3787 Cursor cur = XcursorImageLoadCursor (QX11Info::display(), img);
3788 Assert (cur);
3789 if (cur)
3790 {
3791 viewport()->setCursor (QCursor (cur));
3792 ok = true;
3793 }
3794
3795 XcursorImageDestroy (img);
3796 }
3797
3798#elif defined(Q_WS_MAC)
3799
3800 /* Create a ARGB image out of the shape data. */
3801 QImage image (me->width(), me->height(), QImage::Format_ARGB32);
3802 const uint8_t* pbSrcMask = static_cast<const uint8_t*> (srcAndMaskPtr);
3803 unsigned cbSrcMaskLine = RT_ALIGN (me->width(), 8) / 8;
3804 for (unsigned int y = 0; y < me->height(); ++y)
3805 {
3806 for (unsigned int x = 0; x < me->width(); ++x)
3807 {
3808 unsigned int color = ((unsigned int*)srcShapePtr)[y*me->width()+x];
3809 /* If the alpha channel isn't in the shape data, we have to
3810 * create them from the and-mask. This is a bit field where 1
3811 * represent transparency & 0 opaque respectively. */
3812 if (!me->hasAlpha())
3813 {
3814 if (!(pbSrcMask[x / 8] & (1 << (7 - (x % 8)))))
3815 color |= 0xff000000;
3816 else
3817 {
3818 /* This isn't quite right, but it's the best we can do I
3819 * think... */
3820 if (color & 0x00ffffff)
3821 color = 0xff000000;
3822 else
3823 color = 0x00000000;
3824 }
3825 }
3826 image.setPixel (x, y, color);
3827 }
3828 /* Move one scanline forward. */
3829 pbSrcMask += cbSrcMaskLine;
3830 }
3831 /* Set the new cursor */
3832 QCursor cursor (QPixmap::fromImage (image),
3833 me->xHot(), me->yHot());
3834 viewport()->setCursor (cursor);
3835 ok = true;
3836 NOREF (srcShapePtrScan);
3837
3838#else
3839
3840# warning "port me"
3841
3842#endif
3843 if (!ok)
3844 viewport()->unsetCursor();
3845 }
3846 else
3847 {
3848 /*
3849 * We did not get any shape data
3850 */
3851 if (me->isVisible())
3852 {
3853 /*
3854 * We're supposed to make the last shape we got visible.
3855 * We don't support that for now...
3856 */
3857 /// @todo viewport()->setCursor (QCursor());
3858 }
3859 else
3860 {
3861 viewport()->setCursor (Qt::BlankCursor);
3862 }
3863 }
3864 mHideHostPointer = !me->isVisible();
3865}
3866
3867inline QRgb qRgbIntensity (QRgb rgb, int mul, int div)
3868{
3869 int r = qRed (rgb);
3870 int g = qGreen (rgb);
3871 int b = qBlue (rgb);
3872 return qRgb (mul * r / div, mul * g / div, mul * b / div);
3873}
3874
3875/* static */
3876void VBoxConsoleView::dimImage (QImage &img)
3877{
3878 for (int y = 0; y < img.height(); y ++) {
3879 if (y % 2) {
3880 if (img.depth() == 32) {
3881 for (int x = 0; x < img.width(); x ++) {
3882 int gray = qGray (img.pixel (x, y)) / 2;
3883 img.setPixel (x, y, qRgb (gray, gray, gray));
3884// img.setPixel (x, y, qRgbIntensity (img.pixel (x, y), 1, 2));
3885 }
3886 } else {
3887 ::memset (img.scanLine (y), 0, img.bytesPerLine());
3888 }
3889 } else {
3890 if (img.depth() == 32) {
3891 for (int x = 0; x < img.width(); x ++) {
3892 int gray = (2 * qGray (img.pixel (x, y))) / 3;
3893 img.setPixel (x, y, qRgb (gray, gray, gray));
3894// img.setPixel (x, y, qRgbIntensity (img.pixel(x, y), 2, 3));
3895 }
3896 }
3897 }
3898 }
3899}
3900
3901void VBoxConsoleView::doResizeHint (const QSize &aToSize)
3902{
3903 if (mGuestSupportsGraphics && mAutoresizeGuest)
3904 {
3905 /* If this slot is invoked directly then use the passed size
3906 * otherwise get the available size for the guest display.
3907 * We assume here that the centralWidget() contains this view only
3908 * and gives it all available space. */
3909 QSize sz (aToSize.isValid() ? aToSize : mMainWnd->centralWidget()->size());
3910 if (!aToSize.isValid())
3911 sz -= QSize (frameWidth() * 2, frameWidth() * 2);
3912 /* We only actually send the hint if
3913 * 1) the autoresize property is set to true and
3914 * 2) either an explicit new size was given (e.g. if the request
3915 * was triggered directly by a console resize event) or if no
3916 * explicit size was specified but a resize is flagged as being
3917 * needed (e.g. the autoresize was just enabled and the console
3918 * was resized while it was disabled). */
3919 if (mAutoresizeGuest &&
3920 (aToSize.isValid() || mDoResize))
3921 {
3922 LogFlowFunc (("Will suggest %d x %d\n", sz.width(), sz.height()));
3923
3924 /* Increase the maximum allowed size to the new size if needed. */
3925 setDesktopGeoHint (sz.width(), sz.height());
3926
3927 mConsole.GetDisplay().SetVideoModeHint (sz.width(), sz.height(), 0, 0);
3928 }
3929 /* we have resized now... */
3930 mDoResize = false;
3931 }
3932}
3933
3934
3935/* If the desktop geometry is set automatically, this will update it. */
3936void VBoxConsoleView::doResizeDesktop (int)
3937{
3938 calculateDesktopGeometry();
3939}
3940
3941/**
3942 * Remember a geometry hint sent by the console window. This is used to
3943 * determine the maximum supported guest resolution in the @a desktopGeometry
3944 * method. A hint will always override other restrictions.
3945 *
3946 * @param aWidth width of the resolution hint
3947 * @param aHeight height of the resolution hint
3948 */
3949void VBoxConsoleView::setDesktopGeoHint (int aWidth, int aHeight)
3950{
3951 LogFlowThisFunc (("aWidth=%d, aHeight=%d\n", aWidth, aHeight));
3952 mLastSizeHint = QRect (0, 0, aWidth, aHeight);
3953}
3954
3955/**
3956 * Do initial setup of desktop geometry restrictions on the guest framebuffer.
3957 * These determine the maximum size the guest framebuffer can take on.
3958 *
3959 * @note a hint from the host will always override these restrictions.
3960 *
3961 * @param aGeo Fixed - the guest has a fixed maximum framebuffer size
3962 * Automatic - we calculate the maximum size ourselves. The
3963 * calculations will not actually be done until
3964 * @a calculateDesktopGeometry is called, since
3965 * we don't initially have the information needed.
3966 * Any - any size is allowed
3967 * @param aWidth The maximum width for the guest screen or zero for no change
3968 * (only used for fixed geometry)
3969 * @param aHeight The maximum height for the guest screen or zero for no change
3970 * (only used for fixed geometry)
3971 */
3972void VBoxConsoleView::setDesktopGeometry (DesktopGeo aGeo, int aWidth, int aHeight)
3973{
3974 LogFlowThisFunc (("aGeo=%s, aWidth=%d, aHeight=%d\n",
3975 (aGeo == DesktopGeo_Fixed ? "Fixed" :
3976 aGeo == DesktopGeo_Automatic ? "Automatic" :
3977 aGeo == DesktopGeo_Any ? "Any" : "Invalid"),
3978 aWidth, aHeight));
3979 switch (aGeo)
3980 {
3981 case DesktopGeo_Fixed:
3982 mDesktopGeo = DesktopGeo_Fixed;
3983 if (aWidth != 0 && aHeight != 0)
3984 mDesktopGeometry = QRect (0, 0, aWidth, aHeight);
3985 else
3986 mDesktopGeometry = QRect (0, 0, 0, 0);
3987 setDesktopGeoHint (0, 0);
3988 break;
3989 case DesktopGeo_Automatic:
3990 mDesktopGeo = DesktopGeo_Automatic;
3991 mDesktopGeometry = QRect (0, 0, 0, 0);
3992 setDesktopGeoHint (0, 0);
3993 break;
3994 case DesktopGeo_Any:
3995 mDesktopGeo = DesktopGeo_Any;
3996 mDesktopGeometry = QRect (0, 0, 0, 0);
3997 break;
3998 default:
3999 AssertMsgFailed(("Invalid desktop geometry type %d\n", aGeo));
4000 mDesktopGeo = DesktopGeo_Invalid;
4001 }
4002}
4003
4004
4005/**
4006 * If we are in automatic mode, the geometry restrictions will be recalculated.
4007 * This is needed in particular on the first widget resize, as we can't
4008 * calculate them correctly before that.
4009 *
4010 * @note a hint from the host will always override these restrictions.
4011 * @note we can't do calculations on the fly when they are needed, because
4012 * they require querying the X server on X11 hosts and this must be done
4013 * from within the GUI thread, due to the single threadedness of Xlib.
4014 */
4015void VBoxConsoleView::calculateDesktopGeometry()
4016{
4017 LogFlowThisFunc (("Entering\n"));
4018 /* This method should not get called until we have initially set up the */
4019 Assert ((mDesktopGeo != DesktopGeo_Invalid));
4020 /* If we are not doing automatic geometry calculation then there is
4021 * nothing to do. */
4022 if (DesktopGeo_Automatic == mDesktopGeo)
4023 {
4024 /* Available geometry of the desktop. If the desktop is a single
4025 * screen, this will exclude space taken up by desktop taskbars
4026 * and things, but this is unfortunately not true for the more
4027 * complex case of a desktop spanning multiple screens. */
4028 QRect desktop = QApplication::desktop()->availableGeometry (this);
4029 /* The area taken up by the console window on the desktop,
4030 * including window frame, title and menu bar and whatnot. */
4031 QRect frame = mMainWnd->frameGeometry();
4032 /* The area taken up by the console window, minus all
4033 * decorations. */
4034 QRect window = mMainWnd->centralWidget()->geometry();
4035 /* To work out how big we can make the console window while still
4036 * fitting on the desktop, we calculate desktop - frame + window.
4037 * This works because the difference between frame and window
4038 * (or at least its width and height) is a constant. */
4039 mDesktopGeometry =
4040 QRect (0, 0, desktop.width() - frame.width() + window.width(),
4041 desktop.height() - frame.height() + window.height());
4042 LogFlowThisFunc (("Setting %d, %d\n", mDesktopGeometry.width(),
4043 mDesktopGeometry.height()));
4044 }
4045}
4046
4047/**
4048 * Sets the minimum size restriction depending on the auto-resize feature
4049 * state and the current rendering mode.
4050 *
4051 * Currently, the restriction is set only in SDL mode and only when the
4052 * auto-resize feature is inactive. We need to do that because we cannot
4053 * correctly draw in a scrolled window in SDL mode.
4054 *
4055 * In all other modes, or when auto-resize is in force, this function does
4056 * nothing.
4057 */
4058void VBoxConsoleView::maybeRestrictMinimumSize()
4059{
4060 if (mode == VBoxDefs::SDLMode)
4061 {
4062 if (!mGuestSupportsGraphics || !mAutoresizeGuest)
4063 setMinimumSize (sizeHint());
4064 else
4065 setMinimumSize (0, 0);
4066 }
4067}
4068
4069int VBoxConsoleView::contentsWidth() const
4070{
4071 return mFrameBuf->width();
4072}
4073
4074int VBoxConsoleView::contentsHeight() const
4075{
4076 return mFrameBuf->height();
4077}
4078
4079void VBoxConsoleView::updateSliders()
4080{
4081 QSize p = viewport()->size();
4082 QSize m = maximumViewportSize();
4083
4084 QSize v = QSize (mFrameBuf->width(), mFrameBuf->height());
4085 /* no scroll bars needed */
4086 if (m.expandedTo(v) == m)
4087 p = m;
4088
4089 horizontalScrollBar()->setRange(0, v.width() - p.width());
4090 verticalScrollBar()->setRange(0, v.height() - p.height());
4091 horizontalScrollBar()->setPageStep(p.width());
4092 verticalScrollBar()->setPageStep(p.height());
4093}
4094
4095void VBoxConsoleView::requestToResize (const QSize &aSize)
4096{
4097 mIgnoreFrameBufferResize = true;
4098 mNormalSize = aSize;
4099}
4100
4101#if defined(Q_WS_MAC)
4102
4103void VBoxConsoleView::updateDockIcon()
4104{
4105 if (mDockIconEnabled)
4106 {
4107 if (!mPausedShot.isNull())
4108 {
4109 CGImageRef pauseImg = ::darwinToCGImageRef (&mPausedShot);
4110 /* Use the pause image as background */
4111 mDockIconPreview->updateDockPreview (pauseImg);
4112 CGImageRelease (pauseImg);
4113 }
4114 else
4115 {
4116# if defined (VBOX_GUI_USE_QUARTZ2D)
4117 if (mode == VBoxDefs::Quartz2DMode)
4118 {
4119 /* If the render mode is Quartz2D we could use the CGImageRef
4120 * of the framebuffer for the dock icon creation. This saves
4121 * some conversion time. */
4122 mDockIconPreview->updateDockPreview (static_cast <VBoxQuartz2DFrameBuffer *> (mFrameBuf)->imageRef());
4123 }
4124 else
4125# endif
4126 /* In image mode we have to create the image ref out of the
4127 * framebuffer */
4128 mDockIconPreview->updateDockPreview (mFrameBuf);
4129 }
4130 }
4131}
4132
4133void VBoxConsoleView::updateDockOverlay()
4134{
4135 /* Only to an update to the realtime preview if this is enabled by the user
4136 * & we are in an state where the framebuffer is likely valid. Otherwise to
4137 * the overlay stuff only. */
4138 if (mDockIconEnabled &&
4139 (mLastState == KMachineState_Running ||
4140 mLastState == KMachineState_Paused ||
4141 mLastState == KMachineState_Restoring ||
4142 mLastState == KMachineState_Saving))
4143 updateDockIcon();
4144 else
4145 mDockIconPreview->updateDockOverlay();
4146}
4147
4148/**
4149 * Wrapper for SetMouseCoalescingEnabled().
4150 *
4151 * Called by eventFilter() and darwinGrabKeyboardEvents().
4152 *
4153 * @param aOn Switch it on (true) or off (false).
4154 */
4155void VBoxConsoleView::setMouseCoalescingEnabled (bool aOn)
4156{
4157 /* Enable mouse event compression if we leave the VM view. This
4158 is necessary for having smooth resizing of the VM/other
4159 windows.
4160 Disable mouse event compression if we enter the VM view. So
4161 all mouse events are registered in the VM. Only do this if
4162 the keyboard/mouse is grabbed (this is when we have a valid
4163 event handler). */
4164 if (aOn || mKeyboardGrabbed)
4165 ::darwinSetMouseCoalescingEnabled (aOn);
4166}
4167
4168#endif /* Q_WS_MAC */
4169
Note: See TracBrowser for help on using the repository browser.

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