Update dashboard implementation

fix javascript naming (icinga-url to icingaurl), fix add form, add remove
button, code style changes

refs #4192
This commit is contained in:
Jannis Moßhammer 2013-08-06 11:53:42 +02:00
parent b3e0d5e8ce
commit f8bb478f0e
23 changed files with 745 additions and 282 deletions

View File

@ -84,6 +84,8 @@ class AuthenticationController extends ActionController
} }
} }
/** /**
* Action handle logout * Action handle logout
*/ */

View File

@ -0,0 +1,86 @@
<?php
use Icinga\Web\ActionController;
use Icinga\Application\Config;
use Icinga\Web\Url;
use Icinga\Application\Icinga;
use Icinga\Web\Widget\Dashboard;
use Icinga\Application\Config as IcingaConfig;
use Icinga\Exception\ConfigurationError;
use Icinga\Form\Dashboard\AddUrlForm;
class DashboardController extends ActionController
{
private function getDashboard($config = 'dashboard/dashboard')
{
$dashboard = new Dashboard();
$dashboard->readConfig(IcingaConfig::app($config));
return $dashboard;
}
public function removecomponentAction()
{
$pane = $this->_getParam('pane');
$dashboard = $this->getDashboard();
$dashboard->removeComponent(
$pane,
$this->_getParam('component')
)->store();
// When the pane doesn't exist anymore, display the default pane
if ($dashboard->isEmptyPane($pane)) {
$this->redirectNow(Url::fromPath('dashboard'));
return;
}
$this->redirectNow(Url::fromPath('dashboard', array('pane' => $pane)));
}
public function addurlAction()
{
$form = new AddUrlForm();
$form->setRequest($this->_request);
$this->view->form = $form;
if ($form->isSubmittedAndValid()) {
$dashboard = $this->getDashboard();
$dashboard->setComponentUrl(
$form->getValue('pane'),
$form->getValue('component'),
ltrim($form->getValue('url'), '/')
);
$this->persistDashboard($dashboard);
$this->redirectNow(
Url::fromPath(
'dashboard',
array(
'pane' => $form->getValue('pane')
)
)
);
}
}
private function persistDashboard(Dashboard $dashboard)
{
$dashboard->store();
}
public function indexAction()
{
$dashboard = $this->getDashboard();
if ($this->_getParam('dashboard')) {
$dashboardName = $this->_getParam('dashboard');
$dashboard->activate($dashboardName);
}
$this->view->tabs = $dashboard->getTabs();
$this->view->tabs->add("Add", array(
"title" => "Add Url",
"iconCls" => "plus",
"url" => Url::fromPath("dashboard/addurl")
));
$this->view->dashboard = $dashboard;
}
}

View File

@ -72,8 +72,9 @@ class StaticController extends ActionController
public function imgAction() public function imgAction()
{ {
$module = $this->_getParam('moduleName'); $module = $this->_getParam('module_name');
$file = $this->_getParam('file'); $file = $this->_getParam('file');
$basedir = Icinga::app()->getModuleManager()->getModule($module)->getBaseDir(); $basedir = Icinga::app()->getModuleManager()->getModule($module)->getBaseDir();
$filePath = $basedir . '/public/img/' . $file; $filePath = $basedir . '/public/img/' . $file;
@ -102,7 +103,7 @@ class StaticController extends ActionController
) . ' GMT'); ) . ' GMT');
readfile($filePath); readfile($filePath);
$this->_viewRenderer->setNoRender(); return;
} }
public function javascriptAction() public function javascriptAction()
@ -110,17 +111,16 @@ class StaticController extends ActionController
$module = $this->_getParam('module_name'); $module = $this->_getParam('module_name');
$file = $this->_getParam('file'); $file = $this->_getParam('file');
if (!Icinga::app()->getModuleManager()->hasEnabled($module)) {
echo "/** Module not enabled **/";
return;
}
$basedir = Icinga::app()->getModuleManager()->getModule($module)->getBaseDir(); $basedir = Icinga::app()->getModuleManager()->getModule($module)->getBaseDir();
$filePath = $basedir . '/public/js/' . $file; $filePath = $basedir . '/public/js/' . $file;
if (!file_exists($filePath)) { if (!file_exists($filePath)) {
throw new ActionException( echo "/** Module has no js files **/";
sprintf( return;
'%s does not exist',
$filePath
),
404
);
} }
$hash = md5_file($filePath); $hash = md5_file($filePath);
$response = $this->getResponse(); $response = $this->getResponse();
@ -143,7 +143,8 @@ class StaticController extends ActionController
} else { } else {
readfile($filePath); readfile($filePath);
} }
$this->_viewRenderer->setNoRender();
return;
} }
} }

View File

@ -0,0 +1,102 @@
<?php
namespace Icinga\Form\Dashboard;
use Icinga\Application\Config as IcingaConfig;
use Icinga\Web\Form;
use Icinga\Web\Widget\Dashboard;
use Icinga\Web\Widget\Dashboard\Component;
use Zend_Form_Element_Text;
use Zend_Form_Element_Submit;
use Zend_Form_Element_Hidden;
class AddUrlForm extends Form
{
private function addPaneSelectionBox(Dashboard $dashboard)
{
$selectPane = new \Zend_Form_Element_Select('pane', array(
'label' => 'Dashboard',
'required' => true,
'style' => 'display:inline-block',
'multiOptions' => $dashboard->getPaneKeyTitleArray()
));
$newDashboardBtn = new \Zend_Form_Element_Submit('create_new_pane', array(
'label' => '+',
'required' => false,
'style' => 'display:inline-block'
));
$newDashboardBtn->removeDecorator('DtDdWrapper');
$selectPane->removeDecorator('DtDdWrapper');
$selectPane->removeDecorator('htmlTag');
$this->addElement($selectPane);
$this->addElement($newDashboardBtn);
$this->enableAutoSubmit(array('create_new_pane'));
}
private function addNewPaneTextField()
{
$txtCreatePane = new Zend_Form_Element_Text('pane', array(
'label' => 'New dashboard title',
'required' => true,
'style' => 'display:inline-block'
));
/**
* Marks this field as a new pane (and prevents the checkbox being displayed when validation errors occur)
*/
$markAsNewPane = new Zend_Form_Element_Hidden('create_new_pane', array(
'required' => true,
'value' => 1
));
$cancelDashboardBtn = new Zend_Form_Element_Submit('use_existing_dashboard', array(
'label' => 'X',
'required' => false,
'style' => 'display:inline-block'
));
$cancelDashboardBtn->removeDecorator('DtDdWrapper');
$txtCreatePane->removeDecorator('DtDdWrapper');
$txtCreatePane->removeDecorator('htmlTag');
$this->addElement($txtCreatePane);
$this->addElement($cancelDashboardBtn);
$this->addElement($markAsNewPane);
}
/**
* Add elements to this form (used by extending classes)
*/
protected function create()
{
$dashboard = new Dashboard();
$dashboard->readConfig(IcingaConfig::app('dashboard/dashboard'));
$this->addElement('text', 'url', array(
'label' => 'Url',
'required' => true,
));
$elems = $dashboard->getPaneKeyTitleArray();
if (empty($elems) || // show textfield instead of combobox when no pane is available
($this->getRequest()->getPost('create_new_pane', '0') && // or when a new pane should be created (+ button)
! $this->getRequest()->getPost('use_existing_dashboard', '0')) // and the user didn't click the 'use existing' button
) {
$this->addNewPaneTextField();
} else {
$this->addPaneSelectionBox($dashboard);
}
$this->addElement('text', 'component', array(
'label' => 'Title',
'required' => true,
));
$this->setSubmitLabel("Add to dashboard");
}
}

