VirtualBox

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

Last change on this file since 5999 was 5999, checked in by vboxsync, 17 years ago

The Giant CDDL Dual-License Header Change.

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 107.9 KB
Line 
1/** @file
2 *
3 * VBox frontends: Qt GUI ("VirtualBox"):
4 * VBoxConsoleView class implementation
5 */
6
7/*
8 * Copyright (C) 22006-2007 innotek GmbH
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
19#include "VBoxConsoleView.h"
20#include "VBoxConsoleWnd.h"
21
22#include "VBoxFrameBuffer.h"
23#include "VBoxGlobal.h"
24#include "VBoxProblemReporter.h"
25
26#ifdef Q_WS_PM
27#include "QIHotKeyEdit.h"
28#endif
29
30#include <qapplication.h>
31#include <qstatusbar.h>
32#include <qlabel.h>
33#include <qpainter.h>
34#include <qpixmap.h>
35#include <qimage.h>
36#include <qbitmap.h>
37#include <qcursor.h>
38#include <qthread.h>
39
40#include <qmenudata.h>
41#include <qmenubar.h>
42#include <qwidgetlist.h>
43#include <qtimer.h>
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// We need to capture some X11 events directly which
56// requires the XEvent structure to be defined. However,
57// including the Xlib header file will cause some nasty
58// conflicts with Qt. Therefore we use the following hack
59// to redefine those conflicting identifiers.
60#define XK_XKB_KEYS
61#define XK_MISCELLANY
62#include <X11/Xlib.h>
63#include <X11/Xutil.h>
64#include <X11/XKBlib.h>
65#include <X11/keysym.h>
66#ifdef KeyPress
67const int XFocusOut = FocusOut;
68const int XFocusIn = FocusIn;
69const int XKeyPress = KeyPress;
70const int XKeyRelease = KeyRelease;
71#undef KeyRelease
72#undef KeyPress
73#undef FocusOut
74#undef FocusIn
75#endif
76#include "XKeyboard.h"
77#ifndef VBOX_WITHOUT_XCURSOR
78# include <X11/Xcursor/Xcursor.h>
79#endif
80#endif // Q_WS_X11
81
82#if defined (Q_WS_MAC)
83# include "DarwinKeyboard.h"
84# include "DarwinCursor.h"
85# ifdef VBOX_WITH_HACKED_QT
86# include "QIApplication.h"
87# endif
88# include <VBox/err.h>
89#endif /* defined (Q_WS_MAC) */
90
91#if defined (VBOX_GUI_USE_REFRESH_TIMER)
92enum { UPDATE_FREQ = 1000 / 60 }; // a-la 60Hz
93#endif
94
95#if defined (Q_WS_WIN32)
96
97static HHOOK g_kbdhook = NULL;
98static VBoxConsoleView *g_view = 0;
99
100LRESULT CALLBACK VBoxConsoleView::lowLevelKeyboardProc (int nCode,
101 WPARAM wParam, LPARAM lParam)
102{
103 Assert (g_view);
104 if (g_view && nCode == HC_ACTION &&
105 g_view->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
115# ifndef VBOX_WITH_HACKED_QT
116/**
117 * Event handler callback for Mac OS X.
118 */
119/* static */
120pascal OSStatus VBoxConsoleView::darwinEventHandlerProc (EventHandlerCallRef inHandlerCallRef,
121 EventRef inEvent, void *inUserData)
122{
123 VBoxConsoleView *view = (VBoxConsoleView *)inUserData;
124 UInt32 EventClass = ::GetEventClass (inEvent);
125 if (EventClass == kEventClassKeyboard)
126 {
127 if (view->darwinKeyboardEvent (inEvent))
128 return 0;
129 }
130 /*
131 * Command-H and Command-Q aren't properly disabled yet, and it's still
132 * possible to use the left command key to invoke them when the keyboard
133 * is captured. We discard the events these if the keyboard is captured
134 * as a half measure to prevent unexpected behaviour. However, we don't
135 * get any key down/up events, so these combinations are dead to the guest...
136 */
137 else if (EventClass == kEventClassCommand)
138 {
139 if (view->kbd_captured)
140 return 0;
141 }
142 return ::CallNextEventHandler (inHandlerCallRef, inEvent);
143}
144
145# else /* VBOX_WITH_HACKED_QT */
146
147/**
148 * Event handler callback for Mac OS X.
149 */
150/* static */
151bool VBoxConsoleView::macEventFilter (EventRef inEvent, void *inUserData)
152{
153 VBoxConsoleView *view = (VBoxConsoleView *)inUserData;
154 UInt32 EventClass = ::GetEventClass (inEvent);
155 if (EventClass == kEventClassKeyboard)
156 {
157 if (view->darwinKeyboardEvent (inEvent))
158 return true;
159 }
160 return false;
161}
162# endif /* VBOX_WITH_HACKED_QT */
163
164#endif /* Q_WS_MAC */
165
166/** Guest mouse pointer shape change event. */
167class MousePointerChangeEvent : public QEvent
168{
169public:
170 MousePointerChangeEvent (bool visible, bool alpha, uint xhot, uint yhot,
171 uint width, uint height,
172 const uchar *shape) :
173 QEvent ((QEvent::Type) VBoxDefs::MousePointerChangeEventType),
174 vis (visible), alph (alpha), xh (xhot), yh (yhot), w (width), h (height),
175 data (NULL)
176 {
177 // make a copy of shape
178 uint dataSize = ((((width + 7) / 8 * height) + 3) & ~3) + width * 4 * height;
179
180 if (shape) {
181 data = new uchar [dataSize];
182 memcpy ((void *) data, (void *) shape, dataSize);
183 }
184 }
185 ~MousePointerChangeEvent()
186 {
187 if (data) delete[] data;
188 }
189 bool isVisible() const { return vis; }
190 bool hasAlpha() const { return alph; }
191 uint xHot() const { return xh; }
192 uint yHot() const { return yh; }
193 uint width() const { return w; }
194 uint height() const { return h; }
195 const uchar *shapeData() const { return data; }
196private:
197 bool vis, alph;
198 uint xh, yh, w, h;
199 const uchar *data;
200};
201
202/** Guest mouse absolute positioning capability change event. */
203class MouseCapabilityEvent : public QEvent
204{
205public:
206 MouseCapabilityEvent (bool supportsAbsolute, bool needsHostCursor) :
207 QEvent ((QEvent::Type) VBoxDefs::MouseCapabilityEventType),
208 can_abs (supportsAbsolute),
209 needs_host_cursor (needsHostCursor) {}
210 bool supportsAbsolute() const { return can_abs; }
211 bool needsHostCursor() const { return needs_host_cursor; }
212private:
213 bool can_abs;
214 bool needs_host_cursor;
215};
216
217/** Machine state change. */
218class StateChangeEvent : public QEvent
219{
220public:
221 StateChangeEvent (CEnums::MachineState state) :
222 QEvent ((QEvent::Type) VBoxDefs::MachineStateChangeEventType),
223 s (state) {}
224 CEnums::MachineState machineState() const { return s; }
225private:
226 CEnums::MachineState s;
227};
228
229/** Guest Additions property changes. */
230class GuestAdditionsEvent : public QEvent
231{
232public:
233 GuestAdditionsEvent (const QString &aOsTypeId,
234 const QString &aAddVersion,
235 bool aAddActive,
236 bool aSupportsSeamless) :
237 QEvent ((QEvent::Type) VBoxDefs::AdditionsStateChangeEventType),
238 mOsTypeId (aOsTypeId), mAddVersion (aAddVersion),
239 mAddActive (aAddActive), mSupportsSeamless (aSupportsSeamless) {}
240 const QString &osTypeId() const { return mOsTypeId; }
241 const QString &additionVersion() const { return mAddVersion; }
242 bool additionActive() const { return mAddActive; }
243 bool supportsSeamless() const { return mSupportsSeamless; }
244private:
245 QString mOsTypeId;
246 QString mAddVersion;
247 bool mAddActive;
248 bool mSupportsSeamless;
249};
250
251/** DVD/FD change event */
252class MediaChangeEvent : public QEvent
253{
254public:
255 MediaChangeEvent (VBoxDefs::DiskType aType)
256 : QEvent ((QEvent::Type) VBoxDefs::MediaChangeEventType)
257 , mType (aType) {}
258 VBoxDefs::DiskType diskType() const { return mType; }
259private:
260 VBoxDefs::DiskType mType;
261};
262
263/** Menu activation event */
264class ActivateMenuEvent : public QEvent
265{
266public:
267 ActivateMenuEvent (QMenuData *menuData, uint index) :
268 QEvent ((QEvent::Type) VBoxDefs::ActivateMenuEventType),
269 md (menuData), i (index) {}
270 QMenuData *menuData() const { return md; }
271 uint index() const { return i; }
272private:
273 QMenuData *md;
274 uint i;
275};
276
277/** VM Runtime error event */
278class RuntimeErrorEvent : public QEvent
279{
280public:
281 RuntimeErrorEvent (bool aFatal, const QString &aErrorID,
282 const QString &aMessage) :
283 QEvent ((QEvent::Type) VBoxDefs::RuntimeErrorEventType),
284 mFatal (aFatal), mErrorID (aErrorID), mMessage (aMessage) {}
285 bool fatal() const { return mFatal; }
286 QString errorID() const { return mErrorID; }
287 QString message() const { return mMessage; }
288private:
289 bool mFatal;
290 QString mErrorID;
291 QString mMessage;
292};
293
294/** Modifier key change event */
295class ModifierKeyChangeEvent : public QEvent
296{
297public:
298 ModifierKeyChangeEvent (bool fNumLock, bool fCapsLock, bool fScrollLock) :
299 QEvent ((QEvent::Type) VBoxDefs::ModifierKeyChangeEventType),
300 mNumLock (fNumLock), mCapsLock (fCapsLock), mScrollLock (fScrollLock) {}
301 bool numLock() const { return mNumLock; }
302 bool capsLock() const { return mCapsLock; }
303 bool scrollLock() const { return mScrollLock; }
304private:
305 bool mNumLock, mCapsLock, mScrollLock;
306};
307
308/** Network adapter change event */
309class NetworkAdapterChangeEvent : public QEvent
310{
311public:
312 NetworkAdapterChangeEvent (INetworkAdapter *aAdapter) :
313 QEvent ((QEvent::Type) VBoxDefs::NetworkAdapterChangeEventType),
314 mAdapter (aAdapter) {}
315 INetworkAdapter* networkAdapter() { return mAdapter; }
316private:
317 INetworkAdapter *mAdapter;
318};
319
320/** USB controller state change event */
321class USBControllerStateChangeEvent : public QEvent
322{
323public:
324 USBControllerStateChangeEvent()
325 : QEvent ((QEvent::Type) VBoxDefs::USBCtlStateChangeEventType) {}
326};
327
328/** USB device state change event */
329class USBDeviceStateChangeEvent : public QEvent
330{
331public:
332 USBDeviceStateChangeEvent (const CUSBDevice &aDevice, bool aAttached,
333 const CVirtualBoxErrorInfo &aError) :
334 QEvent ((QEvent::Type) VBoxDefs::USBDeviceStateChangeEventType),
335 mDevice (aDevice), mAttached (aAttached), mError (aError) {}
336 CUSBDevice device() const { return mDevice; }
337 bool attached() const { return mAttached; }
338 CVirtualBoxErrorInfo error() const { return mError; }
339private:
340 CUSBDevice mDevice;
341 bool mAttached;
342 CVirtualBoxErrorInfo mError;
343};
344
345//
346// VBoxConsoleCallback class
347/////////////////////////////////////////////////////////////////////////////
348
349class VBoxConsoleCallback : public IConsoleCallback
350{
351public:
352
353 VBoxConsoleCallback (VBoxConsoleView *v) {
354#if defined (Q_WS_WIN)
355 mRefCnt = 0;
356#endif
357 mView = v;
358 }
359
360 virtual ~VBoxConsoleCallback() {}
361
362 NS_DECL_ISUPPORTS
363
364#if defined (Q_WS_WIN)
365 STDMETHOD_(ULONG, AddRef)() {
366 return ::InterlockedIncrement (&mRefCnt);
367 }
368 STDMETHOD_(ULONG, Release)()
369 {
370 long cnt = ::InterlockedDecrement (&mRefCnt);
371 if (cnt == 0)
372 delete this;
373 return cnt;
374 }
375 STDMETHOD(QueryInterface) (REFIID riid , void **ppObj)
376 {
377 if (riid == IID_IUnknown) {
378 *ppObj = this;
379 AddRef();
380 return S_OK;
381 }
382 if (riid == IID_IConsoleCallback) {
383 *ppObj = this;
384 AddRef();
385 return S_OK;
386 }
387 *ppObj = NULL;
388 return E_NOINTERFACE;
389 }
390#endif
391
392 STDMETHOD(OnMousePointerShapeChange) (BOOL visible, BOOL alpha,
393 ULONG xhot, ULONG yhot,
394 ULONG width, ULONG height,
395 BYTE *shape)
396 {
397 QApplication::postEvent (mView,
398 new MousePointerChangeEvent (visible, alpha,
399 xhot, yhot,
400 width, height, shape));
401 return S_OK;
402 }
403
404 STDMETHOD(OnMouseCapabilityChange)(BOOL supportsAbsolute, BOOL needsHostCursor)
405 {
406 QApplication::postEvent (mView,
407 new MouseCapabilityEvent (supportsAbsolute,
408 needsHostCursor));
409 return S_OK;
410 }
411
412 STDMETHOD(OnKeyboardLedsChange)(BOOL fNumLock, BOOL fCapsLock, BOOL fScrollLock)
413 {
414 QApplication::postEvent (mView,
415 new ModifierKeyChangeEvent (fNumLock, fCapsLock,
416 fScrollLock));
417 return S_OK;
418 }
419
420 STDMETHOD(OnStateChange)(MachineState_T machineState)
421 {
422 LogFlowFunc (("machineState=%d\n", machineState));
423 QApplication::postEvent (mView,
424 new StateChangeEvent (
425 (CEnums::MachineState) machineState));
426 return S_OK;
427 }
428
429 STDMETHOD(OnAdditionsStateChange)()
430 {
431 CGuest guest = mView->console().GetGuest();
432 LogFlowFunc (("ver=%s, active=%d\n",
433 guest.GetAdditionsVersion().latin1(),
434 guest.GetAdditionsActive()));
435 QApplication::postEvent (mView,
436 new GuestAdditionsEvent (
437 guest.GetOSTypeId(),
438 guest.GetAdditionsVersion(),
439 guest.GetAdditionsActive(),
440 guest.GetSupportsSeamless()));
441 return S_OK;
442 }
443
444 STDMETHOD(OnDVDDriveChange)()
445 {
446 LogFlowFunc (("DVD Drive changed\n"));
447 QApplication::postEvent (mView, new MediaChangeEvent (VBoxDefs::CD));
448 return S_OK;
449 }
450
451 STDMETHOD(OnFloppyDriveChange)()
452 {
453 LogFlowFunc (("Floppy Drive changed\n"));
454 QApplication::postEvent (mView, new MediaChangeEvent (VBoxDefs::FD));
455 return S_OK;
456 }
457
458 STDMETHOD(OnNetworkAdapterChange) (INetworkAdapter *aNetworkAdapter)
459 {
460 QApplication::postEvent (mView,
461 new NetworkAdapterChangeEvent (aNetworkAdapter));
462 return S_OK;
463 }
464
465 STDMETHOD(OnSerialPortChange) (ISerialPort *aSerialPort)
466 {
467 NOREF(aSerialPort);
468 return S_OK;
469 }
470
471 STDMETHOD(OnParallelPortChange) (IParallelPort *aParallelPort)
472 {
473 NOREF(aParallelPort);
474 return S_OK;
475 }
476
477 STDMETHOD(OnVRDPServerChange)()
478 {
479 return S_OK;
480 }
481
482 STDMETHOD(OnUSBControllerChange)()
483 {
484 QApplication::postEvent (mView,
485 new USBControllerStateChangeEvent());
486 return S_OK;
487 }
488
489 STDMETHOD(OnUSBDeviceStateChange)(IUSBDevice *aDevice, BOOL aAttached,
490 IVirtualBoxErrorInfo *aError)
491 {
492 QApplication::postEvent (mView,
493 new USBDeviceStateChangeEvent (
494 CUSBDevice (aDevice),
495 bool (aAttached),
496 CVirtualBoxErrorInfo (aError)));
497 return S_OK;
498 }
499
500 STDMETHOD(OnSharedFolderChange) (Scope_T aScope)
501 {
502 NOREF(aScope);
503 QApplication::postEvent (mView,
504 new QEvent ((QEvent::Type)
505 VBoxDefs::SharedFolderChangeEventType));
506 return S_OK;
507 }
508
509 STDMETHOD(OnRuntimeError)(BOOL fatal, IN_BSTRPARAM id, IN_BSTRPARAM message)
510 {
511 QApplication::postEvent (mView,
512 new RuntimeErrorEvent (!!fatal,
513 QString::fromUcs2 (id),
514 QString::fromUcs2 (message)));
515 return S_OK;
516 }
517
518 STDMETHOD(OnCanShowWindow) (BOOL *canShow)
519 {
520 if (!canShow)
521 return E_POINTER;
522
523 /* as long as there is VBoxConsoleView (which creates/destroys us), it
524 * can be shown */
525 *canShow = TRUE;
526 return S_OK;
527 }
528
529 STDMETHOD(OnShowWindow) (ULONG64 *winId)
530 {
531 if (!winId)
532 return E_POINTER;
533
534#if defined (Q_WS_MAC)
535 /*
536 * Let's try the simple approach first - grab the focus.
537 * Getting a window out of the dock (minimized or whatever it's called)
538 * needs to be done on the GUI thread, so post it a note.
539 */
540 *winId = 0;
541 if (!mView)
542 return S_OK;
543
544 ProcessSerialNumber psn = { 0, kCurrentProcess };
545 OSErr rc = ::SetFrontProcess (&psn);
546 if (!rc)
547 QApplication::postEvent (mView, new QEvent ((QEvent::Type)VBoxDefs::ShowWindowEventType));
548 else
549 {
550 /*
551 * It failed for some reason, send the other process our PSN so it can try.
552 * (This is just a precaution should Mac OS X start imposing the same sensible
553 * focus stealing restrictions that other window managers implement.)
554 */
555 AssertMsgFailed(("SetFrontProcess -> %#x\n", rc));
556 if (::GetCurrentProcess (&psn))
557 *winId = RT_MAKE_U64 (psn.lowLongOfPSN, psn.highLongOfPSN);
558 }
559
560#else
561 /* Return the ID of the top-level console window. */
562 *winId = (ULONG64) mView->topLevelWidget()->winId();
563#endif
564
565 return S_OK;
566 }
567
568protected:
569
570 VBoxConsoleView *mView;
571
572#if defined (Q_WS_WIN)
573private:
574 long mRefCnt;
575#endif
576};
577
578#if !defined (Q_WS_WIN)
579NS_DECL_CLASSINFO (VBoxConsoleCallback)
580NS_IMPL_THREADSAFE_ISUPPORTS1_CI (VBoxConsoleCallback, IConsoleCallback)
581#endif
582
583//
584// VBoxConsoleView class
585/////////////////////////////////////////////////////////////////////////////
586
587/** @class VBoxConsoleView
588 *
589 * The VBoxConsoleView class is a widget that implements a console
590 * for the running virtual machine.
591 */
592
593VBoxConsoleView::VBoxConsoleView (VBoxConsoleWnd *mainWnd,
594 const CConsole &console,
595 VBoxDefs::RenderMode rm,
596 QWidget *parent, const char *name, WFlags f)
597 : QScrollView (parent, name, f | WStaticContents | WNoAutoErase)
598 , mMainWnd (mainWnd)
599 , mConsole (console)
600 , gs (vboxGlobal().settings())
601 , attached (false)
602 , kbd_captured (false)
603 , mouse_captured (false)
604 , mouse_absolute (false)
605 , mouse_integration (true)
606 , mDisableAutoCapture (false)
607 , mIsHostkeyPressed (false)
608 , mIsHostkeyAlone (false)
609 , mIgnoreMainwndResize (true)
610 , mAutoresizeGuest (false)
611 , mIsAdditionsActive (false)
612 , mNumLock (false)
613 , mScrollLock (false)
614 , mCapsLock (false)
615 , muNumLockAdaptionCnt (2)
616 , muCapsLockAdaptionCnt (2)
617 , mode (rm)
618#if defined(Q_WS_WIN)
619 , mAlphaCursor (NULL)
620#endif
621#if defined(Q_WS_MAC)
622# ifndef VBOX_WITH_HACKED_QT
623 , mDarwinEventHandlerRef (NULL)
624# endif
625 , mDarwinKeyModifiers (0)
626#endif
627{
628 Assert (!mConsole.isNull() &&
629 !mConsole.GetDisplay().isNull() &&
630 !mConsole.GetKeyboard().isNull() &&
631 !mConsole.GetMouse().isNull());
632
633 /* enable MouseMove events */
634 viewport()->setMouseTracking (true);
635
636 /*
637 * QScrollView does the below on its own, but let's do it anyway
638 * for the case it will not do it in the future.
639 */
640 viewport()->installEventFilter (this);
641
642 /* to fix some focus issues */
643 mMainWnd->menuBar()->installEventFilter (this);
644
645 /* we want to be notified on some parent's events */
646 mMainWnd->installEventFilter (this);
647
648#ifdef Q_WS_X11
649 /* initialize the X keyboard subsystem */
650 initXKeyboard (this->x11Display());
651#endif
652
653 ::memset (mPressedKeys, 0, SIZEOF_ARRAY (mPressedKeys));
654
655 resize_hint_timer = new QTimer (this);
656 connect (resize_hint_timer, SIGNAL (timeout()),
657 this, SLOT (doResizeHint()));
658
659 mToggleFSModeTimer = new QTimer (this);
660 connect (mToggleFSModeTimer, SIGNAL (timeout()),
661 this, SIGNAL (resizeHintDone()));
662
663 /* setup rendering */
664
665 CDisplay display = mConsole.GetDisplay();
666 Assert (!display.isNull());
667
668#if defined (VBOX_GUI_USE_REFRESH_TIMER)
669 tid = 0;
670#endif
671 mFrameBuf = 0;
672
673 LogFlowFunc (("Rendering mode: %d\n", mode));
674
675 switch (mode)
676 {
677#if defined (VBOX_GUI_USE_REFRESH_TIMER)
678 case VBoxDefs::TimerMode:
679 display.SetupInternalFramebuffer (32);
680 tid = startTimer (UPDATE_FREQ);
681 break;
682#endif
683#if defined (VBOX_GUI_USE_QIMAGE)
684 case VBoxDefs::QImageMode:
685 mFrameBuf = new VBoxQImageFrameBuffer (this);
686 break;
687#endif
688#if defined (VBOX_GUI_USE_SDL)
689 case VBoxDefs::SDLMode:
690 mFrameBuf = new VBoxSDLFrameBuffer (this);
691 /*
692 * disable scrollbars because we cannot correctly draw in a
693 * scrolled window using SDL
694 */
695 horizontalScrollBar()->setEnabled (false);
696 verticalScrollBar()->setEnabled (false);
697 break;
698#endif
699#if defined (VBOX_GUI_USE_DDRAW)
700 case VBoxDefs::DDRAWMode:
701 mFrameBuf = new VBoxDDRAWFrameBuffer (this);
702 break;
703#endif
704 default:
705 AssertReleaseMsgFailed (("Render mode must be valid: %d\n", mode));
706 LogRel (("Invalid render mode: %d\n", mode));
707 qApp->exit (1);
708 break;
709 }
710
711#if defined (VBOX_GUI_USE_DDRAW)
712 if (!mFrameBuf || mFrameBuf->address () == NULL)
713 {
714 if (mFrameBuf)
715 delete mFrameBuf;
716 mode = VBoxDefs::QImageMode;
717 mFrameBuf = new VBoxQImageFrameBuffer (this);
718 }
719#endif
720
721 if (mFrameBuf)
722 {
723 mFrameBuf->AddRef();
724 display.RegisterExternalFramebuffer (CFramebuffer (mFrameBuf));
725 }
726
727 /* setup the callback */
728 mCallback = CConsoleCallback (new VBoxConsoleCallback (this));
729 mConsole.RegisterCallback (mCallback);
730 AssertWrapperOk (mConsole);
731
732 viewport()->setEraseColor (black);
733
734 setSizePolicy (QSizePolicy (QSizePolicy::Maximum, QSizePolicy::Maximum));
735 setMaximumSize (sizeHint());
736
737 setFocusPolicy (WheelFocus);
738
739#if defined (VBOX_GUI_DEBUG) && defined (VBOX_GUI_FRAMEBUF_STAT)
740 VMCPUTimer::calibrate (200);
741#endif
742
743#if defined (Q_WS_WIN)
744 g_view = this;
745#endif
746
747#ifdef Q_WS_MAC
748 DarwinCursorClearHandle (&mDarwinCursor);
749#endif
750}
751
752VBoxConsoleView::~VBoxConsoleView()
753{
754#if defined (Q_WS_WIN)
755 if (g_kbdhook)
756 UnhookWindowsHookEx (g_kbdhook);
757 g_view = 0;
758 if (mAlphaCursor)
759 DestroyIcon (mAlphaCursor);
760#endif
761
762#if defined (VBOX_GUI_USE_REFRESH_TIMER)
763 if (tid)
764 killTimer (tid);
765#endif
766 if (mFrameBuf)
767 {
768 /* detach our framebuffer from Display */
769 CDisplay display = mConsole.GetDisplay();
770 Assert (!display.isNull());
771 display.SetupInternalFramebuffer (0);
772 /* release the reference */
773 mFrameBuf->Release();
774 }
775
776 mConsole.UnregisterCallback (mCallback);
777}
778
779//
780// Public members
781/////////////////////////////////////////////////////////////////////////////
782
783QSize VBoxConsoleView::sizeHint() const
784{
785#if defined (VBOX_GUI_USE_REFRESH_TIMER)
786 if (mode == VBoxDefs::TimerMode)
787 {
788 CDisplay display = mConsole.GetDisplay();
789 return QSize (display.GetWidth() + frameWidth() * 2,
790 display.GetHeight() + frameWidth() * 2);
791 }
792 else
793#endif
794 {
795 return QSize (mFrameBuf->width() + frameWidth() * 2,
796 mFrameBuf->height() + frameWidth() * 2);
797 }
798}
799
800/**
801 * Attaches this console view to the managed virtual machine.
802 *
803 * @note This method is not really necessary these days -- the only place where
804 * it gets called is VBoxConsole::openView(), right after powering the
805 * VM up. We leave it as is just in case attaching/detaching will become
806 * necessary some day (there are useful attached checks everywhere in the
807 * code).
808 */
809void VBoxConsoleView::attach()
810{
811 if (!attached)
812 {
813 attached = true;
814 }
815}
816
817/**
818 * Detaches this console view from the VM. Must be called to indicate
819 * that the virtual machine managed by this instance will be no more valid
820 * after this call.
821 *
822 * @note This method is not really necessary these days -- the only place where
823 * it gets called is VBoxConsole::closeView(), when the VM is powered
824 * down, before deleting VBoxConsoleView. We leave it as is just in case
825 * attaching/detaching will become necessary some day (there are useful
826 * attached checks everywhere in the code).
827 */
828void VBoxConsoleView::detach()
829{
830 if (attached)
831 {
832 /* reuse the focus event handler to uncapture everything */
833 focusEvent (false);
834 attached = false;
835 }
836}
837
838/**
839 * Resizes the toplevel widget to fit the console view w/o scrollbars.
840 * If adjustPosition is true and such resize is not possible (because the
841 * console view size is lagrer then the available screen space) the toplevel
842 * widget is resized and moved to become as large as possible while staying
843 * fully visible.
844 */
845void VBoxConsoleView::normalizeGeometry (bool adjustPosition /* = false */)
846{
847 /* Make no normalizeGeometry in case we are in manual resize
848 * mode or main window is maximized */
849 if (mMainWnd->isMaximized() || mMainWnd->isFullScreen())
850 return;
851
852 QWidget *tlw = topLevelWidget();
853
854 /* calculate client window offsets */
855 QRect fr = tlw->frameGeometry();
856 QRect r = tlw->geometry();
857 int dl = r.left() - fr.left();
858 int dt = r.top() - fr.top();
859 int dr = fr.right() - r.right();
860 int db = fr.bottom() - r.bottom();
861
862 /* get the best size w/o scroll bars */
863 QSize s = tlw->sizeHint();
864
865 /* resize the frame to fit the contents */
866 s -= tlw->size();
867 fr.rRight() += s.width();
868 fr.rBottom() += s.height();
869
870 if (adjustPosition)
871 {
872 QRect ar = QApplication::desktop()->availableGeometry (tlw->pos());
873 fr = VBoxGlobal::normalizeGeometry (
874 fr, ar, mode != VBoxDefs::SDLMode /* canResize */);
875 }
876
877#if 0
878 /* center the frame on the desktop */
879 fr.moveCenter (ar.center());
880#endif
881
882 /* finally, set the frame geometry */
883 tlw->setGeometry (fr.left() + dl, fr.top() + dt,
884 fr.width() - dl - dr, fr.height() - dt - db);
885}
886
887/**
888 * Pauses or resumes the VM execution.
889 */
890bool VBoxConsoleView::pause (bool on)
891{
892 /* QAction::setOn() emits the toggled() signal, so avoid recursion when
893 * QAction::setOn() is called from VBoxConsoleWnd::updateMachineState() */
894 if (isPaused() == on)
895 return true;
896
897 if (on)
898 mConsole.Pause();
899 else
900 mConsole.Resume();
901
902 bool ok = mConsole.isOk();
903 if (!ok)
904 {
905 if (on)
906 vboxProblem().cannotPauseMachine (mConsole);
907 else
908 vboxProblem().cannotResumeMachine (mConsole);
909 }
910
911 return ok;
912}
913
914/**
915 * Temporarily disables the mouse integration (or enables it back).
916 */
917void VBoxConsoleView::setMouseIntegrationEnabled (bool enabled)
918{
919 if (mouse_integration == enabled)
920 return;
921
922 if (mouse_absolute)
923 captureMouse (!enabled, false);
924
925 /* Hiding host cursor in case we are entering mouse integration
926 * mode until it's shape is set to the guest cursor shape in
927 * OnMousePointerShapeChange event handler.
928 *
929 * This is necessary to avoid double-cursor issue when both the
930 * guest and the host cursors are displayed in one place one-above-one.
931 *
932 * This is a workaround because the correct decision is to notify
933 * the Guest Additions about we are entering the mouse integration
934 * mode. The GuestOS should hide it's cursor to allow using of
935 * host cursor for the guest's manipulation.
936 *
937 * This notification is not possible right now due to there is
938 * no the required API. */
939 if (enabled)
940 viewport()->setCursor (QCursor (BlankCursor));
941
942 mouse_integration = enabled;
943
944 emitMouseStateChanged();
945}
946
947void VBoxConsoleView::setAutoresizeGuest (bool on)
948{
949 if (mAutoresizeGuest != on)
950 {
951 mAutoresizeGuest = on;
952
953 maybeRestrictMinimumSize();
954
955 if (mIsAdditionsActive && mAutoresizeGuest)
956 doResizeHint();
957 }
958}
959
960/**
961 * This method is called by VBoxConsoleWnd after it does everything necessary
962 * on its side to go to or from fullscreen, but before it is shown.
963 */
964void VBoxConsoleView::onFullscreenChange (bool /* on */)
965{
966 /* Nothing to do here so far */
967}
968
969/**
970 * Notify the console scroll-view about the console-window is opened.
971 */
972void VBoxConsoleView::onViewOpened()
973{
974 /* Variable mIgnoreMainwndResize was initially "true" to ignore QT
975 * initial resize event in case of auto-resize feature is on.
976 * Currently, initial resize event is already processed, so we set
977 * mIgnoreMainwndResize to "false" to process all further resize
978 * events as user-initiated window resize events. */
979 mIgnoreMainwndResize = false;
980}
981
982//
983// Protected Events
984/////////////////////////////////////////////////////////////////////////////
985
986bool VBoxConsoleView::event (QEvent *e)
987{
988 if (attached)
989 {
990 switch (e->type())
991 {
992 case QEvent::FocusIn:
993 {
994 if (isRunning())
995 focusEvent (true);
996 break;
997 }
998 case QEvent::FocusOut:
999 {
1000 if (isRunning())
1001 focusEvent (false);
1002 else
1003 {
1004 /* release the host key even when paused */
1005 mIsHostkeyPressed = false;
1006 emitKeyboardStateChanged();
1007 }
1008 break;
1009 }
1010
1011 case VBoxDefs::ResizeEventType:
1012 {
1013 bool oldIgnoreMainwndResize = mIgnoreMainwndResize;
1014 mIgnoreMainwndResize = true;
1015
1016 VBoxResizeEvent *re = (VBoxResizeEvent *) e;
1017 LogFlow (("VBoxDefs::ResizeEventType: %d x %d x %d bpp\n",
1018 re->width(), re->height(), re->bitsPerPixel()));
1019
1020 if (mToggleFSModeTimer->isActive())
1021 mToggleFSModeTimer->stop();
1022
1023 /* do frame buffer dependent resize */
1024 mFrameBuf->resizeEvent (re);
1025 viewport()->unsetCursor();
1026
1027 /* This event appears in case of guest video was changed
1028 * for somehow even without video resolution change.
1029 * In this last case the host VM window will not be resized
1030 * according this event and the host mouse cursor which was
1031 * unset to default here will not be hidden in capture state.
1032 * So it is necessary to perform updateMouseClipping() for
1033 * the guest resize event if the mouse cursor was captured. */
1034 if (mouse_captured)
1035 updateMouseClipping();
1036
1037 /* apply maximum size restriction */
1038 setMaximumSize (sizeHint());
1039
1040 maybeRestrictMinimumSize();
1041
1042 /* resize the guest canvas */
1043 resizeContents (re->width(), re->height());
1044 /* let our toplevel widget calculate its sizeHint properly */
1045 QApplication::sendPostedEvents (0, QEvent::LayoutHint);
1046
1047 normalizeGeometry (true /* adjustPosition */);
1048
1049 /* report to the VM thread that we finished resizing */
1050 mConsole.GetDisplay().ResizeCompleted (0);
1051
1052 mIgnoreMainwndResize = oldIgnoreMainwndResize;
1053
1054 /* emit a signal about guest was resized */
1055 emit resizeHintDone();
1056
1057 /* update geometry after entering fullscreen | seamless */
1058 if (mMainWnd->isTrueFullscreen() || mMainWnd->isTrueSeamless())
1059 updateGeometry();
1060
1061 return true;
1062 }
1063
1064#if !defined (Q_WS_WIN) && !defined (Q_WS_PM)
1065 /* see VBox[QImage|SDL]FrameBuffer::NotifyUpdate(). */
1066 case VBoxDefs::RepaintEventType:
1067 {
1068 VBoxRepaintEvent *re = (VBoxRepaintEvent *) e;
1069 viewport()->repaint (re->x() - contentsX(),
1070 re->y() - contentsY(),
1071 re->width(), re->height(), false);
1072 /* mConsole.GetDisplay().UpdateCompleted(); - the event was acked already */
1073 return true;
1074 }
1075#endif
1076
1077 case VBoxDefs::SetRegionEventType:
1078 {
1079 VBoxSetRegionEvent *sre = (VBoxSetRegionEvent*) e;
1080 if (mMainWnd->isTrueSeamless() &&
1081 sre->region() != mLastVisibleRegion)
1082 {
1083 mLastVisibleRegion = sre->region();
1084 mMainWnd->setMask (sre->region());
1085 }
1086 else if (!mLastVisibleRegion.isNull() &&
1087 !mMainWnd->isTrueSeamless())
1088 mLastVisibleRegion = QRegion();
1089 return true;
1090 }
1091
1092 case VBoxDefs::MousePointerChangeEventType:
1093 {
1094 MousePointerChangeEvent *me = (MousePointerChangeEvent *) e;
1095 /* change cursor shape only when mouse integration is
1096 * supported (change mouse shape type event may arrive after
1097 * mouse capability change that disables integration */
1098 if (mouse_absolute)
1099 setPointerShape (me);
1100 return true;
1101 }
1102 case VBoxDefs::MouseCapabilityEventType:
1103 {
1104 MouseCapabilityEvent *me = (MouseCapabilityEvent *) e;
1105 if (mouse_absolute != me->supportsAbsolute())
1106 {
1107 mouse_absolute = me->supportsAbsolute();
1108 /* correct the mouse capture state and reset the cursor
1109 * to the default shape if necessary */
1110 if (mouse_absolute)
1111 {
1112 CMouse mouse = mConsole.GetMouse();
1113 mouse.PutMouseEventAbsolute (-1, -1, 0, 0);
1114 captureMouse (false, false);
1115 }
1116 else
1117 viewport()->unsetCursor();
1118 emitMouseStateChanged();
1119 vboxProblem().remindAboutMouseIntegration (mouse_absolute);
1120 }
1121 if (me->needsHostCursor())
1122 mMainWnd->setMouseIntegrationLocked (false);
1123 return true;
1124 }
1125
1126 case VBoxDefs::ModifierKeyChangeEventType:
1127 {
1128 ModifierKeyChangeEvent *me = (ModifierKeyChangeEvent* )e;
1129 if (me->numLock() != mNumLock)
1130 muNumLockAdaptionCnt = 2;
1131 if (me->capsLock() != mCapsLock)
1132 muCapsLockAdaptionCnt = 2;
1133 mNumLock = me->numLock();
1134 mCapsLock = me->capsLock();
1135 mScrollLock = me->scrollLock();
1136 return true;
1137 }
1138
1139 case VBoxDefs::MachineStateChangeEventType:
1140 {
1141 StateChangeEvent *me = (StateChangeEvent *) e;
1142 LogFlowFunc (("MachineStateChangeEventType: state=%d\n",
1143 me->machineState()));
1144 onStateChange (me->machineState());
1145 emit machineStateChanged (me->machineState());
1146 return true;
1147 }
1148
1149 case VBoxDefs::AdditionsStateChangeEventType:
1150 {
1151 GuestAdditionsEvent *ge = (GuestAdditionsEvent *) e;
1152 LogFlowFunc (("AdditionsStateChangeEventType\n"));
1153
1154 mIsAdditionsActive = ge->additionActive();
1155
1156 maybeRestrictMinimumSize();
1157
1158 emit additionsStateChanged (ge->additionVersion(),
1159 ge->additionActive(),
1160 ge->supportsSeamless());
1161 return true;
1162 }
1163
1164 case VBoxDefs::MediaChangeEventType:
1165 {
1166 MediaChangeEvent *mce = (MediaChangeEvent *) e;
1167 LogFlowFunc (("MediaChangeEvent\n"));
1168
1169 emit mediaChanged (mce->diskType());
1170 return true;
1171 }
1172
1173 case VBoxDefs::ActivateMenuEventType:
1174 {
1175 ActivateMenuEvent *ame = (ActivateMenuEvent *) e;
1176 ame->menuData()->activateItemAt (ame->index());
1177
1178 /*
1179 * The main window and its children can be destroyed at this
1180 * point (if, for example, the activated menu item closes the
1181 * main window). Detect this situation to prevent calls to
1182 * destroyed widgets.
1183 */
1184 QWidgetList *list = QApplication::topLevelWidgets ();
1185 bool destroyed = list->find (mMainWnd) < 0;
1186 delete list;
1187 if (!destroyed && mMainWnd->statusBar())
1188 mMainWnd->statusBar()->clear();
1189
1190 return true;
1191 }
1192
1193 case VBoxDefs::NetworkAdapterChangeEventType:
1194 {
1195 /* no specific adapter information stored in this
1196 * event is currently used */
1197 emit networkStateChange();
1198 return true;
1199 }
1200
1201 case VBoxDefs::USBCtlStateChangeEventType:
1202 {
1203 emit usbStateChange();
1204 return true;
1205 }
1206
1207 case VBoxDefs::USBDeviceStateChangeEventType:
1208 {
1209 USBDeviceStateChangeEvent *ue = (USBDeviceStateChangeEvent *)e;
1210
1211 bool success = ue->error().isNull();
1212
1213 if (!success)
1214 {
1215 if (ue->attached())
1216 vboxProblem().cannotAttachUSBDevice (
1217 mConsole,
1218 vboxGlobal().details (ue->device()), ue->error());
1219 else
1220 vboxProblem().cannotDetachUSBDevice (
1221 mConsole,
1222 vboxGlobal().details (ue->device()), ue->error());
1223 }
1224
1225 emit usbStateChange();
1226
1227 return true;
1228 }
1229
1230 case VBoxDefs::SharedFolderChangeEventType:
1231 {
1232 emit sharedFoldersChanged();
1233 return true;
1234 }
1235
1236 case VBoxDefs::RuntimeErrorEventType:
1237 {
1238 RuntimeErrorEvent *ee = (RuntimeErrorEvent *) e;
1239 vboxProblem().showRuntimeError (mConsole, ee->fatal(),
1240 ee->errorID(), ee->message());
1241 return true;
1242 }
1243
1244 case QEvent::KeyPress:
1245 {
1246 QKeyEvent *ke = (QKeyEvent *) e;
1247 if (mIsHostkeyPressed)
1248 {
1249 if (ke->key() >= Key_F1 && ke->key() <= Key_F12)
1250 {
1251 LONG combo [6];
1252 combo [0] = 0x1d; /* Ctrl down */
1253 combo [1] = 0x38; /* Alt down */
1254 combo [4] = 0xb8; /* Alt up */
1255 combo [5] = 0x9d; /* Ctrl up */
1256 if (ke->key() >= Key_F1 && ke->key() <= Key_F10)
1257 {
1258 combo [2] = 0x3b + (ke->key() - Key_F1); /* F1-F10 down */
1259 combo [3] = 0xbb + (ke->key() - Key_F1); /* F1-F10 up */
1260 }
1261 /* some scan slice */
1262 else if (ke->key() >= Key_F11 && ke->key() <= Key_F12)
1263 {
1264 combo [2] = 0x57 + (ke->key() - Key_F11); /* F11-F12 down */
1265 combo [3] = 0xd7 + (ke->key() - Key_F11); /* F11-F12 up */
1266 }
1267 else
1268 Assert (0);
1269
1270 CKeyboard keyboard = mConsole.GetKeyboard();
1271 Assert (!keyboard.isNull());
1272 keyboard.PutScancodes (combo, 6);
1273 }
1274 else if (ke->key() == Key_Home)
1275 {
1276 /* activate the main menu */
1277 if (mMainWnd->isTrueSeamless() || mMainWnd->isTrueFullscreen())
1278 mMainWnd->popupMainMenu (mouse_captured);
1279 else
1280 mMainWnd->menuBar()->setFocus();
1281 }
1282 else
1283 {
1284 /* process hot keys not processed in keyEvent()
1285 * (as in case of non-alphanumeric keys) */
1286 processHotKey (QKeySequence (ke->key()),
1287 mMainWnd->menuBar());
1288 }
1289 }
1290 else
1291 {
1292 if (isPaused())
1293 {
1294 /* if the reminder is disabled we pass the event to
1295 * Qt to enable normal keyboard functionality
1296 * (for example, menu access with Alt+Letter) */
1297 if (!vboxProblem().remindAboutPausedVMInput())
1298 break;
1299 }
1300 }
1301 ke->accept();
1302 return true;
1303 }
1304
1305#ifdef Q_WS_MAC
1306 /* posted OnShowWindow */
1307 case VBoxDefs::ShowWindowEventType:
1308 {
1309 /*
1310 * Dunno what Qt3 thinks a window that has minimized to the dock
1311 * should be - it is not hidden, neither is it minimized. OTOH it is
1312 * marked shown and visible, but not activated. This latter isn't of
1313 * much help though, since at this point nothing is marked activated.
1314 * I might have overlooked something, but I'm buggered what if I know
1315 * what. So, I'll just always show & activate the stupid window to
1316 * make it get out of the dock when the user wishes to show a VM.
1317 */
1318 topLevelWidget()->show();
1319 topLevelWidget()->setActiveWindow();
1320 return true;
1321 }
1322#endif
1323 default:
1324 break;
1325 }
1326 }
1327
1328 return QScrollView::event (e);
1329}
1330
1331bool VBoxConsoleView::eventFilter (QObject *watched, QEvent *e)
1332{
1333 if (attached && watched == viewport())
1334 {
1335 switch (e->type())
1336 {
1337 case QEvent::MouseMove:
1338 case QEvent::MouseButtonPress:
1339 case QEvent::MouseButtonDblClick:
1340 case QEvent::MouseButtonRelease:
1341 {
1342 QMouseEvent *me = (QMouseEvent *) e;
1343 if (mouseEvent (me->type(), me->pos(), me->globalPos(),
1344 me->button(), me->state(), me->stateAfter(),
1345 0, Horizontal))
1346 return true; /* stop further event handling */
1347 break;
1348 }
1349 case QEvent::Wheel:
1350 {
1351 QWheelEvent *we = (QWheelEvent *) e;
1352 if (mouseEvent (we->type(), we->pos(), we->globalPos(),
1353 NoButton, we->state(), we->state(),
1354 we->delta(), we->orientation()))
1355 return true; /* stop further event handling */
1356 break;
1357 }
1358 case QEvent::Resize:
1359 {
1360 if (mouse_captured)
1361 updateMouseClipping();
1362 }
1363 default:
1364 break;
1365 }
1366 }
1367 else if (watched == mMainWnd)
1368 {
1369 switch (e->type())
1370 {
1371#if defined (Q_WS_WIN32)
1372#if defined (VBOX_GUI_USE_DDRAW)
1373 case QEvent::Move:
1374 {
1375 /*
1376 * notification from our parent that it has moved. We need this
1377 * in order to possibly adjust the direct screen blitting.
1378 */
1379 if (mFrameBuf)
1380 mFrameBuf->moveEvent ((QMoveEvent *) e);
1381 break;
1382 }
1383#endif
1384 /*
1385 * install/uninstall low-level kbd hook on every
1386 * activation/deactivation to:
1387 * a) avoid excess hook calls when we're not active and
1388 * b) be always in front of any other possible hooks
1389 */
1390 case QEvent::WindowActivate:
1391 {
1392 g_kbdhook = SetWindowsHookEx (WH_KEYBOARD_LL, lowLevelKeyboardProc,
1393 GetModuleHandle (NULL), 0);
1394 AssertMsg (g_kbdhook, ("SetWindowsHookEx(): err=%d", GetLastError()));
1395 break;
1396 }
1397 case QEvent::WindowDeactivate:
1398 {
1399 if (g_kbdhook)
1400 {
1401 UnhookWindowsHookEx (g_kbdhook);
1402 g_kbdhook = NULL;
1403 }
1404 break;
1405 }
1406#endif /* defined (Q_WS_WIN32) */
1407#if defined (Q_WS_MAC)
1408 /*
1409 * Install/remove the keyboard event handler.
1410 */
1411 case QEvent::WindowActivate:
1412 darwinGrabKeyboardEvents (true);
1413 break;
1414 case QEvent::WindowDeactivate:
1415 darwinGrabKeyboardEvents (false);
1416 break;
1417#endif /* defined (Q_WS_MAC) */
1418 case QEvent::Resize:
1419 {
1420 if (!mIgnoreMainwndResize &&
1421 mIsAdditionsActive && mAutoresizeGuest)
1422 resize_hint_timer->start (300, TRUE);
1423 break;
1424 }
1425
1426 default:
1427 break;
1428 }
1429 }
1430 else if (watched == mMainWnd->menuBar())
1431 {
1432 /*
1433 * sometimes when we press ESC in the menu it brings the
1434 * focus away (Qt bug?) causing no widget to have a focus,
1435 * or holds the focus itself, instead of returning the focus
1436 * to the console window. here we fix this.
1437 */
1438 switch (e->type())
1439 {
1440 case QEvent::FocusOut:
1441 {
1442 if (qApp->focusWidget() == 0)
1443 setFocus();
1444 break;
1445 }
1446 case QEvent::KeyPress:
1447 {
1448 QKeyEvent *ke = (QKeyEvent *) e;
1449 if (ke->key() == Key_Escape && !(ke->state() & KeyButtonMask))
1450 if (mMainWnd->menuBar()->hasFocus())
1451 setFocus();
1452 break;
1453 }
1454 default:
1455 break;
1456 }
1457 }
1458
1459 return QScrollView::eventFilter (watched, e);
1460}
1461
1462#if defined(Q_WS_WIN32)
1463
1464/**
1465 * Low-level keyboard event handler,
1466 * @return
1467 * true to indicate that the message is processed and false otherwise
1468 */
1469bool VBoxConsoleView::winLowKeyboardEvent (UINT msg, const KBDLLHOOKSTRUCT &event)
1470{
1471#if 0
1472 LogFlow (("### vkCode=%08X, scanCode=%08X, flags=%08X, dwExtraInfo=%08X (kbd_captured=%d)\n",
1473 event.vkCode, event.scanCode, event.flags, event.dwExtraInfo, kbd_captured));
1474 char buf [256];
1475 sprintf (buf, "### vkCode=%08X, scanCode=%08X, flags=%08X, dwExtraInfo=%08X",
1476 event.vkCode, event.scanCode, event.flags, event.dwExtraInfo);
1477 mMainWnd->statusBar()->message (buf);
1478#endif
1479
1480 /* Sometimes it happens that Win inserts additional events on some key
1481 * press/release. For example, it prepends ALT_GR in German layout with
1482 * the VK_LCONTROL vkey with curious 0x21D scan code (seems to be necessary
1483 * to specially treat ALT_GR to enter additional chars to regular apps).
1484 * These events are definitely unwanted in VM, so filter them out. */
1485 if (hasFocus() && (event.scanCode & ~0xFF))
1486 return true;
1487
1488 if (!kbd_captured)
1489 return false;
1490
1491 /* it's possible that a key has been pressed while the keyboard was not
1492 * captured, but is being released under the capture. Detect this situation
1493 * and return false to let Windows process the message normally and update
1494 * its key state table (to avoid the stuck key effect). */
1495 uint8_t what_pressed = (event.flags & 0x01) && (event.vkCode != VK_RSHIFT)
1496 ? IsExtKeyPressed
1497 : IsKeyPressed;
1498 if ((event.flags & 0x80) /* released */ &&
1499 ((event.vkCode == gs.hostKey() && !hostkey_in_capture) ||
1500 (mPressedKeys [event.scanCode] & (IsKbdCaptured | what_pressed)) == what_pressed))
1501 return false;
1502
1503 MSG message;
1504 message.hwnd = winId();
1505 message.message = msg;
1506 message.wParam = event.vkCode;
1507 message.lParam =
1508 1 |
1509 (event.scanCode & 0xFF) << 16 |
1510 (event.flags & 0xFF) << 24;
1511
1512 /* Windows sets here the extended bit when the Right Shift key is pressed,
1513 * which is totally wrong. Undo it. */
1514 if (event.vkCode == VK_RSHIFT)
1515 message.lParam &= ~0x1000000;
1516
1517 /* we suppose here that this hook is always called on the main GUI thread */
1518 return winEvent (&message);
1519}
1520
1521/**
1522 * Get Win32 messages before they are passed to Qt. This allows us to get
1523 * the keyboard events directly and bypass the harmful Qt translation. A
1524 * return value of @c true indicates to Qt that the event has been handled.
1525 */
1526bool VBoxConsoleView::winEvent (MSG *msg)
1527{
1528 if (!attached || ! (
1529 msg->message == WM_KEYDOWN || msg->message == WM_SYSKEYDOWN ||
1530 msg->message == WM_KEYUP || msg->message == WM_SYSKEYUP
1531 ))
1532 return false;
1533
1534 /* check for the special flag possibly set at the end of this function */
1535 if ((msg->lParam >> 25) & 0x1)
1536 return false;
1537
1538#if 0
1539 char buf [256];
1540 sprintf (buf, "WM_%04X: vk=%04X rep=%05d scan=%02X ext=%01d rzv=%01X ctx=%01d prev=%01d tran=%01d",
1541 msg->message, msg->wParam,
1542 (msg->lParam & 0xFFFF),
1543 ((msg->lParam >> 16) & 0xFF),
1544 ((msg->lParam >> 24) & 0x1),
1545 ((msg->lParam >> 25) & 0xF),
1546 ((msg->lParam >> 29) & 0x1),
1547 ((msg->lParam >> 30) & 0x1),
1548 ((msg->lParam >> 31) & 0x1));
1549 mMainWnd->statusBar()->message (buf);
1550 LogFlow (("%s\n", buf));
1551#endif
1552
1553 int scan = (msg->lParam >> 16) & 0x7F;
1554 /* scancodes 0x80 and 0x00 are ignored */
1555 if (!scan)
1556 return true;
1557
1558 int vkey = msg->wParam;
1559
1560 /* When one of the SHIFT keys is held and one of the cursor movement
1561 * keys is pressed, Windows duplicates SHIFT press/release messages,
1562 * but with the virtual key code set to 0xFF. These virtual keys are also
1563 * sent in some other situations (Pause, PrtScn, etc.). Ignore such
1564 * messages. */
1565 if (vkey == 0xFF)
1566 return true;
1567
1568 int flags = 0;
1569 if (msg->lParam & 0x1000000)
1570 flags |= KeyExtended;
1571 if (!(msg->lParam & 0x80000000))
1572 flags |= KeyPressed;
1573
1574 switch (vkey)
1575 {
1576 case VK_SHIFT:
1577 case VK_CONTROL:
1578 case VK_MENU:
1579 {
1580 /* overcome stupid Win32 modifier key generalization */
1581 int keyscan = scan;
1582 if (flags & KeyExtended)
1583 keyscan |= 0xE000;
1584 switch (keyscan)
1585 {
1586 case 0x002A: vkey = VK_LSHIFT; break;
1587 case 0x0036: vkey = VK_RSHIFT; break;
1588 case 0x001D: vkey = VK_LCONTROL; break;
1589 case 0xE01D: vkey = VK_RCONTROL; break;
1590 case 0x0038: vkey = VK_LMENU; break;
1591 case 0xE038: vkey = VK_RMENU; break;
1592 }
1593 break;
1594 }
1595 case VK_NUMLOCK:
1596 /* Win32 sets the extended bit for the NumLock key. Reset it. */
1597 flags &= ~KeyExtended;
1598 break;
1599 case VK_SNAPSHOT:
1600 flags |= KeyPrint;
1601 break;
1602 case VK_PAUSE:
1603 flags |= KeyPause;
1604 break;
1605 }
1606
1607 bool result = keyEvent (vkey, scan, flags);
1608 if (!result && kbd_captured)
1609 {
1610 /* keyEvent() returned that it didn't process the message, but since the
1611 * keyboard is captured, we don't want to pass it to Windows. We just want
1612 * to let Qt process the message (to handle non-alphanumeric <HOST>+key
1613 * shortcuts for example). So send it direcltly to the window with the
1614 * special flag in the reserved area of lParam (to avoid recursion). */
1615 ::SendMessage (msg->hwnd, msg->message,
1616 msg->wParam, msg->lParam | (0x1 << 25));
1617 return true;
1618 }
1619
1620 /* These special keys have to be handled by Windows as well to update the
1621 * internal modifier state and to enable/disable the keyboard LED */
1622 if (vkey == VK_NUMLOCK || vkey == VK_CAPITAL)
1623 return false;
1624
1625 return result;
1626}
1627
1628#elif defined (Q_WS_PM)
1629
1630/**
1631 * Get PM messages before they are passed to Qt. This allows us to get
1632 * the keyboard events directly and bypass the harmful Qt translation. A
1633 * return value of @c true indicates to Qt that the event has been handled.
1634 */
1635bool VBoxConsoleView::pmEvent (QMSG *aMsg)
1636{
1637 if (!attached || aMsg->msg != WM_CHAR)
1638 return false;
1639
1640 /* check for the special flag possibly set at the end of this function */
1641 if (SHORT2FROMMP (aMsg->mp2) & 0x8000)
1642 return false;
1643
1644#if 0
1645 {
1646 char buf [256];
1647 sprintf (buf, "*** WM_CHAR: f=%04X rep=%03d scan=%02X ch=%04X vk=%04X",
1648 SHORT1FROMMP (aMsg->mp1), CHAR3FROMMP (aMsg->mp1),
1649 CHAR4FROMMP (aMsg->mp1), SHORT1FROMMP (aMsg->mp2),
1650 SHORT2FROMMP (aMsg->mp2));
1651 mMainWnd->statusBar()->message (buf);
1652 LogFlow (("%s\n", buf));
1653 }
1654#endif
1655
1656 USHORT ch = SHORT1FROMMP (aMsg->mp2);
1657 USHORT f = SHORT1FROMMP (aMsg->mp1);
1658
1659 int scan = (unsigned int) CHAR4FROMMP (aMsg->mp1);
1660 if (!scan || scan > 0x7F)
1661 return true;
1662
1663 int vkey = QIHotKeyEdit::virtualKey (aMsg);
1664
1665 int flags = 0;
1666
1667 if ((ch & 0xFF) == 0xE0)
1668 {
1669 flags |= KeyExtended;
1670 scan = ch >> 8;
1671 }
1672 else if (scan == 0x5C && (ch & 0xFF) == '/')
1673 {
1674 /* this is the '/' key on the keypad */
1675 scan = 0x35;
1676 flags |= KeyExtended;
1677 }
1678 else
1679 {
1680 /* For some keys, the scan code passed in QMSG is a pseudo scan
1681 * code. We replace it with a real hardware scan code, according to
1682 * http://www.computer-engineering.org/ps2keyboard/scancodes1.html.
1683 * Also detect Pause and PrtScn and set flags. */
1684 switch (vkey)
1685 {
1686 case VK_ENTER: scan = 0x1C; flags |= KeyExtended; break;
1687 case VK_CTRL: scan = 0x1D; flags |= KeyExtended; break;
1688 case VK_ALTGRAF: scan = 0x38; flags |= KeyExtended; break;
1689 case VK_LWIN: scan = 0x5B; flags |= KeyExtended; break;
1690 case VK_RWIN: scan = 0x5C; flags |= KeyExtended; break;
1691 case VK_WINMENU: scan = 0x5D; flags |= KeyExtended; break;
1692 case VK_FORWARD: scan = 0x69; flags |= KeyExtended; break;
1693 case VK_BACKWARD: scan = 0x6A; flags |= KeyExtended; break;
1694#if 0
1695 /// @todo this would send 0xE0 0x46 0xE0 0xC6. It's not fully
1696 // clear what is more correct
1697 case VK_BREAK: scan = 0x46; flags |= KeyExtended; break;
1698#else
1699 case VK_BREAK: scan = 0; flags |= KeyPause; break;
1700#endif
1701 case VK_PAUSE: scan = 0; flags |= KeyPause; break;
1702 case VK_PRINTSCRN: scan = 0; flags |= KeyPrint; break;
1703 default:;
1704 }
1705 }
1706
1707 if (!(f & KC_KEYUP))
1708 flags |= KeyPressed;
1709
1710 bool result = keyEvent (vkey, scan, flags);
1711 if (!result && kbd_captured)
1712 {
1713 /* keyEvent() returned that it didn't process the message, but since the
1714 * keyboard is captured, we don't want to pass it to PM. We just want
1715 * to let Qt process the message (to handle non-alphanumeric <HOST>+key
1716 * shortcuts for example). So send it direcltly to the window with the
1717 * special flag in the reserved area of lParam (to avoid recursion). */
1718 ::WinSendMsg (aMsg->hwnd, aMsg->msg,
1719 aMsg->mp1,
1720 MPFROM2SHORT (SHORT1FROMMP (aMsg->mp2),
1721 SHORT2FROMMP (aMsg->mp2) | 0x8000));
1722 return true;
1723 }
1724 return result;
1725}
1726
1727#elif defined(Q_WS_X11)
1728
1729/**
1730 * This routine gets X11 events before they are processed by Qt. This is
1731 * used for our platform specific keyboard implementation. A return value
1732 * of TRUE indicates that the event has been processed by us.
1733 */
1734bool VBoxConsoleView::x11Event (XEvent *event)
1735{
1736 static WINEKEYBOARDINFO wineKeyboardInfo;
1737
1738 switch (event->type)
1739 {
1740 case XKeyPress:
1741 case XKeyRelease:
1742 if (attached)
1743 break;
1744 /* else fall through */
1745 /// @todo (AH) later, we might want to handle these as well
1746 case KeymapNotify:
1747 case MappingNotify:
1748 default:
1749 return false; /* pass the event to Qt */
1750 }
1751
1752 /* perform the mega-complex translation using the wine algorithms */
1753 handleXKeyEvent (this->x11Display(), event, &wineKeyboardInfo);
1754
1755#if 0
1756 char buf [256];
1757 sprintf (buf, "pr=%d kc=%08X st=%08X fl=%08lX scan=%04X",
1758 event->type == XKeyPress ? 1 : 0, event->xkey.keycode,
1759 event->xkey.state, wineKeyboardInfo.dwFlags, wineKeyboardInfo.wScan);
1760 mMainWnd->statusBar()->message (buf);
1761 LogFlow (("### %s\n", buf));
1762#endif
1763
1764 int scan = wineKeyboardInfo.wScan & 0x7F;
1765 // scancodes 0x00 (no valid translation) and 0x80 are ignored
1766 if (!scan)
1767 return true;
1768
1769 KeySym ks = ::XKeycodeToKeysym (event->xkey.display, event->xkey.keycode, 0);
1770
1771 int flags = 0;
1772 if (wineKeyboardInfo.dwFlags & 0x0001)
1773 flags |= KeyExtended;
1774 if (event->type == XKeyPress)
1775 flags |= KeyPressed;
1776
1777 switch (ks)
1778 {
1779 case XK_Num_Lock:
1780 // Wine sets the extended bit for the NumLock key. Reset it.
1781 flags &= ~KeyExtended;
1782 break;
1783 case XK_Print:
1784 flags |= KeyPrint;
1785 break;
1786 case XK_Pause:
1787 flags |= KeyPause;
1788 break;
1789 }
1790
1791 return keyEvent (ks, scan, flags);
1792}
1793
1794#elif defined (Q_WS_MAC)
1795
1796/**
1797 * Invoked by VBoxConsoleView::darwinEventHandlerProc / VBoxConsoleView::macEventFilter when
1798 * it receives a raw keyboard event.
1799 *
1800 * @param inEvent The keyboard event.
1801 *
1802 * @return true if the key was processed, false if it wasn't processed and should be passed on.
1803 */
1804bool VBoxConsoleView::darwinKeyboardEvent (EventRef inEvent)
1805{
1806 bool ret = false;
1807 UInt32 EventKind = ::GetEventKind (inEvent);
1808 if (EventKind != kEventRawKeyModifiersChanged)
1809 {
1810 /* convert keycode to set 1 scan code. */
1811 UInt32 keyCode = ~0U;
1812 ::GetEventParameter (inEvent, kEventParamKeyCode, typeUInt32, NULL, sizeof (keyCode), NULL, &keyCode);
1813 unsigned scanCode = ::DarwinKeycodeToSet1Scancode (keyCode);
1814 if (scanCode)
1815 {
1816 /* calc flags. */
1817 int flags = 0;
1818 if (EventKind != kEventRawKeyUp)
1819 flags |= KeyPressed;
1820 if (scanCode & VBOXKEY_EXTENDED)
1821 flags |= KeyExtended;
1822 /** @todo KeyPause, KeyPrint. */
1823 scanCode &= VBOXKEY_SCANCODE_MASK;
1824
1825 /* get the unicode string (if present). */
1826 AssertCompileSize (wchar_t, 2);
1827 AssertCompileSize (UniChar, 2);
1828 UInt32 cbWritten = 0;
1829 wchar_t ucs[8];
1830 if (::GetEventParameter (inEvent, kEventParamKeyUnicodes, typeUnicodeText, NULL,
1831 sizeof (ucs), &cbWritten, &ucs[0]) != 0)
1832 cbWritten = 0;
1833 ucs[cbWritten / sizeof(wchar_t)] = 0; /* The api doesn't terminate it. */
1834
1835 ret = keyEvent (keyCode, scanCode, flags, ucs[0] ? ucs : NULL);
1836 }
1837 }
1838 else
1839 {
1840 /* May contain multiple modifier changes, kind of annoying. */
1841 UInt32 newMask = 0;
1842 ::GetEventParameter (inEvent, kEventParamKeyModifiers, typeUInt32, NULL,
1843 sizeof (newMask), NULL, &newMask);
1844 newMask = ::DarwinAdjustModifierMask (newMask);
1845 UInt32 changed = newMask ^ mDarwinKeyModifiers;
1846 if (changed)
1847 {
1848 for (UInt32 bit = 0; bit < 32; bit++)
1849 {
1850 if (!(changed & (1 << bit)))
1851 continue;
1852 unsigned scanCode = ::DarwinModifierMaskToSet1Scancode (1 << bit);
1853 if (!scanCode)
1854 continue;
1855 unsigned keyCode = ::DarwinModifierMaskToDarwinKeycode (1 << bit);
1856 Assert (keyCode);
1857
1858 if (!(scanCode & VBOXKEY_LOCK))
1859 {
1860 unsigned flags = (newMask & (1 << bit)) ? KeyPressed : 0;
1861 if (scanCode & VBOXKEY_EXTENDED)
1862 flags |= KeyExtended;
1863 scanCode &= VBOXKEY_SCANCODE_MASK;
1864 ret |= keyEvent (keyCode, scanCode & 0xff, flags);
1865 }
1866 else
1867 {
1868 unsigned flags = 0;
1869 if (scanCode & VBOXKEY_EXTENDED)
1870 flags |= KeyExtended;
1871 scanCode &= VBOXKEY_SCANCODE_MASK;
1872 keyEvent (keyCode, scanCode, flags | KeyPressed);
1873 keyEvent (keyCode, scanCode, flags);
1874 }
1875 }
1876 }
1877
1878 mDarwinKeyModifiers = newMask;
1879
1880 /* Always return true here because we'll otherwise getting a Qt event
1881 we don't want and that will only cause the Pause warning to pop up. */
1882 ret = true;
1883 }
1884
1885 return ret;
1886}
1887
1888
1889/**
1890 * Installs or removes the keyboard event handler.
1891 *
1892 * @param fGrab True if we're to grab the events, false if we're not to.
1893 */
1894void VBoxConsoleView::darwinGrabKeyboardEvents (bool fGrab)
1895{
1896 if (fGrab)
1897 {
1898 ::SetMouseCoalescingEnabled (false, NULL); //??
1899 ::CGSetLocalEventsSuppressionInterval (0.0); //??
1900
1901#ifndef VBOX_WITH_HACKED_QT
1902
1903 EventTypeSpec eventTypes[6];
1904 eventTypes[0].eventClass = kEventClassKeyboard;
1905 eventTypes[0].eventKind = kEventRawKeyDown;
1906 eventTypes[1].eventClass = kEventClassKeyboard;
1907 eventTypes[1].eventKind = kEventRawKeyUp;
1908 eventTypes[2].eventClass = kEventClassKeyboard;
1909 eventTypes[2].eventKind = kEventRawKeyRepeat;
1910 eventTypes[3].eventClass = kEventClassKeyboard;
1911 eventTypes[3].eventKind = kEventRawKeyModifiersChanged;
1912 /* For ignorning Command-H and Command-Q which aren't affected by the
1913 * global hotkey stuff (doesn't work well): */
1914 eventTypes[4].eventClass = kEventClassCommand;
1915 eventTypes[4].eventKind = kEventCommandProcess;
1916 eventTypes[5].eventClass = kEventClassCommand;
1917 eventTypes[5].eventKind = kEventCommandUpdateStatus;
1918
1919 EventHandlerUPP eventHandler = ::NewEventHandlerUPP (VBoxConsoleView::darwinEventHandlerProc);
1920
1921 mDarwinEventHandlerRef = NULL;
1922 ::InstallApplicationEventHandler (eventHandler, RT_ELEMENTS (eventTypes), &eventTypes[0],
1923 this, &mDarwinEventHandlerRef);
1924 ::DisposeEventHandlerUPP (eventHandler);
1925
1926#else /* VBOX_WITH_HACKED_QT */
1927 ((QIApplication *)qApp)->setEventFilter (VBoxConsoleView::macEventFilter, this);
1928#endif /* VBOX_WITH_HACKED_QT */
1929
1930 ::DarwinGrabKeyboard (false);
1931 }
1932 else
1933 {
1934 ::DarwinReleaseKeyboard ();
1935#ifndef VBOX_WITH_HACKED_QT
1936 if (mDarwinEventHandlerRef)
1937 {
1938 ::RemoveEventHandler (mDarwinEventHandlerRef);
1939 mDarwinEventHandlerRef = NULL;
1940 }
1941#else
1942 ((QIApplication *)qApp)->setEventFilter (NULL, NULL);
1943#endif
1944 }
1945}
1946
1947#endif // defined (Q_WS_WIN)
1948
1949//
1950// Private members
1951/////////////////////////////////////////////////////////////////////////////
1952
1953/**
1954 * Called on every focus change and also to forcibly capture/uncapture the
1955 * input in situations similar to gaining or losing focus.
1956 *
1957 * @param aHasFocus true if the window got focus and false otherwise.
1958 * @param aReleaseHostKey true to release the host key (used only when
1959 * @a aHasFocus is false.
1960 */
1961void VBoxConsoleView::focusEvent (bool aHasFocus,
1962 bool aReleaseHostKey /* = true */)
1963{
1964 if (aHasFocus)
1965 {
1966#ifdef RT_OS_WINDOWS
1967 if ( !mDisableAutoCapture && gs.autoCapture()
1968 && GetAncestor (winId(), GA_ROOT) == GetForegroundWindow())
1969#else
1970 if (!mDisableAutoCapture && gs.autoCapture())
1971#endif /* RT_OS_WINDOWS */
1972 {
1973 captureKbd (true);
1974/// @todo (dmik)
1975// the below is for the mouse auto-capture. disabled for now. in order to
1976// properly support it, we need to know when *all* mouse buttons are
1977// released after we got focus, and grab the mouse only after then.
1978// btw, the similar would be good the for keyboard auto-capture, too.
1979// if (!(mouse_absolute && mouse_integration))
1980// captureMouse (true);
1981 }
1982
1983 /* reset the single-time disable capture flag */
1984 if (mDisableAutoCapture)
1985 mDisableAutoCapture = false;
1986 }
1987 else
1988 {
1989 captureMouse (false);
1990 captureKbd (false, false);
1991 releaseAllPressedKeys (aReleaseHostKey);
1992 }
1993}
1994
1995/**
1996 * Synchronize the views of the host and the guest to the modifier keys.
1997 * This function will add up to 6 additional keycodes to codes.
1998 *
1999 * @param codes pointer to keycodes which are sent to the keyboard
2000 * @param count pointer to the keycodes counter
2001 */
2002void VBoxConsoleView::fixModifierState(LONG *codes, uint *count)
2003{
2004#if defined(Q_WS_X11)
2005
2006 Window wDummy1, wDummy2;
2007 int iDummy3, iDummy4, iDummy5, iDummy6;
2008 unsigned uMask;
2009 unsigned uKeyMaskNum = 0, uKeyMaskCaps = 0, uKeyMaskScroll = 0;
2010
2011 uKeyMaskCaps = LockMask;
2012 XModifierKeymap* map = XGetModifierMapping(qt_xdisplay());
2013 KeyCode keyCodeNum = XKeysymToKeycode(qt_xdisplay(), XK_Num_Lock);
2014 KeyCode keyCodeScroll = XKeysymToKeycode(qt_xdisplay(), XK_Scroll_Lock);
2015
2016 for (int i = 0; i < 8; i++)
2017 {
2018 if ( keyCodeNum != NoSymbol
2019 && map->modifiermap[map->max_keypermod * i] == keyCodeNum)
2020 uKeyMaskNum = 1 << i;
2021 else if ( keyCodeScroll != NoSymbol
2022 && map->modifiermap[map->max_keypermod * i] == keyCodeScroll)
2023 uKeyMaskScroll = 1 << i;
2024 }
2025 XQueryPointer(qt_xdisplay(), DefaultRootWindow(qt_xdisplay()), &wDummy1, &wDummy2,
2026 &iDummy3, &iDummy4, &iDummy5, &iDummy6, &uMask);
2027 XFreeModifiermap(map);
2028
2029 if (muNumLockAdaptionCnt && (mNumLock ^ !!(uMask & uKeyMaskNum)))
2030 {
2031 muNumLockAdaptionCnt--;
2032 codes[(*count)++] = 0x45;
2033 codes[(*count)++] = 0x45 | 0x80;
2034 }
2035 if (muCapsLockAdaptionCnt && (mCapsLock ^ !!(uMask & uKeyMaskCaps)))
2036 {
2037 muCapsLockAdaptionCnt--;
2038 codes[(*count)++] = 0x3a;
2039 codes[(*count)++] = 0x3a | 0x80;
2040 }
2041
2042#elif defined(Q_WS_WIN32)
2043
2044 if (muNumLockAdaptionCnt && (mNumLock ^ !!(GetKeyState(VK_NUMLOCK))))
2045 {
2046 muNumLockAdaptionCnt--;
2047 codes[(*count)++] = 0x45;
2048 codes[(*count)++] = 0x45 | 0x80;
2049 }
2050 if (muCapsLockAdaptionCnt && (mCapsLock ^ !!(GetKeyState(VK_CAPITAL))))
2051 {
2052 muCapsLockAdaptionCnt--;
2053 codes[(*count)++] = 0x3a;
2054 codes[(*count)++] = 0x3a | 0x80;
2055 }
2056
2057#elif defined(Q_WS_MAC)
2058
2059 /* if (muNumLockAdaptionCnt) ... - NumLock isn't implemented by Mac OS X so ignore it. */
2060 if (muCapsLockAdaptionCnt && (mCapsLock ^ !!(::GetCurrentEventKeyModifiers() & alphaLock)))
2061 {
2062 muCapsLockAdaptionCnt--;
2063 codes[(*count)++] = 0x3a;
2064 codes[(*count)++] = 0x3a | 0x80;
2065 }
2066
2067#else
2068
2069#warning Adapt VBoxConsoleView::fixModifierState
2070
2071#endif
2072
2073
2074}
2075
2076/**
2077 * Called on enter/exit seamless/fullscreen mode.
2078 */
2079void VBoxConsoleView::toggleFSMode()
2080{
2081 if (mIsAdditionsActive && mAutoresizeGuest)
2082 {
2083 QSize newSize = QSize();
2084 if (mMainWnd->isTrueFullscreen() || mMainWnd->isTrueSeamless())
2085 {
2086 mNormalSize = frameSize();
2087 newSize = maximumSize();
2088 }
2089 else
2090 newSize = mNormalSize;
2091 doResizeHint (newSize);
2092 }
2093 mToggleFSModeTimer->start (400, true);
2094}
2095
2096/**
2097 * Called on every key press and release (while in focus).
2098 *
2099 * @key virtual scan code (virtual key on Win32 and KeySym on X11)
2100 * @scan hardware scan code
2101 * @flags flags, a combination of Key* constants
2102 * @aUniKey Unicode translation of the key. Optional.
2103 *
2104 * @return true to consume the event and false to pass it to Qt
2105 */
2106bool VBoxConsoleView::keyEvent (int aKey, uint8_t aScan, int aFlags,
2107 wchar_t *aUniKey/* = NULL*/)
2108{
2109#if 0
2110 {
2111 char buf [256];
2112 sprintf (buf, "aKey=%08X aScan=%02X aFlags=%08X",
2113 aKey, aScan, aFlags);
2114 mMainWnd->statusBar()->message (buf);
2115 }
2116#endif
2117
2118 const bool isHostKey = aKey == gs.hostKey();
2119
2120 LONG buf [16];
2121 LONG *codes = buf;
2122 uint count = 0;
2123 uint8_t whatPressed = 0;
2124
2125 if (!isHostKey)
2126 {
2127 if (aFlags & KeyPrint)
2128 {
2129 static LONG PrintMake[] = { 0xE0, 0x2A, 0xE0, 0x37 };
2130 static LONG PrintBreak[] = { 0xE0, 0xB7, 0xE0, 0xAA };
2131 if (aFlags & KeyPressed)
2132 {
2133 codes = PrintMake;
2134 count = SIZEOF_ARRAY (PrintMake);
2135 }
2136 else
2137 {
2138 codes = PrintBreak;
2139 count = SIZEOF_ARRAY (PrintBreak);
2140 }
2141 }
2142 else if (aFlags & KeyPause)
2143 {
2144 if (aFlags & KeyPressed)
2145 {
2146 static LONG Pause[] = { 0xE1, 0x1D, 0x45, 0xE1, 0x9D, 0xC5 };
2147 codes = Pause;
2148 count = SIZEOF_ARRAY (Pause);
2149 }
2150 else
2151 {
2152 /* Pause shall not produce a break code */
2153 return true;
2154 }
2155 }
2156 else
2157 {
2158 if (aFlags & KeyPressed)
2159 {
2160 /* Check if the guest has the same view on the modifier keys (NumLock,
2161 * CapsLock, ScrollLock) as the X server. If not, send KeyPress events
2162 * to synchronize the state. */
2163 fixModifierState (codes, &count);
2164 }
2165
2166 /* process the scancode and update the table of pressed keys */
2167 whatPressed = IsKeyPressed;
2168
2169 if (aFlags & KeyExtended)
2170 {
2171 codes [count++] = 0xE0;
2172 whatPressed = IsExtKeyPressed;
2173 }
2174
2175 if (aFlags & KeyPressed)
2176 {
2177 codes [count++] = aScan;
2178 mPressedKeys [aScan] |= whatPressed;
2179 }
2180 else
2181 {
2182 /* if we haven't got this key's press message, we ignore its
2183 * release */
2184 if (!(mPressedKeys [aScan] & whatPressed))
2185 return true;
2186 codes [count++] = aScan | 0x80;
2187 mPressedKeys [aScan] &= ~whatPressed;
2188 }
2189
2190 if (kbd_captured)
2191 mPressedKeys [aScan] |= IsKbdCaptured;
2192 else
2193 mPressedKeys [aScan] &= ~IsKbdCaptured;
2194 }
2195 }
2196 else
2197 {
2198 /* currently this is used in winLowKeyboardEvent() only */
2199 hostkey_in_capture = kbd_captured;
2200 }
2201
2202 bool emitSignal = false;
2203 int hotkey = 0;
2204
2205 /* process the host key */
2206 if (aFlags & KeyPressed)
2207 {
2208 if (isHostKey)
2209 {
2210 if (!mIsHostkeyPressed)
2211 {
2212 mIsHostkeyPressed = mIsHostkeyAlone = true;
2213 if (isRunning())
2214 saveKeyStates();
2215 emitSignal = true;
2216 }
2217 }
2218 else
2219 {
2220 if (mIsHostkeyPressed)
2221 {
2222 if (mIsHostkeyAlone)
2223 {
2224 hotkey = aKey;
2225 mIsHostkeyAlone = false;
2226 }
2227 }
2228 }
2229 }
2230 else
2231 {
2232 if (isHostKey)
2233 {
2234 if (mIsHostkeyPressed)
2235 {
2236 mIsHostkeyPressed = false;
2237
2238 if (mIsHostkeyAlone)
2239 {
2240 if (isPaused())
2241 {
2242 vboxProblem().remindAboutPausedVMInput();
2243 }
2244 else
2245 if (isRunning())
2246 {
2247 bool captured = kbd_captured;
2248 bool ok = true;
2249 if (!captured)
2250 {
2251 /* temporarily disable auto capture that will take
2252 * place after this dialog is dismissed because
2253 * the capture state is to be defined by the
2254 * dialog result itself */
2255 mDisableAutoCapture = true;
2256 bool autoConfirmed = false;
2257 ok = vboxProblem().confirmInputCapture (&autoConfirmed);
2258 if (autoConfirmed)
2259 mDisableAutoCapture = false;
2260 /* otherwise, the disable flag will be reset in
2261 * the next console view's foucs in event (since
2262 * may happen asynchronously on some platforms,
2263 * after we return from this code) */
2264 }
2265
2266 if (ok)
2267 {
2268 captureKbd (!captured, false);
2269 if (!(mouse_absolute && mouse_integration))
2270 captureMouse (kbd_captured);
2271 }
2272 }
2273 }
2274
2275 if (isRunning())
2276 sendChangedKeyStates();
2277
2278 emitSignal = true;
2279 }
2280 }
2281 else
2282 {
2283 if (mIsHostkeyPressed)
2284 mIsHostkeyAlone = false;
2285 }
2286 }
2287
2288 /* emit the keyboard state change signal */
2289 if (emitSignal)
2290 emitKeyboardStateChanged();
2291
2292 /* process HOST+<key> shortcuts. currently, <key> is limited to
2293 * alphanumeric chars. */
2294 if (hotkey)
2295 {
2296 bool processed = false;
2297#if defined (Q_WS_WIN32)
2298 NOREF(aUniKey);
2299 int n = GetKeyboardLayoutList (0, NULL);
2300 Assert (n);
2301 HKL *list = new HKL [n];
2302 GetKeyboardLayoutList (n, list);
2303 for (int i = 0; i < n && !processed; i++)
2304 {
2305 wchar_t ch;
2306 static BYTE keys [256] = {0};
2307 if (!ToUnicodeEx (hotkey, 0, keys, &ch, 1, 0, list [i]) == 1)
2308 ch = 0;
2309 if (ch)
2310 processed = processHotKey (QKeySequence (UNICODE_ACCEL +
2311 QChar (ch).upper().unicode()),
2312 mMainWnd->menuBar());
2313 }
2314 delete[] list;
2315#elif defined (Q_WS_X11)
2316 NOREF(aUniKey);
2317 Display *display = x11Display();
2318 int keysyms_per_keycode = getKeysymsPerKeycode();
2319 KeyCode kc = XKeysymToKeycode (display, aKey);
2320 // iterate over the first level (not shifted) keysyms in every group
2321 for (int i = 0; i < keysyms_per_keycode && !processed; i += 2)
2322 {
2323 KeySym ks = XKeycodeToKeysym (display, kc, i);
2324 char ch = 0;
2325 if (!XkbTranslateKeySym (display, &ks, 0, &ch, 1, NULL) == 1)
2326 ch = 0;
2327 if (ch)
2328 {
2329 QChar c = QString::fromLocal8Bit (&ch, 1) [0];
2330 processed = processHotKey (QKeySequence (UNICODE_ACCEL +
2331 c.upper().unicode()),
2332 mMainWnd->menuBar());
2333 }
2334 }
2335#elif defined (Q_WS_MAC)
2336 if (aUniKey && aUniKey [0] && !aUniKey [1])
2337 processed = processHotKey (QKeySequence (UNICODE_ACCEL +
2338 QChar (aUniKey [0]).upper().unicode()),
2339 mMainWnd->menuBar());
2340
2341 /* Don't consider the hot key as pressed since the guest never saw
2342 * it. (probably a generic thing) */
2343 mPressedKeys [aScan] &= ~whatPressed;
2344#endif
2345
2346 /* grab the key from Qt if processed, or pass it to Qt otherwise
2347 * in order to process non-alphanumeric keys in event(), after they are
2348 * converted to Qt virtual keys. */
2349 return processed;
2350 }
2351
2352 /* no more to do, if the host key is in action or the VM is paused */
2353 if (mIsHostkeyPressed || isHostKey || isPaused())
2354 {
2355 /* grab the key from Qt and from VM if it's a host key,
2356 * otherwise just pass it to Qt */
2357 return isHostKey;
2358 }
2359
2360 CKeyboard keyboard = mConsole.GetKeyboard();
2361 Assert (!keyboard.isNull());
2362
2363#if defined (Q_WS_WIN32)
2364 /* send pending WM_PAINT events */
2365 ::UpdateWindow (viewport()->winId());
2366#endif
2367
2368#if 0
2369 {
2370 char buf [256];
2371 sprintf (buf, "*** SCANS: ");
2372 for (uint i = 0; i < count; ++ i)
2373 sprintf (buf + strlen (buf), "%02X ", codes [i]);
2374 mMainWnd->statusBar()->message (buf);
2375 LogFlow (("%s\n", buf));
2376 }
2377#endif
2378
2379 keyboard.PutScancodes (codes, count);
2380
2381 /* grab the key from Qt */
2382 return true;
2383}
2384
2385/**
2386 * Called on every mouse/wheel move and button press/release.
2387 *
2388 * @return true to consume the event and false to pass it to Qt
2389 */
2390bool VBoxConsoleView::mouseEvent (int aType, const QPoint &aPos,
2391 const QPoint &aGlobalPos, ButtonState aButton,
2392 ButtonState aState, ButtonState aStateAfter,
2393 int aWheelDelta, Orientation aWheelDir)
2394{
2395#if 0
2396 char buf [256];
2397 sprintf (buf,
2398 "MOUSE: type=%03d x=%03d y=%03d btn=%03d st=%08X stAfter=%08X "
2399 "wdelta=%03d wdir=%03d",
2400 aType, aPos.x(), aPos.y(), aButton, aState, aStateAfter,
2401 aWheelDelta, aWheelDir);
2402 mMainWnd->statusBar()->message (buf);
2403#else
2404 Q_UNUSED (aButton);
2405 Q_UNUSED (aState);
2406#endif
2407
2408 int state = 0;
2409 if (aStateAfter & LeftButton)
2410 state |= CEnums::LeftButton;
2411 if (aStateAfter & RightButton)
2412 state |= CEnums::RightButton;
2413 if (aStateAfter & MidButton)
2414 state |= CEnums::MiddleButton;
2415
2416 int wheel = 0;
2417 if (aWheelDir == Vertical)
2418 {
2419 /* the absolute value of wheel delta is 120 units per every wheel
2420 * move; positive deltas correspond to counterclockwize rotations
2421 * (usually up), negative -- to clockwize (usually down). */
2422 wheel = - (aWheelDelta / 120);
2423 }
2424
2425 if (mouse_captured)
2426 {
2427#ifdef Q_WS_WIN32
2428 /* send pending WM_PAINT events */
2429 ::UpdateWindow (viewport()->winId());
2430#endif
2431
2432 CMouse mouse = mConsole.GetMouse();
2433 mouse.PutMouseEvent (aGlobalPos.x() - last_pos.x(),
2434 aGlobalPos.y() - last_pos.y(),
2435 wheel, state);
2436
2437#if defined (Q_WS_MAC)
2438 /*
2439 * Keep the mouse from leaving the widget.
2440 *
2441 * This is a bit tricky to get right because if it escapes we won't necessarily
2442 * get mouse events any longer and can warp it back. So, we keep safety zone
2443 * of up to 300 pixels around the borders of the widget to prevent this from
2444 * happening. Also, the mouse is warped back to the center of the widget.
2445 *
2446 * (Note, aPos seems to be unreliable, it caused endless recursion here at one points...)
2447 * (Note, synergy and other remote clients might not like this cursor warping.)
2448 */
2449 QRect rect = viewport()->visibleRect();
2450 QPoint pw = viewport()->mapToGlobal (viewport()->pos());
2451 rect.moveBy (pw.x(), pw.y());
2452
2453 QRect dpRect = QApplication::desktop()->screenGeometry (viewport());
2454 if (rect.intersects (dpRect))
2455 rect = rect.intersect (dpRect);
2456
2457 int wsafe = rect.width() / 6;
2458 rect.setWidth (rect.width() - wsafe * 2);
2459 rect.setLeft (rect.left() + wsafe);
2460
2461 int hsafe = rect.height() / 6;
2462 rect.setWidth (rect.height() - hsafe * 2);
2463 rect.setTop (rect.top() + hsafe);
2464
2465 if (rect.contains (aGlobalPos, true))
2466 last_pos = aGlobalPos;
2467 else
2468 {
2469 last_pos = rect.center();
2470 QCursor::setPos (last_pos);
2471 }
2472
2473#else /* !Q_WS_MAC */
2474
2475 /* "jerk" the mouse by bringing it to the opposite side
2476 * to simulate the endless moving */
2477
2478#ifdef Q_WS_WIN32
2479 int we = viewport()->width() - 1;
2480 int he = viewport()->height() - 1;
2481 QPoint p = aPos;
2482 if (aPos.x() == 0)
2483 p.setX (we - 1);
2484 else if (aPos.x() == we)
2485 p.setX (1);
2486 if (aPos.y() == 0 )
2487 p.setY (he - 1);
2488 else if (aPos.y() == he)
2489 p.setY (1);
2490
2491 if (p != aPos)
2492 {
2493 last_pos = viewport()->mapToGlobal (p);
2494 QCursor::setPos (last_pos);
2495 }
2496 else
2497 {
2498 last_pos = aGlobalPos;
2499 }
2500#else
2501 int we = QApplication::desktop()->width() - 1;
2502 int he = QApplication::desktop()->height() - 1;
2503 QPoint p = aGlobalPos;
2504 if (aGlobalPos.x() == 0)
2505 p.setX (we - 1);
2506 else if (aGlobalPos.x() == we)
2507 p.setX( 1 );
2508 if (aGlobalPos.y() == 0)
2509 p.setY (he - 1);
2510 else if (aGlobalPos.y() == he)
2511 p.setY (1);
2512
2513 if (p != aGlobalPos)
2514 {
2515 last_pos = p;
2516 QCursor::setPos (last_pos);
2517 }
2518 else
2519 {
2520 last_pos = aGlobalPos;
2521 }
2522#endif
2523#endif /* !Q_WS_MAC */
2524 return true; /* stop further event handling */
2525 }
2526 else /* !mouse_captured */
2527 {
2528#ifdef Q_WS_MAC
2529 /* Update the mouse cursor; this is a bit excessive really... */
2530 if (!DarwinCursorIsNull (&mDarwinCursor))
2531 DarwinCursorSet (&mDarwinCursor);
2532#endif
2533 if (mMainWnd->isTrueFullscreen())
2534 {
2535 if (mode != VBoxDefs::SDLMode)
2536 {
2537 /* try to automatically scroll the guest canvas if the
2538 * mouse is on the screen border */
2539 /// @todo (r=dmik) better use a timer for autoscroll
2540 QRect scrGeo = QApplication::desktop()->screenGeometry (this);
2541 int dx = 0, dy = 0;
2542 if (scrGeo.width() < contentsWidth())
2543 {
2544 if (scrGeo.rLeft() == aGlobalPos.x()) dx = -1;
2545 if (scrGeo.rRight() == aGlobalPos.x()) dx = +1;
2546 }
2547 if (scrGeo.height() < contentsHeight())
2548 {
2549 if (scrGeo.rTop() == aGlobalPos.y()) dy = -1;
2550 if (scrGeo.rBottom() == aGlobalPos.y()) dy = +1;
2551 }
2552 if (dx || dy)
2553 scrollBy (dx, dy);
2554 }
2555 }
2556
2557 if (mouse_absolute && mouse_integration)
2558 {
2559 int cw = contentsWidth(), ch = contentsHeight();
2560 int vw = visibleWidth(), vh = visibleHeight();
2561
2562 if (mode != VBoxDefs::SDLMode)
2563 {
2564 /* try to automatically scroll the guest canvas if the
2565 * mouse goes outside its visible part */
2566
2567 int dx = 0;
2568 if (aPos.x() > vw) dx = aPos.x() - vw;
2569 else if (aPos.x() < 0) dx = aPos.x();
2570 int dy = 0;
2571 if (aPos.y() > vh) dy = aPos.y() - vh;
2572 else if (aPos.y() < 0) dy = aPos.y();
2573 if (dx != 0 || dy != 0) scrollBy (dx, dy);
2574 }
2575
2576 QPoint cpnt = viewportToContents (aPos);
2577 if (cpnt.x() < 0) cpnt.setX (0);
2578 else if (cpnt.x() >= cw) cpnt.setX (cw - 1);
2579 if (cpnt.y() < 0) cpnt.setY (0);
2580 else if (cpnt.y() >= ch) cpnt.setY (ch - 1);
2581
2582 CMouse mouse = mConsole.GetMouse();
2583 mouse.PutMouseEventAbsolute (cpnt.x() + 1, cpnt.y() + 1,
2584 wheel, state);
2585 return true; /* stop further event handling */
2586 }
2587 else
2588 {
2589 if (hasFocus() &&
2590 (aType == QEvent::MouseButtonRelease &&
2591 !aStateAfter))
2592 {
2593 if (isPaused())
2594 {
2595 vboxProblem().remindAboutPausedVMInput();
2596 }
2597 else if (isRunning())
2598 {
2599 /* temporarily disable auto capture that will take
2600 * place after this dialog is dismissed because
2601 * the capture state is to be defined by the
2602 * dialog result itself */
2603 mDisableAutoCapture = true;
2604 bool autoConfirmed = false;
2605 bool ok = vboxProblem().confirmInputCapture (&autoConfirmed);
2606 if (autoConfirmed)
2607 mDisableAutoCapture = false;
2608 /* otherwise, the disable flag will be reset in
2609 * the next console view's foucs in event (since
2610 * may happen asynchronously on some platforms,
2611 * after we return from this code) */
2612
2613 if (ok)
2614 {
2615 captureKbd (true);
2616 captureMouse (true);
2617 }
2618 }
2619 }
2620 }
2621 }
2622
2623 return false;
2624}
2625
2626void VBoxConsoleView::onStateChange (CEnums::MachineState state)
2627{
2628 switch (state)
2629 {
2630 case CEnums::Paused:
2631 {
2632 if (mode != VBoxDefs::TimerMode && mFrameBuf)
2633 {
2634 /*
2635 * Take a screen snapshot. Note that TakeScreenShot() always
2636 * needs a 32bpp image
2637 */
2638 QImage shot = QImage (mFrameBuf->width(), mFrameBuf->height(), 32, 0);
2639 CDisplay dsp = mConsole.GetDisplay();
2640 dsp.TakeScreenShot (shot.bits(), shot.width(), shot.height());
2641 /*
2642 * TakeScreenShot() may fail if, e.g. the Paused notification
2643 * was delivered after the machine execution was resumed. It's
2644 * not fatal.
2645 */
2646 if (dsp.isOk())
2647 {
2648 dimImage (shot);
2649 mPausedShot = shot;
2650 /* fully repaint to pick up mPausedShot */
2651 viewport()->repaint();
2652 }
2653 }
2654 /* fall through */
2655 }
2656 case CEnums::Stuck:
2657 {
2658 /* reuse the focus event handler to uncapture everything */
2659 if (hasFocus())
2660 focusEvent (false /* aHasFocus*/, false /* aReleaseHostKey */);
2661 break;
2662 }
2663 case CEnums::Running:
2664 {
2665 if (last_state == CEnums::Paused)
2666 {
2667 if (mode != VBoxDefs::TimerMode && mFrameBuf)
2668 {
2669 /* reset the pixmap to free memory */
2670 mPausedShot.resize (0, 0);
2671 /*
2672 * ask for full guest display update (it will also update
2673 * the viewport through IFramebuffer::NotifyUpdate)
2674 */
2675 CDisplay dsp = mConsole.GetDisplay();
2676 dsp.InvalidateAndUpdate();
2677 }
2678 }
2679 /* reuse the focus event handler to capture input */
2680 if (hasFocus())
2681 focusEvent (true /* aHasFocus */);
2682 break;
2683 }
2684 default:
2685 break;
2686 }
2687
2688 last_state = state;
2689}
2690
2691void VBoxConsoleView::doRefresh()
2692{
2693#if defined (VBOX_GUI_USE_REFRESH_TIMER)
2694 if (mode == VBoxDefs::TimerMode)
2695 {
2696 FRAMEBUF_DEBUG_START (xxx);
2697 QSize last_sz = pm.size();
2698 bool rc = display_to_pixmap (mConsole, pm);
2699 if (rc)
2700 {
2701 if (pm.size() != last_sz)
2702 {
2703 int pw = pm.width(), ph = pm.height();
2704 resizeContents (pw, ph);
2705 updateGeometry();
2706 setMaximumSize (sizeHint());
2707 /* let our toplevel widget calculate its sizeHint properly */
2708 QApplication::sendPostedEvents (0, QEvent::LayoutHint);
2709 normalizeGeometry();
2710 }
2711 else
2712 {
2713 /* the alternative is to update, so we will be repainted
2714 * on the next event loop iteration. currently disabled.
2715 * updateContents(); */
2716 repaintContents (false);
2717 }
2718 }
2719 if (rc)
2720 FRAMEBUF_DEBUG_STOP (xxx, pm.width(), pm.height());
2721 }
2722 else
2723#endif
2724 repaintContents (false);
2725}
2726
2727void VBoxConsoleView::viewportPaintEvent (QPaintEvent *pe)
2728{
2729#if defined (VBOX_GUI_USE_REFRESH_TIMER)
2730 if (mode == VBoxDefs::TimerMode)
2731 {
2732 if (!pm.isNull())
2733 {
2734 /* draw a part of vbuf */
2735 const QRect &r = pe->rect();
2736 ::bitBlt (viewport(), r.x(), r.y(),
2737 &pm, r.x() + contentsX(), r.y() + contentsY(),
2738 r.width(), r.height(),
2739 CopyROP, TRUE);
2740 }
2741 else
2742 {
2743 viewport()->erase (pe->rect());
2744 }
2745 }
2746 else
2747#endif
2748 {
2749 if (mPausedShot.isNull())
2750 {
2751 /* delegate the paint function to the VBoxFrameBuffer interface */
2752 mFrameBuf->paintEvent (pe);
2753 return;
2754 }
2755
2756 /* we have a snapshot for the paused state */
2757 QRect r = pe->rect().intersect (viewport()->rect());
2758 QPainter pnt (viewport());
2759 pnt.drawPixmap (r.x(), r.y(), mPausedShot,
2760 r.x() + contentsX(), r.y() + contentsY(),
2761 r.width(), r.height());
2762 }
2763}
2764
2765#ifdef VBOX_GUI_USE_REFRESH_TIMER
2766void VBoxConsoleView::timerEvent( QTimerEvent * )
2767{
2768 doRefresh();
2769}
2770#endif
2771
2772/**
2773 * Captures the keyboard. When captured, no keyboard input reaches the host
2774 * system (including most system combinations like Alt-Tab).
2775 *
2776 * @capture true to capture, false to uncapture
2777 * @emit_signal whether to emit keyboardStateChanged() or not
2778 */
2779void VBoxConsoleView::captureKbd (bool capture, bool emit_signal)
2780{
2781 AssertMsg (attached, ("Console must be attached"));
2782
2783 if (kbd_captured == capture)
2784 return;
2785
2786 /* On Win32, keyboard grabbing is ineffective, a low-level keyboard hook is
2787 * used instead. On X11, we use XGrabKey instead of XGrabKeyboard (called
2788 * by QWidget::grabKeyboard()) because the latter causes problems under
2789 * metacity 2.16 (in particular, due to a bug, a window cannot be moved
2790 * using the mouse if it is currently grabing the keyboard). On Mac OS X,
2791 * we use the Qt methods + disabling global hot keys + watching modifiers
2792 * (for right/left separation). */
2793#if defined (Q_WS_WIN32)
2794 /**/
2795#elif defined (Q_WS_X11)
2796 if (capture)
2797 XGrabKey (x11Display(), AnyKey, AnyModifier,
2798 topLevelWidget()->winId(), False,
2799 GrabModeAsync, GrabModeAsync);
2800 else
2801 XUngrabKey (x11Display(), AnyKey, AnyModifier,
2802 topLevelWidget()->winId());
2803#elif defined (Q_WS_MAC)
2804 if (capture)
2805 {
2806 ::DarwinDisableGlobalHotKeys (true);
2807 grabKeyboard();
2808 }
2809 else
2810 {
2811 ::DarwinDisableGlobalHotKeys (false);
2812 releaseKeyboard();
2813 }
2814#else
2815 if (capture)
2816 grabKeyboard();
2817 else
2818 releaseKeyboard();
2819#endif
2820
2821 kbd_captured = capture;
2822
2823 if (emit_signal)
2824 emitKeyboardStateChanged ();
2825}
2826
2827/**
2828 * Captures the host mouse pointer. When captured, the mouse pointer is
2829 * unavailable to the host applications.
2830 *
2831 * @capture true to capture, false to uncapture
2832 */
2833void VBoxConsoleView::captureMouse (bool capture, bool emit_signal)
2834{
2835 AssertMsg (attached, ("Console must be attached"));
2836
2837 if (mouse_captured == capture)
2838 return;
2839
2840 if (capture)
2841 {
2842 /* memorize the host position where the cursor was captured */
2843 captured_pos = QCursor::pos();
2844#ifdef Q_WS_WIN32
2845 viewport()->setCursor (QCursor (BlankCursor));
2846 /* move the mouse to the center of the visible area */
2847 QCursor::setPos (mapToGlobal (visibleRect().center()));
2848 last_pos = QCursor::pos();
2849#elif defined (Q_WS_MAC)
2850 /* move the mouse to the center of the visible area */
2851 last_pos = mapToGlobal (visibleRect().center());
2852 QCursor::setPos (last_pos);
2853 /* grab all mouse events. */
2854 viewport()->grabMouse();
2855#else
2856 viewport()->grabMouse();
2857 last_pos = QCursor::pos();
2858#endif
2859 }
2860 else
2861 {
2862#ifndef Q_WS_WIN32
2863 viewport()->releaseMouse();
2864#endif
2865 /* release mouse buttons */
2866 CMouse mouse = mConsole.GetMouse();
2867 mouse.PutMouseEvent (0, 0, 0, 0);
2868 }
2869
2870 mouse_captured = capture;
2871
2872 updateMouseClipping();
2873
2874 if (emit_signal)
2875 emitMouseStateChanged ();
2876}
2877
2878/**
2879 * Searches for a menu item with a given hot key (shortcut). If the item
2880 * is found, activates it and returns true. Otherwise returns false.
2881 */
2882bool VBoxConsoleView::processHotKey (const QKeySequence &key, QMenuData *data)
2883{
2884 if (!data) return false;
2885
2886 /*
2887 * Note: below, we use the internal class QMenuItem, that is subject
2888 * to change w/o notice... (the alternative would be to explicitly assign
2889 * specific IDs to all popup submenus in VBoxConsoleWnd and then
2890 * look through its children in order to find a popup with a given ID,
2891 * which is unconvenient).
2892 */
2893
2894 for (uint i = 0; i < data->count(); i++)
2895 {
2896 int id = data->idAt (i);
2897 QMenuItem *item = data->findItem (id);
2898 if (item->popup())
2899 {
2900 if (processHotKey (key, item->popup()))
2901 return true;
2902 }
2903 else
2904 {
2905 QStringList list = QStringList::split ("\tHost+", data->text (id));
2906 if (list.count() == 2)
2907 {
2908 if (key.matches (QKeySequence (list[1])) == Identical)
2909 {
2910 /*
2911 * we asynchronously post a special event instead of calling
2912 * data->activateItemAt (i) directly, to let key presses
2913 * and releases be processed correctly by Qt first.
2914 * Note: we assume that nobody will delete the menu item
2915 * corresponding to the key sequence, so that the pointer to
2916 * menu data posted along with the event will remain valid in
2917 * the event handler, at least until the main window is closed.
2918 */
2919
2920 QApplication::postEvent (this,
2921 new ActivateMenuEvent (data, i));
2922 return true;
2923 }
2924 }
2925 }
2926 }
2927
2928 return false;
2929}
2930
2931/**
2932 * Send the KEY BREAK code to the VM for all currently pressed keys.
2933 *
2934 * @param aReleaseHostKey @c true to set the host key state to unpressed.
2935 */
2936void VBoxConsoleView::releaseAllPressedKeys (bool aReleaseHostKey)
2937{
2938 AssertMsg (attached, ("Console must be attached"));
2939
2940 CKeyboard keyboard = mConsole.GetKeyboard();
2941 bool fSentRESEND = false;
2942
2943 /* send a dummy scan code (RESEND) to prevent the guest OS from recognizing
2944 * a single key click (for ex., Alt) and performing an unwanted action
2945 * (for ex., activating the menu) when we release all pressed keys below.
2946 * Note, that it's just a guess that sending RESEND will give the desired
2947 * effect :), but at least it works with NT and W2k guests. */
2948
2949 /// @todo Sending 0xFE is responsible for the warning
2950 //
2951 // ``atkbd.c: Spurious NAK on isa0060/serio0. Some program might
2952 // be trying access hardware directly''
2953 //
2954 // on Linux guests (#1944). It might also be responsible for #1949. Don't
2955 // send this command unless we really have to release any key modifier.
2956 // --frank
2957
2958 for (uint i = 0; i < SIZEOF_ARRAY (mPressedKeys); i++)
2959 {
2960 if (mPressedKeys [i] & IsKeyPressed)
2961 {
2962 if (!fSentRESEND)
2963 {
2964 keyboard.PutScancode (0xFE);
2965 fSentRESEND = true;
2966 }
2967 keyboard.PutScancode (i | 0x80);
2968 }
2969 else if (mPressedKeys [i] & IsExtKeyPressed)
2970 {
2971 if (!fSentRESEND)
2972 {
2973 keyboard.PutScancode (0xFE);
2974 fSentRESEND = true;
2975 }
2976 LONG codes [2];
2977 codes[0] = 0xE0;
2978 codes[1] = i | 0x80;
2979 keyboard.PutScancodes (codes, 2);
2980 }
2981 mPressedKeys [i] = 0;
2982 }
2983
2984 if (aReleaseHostKey)
2985 mIsHostkeyPressed = false;
2986
2987#ifdef Q_WS_MAC
2988 /* clear most of the modifiers. */
2989 mDarwinKeyModifiers &=
2990 alphaLock | kEventKeyModifierNumLockMask |
2991 (aReleaseHostKey ? 0 : ::DarwinKeyCodeToDarwinModifierMask (gs.hostKey()));
2992#endif
2993
2994 emitKeyboardStateChanged ();
2995}
2996
2997void VBoxConsoleView::saveKeyStates()
2998{
2999 ::memcpy (mPressedKeysCopy, mPressedKeys,
3000 SIZEOF_ARRAY (mPressedKeys));
3001}
3002
3003void VBoxConsoleView::sendChangedKeyStates()
3004{
3005 AssertMsg (attached, ("Console must be attached"));
3006
3007 LONG codes [2];
3008 CKeyboard keyboard = mConsole.GetKeyboard();
3009 for (uint i = 0; i < SIZEOF_ARRAY (mPressedKeys); ++ i)
3010 {
3011 uint8_t os = mPressedKeysCopy [i];
3012 uint8_t ns = mPressedKeys [i];
3013 if ((os & IsKeyPressed) != (ns & IsKeyPressed))
3014 {
3015 codes [0] = i;
3016 if (!(ns & IsKeyPressed))
3017 codes[0] |= 0x80;
3018 keyboard.PutScancode (codes[0]);
3019 }
3020 else if ((os & IsExtKeyPressed) != (ns & IsExtKeyPressed))
3021 {
3022 codes [0] = 0xE0;
3023 codes [1] = i;
3024 if (!(ns & IsExtKeyPressed))
3025 codes [1] |= 0x80;
3026 keyboard.PutScancodes (codes, 2);
3027 }
3028 }
3029}
3030
3031void VBoxConsoleView::updateMouseClipping()
3032{
3033 AssertMsg (attached, ("Console must be attached"));
3034
3035 if ( mouse_captured ) {
3036 viewport()->setCursor( QCursor( BlankCursor ) );
3037#ifdef Q_WS_WIN32
3038 QRect r = viewport()->rect();
3039 r.moveTopLeft( viewport()->mapToGlobal( QPoint( 0, 0 ) ) );
3040 RECT rect = { r.left(), r.top(), r.right() + 1, r.bottom() + 1 };
3041 ::ClipCursor( &rect );
3042#endif
3043 } else {
3044#ifdef Q_WS_WIN32
3045 ::ClipCursor( NULL );
3046#endif
3047 /* return the cursor to where it was when we captured it and show it */
3048 QCursor::setPos( captured_pos );
3049 viewport()->unsetCursor();
3050 }
3051}
3052
3053void VBoxConsoleView::setPointerShape (MousePointerChangeEvent *me)
3054{
3055 if (me->shapeData () != NULL)
3056 {
3057 bool ok = false;
3058
3059 const uchar *srcAndMaskPtr = me->shapeData();
3060 uint andMaskSize = (me->width() + 7) / 8 * me->height();
3061 const uchar *srcShapePtr = me->shapeData() + ((andMaskSize + 3) & ~3);
3062 uint srcShapePtrScan = me->width() * 4;
3063
3064#if defined (Q_WS_WIN)
3065
3066 BITMAPV5HEADER bi;
3067 HBITMAP hBitmap;
3068 void *lpBits;
3069
3070 ::ZeroMemory (&bi, sizeof (BITMAPV5HEADER));
3071 bi.bV5Size = sizeof (BITMAPV5HEADER);
3072 bi.bV5Width = me->width();
3073 bi.bV5Height = - (LONG) me->height();
3074 bi.bV5Planes = 1;
3075 bi.bV5BitCount = 32;
3076 bi.bV5Compression = BI_BITFIELDS;
3077 // specifiy a supported 32 BPP alpha format for Windows XP
3078 bi.bV5RedMask = 0x00FF0000;
3079 bi.bV5GreenMask = 0x0000FF00;
3080 bi.bV5BlueMask = 0x000000FF;
3081 if (me->hasAlpha())
3082 bi.bV5AlphaMask = 0xFF000000;
3083 else
3084 bi.bV5AlphaMask = 0;
3085
3086 HDC hdc = GetDC (NULL);
3087
3088 // create the DIB section with an alpha channel
3089 hBitmap = CreateDIBSection (hdc, (BITMAPINFO *) &bi, DIB_RGB_COLORS,
3090 (void **) &lpBits, NULL, (DWORD) 0);
3091
3092 ReleaseDC (NULL, hdc);
3093
3094 HBITMAP hMonoBitmap = NULL;
3095 if (me->hasAlpha())
3096 {
3097 // create an empty mask bitmap
3098 hMonoBitmap = CreateBitmap (me->width(), me->height(), 1, 1, NULL);
3099 }
3100 else
3101 {
3102 /* Word aligned AND mask. Will be allocated and created if necessary. */
3103 uint8_t *pu8AndMaskWordAligned = NULL;
3104
3105 /* Width in bytes of the original AND mask scan line. */
3106 uint32_t cbAndMaskScan = (me->width() + 7) / 8;
3107
3108 if (cbAndMaskScan & 1)
3109 {
3110 /* Original AND mask is not word aligned. */
3111
3112 /* Allocate memory for aligned AND mask. */
3113 pu8AndMaskWordAligned = (uint8_t *)RTMemTmpAllocZ ((cbAndMaskScan + 1) * me->height());
3114
3115 Assert(pu8AndMaskWordAligned);
3116
3117 if (pu8AndMaskWordAligned)
3118 {
3119 /* According to MSDN the padding bits must be 0.
3120 * Compute the bit mask to set padding bits to 0 in the last byte of original AND mask.
3121 */
3122 uint32_t u32PaddingBits = cbAndMaskScan * 8 - me->width();
3123 Assert(u32PaddingBits < 8);
3124 uint8_t u8LastBytesPaddingMask = (uint8_t)(0xFF << u32PaddingBits);
3125
3126 Log(("u8LastBytesPaddingMask = %02X, aligned w = %d, width = %d, cbAndMaskScan = %d\n",
3127 u8LastBytesPaddingMask, (cbAndMaskScan + 1) * 8, me->width(), cbAndMaskScan));
3128
3129 uint8_t *src = (uint8_t *)srcAndMaskPtr;
3130 uint8_t *dst = pu8AndMaskWordAligned;
3131
3132 unsigned i;
3133 for (i = 0; i < me->height(); i++)
3134 {
3135 memcpy (dst, src, cbAndMaskScan);
3136
3137 dst[cbAndMaskScan - 1] &= u8LastBytesPaddingMask;
3138
3139 src += cbAndMaskScan;
3140 dst += cbAndMaskScan + 1;
3141 }
3142 }
3143 }
3144
3145 /* create the AND mask bitmap */
3146 hMonoBitmap = ::CreateBitmap (me->width(), me->height(), 1, 1,
3147 pu8AndMaskWordAligned? pu8AndMaskWordAligned: srcAndMaskPtr);
3148
3149 if (pu8AndMaskWordAligned)
3150 {
3151 RTMemTmpFree (pu8AndMaskWordAligned);
3152 }
3153 }
3154
3155 Assert (hBitmap);
3156 Assert (hMonoBitmap);
3157 if (hBitmap && hMonoBitmap)
3158 {
3159 DWORD *dstShapePtr = (DWORD *) lpBits;
3160
3161 for (uint y = 0; y < me->height(); y ++)
3162 {
3163 memcpy (dstShapePtr, srcShapePtr, srcShapePtrScan);
3164 srcShapePtr += srcShapePtrScan;
3165 dstShapePtr += me->width();
3166 }
3167
3168 ICONINFO ii;
3169 ii.fIcon = FALSE;
3170 ii.xHotspot = me->xHot();
3171 ii.yHotspot = me->yHot();
3172 ii.hbmMask = hMonoBitmap;
3173 ii.hbmColor = hBitmap;
3174
3175 HCURSOR hAlphaCursor = CreateIconIndirect (&ii);
3176 Assert (hAlphaCursor);
3177 if (hAlphaCursor)
3178 {
3179 viewport()->setCursor (QCursor (hAlphaCursor));
3180 ok = true;
3181 if (mAlphaCursor)
3182 DestroyIcon (mAlphaCursor);
3183 mAlphaCursor = hAlphaCursor;
3184 }
3185 }
3186
3187 if (hMonoBitmap)
3188 DeleteObject (hMonoBitmap);
3189 if (hBitmap)
3190 DeleteObject (hBitmap);
3191
3192#elif defined (Q_WS_X11) && !defined (VBOX_WITHOUT_XCURSOR)
3193
3194 XcursorImage *img = XcursorImageCreate (me->width(), me->height());
3195 Assert (img);
3196 if (img)
3197 {
3198 img->xhot = me->xHot();
3199 img->yhot = me->yHot();
3200
3201 XcursorPixel *dstShapePtr = img->pixels;
3202
3203 for (uint y = 0; y < me->height(); y ++)
3204 {
3205 memcpy (dstShapePtr, srcShapePtr, srcShapePtrScan);
3206
3207 if (!me->hasAlpha())
3208 {
3209 /* convert AND mask to the alpha channel */
3210 uchar byte = 0;
3211 for (uint x = 0; x < me->width(); x ++)
3212 {
3213 if (!(x % 8))
3214 byte = *(srcAndMaskPtr ++);
3215 else
3216 byte <<= 1;
3217
3218 if (byte & 0x80)
3219 {
3220 /* Linux doesn't support inverted pixels (XOR ops,
3221 * to be exact) in cursor shapes, so we detect such
3222 * pixels and always replace them with black ones to
3223 * make them visible at least over light colors */
3224 if (dstShapePtr [x] & 0x00FFFFFF)
3225 dstShapePtr [x] = 0xFF000000;
3226 else
3227 dstShapePtr [x] = 0x00000000;
3228 }
3229 else
3230 dstShapePtr [x] |= 0xFF000000;
3231 }
3232 }
3233
3234 srcShapePtr += srcShapePtrScan;
3235 dstShapePtr += me->width();
3236 }
3237
3238 Cursor cur = XcursorImageLoadCursor (x11Display(), img);
3239 Assert (cur);
3240 if (cur)
3241 {
3242 viewport()->setCursor (QCursor (cur));
3243 ok = true;
3244 }
3245
3246 XcursorImageDestroy (img);
3247 }
3248
3249#elif defined(Q_WS_MAC)
3250
3251 /*
3252 * Qt3/Mac only supports black/white cursors and it offers no way
3253 * to create your own cursors here unlike on X11 and Windows.
3254 * Which means we're pretty much forced to do it our own way.
3255 */
3256 int rc;
3257
3258 /* dispose of the old cursor. */
3259 if (!DarwinCursorIsNull (&mDarwinCursor))
3260 {
3261 rc = DarwinCursorDestroy (&mDarwinCursor);
3262 AssertRC (rc);
3263 }
3264
3265 /* create the new cursor */
3266 rc = DarwinCursorCreate (me->width(), me->height(), me->xHot(), me->yHot(), me->hasAlpha(),
3267 srcAndMaskPtr, srcShapePtr, &mDarwinCursor);
3268 AssertRC (rc);
3269 if (VBOX_SUCCESS (rc))
3270 {
3271 /** @todo check current mouse coordinates. */
3272 rc = DarwinCursorSet (&mDarwinCursor);
3273 AssertRC (rc);
3274 }
3275 ok = VBOX_SUCCESS (rc);
3276 NOREF (srcShapePtrScan);
3277
3278#else
3279
3280# warning "port me"
3281
3282#endif
3283 if (!ok)
3284 viewport()->unsetCursor();
3285 }
3286 else
3287 {
3288 /*
3289 * We did not get any shape data
3290 */
3291 if (me->isVisible ())
3292 {
3293 /*
3294 * We're supposed to make the last shape we got visible.
3295 * We don't support that for now...
3296 */
3297 /// @todo viewport()->setCursor (QCursor());
3298 }
3299 else
3300 {
3301 viewport()->setCursor (QCursor::BlankCursor);
3302 }
3303 }
3304}
3305
3306inline QRgb qRgbIntensity (QRgb rgb, int mul, int div)
3307{
3308 int r = qRed (rgb);
3309 int g = qGreen (rgb);
3310 int b = qBlue (rgb);
3311 return qRgb (mul * r / div, mul * g / div, mul * b / div);
3312}
3313
3314/* static */
3315void VBoxConsoleView::dimImage (QImage &img)
3316{
3317 for (int y = 0; y < img.height(); y ++) {
3318 if (y % 2) {
3319 if (img.depth() == 32) {
3320 for (int x = 0; x < img.width(); x ++) {
3321 int gray = qGray (img.pixel (x, y)) / 2;
3322 img.setPixel (x, y, qRgb (gray, gray, gray));
3323// img.setPixel (x, y, qRgbIntensity (img.pixel (x, y), 1, 2));
3324 }
3325 } else {
3326 ::memset (img.scanLine (y), 0, img.bytesPerLine());
3327 }
3328 } else {
3329 if (img.depth() == 32) {
3330 for (int x = 0; x < img.width(); x ++) {
3331 int gray = (2 * qGray (img.pixel (x, y))) / 3;
3332 img.setPixel (x, y, qRgb (gray, gray, gray));
3333// img.setPixel (x, y, qRgbIntensity (img.pixel(x, y), 2, 3));
3334 }
3335 }
3336 }
3337 }
3338}
3339
3340void VBoxConsoleView::doResizeHint (const QSize &aToSize)
3341{
3342 if (mIsAdditionsActive && mAutoresizeGuest)
3343 {
3344 /* If this slot is invoked directly then use the passed size
3345 * otherwise get the available size for the guest display.
3346 * We assume here that the centralWidget() contains this view only
3347 * and gives it all available space. */
3348 QSize sz (aToSize.isValid() ? aToSize : mMainWnd->centralWidget()->size());
3349 if (!aToSize.isValid())
3350 sz -= QSize (frameWidth() * 2, frameWidth() * 2);
3351 LogFlowFunc (("Will suggest %d x %d\n", sz.width(), sz.height()));
3352
3353 mConsole.GetDisplay().SetVideoModeHint (sz.width(), sz.height(), 0, 0);
3354 }
3355}
3356
3357/**
3358 * Sets the the minimum size restriction depending on the auto-resize feature
3359 * state and the current rendering mode.
3360 *
3361 * Currently, the restriction is set only in SDL mode and only when the
3362 * auto-resize feature is inactive. We need to do that because we cannot
3363 * correctly draw in a scrolled window in SDL mode.
3364 *
3365 * In all other modes, or when auto-resize is in force, this function does
3366 * nothing.
3367 */
3368void VBoxConsoleView::maybeRestrictMinimumSize()
3369{
3370 if (mode == VBoxDefs::SDLMode)
3371 {
3372 if (!mIsAdditionsActive || !mAutoresizeGuest)
3373 setMinimumSize (sizeHint());
3374 else
3375 setMinimumSize (0, 0);
3376 }
3377}
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