Skip to content

Commit

Permalink
Configurable logo and title (#160)
Browse files Browse the repository at this point in the history
* added asynchronous config for settings.header to specify logo, title and favicon
* refactored navigation stuff to separate folder, excluded logo and title as separate component
* added Promise testcase
* hide logo container if there is no logo
* fixed tests and added assertions for logo and title
* adjusted left header spacing for title only
* added docu for configurable logo and title
* added support for data:image favicons
  • Loading branch information
maxmarkus authored Nov 5, 2018
1 parent 6a288e9 commit b3c41cb
Show file tree
Hide file tree
Showing 16 changed files with 251 additions and 38 deletions.
1 change: 0 additions & 1 deletion core/examples/luigi-sample-angular/angular.json
Original file line number Diff line number Diff line change
Expand Up @@ -18,7 +18,6 @@
"polyfills": "src/polyfills.ts",
"assets": [
"src/assets",
"src/favicon.ico",
"src/index.html",
"src/logout.html",
{
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -6,7 +6,7 @@ <h1 class="fd-section__title">Child node 2</h1>
<p>Navigation node config for this <strong>defaultChildNode</strong> demo:</p>
<app-code-snippet data="{
pathSegment: 'dps',
label: 'Default Child Node Example',
label: 'Default Child node Example',
defaultChildNode: 'dps2',
children: [
{
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -480,6 +480,13 @@ Luigi.setConfig({
useHashRouting: true
},
settings: {
header: () => ({
logo:
'',
title: 'Luigi Demo',
favicon: '/assets/favicon-sap.ico'
// favicon: ''
})
// hideNavigation: true
// backdropDisabled: true
}
Expand Down
Binary file not shown.
Binary file removed core/examples/luigi-sample-angular/src/favicon.ico
Binary file not shown.
2 changes: 1 addition & 1 deletion core/examples/luigi-sample-angular/src/index.html
Original file line number Diff line number Diff line change
@@ -1,7 +1,7 @@
<!DOCTYPE html>
<html>
<head>
<title>Luigi</title>
<title></title>
<link rel='stylesheet' href='/luigi-core/luigi.css'>
</head>
<body>
Expand Down
5 changes: 2 additions & 3 deletions core/src/App.html
Original file line number Diff line number Diff line change
Expand Up @@ -13,12 +13,11 @@
<LeftNav pathData={navigationPath} />
</div>


<script type="text/javascript">
import Backdrop from './Backdrop.html';
import { fade } from 'svelte-transitions';
import TopNav from './TopNav.html';
import LeftNav from './LeftNav.html';
import TopNav from './navigation/TopNav.html';
import LeftNav from './navigation/LeftNav.html';
import * as Routing from './services/routing.js';
import { getConfigValue, getConfigValueFromObject } from './services/config.js';
import { isFunction } from './utilities/helpers.js';
Expand Down
5 changes: 2 additions & 3 deletions core/src/LeftNav.html → core/src/navigation/LeftNav.html
Original file line number Diff line number Diff line change
Expand Up @@ -53,10 +53,9 @@ <h1 class="fd-side-nav__title">{key}</h1>
</div>

<script type="text/javascript">
import Backdrop from './Backdrop.html';
import { getChildren, getLeftNavData } from './services/navigation.js';
import { handleRouteClick } from './services/routing.js';
import { getConfigValue, getConfigBooleanValue } from './services/config.js';
import { handleRouteClick } from '../services/routing.js';
import { getConfigValue, getConfigBooleanValue } from '../services/config.js';

const setLeftNavData = async (current, component) => {
const componentData = component.get();
Expand Down
46 changes: 46 additions & 0 deletions core/src/navigation/LogoTitle.html
Original file line number Diff line number Diff line change
@@ -0,0 +1,46 @@


{#if hasLogo}
<div on:click="goTo('/')" class="fd-global-nav__logo fd-has-margin-left-none" ref:logo></div>
{/if}
<div class="fd-global-nav__product-name">
{#if title}
<a on:click="goTo('/')">{title}</a>
{/if}
</div>

<script type="text/javascript">
import { processHeaderSettings } from './services/header.js';
import { getConfigBooleanValue } from '../services/config.js';

export default {
oncreate() {
processHeaderSettings(this);
},
methods: {
goTo(path) {
if (getConfigBooleanValue('routing.useHashRouting')) {
window.location.hash = path;
} else {
window.location.pathname = path;
}
}
}
};
</script>

<style type="text/scss">
@import 'node_modules/fundamental-ui/scss/components/global-nav';
.fd-global-nav__logo {
cursor: pointer;
background-image: none;
margin-left: 0px;
}
.fd-global-nav__product-name {
color: #0a6ed1;
height: 50px;
font-size: 24px;
line-height: 1em;
padding: 13px 0;
}
</style>
34 changes: 11 additions & 23 deletions core/src/TopNav.html → core/src/navigation/TopNav.html
Original file line number Diff line number Diff line change
@@ -1,11 +1,9 @@
<div class="fd-ui__header {hideNavComponent ? 'hideNavComponent' : ''}">
<nav class="fd-global-nav">
<div class="fd-global-nav__group fd-global-nav__group--left">
<div class="fd-global-nav__logo fd-has-margin-left-none"></div>
<div class="fd-global-nav__product-name">
<a on:click="goTo('/overview')">LUIGI</a>
</div>
<LogoTitle />
</div>

<div class="fd-global-nav__group fd-global-nav__group--right">
<div class="fd-global-nav__actions">
{#if children && pathData.length > 0}
Expand Down Expand Up @@ -45,10 +43,10 @@
</div>

<script type="text/javascript">
import Backdrop from './Backdrop.html';
import Authorization from './Authorization.html';
import { handleRouteClick } from './services/routing.js';
import { getConfigValue, getConfigBooleanValue } from './services/config.js';
import LogoTitle from './LogoTitle.html';
import Authorization from '../Authorization.html';
import { handleRouteClick } from '../services/routing.js';
import { getConfigValue, getConfigValueAsync, getConfigBooleanValue } from '../services/config.js';

const getNegatedBoolString = str => {
return str === 'true' ? 'false' : 'true';
Expand Down Expand Up @@ -107,13 +105,6 @@
},
methods: {
handleClick: node => handleRouteClick(node),
goTo(path) {
if (getConfigBooleanValue('routing.useHashRouting')) {
window.location.hash = path;
} else {
window.location.pathname = path;
}
},
toggleDropdownState(name) {
const dropDownStates = this.get().dropDownStates || {};
dropDownStates[name] = getNegatedBoolString(dropDownStates[name]);
Expand All @@ -130,7 +121,8 @@
}
},
components: {
Authorization
Authorization,
LogoTitle
}
};
</script>
Expand Down Expand Up @@ -158,18 +150,14 @@
border-bottom: solid 1px #dededf;
padding: 0 10px;
}
.fd-global-nav__group--left {
padding-left: 20px;
}

.fd-global-nav {
height: 100%;
border: none;
}
.fd-global-nav__product-name {
color: #0a6ed1;
height: 50px;
font-size: 24px;
line-height: 1em;
padding: 13px 0;
}

.hideNavComponent {
display: none;
Expand Down
42 changes: 42 additions & 0 deletions core/src/navigation/services/header.js
Original file line number Diff line number Diff line change
@@ -0,0 +1,42 @@
import {
getConfigValueAsync,
getConfigBooleanValue
} from '../../services/config.js';

export const processHeaderSettings = component => {
return getConfigValueAsync('settings.header').then(header => {
if (!header) {
return;
}

// Set Title and Logo
if (header.title) {
component.set({ title: header.title });
document.title = header.title;
}

const hasLogo = Boolean(header.logo);
component.set({ hasLogo });
if (hasLogo) {
component.refs.logo.style.backgroundImage = 'url(' + header.logo + ')';
}

// Set Favicon
if (header.favicon) {
const isInvalidFaviconFormat =
!header.favicon.split('?')[0].endsWith('.ico') &&
!header.favicon.startsWith('data:image');
if (isInvalidFaviconFormat) {
console.warn(
'Favicon is not an .ico filetype and might get displayed wrong.'
);
}
const link = Object.assign(document.createElement('link'), {
type: 'image/x-icon',
rel: 'shortcut icon',
href: header.favicon
});
document.getElementsByTagName('head')[0].appendChild(link);
}
});
};
Original file line number Diff line number Diff line change
@@ -1,4 +1,4 @@
import { getConfigValue, getConfigValueFromObjectAsync } from './config';
import { getConfigValue, getConfigValueFromObjectAsync } from '../../services/config';

const isNodeAccessPermitted = (
nodeToCheckPermissionFor,
Expand Down
2 changes: 1 addition & 1 deletion core/src/services/routing.js
Original file line number Diff line number Diff line change
@@ -1,4 +1,4 @@
import { getNavigationPath } from './navigation';
import { getNavigationPath } from '../navigation/services/navigation';
import {
getConfigValue,
getConfigValueAsync,
Expand Down
126 changes: 126 additions & 0 deletions core/test/header.spec.js
Original file line number Diff line number Diff line change
@@ -0,0 +1,126 @@
const headerService = require('../src/navigation/services/header');
const assert = require('chai').assert;
const sinon = require('sinon');

describe('LogoTitle', function() {
afterEach(() => {
sinon.restore();
});

describe('processHeaderSettings()', function() {
let component;
const setHeaderSettings = headerSettings => {
window.Luigi.config = {
settings: {
header: Object.assign({}, headerSettings)
}
};
};

beforeEach(() => {
component = {
set: sinon.spy()
};
});

it('should not fail for undefined arguments', async () => {
window.Luigi.config = {};
await headerService.processHeaderSettings();
});

it('should resolve title', async () => {
// given
const headerSettings = {
title: 'Luigi Demo'
};
setHeaderSettings(headerSettings);

document.title = '';

// when
await headerService.processHeaderSettings(component);

// then
assert.equal(document.title, headerSettings.title, 'document title');
assert(
component.set.calledWith({ title: headerSettings.title }),
'component set() title'
);
});

it('should hide logo if not defined', async () => {
setHeaderSettings({});

// when
await headerService.processHeaderSettings(component);

// then
assert(
component.set.calledWith({ hasLogo: false }),
'component set() hasLogo false'
);
});

it('should resolve logo and set hasLogo', async () => {
// given
const headerSettings = {
logo: ''
};
setHeaderSettings(headerSettings);

component.refs = {
logo: {
style: {
backgroundImage: null
}
}
};

// when
await headerService.processHeaderSettings(component);

// then
assert.equal(
component.refs.logo.style.backgroundImage,
'url(' + headerSettings.logo + ')',
'backgroundImage logo'
);
assert(
component.set.calledOnceWith({ hasLogo: true }),
'component set() hasLogo'
);
});

it('should resolve favicon', async () => {
// given
const headerSettings = {
favicon: '/assets/favicon.ico'
};
setHeaderSettings(headerSettings);

sinon.stub(document, 'createElement').returns({});
const appendChild = sinon.spy();
sinon.stub(document, 'getElementsByTagName').returns([{ appendChild }]);

const expectedLink = {
type: 'image/x-icon',
rel: 'shortcut icon',
href: headerSettings.favicon
};

// when
await headerService.processHeaderSettings(component);

// then
assert(
document.createElement.calledOnce,
'document.createElement() call'
);
assert(
document.getElementsByTagName.calledOnceWith('head'),
'document.getElementsByTagName() call'
);
assert(appendChild.calledOnceWith(expectedLink), 'appendChild() call');
});
});
});
2 changes: 1 addition & 1 deletion core/test/navigation.spec.js
Original file line number Diff line number Diff line change
@@ -1,4 +1,4 @@
const navigation = require('../src/services/navigation');
const navigation = require('../src/navigation/services/navigation');
const chai = require('chai');
const expect = chai.expect;
const assert = chai.assert;
Expand Down
13 changes: 10 additions & 3 deletions docs/general-settings.md
Original file line number Diff line number Diff line change
Expand Up @@ -5,9 +5,16 @@ The configuration file also contains a section called **Settings** in which you
````
settings: {
hideNavigation: false
backdropDisabled : false
backdropDisabled : false,
header: { object / function / Promise
logo: 'path/to/image.png',
title: 'Luigi Demo',
favicon: 'path/to/favicon.ico'
}
}
````
**hideNavigation** disables Luigi's default out-of-the-box navigation when set to `true`. Subsequently, neither the left nor top navigation are visible in your application. You are then free to provide your own navigation UI. By default, the navigation is enabled. If you want to keep it, set the value to `false` or leave this option unconfigured.

**backdropDisabled** prevents the backdrop layer from covering the top and left navigation when showing modal windows, when set to `true`. By default, the backdrop is enabled and covers the top and left navigation when showing modal windows. To keep the default configuration, set the value to `false` or leave this option unconfigured.
**backdropDisabled** When set to `true`, prevents the backdrop layer from covering the top and left navigation when showing modal windows. By default, the backdrop is enabled and covers the top and left navigation when showing modal windows. To keep the default configuration, set the value to `false` or leave this option unconfigured.
**header.logo** defines the top left navigation logo. It has a fixed height of 28px.
**header.title** defines the top left navigation title.
**header.favicon** defines the favicon. Requires a standard favicon file with the `.ico` extension, and 16x16px or 32x32px dimensions.

0 comments on commit b3c41cb

Please sign in to comment.