menunode.class.inc.php 26 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666
  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. * Construction and display of the application's main menu
  18. *
  19. * @author Erwan Taloc <erwan.taloc@combodo.com>
  20. * @author Romain Quetiez <romain.quetiez@combodo.com>
  21. * @author Denis Flaven <denis.flaven@combodo.com>
  22. * @license http://www.opensource.org/licenses/gpl-3.0.html LGPL
  23. */
  24. require_once(APPROOT.'/application/utils.inc.php');
  25. require_once(APPROOT.'/application/template.class.inc.php');
  26. /**
  27. * This class manipulates, stores and displays the navigation menu used in the application
  28. * In order to improve the modularity of the data model and to ease the update/migration
  29. * between evolving data models, the menus are no longer stored in the database, but are instead
  30. * built on the fly each time a page is loaded.
  31. * The application's menu is organized into top-level groups with, inside each group, a tree of menu items.
  32. * Top level groups do not display any content, they just expand/collapse.
  33. * Sub-items drive the actual content of the page, they are based either on templates, OQL queries or full (external?) web pages.
  34. *
  35. * Example:
  36. * Here is how to insert the following items in the application's menu:
  37. * +----------------------------------------+
  38. * | Configuration Management Group | >> Top level group
  39. * +----------------------------------------+
  40. * + Configuration Management Overview >> Template based menu item
  41. * + Contacts >> Template based menu item
  42. * + Persons >> Plain list (OQL based)
  43. * + Teams >> Plain list (OQL based)
  44. *
  45. * // Create the top-level group. fRank = 1, means it will be inserted after the group '0', which is usually 'Welcome'
  46. * $oConfigMgmtMenu = new MenuGroup('ConfigurationManagementMenu', 1);
  47. * // Create an entry, based on a custom template, for the Configuration management overview, under the top-level group
  48. * new TemplateMenuNode('ConfigurationManagementMenu', '../somedirectory/configuration_management_menu.html', $oConfigMgmtMenu->GetIndex(), 0);
  49. * // Create an entry (template based) for the overview of contacts
  50. * $oContactsMenu = new TemplateMenuNode('ContactsMenu', '../somedirectory/configuration_management_menu.html',$oConfigMgmtMenu->GetIndex(), 1);
  51. * // Plain list of persons
  52. * new OQLMenuNode('PersonsMenu', 'SELECT bizPerson', $oContactsMenu->GetIndex(), 0);
  53. *
  54. */
  55. class ApplicationMenu
  56. {
  57. static $aRootMenus = array();
  58. static $aMenusIndex = array();
  59. /**
  60. * Main function to add a menu entry into the application, can be called during the definition
  61. * of the data model objects
  62. */
  63. static public function InsertMenu(MenuNode $oMenuNode, $iParentIndex = -1, $fRank)
  64. {
  65. $index = self::GetMenuIndexById($oMenuNode->GetMenuId());
  66. if ($index == -1)
  67. {
  68. // The menu does not already exist, insert it
  69. $index = count(self::$aMenusIndex);
  70. self::$aMenusIndex[$index] = array( 'node' => $oMenuNode, 'children' => array());
  71. if ($iParentIndex == -1)
  72. {
  73. self::$aRootMenus[] = array ('rank' => $fRank, 'index' => $index);
  74. }
  75. else
  76. {
  77. self::$aMenusIndex[$iParentIndex]['children'][] = array ('rank' => $fRank, 'index' => $index);
  78. }
  79. }
  80. return $index;
  81. }
  82. /**
  83. * Entry point to display the whole menu into the web page, used by iTopWebPage
  84. */
  85. static public function DisplayMenu(iTopWebPage $oPage, $aExtraParams)
  86. {
  87. // Sort the root menu based on the rank
  88. usort(self::$aRootMenus, array('ApplicationMenu', 'CompareOnRank'));
  89. $iAccordion = 0;
  90. $iActiveMenu = ApplicationMenu::GetActiveNodeId();
  91. foreach(self::$aRootMenus as $aMenu)
  92. {
  93. $oMenuNode = self::GetMenuNode($aMenu['index']);
  94. if (($oMenuNode->GetMenuId() == 'AdminTools') && (!UserRights::IsAdministrator())) continue; // Don't display the admin menu for non admin users
  95. if (!$oMenuNode->IsEnabled()) continue; // Don't display a non-enabled menu
  96. $oPage->AddToMenu('<h3>'.$oMenuNode->GetTitle().'</h3>');
  97. $oPage->AddToMenu('<div>');
  98. $aChildren = self::GetChildren($aMenu['index']);
  99. if (count($aChildren) > 0)
  100. {
  101. $oPage->AddToMenu('<ul>');
  102. $bActive = self::DisplaySubMenu($oPage, $aChildren, $aExtraParams, $iActiveMenu);
  103. $oPage->AddToMenu('</ul>');
  104. if ($bActive)
  105. {
  106. $oPage->add_ready_script("$('#accordion').accordion('activate', $iAccordion);");
  107. $oPage->add_ready_script("$('#accordion').accordion('option', {collapsible: true});"); // Make it auto-collapsible once it has been opened properly
  108. }
  109. }
  110. $oPage->AddToMenu('</div>');
  111. $iAccordion++;
  112. }
  113. }
  114. /**
  115. * Handles the display of the sub-menus (called recursively if necessary)
  116. * @return true if the currently selected menu is one of the submenus
  117. */
  118. static protected function DisplaySubMenu($oPage, $aMenus, $aExtraParams, $iActiveMenu = -1)
  119. {
  120. // Sort the menu based on the rank
  121. $bActive = false;
  122. usort($aMenus, array('ApplicationMenu', 'CompareOnRank'));
  123. foreach($aMenus as $aMenu)
  124. {
  125. $index = $aMenu['index'];
  126. $oMenu = self::GetMenuNode($index);
  127. if ($oMenu->IsEnabled())
  128. {
  129. $aChildren = self::GetChildren($index);
  130. $sCSSClass = (count($aChildren) > 0) ? ' class="submenu"' : '';
  131. $sHyperlink = $oMenu->GetHyperlink($aExtraParams);
  132. if ($sHyperlink != '')
  133. {
  134. $oPage->AddToMenu('<li'.$sCSSClass.'><a href="'.$oMenu->GetHyperlink($aExtraParams).'">'.$oMenu->GetTitle().'</a></li>');
  135. }
  136. else
  137. {
  138. $oPage->AddToMenu('<li'.$sCSSClass.'>'.$oMenu->GetTitle().'</li>');
  139. }
  140. $aCurrentMenu = self::$aMenusIndex[$index];
  141. if ($iActiveMenu == $index)
  142. {
  143. $bActive = true;
  144. }
  145. if (count($aChildren) > 0)
  146. {
  147. $oPage->AddToMenu('<ul>');
  148. $bActive |= self::DisplaySubMenu($oPage, $aChildren, $aExtraParams, $iActiveMenu);
  149. $oPage->AddToMenu('</ul>');
  150. }
  151. }
  152. }
  153. return $bActive;
  154. }
  155. /**
  156. * Helper function to sort the menus based on their rank
  157. */
  158. static public function CompareOnRank($a, $b)
  159. {
  160. $result = 1;
  161. if ($a['rank'] == $b['rank'])
  162. {
  163. $result = 0;
  164. }
  165. if ($a['rank'] < $b['rank'])
  166. {
  167. $result = -1;
  168. }
  169. return $result;
  170. }
  171. /**
  172. * Helper function to retrieve the MenuNodeObject based on its ID
  173. */
  174. static public function GetMenuNode($index)
  175. {
  176. return isset(self::$aMenusIndex[$index]) ? self::$aMenusIndex[$index]['node'] : null;
  177. }
  178. /**
  179. * Helper function to get the list of child(ren) of a menu
  180. */
  181. static protected function GetChildren($index)
  182. {
  183. return self::$aMenusIndex[$index]['children'];
  184. }
  185. /**
  186. * Helper function to get the ID of a menu based on its name
  187. * @param string $sTitle Title of the menu (as passed when creating the menu)
  188. * @return integer ID of the menu, or -1 if not found
  189. */
  190. static public function GetMenuIndexById($sTitle)
  191. {
  192. $index = -1;
  193. foreach(self::$aMenusIndex as $aMenu)
  194. {
  195. if ($aMenu['node']->GetMenuId() == $sTitle)
  196. {
  197. $index = $aMenu['node']->GetIndex();
  198. break;
  199. }
  200. }
  201. return $index;
  202. }
  203. /**
  204. * Retrieves the currently active menu (if any, otherwise the first menu is the default)
  205. * @return MenuNode or null if there is no menu at all !
  206. */
  207. static public function GetActiveNodeId()
  208. {
  209. $oAppContext = new ApplicationContext();
  210. $iMenuIndex = $oAppContext->GetCurrentValue('menu', -1);
  211. if ($iMenuIndex == -1)
  212. {
  213. // Make sure the root menu is sorted on 'rank'
  214. usort(self::$aRootMenus, array('ApplicationMenu', 'CompareOnRank'));
  215. $oFirstGroup = self::GetMenuNode(self::$aRootMenus[0]['index']);
  216. $oMenuNode = self::GetMenuNode(self::$aMenusIndex[$oFirstGroup->GetIndex()]['children'][0]['index']);
  217. $iMenuIndex = $oMenuNode->GetIndex();
  218. }
  219. return $iMenuIndex;
  220. }
  221. }
  222. /**
  223. * Root class for all the kind of node in the menu tree, data model providers are responsible for instantiating
  224. * MenuNodes (i.e instances from derived classes) in order to populate the application's menu. Creating an objet
  225. * derived from MenuNode is enough to have it inserted in the application's main menu.
  226. * The class iTopWebPage, takes care of 3 items:
  227. * +--------------------+
  228. * | Welcome |
  229. * +--------------------+
  230. * Welcome To iTop
  231. * +--------------------+
  232. * | Tools |
  233. * +--------------------+
  234. * CSV Import
  235. * +--------------------+
  236. * | Admin Tools |
  237. * +--------------------+
  238. * User Accounts
  239. * Profiles
  240. * Notifications
  241. * Run Queries
  242. * Export
  243. * Data Model
  244. * Universal Search
  245. *
  246. * All the other menu items must constructed along with the various data model modules
  247. */
  248. abstract class MenuNode
  249. {
  250. protected $sMenuId;
  251. protected $index;
  252. /**
  253. * Class of objects to check if the menu is enabled, null if none
  254. */
  255. protected $m_sEnableClass;
  256. /**
  257. * User Rights Action code to check if the menu is enabled, null if none
  258. */
  259. protected $m_iEnableAction;
  260. /**
  261. * User Rights allowed results (actually a bitmask) to check if the menu is enabled, null if none
  262. */
  263. protected $m_iEnableActionResults;
  264. /**
  265. * Create a menu item, sets the condition to have it displayed and inserts it into the application's main menu
  266. * @param string $sMenuId Unique identifier of the menu (used to identify the menu for bookmarking, and for getting the labels from the dictionary)
  267. * @param integer $iParentIndex ID of the parent menu, pass -1 for top level (group) items
  268. * @param float $fRank Number used to order the list, any number will do, but for a given level (i.e same parent) all menus are sorted based on this value
  269. * @param string $sEnableClass Name of class of object
  270. * @param integer $iActionCode Either UR_ACTION_READ, UR_ACTION_MODIFY, UR_ACTION_DELETE, UR_ACTION_BULKREAD, UR_ACTION_BULKMODIFY or UR_ACTION_BULKDELETE
  271. * @param integer $iAllowedResults Expected "rights" for the action: either UR_ALLOWED_YES, UR_ALLOWED_NO, UR_ALLOWED_DEPENDS or a mix of them...
  272. * @return MenuNode
  273. */
  274. public function __construct($sMenuId, $iParentIndex = -1, $fRank = 0, $sEnableClass = null, $iActionCode = null, $iAllowedResults = UR_ALLOWED_YES)
  275. {
  276. $this->sMenuId = $sMenuId;
  277. $this->m_sEnableClass = $sEnableClass;
  278. $this->m_iEnableAction = $iActionCode;
  279. $this->m_iEnableActionResults = $iAllowedResults;
  280. $this->index = ApplicationMenu::InsertMenu($this, $iParentIndex, $fRank);
  281. }
  282. public function GetMenuId()
  283. {
  284. return $this->sMenuId;
  285. }
  286. public function GetTitle()
  287. {
  288. return Dict::S("Menu:$this->sMenuId");
  289. }
  290. public function GetLabel()
  291. {
  292. return Dict::S("Menu:$this->sMenuId+");
  293. }
  294. public function GetIndex()
  295. {
  296. return $this->index;
  297. }
  298. public function GetHyperlink($aExtraParams)
  299. {
  300. $aExtraParams['c[menu]'] = $this->GetIndex();
  301. return $this->AddParams('../pages/UI.php', $aExtraParams);
  302. }
  303. /**
  304. * Tells whether the menu is enabled (i.e. displayed) for the current user
  305. * @return bool True if enabled, false otherwise
  306. */
  307. public function IsEnabled()
  308. {
  309. if ($this->m_sEnableClass != null)
  310. {
  311. if (MetaModel::IsValidClass($this->m_sEnableClass))
  312. {
  313. $iResult = UserRights::IsActionAllowed($this->m_sEnableClass, $this->m_iEnableAction);
  314. if (($iResult & $this->m_iEnableActionResults))
  315. {
  316. return true;
  317. }
  318. }
  319. return false;
  320. }
  321. return true;
  322. }
  323. public abstract function RenderContent(WebPage $oPage, $aExtraParams = array());
  324. protected function AddParams($sHyperlink, $aExtraParams)
  325. {
  326. if (count($aExtraParams) > 0)
  327. {
  328. $aQuery = array();
  329. $sSeparator = '?';
  330. if (strpos($sHyperlink, '?') !== false)
  331. {
  332. $sSeparator = '&';
  333. }
  334. foreach($aExtraParams as $sName => $sValue)
  335. {
  336. $aQuery[] = urlencode($sName).'='.urlencode($sValue);
  337. }
  338. $sHyperlink .= $sSeparator.implode('&', $aQuery);
  339. }
  340. return $sHyperlink;
  341. }
  342. }
  343. /**
  344. * This class implements a top-level menu group. A group is just a container for sub-items
  345. * it does not display a page by itself
  346. */
  347. class MenuGroup extends MenuNode
  348. {
  349. /**
  350. * Create a top-level menu group and inserts it into the application's main menu
  351. * @param string $sMenuId Unique identifier of the menu (used to identify the menu for bookmarking, and for getting the labels from the dictionary)
  352. * @param float $fRank Number used to order the list, the groups are sorted based on this value
  353. * @param string $sEnableClass Name of class of object
  354. * @param integer $iActionCode Either UR_ACTION_READ, UR_ACTION_MODIFY, UR_ACTION_DELETE, UR_ACTION_BULKREAD, UR_ACTION_BULKMODIFY or UR_ACTION_BULKDELETE
  355. * @param integer $iAllowedResults Expected "rights" for the action: either UR_ALLOWED_YES, UR_ALLOWED_NO, UR_ALLOWED_DEPENDS or a mix of them...
  356. * @return MenuGroup
  357. */
  358. public function __construct($sMenuId, $fRank, $sEnableClass = null, $iActionCode = null, $iAllowedResults = UR_ALLOWED_YES)
  359. {
  360. parent::__construct($sMenuId, -1 /* no parent, groups are at root level */, $fRank, $sEnableClass, $iActionCode, $iAllowedResults);
  361. }
  362. public function RenderContent(WebPage $oPage, $aExtraParams = array())
  363. {
  364. assert(false); // Shall never be called, groups do not display any content
  365. }
  366. }
  367. /**
  368. * This class defines a menu item which content is based on a custom template.
  369. * Note the template can be either a local file or an URL !
  370. */
  371. class TemplateMenuNode extends MenuNode
  372. {
  373. protected $sTemplateFile;
  374. /**
  375. * Create a menu item based on a custom template and inserts it into the application's main menu
  376. * @param string $sMenuId Unique identifier of the menu (used to identify the menu for bookmarking, and for getting the labels from the dictionary)
  377. * @param string $sTemplateFile Path (or URL) to the file that will be used as a template for displaying the page's content
  378. * @param integer $iParentIndex ID of the parent menu
  379. * @param float $fRank Number used to order the list, any number will do, but for a given level (i.e same parent) all menus are sorted based on this value
  380. * @param string $sEnableClass Name of class of object
  381. * @param integer $iActionCode Either UR_ACTION_READ, UR_ACTION_MODIFY, UR_ACTION_DELETE, UR_ACTION_BULKREAD, UR_ACTION_BULKMODIFY or UR_ACTION_BULKDELETE
  382. * @param integer $iAllowedResults Expected "rights" for the action: either UR_ALLOWED_YES, UR_ALLOWED_NO, UR_ALLOWED_DEPENDS or a mix of them...
  383. * @return MenuNode
  384. */
  385. public function __construct($sMenuId, $sTemplateFile, $iParentIndex, $fRank = 0, $sEnableClass = null, $iActionCode = null, $iAllowedResults = UR_ALLOWED_YES)
  386. {
  387. parent::__construct($sMenuId, $iParentIndex, $fRank, $sEnableClass, $iActionCode, $iAllowedResults);
  388. $this->sTemplateFile = $sTemplateFile;
  389. }
  390. public function GetHyperlink($aExtraParams)
  391. {
  392. if ($this->sTemplateFile == '') return '';
  393. return parent::GetHyperlink($aExtraParams);
  394. }
  395. public function RenderContent(WebPage $oPage, $aExtraParams = array())
  396. {
  397. $sTemplate = @file_get_contents($this->sTemplateFile);
  398. if ($sTemplate !== false)
  399. {
  400. $oTemplate = new DisplayTemplate($sTemplate);
  401. $oTemplate->Render($oPage, $aExtraParams);
  402. }
  403. else
  404. {
  405. $oPage->p("Error: failed to load template file: '{$this->sTemplateFile}'"); // No need to translate ?
  406. }
  407. }
  408. }
  409. /**
  410. * This class defines a menu item that uses a standard template to display a list of items therefore it allows
  411. * only two parameters: the page's title and the OQL expression defining the list of items to be displayed
  412. */
  413. class OQLMenuNode extends MenuNode
  414. {
  415. protected $sPageTitle;
  416. protected $sOQL;
  417. protected $bSearch;
  418. /**
  419. * Extra parameters to be passed to the display block to fine tune its appearence
  420. */
  421. protected $m_aParams;
  422. /**
  423. * Create a menu item based on an OQL query and inserts it into the application's main menu
  424. * @param string $sMenuId Unique identifier of the menu (used to identify the menu for bookmarking, and for getting the labels from the dictionary)
  425. * @param string $sOQL OQL query defining the set of objects to be displayed
  426. * @param integer $iParentIndex ID of the parent menu
  427. * @param float $fRank Number used to order the list, any number will do, but for a given level (i.e same parent) all menus are sorted based on this value
  428. * @param bool $bSearch Whether or not to display a (collapsed) search frame at the top of the page
  429. * @param string $sEnableClass Name of class of object
  430. * @param integer $iActionCode Either UR_ACTION_READ, UR_ACTION_MODIFY, UR_ACTION_DELETE, UR_ACTION_BULKREAD, UR_ACTION_BULKMODIFY or UR_ACTION_BULKDELETE
  431. * @param integer $iAllowedResults Expected "rights" for the action: either UR_ALLOWED_YES, UR_ALLOWED_NO, UR_ALLOWED_DEPENDS or a mix of them...
  432. * @return MenuNode
  433. */
  434. public function __construct($sMenuId, $sOQL, $iParentIndex, $fRank = 0, $bSearch = false, $sEnableClass = null, $iActionCode = null, $iAllowedResults = UR_ALLOWED_YES)
  435. {
  436. parent::__construct($sMenuId, $iParentIndex, $fRank, $sEnableClass, $iActionCode, $iAllowedResults);
  437. $this->sPageTitle = "Menu:$sMenuId+";
  438. $this->sOQL = $sOQL;
  439. $this->bSearch = $bSearch;
  440. $this->m_aParams = array();
  441. // Enhancement: we could set as the "enable" condition that the user has enough rights to "read" the objects
  442. // of the class specified by the OQL...
  443. }
  444. /**
  445. * Set some extra parameters to be passed to the display block to fine tune its appearence
  446. * @param Hash $aParams paramCode => value. See DisplayBlock::GetDisplay for the meaning of the parameters
  447. */
  448. public function SetParameters($aParams)
  449. {
  450. $this->m_aParams = $aParams;
  451. }
  452. public function RenderContent(WebPage $oPage, $aExtraParams = array())
  453. {
  454. $aExtraParams = array_merge($aExtraParams, $this->m_aParams);
  455. try
  456. {
  457. $oSearch = DBObjectSearch::FromOQL($this->sOQL);
  458. $sIcon = MetaModel::GetClassIcon($oSearch->GetClass());
  459. }
  460. catch(Exception $e)
  461. {
  462. $sIcon = '';
  463. }
  464. // The standard template used for all such pages: a (closed) search form at the top and a list of results at the bottom
  465. $sTemplate = '';
  466. if ($this->bSearch)
  467. {
  468. $sTemplate .= <<<EOF
  469. <itopblock BlockClass="DisplayBlock" type="search" asynchronous="false" encoding="text/oql">$this->sOQL</itopblock>
  470. EOF;
  471. }
  472. $sParams = '';
  473. if (!empty($this->m_aParams))
  474. {
  475. $sParams = 'parameters="';
  476. foreach($this->m_aParams as $sName => $sValue)
  477. {
  478. $sParams .= $sName.':'.$sValue.';';
  479. }
  480. $sParams .= '"';
  481. }
  482. $sTemplate .= <<<EOF
  483. <p class="page-header">$sIcon<itopstring>$this->sPageTitle</itopstring></p>
  484. <itopblock BlockClass="DisplayBlock" type="list" asynchronous="false" encoding="text/oql" $sParams>$this->sOQL</itopblock>
  485. EOF;
  486. $oTemplate = new DisplayTemplate($sTemplate);
  487. $oTemplate->Render($oPage, $aExtraParams);
  488. }
  489. }
  490. /**
  491. * This class defines a menu item that displays a search form for the given class of objects
  492. */
  493. class SearchMenuNode extends MenuNode
  494. {
  495. protected $sPageTitle;
  496. protected $sClass;
  497. /**
  498. * Create a menu item based on an OQL query and inserts it into the application's main menu
  499. * @param string $sMenuId Unique identifier of the menu (used to identify the menu for bookmarking, and for getting the labels from the dictionary)
  500. * @param string $sClass The class of objects to search for
  501. * @param string $sPageTitle Title displayed into the page's content (will be looked-up in the dictionnary for translation)
  502. * @param integer $iParentIndex ID of the parent menu
  503. * @param float $fRank Number used to order the list, any number will do, but for a given level (i.e same parent) all menus are sorted based on this value
  504. * @param string $sEnableClass Name of class of object
  505. * @param integer $iActionCode Either UR_ACTION_READ, UR_ACTION_MODIFY, UR_ACTION_DELETE, UR_ACTION_BULKREAD, UR_ACTION_BULKMODIFY or UR_ACTION_BULKDELETE
  506. * @param integer $iAllowedResults Expected "rights" for the action: either UR_ALLOWED_YES, UR_ALLOWED_NO, UR_ALLOWED_DEPENDS or a mix of them...
  507. * @return MenuNode
  508. */
  509. public function __construct($sMenuId, $sClass, $iParentIndex, $fRank = 0, $bSearch = false, $sEnableClass = null, $iActionCode = null, $iAllowedResults = UR_ALLOWED_YES)
  510. {
  511. parent::__construct($sMenuId, $iParentIndex, $fRank, $sEnableClass, $iActionCode, $iAllowedResults);
  512. $this->sPageTitle = "Menu:$sMenuId+";
  513. $this->sClass = $sClass;
  514. }
  515. public function RenderContent(WebPage $oPage, $aExtraParams = array())
  516. {
  517. // The standard template used for all such pages: an open search form at the top
  518. $sTemplate = <<<EOF
  519. <itopblock BlockClass="DisplayBlock" type="search" asynchronous="false" encoding="text/oql" parameters="open:true">SELECT $this->sClass</itopblock>
  520. EOF;
  521. $oTemplate = new DisplayTemplate($sTemplate);
  522. $oTemplate->Render($oPage, $aExtraParams);
  523. }
  524. }
  525. /**
  526. * This class defines a menu that points to any web page. It takes only two parameters:
  527. * - The hyperlink to point to
  528. * - The name of the menu
  529. * Note: the parameter menu=xxx (where xxx is the id of the menu itself) will be added to the hyperlink
  530. * in order to make it the active one, if the target page is based on iTopWebPage and therefore displays the menu
  531. */
  532. class WebPageMenuNode extends MenuNode
  533. {
  534. protected $sHyperlink;
  535. /**
  536. * Create a menu item that points to any web page (not only UI.php)
  537. * @param string $sMenuId Unique identifier of the menu (used to identify the menu for bookmarking, and for getting the labels from the dictionary)
  538. * @param string $sHyperlink URL to the page to load. Use relative URL if you want to keep the application portable !
  539. * @param integer $iParentIndex ID of the parent menu
  540. * @param float $fRank Number used to order the list, any number will do, but for a given level (i.e same parent) all menus are sorted based on this value
  541. * @param string $sEnableClass Name of class of object
  542. * @param integer $iActionCode Either UR_ACTION_READ, UR_ACTION_MODIFY, UR_ACTION_DELETE, UR_ACTION_BULKREAD, UR_ACTION_BULKMODIFY or UR_ACTION_BULKDELETE
  543. * @param integer $iAllowedResults Expected "rights" for the action: either UR_ALLOWED_YES, UR_ALLOWED_NO, UR_ALLOWED_DEPENDS or a mix of them...
  544. * @return MenuNode
  545. */
  546. public function __construct($sMenuId, $sHyperlink, $iParentIndex, $fRank = 0, $sEnableClass = null, $iActionCode = null, $iAllowedResults = UR_ALLOWED_YES)
  547. {
  548. parent::__construct($sMenuId, $iParentIndex, $fRank, $sEnableClass, $iActionCode, $iAllowedResults);
  549. $this->sHyperlink = $sHyperlink;
  550. }
  551. public function GetHyperlink($aExtraParams)
  552. {
  553. $aExtraParams['c[menu]'] = $this->GetIndex();
  554. return $this->AddParams( $this->sHyperlink, $aExtraParams);
  555. }
  556. public function RenderContent(WebPage $oPage, $aExtraParams = array())
  557. {
  558. assert(false); // Shall never be called, the external web page will handle the display by itself
  559. }
  560. }
  561. /**
  562. * This class defines a menu that points to the page for creating a new object of the specified class.
  563. * It take only one parameter: the name of the class
  564. * Note: the parameter menu=xxx (where xxx is the id of the menu itself) will be added to the hyperlink
  565. * in order to make it the active one
  566. */
  567. class NewObjectMenuNode extends MenuNode
  568. {
  569. protected $sClass;
  570. /**
  571. * Create a menu item that points to the URL for creating a new object, the menu will be added only if the current user has enough
  572. * rights to create such an object (or an object of a child class)
  573. * @param string $sMenuId Unique identifier of the menu (used to identify the menu for bookmarking, and for getting the labels from the dictionary)
  574. * @param string $sClass URL to the page to load. Use relative URL if you want to keep the application portable !
  575. * @param integer $iParentIndex ID of the parent menu
  576. * @param float $fRank Number used to order the list, any number will do, but for a given level (i.e same parent) all menus are sorted based on this value
  577. * @return MenuNode
  578. */
  579. public function __construct($sMenuId, $sClass, $iParentIndex, $fRank = 0)
  580. {
  581. parent::__construct($sMenuId, $iParentIndex, $fRank);
  582. $this->sClass = $sClass;
  583. }
  584. public function GetHyperlink($aExtraParams)
  585. {
  586. $sHyperlink = '../pages/UI.php?operation=new&class='.$this->sClass;
  587. $aExtraParams['c[menu]'] = $this->GetIndex();
  588. return $this->AddParams($sHyperlink, $aExtraParams);
  589. }
  590. /**
  591. * Overload the check of the "enable" state of this menu to take into account
  592. * derived classes of objects
  593. */
  594. public function IsEnabled()
  595. {
  596. // Enable this menu, only if the current user has enough rights to create such an object, or an object of
  597. // any child class
  598. $aSubClasses = MetaModel::EnumChildClasses($this->sClass, ENUM_CHILD_CLASSES_ALL); // Including the specified class itself
  599. $bActionIsAllowed = false;
  600. foreach($aSubClasses as $sCandidateClass)
  601. {
  602. if (!MetaModel::IsAbstract($sCandidateClass) && (UserRights::IsActionAllowed($sCandidateClass, UR_ACTION_MODIFY) == UR_ALLOWED_YES))
  603. {
  604. $bActionIsAllowed = true;
  605. break; // Enough for now
  606. }
  607. }
  608. return $bActionIsAllowed;
  609. }
  610. public function RenderContent(WebPage $oPage, $aExtraParams = array())
  611. {
  612. assert(false); // Shall never be called, the external web page will handle the display by itself
  613. }
  614. }
  615. ?>