ui.extkeywidget.class.inc.php 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267
  1. <?php
  2. // Copyright (C) 2010 Combodo SARL
  3. //
  4. // This program is free software; you can redistribute it and/or modify
  5. // it under the terms of the GNU General Public License as published by
  6. // the Free Software Foundation; version 3 of the License.
  7. //
  8. // This program is distributed in the hope that it will be useful,
  9. // but WITHOUT ANY WARRANTY; without even the implied warranty of
  10. // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  11. // GNU General Public License for more details.
  12. //
  13. // You should have received a copy of the GNU General Public License
  14. // along with this program; if not, write to the Free Software
  15. // Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
  16. /**
  17. * Class UIExtKeyWidget
  18. * UI wdiget for displaying and editing external keys when
  19. * A simple drop-down list is not enough...
  20. *
  21. * The layout is the following
  22. *
  23. * +-- #label_<id> (input)-------+ +-----------+
  24. * | | | Browse... |
  25. * +-----------------------------+ +-----------+
  26. *
  27. * And the popup dialog has the following layout:
  28. *
  29. * +------------------- ac_dlg_<id> (div)-----------+
  30. * + +--- ds_<id> (div)---------------------------+ |
  31. * | | +------------- fs_<id> (form)------------+ | |
  32. * | | | +--------+---+ | | |
  33. * | | | | Class | V | | | |
  34. * | | | +--------+---+ | | |
  35. * | | | | | |
  36. * | | | S e a r c h F o r m | | |
  37. * | | | +--------+ | | |
  38. * | | | | Search | | | |
  39. * | | | +--------+ | | |
  40. * | | +----------------------------------------+ | |
  41. * | +--------------+-dh_<id>-+--------------------+ |
  42. * | \ Search / |
  43. * | +------+ |
  44. * | +--- fr_<id> (form)--------------------------+ |
  45. * | | +------------ dr_<id> (div)--------------+ | |
  46. * | | | | | |
  47. * | | | S e a r c h R e s u l t s | | |
  48. * | | | | | |
  49. * | | +----------------------------------------+ | |
  50. * | | +--------+ +-----+ | |
  51. * | | | Cancel | | Add | | |
  52. * | | +--------+ +-----+ | |
  53. * | +--------------------------------------------+ |
  54. * +------------------------------------------------+
  55. * @author Erwan Taloc <erwan.taloc@combodo.com>
  56. * @author Romain Quetiez <romain.quetiez@combodo.com>
  57. * @author Denis Flaven <denis.flaven@combodo.com>
  58. * @license http://www.opensource.org/licenses/gpl-3.0.html LGPL
  59. */
  60. require_once(APPROOT.'/application/webpage.class.inc.php');
  61. require_once(APPROOT.'/application/displayblock.class.inc.php');
  62. class UIExtKeyWidget
  63. {
  64. protected static $iWidgetIndex = 0;
  65. protected $sAttCode;
  66. protected $sNameSuffix;
  67. protected $iId;
  68. protected $sTitle;
  69. public function __construct($sAttCode, $sClass, $sTitle, $aAllowedValues, $value, $iInputId, $bMandatory, $sNameSuffix = '', $sFieldPrefix = '', $sFormPrefix = '')
  70. {
  71. self::$iWidgetIndex++;
  72. $this->sAttCode = $sAttCode;
  73. $this->sClass = $sClass;
  74. $this->oAttDef = MetaModel::GetAttributeDef($sClass, $sAttCode);
  75. $this->sNameSuffix = $sNameSuffix;
  76. $this->iId = $iInputId;
  77. $this->aAllowedValues = $aAllowedValues;
  78. $this->value = $value;
  79. $this->sFieldPrefix = $sFieldPrefix;
  80. $this->sTargetClass = $this->oAttDef->GetTargetClass();
  81. $this->sTitle = $sTitle;
  82. $this->sFormPrefix = $sFormPrefix;
  83. $this->bMandatory = $bMandatory;
  84. }
  85. /**
  86. * Get the HTML fragment corresponding to the linkset editing widget
  87. * @param WebPage $oP The web page used for all the output
  88. * @param Hash $aArgs Extra context arguments
  89. * @return string The HTML fragment to be inserted into the page
  90. */
  91. public function Display(WebPage $oPage, $aArgs = array())
  92. {
  93. $bCreate = (UserRights::IsActionAllowed($this->sTargetClass, UR_ACTION_BULK_MODIFY) && $this->oAttDef->AllowTargetCreation());
  94. if (count($this->aAllowedValues) < $this->oAttDef->GetMaximumComboLength())
  95. {
  96. // Few choices, use a normal 'select'
  97. $sSelectMode = 'true';
  98. $sHelpText = $this->oAttDef->GetHelpOnEdition();
  99. // In case there are no valid values, the select will be empty, thus blocking the user from validating the form
  100. $sHTMLValue = "<select title=\"$sHelpText\" name=\"attr_{$this->sFieldPrefix}{$this->sAttCode}{$this->sNameSuffix}\" id=\"$this->iId\">\n";
  101. $sHTMLValue .= "<option value=\"\">".Dict::S('UI:SelectOne')."</option>\n";
  102. foreach($this->aAllowedValues as $key => $display_value)
  103. {
  104. if ((count($this->aAllowedValues) == 1) && ($this->bMandatory == 'true') )
  105. {
  106. // When there is only once choice, select it by default
  107. $sSelected = ' selected';
  108. }
  109. else
  110. {
  111. $sSelected = ($this->value == $key) ? ' selected' : '';
  112. }
  113. $sHTMLValue .= "<option value=\"$key\"$sSelected>$display_value</option>\n";
  114. }
  115. $sHTMLValue .= "</select>\n";
  116. }
  117. else
  118. {
  119. // Too many choices, use an autocomplete
  120. $sSelectMode = 'false';
  121. if ($this->oAttDef->IsNull($this->value)) // Null values are displayed as ''
  122. {
  123. $sDisplayValue = '';
  124. }
  125. else
  126. {
  127. $sDisplayValue = $this->GetObjectName($this->value);
  128. }
  129. $sFormPrefix = $this->sFormPrefix;
  130. $iMinChars = $this->oAttDef->GetMinAutoCompleteChars();
  131. $iFieldSize = $this->oAttDef->GetMaxSize();
  132. // the input for the auto-complete
  133. $sHTMLValue = "<input count=\"".count($this->aAllowedValues)."\" type=\"text\" id=\"label_$this->iId\" size=\"30\" maxlength=\"$iFieldSize\" value=\"$sDisplayValue\"/>&nbsp;";
  134. $sHTMLValue .= "<a class=\"no-arrow\" href=\"javascript:oACWidget_{$this->iId}.Search();\"><img style=\"border:0;vertical-align:middle;\" src=\"../images/mini_search.gif\" /></a>&nbsp;";
  135. // another hidden input to store & pass the object's Id
  136. $sHTMLValue .= "<input type=\"hidden\" id=\"$this->iId\" name=\"attr_{$this->sFieldPrefix}{$this->sAttCode}{$this->sNameSuffix}\" value=\"$this->value\" />\n";
  137. // Scripts to start the autocomplete and bind some events to it
  138. $oPage->add_ready_script("\$('#label_$this->iId').autocomplete('./ajax.render.php', { scroll:true, minChars:{$iMinChars}, formatItem:formatItem, autoFill:false, matchContains:true, keyHolder:'#{$this->iId}', extraParams:{operation:'autocomplete', sclass:'$this->sClass',attCode:'".$this->sAttCode."'}});");
  139. $oPage->add_ready_script("\$('#label_$this->iId').blur(function() { $(this).search(); } );");
  140. $oPage->add_ready_script("\$('#label_$this->iId').result( function(event, data, formatted) { OnAutoComplete('$this->iId', event, data, formatted); } );");
  141. $oPage->add_ready_script("\$('#ac_dlg_$this->iId').dialog({ width: $(window).width()*0.8, height: $(window).height()*0.8, autoOpen: false, modal: true, title: '$this->sTitle', resizeStop: oACWidget_{$this->iId}.UpdateSizes, close: oACWidget_{$this->iId}.OnClose });\n");
  142. $oPage->add_at_the_end($this->GetSearchDialog($oPage)); // To prevent adding forms inside the main form
  143. }
  144. if ($bCreate)
  145. {
  146. $sHTMLValue .= "<a class=\"no-arrow\" href=\"javascript:oACWidget_{$this->iId}.CreateObject();\"><img style=\"border:0;vertical-align:middle;\" src=\"../images/mini_add.gif\" /></a>&nbsp;";
  147. $oPage->add_at_the_end('<div id="ajax_'.$this->iId.'"></div>');
  148. }
  149. $sMessage = Dict::S('UI:Message:EmptyList:UseSearchForm');
  150. $sHTMLValue .= "<span id=\"v_{$this->iId}\"></span>";
  151. $oPage->add_ready_script(
  152. <<<EOF
  153. oACWidget_{$this->iId} = new ExtKeyWidget('$this->iId', '$this->sClass', '$this->sAttCode', '$this->sNameSuffix', $sSelectMode, oWizardHelper{$this->sFormPrefix});
  154. oACWidget_{$this->iId}.emptyHtml = "<div style=\"background: #fff; border:0; text-align:center; vertical-align:middle;\"><p>$sMessage</p></div>";
  155. EOF
  156. );
  157. return $sHTMLValue;
  158. }
  159. protected function GetSearchDialog(WebPage $oPage)
  160. {
  161. $sHTML = '<div id="ac_dlg_'.$this->iId.'"><div class="wizContainer" style="vertical-align:top;"><div id="dc_'.$this->iId.'">';
  162. $oFilter = new DBObjectSearch($this->sTargetClass);
  163. $oSet = new CMDBObjectSet($oFilter);
  164. $oBlock = new DisplayBlock($oFilter, 'search', false);
  165. $sHTML .= $oBlock->GetDisplay($oPage, $this->iId, array('open' => true, 'currentId' => $this->iId));
  166. $sHTML .= "<form id=\"fr_{$this->iId}\" OnSubmit=\"return oACWidget_{$this->iId}.DoOk();\">\n";
  167. $sHTML .= "<div id=\"dr_{$this->iId}\" style=\"vertical-align:top;background: #fff;height:100%;overflow:auto;padding:0;border:0;\">\n";
  168. $sHTML .= "<div style=\"background: #fff; border:0; text-align:center; vertical-align:middle;\"><p>".Dict::S('UI:Message:EmptyList:UseSearchForm')."</p></div>\n";
  169. $sHTML .= "</div>\n";
  170. $sHTML .= "<input type=\"button\" id=\"btn_cancel_{$this->iId}\" value=\"".Dict::S('UI:Button:Cancel')."\" onClick=\"$('#ac_dlg_{$this->iId}').dialog('close');\">&nbsp;&nbsp;";
  171. $sHTML .= "<input type=\"button\" id=\"btn_ok_{$this->iId}\" value=\"".Dict::S('UI:Button:Ok')."\" onClick=\"oACWidget_{$this->iId}.DoOk();\">";
  172. $sHTML .= "</form>\n";
  173. $sHTML .= '</div></div></div>';
  174. $oPage->add_ready_script("$('#fs_{$this->iId}').bind('submit.uiAutocomplete', oACWidget_{$this->iId}.DoSearchObjects);");
  175. $oPage->add_ready_script("$('#dc_{$this->iId}').resize(oACWidget_{$this->iId}.UpdateSizes);");
  176. return $sHTML;
  177. }
  178. /**
  179. * Search for objects to be selected
  180. * @param WebPage $oP The page used for the output (usually an AjaxWebPage)
  181. * @param string $sRemoteClass Name of the "remote" class to perform the search on, must be a derived class of m_sRemoteClass
  182. * @param Array $aAlreadyLinkedIds List of IDs of objects of "remote" class already linked, to be filtered out of the search
  183. */
  184. public function SearchObjectsToSelect(WebPage $oP, $sTargetClass = '')
  185. {
  186. if ($sTargetClass != '')
  187. {
  188. // assert(MetaModel::IsParentClass($this->m_sRemoteClass, $sRemoteClass));
  189. $oFilter = new DBObjectSearch($sTargetClass);
  190. }
  191. else
  192. {
  193. // No remote class specified use the one defined in the linkedset
  194. $oFilter = new DBObjectSearch($this->sTargetClass);
  195. }
  196. $oFilter->AddCondition('id', array_keys($this->aAllowedValues), 'IN');
  197. $oSet = new CMDBObjectSet($oFilter);
  198. $oBlock = new DisplayBlock($oFilter, 'list', false);
  199. $oBlock->Display($oP, $this->iId, array('menu' => false, 'selection_mode' => true, 'selection_type' => 'single', 'display_limit' => false)); // Don't display the 'Actions' menu on the results
  200. }
  201. /**
  202. * Get the display name of the selected object, to fill back the autocomplete
  203. */
  204. public function GetObjectName($iObjId)
  205. {
  206. $oObj = MetaModel::GetObject($this->sTargetClass, $iObjId);
  207. return $oObj->GetName();
  208. }
  209. /**
  210. * Get the form to create a new object of the 'target' class
  211. */
  212. public function GetObjectCreationForm(WebPage $oPage)
  213. {
  214. $oPage->add('<div id="ac_create_'.$this->iId.'"><div class="wizContainer" style="vertical-align:top;"><div id="dcr_'.$this->iId.'">');
  215. $oPage->add("<h1>".MetaModel::GetClassIcon($this->sTargetClass)."&nbsp;".Dict::Format('UI:CreationTitle_Class', MetaModel::GetName($this->sTargetClass))."</h1>\n");
  216. cmdbAbstractObject::DisplayCreationForm($oPage, $this->sTargetClass, null, array(), array('formPrefix' => $this->iId, 'noRelations' => true));
  217. $oPage->add('</div></div></div>');
  218. $oPage->add_ready_script("\$('#ac_create_$this->iId').dialog({ width: $(window).width()*0.8, height: 'auto', autoOpen: false, modal: true, title: '$this->sTitle'});\n");
  219. $oPage->add_ready_script("$('#dcr_{$this->iId} form').removeAttr('onsubmit');");
  220. $oPage->add_ready_script("$('#dcr_{$this->iId} form').bind('submit.uilinksWizard', oACWidget_{$this->iId}.DoCreateObject);");
  221. }
  222. /**
  223. * Get the form to create a new object of the 'target' class
  224. */
  225. public function DoCreateObject($oPage)
  226. {
  227. $oObj = MetaModel::NewObject($this->sTargetClass);
  228. $oObj->UpdateObject($this->sFormPrefix.$this->iId);
  229. $oMyChange = MetaModel::NewObject("CMDBChange");
  230. $oMyChange->Set("date", time());
  231. if (UserRights::IsImpersonated())
  232. {
  233. $sUserString = Dict::Format('UI:Archive_User_OnBehalfOf_User', UserRights::GetRealUser(), UserRights::GetUser());
  234. }
  235. else
  236. {
  237. $sUserString = UserRights::GetUser();
  238. }
  239. $oMyChange->Set("userinfo", $sUserString);
  240. $iChangeId = $oMyChange->DBInsert();
  241. $oObj->DBInsertTracked($oMyChange);
  242. return array('name' => $oObj->GetName(), 'id' => $oObj->GetKey());
  243. //return array('name' => 'test', 'id' => '42');
  244. }
  245. }
  246. ?>