View File

@ -0,0 +1,11 @@
[test]
title = "test"
[test.test]
url = "test"
[test.test2]
url = "test2"
[test.dsgdgs]
url = "dsdsgdsg"

View File

@ -1,3 +1,4 @@
[menu] [menu]
Dashboard = "/dashboard/index"
Configuration = "/configuration/index" Configuration = "/configuration/index"

View File

@ -41,6 +41,10 @@ some fields in the form) the form is repopulated but not validated at this time.
in this case, but no errors are added to the created form. in this case, but no errors are added to the created form.
In order to be able to use isSubmittedAndValid, you have to define a submitbutton in the form.
This is done with the *setSubmitLabel(string)* function, with the first parameter being the
label set to the submit button.
#### Pre validation #### Pre validation
To handle dependend fields you can just override *preValid()* or *postValid()* To handle dependend fields you can just override *preValid()* or *postValid()*

View File

@ -45,7 +45,7 @@ class Config extends Zend_Config_Ini
* The INI file this configuration has been loaded from * The INI file this configuration has been loaded from
* @var string * @var string
*/ */
protected $configFile; private $configFile;
/** /**
* Application config instances per file * Application config instances per file
@ -135,4 +135,11 @@ class Config extends Zend_Config_Ini
return array_keys($this->$name->toArray()); return array_keys($this->$name->toArray());
} }
} }
public function getConfigFile()
{
return $this->configFile;
}
} }

View File

@ -0,0 +1,114 @@
<?php
/**
* Created by JetBrains PhpStorm.
* User: moja
* Date: 8/6/13
* Time: 11:13 AM
* To change this template use File | Settings | File Templates.
*/
namespace Icinga\Util;
use InvalidArgumentException;
class Dimension
{
/**
* Defines this dimension as nr of pixels
*/
const UNIT_PX = "px";
/**
* Defines this dimension as width of 'M' in current font
*/
const UNIT_EM = "em";
/**
* Defines this dimension as a percentage value
*/
const UNIT_PERCENT = "%";
/**
* Defines this dimension in points
*/
const UNIT_PT = "pt";
/**
* The current set value for this dimension
*
* @var int
*/
private $value = 0;
/**
* The unit to interpret the value with
*
* @var string
*/
private $unit = self::UNIT_PX;
/**
* Creates a new Dimension object with the given size and unit
*
* @param int $value The new value
* @param string $unit The unit to use (default: px)
*/
public function __construct($value, $unit = self::UNIT_PX)
{
$this->setValue($value, $unit);
}
/**
* Change the value and unit of this dimension
*
* @param int $value The new value
* @param string $unit The unit to use (default: px)
*/
public function setValue($value, $unit = self::UNIT_PX)
{
$this->value = intval($value);
$this->unit = $unit;
}
/**
* Returns true when the value is > 0
*
* @return bool
*/
public function isDefined()
{
return $this->value > 0;
}
/**
* Returns the underlying value without unit information
*
* @return int
*/
public function getValue()
{
return $this->value;
}
/**
* Returns this value with it's according unit as a string
*
* @return string
*/
public function __toString()
{
if (!$this->isDefined()) {
return "";
}
return $this->value.$this->unit;
}
public static function fromString($string)
{
$matches = array();
if (!preg_match_all('/^ *([0-9]+)(px|pt|em|\%) */i', $string, $matches)) {
throw new InvalidArgumentException($string.' is not a valid dimension');
}
return new Dimension(intval($matches[1]), $matches[2]);
}
}

View File

@ -237,7 +237,7 @@ class ActionController extends ZfController
public function redirectNow($url, array $params = array()) public function redirectNow($url, array $params = array())
{ {
if ($url instanceof Url) { if ($url instanceof Url) {
$url = $url->getRelative(); $url = $url->getRelativeUrl();
} }
$this->_helper->Redirector->gotoUrlAndExit($url); $this->_helper->Redirector->gotoUrlAndExit($url);
} }

View File

