ctkWorkflowButtonBoxWidget.cpp 14 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407
  1. /*=========================================================================
  2. Library: CTK
  3. Copyright (c) 2010 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.commontk.org/LICENSE
  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 <QWidget>
  16. #include <QList>
  17. #include <QPushButton>
  18. #include <QBoxLayout>
  19. #include <QDebug>
  20. #include <QPointer>
  21. #include <QStyle>
  22. // CTK includes
  23. #include "ctkWorkflowButtonBoxWidget.h"
  24. #include "ctkWorkflowStep.h"
  25. #include "ctkWorkflowAbstractWidgetStep.h"
  26. #include "ctkWorkflow.h"
  27. // STD includes
  28. #include <iostream>
  29. //-----------------------------------------------------------------------------
  30. class ctkWorkflowButtonBoxWidgetPrivate : public ctkPrivate<ctkWorkflowButtonBoxWidget>
  31. {
  32. public:
  33. CTK_DECLARE_PUBLIC(ctkWorkflowButtonBoxWidget);
  34. ctkWorkflowButtonBoxWidgetPrivate();
  35. ~ctkWorkflowButtonBoxWidgetPrivate(){}
  36. void setupUi(QWidget * newParent);
  37. QPointer<ctkWorkflow> Workflow;
  38. // Convenient typedefs
  39. typedef QMap<QPushButton*, ctkWorkflowStep*> ButtonToStepMapType;
  40. // The buttons on the widget (maintain maps associating each forward/goTo button with its step)
  41. QPushButton* BackButton;
  42. QString BackButtonDefaultText;
  43. QPushButton* NextButton;
  44. QString NextButtonDefaultText;
  45. ButtonToStepMapType GoToButtonToStepMap;
  46. // Direction for layout (for use with QBoxLayout only)
  47. QBoxLayout::Direction Direction;
  48. bool HideInvalidButtons;
  49. // Functions to add/remove buttons
  50. void updateBackButton(ctkWorkflowStep* currentStep);
  51. void updateNextButton(ctkWorkflowStep* currentStep);
  52. void updateGoToButtons(ctkWorkflowStep* currentStep);
  53. };
  54. //-----------------------------------------------------------------------------
  55. // ctkWorkflowButtonBoxWidgetPrivate methods
  56. //-----------------------------------------------------------------------------
  57. ctkWorkflowButtonBoxWidgetPrivate::ctkWorkflowButtonBoxWidgetPrivate()
  58. {
  59. this->BackButton = 0;
  60. this->BackButtonDefaultText = "Back";
  61. this->NextButton = 0;
  62. this->NextButtonDefaultText = "Next";
  63. this->Direction = QBoxLayout::LeftToRight;
  64. this->HideInvalidButtons = false;
  65. }
  66. //-----------------------------------------------------------------------------
  67. void ctkWorkflowButtonBoxWidgetPrivate::setupUi(QWidget * newParent)
  68. {
  69. QBoxLayout * boxLayout = new QBoxLayout(this->Direction, newParent);
  70. boxLayout->setSpacing(0);
  71. boxLayout->setContentsMargins(0, 0, 0, 0);
  72. newParent->setLayout(boxLayout);
  73. // Retrieve standard icons
  74. QIcon backIcon = newParent->style()->standardIcon(QStyle::SP_ArrowLeft);
  75. QIcon nextIcon = newParent->style()->standardIcon(QStyle::SP_ArrowRight);
  76. // Setup the buttons
  77. this->BackButton = new QPushButton(backIcon, this->BackButtonDefaultText, newParent);
  78. newParent->layout()->addWidget(this->BackButton);
  79. this->NextButton = new QPushButton(nextIcon, this->NextButtonDefaultText, newParent);
  80. this->NextButton->setLayoutDirection(Qt::RightToLeft);
  81. newParent->layout()->addWidget(this->NextButton);
  82. }
  83. //-----------------------------------------------------------------------------
  84. void ctkWorkflowButtonBoxWidgetPrivate::updateBackButton(ctkWorkflowStep* currentStep)
  85. {
  86. CTK_P(ctkWorkflowButtonBoxWidget);
  87. Q_ASSERT(this->Workflow);
  88. Q_ASSERT(p->layout());
  89. ctkWorkflowAbstractWidgetStep* step = dynamic_cast<ctkWorkflowAbstractWidgetStep*>(currentStep);
  90. // Set the back button text
  91. QString backButtonText = this->BackButtonDefaultText;
  92. if (step && !step->backButtonText().isEmpty())
  93. {
  94. backButtonText = step->backButtonText();
  95. }
  96. this->BackButton->setText(backButtonText);
  97. // Enable and show the back button if we can go backward
  98. if (currentStep && this->Workflow->canGoBackward(currentStep))
  99. {
  100. this->BackButton->setEnabled(true);
  101. this->BackButton->show();
  102. // Apply the buttonBox hints if possible
  103. if (step)
  104. {
  105. this->BackButton->setDisabled(
  106. step->buttonBoxHints() & ctkWorkflowAbstractWidgetStep::BackButtonDisabled);
  107. this->BackButton->setHidden(
  108. step->buttonBoxHints() & ctkWorkflowAbstractWidgetStep::BackButtonHidden);
  109. }
  110. }
  111. // Disable the back button if we can't go backward, and optionally hide it
  112. else
  113. {
  114. this->BackButton->setEnabled(false);
  115. this->HideInvalidButtons ? this->BackButton->hide() : this->BackButton->show();
  116. }
  117. }
  118. //-----------------------------------------------------------------------------
  119. // This will change for branching workflows, to look more like updateGoToButtons()
  120. void ctkWorkflowButtonBoxWidgetPrivate::updateNextButton(ctkWorkflowStep* currentStep)
  121. {
  122. CTK_P(ctkWorkflowButtonBoxWidget);
  123. Q_ASSERT(this->Workflow);
  124. Q_ASSERT(p->layout());
  125. ctkWorkflowAbstractWidgetStep* step = dynamic_cast<ctkWorkflowAbstractWidgetStep*>(currentStep);
  126. // Set the next button text
  127. QString nextButtonText = this->NextButtonDefaultText;
  128. if (step && !step->nextButtonText().isEmpty())
  129. {
  130. nextButtonText = step->nextButtonText();
  131. }
  132. this->NextButton->setText(nextButtonText);
  133. // Enable and show the next button if we can go forward
  134. if (currentStep && this->Workflow->canGoForward(currentStep))
  135. {
  136. this->NextButton->setEnabled(true);
  137. this->NextButton->show();
  138. // Apply the buttonBox hints if possible
  139. if (step)
  140. {
  141. this->NextButton->setDisabled(
  142. step->buttonBoxHints() & ctkWorkflowAbstractWidgetStep::NextButtonDisabled);
  143. this->NextButton->setHidden(
  144. step->buttonBoxHints() & ctkWorkflowAbstractWidgetStep::NextButtonHidden);
  145. }
  146. }
  147. // Disable the next button if we can't go forward, and optionally hide it
  148. else
  149. {
  150. this->NextButton->setEnabled(false);
  151. this->HideInvalidButtons ? this->NextButton->hide() : this->NextButton->show();
  152. }
  153. }
  154. //-----------------------------------------------------------------------------
  155. void ctkWorkflowButtonBoxWidgetPrivate::updateGoToButtons(ctkWorkflowStep* currentStep)
  156. {
  157. CTK_P(ctkWorkflowButtonBoxWidget);
  158. Q_ASSERT(this->Workflow);
  159. Q_ASSERT(p->layout());
  160. // Change the buttons only if the set of steps to have goTo buttons is either empty or has changed
  161. QSet<ctkWorkflowStep*> goToStepsToHaveButtons = QSet<ctkWorkflowStep*>::fromList(this->Workflow->finishSteps());
  162. QSet<ctkWorkflowStep*> goToStepsThatHaveButtons = QSet<ctkWorkflowStep*>::fromList(this->GoToButtonToStepMap.values());
  163. // Remove the buttons if the set of steps to have goTo buttons has changed
  164. if (!this->GoToButtonToStepMap.isEmpty() && goToStepsThatHaveButtons != goToStepsToHaveButtons)
  165. {
  166. foreach (QPushButton* goToButton, this->GoToButtonToStepMap.keys())
  167. {
  168. p->layout()->removeWidget(goToButton);
  169. delete goToButton;
  170. }
  171. this->GoToButtonToStepMap.clear();
  172. }
  173. // Create the buttons, either for the first time or after removing them above
  174. if (this->GoToButtonToStepMap.isEmpty())
  175. {
  176. // Create a button for each of the goToSteps
  177. foreach (ctkWorkflowStep* step, goToStepsToHaveButtons)
  178. {
  179. // TODO shouldn't have id here
  180. QPushButton* goToButton = new QPushButton(step->id());
  181. p->layout()->addWidget(goToButton);
  182. QObject::connect(goToButton, SIGNAL(clicked()), p, SLOT(prepareGoToStep()));
  183. this->GoToButtonToStepMap[goToButton] = step;
  184. // if the goTo step has an icon associated with it, then add it to the button
  185. if (ctkWorkflowAbstractWidgetStep* wwStep = dynamic_cast<ctkWorkflowAbstractWidgetStep*>(step))
  186. {
  187. goToButton->setIcon(wwStep->icon());
  188. }
  189. }
  190. }
  191. // Show/hide the goTo buttons depending on whether they are accessible from the current step
  192. ctkWorkflowStep* goToStep;
  193. foreach (QPushButton* goToButton, this->GoToButtonToStepMap.keys())
  194. {
  195. // TODO enable and show the goTo button if we can go to it
  196. // ctkWorkflowStep* goToStep = this->GoToButtonToStepMap[goToButton];
  197. // if (this->Workflow->canGoToStep(currentStep, goToStep))
  198. // for now we'll assume we can go to the step
  199. goToStep = this->GoToButtonToStepMap[goToButton];
  200. Q_ASSERT(goToStep);
  201. if (currentStep && this->Workflow->canGoToStep(goToStep->id(), currentStep))
  202. {
  203. goToButton->setEnabled(true);
  204. goToButton->show();
  205. }
  206. // disable the goTo button if we can't go to it, and optionally hide it
  207. else
  208. {
  209. goToButton->setEnabled(false);
  210. this->HideInvalidButtons ? goToButton->hide() : goToButton->show();
  211. }
  212. }
  213. }
  214. //-----------------------------------------------------------------------------
  215. // ctkWorkflowButtonBoxWidget methods
  216. //-----------------------------------------------------------------------------
  217. ctkWorkflowButtonBoxWidget::ctkWorkflowButtonBoxWidget(ctkWorkflow* newWorkflow, QWidget* newParent) :
  218. Superclass(newParent)
  219. {
  220. CTK_INIT_PRIVATE(ctkWorkflowButtonBoxWidget);
  221. CTK_D(ctkWorkflowButtonBoxWidget);
  222. d->setupUi(this);
  223. this->setWorkflow(newWorkflow);
  224. }
  225. //-----------------------------------------------------------------------------
  226. ctkWorkflowButtonBoxWidget::ctkWorkflowButtonBoxWidget(QWidget* newParent) :
  227. Superclass(newParent)
  228. {
  229. CTK_INIT_PRIVATE(ctkWorkflowButtonBoxWidget);
  230. CTK_D(ctkWorkflowButtonBoxWidget);
  231. d->setupUi(this);
  232. }
  233. //-----------------------------------------------------------------------------
  234. void ctkWorkflowButtonBoxWidget::setWorkflow(ctkWorkflow * newWorkflow)
  235. {
  236. CTK_D(ctkWorkflowButtonBoxWidget);
  237. if (d->Workflow == newWorkflow)
  238. {
  239. return;
  240. }
  241. // Disconnect
  242. if (!d->Workflow.isNull())
  243. {
  244. QObject::disconnect(d->BackButton, SIGNAL(clicked()), d->Workflow, SLOT(goBackward()));
  245. QObject::disconnect(d->NextButton, SIGNAL(clicked()), d->Workflow, SLOT(goForward()));
  246. }
  247. // Connect
  248. if (newWorkflow)
  249. {
  250. QObject::connect(d->BackButton, SIGNAL(clicked()), newWorkflow, SLOT(goBackward()));
  251. QObject::connect(d->NextButton, SIGNAL(clicked()), newWorkflow, SLOT(goForward()));
  252. }
  253. d->Workflow = newWorkflow;
  254. }
  255. //-----------------------------------------------------------------------------
  256. CTK_GET_CXX(ctkWorkflowButtonBoxWidget, QString, backButtonDefaultText,
  257. BackButtonDefaultText);
  258. //-----------------------------------------------------------------------------
  259. void ctkWorkflowButtonBoxWidget::setBackButtonDefaultText(const QString& defaultText)
  260. {
  261. CTK_D(ctkWorkflowButtonBoxWidget);
  262. d->BackButtonDefaultText = defaultText;
  263. if (d->Workflow)
  264. {
  265. this->updateButtons(d->Workflow->currentStep());
  266. }
  267. else
  268. {
  269. d->BackButton->setText(d->BackButtonDefaultText);
  270. }
  271. }
  272. //-----------------------------------------------------------------------------
  273. CTK_GET_CXX(ctkWorkflowButtonBoxWidget, QString, nextButtonDefaultText,
  274. NextButtonDefaultText);
  275. //-----------------------------------------------------------------------------
  276. void ctkWorkflowButtonBoxWidget::setNextButtonDefaultText(const QString& defaultText)
  277. {
  278. CTK_D(ctkWorkflowButtonBoxWidget);
  279. d->NextButtonDefaultText = defaultText;
  280. if (d->Workflow)
  281. {
  282. this->updateButtons(d->Workflow->currentStep());
  283. }
  284. else
  285. {
  286. d->NextButton->setText(d->NextButtonDefaultText);
  287. }
  288. }
  289. //-----------------------------------------------------------------------------
  290. CTK_GET_CXX(ctkWorkflowButtonBoxWidget, ctkWorkflow*, workflow, Workflow);
  291. CTK_GET_CXX(ctkWorkflowButtonBoxWidget, QPushButton*, backButton, BackButton);
  292. CTK_GET_CXX(ctkWorkflowButtonBoxWidget, QBoxLayout::Direction, direction, Direction);
  293. CTK_GET_CXX(ctkWorkflowButtonBoxWidget, bool, hideInvalidButtons, HideInvalidButtons);
  294. CTK_SET_CXX(ctkWorkflowButtonBoxWidget, bool, setHideInvalidButtons, HideInvalidButtons);
  295. //-----------------------------------------------------------------------------
  296. // TODO will be list of next buttons for branching workflow
  297. QPushButton* ctkWorkflowButtonBoxWidget::nextButton()const
  298. {
  299. CTK_D(const ctkWorkflowButtonBoxWidget);
  300. return d->NextButton;
  301. }
  302. //-----------------------------------------------------------------------------
  303. QList<QPushButton*> ctkWorkflowButtonBoxWidget::goToButtons()const
  304. {
  305. CTK_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. CTK_D(ctkWorkflowButtonBoxWidget);
  320. // hide aspects of the button bar if specified by the current step
  321. if(ctkWorkflowAbstractWidgetStep* currentWidgetStep = dynamic_cast<ctkWorkflowAbstractWidgetStep*>(currentStep))
  322. {
  323. bool hideButtonBar = currentWidgetStep->buttonBoxHints() & ctkWorkflowAbstractWidgetStep::ButtonBoxHidden;
  324. this->setHidden(hideButtonBar);
  325. }
  326. d->updateBackButton(currentStep);
  327. d->updateNextButton(currentStep);
  328. d->updateGoToButtons(currentStep);
  329. }
  330. //--------------------------------------------------------------------------
  331. void ctkWorkflowButtonBoxWidget::prepareGoToStep()
  332. {
  333. CTK_D(ctkWorkflowButtonBoxWidget);
  334. if (QPushButton* button = qobject_cast<QPushButton*>(QObject::sender()))
  335. {
  336. if (ctkWorkflowStep* step = d->GoToButtonToStepMap.value(button))
  337. {
  338. d->Workflow->goToStep(step->id());
  339. }
  340. }
  341. }