123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419 |
- /*=========================================================================
- Library: CTK
- Copyright (c) Kitware Inc.
- Licensed under the Apache License, Version 2.0 (the "License");
- you may not use this file except in compliance with the License.
- You may obtain a copy of the License at
- http://www.apache.org/licenses/LICENSE-2.0.txt
- Unless required by applicable law or agreed to in writing, software
- distributed under the License is distributed on an "AS IS" BASIS,
- WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
- See the License for the specific language governing permissions and
- limitations under the License.
- =========================================================================*/
- // Qt includes
- #include <QWidget>
- #include <QList>
- #include <QPushButton>
- #include <QBoxLayout>
- #include <QDebug>
- #include <QPointer>
- #include <QStyle>
- // CTK includes
- #include "ctkWorkflowButtonBoxWidget.h"
- #include "ctkWorkflowStep.h"
- #include "ctkWorkflowWidgetStep.h"
- #include "ctkWorkflow.h"
- // STD includes
- #include <iostream>
- //-----------------------------------------------------------------------------
- class ctkWorkflowButtonBoxWidgetPrivate
- {
- Q_DECLARE_PUBLIC(ctkWorkflowButtonBoxWidget);
- protected:
- ctkWorkflowButtonBoxWidget* const q_ptr;
- public:
- ctkWorkflowButtonBoxWidgetPrivate(ctkWorkflowButtonBoxWidget& object);
- ~ctkWorkflowButtonBoxWidgetPrivate(){}
- void setupUi(QWidget * newParent);
- QPointer<ctkWorkflow> Workflow;
- // Convenient typedefs
- typedef QMap<QPushButton*, ctkWorkflowStep*> ButtonToStepMapType;
- // The buttons on the widget (maintain maps associating each forward/goTo button with its step)
- QPushButton* BackButton;
- QString BackButtonDefaultText;
- QPushButton* NextButton;
- QString NextButtonDefaultText;
- ButtonToStepMapType GoToButtonToStepMap;
- // Direction for layout (for use with QBoxLayout only)
- QBoxLayout::Direction Direction;
- bool HideInvalidButtons;
- // Functions to add/remove buttons
- void updateBackButton(ctkWorkflowStep* currentStep);
- void updateNextButton(ctkWorkflowStep* currentStep);
- void updateGoToButtons(ctkWorkflowStep* currentStep);
- };
- //-----------------------------------------------------------------------------
- // ctkWorkflowButtonBoxWidgetPrivate methods
- //-----------------------------------------------------------------------------
- ctkWorkflowButtonBoxWidgetPrivate::ctkWorkflowButtonBoxWidgetPrivate(ctkWorkflowButtonBoxWidget& object)
- :q_ptr(&object)
- {
- this->BackButton = 0;
- this->BackButtonDefaultText = "Back";
- this->NextButton = 0;
- this->NextButtonDefaultText = "Next";
- this->Direction = QBoxLayout::LeftToRight;
- this->HideInvalidButtons = false;
- }
- //-----------------------------------------------------------------------------
- void ctkWorkflowButtonBoxWidgetPrivate::setupUi(QWidget * newParent)
- {
- QBoxLayout * boxLayout = new QBoxLayout(this->Direction, newParent);
- boxLayout->setSpacing(0);
- boxLayout->setContentsMargins(0, 0, 0, 0);
- newParent->setLayout(boxLayout);
- // Retrieve standard icons
- QIcon backIcon = newParent->style()->standardIcon(QStyle::SP_ArrowLeft);
- QIcon nextIcon = newParent->style()->standardIcon(QStyle::SP_ArrowRight);
- // Setup the buttons
- this->BackButton = new QPushButton(backIcon, this->BackButtonDefaultText, newParent);
- newParent->layout()->addWidget(this->BackButton);
- this->NextButton = new QPushButton(nextIcon, this->NextButtonDefaultText, newParent);
- this->NextButton->setLayoutDirection(Qt::RightToLeft);
- newParent->layout()->addWidget(this->NextButton);
- }
- //-----------------------------------------------------------------------------
- void ctkWorkflowButtonBoxWidgetPrivate::updateBackButton(ctkWorkflowStep* currentStep)
- {
- #ifndef QT_NO_DEBUG
- Q_Q(ctkWorkflowButtonBoxWidget);
- Q_ASSERT(this->Workflow);
- Q_ASSERT(q->layout());
- #endif
- ctkWorkflowWidgetStep* step = dynamic_cast<ctkWorkflowWidgetStep*>(currentStep);
- // Set the back button text
- QString backButtonText = this->BackButtonDefaultText;
- if (step && !step->backButtonText().isEmpty())
- {
- backButtonText = step->backButtonText();
- }
- this->BackButton->setText(backButtonText);
- // Enable and show the back button if we can go backward
- if (currentStep && this->Workflow->canGoBackward(currentStep))
- {
- this->BackButton->setEnabled(true);
- this->BackButton->show();
- // Apply the buttonBox hints if possible
- if (step)
- {
- this->BackButton->setDisabled(
- step->buttonBoxHints() & ctkWorkflowWidgetStep::BackButtonDisabled);
- this->BackButton->setHidden(
- step->buttonBoxHints() & ctkWorkflowWidgetStep::BackButtonHidden);
- }
- }
- // Disable the back button if we can't go backward, and optionally hide it
- else
- {
- this->BackButton->setEnabled(false);
- this->HideInvalidButtons ? this->BackButton->hide() : this->BackButton->show();
- }
- }
- //-----------------------------------------------------------------------------
- // This will change for branching workflows, to look more like updateGoToButtons()
- void ctkWorkflowButtonBoxWidgetPrivate::updateNextButton(ctkWorkflowStep* currentStep)
- {
- #ifndef QT_NO_DEBUG
- Q_Q(ctkWorkflowButtonBoxWidget);
- Q_ASSERT(this->Workflow);
- Q_ASSERT(q->layout());
- #endif
- ctkWorkflowWidgetStep* step = dynamic_cast<ctkWorkflowWidgetStep*>(currentStep);
- // Set the next button text
- QString nextButtonText = this->NextButtonDefaultText;
- if (step && !step->nextButtonText().isEmpty())
- {
- nextButtonText = step->nextButtonText();
- }
- this->NextButton->setText(nextButtonText);
- // Enable and show the next button if we can go forward
- if (currentStep && this->Workflow->canGoForward(currentStep))
- {
- this->NextButton->setEnabled(true);
- this->NextButton->show();
- // Apply the buttonBox hints if possible
- if (step)
- {
- this->NextButton->setDisabled(
- step->buttonBoxHints() & ctkWorkflowWidgetStep::NextButtonDisabled);
- this->NextButton->setHidden(
- step->buttonBoxHints() & ctkWorkflowWidgetStep::NextButtonHidden);
- }
- }
- // Disable the next button if we can't go forward, and optionally hide it
- else
- {
- this->NextButton->setEnabled(false);
- this->HideInvalidButtons ? this->NextButton->hide() : this->NextButton->show();
- }
- }
- //-----------------------------------------------------------------------------
- void ctkWorkflowButtonBoxWidgetPrivate::updateGoToButtons(ctkWorkflowStep* currentStep)
- {
- Q_Q(ctkWorkflowButtonBoxWidget);
- Q_ASSERT(this->Workflow);
- Q_ASSERT(q->layout());
- // Change the buttons only if the set of steps to have goTo buttons is either empty or has changed
- QSet<ctkWorkflowStep*> goToStepsToHaveButtons = QSet<ctkWorkflowStep*>::fromList(this->Workflow->finishSteps());
- QSet<ctkWorkflowStep*> goToStepsThatHaveButtons = QSet<ctkWorkflowStep*>::fromList(this->GoToButtonToStepMap.values());
- // Remove the buttons if the set of steps to have goTo buttons has changed
- if (!this->GoToButtonToStepMap.isEmpty() && goToStepsThatHaveButtons != goToStepsToHaveButtons)
- {
- foreach (QPushButton* goToButton, this->GoToButtonToStepMap.keys())
- {
- q->layout()->removeWidget(goToButton);
- delete goToButton;
- }
- this->GoToButtonToStepMap.clear();
- }
- // Create the buttons, either for the first time or after removing them above
- if (this->GoToButtonToStepMap.isEmpty())
- {
- // Create a button for each of the goToSteps
- foreach (ctkWorkflowStep* step, goToStepsToHaveButtons)
- {
- // TODO shouldn't have id here
- QPushButton* goToButton = new QPushButton(step->id());
- q->layout()->addWidget(goToButton);
- QObject::connect(goToButton, SIGNAL(clicked()), q, SLOT(prepareGoToStep()));
- this->GoToButtonToStepMap[goToButton] = step;
- // if the goTo step has an icon associated with it, then add it to the button
- if (ctkWorkflowWidgetStep* wwStep = dynamic_cast<ctkWorkflowWidgetStep*>(step))
- {
- goToButton->setIcon(wwStep->icon());
- }
- }
- }
- // Show/hide the goTo buttons depending on whether they are accessible from the current step
- ctkWorkflowStep* goToStep;
- foreach (QPushButton* goToButton, this->GoToButtonToStepMap.keys())
- {
- // TODO enable and show the goTo button if we can go to it
- // ctkWorkflowStep* goToStep = this->GoToButtonToStepMap[goToButton];
- // if (this->Workflow->canGoToStep(currentStep, goToStep))
- // for now we'll assume we can go to the step
- goToStep = this->GoToButtonToStepMap[goToButton];
- Q_ASSERT(goToStep);
- if (currentStep && this->Workflow->canGoToStep(goToStep->id(), currentStep))
- {
- goToButton->setEnabled(true);
- goToButton->show();
- }
- // disable the goTo button if we can't go to it, and optionally hide it
- else
- {
- goToButton->setEnabled(false);
- this->HideInvalidButtons ? goToButton->hide() : goToButton->show();
- }
- }
- }
- //-----------------------------------------------------------------------------
- // ctkWorkflowButtonBoxWidget methods
- //-----------------------------------------------------------------------------
- ctkWorkflowButtonBoxWidget::ctkWorkflowButtonBoxWidget(ctkWorkflow* newWorkflow, QWidget* newParent) :
- Superclass(newParent)
- , d_ptr(new ctkWorkflowButtonBoxWidgetPrivate(*this))
- {
- Q_D(ctkWorkflowButtonBoxWidget);
- d->setupUi(this);
- this->setWorkflow(newWorkflow);
- }
- //-----------------------------------------------------------------------------
- ctkWorkflowButtonBoxWidget::ctkWorkflowButtonBoxWidget(QWidget* newParent) :
- Superclass(newParent)
- , d_ptr(new ctkWorkflowButtonBoxWidgetPrivate(*this))
- {
- Q_D(ctkWorkflowButtonBoxWidget);
- d->setupUi(this);
- }
- //-----------------------------------------------------------------------------
- ctkWorkflowButtonBoxWidget::~ctkWorkflowButtonBoxWidget()
- {
- }
- //-----------------------------------------------------------------------------
- void ctkWorkflowButtonBoxWidget::setWorkflow(ctkWorkflow * newWorkflow)
- {
- Q_D(ctkWorkflowButtonBoxWidget);
- if (d->Workflow == newWorkflow)
- {
- return;
- }
- // Disconnect
- if (!d->Workflow.isNull())
- {
- QObject::disconnect(d->BackButton, SIGNAL(clicked()), d->Workflow, SLOT(goBackward()));
- QObject::disconnect(d->NextButton, SIGNAL(clicked()), d->Workflow, SLOT(goForward()));
- }
- // Connect
- if (newWorkflow)
- {
- QObject::connect(d->BackButton, SIGNAL(clicked()), newWorkflow, SLOT(goBackward()));
- QObject::connect(d->NextButton, SIGNAL(clicked()), newWorkflow, SLOT(goForward()));
- }
- d->Workflow = newWorkflow;
- }
- //-----------------------------------------------------------------------------
- CTK_GET_CPP(ctkWorkflowButtonBoxWidget, QString, backButtonDefaultText,
- BackButtonDefaultText);
- //-----------------------------------------------------------------------------
- void ctkWorkflowButtonBoxWidget::setBackButtonDefaultText(const QString& defaultText)
- {
- Q_D(ctkWorkflowButtonBoxWidget);
- d->BackButtonDefaultText = defaultText;
- if (d->Workflow)
- {
- this->updateButtons(d->Workflow->currentStep());
- }
- else
- {
- d->BackButton->setText(d->BackButtonDefaultText);
- }
- }
- //-----------------------------------------------------------------------------
- CTK_GET_CPP(ctkWorkflowButtonBoxWidget, QString, nextButtonDefaultText,
- NextButtonDefaultText);
- //-----------------------------------------------------------------------------
- void ctkWorkflowButtonBoxWidget::setNextButtonDefaultText(const QString& defaultText)
- {
- Q_D(ctkWorkflowButtonBoxWidget);
- d->NextButtonDefaultText = defaultText;
- if (d->Workflow)
- {
- this->updateButtons(d->Workflow->currentStep());
- }
- else
- {
- d->NextButton->setText(d->NextButtonDefaultText);
- }
- }
- //-----------------------------------------------------------------------------
- CTK_GET_CPP(ctkWorkflowButtonBoxWidget, ctkWorkflow*, workflow, Workflow);
- CTK_GET_CPP(ctkWorkflowButtonBoxWidget, QPushButton*, backButton, BackButton);
- CTK_GET_CPP(ctkWorkflowButtonBoxWidget, QBoxLayout::Direction, direction, Direction);
- CTK_GET_CPP(ctkWorkflowButtonBoxWidget, bool, hideInvalidButtons, HideInvalidButtons);
- CTK_SET_CPP(ctkWorkflowButtonBoxWidget, bool, setHideInvalidButtons, HideInvalidButtons);
- //-----------------------------------------------------------------------------
- // TODO will be list of next buttons for branching workflow
- QPushButton* ctkWorkflowButtonBoxWidget::nextButton()const
- {
- Q_D(const ctkWorkflowButtonBoxWidget);
- return d->NextButton;
- }
- //-----------------------------------------------------------------------------
- QList<QPushButton*> ctkWorkflowButtonBoxWidget::goToButtons()const
- {
- Q_D(const ctkWorkflowButtonBoxWidget);
- return d->GoToButtonToStepMap.keys();
- }
- //-----------------------------------------------------------------------------
- void ctkWorkflowButtonBoxWidget::setDirection(const QBoxLayout::Direction& newDirection)
- {
- if (QBoxLayout* layout = qobject_cast<QBoxLayout*>(this->layout()))
- {
- layout->setDirection(newDirection);
- }
- }
- //-----------------------------------------------------------------------------
- void ctkWorkflowButtonBoxWidget::updateButtons(ctkWorkflowStep* currentStep)
- {
- Q_D(ctkWorkflowButtonBoxWidget);
- // hide aspects of the button bar if specified by the current step
- if(ctkWorkflowWidgetStep* currentWidgetStep = dynamic_cast<ctkWorkflowWidgetStep*>(currentStep))
- {
- bool hideButtonBar = currentWidgetStep->buttonBoxHints() & ctkWorkflowWidgetStep::ButtonBoxHidden;
- this->setHidden(hideButtonBar);
- }
- d->updateBackButton(currentStep);
- d->updateNextButton(currentStep);
- d->updateGoToButtons(currentStep);
- }
- //--------------------------------------------------------------------------
- void ctkWorkflowButtonBoxWidget::prepareGoToStep()
- {
- Q_D(ctkWorkflowButtonBoxWidget);
- if (QPushButton* button = qobject_cast<QPushButton*>(QObject::sender()))
- {
- if (ctkWorkflowStep* step = d->GoToButtonToStepMap.value(button))
- {
- d->Workflow->goToStep(step->id());
- }
- }
- }
|