uiwizard.class.inc.php 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297
  1. <?php
  2. class UIWizard
  3. {
  4. protected $m_oPage;
  5. protected $m_sClass;
  6. protected $m_sTargetState;
  7. protected $m_aWizardSteps;
  8. public function __construct($oPage, $sClass, $sTargetState = '')
  9. {
  10. $this->m_oPage = $oPage;
  11. $this->m_sClass = $sClass;
  12. if (empty($sTargetState))
  13. {
  14. $sTargetState = MetaModel::GetDefaultState($sClass);
  15. }
  16. $this->m_sTargetState = $sTargetState;
  17. $this->m_aWizardSteps = $this->ComputeWizardStructure();
  18. }
  19. public function GetObjectClass() { return $this->m_sClass; }
  20. public function GetTargetState() { return $this->m_sTargetState; }
  21. public function GetWizardStructure() { return $this->m_aWizardSteps; }
  22. /**
  23. * Displays one step of the wizard
  24. */
  25. public function DisplayWizardStep($aStep, $iStepIndex, &$iMaxInputId, &$aFieldsMap, $bFinishEnabled = false, $aArgs = array())
  26. {
  27. if ($iStepIndex == 1) // one big form that contains everything, to make sure that the uploaded files are posted too
  28. {
  29. $this->m_oPage->add("<form method=\"post\" enctype=\"multipart/form-data\" action=\"../pages/UI.php\">\n");
  30. }
  31. $this->m_oPage->add("<div class=\"wizContainer\" id=\"wizStep$iStepIndex\" style=\"display:none;\">\n");
  32. $this->m_oPage->add("<a name=\"step$iStepIndex\" />\n");
  33. $aStates = MetaModel::EnumStates($this->m_sClass);
  34. $aDetails = array();
  35. $sJSHandlerCode = ''; // Javascript code to be executed each time this step of the wizard is entered
  36. foreach($aStep as $sAttCode)
  37. {
  38. if ($sAttCode != 'finalclass') // Do not display the attribute that stores the actual class name
  39. {
  40. $oAttDef = MetaModel::GetAttributeDef($this->m_sClass, $sAttCode);
  41. $sAttLabel = $oAttDef->GetLabel();
  42. $iOptions = isset($aStates[$this->m_sTargetState]['attribute_list'][$sAttCode]) ? $aStates[$this->m_sTargetState]['attribute_list'][$sAttCode] : 0;
  43. $aPrerequisites = $oAttDef->GetPrerequisiteAttributes();
  44. if ($iOptions & (OPT_ATT_MANDATORY | OPT_ATT_MUSTCHANGE | OPT_ATT_MUSTPROMPT))
  45. {
  46. $aFields[$sAttCode] = array();
  47. foreach($aPrerequisites as $sCode)
  48. {
  49. $aFields[$sAttCode][$sCode] = '';
  50. }
  51. }
  52. if (count($aPrerequisites) > 0)
  53. {
  54. $aOptions[] = 'Prerequisites: '.implode(', ', $aPrerequisites);
  55. }
  56. $sFieldFlag = (($iOptions & (OPT_ATT_MANDATORY | OPT_ATT_MUSTCHANGE)) || (!$oAttDef->IsNullAllowed()) )? ' <span class="hilite">*</span>' : '';
  57. $oDefaultValuesSet = $oAttDef->GetDefaultValue(/* $oObject->ToArgs() */); // @@@ TO DO: get the object's current value if the object exists
  58. $sHTMLValue = cmdbAbstractObject::GetFormElementForField($this->m_oPage, $this->m_sClass, $sAttCode, $oAttDef, $oDefaultValuesSet, '', "att_$iMaxInputId", '', $iOptions, $aArgs);
  59. $aFieldsMap[$iMaxInputId] = $sAttCode;
  60. $aDetails[] = array('label' => $oAttDef->GetLabel().$sFieldFlag, 'value' => "<div id=\"field_$iMaxInputId\">$sHTMLValue</div>");
  61. if ($oAttDef->GetValuesDef() != null)
  62. {
  63. $sJSHandlerCode .= "\toWizardHelper.RequestAllowedValues('$sAttCode');\n";
  64. }
  65. if ($oAttDef->GetDefaultValue() != null)
  66. {
  67. $sJSHandlerCode .= "\toWizardHelper.RequestDefaultValue('$sAttCode');\n";
  68. }
  69. if ($oAttDef->IsLinkSet())
  70. {
  71. $sJSHandlerCode .= "\toLinkWidgetatt_$iMaxInputId.Init();";
  72. }
  73. $iMaxInputId++;
  74. }
  75. }
  76. //$aDetails[] = array('label' => '', 'value' => '<input type="button" value="Next &gt;&gt;">');
  77. $this->m_oPage->details($aDetails);
  78. $sBackButtonDisabled = ($iStepIndex <= 1) ? 'disabled' : '';
  79. $sDisabled = $bFinishEnabled ? '' : 'disabled';
  80. $nbSteps = count($this->m_aWizardSteps['mandatory']) + count($this->m_aWizardSteps['optional']);
  81. $this->m_oPage->add("<div style=\"text-align:center\">
  82. <input type=\"button\" value=\"&lt;&lt; Back \" $sBackButtonDisabled onClick=\"GoToStep($iStepIndex, $iStepIndex - 1)\" />
  83. <input type=\"button\" value=\" Next &gt;&gt;\" onClick=\"GoToStep($iStepIndex, 1+$iStepIndex)\" />
  84. <input type=\"button\" value=\" Finish \" $sDisabled onClick=\"GoToStep($iStepIndex, 1+$nbSteps)\" />
  85. </div>\n");
  86. $this->m_oPage->add("
  87. <script type=\"text/javascript\">
  88. function OnEnterStep{$iStepIndex}()
  89. {
  90. oWizardHelper.ResetQuery();
  91. oWizardHelper.UpdateWizard();
  92. $sJSHandlerCode
  93. oWizardHelper.AjaxQueryServer();
  94. }
  95. </script>\n");
  96. $this->m_oPage->add("</div>\n\n");
  97. }
  98. /**
  99. * Display the final step of the wizard: a confirmation screen
  100. */
  101. public function DisplayFinalStep($iStepIndex, $aFieldsMap)
  102. {
  103. $oAppContext = new ApplicationContext();
  104. $this->m_oPage->add("<div class=\"wizContainer\" id=\"wizStep$iStepIndex\" style=\"display:none;\">\n");
  105. $this->m_oPage->add("<a name=\"step$iStepIndex\" />\n");
  106. $this->m_oPage->P("Final step: confirmation");
  107. $this->m_oPage->add("<input type=\"hidden\" name=\"operation\" value=\"wizard_apply_new\" />\n");
  108. $this->m_oPage->add("<input type=\"hidden\" name=\"transaction_id\" value=\"".utils::GetNewTransactionId()."\" />\n");
  109. $this->m_oPage->add("<input type=\"hidden\" id=\"wizard_json_obj\" name=\"json_obj\" value=\"\" />\n");
  110. $this->m_oPage->add("<script type=\"text/javascript\">\n");
  111. $this->m_oPage->add("function OnEnterStep$iStepIndex() {\n");
  112. foreach($aFieldsMap as $iInputId => $sAttCode)
  113. {
  114. $this->m_oPage->add("\toWizardHelper.UpdateCurrentValue('$sAttCode');\n");
  115. }
  116. $this->m_oPage->add("\toWizardHelper.Preview('object_preview');\n");
  117. $this->m_oPage->add("\t$('#wizard_json_obj').val(oWizardHelper.ToJSON());\n");
  118. $this->m_oPage->add("}\n");
  119. $this->m_oPage->add("</script>\n");
  120. $this->m_oPage->add("<div id=\"object_preview\">\n");
  121. $this->m_oPage->add("</div>\n");
  122. $this->m_oPage->add($oAppContext->GetForForm());
  123. $this->m_oPage->add("<input type=\"submit\" value=\"Create ".MetaModel::GetName($this->m_sClass)."\" />\n");
  124. $this->m_oPage->add("</div>\n");
  125. $this->m_oPage->add("</form>\n");
  126. }
  127. /**
  128. * Compute the order of the fields & pages in the wizard
  129. * @param $oPage iTopWebPage The current page (used to display error messages)
  130. * @param $sClass string Name of the class
  131. * @param $sStateCode string Code of the target state of the object
  132. * @return hash Two dimensional array: each element represents the list of fields for a given page
  133. */
  134. protected function ComputeWizardStructure()
  135. {
  136. $aWizardSteps = array( 'mandatory' => array(), 'optional' => array());
  137. $aFieldsDone = array(); // Store all the fields that are already covered by a previous step of the wizard
  138. $aStates = MetaModel::EnumStates($this->m_sClass);
  139. $sStateAttCode = MetaModel::GetStateAttributeCode($this->m_sClass);
  140. $aMandatoryAttributes = array();
  141. // Some attributes are always mandatory independently of the state machine (if any)
  142. foreach(MetaModel::GetAttributesList($this->m_sClass) as $sAttCode)
  143. {
  144. $oAttDef = MetaModel::GetAttributeDef($this->m_sClass, $sAttCode);
  145. if (!$oAttDef->IsExternalField() && !$oAttDef->IsNullAllowed() &&
  146. ($sAttCode != 'finalclass') && ($sAttCode != $sStateAttCode) )
  147. {
  148. $aMandatoryAttributes[$sAttCode] = OPT_ATT_MANDATORY;
  149. }
  150. }
  151. // Now check the attributes that are mandatory in the specified state
  152. if ( (!empty($this->m_sTargetState)) && (count($aStates[$this->m_sTargetState]['attribute_list']) > 0) )
  153. {
  154. // Check all the fields that *must* be included in the wizard for this
  155. // particular target state
  156. $aFields = array();
  157. foreach($aStates[$this->m_sTargetState]['attribute_list'] as $sAttCode => $iOptions)
  158. {
  159. if ( (isset($aMandatoryAttributes[$sAttCode])) &&
  160. ($aMandatoryAttributes[$sAttCode] & (OPT_ATT_MANDATORY | OPT_ATT_MUSTCHANGE | OPT_ATT_MUSTPROMPT)) )
  161. {
  162. $aMandatoryAttributes[$sAttCode] |= $iOptions;
  163. }
  164. else
  165. {
  166. $aMandatoryAttributes[$sAttCode] = $iOptions;
  167. }
  168. }
  169. }
  170. // Check all the fields that *must* be included in the wizard
  171. // i.e. all mandatory, must-change or must-prompt fields that are
  172. // not also read-only or hidden.
  173. // Some fields may be required (null not allowed) from the database
  174. // perspective, but hidden or read-only from the user interface perspective
  175. $aFields = array();
  176. foreach($aMandatoryAttributes as $sAttCode => $iOptions)
  177. {
  178. if ( ($iOptions & (OPT_ATT_MANDATORY | OPT_ATT_MUSTCHANGE | OPT_ATT_MUSTPROMPT)) &&
  179. !($iOptions & (OPT_ATT_READONLY | OPT_ATT_HIDDEN)) )
  180. {
  181. $oAttDef = MetaModel::GetAttributeDef($this->m_sClass, $sAttCode);
  182. $aPrerequisites = $oAttDef->GetPrerequisiteAttributes();
  183. $aFields[$sAttCode] = array();
  184. foreach($aPrerequisites as $sCode)
  185. {
  186. $aFields[$sAttCode][$sCode] = '';
  187. }
  188. }
  189. }
  190. // Now use the dependencies between the fields to order them
  191. while(count($aFields) > 0)
  192. {
  193. $aCurrentStep = array();
  194. foreach($aFields as $sAttCode => $aDependencies)
  195. {
  196. // All fields with no remaining dependencies can be entered at this
  197. // step of the wizard
  198. if (count($aDependencies) == 0)
  199. {
  200. $aCurrentStep[] = $sAttCode;
  201. $aFieldsDone[$sAttCode] = '';
  202. unset($aFields[$sAttCode]);
  203. // Remove this field from the dependencies of the other fields
  204. foreach($aFields as $sUpdatedCode => $aDummy)
  205. {
  206. // remove the dependency
  207. unset($aFields[$sUpdatedCode][$sAttCode]);
  208. }
  209. }
  210. }
  211. if (count($aCurrentStep) == 0)
  212. {
  213. // This step of the wizard would contain NO field !
  214. echo "<strong>Error:</strong> Circular reference in the dependencies between the fields.";
  215. print_r($aFields);
  216. break;
  217. }
  218. $aWizardSteps['mandatory'][] = $aCurrentStep;
  219. }
  220. // Now computes the steps to fill the optional fields
  221. $aFields = array(); // reset
  222. foreach(MetaModel::ListAttributeDefs($this->m_sClass) as $sAttCode=>$oAttDef)
  223. {
  224. $iOptions = (isset($aStates[$this->m_sTargetState]['attribute_list'][$sAttCode])) ? $aStates[$this->m_sTargetState]['attribute_list'][$sAttCode] : 0;
  225. if ( ($sStateAttCode != $sAttCode) &&
  226. (!$oAttDef->IsExternalField()) &&
  227. (($iOptions & (OPT_ATT_HIDDEN | OPT_ATT_READONLY)) == 0) &&
  228. (!isset($aFieldsDone[$sAttCode])) )
  229. {
  230. // 'State', external fields, read-only and hidden fields
  231. // and fields that are already listed in the wizard
  232. // are removed from the 'optional' part of the wizard
  233. $oAttDef = MetaModel::GetAttributeDef($this->m_sClass, $sAttCode);
  234. $aPrerequisites = $oAttDef->GetPrerequisiteAttributes();
  235. $aFields[$sAttCode] = array();
  236. foreach($aPrerequisites as $sCode)
  237. {
  238. if (!isset($aFieldsDone[$sCode]))
  239. {
  240. // retain only the dependencies that were not covered
  241. // in the 'mandatory' part of the wizard
  242. $aFields[$sAttCode][$sCode] = '';
  243. }
  244. }
  245. }
  246. }
  247. // Now use the dependencies between the fields to order them
  248. while(count($aFields) > 0)
  249. {
  250. $aCurrentStep = array();
  251. foreach($aFields as $sAttCode => $aDependencies)
  252. {
  253. // All fields with no remaining dependencies can be entered at this
  254. // step of the wizard
  255. if (count($aDependencies) == 0)
  256. {
  257. $aCurrentStep[] = $sAttCode;
  258. $aFieldsDone[$sAttCode] = '';
  259. unset($aFields[$sAttCode]);
  260. // Remove this field from the dependencies of the other fields
  261. foreach($aFields as $sUpdatedCode => $aDummy)
  262. {
  263. // remove the dependency
  264. unset($aFields[$sUpdatedCode][$sAttCode]);
  265. }
  266. }
  267. }
  268. if (count($aCurrentStep) == 0)
  269. {
  270. // This step of the wizard would contain NO field !
  271. $this->m_oPage->add("<strong>Error:</strong> Circular reference in the dependencies between the fields.");
  272. print_r($aFields);
  273. break;
  274. }
  275. $aWizardSteps['optional'][] = $aCurrentStep;
  276. }
  277. return $aWizardSteps;
  278. }
  279. }
  280. ?>