2015-09-07 16:58:22 +02:00
|
|
|
<?php
|
|
|
|
/* Icinga Web 2 | (c) 2013-2015 Icinga Development Team | GPLv2+ */
|
|
|
|
|
|
|
|
namespace Icinga\Forms\Navigation;
|
|
|
|
|
|
|
|
use InvalidArgumentException;
|
2015-09-15 15:57:03 +02:00
|
|
|
use Icinga\Application\Config;
|
2015-09-07 16:58:22 +02:00
|
|
|
use Icinga\Application\Icinga;
|
|
|
|
use Icinga\Exception\IcingaException;
|
|
|
|
use Icinga\Exception\NotFoundError;
|
|
|
|
use Icinga\Forms\ConfigForm;
|
2015-09-15 15:57:03 +02:00
|
|
|
use Icinga\User;
|
2015-09-07 16:58:22 +02:00
|
|
|
use Icinga\Web\Form;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Form for managing navigation items
|
|
|
|
*/
|
|
|
|
class NavigationConfigForm extends ConfigForm
|
|
|
|
{
|
|
|
|
/**
|
|
|
|
* The default item types provided by Icinga Web 2
|
|
|
|
*
|
|
|
|
* @var array
|
|
|
|
*/
|
|
|
|
protected $defaultItemTypes = array(
|
|
|
|
'menu-item',
|
|
|
|
'dashlet'
|
|
|
|
);
|
|
|
|
|
2015-09-16 10:58:57 +02:00
|
|
|
/**
|
|
|
|
* The secondary configuration to write
|
|
|
|
*
|
|
|
|
* This is always the reduced configuration and is only written to
|
|
|
|
* disk once the main configuration has been successfully written.
|
|
|
|
*
|
|
|
|
* @var Config
|
|
|
|
*/
|
|
|
|
protected $secondaryConfig;
|
|
|
|
|
2015-09-07 16:58:22 +02:00
|
|
|
/**
|
|
|
|
* The navigation item to load when displaying the form for the first time
|
|
|
|
*
|
|
|
|
* @var string
|
|
|
|
*/
|
|
|
|
protected $itemToLoad;
|
|
|
|
|
2015-09-15 15:57:03 +02:00
|
|
|
/**
|
|
|
|
* The user for whom to manage navigation items
|
|
|
|
*
|
|
|
|
* @var User
|
|
|
|
*/
|
|
|
|
protected $user;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* The user's navigation configuration
|
|
|
|
*
|
|
|
|
* @var Config
|
|
|
|
*/
|
|
|
|
protected $userConfig;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* The shared navigation configuration
|
|
|
|
*
|
|
|
|
* @var Config
|
|
|
|
*/
|
|
|
|
protected $shareConfig;
|
|
|
|
|
2015-09-07 16:58:22 +02:00
|
|
|
/**
|
|
|
|
* Initialize this form
|
|
|
|
*/
|
|
|
|
public function init()
|
|
|
|
{
|
|
|
|
$this->setName('form_config_navigation');
|
|
|
|
$this->setSubmitLabel($this->translate('Save Changes'));
|
|
|
|
}
|
|
|
|
|
2015-09-15 15:57:03 +02:00
|
|
|
/**
|
|
|
|
* 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;
|
|
|
|
}
|
|
|
|
|
2015-09-07 16:58:22 +02:00
|
|
|
/**
|
|
|
|
* 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)
|
|
|
|
{
|
2015-09-15 15:57:03 +02:00
|
|
|
if ($this->getConfigForItem($name) === null) {
|
2015-09-07 16:58:22 +02:00
|
|
|
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');
|
|
|
|
}
|
|
|
|
|
2015-09-15 15:57:03 +02:00
|
|
|
$config = $this->getUserConfig();
|
2015-09-16 11:51:57 +02:00
|
|
|
if ((isset($data['users']) && $data['users']) || (isset($data['groups']) && $data['groups'])) {
|
|
|
|
if ($this->getUser()->can('application/share/navigation')) {
|
|
|
|
$data['owner'] = $this->getUser()->getUsername();
|
|
|
|
$config = $this->getShareConfig();
|
|
|
|
} else {
|
|
|
|
unset($data['users']);
|
|
|
|
unset($data['groups']);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
$itemName = $data['name'];
|
2015-09-15 15:57:03 +02:00
|
|
|
if ($config->hasSection($itemName)) {
|
2015-09-07 16:58:22 +02:00
|
|
|
throw new IcingaException(
|
|
|
|
$this->translate('A navigation item with the name "%s" does already exist'),
|
|
|
|
$itemName
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
unset($data['name']);
|
2015-09-15 15:57:03 +02:00
|
|
|
$config->setSection($itemName, $data);
|
|
|
|
$this->setIniConfig($config);
|
2015-09-07 16:58:22 +02:00
|
|
|
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)
|
|
|
|
{
|
2015-09-15 15:57:03 +02:00
|
|
|
$config = $this->getConfigForItem($name);
|
|
|
|
if ($config === null) {
|
2015-09-07 16:58:22 +02:00
|
|
|
throw new NotFoundError('No navigation item called "%s" found', $name);
|
|
|
|
}
|
|
|
|
|
2015-09-15 15:57:03 +02:00
|
|
|
$itemConfig = $config->getSection($name);
|
2015-09-16 11:51:57 +02:00
|
|
|
|
|
|
|
if ($this->hasBeenShared($name)) {
|
|
|
|
if ((! isset($data['users']) || !$data['users']) && (! isset($data['groups']) || !$data['groups'])) {
|
|
|
|
// It is shared but shouldn't anymore
|
|
|
|
$config->removeSection($name);
|
|
|
|
$this->secondaryConfig = $config;
|
|
|
|
|
|
|
|
if (! $itemConfig->owner || $itemConfig->owner === $this->getUser()->getUsername()) {
|
|
|
|
$config = $this->getUserConfig();
|
|
|
|
} else {
|
|
|
|
$owner = new User($itemConfig->owner);
|
|
|
|
$config = $owner->loadNavigationConfig();
|
|
|
|
}
|
|
|
|
|
|
|
|
unset($itemConfig->owner);
|
|
|
|
unset($itemConfig->users);
|
|
|
|
unset($itemConfig->groups);
|
|
|
|
}
|
|
|
|
} elseif ((isset($data['users']) && $data['users']) || (isset($data['groups']) && $data['groups'])) {
|
|
|
|
if ($this->getUser()->can('application/share/navigation')) {
|
|
|
|
// It is not shared yet but should be
|
|
|
|
$config->removeSection($name);
|
|
|
|
$this->secondaryConfig = $config;
|
|
|
|
$config = $this->getShareConfig();
|
|
|
|
$data['owner'] = $this->getUser()->getUsername();
|
|
|
|
} else {
|
|
|
|
unset($data['users']);
|
|
|
|
unset($data['groups']);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-09-07 16:58:22 +02:00
|
|
|
if (isset($data['name'])) {
|
|
|
|
if ($data['name'] !== $name) {
|
2015-09-15 15:57:03 +02:00
|
|
|
$config->removeSection($name);
|
2015-09-07 16:58:22 +02:00
|
|
|
$name = $data['name'];
|
|
|
|
}
|
|
|
|
|
|
|
|
unset($data['name']);
|
|
|
|
}
|
|
|
|
|
|
|
|
$itemConfig->merge($data);
|
|
|
|
foreach ($itemConfig->toArray() as $k => $v) {
|
|
|
|
if ($v === null) {
|
|
|
|
unset($itemConfig->$k);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-09-15 15:57:03 +02:00
|
|
|
$config->setSection($name, $itemConfig);
|
|
|
|
$this->setIniConfig($config);
|
2015-09-07 16:58:22 +02:00
|
|
|
return $this;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Remove a navigation item
|
|
|
|
*
|
|
|
|
* @param string $name
|
|
|
|
*
|
|
|
|
* @return $this
|
|
|
|
*/
|
|
|
|
public function delete($name)
|
|
|
|
{
|
2015-09-15 15:57:03 +02:00
|
|
|
$config = $this->getConfigForItem($name);
|
|
|
|
if ($config === null) {
|
|
|
|
throw new NotFoundError('No navigation item called "%s" found', $name);
|
|
|
|
}
|
|
|
|
|
|
|
|
$config->removeSection($name);
|
|
|
|
$this->setIniConfig($config);
|
2015-09-07 16:58:22 +02:00
|
|
|
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)
|
|
|
|
{
|
2015-09-15 15:57:03 +02:00
|
|
|
throw new NotFoundError('No navigation item called "%s" found', $name);
|
2015-09-07 16:58:22 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* {@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'
|
|
|
|
)
|
|
|
|
)
|
|
|
|
);
|
|
|
|
|
2015-09-16 11:51:57 +02:00
|
|
|
if ($this->getUser()->can('application/share/navigation')) {
|
|
|
|
$checked = isset($formData['shared']) ? null : (isset($formData['users']) || isset($formData['groups']));
|
|
|
|
|
|
|
|
$this->addElement(
|
|
|
|
'checkbox',
|
|
|
|
'shared',
|
|
|
|
array(
|
|
|
|
'autosubmit' => true,
|
|
|
|
'ignore' => true,
|
|
|
|
'value' => $checked,
|
|
|
|
'label' => $this->translate('Shared'),
|
|
|
|
'description' => $this->translate('Tick this box to share this item with others')
|
|
|
|
)
|
|
|
|
);
|
|
|
|
|
|
|
|
if ($checked || (isset($formData['shared']) && $formData['shared'])) {
|
|
|
|
$this->addElement(
|
|
|
|
'text',
|
|
|
|
'users',
|
|
|
|
array(
|
|
|
|
'label' => $this->translate('Users'),
|
|
|
|
'description' => $this->translate(
|
|
|
|
'Comma separated list of usernames to share this item with'
|
|
|
|
)
|
|
|
|
)
|
|
|
|
);
|
|
|
|
$this->addElement(
|
|
|
|
'text',
|
|
|
|
'groups',
|
|
|
|
array(
|
|
|
|
'label' => $this->translate('Groups'),
|
|
|
|
'description' => $this->translate(
|
|
|
|
'Comma separated list of group names to share this item with'
|
|
|
|
)
|
|
|
|
)
|
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-09-07 16:58:22 +02:00
|
|
|
$this->addElement(
|
|
|
|
'select',
|
|
|
|
'type',
|
|
|
|
array(
|
|
|
|
'required' => true,
|
|
|
|
'autosubmit' => true,
|
|
|
|
'label' => $this->translate('Type'),
|
|
|
|
'description' => $this->translate('The type of this navigation item'),
|
2015-09-15 16:16:32 +02:00
|
|
|
'multiOptions' => array_combine($itemTypes, $itemTypes)
|
2015-09-07 16:58:22 +02:00
|
|
|
)
|
|
|
|
);
|
|
|
|
|
|
|
|
$this->addSubForm($this->getItemForm($itemType)->create($formData), 'item_form');
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Populate the configuration of the navigation item to load
|
|
|
|
*/
|
|
|
|
public function onRequest()
|
|
|
|
{
|
|
|
|
if ($this->itemToLoad) {
|
2015-09-15 15:57:03 +02:00
|
|
|
$data = $this->getConfigForItem($this->itemToLoad)->getSection($this->itemToLoad)->toArray();
|
2015-09-07 16:58:22 +02:00
|
|
|
$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;
|
|
|
|
}
|
|
|
|
|
2015-09-16 10:58:57 +02:00
|
|
|
/**
|
|
|
|
* {@inheritdoc}
|
|
|
|
*/
|
|
|
|
protected function writeConfig(Config $config)
|
|
|
|
{
|
|
|
|
parent::writeConfig($config);
|
|
|
|
|
|
|
|
if ($this->secondaryConfig !== null) {
|
|
|
|
$this->config = $this->secondaryConfig; // Causes the config being displayed to the user in case of an error
|
|
|
|
parent::writeConfig($this->secondaryConfig);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-09-07 16:58:22 +02:00
|
|
|
/**
|
|
|
|
* 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;
|
|
|
|
}
|
|
|
|
|
2015-09-15 15:57:03 +02:00
|
|
|
/**
|
|
|
|
* 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();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-09-16 11:51:57 +02:00
|
|
|
/**
|
|
|
|
* Return whether the given navigation item has been shared
|
|
|
|
*
|
|
|
|
* @param string $name
|
|
|
|
*
|
|
|
|
* @return bool
|
|
|
|
*/
|
|
|
|
protected function hasBeenShared($name)
|
|
|
|
{
|
|
|
|
return $this->getConfigForItem($name) === $this->getShareConfig();
|
|
|
|
}
|
|
|
|
|
2015-09-07 16:58:22 +02:00
|
|
|
/**
|
|
|
|
* 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();
|
|
|
|
}
|
|
|
|
}
|