VirtualBox

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

Last change on this file since 4131 was 4131, checked in by vboxsync, 18 years ago

Main/Frontends: Added missing IConsoleCallback notifications (onDVDDriveChange, onFloppyDriveChange, onNetworkAdapterChange, onSerialPortChange, onParallelPortChange, onVRDPServerChange, onUSBControllerChange, onSharedFolderChange) to track third party configuration requests in frontends.

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