QIMainDialog.cpp revision d34409ad02ea0d28e08a6c4b089a412fdb3b4c9c
/* $Id$ */
/** @file
* VBox Qt GUI - VirtualBox Qt extensions: QIMainDialog class implementation.
*/
/*
* Copyright (C) 2008-2012 Oracle Corporation
*
* This file is part of VirtualBox Open Source Edition (OSE), as
* available from http://www.virtualbox.org. This file is free software;
* you can redistribute it and/or modify it under the terms of the GNU
* General Public License (GPL) as published by the Free Software
* Foundation, in version 2 as it comes in the "COPYING" file of the
* VirtualBox OSE distribution. VirtualBox OSE is distributed in the
* hope that it will be useful, but WITHOUT ANY WARRANTY of any kind.
*/
#include "QIMainDialog.h"
#include "VBoxUtils.h"
#include "VBoxGlobal.h"
#include <iprt/assert.h>
/* Qt includes */
#include <QProcess>
#include <QEventLoop>
#include <QApplication>
#include <QDir>
#include <QUrl>
#include <QMenu>
#include <QSizeGrip>
#include <QPushButton>
#include <QDialogButtonBox>
QIMainDialog::QIMainDialog (QWidget *aParent /* = 0 */,
Qt::WindowFlags aFlags /* = Qt::Dialog */)
: QMainWindow (aParent, aFlags)
, mRescode (QDialog::Rejected)
, mPolished (false)
, mIsAutoCentering (true)
, mCenterWidget (aParent)
{
qApp->installEventFilter (this);
}
QDialog::DialogCode QIMainDialog::exec()
{
/* Check for the recursive run: */
AssertMsg(!mEventLoop, ("QIMainDialog::exec() is called recursively!\n"));
/* Reset the result code: */
setResult(QDialog::Rejected);
/* Tune some attributes: */
bool fDeleteOnClose = testAttribute(Qt::WA_DeleteOnClose); NOREF(fDeleteOnClose);
AssertMsg(!fDeleteOnClose, ("QIMainDialog is NOT supposed to be run in 'delete-on-close' mode!"));
setAttribute(Qt::WA_DeleteOnClose, false);
bool fWasShowModal = testAttribute(Qt::WA_ShowModal);
setAttribute(Qt::WA_ShowModal, true);
/* Create a local event-loop: */
QEventLoop eventLoop;
mEventLoop = &eventLoop;
/* Show the window: */
show();
/* A guard to ourself for the case we destroy ourself: */
QPointer<QIMainDialog> guard = this;
/* Start the event-loop: */
eventLoop.exec();
/* Check if dialog is still valid: */
if (guard.isNull())
return QDialog::Rejected;
mEventLoop = 0;
/* Prepare result: */
QDialog::DialogCode res = result();
/* Restore old show-modal attribute: */
setAttribute(Qt::WA_ShowModal, fWasShowModal);
/* Return the final result: */
return res;
}
QDialog::DialogCode QIMainDialog::result() const
{
return mRescode;
}
void QIMainDialog::setSizeGripEnabled (bool aEnabled)
{
if (!mSizeGrip && aEnabled)
{
mSizeGrip = new QSizeGrip (this);
mSizeGrip->resize (mSizeGrip->sizeHint());
mSizeGrip->show();
}
else if (mSizeGrip && !aEnabled)
delete mSizeGrip;
}
bool QIMainDialog::isSizeGripEnabled() const
{
return mSizeGrip;
}
void QIMainDialog::setDefaultButton (QPushButton* aButton)
{
mDefaultButton = aButton;
}
QPushButton* QIMainDialog::defaultButton() const
{
return mDefaultButton;
}
void QIMainDialog::setAutoCenteringEnabled(bool fIsAutoCentering)
{
mIsAutoCentering = fIsAutoCentering;
}
void QIMainDialog::setVisible (bool aVisible)
{
QMainWindow::setVisible (aVisible);
/* Exit from the event loop if there is any and we are changing our state
* from visible to invisible. */
if (mEventLoop && !aVisible)
mEventLoop->exit();
}
bool QIMainDialog::event (QEvent *aEvent)
{
switch (aEvent->type())
{
case QEvent::Polish:
{
/* Initially search for the default button. */
mDefaultButton = searchDefaultButton();
break;
}
default:
break;
}
return QMainWindow::event (aEvent);
}
void QIMainDialog::showEvent (QShowEvent *aEvent)
{
QMainWindow::showEvent (aEvent);
/* Polishing border */
if (mPolished)
return;
mPolished = true;
/* Explicit widget centering relatively to it's centering
* widget if any or desktop if centering widget is missed. */
if (mIsAutoCentering)
VBoxGlobal::centerWidget (this, mCenterWidget, false);
}
void QIMainDialog::resizeEvent (QResizeEvent *aEvent)
{
QMainWindow::resizeEvent (aEvent);
/* Adjust the size-grip location for the current resize event */
if (mSizeGrip)
{
if (isRightToLeft())
mSizeGrip->move (rect().bottomLeft() - mSizeGrip->rect().bottomLeft());
else
mSizeGrip->move (rect().bottomRight() - mSizeGrip->rect().bottomRight());
aEvent->accept();
}
}
void QIMainDialog::keyPressEvent(QKeyEvent *pEvent)
{
/* Make sure that we only proceed if no
* popup or other modal widgets are open. */
if (qApp->activePopupWidget() ||
(qApp->activeModalWidget() && qApp->activeModalWidget() != this))
{
/* Call to base-class: */
return QMainWindow::keyPressEvent(pEvent);
}
/* Special handling for some keys: */
switch (pEvent->key())
{
/* Special handling for Escape key: */
case Qt::Key_Escape:
{
if (pEvent->modifiers() == Qt::NoModifier)
{
reject();
return;
}
break;
}
#ifdef Q_WS_MAC
/* Special handling for Period key: */
case Qt::Key_Period:
{
if (pEvent->modifiers() == Qt::ControlModifier)
{
reject();
return;
}
break;
}
#endif /* Q_WS_MAC */
/* Special handling for Return/Enter key: */
case Qt::Key_Return:
case Qt::Key_Enter:
{
if (((pEvent->modifiers() == Qt::NoModifier) && (pEvent->key() == Qt::Key_Return)) ||
((pEvent->modifiers() & Qt::KeypadModifier) && (pEvent->key() == Qt::Key_Enter)))
{
if (QPushButton *pCurrentDefault = searchDefaultButton())
{
pCurrentDefault->animateClick();
return;
}
}
break;
}
/* Default handling for others: */
default: break;
}
/* Call to base-class: */
return QMainWindow::keyPressEvent(pEvent);
}
bool QIMainDialog::eventFilter (QObject *aObject, QEvent *aEvent)
{
if (!isActiveWindow())
return QMainWindow::eventFilter (aObject, aEvent);
if (qobject_cast<QWidget*> (aObject) &&
qobject_cast<QWidget*> (aObject)->window() != this)
return QMainWindow::eventFilter (aObject, aEvent);
switch (aEvent->type())
{
/* Auto-default button focus-in processor used to move the "default"
* button property into the currently focused button. */
case QEvent::FocusIn:
{
if (qobject_cast<QPushButton*> (aObject) &&
(aObject->parent() == centralWidget() ||
qobject_cast<QDialogButtonBox*> (aObject->parent())))
{
qobject_cast<QPushButton*> (aObject)->setDefault (aObject != mDefaultButton);
if (mDefaultButton)
mDefaultButton->setDefault (aObject == mDefaultButton);
}
break;
}
/* Auto-default button focus-out processor used to remove the "default"
* button property from the previously focused button. */
case QEvent::FocusOut:
{
if (qobject_cast<QPushButton*> (aObject) &&
(aObject->parent() == centralWidget() ||
qobject_cast<QDialogButtonBox*> (aObject->parent())))
{
if (mDefaultButton)
mDefaultButton->setDefault (aObject != mDefaultButton);
qobject_cast<QPushButton*> (aObject)->setDefault (aObject == mDefaultButton);
}
break;
}
default:
break;
}
return QMainWindow::eventFilter (aObject, aEvent);
}
QPushButton* QIMainDialog::searchDefaultButton() const
{
/* Search for the first default button in the dialog. */
QList<QPushButton*> list = qFindChildren<QPushButton*> (this);
foreach (QPushButton *button, list)
if (button->isDefault() &&
(button->parent() == centralWidget() ||
qobject_cast<QDialogButtonBox*> (button->parent())))
return button;
return NULL;
}
void QIMainDialog::accept()
{
done (QDialog::Accepted);
}
void QIMainDialog::reject()
{
done (QDialog::Rejected);
}
void QIMainDialog::done (QDialog::DialogCode aResult)
{
/* Set the final result */
setResult (aResult);
/* Hide this window */
hide();
/* And close the window */
close();
}
void QIMainDialog::setResult (QDialog::DialogCode aRescode)
{
mRescode = aRescode;
}