You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

333 lines
7.4 KiB

1 year ago
<?php declare(strict_types = 0);
/*
** Zabbix
** Copyright (C) 2001-2023 Zabbix SIA
**
** This program is free software; you can redistribute it and/or modify
** it under the terms of the GNU General Public License as published by
** the Free Software Foundation; either version 2 of the License, or
** (at your option) any later version.
**
** This program is distributed in the hope that it will be useful,
** but WITHOUT ANY WARRANTY; without even the implied warranty of
** MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
** GNU General Public License for more details.
**
** You should have received a copy of the GNU General Public License
** along with this program; if not, write to the Free Software
** Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
**/
class CMenuItem extends CTag {
/**
* @var string
*/
private $action;
/**
* @var array
*/
private $aliases = [];
/**
* @var string
*/
private $icon_class;
/**
* @var string
*/
private $label;
/**
* @var CMenu
*/
private $sub_menu;
/**
* @var string
*/
private $title;
/**
* @var string
*/
private $target;
/**
* @var CUrl
*/
private $url;
/**
* @var bool
*/
private $is_selected = false;
/**
* Create menu item.
*
* @param string $label Menu item visual label.
*/
public function __construct(string $label) {
parent::__construct('li', true);
$this->label = $label;
}
/**
* Get action name.
*
* @return string|null
*/
public function getAction(): ?string {
return $this->action;
}
/**
* Set action name and derive a corresponding URL for menu item link.
*
* @param string $action_name Action name.
*
* @return CMenuItem
*/
public function setAction(string $action_name): self {
return $this->setUrl((new CUrl('zabbix.php'))->setArgument('action', $action_name), $action_name);
}
/**
* Get action name aliases.
*
* @return array
*/
public function getAliases(): array {
return $this->aliases;
}
/**
* Set action name aliases.
*
* @param array $aliases The aliases of menu item. Is able to specify the alias in following formats:
* - {action_name} - The alias is applicable to page with specified action name with any GET
* parameters in URL or without them;
* - {action_name}?{param}={value} - The alias is applicable to page with specified action
* when specified GET parameter exists in URL and have the same value;
* - {action_name}?{param}=* - The alias is applicable to page with specified action
* when specified GET parameter exists in URL and have any value;
* - {action_name}?!{param}={value} - The alias is applicable to page with specified action
* when specified GET parameter not exists in URL or have different value;
* - {action_name}?!{param}=* - The alias is applicable to page with specified action
* when specified GET parameter not exists in URL.
*
* @return CMenuItem
*/
public function setAliases(array $aliases): self {
foreach ($aliases as $alias) {
['path' => $action_name, 'query' => $query_string] = parse_url($alias) + ['query' => ''];
parse_str($query_string, $query_params);
$this->aliases[$action_name][] = $query_params;
}
return $this;
}
/**
* Set icon CSS class for menu item link.
*
* @param string $icon_class
*
* @return CMenuItem
*/
public function setIcon(string $icon_class): self {
$this->icon_class = $icon_class;
return $this;
}
/**
* Get visual label of menu item.
*
* @return string
*/
public function getLabel(): string {
return $this->label;
}
/**
* Check if menu item is marked as selected.
*
* @return bool
*/
public function isSelected(): bool {
return $this->is_selected;
}
/**
* Mark menu item as selected.
*
* @return CMenuItem
*/
public function setSelected(): self {
$this->is_selected = true;
$this->addClass('is-selected');
return $this;
}
/**
* Deep find menu item (including this one) by action name and mark the whole chain as selected.
*
* @param string $action_name Action name to search for.
* @param array $request_params Parameters of current HTTP request to compare in search process.
* @param bool $expand Add 'is-expanded' class for selected submenus.
*
* @return bool True, if menu item was selected.
*/
public function setSelectedByAction(string $action_name, array $request_params, bool $expand = false): bool {
if (array_key_exists($action_name, $this->aliases)) {
foreach ($this->aliases[$action_name] as $alias_params) {
$has_mandatory_params = true;
foreach ($alias_params as $name => $value) {
if (!array_key_exists($name, $request_params)
|| ($value !== '*' && $value !== $request_params[$name])) {
$has_mandatory_params = false;
break;
}
}
if ($has_mandatory_params) {
$this->setSelected();
return true;
}
}
}
if ($this->sub_menu !== null && $this->sub_menu->setSelectedByAction($action_name, $request_params, $expand)) {
$this->setSelected();
return true;
}
return false;
}
/**
* Get submenu of menu item or create new one, if not exists.
*
* @return CMenu
*/
public function getSubMenu(): CMenu {
if ($this->sub_menu === null) {
$this->setSubMenu(new CMenu());
}
return $this->sub_menu;
}
/**
* Set submenu for menu item.
*
* @param CMenu $sub_menu
*
* @return CMenuItem
*/
public function setSubMenu(CMenu $sub_menu): self {
$this->sub_menu = $sub_menu->addClass('submenu');
$this->addClass('has-submenu');
return $this;
}
/**
* Check if menu item has submenu.
*
* @return bool
*/
public function hasSubMenu(): bool {
return ($this->sub_menu !== null);
}
/**
* Set target attribute for the menu item link.
*
* @param string $target
*
* @return CMenuItem
*/
public function setTarget(string $target): self {
$this->target = $target;
return $this;
}
/**
* Set title attribute for the menu item link.
*
* @param string $title
*
* @return CMenuItem
*/
public function setTitle($title): self {
$this->title = $title;
return $this;
}
/**
* Get url of the menu item link.
*
* @return CUrl|null
*/
public function getUrl(): ?CUrl {
return $this->url;
}
/**
* Set url for the menu item link.
*
* @param CUrl $url
* @param string|null $action_name Associate action name to be matched by setSelected method.
*
* @return CMenuItem
*/
public function setUrl(CUrl $url, string $action_name = null): self {
$action = null;
if ($action_name !== null) {
$this->setAliases([$action_name]);
['path' => $action] = parse_url($action_name);
}
$this->url = $url;
$this->action = $action;
return $this;
}
public function toString($destroy = true)
{
if ($this->url !== null || $this->sub_menu !== null) {
$this->addItem([
(new CLink($this->label, $this->sub_menu !== null ? '#' : $this->url->getUrl()))
->addClass($this->icon_class)
->setTitle($this->title)
->setTarget($this->target),
$this->sub_menu
]);
}
else {
$this->addItem(
(new CSpan($this->label))
->addClass($this->icon_class)
->setTitle($this->title)
);
}
return parent::toString($destroy);
}
}