@ -121,6 +121,16 @@ class Url
return $this; return $this;
} }
/**
* Return the baseUrl set for this Url
*
* @return string
*/
public function getBaseUrl()
{
return $this->baseUrl;
}
/** /**
* Set the relative path of this url, without query parameters * Set the relative path of this url, without query parameters
* *
@ -302,6 +312,8 @@ class Url
return $url; return $url;
} }
/** /**
* Alias for @see Url::getAbsoluteUrl() * Alias for @see Url::getAbsoluteUrl()
* @return mixed * @return mixed

View File

@ -3,28 +3,30 @@
namespace Icinga\Web\Widget; namespace Icinga\Web\Widget;
use Icinga\Application\Icinga; use Icinga\Application\Icinga;
use Icinga\Config\Config; use Icinga\Config\Config as IcingaConfig;;
use Icinga\Application\Logger;
use Icinga\Exception\ConfigurationError;
use Icinga\Web\Widget\Widget; use Icinga\Web\Widget\Widget;
use Icinga\Web\Widget\Dashboard\Pane; use Icinga\Web\Widget\Dashboard\Pane;
use Icinga\Web\Widget\Dashboard\Component as DashboardComponent;
use Icinga\Web\Url; use Icinga\Web\Url;
use Zend_Config as ZfConfig;
class Dashboard implements Widget class Dashboard implements Widget
{ {
/** /**
* @var Config * @var IcingaConfig;
*/ */
protected $config; private $config;
protected $configfile; private $configfile;
protected $panes = array(); private $panes = array();
protected $tabs; private $tabs;
protected $properties = array( private $url = null;
'url' => null, private $tabParam = 'pane';
'tabParam' => 'pane'
);
protected function init()
public function __construct()
{ {
if ($this->url === null) { if ($this->url === null) {
$this->url = Url::fromRequest()->getUrlWithout($this->tabParam); $this->url = Url::fromRequest()->getUrlWithout($this->tabParam);
@ -33,14 +35,15 @@ class Dashboard implements Widget
public function activate($name) public function activate($name)
{ {
$this->tabs()->activate($name); $this->getTabs()->activate($name);
} }
public function tabs() public function getTabs()
{ {
if ($this->tabs === null) { if ($this->tabs === null) {
$this->tabs = new Tabs(); $this->tabs = new Tabs();
foreach ($this->panes as $key => $pane) { foreach ($this->panes as $key => $pane) {
$this->tabs->add($key, array( $this->tabs->add($key, array(
'title' => $pane->getTitle(), 'title' => $pane->getTitle(),
'url' => clone($this->url), 'url' => clone($this->url),
@ -57,29 +60,57 @@ class Dashboard implements Widget
return is_writable($this->configfile); return is_writable($this->configfile);
} }
public function store() public function store($file = null)
{ {
if ($file === null) {
$file = IcingaConfig::app('dashboard/dashboard')->getConfigFile();
}
$this->configfile = $file;
if (!$this->isWritable()) {
Logger::error("Tried to persist dashboard to %s, but path is not writeable", $this->configfile);
throw new ConfigurationError('Can\'t persist dashboard');
}
if (! @file_put_contents($this->configfile, $this->toIni())) { if (! @file_put_contents($this->configfile, $this->toIni())) {
return false; $error = error_get_last();
if ($error == NULL) {
$error = "Unknown error";
} else {
$error = $error["message"];
}
Logger::error("Tried to persist dashboard to %s, but got error: %s", $this->configfile, $error);
throw new ConfigurationError('Can\'t persist dashboard');
} else { } else {
return $this; return $this;
} }
} }
public function readConfig(ZfConfig $config) public function readConfig(IcingaConfig $config)
{ {
$this->configfile = Icinga::app('dashboard')->getApplicationDir("dashboard");
$this->config = $config; $this->config = $config;
$this->panes = array(); $this->panes = array();
$this->loadConfigPanes(); $this->loadConfigPanes();
return $this; return $this;
} }
public function createPane($title)
{
$pane = new Pane($title);
$pane->setTitle($title);
$this->addPane($pane);
}
public function setComponentUrl($pane, $component, $url) public function setComponentUrl($pane, $component, $url)
{ {
if ($component === null && strpos($pane, '.')) { if ($component === null && strpos($pane, '.')) {
list($pane, $component) = preg_split('~\.~', $pane, 2); list($pane, $component) = preg_split('~\.~', $pane, 2);
} }
if (!isset($this->panes[$pane])) {
$this->createPane($pane);
}
$pane = $this->getPane($pane); $pane = $this->getPane($pane);
if ($pane->hasComponent($component)) { if ($pane->hasComponent($component)) {
$pane->getComponent($component)->setUrl($url); $pane->getComponent($component)->setUrl($url);
@ -89,16 +120,30 @@ class Dashboard implements Widget
return $this; return $this;
} }
public function isEmptyPane($pane)
{
$paneObj = $this->getPane($pane);
if ($paneObj === null) {
return true;
}
$cmps = $paneObj->getComponents();
return !empty($cmps);
}
public function removeComponent($pane, $component) public function removeComponent($pane, $component)
{ {
if ($component === null && strpos($pane, '.')) { if ($component === null && strpos($pane, '.')) {
list($pane, $component) = preg_split('~\.~', $pane, 2); list($pane, $component) = preg_split('~\.~', $pane, 2);
} }
$this->getPane($pane)->removeComponent($component); $pane = $this->getPane($pane);
if ($pane !== null) {
$pane->removeComponent($component);
}
return $this; return $this;
} }
public function paneEnum() public function getPaneKeyTitleArray()
{ {
$list = array(); $list = array();
foreach ($this->panes as $name => $pane) { foreach ($this->panes as $name => $pane) {
@ -127,6 +172,8 @@ class Dashboard implements Widget
public function getPane($name) public function getPane($name)
{ {
if (!isset($this->panes[$name]))
return null;
return $this->panes[$name]; return $this->panes[$name];
} }
@ -136,22 +183,31 @@ class Dashboard implements Widget
return ''; return '';
} }
return $this->tabs() . $this->getActivePane(); return $this->getActivePane()->render($view);
}
private function setDefaultPane()
{
reset($this->panes);
$active = key($this->panes);
$this->activate($active);
return $active;
} }
public function getActivePane() public function getActivePane()
{ {
$active = $this->tabs()->getActiveName(); $active = $this->getTabs()->getActiveName();
if (! $active) { if (! $active) {
if ($active = Url::fromRequest()->getParam($this->tabParam)) { if ($active = Url::fromRequest()->getParam($this->tabParam)) {
$this->activate($active); if ($this->isEmptyPane($active)) {
$active = $this->setDefaultPane();
} else {
$this->activate($active);
}
} else { } else {
reset($this->panes); $active = $this->setDefaultPane();
$active = key($this->panes);
$this->activate($active);
} }
} }
return $this->panes[$active]; return $this->panes[$active];
} }
@ -166,31 +222,20 @@ class Dashboard implements Widget
protected function loadConfigPanes() protected function loadConfigPanes()
{ {
$items = $this->config->keys(); $items = $this->config;
$app = Icinga::app(); $app = Icinga::app();
foreach ($items as $key => $item) { foreach ($items->keys() as $key) {
$item = $this->config->get($key, false);
if (false === strstr($key, '.')) { if (false === strstr($key, '.')) {
$pane = new Pane($key); $this->addPane(Pane::fromIni($key, $item));
if (isset($item['title'])) {
$pane->setTitle($item['title']);
}
$this->addPane($pane);
} else { } else {
list($dashboard, $title) = preg_split('~\.~', $key, 2); list($paneName, $title) = explode('.', $key , 2);
$base_url = $item['base_url']; $pane = $this->getPane($paneName);
$pane->addComponent(DashboardComponent::fromIni($title, $item, $pane));
$module = substr($base_url, 0, strpos($base_url, '/'));
$whitelist = array();
if (! $app->hasModule($module)) {
continue;
}
unset($item['base_url']);
$this->getPane($dashboard)->addComponent(
$title,
Url::fromPath($base_url, $item)
);
} }
} }
} }
} }

View File

@ -2,23 +2,48 @@
namespace Icinga\Web\Widget\Dashboard; namespace Icinga\Web\Widget\Dashboard;
use Icinga\Util\Dimension;
use Icinga\Web\Url; use Icinga\Web\Url;
use Icinga\Web\Widget\Widget;
use Zend_Config;
/** /**
* A dashboard pane component * A dashboard pane component
* *
* Needs a title and an URL * Needs a title and an URL
* // TODO: Rename to "Dashboardlet"
* *
*/ */
class Component class Component implements Widget
{ {
protected $url; private $url;
protected $title; private $title;
private $width;
private $height;
public function __construct($title, $url) /**
* @var Pane
*/
private $pane;
private $template =<<<'EOD'
<div class="icinga-container dashboard" icingatitle="{TITLE}" style="{DIMENSION}">
<a href="{URL}"> {TITLE}</a>
<a class="btn btn-danger btn-mini pull-right" href="{REMOVE_URL}" style="color:black">X</a>
<div class="icinga-container" icingaurl="{URL}">
<noscript>
<iframe src="{URL}" style="height:100%; width:99%" frameborder="no"></iframe>
</noscript>
</div>
</div>
EOD;
public function __construct($title, $url, Pane $pane)
{ {
$this->title = $title; $this->title = $title;
$this->pane = $pane;
if ($url instanceof Url) { if ($url instanceof Url) {
$this->url = $url; $this->url = $url;
} else { } else {
@ -26,6 +51,16 @@ class Component
} }
} }
public function setWidth(Dimension $width = null)
{
$this->width = $width;
}
public function setHeight(Dimension $height = null)
{
$this->height = $height;
}
/** /**
* Retrieve this components title * Retrieve this components title
* *
@ -78,39 +113,74 @@ class Component
public function toIni() public function toIni()
{ {
$ini = $this->iniPair('base_url', $this->url->getScript()); $ini = $this->iniPair('url', $this->url->getRelativeUrl());
foreach ($this->url->getParams() as $key => $val) { foreach ($this->url->getParams() as $key => $val) {
$ini .= $this->iniPair($key, $val); $ini .= $this->iniPair($key, $val);
} }
if ($this->height !== null) {
$ini .= 'height: '.((string) $this->height).'\n';
}
if ($this->width !== null) {
$ini .= 'width: '.((string) $this->width).'\n';
}
return $ini; return $ini;
} }
/** public function render(\Zend_View_Abstract $view)
* Render this components HTML
*/
public function __toString()
{ {
$url = clone($this->url); $url = clone($this->url);
$url->addParams(array('view' => 'compact')); $url->addParams(array('view' => 'compact'));
if (isset($_GET['layout'])) { if (isset($_GET['layout'])) {
$url->addParams(array('layout' => $_GET['layout'])); $url->addParams(array('layout' => $_GET['layout']));
} }
$removeUrl = Url::fromPath(
"/dashboard/removecomponent",
array(
"pane" => $this->pane->getName(),
"component" => $this->getTitle()
)
);
$htm = '<div class="icinga-container dashboard" icingaurl="'
. $url $html = str_replace("{URL}", $url->getAbsoluteUrl(), $this->template);
. '" icingatitle="' $html = str_replace("{REMOVE_URL}", $removeUrl, $html);
. htmlspecialchars($this->title) $html = str_replace("{STYLE}", $this->getBoxSizeAsCSS(), $html);
. '">' $html = str_replace("{TITLE}", $view->escape($this->getTitle()), $html);
. "\n" return $html;
. '<h1><a href="' }
. $this->url
. '">' private function getBoxSizeAsCSS()
. htmlspecialchars($this->title) {
. "</a></h1>\n" $style = "";
. '<noscript><iframe src="' if ($this->height) {
. $url->addParams(array('layout' => 'embedded', 'iframe' => 'true')) $style .= 'height:'.(string) $this->height.';';
. '" style="height:100%; width:99%" frameborder="no"></iframe></noscript>' }
. "\n</div>\n"; if ($this->width) {
return $htm; $style .= 'width:'.(string) $this->width.';';
}
}
public static function fromIni($title, Zend_Config $config, Pane $pane)
{
$height = null;
$width = null;
$url = $config->get('url');
$parameters = $config->toArray();
unset($parameters["url"]); // otherwise there's an url = parameter in the Url
if (isset($parameters["height"])) {
$height = Dimension::fromString($parameters["height"]);
unset($parameters["height"]);
}
if (isset($parameters["width"])) {
$width = Dimension::fromString($parameters["width"]);
unset($parameters["width"]);
}
$cmp = new Component($title, Url::fromPath($url, $parameters), $pane);
$cmp->setHeight($height);
$cmp->setWidth($width);
return $cmp;
} }
} }

View File

@ -4,8 +4,10 @@ namespace Icinga\Web\Widget\Dashboard;
use Icinga\Web\Url; use Icinga\Web\Url;
use Icinga\Exception\ConfigurationError; use Icinga\Exception\ConfigurationError;
use Icinga\Web\Widget\Widget;
use Zend_Config;
class Pane class Pane implements Widget
{ {
protected $name; protected $name;
protected $title; protected $title;
@ -61,13 +63,22 @@ class Pane
{ {
return $this->components; return $this->components;
} }
public function render(\Zend_View_Abstract $view)
{
$html = PHP_EOL;
foreach ($this->getComponents() as $component) {
$html .= PHP_EOL.$component->render($view);
}
return $html;
}
public function addComponent($component, $url = null) public function addComponent($component, $url = null)
{ {
if ($component instanceof Component) { if ($component instanceof Component) {
$this->components[$component->title] = $component; $this->components[$component->getTitle()] = $component;
} elseif (is_string($component) && $url !== null) { } elseif (is_string($component) && $url !== null) {
$this->components[$component] = new Component($component, $url); $this->components[$component] = new Component($component, $url, $this);
} else{ } else{
throw new ConfigurationError('You messed up your dashboard'); throw new ConfigurationError('You messed up your dashboard');
} }
@ -81,24 +92,28 @@ class Pane
public function toIni() public function toIni()
{ {
$ini = sprintf( if (empty($this->components))
"[%s]\ntitle = %s\n", {
$this->getName(), return "";
$this->quoteIni($this->getTitle()) }
) . "\n"; $ini = '['.$this->getName().']'.PHP_EOL.
'title = '.$this->quoteIni($this->getTitle()).PHP_EOL;
foreach ($this->components as $title => $component) { foreach ($this->components as $title => $component) {
$ini .= sprintf( // component header
"[%s.%s]\n", $ini .= '['.$this->getName().'.'.$title.']'.PHP_EOL;
$this->getName(), // component content
$title $ini .= $component->toIni().PHP_EOL;
) . $component->toIni() . "\n";
} }
return $ini; return $ini;
} }
public function __toString() public static function fromIni($title, Zend_Config $config)
{ {
return implode('', $this->components); $pane = new Pane($title);
if ($config->get('title', false)) {
$pane->setTitle($config->get('title'));
}
return $pane;
} }
} }

View File

@ -1,78 +0,0 @@
<?php
namespace Icinga\Web\Widget;
use Icinga\Exception\ProgrammingError;
use Icinga\Application\Icinga;
/**
* A form loader
*
* @copyright Copyright (c) 2013 Icinga-Web Team <info@icinga.org>
* @author Icinga-Web Team <info@icinga.org>
* @license http://www.gnu.org/copyleft/gpl.html GNU General Public License
*/
class Form
{
protected $form;
protected $properties = array(
'name' => null,
'options' => null
);
public function __call($func, $args)
{
return call_user_func_array(array($this->form, $func), $args);
}
protected function init()
{
// Load form by name given in props:
$file = null;
$fparts = array();
$cparts = array();
foreach (preg_split('~/~', $this->name, -1, PREG_SPLIT_NO_EMPTY) as $part) {
$fparts[] = $part;
$cparts[] = ucfirst($part);
}
array_push($fparts, ucfirst(array_pop($fparts)));
$app = Icinga::app();
$module_name = $this->view()->module_name;
if ($module_name === 'default') {
$module_name = null;
}
if ($module_name !== null) {
$fname = $app->getModuleManager()->getModule($module_name)->getBaseDir()
. '/application/forms/'
. implode('/', $fparts)
. 'Form.php';
if (file_exists($fname)) {
$file = $fname;
array_unshift($cparts, ucfirst($module_name));
}
}
if ($file === null) {
$fname = $app->getApplicationDir('forms/')
. implode('/', $fparts)
. 'Form.php';
if (file_exists($fname)) {
$file = $fname;
} else {
throw new ProgrammingError(sprintf(
'Unable to load your form: %s',
$this->name
));
}
}
$class = 'Icinga\\Web\\Form\\' . implode('_', $cparts) . 'Form';
require_once($file);
$this->form = new $class($this->options);
}
public function render()
{
return (string) $this->form;
}
}

View File

@ -41,6 +41,7 @@ class Tab implements Widget
private $url = null; private $url = null;
private $urlParams = array(); private $urlParams = array();
private $icon = null; private $icon = null;
private $iconCls = null;
/** /**
@ -59,6 +60,16 @@ class Tab implements Widget
return $this->icon; return $this->icon;
} }
public function setIconCls($iconCls)
{
$this->iconCls = $iconCls;
}
public function getIconCls()
{
return $this->iconCls;
}
/** /**
* @param mixed $name * @param mixed $name
*/ */
@ -107,6 +118,22 @@ class Tab implements Widget
return $this->url; return $this->url;
} }
/**
* @param mixed $url
*/
public function setUrlParams(array $urlParams)
{
$this->urlParams = $urlParams;
}
/**
* @return mixed
*/
public function getUrlParams()
{
return $this->urlParams;
}
public function __construct(array $properties = array()) public function __construct(array $properties = array())
{ {
foreach ($properties as $name=>$value) { foreach ($properties as $name=>$value) {
@ -115,18 +142,6 @@ class Tab implements Widget
$this->$setter($value); $this->$setter($value);
} }
} }
}
/**
* Health check at initialization time
*
* @throws Icinga\Exception\ProgrammingError if tab name is missing
*
* @return void
*/
protected function init()
{
if ($this->name === null) { if ($this->name === null) {
throw new ProgrammingError('Cannot create a nameless tab'); throw new ProgrammingError('Cannot create a nameless tab');
} }
@ -172,6 +187,8 @@ class Tab implements Widget
'width' => 16, 'width' => 16,
'height' => 16 'height' => 16
)) . ' ' . $caption; )) . ' ' . $caption;
} else if ($this->iconCls !== null) {
$caption = '<i class="icon-'.$this->iconCls.'"></i> ' . $caption;
} }
if ($this->url !== null) { if ($this->url !== null) {
$tab = $view->qlink( $tab = $view->qlink(

View File

@ -197,6 +197,7 @@ class Tabs implements Countable, Widget
$html .= $tab->render($view); $html .= $tab->render($view);
} }
// @TODO: Remove special part from tabs (Bug #4512)
$special = array(); $special = array();
$special[] = $view->qlink( $special[] = $view->qlink(
$view->img('img/classic/application-pdf.png') . ' PDF', $view->img('img/classic/application-pdf.png') . ' PDF',

View File

@ -5,7 +5,7 @@
// Last Update : 2011-04-15 // Last Update : 2011-04-15
// //
// Description : Example 059 for TCPDF class // Description : Example 059 for TCPDF class
// Table Of Content using HTML templates. // Table Of Content using HTML Templates.
// //
// Author: Nicola Asuni // Author: Nicola Asuni
// //
@ -19,7 +19,7 @@
/** /**
* Creates an example PDF TEST document using TCPDF * Creates an example PDF TEST document using TCPDF
* @package com.tecnick.tcpdf * @package com.tecnick.tcpdf
* @abstract TCPDF - Example: Table Of Content using HTML templates. * @abstract TCPDF - Example: Table Of Content using HTML Templates.
* @author Nicola Asuni * @author Nicola Asuni
* @since 2010-05-06 * @since 2010-05-06
*/ */
@ -153,7 +153,7 @@ $bookmark_templates = array();
/* /*
* The key of the $bookmark_templates array represent the bookmark level (from 0 to n). * The key of the $bookmark_templates array represent the bookmark level (from 0 to n).
* The following templates will be replaced with proper content: * The following Templates will be replaced with proper content:
* #TOC_DESCRIPTION# this will be replaced with the bookmark description; * #TOC_DESCRIPTION# this will be replaced with the bookmark description;
* #TOC_PAGE_NUMBER# this will be replaced with page number. * #TOC_PAGE_NUMBER# this will be replaced with page number.
* *
@ -166,7 +166,7 @@ $bookmark_templates = array();
$bookmark_templates[0] = '<table border="0" cellpadding="0" cellspacing="0" style="background-color:#EEFAFF"><tr><td width="155mm"><span style="font-family:times;font-weight:bold;font-size:12pt;color:black;">#TOC_DESCRIPTION#</span></td><td width="25mm"><span style="font-family:courier;font-weight:bold;font-size:12pt;color:black;" align="right">#TOC_PAGE_NUMBER#</span></td></tr></table>'; $bookmark_templates[0] = '<table border="0" cellpadding="0" cellspacing="0" style="background-color:#EEFAFF"><tr><td width="155mm"><span style="font-family:times;font-weight:bold;font-size:12pt;color:black;">#TOC_DESCRIPTION#</span></td><td width="25mm"><span style="font-family:courier;font-weight:bold;font-size:12pt;color:black;" align="right">#TOC_PAGE_NUMBER#</span></td></tr></table>';
$bookmark_templates[1] = '<table border="0" cellpadding="0" cellspacing="0"><tr><td width="5mm">&nbsp;</td><td width="150mm"><span style="font-family:times;font-size:11pt;color:green;">#TOC_DESCRIPTION#</span></td><td width="25mm"><span style="font-family:courier;font-weight:bold;font-size:11pt;color:green;" align="right">#TOC_PAGE_NUMBER#</span></td></tr></table>'; $bookmark_templates[1] = '<table border="0" cellpadding="0" cellspacing="0"><tr><td width="5mm">&nbsp;</td><td width="150mm"><span style="font-family:times;font-size:11pt;color:green;">#TOC_DESCRIPTION#</span></td><td width="25mm"><span style="font-family:courier;font-weight:bold;font-size:11pt;color:green;" align="right">#TOC_PAGE_NUMBER#</span></td></tr></table>';
$bookmark_templates[2] = '<table border="0" cellpadding="0" cellspacing="0"><tr><td width="10mm">&nbsp;</td><td width="145mm"><span style="font-family:times;font-size:10pt;color:#666666;"><i>#TOC_DESCRIPTION#</i></span></td><td width="25mm"><span style="font-family:courier;font-weight:bold;font-size:10pt;color:#666666;" align="right">#TOC_PAGE_NUMBER#</span></td></tr></table>'; $bookmark_templates[2] = '<table border="0" cellpadding="0" cellspacing="0"><tr><td width="10mm">&nbsp;</td><td width="145mm"><span style="font-family:times;font-size:10pt;color:#666666;"><i>#TOC_DESCRIPTION#</i></span></td><td width="25mm"><span style="font-family:courier;font-weight:bold;font-size:10pt;color:#666666;" align="right">#TOC_PAGE_NUMBER#</span></td></tr></table>';
// add other bookmark level templates here ... // add other bookmark level Templates here ...
// add table of content at page 1 // add table of content at page 1
// (check the example n. 45 for a text-only TOC // (check the example n. 45 for a text-only TOC

View File

@ -76,7 +76,7 @@
// dullus for text Justification. // dullus for text Justification.
// Bob Vincent (pillarsdotnet@users.sourceforge.net) for <li> value attribute. // Bob Vincent (pillarsdotnet@users.sourceforge.net) for <li> value attribute.
// Patrick Benny for text stretch suggestion on Cell(). // Patrick Benny for text stretch suggestion on Cell().
// Johannes Güntert for JavaScript support. // Johannes G<EFBFBD>ntert for JavaScript support.
// Denis Van Nuffelen for Dynamic Form. // Denis Van Nuffelen for Dynamic Form.
// Jacek Czekaj for multibyte justification // Jacek Czekaj for multibyte justification
// Anthony Ferrara for the reintroduction of legacy image methods. // Anthony Ferrara for the reintroduction of legacy image methods.
@ -87,7 +87,7 @@
// Mohamad Ali Golkar, Saleh AlMatrafe, Charles Abbott for Arabic and Persian support. // Mohamad Ali Golkar, Saleh AlMatrafe, Charles Abbott for Arabic and Persian support.
// Moritz Wagner and Andreas Wurmser for graphic functions. // Moritz Wagner and Andreas Wurmser for graphic functions.
// Andrew Whitehead for core fonts support. // Andrew Whitehead for core fonts support.
// Esteban Joël Marín for OpenType font conversion. // Esteban Jo<EFBFBD>l Mar<61>n for OpenType font conversion.
// Teus Hagen for several suggestions and fixes. // Teus Hagen for several suggestions and fixes.
// Yukihiro Nakadaira for CID-0 CJK fonts fixes. // Yukihiro Nakadaira for CID-0 CJK fonts fixes.
// Kosmas Papachristos for some CSS improvements. // Kosmas Papachristos for some CSS improvements.
@ -647,7 +647,7 @@ class TCPDF {
protected $footer_font; protected $footer_font;
/** /**
* Language templates. * Language Templates.
* @protected * @protected
*/ */
protected $l; protected $l;
@ -6123,7 +6123,7 @@ class TCPDF {
* @param $cellpadding (float) Internal cell padding, if empty uses default cell padding. * @param $cellpadding (float) Internal cell padding, if empty uses default cell padding.
* @param $border (mixed) Indicates if borders must be drawn around the cell. The value can be a number:<ul><li>0: no border (default)</li><li>1: frame</li></ul> or a string containing some or all of the following characters (in any order):<ul><li>L: left</li><li>T: top</li><li>R: right</li><li>B: bottom</li></ul> or an array of line styles for each border group - for example: array('LTRB' => array('width' => 2, 'cap' => 'butt', 'join' => 'miter', 'dash' => 0, 'color' => array(0, 0, 0))) * @param $border (mixed) Indicates if borders must be drawn around the cell. The value can be a number:<ul><li>0: no border (default)</li><li>1: frame</li></ul> or a string containing some or all of the following characters (in any order):<ul><li>L: left</li><li>T: top</li><li>R: right</li><li>B: bottom</li></ul> or an array of line styles for each border group - for example: array('LTRB' => array('width' => 2, 'cap' => 'butt', 'join' => 'miter', 'dash' => 0, 'color' => array(0, 0, 0)))
* @return float Return the minimal height needed for multicell method for printing the $txt param. * @return float Return the minimal height needed for multicell method for printing the $txt param.
* @author Alexander Escalona Fernández, Nicola Asuni * @author Alexander Escalona Fern<EFBFBD>ndez, Nicola Asuni
* @public * @public
* @since 4.5.011 * @since 4.5.011
*/ */
@ -6230,7 +6230,7 @@ class TCPDF {
* @param $cellpadding (float) Internal cell padding, if empty uses default cell padding. * @param $cellpadding (float) Internal cell padding, if empty uses default cell padding.
* @param $border (mixed) Indicates if borders must be drawn around the cell. The value can be a number:<ul><li>0: no border (default)</li><li>1: frame</li></ul> or a string containing some or all of the following characters (in any order):<ul><li>L: left</li><li>T: top</li><li>R: right</li><li>B: bottom</li></ul> or an array of line styles for each border group - for example: array('LTRB' => array('width' => 2, 'cap' => 'butt', 'join' => 'miter', 'dash' => 0, 'color' => array(0, 0, 0))) * @param $border (mixed) Indicates if borders must be drawn around the cell. The value can be a number:<ul><li>0: no border (default)</li><li>1: frame</li></ul> or a string containing some or all of the following characters (in any order):<ul><li>L: left</li><li>T: top</li><li>R: right</li><li>B: bottom</li></ul> or an array of line styles for each border group - for example: array('LTRB' => array('width' => 2, 'cap' => 'butt', 'join' => 'miter', 'dash' => 0, 'color' => array(0, 0, 0)))
* @return float Return the minimal height needed for multicell method for printing the $txt param. * @return float Return the minimal height needed for multicell method for printing the $txt param.
* @author Nicola Asuni, Alexander Escalona Fernández * @author Nicola Asuni, Alexander Escalona Fern<EFBFBD>ndez
* @public * @public
*/ */
public function getStringHeight($w, $txt, $reseth=false, $autopadding=true, $cellpadding='', $border=0) { public function getStringHeight($w, $txt, $reseth=false, $autopadding=true, $cellpadding='', $border=0) {
@ -11442,7 +11442,7 @@ class TCPDF {
} }
/** /**
* Append a cubic Bézier curve to the current path. The curve shall extend from the current point to the point (x3, y3), using (x1, y1) and (x2, y2) as the Bézier control points. * Append a cubic B<EFBFBD>zier curve to the current path. The curve shall extend from the current point to the point (x3, y3), using (x1, y1) and (x2, y2) as the B<EFBFBD>zier control points.
* The new current point shall be (x3, y3). * The new current point shall be (x3, y3).
* @param $x1 (float) Abscissa of control point 1. * @param $x1 (float) Abscissa of control point 1.
* @param $y1 (float) Ordinate of control point 1. * @param $y1 (float) Ordinate of control point 1.
@ -11460,7 +11460,7 @@ class TCPDF {
} }
/** /**
* Append a cubic Bézier curve to the current path. The curve shall extend from the current point to the point (x3, y3), using the current point and (x2, y2) as the Bézier control points. * Append a cubic B<EFBFBD>zier curve to the current path. The curve shall extend from the current point to the point (x3, y3), using the current point and (x2, y2) as the B<EFBFBD>zier control points.
* The new current point shall be (x3, y3). * The new current point shall be (x3, y3).
* @param $x2 (float) Abscissa of control point 2. * @param $x2 (float) Abscissa of control point 2.
* @param $y2 (float) Ordinate of control point 2. * @param $y2 (float) Ordinate of control point 2.
@ -11476,7 +11476,7 @@ class TCPDF {
} }
/** /**
* Append a cubic Bézier curve to the current path. The curve shall extend from the current point to the point (x3, y3), using (x1, y1) and (x3, y3) as the Bézier control points. * Append a cubic B<EFBFBD>zier curve to the current path. The curve shall extend from the current point to the point (x3, y3), using (x1, y1) and (x3, y3) as the B<EFBFBD>zier control points.
* The new current point shall be (x3, y3). * The new current point shall be (x3, y3).
* @param $x1 (float) Abscissa of control point 1. * @param $x1 (float) Abscissa of control point 1.
* @param $y1 (float) Ordinate of control point 1. * @param $y1 (float) Ordinate of control point 1.
@ -12271,7 +12271,7 @@ class TCPDF {
/** /**
* Insert Named Destinations. * Insert Named Destinations.
* @protected * @protected
* @author Johannes Güntert, Nicola Asuni * @author Johannes G<EFBFBD>ntert, Nicola Asuni
* @since 5.9.098 (2011-06-23) * @since 5.9.098 (2011-06-23)
*/ */
protected function _putdests() { protected function _putdests() {
@ -12499,7 +12499,7 @@ class TCPDF {
* Adds a javascript * Adds a javascript
* @param $script (string) Javascript code * @param $script (string) Javascript code
* @public * @public
* @author Johannes Güntert, Nicola Asuni * @author Johannes G<EFBFBD>ntert, Nicola Asuni
* @since 2.1.002 (2008-02-12) * @since 2.1.002 (2008-02-12)
*/ */
public function IncludeJS($script) { public function IncludeJS($script) {
@ -12528,7 +12528,7 @@ class TCPDF {
/** /**
* Create a javascript PDF string. * Create a javascript PDF string.
* @protected * @protected
* @author Johannes Güntert, Nicola Asuni * @author Johannes G<EFBFBD>ntert, Nicola Asuni
* @since 2.1.002 (2008-02-12) * @since 2.1.002 (2008-02-12)
*/ */
protected function _putjavascript() { protected function _putjavascript() {
@ -13414,7 +13414,7 @@ class TCPDF {
* @param $private_key (mixed) private key (string or filename prefixed with 'file://') * @param $private_key (mixed) private key (string or filename prefixed with 'file://')
* @param $private_key_password (string) password * @param $private_key_password (string) password
* @param $extracerts (string) specifies the name of a file containing a bunch of extra certificates to include in the signature which can for example be used to help the recipient to verify the certificate that you used. * @param $extracerts (string) specifies the name of a file containing a bunch of extra certificates to include in the signature which can for example be used to help the recipient to verify the certificate that you used.
* @param $cert_type (int) The access permissions granted for this document. Valid values shall be: 1 = No changes to the document shall be permitted; any change to the document shall invalidate the signature; 2 = Permitted changes shall be filling in forms, instantiating page templates, and signing; other changes shall invalidate the signature; 3 = Permitted changes shall be the same as for 2, as well as annotation creation, deletion, and modification; other changes shall invalidate the signature. * @param $cert_type (int) The access permissions granted for this document. Valid values shall be: 1 = No changes to the document shall be permitted; any change to the document shall invalidate the signature; 2 = Permitted changes shall be filling in forms, instantiating page Templates, and signing; other changes shall invalidate the signature; 3 = Permitted changes shall be the same as for 2, as well as annotation creation, deletion, and modification; other changes shall invalidate the signature.
* @param $info (array) array of option information: Name, Location, Reason, ContactInfo. * @param $info (array) array of option information: Name, Location, Reason, ContactInfo.
* @public * @public
* @author Nicola Asuni * @author Nicola Asuni
@ -14174,7 +14174,7 @@ class TCPDF {
* @param $col1 (array) first color (Grayscale, RGB or CMYK components). * @param $col1 (array) first color (Grayscale, RGB or CMYK components).
* @param $col2 (array) second color (Grayscale, RGB or CMYK components). * @param $col2 (array) second color (Grayscale, RGB or CMYK components).
* @param $coords (array) array of the form (x1, y1, x2, y2) which defines the gradient vector (see linear_gradient_coords.jpg). The default value is from left to right (x1=0, y1=0, x2=1, y2=0). * @param $coords (array) array of the form (x1, y1, x2, y2) which defines the gradient vector (see linear_gradient_coords.jpg). The default value is from left to right (x1=0, y1=0, x2=1, y2=0).
* @author Andreas Würmser, Nicola Asuni * @author Andreas W<EFBFBD>rmser, Nicola Asuni
* @since 3.1.000 (2008-06-09) * @since 3.1.000 (2008-06-09)
* @public * @public
*/ */
@ -14192,7 +14192,7 @@ class TCPDF {
* @param $col1 (array) first color (Grayscale, RGB or CMYK components). * @param $col1 (array) first color (Grayscale, RGB or CMYK components).
* @param $col2 (array) second color (Grayscale, RGB or CMYK components). * @param $col2 (array) second color (Grayscale, RGB or CMYK components).
* @param $coords (array) array of the form (fx, fy, cx, cy, r) where (fx, fy) is the starting point of the gradient with color1, (cx, cy) is the center of the circle with color2, and r is the radius of the circle (see radial_gradient_coords.jpg). (fx, fy) should be inside the circle, otherwise some areas will not be defined. * @param $coords (array) array of the form (fx, fy, cx, cy, r) where (fx, fy) is the starting point of the gradient with color1, (cx, cy) is the center of the circle with color2, and r is the radius of the circle (see radial_gradient_coords.jpg). (fx, fy) should be inside the circle, otherwise some areas will not be defined.
* @author Andreas Würmser, Nicola Asuni * @author Andreas W<EFBFBD>rmser, Nicola Asuni
* @since 3.1.000 (2008-06-09) * @since 3.1.000 (2008-06-09)
* @public * @public
*/ */
@ -14215,7 +14215,7 @@ class TCPDF {
* @param $coords_min (array) minimum value used by the coordinates. If a coordinate's value is smaller than this it will be cut to coords_min. default: 0 * @param $coords_min (array) minimum value used by the coordinates. If a coordinate's value is smaller than this it will be cut to coords_min. default: 0
* @param $coords_max (array) maximum value used by the coordinates. If a coordinate's value is greater than this it will be cut to coords_max. default: 1 * @param $coords_max (array) maximum value used by the coordinates. If a coordinate's value is greater than this it will be cut to coords_max. default: 1
* @param $antialias (boolean) A flag indicating whether to filter the shading function to prevent aliasing artifacts. * @param $antialias (boolean) A flag indicating whether to filter the shading function to prevent aliasing artifacts.
* @author Andreas Würmser, Nicola Asuni * @author Andreas W<EFBFBD>rmser, Nicola Asuni
* @since 3.1.000 (2008-06-09) * @since 3.1.000 (2008-06-09)
* @public * @public
*/ */
@ -14307,7 +14307,7 @@ class TCPDF {
* @param $y (float) ordinate of the top left corner of the rectangle. * @param $y (float) ordinate of the top left corner of the rectangle.
* @param $w (float) width of the rectangle. * @param $w (float) width of the rectangle.
* @param $h (float) height of the rectangle. * @param $h (float) height of the rectangle.
* @author Andreas Würmser, Nicola Asuni * @author Andreas W<EFBFBD>rmser, Nicola Asuni
* @since 3.1.000 (2008-06-09) * @since 3.1.000 (2008-06-09)
* @protected * @protected
*/ */
@ -21301,13 +21301,13 @@ if (! $size) $size = 10;
} }
/** /**
* Output a Table Of Content Index (TOC) using HTML templates. * Output a Table Of Content Index (TOC) using HTML Templates.
* This method must be called after all Bookmarks were set. * This method must be called after all Bookmarks were set.
* Before calling this method you have to open the page using the addTOCPage() method. * Before calling this method you have to open the page using the addTOCPage() method.
* After calling this method you have to call endTOCPage() to close the TOC page. * After calling this method you have to call endTOCPage() to close the TOC page.
* @param $page (int) page number where this TOC should be inserted (leave empty for current page). * @param $page (int) page number where this TOC should be inserted (leave empty for current page).
* @param $toc_name (string) name to use for TOC bookmark. * @param $toc_name (string) name to use for TOC bookmark.
* @param $templates (array) array of html templates. Use: "#TOC_DESCRIPTION#" for bookmark title, "#TOC_PAGE_NUMBER#" for page number. * @param $templates (array) array of html Templates. Use: "#TOC_DESCRIPTION#" for bookmark title, "#TOC_PAGE_NUMBER#" for page number.
* @param $correct_align (boolean) if true correct the number alignment (numbers must be in monospaced font like courier and right aligned on LTR, or left aligned on RTL) * @param $correct_align (boolean) if true correct the number alignment (numbers must be in monospaced font like courier and right aligned on LTR, or left aligned on RTL)
* @param $style (string) Font style for title: B = Bold, I = Italic, BI = Bold + Italic. * @param $style (string) Font style for title: B = Bold, I = Italic, BI = Bold + Italic.
* @param $color (array) RGB color array for title (values from 0 to 255). * @param $color (array) RGB color array for title (values from 0 to 255).
@ -21352,7 +21352,7 @@ if (! $size) $size = 10;
} }
$maxpage = max($maxpage, $outline['p']); $maxpage = max($maxpage, $outline['p']);
} }
// replace templates with current values // replace Templates with current values
$row = str_replace('#TOC_DESCRIPTION#', $outline['t'], $row); $row = str_replace('#TOC_DESCRIPTION#', $outline['t'], $row);
$row = str_replace('#TOC_PAGE_NUMBER#', $pagenum, $row); $row = str_replace('#TOC_PAGE_NUMBER#', $pagenum, $row);
// add link to page // add link to page
@ -23300,7 +23300,7 @@ if (! $size) $size = 10;
} }
break; break;
} }
case 'Q': { // quadratic Bézier curveto case 'Q': { // quadratic B<EFBFBD>zier curveto
foreach ($params as $ck => $cp) { foreach ($params as $ck => $cp) {
$params[$ck] = $cp; $params[$ck] = $cp;
if ((($ck + 1) % 4) == 0) { if ((($ck + 1) % 4) == 0) {
@ -23326,7 +23326,7 @@ if (! $size) $size = 10;
} }
break; break;
} }
case 'T': { // shorthand/smooth quadratic Bézier curveto case 'T': { // shorthand/smooth quadratic B<EFBFBD>zier curveto
foreach ($params as $ck => $cp) { foreach ($params as $ck => $cp) {
$params[$ck] = $cp; $params[$ck] = $cp;
if (($ck % 2) != 0) { if (($ck % 2) != 0) {

View File

@ -10,6 +10,7 @@ use Icinga\Web\Widget\Tabs;
class Monitoring_ListController extends ModuleActionController class Monitoring_ListController extends ModuleActionController
{ {
protected $backend; protected $backend;
private $compactView = null;
public function init() public function init()
{ {
@ -23,6 +24,7 @@ class Monitoring_ListController extends ModuleActionController
public function hostsAction() public function hostsAction()
{ {
Benchmark::measure("hostsAction::query()"); Benchmark::measure("hostsAction::query()");
$this->compactView = "hosts-compact";
$this->view->hosts = $this->query( $this->view->hosts = $this->query(
'status', 'status',
array( array(
@ -192,6 +194,10 @@ class Monitoring_ListController extends ModuleActionController
protected function handleFormatRequest($query) protected function handleFormatRequest($query)
{ {
if ($this->compactView !== null && $this->_getParam("view", false) === "compact") {
$this->_helper->viewRenderer($this->compactView);
}
if ($this->_getParam('format') === 'sql') { if ($this->_getParam('format') === 'sql') {
echo '<pre>' echo '<pre>'
. htmlspecialchars(wordwrap($query->getQuery()->dump())) . htmlspecialchars(wordwrap($query->getQuery()->dump()))

View File

@ -1,64 +1,110 @@
<?php <?php
$hosts = $this->hosts->paginate();
$count = $hosts->count(); $viewHelper = $this->getHelper('MonitoringState');
if (! $count) { $trimArea = $this->getHelper('Trim');
echo '- no host is matching this filter -';
return;
}
$hosts->paginate();
?><table class="hosts action">
<thead>
<tr>
<th style="width: 6em;" >State</th>
<th >Host</th>
</tr>
</thead>
<tbody>
<?php foreach ($hosts->fetchAll() as $host):
$icons = array();
if ($host->host_acknowledged) {
$icons['ack.gif'] = 'Problem has been acknowledged';
}
if ($host->host_in_downtime) {
$icons['downtime.gif'] = 'Host is in a scheduled downtime';
}
$state_classes = array($this->monitoringState($host, 'host'));
if ($host->host_acknowledged || $host->host_in_downtime) {
$state_classes[] = 'handled';
}
if ($host->host_last_state_change > (time() - 600)) {
$state_classes[] = 'new';
}
$state_title = strtoupper($this->monitoringState($host, 'host'))
. ' since '
. date('Y-m-d H:i:s', $host->host_last_state_change);
?> ?>
<tr class="<?= implode(' ', $state_classes) ?>">
<td style="width: 20%;" class="state" title="<?= $state_title ?>"><?= $this->qlink( <?= $this->paginationControl($hosts, null, null, array('preserve' => $this->preserve)) ?>
$this->timeSince($host->host_last_state_change),
'monitoring/show/history', <table class="statustable action hosts">
array('host' => $host->host_name),
array('quote' => false)) ?></td> <thead>
<td style="width: 80%;"><? <tr>
foreach ($icons as $icon => $alt) <th colspan="3">Status</th>
echo $this->img('img/classic/' . $icon, array( <th>Host</th>
'class' => 'icon', <th>Output</th>
'title' => $alt <th></th>
)); </tr>
?> </thead>
<?= $this->qlink( <tbody>
$host->host_name, <?php foreach($hosts as $host): ?>
'monitoring/show/host', <tr class="<?= implode(' ', $viewHelper->getStateFlags($host, 'host')); ?>">
array('host' => $host->host_name), <td class="icons indicator">
array('class' => 'row-action') <div class="img-box"><?php $trimArea->start(); ?>
) ?></td> <?php if ($host->host_icon_image) : ?>
</tr> <img src="<?= $host->host_icon_image; ?>"/>
<? endforeach ?> <?php endif; ?>
</tbody> <?php $trimArea->end(); ?></div>
</td>
<td class="icons indicator">
<div class="icon-box"><?php $trimArea->start(); ?>
<?php if (!$host->host_handled && $host->host_state > 0): ?>
<a href="#" title="<?= 'Unhandled host' ?>">
<i class="icon-warning-sign"></i>
</a>
<?php endif; ?>
<?php if ($host->host_acknowledged && !$host->host_in_downtime): ?>
<a href="#" title="<?= 'Acknowledged' ?>">
<i class="icon-ok-sign"></i>
</a>
<?php endif; ?>
<?php if ($host->host_is_flapping): ?>
<a href="#" title="<?= 'Flapping' ?>">
<i class="icon-random"></i>
</a>
<?php endif; ?>
<?php if (!$host->host_notifications_enabled): ?>
<a href="#" title="<?= 'Notifications disabled' ?>">
<i class="icon-volume-off"></i>
</a>
<?php endif; ?>
<?php if ($host->host_in_downtime): ?>
<a href="#" title="<?= 'In downtime' ?>">
<i class="icon-wrench"></i>
</a>
<?php endif; ?>
<?php $trimArea->end(); ?></div>
</td>
<td class="indicator state" title="<?= $viewHelper->getStateTitle($host, 'host'); ?>">
<div class="statetext">
<?= $this->qlink(
"<b>".ucfirst($viewHelper->monitoringState($host, 'host'))."</b>".
'<div class="nowrap"> since&nbsp;'.
$this->timeSince($host->host_last_state_change),
'monitoring/show/history', array(
'host' => $host->host_name
),
array('quote' => false)
);?>
<?php if ($host->host_state_type == 0): ?>
<a href="#" title="<?= 'Soft state' ?>">
<i class="icon-gears"></i>
</a>
<?php endif; ?>
</div>
</td>
<td class="hostname">
<?php if ($host->host_last_comment !== null): ?>
<a href="#" title="<?= 'Comments' ?>">
<i class="icon-comment"> </i>
</a>
<?php endif; ?>
<?= $this->qlink(
"<b>".$host->host_name."</b><br/>".
"<i>".$host->host_address."</i>",
'monitoring/show/host', array(
'host' => $host->host_name
), array(
'class' => 'row-action',
'quote' => false
)
); ?>
<?php if ($host->host_action_url != ""): ?>
<a href="<?= $host->host_action_url; ?>">Action</a>
<?php endif; ?>
<?php if ($host->host_notes_url != ""): ?>
<a href="<?= $host->host_notes_url; ?>">Notes</a>
<?php endif; ?>
</td>
<td class="expand-full">
<?= $this->escape(substr(strip_tags($host->host_output), 0, 500)); ?>
</td>
</tr>
<?php endforeach; ?>
</tbody>
</table> </table>

View File

@ -18,7 +18,7 @@ require_once(dirname(__FILE__).'/../schemes/StatusdatTemplates.php');
* to according objects.cache and status.dat files which then can be read * to according objects.cache and status.dat files which then can be read
* by the Statusdat parser and used in tests. * by the Statusdat parser and used in tests.
* *
* The templates for insertion can be found under schemes/objectsCacheTemplates.php * The Templates for insertion can be found under schemes/objectsCacheTemplates.php
* and schemes/StatusdatTempaltes.php * and schemes/StatusdatTempaltes.php
* *
*/ */

View File

@ -71,12 +71,13 @@
}; };
this.loadAsyncContainers = function(root) { this.loadAsyncContainers = function(root) {
$('.icinga-container[icinga-url]',root).each(function() { $('.icinga-container[icingaurl]',root).each(function() {
var el = $(this); var el = $(this);
var url = el.attr('icinga-url'); var url = el.attr('icingaurl');
el.attr('loaded',true); el.attr('loaded',true);
async.loadToTarget(el,url); async.loadToTarget(el,url);
}); });
log.debug("Loading async");
}; };
this.initializeContainers = function(root) { this.initializeContainers = function(root) {