Merge pull request #3504 from Icinga/fix/collapsible-sidebar-issues-3187

Fix collapsible sidebar issues
This commit is contained in:
Eric Lippmann 2019-03-21 09:09:27 +01:00 committed by GitHub
commit 915c7b8fe5
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
10 changed files with 187 additions and 61 deletions

View File

@ -37,6 +37,13 @@ class NavigationItem implements IteratorAggregate
*/ */
protected $active; protected $active;
/**
* Whether this item is selected
*
* @var bool
*/
protected $selected;
/** /**
* The CSS class used for the outer li element * The CSS class used for the outer li element
* *
@ -213,6 +220,39 @@ class NavigationItem implements IteratorAggregate
return $this; return $this;
} }
/**
* Return whether this item is selected
*
* @return bool
*/
public function getSelected()
{
if ($this->selected === null) {
$this->active = false;
if ($this->getUrl() !== null && Icinga::app()->getRequest()->getUrl()->matches($this->getUrl())) {
$this->setSelected();
}
}
return $this->selected;
}
/**
* Set whether this item is active
*
* If it's active and has a parent, the parent gets activated as well.
*
* @param bool $selected
*
* @return $this
*/
public function setSelected($selected = true)
{
$this->selected = (bool) $selected;
return $this;
}
/** /**
* Get the CSS class used for the outer li element * Get the CSS class used for the outer li element
* *

View File

@ -100,15 +100,8 @@ abstract class BadgeNavigationItemRenderer extends NavigationItemRenderer
$item->setCssClass('badge-nav-item'); $item->setCssClass('badge-nav-item');
$this->setEscapeLabel(false); $this->setEscapeLabel(false);
$label = $this->view()->escape($item->getLabel()); $label = $this->view()->escape($item->getLabel());
if (($icon = $item->getIcon()) !== null) {
$label = $this->view()->icon($icon) . $label;
$item->setIcon(null);
}
$item->setLabel($this->renderBadge() . $label); $item->setLabel($this->renderBadge() . $label);
$html = parent::render($item); $html = parent::render($item);
if ($icon) {
$item->setIcon(true);
}
return $html; return $html;
} }

View File

@ -178,6 +178,9 @@ class NavigationItemRenderer
: $item->getLabel(); : $item->getLabel();
if (($icon = $item->getIcon()) !== null) { if (($icon = $item->getIcon()) !== null) {
$label = $this->view()->icon($icon) . $label; $label = $this->view()->icon($icon) . $label;
} else {
$firstLetter = $item->getName()[0];
$label = $this->view()->icon('letter', null, ['data-letter' => strtolower($firstLetter)]) . $label;
} }
if (($url = $item->getUrl()) !== null) { if (($url = $item->getUrl()) !== null) {

View File

@ -322,9 +322,8 @@ class NavigationRenderer implements RecursiveIterator, NavigationRendererInterfa
$cssClasses[] = static::CSS_CLASS_ACTIVE; $cssClasses[] = static::CSS_CLASS_ACTIVE;
} }
if ($item->getIcon() === null) { if ($item->getSelected()) {
// @TODO(el): Add constant $cssClasses[] = static::CSS_CLASS_SELECTED;
$cssClasses[] = 'no-icon';
} }
if ($cssClass = $item->getCssClass()) { if ($cssClass = $item->getCssClass()) {

View File

@ -22,6 +22,13 @@ interface NavigationRendererInterface
*/ */
const CSS_CLASS_ACTIVE = 'active'; const CSS_CLASS_ACTIVE = 'active';
/**
* CSS class for selected items
*
* @var string
*/
const CSS_CLASS_SELECTED = 'selected';
/** /**
* CSS class for dropdown items * CSS class for dropdown items
* *

View File

@ -60,7 +60,7 @@ input.search {
&:focus { &:focus {
background-color: @body-bg-color; background-color: @body-bg-color;
background-image: url('../img/icons/search_icinga_blue.png') !important; background-image: url('../img/icons/search_icinga_blue.png');
} }
} }

View File

@ -25,11 +25,6 @@
position: absolute; position: absolute;
} }
#layout:not(.minimal-layout).sidebar-collapsed #menu {
min-width: 6em;
width: auto;
}
#layout:not(.minimal-layout) #menu { #layout:not(.minimal-layout) #menu {
// Space for the #toggle-sidebar button // Space for the #toggle-sidebar button
&:after { &:after {
@ -71,14 +66,9 @@
color: @menu-color; color: @menu-color;
&.active { &.active {
> a { > a > .badge {
font-weight: @font-weight-bold;
letter-spacing: .02em;
> .badge {
display: none; display: none;
} }
}
background-color: @menu-active-bg-color; background-color: @menu-active-bg-color;
} }
@ -89,58 +79,89 @@
> a { > a {
padding: 0.5em 0.5em 0.5em .75em; padding: 0.5em 0.5em 0.5em .75em;
&:focus, &:hover {
color: @menu-highlight-color;
} }
&.active:not(.selected) > a:focus,
&.active:not(.selected) > a:hover {
background-color: mix(#000, @menu-active-bg-color, 20);
}
&:not(.selected) > a:hover,
&:not(.selected) > a:focus {
background-color: mix(#000, @menu-bg-color, 20);
} }
// Balance icon weight for non active menu items // Balance icon weight for non active menu items
&:not(.active) > a > i { &:not(.active) > a > i {
opacity: .8; opacity: .8;
} }
& > a > .icon-letter:before {
content: attr(data-letter);
font-family: @font-family;
font-weight: 800;
text-transform: uppercase;
}
}
ul:not(.nav-level-2) > .selected > a {
background-color: @menu-highlight-color;
color: @text-color-inverted;
&:hover {
color: @text-color-inverted;
}
&:after {
.transform(rotate(45deg));
position: absolute;
right: -.75em;
background-color: #fff;
box-shadow: 0 0 1em 0 rgba(0,0,0,0.6);
content: "";
display: block;
height: 1.25em;
margin-top: -1.75em;
width: 1.25em;
}
} }
#menu .nav-level-2 > .nav-item { #menu .nav-level-2 > .nav-item {
// Collapse menu by default // Collapse menu by default
display: none; display: none;
line-height: 1.833em; // 22px line-height: 1.833em; // 22px
padding-left: @icon-width;
> a { > a {
color: @menu-2ndlvl-color; color: @menu-2ndlvl-color;
font-size: @font-size-small; font-size: @font-size-small;
padding: 0.364em 0.545em 0.364em .8em; padding: 0.364em 0.545em 0.364em (@icon-width + .75em);
&:hover, &:focus {
color: @menu-2ndlvl-highlight-color;
} }
&:not(.selected):not(.active) > a:hover,
&:not(.selected):not(.active) > a:focus {
background-color: mix(#000, @menu-active-bg-color, 20);
color: @menu-2ndlvl-highlight-color;
} }
&.active { &.active {
background-color: @menu-highlight-color; background-color: @menu-highlight-color;
overflow: hidden; overflow: hidden;
position: relative; position: relative;
> a {
&:focus, &:hover {
opacity: .6;
}
}
} }
// Little caret on active level-2 item // Little caret on active level-2 item
&.active:after { &.active:after {
.transform(rotate(45deg));
background-color: @body-bg-color; background-color: @body-bg-color;
box-shadow: 0 0 1em 0 rgba(0,0,0,.6); box-shadow: 0 0 1em 0 rgba(0,0,0,.6);
content: ""; content: "";
display: block; display: block;
height: 1.25em; height: 1.25em;
width: 1.25em; width: 1.25em;
-webkit-transform: rotate(45deg);
-moz-transform: rotate(45deg);
-ms-transform: rotate(45deg);
transform: rotate(45deg);
position: absolute; position: absolute;
top: .5em; top: .5em;
right: -.75em; right: -.75em;
@ -199,6 +220,11 @@
&.active { &.active {
background-color: @menu-active-bg-color; background-color: @menu-active-bg-color;
} }
&:hover,
&:focus {
background-color: mix(#000, @menu-bg-color, 20);
}
} }
// Badge offset correction // Badge offset correction
@ -223,6 +249,21 @@
width: 14em; width: 14em;
position: fixed; position: fixed;
&:after {
.transform(rotate(45deg));
background-color: @body-bg-color;
border-bottom: 1px solid @gray-light;
border-left: 1px solid @gray-light;
content: "";
display: block;
height: 1.1em;
width: 1.1em;
position: absolute;
top: 1em;
left: -.6em;
z-index: -1;
}
> .nav-item { > .nav-item {
display: block; display: block;
padding-left: 0; padding-left: 0;
@ -242,7 +283,6 @@
&:focus, &:hover { &:focus, &:hover {
background-color: @menu-highlight-color; background-color: @menu-highlight-color;
opacity: .6;
} }
} }
@ -316,6 +356,10 @@
position: relative; position: relative;
} }
.search-input:focus ~ .search-reset:hover {
background-color: mix(#000, @menu-active-bg-color, 20);
}
.search-reset { .search-reset {
background: none; background: none;
border: 0; border: 0;
@ -329,8 +373,9 @@
right: 0; right: 0;
top: 0; top: 0;
&:focus, &:hover { &:focus,
color: @menu-highlight-color; &:hover {
background-color: mix(#000, @menu-bg-color, 20);
outline: none; outline: none;
} }
} }
@ -390,7 +435,7 @@ input[type=text].search-input {
padding: 0; padding: 0;
color: @gray-light; color: @gray-light;
position: absolute; position: absolute;
bottom: 0; bottom: 0.2em;
right: 0; right: 0;
i { i {
@ -418,3 +463,9 @@ input[type=text].search-input {
#open-sidebar { #open-sidebar {
display: none; display: none;
} }
#open-sidebar:before,
#close-sidebar:before {
width: 1.4em;
margin-right: 0;
}

View File

@ -52,11 +52,11 @@
var $active = _this.$menu.find('li.active'); var $active = _this.$menu.find('li.active');
if ($active.length) { if ($active.length) {
$active.each(function() { $active.each(function() {
_this.setActive($(this)); _this.setActiveAndSelected($(this));
}); });
} else { } else {
// if no item is marked as active, try to select the menu from the current URL // if no item is marked as active, try to select the menu from the current URL
_this.setActiveByUrl($('#col1').data('icingaUrl')); _this.setActiveAndSelectedByUrl($('#col1').data('icingaUrl'));
} }
} }
@ -70,7 +70,7 @@
// restore selection to current active element // restore selection to current active element
if (this.active) { if (this.active) {
var $el = $(this.icinga.utils.getElementByDomPath(this.active)); var $el = $(this.icinga.utils.getElementByDomPath(this.active));
this.setActive($el); this.setActiveAndSelected($el);
/* /*
* Recreate the html content of the menu item to force the browser to update the layout, or else * Recreate the html content of the menu item to force the browser to update the layout, or else
@ -98,9 +98,9 @@
if (href.match(/#/)) { if (href.match(/#/)) {
// ...it may be a menu section without a dedicated link. // ...it may be a menu section without a dedicated link.
// Switch the active menu item: // Switch the active menu item:
_this.setActive($a); _this.setActiveAndSelected($a);
} else { } else {
_this.setActive($(event.target)); _this.setActiveAndSelected($(event.target));
} }
// update target url of the menu container to the clicked link // update target url of the menu container to the clicked link
var $menu = $('#menu'); var $menu = $('#menu');
@ -116,7 +116,7 @@
* *
* @param url {String} The url to match * @param url {String} The url to match
*/ */
Navigation.prototype.setActiveByUrl = function(url) { Navigation.prototype.setActiveAndSelectedByUrl = function(url) {
var $menu = $('#menu'); var $menu = $('#menu');
if (! $menu.length) { if (! $menu.length) {
@ -124,17 +124,17 @@
} }
// try to active the first item that has an exact URL match // try to active the first item that has an exact URL match
this.setActive($menu.find('[href="' + url + '"]')); this.setActiveAndSelected($menu.find('[href="' + url + '"]'));
// the url may point to the search field, which must be activated too // the url may point to the search field, which must be activated too
if (! this.active) { if (! this.active) {
this.setActive($menu.find('form[action="' + this.icinga.utils.parseUrl(url).path + '"]')); this.setActiveAndSelected($menu.find('form[action="' + this.icinga.utils.parseUrl(url).path + '"]'));
} }
// some urls may have custom filters which won't match any menu item, in that case search // some urls may have custom filters which won't match any menu item, in that case search
// for a menu item that points to the base action without any filters // for a menu item that points to the base action without any filters
if (! this.active) { if (! this.active) {
this.setActive($menu.find('[href="' + this.icinga.utils.parseUrl(url).path + '"]').first()); this.setActiveAndSelected($menu.find('[href="' + this.icinga.utils.parseUrl(url).path + '"]').first());
} }
}; };
@ -143,11 +143,12 @@
* *
* @param url * @param url
*/ */
Navigation.prototype.trySetActiveByUrl = function(url) { Navigation.prototype.trySetActiveAndSelectedByUrl = function(url) {
var active = this.active; var active = this.active;
this.setActiveByUrl(url); this.setActiveAndSelectedByUrl(url);
if (! this.active && active) { if (! this.active && active) {
this.setActive($(this.icinga.utils.getElementByDomPath(active))); this.setActiveAndSelected($(this.icinga.utils.getElementByDomPath(active)));
} }
}; };
@ -160,6 +161,15 @@
} }
}; };
/**
* Remove all selected elements
*/
Navigation.prototype.clearSelected = function() {
if (this.$menu) {
this.$menu.find('.selected').removeClass('selected');
}
};
/** /**
* Select all menu items in the selector as active and unfold surrounding menus when necessary * Select all menu items in the selector as active and unfold surrounding menus when necessary
* *
@ -184,6 +194,11 @@
} }
}; };
Navigation.prototype.setActiveAndSelected = function ($el) {
this.setActive($el);
this.setSelected($el);
};
/** /**
* Change the active menu element * Change the active menu element
* *
@ -202,6 +217,15 @@
// TODO: push to history // TODO: push to history
}; };
Navigation.prototype.setSelected = function($el) {
this.clearSelected();
$el = $el.closest('li');
if ($el.length) {
$el.addClass('selected');
}
};
/** /**
* Reset the active element to nothing * Reset the active element to nothing
*/ */
@ -210,6 +234,14 @@
this.active = null; this.active = null;
}; };
/**
* Reset the selected element to nothing
*/
Navigation.prototype.resetSelected = function() {
this.clearSelected();
this.selected = null;
};
/** /**
* Show the fly-out menu * Show the fly-out menu
* *
@ -231,7 +263,7 @@
return; return;
} }
var delay = 600; var delay = 300;
if ($layout.hasClass('menu-hovered')) { if ($layout.hasClass('menu-hovered')) {
delay = 0; delay = 0;
@ -318,9 +350,10 @@
); );
return; return;
} }
this.setActive($(active)); this.setActiveAndSelected($(active))
} else { } else {
this.resetActive(); this.resetActive();
this.resetSelected();
} }
}; };

View File

@ -610,7 +610,7 @@
var url = req.url; var url = req.url;
if (req.$target[0].id === 'col1') { if (req.$target[0].id === 'col1') {
this.icinga.behaviors.navigation.trySetActiveByUrl(url); this.icinga.behaviors.navigation.trySetActiveAndSelectedByUrl(url);
} }
var $forms = $('[action="' + this.icinga.utils.parseUrl(url).path + '"]'); var $forms = $('[action="' + this.icinga.utils.parseUrl(url).path + '"]');

View File

@ -186,7 +186,7 @@
var kill = this.cutContainer($('#col1')); var kill = this.cutContainer($('#col1'));
this.pasteContainer($('#col1'), col2); this.pasteContainer($('#col1'), col2);
this.fixControls(); this.fixControls();
this.icinga.behaviors.navigation.trySetActiveByUrl($('#col1').data('icingaUrl')); this.icinga.behaviors.navigation.trySetActiveAndSelectedByUrl($('#col1').data('icingaUrl'));
}, },
cutContainer: function ($col) { cutContainer: function ($col) {