qskinny/src/dialogs/QskDialogSubWindow.cpp

503 lines
12 KiB
C++

/******************************************************************************
* QSkinny - Copyright (C) 2016 Uwe Rathmann
* This file may be used under the terms of the QSkinny License, Version 1.0
*****************************************************************************/
#include "QskDialogSubWindow.h"
#include "QskDialogButtonBox.h"
#include "QskLayoutConstraint.h"
#include "QskPushButton.h"
#include "QskQuick.h"
#include <qeventloop.h>
#include <qquickwindow.h>
#include <qpointer.h>
static inline void qskSetRejectOnClose( QskDialogSubWindow* subWindow, bool on )
{
if ( on )
{
QObject::connect( subWindow, &QskPopup::closed,
subWindow, &QskDialogSubWindow::reject );
}
else
{
QObject::disconnect( subWindow, &QskPopup::closed,
subWindow, &QskDialogSubWindow::reject );
}
}
static inline qreal qskConstrainedValue( QskLayoutConstraint::Type type,
const QskControl* control, qreal widthOrHeight )
{
auto subWindow = static_cast< const QskDialogSubWindow* >( control );
if ( type == QskLayoutConstraint::WidthForHeight )
{
qreal width = -1.0;
qreal height = widthOrHeight;
if ( auto buttonBox = subWindow->buttonBox() )
{
if ( buttonBox->isVisibleTo( subWindow ) )
{
const auto hint = buttonBox->sizeHint();
width = hint.width();
height -= hint.height();
}
}
if ( auto contentItem = qskControlCast( subWindow->contentItem() ) )
{
if ( contentItem->isVisibleTo( subWindow ) )
{
const auto m = subWindow->contentPadding();
height -= m.top() + m.bottom();
qreal w = contentItem->widthForHeight( height );
if ( w >= 0 )
{
w += m.left() + m.right();
width = qMax( width, w );
}
}
}
return width;
}
else
{
qreal width = widthOrHeight;
qreal height = -1.0;
if ( auto buttonBox = subWindow->buttonBox() )
{
if ( buttonBox->isVisibleTo( subWindow ) )
height += buttonBox->sizeHint().height();
}
if ( auto contentItem = qskControlCast( subWindow->contentItem() ) )
{
if ( qskIsVisibleTo( contentItem, subWindow ) )
{
const auto& m = subWindow->contentPadding();
width -= m.left() + m.right();
const qreal h = contentItem->heightForWidth( width );
if ( h >= 0 )
height += h + m.top() + m.bottom();
}
}
return height;
}
}
class QskDialogSubWindow::PrivateData
{
public:
QskDialog::Actions actions = QskDialog::NoAction;
QPointer< QQuickItem > contentItem;
QskDialogButtonBox* buttonBox = nullptr;
QMarginsF contentPadding;
QskDialog::DialogCode result = QskDialog::Rejected;
};
QskDialogSubWindow::QskDialogSubWindow( QQuickItem* parent )
: Inherited( parent )
, m_data( new PrivateData() )
{
setPolishOnResize( true );
qskSetRejectOnClose( this, true );
}
QskDialogSubWindow::~QskDialogSubWindow()
{
}
void QskDialogSubWindow::addDialogAction( QskDialog::Action action )
{
if ( action != QskDialog::NoAction )
{
if ( m_data->buttonBox == nullptr )
initButtonBox();
if ( m_data->buttonBox )
m_data->buttonBox->addAction( action );
}
}
void QskDialogSubWindow::addDialogButton(
QskPushButton* button, QskDialog::ActionRole actionRole )
{
if ( button )
{
if ( m_data->buttonBox == nullptr )
initButtonBox();
if ( m_data->buttonBox )
m_data->buttonBox->addButton( button, actionRole );
}
}
void QskDialogSubWindow::setDialogActions( QskDialog::Actions actions )
{
if ( m_data->actions == actions )
return;
m_data->actions = actions;
if ( actions == QskDialog::NoAction )
{
if ( m_data->buttonBox->parent() == this )
{
delete m_data->buttonBox;
}
else
{
m_data->buttonBox->setParentItem( nullptr );
disconnect( m_data->buttonBox, &QskDialogButtonBox::accepted,
this, &QskDialogSubWindow::accept );
disconnect( m_data->buttonBox, &QskDialogButtonBox::rejected,
this, &QskDialogSubWindow::reject );
}
m_data->buttonBox = nullptr;
}
else
{
if ( m_data->buttonBox == nullptr )
{
initButtonBox();
}
if ( m_data->buttonBox )
m_data->buttonBox->setActions( actions );
}
resetImplicitSize();
polish();
}
QskDialog::Actions QskDialogSubWindow::dialogActions() const
{
if ( m_data->buttonBox )
return m_data->buttonBox->actions();
return QskDialog::NoAction;
}
void QskDialogSubWindow::setContentItem( QQuickItem* item )
{
if ( item == m_data->contentItem )
return;
if ( m_data->contentItem )
{
if ( m_data->contentItem->parent() == this )
delete m_data->contentItem;
else
m_data->contentItem->setParentItem( nullptr );
}
m_data->contentItem = item;
if ( item )
{
item->setParentItem( this );
if ( item->parent() == nullptr )
item->setParent( this );
}
resetImplicitSize();
polish();
}
QQuickItem* QskDialogSubWindow::contentItem() const
{
return m_data->contentItem;
}
void QskDialogSubWindow::setContentPadding( const QMarginsF& padding )
{
// should be a skin hint ???
if ( m_data->contentPadding != padding )
{
m_data->contentPadding = padding;
resetImplicitSize();
polish();
}
}
QMarginsF QskDialogSubWindow::contentPadding() const
{
return m_data->contentPadding;
}
void QskDialogSubWindow::setDefaultDialogAction( QskDialog::Action action )
{
QskPushButton* button = nullptr;
if ( m_data->buttonBox )
button = m_data->buttonBox->button( action );
setDefaultButton( button );
}
void QskDialogSubWindow::setDefaultButton( QskPushButton* button )
{
if ( !qskIsAncestorOf( m_data->buttonBox, button ) )
{
#if defined( QT_DEBUG )
qWarning( "Only buttons of the QskDialogButtonBox can be the default button." );
#endif
return;
}
m_data->buttonBox->setDefaultButton( button );
}
QskPushButton* QskDialogSubWindow::defaultButton() const
{
if ( m_data->buttonBox == nullptr )
return nullptr;
return m_data->buttonBox->defaultButton();
}
QskDialogButtonBox* QskDialogSubWindow::buttonBox()
{
return m_data->buttonBox;
}
const QskDialogButtonBox* QskDialogSubWindow::buttonBox() const
{
return m_data->buttonBox;
}
QskDialog::Action QskDialogSubWindow::clickedAction() const
{
if ( m_data->buttonBox )
return m_data->buttonBox->clickedAction();
return QskDialog::NoAction;
}
void QskDialogSubWindow::setResult( QskDialog::DialogCode result )
{
m_data->result = result;
}
QskDialog::DialogCode QskDialogSubWindow::result() const
{
return m_data->result;
}
QskDialog::DialogCode QskDialogSubWindow::exec()
{
if ( priority() > 0 )
{
qWarning( "illegal call of QskDialogSubWindow::exec "
"for a subwindow with non default priority." );
}
open();
//check for window after call to open(), because maybe a popupmanager assigns a window on open.
if ( window() == nullptr )
{
qWarning( "trying to exec a subwindow without window." );
return QskDialog::Rejected;
}
if ( QQuickItem* mouseGrabber = window()->mouseGrabberItem() )
{
// when being called from QQuickWindow::mouseReleaseEvent
// the mouse grabber has not yet been released.
if( !qskIsAncestorOf( this, mouseGrabber ) )
mouseGrabber->ungrabMouse();
}
QEventLoop eventLoop;
connect( this, &QskDialogSubWindow::finished, &eventLoop, &QEventLoop::quit );
( void ) eventLoop.exec( QEventLoop::DialogExec );
return m_data->result;
}
void QskDialogSubWindow::done( QskDialog::DialogCode result )
{
m_data->result = result;
if ( !isOpen() )
return;
qskSetRejectOnClose( this, false );
close();
Q_EMIT finished( result );
if ( result == QskDialog::Accepted )
Q_EMIT accepted();
else
Q_EMIT rejected();
}
void QskDialogSubWindow::accept()
{
done( QskDialog::Accepted );
}
void QskDialogSubWindow::reject()
{
done( QskDialog::Rejected );
}
void QskDialogSubWindow::keyPressEvent( QKeyEvent* event )
{
if ( m_data->buttonBox &&
QskDialogButtonBox::isDefaultButtonKeyEvent( event ) )
{
auto button = m_data->buttonBox->defaultButton();
if ( button && button->isEnabled() )
button->click();
}
if ( event->matches( QKeySequence::Cancel ) )
{
// using shortcuts instead ???
reject();
return;
}
Inherited::keyPressEvent( event );
}
QskDialogButtonBox* QskDialogSubWindow::createButtonBox()
{
return new QskDialogButtonBox();
}
void QskDialogSubWindow::initButtonBox()
{
m_data->buttonBox = createButtonBox();
if ( m_data->buttonBox )
{
m_data->buttonBox->setParentItem( this );
if ( m_data->buttonBox->parent() == nullptr )
m_data->buttonBox->setParent( this );
connect( m_data->buttonBox, &QskDialogButtonBox::accepted,
this, &QskDialogSubWindow::accept, Qt::UniqueConnection );
connect( m_data->buttonBox, &QskDialogButtonBox::rejected,
this, &QskDialogSubWindow::reject, Qt::UniqueConnection );
}
}
void QskDialogSubWindow::aboutToShow()
{
if ( size().isEmpty() )
{
// setting an initial size from the hint, centered inside the window
const qreal cx = 0.5 * parentItem()->width();
const qreal cy = 0.5 * parentItem()->height();
QRectF rect;
rect.setSize( sizeHint() );
rect.moveCenter( QPointF( cx, cy ) );
setGeometry( rect );
}
Inherited::aboutToShow();
}
void QskDialogSubWindow::updateLayout()
{
Inherited::updateLayout();
auto rect = layoutRect();
if ( m_data->buttonBox && m_data->buttonBox->isVisibleTo( this ) )
{
const auto h = m_data->buttonBox->sizeHint().height();
rect.setBottom( rect.bottom() - h );
m_data->buttonBox->setGeometry( rect.x(), rect.bottom(), rect.width(), h );
}
if ( m_data->contentItem )
{
rect = rect.marginsRemoved( m_data->contentPadding );
qskSetItemGeometry( m_data->contentItem, rect );
}
}
qreal QskDialogSubWindow::heightForWidth( qreal width ) const
{
return QskLayoutConstraint::constrainedMetric(
QskLayoutConstraint::HeightForWidth, this, width, qskConstrainedValue );
}
qreal QskDialogSubWindow::widthForHeight( qreal height ) const
{
return QskLayoutConstraint::constrainedMetric(
QskLayoutConstraint::WidthForHeight, this, height, qskConstrainedValue );
}
QSizeF QskDialogSubWindow::contentsSizeHint() const
{
qreal w = -1;
qreal h = -1;
if ( m_data->buttonBox && m_data->buttonBox->isVisibleTo( this ) )
{
const auto hint = m_data->buttonBox->sizeHint();
w = hint.width();
h = hint.height();
}
if ( auto* control = qskControlCast( m_data->contentItem ) )
{
const auto hint = control->sizeHint();
const auto& m = m_data->contentPadding;
if ( hint.width() >= 0 )
w = qMax( w, hint.width() + m.left() + m.right() );
if ( hint.height() >= 0 )
h += hint.height() + m.top() + m.bottom();
}
const auto innerSize = layoutRect().size();
const auto outerSize = size();
w += outerSize.width() - innerSize.width();
h += outerSize.height() - innerSize.height();
return QSizeF( w, h );
}
#include "moc_QskDialogSubWindow.cpp"