VirtualBox

source: vbox/trunk/src/VBox/Frontends/VirtualBox4/src/VBoxGlobal.cpp@ 7372

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

FE/Qt4: Fixed images in the state indicator tooltips and right click of mouse integration as well as shared folders.

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 139.8 KB
Line 
1/** @file
2 *
3 * VBox frontends: Qt GUI ("VirtualBox"):
4 * VBoxGlobal 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 (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 "VBoxGlobal.h"
20
21#include "VBoxDefs.h"
22#include "VBoxSelectorWnd.h"
23#include "VBoxConsoleWnd.h"
24#include "VBoxProblemReporter.h"
25#include "QIHotKeyEdit.h"
26#include "QIMessageBox.h"
27
28//Added by qt3to4:
29#include <QDesktopWidget>
30#include <QTranslator>
31#include <Q3VBoxLayout>
32#include <Q3Frame>
33#include <QEvent>
34#include <QShowEvent>
35#include <Q3PopupMenu>
36#include <Q3HBoxLayout>
37
38#ifdef VBOX_WITH_REGISTRATION
39#include "VBoxRegistrationDlg.h"
40#endif
41
42#include <qapplication.h>
43#include <qmessagebox.h>
44#include <qpixmap.h>
45#include <qicon.h>
46#include <qwidget.h>
47#include <q3filedialog.h>
48#include <qimage.h>
49#include <qlabel.h>
50#include <qtoolbutton.h>
51#include <qfileinfo.h>
52#include <qdir.h>
53#include <qmutex.h>
54#include <qregexp.h>
55#include <qlocale.h>
56#include <QProcess>
57#include <QList>
58
59#ifdef Q_WS_X11
60#include <q3textbrowser.h>
61#include <qpushbutton.h>
62#include <qlayout.h>
63#endif
64
65
66#if defined (Q_WS_MAC)
67#include <Carbon/Carbon.h> // for HIToolbox/InternetConfig
68#endif
69
70#if defined (Q_WS_WIN)
71#include "shlobj.h"
72#include <qeventloop.h>
73#endif
74
75#if defined (Q_WS_X11)
76#undef BOOL /* typedef CARD8 BOOL in Xmd.h conflicts with #define BOOL PRBool
77 * in COMDefs.h. A better fix would be to isolate X11-specific
78 * stuff by placing XX* helpers below to a separate source file. */
79#include <X11/X.h>
80#include <X11/Xmd.h>
81#include <X11/Xlib.h>
82#include <X11/Xatom.h>
83#define BOOL PRBool
84#endif
85
86#include <iprt/err.h>
87#include <iprt/param.h>
88#include <iprt/path.h>
89#include <iprt/env.h>
90#include <iprt/file.h>
91
92#if defined (Q_WS_X11)
93#include <iprt/mem.h>
94#endif
95
96#if defined (VBOX_GUI_DEBUG)
97uint64_t VMCPUTimer::ticks_per_msec = (uint64_t) -1LL; // declared in VBoxDefs.h
98#endif
99
100// VBoxEnumerateMediaEvent
101/////////////////////////////////////////////////////////////////////////////
102
103class VBoxEnumerateMediaEvent : public QEvent
104{
105public:
106
107 /** Constructs a regular enum event */
108 VBoxEnumerateMediaEvent (const VBoxMedia &aMedia, int aIndex)
109 : QEvent ((QEvent::Type) VBoxDefs::EnumerateMediaEventType)
110 , mMedia (aMedia), mLast (false), mIndex (aIndex)
111 {}
112 /** Constructs the last enum event */
113 VBoxEnumerateMediaEvent()
114 : QEvent ((QEvent::Type) VBoxDefs::EnumerateMediaEventType)
115 , mLast (true), mIndex (-1)
116 {}
117
118 /** the last enumerated media (not valid when #last is true) */
119 const VBoxMedia mMedia;
120 /** whether this is the last event for the given enumeration or not */
121 const bool mLast;
122 /** last enumerated media index (-1 when #last is true) */
123 const int mIndex;
124};
125
126#if defined (Q_WS_WIN)
127class VBoxShellExecuteEvent : public QEvent
128{
129public:
130
131 /** Constructs a regular enum event */
132 VBoxShellExecuteEvent (QThread *aThread, const QString &aURL,
133 bool aOk)
134 : QEvent ((QEvent::Type) VBoxDefs::ShellExecuteEventType)
135 , mThread (aThread), mURL (aURL), mOk (aOk)
136 {}
137
138 QThread *mThread;
139 QString mURL;
140 bool mOk;
141};
142#endif
143
144// VirtualBox callback class
145/////////////////////////////////////////////////////////////////////////////
146
147class VBoxCallback : public IVirtualBoxCallback
148{
149public:
150
151 VBoxCallback (VBoxGlobal &aGlobal)
152 : mGlobal (aGlobal), mIsRegDlgOwner (false)
153 {
154#if defined (Q_OS_WIN32)
155 refcnt = 0;
156#endif
157 }
158
159 virtual ~VBoxCallback() {}
160
161 NS_DECL_ISUPPORTS
162
163#if defined (Q_OS_WIN32)
164 STDMETHOD_(ULONG, AddRef)()
165 {
166 return ::InterlockedIncrement (&refcnt);
167 }
168 STDMETHOD_(ULONG, Release)()
169 {
170 long cnt = ::InterlockedDecrement (&refcnt);
171 if (cnt == 0)
172 delete this;
173 return cnt;
174 }
175 STDMETHOD(QueryInterface) (REFIID riid , void **ppObj)
176 {
177 if (riid == IID_IUnknown) {
178 *ppObj = this;
179 AddRef();
180 return S_OK;
181 }
182 if (riid == IID_IVirtualBoxCallback) {
183 *ppObj = this;
184 AddRef();
185 return S_OK;
186 }
187 *ppObj = NULL;
188 return E_NOINTERFACE;
189 }
190#endif
191
192 // IVirtualBoxCallback methods
193
194 // Note: we need to post custom events to the GUI event queue
195 // instead of doing what we need directly from here because on Win32
196 // these callback methods are never called on the main GUI thread.
197 // Another reason to handle events asynchronously is that internally
198 // most callback interface methods are called from under the initiator
199 // object's lock, so accessing the initiator object (for example, reading
200 // some property) directly from the callback method will definitely cause
201 // a deadlock.
202
203 STDMETHOD(OnMachineStateChange) (IN_GUIDPARAM id, MachineState_T state)
204 {
205 postEvent (new VBoxMachineStateChangeEvent (COMBase::ToQUuid (id),
206 (KMachineState) state));
207 return S_OK;
208 }
209
210 STDMETHOD(OnMachineDataChange) (IN_GUIDPARAM id)
211 {
212 postEvent (new VBoxMachineDataChangeEvent (COMBase::ToQUuid (id)));
213 return S_OK;
214 }
215
216 STDMETHOD(OnExtraDataCanChange)(IN_GUIDPARAM id,
217 IN_BSTRPARAM key, IN_BSTRPARAM value,
218 BSTR *error, BOOL *allowChange)
219 {
220 if (!error || !allowChange)
221 return E_INVALIDARG;
222
223 if (COMBase::ToQUuid (id).isNull())
224 {
225 /* it's a global extra data key someone wants to change */
226 QString sKey = QString::fromUcs2 (key);
227 QString sVal = QString::fromUcs2 (value);
228 if (sKey.startsWith ("GUI/"))
229 {
230 if (sKey == VBoxDefs::GUI_RegistrationDlgWinID)
231 {
232 if (mIsRegDlgOwner)
233 {
234 if (sVal.isEmpty() ||
235 sVal == QString ("%1").arg ((long)qApp->mainWidget()->winId()))
236 *allowChange = TRUE;
237 else
238 *allowChange = FALSE;
239 }
240 else
241 *allowChange = TRUE;
242 return S_OK;
243 }
244
245 /* try to set the global setting to check its syntax */
246 VBoxGlobalSettings gs (false /* non-null */);
247 if (gs.setPublicProperty (sKey, sVal))
248 {
249 /* this is a known GUI property key */
250 if (!gs)
251 {
252 /* disallow the change when there is an error*/
253 *error = SysAllocString ((const OLECHAR *)
254 gs.lastError().ucs2());
255 *allowChange = FALSE;
256 }
257 else
258 *allowChange = TRUE;
259 return S_OK;
260 }
261 }
262 }
263
264 /* not interested in this key -- never disagree */
265 *allowChange = TRUE;
266 return S_OK;
267 }
268
269 STDMETHOD(OnExtraDataChange) (IN_GUIDPARAM id,
270 IN_BSTRPARAM key, IN_BSTRPARAM value)
271 {
272 if (COMBase::ToQUuid (id).isNull())
273 {
274 QString sKey = QString::fromUcs2 (key);
275 QString sVal = QString::fromUcs2 (value);
276 if (sKey.startsWith ("GUI/"))
277 {
278 if (sKey == VBoxDefs::GUI_RegistrationDlgWinID)
279 {
280 if (sVal.isEmpty())
281 {
282 mIsRegDlgOwner = false;
283 QApplication::postEvent (&mGlobal, new VBoxCanShowRegDlgEvent (true));
284 }
285 else if (sVal == QString ("%1").arg ((long) qApp->mainWidget()->winId()))
286 {
287 mIsRegDlgOwner = true;
288 QApplication::postEvent (&mGlobal, new VBoxCanShowRegDlgEvent (true));
289 }
290 else
291 QApplication::postEvent (&mGlobal, new VBoxCanShowRegDlgEvent (false));
292 }
293
294 mMutex.lock();
295 mGlobal.gset.setPublicProperty (sKey, sVal);
296 mMutex.unlock();
297 Assert (!!mGlobal.gset);
298 }
299 }
300 return S_OK;
301 }
302
303 STDMETHOD(OnMediaRegistered) (IN_GUIDPARAM id, DeviceType_T type,
304 BOOL registered)
305 {
306 /** @todo */
307 Q_UNUSED (id);
308 Q_UNUSED (type);
309 Q_UNUSED (registered);
310 return S_OK;
311 }
312
313 STDMETHOD(OnMachineRegistered) (IN_GUIDPARAM id, BOOL registered)
314 {
315 postEvent (new VBoxMachineRegisteredEvent (COMBase::ToQUuid (id),
316 registered));
317 return S_OK;
318 }
319
320 STDMETHOD(OnSessionStateChange) (IN_GUIDPARAM id, SessionState_T state)
321 {
322 postEvent (new VBoxSessionStateChangeEvent (COMBase::ToQUuid (id),
323 (KSessionState) state));
324 return S_OK;
325 }
326
327 STDMETHOD(OnSnapshotTaken) (IN_GUIDPARAM aMachineId, IN_GUIDPARAM aSnapshotId)
328 {
329 postEvent (new VBoxSnapshotEvent (COMBase::ToQUuid (aMachineId),
330 COMBase::ToQUuid (aSnapshotId),
331 VBoxSnapshotEvent::Taken));
332 return S_OK;
333 }
334
335 STDMETHOD(OnSnapshotDiscarded) (IN_GUIDPARAM aMachineId, IN_GUIDPARAM aSnapshotId)
336 {
337 postEvent (new VBoxSnapshotEvent (COMBase::ToQUuid (aMachineId),
338 COMBase::ToQUuid (aSnapshotId),
339 VBoxSnapshotEvent::Discarded));
340 return S_OK;
341 }
342
343 STDMETHOD(OnSnapshotChange) (IN_GUIDPARAM aMachineId, IN_GUIDPARAM aSnapshotId)
344 {
345 postEvent (new VBoxSnapshotEvent (COMBase::ToQUuid (aMachineId),
346 COMBase::ToQUuid (aSnapshotId),
347 VBoxSnapshotEvent::Changed));
348 return S_OK;
349 }
350
351private:
352
353 void postEvent (QEvent *e)
354 {
355 // currently, we don't post events if we are in the VM execution
356 // console mode, to save some CPU ticks (so far, there was no need
357 // to handle VirtualBox callback events in the execution console mode)
358
359 if (!mGlobal.isVMConsoleProcess())
360 QApplication::postEvent (&mGlobal, e);
361 }
362
363 VBoxGlobal &mGlobal;
364
365 /** protects #OnExtraDataChange() */
366 QMutex mMutex;
367
368 bool mIsRegDlgOwner;
369
370#if defined (Q_OS_WIN32)
371private:
372 long refcnt;
373#endif
374};
375
376#if !defined (Q_OS_WIN32)
377NS_DECL_CLASSINFO (VBoxCallback)
378NS_IMPL_THREADSAFE_ISUPPORTS1_CI (VBoxCallback, IVirtualBoxCallback)
379#endif
380
381// Helpers for VBoxGlobal::getOpenFileName() & getExistingDirectory()
382/////////////////////////////////////////////////////////////////////////////
383
384#if defined Q_WS_WIN
385
386extern void qt_enter_modal (QWidget*);
387extern void qt_leave_modal (QWidget*);
388
389static QString extractFilter (const QString &aRawFilter)
390{
391 static const char qt_file_dialog_filter_reg_exp[] =
392 "([a-zA-Z0-9 ]*)\\(([a-zA-Z0-9_.*? +;#\\[\\]]*)\\)$";
393
394 QString result = aRawFilter;
395 QRegExp r (QString::fromLatin1 (qt_file_dialog_filter_reg_exp));
396 int index = r.search (result);
397 if (index >= 0)
398 result = r.cap (2);
399 return result.replace (QChar (' '), QChar (';'));
400}
401
402/**
403 * Converts QFileDialog filter list to Win32 API filter list.
404 */
405static QString winFilter (const QString &aFilter)
406{
407 QStringList filterLst;
408
409 if (!aFilter.isEmpty())
410 {
411 int i = aFilter.find (";;", 0);
412 QString sep (";;");
413 if (i == -1)
414 {
415 if (aFilter.find ("\n", 0) != -1)
416 {
417 sep = "\n";
418 i = aFilter.find (sep, 0);
419 }
420 }
421
422 filterLst = QStringList::split (sep, aFilter);
423 }
424
425 QStringList::Iterator it = filterLst.begin();
426 QString winfilters;
427 for (; it != filterLst.end(); ++it)
428 {
429 winfilters += *it;
430 winfilters += QChar::Null;
431 winfilters += extractFilter (*it);
432 winfilters += QChar::Null;
433 }
434 winfilters += QChar::Null;
435 return winfilters;
436}
437
438/*
439 * Callback function to control the native Win32 API file dialog
440 */
441UINT_PTR CALLBACK OFNHookProc (HWND aHdlg, UINT aUiMsg, WPARAM aWParam, LPARAM aLParam)
442{
443 if (aUiMsg == WM_NOTIFY)
444 {
445 OFNOTIFY *notif = (OFNOTIFY*) aLParam;
446 if (notif->hdr.code == CDN_TYPECHANGE)
447 {
448 /* locate native dialog controls */
449 HWND parent = GetParent (aHdlg);
450 HWND button = GetDlgItem (parent, IDOK);
451 HWND textfield = ::GetDlgItem (parent, cmb13);
452 if (textfield == NULL)
453 textfield = ::GetDlgItem (parent, edt1);
454 if (textfield == NULL)
455 return FALSE;
456 HWND selector = ::GetDlgItem (parent, cmb1);
457
458 /* simulate filter change by pressing apply-key */
459 int size = 256;
460 TCHAR *buffer = (TCHAR*)malloc (size);
461 SendMessage (textfield, WM_GETTEXT, size, (LPARAM)buffer);
462 SendMessage (textfield, WM_SETTEXT, 0, (LPARAM)"\0");
463 SendMessage (button, BM_CLICK, 0, 0);
464 SendMessage (textfield, WM_SETTEXT, 0, (LPARAM)buffer);
465 free (buffer);
466
467 /* make request for focus moving to filter selector combo-box */
468 HWND curFocus = GetFocus();
469 PostMessage (curFocus, WM_KILLFOCUS, (WPARAM)selector, 0);
470 PostMessage (selector, WM_SETFOCUS, (WPARAM)curFocus, 0);
471 WPARAM wParam = MAKEWPARAM (WA_ACTIVE, 0);
472 PostMessage (selector, WM_ACTIVATE, wParam, (LPARAM)curFocus);
473 }
474 }
475 return FALSE;
476}
477
478/*
479 * Callback function to control the native Win32 API folders dialog
480 */
481static int __stdcall winGetExistDirCallbackProc (HWND hwnd, UINT uMsg,
482 LPARAM lParam, LPARAM lpData)
483{
484 if (uMsg == BFFM_INITIALIZED && lpData != 0)
485 {
486 QString *initDir = (QString *)(lpData);
487 if (!initDir->isEmpty())
488 {
489 SendMessage (hwnd, BFFM_SETSELECTION, TRUE, Q_ULONG (initDir->ucs2()));
490 //SendMessage (hwnd, BFFM_SETEXPANDED, TRUE, Q_ULONG (initDir->ucs2()));
491 }
492 }
493 else if (uMsg == BFFM_SELCHANGED)
494 {
495 TCHAR path [MAX_PATH];
496 SHGetPathFromIDList (LPITEMIDLIST (lParam), path);
497 QString tmpStr = QString::fromUcs2 ((ushort*)path);
498 if (!tmpStr.isEmpty())
499 SendMessage (hwnd, BFFM_ENABLEOK, 1, 1);
500 else
501 SendMessage (hwnd, BFFM_ENABLEOK, 0, 0);
502 SendMessage (hwnd, BFFM_SETSTATUSTEXT, 1, Q_ULONG (path));
503 }
504 return 0;
505}
506
507/**
508 * QEvent class to carry Win32 API native dialog's result information
509 */
510class OpenNativeDialogEvent : public QEvent
511{
512public:
513
514 OpenNativeDialogEvent (const QString &aResult, QEvent::Type aType)
515 : QEvent (aType), mResult (aResult) {}
516
517 const QString& result() { return mResult; }
518
519private:
520
521 QString mResult;
522};
523
524/**
525 * QObject class reimplementation which is the target for OpenNativeDialogEvent
526 * event. It receives OpenNativeDialogEvent event from another thread,
527 * stores result information and exits event processing loop.
528 */
529class LoopObject : public QObject
530{
531public:
532
533 LoopObject (QEvent::Type aType) : mType (aType), mResult (QString::null) {}
534 const QString& result() { return mResult; }
535
536private:
537
538 bool event (QEvent *aEvent)
539 {
540 if (aEvent->type() == mType)
541 {
542 OpenNativeDialogEvent *ev = (OpenNativeDialogEvent*) aEvent;
543 mResult = ev->result();
544 qApp->eventLoop()->exitLoop();
545 return true;
546 }
547 return QObject::event (aEvent);
548 }
549
550 QEvent::Type mType;
551 QString mResult;
552};
553
554#endif /* Q_WS_WIN */
555
556#ifdef Q_WS_X11
557/**
558 * This class is used to show a user license under linux.
559 */
560class VBoxLicenseViewer : public QDialog
561{
562 Q_OBJECT
563
564public:
565
566 VBoxLicenseViewer (const QString &aFilePath)
567 : QDialog (0, "VBoxLicenseViewerObject")
568 , mFilePath (aFilePath)
569 , mLicenseText (0), mAgreeButton (0), mDisagreeButton (0)
570 {
571 setCaption ("VirtualBox License");
572 setIcon (QPixmap (":/ico40x01.png"));
573
574 mLicenseText = new Q3TextBrowser (this);
575 mAgreeButton = new QPushButton (tr ("I &Agree"), this);
576 mDisagreeButton = new QPushButton (tr ("I &Disagree"), this);
577
578 mLicenseText->setTextFormat (Qt::RichText);
579
580 connect (mLicenseText->verticalScrollBar(), SIGNAL (valueChanged (int)),
581 SLOT (onScrollBarMoving (int)));
582 connect (mAgreeButton, SIGNAL (clicked()), SLOT (accept()));
583 connect (mDisagreeButton, SIGNAL (clicked()), SLOT (reject()));
584
585 Q3VBoxLayout *mainLayout = new Q3VBoxLayout (this, 10, 10);
586 mainLayout->addWidget (mLicenseText);
587
588 Q3HBoxLayout *buttonLayout = new Q3HBoxLayout (mainLayout, 10);
589 buttonLayout->addItem (new QSpacerItem (0, 0, QSizePolicy::Expanding,
590 QSizePolicy::Preferred));
591 buttonLayout->addWidget (mAgreeButton);
592 buttonLayout->addWidget (mDisagreeButton);
593
594 mLicenseText->verticalScrollBar()->installEventFilter (this);
595
596 resize (600, 450);
597 }
598
599public slots:
600
601 int exec()
602 {
603 /* read & show the license file */
604 QFile file (mFilePath);
605 if (file.open (QIODevice::ReadOnly))
606 {
607 mLicenseText->setText (file.readAll());
608 return QDialog::exec();
609 }
610 else
611 {
612 vboxProblem().cannotOpenLicenseFile (this, mFilePath);
613 return QDialog::Rejected;
614 }
615 }
616
617private slots:
618
619 void onScrollBarMoving (int aValue)
620 {
621 if (aValue == mLicenseText->verticalScrollBar()->maxValue())
622 unlockButtons();
623 }
624
625 void unlockButtons()
626 {
627 mAgreeButton->setEnabled (true);
628 mDisagreeButton->setEnabled (true);
629 }
630
631private:
632
633 void showEvent (QShowEvent *aEvent)
634 {
635 QDialog::showEvent (aEvent);
636 bool isScrollBarHidden = mLicenseText->verticalScrollBar()->isHidden()
637 && !(windowState() & Qt::WindowMinimized);
638 mAgreeButton->setEnabled (isScrollBarHidden);
639 mDisagreeButton->setEnabled (isScrollBarHidden);
640 }
641
642 bool eventFilter (QObject *aObject, QEvent *aEvent)
643 {
644 switch (aEvent->type())
645 {
646 case QEvent::Hide:
647 if (aObject == mLicenseText->verticalScrollBar() &&
648 (windowState() & Qt::WindowActive))
649 unlockButtons();
650 default:
651 break;
652 }
653 return QDialog::eventFilter (aObject, aEvent);
654 }
655
656 QString mFilePath;
657 Q3TextBrowser *mLicenseText;
658 QPushButton *mAgreeButton;
659 QPushButton *mDisagreeButton;
660};
661#endif
662
663
664// VBoxGlobal
665////////////////////////////////////////////////////////////////////////////////
666
667static bool sVBoxGlobalInited = false;
668static bool sVBoxGlobalInCleanup = false;
669
670/** @internal
671 *
672 * Special routine to do VBoxGlobal cleanup when the application is being
673 * terminated. It is called before some essential Qt functionality (for
674 * instance, QThread) becomes unavailable, allowing us to use it from
675 * VBoxGlobal::cleanup() if necessary.
676 */
677static void vboxGlobalCleanup()
678{
679 Assert (!sVBoxGlobalInCleanup);
680 sVBoxGlobalInCleanup = true;
681 vboxGlobal().cleanup();
682}
683
684/** @internal
685 *
686 * Determines the rendering mode from the argument. Sets the appropriate
687 * default rendering mode if the argumen is NULL.
688 */
689static VBoxDefs::RenderMode vboxGetRenderMode (const char *aModeStr)
690{
691 VBoxDefs::RenderMode mode = VBoxDefs::InvalidRenderMode;
692
693#if defined (Q_WS_MAC) && defined (VBOX_GUI_USE_QUARTZ2D)
694 mode = VBoxDefs::Quartz2DMode;
695#elif (defined (Q_WS_WIN32) || defined (Q_WS_PM)) && defined (VBOX_GUI_USE_QIMAGE)
696 mode = VBoxDefs::QImageMode;
697#elif defined (Q_WS_X11) && defined (VBOX_GUI_USE_SDL)
698 mode = VBoxDefs::SDLMode;
699#elif defined (VBOX_GUI_USE_QIMAGE)
700 mode = VBoxDefs::QImageMode;
701#else
702# error "Cannot determine the default render mode!"
703#endif
704
705 if (aModeStr)
706 {
707 if (0) ;
708#if defined (VBOX_GUI_USE_REFRESH_TIMER)
709 else if (::strcmp (aModeStr, "timer") == 0)
710 mode = VBoxDefs::TimerMode;
711#endif
712#if defined (VBOX_GUI_USE_QIMAGE)
713 else if (::strcmp (aModeStr, "image") == 0)
714 mode = VBoxDefs::QImageMode;
715#endif
716#if defined (VBOX_GUI_USE_SDL)
717 else if (::strcmp (aModeStr, "sdl") == 0)
718 mode = VBoxDefs::SDLMode;
719#endif
720#if defined (VBOX_GUI_USE_DDRAW)
721 else if (::strcmp (aModeStr, "ddraw") == 0)
722 mode = VBoxDefs::DDRAWMode;
723#endif
724#if defined (VBOX_GUI_USE_QUARTZ2D)
725 else if (::strcmp (aModeStr, "quartz2d") == 0)
726 mode = VBoxDefs::Quartz2DMode;
727#endif
728 }
729
730 return mode;
731}
732
733/** @class VBoxGlobal
734 *
735 * The VBoxGlobal class incapsulates the global VirtualBox data.
736 *
737 * There is only one instance of this class per VirtualBox application,
738 * the reference to it is returned by the static instance() method, or by
739 * the global vboxGlobal() function, that is just an inlined shortcut.
740 */
741
742VBoxGlobal::VBoxGlobal()
743 : mValid (false)
744 , mSelectorWnd (NULL), mConsoleWnd (NULL)
745#ifdef VBOX_WITH_REGISTRATION
746 , mRegDlg (NULL)
747#endif
748 , media_enum_thread (NULL)
749 , verString ("1.0")
750 , vm_state_color (KMachineState_COUNT)
751 , machineStates (KMachineState_COUNT)
752 , sessionStates (KSessionState_COUNT)
753 , deviceTypes (KDeviceType_COUNT)
754 , diskControllerTypes (KDiskControllerType_COUNT)
755 , diskTypes (KHardDiskType_COUNT)
756 , diskStorageTypes (KHardDiskStorageType_COUNT)
757 , vrdpAuthTypes (KVRDPAuthType_COUNT)
758 , portModeTypes (KPortMode_COUNT)
759 , usbFilterActionTypes (KUSBDeviceFilterAction_COUNT)
760 , diskControllerDevices (3)
761 , audioDriverTypes (KAudioDriverType_COUNT)
762 , audioControllerTypes (KAudioControllerType_COUNT)
763 , networkAdapterTypes (KNetworkAdapterType_COUNT)
764 , networkAttachmentTypes (KNetworkAttachmentType_COUNT)
765 , clipboardTypes (KClipboardMode_COUNT)
766 , ideControllerTypes (KIDEControllerType_COUNT)
767 , USBDeviceStates (KUSBDeviceState_COUNT)
768 , detailReportTemplatesReady (false)
769{
770}
771
772//
773// Public members
774/////////////////////////////////////////////////////////////////////////////
775
776/**
777 * Returns a reference to the global VirtualBox data, managed by this class.
778 *
779 * The main() function of the VBox GUI must call this function soon after
780 * creating a QApplication instance but before opening any of the main windows
781 * (to let the VBoxGlobal initialization procedure use various Qt facilities),
782 * and continue execution only when the isValid() method of the returned
783 * instancereturns true, i.e. do something like:
784 *
785 * @code
786 * if ( !VBoxGlobal::instance().isValid() )
787 * return 1;
788 * @endcode
789 * or
790 * @code
791 * if ( !vboxGlobal().isValid() )
792 * return 1;
793 * @endcode
794 *
795 * @note Some VBoxGlobal methods can be used on a partially constructed
796 * VBoxGlobal instance, i.e. from constructors and methods called
797 * from the VBoxGlobal::init() method, which obtain the instance
798 * using this instance() call or the ::vboxGlobal() function. Currently, such
799 * methods are:
800 * #vmStateText, #vmTypeIcon, #vmTypeText, #vmTypeTextList, #vmTypeFromText.
801 *
802 * @see ::vboxGlobal
803 */
804VBoxGlobal &VBoxGlobal::instance()
805{
806 static VBoxGlobal vboxGlobal_instance;
807
808 if (!sVBoxGlobalInited)
809 {
810 /* check that a QApplication instance is created */
811 if (qApp)
812 {
813 sVBoxGlobalInited = true;
814 vboxGlobal_instance.init();
815 /* add our cleanup handler to the list of Qt post routines */
816 qAddPostRoutine (vboxGlobalCleanup);
817 }
818 else
819 AssertMsgFailed (("Must construct a QApplication first!"));
820 }
821 return vboxGlobal_instance;
822}
823
824/**
825 * Sets the new global settings and saves them to the VirtualBox server.
826 */
827bool VBoxGlobal::setSettings (const VBoxGlobalSettings &gs)
828{
829 gs.save (mVBox);
830
831 if (!mVBox.isOk())
832 {
833 vboxProblem().cannotSaveGlobalConfig (mVBox);
834 return false;
835 }
836
837 /* We don't assign gs to our gset member here, because VBoxCallback
838 * will update gset as necessary when new settings are successfullly
839 * sent to the VirtualBox server by gs.save(). */
840
841 return true;
842}
843
844/**
845 * Returns a reference to the main VBox VM Selector window.
846 * The reference is valid until application termination.
847 *
848 * There is only one such a window per VirtualBox application.
849 */
850VBoxSelectorWnd &VBoxGlobal::selectorWnd()
851{
852#if defined (VBOX_GUI_SEPARATE_VM_PROCESS)
853 AssertMsg (!vboxGlobal().isVMConsoleProcess(),
854 ("Must NOT be a VM console process"));
855#endif
856
857 Assert (mValid);
858
859 if (!mSelectorWnd)
860 {
861 /*
862 * We pass the address of mSelectorWnd to the constructor to let it be
863 * initialized right after the constructor is called. It is necessary
864 * to avoid recursion, since this method may be (and will be) called
865 * from the below constructor or from constructors/methods it calls.
866 */
867 VBoxSelectorWnd *w = new VBoxSelectorWnd (&mSelectorWnd, 0);
868 Assert (w == mSelectorWnd);
869 NOREF(w);
870 }
871
872 return *mSelectorWnd;
873}
874
875/**
876 * Returns a reference to the main VBox VM Console window.
877 * The reference is valid until application termination.
878 *
879 * There is only one such a window per VirtualBox application.
880 */
881VBoxConsoleWnd &VBoxGlobal::consoleWnd()
882{
883#if defined (VBOX_GUI_SEPARATE_VM_PROCESS)
884 AssertMsg (vboxGlobal().isVMConsoleProcess(),
885 ("Must be a VM console process"));
886#endif
887
888 Assert (mValid);
889
890 if (!mConsoleWnd)
891 {
892 /*
893 * We pass the address of mConsoleWnd to the constructor to let it be
894 * initialized right after the constructor is called. It is necessary
895 * to avoid recursion, since this method may be (and will be) called
896 * from the below constructor or from constructors/methods it calls.
897 */
898 VBoxConsoleWnd *w = new VBoxConsoleWnd (&mConsoleWnd, 0, "consoleWnd");
899 Assert (w == mConsoleWnd);
900 NOREF(w);
901 }
902
903 return *mConsoleWnd;
904}
905
906/**
907 * Returns the list of all guest OS type descriptions, queried from
908 * IVirtualBox.
909 */
910QStringList VBoxGlobal::vmGuestOSTypeDescriptions() const
911{
912 static QStringList list;
913 if (list.empty()) {
914 for (int i = 0; i < vm_os_types.count(); i++) {
915 list += vm_os_types [i].GetDescription();
916 }
917 }
918 return list;
919}
920
921/**
922 * Returns the guest OS type object corresponding to the given index.
923 * The index argument corresponds to the index in the list of OS type
924 * descriptions as returnded by #vmGuestOSTypeDescriptions().
925 *
926 * If the index is invalid a null object is returned.
927 */
928CGuestOSType VBoxGlobal::vmGuestOSType (int aIndex) const
929{
930 CGuestOSType type;
931 if (aIndex >= 0 && aIndex < (int) vm_os_types.count())
932 type = vm_os_types [aIndex];
933 AssertMsg (!type.isNull(), ("Index for OS type must be valid: %d", aIndex));
934 return type;
935}
936
937/**
938 * Returns the index corresponding to the given guest OS type ID.
939 * The returned index corresponds to the index in the list of OS type
940 * descriptions as returnded by #vmGuestOSTypeDescriptions().
941 *
942 * If the guest OS type ID is invalid, -1 is returned.
943 */
944int VBoxGlobal::vmGuestOSTypeIndex (const QString &aId) const
945{
946 for (int i = 0; i < (int) vm_os_types.count(); i++) {
947 if (!vm_os_types [i].GetId().compare (aId))
948 return i;
949 }
950 return -1;
951}
952
953/**
954 * Returns the icon corresponding to the given guest OS type ID.
955 */
956QPixmap VBoxGlobal::vmGuestOSTypeIcon (const QString &aId) const
957{
958 static const QPixmap none;
959 QPixmap *p = vm_os_type_icons [aId];
960 AssertMsg (p, ("Icon for type `%s' must be defined", aId.latin1()));
961 return p ? *p : none;
962}
963
964/**
965 * Returns the description corresponding to the given guest OS type ID.
966 */
967QString VBoxGlobal::vmGuestOSTypeDescription (const QString &aId) const
968{
969 for (int i = 0; i < (int) vm_os_types.count(); i++) {
970 if (!vm_os_types [i].GetId().compare (aId))
971 return vm_os_types [i].GetDescription();
972 }
973 return QString::null;
974}
975
976QString VBoxGlobal::toString (KDiskControllerType t, LONG d) const
977{
978 Assert (diskControllerDevices.count() == 3);
979 QString dev;
980 switch (t)
981 {
982 case KDiskControllerType_IDE0:
983 case KDiskControllerType_IDE1:
984 {
985 if (d == 0 || d == 1)
986 {
987 dev = diskControllerDevices [d];
988 break;
989 }
990 // fall through
991 }
992 default:
993 dev = diskControllerDevices [2].arg (d);
994 }
995 return dev;
996}
997
998/**
999 * Returns the list of all device types (VurtialBox::DeviceType COM enum).
1000 */
1001QStringList VBoxGlobal::deviceTypeStrings() const
1002{
1003 static QStringList list;
1004 if (list.empty())
1005 for (int i = 0; i < deviceTypes.count() - 1 /* usb=n/a */; i++)
1006 list += deviceTypes [i];
1007 return list;
1008}
1009
1010struct PortConfig
1011{
1012 const char *name;
1013 const ulong IRQ;
1014 const ulong IOBase;
1015};
1016
1017static const PortConfig comKnownPorts[] =
1018{
1019 { "COM1", 4, 0x3F8 },
1020 { "COM2", 3, 0x2F8 },
1021 { "COM3", 4, 0x3E8 },
1022 { "COM4", 3, 0x2E8 },
1023 /* must not contain an element with IRQ=0 and IOBase=0 used to cause
1024 * toCOMPortName() to return the "User-defined" string for these values. */
1025};
1026
1027static const PortConfig lptKnownPorts[] =
1028{
1029 { "LPT1", 7, 0x3BC },
1030 { "LPT2", 5, 0x378 },
1031 { "LPT3", 5, 0x278 },
1032 /* must not contain an element with IRQ=0 and IOBase=0 used to cause
1033 * toLPTPortName() to return the "User-defined" string for these values. */
1034};
1035
1036/**
1037 * Returns the list of the standard COM port names (i.e. "COMx").
1038 */
1039QStringList VBoxGlobal::COMPortNames() const
1040{
1041 QStringList list;
1042 for (size_t i = 0; i < ELEMENTS (comKnownPorts); ++ i)
1043 list << comKnownPorts [i].name;
1044
1045 return list;
1046}
1047
1048/**
1049 * Returns the list of the standard LPT port names (i.e. "LPTx").
1050 */
1051QStringList VBoxGlobal::LPTPortNames() const
1052{
1053 QStringList list;
1054 for (size_t i = 0; i < ELEMENTS (lptKnownPorts); ++ i)
1055 list << lptKnownPorts [i].name;
1056
1057 return list;
1058}
1059
1060/**
1061 * Returns the name of the standard COM port corresponding to the given
1062 * parameters, or "User-defined" (which is also returned when both
1063 * @a aIRQ and @a aIOBase are 0).
1064 */
1065QString VBoxGlobal::toCOMPortName (ulong aIRQ, ulong aIOBase) const
1066{
1067 for (size_t i = 0; i < ELEMENTS (comKnownPorts); ++ i)
1068 if (comKnownPorts [i].IRQ == aIRQ &&
1069 comKnownPorts [i].IOBase == aIOBase)
1070 return comKnownPorts [i].name;
1071
1072 return mUserDefinedPortName;
1073}
1074
1075/**
1076 * Returns the name of the standard LPT port corresponding to the given
1077 * parameters, or "User-defined" (which is also returned when both
1078 * @a aIRQ and @a aIOBase are 0).
1079 */
1080QString VBoxGlobal::toLPTPortName (ulong aIRQ, ulong aIOBase) const
1081{
1082 for (size_t i = 0; i < ELEMENTS (lptKnownPorts); ++ i)
1083 if (lptKnownPorts [i].IRQ == aIRQ &&
1084 lptKnownPorts [i].IOBase == aIOBase)
1085 return lptKnownPorts [i].name;
1086
1087 return mUserDefinedPortName;
1088}
1089
1090/**
1091 * Returns port parameters corresponding to the given standard COM name.
1092 * Returns @c true on success, or @c false if the given port name is not one
1093 * of the standard names (i.e. "COMx").
1094 */
1095bool VBoxGlobal::toCOMPortNumbers (const QString &aName, ulong &aIRQ,
1096 ulong &aIOBase) const
1097{
1098 for (size_t i = 0; i < ELEMENTS (comKnownPorts); ++ i)
1099 if (strcmp (comKnownPorts [i].name, aName.utf8().data()) == 0)
1100 {
1101 aIRQ = comKnownPorts [i].IRQ;
1102 aIOBase = comKnownPorts [i].IOBase;
1103 return true;
1104 }
1105
1106 return false;
1107}
1108
1109/**
1110 * Returns port parameters corresponding to the given standard LPT name.
1111 * Returns @c true on success, or @c false if the given port name is not one
1112 * of the standard names (i.e. "LPTx").
1113 */
1114bool VBoxGlobal::toLPTPortNumbers (const QString &aName, ulong &aIRQ,
1115 ulong &aIOBase) const
1116{
1117 for (size_t i = 0; i < ELEMENTS (lptKnownPorts); ++ i)
1118 if (strcmp (lptKnownPorts [i].name, aName.utf8().data()) == 0)
1119 {
1120 aIRQ = lptKnownPorts [i].IRQ;
1121 aIOBase = lptKnownPorts [i].IOBase;
1122 return true;
1123 }
1124
1125 return false;
1126}
1127
1128/**
1129 * Returns the details of the given hard disk as a single-line string
1130 * to be used in the VM details view.
1131 *
1132 * The details include the type and the virtual size of the hard disk.
1133 * Note that for differencing hard disks based on immutable hard disks,
1134 * the Immutable hard disk type is returned.
1135 *
1136 * @param aHD hard disk image (when predict = true, must be a top-level image)
1137 * @param aPredict when true, the function predicts the type of the resulting
1138 * image after attaching the given image to the machine.
1139 * Otherwise, a real type of the given image is returned
1140 * (with the exception mentioned above).
1141 *
1142 * @note The hard disk object may become uninitialized by a third party
1143 * while this method is reading its properties. In this case, the method will
1144 * return an empty string.
1145 */
1146QString VBoxGlobal::details (const CHardDisk &aHD, bool aPredict /* = false */,
1147 bool aDoRefresh)
1148{
1149 Assert (!aPredict || aHD.GetParent().isNull());
1150
1151 VBoxMedia media;
1152 if (!aDoRefresh)
1153 media = VBoxMedia (CUnknown (aHD), VBoxDefs::HD, VBoxMedia::Ok);
1154 else if (!findMedia (CUnknown (aHD), media))
1155 {
1156 /* media may be new and not alredy in the media list, request refresh */
1157 startEnumeratingMedia();
1158 if (!findMedia (CUnknown (aHD), media))
1159 AssertFailed();
1160 }
1161
1162 CHardDisk root = aHD.GetRoot();
1163
1164 // @todo *** this check is rough; if aHD becomes uninitialized, any of aHD
1165 // getters called afterwards will also fail. The same relates to the root
1166 // object (that will be aHD itself in case of non-differencing
1167 // disks). However, this check was added to fix a particular use case:
1168 // when aHD is a differencing hard disk and it happens to be discarded
1169 // (and uninitialized) after this method is called but before we read all
1170 // its properties (yes, it's possible!), the root object will be null and
1171 // calling methods on it will assert in the debug builds. This check seems
1172 // to be enough as a quick solution (fresh hard disk attachments will be
1173 // re-read by a state change signal after the discard operation is
1174 // finished, so the user will eventually see correct data), but in order
1175 // to solve the problem properly we need to use exceptions everywhere (or
1176 // check the result after every method call). See also Comment #17 and
1177 // below in Defect #2126.
1178 if (!aHD.isOk())
1179 return QString::null;
1180
1181 QString details;
1182
1183 KHardDiskType type = root.GetType();
1184
1185 if (type == KHardDiskType_Normal &&
1186 (aHD != root || (aPredict && root.GetChildren().GetCount() != 0)))
1187 details = tr ("Differencing", "hard disk");
1188 else
1189 details = hardDiskTypeString (root);
1190
1191 details += ", ";
1192
1193 /// @todo prepend the details with the warning/error
1194 // icon when not accessible
1195
1196 switch (media.status)
1197 {
1198 case VBoxMedia::Unknown:
1199 details += tr ("<i>Checking...</i>", "hard disk");
1200 break;
1201 case VBoxMedia::Ok:
1202 details += formatSize (root.GetSize() * _1M);
1203 break;
1204 case VBoxMedia::Error:
1205 case VBoxMedia::Inaccessible:
1206 details += tr ("<i>Inaccessible</i>", "hard disk");
1207 break;
1208 }
1209
1210 return details;
1211}
1212
1213/**
1214 * Returns the details of the given USB device as a single-line string.
1215 */
1216QString VBoxGlobal::details (const CUSBDevice &aDevice) const
1217{
1218 QString details;
1219 QString m = aDevice.GetManufacturer().stripWhiteSpace();
1220 QString p = aDevice.GetProduct().stripWhiteSpace();
1221 if (m.isEmpty() && p.isEmpty())
1222 {
1223 details =
1224 tr ("Unknown device %1:%2", "USB device details")
1225 .arg (QString().sprintf ("%04hX", aDevice.GetVendorId()))
1226 .arg (QString().sprintf ("%04hX", aDevice.GetProductId()));
1227 }
1228 else
1229 {
1230 if (p.upper().startsWith (m.upper()))
1231 details = p;
1232 else
1233 details = m + " " + p;
1234 }
1235 ushort r = aDevice.GetRevision();
1236 if (r != 0)
1237 details += QString().sprintf (" [%04hX]", r);
1238
1239 return details.stripWhiteSpace();
1240}
1241
1242/**
1243 * Returns the multi-line description of the given USB device.
1244 */
1245QString VBoxGlobal::toolTip (const CUSBDevice &aDevice) const
1246{
1247 QString tip =
1248 tr ("<nobr>Vendor ID: %1</nobr><br>"
1249 "<nobr>Product ID: %2</nobr><br>"
1250 "<nobr>Revision: %3</nobr>", "USB device tooltip")
1251 .arg (QString().sprintf ("%04hX", aDevice.GetVendorId()))
1252 .arg (QString().sprintf ("%04hX", aDevice.GetProductId()))
1253 .arg (QString().sprintf ("%04hX", aDevice.GetRevision()));
1254
1255 QString ser = aDevice.GetSerialNumber();
1256 if (!ser.isEmpty())
1257 tip += QString (tr ("<br><nobr>Serial No. %1</nobr>", "USB device tooltip"))
1258 .arg (ser);
1259
1260 /* add the state field if it's a host USB device */
1261 CHostUSBDevice hostDev = CUnknown (aDevice);
1262 if (!hostDev.isNull())
1263 {
1264 tip += QString (tr ("<br><nobr>State: %1</nobr>", "USB device tooltip"))
1265 .arg (vboxGlobal().toString (hostDev.GetState()));
1266 }
1267
1268 return tip;
1269}
1270
1271/**
1272 * Puts soft hyphens after every path component in the given file name.
1273 * @param fn file name (must be a full path name)
1274 */
1275QString VBoxGlobal::prepareFileNameForHTML (const QString &fn) const
1276{
1277/// @todo (dmik) remove?
1278// QString result = QDir::convertSeparators (fn);
1279//#ifdef Q_OS_LINUX
1280// result.replace ('/', "/<font color=red>&shy;</font>");
1281//#else
1282// result.replace ('\\', "\\<font color=red>&shy;</font>");
1283//#endif
1284// return result;
1285 QFileInfo fi (fn);
1286 return fi.fileName();
1287}
1288
1289/**
1290 * Returns a details report on a given VM enclosed in a HTML table.
1291 *
1292 * @param m machine to create a report for
1293 * @param isNewVM true when called by the New VM Wizard
1294 * @param withLinks true if section titles should be hypertext links
1295 */
1296QString VBoxGlobal::detailsReport (const CMachine &m, bool isNewVM,
1297 bool withLinks, bool aDoRefresh)
1298{
1299 static const char *sTableTpl =
1300 "<table border=0 cellspacing=0 cellpadding=0>%1</table>";
1301 static const char *sSectionHrefTpl =
1302 "<tr><td rowspan=%1 align=left><img src='%2'></td>"
1303 "<td colspan=2><b><a href='%3'><nobr>%4</nobr></a></b></td></tr>"
1304 "%5"
1305 "<tr><td colspan=2><font size=1>&nbsp;</font></td></tr>";
1306 static const char *sSectionBoldTpl =
1307 "<tr><td rowspan=%1 align=left><img src='%2'></td>"
1308 "<td colspan=2><!-- %3 --><b><nobr>%4</nobr></b></td></tr>"
1309 "%5"
1310 "<tr><td colspan=2><font size=1>&nbsp;</font></td></tr>";
1311 static const char *sSectionItemTpl =
1312 "<tr><td width=30%><nobr>%1</nobr></td><td>%2</td></tr>";
1313
1314 static QString sGeneralBasicHrefTpl, sGeneralBasicBoldTpl;
1315 static QString sGeneralFullHrefTpl, sGeneralFullBoldTpl;
1316
1317 /* generate templates after every language change */
1318
1319 if (!detailReportTemplatesReady)
1320 {
1321 detailReportTemplatesReady = true;
1322
1323 QString generalItems
1324 = QString (sSectionItemTpl).arg (tr ("Name", "details report"), "%1")
1325 + QString (sSectionItemTpl).arg (tr ("OS Type", "details report"), "%2")
1326 + QString (sSectionItemTpl).arg (tr ("Base Memory", "details report"),
1327 tr ("<nobr>%3 MB</nobr>", "details report"));
1328 sGeneralBasicHrefTpl = QString (sSectionHrefTpl)
1329 .arg (2 + 3) /* rows */
1330 .arg (":/machine_16px.png", /* icon */
1331 "#general", /* link */
1332 tr ("General", "details report"), /* title */
1333 generalItems); /* items */
1334 sGeneralBasicBoldTpl = QString (sSectionBoldTpl)
1335 .arg (2 + 3) /* rows */
1336 .arg (":/machine_16px.png", /* icon */
1337 "#general", /* link */
1338 tr ("General", "details report"), /* title */
1339 generalItems); /* items */
1340
1341 generalItems
1342 += QString (sSectionItemTpl).arg (tr ("Video Memory", "details report"),
1343 tr ("<nobr>%4 MB</nobr>", "details report"))
1344 + QString (sSectionItemTpl).arg (tr ("Boot Order", "details report"), "%5")
1345 + QString (sSectionItemTpl).arg (tr ("ACPI", "details report"), "%6")
1346 + QString (sSectionItemTpl).arg (tr ("IO APIC", "details report"), "%7")
1347 + QString (sSectionItemTpl).arg (tr ("VT-x/AMD-V", "details report"), "%8");
1348
1349 sGeneralFullHrefTpl = QString (sSectionHrefTpl)
1350 .arg (2 + 8) /* rows */
1351 .arg (":/machine_16px.png", /* icon */
1352 "#general", /* link */
1353 tr ("General", "details report"), /* title */
1354 generalItems); /* items */
1355 sGeneralFullBoldTpl = QString (sSectionBoldTpl)
1356 .arg (2 + 8) /* rows */
1357 .arg (":/machine_16px.png", /* icon */
1358 "#general", /* link */
1359 tr ("General", "details report"), /* title */
1360 generalItems); /* items */
1361 }
1362
1363 /* common generated content */
1364
1365 const QString &sectionTpl = withLinks
1366 ? sSectionHrefTpl
1367 : sSectionBoldTpl;
1368
1369 QString hardDisks;
1370 {
1371 int rows = 2; /* including section header and footer */
1372
1373 CHardDiskAttachmentEnumerator aen = m.GetHardDiskAttachments().Enumerate();
1374 while (aen.HasMore())
1375 {
1376 CHardDiskAttachment hda = aen.GetNext();
1377 CHardDisk hd = hda.GetHardDisk();
1378 /// @todo for the explaination of the below isOk() checks, see
1379 /// @todo *** in #details (const CHardDisk &, bool).
1380 if (hda.isOk())
1381 {
1382 CHardDisk root = hd.GetRoot();
1383 if (hd.isOk())
1384 {
1385 QString src = root.GetLocation();
1386 hardDisks += QString (sSectionItemTpl)
1387 .arg (QString ("%1 %2")
1388 .arg (toString (hda.GetController()))
1389 .arg (toString (hda.GetController(),
1390 hda.GetDeviceNumber())))
1391 .arg (QString ("%1 [<nobr>%2</nobr>]")
1392 .arg (prepareFileNameForHTML (src))
1393 .arg (details (hd, isNewVM /* predict */, aDoRefresh)));
1394 ++ rows;
1395 }
1396 }
1397 }
1398
1399 if (hardDisks.isNull())
1400 {
1401 hardDisks = QString (sSectionItemTpl)
1402 .arg (tr ("Not Attached", "details report (HDDs)")).arg ("");
1403 ++ rows;
1404 }
1405
1406 hardDisks = sectionTpl
1407 .arg (rows) /* rows */
1408 .arg (":/hd_16px.png", /* icon */
1409 "#hdds", /* link */
1410 tr ("Hard Disks", "details report"), /* title */
1411 hardDisks); /* items */
1412 }
1413
1414 /* compose details report */
1415
1416 const QString &generalBasicTpl = withLinks
1417 ? sGeneralBasicHrefTpl
1418 : sGeneralBasicBoldTpl;
1419
1420 const QString &generalFullTpl = withLinks
1421 ? sGeneralFullHrefTpl
1422 : sGeneralFullBoldTpl;
1423
1424 QString detailsReport;
1425
1426 if (isNewVM)
1427 {
1428 detailsReport
1429 = generalBasicTpl
1430 .arg (m.GetName())
1431 .arg (vmGuestOSTypeDescription (m.GetOSTypeId()))
1432 .arg (m.GetMemorySize())
1433 + hardDisks;
1434 }
1435 else
1436 {
1437 /* boot order */
1438 QString bootOrder;
1439 for (ulong i = 1; i <= mVBox.GetSystemProperties().GetMaxBootPosition(); i++)
1440 {
1441 KDeviceType device = m.GetBootOrder (i);
1442 if (device == KDeviceType_Null)
1443 continue;
1444 if (!bootOrder.isEmpty())
1445 bootOrder += ", ";
1446 bootOrder += toString (device);
1447 }
1448 if (bootOrder.isEmpty())
1449 bootOrder = toString (KDeviceType_Null);
1450
1451 CBIOSSettings biosSettings = m.GetBIOSSettings();
1452
1453 /* ACPI */
1454 QString acpi = biosSettings.GetACPIEnabled()
1455 ? tr ("Enabled", "details report (ACPI)")
1456 : tr ("Disabled", "details report (ACPI)");
1457
1458 /* IO APIC */
1459 QString ioapic = biosSettings.GetIOAPICEnabled()
1460 ? tr ("Enabled", "details report (IO APIC)")
1461 : tr ("Disabled", "details report (IO APIC)");
1462
1463 /* VT-x/AMD-V */
1464 CSystemProperties props = vboxGlobal().virtualBox().GetSystemProperties();
1465 QString virt = m.GetHWVirtExEnabled() == KTSBool_True ?
1466 tr ("Enabled", "details report (VT-x/AMD-V)") :
1467 m.GetHWVirtExEnabled() == KTSBool_False ?
1468 tr ("Disabled", "details report (VT-x/AMD-V)") :
1469 props.GetHWVirtExEnabled() ?
1470 tr ("Enabled", "details report (VT-x/AMD-V)") :
1471 tr ("Disabled", "details report (VT-x/AMD-V)");
1472
1473 /* General + Hard Disks */
1474 detailsReport
1475 = generalFullTpl
1476 .arg (m.GetName())
1477 .arg (vmGuestOSTypeDescription (m.GetOSTypeId()))
1478 .arg (m.GetMemorySize())
1479 .arg (m.GetVRAMSize())
1480 .arg (bootOrder)
1481 .arg (acpi)
1482 .arg (ioapic)
1483 .arg (virt)
1484 + hardDisks;
1485
1486 QString item;
1487
1488 /* DVD */
1489 CDVDDrive dvd = m.GetDVDDrive();
1490 item = QString (sSectionItemTpl);
1491 switch (dvd.GetState())
1492 {
1493 case KDriveState_NotMounted:
1494 item = item.arg (tr ("Not mounted", "details report (DVD)"), "");
1495 break;
1496 case KDriveState_ImageMounted:
1497 {
1498 CDVDImage img = dvd.GetImage();
1499 item = item.arg (tr ("Image", "details report (DVD)"),
1500 prepareFileNameForHTML (img.GetFilePath()));
1501 break;
1502 }
1503 case KDriveState_HostDriveCaptured:
1504 {
1505 CHostDVDDrive drv = dvd.GetHostDrive();
1506 QString drvName = drv.GetName();
1507 QString description = drv.GetDescription();
1508 QString fullName = description.isEmpty() ?
1509 drvName :
1510 QString ("%1 (%2)").arg (description, drvName);
1511 item = item.arg (tr ("Host Drive", "details report (DVD)"),
1512 fullName);
1513 break;
1514 }
1515 default:
1516 AssertMsgFailed (("Invalid DVD state: %d", dvd.GetState()));
1517 }
1518 detailsReport += sectionTpl
1519 .arg (2 + 1) /* rows */
1520 .arg (":/cd_16px.png", /* icon */
1521 "#dvd", /* link */
1522 tr ("CD/DVD-ROM", "details report"), /* title */
1523 item); // items
1524
1525 /* Floppy */
1526 CFloppyDrive floppy = m.GetFloppyDrive();
1527 item = QString (sSectionItemTpl);
1528 switch (floppy.GetState())
1529 {
1530 case KDriveState_NotMounted:
1531 item = item.arg (tr ("Not mounted", "details report (floppy)"), "");
1532 break;
1533 case KDriveState_ImageMounted:
1534 {
1535 CFloppyImage img = floppy.GetImage();
1536 item = item.arg (tr ("Image", "details report (floppy)"),
1537 prepareFileNameForHTML (img.GetFilePath()));
1538 break;
1539 }
1540 case KDriveState_HostDriveCaptured:
1541 {
1542 CHostFloppyDrive drv = floppy.GetHostDrive();
1543 QString drvName = drv.GetName();
1544 QString description = drv.GetDescription();
1545 QString fullName = description.isEmpty() ?
1546 drvName :
1547 QString ("%1 (%2)").arg (description, drvName);
1548 item = item.arg (tr ("Host Drive", "details report (floppy)"),
1549 fullName);
1550 break;
1551 }
1552 default:
1553 AssertMsgFailed (("Invalid floppy state: %d", floppy.GetState()));
1554 }
1555 detailsReport += sectionTpl
1556 .arg (2 + 1) /* rows */
1557 .arg (":/fd_16px.png", /* icon */
1558 "#floppy", /* link */
1559 tr ("Floppy", "details report"), /* title */
1560 item); /* items */
1561
1562 /* audio */
1563 {
1564 CAudioAdapter audio = m.GetAudioAdapter();
1565 int rows = audio.GetEnabled() ? 3 : 2;
1566 if (audio.GetEnabled())
1567 item = QString (sSectionItemTpl)
1568 .arg (tr ("Host Driver", "details report (audio)"),
1569 toString (audio.GetAudioDriver())) +
1570 QString (sSectionItemTpl)
1571 .arg (tr ("Controller", "details report (audio)"),
1572 toString (audio.GetAudioController()));
1573 else
1574 item = QString (sSectionItemTpl)
1575 .arg (tr ("Disabled", "details report (audio)"), "");
1576
1577 detailsReport += sectionTpl
1578 .arg (rows + 1) /* rows */
1579 .arg (":/sound_16px.png", /* icon */
1580 "#audio", /* link */
1581 tr ("Audio", "details report"), /* title */
1582 item); /* items */
1583 }
1584 /* network */
1585 {
1586 item = QString::null;
1587 ulong count = mVBox.GetSystemProperties().GetNetworkAdapterCount();
1588 int rows = 2; /* including section header and footer */
1589 for (ulong slot = 0; slot < count; slot ++)
1590 {
1591 CNetworkAdapter adapter = m.GetNetworkAdapter (slot);
1592 if (adapter.GetEnabled())
1593 {
1594 KNetworkAttachmentType type = adapter.GetAttachmentType();
1595 QString attType = toString (adapter.GetAdapterType())
1596 .replace (QRegExp ("\\s\\(.+\\)"), " (%1)");
1597 /* don't use the adapter type string for types that have
1598 * an additional symbolic network/interface name field, use
1599 * this name instead */
1600 if (type == KNetworkAttachmentType_HostInterface)
1601 attType = attType.arg (tr ("host interface, %1",
1602 "details report (network)").arg (adapter.GetHostInterface()));
1603 else if (type == KNetworkAttachmentType_Internal)
1604 attType = attType.arg (tr ("internal network, '%1'",
1605 "details report (network)").arg (adapter.GetInternalNetwork()));
1606 else
1607 attType = attType.arg (vboxGlobal().toString (type));
1608
1609 item += QString (sSectionItemTpl)
1610 .arg (tr ("Adapter %1", "details report (network)")
1611 .arg (adapter.GetSlot()))
1612 .arg (attType);
1613 ++ rows;
1614 }
1615 }
1616 if (item.isNull())
1617 {
1618 item = QString (sSectionItemTpl)
1619 .arg (tr ("Disabled", "details report (network)"), "");
1620 ++ rows;
1621 }
1622
1623 detailsReport += sectionTpl
1624 .arg (rows) /* rows */
1625 .arg (":/nw_16px.png", /* icon */
1626 "#network", /* link */
1627 tr ("Network", "details report"), /* title */
1628 item); /* items */
1629 }
1630 /* serial ports */
1631 {
1632 item = QString::null;
1633 ulong count = mVBox.GetSystemProperties().GetSerialPortCount();
1634 int rows = 2; /* including section header and footer */
1635 for (ulong slot = 0; slot < count; slot ++)
1636 {
1637 CSerialPort port = m.GetSerialPort (slot);
1638 if (port.GetEnabled())
1639 {
1640 KPortMode mode = port.GetHostMode();
1641 QString data =
1642 toCOMPortName (port.GetIRQ(), port.GetIOBase()) + ", ";
1643 if (mode == KPortMode_HostPipe ||
1644 mode == KPortMode_HostDevice)
1645 data += QString ("%1 (<nobr>%2</nobr>)")
1646 .arg (vboxGlobal().toString (mode))
1647 .arg (QDir::convertSeparators (port.GetPath()));
1648 else
1649 data += toString (mode);
1650
1651 item += QString (sSectionItemTpl)
1652 .arg (tr ("Port %1", "details report (serial ports)")
1653 .arg (port.GetSlot()))
1654 .arg (data);
1655 ++ rows;
1656 }
1657 }
1658 if (item.isNull())
1659 {
1660 item = QString (sSectionItemTpl)
1661 .arg (tr ("Disabled", "details report (serial ports)"), "");
1662 ++ rows;
1663 }
1664
1665 detailsReport += sectionTpl
1666 .arg (rows) /* rows */
1667 .arg (":/serial_port_16px.png", /* icon */
1668 "#serialPorts", /* link */
1669 tr ("Serial Ports", "details report"), /* title */
1670 item); /* items */
1671 }
1672 /* parallel ports */
1673 {
1674 item = QString::null;
1675 ulong count = mVBox.GetSystemProperties().GetParallelPortCount();
1676 int rows = 2; /* including section header and footer */
1677 for (ulong slot = 0; slot < count; slot ++)
1678 {
1679 CParallelPort port = m.GetParallelPort (slot);
1680 if (port.GetEnabled())
1681 {
1682 QString data =
1683 toLPTPortName (port.GetIRQ(), port.GetIOBase()) +
1684 QString (" (<nobr>%1</nobr>)")
1685 .arg (QDir::convertSeparators (port.GetPath()));
1686
1687 item += QString (sSectionItemTpl)
1688 .arg (tr ("Port %1", "details report (parallel ports)")
1689 .arg (port.GetSlot()))
1690 .arg (data);
1691 ++ rows;
1692 }
1693 }
1694 if (item.isNull())
1695 {
1696 item = QString (sSectionItemTpl)
1697 .arg (tr ("Disabled", "details report (parallel ports)"), "");
1698 ++ rows;
1699 }
1700
1701 /* Temporary disabled */
1702 QString dummy = sectionTpl /* detailsReport += sectionTpl */
1703 .arg (rows) /* rows */
1704 .arg (":/parallel_port_16px.png", /* icon */
1705 "#parallelPorts", /* link */
1706 tr ("Parallel Ports", "details report"), /* title */
1707 item); /* items */
1708 }
1709 /* USB */
1710 {
1711 CUSBController ctl = m.GetUSBController();
1712 if (!ctl.isNull())
1713 {
1714 /* the USB controller may be unavailable (i.e. in VirtualBox OSE) */
1715
1716 if (ctl.GetEnabled())
1717 {
1718 CUSBDeviceFilterCollection coll = ctl.GetDeviceFilters();
1719 CUSBDeviceFilterEnumerator en = coll.Enumerate();
1720 uint active = 0;
1721 while (en.HasMore())
1722 if (en.GetNext().GetActive())
1723 active ++;
1724
1725 item = QString (sSectionItemTpl)
1726 .arg (tr ("Device Filters", "details report (USB)"),
1727 tr ("%1 (%2 active)", "details report (USB)")
1728 .arg (coll.GetCount()).arg (active));
1729 }
1730 else
1731 item = QString (sSectionItemTpl)
1732 .arg (tr ("Disabled", "details report (USB)"), "");
1733
1734 detailsReport += sectionTpl
1735 .arg (2 + 1) /* rows */
1736 .arg (":/usb_16px.png", /* icon */
1737 "#usb", /* link */
1738 tr ("USB", "details report"), /* title */
1739 item); /* items */
1740 }
1741 }
1742 /* Shared folders */
1743 {
1744 ulong count = m.GetSharedFolders().GetCount();
1745 if (count > 0)
1746 {
1747 item = QString (sSectionItemTpl)
1748 .arg (tr ("Shared Folders", "details report (shared folders)"),
1749 tr ("%1", "details report (shadef folders)")
1750 .arg (count));
1751 }
1752 else
1753 item = QString (sSectionItemTpl)
1754 .arg (tr ("None", "details report (shared folders)"), "");
1755
1756 detailsReport += sectionTpl
1757 .arg (2 + 1) /* rows */
1758 .arg (":/shared_folder_16px.png", /* icon */
1759 "#sfolders", /* link */
1760 tr ("Shared Folders", "details report"), /* title */
1761 item); /* items */
1762 }
1763 /* VRDP */
1764 {
1765 CVRDPServer srv = m.GetVRDPServer();
1766 if (!srv.isNull())
1767 {
1768 /* the VRDP server may be unavailable (i.e. in VirtualBox OSE) */
1769
1770 if (srv.GetEnabled())
1771 item = QString (sSectionItemTpl)
1772 .arg (tr ("VRDP Server Port", "details report (VRDP)"),
1773 tr ("%1", "details report (VRDP)")
1774 .arg (srv.GetPort()));
1775 else
1776 item = QString (sSectionItemTpl)
1777 .arg (tr ("Disabled", "details report (VRDP)"), "");
1778
1779 detailsReport += sectionTpl
1780 .arg (2 + 1) /* rows */
1781 .arg (":/vrdp_16px.png", /* icon */
1782 "#vrdp", /* link */
1783 tr ("Remote Display", "details report"), /* title */
1784 item); /* items */
1785 }
1786 }
1787 }
1788
1789 return QString (sTableTpl). arg (detailsReport);
1790}
1791
1792#ifdef Q_WS_X11
1793bool VBoxGlobal::showVirtualBoxLicense()
1794{
1795 /* get the apps doc path */
1796 int size = 256;
1797 char *buffer = (char*) RTMemTmpAlloc (size);
1798 RTPathAppDocs (buffer, size);
1799 QString path (buffer);
1800 RTMemTmpFree (buffer);
1801 QDir docDir (path);
1802 docDir.setFilter (QDir::Files);
1803 docDir.setNameFilter ("License-*.html");
1804
1805 /* get the license files list and search for the latest license */
1806 QStringList filesList = docDir.entryList();
1807 double maxVersionNumber = 0;
1808 for (int index = 0; index < filesList.count(); ++ index)
1809 {
1810 QRegExp regExp ("License-([\\d\\.]+).html");
1811 regExp.search (filesList [index]);
1812 QString version = regExp.cap (1);
1813 if (maxVersionNumber < version.toDouble())
1814 maxVersionNumber = version.toDouble();
1815 }
1816 if (!maxVersionNumber)
1817 {
1818 vboxProblem().cannotFindLicenseFiles (path);
1819 return false;
1820 }
1821
1822 /* compose the latest license file full path */
1823 QString latestVersion = QString::number (maxVersionNumber);
1824 QString latestFilePath = docDir.absFilePath (
1825 QString ("License-%1.html").arg (latestVersion));
1826
1827 /* check for the agreed license version */
1828 QString licenseAgreed = virtualBox().GetExtraData (VBoxDefs::GUI_LicenseKey);
1829 if (licenseAgreed == latestVersion)
1830 return true;
1831
1832 VBoxLicenseViewer licenseDialog (latestFilePath);
1833 bool result = licenseDialog.exec() == QDialog::Accepted;
1834 if (result)
1835 virtualBox().SetExtraData (VBoxDefs::GUI_LicenseKey, latestVersion);
1836 return result;
1837}
1838#endif
1839
1840/**
1841 * Checks if any of the settings files were auto-converted and informs the user
1842 * if so.
1843 */
1844void VBoxGlobal::checkForAutoConvertedSettings()
1845{
1846 QString formatVersion = mVBox.GetSettingsFormatVersion();
1847
1848 bool isGlobalConverted = false;
1849 QList <CMachine> machines;
1850 QString fileList;
1851 QString version;
1852
1853 CMachineVector vec = mVBox.GetMachines2();
1854 for (CMachineVector::ConstIterator m = vec.begin();
1855 m != vec.end(); ++ m)
1856 {
1857 if (!m->GetAccessible())
1858 continue;
1859
1860 version = m->GetSettingsFileVersion();
1861 if (version != formatVersion)
1862 {
1863 machines.append (*m);
1864 fileList += QString ("<nobr>%1&nbsp;&nbsp;&nbsp;(<i>%2</i>)</nobr><br>")
1865 .arg (m->GetSettingsFilePath())
1866 .arg (version);
1867 }
1868 }
1869
1870 version = mVBox.GetSettingsFileVersion();
1871 if (version != formatVersion)
1872 {
1873 isGlobalConverted = true;
1874 fileList += QString ("<nobr>%1&nbsp;&nbsp;&nbsp;(<i>%2</i>)</nobr><br>")
1875 .arg (mVBox.GetSettingsFilePath())
1876 .arg (version);
1877 }
1878
1879
1880 if (!fileList.isNull())
1881 {
1882 int rc = vboxProblem()
1883 .warnAboutAutoConvertedSettings (formatVersion, fileList);
1884
1885 if (rc == QIMessageBox::No || rc == QIMessageBox::Yes)
1886 {
1887 /* backup (optionally) and save all settings files
1888 * (QIMessageBox::No = Backup, QIMessageBox::Yes = Save) */
1889
1890 for (QList <CMachine>::Iterator m = machines.begin();
1891 /* machines.end() means global config => manual loop break */ ;)
1892 {
1893 if (m == machines.end())
1894 {
1895 if (isGlobalConverted)
1896 {
1897 if (rc == QIMessageBox::No)
1898 mVBox.SaveSettingsWithBackup();
1899 else
1900 mVBox.SaveSettings();
1901
1902 if (!mVBox.isOk())
1903 vboxProblem().cannotSaveGlobalSettings (mVBox);
1904 }
1905 }
1906 else
1907 {
1908 CSession session = openSession ((*m).GetId());
1909 if (!session.isNull())
1910 {
1911 CMachine sm = session.GetMachine();
1912 if (rc == QIMessageBox::No)
1913 sm.SaveSettingsWithBackup();
1914 else
1915 sm.SaveSettings();
1916 ;
1917 if (!sm.isOk())
1918 vboxProblem().cannotSaveMachineSettings (sm);
1919 session.Close();
1920 }
1921 }
1922
1923 if (m == machines.end())
1924 break;
1925
1926 ++ m;
1927 }
1928 }
1929 }
1930}
1931
1932/**
1933 * Opens a direct session for a machine with the given ID.
1934 * This method does user-friendly error handling (display error messages, etc.).
1935 * and returns a null CSession object in case of any error.
1936 * If this method succeeds, don't forget to close the returned session when
1937 * it is no more necessary.
1938 *
1939 * @param aId Machine ID.
1940 * @param aExisting @c true to open an existing session with the machine
1941 * which is already running, @c false to open a new direct
1942 * session.
1943 */
1944CSession VBoxGlobal::openSession (const QUuid &aId, bool aExisting /* = false */)
1945{
1946 CSession session;
1947 session.createInstance (CLSID_Session);
1948 if (session.isNull())
1949 {
1950 vboxProblem().cannotOpenSession (session);
1951 return session;
1952 }
1953
1954 aExisting ? mVBox.OpenExistingSession (session, aId) :
1955 mVBox.OpenSession (session, aId);
1956
1957 if (!mVBox.isOk())
1958 {
1959 CMachine machine = CVirtualBox (mVBox).GetMachine (aId);
1960 vboxProblem().cannotOpenSession (mVBox, machine);
1961 session.detach();
1962 }
1963
1964 return session;
1965}
1966
1967/**
1968 * Starts a machine with the given ID.
1969 */
1970bool VBoxGlobal::startMachine (const QUuid &id)
1971{
1972 AssertReturn (mValid, false);
1973
1974 CSession session = vboxGlobal().openSession (id);
1975 if (session.isNull())
1976 return false;
1977
1978 return consoleWnd().openView (session);
1979}
1980
1981/**
1982 * Appends the disk object and all its children to the media list.
1983 */
1984static
1985void addMediaToList (VBoxMediaList &aList,
1986 const CUnknown &aDisk,
1987 VBoxDefs::DiskType aType)
1988{
1989 VBoxMedia media (aDisk, aType, VBoxMedia::Unknown);
1990 aList += media;
1991 /* append all vdi children */
1992 if (aType == VBoxDefs::HD)
1993 {
1994 CHardDisk hd = aDisk;
1995 CHardDiskEnumerator enumerator = hd.GetChildren().Enumerate();
1996 while (enumerator.HasMore())
1997 {
1998 CHardDisk subHd = enumerator.GetNext();
1999 addMediaToList (aList, CUnknown (subHd), VBoxDefs::HD);
2000 }
2001 }
2002}
2003
2004/**
2005 * Starts a thread that asynchronously enumerates all currently registered
2006 * media, checks for its accessibility and posts VBoxEnumerateMediaEvent
2007 * events to the VBoxGlobal object until all media is enumerated.
2008 *
2009 * If the enumeration is already in progress, no new thread is started.
2010 *
2011 * @sa #currentMediaList()
2012 * @sa #isMediaEnumerationStarted()
2013 */
2014void VBoxGlobal::startEnumeratingMedia()
2015{
2016 Assert (mValid);
2017
2018 /* check if already started but not yet finished */
2019 if (media_enum_thread)
2020 return;
2021
2022 /* ignore the request during application termination */
2023 if (sVBoxGlobalInCleanup)
2024 return;
2025
2026 /* composes a list of all currently known media & their children */
2027 media_list.clear();
2028 {
2029 CHardDiskEnumerator enHD = mVBox.GetHardDisks().Enumerate();
2030 while (enHD.HasMore())
2031 addMediaToList (media_list, CUnknown (enHD.GetNext()), VBoxDefs::HD);
2032
2033 CDVDImageEnumerator enCD = mVBox.GetDVDImages().Enumerate();
2034 while (enCD.HasMore())
2035 addMediaToList (media_list, CUnknown (enCD.GetNext()), VBoxDefs::CD);
2036
2037 CFloppyImageEnumerator enFD = mVBox.GetFloppyImages().Enumerate();
2038 while (enFD.HasMore())
2039 addMediaToList (media_list, CUnknown (enFD.GetNext()), VBoxDefs::FD);
2040 }
2041
2042 /* enumeration thread class */
2043 class Thread : public QThread
2044 {
2045 public:
2046
2047 Thread (const VBoxMediaList &aList) : mList (aList) {}
2048
2049 virtual void run()
2050 {
2051 LogFlow (("MediaEnumThread started.\n"));
2052 COMBase::InitializeCOM();
2053
2054 CVirtualBox mVBox = vboxGlobal().virtualBox();
2055 QObject *target = &vboxGlobal();
2056
2057 /* enumerating list */
2058 int index = 0;
2059 VBoxMediaList::const_iterator it;
2060 for (it = mList.begin();
2061 it != mList.end() && !sVBoxGlobalInCleanup;
2062 ++ it, ++ index)
2063 {
2064 VBoxMedia media = *it;
2065 switch (media.type)
2066 {
2067 case VBoxDefs::HD:
2068 {
2069 CHardDisk hd = media.disk;
2070 media.status =
2071 hd.GetAccessible() == TRUE ? VBoxMedia::Ok :
2072 hd.isOk() ? VBoxMedia::Inaccessible :
2073 VBoxMedia::Error;
2074 /* assign back to store error info if any */
2075 media.disk = hd;
2076 if (media.status == VBoxMedia::Inaccessible)
2077 {
2078 QUuid machineId = hd.GetMachineId();
2079 if (!machineId.isNull())
2080 {
2081 CMachine machine = mVBox.GetMachine (machineId);
2082 if (!machine.isNull() && (machine.GetState() >= KMachineState_Running))
2083 media.status = VBoxMedia::Ok;
2084 }
2085 }
2086 QApplication::postEvent (target,
2087 new VBoxEnumerateMediaEvent (media, index));
2088 break;
2089 }
2090 case VBoxDefs::CD:
2091 {
2092 CDVDImage cd = media.disk;
2093 media.status =
2094 cd.GetAccessible() == TRUE ? VBoxMedia::Ok :
2095 cd.isOk() ? VBoxMedia::Inaccessible :
2096 VBoxMedia::Error;
2097 /* assign back to store error info if any */
2098 media.disk = cd;
2099 QApplication::postEvent (target,
2100 new VBoxEnumerateMediaEvent (media, index));
2101 break;
2102 }
2103 case VBoxDefs::FD:
2104 {
2105 CFloppyImage fd = media.disk;
2106 media.status =
2107 fd.GetAccessible() == TRUE ? VBoxMedia::Ok :
2108 fd.isOk() ? VBoxMedia::Inaccessible :
2109 VBoxMedia::Error;
2110 /* assign back to store error info if any */
2111 media.disk = fd;
2112 QApplication::postEvent (target,
2113 new VBoxEnumerateMediaEvent (media, index));
2114 break;
2115 }
2116 default:
2117 {
2118 AssertMsgFailed (("Invalid aMedia type\n"));
2119 break;
2120 }
2121 }
2122 }
2123
2124 /* post the last message to indicate the end of enumeration */
2125 if (!sVBoxGlobalInCleanup)
2126 QApplication::postEvent (target, new VBoxEnumerateMediaEvent());
2127
2128 COMBase::CleanupCOM();
2129 LogFlow (("MediaEnumThread finished.\n"));
2130 }
2131
2132 private:
2133
2134 const VBoxMediaList &mList;
2135 };
2136
2137 media_enum_thread = new Thread (media_list);
2138 AssertReturnVoid (media_enum_thread);
2139
2140 /* emit mediaEnumStarted() after we set media_enum_thread to != NULL
2141 * to cause isMediaEnumerationStarted() to return TRUE from slots */
2142 emit mediaEnumStarted();
2143
2144 media_enum_thread->start();
2145}
2146
2147/**
2148 * Adds a new media to the current media list.
2149 * @note Currently, this method does nothing but emits the mediaAdded() signal.
2150 * Later, it will be used to synchronize the current media list with
2151 * the actial media list on the server after a single media opetartion
2152 * performed from within one of our UIs.
2153 * @sa #currentMediaList()
2154 */
2155void VBoxGlobal::addMedia (const VBoxMedia &aMedia)
2156{
2157 emit mediaAdded (aMedia);
2158}
2159
2160/**
2161 * Updates the media in the current media list.
2162 * @note Currently, this method does nothing but emits the mediaUpdated() signal.
2163 * Later, it will be used to synchronize the current media list with
2164 * the actial media list on the server after a single media opetartion
2165 * performed from within one of our UIs.
2166 * @sa #currentMediaList()
2167 */
2168void VBoxGlobal::updateMedia (const VBoxMedia &aMedia)
2169{
2170 emit mediaUpdated (aMedia);
2171}
2172
2173/**
2174 * Removes the media from the current media list.
2175 * @note Currently, this method does nothing but emits the mediaRemoved() signal.
2176 * Later, it will be used to synchronize the current media list with
2177 * the actial media list on the server after a single media opetartion
2178 * performed from within one of our UIs.
2179 * @sa #currentMediaList()
2180 */
2181void VBoxGlobal::removeMedia (VBoxDefs::DiskType aType, const QUuid &aId)
2182{
2183 emit mediaRemoved (aType, aId);
2184}
2185
2186/**
2187 * Searches for a VBoxMedia object representing the given COM media object.
2188 *
2189 * @return true if found and false otherwise.
2190 */
2191bool VBoxGlobal::findMedia (const CUnknown &aObj, VBoxMedia &aMedia) const
2192{
2193 for (VBoxMediaList::ConstIterator it = media_list.begin();
2194 it != media_list.end(); ++ it)
2195 {
2196 if ((*it).disk == aObj)
2197 {
2198 aMedia = (*it);
2199 return true;
2200 }
2201 }
2202
2203 return false;
2204}
2205
2206/**
2207 * Native language name of the currently installed translation.
2208 * Returns "English" if no translation is installed
2209 * or if the translation file is invalid.
2210 */
2211QString VBoxGlobal::languageName() const
2212{
2213
2214 return qApp->translate ("@@@", "English",
2215 "Native language name");
2216}
2217
2218/**
2219 * Native language country name of the currently installed translation.
2220 * Returns "--" if no translation is installed or if the translation file is
2221 * invalid, or if the language is independent on the country.
2222 */
2223QString VBoxGlobal::languageCountry() const
2224{
2225 return qApp->translate ("@@@", "--",
2226 "Native language country name "
2227 "(empty if this language is for all countries)");
2228}
2229
2230/**
2231 * Language name of the currently installed translation, in English.
2232 * Returns "English" if no translation is installed
2233 * or if the translation file is invalid.
2234 */
2235QString VBoxGlobal::languageNameEnglish() const
2236{
2237
2238 return qApp->translate ("@@@", "English",
2239 "Language name, in English");
2240}
2241
2242/**
2243 * Language country name of the currently installed translation, in English.
2244 * Returns "--" if no translation is installed or if the translation file is
2245 * invalid, or if the language is independent on the country.
2246 */
2247QString VBoxGlobal::languageCountryEnglish() const
2248{
2249 return qApp->translate ("@@@", "--",
2250 "Language country name, in English "
2251 "(empty if native country name is empty)");
2252}
2253
2254/**
2255 * Comma-separated list of authors of the currently installed translation.
2256 * Returns "innotek" if no translation is installed or if the translation
2257 * file is invalid, or if the translation is supplied by innotek.
2258 */
2259QString VBoxGlobal::languageTranslators() const
2260{
2261 return qApp->translate ("@@@", "innotek",
2262 "Comma-separated list of translators");
2263}
2264
2265/**
2266 * Changes the language of all global string constants according to the
2267 * currently installed translations tables.
2268 */
2269void VBoxGlobal::languageChange()
2270{
2271 machineStates [KMachineState_PoweredOff] = tr ("Powered Off", "MachineState");
2272 machineStates [KMachineState_Saved] = tr ("Saved", "MachineState");
2273 machineStates [KMachineState_Aborted] = tr ("Aborted", "MachineState");
2274 machineStates [KMachineState_Running] = tr ("Running", "MachineState");
2275 machineStates [KMachineState_Paused] = tr ("Paused", "MachineState");
2276 machineStates [KMachineState_Stuck] = tr ("Stuck", "MachineState");
2277 machineStates [KMachineState_Starting] = tr ("Starting", "MachineState");
2278 machineStates [KMachineState_Stopping] = tr ("Stopping", "MachineState");
2279 machineStates [KMachineState_Saving] = tr ("Saving", "MachineState");
2280 machineStates [KMachineState_Restoring] = tr ("Restoring", "MachineState");
2281 machineStates [KMachineState_Discarding] = tr ("Discarding", "MachineState");
2282
2283 sessionStates [KSessionState_Closed] = tr ("Closed", "SessionState");
2284 sessionStates [KSessionState_Open] = tr ("Open", "SessionState");
2285 sessionStates [KSessionState_Spawning] = tr ("Spawning", "SessionState");
2286 sessionStates [KSessionState_Closing] = tr ("Closing", "SessionState");
2287
2288 deviceTypes [KDeviceType_Null] = tr ("None", "DeviceType");
2289 deviceTypes [KDeviceType_Floppy] = tr ("Floppy", "DeviceType");
2290 deviceTypes [KDeviceType_DVD] = tr ("CD/DVD-ROM", "DeviceType");
2291 deviceTypes [KDeviceType_HardDisk] = tr ("Hard Disk", "DeviceType");
2292 deviceTypes [KDeviceType_Network] = tr ("Network", "DeviceType");
2293
2294 diskControllerTypes [KDiskControllerType_IDE0] =
2295 tr ("Primary", "DiskControllerType");
2296 diskControllerTypes [KDiskControllerType_IDE1] =
2297 tr ("Secondary", "DiskControllerType");
2298
2299 diskTypes [KHardDiskType_Normal] =
2300 tr ("Normal", "DiskType");
2301 diskTypes [KHardDiskType_Immutable] =
2302 tr ("Immutable", "DiskType");
2303 diskTypes [KHardDiskType_Writethrough] =
2304 tr ("Writethrough", "DiskType");
2305
2306 diskStorageTypes [KHardDiskStorageType_VirtualDiskImage] =
2307 tr ("Virtual Disk Image", "DiskStorageType");
2308 diskStorageTypes [KHardDiskStorageType_ISCSIHardDisk] =
2309 tr ("iSCSI", "DiskStorageType");
2310 diskStorageTypes [KHardDiskStorageType_VMDKImage] =
2311 tr ("VMDK Image", "DiskStorageType");
2312 diskStorageTypes [KHardDiskStorageType_CustomHardDisk] =
2313 tr ("Custom Hard Disk", "DiskStorageType");
2314 diskStorageTypes [KHardDiskStorageType_VHDImage] =
2315 tr ("VHD Image", "DiskStorageType");
2316
2317 vrdpAuthTypes [KVRDPAuthType_Null] =
2318 tr ("Null", "VRDPAuthType");
2319 vrdpAuthTypes [KVRDPAuthType_External] =
2320 tr ("External", "VRDPAuthType");
2321 vrdpAuthTypes [KVRDPAuthType_Guest] =
2322 tr ("Guest", "VRDPAuthType");
2323
2324 portModeTypes [KPortMode_Disconnected] =
2325 tr ("Disconnected", "PortMode");
2326 portModeTypes [KPortMode_HostPipe] =
2327 tr ("Host Pipe", "PortMode");
2328 portModeTypes [KPortMode_HostDevice] =
2329 tr ("Host Device", "PortMode");
2330
2331 usbFilterActionTypes [KUSBDeviceFilterAction_Ignore] =
2332 tr ("Ignore", "USBFilterActionType");
2333 usbFilterActionTypes [KUSBDeviceFilterAction_Hold] =
2334 tr ("Hold", "USBFilterActionType");
2335
2336 Assert (diskControllerDevices.count() == 3);
2337 diskControllerDevices [0] = tr ("Master", "DiskControllerDevice");
2338 diskControllerDevices [1] = tr ("Slave", "DiskControllerDevice");
2339 diskControllerDevices [2] = tr ("Device&nbsp;%1", "DiskControllerDevice");
2340
2341 audioDriverTypes [KAudioDriverType_Null] =
2342 tr ("Null Audio Driver", "AudioDriverType");
2343 audioDriverTypes [KAudioDriverType_WINMM] =
2344 tr ("Windows Multimedia", "AudioDriverType");
2345 audioDriverTypes [KAudioDriverType_OSS] =
2346 tr ("OSS Audio Driver", "AudioDriverType");
2347 audioDriverTypes [KAudioDriverType_ALSA] =
2348 tr ("ALSA Audio Driver", "AudioDriverType");
2349 audioDriverTypes [KAudioDriverType_DSOUND] =
2350 tr ("Windows DirectSound", "AudioDriverType");
2351 audioDriverTypes [KAudioDriverType_Core] =
2352 tr ("CoreAudio", "AudioDriverType");
2353 audioDriverTypes [KAudioDriverType_Pulse] =
2354 tr ("PulseAudio", "AudioDriverType");
2355
2356 audioControllerTypes [KAudioControllerType_AC97] =
2357 tr ("ICH AC97", "AudioControllerType");
2358 audioControllerTypes [KAudioControllerType_SB16] =
2359 tr ("SoundBlaster 16", "AudioControllerType");
2360
2361 networkAdapterTypes [KNetworkAdapterType_Am79C970A] =
2362 tr ("PCnet-PCI II (Am79C970A)", "NetworkAdapterType");
2363 networkAdapterTypes [KNetworkAdapterType_Am79C973] =
2364 tr ("PCnet-FAST III (Am79C973)", "NetworkAdapterType");
2365 networkAdapterTypes [KNetworkAdapterType_I82540EM] =
2366 tr ("Intel PRO/1000 MT Desktop (82540EM)", "NetworkAdapterType");
2367
2368 networkAttachmentTypes [KNetworkAttachmentType_Null] =
2369 tr ("Not attached", "NetworkAttachmentType");
2370 networkAttachmentTypes [KNetworkAttachmentType_NAT] =
2371 tr ("NAT", "NetworkAttachmentType");
2372 networkAttachmentTypes [KNetworkAttachmentType_HostInterface] =
2373 tr ("Host Interface", "NetworkAttachmentType");
2374 networkAttachmentTypes [KNetworkAttachmentType_Internal] =
2375 tr ("Internal Network", "NetworkAttachmentType");
2376
2377 clipboardTypes [KClipboardMode_Disabled] =
2378 tr ("Disabled", "ClipboardType");
2379 clipboardTypes [KClipboardMode_HostToGuest] =
2380 tr ("Host To Guest", "ClipboardType");
2381 clipboardTypes [KClipboardMode_GuestToHost] =
2382 tr ("Guest To Host", "ClipboardType");
2383 clipboardTypes [KClipboardMode_Bidirectional] =
2384 tr ("Bidirectional", "ClipboardType");
2385
2386 ideControllerTypes [KIDEControllerType_PIIX3] =
2387 tr ("PIIX3", "IDEControllerType");
2388 ideControllerTypes [KIDEControllerType_PIIX4] =
2389 tr ("PIIX4", "IDEControllerType");
2390
2391 USBDeviceStates [KUSBDeviceState_NotSupported] =
2392 tr ("Not supported", "USBDeviceState");
2393 USBDeviceStates [KUSBDeviceState_Unavailable] =
2394 tr ("Unavailable", "USBDeviceState");
2395 USBDeviceStates [KUSBDeviceState_Busy] =
2396 tr ("Busy", "USBDeviceState");
2397 USBDeviceStates [KUSBDeviceState_Available] =
2398 tr ("Available", "USBDeviceState");
2399 USBDeviceStates [KUSBDeviceState_Held] =
2400 tr ("Held", "USBDeviceState");
2401 USBDeviceStates [KUSBDeviceState_Captured] =
2402 tr ("Captured", "USBDeviceState");
2403
2404 mUserDefinedPortName = tr ("User-defined", "serial port");
2405
2406 detailReportTemplatesReady = false;
2407
2408#if defined (Q_WS_PM) || defined (Q_WS_X11)
2409 /* As PM and X11 do not (to my knowledge) have functionality for providing
2410 * human readable key names, we keep a table of them, which must be
2411 * updated when the language is changed. */
2412#warning port me
2413 QIHotKeyEdit::languageChange_qt3();
2414#endif
2415}
2416
2417// public static stuff
2418////////////////////////////////////////////////////////////////////////////////
2419
2420/* static */
2421bool VBoxGlobal::isDOSType (const QString &aOSTypeId)
2422{
2423 if (aOSTypeId.left (3) == "dos" ||
2424 aOSTypeId.left (3) == "win" ||
2425 aOSTypeId.left (3) == "os2")
2426 return true;
2427
2428 return false;
2429}
2430
2431/**
2432 * Sets the QLabel background and frame colors according tho the pixmap
2433 * contents. The bottom right pixel of the label pixmap defines the
2434 * background color of the label, the top right pixel defines the color of
2435 * the one-pixel frame around it. This function also sets the alignment of
2436 * the pixmap to AlignVTop (to correspond to the color choosing logic).
2437 *
2438 * This method is useful to provide nice scaling of pixmal labels without
2439 * scaling pixmaps themselves. To see th eeffect, the size policy of the
2440 * label in the corresponding direction (vertical, for now) should be set to
2441 * something like MinimumExpanding.
2442 *
2443 * @todo Parametrize corners to select pixels from and set the alignment
2444 * accordingly.
2445 */
2446/* static */
2447void VBoxGlobal::adoptLabelPixmap (QLabel *aLabel)
2448{
2449 AssertReturnVoid (aLabel);
2450
2451 const QPixmap *pix = aLabel->pixmap();
2452 QImage img = pix->convertToImage();
2453 QRgb rgbBack = img.pixel (img.width() - 1, img.height() - 1);
2454 QRgb rgbFrame = img.pixel (img.width() - 1, 0);
2455
2456 aLabel->setAlignment (Qt::AlignTop);
2457
2458 aLabel->setPaletteBackgroundColor (QColor (rgbBack));
2459 aLabel->setFrameShadow (Q3Frame::Plain);
2460 aLabel->setFrameShape (Q3Frame::Box);
2461 aLabel->setPaletteForegroundColor (QColor (rgbFrame));
2462}
2463
2464extern const char *gVBoxLangSubDir = "/nls";
2465extern const char *gVBoxLangFileBase = "VirtualBox_";
2466extern const char *gVBoxLangFileExt = ".qm";
2467extern const char *gVBoxLangIDRegExp = "(([a-z]{2})(?:_([A-Z]{2}))?)|(C)";
2468extern const char *gVBoxBuiltInLangName = "C";
2469
2470class VBoxTranslator : public QTranslator
2471{
2472public:
2473
2474 VBoxTranslator (QObject *aParent = 0)
2475 : QTranslator (aParent, "VBoxTranslatorObject") {}
2476
2477 bool loadFile (const QString &aFileName)
2478 {
2479 QFile file (aFileName);
2480 if (!file.open (QIODevice::ReadOnly))
2481 return false;
2482 mData = file.readAll();
2483 return load ((uchar*) mData.data(), mData.size());
2484 }
2485
2486private:
2487
2488 QByteArray mData;
2489};
2490
2491static VBoxTranslator *sTranslator = 0;
2492static QString sLoadedLangId = gVBoxBuiltInLangName;
2493
2494/**
2495 * Returns the loaded (active) language ID.
2496 * Note that it may not match with VBoxGlobalSettings::languageId() if the
2497 * specified language cannot be loaded.
2498 * If the built-in language is active, this method returns "C".
2499 *
2500 * @note "C" is treated as the built-in language for simplicity -- the C
2501 * locale is used in unix environments as a fallback when the requested
2502 * locale is invalid. This way we don't need to process both the "built_in"
2503 * language and the "C" language (which is a valid environment setting)
2504 * separately.
2505 */
2506/* static */
2507QString VBoxGlobal::languageId()
2508{
2509 return sLoadedLangId;
2510}
2511
2512/**
2513 * Loads the language by language ID.
2514 *
2515 * @param aLangId Language ID in in form of xx_YY. QString::null means the
2516 * system default language.
2517 */
2518/* static */
2519void VBoxGlobal::loadLanguage (const QString &aLangId)
2520{
2521 QString langId = aLangId.isNull() ?
2522 VBoxGlobal::systemLanguageId() : aLangId;
2523 QString languageFileName;
2524 QString selectedLangId = gVBoxBuiltInLangName;
2525
2526 char szNlsPath[RTPATH_MAX];
2527 int rc;
2528
2529 rc = RTPathAppPrivateNoArch(szNlsPath, sizeof(szNlsPath));
2530 Assert(RT_SUCCESS(rc));
2531
2532 QString nlsPath = QString(szNlsPath) + gVBoxLangSubDir;
2533 QDir nlsDir (nlsPath);
2534
2535 Assert (!langId.isEmpty());
2536 if (!langId.isEmpty() && langId != gVBoxBuiltInLangName)
2537 {
2538 QRegExp regExp (gVBoxLangIDRegExp);
2539 int pos = regExp.search (langId);
2540 /* the language ID should match the regexp completely */
2541 AssertReturnVoid (pos == 0);
2542
2543 QString lang = regExp.cap (2);
2544
2545 if (nlsDir.exists (gVBoxLangFileBase + langId + gVBoxLangFileExt))
2546 {
2547 languageFileName = nlsDir.absFilePath (gVBoxLangFileBase + langId +
2548 gVBoxLangFileExt);
2549 selectedLangId = langId;
2550 }
2551 else if (nlsDir.exists (gVBoxLangFileBase + lang + gVBoxLangFileExt))
2552 {
2553 languageFileName = nlsDir.absFilePath (gVBoxLangFileBase + lang +
2554 gVBoxLangFileExt);
2555 selectedLangId = lang;
2556 }
2557 else
2558 {
2559 /* Never complain when the default language is requested. In any
2560 * case, if no explicit language file exists, we will simply
2561 * fall-back to English (built-in). */
2562 if (!aLangId.isNull())
2563 vboxProblem().cannotFindLanguage (langId, nlsPath);
2564 /* selectedLangId remains built-in here */
2565 AssertReturnVoid (selectedLangId == gVBoxBuiltInLangName);
2566 }
2567 }
2568
2569 /* delete the old translator if there is one */
2570 if (sTranslator)
2571 {
2572 /* QTranslator destructor will call qApp->removeTranslator() for
2573 * us. It will also delete all its child translations we attach to it
2574 * below, so we don't have to care about them specially. */
2575 delete sTranslator;
2576 }
2577
2578 /* load new language files */
2579 sTranslator = new VBoxTranslator (qApp);
2580 Assert (sTranslator);
2581 bool loadOk = true;
2582 if (sTranslator)
2583 {
2584 if (selectedLangId != gVBoxBuiltInLangName)
2585 {
2586 Assert (!languageFileName.isNull());
2587 loadOk = sTranslator->loadFile (languageFileName);
2588 }
2589 /* we install the translator in any case: on failure, this will
2590 * activate an empty translator that will give us English
2591 * (built-in) */
2592 qApp->installTranslator (sTranslator);
2593 }
2594 else
2595 loadOk = false;
2596
2597 if (loadOk)
2598 sLoadedLangId = selectedLangId;
2599 else
2600 {
2601 vboxProblem().cannotLoadLanguage (languageFileName);
2602 sLoadedLangId = gVBoxBuiltInLangName;
2603 }
2604
2605 /* Try to load the corresponding Qt translation */
2606 if (sLoadedLangId != gVBoxBuiltInLangName)
2607 {
2608#ifdef Q_OS_UNIX
2609 /* We use system installations of Qt on Linux systems, so first, try
2610 * to load the Qt translation from the system location. */
2611 languageFileName = QString (qInstallPathTranslations()) + "/qt_" +
2612 sLoadedLangId + gVBoxLangFileExt;
2613 QTranslator *qtSysTr = new QTranslator (sTranslator);
2614 Assert (qtSysTr);
2615 if (qtSysTr && qtSysTr->load (languageFileName))
2616 qApp->installTranslator (qtSysTr);
2617 /* Note that the Qt translation supplied by innotek is always loaded
2618 * afterwards to make sure it will take precedence over the system
2619 * translation (it may contain more decent variants of translation
2620 * that better correspond to VirtualBox UI). We need to load both
2621 * because a newer version of Qt may be installed on the user computer
2622 * and the innotek version may not fully support it. We don't do it on
2623 * Win32 because we supply a Qt library there and therefore the
2624 * innotek translation is always the best one. */
2625#endif
2626 languageFileName = nlsDir.absFilePath (QString ("qt_") +
2627 sLoadedLangId +
2628 gVBoxLangFileExt);
2629 QTranslator *qtTr = new QTranslator (sTranslator);
2630 Assert (qtTr);
2631 if (qtTr && (loadOk = qtTr->load (languageFileName)))
2632 qApp->installTranslator (qtTr);
2633 /* The below message doesn't fit 100% (because it's an additonal
2634 * language and the main one won't be reset to built-in on failure)
2635 * but the load failure is so rare here that it's not worth a separate
2636 * message (but still, having something is better than having none) */
2637 if (!loadOk && !aLangId.isNull())
2638 vboxProblem().cannotLoadLanguage (languageFileName);
2639 }
2640}
2641
2642/* static */
2643QIcon VBoxGlobal::iconSet (const char *aNormal,
2644 const char *aDisabled /* = NULL */,
2645 const char *aActive /* = NULL */)
2646{
2647 QIcon iconSet;
2648
2649 iconSet.setPixmap (QPixmap (aNormal),
2650 QIcon::Automatic, QIcon::Normal);
2651 if (aDisabled != NULL)
2652 iconSet.setPixmap (QPixmap (aDisabled),
2653 QIcon::Automatic, QIcon::Disabled);
2654 if (aActive != NULL)
2655 iconSet.setPixmap (QPixmap (aActive),
2656 QIcon::Automatic, QIcon::Active);
2657 return iconSet;
2658}
2659
2660/* static */
2661QIcon VBoxGlobal::
2662iconSetEx (const char *aNormal, const char *aSmallNormal,
2663 const char *aDisabled /* = NULL */,
2664 const char *aSmallDisabled /* = NULL */,
2665 const char *aActive /* = NULL */,
2666 const char *aSmallActive /* = NULL */)
2667{
2668 QIcon iconSet;
2669
2670 iconSet.setPixmap (QPixmap (aNormal),
2671 QIcon::Large, QIcon::Normal);
2672 iconSet.setPixmap (QPixmap (aSmallNormal),
2673 QIcon::Small, QIcon::Normal);
2674 if (aSmallDisabled != NULL)
2675 {
2676 iconSet.setPixmap (QPixmap (aDisabled),
2677 QIcon::Large, QIcon::Disabled);
2678 iconSet.setPixmap (QPixmap (aSmallDisabled),
2679 QIcon::Small, QIcon::Disabled);
2680 }
2681 if (aSmallActive != NULL)
2682 {
2683 iconSet.setPixmap (QPixmap (aActive),
2684 QIcon::Large, QIcon::Active);
2685 iconSet.setPixmap (QPixmap (aSmallActive),
2686 QIcon::Small, QIcon::Active);
2687 }
2688
2689 return iconSet;
2690}
2691
2692/**
2693 * Replacement for QToolButton::setTextLabel() that handles the shortcut
2694 * letter (if it is present in the argument string) as if it were a setText()
2695 * call: the shortcut letter is used to automatically assign an "Alt+<letter>"
2696 * accelerator key sequence to the given tool button.
2697 *
2698 * @note This method preserves the icon set if it was assigned before. Only
2699 * the text label and the accelerator are changed.
2700 *
2701 * @param aToolButton Tool button to set the text label on.
2702 * @param aTextLabel Text label to set.
2703 */
2704/* static */
2705void VBoxGlobal::setTextLabel (QToolButton *aToolButton,
2706 const QString &aTextLabel)
2707{
2708 AssertReturnVoid (aToolButton != NULL);
2709
2710 /* remember the icon set as setText() will kill it */
2711 QIcon iset = aToolButton->iconSet();
2712 /* re-use the setText() method to detect and set the accelerator */
2713 aToolButton->setText (aTextLabel);
2714 QKeySequence accel = aToolButton->accel();
2715 aToolButton->setTextLabel (aTextLabel);
2716 aToolButton->setIconSet (iset);
2717 /* set the accel last as setIconSet() would kill it */
2718 aToolButton->setAccel (accel);
2719}
2720
2721/**
2722 * Ensures that the given rectangle \a aRect is fully contained within the
2723 * rectangle \a aBoundRect by moving \a aRect if necessary. If \a aRect is
2724 * larger than \a aBoundRect, its top left corner is simply aligned with the
2725 * top left corner of \a aRect and, if \a aCanResize is true, \a aRect is
2726 * shrinked to become fully visible.
2727 */
2728/* static */
2729QRect VBoxGlobal::normalizeGeometry (const QRect &aRect, const QRect &aBoundRect,
2730 bool aCanResize /* = true */)
2731{
2732 QRect fr = aRect;
2733
2734 /* make the bottom right corner visible */
2735 int rd = aBoundRect.right() - fr.right();
2736 int bd = aBoundRect.bottom() - fr.bottom();
2737 fr.moveBy (rd < 0 ? rd : 0, bd < 0 ? bd : 0);
2738
2739 /* ensure the top left corner is visible */
2740 int ld = fr.left() - aBoundRect.left();
2741 int td = fr.top() - aBoundRect.top();
2742 fr.moveBy (ld < 0 ? -ld : 0, td < 0 ? -td : 0);
2743
2744 if (aCanResize)
2745 {
2746 /* adjust the size to make the rectangle fully contained */
2747 rd = aBoundRect.right() - fr.right();
2748 bd = aBoundRect.bottom() - fr.bottom();
2749 if (rd < 0)
2750 fr.rRight() += rd;
2751 if (bd < 0)
2752 fr.rBottom() += bd;
2753 }
2754
2755 return fr;
2756}
2757
2758/**
2759 * Aligns the center of \a aWidget with the center of \a aRelative.
2760 *
2761 * If necessary, \a aWidget's position is adjusted to make it fully visible
2762 * within the available desktop area. If \a aWidget is bigger then this area,
2763 * it will also be resized unless \a aCanResize is false or there is an
2764 * inappropriate minimum size limit (in which case the top left corner will be
2765 * simply aligned with the top left corner of the available desktop area).
2766 *
2767 * \a aWidget must be a top-level widget. \a aRelative may be any widget, but
2768 * if it's not top-level itself, its top-level widget will be used for
2769 * calculations. \a aRelative can also be NULL, in which case \a aWidget will
2770 * be centered relative to the available desktop area.
2771 */
2772/* static */
2773void VBoxGlobal::centerWidget (QWidget *aWidget, QWidget *aRelative,
2774 bool aCanResize /* = true */)
2775{
2776 AssertReturnVoid (aWidget);
2777 AssertReturnVoid (aWidget->isTopLevel());
2778
2779 QRect deskGeo, parentGeo;
2780 QWidget *w = aRelative;
2781 if (w)
2782 {
2783 w = w->topLevelWidget();
2784 deskGeo = QApplication::desktop()->availableGeometry (w);
2785 parentGeo = w->frameGeometry();
2786 /* On X11/Gnome, geo/frameGeo.x() and y() are always 0 for top level
2787 * widgets with parents, what a shame. Use mapToGlobal() to workaround. */
2788 QPoint d = w->mapToGlobal (QPoint (0, 0));
2789 d.rx() -= w->geometry().x() - w->x();
2790 d.ry() -= w->geometry().y() - w->y();
2791 parentGeo.moveTopLeft (d);
2792 }
2793 else
2794 {
2795 deskGeo = QApplication::desktop()->availableGeometry();
2796 parentGeo = deskGeo;
2797 }
2798
2799 /* On X11, there is no way to determine frame geometry (including WM
2800 * decorations) before the widget is shown for the first time. Stupidly
2801 * enumerate other top level widgets to find the thickest frame. The code
2802 * is based on the idea taken from QDialog::adjustPositionInternal(). */
2803
2804 int extraw = 0, extrah = 0;
2805
2806 QWidgetList list = QApplication::topLevelWidgets();
2807 QListIterator<QWidget*> it (list);
2808 while ((extraw == 0 || extrah == 0) && it.hasNext())
2809 {
2810 int framew, frameh;
2811 QWidget *current = it.next();
2812 if (!current->isVisible())
2813 continue;
2814
2815 framew = current->frameGeometry().width() - current->width();
2816 frameh = current->frameGeometry().height() - current->height();
2817
2818 extraw = QMAX (extraw, framew);
2819 extrah = QMAX (extrah, frameh);
2820 }
2821
2822 /// @todo (r=dmik) not sure if we really need this
2823#if 0
2824 /* sanity check for decoration frames. With embedding, we
2825 * might get extraordinary values */
2826 if (extraw == 0 || extrah == 0 || extraw > 20 || extrah > 50)
2827 {
2828 extrah = 50;
2829 extraw = 20;
2830 }
2831#endif
2832
2833 /* On non-X11 platforms, the following would be enough instead of the
2834 * above workaround: */
2835 // QRect geo = frameGeometry();
2836 QRect geo = QRect (0, 0, aWidget->width() + extraw,
2837 aWidget->height() + extrah);
2838
2839 geo.moveCenter (QPoint (parentGeo.x() + (parentGeo.width() - 1) / 2,
2840 parentGeo.y() + (parentGeo.height() - 1) / 2));
2841
2842 /* ensure the widget is within the available desktop area */
2843 QRect newGeo = normalizeGeometry (geo, deskGeo, aCanResize);
2844
2845 aWidget->move (newGeo.topLeft());
2846
2847 if (aCanResize &&
2848 (geo.width() != newGeo.width() || geo.height() != newGeo.height()))
2849 aWidget->resize (newGeo.width() - extraw, newGeo.height() - extrah);
2850}
2851
2852/**
2853 * Returns the decimal separator for the current locale.
2854 */
2855/* static */
2856QChar VBoxGlobal::decimalSep()
2857{
2858 QString n = QLocale::system().toString (0.0, 'f', 1).stripWhiteSpace();
2859 return n [1];
2860}
2861
2862/**
2863 * Returns the regexp string that defines the format of the human-readable
2864 * size representation, <tt>####[.##] B|KB|MB|GB|TB|PB</tt>.
2865 *
2866 * This regexp will capture 5 groups of text:
2867 * - cap(1): integer number in case when no decimal point is present
2868 * (if empty, it means that decimal point is present)
2869 * - cap(2): size suffix in case when no decimal point is present (may be empty)
2870 * - cap(3): integer number in case when decimal point is present (may be empty)
2871 * - cap(4): fraction number (hundredth) in case when decimal point is present
2872 * - cap(5): size suffix in case when decimal point is present (note that
2873 * B cannot appear there)
2874 */
2875/* static */
2876QString VBoxGlobal::sizeRegexp()
2877{
2878 QString regexp =
2879 QString ("^(?:(?:(\\d+)(?:\\s?([KMGTP]?B))?)|(?:(\\d*)%1(\\d{1,2})(?:\\s?([KMGTP]B))))$")
2880 .arg (decimalSep());
2881 return regexp;
2882}
2883
2884/**
2885 * Parses the given size string that should be in form of
2886 * <tt>####[.##] B|KB|MB|GB|TB|PB</tt> and returns the size value
2887 * in bytes. Zero is returned on error.
2888 */
2889/* static */
2890Q_UINT64 VBoxGlobal::parseSize (const QString &aText)
2891{
2892 QRegExp regexp (sizeRegexp());
2893 int pos = regexp.search (aText);
2894 if (pos != -1)
2895 {
2896 QString intgS = regexp.cap (1);
2897 QString hundS;
2898 QString suff = regexp.cap (2);
2899 if (intgS.isEmpty())
2900 {
2901 intgS = regexp.cap (3);
2902 hundS = regexp.cap (4);
2903 suff = regexp.cap (5);
2904 }
2905
2906 Q_UINT64 denom = 0;
2907 if (suff.isEmpty() || suff == "B")
2908 denom = 1;
2909 else if (suff == "KB")
2910 denom = _1K;
2911 else if (suff == "MB")
2912 denom = _1M;
2913 else if (suff == "GB")
2914 denom = _1G;
2915 else if (suff == "TB")
2916 denom = _1T;
2917 else if (suff == "PB")
2918 denom = _1P;
2919
2920 Q_UINT64 intg = intgS.toULongLong();
2921 if (denom == 1)
2922 return intg;
2923
2924 Q_UINT64 hund = hundS.rightJustify (2, '0').toULongLong();
2925 hund = hund * denom / 100;
2926 intg = intg * denom + hund;
2927 return intg;
2928 }
2929 else
2930 return 0;
2931}
2932
2933/**
2934 * Formats the given \a size value in bytes to a human readable string
2935 * in form of <tt>####[.##] B|KB|MB|GB|TB|PB</tt>.
2936 *
2937 * The \a mode parameter is used for resulting numbers that get a fractional
2938 * part after converting the \a size to KB, MB etc:
2939 * <ul>
2940 * <li>When \a mode is 0, the result is rounded to the closest number
2941 * containing two decimal digits.
2942 * </li>
2943 * <li>When \a mode is -1, the result is rounded to the largest two decimal
2944 * digit number that is not greater than the result. This guarantees that
2945 * converting the resulting string back to the integer value in bytes
2946 * will not produce a value greater that the initial \a size parameter.
2947 * </li>
2948 * <li>When \a mode is 1, the result is rounded to the smallest two decimal
2949 * digit number that is not less than the result. This guarantees that
2950 * converting the resulting string back to the integer value in bytes
2951 * will not produce a value less that the initial \a size parameter.
2952 * </li>
2953 * </ul>
2954 *
2955 * @param aSize size value in bytes
2956 * @param aMode convertion mode (-1, 0 or 1)
2957 * @return human-readable size string
2958 */
2959/* static */
2960QString VBoxGlobal::formatSize (Q_UINT64 aSize, int aMode /* = 0 */)
2961{
2962 static const char *Suffixes [] = { "B", "KB", "MB", "GB", "TB", "PB", NULL };
2963
2964 Q_UINT64 denom = 0;
2965 int suffix = 0;
2966
2967 if (aSize < _1K)
2968 {
2969 denom = 1;
2970 suffix = 0;
2971 }
2972 else if (aSize < _1M)
2973 {
2974 denom = _1K;
2975 suffix = 1;
2976 }
2977 else if (aSize < _1G)
2978 {
2979 denom = _1M;
2980 suffix = 2;
2981 }
2982 else if (aSize < _1T)
2983 {
2984 denom = _1G;
2985 suffix = 3;
2986 }
2987 else if (aSize < _1P)
2988 {
2989 denom = _1T;
2990 suffix = 4;
2991 }
2992 else
2993 {
2994 denom = _1P;
2995 suffix = 5;
2996 }
2997
2998 Q_UINT64 intg = aSize / denom;
2999 Q_UINT64 hund = aSize % denom;
3000
3001 QString number;
3002 if (denom > 1)
3003 {
3004 if (hund)
3005 {
3006 hund *= 100;
3007 /* not greater */
3008 if (aMode < 0) hund = hund / denom;
3009 /* not less */
3010 else if (aMode > 0) hund = (hund + denom - 1) / denom;
3011 /* nearest */
3012 else hund = (hund + denom / 2) / denom;
3013 }
3014 /* check for the fractional part overflow due to rounding */
3015 if (hund == 100)
3016 {
3017 hund = 0;
3018 ++ intg;
3019 /* check if we've got 1024 XB after rounding and scale down if so */
3020 if (intg == 1024 && Suffixes [suffix + 1] != NULL)
3021 {
3022 intg /= 1024;
3023 ++ suffix;
3024 }
3025 }
3026 number = QString ("%1%2%3").arg (intg).arg (decimalSep())
3027 .arg (QString::number (hund).rightJustify (2, '0'));
3028 }
3029 else
3030 {
3031 number = QString::number (intg);
3032 }
3033
3034 return QString ("%1 %2").arg (number).arg (Suffixes [suffix]);
3035}
3036
3037/**
3038 * Reformats the input string @a aStr so that:
3039 * - strings in single quotes will be put inside <nobr> and marked
3040 * with blue color;
3041 * - UUIDs be put inside <nobr> and marked
3042 * with green color;
3043 * - replaces new line chars with </p><p> constructs to form paragraphs
3044 * (note that <p> and </p> are not appended to the beginnign and to the
3045 * end of the string respectively, to allow the result be appended
3046 * or prepended to the existing paragraph).
3047 *
3048 * If @a aToolTip is true, colouring is not applied, only the <nobr> tag
3049 * is added. Also, new line chars are replaced with <br> instead of <p>.
3050 */
3051/* static */
3052QString VBoxGlobal::highlight (const QString &aStr, bool aToolTip /* = false */)
3053{
3054 QString strFont;
3055 QString uuidFont;
3056 QString endFont;
3057 if (!aToolTip)
3058 {
3059 strFont = "<font color=#0000CC>";
3060 uuidFont = "<font color=#008000>";
3061 endFont = "</font>";
3062 }
3063
3064 QString text = aStr;
3065
3066 /* replace special entities, '&' -- first! */
3067 text.replace ('&', "&amp;");
3068 text.replace ('<', "&lt;");
3069 text.replace ('>', "&gt;");
3070 text.replace ('\"', "&quot;");
3071
3072 /* mark strings in single quotes with color */
3073 QRegExp rx = QRegExp ("((?:^|\\s)[(]?)'([^']*)'(?=[:.-!);]?(?:\\s|$))");
3074 rx.setMinimal (true);
3075 text.replace (rx,
3076 QString ("\\1%1<nobr>'\\2'</nobr>%2")
3077 .arg (strFont). arg (endFont));
3078
3079 /* mark UUIDs with color */
3080 text.replace (QRegExp (
3081 "((?:^|\\s)[(]?)"
3082 "(\\{[0-9A-Fa-f]{8}-[0-9A-Fa-f]{4}-[0-9A-Fa-f]{4}-[0-9A-Fa-f]{4}-[0-9A-Fa-f]{12}\\})"
3083 "(?=[:.-!);]?(?:\\s|$))"),
3084 QString ("\\1%1<nobr>\\2</nobr>%2")
3085 .arg (uuidFont). arg (endFont));
3086
3087 /* split to paragraphs at \n chars */
3088 if (!aToolTip)
3089 text.replace ('\n', "</p><p>");
3090 else
3091 text.replace ('\n', "<br>");
3092
3093 return text;
3094}
3095
3096/**
3097 * This does exactly the same as QLocale::system().name() but corrects its
3098 * wrong behavior on Linux systems (LC_NUMERIC for some strange reason takes
3099 * precedence over any other locale setting in the QLocale::system()
3100 * implementation). This implementation first looks at LC_ALL (as defined by
3101 * SUS), then looks at LC_MESSAGES which is designed to define a language for
3102 * program messages in case if it differs from the language for other locale
3103 * categories. Then it looks for LANG and finally falls back to
3104 * QLocale::system().name().
3105 *
3106 * The order of precedence is well defined here:
3107 * http://opengroup.org/onlinepubs/007908799/xbd/envvar.html
3108 *
3109 * @note This method will return "C" when the requested locale is invalid or
3110 * when the "C" locale is set explicitly.
3111 */
3112/* static */
3113QString VBoxGlobal::systemLanguageId()
3114{
3115#ifdef Q_OS_UNIX
3116 const char *s = RTEnvGet ("LC_ALL");
3117 if (s == 0)
3118 s = RTEnvGet ("LC_MESSAGES");
3119 if (s == 0)
3120 s = RTEnvGet ("LANG");
3121 if (s != 0)
3122 return QLocale (s).name();
3123#endif
3124 return QLocale::system().name();
3125}
3126
3127/**
3128 * Reimplementation of QFileDialog::getExistingDirectory() that removes some
3129 * oddities and limitations.
3130 *
3131 * On Win32, this function makes sure a native dialog is launched in
3132 * another thread to avoid dialog visualization errors occuring due to
3133 * multi-threaded COM apartment initialization on the main UI thread while
3134 * the appropriate native dialog function expects a single-threaded one.
3135 *
3136 * On all other platforms, this function is equivalent to
3137 * QFileDialog::getExistingDirectory().
3138 */
3139QString VBoxGlobal::getExistingDirectory (const QString &aDir,
3140 QWidget *aParent, const char *aName,
3141 const QString &aCaption,
3142 bool aDirOnly,
3143 bool aResolveSymlinks)
3144{
3145#if defined Q_WS_WIN
3146
3147 /**
3148 * QEvent class reimplementation to carry Win32 API native dialog's
3149 * result folder information
3150 */
3151 class GetExistDirectoryEvent : public OpenNativeDialogEvent
3152 {
3153 public:
3154
3155 enum { TypeId = QEvent::User + 300 };
3156
3157 GetExistDirectoryEvent (const QString &aResult)
3158 : OpenNativeDialogEvent (aResult, (QEvent::Type) TypeId) {}
3159 };
3160
3161 /**
3162 * QThread class reimplementation to open Win32 API native folder's dialog
3163 */
3164 class Thread : public QThread
3165 {
3166 public:
3167
3168 Thread (QWidget *aParent, QObject *aTarget,
3169 const QString &aDir, const QString &aCaption)
3170 : mParent (aParent), mTarget (aTarget), mDir (aDir), mCaption (aCaption) {}
3171
3172 virtual void run()
3173 {
3174 QString result;
3175
3176 QWidget *topParent = mParent ? mParent->topLevelWidget() : qApp->mainWidget();
3177 QString title = mCaption.isNull() ? tr ("Select a directory") : mCaption;
3178
3179 TCHAR path[MAX_PATH];
3180 path [0] = 0;
3181 TCHAR initPath [MAX_PATH];
3182 initPath [0] = 0;
3183
3184 BROWSEINFO bi;
3185 bi.hwndOwner = topParent ? topParent->winId() : 0;
3186 bi.pidlRoot = NULL;
3187 bi.lpszTitle = (TCHAR*)title.ucs2();
3188 bi.pszDisplayName = initPath;
3189 bi.ulFlags = BIF_RETURNONLYFSDIRS | BIF_STATUSTEXT | BIF_NEWDIALOGSTYLE;
3190 bi.lpfn = winGetExistDirCallbackProc;
3191 bi.lParam = Q_ULONG (&mDir);
3192
3193 /* Qt is uncapable to properly handle modal state if the modal
3194 * window is not a QWidget. For example, if we have the W1->W2->N
3195 * ownership where Wx are QWidgets (W2 is modal), and N is a
3196 * native modal window, cliking on the title bar of W1 will still
3197 * activate W2 and redirect keyboard/mouse to it. The dirty hack
3198 * to prevent it is to disable the entire widget... */
3199 if (mParent)
3200 mParent->setEnabled (false);
3201
3202 LPITEMIDLIST itemIdList = SHBrowseForFolder (&bi);
3203 if (itemIdList)
3204 {
3205 SHGetPathFromIDList (itemIdList, path);
3206 IMalloc *pMalloc;
3207 if (SHGetMalloc (&pMalloc) != NOERROR)
3208 result = QString::null;
3209 else
3210 {
3211 pMalloc->Free (itemIdList);
3212 pMalloc->Release();
3213 result = QString::fromUcs2 ((ushort*)path);
3214 }
3215 }
3216 else
3217 result = QString::null;
3218 QApplication::postEvent (mTarget, new GetExistDirectoryEvent (result));
3219
3220 /* Enable the parent widget again. */
3221 if (mParent)
3222 mParent->setEnabled (true);
3223 }
3224
3225 private:
3226
3227 QWidget *mParent;
3228 QObject *mTarget;
3229 QString mDir;
3230 QString mCaption;
3231 };
3232
3233 QString dir = QDir::convertSeparators (aDir);
3234 LoopObject loopObject ((QEvent::Type) GetExistDirectoryEvent::TypeId);
3235 Thread openDirThread (aParent, &loopObject, dir, aCaption);
3236 openDirThread.start();
3237 qApp->eventLoop()->enterLoop();
3238 openDirThread.wait();
3239 return loopObject.result();
3240
3241#else
3242
3243 return Q3FileDialog::getExistingDirectory (aDir, aParent, aName, aCaption,
3244 aDirOnly, aResolveSymlinks);
3245
3246#endif
3247}
3248
3249/**
3250 * Reimplementation of QFileDialog::getOpenFileName() that removes some
3251 * oddities and limitations.
3252 *
3253 * On Win32, this funciton makes sure a file filter is applied automatically
3254 * right after it is selected from the drop-down list, to conform to common
3255 * experience in other applications. Note that currently, @a selectedFilter
3256 * is always set to null on return.
3257 *
3258 * On all other platforms, this function is equivalent to
3259 * QFileDialog::getOpenFileName().
3260 */
3261/* static */
3262QString VBoxGlobal::getOpenFileName (const QString &aStartWith,
3263 const QString &aFilters,
3264 QWidget *aParent,
3265 const char *aName,
3266 const QString &aCaption,
3267 QString *aSelectedFilter,
3268 bool aResolveSymlinks)
3269{
3270#if defined Q_WS_WIN
3271
3272 /**
3273 * QEvent class reimplementation to carry Win32 API native dialog's
3274 * result folder information
3275 */
3276 class GetOpenFileNameEvent : public OpenNativeDialogEvent
3277 {
3278 public:
3279
3280 enum { TypeId = QEvent::User + 301 };
3281
3282 GetOpenFileNameEvent (const QString &aResult)
3283 : OpenNativeDialogEvent (aResult, (QEvent::Type) TypeId) {}
3284 };
3285
3286 /**
3287 * QThread class reimplementation to open Win32 API native file dialog
3288 */
3289 class Thread : public QThread
3290 {
3291 public:
3292
3293 Thread (QWidget *aParent, QObject *aTarget,
3294 const QString &aStartWith, const QString &aFilters,
3295 const QString &aCaption) :
3296 mParent (aParent), mTarget (aTarget),
3297 mStartWith (aStartWith), mFilters (aFilters),
3298 mCaption (aCaption) {}
3299
3300 virtual void run()
3301 {
3302 QString result;
3303
3304 QString workDir;
3305 QString initSel;
3306 QFileInfo fi (mStartWith);
3307
3308 if (fi.isDir())
3309 workDir = mStartWith;
3310 else
3311 {
3312 workDir = fi.dirPath (true);
3313 initSel = fi.fileName();
3314 }
3315
3316 workDir = QDir::convertSeparators (workDir);
3317 if (!workDir.endsWith ("\\"))
3318 workDir += "\\";
3319
3320 QString title = mCaption.isNull() ? tr ("Select a file") : mCaption;
3321
3322 QWidget *topParent = mParent ? mParent->topLevelWidget() : qApp->mainWidget();
3323 QString winFilters = winFilter (mFilters);
3324 AssertCompile (sizeof (TCHAR) == sizeof (QChar));
3325 TCHAR buf [1024];
3326 if (initSel.length() > 0 && initSel.length() < sizeof (buf))
3327 memcpy (buf, initSel.ucs2(), (initSel.length() + 1) * sizeof (TCHAR));
3328 else
3329 buf [0] = 0;
3330
3331 OPENFILENAME ofn;
3332 memset (&ofn, 0, sizeof (OPENFILENAME));
3333
3334 ofn.lStructSize = sizeof (OPENFILENAME);
3335 ofn.hwndOwner = topParent ? topParent->winId() : 0;
3336 ofn.lpstrFilter = (TCHAR *) winFilters.ucs2();
3337 ofn.lpstrFile = buf;
3338 ofn.nMaxFile = sizeof (buf) - 1;
3339 ofn.lpstrInitialDir = (TCHAR *) workDir.ucs2();
3340 ofn.lpstrTitle = (TCHAR *) title.ucs2();
3341 ofn.Flags = (OFN_NOCHANGEDIR | OFN_HIDEREADONLY |
3342 OFN_EXPLORER | OFN_ENABLEHOOK |
3343 OFN_FILEMUSTEXIST | OFN_PATHMUSTEXIST);
3344 ofn.lpfnHook = OFNHookProc;
3345
3346 if (GetOpenFileName (&ofn))
3347 {
3348 result = QString::fromUcs2 ((ushort *) ofn.lpstrFile);
3349 }
3350
3351 // qt_win_eatMouseMove();
3352 MSG msg = {0, 0, 0, 0, 0, 0, 0};
3353 while (PeekMessage (&msg, 0, WM_MOUSEMOVE, WM_MOUSEMOVE, PM_REMOVE));
3354 if (msg.message == WM_MOUSEMOVE)
3355 PostMessage (msg.hwnd, msg.message, 0, msg.lParam);
3356
3357 result = result.isEmpty() ? result : QFileInfo (result).absFilePath();
3358
3359 QApplication::postEvent (mTarget, new GetOpenFileNameEvent (result));
3360 }
3361
3362 private:
3363
3364 QWidget *mParent;
3365 QObject *mTarget;
3366 QString mStartWith;
3367 QString mFilters;
3368 QString mCaption;
3369 };
3370
3371 if (aSelectedFilter)
3372 *aSelectedFilter = QString::null;
3373 QString startWith = QDir::convertSeparators (aStartWith);
3374 LoopObject loopObject ((QEvent::Type) GetOpenFileNameEvent::TypeId);
3375 if (aParent) qt_enter_modal (aParent);
3376 Thread openDirThread (aParent, &loopObject, startWith, aFilters, aCaption);
3377 openDirThread.start();
3378 qApp->eventLoop()->enterLoop();
3379 openDirThread.wait();
3380 if (aParent) qt_leave_modal (aParent);
3381 return loopObject.result();
3382
3383#else
3384
3385 return Q3FileDialog::getOpenFileName (aStartWith, aFilters, aParent, aName,
3386 aCaption, aSelectedFilter, aResolveSymlinks);
3387
3388#endif
3389}
3390
3391/**
3392 * Search for the first directory that exists starting from the passed one
3393 * and going up through its parents. In case if none of the directories
3394 * exist (except the root one), the function returns QString::null.
3395 */
3396/* static */
3397QString VBoxGlobal::getFirstExistingDir (const QString &aStartDir)
3398{
3399 QString result = QString::null;
3400 QDir dir (aStartDir);
3401 while (!dir.exists() && !dir.isRoot())
3402 {
3403 QFileInfo dirInfo (dir.absPath());
3404 dir = dirInfo.dirPath (true);
3405 }
3406 if (dir.exists() && !dir.isRoot())
3407 result = dir.absPath();
3408 return result;
3409}
3410
3411#if defined (Q_WS_X11)
3412
3413static char *XXGetProperty (Display *aDpy, Window aWnd,
3414 Atom aPropType, const char *aPropName)
3415{
3416 Atom propNameAtom = XInternAtom (aDpy, aPropName,
3417 True /* only_if_exists */);
3418 if (propNameAtom == None)
3419 return NULL;
3420
3421 Atom actTypeAtom = None;
3422 int actFmt = 0;
3423 unsigned long nItems = 0;
3424 unsigned long nBytesAfter = 0;
3425 unsigned char *propVal = NULL;
3426 int rc = XGetWindowProperty (aDpy, aWnd, propNameAtom,
3427 0, LONG_MAX, False /* delete */,
3428 aPropType, &actTypeAtom, &actFmt,
3429 &nItems, &nBytesAfter, &propVal);
3430 if (rc != Success)
3431 return NULL;
3432
3433 return reinterpret_cast <char *> (propVal);
3434}
3435
3436static Bool XXSendClientMessage (Display *aDpy, Window aWnd, const char *aMsg,
3437 unsigned long aData0 = 0, unsigned long aData1 = 0,
3438 unsigned long aData2 = 0, unsigned long aData3 = 0,
3439 unsigned long aData4 = 0)
3440{
3441 Atom msgAtom = XInternAtom (aDpy, aMsg, True /* only_if_exists */);
3442 if (msgAtom == None)
3443 return False;
3444
3445 XEvent ev;
3446
3447 ev.xclient.type = ClientMessage;
3448 ev.xclient.serial = 0;
3449 ev.xclient.send_event = True;
3450 ev.xclient.display = aDpy;
3451 ev.xclient.window = aWnd;
3452 ev.xclient.message_type = msgAtom;
3453
3454 /* always send as 32 bit for now */
3455 ev.xclient.format = 32;
3456 ev.xclient.data.l [0] = aData0;
3457 ev.xclient.data.l [1] = aData1;
3458 ev.xclient.data.l [2] = aData2;
3459 ev.xclient.data.l [3] = aData3;
3460 ev.xclient.data.l [4] = aData4;
3461
3462 return XSendEvent (aDpy, DefaultRootWindow (aDpy), False,
3463 SubstructureRedirectMask, &ev) != 0;
3464}
3465
3466#endif
3467
3468/**
3469 * Activates the specified window. If necessary, the window will be
3470 * de-iconified activation.
3471 *
3472 * @note On X11, it is implied that @a aWid represents a window of the same
3473 * display the application was started on.
3474 *
3475 * @param aWId Window ID to activate.
3476 * @param aSwitchDesktop @c true to switch to the window's desktop before
3477 * activation.
3478 *
3479 * @return @c true on success and @c false otherwise.
3480 */
3481/* static */
3482bool VBoxGlobal::activateWindow (WId aWId, bool aSwitchDesktop /* = true */)
3483{
3484 bool result = true;
3485
3486#if defined (Q_WS_WIN32)
3487
3488 if (IsIconic (aWId))
3489 result &= !!ShowWindow (aWId, SW_RESTORE);
3490 else if (!IsWindowVisible (aWId))
3491 result &= !!ShowWindow (aWId, SW_SHOW);
3492
3493 result &= !!SetForegroundWindow (aWId);
3494
3495#elif defined (Q_WS_X11)
3496
3497 Display *dpy = QPaintDevice::x11AppDisplay();
3498
3499 if (aSwitchDesktop)
3500 {
3501 /* try to find the desktop ID using the NetWM property */
3502 CARD32 *desktop = (CARD32 *) XXGetProperty (dpy, aWId, XA_CARDINAL,
3503 "_NET_WM_DESKTOP");
3504 if (desktop == NULL)
3505 /* if the NetWM propery is not supported try to find the desktop
3506 * ID using the GNOME WM property */
3507 desktop = (CARD32 *) XXGetProperty (dpy, aWId, XA_CARDINAL,
3508 "_WIN_WORKSPACE");
3509
3510 if (desktop != NULL)
3511 {
3512 Bool ok = XXSendClientMessage (dpy, DefaultRootWindow (dpy),
3513 "_NET_CURRENT_DESKTOP",
3514 *desktop);
3515 if (!ok)
3516 {
3517 LogWarningFunc (("Couldn't switch to desktop=%08X\n",
3518 desktop));
3519 result = false;
3520 }
3521 XFree (desktop);
3522 }
3523 else
3524 {
3525 LogWarningFunc (("Couldn't find a desktop ID for aWId=%08X\n",
3526 aWId));
3527 result = false;
3528 }
3529 }
3530
3531 Bool ok = XXSendClientMessage (dpy, aWId, "_NET_ACTIVE_WINDOW");
3532 result &= !!ok;
3533
3534 XRaiseWindow (dpy, aWId);
3535
3536#else
3537
3538 AssertFailed();
3539 result = false;
3540
3541#endif
3542
3543 if (!result)
3544 LogWarningFunc (("Couldn't activate aWId=%08X\n", aWId));
3545
3546 return result;
3547}
3548
3549/**
3550 * Removes the acceletartor mark (the ampersand symbol) from the given string
3551 * and returns the result. The string is supposed to be a menu item's text
3552 * that may (or may not) contain the accelerator mark.
3553 *
3554 * In order to support accelerators used in non-alphabet languages
3555 * (e.g. Japanese) that has a form of "(&<L>)" (where <L> is a latin letter),
3556 * this method first searches for this pattern and, if found, removes it as a
3557 * whole. If such a pattern is not found, then the '&' character is simply
3558 * removed from the string.
3559 *
3560 * @note This function removes only the first occurense of the accelerator
3561 * mark.
3562 *
3563 * @param aText Menu item's text to remove the acceletaror mark from.
3564 *
3565 * @return The resulting string.
3566 */
3567/* static */
3568QString VBoxGlobal::removeAccelMark (const QString &aText)
3569{
3570 QString result = aText;
3571
3572 QRegExp accel ("\\(&[a-zA-Z]\\)");
3573 int pos = accel.search (result);
3574 if (pos >= 0)
3575 result.remove (pos, accel.cap().length());
3576 else
3577 {
3578 pos = result.find ('&');
3579 if (pos >= 0)
3580 result.remove (pos, 1);
3581 }
3582
3583 return result;
3584}
3585
3586/**
3587 * Searches for a widget that with @a aName (if it is not NULL) which inherits
3588 * @a aClassName (if it is not NULL) and among children of @a aParent. If @a
3589 * aParent is NULL, all top-level widgets are searched. If @a aRecursive is
3590 * true, child widgets are recursively searched as well.
3591 */
3592/* static */
3593QWidget *VBoxGlobal::findWidget (QWidget *aParent, const char *aName,
3594 const char *aClassName /* = NULL */,
3595 bool aRecursive /* = false */)
3596{
3597 if (aParent == NULL)
3598 {
3599 QWidgetList list = QApplication::topLevelWidgets();
3600 QWidget* w = NULL;
3601 foreach(w, list)
3602 {
3603 if ((!aName || strcmp (w->name(), aName) == 0) &&
3604 (!aClassName || strcmp (w->className(), aClassName) == 0))
3605 break;
3606 if (aRecursive)
3607 {
3608 w = findWidget (w, aName, aClassName, aRecursive);
3609 if (w)
3610 break;
3611 }
3612 }
3613 return w;
3614 }
3615
3616 QObjectList list = aParent->queryList (aName, aClassName, false, true);
3617 QObject *obj = NULL;
3618 foreach(obj, list)
3619 {
3620 if (obj->isWidgetType())
3621 break;
3622 }
3623 return (QWidget *) obj;
3624}
3625
3626// Public slots
3627////////////////////////////////////////////////////////////////////////////////
3628
3629/**
3630 * Opens the specified URL using OS/Desktop capabilities.
3631 *
3632 * @param aURL URL to open
3633 *
3634 * @return true on success and false otherwise
3635 */
3636bool VBoxGlobal::openURL (const QString &aURL)
3637{
3638#if defined (Q_WS_WIN)
3639 /* We cannot use ShellExecute() on the main UI thread because we've
3640 * initialized COM with CoInitializeEx(COINIT_MULTITHREADED). See
3641 * http://support.microsoft.com/default.aspx?scid=kb;en-us;287087
3642 * for more details. */
3643 class Thread : public QThread
3644 {
3645 public:
3646
3647 Thread (const QString &aURL, QObject *aObject)
3648 : mObject (aObject), mURL (aURL) {}
3649
3650 void run()
3651 {
3652 int rc = (int) ShellExecute (NULL, NULL, mURL.ucs2(), NULL, NULL, SW_SHOW);
3653 bool ok = rc > 32;
3654 QApplication::postEvent
3655 (mObject,
3656 new VBoxShellExecuteEvent (this, mURL, ok));
3657 }
3658
3659 QString mURL;
3660 QObject *mObject;
3661 };
3662
3663 Thread *thread = new Thread (aURL, this);
3664 thread->start();
3665 /* thread will be deleted in the VBoxShellExecuteEvent handler */
3666
3667 return true;
3668
3669#elif defined (Q_WS_X11)
3670
3671 static const char * const commands[] =
3672 { "kfmclient:exec", "gnome-open", "x-www-browser", "firefox", "konqueror" };
3673
3674 for (size_t i = 0; i < ELEMENTS (commands); ++ i)
3675 {
3676 QStringList args = QString(commands [i]).split (':');
3677 args += aURL;
3678 QString command = args.takeFirst();
3679 if (QProcess::startDetached (command, args))
3680 return true;
3681 }
3682
3683#elif defined (Q_WS_MAC)
3684
3685 /* The code below is taken from Psi 0.10 sources
3686 * (http://www.psi-im.org) */
3687
3688 /* Use Internet Config to hand the URL to the appropriate application, as
3689 * set by the user in the Internet Preferences pane.
3690 * NOTE: ICStart could be called once at Psi startup, saving the
3691 * ICInstance in a global variable, as a minor optimization.
3692 * ICStop should then be called at Psi shutdown if ICStart
3693 * succeeded. */
3694 ICInstance icInstance;
3695 OSType psiSignature = 'psi ';
3696 OSStatus error = ::ICStart (&icInstance, psiSignature);
3697 if (error == noErr)
3698 {
3699 ConstStr255Param hint (0x0);
3700 QByteArray cs = aURL.toLocal8Bit();
3701 const char* data = cs.data();
3702 long length = cs.length();
3703 long start (0);
3704 long end (length);
3705 /* Don't bother testing return value (error); launched application
3706 * will report problems. */
3707 ::ICLaunchURL (icInstance, hint, data, length, &start, &end);
3708 ICStop (icInstance);
3709 return true;
3710 }
3711
3712#else
3713 vboxProblem().message
3714 (NULL, VBoxProblemReporter::Error,
3715 tr ("Opening URLs is not implemented yet."));
3716 return false;
3717#endif
3718
3719 /* if we go here it means we couldn't open the URL */
3720 vboxProblem().cannotOpenURL (aURL);
3721
3722 return false;
3723}
3724
3725void VBoxGlobal::showRegistrationDialog (bool aForce)
3726{
3727#ifdef VBOX_WITH_REGISTRATION
3728 if (!aForce && !VBoxRegistrationDlg::hasToBeShown())
3729 return;
3730
3731 if (mRegDlg)
3732 {
3733 /* Show the already opened registration dialog */
3734 mRegDlg->setWindowState (mRegDlg->windowState() & ~Qt::WindowMinimized);
3735 mRegDlg->raise();
3736 mRegDlg->setActiveWindow();
3737 }
3738 else
3739 {
3740 /* Store the ID of the main window to ensure that only one
3741 * registration dialog is shown at a time. Due to manipulations with
3742 * OnExtraDataCanChange() and OnExtraDataChange() signals, this extra
3743 * data item acts like an inter-process mutex, so the first process
3744 * that attempts to set it will win, the rest will get a failure from
3745 * the SetExtraData() call. */
3746 mVBox.SetExtraData (VBoxDefs::GUI_RegistrationDlgWinID,
3747 QString ("%1").arg ((long) qApp->mainWidget()->winId()));
3748
3749 if (mVBox.isOk())
3750 {
3751 /* We've got the "mutex", create a new registration dialog */
3752#warning port me
3753// VBoxRegistrationDlg *dlg =
3754// new VBoxRegistrationDlg (0, 0, false, Qt::WDestructiveClose);
3755// dlg->setup (&mRegDlg);
3756// Assert (dlg == mRegDlg);
3757// mRegDlg->show();
3758 }
3759 }
3760#endif
3761}
3762
3763// Protected members
3764////////////////////////////////////////////////////////////////////////////////
3765
3766bool VBoxGlobal::event (QEvent *e)
3767{
3768 switch (e->type())
3769 {
3770#if defined (Q_WS_WIN)
3771 case VBoxDefs::ShellExecuteEventType:
3772 {
3773 VBoxShellExecuteEvent *ev = (VBoxShellExecuteEvent *) e;
3774 if (!ev->mOk)
3775 vboxProblem().cannotOpenURL (ev->mURL);
3776 /* wait for the thread and free resources */
3777 ev->mThread->wait();
3778 delete ev->mThread;
3779 return true;
3780 }
3781#endif
3782
3783 case VBoxDefs::AsyncEventType:
3784 {
3785 VBoxAsyncEvent *ev = (VBoxAsyncEvent *) e;
3786 ev->handle();
3787 return true;
3788 }
3789
3790 case VBoxDefs::EnumerateMediaEventType:
3791 {
3792 VBoxEnumerateMediaEvent *ev = (VBoxEnumerateMediaEvent *) e;
3793
3794 if (!ev->mLast)
3795 {
3796 if (ev->mMedia.status == VBoxMedia::Error)
3797 vboxProblem().cannotGetMediaAccessibility (ev->mMedia.disk);
3798 media_list [ev->mIndex] = ev->mMedia;
3799 emit mediaEnumerated (media_list [ev->mIndex], ev->mIndex);
3800 }
3801 else
3802 {
3803 /* the thread has posted the last message, wait for termination */
3804 media_enum_thread->wait();
3805 delete media_enum_thread;
3806 media_enum_thread = 0;
3807
3808 emit mediaEnumFinished (media_list);
3809 }
3810
3811 return true;
3812 }
3813
3814 /* VirtualBox callback events */
3815
3816 case VBoxDefs::MachineStateChangeEventType:
3817 {
3818 emit machineStateChanged (*(VBoxMachineStateChangeEvent *) e);
3819 return true;
3820 }
3821 case VBoxDefs::MachineDataChangeEventType:
3822 {
3823 emit machineDataChanged (*(VBoxMachineDataChangeEvent *) e);
3824 return true;
3825 }
3826 case VBoxDefs::MachineRegisteredEventType:
3827 {
3828 emit machineRegistered (*(VBoxMachineRegisteredEvent *) e);
3829 return true;
3830 }
3831 case VBoxDefs::SessionStateChangeEventType:
3832 {
3833 emit sessionStateChanged (*(VBoxSessionStateChangeEvent *) e);
3834 return true;
3835 }
3836 case VBoxDefs::SnapshotEventType:
3837 {
3838 emit snapshotChanged (*(VBoxSnapshotEvent *) e);
3839 return true;
3840 }
3841 case VBoxDefs::CanShowRegDlgEventType:
3842 {
3843 emit canShowRegDlg (((VBoxCanShowRegDlgEvent *) e)->mCanShow);
3844 return true;
3845 }
3846
3847 default:
3848 break;
3849 }
3850
3851 return QObject::event (e);
3852}
3853
3854bool VBoxGlobal::eventFilter (QObject *aObject, QEvent *aEvent)
3855{
3856 if (aEvent->type() == QEvent::LanguageChange &&
3857 aObject->isWidgetType() &&
3858 static_cast <QWidget *> (aObject)->isTopLevel())
3859 {
3860 /* Catch the language change event before any other widget gets it in
3861 * order to invalidate cached string resources (like the details view
3862 * templates) that may be used by other widgets. */
3863 QWidgetList list = QApplication::topLevelWidgets();
3864 if (list.first() == aObject)
3865 {
3866 /* call this only once per every language change (see
3867 * QApplication::installTranslator() for details) */
3868 languageChange();
3869 }
3870 }
3871
3872 return QObject::eventFilter (aObject, aEvent);
3873}
3874
3875// Private members
3876////////////////////////////////////////////////////////////////////////////////
3877
3878void VBoxGlobal::init()
3879{
3880#ifdef DEBUG
3881 verString += " [DEBUG]";
3882#endif
3883
3884#ifdef Q_WS_WIN
3885 /* COM for the main thread is initialized in main() */
3886#else
3887 HRESULT rc = COMBase::InitializeCOM();
3888 if (FAILED (rc))
3889 {
3890 vboxProblem().cannotInitCOM (rc);
3891 return;
3892 }
3893#endif
3894
3895 mVBox.createInstance (CLSID_VirtualBox);
3896 if (!mVBox.isOk())
3897 {
3898 vboxProblem().cannotCreateVirtualBox (mVBox);
3899 return;
3900 }
3901
3902 /* initialize guest OS type vector */
3903 CGuestOSTypeCollection coll = mVBox.GetGuestOSTypes();
3904 int osTypeCount = coll.GetCount();
3905 AssertMsg (osTypeCount > 0, ("Number of OS types must not be zero"));
3906 if (osTypeCount > 0)
3907 {
3908 vm_os_types.resize (osTypeCount);
3909 int i = 0;
3910 CGuestOSTypeEnumerator en = coll.Enumerate();
3911 while (en.HasMore())
3912 vm_os_types [i++] = en.GetNext();
3913 }
3914
3915 /* fill in OS type icon dictionary */
3916 static const char *osTypeIcons[][2] =
3917 {
3918 {"unknown", ":/os_other.png"},
3919 {"dos", ":/os_dos.png"},
3920 {"win31", ":/os_win31.png"},
3921 {"win95", ":/os_win95.png"},
3922 {"win98", ":/os_win98.png"},
3923 {"winme", ":/os_winme.png"},
3924 {"winnt4", ":/os_winnt.png"},
3925 {"win2k", ":/os_win2000.png"},
3926 {"winxp", ":/os_winxp.png"},
3927 {"win2k3", ":/os_win2003.png"},
3928 {"winvista", ":/os_winvista.png"},
3929 {"os2warp3", ":/os_os2.png"},
3930 {"os2warp4", ":/os_os2.png"},
3931 {"os2warp45", ":/os_os2.png"},
3932 {"linux22", ":/os_linux.png"},
3933 {"linux24", ":/os_linux.png"},
3934 {"linux26", ":/os_linux.png"},
3935 {"freebsd", ":/os_freebsd.png"},
3936 {"openbsd", ":/os_openbsd.png"},
3937 {"netbsd", ":/os_netbsd.png"},
3938 {"netware", ":/os_netware.png"},
3939 {"solaris", ":/os_solaris.png"},
3940 {"l4", ":/os_l4.png"},
3941 };
3942 vm_os_type_icons.setAutoDelete (true); /* takes ownership of elements */
3943 for (uint n = 0; n < SIZEOF_ARRAY (osTypeIcons); n ++)
3944 {
3945 vm_os_type_icons.insert (osTypeIcons [n][0],
3946 new QPixmap (osTypeIcons [n][1]));
3947 }
3948
3949 /* fill in VM state icon dictionary */
3950 static struct
3951 {
3952 KMachineState state;
3953 const char *name;
3954 }
3955 vmStateIcons[] =
3956 {
3957 {KMachineState_Null, NULL},
3958 {KMachineState_PoweredOff, ":/state_powered_off_16px.png"},
3959 {KMachineState_Saved, ":/state_saved_16px.png"},
3960 {KMachineState_Aborted, ":/state_aborted_16px.png"},
3961 {KMachineState_Running, ":/state_running_16px.png"},
3962 {KMachineState_Paused, ":/state_paused_16px.png"},
3963 {KMachineState_Stuck, ":/state_stuck_16px.png"},
3964 {KMachineState_Starting, ":/state_running_16px.png"}, /// @todo (dmik) separate icon?
3965 {KMachineState_Stopping, ":/state_running_16px.png"}, /// @todo (dmik) separate icon?
3966 {KMachineState_Saving, ":/state_saving_16px.png"},
3967 {KMachineState_Restoring, ":/state_restoring_16px.png"},
3968 {KMachineState_Discarding, ":/state_discarding_16px.png"},
3969 };
3970 mStateIcons.setAutoDelete (true); // takes ownership of elements
3971 for (uint n = 0; n < SIZEOF_ARRAY (vmStateIcons); n ++)
3972 {
3973 mStateIcons.insert (vmStateIcons [n].state,
3974 new QPixmap (vmStateIcons [n].name));
3975 }
3976
3977 /* online/offline snapshot icons */
3978 mOfflineSnapshotIcon = QPixmap (":/offline_snapshot_16px.png");
3979 mOnlineSnapshotIcon = QPixmap (":/online_snapshot_16px.png");
3980
3981 /* initialize state colors vector */
3982 vm_state_color.setAutoDelete (true); /* takes ownership of elements */
3983 vm_state_color.insert (KMachineState_Null, new QColor(Qt::red));
3984 vm_state_color.insert (KMachineState_PoweredOff, new QColor(Qt::gray));
3985 vm_state_color.insert (KMachineState_Saved, new QColor(Qt::yellow));
3986 vm_state_color.insert (KMachineState_Aborted, new QColor(Qt::darkRed));
3987 vm_state_color.insert (KMachineState_Running, new QColor(Qt::green));
3988 vm_state_color.insert (KMachineState_Paused, new QColor(Qt::darkGreen));
3989 vm_state_color.insert (KMachineState_Stuck, new QColor(Qt::darkMagenta));
3990 vm_state_color.insert (KMachineState_Starting, new QColor(Qt::green));
3991 vm_state_color.insert (KMachineState_Stopping, new QColor(Qt::green));
3992 vm_state_color.insert (KMachineState_Saving, new QColor(Qt::green));
3993 vm_state_color.insert (KMachineState_Restoring, new QColor(Qt::green));
3994 vm_state_color.insert (KMachineState_Discarding, new QColor(Qt::green));
3995
3996 /* Redefine default large and small icon sizes. In particular, it is
3997 * necessary to consider both 32px and 22px icon sizes as Large when we
3998 * explicitly define them as Large (seems to be a bug in
3999 * QToolButton::sizeHint()). */
4000#warning port me
4001// QIcon::setIconSize (QIcon::Small, QSize (16, 16));
4002// QIcon::setIconSize (QIcon::Large, QSize (22, 22));
4003
4004 qApp->installEventFilter (this);
4005
4006 /* create default non-null global settings */
4007 gset = VBoxGlobalSettings (false);
4008
4009 /* try to load global settings */
4010 gset.load (mVBox);
4011 if (!mVBox.isOk() || !gset)
4012 {
4013 vboxProblem().cannotLoadGlobalConfig (mVBox, gset.lastError());
4014 return;
4015 }
4016
4017 /* Load customized language if any */
4018 QString languageId = gset.languageId();
4019 if (!languageId.isNull())
4020 loadLanguage (languageId);
4021
4022 languageChange();
4023
4024 /* process command line */
4025
4026 vm_render_mode_str = 0;
4027#ifdef VBOX_WITH_DEBUGGER_GUI
4028#ifdef VBOX_WITH_DEBUGGER_GUI_MENU
4029 dbg_enabled = true;
4030#else
4031 dbg_enabled = false;
4032#endif
4033 dbg_visible_at_startup = false;
4034#endif
4035
4036 int argc = qApp->argc();
4037 int i = 1;
4038 while (i < argc)
4039 {
4040 const char *arg = qApp->argv() [i];
4041 if ( !::strcmp (arg, "-startvm"))
4042 {
4043 if (++i < argc)
4044 {
4045 QString param = QString (qApp->argv() [i]);
4046 QUuid uuid = QUuid (param);
4047 if (!uuid.isNull())
4048 {
4049 vmUuid = uuid;
4050 }
4051 else
4052 {
4053 CMachine m = mVBox.FindMachine (param);
4054 if (m.isNull())
4055 {
4056 vboxProblem().cannotFindMachineByName (mVBox, param);
4057 return;
4058 }
4059 vmUuid = m.GetId();
4060 }
4061 }
4062 }
4063 else if (!::strcmp (arg, "-comment"))
4064 {
4065 ++i;
4066 }
4067 else if (!::strcmp (arg, "-rmode"))
4068 {
4069 if (++i < argc)
4070 vm_render_mode_str = qApp->argv() [i];
4071 }
4072#ifdef VBOX_WITH_DEBUGGER_GUI
4073 else if (!::strcmp (arg, "-dbg"))
4074 {
4075 dbg_enabled = true;
4076 }
4077#ifdef DEBUG
4078 else if (!::strcmp (arg, "-nodebug"))
4079 {
4080 dbg_enabled = false;
4081 dbg_visible_at_startup = false;
4082 }
4083#else
4084 else if (!::strcmp( arg, "-debug"))
4085 {
4086 dbg_enabled = true;
4087 dbg_visible_at_startup = true;
4088 }
4089#endif
4090#endif
4091 i++;
4092 }
4093
4094 vm_render_mode = vboxGetRenderMode( vm_render_mode_str );
4095
4096 /* setup the callback */
4097 callback = CVirtualBoxCallback (new VBoxCallback (*this));
4098 mVBox.RegisterCallback (callback);
4099 AssertWrapperOk (mVBox);
4100 if (!mVBox.isOk())
4101 return;
4102
4103 mValid = true;
4104}
4105
4106/** @internal
4107 *
4108 * This method should be never called directly. It is called automatically
4109 * when the application terminates.
4110 */
4111void VBoxGlobal::cleanup()
4112{
4113 /* sanity check */
4114 if (!sVBoxGlobalInCleanup)
4115 {
4116 AssertMsgFailed (("Should never be called directly\n"));
4117 return;
4118 }
4119
4120 if (!callback.isNull())
4121 {
4122 mVBox.UnregisterCallback (callback);
4123 AssertWrapperOk (mVBox);
4124 callback.detach();
4125 }
4126
4127 if (media_enum_thread)
4128 {
4129 /* sVBoxGlobalInCleanup is true here, so just wait for the thread */
4130 media_enum_thread->wait();
4131 delete media_enum_thread;
4132 media_enum_thread = 0;
4133 }
4134
4135#ifdef VBOX_WITH_REGISTRATION
4136 if (mRegDlg)
4137 mRegDlg->close();
4138#endif
4139
4140 if (mConsoleWnd)
4141 delete mConsoleWnd;
4142 if (mSelectorWnd)
4143 delete mSelectorWnd;
4144
4145 /* ensure CGuestOSType objects are no longer used */
4146 vm_os_types.clear();
4147 /* media list contains a lot of CUUnknown, release them */
4148 media_list.clear();
4149 /* the last step to ensure we don't use COM any more */
4150 mVBox.detach();
4151
4152 /* There may be VBoxEnumerateMediaEvent instances still in the message
4153 * queue which reference COM objects. Remove them to release those objects
4154 * before uninitializing the COM subsystem. */
4155 QApplication::removePostedEvents (this);
4156
4157#ifdef Q_WS_WIN
4158 /* COM for the main thread is shutdown in main() */
4159#else
4160 COMBase::CleanupCOM();
4161#endif
4162
4163 mValid = false;
4164}
4165
4166/** @fn vboxGlobal
4167 *
4168 * Shortcut to the static VBoxGlobal::instance() method, for convenience.
4169 */
4170
4171
4172/**
4173 * USB Popup Menu class methods
4174 * This class provides the list of USB devices attached to the host.
4175 */
4176VBoxUSBMenu::VBoxUSBMenu (QWidget *aParent) : Q3PopupMenu (aParent)
4177{
4178 connect (this, SIGNAL (aboutToShow()),
4179 this, SLOT (processAboutToShow()));
4180 connect (this, SIGNAL (highlighted (int)),
4181 this, SLOT (processHighlighted (int)));
4182}
4183
4184const CUSBDevice& VBoxUSBMenu::getUSB (int aIndex)
4185{
4186 return mUSBDevicesMap [aIndex];
4187}
4188
4189void VBoxUSBMenu::setConsole (const CConsole &aConsole)
4190{
4191 mConsole = aConsole;
4192}
4193
4194void VBoxUSBMenu::processAboutToShow()
4195{
4196 clear();
4197 mUSBDevicesMap.clear();
4198
4199 CHost host = vboxGlobal().virtualBox().GetHost();
4200
4201 bool isUSBEmpty = host.GetUSBDevices().GetCount() == 0;
4202 if (isUSBEmpty)
4203 {
4204 insertItem (
4205 tr ("<no available devices>", "USB devices"),
4206 USBDevicesMenuNoDevicesId);
4207 setItemEnabled (USBDevicesMenuNoDevicesId, false);
4208 }
4209 else
4210 {
4211 CHostUSBDeviceEnumerator en = host.GetUSBDevices().Enumerate();
4212 while (en.HasMore())
4213 {
4214 CHostUSBDevice iterator = en.GetNext();
4215 CUSBDevice usb = CUnknown (iterator);
4216 int id = insertItem (vboxGlobal().details (usb));
4217 mUSBDevicesMap [id] = usb;
4218 /* check if created item was alread attached to this session */
4219 if (!mConsole.isNull())
4220 {
4221 CUSBDevice attachedUSB =
4222 mConsole.GetUSBDevices().FindById (usb.GetId());
4223 setItemChecked (id, !attachedUSB.isNull());
4224 setItemEnabled (id, iterator.GetState() !=
4225 KUSBDeviceState_Unavailable);
4226 }
4227 }
4228 }
4229}
4230
4231void VBoxUSBMenu::processHighlighted (int aIndex)
4232{
4233 /* the <no available devices> item is highlighted */
4234 if (aIndex == USBDevicesMenuNoDevicesId)
4235 {
4236 QToolTip::add (this,
4237 tr ("No supported devices connected to the host PC",
4238 "USB device tooltip"));
4239 return;
4240 }
4241
4242 CUSBDevice usb = mUSBDevicesMap [aIndex];
4243 /* if null then some other item but a USB device is highlighted */
4244 if (usb.isNull())
4245 {
4246 QToolTip::remove (this);
4247 return;
4248 }
4249
4250 QToolTip::remove (this);
4251 QToolTip::add (this, vboxGlobal().toolTip (usb));
4252}
4253
4254
4255/**
4256 * Enable/Disable Menu class.
4257 * This class provides enable/disable menu items.
4258 */
4259VBoxSwitchMenu::VBoxSwitchMenu (QWidget *aParent, QAction *aAction,
4260 bool aInverted)
4261 : QMenu (aParent), mAction (aAction), mInverted (aInverted)
4262{
4263 /* this menu works only with toggle action */
4264 Assert (aAction->isCheckable());
4265 addAction(aAction);
4266 connect (this, SIGNAL (aboutToShow()),
4267 this, SLOT (processAboutToShow()));
4268}
4269
4270void VBoxSwitchMenu::setToolTip (const QString &aTip)
4271{
4272 mAction->setToolTip (aTip);
4273}
4274
4275void VBoxSwitchMenu::processAboutToShow()
4276{
4277 QString text = mAction->isChecked() ^ mInverted ? tr ("Disable") : tr ("Enable");
4278 mAction->setText (text);
4279}
4280
4281#ifdef Q_WS_X11
4282#include "VBoxGlobal.moc"
4283#endif
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