1 | /**
|
---|
2 | *
|
---|
3 | * VBox frontends: Qt GUI ("VirtualBox"):
|
---|
4 | * "VirtualBox Information Dialog" dialog UI include (Qt Designer)
|
---|
5 | */
|
---|
6 |
|
---|
7 | /*
|
---|
8 | * Copyright (C) 2006 Sun Microsystems, Inc.
|
---|
9 | *
|
---|
10 | * This file is part of VirtualBox Open Source Edition (OSE), as
|
---|
11 | * available from http://www.virtualbox.org. This file is free software;
|
---|
12 | * you can redistribute it and/or modify it under the terms of the GNU
|
---|
13 | * General Public License (GPL) as published by the Free Software
|
---|
14 | * Foundation, in version 2 as it comes in the "COPYING" file of the
|
---|
15 | * VirtualBox OSE distribution. VirtualBox OSE is distributed in the
|
---|
16 | * hope that it will be useful, but WITHOUT ANY WARRANTY of any kind.
|
---|
17 | *
|
---|
18 | * Please contact Sun Microsystems, Inc., 4150 Network Circle, Santa
|
---|
19 | * Clara, CA 95054 USA or visit http://www.sun.com if you need
|
---|
20 | * additional information or have any questions.
|
---|
21 | */
|
---|
22 |
|
---|
23 | /****************************************************************************
|
---|
24 | ** ui.h extension file, included from the uic-generated form implementation.
|
---|
25 | **
|
---|
26 | ** If you wish to add, delete or rename functions or slots use
|
---|
27 | ** Qt Designer which will update this file, preserving your code. Create an
|
---|
28 | ** init() function in place of a constructor, and a destroy() function in
|
---|
29 | ** place of a destructor.
|
---|
30 | *****************************************************************************/
|
---|
31 |
|
---|
32 |
|
---|
33 | VBoxVMInformationDlg::InfoDlgMap VBoxVMInformationDlg::mSelfArray = InfoDlgMap();
|
---|
34 |
|
---|
35 | void VBoxVMInformationDlg::createInformationDlg (const CSession &aSession,
|
---|
36 | VBoxConsoleView *aConsole)
|
---|
37 | {
|
---|
38 | CMachine machine = aSession.GetMachine();
|
---|
39 | if (mSelfArray.find (machine.GetName()) == mSelfArray.end())
|
---|
40 | {
|
---|
41 | /* creating new information dialog if there is no one existing */
|
---|
42 | mSelfArray [machine.GetName()] = new VBoxVMInformationDlg (
|
---|
43 | aConsole,
|
---|
44 | "VBoxVMInformationDlg", WType_TopLevel | WDestructiveClose);
|
---|
45 | /* read new machine data for this information dialog */
|
---|
46 | mSelfArray [machine.GetName()]->setup (aSession, aConsole);
|
---|
47 | }
|
---|
48 |
|
---|
49 | VBoxVMInformationDlg *info = mSelfArray [machine.GetName()];
|
---|
50 | info->show();
|
---|
51 | info->raise();
|
---|
52 | info->setWindowState (info->windowState() & ~WindowMinimized);
|
---|
53 | info->setActiveWindow();
|
---|
54 | }
|
---|
55 |
|
---|
56 |
|
---|
57 | void VBoxVMInformationDlg::init()
|
---|
58 | {
|
---|
59 | /* dialog initially is not polished */
|
---|
60 | mIsPolished = false;
|
---|
61 |
|
---|
62 | /* search the default button */
|
---|
63 | mDefaultButton = searchDefaultButton();
|
---|
64 | qApp->installEventFilter (this);
|
---|
65 |
|
---|
66 | /* setup a dialog icon */
|
---|
67 | setIcon (QPixmap::fromMimeSource ("description_16px.png"));
|
---|
68 |
|
---|
69 | /* statusbar initially disabled */
|
---|
70 | statusBar()->setHidden (true);
|
---|
71 |
|
---|
72 | /* setup size grip */
|
---|
73 | mSizeGrip = new QSizeGrip (centralWidget(), "mSizeGrip");
|
---|
74 | mSizeGrip->resize (mSizeGrip->sizeHint());
|
---|
75 | mSizeGrip->stackUnder (mCloseButton);
|
---|
76 |
|
---|
77 | /* logs list creation */
|
---|
78 | mInfoStack = new QTabWidget (mInfoFrame, "mInfoStack");
|
---|
79 | mInfoStack->setMargin (10);
|
---|
80 | QVBoxLayout *infoFrameLayout = new QVBoxLayout (mInfoFrame);
|
---|
81 | infoFrameLayout->addWidget (mInfoStack);
|
---|
82 |
|
---|
83 | /* details view creation */
|
---|
84 | mDetailsText = new QTextBrowser();
|
---|
85 | mDetailsText->setFrameShape (QFrame::NoFrame);
|
---|
86 | mDetailsText->setPaper (backgroundBrush());
|
---|
87 | mInfoStack->addTab (mDetailsText,
|
---|
88 | VBoxGlobal::iconSet ("settings_16px.png"),
|
---|
89 | QString::null);
|
---|
90 |
|
---|
91 | /* statistic view creation */
|
---|
92 | mStatisticText = new QTextBrowser();
|
---|
93 | mStatisticText->setFrameShape (QFrame::NoFrame);
|
---|
94 | mStatisticText->setPaper (backgroundBrush());
|
---|
95 | mInfoStack->addTab (mStatisticText,
|
---|
96 | VBoxGlobal::iconSet ("state_running_16px.png"),
|
---|
97 | QString::null);
|
---|
98 |
|
---|
99 | /* full list of statistics counters to get total info */
|
---|
100 | // mDefStatText = new QTextBrowser();
|
---|
101 | // mDefStatText->setFrameShape (QFrame::NoFrame);
|
---|
102 | // mDefStatText->setPaper (backgroundBrush());
|
---|
103 | // mInfoStack->addTab (mDefStatText,
|
---|
104 | // VBoxGlobal::iconSet ("show_logs_16px.png"),
|
---|
105 | // QString::null);
|
---|
106 |
|
---|
107 | /* show statistics page and make it focused */
|
---|
108 | connect (mInfoStack, SIGNAL (currentChanged (QWidget*)),
|
---|
109 | this, SLOT (onPageChanged (QWidget*)));
|
---|
110 | mInfoStack->showPage (mStatisticText);
|
---|
111 | }
|
---|
112 |
|
---|
113 |
|
---|
114 | void VBoxVMInformationDlg::destroy()
|
---|
115 | {
|
---|
116 | /* save dialog attributes for this vm */
|
---|
117 | QString dlgsize ("%1,%2,%3");
|
---|
118 | mSession.GetMachine().SetExtraData (VBoxDefs::GUI_InfoDlgState,
|
---|
119 | dlgsize.arg (mWidth).arg (mHeight).arg (isMaximized() ? "max" : "normal"));
|
---|
120 |
|
---|
121 | if (!mSession.isNull() && !mSession.GetMachine().isNull())
|
---|
122 | mSelfArray.erase (mSession.GetMachine().GetName());
|
---|
123 | }
|
---|
124 |
|
---|
125 |
|
---|
126 | void VBoxVMInformationDlg::setup (const CSession &aSession,
|
---|
127 | VBoxConsoleView *aConsole)
|
---|
128 | {
|
---|
129 | /* store related machine pointers */
|
---|
130 | mSession = aSession;
|
---|
131 | mConsole = aConsole;
|
---|
132 |
|
---|
133 | /* loading language constants */
|
---|
134 | languageChangeImp();
|
---|
135 |
|
---|
136 | /* details page update */
|
---|
137 | updateDetails();
|
---|
138 |
|
---|
139 | /* statistics page update */
|
---|
140 | processStatistics();
|
---|
141 | mStatTimer.start (5000);
|
---|
142 |
|
---|
143 | /* setup handlers */
|
---|
144 | connect (&vboxGlobal(), SIGNAL (mediaEnumFinished (const VBoxMediaList &)),
|
---|
145 | this, SLOT (updateDetails()));
|
---|
146 | connect (mConsole, SIGNAL (mediaChanged (VBoxDefs::DiskType)),
|
---|
147 | this, SLOT (updateDetails()));
|
---|
148 | connect (mConsole, SIGNAL (sharedFoldersChanged()),
|
---|
149 | this, SLOT (updateDetails()));
|
---|
150 |
|
---|
151 | connect (&mStatTimer, SIGNAL (timeout()), this, SLOT (processStatistics()));
|
---|
152 | connect (mConsole, SIGNAL (resizeHintDone()), this, SLOT (processStatistics()));
|
---|
153 |
|
---|
154 | /* preload dialog attributes for this vm */
|
---|
155 | QString dlgsize = mSession.GetMachine().GetExtraData (VBoxDefs::GUI_InfoDlgState);
|
---|
156 | if (dlgsize.isNull())
|
---|
157 | {
|
---|
158 | mWidth = 400;
|
---|
159 | mHeight = 450;
|
---|
160 | mMax = false;
|
---|
161 | }
|
---|
162 | else
|
---|
163 | {
|
---|
164 | QStringList list = QStringList::split (',', dlgsize);
|
---|
165 | mWidth = list [0].toInt(), mHeight = list [1].toInt();
|
---|
166 | mMax = list [2] == "max";
|
---|
167 | }
|
---|
168 | }
|
---|
169 |
|
---|
170 |
|
---|
171 | void VBoxVMInformationDlg::languageChangeImp()
|
---|
172 | {
|
---|
173 | AssertReturnVoid (!mSession.isNull());
|
---|
174 |
|
---|
175 | CMachine machine = mSession.GetMachine();
|
---|
176 | AssertReturnVoid (!machine.isNull());
|
---|
177 |
|
---|
178 | /* Setup a dialog caption. */
|
---|
179 | setCaption (tr ("%1 - Session Information").arg (machine.GetName()));
|
---|
180 |
|
---|
181 | /* Setup a tabwidget page names. */
|
---|
182 | mInfoStack->changeTab (mDetailsText, tr ("&Details"));
|
---|
183 | mInfoStack->changeTab (mStatisticText, tr ("&Runtime"));
|
---|
184 | // mInfoStack->changeTab (mDefStatText, tr ("De&fault Stat"));
|
---|
185 |
|
---|
186 | /* Clear counter names initially. */
|
---|
187 | mNamesMap.clear();
|
---|
188 |
|
---|
189 | /* HD statistics: */
|
---|
190 | mNamesMap ["/Devices/ATA0/Unit0/*DMA"] = tr ("DMA Transfers");
|
---|
191 | mNamesMap ["/Devices/ATA0/Unit0/*PIO"] = tr ("PIO Transfers");
|
---|
192 | mNamesMap ["/Devices/ATA0/Unit0/ReadBytes"] = tr ("Data Read");
|
---|
193 | mNamesMap ["/Devices/ATA0/Unit0/WrittenBytes"] = tr ("Data Written");
|
---|
194 |
|
---|
195 | mNamesMap ["/Devices/ATA0/Unit1/*DMA"] = tr ("DMA Transfers");
|
---|
196 | mNamesMap ["/Devices/ATA0/Unit1/*PIO"] = tr ("PIO Transfers");
|
---|
197 | mNamesMap ["/Devices/ATA0/Unit1/ReadBytes"] = tr ("Data Read");
|
---|
198 | mNamesMap ["/Devices/ATA0/Unit1/WrittenBytes"] = tr ("Data Written");
|
---|
199 |
|
---|
200 | mNamesMap ["/Devices/ATA1/Unit0/*DMA"] = tr ("DMA Transfers");
|
---|
201 | mNamesMap ["/Devices/ATA1/Unit0/*PIO"] = tr ("PIO Transfers");
|
---|
202 | mNamesMap ["/Devices/ATA1/Unit0/ReadBytes"] = tr ("Data Read");
|
---|
203 | mNamesMap ["/Devices/ATA1/Unit0/WrittenBytes"] = tr ("Data Written");
|
---|
204 |
|
---|
205 | mNamesMap ["/Devices/ATA1/Unit1/*DMA"] = tr ("DMA Transfers");
|
---|
206 | mNamesMap ["/Devices/ATA1/Unit1/*PIO"] = tr ("PIO Transfers");
|
---|
207 | mNamesMap ["/Devices/ATA1/Unit1/ReadBytes"] = tr ("Data Read");
|
---|
208 | mNamesMap ["/Devices/ATA1/Unit1/WrittenBytes"] = tr ("Data Written");
|
---|
209 |
|
---|
210 | for (int i = 0; i < 4; i++)
|
---|
211 | {
|
---|
212 | CNetworkAdapter na = machine.GetNetworkAdapter (i);
|
---|
213 | KNetworkAdapterType ty = na.GetAdapterType();
|
---|
214 | const char *name;
|
---|
215 |
|
---|
216 | switch (ty)
|
---|
217 | {
|
---|
218 | case KNetworkAdapterType_I82540EM:
|
---|
219 | name = "E1k";
|
---|
220 | break;
|
---|
221 | default:
|
---|
222 | name = "PCNet";
|
---|
223 | break;
|
---|
224 | }
|
---|
225 | mNamesMap [QString("/Devices/%1%2/TransmitBytes")
|
---|
226 | .arg(name) .arg(i)] = tr ("Data Transmitted");
|
---|
227 | mNamesMap [QString("/Devices/%1%2/ReceiveBytes")
|
---|
228 | .arg(name) .arg(i)] = tr ("Data Received");
|
---|
229 | }
|
---|
230 |
|
---|
231 | /* Statistics page update. */
|
---|
232 | refreshStatistics();
|
---|
233 | }
|
---|
234 |
|
---|
235 |
|
---|
236 | QPushButton* VBoxVMInformationDlg::searchDefaultButton()
|
---|
237 | {
|
---|
238 | /* this mechanism is used for searching the default dialog button
|
---|
239 | * and similar the same mechanism in Qt::QDialog inner source */
|
---|
240 | QPushButton *button = 0;
|
---|
241 | QObjectList *list = queryList ("QPushButton");
|
---|
242 | QObjectListIt it (*list);
|
---|
243 | while ((button = (QPushButton*)it.current()) && !button->isDefault())
|
---|
244 | ++ it;
|
---|
245 | return button;
|
---|
246 | }
|
---|
247 |
|
---|
248 |
|
---|
249 | bool VBoxVMInformationDlg::eventFilter (QObject *aObject, QEvent *aEvent)
|
---|
250 | {
|
---|
251 | switch (aEvent->type())
|
---|
252 | {
|
---|
253 | /* auto-default button focus-in processor used to move the "default"
|
---|
254 | * button property into the currently focused button */
|
---|
255 | case QEvent::FocusIn:
|
---|
256 | {
|
---|
257 | if (aObject->inherits ("QPushButton") &&
|
---|
258 | aObject->parent() == centralWidget())
|
---|
259 | {
|
---|
260 | ((QPushButton*)aObject)->setDefault (aObject != mDefaultButton);
|
---|
261 | if (mDefaultButton)
|
---|
262 | mDefaultButton->setDefault (aObject == mDefaultButton);
|
---|
263 | }
|
---|
264 | break;
|
---|
265 | }
|
---|
266 | /* auto-default button focus-out processor used to remove the "default"
|
---|
267 | * button property from the previously focused button */
|
---|
268 | case QEvent::FocusOut:
|
---|
269 | {
|
---|
270 | if (aObject->inherits ("QPushButton") &&
|
---|
271 | aObject->parent() == centralWidget())
|
---|
272 | {
|
---|
273 | if (mDefaultButton)
|
---|
274 | mDefaultButton->setDefault (aObject != mDefaultButton);
|
---|
275 | ((QPushButton*)aObject)->setDefault (aObject == mDefaultButton);
|
---|
276 | }
|
---|
277 | break;
|
---|
278 | }
|
---|
279 | default:
|
---|
280 | break;
|
---|
281 | }
|
---|
282 | return QMainWindow::eventFilter (aObject, aEvent);
|
---|
283 | }
|
---|
284 |
|
---|
285 |
|
---|
286 | bool VBoxVMInformationDlg::event (QEvent *aEvent)
|
---|
287 | {
|
---|
288 | bool result = QMainWindow::event (aEvent);
|
---|
289 | switch (aEvent->type())
|
---|
290 | {
|
---|
291 | case QEvent::LanguageChange:
|
---|
292 | {
|
---|
293 | if (!mSession.isNull())
|
---|
294 | languageChangeImp();
|
---|
295 | break;
|
---|
296 | }
|
---|
297 | case QEvent::WindowStateChange:
|
---|
298 | {
|
---|
299 | if (mIsPolished)
|
---|
300 | mMax = isMaximized();
|
---|
301 | else if (mMax == isMaximized())
|
---|
302 | mIsPolished = true;
|
---|
303 | break;
|
---|
304 | }
|
---|
305 | default:
|
---|
306 | break;
|
---|
307 | }
|
---|
308 | return result;
|
---|
309 | }
|
---|
310 |
|
---|
311 |
|
---|
312 | void VBoxVMInformationDlg::keyPressEvent (QKeyEvent *aEvent)
|
---|
313 | {
|
---|
314 | if (aEvent->state() == 0 ||
|
---|
315 | (aEvent->state() & Keypad && aEvent->key() == Key_Enter))
|
---|
316 | {
|
---|
317 | switch (aEvent->key())
|
---|
318 | {
|
---|
319 | /* processing the return keypress for the auto-default button */
|
---|
320 | case Key_Enter:
|
---|
321 | case Key_Return:
|
---|
322 | {
|
---|
323 | QPushButton *currentDefault = searchDefaultButton();
|
---|
324 | if (currentDefault)
|
---|
325 | currentDefault->animateClick();
|
---|
326 | break;
|
---|
327 | }
|
---|
328 | /* processing the escape keypress as the close dialog action */
|
---|
329 | case Key_Escape:
|
---|
330 | {
|
---|
331 | close();
|
---|
332 | break;
|
---|
333 | }
|
---|
334 | }
|
---|
335 | }
|
---|
336 | else
|
---|
337 | aEvent->ignore();
|
---|
338 | }
|
---|
339 |
|
---|
340 |
|
---|
341 | void VBoxVMInformationDlg::showEvent (QShowEvent *aEvent)
|
---|
342 | {
|
---|
343 | QMainWindow::showEvent (aEvent);
|
---|
344 |
|
---|
345 | /* one may think that QWidget::polish() is the right place to do things
|
---|
346 | * below, but apparently, by the time when QWidget::polish() is called,
|
---|
347 | * the widget style & layout are not fully done, at least the minimum
|
---|
348 | * size hint is not properly calculated. Since this is sometimes necessary,
|
---|
349 | * we provide our own "polish" implementation. */
|
---|
350 |
|
---|
351 | if (mIsPolished)
|
---|
352 | return;
|
---|
353 |
|
---|
354 | /* load window size and state */
|
---|
355 | resize (mWidth, mHeight);
|
---|
356 | if (mMax)
|
---|
357 | QTimer::singleShot (0, this, SLOT (showMaximized()));
|
---|
358 | else
|
---|
359 | mIsPolished = true;
|
---|
360 |
|
---|
361 | VBoxGlobal::centerWidget (this, parentWidget());
|
---|
362 | }
|
---|
363 |
|
---|
364 |
|
---|
365 | void VBoxVMInformationDlg::resizeEvent (QResizeEvent*)
|
---|
366 | {
|
---|
367 | /* adjust the size-grip location for the current resize event */
|
---|
368 | mSizeGrip->move (centralWidget()->rect().bottomRight() -
|
---|
369 | QPoint (mSizeGrip->rect().width() - 1,
|
---|
370 | mSizeGrip->rect().height() - 1));
|
---|
371 |
|
---|
372 | /* store dialog size for this vm */
|
---|
373 | if (mIsPolished && !isMaximized())
|
---|
374 | {
|
---|
375 | mWidth = width();
|
---|
376 | mHeight = height();
|
---|
377 | }
|
---|
378 | }
|
---|
379 |
|
---|
380 |
|
---|
381 | void VBoxVMInformationDlg::updateDetails()
|
---|
382 | {
|
---|
383 | /* details page update */
|
---|
384 | mDetailsText->setText (
|
---|
385 | vboxGlobal().detailsReport (mSession.GetMachine(), false /* isNewVM */,
|
---|
386 | false /* withLinks */, false /* refresh */));
|
---|
387 | }
|
---|
388 |
|
---|
389 |
|
---|
390 | void VBoxVMInformationDlg::onPageChanged (QWidget *aPage)
|
---|
391 | {
|
---|
392 | /* focusing the browser on shown page */
|
---|
393 | aPage->setFocus();
|
---|
394 | }
|
---|
395 |
|
---|
396 |
|
---|
397 | void VBoxVMInformationDlg::processStatistics()
|
---|
398 | {
|
---|
399 | CMachineDebugger dbg = mSession.GetConsole().GetDebugger();
|
---|
400 | QString info;
|
---|
401 |
|
---|
402 | /* Look for all statistics for filtering purposes. */
|
---|
403 | // dbg.GetStats ("*", false, info);
|
---|
404 | // mDefStatText->setText (info);
|
---|
405 |
|
---|
406 | /* Process selected statistics: */
|
---|
407 | for (DataMapType::const_iterator it = mNamesMap.begin();
|
---|
408 | it != mNamesMap.end(); ++ it)
|
---|
409 | {
|
---|
410 | dbg.GetStats (it.key(), true, info);
|
---|
411 | mValuesMap [it.key()] = parseStatistics (info);
|
---|
412 | }
|
---|
413 |
|
---|
414 | /* Statistics page update. */
|
---|
415 | refreshStatistics();
|
---|
416 | }
|
---|
417 |
|
---|
418 |
|
---|
419 | QString VBoxVMInformationDlg::parseStatistics (const QString &aText)
|
---|
420 | {
|
---|
421 | /* Filters the statistic counters body. */
|
---|
422 | QRegExp query ("^.+<Statistics>\n(.+)\n</Statistics>.*$");
|
---|
423 | if (query.search (aText) == -1)
|
---|
424 | return QString::null;
|
---|
425 |
|
---|
426 | QStringList wholeList = QStringList::split ("\n", query.cap (1));
|
---|
427 |
|
---|
428 | ULONG64 summa = 0;
|
---|
429 | for (QStringList::Iterator lineIt = wholeList.begin();
|
---|
430 | lineIt != wholeList.end(); ++ lineIt)
|
---|
431 | {
|
---|
432 | QString text = *lineIt;
|
---|
433 | text.remove (1, 1);
|
---|
434 | text.remove (text.length() - 2, 2);
|
---|
435 |
|
---|
436 | /* Parse incoming counter and fill the counter-element values. */
|
---|
437 | CounterElementType counter;
|
---|
438 | counter.type = text.section (" ", 0, 0);
|
---|
439 | text = text.section (" ", 1);
|
---|
440 | QStringList list = QStringList::split ("\" ", text);
|
---|
441 | for (QStringList::Iterator it = list.begin(); it != list.end(); ++ it)
|
---|
442 | {
|
---|
443 | QString pair = *it;
|
---|
444 | QRegExp regExp ("^(.+)=\"([^\"]*)\"?$");
|
---|
445 | regExp.search (pair);
|
---|
446 | counter.list.insert (regExp.cap (1), regExp.cap (2));
|
---|
447 | }
|
---|
448 |
|
---|
449 | /* Fill the output with the necessary counter's value.
|
---|
450 | * Currently we are using "c" field of simple counter only. */
|
---|
451 | QString result = counter.list.contains ("c") ? counter.list ["c"] : "0";
|
---|
452 | summa += result.toULongLong();
|
---|
453 | }
|
---|
454 |
|
---|
455 | return QString::number (summa);
|
---|
456 | }
|
---|
457 |
|
---|
458 |
|
---|
459 | void VBoxVMInformationDlg::refreshStatistics()
|
---|
460 | {
|
---|
461 | if (mSession.isNull())
|
---|
462 | return;
|
---|
463 |
|
---|
464 | QString table = "<p><table border=0 cellspacing=0 cellpadding=0 width=100%>%1</table></p>";
|
---|
465 | QString hdrRow = "<tr><td align=left><img src='%1'></td><td colspan=3><b>%2</b></td></tr>";
|
---|
466 | QString bdyRow = "<tr><td></td><td><nobr>%1</nobr></td><td colspan=2><nobr>%2</nobr></td></tr>";
|
---|
467 | QString paragraph = "<tr><td colspan=4></td></tr>";
|
---|
468 | QString interline = "<tr><td colspan=4><font size=1> </font></td></tr>";
|
---|
469 | QString result;
|
---|
470 |
|
---|
471 | /* Screen & VT-X Runtime Parameters */
|
---|
472 | {
|
---|
473 | CConsole console = mSession.GetConsole();
|
---|
474 | ULONG bpp = console.GetDisplay().GetBitsPerPixel();
|
---|
475 | QString resolution = QString ("%1x%2")
|
---|
476 | .arg (console.GetDisplay().GetWidth())
|
---|
477 | .arg (console.GetDisplay().GetHeight());
|
---|
478 | if (bpp)
|
---|
479 | resolution += QString ("x%1").arg (bpp);
|
---|
480 | QString virt = console.GetDebugger().GetHWVirtExEnabled() ?
|
---|
481 | VBoxGlobal::tr ("Enabled", "details report (VT-x/AMD-V)") :
|
---|
482 | VBoxGlobal::tr ("Disabled", "details report (VT-x/AMD-V)");
|
---|
483 | QString addInfo = console.GetGuest().GetAdditionsVersion();
|
---|
484 | uint addVersion = addInfo.toUInt();
|
---|
485 | QString addVerisonStr = !addInfo.isNull() ?
|
---|
486 | tr ("Version %1.%2", "guest additions")
|
---|
487 | .arg (RT_HIWORD (addVersion)).arg (RT_LOWORD (addVersion)) :
|
---|
488 | tr ("Not Detected", "guest additions");
|
---|
489 |
|
---|
490 | result += hdrRow.arg ("state_running_16px.png").arg (tr ("Runtime Attributes"));
|
---|
491 | result += bdyRow.arg (tr ("Screen Resolution")).arg (resolution) +
|
---|
492 | bdyRow.arg (VBoxGlobal::tr ("VT-x/AMD-V", "details report")).arg (virt);
|
---|
493 | result += bdyRow.arg (tr ("Guest Additions")).arg (addVerisonStr);
|
---|
494 | result += paragraph;
|
---|
495 | }
|
---|
496 |
|
---|
497 | /* Hard Disk Statistics. */
|
---|
498 | QString primaryMaster = QString ("%1 %2")
|
---|
499 | .arg (vboxGlobal().toString (KStorageBus_IDE, 0))
|
---|
500 | .arg (vboxGlobal().toString (KStorageBus_IDE, 0, 0));
|
---|
501 | QString primarySlave = QString ("%1 %2")
|
---|
502 | .arg (vboxGlobal().toString (KStorageBus_IDE, 0))
|
---|
503 | .arg (vboxGlobal().toString (KStorageBus_IDE, 0, 1));
|
---|
504 | QString secondarySlave = QString ("%1 %2")
|
---|
505 | .arg (vboxGlobal().toString (KStorageBus_IDE, 1))
|
---|
506 | .arg (vboxGlobal().toString (KStorageBus_IDE, 1, 1));
|
---|
507 |
|
---|
508 | result += hdrRow.arg ("hd_16px.png").arg (tr ("IDE Hard Disk Statistics"));
|
---|
509 | result += formatHardDisk (primaryMaster, KStorageBus_IDE, 0, 0, 0, 1);
|
---|
510 | result += interline;
|
---|
511 | result += formatHardDisk (primarySlave, KStorageBus_IDE, 0, 1, 4, 5);
|
---|
512 | result += interline;
|
---|
513 | result += formatHardDisk (secondarySlave, KStorageBus_IDE, 1, 1, 12, 13);
|
---|
514 | result += paragraph;
|
---|
515 |
|
---|
516 | /* CD/DVD-ROM Statistics. */
|
---|
517 | result += hdrRow.arg ("cd_16px.png").arg (tr ("CD/DVD-ROM Statistics"));
|
---|
518 | result += formatHardDisk (QString::null,
|
---|
519 | KStorageBus_IDE, 1, 0, 8, 9);
|
---|
520 | result += paragraph;
|
---|
521 |
|
---|
522 | /* Network Adapters Statistics. */
|
---|
523 | {
|
---|
524 | result += hdrRow.arg ("nw_16px.png")
|
---|
525 | .arg (tr ("Network Adapter Statistics"));
|
---|
526 | ulong count = vboxGlobal().virtualBox()
|
---|
527 | .GetSystemProperties().GetNetworkAdapterCount();
|
---|
528 | for (ulong slot = 0; slot < count; ++ slot)
|
---|
529 | {
|
---|
530 | result += formatAdapter (
|
---|
531 | VBoxGlobal::tr ("Adapter %1", "details report (network)")
|
---|
532 | .arg (slot), slot, 16 + slot * 2, 17 + slot * 2);
|
---|
533 | if (slot < count - 1)
|
---|
534 | result += interline;
|
---|
535 | }
|
---|
536 | }
|
---|
537 |
|
---|
538 | /* Show full composed page. */
|
---|
539 | mStatisticText->setText (table.arg (result));
|
---|
540 | }
|
---|
541 |
|
---|
542 |
|
---|
543 | QString VBoxVMInformationDlg::formatHardDisk (const QString &aName,
|
---|
544 | KStorageBus aBus, LONG aChannel,
|
---|
545 | LONG aDevice, int aStart, int aFinish)
|
---|
546 | {
|
---|
547 | if (mSession.isNull())
|
---|
548 | return QString::null;
|
---|
549 |
|
---|
550 | QString header = "<tr><td></td><td colspan=3><nobr><u>%1</u></nobr></td></tr>";
|
---|
551 | CMachine machine = mSession.GetMachine();
|
---|
552 |
|
---|
553 | QString result = aName.isNull() ? QString::null : header.arg (aName);
|
---|
554 | CHardDisk hd = machine.GetHardDisk (aBus, aChannel, aDevice);
|
---|
555 | if (!hd.isNull() || (aBus == KStorageBus_IDE && aChannel == 1 && aDevice == 0))
|
---|
556 | {
|
---|
557 | result += composeArticle (QString::null, aStart, aFinish);
|
---|
558 | result += composeArticle ("B", aStart + 2, aFinish + 2);
|
---|
559 | }
|
---|
560 | else
|
---|
561 | result += composeArticle (tr ("Not Attached", "hard disk"), -1, -1);
|
---|
562 | return result;
|
---|
563 | }
|
---|
564 |
|
---|
565 | QString VBoxVMInformationDlg::formatAdapter (const QString &aName,
|
---|
566 | ULONG aSlot,
|
---|
567 | int aStart, int aFinish)
|
---|
568 |
|
---|
569 | {
|
---|
570 | if (mSession.isNull())
|
---|
571 | return QString::null;
|
---|
572 |
|
---|
573 | QString header = "<tr><td></td><td colspan=3><nobr><u>%1</u></nobr></td></tr>";
|
---|
574 | CMachine machine = mSession.GetMachine();
|
---|
575 |
|
---|
576 | QString result = header.arg (aName);
|
---|
577 | CNetworkAdapter na = machine.GetNetworkAdapter (aSlot);
|
---|
578 | result += na.GetEnabled() ?
|
---|
579 | composeArticle ("B", aStart, aFinish) :
|
---|
580 | composeArticle (tr ("Disabled", "network adapter"), -1, -1);
|
---|
581 | return result;
|
---|
582 | }
|
---|
583 |
|
---|
584 |
|
---|
585 | QString VBoxVMInformationDlg::composeArticle (const QString &aUnits,
|
---|
586 | int aStart, int aFinish)
|
---|
587 | {
|
---|
588 | QString body = "<tr><td></td><td><nobr>%1</nobr></td><td align=right><nobr>%2%3</nobr></td><td width=100%></td></tr>";
|
---|
589 |
|
---|
590 | QString result;
|
---|
591 |
|
---|
592 | if (aStart == -1 && aFinish == -1)
|
---|
593 | result += body.arg (aUnits).arg (QString::null).arg (QString::null);
|
---|
594 | else for (int id = aStart; id <= aFinish; ++ id)
|
---|
595 | {
|
---|
596 | QString line = body;
|
---|
597 | if (mValuesMap.contains (mNamesMap.keys() [id]))
|
---|
598 | {
|
---|
599 | ULONG64 value = mValuesMap.values() [id].toULongLong();
|
---|
600 | line = line.arg (mNamesMap.values() [id])
|
---|
601 | .arg (QString ("%L1").arg (value));
|
---|
602 | line = aUnits.isNull() ?
|
---|
603 | line.arg (QString ("<img src=tpixel.png width=%1 height=1>")
|
---|
604 | .arg (QApplication::fontMetrics().width (" B"))) :
|
---|
605 | line.arg (QString (" %1").arg (aUnits));
|
---|
606 | }
|
---|
607 | result += line;
|
---|
608 | }
|
---|
609 |
|
---|
610 | return result;
|
---|
611 | }
|
---|
612 |
|
---|
613 |
|
---|
614 | /* Old code for two columns support */
|
---|
615 | #if 0
|
---|
616 | void VBoxVMInformationDlg::refreshStatistics()
|
---|
617 | {
|
---|
618 | QString table = "<p><table border=0 cellspacing=0 cellpadding=0 width=100%>%1</table></p>";
|
---|
619 | QString hdrRow = "<tr><td align=left><img src='%1'></td><td colspan=4><b>%2</b></td></tr>";
|
---|
620 | QString subRow = "<tr><td></td><td colspan=2><nobr><u>%1</u></nobr></td>"
|
---|
621 | "<td colspan=2><nobr><u>%2</u></nobr></td></tr>";
|
---|
622 | QString bdyRow = "<tr><td></td><td><nobr>%1</nobr></td><td width=50%><nobr>%2</nobr></td>"
|
---|
623 | "<td><nobr>%3</nobr></td><td width=50%><nobr>%4</nobr></td></tr>";
|
---|
624 | QString paragraph = "<tr><td colspan=5></td></tr>";
|
---|
625 | QString interline = "<tr><td colspan=5><font size=1> </font></td></tr>";
|
---|
626 | QString result;
|
---|
627 |
|
---|
628 | /* Screen & VT-X Runtime Parameters */
|
---|
629 | if (!mSession.isNull())
|
---|
630 | {
|
---|
631 | CConsole console = mSession.GetConsole();
|
---|
632 | ULONG bpp = console.GetDisplay().GetBitsPerPixel();
|
---|
633 | QString resolution = QString ("%1x%2")
|
---|
634 | .arg (console.GetDisplay().GetWidth())
|
---|
635 | .arg (console.GetDisplay().GetHeight());
|
---|
636 | if (bpp)
|
---|
637 | resolution += QString ("x%1").arg (bpp);
|
---|
638 | QString virt = console.GetDebugger().GetHWVirtExEnabled() ?
|
---|
639 | tr ("Enabled") : tr ("Disabled");
|
---|
640 |
|
---|
641 | result += hdrRow.arg ("state_running_16px.png").arg (tr ("Runtime Attributes"));
|
---|
642 | result += bdyRow.arg (tr ("Screen Resolution")) .arg (resolution)
|
---|
643 | .arg (tr ("Hardware Virtualization")).arg (virt);
|
---|
644 | result += paragraph;
|
---|
645 | }
|
---|
646 |
|
---|
647 | /* Hard Disk Statistics. */
|
---|
648 | result += hdrRow.arg ("hd_16px.png").arg (tr ("Hard Disks Statistics"));
|
---|
649 |
|
---|
650 | result += subRow.arg (tr ("Primary Master")).arg (tr ("Primary Slave"));
|
---|
651 | result += composeArticle (QString::null, 0, 1, 4, 5);
|
---|
652 | result += composeArticle ("B", 2, 3, 6, 7);
|
---|
653 | result += interline;
|
---|
654 |
|
---|
655 | result += subRow.arg (tr ("Secondary Master")).arg (tr ("Secondary Slave"));
|
---|
656 | result += composeArticle (QString::null, 8, 9, 12, 13);
|
---|
657 | result += composeArticle ("B", 10, 11, 14, 15);
|
---|
658 | result += paragraph;
|
---|
659 |
|
---|
660 | /* Network Adapters Statistics. Counters are currently missed. */
|
---|
661 | result += hdrRow.arg ("nw_16px.png").arg (tr ("Network Adapter Statistics"));
|
---|
662 | result += subRow.arg (tr ("Adapter 1")).arg (tr ("Adapter 2"));
|
---|
663 | result += composeArticle ("B", 16, 17, 18, 19);
|
---|
664 |
|
---|
665 | /* Show full composed page. */
|
---|
666 | mStatisticText->setText (table.arg (result));
|
---|
667 | }
|
---|
668 |
|
---|
669 |
|
---|
670 | QString VBoxVMInformationDlg::composeArticle (const QString &aUnits,
|
---|
671 | int aStart1, int aFinish1,
|
---|
672 | int aStart2, int aFinish2)
|
---|
673 | {
|
---|
674 | QString bdyRow = "<tr><td></td><td><nobr>%1</nobr></td><td width=50%><nobr>%2</nobr></td>"
|
---|
675 | "<td><nobr>%3</nobr></td><td width=50%><nobr>%4</nobr></td></tr>";
|
---|
676 |
|
---|
677 | QString result;
|
---|
678 |
|
---|
679 | int id1 = aStart1, id2 = aStart2;
|
---|
680 | while (id1 <= aFinish1 || id2 <= aFinish2)
|
---|
681 | {
|
---|
682 | QString line = bdyRow;
|
---|
683 | /* Processing first column */
|
---|
684 | if (id1 > aFinish1)
|
---|
685 | {
|
---|
686 | line = line.arg (QString::null).arg (QString::null)
|
---|
687 | .arg (QString::null).arg (QString::null);
|
---|
688 | }
|
---|
689 | else if (mValuesMap.contains (mNamesMap.keys() [id1]))
|
---|
690 | {
|
---|
691 | line = line.arg (mNamesMap.values() [id1]);
|
---|
692 | ULONG64 value = mValuesMap.values() [id1].toULongLong();
|
---|
693 | line = aUnits.isNull() ?
|
---|
694 | line.arg (QString ("%L1").arg (value)) :
|
---|
695 | line.arg (QString ("%L1 %2").arg (value).arg (aUnits));
|
---|
696 | }
|
---|
697 | /* Processing second column */
|
---|
698 | if (id2 > aFinish2)
|
---|
699 | {
|
---|
700 | line = line.arg (QString::null).arg (QString::null)
|
---|
701 | .arg (QString::null).arg (QString::null);
|
---|
702 | }
|
---|
703 | else if (mValuesMap.contains (mNamesMap.keys() [id2]))
|
---|
704 | {
|
---|
705 | line = line.arg (mNamesMap.values() [id2]);
|
---|
706 | ULONG64 value = mValuesMap.values() [id2].toULongLong();
|
---|
707 | line = aUnits.isNull() ?
|
---|
708 | line.arg (QString ("%L1").arg (value)) :
|
---|
709 | line.arg (QString ("%L1 %2").arg (value).arg (aUnits));
|
---|
710 | }
|
---|
711 | result += line;
|
---|
712 | ++ id1; ++ id2;
|
---|
713 | }
|
---|
714 |
|
---|
715 | return result;
|
---|
716 | }
|
---|
717 | #endif
|
---|
718 |
|
---|