ctkWorkflowTest2.cpp 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271
  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 <QCoreApplication>
  16. #include <QTimer>
  17. // CTK includes
  18. #include "ctkExampleWorkflowStepUsingSignalsAndSlots.h"
  19. #include "ctkWorkflowStep.h"
  20. #include "ctkWorkflow.h"
  21. // STD includes
  22. #include <cstdlib>
  23. #include <iostream>
  24. //-----------------------------------------------------------------------------
  25. int numberOfTimesEntryExitTest(ctkExampleWorkflowStepUsingSignalsAndSlots* step1=0, int step1Entry=0, int step1Exit=0, ctkExampleWorkflowStepUsingSignalsAndSlots* step2=0, int step2Entry=0, int step2Exit=0, ctkExampleWorkflowStepUsingSignalsAndSlots* step3=0, int step3Entry=0, int step3Exit=0, ctkExampleWorkflowStepUsingSignalsAndSlots* step4=0, int step4Entry=0, int step4Exit=0)
  26. {
  27. if (step1)
  28. {
  29. if (step1->numberOfTimesRanOnEntry() != step1Entry || step1->numberOfTimesRanOnExit() != step1Exit)
  30. {
  31. return 0;
  32. }
  33. }
  34. if (step2)
  35. {
  36. if (step2->numberOfTimesRanOnEntry() != step2Entry || step2->numberOfTimesRanOnExit() != step2Exit)
  37. {
  38. return 0;
  39. }
  40. }
  41. if (step3)
  42. {
  43. if (step3->numberOfTimesRanOnEntry() != step3Entry || step3->numberOfTimesRanOnExit() != step3Exit)
  44. {
  45. return 0;
  46. }
  47. }
  48. if (step4)
  49. {
  50. if (step4->numberOfTimesRanOnEntry() != step4Entry || step4->numberOfTimesRanOnExit() != step4Exit)
  51. {
  52. return 0;
  53. }
  54. }
  55. return 1;
  56. }
  57. //-----------------------------------------------------------------------------
  58. int currentStepAndNumberOfTimesEntryExitTest(ctkWorkflow* workflow, ctkWorkflowStep* expectedStep, ctkExampleWorkflowStepUsingSignalsAndSlots* step1, int step1Entry, int step1Exit, ctkExampleWorkflowStepUsingSignalsAndSlots* step2, int step2Entry, int step2Exit, ctkExampleWorkflowStepUsingSignalsAndSlots* step3=0, int step3Entry=0, int step3Exit=0, ctkExampleWorkflowStepUsingSignalsAndSlots* step4=0, int step4Entry=0, int step4Exit=0)
  59. {
  60. if (expectedStep)
  61. {
  62. if (workflow->currentStep() != expectedStep)
  63. {
  64. return 0;
  65. }
  66. }
  67. return numberOfTimesEntryExitTest(step1, step1Entry, step1Exit, step2, step2Entry, step2Exit, step3, step3Entry, step3Exit, step4, step4Entry, step4Exit);
  68. }
  69. //-----------------------------------------------------------------------------
  70. int transitionTest(ctkWorkflow* workflow, int defaultTime, ctkWorkflowStep* expectedStep, ctkExampleWorkflowStepUsingSignalsAndSlots* step1, int step1Entry, int step1Exit, ctkExampleWorkflowStepUsingSignalsAndSlots* step2, int step2Entry, int step2Exit, ctkExampleWorkflowStepUsingSignalsAndSlots* step3=0, int step3Entry=0, int step3Exit=0, ctkExampleWorkflowStepUsingSignalsAndSlots* step4=0, int step4Entry=0, int step4Exit=0)
  71. {
  72. QTimer::singleShot(defaultTime, qApp, SLOT(quit()));
  73. qApp->exec();
  74. return currentStepAndNumberOfTimesEntryExitTest(workflow, expectedStep, step1, step1Entry, step1Exit, step2, step2Entry, step2Exit, step3, step3Entry, step3Exit, step4, step4Entry, step4Exit);
  75. }
  76. //-----------------------------------------------------------------------------
  77. int testStartWorkflow(ctkWorkflow* workflow, int defaultTime, bool shouldRun, ctkWorkflowStep* expectedStep=0, ctkExampleWorkflowStepUsingSignalsAndSlots* step1=0, int step1Entry=0, int step1Exit=0, ctkExampleWorkflowStepUsingSignalsAndSlots* step2=0, int step2Entry=0, int step2Exit=0, ctkExampleWorkflowStepUsingSignalsAndSlots* step3=0, int step3Entry=0, int step3Exit=0, ctkExampleWorkflowStepUsingSignalsAndSlots* step4=0, int step4Entry=0, int step4Exit=0)
  78. {
  79. workflow->start();
  80. QTimer::singleShot(defaultTime, qApp, SLOT(quit()));
  81. qApp->exec();
  82. if (workflow->isRunning() != shouldRun)
  83. {
  84. return 0;
  85. }
  86. return currentStepAndNumberOfTimesEntryExitTest(workflow, expectedStep, step1, step1Entry, step1Exit, step2, step2Entry, step2Exit, step3, step3Entry, step3Exit, step4, step4Entry, step4Exit);
  87. }
  88. //-----------------------------------------------------------------------------
  89. int testStopWorkflow(ctkWorkflow* workflow, int defaultTime, ctkExampleWorkflowStepUsingSignalsAndSlots* step1, int step1Entry, int step1Exit, ctkExampleWorkflowStepUsingSignalsAndSlots* step2, int step2Entry, int step2Exit, ctkExampleWorkflowStepUsingSignalsAndSlots* step3=0, int step3Entry=0, int step3Exit=0, ctkExampleWorkflowStepUsingSignalsAndSlots* step4=0, int step4Entry=0, int step4Exit=0)
  90. {
  91. workflow->stop();
  92. QTimer::singleShot(defaultTime, qApp, SLOT(quit()));
  93. qApp->exec();
  94. if (workflow->isRunning())
  95. {
  96. return 0;
  97. }
  98. return numberOfTimesEntryExitTest(step1, step1Entry, step1Exit, step2, step2Entry, step2Exit, step3, step3Entry, step3Exit, step4, step4Entry, step4Exit);
  99. }
  100. //-----------------------------------------------------------------------------
  101. int ctkWorkflowTest2(int argc, char * argv [] )
  102. {
  103. QCoreApplication app(argc, argv);
  104. int defaultTime = 100;
  105. // create the steps and the workflow
  106. ctkWorkflow *workflow = new ctkWorkflow();
  107. ctkWorkflowStep *step1 = new ctkWorkflowStep("Step 1");
  108. step1->setName("Step 1");
  109. step1->setId("FirstStep");
  110. step1->setDescription("Description for step 1");
  111. ctkWorkflowStep *step2 = new ctkWorkflowStep("Step 2");
  112. step2->setName("Step 2");
  113. step2->setDescription("Description for step 2");
  114. ctkWorkflowStep *step3 = new ctkWorkflowStep("Step 3");
  115. step3->setName("Step 3");
  116. step3->setDescription("Description for step 3");
  117. ctkWorkflowStep *step4 = new ctkWorkflowStep("Step 4");
  118. step4->setName("Step 4");
  119. step4->setDescription("Description for step 4");
  120. // create the qObjects that implement the required functions, and
  121. // communicate with the workflow using signals and slots
  122. ctkExampleWorkflowStepUsingSignalsAndSlots* qObject1 = new ctkExampleWorkflowStepUsingSignalsAndSlots(step1);
  123. ctkExampleWorkflowStepUsingSignalsAndSlots* qObject2 = new ctkExampleWorkflowStepUsingSignalsAndSlots(step2);
  124. ctkExampleWorkflowStepUsingSignalsAndSlots* qObject3 = new ctkExampleWorkflowStepUsingSignalsAndSlots(step3);
  125. ctkExampleWorkflowStepUsingSignalsAndSlots* qObject4 = new ctkExampleWorkflowStepUsingSignalsAndSlots(step4);
  126. // use the qObjects for validation
  127. QObject::connect(step1->ctkWorkflowStepQObject(), SIGNAL(invokeValidateCommand(QString)),
  128. qObject1, SLOT(validate(QString)));
  129. QObject::connect(step2->ctkWorkflowStepQObject(), SIGNAL(invokeValidateCommand(QString)),
  130. qObject2, SLOT(validate(QString)));
  131. // step 3's validation will always fail
  132. QObject::connect(step3->ctkWorkflowStepQObject(), SIGNAL(invokeValidateCommand(QString)),
  133. qObject3, SLOT(validateFails()));
  134. QObject::connect(step4->ctkWorkflowStepQObject(), SIGNAL(invokeValidateCommand(QString)),
  135. qObject4, SLOT(validate(QString)));
  136. // use the qObjects for entry processing
  137. QObject::connect(
  138. step1->ctkWorkflowStepQObject(), SIGNAL(invokeOnEntryCommand(const ctkWorkflowStep*,ctkWorkflowInterstepTransition::InterstepTransitionType)),
  139. qObject1, SLOT(onEntry(const ctkWorkflowStep*,ctkWorkflowInterstepTransition::InterstepTransitionType)));
  140. QObject::connect(
  141. step2->ctkWorkflowStepQObject(), SIGNAL(invokeOnEntryCommand(const ctkWorkflowStep*,ctkWorkflowInterstepTransition::InterstepTransitionType)),
  142. qObject2, SLOT(onEntry(const ctkWorkflowStep*,ctkWorkflowInterstepTransition::InterstepTransitionType)));
  143. QObject::connect(
  144. step3->ctkWorkflowStepQObject(), SIGNAL(invokeOnEntryCommand(const ctkWorkflowStep*,ctkWorkflowInterstepTransition::InterstepTransitionType)),
  145. qObject3, SLOT(onEntry(const ctkWorkflowStep*,ctkWorkflowInterstepTransition::InterstepTransitionType)));
  146. QObject::connect(
  147. step4->ctkWorkflowStepQObject(), SIGNAL(invokeOnEntryCommand(const ctkWorkflowStep*,ctkWorkflowInterstepTransition::InterstepTransitionType)),
  148. qObject4, SLOT(onEntry(const ctkWorkflowStep*,ctkWorkflowInterstepTransition::InterstepTransitionType)));
  149. // use the qObjects for exit processing
  150. QObject::connect(
  151. step1->ctkWorkflowStepQObject(), SIGNAL(invokeOnExitCommand(const ctkWorkflowStep*,ctkWorkflowInterstepTransition::InterstepTransitionType)),
  152. qObject1, SLOT(onExit(const ctkWorkflowStep*,ctkWorkflowInterstepTransition::InterstepTransitionType)));
  153. QObject::connect(
  154. step2->ctkWorkflowStepQObject(), SIGNAL(invokeOnExitCommand(const ctkWorkflowStep*,ctkWorkflowInterstepTransition::InterstepTransitionType)),
  155. qObject2, SLOT(onExit(const ctkWorkflowStep*,ctkWorkflowInterstepTransition::InterstepTransitionType)));
  156. QObject::connect(
  157. step3->ctkWorkflowStepQObject(), SIGNAL(invokeOnExitCommand(const ctkWorkflowStep*,ctkWorkflowInterstepTransition::InterstepTransitionType)),
  158. qObject3, SLOT(onExit(const ctkWorkflowStep*,ctkWorkflowInterstepTransition::InterstepTransitionType)));
  159. QObject::connect(
  160. step4->ctkWorkflowStepQObject(), SIGNAL(invokeOnExitCommand(const ctkWorkflowStep*,ctkWorkflowInterstepTransition::InterstepTransitionType)),
  161. qObject4, SLOT(onExit(const ctkWorkflowStep*,ctkWorkflowInterstepTransition::InterstepTransitionType)));
  162. step1->setHasValidateCommand(1);
  163. step2->setHasValidateCommand(1);
  164. step3->setHasValidateCommand(1);
  165. step4->setHasValidateCommand(1);
  166. step1->setHasOnEntryCommand(1);
  167. step2->setHasOnEntryCommand(1);
  168. step3->setHasOnEntryCommand(1);
  169. step4->setHasOnEntryCommand(1);
  170. step1->setHasOnExitCommand(1);
  171. step2->setHasOnExitCommand(1);
  172. step3->setHasOnExitCommand(1);
  173. step4->setHasOnExitCommand(1);
  174. // set the initial step (which sets the initial state)
  175. workflow->setInitialStep(step1);
  176. // add the first and second steps
  177. if (!workflow->addTransition(step1, step2))
  178. {
  179. std::cerr << "could not add 1st and 2nd step" << std::endl;
  180. return EXIT_FAILURE;
  181. }
  182. // add the third step
  183. if (!workflow->addTransition(step2, step3))
  184. {
  185. std::cerr << "could not add 3rd step" << std::endl;
  186. return EXIT_FAILURE;
  187. }
  188. // add the fourth step
  189. if (!workflow->addTransition(step3, step4))
  190. {
  191. std::cerr << "could not add 4rd step" << std::endl;
  192. return EXIT_FAILURE;
  193. }
  194. // start the workflow
  195. if (!testStartWorkflow(workflow, defaultTime, 1, step1, qObject1, true, 0, qObject2, 0, 0, qObject3, 0, 0, qObject4, 0, 0))
  196. {
  197. std::cerr << "error starting workflow";
  198. return EXIT_FAILURE;
  199. }
  200. // trigger transition to the next step
  201. workflow->goForward();
  202. if (!transitionTest(workflow, defaultTime, step2, qObject1, 1, 1, qObject2, 1, 0, qObject3, 0, 0, qObject4, 0, 0))
  203. {
  204. std::cerr << "error transitioning to step 2";
  205. return EXIT_FAILURE;
  206. }
  207. // trigger transition to the next step
  208. workflow->goForward();
  209. if (!transitionTest(workflow, defaultTime, step3, qObject1, 1, 1, qObject2, 1, 1, qObject3, 1, 0, qObject4, 0, 0))
  210. {
  211. std::cerr << "error transitioning to step 3";
  212. return EXIT_FAILURE;
  213. }
  214. // trigger transition to the next state (this should fail!)
  215. workflow->goForward();
  216. if (!transitionTest(workflow, defaultTime, step3, qObject1, 1, 1, qObject2, 1, 1, qObject3, 1, 0, qObject4, 0, 0))
  217. {
  218. std::cerr << "error after transition failure at step 3";
  219. return EXIT_FAILURE;
  220. }
  221. // make sure the workflow stops properly
  222. if (!testStopWorkflow(workflow, defaultTime, qObject1, 1, 1, qObject2, 1, 1, qObject3, 1, 1, qObject4, 0, 0))
  223. {
  224. std::cerr << "error after stopping workflow";
  225. return EXIT_FAILURE;
  226. }
  227. // handles deletions of the workflow, steps, states and transitions
  228. delete workflow;
  229. return EXIT_SUCCESS;
  230. }