ui.extkeywidget.class.inc.php 16 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349
  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 $iId;
  65. protected $sTargetClass;
  66. //public function __construct($sAttCode, $sClass, $sTitle, $oAllowedValues, $value, $iInputId, $bMandatory, $sNameSuffix = '', $sFieldPrefix = '', $sFormPrefix = '')
  67. static public function DisplayFromAttCode($oPage, $sAttCode, $sClass, $sTitle, $oAllowedValues, $value, $iInputId, $bMandatory, $sFieldName = '', $sFormPrefix = '', $aArgs, $bSearchMode = false)
  68. {
  69. $oAttDef = MetaModel::GetAttributeDef($sClass, $sAttCode);
  70. $sTargetClass = $oAttDef->GetTargetClass();
  71. $iMaxComboLength = $oAttDef->GetMaximumComboLength();
  72. $bAllowTargetCreation = $oAttDef->AllowTargetCreation();
  73. $oWidget = new UIExtKeyWidget($sTargetClass, $iInputId);
  74. return $oWidget->Display($oPage, $iMaxComboLength, $bAllowTargetCreation, $sTitle, $oAllowedValues, $value, $iInputId, $bMandatory, $sFieldName, $sFormPrefix, $aArgs, $bSearchMode);
  75. }
  76. public function __construct($sTargetClass, $iInputId)
  77. {
  78. $this->sTargetClass = $sTargetClass;
  79. $this->iId = $iInputId;
  80. }
  81. /**
  82. * Get the HTML fragment corresponding to the linkset editing widget
  83. * @param WebPage $oP The web page used for all the output
  84. * @param Hash $aArgs Extra context arguments
  85. * @return string The HTML fragment to be inserted into the page
  86. */
  87. public function Display(WebPage $oPage, $iMaxComboLength, $bAllowTargetCreation, $sTitle, $oAllowedValues, $value, $iInputId, $bMandatory, $sFieldName, $sFormPrefix = '', $aArgs = array(), $bSearchMode = false)
  88. {
  89. $sTitle = addslashes($sTitle);
  90. $oPage->add_linked_script('../js/extkeywidget.js');
  91. $oPage->add_linked_script('../js/forms-json-utils.js');
  92. $bCreate = (!$bSearchMode) && (!MetaModel::IsAbstract($this->sTargetClass)) && (UserRights::IsActionAllowed($this->sTargetClass, UR_ACTION_BULK_MODIFY) && $bAllowTargetCreation);
  93. $sMessage = Dict::S('UI:Message:EmptyList:UseSearchForm');
  94. $sAttrFieldPrefix = ($bSearchMode) ? '' : 'attr_';
  95. $sHTMLValue = "<span style=\"white-space:nowrap\">"; // no wrap
  96. $sFilter = addslashes($oAllowedValues->GetFilter()->ToOQL());
  97. if($bSearchMode)
  98. {
  99. $sWizHelper = 'null';
  100. $sWizHelperJSON = "''";
  101. }
  102. else
  103. {
  104. $sWizHelper = 'oWizardHelper'.$sFormPrefix;
  105. $sWizHelperJSON = $sWizHelper.'.ToJSON()';
  106. }
  107. if (is_null($oAllowedValues))
  108. {
  109. throw new Exception('Implementation: null value for allowed values definition');
  110. }
  111. elseif ($oAllowedValues->Count() < $iMaxComboLength)
  112. {
  113. // Few choices, use a normal 'select'
  114. $sSelectMode = 'true';
  115. $sHelpText = ''; //$this->oAttDef->GetHelpOnEdition();
  116. $sHTMLValue = "<select title=\"$sHelpText\" name=\"{$sAttrFieldPrefix}{$sFieldName}\" id=\"$this->iId\">\n";
  117. if ($bSearchMode)
  118. {
  119. $sDisplayValue = isset($aArgs['sDefaultValue']) ? $aArgs['sDefaultValue'] : Dict::S('UI:SearchValue:Any');
  120. $sHTMLValue .= "<option value=\"\">$sDisplayValue</option>\n";
  121. }
  122. else
  123. {
  124. $sHTMLValue .= "<option value=\"\">".Dict::S('UI:SelectOne')."</option>\n";
  125. }
  126. $oAllowedValues->Rewind();
  127. while($oObj = $oAllowedValues->Fetch())
  128. {
  129. $key = $oObj->GetKey();
  130. $display_value = $oObj->Get('friendlyname');
  131. if (($oAllowedValues->Count() == 1) && ($bMandatory == 'true') )
  132. {
  133. // When there is only once choice, select it by default
  134. $sSelected = ' selected';
  135. }
  136. else
  137. {
  138. $sSelected = ($value == $key) ? ' selected' : '';
  139. }
  140. $sHTMLValue .= "<option value=\"$key\"$sSelected>$display_value</option>\n";
  141. }
  142. $sHTMLValue .= "</select>\n";
  143. $oPage->add_ready_script(
  144. <<<EOF
  145. oACWidget_{$this->iId} = new ExtKeyWidget('{$this->iId}', '{$this->sTargetClass}', '$sFilter', '$sTitle', true, $sWizHelper);
  146. oACWidget_{$this->iId}.emptyHtml = "<div style=\"background: #fff; border:0; text-align:center; vertical-align:middle;\"><p>$sMessage</p></div>";
  147. $('#$this->iId').bind('update', function() { oACWidget_{$this->iId}.Update(); } );
  148. EOF
  149. );
  150. }
  151. else
  152. {
  153. // Too many choices, use an autocomplete
  154. $sSelectMode = 'false';
  155. if (is_null($value) || ($value == 0)) // Null values are displayed as ''
  156. {
  157. $sDisplayValue = isset($aArgs['sDefaultValue']) ? $aArgs['sDefaultValue'] : '';
  158. }
  159. else
  160. {
  161. $sDisplayValue = $this->GetObjectName($value);
  162. }
  163. $iMinChars = isset($aArgs['iMinChars']) ? $aArgs['iMinChars'] : 3; //@@@ $this->oAttDef->GetMinAutoCompleteChars();
  164. $iFieldSize = isset($aArgs['iFieldSize']) ? $aArgs['iFieldSize'] : 30; //@@@ $this->oAttDef->GetMaxSize();
  165. $iDropDownSize = 10;
  166. // the input for the auto-complete
  167. $sHTMLValue = "<input count=\"".$oAllowedValues->Count()."\" type=\"text\" id=\"label_$this->iId\" size=\"$iFieldSize\" maxlength=\"$iDropDownSize\" value=\"$sDisplayValue\"/>&nbsp;";
  168. $sHTMLValue .= "<a class=\"no-arrow\" href=\"javascript:oACWidget_{$this->iId}.Search();\"><img id=\"mini_search_{$this->iId}\" style=\"border:0;vertical-align:middle;\" src=\"../images/mini_search.gif\" /></a>&nbsp;";
  169. // another hidden input to store & pass the object's Id
  170. $sHTMLValue .= "<input type=\"hidden\" id=\"$this->iId\" name=\"{$sAttrFieldPrefix}{$sFieldName}\" value=\"$value\" />\n";
  171. // Scripts to start the autocomplete and bind some events to it
  172. $oPage->add_ready_script(
  173. <<<EOF
  174. oACWidget_{$this->iId} = new ExtKeyWidget('{$this->iId}', '{$this->sTargetClass}', '$sFilter', '$sTitle', false, $sWizHelper);
  175. oACWidget_{$this->iId}.emptyHtml = "<div style=\"background: #fff; border:0; text-align:center; vertical-align:middle;\"><p>$sMessage</p></div>";
  176. $('#label_$this->iId').autocomplete('../pages/ajax.render.php', { scroll:true, minChars:{$iMinChars}, autoFill:false, matchContains:true, keyHolder:'#{$this->iId}', extraParams:{operation:'ac_extkey', sTargetClass:'{$this->sTargetClass}',sFilter:'$sFilter', json: $sWizHelperJSON }});
  177. $('#label_$this->iId').blur(function() { $(this).search(); } );
  178. $('#label_$this->iId').keyup(function() { if ($(this).val() == '') { $('#$this->iId').val(''); } } ); // Useful for search forms: empty value in the "label", means no value, immediatly !
  179. $('#label_$this->iId').result( function(event, data, formatted) { OnAutoComplete('{$this->iId}', event, data, formatted); } );
  180. $('#$this->iId').bind('update', function() { oACWidget_{$this->iId}.Update(); } );
  181. if ($('#ac_dlg_{$this->iId}').length == 0)
  182. {
  183. $('body').append('<div id="ac_dlg_{$this->iId}"></div>');
  184. }
  185. EOF
  186. );
  187. //$oPage->add_at_the_end($this->GetSearchDialog($oPage)); // To prevent adding forms inside the main form
  188. //$oPage->add_at_the_end('<div id="ac_dlg_'.$this->iId.'"></div>'); // The place where to download the search dialog is outside of the main form (to prevent nested forms)
  189. }
  190. if ($bCreate)
  191. {
  192. $sHTMLValue .= "<a class=\"no-arrow\" href=\"javascript:oACWidget_{$this->iId}.CreateObject();\"><img id=\"mini_add_{$this->iId}\" style=\"border:0;vertical-align:middle;\" src=\"../images/mini_add.gif\" /></a>&nbsp;";
  193. $oPage->add_ready_script(
  194. <<<EOF
  195. if ($('#ajax_{$this->iId}').length == 0)
  196. {
  197. $('body').append('<div id="ajax_{$this->iId}"></div>');
  198. }
  199. EOF
  200. );
  201. }
  202. $sHTMLValue .= "<span id=\"v_{$this->iId}\"></span>";
  203. $sHTMLValue .= "</span>"; // end of no wrap
  204. return $sHTMLValue;
  205. }
  206. public function GetSearchDialog(WebPage $oPage, $sTitle)
  207. {
  208. $sHTML = '<div class="wizContainer" style="vertical-align:top;"><div id="dc_'.$this->iId.'">';
  209. $oFilter = new DBObjectSearch($this->sTargetClass);
  210. $oSet = new CMDBObjectSet($oFilter);
  211. $oBlock = new DisplayBlock($oFilter, 'search', false);
  212. $sHTML .= $oBlock->GetDisplay($oPage, $this->iId, array('open' => true, 'currentId' => $this->iId));
  213. $sHTML .= "<form id=\"fr_{$this->iId}\" OnSubmit=\"return oACWidget_{$this->iId}.DoOk();\">\n";
  214. $sHTML .= "<div id=\"dr_{$this->iId}\" style=\"vertical-align:top;background: #fff;height:100%;overflow:auto;padding:0;border:0;\">\n";
  215. $sHTML .= "<div style=\"background: #fff; border:0; text-align:center; vertical-align:middle;\"><p>".Dict::S('UI:Message:EmptyList:UseSearchForm')."</p></div>\n";
  216. $sHTML .= "</div>\n";
  217. $sHTML .= "<input type=\"button\" id=\"btn_cancel_{$this->iId}\" value=\"".Dict::S('UI:Button:Cancel')."\" onClick=\"$('#ac_dlg_{$this->iId}').dialog('close');\">&nbsp;&nbsp;";
  218. $sHTML .= "<input type=\"button\" id=\"btn_ok_{$this->iId}\" value=\"".Dict::S('UI:Button:Ok')."\" onClick=\"oACWidget_{$this->iId}.DoOk();\">";
  219. $sHTML .= "</form>\n";
  220. $sHTML .= '</div></div>';
  221. $sDialogTitle = addslashes($sTitle);
  222. $oPage->add_ready_script(
  223. <<<EOF
  224. $('#ac_dlg_{$this->iId}').dialog({ width: $(window).width()*0.8, height: $(window).height()*0.8, autoOpen: false, modal: true, title: '$sDialogTitle', resizeStop: oACWidget_{$this->iId}.UpdateSizes, close: oACWidget_{$this->iId}.OnClose });
  225. $('#fs_{$this->iId}').bind('submit.uiAutocomplete', oACWidget_{$this->iId}.DoSearchObjects);
  226. $('#dc_{$this->iId}').resize(oACWidget_{$this->iId}.UpdateSizes);
  227. EOF
  228. );
  229. $oPage->add($sHTML);
  230. }
  231. /**
  232. * Search for objects to be selected
  233. * @param WebPage $oP The page used for the output (usually an AjaxWebPage)
  234. * @param string $sRemoteClass Name of the "remote" class to perform the search on, must be a derived class of m_sRemoteClass
  235. * @param Array $aAlreadyLinkedIds List of IDs of objects of "remote" class already linked, to be filtered out of the search
  236. */
  237. public function SearchObjectsToSelect(WebPage $oP, $sFilter, $sRemoteClass = '', $oObj = null)
  238. {
  239. if (is_null($sFilter))
  240. {
  241. throw new Exception('Implementation: null value for allowed values definition');
  242. }
  243. try
  244. {
  245. $oFilter = DBObjectSearch::FromOQL($sFilter);
  246. $oBlock = new DisplayBlock($oFilter, 'list', false);
  247. $oBlock->Display($oP, $this->iId, array('this' => $oObj, 'menu' => false, 'selection_mode' => true, 'selection_type' => 'single', 'display_limit' => false)); // Don't display the 'Actions' menu on the results
  248. }
  249. catch(MissingQueryArgument $e)
  250. {
  251. // When used in a search form the $this parameter may be missing, in this case return all possible values...
  252. // TODO check if we can improve this behavior...
  253. $sOQL = 'SELECT '.$sRemoteClass;
  254. $oFilter = DBObjectSearch::FromOQL($sOQL);
  255. $oBlock = new DisplayBlock($oFilter, 'list', false);
  256. $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
  257. }
  258. }
  259. /**
  260. * Search for objects to be selected
  261. * @param WebPage $oP The page used for the output (usually an AjaxWebPage)
  262. * @param string $sFilter The OQL expression used to define/limit limit the scope of possible values
  263. * @param DBObject $oObj The current object for the OQL context
  264. * @param string $sContains The text of the autocomplete to filter the results
  265. */
  266. public function AutoComplete(WebPage $oP, $sFilter, $oObj = null, $sContains)
  267. {
  268. if (is_null($sFilter))
  269. {
  270. throw new Exception('Implementation: null value for allowed values definition');
  271. }
  272. $oValuesSet = new ValueSetObjects($sFilter);
  273. $aValues = $oValuesSet->GetValues(array('this' => $oObj), $sContains);
  274. foreach($aValues as $sKey => $sFriendlyName)
  275. {
  276. $oP->add($sFriendlyName."|".$sKey."\n");
  277. }
  278. }
  279. /**
  280. * Get the display name of the selected object, to fill back the autocomplete
  281. */
  282. public function GetObjectName($iObjId)
  283. {
  284. $oObj = MetaModel::GetObject($this->sTargetClass, $iObjId);
  285. return $oObj->GetName();
  286. }
  287. /**
  288. * Get the form to create a new object of the 'target' class
  289. */
  290. public function GetObjectCreationForm(WebPage $oPage)
  291. {
  292. $sDialogTitle = addslashes($this->sTitle);
  293. $oPage->add('<div id="ac_create_'.$this->iId.'"><div class="wizContainer" style="vertical-align:top;"><div id="dcr_'.$this->iId.'">');
  294. $oPage->add("<h1>".MetaModel::GetClassIcon($this->sTargetClass)."&nbsp;".Dict::Format('UI:CreationTitle_Class', MetaModel::GetName($this->sTargetClass))."</h1>\n");
  295. cmdbAbstractObject::DisplayCreationForm($oPage, $this->sTargetClass, null, array(), array('formPrefix' => $this->iId, 'noRelations' => true));
  296. $oPage->add('</div></div></div>');
  297. // $oPage->add_ready_script("\$('#ac_create_$this->iId').dialog({ width: $(window).width()*0.8, height: 'auto', autoOpen: false, modal: true, title: '$sDialogTitle'});\n");
  298. $oPage->add_ready_script("\$('#ac_create_$this->iId').dialog({ width: 'auto', height: 'auto', autoOpen: false, modal: true, title: '$sDialogTitle'});\n");
  299. $oPage->add_ready_script("$('#dcr_{$this->iId} form').removeAttr('onsubmit');");
  300. $oPage->add_ready_script("$('#dcr_{$this->iId} form').bind('submit.uilinksWizard', oACWidget_{$this->iId}.DoCreateObject);");
  301. }
  302. /**
  303. * Get the form to create a new object of the 'target' class
  304. */
  305. public function DoCreateObject($oPage)
  306. {
  307. $oObj = MetaModel::NewObject($this->sTargetClass);
  308. $aErrors = $oObj->UpdateObject($this->iId);
  309. if (count($aErrors) == 0)
  310. {
  311. $oMyChange = MetaModel::NewObject("CMDBChange");
  312. $oMyChange->Set("date", time());
  313. $sUserString = CMDBChange::GetCurrentUserName();
  314. $oMyChange->Set("userinfo", $sUserString);
  315. $iChangeId = $oMyChange->DBInsert();
  316. $oObj->DBInsertTracked($oMyChange);
  317. return array('name' => $oObj->GetName(), 'id' => $oObj->GetKey());
  318. }
  319. else
  320. {
  321. return array('name' => implode(' ', $aErrors), 'id' => 0);
  322. }
  323. }
  324. }
  325. ?>