ctkWorkflowButtonBoxWidget.cpp 14 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400
  1. /*=========================================================================
  2. Library: CTK
  3. Copyright (c) Kitware Inc.
  4. Licensed under the Apache License, Version 2.0 (the "License");
  5. you may not use this file except in compliance with the License.
  6. You may obtain a copy of the License at
  7. http://www.apache.org/licenses/LICENSE-2.0.txt
  8. Unless required by applicable law or agreed to in writing, software
  9. distributed under the License is distributed on an "AS IS" BASIS,
  10. WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
  11. See the License for the specific language governing permissions and
  12. limitations under the License.
  13. =========================================================================*/
  14. // Qt includes
  15. #include <QBoxLayout>
  16. #include <QDebug>
  17. #include <QList>
  18. #include <QPointer>
  19. #include <QStyle>
  20. #include <QVariant>
  21. // CTK includes
  22. #include "ctkPushButton.h"
  23. #include "ctkWorkflowButtonBoxWidget.h"
  24. #include "ctkWorkflowStep.h"
  25. #include "ctkWorkflowWidget.h"
  26. #include "ctkWorkflowWidgetStep.h"
  27. #include "ctkWorkflow.h"
  28. // STD includes
  29. #include <iostream>
  30. //-----------------------------------------------------------------------------
  31. class ctkWorkflowButtonBoxWidgetPrivate
  32. {
  33. Q_DECLARE_PUBLIC(ctkWorkflowButtonBoxWidget);
  34. protected:
  35. ctkWorkflowButtonBoxWidget* const q_ptr;
  36. public:
  37. ctkWorkflowButtonBoxWidgetPrivate(ctkWorkflowButtonBoxWidget& object);
  38. ~ctkWorkflowButtonBoxWidgetPrivate(){}
  39. void setupUi(QWidget * newParent);
  40. QPointer<ctkWorkflow> Workflow;
  41. // Convenient typedefs
  42. typedef QMap<ctkPushButton*, ctkWorkflowStep*> ButtonToStepMapType;
  43. // The buttons on the widget (maintain maps associating each forward/goTo button with its step)
  44. ctkPushButton* BackButton;
  45. QString BackButtonFormat;
  46. ctkPushButton* NextButton;
  47. QString NextButtonFormat;
  48. ButtonToStepMapType GoToButtonToStepMap;
  49. QString GoToButtonsFormat;
  50. // Direction for layout (for use with QBoxLayout only)
  51. QBoxLayout::Direction Direction;
  52. QSizePolicy ButtonSizePolicy;
  53. bool HideGoToButtons;
  54. bool HideInvalidButtons;
  55. // Functions to add/remove buttons
  56. void updateBackButton(ctkWorkflowStep* currentStep);
  57. void updateNextButton(ctkWorkflowStep* currentStep);
  58. void updateGoToButtons(ctkWorkflowStep* currentStep);
  59. };
  60. //-----------------------------------------------------------------------------
  61. // ctkWorkflowButtonBoxWidgetPrivate methods
  62. //-----------------------------------------------------------------------------
  63. ctkWorkflowButtonBoxWidgetPrivate::ctkWorkflowButtonBoxWidgetPrivate(ctkWorkflowButtonBoxWidget& object)
  64. :q_ptr(&object)
  65. {
  66. this->BackButton = 0;
  67. this->BackButtonFormat = "[<-]{backButtonText|\"Back\"}(back:description)";
  68. this->NextButton = 0;
  69. this->NextButtonFormat = "{nextButtonText|\"Next\"}(next:description)[->]";
  70. this->GoToButtonsFormat = "[icon]{stepid|\"Finish\"}(description)";
  71. this->Direction = QBoxLayout::LeftToRight;
  72. this->ButtonSizePolicy = QSizePolicy(QSizePolicy::Minimum, QSizePolicy::Minimum);
  73. this->HideGoToButtons = false;
  74. this->HideInvalidButtons = false;
  75. }
  76. //-----------------------------------------------------------------------------
  77. void ctkWorkflowButtonBoxWidgetPrivate::setupUi(QWidget * newParent)
  78. {
  79. QBoxLayout * boxLayout = new QBoxLayout(this->Direction, newParent);
  80. boxLayout->setSpacing(0);
  81. boxLayout->setContentsMargins(0, 0, 0, 0);
  82. newParent->setLayout(boxLayout);
  83. // Retrieve standard icons
  84. QIcon backIcon = newParent->style()->standardIcon(QStyle::SP_ArrowLeft);
  85. QIcon nextIcon = newParent->style()->standardIcon(QStyle::SP_ArrowRight);
  86. // Setup the buttons
  87. this->BackButton = new ctkPushButton(backIcon, this->BackButtonFormat, newParent);
  88. this->BackButton->setSizePolicy(this->ButtonSizePolicy);
  89. this->BackButton->setIconAlignment(Qt::AlignLeft | Qt::AlignVCenter);
  90. newParent->layout()->addWidget(this->BackButton);
  91. this->NextButton = new ctkPushButton(nextIcon, this->NextButtonFormat, newParent);
  92. this->NextButton->setSizePolicy(this->ButtonSizePolicy);
  93. this->NextButton->setIconAlignment(Qt::AlignRight | Qt::AlignVCenter);
  94. newParent->layout()->addWidget(this->NextButton);
  95. }
  96. //-----------------------------------------------------------------------------
  97. void ctkWorkflowButtonBoxWidgetPrivate::updateBackButton(ctkWorkflowStep* currentStep)
  98. {
  99. #ifndef QT_NO_DEBUG
  100. Q_Q(ctkWorkflowButtonBoxWidget);
  101. Q_ASSERT(this->Workflow);
  102. Q_ASSERT(q->layout());
  103. #endif
  104. ctkWorkflowWidgetStep* step = dynamic_cast<ctkWorkflowWidgetStep*>(currentStep);
  105. // Set the back button text and icon
  106. ctkWorkflowWidget::formatButton(this->BackButton, this->BackButtonFormat, step);
  107. // Disable the back button if we can't go backward
  108. bool enable = currentStep && this->Workflow->canGoBackward(currentStep);
  109. bool visible = true;
  110. // Apply the buttonBox hints if possible
  111. if (enable && step)
  112. {
  113. enable = !(step->buttonBoxHints() & ctkWorkflowWidgetStep::BackButtonDisabled);
  114. visible = !(step->buttonBoxHints() & (ctkWorkflowWidgetStep::BackButtonHidden |
  115. ctkWorkflowWidgetStep::ButtonBoxHidden));
  116. }
  117. this->BackButton->setEnabled(enable);
  118. if (!enable && this->HideInvalidButtons)
  119. {
  120. visible = false;
  121. }
  122. this->BackButton->setVisible(visible);
  123. }
  124. //-----------------------------------------------------------------------------
  125. // This will change for branching workflows, to look more like updateGoToButtons()
  126. void ctkWorkflowButtonBoxWidgetPrivate::updateNextButton(ctkWorkflowStep* currentStep)
  127. {
  128. #ifndef QT_NO_DEBUG
  129. Q_Q(ctkWorkflowButtonBoxWidget);
  130. Q_ASSERT(this->Workflow);
  131. Q_ASSERT(q->layout());
  132. #endif
  133. ctkWorkflowWidgetStep* step = dynamic_cast<ctkWorkflowWidgetStep*>(currentStep);
  134. // Set the next button text and icon
  135. ctkWorkflowWidget::formatButton(this->NextButton, this->NextButtonFormat, step);
  136. // Disable the next button if we can't go backward
  137. bool enable = currentStep && this->Workflow->canGoForward(currentStep);
  138. bool visible = true;
  139. // Apply the buttonBox hints if possible
  140. if (enable && step)
  141. {
  142. enable = !(step->buttonBoxHints() & ctkWorkflowWidgetStep::NextButtonDisabled);
  143. visible = !(step->buttonBoxHints() & (ctkWorkflowWidgetStep::NextButtonHidden |
  144. ctkWorkflowWidgetStep::ButtonBoxHidden));
  145. }
  146. this->NextButton->setEnabled(enable);
  147. if (!enable && this->HideInvalidButtons)
  148. {
  149. visible = false;
  150. }
  151. this->NextButton->setVisible(visible);
  152. }
  153. //-----------------------------------------------------------------------------
  154. void ctkWorkflowButtonBoxWidgetPrivate::updateGoToButtons(ctkWorkflowStep* currentStep)
  155. {
  156. Q_Q(ctkWorkflowButtonBoxWidget);
  157. Q_ASSERT(this->Workflow);
  158. Q_ASSERT(q->layout());
  159. // Change the buttons only if the set of steps to have goTo buttons is either empty or has changed
  160. QSet<ctkWorkflowStep*> goToStepsToHaveButtons =
  161. QSet<ctkWorkflowStep*>::fromList(this->Workflow->finishSteps());
  162. QSet<ctkWorkflowStep*> goToStepsThatHaveButtons =
  163. QSet<ctkWorkflowStep*>::fromList(this->GoToButtonToStepMap.values());
  164. // Remove the buttons if the set of steps to have goTo buttons has changed
  165. if (goToStepsThatHaveButtons != goToStepsToHaveButtons)
  166. {
  167. foreach (ctkPushButton* goToButton, this->GoToButtonToStepMap.keys())
  168. {
  169. q->layout()->removeWidget(goToButton);
  170. goToButton->deleteLater();
  171. }
  172. this->GoToButtonToStepMap.clear();
  173. }
  174. // Create the buttons, either for the first time or after removing them above
  175. if (this->GoToButtonToStepMap.isEmpty())
  176. {
  177. // Create a button for each of the goToSteps
  178. foreach (ctkWorkflowStep* step, goToStepsToHaveButtons)
  179. {
  180. // TODO shouldn't have id here
  181. ctkPushButton* goToButton = new ctkPushButton();
  182. goToButton->setSizePolicy(this->ButtonSizePolicy);
  183. ctkWorkflowWidgetStep* wwStep = dynamic_cast<ctkWorkflowWidgetStep*>(step);
  184. ctkWorkflowWidget::formatButton(goToButton, this->GoToButtonsFormat, wwStep);
  185. q->layout()->addWidget(goToButton);
  186. QObject::connect(goToButton, SIGNAL(clicked()), q, SLOT(prepareGoToStep()));
  187. this->GoToButtonToStepMap[goToButton] = step;
  188. }
  189. }
  190. // Show/hide the goTo buttons depending on whether they are accessible from the current step
  191. ctkWorkflowWidgetStep* step = dynamic_cast<ctkWorkflowWidgetStep*>(currentStep);
  192. foreach (ctkPushButton* goToButton, this->GoToButtonToStepMap.keys())
  193. {
  194. // TODO enable and show the goTo button if we can go to it
  195. // ctkWorkflowStep* goToStep = this->GoToButtonToStepMap[goToButton];
  196. // if (this->Workflow->canGoToStep(currentStep, goToStep))
  197. // for now we'll assume we can go to the step
  198. ctkWorkflowStep* goToStep = this->GoToButtonToStepMap[goToButton];
  199. Q_ASSERT(goToStep);
  200. bool enable = currentStep && this->Workflow->canGoToStep(goToStep->id(), currentStep);
  201. bool visible = step ? !(step->buttonBoxHints() & ctkWorkflowWidgetStep::ButtonBoxHidden) : true;
  202. if ((!enable && this->HideInvalidButtons)
  203. || this->HideGoToButtons
  204. )
  205. {
  206. visible = false;
  207. }
  208. goToButton->setEnabled(enable);
  209. goToButton->setVisible(visible);
  210. }
  211. }
  212. //-----------------------------------------------------------------------------
  213. // ctkWorkflowButtonBoxWidget methods
  214. //-----------------------------------------------------------------------------
  215. ctkWorkflowButtonBoxWidget::ctkWorkflowButtonBoxWidget(ctkWorkflow* newWorkflow, QWidget* newParent) :
  216. Superclass(newParent)
  217. , d_ptr(new ctkWorkflowButtonBoxWidgetPrivate(*this))
  218. {
  219. Q_D(ctkWorkflowButtonBoxWidget);
  220. d->setupUi(this);
  221. this->setWorkflow(newWorkflow);
  222. }
  223. //-----------------------------------------------------------------------------
  224. ctkWorkflowButtonBoxWidget::ctkWorkflowButtonBoxWidget(QWidget* newParent) :
  225. Superclass(newParent)
  226. , d_ptr(new ctkWorkflowButtonBoxWidgetPrivate(*this))
  227. {
  228. Q_D(ctkWorkflowButtonBoxWidget);
  229. d->setupUi(this);
  230. }
  231. //-----------------------------------------------------------------------------
  232. ctkWorkflowButtonBoxWidget::~ctkWorkflowButtonBoxWidget()
  233. {
  234. }
  235. //-----------------------------------------------------------------------------
  236. void ctkWorkflowButtonBoxWidget::setWorkflow(ctkWorkflow * newWorkflow)
  237. {
  238. Q_D(ctkWorkflowButtonBoxWidget);
  239. if (d->Workflow == newWorkflow)
  240. {
  241. return;
  242. }
  243. // Disconnect
  244. if (!d->Workflow.isNull())
  245. {
  246. QObject::disconnect(d->BackButton, SIGNAL(clicked()), d->Workflow, SLOT(goBackward()));
  247. QObject::disconnect(d->NextButton, SIGNAL(clicked()), d->Workflow, SLOT(goForward()));
  248. }
  249. // Connect
  250. if (newWorkflow)
  251. {
  252. QObject::connect(d->BackButton, SIGNAL(clicked()), newWorkflow, SLOT(goBackward()));
  253. QObject::connect(d->NextButton, SIGNAL(clicked()), newWorkflow, SLOT(goForward()));
  254. }
  255. d->Workflow = newWorkflow;
  256. }
  257. //-----------------------------------------------------------------------------
  258. CTK_GET_CPP(ctkWorkflowButtonBoxWidget, QString, backButtonFormat,
  259. BackButtonFormat);
  260. //-----------------------------------------------------------------------------
  261. void ctkWorkflowButtonBoxWidget::setBackButtonFormat(const QString& format)
  262. {
  263. Q_D(ctkWorkflowButtonBoxWidget);
  264. d->BackButtonFormat = format;
  265. d->updateBackButton(d->Workflow ? d->Workflow->currentStep() : 0);
  266. }
  267. //-----------------------------------------------------------------------------
  268. CTK_GET_CPP(ctkWorkflowButtonBoxWidget, QString, nextButtonFormat,
  269. NextButtonFormat);
  270. //-----------------------------------------------------------------------------
  271. void ctkWorkflowButtonBoxWidget::setNextButtonFormat(const QString& format)
  272. {
  273. Q_D(ctkWorkflowButtonBoxWidget);
  274. d->NextButtonFormat = format;
  275. d->updateNextButton(d->Workflow ? d->Workflow->currentStep() : 0);
  276. }
  277. //-----------------------------------------------------------------------------
  278. CTK_GET_CPP(ctkWorkflowButtonBoxWidget, QString, goToButtonsFormat,
  279. GoToButtonsFormat);
  280. //-----------------------------------------------------------------------------
  281. void ctkWorkflowButtonBoxWidget::setGoToButtonsFormat(const QString& format)
  282. {
  283. Q_D(ctkWorkflowButtonBoxWidget);
  284. d->GoToButtonsFormat = format;
  285. d->updateGoToButtons(d->Workflow ? d->Workflow->currentStep() : 0);
  286. }
  287. //-----------------------------------------------------------------------------
  288. CTK_GET_CPP(ctkWorkflowButtonBoxWidget, ctkWorkflow*, workflow, Workflow);
  289. CTK_GET_CPP(ctkWorkflowButtonBoxWidget, ctkPushButton*, backButton, BackButton);
  290. CTK_GET_CPP(ctkWorkflowButtonBoxWidget, QBoxLayout::Direction, direction, Direction);
  291. CTK_GET_CPP(ctkWorkflowButtonBoxWidget, bool, hideGoToButtons, HideGoToButtons);
  292. CTK_SET_CPP(ctkWorkflowButtonBoxWidget, bool, setHideGoToButtons, HideGoToButtons);
  293. CTK_GET_CPP(ctkWorkflowButtonBoxWidget, bool, hideInvalidButtons, HideInvalidButtons);
  294. CTK_SET_CPP(ctkWorkflowButtonBoxWidget, bool, setHideInvalidButtons, HideInvalidButtons);
  295. //-----------------------------------------------------------------------------
  296. // TODO will be list of next buttons for branching workflow
  297. ctkPushButton* ctkWorkflowButtonBoxWidget::nextButton()const
  298. {
  299. Q_D(const ctkWorkflowButtonBoxWidget);
  300. return d->NextButton;
  301. }
  302. //-----------------------------------------------------------------------------
  303. QList<ctkPushButton*> ctkWorkflowButtonBoxWidget::goToButtons()const
  304. {
  305. Q_D(const ctkWorkflowButtonBoxWidget);
  306. return d->GoToButtonToStepMap.keys();
  307. }
  308. //-----------------------------------------------------------------------------
  309. void ctkWorkflowButtonBoxWidget::setDirection(const QBoxLayout::Direction& newDirection)
  310. {
  311. if (QBoxLayout* layout = qobject_cast<QBoxLayout*>(this->layout()))
  312. {
  313. layout->setDirection(newDirection);
  314. }
  315. }
  316. //-----------------------------------------------------------------------------
  317. void ctkWorkflowButtonBoxWidget::updateButtons(ctkWorkflowStep* currentStep)
  318. {
  319. Q_D(ctkWorkflowButtonBoxWidget);
  320. d->updateBackButton(currentStep);
  321. d->updateNextButton(currentStep);
  322. d->updateGoToButtons(currentStep);
  323. }
  324. //--------------------------------------------------------------------------
  325. void ctkWorkflowButtonBoxWidget::prepareGoToStep()
  326. {
  327. Q_D(ctkWorkflowButtonBoxWidget);
  328. if (ctkPushButton* button = qobject_cast<ctkPushButton*>(QObject::sender()))
  329. {
  330. if (ctkWorkflowStep* step = d->GoToButtonToStepMap.value(button))
  331. {
  332. d->Workflow->goToStep(step->id());
  333. }
  334. }
  335. }