setName('form_config_navigation'); $this->setSubmitLabel($this->translate('Save Changes')); } /** * Set the user for whom to manage navigation items * * @param User $user * * @return $this */ public function setUser(User $user) { $this->user = $user; return $this; } /** * Return the user for whom to manage navigation items * * @return User */ public function getUser() { return $this->user; } /** * Set the user's navigation configuration * * @param Config $config * * @return $this */ public function setUserConfig(Config $config) { $this->userConfig = $config; return $this; } /** * Return the user's navigation configuration * * @return Config */ public function getUserConfig() { if ($this->userConfig === null) { $this->userConfig = $this->getUser()->loadNavigationConfig(); } return $this->userConfig; } /** * Set the shared navigation configuration * * @param Config $config * * @return $this */ public function setShareConfig(Config $config) { $this->shareConfig = $config; return $this; } /** * Return the shared navigation configuration * * @return Config */ public function getShareConfig() { return $this->shareConfig; } /** * Populate the form with the given navigation item's config * * @param string $name * * @return $this * * @throws NotFoundError In case no navigation item with the given name is found */ public function load($name) { if ($this->getConfigForItem($name) === null) { throw new NotFoundError('No navigation item called "%s" found', $name); } $this->itemToLoad = $name; return $this; } /** * Add a new navigation item * * The navigation item to add is identified by the array-key `name'. * * @param array $data * * @return $this * * @throws InvalidArgumentException In case $data does not contain a navigation item name * @throws IcingaException In case a navigation item with the same name already exists */ public function add(array $data) { if (! isset($data['name'])) { throw new InvalidArgumentException('Key \'name\' missing'); } $itemName = $data['name']; $config = $this->getUserConfig(); if ($config->hasSection($itemName)) { throw new IcingaException( $this->translate('A navigation item with the name "%s" does already exist'), $itemName ); } unset($data['name']); $config->setSection($itemName, $data); $this->setIniConfig($config); return $this; } /** * Edit a navigation item * * @param string $name * @param array $data * * @return $this * * @throws NotFoundError In case no navigation item with the given name is found */ public function edit($name, array $data) { $config = $this->getConfigForItem($name); if ($config === null) { throw new NotFoundError('No navigation item called "%s" found', $name); } $itemConfig = $config->getSection($name); if (isset($data['name'])) { if ($data['name'] !== $name) { $config->removeSection($name); $name = $data['name']; } unset($data['name']); } $itemConfig->merge($data); foreach ($itemConfig->toArray() as $k => $v) { if ($v === null) { unset($itemConfig->$k); } } $config->setSection($name, $itemConfig); $this->setIniConfig($config); return $this; } /** * Remove a navigation item * * @param string $name * * @return $this */ public function delete($name) { $config = $this->getConfigForItem($name); if ($config === null) { throw new NotFoundError('No navigation item called "%s" found', $name); } $config->removeSection($name); $this->setIniConfig($config); return $this; } /** * Unshare the given navigation item * * @param string $name * * @return bool * * @throws NotFoundError In case no navigation item with the given name is found */ public function unshare($name) { throw new NotFoundError('No navigation item called "%s" found', $name); } /** * {@inheritdoc} */ public function createElements(array $formData) { $itemTypes = $this->listItemTypes(); $itemType = isset($formData['type']) ? $formData['type'] : reset($itemTypes); $this->addElement( 'text', 'name', array( 'required' => true, 'label' => $this->translate('Name'), 'description' => $this->translate( 'The name of this navigation item that is used to differentiate it from others' ) ) ); $this->addElement( 'select', 'type', array( 'required' => true, 'autosubmit' => true, 'label' => $this->translate('Type'), 'description' => $this->translate('The type of this navigation item'), 'multiOptions' => array_combine($itemTypes, $itemTypes) ) ); $this->addSubForm($this->getItemForm($itemType)->create($formData), 'item_form'); } /** * Populate the configuration of the navigation item to load */ public function onRequest() { if ($this->itemToLoad) { $data = $this->getConfigForItem($this->itemToLoad)->getSection($this->itemToLoad)->toArray(); $data['name'] = $this->itemToLoad; $this->populate($data); } } /** * {@inheritdoc} */ public function getValues($suppressArrayNotation = false) { $values = parent::getValues(); $values = array_merge($values, $values['item_form']); unset($values['item_form']); return $values; } /** * Return a list of available item types * * @return array */ protected function listItemTypes() { $types = $this->defaultItemTypes; foreach (Icinga::app()->getModuleManager()->getLoadedModules() as $module) { $moduleItems = $module->getNavigationItems(); if (! empty($moduleItems)) { $types = array_merge($types, $moduleItems); } } return $types; } /** * Return the navigation configuration the given item is a part of * * @param string $name * * @return Config|null In case the item is not part of any configuration */ protected function getConfigForItem($name) { if ($this->getUserConfig()->hasSection($name)) { return $this->getUserConfig(); } elseif ($this->getShareConfig()->hasSection($name)) { if ( $this->getShareConfig()->get($name, 'owner') === $this->getUser()->getUsername() || $this->getUser()->can('config/application/navigation') ) { return $this->getShareConfig(); } } } /** * Return the form for the given type of navigation item * * @param string $type * * @return Form */ protected function getItemForm($type) { // TODO: Load form classes dynamically return new NavigationItemForm(); } }