VirtualBox

source: vbox/trunk/src/VBox/Frontends/VirtualBox/ui/VBoxVMSettingsDlg.ui.h@ 3340

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

FE/Qt: Improved whatsthis wording; increased the default height of the whatsthis pane to 4 lines (instead of 3).

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 83.2 KB
Line 
1/**
2 *
3 * VBox frontends: Qt GUI ("VirtualBox"):
4 * "VM settings" dialog UI include (Qt Designer)
5 */
6
7/*
8 * Copyright (C) 2006-2007 innotek GmbH
9 *
10 * This file is part of VirtualBox Open Source Edition (OSE), as
11 * available from http://www.virtualbox.org. This file is free software;
12 * you can redistribute it and/or modify it under the terms of the GNU
13 * General Public License as published by the Free Software Foundation,
14 * in version 2 as it comes in the "COPYING" file of the VirtualBox OSE
15 * distribution. VirtualBox OSE is distributed in the hope that it will
16 * be useful, but WITHOUT ANY WARRANTY of any kind.
17 *
18 * If you received this file as part of a commercial VirtualBox
19 * distribution, then only the terms of your commercial VirtualBox
20 * license agreement apply instead of the previous paragraph.
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
33extern const char *GUI_FirstRun;
34
35
36/**
37 * QDialog class reimplementation to use for adding network interface.
38 * It has one line-edit field for entering network interface's name and
39 * common dialog's ok/cancel buttons.
40 */
41class VBoxAddNIDialog : public QDialog
42{
43 Q_OBJECT
44
45public:
46
47 VBoxAddNIDialog (QWidget *aParent, const QString &aIfaceName) :
48 QDialog (aParent, "VBoxAddNIDialog", true /* modal */),
49 mLeName (0)
50 {
51 setCaption (tr ("Add Host Interface"));
52 QVBoxLayout *mainLayout = new QVBoxLayout (this, 10, 10, "mainLayout");
53
54 /* Setup Input layout */
55 QHBoxLayout *inputLayout = new QHBoxLayout (mainLayout, 10, "inputLayout");
56 QLabel *lbName = new QLabel (tr ("Interface Name"), this);
57 mLeName = new QLineEdit (aIfaceName, this);
58 QWhatsThis::add (mLeName, tr ("Descriptive name of the new network interface"));
59 inputLayout->addWidget (lbName);
60 inputLayout->addWidget (mLeName);
61 connect (mLeName, SIGNAL (textChanged (const QString &)),
62 this, SLOT (validate()));
63
64 /* Setup Button layout */
65 QHBoxLayout *buttonLayout = new QHBoxLayout (mainLayout, 10, "buttonLayout");
66 mBtOk = new QPushButton (tr ("&OK"), this, "mBtOk");
67 QSpacerItem *spacer = new QSpacerItem(0, 0, QSizePolicy::Expanding, QSizePolicy::Minimum);
68 QPushButton *btCancel = new QPushButton (tr ("Cancel"), this, "btCancel");
69 connect (mBtOk, SIGNAL (clicked()), this, SLOT (accept()));
70 connect (btCancel, SIGNAL (clicked()), this, SLOT (reject()));
71 buttonLayout->addWidget (mBtOk);
72 buttonLayout->addItem (spacer);
73 buttonLayout->addWidget (btCancel);
74
75 /* resize to fit the aIfaceName in one string */
76 int requiredWidth = mLeName->fontMetrics().width (aIfaceName) +
77 mLeName->frameWidth() * 2 +
78 mLeName->lineWidth() * 2 +
79 inputLayout->spacing() +
80 lbName->fontMetrics().width (lbName->text()) +
81 lbName->frameWidth() * 2 +
82 lbName->lineWidth() * 2 +
83 mainLayout->margin() * 2;
84 resize (requiredWidth, minimumHeight());
85
86 /* Validate interface name field */
87 validate();
88 }
89
90 ~VBoxAddNIDialog() {}
91
92 QString getName() { return mLeName->text(); }
93
94private slots:
95
96 void validate()
97 {
98 mBtOk->setEnabled (!mLeName->text().isEmpty());
99 }
100
101private:
102
103 void showEvent (QShowEvent *aEvent)
104 {
105 setFixedHeight (height());
106 QDialog::showEvent (aEvent);
107 }
108
109 QPushButton *mBtOk;
110 QLineEdit *mLeName;
111};
112
113
114/**
115 * Calculates a suitable page step size for the given max value.
116 * The returned size is so that there will be no more than 32 pages.
117 * The minimum returned page size is 4.
118 */
119static int calcPageStep (int aMax)
120{
121 /* reasonable max. number of page steps is 32 */
122 uint page = ((uint) aMax + 31) / 32;
123 /* make it a power of 2 */
124 uint p = page, p2 = 0x1;
125 while ((p >>= 1))
126 p2 <<= 1;
127 if (page != p2)
128 p2 <<= 1;
129 if (p2 < 4)
130 p2 = 4;
131 return (int) p2;
132}
133
134
135/**
136 * QListView class reimplementation to use as boot items table.
137 * It has one unsorted column without header with automated width
138 * resize management.
139 * Keymapping handlers for ctrl-up & ctrl-down are translated into
140 * boot-items up/down moving.
141 */
142class BootItemsTable : public QListView
143{
144 Q_OBJECT
145
146public:
147
148 BootItemsTable (QWidget *aParent, const char *aName)
149 : QListView (aParent, aName)
150 {
151 addColumn (QString::null);
152 header()->hide();
153 setSorting (-1);
154 setColumnWidthMode (0, Maximum);
155 setResizeMode (AllColumns);
156 QWhatsThis::add (this, tr ("Defines the boot device order. "
157 "Use checkboxes to the left to enable or disable "
158 "individual boot devices. Move items up and down to "
159 "change the device order."));
160 setSizePolicy (QSizePolicy::Expanding, QSizePolicy::Preferred);
161 connect (this, SIGNAL (pressed (QListViewItem*)),
162 this, SLOT (processPressed (QListViewItem*)));
163 }
164
165 ~BootItemsTable() {}
166
167 void emitItemToggled() { emit itemToggled(); }
168
169signals:
170
171 void moveItemUp();
172 void moveItemDown();
173 void itemToggled();
174
175private slots:
176
177 void processPressed (QListViewItem *aItem)
178 {
179 if (!aItem)
180 setSelected (currentItem(), true);
181 }
182
183 void keyPressEvent (QKeyEvent *aEvent)
184 {
185 if (aEvent->state() == Qt::ControlButton)
186 {
187 switch (aEvent->key())
188 {
189 case Qt::Key_Up:
190 emit moveItemUp();
191 return;
192 case Qt::Key_Down:
193 emit moveItemDown();
194 return;
195 default:
196 break;
197 }
198 }
199 QListView::keyPressEvent (aEvent);
200 }
201};
202
203
204/**
205 * QWidget class reimplementation to use as boot items widget.
206 * It contains BootItemsTable and two tool-buttons for moving
207 * boot-items up/down.
208 * This widget handles saving/loading CMachine information related
209 * to boot sequience.
210 */
211class BootItemsList : public QWidget
212{
213 Q_OBJECT
214
215 class BootItem : public QCheckListItem
216 {
217 public:
218
219 BootItem (BootItemsTable *aParent, QListViewItem *aAfter,
220 const QString &aName, Type aType)
221 : QCheckListItem (aParent, aAfter, aName, aType) {}
222
223 private:
224
225 void stateChange (bool)
226 {
227 BootItemsTable *table = static_cast<BootItemsTable*> (listView());
228 table->emitItemToggled();
229 }
230 };
231
232public:
233
234 BootItemsList (QWidget *aParent, const char *aName)
235 : QWidget (aParent, aName), mBootTable (0)
236 {
237 /* Setup main widget layout */
238 QHBoxLayout *mainLayout = new QHBoxLayout (this, 0, 6, "mainLayout");
239
240 /* Setup settings layout */
241 mBootTable = new BootItemsTable (this, "mBootTable");
242 connect (mBootTable, SIGNAL (currentChanged (QListViewItem*)),
243 this, SLOT (processCurrentChanged (QListViewItem*)));
244 mainLayout->addWidget (mBootTable);
245
246 /* Setup button's layout */
247 QVBoxLayout *buttonLayout = new QVBoxLayout (mainLayout, 0, "buttonLayout");
248 mBtnUp = new QToolButton (this, "mBtnUp");
249 mBtnDown = new QToolButton (this, "mBtnDown");
250 mBtnUp->setSizePolicy (QSizePolicy::Fixed, QSizePolicy::Fixed);
251 mBtnDown->setSizePolicy (QSizePolicy::Fixed, QSizePolicy::Fixed);
252 QWhatsThis::add (mBtnUp, tr ("Moves the selected boot device up."));
253 QWhatsThis::add (mBtnDown, tr ("Moves the selected boot device down."));
254 QToolTip::add (mBtnUp, tr ("Move Up (Ctrl-Up)"));
255 QToolTip::add (mBtnDown, tr ("Move Down (Ctrl-Down)"));
256 mBtnUp->setAutoRaise (true);
257 mBtnDown->setAutoRaise (true);
258 mBtnUp->setFocusPolicy (QWidget::StrongFocus);
259 mBtnDown->setFocusPolicy (QWidget::StrongFocus);
260 mBtnUp->setIconSet (VBoxGlobal::iconSet ("list_moveup_16px.png",
261 "list_moveup_disabled_16px.png"));
262 mBtnDown->setIconSet (VBoxGlobal::iconSet ("list_movedown_16px.png",
263 "list_movedown_disabled_16px.png"));
264 QSpacerItem *spacer = new QSpacerItem (0, 0, QSizePolicy::Minimum,
265 QSizePolicy::Minimum);
266 connect (mBtnUp, SIGNAL (clicked()), this, SLOT (moveItemUp()));
267 connect (mBtnDown, SIGNAL (clicked()), this, SLOT (moveItemDown()));
268 connect (mBootTable, SIGNAL (moveItemUp()), this, SLOT (moveItemUp()));
269 connect (mBootTable, SIGNAL (moveItemDown()), this, SLOT (moveItemDown()));
270 connect (mBootTable, SIGNAL (itemToggled()), this, SLOT (onItemToggled()));
271 buttonLayout->addWidget (mBtnUp);
272 buttonLayout->addWidget (mBtnDown);
273 buttonLayout->addItem (spacer);
274
275 /* Setup focus proxy for BootItemsList */
276 setFocusProxy (mBootTable);
277 }
278
279 ~BootItemsList() {}
280
281 void fixTabStops()
282 {
283 /* Fixing focus order for BootItemsList */
284 setTabOrder (mBootTable, mBtnUp);
285 setTabOrder (mBtnUp, mBtnDown);
286 }
287
288 void getFromMachine (const CMachine &aMachine)
289 {
290 /* Load boot-items of current VM */
291 QStringList uniqueList;
292 int minimumWidth = 0;
293 for (int i = 1; i <= 4; ++ i)
294 {
295 CEnums::DeviceType type = aMachine.GetBootOrder (i);
296 if (type != CEnums::NoDevice)
297 {
298 QString name = vboxGlobal().toString (type);
299 QCheckListItem *item = new BootItem (mBootTable,
300 mBootTable->lastItem(), name, QCheckListItem::CheckBox);
301 item->setOn (true);
302 uniqueList << name;
303 int width = item->width (mBootTable->fontMetrics(), mBootTable, 0);
304 if (width > minimumWidth) minimumWidth = width;
305 }
306 }
307 /* Load other unique boot-items */
308 for (int i = CEnums::FloppyDevice; i < CEnums::USBDevice; ++ i)
309 {
310 QString name = vboxGlobal().toString ((CEnums::DeviceType) i);
311 if (!uniqueList.contains (name))
312 {
313 QCheckListItem *item = new BootItem (mBootTable,
314 mBootTable->lastItem(), name, QCheckListItem::CheckBox);
315 uniqueList << name;
316 int width = item->width (mBootTable->fontMetrics(), mBootTable, 0);
317 if (width > minimumWidth) minimumWidth = width;
318 }
319 }
320 processCurrentChanged (mBootTable->firstChild());
321 mBootTable->setFixedWidth (minimumWidth +
322 4 /* viewport margin */);
323 mBootTable->setFixedHeight (mBootTable->childCount() *
324 mBootTable->firstChild()->totalHeight() +
325 4 /* viewport margin */);
326 }
327
328 void putBackToMachine (CMachine &aMachine)
329 {
330 QCheckListItem *item = 0;
331 /* Search for checked items */
332 int index = 1;
333 item = static_cast<QCheckListItem*> (mBootTable->firstChild());
334 while (item)
335 {
336 if (item->isOn())
337 {
338 CEnums::DeviceType type =
339 vboxGlobal().toDeviceType (item->text (0));
340 aMachine.SetBootOrder (index++, type);
341 }
342 item = static_cast<QCheckListItem*> (item->nextSibling());
343 }
344 /* Search for non-checked items */
345 item = static_cast<QCheckListItem*> (mBootTable->firstChild());
346 while (item)
347 {
348 if (!item->isOn())
349 aMachine.SetBootOrder (index++, CEnums::NoDevice);
350 item = static_cast<QCheckListItem*> (item->nextSibling());
351 }
352 }
353
354 void processFocusIn (QWidget *aWidget)
355 {
356 if (aWidget == mBootTable)
357 {
358 mBootTable->setSelected (mBootTable->currentItem(), true);
359 processCurrentChanged (mBootTable->currentItem());
360 }
361 else if (aWidget != mBtnUp && aWidget != mBtnDown)
362 {
363 mBootTable->setSelected (mBootTable->currentItem(), false);
364 processCurrentChanged (mBootTable->currentItem());
365 }
366 }
367
368signals:
369
370 void bootSequenceChanged();
371
372private slots:
373
374 void moveItemUp()
375 {
376 QListViewItem *item = mBootTable->currentItem();
377 Assert (item);
378 QListViewItem *itemAbove = item->itemAbove();
379 if (!itemAbove) return;
380 itemAbove->moveItem (item);
381 processCurrentChanged (item);
382 emit bootSequenceChanged();
383 }
384
385 void moveItemDown()
386 {
387 QListViewItem *item = mBootTable->currentItem();
388 Assert (item);
389 QListViewItem *itemBelow = item->itemBelow();
390 if (!itemBelow) return;
391 item->moveItem (itemBelow);
392 processCurrentChanged (item);
393 emit bootSequenceChanged();
394 }
395
396 void onItemToggled()
397 {
398 emit bootSequenceChanged();
399 }
400
401 void processCurrentChanged (QListViewItem *aItem)
402 {
403 bool upEnabled = aItem && aItem->isSelected() && aItem->itemAbove();
404 bool downEnabled = aItem && aItem->isSelected() && aItem->itemBelow();
405 if (mBtnUp->hasFocus() && !upEnabled ||
406 mBtnDown->hasFocus() && !downEnabled)
407 mBootTable->setFocus();
408 mBtnUp->setEnabled (upEnabled);
409 mBtnDown->setEnabled (downEnabled);
410 }
411
412private:
413
414 BootItemsTable *mBootTable;
415 QToolButton *mBtnUp;
416 QToolButton *mBtnDown;
417};
418
419
420/// @todo (dmik) remove?
421///**
422// * Returns the through position of the item in the list view.
423// */
424//static int pos (QListView *lv, QListViewItem *li)
425//{
426// QListViewItemIterator it (lv);
427// int p = -1, c = 0;
428// while (it.current() && p < 0)
429// {
430// if (it.current() == li)
431// p = c;
432// ++ it;
433// ++ c;
434// }
435// return p;
436//}
437
438class USBListItem : public QCheckListItem
439{
440public:
441
442 USBListItem (QListView *aParent, QListViewItem *aAfter)
443 : QCheckListItem (aParent, aAfter, QString::null, CheckBox)
444 , mId (-1) {}
445
446 int mId;
447};
448
449/**
450 * Returns the path to the item in the form of 'grandparent > parent > item'
451 * using the text of the first column of every item.
452 */
453static QString path (QListViewItem *li)
454{
455 static QString sep = ": ";
456 QString p;
457 QListViewItem *cur = li;
458 while (cur)
459 {
460 if (!p.isNull())
461 p = sep + p;
462 p = cur->text (0).simplifyWhiteSpace() + p;
463 cur = cur->parent();
464 }
465 return p;
466}
467
468enum
469{
470 /* listView column numbers */
471 listView_Category = 0,
472 listView_Id = 1,
473 listView_Link = 2,
474 /* lvUSBFilters column numbers */
475 lvUSBFilters_Name = 0,
476};
477
478void VBoxVMSettingsDlg::init()
479{
480 polished = false;
481
482 mResetFirstRunFlag = false;
483
484 setIcon (QPixmap::fromMimeSource ("settings_16px.png"));
485
486 /* all pages are initially valid */
487 valid = true;
488 buttonOk->setEnabled( true );
489
490 /* disable unselecting items by clicking in the unused area of the list */
491 new QIListViewSelectionPreserver (this, listView);
492 /* hide the header and internal columns */
493 listView->header()->hide();
494 listView->setColumnWidthMode (listView_Id, QListView::Manual);
495 listView->setColumnWidthMode (listView_Link, QListView::Manual);
496 listView->hideColumn (listView_Id);
497 listView->hideColumn (listView_Link);
498 /* sort by the id column (to have pages in the desired order) */
499 listView->setSorting (listView_Id);
500 listView->sort();
501 /* disable further sorting (important for network adapters) */
502 listView->setSorting (-1);
503 /* set the first item selected */
504 listView->setSelected (listView->firstChild(), true);
505 listView_currentChanged (listView->firstChild());
506 /* setup status bar icon */
507 warningPixmap->setMaximumSize( 16, 16 );
508 warningPixmap->setPixmap( QMessageBox::standardIcon( QMessageBox::Warning ) );
509
510 /* page title font is derived from the system font */
511 QFont f = font();
512 f.setBold (true);
513 f.setPointSize (f.pointSize() + 2);
514 titleLabel->setFont (f);
515
516 /* setup the what's this label */
517 QApplication::setGlobalMouseTracking (true);
518 qApp->installEventFilter (this);
519 whatsThisTimer = new QTimer (this);
520 connect (whatsThisTimer, SIGNAL (timeout()), this, SLOT (updateWhatsThis()));
521 whatsThisCandidate = NULL;
522
523 whatsThisLabel = new QIRichLabel (this, "whatsThisLabel");
524 VBoxVMSettingsDlgLayout->addWidget (whatsThisLabel, 2, 1);
525
526#ifndef DEBUG
527 /* Enforce rich text format to avoid jumping margins (margins of plain
528 * text labels seem to be smaller). We don't do it in the DEBUG builds to
529 * be able to immediately catch badly formatted text (i.e. text that
530 * contains HTML tags but doesn't start with <qt> so that Qt isn't able to
531 * recognize it as rich text and draws all tags as is instead of doing
532 * formatting). We want to catch this text because this is how it will look
533 * in the whatsthis balloon where we cannot enforce rich text. */
534 whatsThisLabel->setTextFormat (Qt::RichText);
535#endif
536
537 whatsThisLabel->setMaxHeightMode (true);
538 whatsThisLabel->setFocusPolicy (QWidget::NoFocus);
539 whatsThisLabel->setSizePolicy (QSizePolicy::Expanding, QSizePolicy::Fixed);
540 whatsThisLabel->setBackgroundMode (QLabel::PaletteMidlight);
541 whatsThisLabel->setFrameShape (QLabel::Box);
542 whatsThisLabel->setFrameShadow (QLabel::Sunken);
543 whatsThisLabel->setMargin (7);
544 whatsThisLabel->setScaledContents (FALSE);
545 whatsThisLabel->setAlignment (int (QLabel::WordBreak |
546 QLabel::AlignJustify |
547 QLabel::AlignTop));
548
549 whatsThisLabel->setFixedHeight (whatsThisLabel->frameWidth() * 2 +
550 6 /* seems that RichText adds some margin */ +
551 whatsThisLabel->fontMetrics().lineSpacing() * 4);
552 whatsThisLabel->setMinimumWidth (whatsThisLabel->frameWidth() * 2 +
553 6 /* seems that RichText adds some margin */ +
554 whatsThisLabel->fontMetrics().width ('m') * 40);
555
556 /*
557 * setup connections and set validation for pages
558 * ----------------------------------------------------------------------
559 */
560
561 /* General page */
562
563 CSystemProperties sysProps = vboxGlobal().virtualBox().GetSystemProperties();
564
565 const uint MinRAM = sysProps.GetMinGuestRAM();
566 const uint MaxRAM = sysProps.GetMaxGuestRAM();
567 const uint MinVRAM = sysProps.GetMinGuestVRAM();
568 const uint MaxVRAM = sysProps.GetMaxGuestVRAM();
569
570 leName->setValidator( new QRegExpValidator( QRegExp( ".+" ), this ) );
571
572 leRAM->setValidator (new QIntValidator (MinRAM, MaxRAM, this));
573 leVRAM->setValidator (new QIntValidator (MinVRAM, MaxVRAM, this));
574
575 wvalGeneral = new QIWidgetValidator( pageGeneral, this );
576 connect (wvalGeneral, SIGNAL (validityChanged (const QIWidgetValidator *)),
577 this, SLOT(enableOk (const QIWidgetValidator *)));
578
579 tbSelectSavedStateFolder->setIconSet (VBoxGlobal::iconSet ("select_file_16px.png",
580 "select_file_dis_16px.png"));
581 tbResetSavedStateFolder->setIconSet (VBoxGlobal::iconSet ("eraser_16px.png",
582 "eraser_disabled_16px.png"));
583
584 teDescription->setTextFormat (Qt::PlainText);
585
586 /* HDD Images page */
587
588 QWhatsThis::add (static_cast <QWidget *> (grbHDA->child ("qt_groupbox_checkbox")),
589 tr ("When checked, attaches the specified virtual hard disk to the "
590 "Master slot of the Primary IDE controller."));
591 QWhatsThis::add (static_cast <QWidget *> (grbHDB->child ("qt_groupbox_checkbox")),
592 tr ("When checked, attaches the specified virtual hard disk to the "
593 "Slave slot of the Primary IDE controller."));
594 QWhatsThis::add (static_cast <QWidget *> (grbHDD->child ("qt_groupbox_checkbox")),
595 tr ("When checked, attaches the specified virtual hard disk to the "
596 "Slave slot of the Secondary IDE controller."));
597 cbHDA = new VBoxMediaComboBox (grbHDA, "cbHDA", VBoxDefs::HD);
598 cbHDB = new VBoxMediaComboBox (grbHDB, "cbHDB", VBoxDefs::HD);
599 cbHDD = new VBoxMediaComboBox (grbHDD, "cbHDD", VBoxDefs::HD);
600 hdaLayout->insertWidget (0, cbHDA);
601 hdbLayout->insertWidget (0, cbHDB);
602 hddLayout->insertWidget (0, cbHDD);
603 /* sometimes the weirdness of Qt just kills... */
604 setTabOrder (static_cast <QWidget *> (grbHDA->child ("qt_groupbox_checkbox")),
605 cbHDA);
606 setTabOrder (static_cast <QWidget *> (grbHDB->child ("qt_groupbox_checkbox")),
607 cbHDB);
608 setTabOrder (static_cast <QWidget *> (grbHDD->child ("qt_groupbox_checkbox")),
609 cbHDD);
610
611 QWhatsThis::add (cbHDB, tr ("Displays the virtual hard disk to attach to this IDE slot "
612 "and allows to quickly select a different hard disk."));
613 QWhatsThis::add (cbHDD, tr ("Displays the virtual hard disk to attach to this IDE slot "
614 "and allows to quickly select a different hard disk."));
615 QWhatsThis::add (cbHDA, tr ("Displays the virtual hard disk to attach to this IDE slot "
616 "and allows to quickly select a different hard disk."));
617 QWhatsThis::add (cbHDB, tr ("Displays the virtual hard disk to attach to this IDE slot "
618 "and allows to quickly select a different hard disk."));
619 QWhatsThis::add (cbHDD, tr ("Displays the virtual hard disk to attach to this IDE slot "
620 "and allows to quickly select a different hard disk."));
621
622 wvalHDD = new QIWidgetValidator( pageHDD, this );
623 connect (wvalHDD, SIGNAL (validityChanged (const QIWidgetValidator *)),
624 this, SLOT (enableOk (const QIWidgetValidator *)));
625 connect (wvalHDD, SIGNAL (isValidRequested (QIWidgetValidator *)),
626 this, SLOT (revalidate (QIWidgetValidator *)));
627
628 connect (grbHDA, SIGNAL (toggled (bool)), this, SLOT (hdaMediaChanged()));
629 connect (grbHDB, SIGNAL (toggled (bool)), this, SLOT (hdbMediaChanged()));
630 connect (grbHDD, SIGNAL (toggled (bool)), this, SLOT (hddMediaChanged()));
631 connect (cbHDA, SIGNAL (activated (int)), this, SLOT (hdaMediaChanged()));
632 connect (cbHDB, SIGNAL (activated (int)), this, SLOT (hdbMediaChanged()));
633 connect (cbHDD, SIGNAL (activated (int)), this, SLOT (hddMediaChanged()));
634 connect (tbHDA, SIGNAL (clicked()), this, SLOT (showImageManagerHDA()));
635 connect (tbHDB, SIGNAL (clicked()), this, SLOT (showImageManagerHDB()));
636 connect (tbHDD, SIGNAL (clicked()), this, SLOT (showImageManagerHDD()));
637
638 /* setup iconsets -- qdesigner is not capable... */
639 tbHDA->setIconSet (VBoxGlobal::iconSet ("select_file_16px.png",
640 "select_file_dis_16px.png"));
641 tbHDB->setIconSet (VBoxGlobal::iconSet ("select_file_16px.png",
642 "select_file_dis_16px.png"));
643 tbHDD->setIconSet (VBoxGlobal::iconSet ("select_file_16px.png",
644 "select_file_dis_16px.png"));
645
646 /* CD/DVD-ROM Drive Page */
647
648 QWhatsThis::add (static_cast <QWidget *> (bgDVD->child ("qt_groupbox_checkbox")),
649 tr ("When checked, mounts the specified media to the CD/DVD drive of the "
650 "virtual machine. Note that the CD/DVD drive is always connected to the "
651 "Secondary Master IDE controller of the machine."));
652 cbISODVD = new VBoxMediaComboBox (bgDVD, "cbISODVD", VBoxDefs::CD);
653 cdLayout->insertWidget(0, cbISODVD);
654 QWhatsThis::add (cbISODVD, tr ("Displays the image file to mount to the virtual CD/DVD "
655 "drive and allows to quickly select a different image."));
656
657 wvalDVD = new QIWidgetValidator (pageDVD, this);
658 connect (wvalDVD, SIGNAL (validityChanged (const QIWidgetValidator *)),
659 this, SLOT (enableOk (const QIWidgetValidator *)));
660 connect (wvalDVD, SIGNAL (isValidRequested (QIWidgetValidator *)),
661 this, SLOT (revalidate( QIWidgetValidator *)));
662
663 connect (bgDVD, SIGNAL (toggled (bool)), this, SLOT (cdMediaChanged()));
664 connect (rbHostDVD, SIGNAL (stateChanged (int)), wvalDVD, SLOT (revalidate()));
665 connect (rbISODVD, SIGNAL (stateChanged (int)), wvalDVD, SLOT (revalidate()));
666 connect (cbISODVD, SIGNAL (activated (int)), this, SLOT (cdMediaChanged()));
667 connect (tbISODVD, SIGNAL (clicked()), this, SLOT (showImageManagerISODVD()));
668
669 /* setup iconsets -- qdesigner is not capable... */
670 tbISODVD->setIconSet (VBoxGlobal::iconSet ("select_file_16px.png",
671 "select_file_dis_16px.png"));
672
673 /* Floppy Drive Page */
674
675 QWhatsThis::add (static_cast <QWidget *> (bgFloppy->child ("qt_groupbox_checkbox")),
676 tr ("When checked, mounts the specified media to the Floppy drive of the "
677 "virtual machine."));
678 cbISOFloppy = new VBoxMediaComboBox (bgFloppy, "cbISOFloppy", VBoxDefs::FD);
679 fdLayout->insertWidget(0, cbISOFloppy);
680 QWhatsThis::add (cbISOFloppy, tr ("Displays the image file to mount to the virtual Floppy "
681 "drive and allows to quickly select a different image."));
682
683 wvalFloppy = new QIWidgetValidator (pageFloppy, this);
684 connect (wvalFloppy, SIGNAL (validityChanged (const QIWidgetValidator *)),
685 this, SLOT (enableOk (const QIWidgetValidator *)));
686 connect (wvalFloppy, SIGNAL (isValidRequested (QIWidgetValidator *)),
687 this, SLOT (revalidate( QIWidgetValidator *)));
688
689 connect (bgFloppy, SIGNAL (toggled (bool)), this, SLOT (fdMediaChanged()));
690 connect (rbHostFloppy, SIGNAL (stateChanged (int)), wvalFloppy, SLOT (revalidate()));
691 connect (rbISOFloppy, SIGNAL (stateChanged (int)), wvalFloppy, SLOT (revalidate()));
692 connect (cbISOFloppy, SIGNAL (activated (int)), this, SLOT (fdMediaChanged()));
693 connect (tbISOFloppy, SIGNAL (clicked()), this, SLOT (showImageManagerISOFloppy()));
694
695 /* setup iconsets -- qdesigner is not capable... */
696 tbISOFloppy->setIconSet (VBoxGlobal::iconSet ("select_file_16px.png",
697 "select_file_dis_16px.png"));
698
699 /* Audio Page */
700
701 QWhatsThis::add (static_cast <QWidget *> (grbAudio->child ("qt_groupbox_checkbox")),
702 tr ("When checked, the virtual PCI audio card is plugged into the "
703 "virtual machine that uses the specified driver to communicate "
704 "to the host audio card."));
705
706 /* Network Page */
707#ifndef Q_WS_WIN
708 gbInterfaceList->setHidden (true);
709#endif
710
711 /* setup tab widget */
712 QVBoxLayout* pageNetworkLayout = static_cast<QVBoxLayout*> (pageNetwork->layout());
713 tbwNetwork = new QTabWidget (pageNetwork, "tbwNetwork");
714 pageNetworkLayout->insertWidget (0, tbwNetwork);
715 tbwNetwork->setSizePolicy (QSizePolicy::Minimum, QSizePolicy::Minimum);
716 mNoInterfaces = tr ("<No suitable interfaces>");
717 /* setup iconsets */
718 pbHostAdd->setIconSet (VBoxGlobal::iconSet ("add_host_iface_16px.png",
719 "add_host_iface_disabled_16px.png"));
720 pbHostRemove->setIconSet (VBoxGlobal::iconSet ("remove_host_iface_16px.png",
721 "remove_host_iface_disabled_16px.png"));
722 /* setup languages */
723 QToolTip::add (pbHostAdd, tr ("Add"));
724 QToolTip::add (pbHostRemove, tr ("Remove"));
725
726 /* USB Page */
727
728 lvUSBFilters->header()->hide();
729 /* disable sorting */
730 lvUSBFilters->setSorting (-1);
731 /* disable unselecting items by clicking in the unused area of the list */
732 new QIListViewSelectionPreserver (this, lvUSBFilters);
733 /* create the widget stack for filter settings */
734 /// @todo (r=dmik) having a separate settings widget for every USB filter
735 // is not that smart if there are lots of USB filters. The reason for
736 // stacking here is that the stacked widget is used to temporarily store
737 // data of the associated USB filter until the dialog window is accepted.
738 // If we remove stacking, we will have to create a structure to store
739 // editable data of all USB filters while the dialog is open.
740 wstUSBFilters = new QWidgetStack (grbUSBFilters, "wstUSBFilters");
741 grbUSBFiltersLayout->addWidget (wstUSBFilters);
742 /* create a default (disabled) filter settings widget at index 0 */
743 VBoxUSBFilterSettings *settings = new VBoxUSBFilterSettings (wstUSBFilters);
744 settings->setup (VBoxUSBFilterSettings::MachineType);
745 wstUSBFilters->addWidget (settings, 0);
746 lvUSBFilters_currentChanged (NULL);
747
748 /* setup iconsets -- qdesigner is not capable... */
749 tbAddUSBFilter->setIconSet (VBoxGlobal::iconSet ("usb_new_16px.png",
750 "usb_new_disabled_16px.png"));
751 tbAddUSBFilterFrom->setIconSet (VBoxGlobal::iconSet ("usb_add_16px.png",
752 "usb_add_disabled_16px.png"));
753 tbRemoveUSBFilter->setIconSet (VBoxGlobal::iconSet ("usb_remove_16px.png",
754 "usb_remove_disabled_16px.png"));
755 tbUSBFilterUp->setIconSet (VBoxGlobal::iconSet ("usb_moveup_16px.png",
756 "usb_moveup_disabled_16px.png"));
757 tbUSBFilterDown->setIconSet (VBoxGlobal::iconSet ("usb_movedown_16px.png",
758 "usb_movedown_disabled_16px.png"));
759 usbDevicesMenu = new VBoxUSBMenu (this);
760 connect (usbDevicesMenu, SIGNAL(activated(int)), this, SLOT(menuAddUSBFilterFrom_activated(int)));
761 mUSBFilterListModified = false;
762
763 /* VRDP Page */
764
765 QWhatsThis::add (static_cast <QWidget *> (grbVRDP->child ("qt_groupbox_checkbox")),
766 tr ("When checked, the VM will act as a Remote Desktop "
767 "Protocol (RDP) server, allowing remote clients to connect "
768 "and operate the VM (when it is running) "
769 "using a standard RDP client."));
770
771 ULONG maxPort = 65535;
772 leVRDPPort->setValidator (new QIntValidator (0, maxPort, this));
773 leVRDPTimeout->setValidator (new QIntValidator (0, maxPort, this));
774 wvalVRDP = new QIWidgetValidator (pageVRDP, this);
775 connect (wvalVRDP, SIGNAL (validityChanged (const QIWidgetValidator *)),
776 this, SLOT (enableOk (const QIWidgetValidator *)));
777 connect (wvalVRDP, SIGNAL (isValidRequested (QIWidgetValidator *)),
778 this, SLOT (revalidate( QIWidgetValidator *)));
779
780 connect (grbVRDP, SIGNAL (toggled (bool)), wvalFloppy, SLOT (revalidate()));
781 connect (leVRDPPort, SIGNAL (textChanged (const QString&)), wvalFloppy, SLOT (revalidate()));
782 connect (leVRDPTimeout, SIGNAL (textChanged (const QString&)), wvalFloppy, SLOT (revalidate()));
783
784 /* Shared Folders Page */
785
786 QVBoxLayout* pageFoldersLayout = new QVBoxLayout (pageFolders, 0, 10, "pageFoldersLayout");
787 mSharedFolders = new VBoxSharedFoldersSettings (pageFolders, "sharedFolders");
788 mSharedFolders->setDialogType (VBoxSharedFoldersSettings::MachineType);
789 pageFoldersLayout->addWidget (mSharedFolders);
790
791 /*
792 * set initial values
793 * ----------------------------------------------------------------------
794 */
795
796 /* General page */
797
798 cbOS->insertStringList (vboxGlobal().vmGuestOSTypeDescriptions());
799
800 slRAM->setPageStep (calcPageStep (MaxRAM));
801 slRAM->setLineStep (slRAM->pageStep() / 4);
802 slRAM->setTickInterval (slRAM->pageStep());
803 /* setup the scale so that ticks are at page step boundaries */
804 slRAM->setMinValue ((MinRAM / slRAM->pageStep()) * slRAM->pageStep());
805 slRAM->setMaxValue (MaxRAM);
806 txRAMMin->setText (tr ("<qt>%1&nbsp;MB</qt>").arg (MinRAM));
807 txRAMMax->setText (tr ("<qt>%1&nbsp;MB</qt>").arg (MaxRAM));
808 /* limit min/max. size of QLineEdit */
809 leRAM->setMaximumSize (leRAM->fontMetrics().width ("99999")
810 + leRAM->frameWidth() * 2,
811 leRAM->minimumSizeHint().height());
812 leRAM->setMinimumSize (leRAM->maximumSize());
813 /* ensure leRAM value and validation is updated */
814 slRAM_valueChanged (slRAM->value());
815
816 slVRAM->setPageStep (calcPageStep (MaxVRAM));
817 slVRAM->setLineStep (slVRAM->pageStep() / 4);
818 slVRAM->setTickInterval (slVRAM->pageStep());
819 /* setup the scale so that ticks are at page step boundaries */
820 slVRAM->setMinValue ((MinVRAM / slVRAM->pageStep()) * slVRAM->pageStep());
821 slVRAM->setMaxValue (MaxVRAM);
822 txVRAMMin->setText (tr ("<qt>%1&nbsp;MB</qt>").arg (MinVRAM));
823 txVRAMMax->setText (tr ("<qt>%1&nbsp;MB</qt>").arg (MaxVRAM));
824 /* limit min/max. size of QLineEdit */
825 leVRAM->setMaximumSize (leVRAM->fontMetrics().width ("99999")
826 + leVRAM->frameWidth() * 2,
827 leVRAM->minimumSizeHint().height());
828 leVRAM->setMinimumSize (leVRAM->maximumSize());
829 /* ensure leVRAM value and validation is updated */
830 slVRAM_valueChanged (slVRAM->value());
831
832 /* Boot-order table */
833 tblBootOrder = new BootItemsList (groupBox12, "tblBootOrder");
834 connect (tblBootOrder, SIGNAL (bootSequenceChanged()),
835 this, SLOT (resetFirstRunFlag()));
836 /* Fixing focus order for BootItemsList */
837 setTabOrder (tbwGeneral, tblBootOrder);
838 setTabOrder (tblBootOrder->focusProxy(), chbEnableACPI);
839 groupBox12Layout->addWidget (tblBootOrder);
840 tblBootOrder->fixTabStops();
841 /* Shared Clipboard mode */
842 cbSharedClipboard->insertItem (vboxGlobal().toString (CEnums::ClipDisabled));
843 cbSharedClipboard->insertItem (vboxGlobal().toString (CEnums::ClipHostToGuest));
844 cbSharedClipboard->insertItem (vboxGlobal().toString (CEnums::ClipGuestToHost));
845 cbSharedClipboard->insertItem (vboxGlobal().toString (CEnums::ClipBidirectional));
846
847 /* HDD Images page */
848
849 /* CD-ROM Drive Page */
850
851 /* Audio Page */
852
853 cbAudioDriver->insertItem (vboxGlobal().toString (CEnums::NullAudioDriver));
854#if defined Q_WS_WIN32
855 cbAudioDriver->insertItem (vboxGlobal().toString (CEnums::DSOUNDAudioDriver));
856#ifdef VBOX_WITH_WINMM
857 cbAudioDriver->insertItem (vboxGlobal().toString (CEnums::WINMMAudioDriver));
858#endif
859#elif defined Q_OS_LINUX
860 cbAudioDriver->insertItem (vboxGlobal().toString (CEnums::OSSAudioDriver));
861#ifdef VBOX_WITH_ALSA
862 cbAudioDriver->insertItem (vboxGlobal().toString (CEnums::ALSAAudioDriver));
863#endif
864#elif defined Q_OS_MACX
865 cbAudioDriver->insertItem (vboxGlobal().toString (CEnums::CoreAudioDriver));
866#endif
867
868 /* Network Page */
869
870 loadInterfacesList();
871
872 /*
873 * update the Ok button state for pages with validation
874 * (validityChanged() connected to enableNext() will do the job)
875 */
876 wvalGeneral->revalidate();
877 wvalHDD->revalidate();
878 wvalDVD->revalidate();
879 wvalFloppy->revalidate();
880
881 /* VRDP Page */
882
883 leVRDPPort->setAlignment (Qt::AlignRight);
884 cbVRDPAuthType->insertItem (vboxGlobal().toString (CEnums::VRDPAuthNull));
885 cbVRDPAuthType->insertItem (vboxGlobal().toString (CEnums::VRDPAuthExternal));
886 cbVRDPAuthType->insertItem (vboxGlobal().toString (CEnums::VRDPAuthGuest));
887 leVRDPTimeout->setAlignment (Qt::AlignRight);
888}
889
890bool VBoxVMSettingsDlg::eventFilter (QObject *object, QEvent *event)
891{
892 if (!object->isWidgetType())
893 return QDialog::eventFilter (object, event);
894
895 QWidget *widget = static_cast <QWidget *> (object);
896 if (widget->topLevelWidget() != this)
897 return QDialog::eventFilter (object, event);
898
899 switch (event->type())
900 {
901 case QEvent::Enter:
902 case QEvent::Leave:
903 {
904 if (event->type() == QEvent::Enter)
905 whatsThisCandidate = widget;
906 else
907 whatsThisCandidate = NULL;
908 whatsThisTimer->start (100, true /* sshot */);
909 break;
910 }
911 case QEvent::FocusIn:
912 {
913 updateWhatsThis (true /* gotFocus */);
914 tblBootOrder->processFocusIn (widget);
915 break;
916 }
917 default:
918 break;
919 }
920
921 return QDialog::eventFilter (object, event);
922}
923
924void VBoxVMSettingsDlg::showEvent (QShowEvent *e)
925{
926 QDialog::showEvent (e);
927
928 /* one may think that QWidget::polish() is the right place to do things
929 * below, but apparently, by the time when QWidget::polish() is called,
930 * the widget style & layout are not fully done, at least the minimum
931 * size hint is not properly calculated. Since this is sometimes necessary,
932 * we provide our own "polish" implementation. */
933
934 if (polished)
935 return;
936
937 polished = true;
938
939 /* update geometry for the dynamically added usb-page to ensure proper
940 * sizeHint calculation by the Qt layout manager */
941 wstUSBFilters->updateGeometry();
942 /* let our toplevel widget calculate its sizeHint properly */
943 QApplication::sendPostedEvents (0, 0);
944
945 layout()->activate();
946
947 /* resize to the miminum possible size */
948 resize (minimumSize());
949
950 VBoxGlobal::centerWidget (this, parentWidget());
951}
952
953void VBoxVMSettingsDlg::updateShortcuts()
954{
955 /* setup necessary combobox item */
956 cbHDA->setCurrentItem (uuidHDA);
957 cbHDB->setCurrentItem (uuidHDB);
958 cbHDD->setCurrentItem (uuidHDD);
959 cbISODVD->setCurrentItem (uuidISODVD);
960 cbISOFloppy->setCurrentItem (uuidISOFloppy);
961 /* check if the enumeration process has been started yet */
962 if (!vboxGlobal().isMediaEnumerationStarted())
963 vboxGlobal().startEnumeratingMedia();
964 else
965 {
966 cbHDA->refresh();
967 cbHDB->refresh();
968 cbHDD->refresh();
969 cbISODVD->refresh();
970 cbISOFloppy->refresh();
971 }
972}
973
974void VBoxVMSettingsDlg::loadInterfacesList()
975{
976#if defined Q_WS_WIN
977 /* clear inner list */
978 mInterfaceList.clear();
979 /* load current inner list */
980 CHostNetworkInterfaceEnumerator en =
981 vboxGlobal().virtualBox().GetHost().GetNetworkInterfaces().Enumerate();
982 while (en.HasMore())
983 mInterfaceList += en.GetNext().GetName();
984 /* save current list item name */
985 QString currentListItemName = lbHostInterface->currentText();
986 /* load current list items */
987 lbHostInterface->clear();
988 if (mInterfaceList.count())
989 lbHostInterface->insertStringList (mInterfaceList);
990 else
991 lbHostInterface->insertItem (mNoInterfaces);
992 /* select current list item */
993 int index = lbHostInterface->index (
994 lbHostInterface->findItem (currentListItemName));
995 if (index == -1)
996 index = 0;
997 lbHostInterface->setCurrentItem (index);
998 lbHostInterface->setSelected (index, true);
999 /* enable/disable interface delete button */
1000 pbHostRemove->setEnabled (!mInterfaceList.isEmpty());
1001#endif
1002}
1003
1004void VBoxVMSettingsDlg::hostInterfaceAdd()
1005{
1006#if defined Q_WS_WIN
1007
1008 /* allow the started helper process to make itself the foreground window */
1009 AllowSetForegroundWindow (ASFW_ANY);
1010
1011 /* search for the max available interface index */
1012 int ifaceNumber = 0;
1013 QString ifaceName = tr ("VirtualBox Host Interface %1");
1014 QRegExp regExp (QString ("^") + ifaceName.arg ("([0-9]+)") + QString ("$"));
1015 for (uint index = 0; index < lbHostInterface->count(); ++ index)
1016 {
1017 QString iface = lbHostInterface->text (index);
1018 int pos = regExp.search (iface);
1019 if (pos != -1)
1020 ifaceNumber = regExp.cap (1).toInt() > ifaceNumber ?
1021 regExp.cap (1).toInt() : ifaceNumber;
1022 }
1023
1024 /* creating add host interface dialog */
1025 VBoxAddNIDialog dlg (this, ifaceName.arg (++ ifaceNumber));
1026 if (dlg.exec() != QDialog::Accepted)
1027 return;
1028 QString iName = dlg.getName();
1029
1030 /* create interface */
1031 CHost host = vboxGlobal().virtualBox().GetHost();
1032 CHostNetworkInterface iFace;
1033 CProgress progress = host.CreateHostNetworkInterface (iName, iFace);
1034 if (host.isOk())
1035 {
1036 vboxProblem().showModalProgressDialog (progress, iName, this);
1037 if (progress.GetResultCode() == 0)
1038 {
1039 /* add&select newly created interface */
1040 delete lbHostInterface->findItem (mNoInterfaces);
1041 lbHostInterface->insertItem (iName);
1042 mInterfaceList += iName;
1043 lbHostInterface->setCurrentItem (lbHostInterface->count() - 1);
1044 lbHostInterface->setSelected (lbHostInterface->count() - 1, true);
1045 for (int index = 0; index < tbwNetwork->count(); ++ index)
1046 networkPageUpdate (tbwNetwork->page (index));
1047 /* enable interface delete button */
1048 pbHostRemove->setEnabled (true);
1049 }
1050 else
1051 vboxProblem().cannotCreateHostInterface (progress, iName, this);
1052 }
1053 else
1054 vboxProblem().cannotCreateHostInterface (host, iName, this);
1055
1056 /* allow the started helper process to make itself the foreground window */
1057 AllowSetForegroundWindow (ASFW_ANY);
1058
1059#endif
1060}
1061
1062void VBoxVMSettingsDlg::hostInterfaceRemove()
1063{
1064#if defined Q_WS_WIN
1065
1066 /* allow the started helper process to make itself the foreground window */
1067 AllowSetForegroundWindow (ASFW_ANY);
1068
1069 /* check interface name */
1070 QString iName = lbHostInterface->currentText();
1071 if (iName.isEmpty())
1072 return;
1073
1074 /* asking user about deleting selected network interface */
1075 int delNetIface = vboxProblem().message (this, VBoxProblemReporter::Question,
1076 tr ("<p>Do you want to remove the selected host network interface "
1077 "<nobr><b>%1</b>?</nobr></p>"
1078 "<p><b>Note:</b> This interface may be in use by one or more "
1079 "network adapters of this or another VM. After it is removed, these "
1080 "adapters will no longer work until you correct their settings by "
1081 "either choosing a different interface name or a different adapter "
1082 "attachment type.</p>").arg (iName),
1083 0, /* autoConfirmId */
1084 QIMessageBox::Ok | QIMessageBox::Default,
1085 QIMessageBox::Cancel | QIMessageBox::Escape);
1086 if (delNetIface == QIMessageBox::Cancel)
1087 return;
1088
1089 CHost host = vboxGlobal().virtualBox().GetHost();
1090 CHostNetworkInterface iFace = host.GetNetworkInterfaces().FindByName (iName);
1091 if (host.isOk())
1092 {
1093 /* delete interface */
1094 CProgress progress = host.RemoveHostNetworkInterface (iFace.GetId(), iFace);
1095 if (host.isOk())
1096 {
1097 vboxProblem().showModalProgressDialog (progress, iName, this);
1098 if (progress.GetResultCode() == 0)
1099 {
1100 if (lbHostInterface->count() == 1)
1101 {
1102 lbHostInterface->insertItem (mNoInterfaces);
1103 /* disable interface delete button */
1104 pbHostRemove->setEnabled (false);
1105 }
1106 delete lbHostInterface->findItem (iName);
1107 lbHostInterface->setSelected (lbHostInterface->currentItem(), true);
1108 mInterfaceList.erase (mInterfaceList.find (iName));
1109 for (int index = 0; index < tbwNetwork->count(); ++ index)
1110 networkPageUpdate (tbwNetwork->page (index));
1111 }
1112 else
1113 vboxProblem().cannotRemoveHostInterface (progress, iFace, this);
1114 }
1115 }
1116
1117 if (!host.isOk())
1118 vboxProblem().cannotRemoveHostInterface (host, iFace, this);
1119#endif
1120}
1121
1122void VBoxVMSettingsDlg::networkPageUpdate (QWidget *aWidget)
1123{
1124 if (!aWidget) return;
1125#if defined Q_WS_WIN
1126 VBoxVMNetworkSettings *set = static_cast<VBoxVMNetworkSettings*> (aWidget);
1127 set->loadList (mInterfaceList, mNoInterfaces);
1128 set->revalidate();
1129#endif
1130}
1131
1132
1133void VBoxVMSettingsDlg::resetFirstRunFlag()
1134{
1135 mResetFirstRunFlag = true;
1136}
1137
1138
1139void VBoxVMSettingsDlg::hdaMediaChanged()
1140{
1141 resetFirstRunFlag();
1142 uuidHDA = grbHDA->isChecked() ? cbHDA->getId() : QUuid();
1143 txHDA->setText (getHdInfo (grbHDA, uuidHDA));
1144 /* revailidate */
1145 wvalHDD->revalidate();
1146}
1147
1148
1149void VBoxVMSettingsDlg::hdbMediaChanged()
1150{
1151 resetFirstRunFlag();
1152 uuidHDB = grbHDB->isChecked() ? cbHDB->getId() : QUuid();
1153 txHDB->setText (getHdInfo (grbHDB, uuidHDB));
1154 /* revailidate */
1155 wvalHDD->revalidate();
1156}
1157
1158
1159void VBoxVMSettingsDlg::hddMediaChanged()
1160{
1161 resetFirstRunFlag();
1162 uuidHDD = grbHDD->isChecked() ? cbHDD->getId() : QUuid();
1163 txHDD->setText (getHdInfo (grbHDD, uuidHDD));
1164 /* revailidate */
1165 wvalHDD->revalidate();
1166}
1167
1168
1169void VBoxVMSettingsDlg::cdMediaChanged()
1170{
1171 resetFirstRunFlag();
1172 uuidISODVD = bgDVD->isChecked() ? cbISODVD->getId() : QUuid();
1173 /* revailidate */
1174 wvalDVD->revalidate();
1175}
1176
1177
1178void VBoxVMSettingsDlg::fdMediaChanged()
1179{
1180 resetFirstRunFlag();
1181 uuidISOFloppy = bgFloppy->isChecked() ? cbISOFloppy->getId() : QUuid();
1182 /* revailidate */
1183 wvalFloppy->revalidate();
1184}
1185
1186
1187QString VBoxVMSettingsDlg::getHdInfo (QGroupBox *aGroupBox, QUuid aId)
1188{
1189 QString notAttached = tr ("<not attached>", "hard disk");
1190 if (aId.isNull())
1191 return notAttached;
1192 return aGroupBox->isChecked() ?
1193 vboxGlobal().details (vboxGlobal().virtualBox().GetHardDisk (aId), true) :
1194 notAttached;
1195}
1196
1197void VBoxVMSettingsDlg::updateWhatsThis (bool gotFocus /* = false */)
1198{
1199 QString text;
1200
1201 QWidget *widget = NULL;
1202 if (!gotFocus)
1203 {
1204 if (whatsThisCandidate != NULL && whatsThisCandidate != this)
1205 widget = whatsThisCandidate;
1206 }
1207 else
1208 {
1209 widget = focusData()->focusWidget();
1210 }
1211 /* if the given widget lacks the whats'this text, look at its parent */
1212 while (widget && widget != this)
1213 {
1214 text = QWhatsThis::textFor (widget);
1215 if (!text.isEmpty())
1216 break;
1217 widget = widget->parentWidget();
1218 }
1219
1220 if (text.isEmpty() && !warningString.isEmpty())
1221 text = warningString;
1222 if (text.isEmpty())
1223 text = QWhatsThis::textFor (this);
1224
1225 whatsThisLabel->setText (text);
1226}
1227
1228void VBoxVMSettingsDlg::setWarning (const QString &warning)
1229{
1230 warningString = warning;
1231 if (!warning.isEmpty())
1232 warningString = QString ("<font color=red>%1</font>").arg (warning);
1233
1234 if (!warningString.isEmpty())
1235 whatsThisLabel->setText (warningString);
1236 else
1237 updateWhatsThis (true);
1238}
1239
1240/**
1241 * Sets up this dialog.
1242 *
1243 * If @a aCategory is non-null, it should be one of values from the hidden
1244 * '[cat]' column of #listView (see VBoxVMSettingsDlg.ui in qdesigner)
1245 * prepended with the '#' sign. In this case, the specified category page
1246 * will be activated when the dialog is open.
1247 *
1248 * If @a aWidget is non-null, it should be a name of one of widgets
1249 * from the given category page. In this case, the specified widget
1250 * will get focus when the dialog is open.
1251 *
1252 * @note Calling this method after the dialog is open has no sense.
1253 *
1254 * @param aCategory Category to select when the dialog is open or null.
1255 * @param aWidget Category to select when the dialog is open or null.
1256 */
1257void VBoxVMSettingsDlg::setup (const QString &aCategory, const QString &aControl)
1258{
1259 if (!aCategory.isNull())
1260 {
1261 /* search for a list view item corresponding to the category */
1262 QListViewItem *item = listView->findItem (aCategory, listView_Link);
1263 if (item)
1264 {
1265 listView->setSelected (item, true);
1266
1267 /* search for a widget with the given name */
1268 if (!aControl.isNull())
1269 {
1270 QObject *obj = widgetStack->visibleWidget()->child (aControl);
1271 if (obj && obj->isWidgetType())
1272 {
1273 QWidget *w = static_cast <QWidget *> (obj);
1274 QWidgetList parents;
1275 QWidget *p = w;
1276 while ((p = p->parentWidget()) != NULL)
1277 {
1278 if (!strcmp (p->className(), "QTabWidget"))
1279 {
1280 /* the tab contents widget is two steps down
1281 * (QTabWidget -> QWidgetStack -> QWidget) */
1282 QWidget *c = parents.last();
1283 if (c)
1284 c = parents.prev();
1285 if (c)
1286 static_cast <QTabWidget *> (p)->showPage (c);
1287 }
1288 parents.append (p);
1289 }
1290
1291 w->setFocus();
1292 }
1293 }
1294 }
1295 }
1296}
1297
1298void VBoxVMSettingsDlg::listView_currentChanged (QListViewItem *item)
1299{
1300 Assert (item);
1301 int id = item->text (1).toInt();
1302 Assert (id >= 0);
1303 titleLabel->setText (::path (item));
1304 widgetStack->raiseWidget (id);
1305}
1306
1307
1308void VBoxVMSettingsDlg::enableOk( const QIWidgetValidator *wval )
1309{
1310 Q_UNUSED (wval);
1311
1312 /* detect the overall validity */
1313 bool newValid = true;
1314 {
1315 QObjectList *l = this->queryList ("QIWidgetValidator");
1316 QObjectListIt it (*l);
1317 QObject *obj;
1318 while ((obj = it.current()) != 0)
1319 {
1320 newValid &= ((QIWidgetValidator *) obj)->isValid();
1321 ++it;
1322 }
1323 delete l;
1324 }
1325
1326 if (valid != newValid)
1327 {
1328 valid = newValid;
1329 buttonOk->setEnabled (valid);
1330 if (valid)
1331 setWarning(0);
1332 warningLabel->setHidden(valid);
1333 warningPixmap->setHidden(valid);
1334 }
1335}
1336
1337
1338void VBoxVMSettingsDlg::revalidate( QIWidgetValidator *wval )
1339{
1340 /* do individual validations for pages */
1341 QWidget *pg = wval->widget();
1342 bool valid = wval->isOtherValid();
1343
1344 if (pg == pageHDD)
1345 {
1346 CVirtualBox vbox = vboxGlobal().virtualBox();
1347 valid = true;
1348
1349 QValueList <QUuid> uuids;
1350
1351 if (valid && grbHDA->isChecked())
1352 {
1353 if (uuidHDA.isNull())
1354 {
1355 valid = false;
1356 setWarning (tr ("Primary Master hard disk is not selected."));
1357 }
1358 else uuids << uuidHDA;
1359 }
1360
1361 if (valid && grbHDB->isChecked())
1362 {
1363 if (uuidHDB.isNull())
1364 {
1365 valid = false;
1366 setWarning (tr ("Primary Slave hard disk is not selected."));
1367 }
1368 else
1369 {
1370 bool found = uuids.findIndex (uuidHDB) >= 0;
1371 if (found)
1372 {
1373 CHardDisk hd = vbox.GetHardDisk (uuidHDB);
1374 valid = hd.GetType() == CEnums::ImmutableHardDisk;
1375 }
1376 if (valid)
1377 uuids << uuidHDB;
1378 else
1379 setWarning (tr ("Primary Slave hard disk is already attached "
1380 "to a different slot."));
1381 }
1382 }
1383
1384 if (valid && grbHDD->isChecked())
1385 {
1386 if (uuidHDD.isNull())
1387 {
1388 valid = false;
1389 setWarning (tr ("Secondary Slave hard disk is not selected."));
1390 }
1391 else
1392 {
1393 bool found = uuids.findIndex (uuidHDD) >= 0;
1394 if (found)
1395 {
1396 CHardDisk hd = vbox.GetHardDisk (uuidHDD);
1397 valid = hd.GetType() == CEnums::ImmutableHardDisk;
1398 }
1399 if (valid)
1400 uuids << uuidHDB;
1401 else
1402 setWarning (tr ("Secondary Slave hard disk is already attached "
1403 "to a different slot."));
1404 }
1405 }
1406
1407 cbHDA->setEnabled (grbHDA->isChecked());
1408 cbHDB->setEnabled (grbHDB->isChecked());
1409 cbHDD->setEnabled (grbHDD->isChecked());
1410 tbHDA->setEnabled (grbHDA->isChecked());
1411 tbHDB->setEnabled (grbHDB->isChecked());
1412 tbHDD->setEnabled (grbHDD->isChecked());
1413 }
1414 else if (pg == pageDVD)
1415 {
1416 if (!bgDVD->isChecked())
1417 rbHostDVD->setChecked(false), rbISODVD->setChecked(false);
1418 else if (!rbHostDVD->isChecked() && !rbISODVD->isChecked())
1419 rbHostDVD->setChecked(true);
1420
1421 valid = !(rbISODVD->isChecked() && uuidISODVD.isNull());
1422
1423 cbHostDVD->setEnabled (rbHostDVD->isChecked());
1424 cbPassthrough->setEnabled (rbHostDVD->isChecked());
1425
1426 cbISODVD->setEnabled (rbISODVD->isChecked());
1427 tbISODVD->setEnabled (rbISODVD->isChecked());
1428
1429 if (!valid)
1430 setWarning (tr ("CD/DVD drive image file is not selected."));
1431 }
1432 else if (pg == pageFloppy)
1433 {
1434 if (!bgFloppy->isChecked())
1435 rbHostFloppy->setChecked(false), rbISOFloppy->setChecked(false);
1436 else if (!rbHostFloppy->isChecked() && !rbISOFloppy->isChecked())
1437 rbHostFloppy->setChecked(true);
1438
1439 valid = !(rbISOFloppy->isChecked() && uuidISOFloppy.isNull());
1440
1441 cbHostFloppy->setEnabled (rbHostFloppy->isChecked());
1442
1443 cbISOFloppy->setEnabled (rbISOFloppy->isChecked());
1444 tbISOFloppy->setEnabled (rbISOFloppy->isChecked());
1445
1446 if (!valid)
1447 setWarning (tr ("Floppy drive image file is not selected."));
1448 }
1449 else if (pg == pageNetwork)
1450 {
1451 int index = 0;
1452 for (; index < tbwNetwork->count(); ++index)
1453 {
1454 QWidget *tab = tbwNetwork->page (index);
1455 VBoxVMNetworkSettings *set = static_cast<VBoxVMNetworkSettings*> (tab);
1456 valid = set->isPageValid (mInterfaceList);
1457 if (!valid) break;
1458 }
1459 if (!valid)
1460 setWarning (tr ("Incorrect host network interface is selected "
1461 "for Adapter %1.").arg (index));
1462 }
1463 else if (pg == pageVRDP)
1464 {
1465 if (pageVRDP->isEnabled())
1466 {
1467 valid = !(grbVRDP->isChecked() &&
1468 (leVRDPPort->text().isEmpty() || leVRDPTimeout->text().isEmpty()));
1469 if (!valid && leVRDPPort->text().isEmpty())
1470 setWarning (tr ("VRDP Port is not set."));
1471 if (!valid && leVRDPTimeout->text().isEmpty())
1472 setWarning (tr ("VRDP Timeout is not set."));
1473 }
1474 else
1475 valid = true;
1476 }
1477
1478 wval->setOtherValid (valid);
1479}
1480
1481
1482void VBoxVMSettingsDlg::getFromMachine (const CMachine &machine)
1483{
1484 cmachine = machine;
1485
1486 setCaption (machine.GetName() + tr (" - Settings"));
1487
1488 CVirtualBox vbox = vboxGlobal().virtualBox();
1489 CBIOSSettings biosSettings = cmachine.GetBIOSSettings();
1490
1491 /* name */
1492 leName->setText (machine.GetName());
1493
1494 /* OS type */
1495 QString typeId = machine.GetOSTypeId();
1496 cbOS->setCurrentItem (vboxGlobal().vmGuestOSTypeIndex (typeId));
1497 cbOS_activated (cbOS->currentItem());
1498
1499 /* RAM size */
1500 slRAM->setValue (machine.GetMemorySize());
1501
1502 /* VRAM size */
1503 slVRAM->setValue (machine.GetVRAMSize());
1504
1505 /* Boot-order */
1506 tblBootOrder->getFromMachine (machine);
1507
1508 /* ACPI */
1509 chbEnableACPI->setChecked (biosSettings.GetACPIEnabled());
1510
1511 /* IO APIC */
1512 chbEnableIOAPIC->setChecked (biosSettings.GetIOAPICEnabled());
1513
1514 /* VT-x/AMD-V */
1515 machine.GetHWVirtExEnabled() == CEnums::False ? chbVTX->setChecked (false) :
1516 machine.GetHWVirtExEnabled() == CEnums::True ? chbVTX->setChecked (true) :
1517 chbVTX->setNoChange();
1518
1519 /* Saved state folder */
1520 leSnapshotFolder->setText (machine.GetSnapshotFolder());
1521
1522 /* Description */
1523 teDescription->setText (machine.GetDescription());
1524
1525 /* Shared clipboard mode */
1526 cbSharedClipboard->setCurrentItem (machine.GetClipboardMode());
1527
1528 /* hard disk images */
1529 {
1530 struct
1531 {
1532 CEnums::DiskControllerType ctl;
1533 LONG dev;
1534 struct {
1535 QGroupBox *grb;
1536 QComboBox *cbb;
1537 QLabel *tx;
1538 QUuid *uuid;
1539 } data;
1540 }
1541 diskSet[] =
1542 {
1543 { CEnums::IDE0Controller, 0, {grbHDA, cbHDA, txHDA, &uuidHDA} },
1544 { CEnums::IDE0Controller, 1, {grbHDB, cbHDB, txHDB, &uuidHDB} },
1545 { CEnums::IDE1Controller, 1, {grbHDD, cbHDD, txHDD, &uuidHDD} },
1546 };
1547
1548 grbHDA->setChecked (false);
1549 grbHDB->setChecked (false);
1550 grbHDD->setChecked (false);
1551
1552 CHardDiskAttachmentEnumerator en =
1553 machine.GetHardDiskAttachments().Enumerate();
1554 while (en.HasMore())
1555 {
1556 CHardDiskAttachment hda = en.GetNext();
1557 for (uint i = 0; i < SIZEOF_ARRAY (diskSet); i++)
1558 {
1559 if (diskSet [i].ctl == hda.GetController() &&
1560 diskSet [i].dev == hda.GetDeviceNumber())
1561 {
1562 CHardDisk hd = hda.GetHardDisk();
1563 CHardDisk root = hd.GetRoot();
1564 QString src = root.GetLocation();
1565 if (hd.GetStorageType() == CEnums::VirtualDiskImage)
1566 {
1567 QFileInfo fi (src);
1568 src = fi.fileName() + " (" +
1569 QDir::convertSeparators (fi.dirPath (true)) + ")";
1570 }
1571 diskSet [i].data.grb->setChecked (true);
1572 diskSet [i].data.tx->setText (vboxGlobal().details (hd));
1573 *(diskSet [i].data.uuid) = QUuid (root.GetId());
1574 }
1575 }
1576 }
1577 }
1578
1579 /* floppy image */
1580 {
1581 /* read out the host floppy drive list and prepare the combobox */
1582 CHostFloppyDriveCollection coll =
1583 vboxGlobal().virtualBox().GetHost().GetFloppyDrives();
1584 hostFloppies.resize (coll.GetCount());
1585 cbHostFloppy->clear();
1586 int id = 0;
1587 CHostFloppyDriveEnumerator en = coll.Enumerate();
1588 while (en.HasMore())
1589 {
1590 CHostFloppyDrive hostFloppy = en.GetNext();
1591 /** @todo set icon? */
1592 QString name = hostFloppy.GetName();
1593 QString description = hostFloppy.GetDescription();
1594 QString fullName = description.isEmpty() ?
1595 name :
1596 QString ("%1 (%2)").arg (description, name);
1597 cbHostFloppy->insertItem (fullName, id);
1598 hostFloppies [id] = hostFloppy;
1599 ++ id;
1600 }
1601
1602 CFloppyDrive floppy = machine.GetFloppyDrive();
1603 switch (floppy.GetState())
1604 {
1605 case CEnums::HostDriveCaptured:
1606 {
1607 CHostFloppyDrive drv = floppy.GetHostDrive();
1608 QString name = drv.GetName();
1609 QString description = drv.GetDescription();
1610 QString fullName = description.isEmpty() ?
1611 name :
1612 QString ("%1 (%2)").arg (description, name);
1613 if (coll.FindByName (name).isNull())
1614 {
1615 /*
1616 * if the floppy drive is not currently available,
1617 * add it to the end of the list with a special mark
1618 */
1619 cbHostFloppy->insertItem ("* " + fullName);
1620 cbHostFloppy->setCurrentItem (cbHostFloppy->count() - 1);
1621 }
1622 else
1623 {
1624 /* this will select the correct item from the prepared list */
1625 cbHostFloppy->setCurrentText (fullName);
1626 }
1627 rbHostFloppy->setChecked (true);
1628 break;
1629 }
1630 case CEnums::ImageMounted:
1631 {
1632 CFloppyImage img = floppy.GetImage();
1633 QString src = img.GetFilePath();
1634 AssertMsg (!src.isNull(), ("Image file must not be null"));
1635 QFileInfo fi (src);
1636 rbISOFloppy->setChecked (true);
1637 uuidISOFloppy = QUuid (img.GetId());
1638 break;
1639 }
1640 case CEnums::NotMounted:
1641 {
1642 bgFloppy->setChecked(false);
1643 break;
1644 }
1645 default:
1646 AssertMsgFailed (("invalid floppy state: %d\n", floppy.GetState()));
1647 }
1648 }
1649
1650 /* CD/DVD-ROM image */
1651 {
1652 /* read out the host DVD drive list and prepare the combobox */
1653 CHostDVDDriveCollection coll =
1654 vboxGlobal().virtualBox().GetHost().GetDVDDrives();
1655 hostDVDs.resize (coll.GetCount());
1656 cbHostDVD->clear();
1657 int id = 0;
1658 CHostDVDDriveEnumerator en = coll.Enumerate();
1659 while (en.HasMore())
1660 {
1661 CHostDVDDrive hostDVD = en.GetNext();
1662 /// @todo (r=dmik) set icon?
1663 QString name = hostDVD.GetName();
1664 QString description = hostDVD.GetDescription();
1665 QString fullName = description.isEmpty() ?
1666 name :
1667 QString ("%1 (%2)").arg (description, name);
1668 cbHostDVD->insertItem (fullName, id);
1669 hostDVDs [id] = hostDVD;
1670 ++ id;
1671 }
1672
1673 CDVDDrive dvd = machine.GetDVDDrive();
1674 switch (dvd.GetState())
1675 {
1676 case CEnums::HostDriveCaptured:
1677 {
1678 CHostDVDDrive drv = dvd.GetHostDrive();
1679 QString name = drv.GetName();
1680 QString description = drv.GetDescription();
1681 QString fullName = description.isEmpty() ?
1682 name :
1683 QString ("%1 (%2)").arg (description, name);
1684 if (coll.FindByName (name).isNull())
1685 {
1686 /*
1687 * if the DVD drive is not currently available,
1688 * add it to the end of the list with a special mark
1689 */
1690 cbHostDVD->insertItem ("* " + fullName);
1691 cbHostDVD->setCurrentItem (cbHostDVD->count() - 1);
1692 }
1693 else
1694 {
1695 /* this will select the correct item from the prepared list */
1696 cbHostDVD->setCurrentText (fullName);
1697 }
1698 rbHostDVD->setChecked (true);
1699 cbPassthrough->setChecked (dvd.GetPassthrough());
1700 break;
1701 }
1702 case CEnums::ImageMounted:
1703 {
1704 CDVDImage img = dvd.GetImage();
1705 QString src = img.GetFilePath();
1706 AssertMsg (!src.isNull(), ("Image file must not be null"));
1707 QFileInfo fi (src);
1708 rbISODVD->setChecked (true);
1709 uuidISODVD = QUuid (img.GetId());
1710 break;
1711 }
1712 case CEnums::NotMounted:
1713 {
1714 bgDVD->setChecked(false);
1715 break;
1716 }
1717 default:
1718 AssertMsgFailed (("invalid DVD state: %d\n", dvd.GetState()));
1719 }
1720 }
1721
1722 /* audio */
1723 {
1724 CAudioAdapter audio = machine.GetAudioAdapter();
1725 grbAudio->setChecked (audio.GetEnabled());
1726 cbAudioDriver->setCurrentText (vboxGlobal().toString (audio.GetAudioDriver()));
1727 }
1728
1729 /* network */
1730 {
1731 ulong count = vbox.GetSystemProperties().GetNetworkAdapterCount();
1732 for (ulong slot = 0; slot < count; ++ slot)
1733 {
1734 CNetworkAdapter adapter = machine.GetNetworkAdapter (slot);
1735 addNetworkAdapter (adapter);
1736 }
1737 }
1738
1739 /* USB */
1740 {
1741 CUSBController ctl = machine.GetUSBController();
1742
1743 if (ctl.isNull())
1744 {
1745 /* disable the USB controller category if the USB controller is
1746 * not available (i.e. in VirtualBox OSE) */
1747
1748 QListViewItem *usbItem = listView->findItem ("#usb", listView_Link);
1749 Assert (usbItem);
1750 if (usbItem)
1751 usbItem->setVisible (false);
1752
1753 /* disable validators if any */
1754 pageUSB->setEnabled (false);
1755
1756 /* Show an error message (if there is any).
1757 * Note that we don't use the generic cannotLoadMachineSettings()
1758 * call here because we want this message to be suppressable. */
1759 vboxProblem().cannotAccessUSB (machine);
1760 }
1761 else
1762 {
1763 cbEnableUSBController->setChecked (ctl.GetEnabled());
1764
1765 CUSBDeviceFilterEnumerator en = ctl.GetDeviceFilters().Enumerate();
1766 while (en.HasMore())
1767 addUSBFilter (en.GetNext(), false /* isNew */);
1768
1769 lvUSBFilters->setCurrentItem (lvUSBFilters->firstChild());
1770 /* silly Qt -- doesn't emit currentChanged after adding the
1771 * first item to an empty list */
1772 lvUSBFilters_currentChanged (lvUSBFilters->firstChild());
1773 }
1774 }
1775
1776 /* vrdp */
1777 {
1778 CVRDPServer vrdp = machine.GetVRDPServer();
1779
1780 if (vrdp.isNull())
1781 {
1782 /* disable the VRDP category if VRDP is
1783 * not available (i.e. in VirtualBox OSE) */
1784
1785 QListViewItem *vrdpItem = listView->findItem ("#vrdp", listView_Link);
1786 Assert (vrdpItem);
1787 if (vrdpItem)
1788 vrdpItem->setVisible (false);
1789
1790 /* disable validators if any */
1791 pageVRDP->setEnabled (false);
1792
1793 /* if machine has something to say, show the message */
1794 vboxProblem().cannotLoadMachineSettings (machine, false /* strict */);
1795 }
1796 else
1797 {
1798 grbVRDP->setChecked (vrdp.GetEnabled());
1799 leVRDPPort->setText (QString::number (vrdp.GetPort()));
1800 cbVRDPAuthType->setCurrentText (vboxGlobal().toString (vrdp.GetAuthType()));
1801 leVRDPTimeout->setText (QString::number (vrdp.GetAuthTimeout()));
1802 }
1803 }
1804
1805 /* shared folders */
1806 {
1807 mSharedFolders->getFromMachine (machine);
1808 }
1809
1810 /* request for media shortcuts update */
1811 cbHDA->setBelongsTo (machine.GetId());
1812 cbHDB->setBelongsTo (machine.GetId());
1813 cbHDD->setBelongsTo (machine.GetId());
1814 updateShortcuts();
1815
1816 /* revalidate pages with custom validation */
1817 wvalHDD->revalidate();
1818 wvalDVD->revalidate();
1819 wvalFloppy->revalidate();
1820 wvalVRDP->revalidate();
1821}
1822
1823
1824COMResult VBoxVMSettingsDlg::putBackToMachine()
1825{
1826 CVirtualBox vbox = vboxGlobal().virtualBox();
1827 CBIOSSettings biosSettings = cmachine.GetBIOSSettings();
1828
1829 /* name */
1830 cmachine.SetName (leName->text());
1831
1832 /* OS type */
1833 CGuestOSType type = vboxGlobal().vmGuestOSType (cbOS->currentItem());
1834 AssertMsg (!type.isNull(), ("vmGuestOSType() must return non-null type"));
1835 cmachine.SetOSTypeId (type.GetId());
1836
1837 /* RAM size */
1838 cmachine.SetMemorySize (slRAM->value());
1839
1840 /* VRAM size */
1841 cmachine.SetVRAMSize (slVRAM->value());
1842
1843 /* boot order */
1844 tblBootOrder->putBackToMachine (cmachine);
1845
1846 /* ACPI */
1847 biosSettings.SetACPIEnabled (chbEnableACPI->isChecked());
1848
1849 /* IO APIC */
1850 biosSettings.SetIOAPICEnabled (chbEnableIOAPIC->isChecked());
1851
1852 /* VT-x/AMD-V */
1853 cmachine.SetHWVirtExEnabled (
1854 chbVTX->state() == QButton::Off ? CEnums::False :
1855 chbVTX->state() == QButton::On ? CEnums::True : CEnums::Default);
1856
1857 /* Saved state folder */
1858 if (leSnapshotFolder->isModified())
1859 cmachine.SetSnapshotFolder (leSnapshotFolder->text());
1860
1861 /* Description */
1862 cmachine.SetDescription (teDescription->text());
1863
1864 /* Shared clipboard mode */
1865 cmachine.SetClipboardMode ((CEnums::ClipboardMode)cbSharedClipboard->currentItem());
1866
1867 /* hard disk images */
1868 {
1869 struct
1870 {
1871 CEnums::DiskControllerType ctl;
1872 LONG dev;
1873 struct {
1874 QGroupBox *grb;
1875 QUuid *uuid;
1876 } data;
1877 }
1878 diskSet[] =
1879 {
1880 { CEnums::IDE0Controller, 0, {grbHDA, &uuidHDA} },
1881 { CEnums::IDE0Controller, 1, {grbHDB, &uuidHDB} },
1882 { CEnums::IDE1Controller, 1, {grbHDD, &uuidHDD} }
1883 };
1884
1885 /*
1886 * first, detach all disks (to ensure we can reattach them to different
1887 * controllers / devices, when appropriate)
1888 */
1889 CHardDiskAttachmentEnumerator en =
1890 cmachine.GetHardDiskAttachments().Enumerate();
1891 while (en.HasMore())
1892 {
1893 CHardDiskAttachment hda = en.GetNext();
1894 for (uint i = 0; i < SIZEOF_ARRAY (diskSet); i++)
1895 {
1896 if (diskSet [i].ctl == hda.GetController() &&
1897 diskSet [i].dev == hda.GetDeviceNumber())
1898 {
1899 cmachine.DetachHardDisk (diskSet [i].ctl, diskSet [i].dev);
1900 if (!cmachine.isOk())
1901 vboxProblem().cannotDetachHardDisk (
1902 this, cmachine, diskSet [i].ctl, diskSet [i].dev);
1903 }
1904 }
1905 }
1906
1907 /* now, attach new disks */
1908 for (uint i = 0; i < SIZEOF_ARRAY (diskSet); i++)
1909 {
1910 QUuid *newId = diskSet [i].data.uuid;
1911 if (diskSet [i].data.grb->isChecked() && !(*newId).isNull())
1912 {
1913 cmachine.AttachHardDisk (*newId, diskSet [i].ctl, diskSet [i].dev);
1914 if (!cmachine.isOk())
1915 vboxProblem().cannotAttachHardDisk (
1916 this, cmachine, *newId, diskSet [i].ctl, diskSet [i].dev);
1917 }
1918 }
1919 }
1920
1921 /* floppy image */
1922 {
1923 CFloppyDrive floppy = cmachine.GetFloppyDrive();
1924 if (!bgFloppy->isChecked())
1925 {
1926 floppy.Unmount();
1927 }
1928 else if (rbHostFloppy->isChecked())
1929 {
1930 int id = cbHostFloppy->currentItem();
1931 Assert (id >= 0);
1932 if (id < (int) hostFloppies.count())
1933 floppy.CaptureHostDrive (hostFloppies [id]);
1934 /*
1935 * otherwise the selected drive is not yet available, leave it
1936 * as is
1937 */
1938 }
1939 else if (rbISOFloppy->isChecked())
1940 {
1941 Assert (!uuidISOFloppy.isNull());
1942 floppy.MountImage (uuidISOFloppy);
1943 }
1944 }
1945
1946 /* CD/DVD-ROM image */
1947 {
1948 CDVDDrive dvd = cmachine.GetDVDDrive();
1949 if (!bgDVD->isChecked())
1950 {
1951 dvd.SetPassthrough (false);
1952 dvd.Unmount();
1953 }
1954 else if (rbHostDVD->isChecked())
1955 {
1956 dvd.SetPassthrough (cbPassthrough->isChecked());
1957 int id = cbHostDVD->currentItem();
1958 Assert (id >= 0);
1959 if (id < (int) hostDVDs.count())
1960 dvd.CaptureHostDrive (hostDVDs [id]);
1961 /*
1962 * otherwise the selected drive is not yet available, leave it
1963 * as is
1964 */
1965 }
1966 else if (rbISODVD->isChecked())
1967 {
1968 dvd.SetPassthrough (false);
1969 Assert (!uuidISODVD.isNull());
1970 dvd.MountImage (uuidISODVD);
1971 }
1972 }
1973
1974 /* Clear the "GUI_FirstRun" extra data key in case if the boot order
1975 * and/or disk configuration were changed */
1976 if (mResetFirstRunFlag)
1977 cmachine.SetExtraData (GUI_FirstRun, QString::null);
1978
1979 /* audio */
1980 {
1981 CAudioAdapter audio = cmachine.GetAudioAdapter();
1982 audio.SetAudioDriver (vboxGlobal().toAudioDriverType (cbAudioDriver->currentText()));
1983 audio.SetEnabled (grbAudio->isChecked());
1984 AssertWrapperOk (audio);
1985 }
1986
1987 /* network */
1988 {
1989 for (int index = 0; index < tbwNetwork->count(); index++)
1990 {
1991 VBoxVMNetworkSettings *page =
1992 (VBoxVMNetworkSettings *) tbwNetwork->page (index);
1993 Assert (page);
1994 page->putBackToAdapter();
1995 }
1996 }
1997
1998 /* usb */
1999 {
2000 CUSBController ctl = cmachine.GetUSBController();
2001
2002 if (!ctl.isNull())
2003 {
2004 /* the USB controller may be unavailable (i.e. in VirtualBox OSE) */
2005
2006 ctl.SetEnabled (cbEnableUSBController->isChecked());
2007
2008 /*
2009 * first, remove all old filters (only if the list is changed,
2010 * not only individual properties of filters)
2011 */
2012 if (mUSBFilterListModified)
2013 for (ulong count = ctl.GetDeviceFilters().GetCount(); count; -- count)
2014 ctl.RemoveDeviceFilter (0);
2015
2016 /* then add all new filters */
2017 for (QListViewItem *item = lvUSBFilters->firstChild(); item;
2018 item = item->nextSibling())
2019 {
2020 USBListItem *uli = static_cast <USBListItem *> (item);
2021 VBoxUSBFilterSettings *settings =
2022 static_cast <VBoxUSBFilterSettings *>
2023 (wstUSBFilters->widget (uli->mId));
2024 Assert (settings);
2025
2026 COMResult res = settings->putBackToFilter();
2027 if (!res.isOk())
2028 return res;
2029
2030 CUSBDeviceFilter filter = settings->filter();
2031 filter.SetActive (uli->isOn());
2032
2033 if (mUSBFilterListModified)
2034 ctl.InsertDeviceFilter (~0, filter);
2035 }
2036 }
2037
2038 mUSBFilterListModified = false;
2039 }
2040
2041 /* vrdp */
2042 {
2043 CVRDPServer vrdp = cmachine.GetVRDPServer();
2044
2045 if (!vrdp.isNull())
2046 {
2047 /* VRDP may be unavailable (i.e. in VirtualBox OSE) */
2048 vrdp.SetEnabled (grbVRDP->isChecked());
2049 vrdp.SetPort (leVRDPPort->text().toULong());
2050 vrdp.SetAuthType (vboxGlobal().toVRDPAuthType (cbVRDPAuthType->currentText()));
2051 vrdp.SetAuthTimeout (leVRDPTimeout->text().toULong());
2052 }
2053 }
2054
2055 /* shared folders */
2056 {
2057 mSharedFolders->putBackToMachine();
2058 }
2059
2060 return COMResult();
2061}
2062
2063
2064void VBoxVMSettingsDlg::showImageManagerHDA() { showVDImageManager (&uuidHDA, cbHDA); }
2065void VBoxVMSettingsDlg::showImageManagerHDB() { showVDImageManager (&uuidHDB, cbHDB); }
2066void VBoxVMSettingsDlg::showImageManagerHDD() { showVDImageManager (&uuidHDD, cbHDD); }
2067void VBoxVMSettingsDlg::showImageManagerISODVD() { showVDImageManager (&uuidISODVD, cbISODVD); }
2068void VBoxVMSettingsDlg::showImageManagerISOFloppy() { showVDImageManager(&uuidISOFloppy, cbISOFloppy); }
2069
2070void VBoxVMSettingsDlg::showVDImageManager (QUuid *id, VBoxMediaComboBox *cbb, QLabel*)
2071{
2072 VBoxDefs::DiskType type = VBoxDefs::InvalidType;
2073 if (cbb == cbISODVD)
2074 type = VBoxDefs::CD;
2075 else if (cbb == cbISOFloppy)
2076 type = VBoxDefs::FD;
2077 else
2078 type = VBoxDefs::HD;
2079
2080 VBoxDiskImageManagerDlg dlg (this, "VBoxDiskImageManagerDlg",
2081 WType_Dialog | WShowModal);
2082 QUuid machineId = cmachine.GetId();
2083 dlg.setup (type, true, &machineId, true /* aRefresh */, cmachine);
2084 if (dlg.exec() == VBoxDiskImageManagerDlg::Accepted)
2085 {
2086 *id = dlg.getSelectedUuid();
2087 resetFirstRunFlag();
2088 }
2089 else
2090 {
2091 *id = cbb->getId();
2092 }
2093
2094 cbb->setCurrentItem (*id);
2095 cbb->setFocus();
2096
2097 /* revalidate pages with custom validation */
2098 wvalHDD->revalidate();
2099 wvalDVD->revalidate();
2100 wvalFloppy->revalidate();
2101}
2102
2103void VBoxVMSettingsDlg::addNetworkAdapter (const CNetworkAdapter &aAdapter)
2104{
2105 VBoxVMNetworkSettings *page = new VBoxVMNetworkSettings();
2106 page->loadList (mInterfaceList, mNoInterfaces);
2107 page->getFromAdapter (aAdapter);
2108 tbwNetwork->addTab (page, QString (tr ("Adapter %1", "network"))
2109 .arg (aAdapter.GetSlot()));
2110
2111 /* fix the tab order so that main dialog's buttons are always the last */
2112 setTabOrder (page->leTAPTerminate, buttonHelp);
2113 setTabOrder (buttonHelp, buttonOk);
2114 setTabOrder (buttonOk, buttonCancel);
2115
2116 /* setup validation */
2117 QIWidgetValidator *wval = new QIWidgetValidator (pageNetwork, this);
2118 connect (page->grbEnabled, SIGNAL (toggled (bool)), wval, SLOT (revalidate()));
2119 connect (page->cbNetworkAttachment, SIGNAL (activated (const QString &)),
2120 wval, SLOT (revalidate()));
2121 connect (wval, SIGNAL (validityChanged (const QIWidgetValidator *)),
2122 this, SLOT (enableOk (const QIWidgetValidator *)));
2123 connect (wval, SIGNAL (isValidRequested (QIWidgetValidator *)),
2124 this, SLOT (revalidate( QIWidgetValidator *)));
2125
2126 page->setValidator (wval);
2127 page->revalidate();
2128}
2129
2130void VBoxVMSettingsDlg::slRAM_valueChanged( int val )
2131{
2132 leRAM->setText( QString().setNum( val ) );
2133}
2134
2135void VBoxVMSettingsDlg::leRAM_textChanged( const QString &text )
2136{
2137 slRAM->setValue( text.toInt() );
2138}
2139
2140void VBoxVMSettingsDlg::slVRAM_valueChanged( int val )
2141{
2142 leVRAM->setText( QString().setNum( val ) );
2143}
2144
2145void VBoxVMSettingsDlg::leVRAM_textChanged( const QString &text )
2146{
2147 slVRAM->setValue( text.toInt() );
2148}
2149
2150void VBoxVMSettingsDlg::cbOS_activated (int item)
2151{
2152 Q_UNUSED (item);
2153/// @todo (dmik) remove?
2154// CGuestOSType type = vboxGlobal().vmGuestOSType (item);
2155// txRAMBest->setText (tr ("<qt>Best&nbsp;%1&nbsp;MB<qt>")
2156// .arg (type.GetRecommendedRAM()));
2157// txVRAMBest->setText (tr ("<qt>Best&nbsp;%1&nbsp;MB</qt>")
2158// .arg (type.GetRecommendedVRAM()));
2159 txRAMBest->setText (QString::null);
2160 txVRAMBest->setText (QString::null);
2161}
2162
2163void VBoxVMSettingsDlg::tbResetSavedStateFolder_clicked()
2164{
2165 /*
2166 * do this instead of le->setText (QString::null) to cause
2167 * isModified() return true
2168 */
2169 leSnapshotFolder->selectAll();
2170 leSnapshotFolder->del();
2171}
2172
2173void VBoxVMSettingsDlg::tbSelectSavedStateFolder_clicked()
2174{
2175 QString settingsFolder = VBoxGlobal::getFirstExistingDir (leSnapshotFolder->text());
2176 if (settingsFolder.isNull())
2177 settingsFolder = QFileInfo (cmachine.GetSettingsFilePath()).dirPath (true);
2178
2179 QString folder = vboxGlobal().getExistingDirectory (settingsFolder, this);
2180 if (folder.isNull())
2181 return;
2182
2183 folder = QDir::convertSeparators (folder);
2184 /* remove trailing slash if any */
2185 folder.remove (QRegExp ("[\\\\/]$"));
2186
2187 /*
2188 * do this instead of le->setText (folder) to cause
2189 * isModified() return true
2190 */
2191 leSnapshotFolder->selectAll();
2192 leSnapshotFolder->insert (folder);
2193}
2194
2195// USB Filter stuff
2196////////////////////////////////////////////////////////////////////////////////
2197
2198void VBoxVMSettingsDlg::addUSBFilter (const CUSBDeviceFilter &aFilter, bool isNew)
2199{
2200 QListViewItem *currentItem = isNew
2201 ? lvUSBFilters->currentItem()
2202 : lvUSBFilters->lastItem();
2203
2204 VBoxUSBFilterSettings *settings = new VBoxUSBFilterSettings (wstUSBFilters);
2205 settings->setup (VBoxUSBFilterSettings::MachineType);
2206 settings->getFromFilter (aFilter);
2207
2208 USBListItem *item = new USBListItem (lvUSBFilters, currentItem);
2209 item->setOn (aFilter.GetActive());
2210 item->setText (lvUSBFilters_Name, aFilter.GetName());
2211
2212 item->mId = wstUSBFilters->addWidget (settings);
2213
2214 /* fix the tab order so that main dialog's buttons are always the last */
2215 setTabOrder (settings->focusProxy(), buttonHelp);
2216 setTabOrder (buttonHelp, buttonOk);
2217 setTabOrder (buttonOk, buttonCancel);
2218
2219 if (isNew)
2220 {
2221 lvUSBFilters->setSelected (item, true);
2222 lvUSBFilters_currentChanged (item);
2223 settings->leUSBFilterName->setFocus();
2224 }
2225
2226 connect (settings->leUSBFilterName, SIGNAL (textChanged (const QString &)),
2227 this, SLOT (lvUSBFilters_setCurrentText (const QString &)));
2228
2229 /* setup validation */
2230
2231 QIWidgetValidator *wval = new QIWidgetValidator (settings, settings);
2232 connect (wval, SIGNAL (validityChanged (const QIWidgetValidator *)),
2233 this, SLOT (enableOk (const QIWidgetValidator *)));
2234
2235 wval->revalidate();
2236}
2237
2238void VBoxVMSettingsDlg::lvUSBFilters_currentChanged (QListViewItem *item)
2239{
2240 if (item && lvUSBFilters->selectedItem() != item)
2241 lvUSBFilters->setSelected (item, true);
2242
2243 tbRemoveUSBFilter->setEnabled (!!item);
2244
2245 tbUSBFilterUp->setEnabled (!!item && item->itemAbove());
2246 tbUSBFilterDown->setEnabled (!!item && item->itemBelow());
2247
2248 if (item)
2249 {
2250 USBListItem *uli = static_cast <USBListItem *> (item);
2251 wstUSBFilters->raiseWidget (uli->mId);
2252 }
2253 else
2254 {
2255 /* raise the disabled widget */
2256 wstUSBFilters->raiseWidget (0);
2257 }
2258}
2259
2260void VBoxVMSettingsDlg::lvUSBFilters_setCurrentText (const QString &aText)
2261{
2262 QListViewItem *item = lvUSBFilters->currentItem();
2263 Assert (item);
2264
2265 item->setText (lvUSBFilters_Name, aText);
2266}
2267
2268void VBoxVMSettingsDlg::tbAddUSBFilter_clicked()
2269{
2270 /* search for the max available filter index */
2271 int maxFilterIndex = 0;
2272 QString usbFilterName = tr ("New Filter %1", "usb");
2273 QRegExp regExp (QString ("^") + usbFilterName.arg ("([0-9]+)") + QString ("$"));
2274 QListViewItemIterator iterator (lvUSBFilters);
2275 while (*iterator)
2276 {
2277 QString filterName = (*iterator)->text (lvUSBFilters_Name);
2278 int pos = regExp.search (filterName);
2279 if (pos != -1)
2280 maxFilterIndex = regExp.cap (1).toInt() > maxFilterIndex ?
2281 regExp.cap (1).toInt() : maxFilterIndex;
2282 ++ iterator;
2283 }
2284
2285 /* creating new usb filter */
2286 CUSBDeviceFilter filter = cmachine.GetUSBController()
2287 .CreateDeviceFilter (usbFilterName.arg (maxFilterIndex + 1));
2288
2289 filter.SetActive (true);
2290 addUSBFilter (filter, true /* isNew */);
2291
2292 mUSBFilterListModified = true;
2293}
2294
2295void VBoxVMSettingsDlg::tbAddUSBFilterFrom_clicked()
2296{
2297 usbDevicesMenu->exec (QCursor::pos());
2298}
2299
2300void VBoxVMSettingsDlg::menuAddUSBFilterFrom_activated (int aIndex)
2301{
2302 CUSBDevice usb = usbDevicesMenu->getUSB (aIndex);
2303 /* if null then some other item but a USB device is selected */
2304 if (usb.isNull())
2305 return;
2306
2307 CUSBDeviceFilter filter = cmachine.GetUSBController()
2308 .CreateDeviceFilter (vboxGlobal().details (usb));
2309
2310 filter.SetVendorId (QString().sprintf ("%04hX", usb.GetVendorId()));
2311 filter.SetProductId (QString().sprintf ("%04hX", usb.GetProductId()));
2312 filter.SetRevision (QString().sprintf ("%04hX", usb.GetRevision()));
2313 /* The port property depends on the host computer rather than on the USB
2314 * device itself; for this reason only a few people will want to use it in
2315 * the filter since the same device plugged into a different socket will
2316 * not match the filter in this case. */
2317#if 0
2318 /// @todo set it anyway if Alt is currently pressed
2319 filter.SetPort (QString().sprintf ("%04hX", usb.GetPort()));
2320#endif
2321 filter.SetManufacturer (usb.GetManufacturer());
2322 filter.SetProduct (usb.GetProduct());
2323 filter.SetSerialNumber (usb.GetSerialNumber());
2324 filter.SetRemote (usb.GetRemote() ? "yes" : "no");
2325
2326 filter.SetActive (true);
2327 addUSBFilter (filter, true /* isNew */);
2328
2329 mUSBFilterListModified = true;
2330}
2331
2332void VBoxVMSettingsDlg::tbRemoveUSBFilter_clicked()
2333{
2334 QListViewItem *item = lvUSBFilters->currentItem();
2335 Assert (item);
2336
2337 USBListItem *uli = static_cast <USBListItem *> (item);
2338 QWidget *settings = wstUSBFilters->widget (uli->mId);
2339 Assert (settings);
2340 wstUSBFilters->removeWidget (settings);
2341 delete settings;
2342
2343 delete item;
2344
2345 lvUSBFilters->setSelected (lvUSBFilters->currentItem(), true);
2346 mUSBFilterListModified = true;
2347}
2348
2349void VBoxVMSettingsDlg::tbUSBFilterUp_clicked()
2350{
2351 QListViewItem *item = lvUSBFilters->currentItem();
2352 Assert (item);
2353
2354 QListViewItem *itemAbove = item->itemAbove();
2355 Assert (itemAbove);
2356 itemAbove = itemAbove->itemAbove();
2357
2358 if (!itemAbove)
2359 {
2360 /* overcome Qt stupidity */
2361 item->itemAbove()->moveItem (item);
2362 }
2363 else
2364 item->moveItem (itemAbove);
2365
2366 lvUSBFilters_currentChanged (item);
2367 mUSBFilterListModified = true;
2368}
2369
2370void VBoxVMSettingsDlg::tbUSBFilterDown_clicked()
2371{
2372 QListViewItem *item = lvUSBFilters->currentItem();
2373 Assert (item);
2374
2375 QListViewItem *itemBelow = item->itemBelow();
2376 Assert (itemBelow);
2377
2378 item->moveItem (itemBelow);
2379
2380 lvUSBFilters_currentChanged (item);
2381 mUSBFilterListModified = true;
2382}
2383
2384#include "VBoxVMSettingsDlg.ui.moc"
2385
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