Merge branch 'dev/new-fuse'

This commit is contained in:
Sercan Yemen 2018-06-15 17:56:36 +03:00
commit 75b7e4e270
878 changed files with 30083 additions and 14944 deletions

View File

@ -1,13 +1,6 @@
# Editor configuration, see http://editorconfig.org
root = true
[*] [*]
charset = utf-8 charset=utf-8
indent_style = space end_of_line=lf
indent_size = 2 insert_final_newline=false
insert_final_newline = true indent_style=space
trim_trailing_whitespace = true indent_size=4
[*.md]
max_line_length = off
trim_trailing_whitespace = false

View File

@ -1,6 +1,6 @@
# Fuse2 # Fuse - Angular
Material Design Admin Template with Angular 6+ and Angular Material 2 Material Design Admin Template with Angular 6+ and Angular Material
## The Community ## The Community

View File

@ -25,7 +25,7 @@
"assets": [ "assets": [
"src/favicon.ico", "src/favicon.ico",
"src/assets", "src/assets",
"src/app/main/content/components/angular-material" "src/app/main/angular-material-elements"
], ],
"styles": [ "styles": [
"src/styles.scss" "src/styles.scss"
@ -104,7 +104,7 @@
], ],
"exclude": [ "exclude": [
"**/node_modules/**", "**/node_modules/**",
"**/src/app/fuse-fake-db/**/*", "**/src/app/fake-db/**/*",
"**/src/assets/angular-material-examples/**/*" "**/src/assets/angular-material-examples/**/*"
] ]
} }
@ -128,7 +128,7 @@
"tsConfig": "e2e/tsconfig.e2e.json", "tsConfig": "e2e/tsconfig.e2e.json",
"exclude": [ "exclude": [
"**/node_modules/**", "**/node_modules/**",
"**/src/app/fuse-fake-db/**/*", "**/src/app/fake-db/**/*",
"**/src/assets/angular-material-examples/**/*" "**/src/assets/angular-material-examples/**/*"
] ]
} }

4918
package-lock.json generated

File diff suppressed because it is too large Load Diff

View File

@ -1,6 +1,6 @@
{ {
"name": "fuse", "name": "fuse",
"version": "6.0.2", "version": "6.1.0",
"license": "https://themeforest.net/licenses/terms/regular", "license": "https://themeforest.net/licenses/terms/regular",
"scripts": { "scripts": {
"ng": "ng", "ng": "ng",
@ -18,56 +18,56 @@
}, },
"private": true, "private": true,
"dependencies": { "dependencies": {
"@agm/core": "1.0.0-beta.2", "@agm/core": "1.0.0-beta.3",
"@angular/animations": "6.0.2", "@angular/animations": "6.0.5",
"@angular/cdk": "6.0.2", "@angular/cdk": "6.2.1",
"@angular/common": "6.0.2", "@angular/common": "6.0.5",
"@angular/compiler": "6.0.2", "@angular/compiler": "6.0.5",
"@angular/core": "6.0.2", "@angular/core": "6.0.5",
"@angular/flex-layout": "6.0.0-beta.15", "@angular/flex-layout": "6.0.0-beta.16",
"@angular/forms": "6.0.2", "@angular/forms": "6.0.5",
"@angular/http": "6.0.2", "@angular/http": "6.0.5",
"@angular/material": "6.0.2", "@angular/material": "6.2.1",
"@angular/material-moment-adapter": "6.0.2", "@angular/material-moment-adapter": "6.2.1",
"@angular/platform-browser": "6.0.2", "@angular/platform-browser": "6.0.5",
"@angular/platform-browser-dynamic": "6.0.2", "@angular/platform-browser-dynamic": "6.0.5",
"@angular/router": "6.0.2", "@angular/router": "6.0.5",
"@ngrx/effects": "6.0.0-beta.3", "@ngrx/effects": "6.0.1",
"@ngrx/router-store": "6.0.0-beta.3", "@ngrx/router-store": "6.0.1",
"@ngrx/store": "6.0.0-beta.3", "@ngrx/store": "6.0.1",
"@ngrx/store-devtools": "6.0.0-beta.3", "@ngrx/store-devtools": "6.0.1",
"@ngx-translate/core": "10.0.1", "@ngx-translate/core": "10.0.2",
"@swimlane/ngx-charts": "8.0.0", "@swimlane/ngx-charts": "8.1.0",
"@swimlane/ngx-datatable": "12.0.0", "@swimlane/ngx-datatable": "13.0.1",
"@swimlane/ngx-dnd": "4.0.1", "@swimlane/ngx-dnd": "4.0.0",
"@types/prismjs": "1.9.0", "@types/prismjs": "1.9.0",
"angular-calendar": "0.25.2", "angular-calendar": "0.25.2",
"angular-in-memory-web-api": "0.6.0", "angular-in-memory-web-api": "0.6.0",
"chart.js": "2.7.2", "chart.js": "2.7.2",
"classlist.js": "1.1.20150312", "classlist.js": "1.1.20150312",
"core-js": "2.5.6", "core-js": "2.5.7",
"d3": "5.4.0", "d3": "5.4.0",
"hammerjs": "2.0.8", "hammerjs": "2.0.8",
"lodash": "4.17.10", "lodash": "4.17.10",
"moment": "2.22.1", "moment": "2.22.2",
"ng2-charts": "1.6.0", "ng2-charts": "1.6.0",
"ngrx-store-freeze": "0.2.3", "ngrx-store-freeze": "0.2.4",
"ngx-color-picker": "6.1.0", "ngx-color-picker": "6.3.3",
"ngx-cookie-service": "1.0.10", "ngx-cookie-service": "1.0.10",
"perfect-scrollbar": "1.3.0", "perfect-scrollbar": "1.4.0",
"prismjs": "1.14.0", "prismjs": "1.14.0",
"rxjs": "6.1.0", "rxjs": "6.2.1",
"rxjs-compat": "6.1.0", "rxjs-compat": "6.2.1",
"web-animations-js": "2.3.1", "web-animations-js": "2.3.1",
"zone.js": "0.8.26" "zone.js": "0.8.26"
}, },
"devDependencies": { "devDependencies": {
"@angular/cli": "6.0.3", "@angular/cli": "6.0.8",
"@angular/compiler-cli": "6.0.2", "@angular/compiler-cli": "6.0.5",
"@angular/language-service": "6.0.2", "@angular/language-service": "6.0.5",
"@angular-devkit/build-angular": "0.6.3", "@angular-devkit/build-angular": "0.6.8",
"@angularclass/hmr": "2.1.3", "@angularclass/hmr": "2.1.3",
"@types/jasmine": "2.8.7", "@types/jasmine": "2.8.8",
"@types/jasminewd2": "2.0.3", "@types/jasminewd2": "2.0.3",
"@types/lodash": "4.14.109", "@types/lodash": "4.14.109",
"@types/node": "8.9.5", "@types/node": "8.9.5",
@ -76,13 +76,13 @@
"jasmine-spec-reporter": "4.2.1", "jasmine-spec-reporter": "4.2.1",
"karma": "1.7.1", "karma": "1.7.1",
"karma-chrome-launcher": "2.2.0", "karma-chrome-launcher": "2.2.0",
"karma-coverage-istanbul-reporter": "1.4.2", "karma-coverage-istanbul-reporter": "2.0.1",
"karma-jasmine": "1.1.2", "karma-jasmine": "1.1.2",
"karma-jasmine-html-reporter": "0.2.2", "karma-jasmine-html-reporter": "0.2.2",
"protractor": "5.3.2", "protractor": "5.3.2",
"ts-node": "5.0.1", "ts-node": "5.0.1",
"tslint": "5.9.1", "tslint": "5.9.1",
"typescript": "2.7.2", "typescript": "2.7.2",
"webpack-bundle-analyzer": "2.11.3" "webpack-bundle-analyzer": "2.13.1"
} }
} }

View File

@ -187,10 +187,14 @@ export const fuseAnimations = [
transition('* => void', animate('300ms ease-in')) transition('* => void', animate('300ms ease-in'))
]), ]),
// -----------------------------------------------------------------------------------------------------
// @ Router animations
// -----------------------------------------------------------------------------------------------------
trigger('routerTransitionLeft', [ trigger('routerTransitionLeft', [
transition('* => *', [ transition('* => *', [
query('fuse-content > :enter, fuse-content > :leave', [ query('content > :enter, content > :leave', [
style({ style({
position: 'absolute', position: 'absolute',
top : 0, top : 0,
@ -199,7 +203,7 @@ export const fuseAnimations = [
right : 0 right : 0
}) })
], {optional: true}), ], {optional: true}),
query('fuse-content > :enter', [ query('content > :enter', [
style({ style({
transform: 'translateX(100%)', transform: 'translateX(100%)',
opacity : 0 opacity : 0
@ -207,7 +211,7 @@ export const fuseAnimations = [
], {optional: true}), ], {optional: true}),
sequence([ sequence([
group([ group([
query('fuse-content > :leave', [ query('content > :leave', [
style({ style({
transform: 'translateX(0)', transform: 'translateX(0)',
opacity : 1 opacity : 1
@ -218,7 +222,7 @@ export const fuseAnimations = [
opacity : 0 opacity : 0
})) }))
], {optional: true}), ], {optional: true}),
query('fuse-content > :enter', [ query('content > :enter', [
style({transform: 'translateX(100%)'}), style({transform: 'translateX(100%)'}),
animate('600ms cubic-bezier(0.0, 0.0, 0.2, 1)', animate('600ms cubic-bezier(0.0, 0.0, 0.2, 1)',
style({ style({
@ -227,8 +231,8 @@ export const fuseAnimations = [
})) }))
], {optional: true}) ], {optional: true})
]), ]),
query('fuse-content > :leave', animateChild(), {optional: true}), query('content > :leave', animateChild(), {optional: true}),
query('fuse-content > :enter', animateChild(), {optional: true}) query('content > :enter', animateChild(), {optional: true})
]) ])
]) ])
]), ]),
@ -236,7 +240,7 @@ export const fuseAnimations = [
trigger('routerTransitionRight', [ trigger('routerTransitionRight', [
transition('* => *', [ transition('* => *', [
query('fuse-content > :enter, fuse-content > :leave', [ query('content > :enter, content > :leave', [
style({ style({
position: 'absolute', position: 'absolute',
top : 0, top : 0,
@ -245,7 +249,7 @@ export const fuseAnimations = [
right : 0 right : 0
}) })
], {optional: true}), ], {optional: true}),
query('fuse-content > :enter', [ query('content > :enter', [
style({ style({
transform: 'translateX(-100%)', transform: 'translateX(-100%)',
opacity : 0 opacity : 0
@ -253,7 +257,7 @@ export const fuseAnimations = [
], {optional: true}), ], {optional: true}),
sequence([ sequence([
group([ group([
query('fuse-content > :leave', [ query('content > :leave', [
style({ style({
transform: 'translateX(0)', transform: 'translateX(0)',
opacity : 1 opacity : 1
@ -264,7 +268,7 @@ export const fuseAnimations = [
opacity : 0 opacity : 0
})) }))
], {optional: true}), ], {optional: true}),
query('fuse-content > :enter', [ query('content > :enter', [
style({transform: 'translateX(-100%)'}), style({transform: 'translateX(-100%)'}),
animate('600ms cubic-bezier(0.0, 0.0, 0.2, 1)', animate('600ms cubic-bezier(0.0, 0.0, 0.2, 1)',
style({ style({
@ -273,8 +277,8 @@ export const fuseAnimations = [
})) }))
], {optional: true}) ], {optional: true})
]), ]),
query('fuse-content > :leave', animateChild(), {optional: true}), query('content > :leave', animateChild(), {optional: true}),
query('fuse-content > :enter', animateChild(), {optional: true}) query('content > :enter', animateChild(), {optional: true})
]) ])
]) ])
]), ]),
@ -282,7 +286,7 @@ export const fuseAnimations = [
trigger('routerTransitionUp', [ trigger('routerTransitionUp', [
transition('* => *', [ transition('* => *', [
query('fuse-content > :enter, fuse-content > :leave', [ query('content > :enter, content > :leave', [
style({ style({
position: 'absolute', position: 'absolute',
top : 0, top : 0,
@ -291,14 +295,14 @@ export const fuseAnimations = [
right : 0 right : 0
}) })
], {optional: true}), ], {optional: true}),
query('fuse-content > :enter', [ query('content > :enter', [
style({ style({
transform: 'translateY(100%)', transform: 'translateY(100%)',
opacity : 0 opacity : 0
}) })
], {optional: true}), ], {optional: true}),
group([ group([
query('fuse-content > :leave', [ query('content > :leave', [
style({ style({
transform: 'translateY(0)', transform: 'translateY(0)',
opacity : 1 opacity : 1
@ -309,7 +313,7 @@ export const fuseAnimations = [
opacity : 0 opacity : 0
})) }))
], {optional: true}), ], {optional: true}),
query('fuse-content > :enter', [ query('content > :enter', [
style({transform: 'translateY(100%)'}), style({transform: 'translateY(100%)'}),
animate('600ms cubic-bezier(0.0, 0.0, 0.2, 1)', animate('600ms cubic-bezier(0.0, 0.0, 0.2, 1)',
style({ style({
@ -318,15 +322,15 @@ export const fuseAnimations = [
})) }))
], {optional: true}) ], {optional: true})
]), ]),
query('fuse-content > :leave', animateChild(), {optional: true}), query('content > :leave', animateChild(), {optional: true}),
query('fuse-content > :enter', animateChild(), {optional: true}) query('content > :enter', animateChild(), {optional: true})
]) ])
]), ]),
trigger('routerTransitionDown', [ trigger('routerTransitionDown', [
transition('* => *', [ transition('* => *', [
query('fuse-content > :enter, fuse-content > :leave', [ query('content > :enter, content > :leave', [
style({ style({
position: 'absolute', position: 'absolute',
top : 0, top : 0,
@ -335,7 +339,7 @@ export const fuseAnimations = [
right : 0 right : 0
}) })
], {optional: true}), ], {optional: true}),
query('fuse-content > :enter', [ query('content > :enter', [
style({ style({
transform: 'translateY(-100%)', transform: 'translateY(-100%)',
opacity : 0 opacity : 0
@ -343,7 +347,7 @@ export const fuseAnimations = [
], {optional: true}), ], {optional: true}),
sequence([ sequence([
group([ group([
query('fuse-content > :leave', [ query('content > :leave', [
style({ style({
transform: 'translateY(0)', transform: 'translateY(0)',
opacity : 1 opacity : 1
@ -354,7 +358,7 @@ export const fuseAnimations = [
opacity : 0 opacity : 0
})) }))
], {optional: true}), ], {optional: true}),
query('fuse-content > :enter', [ query('content > :enter', [
style({transform: 'translateY(-100%)'}), style({transform: 'translateY(-100%)'}),
animate('600ms cubic-bezier(0.0, 0.0, 0.2, 1)', animate('600ms cubic-bezier(0.0, 0.0, 0.2, 1)',
style({ style({
@ -363,8 +367,8 @@ export const fuseAnimations = [
})) }))
], {optional: true}) ], {optional: true})
]), ]),
query('fuse-content > :leave', animateChild(), {optional: true}), query('content > :leave', animateChild(), {optional: true}),
query('fuse-content > :enter', animateChild(), {optional: true}) query('content > :enter', animateChild(), {optional: true})
]) ])
]) ])
]), ]),
@ -373,7 +377,7 @@ export const fuseAnimations = [
transition('* => *', group([ transition('* => *', group([
query('fuse-content > :enter, fuse-content > :leave ', [ query('content > :enter, content > :leave ', [
style({ style({
position: 'absolute', position: 'absolute',
top : 0, top : 0,
@ -383,12 +387,12 @@ export const fuseAnimations = [
}) })
], {optional: true}), ], {optional: true}),
query('fuse-content > :enter', [ query('content > :enter', [
style({ style({
opacity: 0 opacity: 0
}) })
], {optional: true}), ], {optional: true}),
query('fuse-content > :leave', [ query('content > :leave', [
style({ style({
opacity: 1 opacity: 1
}), }),
@ -397,7 +401,7 @@ export const fuseAnimations = [
opacity: 0 opacity: 0
})) }))
], {optional: true}), ], {optional: true}),
query('fuse-content > :enter', [ query('content > :enter', [
style({ style({
opacity: 0 opacity: 0
}), }),
@ -406,8 +410,8 @@ export const fuseAnimations = [
opacity: 1 opacity: 1
})) }))
], {optional: true}), ], {optional: true}),
query('fuse-content > :enter', animateChild(), {optional: true}), query('content > :enter', animateChild(), {optional: true}),
query('fuse-content > :leave', animateChild(), {optional: true}) query('content > :leave', animateChild(), {optional: true})
])) ]))
]) ])
]; ];

View File

@ -10,7 +10,14 @@ export class FuseConfirmDialogComponent
{ {
public confirmMessage: string; public confirmMessage: string;
constructor(public dialogRef: MatDialogRef<FuseConfirmDialogComponent>) /**
* Constructor
*
* @param {MatDialogRef<FuseConfirmDialogComponent>} dialogRef
*/
constructor(
public dialogRef: MatDialogRef<FuseConfirmDialogComponent>
)
{ {
} }

View File

@ -1,8 +1,6 @@
import { Component, Input, OnInit } from '@angular/core'; import { Component, Input, OnDestroy, OnInit } from '@angular/core';
import { interval, Subject } from 'rxjs';
import { interval } from 'rxjs'; import { map, takeUntil } from 'rxjs/operators';
import { map } from 'rxjs/operators';
import * as moment from 'moment'; import * as moment from 'moment';
@Component({ @Component({
@ -10,46 +8,82 @@ import * as moment from 'moment';
templateUrl: './countdown.component.html', templateUrl: './countdown.component.html',
styleUrls : ['./countdown.component.scss'] styleUrls : ['./countdown.component.scss']
}) })
export class FuseCountdownComponent implements OnInit export class FuseCountdownComponent implements OnInit, OnDestroy
{ {
@Input('eventDate') eventDate; // Event date
@Input('eventDate')
eventDate;
countdown: any; countdown: any;
// Private
private _unsubscribeAll: Subject<any>;
/**
* Constructor
*/
constructor() constructor()
{ {
// Set the defaults
this.countdown = { this.countdown = {
days : '', days : '',
hours : '', hours : '',
minutes: '', minutes: '',
seconds: '' seconds: ''
}; };
// Set the private defaults
this._unsubscribeAll = new Subject();
} }
ngOnInit() // -----------------------------------------------------------------------------------------------------
// @ Lifecycle hooks
// -----------------------------------------------------------------------------------------------------
/**
* On init
*/
ngOnInit(): void
{ {
const currDate = moment(); const currDate = moment();
const eventDate = moment(this.eventDate); const eventDate = moment(this.eventDate);
// Get the difference in between the current date and event date
let diff = eventDate.diff(currDate, 'seconds'); let diff = eventDate.diff(currDate, 'seconds');
const countDown = interval(1000).pipe( // Create a subscribable interval
map(value => { const countDown = interval(1000)
return diff = diff - 1; .pipe(
}), map(value => {
map(value => { return diff = diff - 1;
const timeLeft = moment.duration(value, 'seconds'); }),
map(value => {
const timeLeft = moment.duration(value, 'seconds');
return { return {
days : timeLeft.asDays().toFixed(0), days : timeLeft.asDays().toFixed(0),
hours : timeLeft.hours(), hours : timeLeft.hours(),
minutes: timeLeft.minutes(), minutes: timeLeft.minutes(),
seconds: timeLeft.seconds() seconds: timeLeft.seconds()
}; };
}) })
); );
countDown.subscribe(value => { // Subscribe to the countdown interval
this.countdown = value; countDown
}); .pipe(takeUntil(this._unsubscribeAll))
.subscribe(value => {
this.countdown = value;
});
}
/**
* On destroy
*/
ngOnDestroy(): void
{
// Unsubscribe from all subscriptions
this._unsubscribeAll.next();
this._unsubscribeAll.complete();
} }
} }

View File

@ -7,6 +7,9 @@ import { Component } from '@angular/core';
}) })
export class FuseDemoContentComponent export class FuseDemoContentComponent
{ {
/**
* Constructor
*/
constructor() constructor()
{ {
} }

View File

@ -1,99 +1,99 @@
<div class="demo-sidenav"> <div class="demo-sidebar">
<mat-list> <mat-list>
<h3 matSubheader>Sidenav Demo</h3> <h3 matSubheader>Sidebar Demo</h3>
<mat-list-item> <mat-list-item>
<span>Sidenav Item 1</span> <span>Sidebar Item 1</span>
</mat-list-item> </mat-list-item>
<mat-divider></mat-divider> <mat-divider></mat-divider>
<mat-list-item> <mat-list-item>
<span>Sidenav Item 2</span> <span>Sidebar Item 2</span>
</mat-list-item> </mat-list-item>
<mat-divider></mat-divider> <mat-divider></mat-divider>
<mat-list-item> <mat-list-item>
<span>Sidenav Item 3</span> <span>Sidebar Item 3</span>
</mat-list-item> </mat-list-item>
<mat-divider></mat-divider> <mat-divider></mat-divider>
<mat-list-item> <mat-list-item>
<span>Sidenav Item 4</span> <span>Sidebar Item 4</span>
</mat-list-item> </mat-list-item>
<mat-divider></mat-divider> <mat-divider></mat-divider>
<mat-list-item> <mat-list-item>
<span>Sidenav Item 5</span> <span>Sidebar Item 5</span>
</mat-list-item> </mat-list-item>
<mat-divider></mat-divider> <mat-divider></mat-divider>
<mat-list-item> <mat-list-item>
<span>Sidenav Item 6</span> <span>Sidebar Item 6</span>
</mat-list-item> </mat-list-item>
<mat-divider></mat-divider> <mat-divider></mat-divider>
<mat-list-item> <mat-list-item>
<span>Sidenav Item 7</span> <span>Sidebar Item 7</span>
</mat-list-item> </mat-list-item>
<mat-divider></mat-divider> <mat-divider></mat-divider>
<mat-list-item> <mat-list-item>
<span>Sidenav Item 8</span> <span>Sidebar Item 8</span>
</mat-list-item> </mat-list-item>
<mat-divider></mat-divider> <mat-divider></mat-divider>
<mat-list-item> <mat-list-item>
<span>Sidenav Item 9</span> <span>Sidebar Item 9</span>
</mat-list-item> </mat-list-item>
<mat-divider></mat-divider> <mat-divider></mat-divider>
<mat-list-item> <mat-list-item>
<span>Sidenav Item 10</span> <span>Sidebar Item 10</span>
</mat-list-item> </mat-list-item>
<mat-divider></mat-divider> <mat-divider></mat-divider>
<mat-list-item> <mat-list-item>
<span>Sidenav Item 11</span> <span>Sidebar Item 11</span>
</mat-list-item> </mat-list-item>
<mat-divider></mat-divider> <mat-divider></mat-divider>
<mat-list-item> <mat-list-item>
<span>Sidenav Item 12</span> <span>Sidebar Item 12</span>
</mat-list-item> </mat-list-item>
<mat-divider></mat-divider> <mat-divider></mat-divider>
<mat-list-item> <mat-list-item>
<span>Sidenav Item 13</span> <span>Sidebar Item 13</span>
</mat-list-item> </mat-list-item>
<mat-divider></mat-divider> <mat-divider></mat-divider>
<mat-list-item> <mat-list-item>
<span>Sidenav Item 14</span> <span>Sidebar Item 14</span>
</mat-list-item> </mat-list-item>
<mat-divider></mat-divider> <mat-divider></mat-divider>
<mat-list-item> <mat-list-item>
<span>Sidenav Item 15</span> <span>Sidebar Item 15</span>
</mat-list-item> </mat-list-item>
<mat-divider></mat-divider> <mat-divider></mat-divider>
<mat-list-item> <mat-list-item>
<span>Sidenav Item 16</span> <span>Sidebar Item 16</span>
</mat-list-item> </mat-list-item>
</mat-list> </mat-list>

View File

@ -0,0 +1,16 @@
import { Component } from '@angular/core';
@Component({
selector : 'fuse-demo-sidebar',
templateUrl: './demo-sidebar.component.html',
styleUrls : ['./demo-sidebar.component.scss']
})
export class FuseDemoSidebarComponent
{
/**
* Constructor
*/
constructor()
{
}
}

View File

@ -1,13 +0,0 @@
import { Component } from '@angular/core';
@Component({
selector : 'fuse-demo-sidenav',
templateUrl: './demo-sidenav.component.html',
styleUrls : ['./demo-sidenav.component.scss']
})
export class FuseDemoSidenavComponent
{
constructor()
{
}
}

View File

@ -4,12 +4,12 @@ import { RouterModule } from '@angular/router';
import { MatDividerModule, MatListModule } from '@angular/material'; import { MatDividerModule, MatListModule } from '@angular/material';
import { FuseDemoContentComponent } from './demo-content/demo-content.component'; import { FuseDemoContentComponent } from './demo-content/demo-content.component';
import { FuseDemoSidenavComponent } from './demo-sidenav/demo-sidenav.component'; import { FuseDemoSidebarComponent } from './demo-sidebar/demo-sidebar.component';
@NgModule({ @NgModule({
declarations: [ declarations: [
FuseDemoContentComponent, FuseDemoContentComponent,
FuseDemoSidenavComponent FuseDemoSidebarComponent
], ],
imports : [ imports : [
RouterModule, RouterModule,
@ -19,7 +19,7 @@ import { FuseDemoSidenavComponent } from './demo-sidenav/demo-sidenav.component'
], ],
exports : [ exports : [
FuseDemoContentComponent, FuseDemoContentComponent,
FuseDemoSidenavComponent FuseDemoSidebarComponent
] ]
}) })
export class FuseDemoModule export class FuseDemoModule

View File

@ -3,4 +3,5 @@
padding: 8px; padding: 8px;
background: #263238; background: #263238;
cursor: text; cursor: text;
overflow: auto;
} }

View File

@ -1,28 +1,55 @@
import { Component, ContentChild, ElementRef, Input, OnInit } from '@angular/core'; import { Component, ContentChild, ElementRef, Input, OnDestroy, OnInit } from '@angular/core';
import { HttpClient } from '@angular/common/http'; import { HttpClient } from '@angular/common/http';
import { Subject } from 'rxjs';
import { takeUntil } from 'rxjs/operators';
import * as Prism from 'prismjs/prism'; import * as Prism from 'prismjs/prism';
import './prism-languages'; import '@fuse/components/highlight/prism-languages';
@Component({ @Component({
selector : 'fuse-highlight', selector : 'fuse-highlight',
template : ' ', template : '',
styleUrls: ['./highlight.component.scss'] styleUrls: ['./highlight.component.scss']
}) })
export class FuseHighlightComponent implements OnInit export class FuseHighlightComponent implements OnInit, OnDestroy
{ {
@ContentChild('source') source: ElementRef; // Source
@Input('lang') lang: string; @ContentChild('source')
@Input('path') path: string; source: ElementRef;
// Lang
@Input('lang')
lang: string;
// Path
@Input('path')
path: string;
// Private
private _unsubscribeAll: Subject<any>;
/**
* Constructor
*
* @param {ElementRef} _elementRef
* @param {HttpClient} _httpClient
*/
constructor( constructor(
private elementRef: ElementRef, private _elementRef: ElementRef,
private http: HttpClient private _httpClient: HttpClient
) )
{ {
// Set the private defaults
this._unsubscribeAll = new Subject();
} }
ngOnInit() // -----------------------------------------------------------------------------------------------------
// @ Lifecycle hooks
// -----------------------------------------------------------------------------------------------------
/**
* On init
*/
ngOnInit(): void
{ {
// If there is no language defined, return... // If there is no language defined, return...
if ( !this.lang ) if ( !this.lang )
@ -34,11 +61,13 @@ export class FuseHighlightComponent implements OnInit
if ( this.path ) if ( this.path )
{ {
// Get the source // Get the source
this.http.get(this.path, {responseType: 'text'}).subscribe((response) => { this._httpClient.get(this.path, {responseType: 'text'})
.pipe(takeUntil(this._unsubscribeAll))
.subscribe((response) => {
// Highlight it // Highlight it
this.highlight(response); this.highlight(response);
}); });
} }
// If the path is not defined and the source element exists... // If the path is not defined and the source element exists...
@ -49,7 +78,26 @@ export class FuseHighlightComponent implements OnInit
} }
} }
highlight(sourceCode) /**
* On destroy
*/
ngOnDestroy(): void
{
// Unsubscribe from all subscriptions
this._unsubscribeAll.next();
this._unsubscribeAll.complete();
}
// -----------------------------------------------------------------------------------------------------
// @ Public methods
// -----------------------------------------------------------------------------------------------------
/**
* Highlight the given source code
*
* @param sourceCode
*/
highlight(sourceCode): void
{ {
// Split the source into lines // Split the source into lines
const sourceLines = sourceCode.split('\n'); const sourceLines = sourceCode.split('\n');
@ -94,9 +142,8 @@ export class FuseHighlightComponent implements OnInit
const highlightedCode = Prism.highlight(source, Prism.languages[this.lang]); const highlightedCode = Prism.highlight(source, Prism.languages[this.lang]);
// Replace the innerHTML of the component with the highlighted code // Replace the innerHTML of the component with the highlighted code
this.elementRef.nativeElement.innerHTML = this._elementRef.nativeElement.innerHTML =
'<pre><code class="highlight language-' + this.lang + '">' + highlightedCode + '</code></pre>'; '<pre><code class="highlight language-' + this.lang + '">' + highlightedCode + '</code></pre>';
} }
} }

View File

@ -1,4 +1,5 @@
import 'prismjs/prism'; import 'prismjs/prism';
import 'prismjs/components/prism-bash';
import 'prismjs/components/prism-c'; import 'prismjs/components/prism-c';
import 'prismjs/components/prism-cpp'; import 'prismjs/components/prism-cpp';
import 'prismjs/components/prism-csharp'; import 'prismjs/components/prism-csharp';

View File

@ -13,22 +13,78 @@ import { MatColors } from '@fuse/mat-colors';
export class FuseMaterialColorPickerComponent implements OnChanges export class FuseMaterialColorPickerComponent implements OnChanges
{ {
colors: any; colors: any;
selectedColor: any;
hues: string[]; hues: string[];
view = 'palettes'; selectedColor: any;
view: string;
@Input() selectedPalette = ''; @Input()
@Input() selectedHue = ''; selectedPalette: string;
@Input() selectedFg = '';
@Input() value: any;
@Output() onValueChange = new EventEmitter();
@Output() selectedPaletteChange = new EventEmitter();
@Output() selectedHueChange = new EventEmitter();
@Output() selectedClassChange = new EventEmitter();
@Output() selectedBgChange = new EventEmitter();
@Output() selectedFgChange = new EventEmitter();
_selectedClass = ''; @Input()
selectedHue: string;
@Input()
selectedFg: string;
@Input()
value: any;
@Output()
onValueChange: EventEmitter<any>;
@Output()
selectedPaletteChange: EventEmitter<any>;
@Output()
selectedHueChange: EventEmitter<any>;
@Output()
selectedClassChange: EventEmitter<any>;
@Output()
selectedBgChange: EventEmitter<any>;
@Output()
selectedFgChange: EventEmitter<any>;
// Private
_selectedClass: string;
_selectedBg: string;
/**
* Constructor
*/
constructor()
{
// Set the defaults
this.colors = MatColors.all;
this.hues = ['50', '100', '200', '300', '400', '500', '600', '700', '800', '900', 'A100', 'A200', 'A400', 'A700'];
this.selectedFg = '';
this.selectedHue = '';
this.selectedPalette = '';
this.view = 'palettes';
this.onValueChange = new EventEmitter();
this.selectedPaletteChange = new EventEmitter();
this.selectedHueChange = new EventEmitter();
this.selectedClassChange = new EventEmitter();
this.selectedBgChange = new EventEmitter();
this.selectedFgChange = new EventEmitter();
// Set the private defaults
this._selectedClass = '';
this._selectedBg = '';
}
// -----------------------------------------------------------------------------------------------------
// @ Accessors
// -----------------------------------------------------------------------------------------------------
/**
* Selected class
*
* @param value
*/
@Input() @Input()
set selectedClass(value) set selectedClass(value)
{ {
@ -54,7 +110,11 @@ export class FuseMaterialColorPickerComponent implements OnChanges
return this._selectedClass; return this._selectedClass;
} }
_selectedBg = ''; /**
* Selected bg
*
* @param value
*/
@Input() @Input()
set selectedBg(value) set selectedBg(value)
{ {
@ -86,13 +146,16 @@ export class FuseMaterialColorPickerComponent implements OnChanges
return this._selectedBg; return this._selectedBg;
} }
constructor() // -----------------------------------------------------------------------------------------------------
{ // @ Lifecycle hooks
this.colors = MatColors.all; // -----------------------------------------------------------------------------------------------------
this.hues = ['50', '100', '200', '300', '400', '500', '600', '700', '800', '900', 'A100', 'A200', 'A400', 'A700'];
}
ngOnChanges(changes: any) /**
* On changes
*
* @param changes
*/
ngOnChanges(changes: any): void
{ {
if ( changes.selectedBg && changes.selectedBg.currentValue === '' || if ( changes.selectedBg && changes.selectedBg.currentValue === '' ||
changes.selectedClass && changes.selectedClass.currentValue === '' || changes.selectedClass && changes.selectedClass.currentValue === '' ||
@ -107,20 +170,37 @@ export class FuseMaterialColorPickerComponent implements OnChanges
} }
} }
selectPalette(palette) // -----------------------------------------------------------------------------------------------------
// @ Public methods
// -----------------------------------------------------------------------------------------------------
/**
* Select palette
*
* @param palette
*/
selectPalette(palette): void
{ {
this.selectedPalette = palette; this.selectedPalette = palette;
this.updateSelectedColor(); this.updateSelectedColor();
this.view = 'hues'; this.view = 'hues';
} }
selectHue(hue) /**
* Select hue
*
* @param hue
*/
selectHue(hue): void
{ {
this.selectedHue = hue; this.selectedHue = hue;
this.updateSelectedColor(); this.updateSelectedColor();
} }
removeColor() /**
* Remove color
*/
removeColor(): void
{ {
this.selectedPalette = ''; this.selectedPalette = '';
this.selectedHue = ''; this.selectedHue = '';
@ -128,7 +208,10 @@ export class FuseMaterialColorPickerComponent implements OnChanges
this.view = 'palettes'; this.view = 'palettes';
} }
updateSelectedColor() /**
* Update selected color
*/
updateSelectedColor(): void
{ {
setTimeout(() => { setTimeout(() => {
@ -168,12 +251,18 @@ export class FuseMaterialColorPickerComponent implements OnChanges
}); });
} }
backToPaletteSelection() /**
* Go back to palette selection
*/
backToPaletteSelection(): void
{ {
this.view = 'palettes'; this.view = 'palettes';
} }
onMenuOpen() /**
* On menu open
*/
onMenuOpen(): void
{ {
if ( this.selectedPalette === '' ) if ( this.selectedPalette === '' )
{ {

View File

@ -1,7 +1,6 @@
import { NgModule } from '@angular/core'; import { NgModule } from '@angular/core';
import { CommonModule } from '@angular/common'; import { CommonModule } from '@angular/common';
import { FlexLayoutModule } from '@angular/flex-layout'; import { FlexLayoutModule } from '@angular/flex-layout';
import { MatButtonModule, MatIconModule, MatMenuModule, MatRippleModule } from '@angular/material'; import { MatButtonModule, MatIconModule, MatMenuModule, MatRippleModule } from '@angular/material';
import { FusePipesModule } from '@fuse/pipes/pipes.module'; import { FusePipesModule } from '@fuse/pipes/pipes.module';

View File

@ -31,18 +31,19 @@
[ngStyle]="{'background-color': item.badge.bg,'color': item.badge.fg}"> [ngStyle]="{'background-color': item.badge.bg,'color': item.badge.fg}">
{{item.badge.title}} {{item.badge.title}}
</span> </span>
<mat-icon class="collapse-arrow">keyboard_arrow_right</mat-icon> <mat-icon class="collapsable-arrow">keyboard_arrow_right</mat-icon>
</ng-template> </ng-template>
<div class="children" [ngClass]="{'open': isOpen}"> <div class="children" [ngClass]="{'open': isOpen}">
<div class="{{fuseSettings.colorClasses.navbar}}"> <div class="{{fuseConfig.layout.navbar.background}}">
<ng-container *ngFor="let item of item.children"> <ng-container *ngFor="let item of item.children">
<fuse-nav-horizontal-item *ngIf="item.type=='item'" [item]="item"></fuse-nav-horizontal-item> <fuse-nav-horizontal-item *ngIf="item.type=='item'" [item]="item"></fuse-nav-horizontal-item>
<fuse-nav-horizontal-collapse *ngIf="item.type=='collapse'" <fuse-nav-horizontal-collapsable *ngIf="item.type=='collapsable'"
[item]="item"></fuse-nav-horizontal-collapse> [item]="item"></fuse-nav-horizontal-collapsable>
<fuse-nav-horizontal-collapse *ngIf="item.type=='group'" [item]="item"></fuse-nav-horizontal-collapse> <fuse-nav-horizontal-collapsable *ngIf="item.type=='group'"
[item]="item"></fuse-nav-horizontal-collapsable>
</ng-container> </ng-container>
</div> </div>

View File

@ -0,0 +1,86 @@
import { Component, HostBinding, HostListener, Input, OnDestroy, OnInit } from '@angular/core';
import { Subject } from 'rxjs';
import { takeUntil } from 'rxjs/operators';
import { fuseAnimations } from '@fuse/animations';
import { FuseConfigService } from '@fuse/services/config.service';
@Component({
selector : 'fuse-nav-horizontal-collapsable',
templateUrl: './collapsable.component.html',
styleUrls : ['./collapsable.component.scss'],
animations : fuseAnimations
})
export class FuseNavHorizontalCollapsableComponent implements OnInit, OnDestroy
{
fuseConfig: any;
isOpen = false;
@HostBinding('class')
classes = 'nav-collapsable nav-item';
@Input()
item: any;
// Private
private _unsubscribeAll: Subject<any>;
constructor(
private _fuseConfigService: FuseConfigService
)
{
// Set the private defaults
this._unsubscribeAll = new Subject();
}
// -----------------------------------------------------------------------------------------------------
// @ Lifecycle hooks
// -----------------------------------------------------------------------------------------------------
/**
* On init
*/
ngOnInit(): void
{
// Subscribe to config changes
this._fuseConfigService.config
.pipe(takeUntil(this._unsubscribeAll))
.subscribe(
(config) => {
this.fuseConfig = config;
}
);
}
/**
* On destroy
*/
ngOnDestroy(): void
{
// Unsubscribe from all subscriptions
this._unsubscribeAll.next();
this._unsubscribeAll.complete();
}
// -----------------------------------------------------------------------------------------------------
// @ Public methods
// -----------------------------------------------------------------------------------------------------
/**
* Open
*/
@HostListener('mouseenter')
open(): void
{
this.isOpen = true;
}
/**
* Close
*/
@HostListener('mouseleave')
close(): void
{
this.isOpen = false;
}
}

View File

@ -0,0 +1,23 @@
import { Component, HostBinding, Input } from '@angular/core';
@Component({
selector : 'fuse-nav-horizontal-item',
templateUrl: './item.component.html',
styleUrls : ['./item.component.scss']
})
export class FuseNavHorizontalItemComponent
{
@HostBinding('class')
classes = 'nav-item';
@Input()
item: any;
/**
* Constructor
*/
constructor()
{
}
}

View File

@ -1,51 +0,0 @@
import { Component, HostBinding, HostListener, Input, OnDestroy } from '@angular/core';
import { Subscription } from 'rxjs';
import { fuseAnimations } from '../../../../animations/index';
import { FuseConfigService } from '../../../../services/config.service';
@Component({
selector : 'fuse-nav-horizontal-collapse',
templateUrl: './nav-horizontal-collapse.component.html',
styleUrls : ['./nav-horizontal-collapse.component.scss'],
animations : fuseAnimations
})
export class FuseNavHorizontalCollapseComponent implements OnDestroy
{
onConfigChanged: Subscription;
fuseSettings: any;
isOpen = false;
@HostBinding('class') classes = 'nav-item nav-collapse';
@Input() item: any;
@HostListener('mouseenter')
open()
{
this.isOpen = true;
}
@HostListener('mouseleave')
close()
{
this.isOpen = false;
}
constructor(
private fuseConfig: FuseConfigService
)
{
this.onConfigChanged =
this.fuseConfig.onConfigChanged
.subscribe(
(newSettings) => {
this.fuseSettings = newSettings;
}
);
}
ngOnDestroy()
{
this.onConfigChanged.unsubscribe();
}
}

View File

@ -1,12 +0,0 @@
import { Component, HostBinding, Input } from '@angular/core';
@Component({
selector : 'fuse-nav-horizontal-item',
templateUrl: './nav-horizontal-item.component.html',
styleUrls : ['./nav-horizontal-item.component.scss']
})
export class FuseNavHorizontalItemComponent
{
@HostBinding('class') classes = 'nav-item';
@Input() item: any;
}

View File

@ -7,7 +7,7 @@
<ng-container *ngFor="let item of navigation"> <ng-container *ngFor="let item of navigation">
<fuse-nav-vertical-group *ngIf="item.type=='group'" [item]="item"></fuse-nav-vertical-group> <fuse-nav-vertical-group *ngIf="item.type=='group'" [item]="item"></fuse-nav-vertical-group>
<fuse-nav-vertical-collapse *ngIf="item.type=='collapse'" [item]="item"></fuse-nav-vertical-collapse> <fuse-nav-vertical-collapsable *ngIf="item.type=='collapse'" [item]="item"></fuse-nav-vertical-collapsable>
<fuse-nav-vertical-item *ngIf="item.type=='item'" [item]="item"></fuse-nav-vertical-item> <fuse-nav-vertical-item *ngIf="item.type=='item'" [item]="item"></fuse-nav-vertical-item>
</ng-container> </ng-container>
@ -20,8 +20,8 @@
<ng-container *ngFor="let item of navigation"> <ng-container *ngFor="let item of navigation">
<fuse-nav-horizontal-collapse *ngIf="item.type=='group'" [item]="item"></fuse-nav-horizontal-collapse> <fuse-nav-horizontal-collapsable *ngIf="item.type=='group'" [item]="item"></fuse-nav-horizontal-collapsable>
<fuse-nav-horizontal-collapse *ngIf="item.type=='collapse'" [item]="item"></fuse-nav-horizontal-collapse> <fuse-nav-horizontal-collapsable *ngIf="item.type=='collapse'" [item]="item"></fuse-nav-horizontal-collapsable>
<fuse-nav-horizontal-item *ngIf="item.type=='item'" [item]="item"></fuse-nav-horizontal-item> <fuse-nav-horizontal-item *ngIf="item.type=='item'" [item]="item"></fuse-nav-horizontal-item>
</ng-container> </ng-container>

View File

@ -1,4 +1,8 @@
import { Component, Input, ViewEncapsulation } from '@angular/core'; import { Component, Input, OnInit, ViewEncapsulation } from '@angular/core';
import { Subject } from 'rxjs';
import { takeUntil } from 'rxjs/operators';
import { FuseNavigationService } from '@fuse/components/navigation/navigation.service';
@Component({ @Component({
selector : 'fuse-navigation', selector : 'fuse-navigation',
@ -6,13 +10,45 @@ import { Component, Input, ViewEncapsulation } from '@angular/core';
styleUrls : ['./navigation.component.scss'], styleUrls : ['./navigation.component.scss'],
encapsulation: ViewEncapsulation.None encapsulation: ViewEncapsulation.None
}) })
export class FuseNavigationComponent export class FuseNavigationComponent implements OnInit
{ {
@Input() layout = 'vertical'; @Input()
@Input() navigation: any; layout = 'vertical';
constructor() @Input()
navigation: any;
// Private
private _unsubscribeAll: Subject<any>;
/**
* Constructor
*/
constructor(
private _fuseNavigationService: FuseNavigationService
)
{ {
// Set the private defaults
this._unsubscribeAll = new Subject();
}
// -----------------------------------------------------------------------------------------------------
// @ Lifecycle hooks
// -----------------------------------------------------------------------------------------------------
/**
* On init
*/
ngOnInit(): void
{
// Load the navigation either from the input or from the service
this.navigation = this.navigation || this._fuseNavigationService.getCurrentNavigation();
// Subscribe to the current navigation changes
this._fuseNavigationService.onNavigationChanged
.pipe(takeUntil(this._unsubscribeAll))
.subscribe(() => {
this.navigation = this._fuseNavigationService.getCurrentNavigation();
});
} }
} }

View File

@ -6,11 +6,11 @@ import { MatIconModule, MatRippleModule } from '@angular/material';
import { TranslateModule } from '@ngx-translate/core'; import { TranslateModule } from '@ngx-translate/core';
import { FuseNavigationComponent } from './navigation.component'; import { FuseNavigationComponent } from './navigation.component';
import { FuseNavVerticalItemComponent } from './vertical/nav-item/nav-vertical-item.component'; import { FuseNavVerticalItemComponent } from './vertical/item/item.component';
import { FuseNavVerticalCollapseComponent } from './vertical/nav-collapse/nav-vertical-collapse.component'; import { FuseNavVerticalCollapsableComponent } from './vertical/collapsable/collapsable.component';
import { FuseNavVerticalGroupComponent } from './vertical/nav-group/nav-vertical-group.component'; import { FuseNavVerticalGroupComponent } from './vertical/group/group.component';
import { FuseNavHorizontalItemComponent } from './horizontal/nav-item/nav-horizontal-item.component'; import { FuseNavHorizontalItemComponent } from './horizontal/item/item.component';
import { FuseNavHorizontalCollapseComponent } from './horizontal/nav-collapse/nav-horizontal-collapse.component'; import { FuseNavHorizontalCollapsableComponent } from './horizontal/collapsable/collapsable.component';
@NgModule({ @NgModule({
imports : [ imports : [
@ -29,9 +29,9 @@ import { FuseNavHorizontalCollapseComponent } from './horizontal/nav-collapse/na
FuseNavigationComponent, FuseNavigationComponent,
FuseNavVerticalGroupComponent, FuseNavVerticalGroupComponent,
FuseNavVerticalItemComponent, FuseNavVerticalItemComponent,
FuseNavVerticalCollapseComponent, FuseNavVerticalCollapsableComponent,
FuseNavHorizontalItemComponent, FuseNavHorizontalItemComponent,
FuseNavHorizontalCollapseComponent FuseNavHorizontalCollapsableComponent
] ]
}) })
export class FuseNavigationModule export class FuseNavigationModule

View File

@ -1,34 +1,161 @@
import { Injectable } from '@angular/core'; import { Injectable } from '@angular/core';
import { Subject } from 'rxjs'; import { BehaviorSubject, Observable, Subject } from 'rxjs';
@Injectable() @Injectable()
export class FuseNavigationService export class FuseNavigationService
{ {
flatNavigation: any[] = []; flatNavigation: any[] = [];
onItemCollapsed: Subject<any> = new Subject; onItemCollapsed: Subject<any>;
onItemCollapseToggled: Subject<any> = new Subject; onItemCollapseToggled: Subject<any>;
// Private
private _onNavigationChanged: BehaviorSubject<any>;
private _onNavigationRegistered: BehaviorSubject<any>;
private _onNavigationUnregistered: BehaviorSubject<any>;
private _currentNavigationKey: string;
private _registry: { [key: string]: any } = {};
/**
* Constructor
*/
constructor() constructor()
{ {
// Set the defaults
this.onItemCollapsed = new Subject();
this.onItemCollapseToggled = new Subject();
// Set the private defaults
this._currentNavigationKey = null;
this._onNavigationChanged = new BehaviorSubject(null);
this._onNavigationRegistered = new BehaviorSubject(null);
this._onNavigationUnregistered = new BehaviorSubject(null);
}
// -----------------------------------------------------------------------------------------------------
// @ Accessors
// -----------------------------------------------------------------------------------------------------
/**
* Get onNavigationChanged
*
* @returns {Observable<any>}
*/
get onNavigationChanged(): Observable<any>
{
return this._onNavigationChanged.asObservable();
}
/**
* Get onNavigationRegistered
*
* @returns {Observable<any>}
*/
get onNavigationRegistered(): Observable<any>
{
return this._onNavigationRegistered.asObservable();
}
/**
* Get onNavigationUnregistered
*
* @returns {Observable<any>}
*/
get onNavigationUnregistered(): Observable<any>
{
return this._onNavigationUnregistered.asObservable();
}
// -----------------------------------------------------------------------------------------------------
// @ Public methods
// -----------------------------------------------------------------------------------------------------
/**
* Register the given navigation
* with the given key
*
* @param key
* @param navigation
*/
register(key, navigation): void
{
// Check if the key already being used
if ( this._registry[key] )
{
console.error(`The navigation with the key '${key}' already exists. Either unregister it first or use a unique key.`);
return;
}
// Add to the registry
this._registry[key] = navigation;
// Notify the subject
this._onNavigationRegistered.next([key, navigation]);
}
/**
* Unregister the navigation from the registry
* @param key
*/
unregister(key): void
{
// Check if the navigation exists
if ( !this._registry[key] )
{
console.warn(`The navigation with the key '${key}' doesn't exist in the registry.`);
}
// Unregister the sidebar
delete this._registry[key];
// Notify the subject
this._onNavigationUnregistered.next(key);
}
/**
* Get navigation from registry by key
*
* @param key
* @returns {any}
*/
getNavigation(key): any
{
// Check if the navigation exists
if ( !this._registry[key] )
{
console.warn(`The navigation with the key '${key}' doesn't exist in the registry.`);
return;
}
// Return the sidebar
return this._registry[key];
} }
/** /**
* Get flattened navigation array * Get flattened navigation array
*
* @param navigation * @param navigation
* @returns {any[]} * @returns {any[]}
*/ */
getFlatNavigation(navigation) getFlatNavigation(navigation): any
{ {
for ( const navItem of navigation ) for ( const navItem of navigation )
{ {
if ( navItem.type === 'item' ) if ( navItem.type === 'item' )
{ {
this.flatNavigation.push({ this.flatNavigation.push({
title: navItem.title, id : navItem.id || null,
type : navItem.type, title : navItem.title || null,
icon : navItem.icon || false, translate : navItem.translate || null,
url : navItem.url type : navItem.type,
icon : navItem.icon || null,
url : navItem.url || null,
function : navItem.function || null,
exactMatch: navItem.exactMatch || false,
badge : navItem.badge || null
}); });
continue; continue;
@ -45,4 +172,186 @@ export class FuseNavigationService
return this.flatNavigation; return this.flatNavigation;
} }
/**
* Get the current navigation
*
* @returns {any}
*/
getCurrentNavigation(): any
{
if ( !this._currentNavigationKey )
{
console.warn(`The current navigation is not set.`);
return;
}
return this.getNavigation(this._currentNavigationKey);
}
/**
* Set the navigation with the key
* as the current navigation
*
* @param key
*/
setCurrentNavigation(key): void
{
// Check if the sidebar exists
if ( !this._registry[key] )
{
console.warn(`The navigation with the key '${key}' doesn't exist in the registry.`);
return;
}
// Set the current navigation key
this._currentNavigationKey = key;
// Notify the subject
this._onNavigationChanged.next(key);
}
/**
* Get navigation item by id from the
* current navigation
*
* @param id
* @param {any} navigation
* @returns {any | boolean}
*/
getNavigationItem(id, navigation = null): any | boolean
{
if ( !navigation )
{
navigation = this.getCurrentNavigation();
}
for ( const item of navigation )
{
if ( item.id === id )
{
return item;
}
if ( item.children )
{
const childItem = this.getNavigationItem(id, item.children);
if ( childItem )
{
return childItem;
}
}
}
return false;
}
/**
* Get the parent of the navigation item
* with the id
*
* @param id
* @param {any} navigation
* @param parent
*/
getNavigationItemParent(id, navigation = null, parent = null): any
{
if ( !navigation )
{
navigation = this.getCurrentNavigation();
parent = navigation;
}
for ( const item of navigation )
{
if ( item.id === id )
{
return parent;
}
if ( item.children )
{
const childItem = this.getNavigationItemParent(id, item.children, item);
if ( childItem )
{
return childItem;
}
}
}
return false;
}
/**
* Add a navigation item to the specified location
*
* @param item
* @param id
*/
addNavigationItem(item, id): void
{
// Get the current navigation
const navigation: any[] = this.getCurrentNavigation();
// Add to the end of the navigation
if ( id === 'end' )
{
navigation.push(item);
return;
}
// Add to the start of the navigation
if ( id === 'start' )
{
navigation.unshift(item);
}
// Add it to a specific location
const parent: any = this.getNavigationItem(id);
if ( parent )
{
// Check if parent has a children entry,
// and add it if it doesn't
if ( !parent.children )
{
parent.children = [];
}
// Add the item
parent.children.push(item);
}
}
/**
* Remove navigation item with the given id
*
* @param id
*/
removeNavigationItem(id): void
{
const item = this.getNavigationItem(id);
// Return, if there is not such an item
if ( !item )
{
return;
}
// Get the parent of the item
let parent = this.getNavigationItemParent(id);
// This check is required because of the first level
// of the navigation, since the first level is not
// inside the 'children' array
parent = parent.children || parent;
// Remove the item
parent.splice(parent.indexOf(item), 1);
}
} }

View File

@ -6,24 +6,40 @@
</a> </a>
<!-- item.url --> <!-- item.url -->
<a class="nav-link" *ngIf="item.url && !item.function" (click)="toggleOpen($event)" <a class="nav-link" *ngIf="item.url && !item.externalUrl && !item.function" (click)="toggleOpen($event)"
[routerLink]="[item.url]" routerLinkActive="active" [routerLink]="[item.url]" [routerLinkActive]="['active', 'mat-accent-bg']"
[routerLinkActiveOptions]="{exact: item.exactMatch || false}" matRipple> [routerLinkActiveOptions]="{exact: item.exactMatch || false}"
[target]="item.openInNewTab ? '_blank' : '_self'" matRipple>
<ng-container *ngTemplateOutlet="itemContent"></ng-container>
</a>
<!-- item.externalUrl -->
<a class="nav-link" *ngIf="item.url && item.externalUrl && !item.function" (click)="toggleOpen($event)"
[href]="item.url" [target]="item.openInNewTab ? '_blank' : '_self'" matRipple>
<ng-container *ngTemplateOutlet="itemContent"></ng-container> <ng-container *ngTemplateOutlet="itemContent"></ng-container>
</a> </a>
<!-- item.function --> <!-- item.function -->
<span class="nav-link" *ngIf="!item.url && item.function" (click)="toggleOpen($event);item.function()" matRipple> <span class="nav-link" *ngIf="!item.url && item.function"
(click)="toggleOpen($event);item.function()" matRipple>
<ng-container *ngTemplateOutlet="itemContent"></ng-container> <ng-container *ngTemplateOutlet="itemContent"></ng-container>
</span> </span>
<!-- item.url && item.function --> <!-- item.url && item.function -->
<a class="nav-link" *ngIf="item.url && item.function" (click)="toggleOpen($event);item.function()" <a class="nav-link" *ngIf="item.url && !item.externalUrl && item.function"
[routerLink]="[item.url]" routerLinkActive="active" (click)="toggleOpen($event);item.function()"
[routerLink]="[item.url]" [routerLinkActive]="['active', 'mat-accent-bg']"
[routerLinkActiveOptions]="{exact: item.exactMatch || false}" matRipple> [routerLinkActiveOptions]="{exact: item.exactMatch || false}" matRipple>
<ng-container *ngTemplateOutlet="itemContent"></ng-container> <ng-container *ngTemplateOutlet="itemContent"></ng-container>
</a> </a>
<!-- item.externalUrl && item.function -->
<a class="nav-link" *ngIf="item.url && item.externalUrl && item.function"
(click)="toggleOpen($event);item.function()"
[href]="item.url" [target]="item.openInNewTab ? '_blank' : '_self'" matRipple>
<ng-container *ngTemplateOutlet="itemContent"></ng-container>
</a>
<ng-template #itemContent> <ng-template #itemContent>
<mat-icon class="nav-link-icon" *ngIf="item.icon">{{item.icon}}</mat-icon> <mat-icon class="nav-link-icon" *ngIf="item.icon">{{item.icon}}</mat-icon>
<span class="nav-link-title" [translate]="item.translate">{{item.title}}</span> <span class="nav-link-title" [translate]="item.translate">{{item.title}}</span>
@ -31,13 +47,14 @@
[ngStyle]="{'background-color': item.badge.bg,'color': item.badge.fg}"> [ngStyle]="{'background-color': item.badge.bg,'color': item.badge.fg}">
{{item.badge.title}} {{item.badge.title}}
</span> </span>
<mat-icon class="collapse-arrow">keyboard_arrow_right</mat-icon> <mat-icon class="collapsable-arrow">keyboard_arrow_right</mat-icon>
</ng-template> </ng-template>
<div class="children" [@slideInOut]="isOpen"> <div class="children" [@slideInOut]="isOpen">
<ng-container *ngFor="let item of item.children"> <ng-container *ngFor="let item of item.children">
<fuse-nav-vertical-item *ngIf="item.type=='item'" [item]="item"></fuse-nav-vertical-item> <fuse-nav-vertical-item *ngIf="item.type=='item'" [item]="item"></fuse-nav-vertical-item>
<fuse-nav-vertical-collapse *ngIf="item.type=='collapse'" [item]="item"></fuse-nav-vertical-collapse> <fuse-nav-vertical-collapsable *ngIf="item.type=='collapsable'"
[item]="item"></fuse-nav-vertical-collapsable>
<fuse-nav-vertical-group *ngIf="item.type=='group'" [item]="item"></fuse-nav-vertical-group> <fuse-nav-vertical-group *ngIf="item.type=='group'" [item]="item"></fuse-nav-vertical-group>
</ng-container> </ng-container>
</div> </div>

View File

@ -20,7 +20,7 @@
.nav-link { .nav-link {
.collapse-arrow { .collapsable-arrow {
transition: transform .3s ease-in-out, opacity .25s ease-in-out .1s; transition: transform .3s ease-in-out, opacity .25s ease-in-out .1s;
transform: rotate(0); transform: rotate(0);
} }
@ -34,13 +34,9 @@
> .nav-link { > .nav-link {
.collapse-arrow { .collapsable-arrow {
transform: rotate(90deg); transform: rotate(90deg);
} }
} }
> .children {
}
} }
} }

View File

@ -1,46 +1,79 @@
import { Component, HostBinding, Input, OnInit } from '@angular/core'; import { Component, HostBinding, Input, OnDestroy, OnInit } from '@angular/core';
import { FuseNavigationService } from '../../navigation.service';
import { NavigationEnd, Router } from '@angular/router'; import { NavigationEnd, Router } from '@angular/router';
import { fuseAnimations } from '../../../../animations/index'; import { Subject } from 'rxjs';
import { filter, takeUntil } from 'rxjs/operators';
import { FuseNavigationItem } from '@fuse/types';
import { fuseAnimations } from '@fuse/animations';
import { FuseNavigationService } from '@fuse/components/navigation/navigation.service';
@Component({ @Component({
selector : 'fuse-nav-vertical-collapse', selector : 'fuse-nav-vertical-collapsable',
templateUrl: './nav-vertical-collapse.component.html', templateUrl: './collapsable.component.html',
styleUrls : ['./nav-vertical-collapse.component.scss'], styleUrls : ['./collapsable.component.scss'],
animations : fuseAnimations animations : fuseAnimations
}) })
export class FuseNavVerticalCollapseComponent implements OnInit export class FuseNavVerticalCollapsableComponent implements OnInit, OnDestroy
{ {
@Input() item: any; @Input()
@HostBinding('class') classes = 'nav-collapse nav-item'; item: FuseNavigationItem;
@HostBinding('class.open') public isOpen = false;
@HostBinding('class')
classes = 'nav-collapsable nav-item';
@HostBinding('class.open')
public isOpen = false;
// Private
private _unsubscribeAll: Subject<any>;
/**
* Constructor
*
* @param {FuseNavigationService} _fuseNavigationService
* @param {Router} _router
*/
constructor( constructor(
private navigationService: FuseNavigationService, private _fuseNavigationService: FuseNavigationService,
private router: Router private _router: Router
) )
{ {
// Listen for route changes // Set the private defaults
router.events.subscribe( this._unsubscribeAll = new Subject();
(event) => { }
if ( event instanceof NavigationEnd )
// -----------------------------------------------------------------------------------------------------
// @ Lifecycle hooks
// -----------------------------------------------------------------------------------------------------
/**
* On init
*/
ngOnInit(): void
{
// Listen for router events
this._router.events
.pipe(
filter(event => event instanceof NavigationEnd),
takeUntil(this._unsubscribeAll)
)
.subscribe((event: NavigationEnd) => {
// Check if the url can be found in
// one of the children of this item
if ( this.isUrlInChildren(this.item, event.urlAfterRedirects) )
{ {
// Check if the url can be found in this.expand();
// one of the children of this item
if ( this.isUrlInChildren(this.item, event.urlAfterRedirects) )
{
this.expand();
}
else
{
this.collapse();
}
} }
} else
); {
this.collapse();
}
});
// Listen for collapsing of any navigation item // Listen for collapsing of any navigation item
this.navigationService.onItemCollapsed this._fuseNavigationService.onItemCollapsed
.pipe(takeUntil(this._unsubscribeAll))
.subscribe( .subscribe(
(clickedItem) => { (clickedItem) => {
if ( clickedItem && clickedItem.children ) if ( clickedItem && clickedItem.children )
@ -54,7 +87,7 @@ export class FuseNavVerticalCollapseComponent implements OnInit
// Check if the url can be found in // Check if the url can be found in
// one of the children of this item // one of the children of this item
if ( this.isUrlInChildren(this.item, this.router.url) ) if ( this.isUrlInChildren(this.item, this._router.url) )
{ {
return; return;
} }
@ -67,13 +100,10 @@ export class FuseNavVerticalCollapseComponent implements OnInit
} }
} }
); );
}
ngOnInit()
{
// Check if the url can be found in // Check if the url can be found in
// one of the children of this item // one of the children of this item
if ( this.isUrlInChildren(this.item, this.router.url) ) if ( this.isUrlInChildren(this.item, this._router.url) )
{ {
this.expand(); this.expand();
} }
@ -83,26 +113,40 @@ export class FuseNavVerticalCollapseComponent implements OnInit
} }
} }
/**
* On destroy
*/
ngOnDestroy(): void
{
// Unsubscribe from all subscriptions
this._unsubscribeAll.next();
this._unsubscribeAll.complete();
}
// -----------------------------------------------------------------------------------------------------
// @ Public methods
// -----------------------------------------------------------------------------------------------------
/** /**
* Toggle collapse * Toggle collapse
* *
* @param ev * @param ev
*/ */
toggleOpen(ev) toggleOpen(ev): void
{ {
ev.preventDefault(); ev.preventDefault();
this.isOpen = !this.isOpen; this.isOpen = !this.isOpen;
// Navigation collapse toggled... // Navigation collapse toggled...
this.navigationService.onItemCollapsed.next(this.item); this._fuseNavigationService.onItemCollapsed.next(this.item);
this.navigationService.onItemCollapseToggled.next(); this._fuseNavigationService.onItemCollapseToggled.next();
} }
/** /**
* Expand the collapsable navigation * Expand the collapsable navigation
*/ */
expand() expand(): void
{ {
if ( this.isOpen ) if ( this.isOpen )
{ {
@ -110,13 +154,13 @@ export class FuseNavVerticalCollapseComponent implements OnInit
} }
this.isOpen = true; this.isOpen = true;
this.navigationService.onItemCollapseToggled.next(); this._fuseNavigationService.onItemCollapseToggled.next();
} }
/** /**
* Collapse the collapsable navigation * Collapse the collapsable navigation
*/ */
collapse() collapse(): void
{ {
if ( !this.isOpen ) if ( !this.isOpen )
{ {
@ -124,7 +168,7 @@ export class FuseNavVerticalCollapseComponent implements OnInit
} }
this.isOpen = false; this.isOpen = false;
this.navigationService.onItemCollapseToggled.next(); this._fuseNavigationService.onItemCollapseToggled.next();
} }
/** /**
@ -133,9 +177,9 @@ export class FuseNavVerticalCollapseComponent implements OnInit
* *
* @param parent * @param parent
* @param item * @param item
* @return {any} * @returns {boolean}
*/ */
isChildrenOf(parent, item) isChildrenOf(parent, item): boolean
{ {
if ( !parent.children ) if ( !parent.children )
{ {
@ -162,9 +206,9 @@ export class FuseNavVerticalCollapseComponent implements OnInit
* *
* @param parent * @param parent
* @param url * @param url
* @returns {any} * @returns {boolean}
*/ */
isUrlInChildren(parent, url) isUrlInChildren(parent, url): boolean
{ {
if ( !parent.children ) if ( !parent.children )
{ {

View File

@ -7,7 +7,8 @@
<div class="group-items"> <div class="group-items">
<ng-container *ngFor="let item of item.children"> <ng-container *ngFor="let item of item.children">
<fuse-nav-vertical-group *ngIf="item.type=='group'" [item]="item"></fuse-nav-vertical-group> <fuse-nav-vertical-group *ngIf="item.type=='group'" [item]="item"></fuse-nav-vertical-group>
<fuse-nav-vertical-collapse *ngIf="item.type=='collapse'" [item]="item"></fuse-nav-vertical-collapse> <fuse-nav-vertical-collapsable *ngIf="item.type=='collapsable'"
[item]="item"></fuse-nav-vertical-collapsable>
<fuse-nav-vertical-item *ngIf="item.type=='item'" [item]="item"></fuse-nav-vertical-item> <fuse-nav-vertical-item *ngIf="item.type=='item'" [item]="item"></fuse-nav-vertical-item>
</ng-container> </ng-container>
</div> </div>

View File

@ -0,0 +1,25 @@
import { Component, HostBinding, Input } from '@angular/core';
import { FuseNavigationItem } from '@fuse/types';
@Component({
selector : 'fuse-nav-vertical-group',
templateUrl: './group.component.html',
styleUrls : ['./group.component.scss']
})
export class FuseNavVerticalGroupComponent
{
@HostBinding('class')
classes = 'nav-group nav-item';
@Input()
item: FuseNavigationItem;
/**
* Constructor
*/
constructor()
{
}
}

View File

@ -0,0 +1,46 @@
<ng-container *ngIf="!item.hidden">
<!-- item.url -->
<a class="nav-link" *ngIf="item.url && !item.externalUrl && !item.function"
[routerLink]="[item.url]" [routerLinkActive]="['active', 'mat-accent-bg']"
[routerLinkActiveOptions]="{exact: item.exactMatch || false}"
[target]="item.openInNewTab ? '_blank' : '_self'" matRipple>
<ng-container *ngTemplateOutlet="itemContent"></ng-container>
</a>
<!-- item.externalUrl -->
<a class="nav-link" *ngIf="item.url && item.externalUrl && !item.function"
[href]="item.url" [target]="item.openInNewTab ? '_blank' : '_self'" matRipple>
<ng-container *ngTemplateOutlet="itemContent"></ng-container>
</a>
<!-- item.function -->
<span class="nav-link" *ngIf="!item.url && item.function"
(click)="item.function()" matRipple>
<ng-container *ngTemplateOutlet="itemContent"></ng-container>
</span>
<!-- item.url && item.function -->
<a class="nav-link" *ngIf="item.url && !item.externalUrl && item.function" (click)="item.function()"
[routerLink]="[item.url]" [routerLinkActive]="['active', 'mat-accent-bg']"
[routerLinkActiveOptions]="{exact: item.exactMatch || false}"
[target]="item.openInNewTab ? '_blank' : '_self'" matRipple>
<ng-container *ngTemplateOutlet="itemContent"></ng-container>
</a>
<!-- item.externalUrl && item.function -->
<a class="nav-link" *ngIf="item.url && item.externalUrl && item.function" (click)="item.function()"
[href]="item.url" [target]="item.openInNewTab ? '_blank' : '_self'" matRipple>
<ng-container *ngTemplateOutlet="itemContent"></ng-container>
</a>
<ng-template #itemContent>
<mat-icon class="nav-link-icon" *ngIf="item.icon">{{item.icon}}</mat-icon>
<span class="nav-link-title" [translate]="item.translate">{{item.title}}</span>
<span class="nav-link-badge" *ngIf="item.badge" [translate]="item.badge.translate"
[ngStyle]="{'background-color': item.badge.bg,'color': item.badge.fg}">
{{item.badge.title}}
</span>
</ng-template>
</ng-container>

View File

@ -0,0 +1,24 @@
import { Component, HostBinding, Input } from '@angular/core';
import { FuseNavigationItem } from '@fuse/types';
@Component({
selector : 'fuse-nav-vertical-item',
templateUrl: './item.component.html',
styleUrls : ['./item.component.scss']
})
export class FuseNavVerticalItemComponent
{
@HostBinding('class')
classes = 'nav-item';
@Input()
item: FuseNavigationItem;
/**
* Constructor
*/
constructor()
{
}
}

View File

@ -1,17 +0,0 @@
import { Component, HostBinding, Input } from '@angular/core';
@Component({
selector : 'fuse-nav-vertical-group',
templateUrl: './nav-vertical-group.component.html',
styleUrls : ['./nav-vertical-group.component.scss']
})
export class FuseNavVerticalGroupComponent
{
@HostBinding('class') classes = 'nav-group nav-item';
@Input() item: any;
constructor()
{
}
}

View File

@ -1,31 +0,0 @@
<ng-container *ngIf="!item.hidden">
<!-- item.url -->
<a class="nav-link" *ngIf="item.url && !item.function"
[routerLink]="[item.url]" routerLinkActive="active"
[routerLinkActiveOptions]="{exact: item.exactMatch || false}" matRipple>
<ng-container *ngTemplateOutlet="itemContent"></ng-container>
</a>
<!-- item.function -->
<span class="nav-link" *ngIf="!item.url && item.function" (click)="item.function()" matRipple>
<ng-container *ngTemplateOutlet="itemContent"></ng-container>
</span>
<!-- item.url && item.function -->
<a class="nav-link" *ngIf="item.url && item.function" (click)="item.function()"
[routerLink]="[item.url]" routerLinkActive="active"
[routerLinkActiveOptions]="{exact: item.exactMatch || false}" matRipple>
<ng-container *ngTemplateOutlet="itemContent"></ng-container>
</a>
<ng-template #itemContent>
<mat-icon class="nav-link-icon" *ngIf="item.icon">{{item.icon}}</mat-icon>
<span class="nav-link-title" [translate]="item.translate">{{item.title}}</span>
<span class="nav-link-badge" *ngIf="item.badge" [translate]="item.badge.translate"
[ngStyle]="{'background-color': item.badge.bg,'color': item.badge.fg}">
{{item.badge.title}}
</span>
</ng-template>
</ng-container>

View File

@ -1,16 +0,0 @@
import { Component, HostBinding, Input } from '@angular/core';
@Component({
selector : 'fuse-nav-vertical-item',
templateUrl: './nav-vertical-item.component.html',
styleUrls : ['./nav-vertical-item.component.scss']
})
export class FuseNavVerticalItemComponent
{
@HostBinding('class') classes = 'nav-item';
@Input() item: any;
constructor()
{
}
}

View File

@ -1,11 +1,11 @@
<div class="fuse-search-bar" [ngClass]="{'expanded':!collapsed}"> <div class="fuse-search-bar" [ngClass]="{'expanded':!collapsed}">
<div class="fuse-search-bar-content" [ngClass]="toolbarColor"> <div class="fuse-search-bar-content" [ngClass]="fuseConfig.layout.toolbar.background">
<label for="fuse-search-bar-input"> <label for="fuse-search-bar-input">
<button mat-icon-button class="fuse-search-bar-expander" aria-label="Expand Search Bar" (click)="expand()" <button mat-icon-button class="fuse-search-bar-expander" aria-label="Expand Search Bar" (click)="expand()"
*ngIf="collapsed"> *ngIf="collapsed">
<mat-icon class="s-24">search</mat-icon> <mat-icon class="s-24 secondary-text">search</mat-icon>
</button> </button>
<!--<span class="fuse-search-bar-loader" fxLayout="row" fxLayoutAlign="center center" *ngIf="!collapsed"> <!--<span class="fuse-search-bar-loader" fxLayout="row" fxLayoutAlign="center center" *ngIf="!collapsed">
<mat-progress-spinner color="mat-accent" mode="indeterminate"></mat-progress-spinner> <mat-progress-spinner color="mat-accent" mode="indeterminate"></mat-progress-spinner>
@ -17,7 +17,7 @@
<button mat-icon-button class="fuse-search-bar-collapser" (click)="collapse()" <button mat-icon-button class="fuse-search-bar-collapser" (click)="collapse()"
aria-label="Collapse Search Bar"> aria-label="Collapse Search Bar">
<mat-icon class="s-24">close</mat-icon> <mat-icon class="s-24 secondary-text">close</mat-icon>
</button> </button>
</div> </div>

View File

@ -1,5 +1,6 @@
import { Component, EventEmitter, Output } from '@angular/core'; import { Component, EventEmitter, OnDestroy, OnInit, Output } from '@angular/core';
import { Subscription } from 'rxjs'; import { Subject } from 'rxjs';
import { takeUntil } from 'rxjs/operators';
import { FuseConfigService } from '@fuse/services/config.service'; import { FuseConfigService } from '@fuse/services/config.service';
@ -8,42 +9,91 @@ import { FuseConfigService } from '@fuse/services/config.service';
templateUrl: './search-bar.component.html', templateUrl: './search-bar.component.html',
styleUrls : ['./search-bar.component.scss'] styleUrls : ['./search-bar.component.scss']
}) })
export class FuseSearchBarComponent export class FuseSearchBarComponent implements OnInit, OnDestroy
{ {
collapsed: boolean; collapsed: boolean;
toolbarColor: string; fuseConfig: any;
@Output() onInput: EventEmitter<any> = new EventEmitter();
onConfigChanged: Subscription;
@Output()
input: EventEmitter<any>;
// Private
private _unsubscribeAll: Subject<any>;
/**
* Constructor
*
* @param {FuseConfigService} _fuseConfigService
*/
constructor( constructor(
private fuseConfig: FuseConfigService private _fuseConfigService: FuseConfigService
) )
{ {
// Set the defaults
this.input = new EventEmitter();
this.collapsed = true; this.collapsed = true;
this.onConfigChanged =
this.fuseConfig.onConfigChanged // Set the private defaults
.subscribe( this._unsubscribeAll = new Subject();
(newSettings) => {
this.toolbarColor = newSettings.colorClasses.toolbar;
}
);
} }
collapse() // -----------------------------------------------------------------------------------------------------
// @ Lifecycle hooks
// -----------------------------------------------------------------------------------------------------
/**
* On init
*/
ngOnInit(): void
{
// Subscribe to config changes
this._fuseConfigService.config
.pipe(takeUntil(this._unsubscribeAll))
.subscribe(
(config) => {
this.fuseConfig = config;
}
);
}
/**
* On destroy
*/
ngOnDestroy(): void
{
// Unsubscribe from all subscriptions
this._unsubscribeAll.next();
this._unsubscribeAll.complete();
}
// -----------------------------------------------------------------------------------------------------
// @ Public methods
// -----------------------------------------------------------------------------------------------------
/**
* Collapse
*/
collapse(): void
{ {
this.collapsed = true; this.collapsed = true;
} }
expand() /**
* Expand
*/
expand(): void
{ {
this.collapsed = false; this.collapsed = false;
} }
search(event) /**
* Search
*
* @param event
*/
search(event): void
{ {
const value = event.target.value; this.input.emit(event.target.value);
this.onInput.emit(value);
} }
} }

View File

@ -7,7 +7,7 @@
</button> </button>
</div> </div>
<div class="shortcuts" fxHide fxShow.gt-sm [ngClass]="toolbarColor"> <div class="shortcuts" fxHide fxShow.gt-sm>
<div fxLayout="row" fxLayoutAlign="space-between center" fxFlex="0 1 auto"> <div fxLayout="row" fxLayoutAlign="space-between center" fxFlex="0 1 auto">

View File

@ -1,12 +1,11 @@
import { Component, ElementRef, Input, OnDestroy, OnInit, Renderer2, ViewChild } from '@angular/core'; import { Component, ElementRef, Input, OnDestroy, OnInit, Renderer2, ViewChild } from '@angular/core';
import { ObservableMedia } from '@angular/flex-layout'; import { ObservableMedia } from '@angular/flex-layout';
import { Subscription } from 'rxjs';
import { CookieService } from 'ngx-cookie-service'; import { CookieService } from 'ngx-cookie-service';
import { Subject } from 'rxjs';
import { takeUntil } from 'rxjs/operators';
import { FuseMatchMediaService } from '@fuse/services/match-media.service'; import { FuseMatchMediaService } from '@fuse/services/match-media.service';
import { FuseNavigationService } from '@fuse/components/navigation/navigation.service'; import { FuseNavigationService } from '@fuse/components/navigation/navigation.service';
import { FuseConfigService } from '@fuse/services/config.service';
@Component({ @Component({
selector : 'fuse-shortcuts', selector : 'fuse-shortcuts',
@ -15,48 +14,67 @@ import { FuseConfigService } from '@fuse/services/config.service';
}) })
export class FuseShortcutsComponent implements OnInit, OnDestroy export class FuseShortcutsComponent implements OnInit, OnDestroy
{ {
shortcutItems: any[] = []; shortcutItems: any[];
navigationItems: any[]; navigationItems: any[];
filteredNavigationItems: any[]; filteredNavigationItems: any[];
searching = false; searching: boolean;
mobileShortcutsPanelActive = false; mobileShortcutsPanelActive: boolean;
toolbarColor: string;
matchMediaSubscription: Subscription;
onConfigChanged: Subscription;
@Input() navigation: any; @Input()
navigation: any;
@ViewChild('searchInput') searchInputField; @ViewChild('searchInput')
@ViewChild('shortcuts') shortcutsEl: ElementRef; searchInputField;
@ViewChild('shortcuts')
shortcutsEl: ElementRef;
// Private
private _unsubscribeAll: Subject<any>;
/**
* Constructor
*
* @param {Renderer2} _renderer
* @param {CookieService} _cookieService
* @param {FuseMatchMediaService} _fuseMatchMediaService
* @param {FuseNavigationService} _fuseNavigationService
* @param {ObservableMedia} _observableMedia
*/
constructor( constructor(
private renderer: Renderer2, private _cookieService: CookieService,
private observableMedia: ObservableMedia, private _fuseMatchMediaService: FuseMatchMediaService,
private fuseMatchMedia: FuseMatchMediaService, private _fuseNavigationService: FuseNavigationService,
private fuseNavigationService: FuseNavigationService, private _observableMedia: ObservableMedia,
private fuseConfig: FuseConfigService, private _renderer: Renderer2
private cookieService: CookieService
) )
{ {
this.onConfigChanged = // Set the defaults
this.fuseConfig.onConfigChanged this.shortcutItems = [];
.subscribe( this.searching = false;
(newSettings) => { this.mobileShortcutsPanelActive = false;
this.toolbarColor = newSettings.colorClasses.toolbar;
} // Set the private defaults
); this._unsubscribeAll = new Subject();
} }
ngOnInit() // -----------------------------------------------------------------------------------------------------
// @ Lifecycle hooks
// -----------------------------------------------------------------------------------------------------
/**
* On init
*/
ngOnInit(): void
{ {
// Get the navigation items and flatten them // Get the navigation items and flatten them
this.filteredNavigationItems = this.navigationItems = this.fuseNavigationService.getFlatNavigation(this.navigation); this.filteredNavigationItems = this.navigationItems = this._fuseNavigationService.getFlatNavigation(this.navigation);
const cookieExists = this.cookieService.check('FUSE2.shortcuts'); const cookieExists = this._cookieService.check('FUSE2.shortcuts');
if ( cookieExists ) if ( cookieExists )
{ {
this.shortcutItems = JSON.parse(this.cookieService.get('FUSE2.shortcuts')); this.shortcutItems = JSON.parse(this._cookieService.get('FUSE2.shortcuts'));
} }
else else
{ {
@ -64,46 +82,61 @@ export class FuseShortcutsComponent implements OnInit, OnDestroy
this.shortcutItems = [ this.shortcutItems = [
{ {
'title': 'Calendar', 'title': 'Calendar',
'type' : 'nav-item', 'type' : 'item',
'icon' : 'today', 'icon' : 'today',
'url' : '/apps/calendar' 'url' : '/apps/calendar'
}, },
{ {
'title': 'Mail', 'title': 'Mail',
'type' : 'nav-item', 'type' : 'item',
'icon' : 'email', 'icon' : 'email',
'url' : '/apps/mail' 'url' : '/apps/mail'
}, },
{ {
'title': 'Contacts', 'title': 'Contacts',
'type' : 'nav-item', 'type' : 'item',
'icon' : 'account_box', 'icon' : 'account_box',
'url' : '/apps/contacts' 'url' : '/apps/contacts'
}, },
{ {
'title': 'To-Do', 'title': 'To-Do',
'type' : 'nav-item', 'type' : 'item',
'icon' : 'check_box', 'icon' : 'check_box',
'url' : '/apps/todo' 'url' : '/apps/todo'
} }
]; ];
} }
this.matchMediaSubscription = this._fuseMatchMediaService.onMediaChange
this.fuseMatchMedia.onMediaChange.subscribe(() => { .pipe(takeUntil(this._unsubscribeAll))
if ( this.observableMedia.isActive('gt-sm') ) .subscribe(() => {
if ( this._observableMedia.isActive('gt-sm') )
{ {
this.hideMobileShortcutsPanel(); this.hideMobileShortcutsPanel();
} }
}); });
} }
ngOnDestroy() /**
* On destroy
*/
ngOnDestroy(): void
{ {
this.matchMediaSubscription.unsubscribe(); // Unsubscribe from all subscriptions
this._unsubscribeAll.next();
this._unsubscribeAll.complete();
} }
search(event) // -----------------------------------------------------------------------------------------------------
// @ Public methods
// -----------------------------------------------------------------------------------------------------
/**
* Search
*
* @param event
*/
search(event): void
{ {
const value = event.target.value.toLowerCase(); const value = event.target.value.toLowerCase();
@ -122,7 +155,13 @@ export class FuseShortcutsComponent implements OnInit, OnDestroy
}); });
} }
toggleShortcut(event, itemToToggle) /**
* Toggle shortcut
*
* @param event
* @param itemToToggle
*/
toggleShortcut(event, itemToToggle): void
{ {
event.stopPropagation(); event.stopPropagation();
@ -133,7 +172,7 @@ export class FuseShortcutsComponent implements OnInit, OnDestroy
this.shortcutItems.splice(i, 1); this.shortcutItems.splice(i, 1);
// Save to the cookies // Save to the cookies
this.cookieService.set('FUSE2.shortcuts', JSON.stringify(this.shortcutItems)); this._cookieService.set('FUSE2.shortcuts', JSON.stringify(this.shortcutItems));
return; return;
} }
@ -142,32 +181,47 @@ export class FuseShortcutsComponent implements OnInit, OnDestroy
this.shortcutItems.push(itemToToggle); this.shortcutItems.push(itemToToggle);
// Save to the cookies // Save to the cookies
this.cookieService.set('FUSE2.shortcuts', JSON.stringify(this.shortcutItems)); this._cookieService.set('FUSE2.shortcuts', JSON.stringify(this.shortcutItems));
} }
isInShortcuts(navigationItem) /**
* Is in shortcuts?
*
* @param navigationItem
* @returns {any}
*/
isInShortcuts(navigationItem): any
{ {
return this.shortcutItems.find(item => { return this.shortcutItems.find(item => {
return item.url === navigationItem.url; return item.url === navigationItem.url;
}); });
} }
onMenuOpen() /**
* On menu open
*/
onMenuOpen(): void
{ {
setTimeout(() => { setTimeout(() => {
this.searchInputField.nativeElement.focus(); this.searchInputField.nativeElement.focus();
}); });
} }
showMobileShortcutsPanel() /**
* Show mobile shortcuts
*/
showMobileShortcutsPanel(): void
{ {
this.mobileShortcutsPanelActive = true; this.mobileShortcutsPanelActive = true;
this.renderer.addClass(this.shortcutsEl.nativeElement, 'show-mobile-panel'); this._renderer.addClass(this.shortcutsEl.nativeElement, 'show-mobile-panel');
} }
hideMobileShortcutsPanel() /**
* Hide mobile shortcuts
*/
hideMobileShortcutsPanel(): void
{ {
this.mobileShortcutsPanelActive = false; this.mobileShortcutsPanelActive = false;
this.renderer.removeClass(this.shortcutsEl.nativeElement, 'show-mobile-panel'); this._renderer.removeClass(this.shortcutsEl.nativeElement, 'show-mobile-panel');
} }
} }

View File

@ -11,17 +11,15 @@ fuse-sidebar {
min-width: 280px; min-width: 280px;
max-width: 280px; max-width: 280px;
z-index: 1000; z-index: 1000;
transition-property: transform, width, min-width, max-width;
transition-duration: 150ms;
transition-timing-function: ease-in-out;
box-shadow: 0 2px 8px 0 rgba(0, 0, 0, 0.35); box-shadow: 0 2px 8px 0 rgba(0, 0, 0, 0.35);
background: white;
&.left-aligned { &.left-positioned {
left: 0; left: 0;
transform: translateX(-100%); transform: translateX(-100%);
} }
&.right-aligned { &.right-positioned {
right: 0; right: 0;
transform: translateX(100%); transform: translateX(100%);
} }
@ -46,6 +44,12 @@ fuse-sidebar {
max-width: 64px; max-width: 64px;
} }
} }
&.animations-enabled {
transition-property: transform, width, min-width, max-width;
transition-duration: 150ms;
transition-timing-function: ease-in-out;
}
} }
.fuse-sidebar-overlay { .fuse-sidebar-overlay {
@ -54,7 +58,11 @@ fuse-sidebar {
bottom: 0; bottom: 0;
left: 0; left: 0;
right: 0; right: 0;
z-index: 3; z-index: 999;
background-color: rgba(0, 0, 0, 0.6); background-color: rgba(0, 0, 0, 0.6);
opacity: 0; opacity: 0;
&.fuse-sidebar-overlay-invisible {
background-color: transparent;
}
} }

View File

@ -1,11 +1,12 @@
import { Component, ElementRef, HostBinding, HostListener, Input, OnDestroy, OnInit, Renderer2, ViewEncapsulation } from '@angular/core'; import { ChangeDetectorRef, Component, ElementRef, HostBinding, HostListener, Input, OnDestroy, OnInit, Renderer2, RendererStyleFlags2, ViewEncapsulation } from '@angular/core';
import { animate, AnimationBuilder, AnimationPlayer, style } from '@angular/animations'; import { animate, AnimationBuilder, AnimationPlayer, style } from '@angular/animations';
import { ObservableMedia } from '@angular/flex-layout'; import { ObservableMedia } from '@angular/flex-layout';
import { Subscription } from 'rxjs'; import { Subject } from 'rxjs';
import { FuseSidebarService } from './sidebar.service'; import { FuseSidebarService } from './sidebar.service';
import { FuseMatchMediaService } from '@fuse/services/match-media.service'; import { FuseMatchMediaService } from '@fuse/services/match-media.service';
import { FuseConfigService } from '@fuse/services/config.service'; import { FuseConfigService } from '@fuse/services/config.service';
import { takeUntil } from 'rxjs/internal/operators';
@Component({ @Component({
selector : 'fuse-sidebar', selector : 'fuse-sidebar',
@ -19,9 +20,9 @@ export class FuseSidebarComponent implements OnInit, OnDestroy
@Input() @Input()
name: string; name: string;
// Align // Position
@Input() @Input()
align: 'left' | 'right'; position: 'left' | 'right';
// Open // Open
@HostBinding('class.open') @HostBinding('class.open')
@ -35,6 +36,63 @@ export class FuseSidebarComponent implements OnInit, OnDestroy
@HostBinding('class.locked-open') @HostBinding('class.locked-open')
isLockedOpen: boolean; isLockedOpen: boolean;
// Folded unfolded
@HostBinding('class.unfolded')
unfolded: boolean;
// Invisible overlay
@Input()
invisibleOverlay: boolean;
// Private
private _folded: boolean;
private _fuseConfig: any;
private _wasActive: boolean;
private _backdrop: HTMLElement | null = null;
private _player: AnimationPlayer;
private _unsubscribeAll: Subject<any>;
@HostBinding('class.animations-enabled')
private _animationsEnabled: boolean;
/**
* Constructor
*
* @param {AnimationBuilder} _animationBuilder
* @param {ChangeDetectorRef} _changeDetectorRef
* @param {ElementRef} _elementRef
* @param {FuseConfigService} _fuseConfigService
* @param {FuseMatchMediaService} _fuseMatchMediaService
* @param {FuseSidebarService} _fuseSidebarService
* @param {ObservableMedia} _observableMedia
* @param {Renderer2} _renderer
*/
constructor(
private _animationBuilder: AnimationBuilder,
private _changeDetectorRef: ChangeDetectorRef,
private _elementRef: ElementRef,
private _fuseConfigService: FuseConfigService,
private _fuseMatchMediaService: FuseMatchMediaService,
private _fuseSidebarService: FuseSidebarService,
private _observableMedia: ObservableMedia,
private _renderer: Renderer2
)
{
// Set the defaults
this.folded = false;
this.opened = false;
this.position = 'left';
this.invisibleOverlay = false;
// Set the private defaults
this._animationsEnabled = false;
this._unsubscribeAll = new Subject();
}
// -----------------------------------------------------------------------------------------------------
// @ Accessors
// -----------------------------------------------------------------------------------------------------
// Folded // Folded
@HostBinding('class.folded') @HostBinding('class.folded')
@Input() @Input()
@ -50,22 +108,22 @@ export class FuseSidebarComponent implements OnInit, OnDestroy
this._folded = value; this._folded = value;
// Programmatically add/remove margin to the element // Programmatically add/remove margin to the element
// that comes after or before based on the alignment // that comes after or before based on the position
let sibling, let sibling,
styleRule; styleRule;
const styleValue = '64px'; const styleValue = '64px';
// Get the sibling and set the style rule // Get the sibling and set the style rule
if ( this.align === 'left' ) if ( this.position === 'left' )
{ {
sibling = this.elementRef.nativeElement.nextElementSibling; sibling = this._elementRef.nativeElement.nextElementSibling;
styleRule = 'marginLeft'; styleRule = 'margin-left';
} }
else else
{ {
sibling = this.elementRef.nativeElement.previousElementSibling; sibling = this._elementRef.nativeElement.previousElementSibling;
styleRule = 'marginRight'; styleRule = 'margin-right';
} }
// If there is no sibling, return... // If there is no sibling, return...
@ -78,13 +136,13 @@ export class FuseSidebarComponent implements OnInit, OnDestroy
if ( value ) if ( value )
{ {
// Set the style // Set the style
this.renderer.setStyle(sibling, styleRule, styleValue); this._renderer.setStyle(sibling, styleRule, styleValue, RendererStyleFlags2.Important + RendererStyleFlags2.DashCase);
} }
// If unfolded... // If unfolded...
else else
{ {
// Remove the style // Remove the style
this.renderer.removeStyle(sibling, styleRule); this._renderer.removeStyle(sibling, styleRule);
} }
} }
@ -93,54 +151,30 @@ export class FuseSidebarComponent implements OnInit, OnDestroy
return this._folded; return this._folded;
} }
// Folded unfolded // -----------------------------------------------------------------------------------------------------
@HostBinding('class.unfolded') // @ Lifecycle hooks
unfolded: boolean; // -----------------------------------------------------------------------------------------------------
// Private
private _folded: boolean;
private _wasActive: boolean;
private _backdrop: HTMLElement | null = null;
private _player: AnimationPlayer;
private _onMediaChangeSubscription: Subscription;
/**
* Constructor
*
* @param {Renderer2} renderer
* @param {ElementRef} elementRef
* @param {AnimationBuilder} animationBuilder
* @param {ObservableMedia} observableMedia
* @param {FuseConfigService} fuseConfigService
* @param {FuseSidebarService} fuseSidebarService
* @param {FuseMatchMediaService} fuseMatchMediaService
*/
constructor(
private renderer: Renderer2,
private elementRef: ElementRef,
private animationBuilder: AnimationBuilder,
private observableMedia: ObservableMedia,
private fuseConfigService: FuseConfigService,
private fuseSidebarService: FuseSidebarService,
private fuseMatchMediaService: FuseMatchMediaService
)
{
// Set the defaults
this.opened = false;
this.folded = false;
this.align = 'left';
}
/** /**
* On init * On init
*/ */
ngOnInit(): void ngOnInit(): void
{ {
// Register the sidebar // Subscribe to config changes
this.fuseSidebarService.register(this.name, this); this._fuseConfigService.config
.pipe(takeUntil(this._unsubscribeAll))
.subscribe((config) => {
this._fuseConfig = config;
});
// Setup alignment // Register the sidebar
this._setupAlignment(); this._fuseSidebarService.register(this.name, this);
// Setup visibility
this._setupVisibility();
// Setup position
this._setupPosition();
// Setup lockedOpen // Setup lockedOpen
this._setupLockedOpen(); this._setupLockedOpen();
@ -158,28 +192,47 @@ export class FuseSidebarComponent implements OnInit, OnDestroy
} }
// Unregister the sidebar // Unregister the sidebar
this.fuseSidebarService.unregister(this.name); this._fuseSidebarService.unregister(this.name);
// Unsubscribe from the media watcher subscription // Unsubscribe from all subscriptions
this._onMediaChangeSubscription.unsubscribe(); this._unsubscribeAll.next();
this._unsubscribeAll.complete();
} }
// -----------------------------------------------------------------------------------------------------
// @ Private methods
// -----------------------------------------------------------------------------------------------------
/** /**
* Set the sidebar alignment * Setup the visibility of the sidebar
* *
* @private * @private
*/ */
private _setupAlignment(): void private _setupVisibility(): void
{
// Remove the existing box-shadow
this._renderer.setStyle(this._elementRef.nativeElement, 'box-shadow', 'none');
// Make the sidebar invisible
this._renderer.setStyle(this._elementRef.nativeElement, 'visibility', 'hidden');
}
/**
* Setup the sidebar position
*
* @private
*/
private _setupPosition(): void
{ {
// Add the correct class name to the sidebar // Add the correct class name to the sidebar
// element depending on the align attribute // element depending on the position attribute
if ( this.align === 'right' ) if ( this.position === 'right' )
{ {
this.renderer.addClass(this.elementRef.nativeElement, 'right-aligned'); this._renderer.addClass(this._elementRef.nativeElement, 'right-positioned');
} }
else else
{ {
this.renderer.addClass(this.elementRef.nativeElement, 'left-aligned'); this._renderer.addClass(this._elementRef.nativeElement, 'left-positioned');
} }
} }
@ -193,19 +246,23 @@ export class FuseSidebarComponent implements OnInit, OnDestroy
// Return if the lockedOpen wasn't set // Return if the lockedOpen wasn't set
if ( !this.lockedOpen ) if ( !this.lockedOpen )
{ {
// Return
return; return;
} }
// Set the wasActive for the first time // Set the wasActive for the first time
this._wasActive = false; this._wasActive = false;
// Act on every media change // Show the sidebar
this._onMediaChangeSubscription = this._showSidebar();
this.fuseMatchMediaService.onMediaChange.subscribe(() => { // Act on every media change
this._fuseMatchMediaService.onMediaChange
.pipe(takeUntil(this._unsubscribeAll))
.subscribe(() => {
// Get the active status // Get the active status
const isActive = this.observableMedia.isActive(this.lockedOpen); const isActive = this._observableMedia.isActive(this.lockedOpen);
// If the both status are the same, don't act // If the both status are the same, don't act
if ( this._wasActive === isActive ) if ( this._wasActive === isActive )
@ -219,18 +276,21 @@ export class FuseSidebarComponent implements OnInit, OnDestroy
// Set the lockedOpen status // Set the lockedOpen status
this.isLockedOpen = true; this.isLockedOpen = true;
// Show the sidebar
this._showSidebar();
// Force the the opened status to true // Force the the opened status to true
this.opened = true; this.opened = true;
// Read the folded setting from the config // Read the folded setting from the config
// and fold the sidebar if it's true // and fold the sidebar if it's true
if ( this.fuseConfigService.config.layout.navigationFolded ) if ( this._fuseConfig.layout.navbar.folded )
{ {
this.fold(); this.fold();
} }
// Hide the backdrop if any exists // Hide the backdrop if any exists
this.hideBackdrop(); this._hideBackdrop();
} }
// De-Activate the lockedOpen // De-Activate the lockedOpen
else else
@ -243,6 +303,9 @@ export class FuseSidebarComponent implements OnInit, OnDestroy
// Force the the opened status to close // Force the the opened status to close
this.opened = false; this.opened = false;
// Hide the sidebar
this._hideSidebar();
} }
// Store the new active status // Store the new active status
@ -250,6 +313,152 @@ export class FuseSidebarComponent implements OnInit, OnDestroy
}); });
} }
/**
* Show the backdrop
*
* @private
*/
private _showBackdrop(): void
{
// Create the backdrop element
this._backdrop = this._renderer.createElement('div');
// Add a class to the backdrop element
this._backdrop.classList.add('fuse-sidebar-overlay');
// Add a class depending on the invisibleOverlay option
if ( this.invisibleOverlay )
{
this._backdrop.classList.add('fuse-sidebar-overlay-invisible');
}
// Append the backdrop to the parent of the sidebar
this._renderer.appendChild(this._elementRef.nativeElement.parentElement, this._backdrop);
// Create the enter animation and attach it to the player
this._player =
this._animationBuilder
.build([
animate('300ms ease', style({opacity: 1}))
]).create(this._backdrop);
// Play the animation
this._player.play();
// Add an event listener to the overlay
this._backdrop.addEventListener('click', () => {
this.close();
}
);
// Mark for check
this._changeDetectorRef.markForCheck();
}
/**
* Hide the backdrop
*
* @private
*/
private _hideBackdrop(): void
{
if ( !this._backdrop )
{
return;
}
// Create the leave animation and attach it to the player
this._player =
this._animationBuilder
.build([
animate('300ms ease', style({opacity: 0}))
]).create(this._backdrop);
// Play the animation
this._player.play();
// Once the animation is done...
this._player.onDone(() => {
// If the backdrop still exists...
if ( this._backdrop )
{
// Remove the backdrop
this._backdrop.parentNode.removeChild(this._backdrop);
this._backdrop = null;
}
});
// Mark for check
this._changeDetectorRef.markForCheck();
}
/**
* Change some properties of the sidebar
* and make it visible
*
* @private
*/
private _showSidebar(): void
{
// Remove the box-shadow style
this._renderer.removeStyle(this._elementRef.nativeElement, 'box-shadow');
// Make the sidebar invisible
this._renderer.removeStyle(this._elementRef.nativeElement, 'visibility');
// Mark for check
this._changeDetectorRef.markForCheck();
}
/**
* Change some properties of the sidebar
* and make it invisible
*
* @private
*/
private _hideSidebar(delay = true): void
{
const delayAmount = delay ? 300 : 0;
// Add a delay so close animation can play
setTimeout(() => {
// Remove the box-shadow
this._renderer.setStyle(this._elementRef.nativeElement, 'box-shadow', 'none');
// Make the sidebar invisible
this._renderer.setStyle(this._elementRef.nativeElement, 'visibility', 'hidden');
}, delayAmount);
// Mark for check
this._changeDetectorRef.markForCheck();
}
/**
* Enable the animations
*
* @private
*/
private _enableAnimations(): void
{
// Return if animations already enabled
if ( this._animationsEnabled )
{
return;
}
// Enable the animations
this._animationsEnabled = true;
// Mark for check
this._changeDetectorRef.markForCheck();
}
// -----------------------------------------------------------------------------------------------------
// @ Public methods
// -----------------------------------------------------------------------------------------------------
/** /**
* Open the sidebar * Open the sidebar
*/ */
@ -260,11 +469,20 @@ export class FuseSidebarComponent implements OnInit, OnDestroy
return; return;
} }
// Enable the animations
this._enableAnimations();
// Show the sidebar
this._showSidebar();
// Show the backdrop // Show the backdrop
this.showBackdrop(); this._showBackdrop();
// Set the opened status // Set the opened status
this.opened = true; this.opened = true;
// Mark for check
this._changeDetectorRef.markForCheck();
} }
/** /**
@ -277,11 +495,20 @@ export class FuseSidebarComponent implements OnInit, OnDestroy
return; return;
} }
// Enable the animations
this._enableAnimations();
// Hide the backdrop // Hide the backdrop
this.hideBackdrop(); this._hideBackdrop();
// Set the opened status // Set the opened status
this.opened = false; this.opened = false;
// Hide the sidebar
this._hideSidebar();
// Mark for check
this._changeDetectorRef.markForCheck();
} }
/** /**
@ -311,8 +538,14 @@ export class FuseSidebarComponent implements OnInit, OnDestroy
return; return;
} }
// Enable the animations
this._enableAnimations();
// Unfold the sidebar temporarily // Unfold the sidebar temporarily
this.unfolded = true; this.unfolded = true;
// Mark for check
this._changeDetectorRef.markForCheck();
} }
/** /**
@ -327,8 +560,14 @@ export class FuseSidebarComponent implements OnInit, OnDestroy
return; return;
} }
// Enable the animations
this._enableAnimations();
// Fold the sidebar back // Fold the sidebar back
this.unfolded = false; this.unfolded = false;
// Mark for check
this._changeDetectorRef.markForCheck();
} }
/** /**
@ -342,8 +581,14 @@ export class FuseSidebarComponent implements OnInit, OnDestroy
return; return;
} }
// Enable the animations
this._enableAnimations();
// Fold // Fold
this.folded = true; this.folded = true;
// Mark for check
this._changeDetectorRef.markForCheck();
} }
/** /**
@ -357,8 +602,14 @@ export class FuseSidebarComponent implements OnInit, OnDestroy
return; return;
} }
// Enable the animations
this._enableAnimations();
// Unfold // Unfold
this.folded = false; this.folded = false;
// Mark for check
this._changeDetectorRef.markForCheck();
} }
/** /**
@ -375,68 +626,4 @@ export class FuseSidebarComponent implements OnInit, OnDestroy
this.fold(); this.fold();
} }
} }
/**
* Show the backdrop
*/
showBackdrop(): void
{
// Create the backdrop element
this._backdrop = this.renderer.createElement('div');
// Add a class to the backdrop element
this._backdrop.classList.add('fuse-sidebar-overlay');
// Append the backdrop to the parent of the sidebar
this.renderer.appendChild(this.elementRef.nativeElement.parentElement, this._backdrop);
// Create the enter animation and attach it to the player
this._player =
this.animationBuilder
.build([
animate('300ms ease', style({opacity: 1}))
]).create(this._backdrop);
// Play the animation
this._player.play();
// Add an event listener to the overlay
this._backdrop.addEventListener('click', () => {
this.close();
}
);
}
/**
* Hide the backdrop
*/
hideBackdrop(): void
{
if ( !this._backdrop )
{
return;
}
// Create the leave animation and attach it to the player
this._player =
this.animationBuilder
.build([
animate('300ms ease', style({opacity: 0}))
]).create(this._backdrop);
// Play the animation
this._player.play();
// Once the animation is done...
this._player.onDone(() => {
// If the backdrop still exists...
if ( this._backdrop )
{
// Remove the backdrop
this._backdrop.parentNode.removeChild(this._backdrop);
this._backdrop = null;
}
});
}
} }

View File

@ -1,105 +1,372 @@
<button #openButton mat-icon-button class="open-button mat-primary-bg mat-elevation-z2" (click)="openBar()"> <div class="theme-options-panel" fusePerfectScrollbar>
<mat-icon>settings</mat-icon>
</button>
<div class="theme-options-panel-overlay" #overlay [fxHide]="barClosed" [@fadeInOut]="!barClosed"></div> <div class="header">
<div #panel class="theme-options-panel mat-white-bg mat-elevation-z8"> <span class="title">Theme Options</span>
<button mat-icon-button class="close-button" (click)="closeBar()"> <button mat-icon-button class="close-button" (click)="toggleSidebarOpen('themeOptionsPanel')">
<mat-icon>close</mat-icon> <mat-icon>close</mat-icon>
</button> </button>
<div class="theme-options-panel-inner" fxLayout="column" fxLayoutAlign="start start"> </div>
<h3>Navigation:</h3> <form [formGroup]="form">
<mat-radio-group [(ngModel)]="config.layout.navigation" (ngModelChange)="onSettingsChange()"
fxLayout="column" fxLayout.gt-xs="row wrap" fxLayoutAlign="start start">
<mat-radio-button class="mr-8 mb-8" value="top">Top</mat-radio-button>
<mat-radio-button class="mr-8 mb-8" value="left">Left</mat-radio-button>
<mat-radio-button class="mr-8 mb-8" value="right">Right</mat-radio-button>
<mat-radio-button class="mr-8 mb-8" value="none">None</mat-radio-button>
</mat-radio-group>
<h3>Navigation Fold (for vertical navigation):</h3> <!-- LAYOUT STYLES -->
<mat-slide-toggle [(ngModel)]="config.layout.navigationFolded" <div class="group" formGroupName="layout">
(change)="onSettingsChange()">
Folded
</mat-slide-toggle>
<h3 class="mt-24">Toolbar:</h3> <h2>Layout Styles</h2>
<mat-radio-group [(ngModel)]="config.layout.toolbar" (ngModelChange)="onSettingsChange()"
fxLayout="column" fxLayout.gt-xs="row wrap" fxLayoutAlign="start start">
<mat-radio-button class="mr-8 mb-8" value="below">Below</mat-radio-button>
<mat-radio-button class="mr-8 mb-8" value="above">Above</mat-radio-button>
<mat-radio-button class="mr-8 mb-8" value="none">None</mat-radio-button>
</mat-radio-group>
<h3 class="mt-24">Footer:</h3> <mat-radio-group fxLayout="column" fxLayoutAlign="start start" formControlName="style">
<mat-radio-group [(ngModel)]="config.layout.footer" (ngModelChange)="onSettingsChange()"
fxLayout="column" fxLayout.gt-xs="row wrap" fxLayoutAlign="start start">
<mat-radio-button class="mr-8 mb-8" value="below">Below</mat-radio-button>
<mat-radio-button class="mr-8 mb-8" value="above">Above</mat-radio-button>
<mat-radio-button class="mr-8 mb-8" value="none">None</mat-radio-button>
</mat-radio-group>
<h3 class="mt-24">Layout Mode:</h3> <mat-radio-button class="mb-12" value="vertical-layout-1">
<mat-radio-group [(ngModel)]="config.layout.mode" (ngModelChange)="onSettingsChange()" Vertical Layout #1
fxLayout="column" fxLayout.gt-xs="row wrap" fxLayoutAlign="start start"> </mat-radio-button>
<mat-radio-button class="mr-8 mb-8" value="boxed">Boxed</mat-radio-button>
<mat-radio-button class="mr-8 mb-8" value="fullwidth">Fullwidth</mat-radio-button>
</mat-radio-group>
<mat-divider></mat-divider> <mat-radio-button class="mb-12" value="vertical-layout-2">
Vertical Layout #2
</mat-radio-button>
<h3>Colors:</h3> <mat-radio-button class="mb-12" value="vertical-layout-3">
<div class="colors"> Vertical Layout #3
</mat-radio-button>
<div fxFlex fxLayout="row" fxLayoutAlign="space-between center"> <mat-radio-button class="mb-12" value="horizontal-layout-1">
<h4 class="mr-8">Toolbar Color</h4> Horizontal Layout #1
<fuse-material-color-picker [(selectedClass)]="config.colorClasses.toolbar" </mat-radio-button>
(onValueChange)="onSettingsChange()"></fuse-material-color-picker>
</div>
<div fxFlex fxLayout="row" fxLayoutAlign="space-between center"> </mat-radio-group>
<h4 class="mr-8">Navigation Bar Color</h4>
<fuse-material-color-picker [(selectedClass)]="config.colorClasses.navbar" <!-- DIFFERENT FORMS BASED ON LAYOUT STYLES -->
(onValueChange)="onSettingsChange()"></fuse-material-color-picker> <ng-container [ngSwitch]="fuseConfig.layout.style">
</div>
<!-- VERTICAL LAYOUT #1 -->
<ng-container *ngSwitchCase="'vertical-layout-1'">
<!-- LAYOUT WIDTH -->
<div class="group mt-32">
<h2>Layout Width</h2>
<mat-radio-group fxLayout="column" fxLayoutAlign="start start" formControlName="width">
<mat-radio-button class="mb-12" value="fullwidth">Fullwidth</mat-radio-button>
<mat-radio-button class="mb-12" value="boxed">Boxed</mat-radio-button>
</mat-radio-group>
</div>
<!-- NAVBAR -->
<div class="group" formGroupName="navbar">
<h2>Navbar</h2>
<mat-slide-toggle formControlName="hidden">
Hide
</mat-slide-toggle>
<mat-slide-toggle class="mt-24" formControlName="folded">
Folded
</mat-slide-toggle>
<h3 class="mt-24">Position:</h3>
<mat-radio-group fxLayout="column" fxLayoutAlign="start start" formControlName="position">
<mat-radio-button class="mb-16" value="left">Left</mat-radio-button>
<mat-radio-button class="mb-16" value="right">Right</mat-radio-button>
</mat-radio-group>
</div>
<!-- TOOLBAR -->
<div class="group" formGroupName="toolbar">
<h2>Toolbar</h2>
<mat-slide-toggle formControlName="hidden">
Hide
</mat-slide-toggle>
<h3 class="mt-24">Position:</h3>
<mat-radio-group fxLayout="column" fxLayoutAlign="start start" formControlName="position">
<mat-radio-button class="mb-12" value="above">Above</mat-radio-button>
<mat-radio-button class="mb-12" value="below-static">Below Static</mat-radio-button>
<mat-radio-button class="mb-12" value="below-fixed">Below Fixed</mat-radio-button>
</mat-radio-group>
</div>
<!-- FOOTER -->
<div class="group" formGroupName="footer">
<h2>Footer</h2>
<mat-slide-toggle formControlName="hidden">
Hide
</mat-slide-toggle>
<h3 class="mt-24">Position:</h3>
<mat-radio-group fxLayout="column" fxLayoutAlign="start start" formControlName="position">
<mat-radio-button class="mb-12" value="above">Above</mat-radio-button>
<mat-radio-button class="mb-12" value="below-static">Below Static</mat-radio-button>
<mat-radio-button class="mb-12" value="below-fixed">Below Fixed</mat-radio-button>
</mat-radio-group>
</div>
</ng-container>
<!-- VERTICAL LAYOUT #2 -->
<ng-container *ngSwitchCase="'vertical-layout-2'">
<!-- LAYOUT WIDTH -->
<div class="group mt-32">
<h2>Layout Width</h2>
<mat-radio-group fxLayout="column" fxLayoutAlign="start start" formControlName="width">
<mat-radio-button class="mb-12" value="fullwidth">Fullwidth</mat-radio-button>
<mat-radio-button class="mb-12" value="boxed">Boxed</mat-radio-button>
</mat-radio-group>
</div>
<!-- NAVBAR -->
<div class="group" formGroupName="navbar">
<h2>Navbar</h2>
<mat-slide-toggle formControlName="hidden">
Hide
</mat-slide-toggle>
<mat-slide-toggle class="mt-24" formControlName="folded">
Folded
</mat-slide-toggle>
<h3 class="mt-24">Position:</h3>
<mat-radio-group fxLayout="column" fxLayoutAlign="start start" formControlName="position">
<mat-radio-button class="mb-16" value="left">Left</mat-radio-button>
<mat-radio-button class="mb-16" value="right">Right</mat-radio-button>
</mat-radio-group>
</div>
<!-- TOOLBAR -->
<div class="group" formGroupName="toolbar">
<h2>Toolbar</h2>
<mat-slide-toggle formControlName="hidden">
Hide
</mat-slide-toggle>
<h3 class="mt-24">Position:</h3>
<mat-radio-group fxLayout="column" fxLayoutAlign="start start" formControlName="position">
<mat-radio-button class="mb-12" value="above-static">Above Static</mat-radio-button>
<mat-radio-button class="mb-12" value="above-fixed">Above Fixed</mat-radio-button>
<mat-radio-button class="mb-12" value="below">Below</mat-radio-button>
</mat-radio-group>
</div>
<!-- FOOTER -->
<div class="group" formGroupName="footer">
<h2>Footer</h2>
<mat-slide-toggle formControlName="hidden">
Hide
</mat-slide-toggle>
<h3 class="mt-24">Position:</h3>
<mat-radio-group fxLayout="column" fxLayoutAlign="start start" formControlName="position">
<mat-radio-button class="mb-12" value="above-static">Above Static</mat-radio-button>
<mat-radio-button class="mb-12" value="above-fixed">Above Fixed</mat-radio-button>
<mat-radio-button class="mb-12" value="below">Below</mat-radio-button>
</mat-radio-group>
</div>
</ng-container>
<!-- VERTICAL LAYOUT #3 -->
<ng-container *ngSwitchCase="'vertical-layout-3'">
<!-- LAYOUT WIDTH -->
<div class="group mt-32">
<h2>Layout Width</h2>
<mat-radio-group fxLayout="column" fxLayoutAlign="start start" formControlName="width">
<mat-radio-button class="mb-12" value="fullwidth">Fullwidth</mat-radio-button>
<mat-radio-button class="mb-12" value="boxed">Boxed</mat-radio-button>
</mat-radio-group>
</div>
<!-- NAVBAR -->
<div class="group" formGroupName="navbar">
<h2>Navbar</h2>
<mat-slide-toggle formControlName="hidden">
Hide
</mat-slide-toggle>
<mat-slide-toggle class="mt-24" formControlName="folded">
Folded
</mat-slide-toggle>
<h3 class="mt-24">Position:</h3>
<mat-radio-group fxLayout="column" fxLayoutAlign="start start" formControlName="position">
<mat-radio-button class="mb-16" value="left">Left</mat-radio-button>
<mat-radio-button class="mb-16" value="right">Right</mat-radio-button>
</mat-radio-group>
</div>
<!-- TOOLBAR -->
<div class="group" formGroupName="toolbar">
<h2>Toolbar</h2>
<mat-slide-toggle formControlName="hidden">
Hide
</mat-slide-toggle>
<h3 class="mt-24">Position:</h3>
<mat-radio-group fxLayout="column" fxLayoutAlign="start start" formControlName="position">
<mat-radio-button class="mb-12" value="above-static">Above Static</mat-radio-button>
<mat-radio-button class="mb-12" value="above-fixed">Above Fixed</mat-radio-button>
</mat-radio-group>
</div>
<!-- FOOTER -->
<div class="group" formGroupName="footer">
<h2>Footer</h2>
<mat-slide-toggle formControlName="hidden">
Hide
</mat-slide-toggle>
<h3 class="mt-24">Position:</h3>
<mat-radio-group fxLayout="column" fxLayoutAlign="start start" formControlName="position">
<mat-radio-button class="mb-12" value="above-static">Above Static</mat-radio-button>
<mat-radio-button class="mb-12" value="above-fixed">Above Fixed</mat-radio-button>
</mat-radio-group>
</div>
</ng-container>
<!-- HORIZONTAL LAYOUT #1 -->
<ng-container *ngSwitchCase="'horizontal-layout-1'">
<!-- LAYOUT WIDTH -->
<div class="group mt-32">
<h2>Layout Width</h2>
<mat-radio-group fxLayout="column" fxLayoutAlign="start start" formControlName="width">
<mat-radio-button class="mb-12" value="fullwidth">Fullwidth</mat-radio-button>
<mat-radio-button class="mb-12" value="boxed">Boxed</mat-radio-button>
</mat-radio-group>
</div>
<!-- NAVBAR -->
<div class="group" formGroupName="navbar">
<h2>Navbar</h2>
<mat-slide-toggle formControlName="hidden">
Hide
</mat-slide-toggle>
<h3 class="mt-24">Position:</h3>
<mat-radio-group fxLayout="column" fxLayoutAlign="start start" formControlName="position">
<mat-radio-button class="mb-16" value="top">Top</mat-radio-button>
</mat-radio-group>
</div>
<!-- TOOLBAR -->
<div class="group" formGroupName="toolbar">
<h2>Toolbar</h2>
<mat-slide-toggle formControlName="hidden">
Hide
</mat-slide-toggle>
<h3 class="mt-24">Position:</h3>
<mat-radio-group fxLayout="column" fxLayoutAlign="start start" formControlName="position">
<mat-radio-button class="mb-12" value="above">Above</mat-radio-button>
<mat-radio-button class="mb-12" value="below">Below</mat-radio-button>
</mat-radio-group>
</div>
<!-- FOOTER -->
<div class="group" formGroupName="footer">
<h2>Footer</h2>
<mat-slide-toggle formControlName="hidden">
Hide
</mat-slide-toggle>
<h3 class="mt-24">Position:</h3>
<mat-radio-group fxLayout="column" fxLayoutAlign="start start" formControlName="position">
<mat-radio-button class="mb-12" value="above-fixed">Above Fixed</mat-radio-button>
<mat-radio-button class="mb-12" value="above-static">Above Static</mat-radio-button>
</mat-radio-group>
</div>
</ng-container>
</ng-container>
</div>
<!-- CUSTOM SCROLLBARS -->
<div class="group">
<h2>Custom scrollbars</h2>
<mat-slide-toggle class="mb-12" formControlName="customScrollbars">
Enable custom scrollbars
</mat-slide-toggle>
</div>
<!-- COLORS -->
<div class="group">
<h2>Colors</h2>
<div class="colors">
<div fxFlex fxLayout="row" fxLayoutAlign="space-between center">
<h4 class="mr-8">Toolbar Color</h4>
<fuse-material-color-picker
[(selectedClass)]="fuseConfig.layout.toolbar.background"></fuse-material-color-picker>
</div>
<div fxFlex fxLayout="row" fxLayoutAlign="space-between center">
<h4 class="mr-8">Navbar Color</h4>
<fuse-material-color-picker
[(selectedClass)]="fuseConfig.layout.navbar.background"></fuse-material-color-picker>
</div>
<div fxFlex fxLayout="row" fxLayoutAlign="space-between center">
<h4 class="mr-8">Footer Color</h4>
<fuse-material-color-picker
[(selectedClass)]="fuseConfig.layout.footer.background"></fuse-material-color-picker>
</div>
<div fxFlex fxLayout="row" fxLayoutAlign="space-between center">
<h4 class="mr-8">Footer Color</h4>
<fuse-material-color-picker [(selectedClass)]="config.colorClasses.footer"
(onValueChange)="onSettingsChange()"></fuse-material-color-picker>
</div> </div>
</div> </div>
<mat-divider></mat-divider> </form>
<h3>Router Animation:</h3>
<mat-form-field class="w-100-p">
<mat-select class="p-0" [(ngModel)]="config.routerAnimation">
<mat-option value="none">
None
</mat-option>
<mat-option value="slideUp">
Slide up
</mat-option>
<mat-option value="slideDown">
Slide down
</mat-option>
<mat-option value="slideRight">
Slide right
</mat-option>
<mat-option value="slideLeft">
Slide left
</mat-option>
<mat-option value="fadeIn">
Fade in
</mat-option>
</mat-select>
</mat-form-field>
</div>
</div> </div>

View File

@ -10,106 +10,74 @@
} }
:host { :host {
position: fixed; display: flex;
display: block; overflow: hidden;
right: 0;
top: 160px;
z-index: 998;
&.bar-closed .theme-options-panel {
display: none;
}
.theme-options-panel { .theme-options-panel {
position: absolute; display: flex;
right: 0; flex-direction: column;
top: 0; flex: 1 0 auto;
width: 360px; padding: 40px 24px 24px 24px;
transform: translate3d(100%, 0, 0);
z-index: 999;
max-height: calc(100vh - 200px);
padding: 24px;
overflow: auto; overflow: auto;
@include media-breakpoint-down('xs') { .header {
top: -120px; display: flex;
max-height: calc(100vh - 100px); flex: 0 1 auto;
width: 90vw; margin-bottom: 32px;
align-items: center;
justify-content: space-between;
.title {
font-size: 20px;
font-weight: 500;
padding-left: 4px;
}
} }
.close-button { form {
position: absolute; display: flex;
top: 8px; flex: 1 1 auto;
right: 8px; flex-direction: column;
}
h3 { .group {
font-size: 14px; display: flex;
font-weight: 500; flex: 1 0 auto;
color: rgba(0, 0, 0, 0.54); flex-direction: column;
} position: relative;
border: 1px solid rgba(0, 0, 0, 0.12);
border-radius: 2px;
padding: 28px 16px 8px 16px;
margin: 16px 0;
.mat-divider { h2 {
display: block !important; position: absolute;
width: 100%; top: -11px;
margin: 24px 0 16px 0; left: 8px;
} margin: 0;
padding: 0 8px;
font-size: 16px;
font-weight: 500;
background: white;
color: rgba(0, 0, 0, 0.54);
}
.colors { h3 {
display: block !important; font-size: 14px;
width: 100%; font-weight: 500;
color: rgba(0, 0, 0, 0.54);
margin: 24px 0 16px 0;
padding: 0;
&:first-of-type {
margin-top: 0;
}
}
}
.colors {
display: block !important;
width: 100%;
}
} }
} }
.theme-options-panel-overlay {
position: fixed;
display: block;
background: rgba(0, 0, 0, 0);
top: 0;
right: 0;
left: 0;
bottom: 0;
z-index: 998;
@include media-breakpoint-down('sm') {
background: rgba(0, 0, 0, 0.37);
}
&.hidden {
display: none;
}
}
.mat-list .mat-list-item {
font-size: 15px;
}
.mat-divider {
margin: 16px;
}
.open-button {
position: absolute;
top: 0;
left: -48px;
width: 48px;
height: 48px;
line-height: 48px;
text-align: center;
cursor: pointer;
border-radius: 0;
margin: 0;
pointer-events: auto;
opacity: .75;
z-index: 998;
mat-icon {
animation: rotating 3s linear infinite;
}
&:hover {
opacity: 1;
}
}
} }

View File

@ -1,10 +1,12 @@
import { Component, ElementRef, HostBinding, Input, OnDestroy, OnInit, Renderer2, ViewChild } from '@angular/core'; import { Component, HostBinding, OnDestroy, OnInit, Renderer2 } from '@angular/core';
import { style, animate, AnimationBuilder, AnimationPlayer } from '@angular/animations'; import { FormBuilder, FormControl, FormGroup } from '@angular/forms';
import { Subscription } from 'rxjs'; import { Subject } from 'rxjs';
import { takeUntil } from 'rxjs/operators';
import { fuseAnimations } from '@fuse/animations'; import { fuseAnimations } from '@fuse/animations';
import { FuseConfigService } from '@fuse/services/config.service'; import { FuseConfigService } from '@fuse/services/config.service';
import { FuseNavigationService } from '@fuse/components/navigation/navigation.service'; import { FuseNavigationService } from '@fuse/components/navigation/navigation.service';
import { FuseSidebarService } from '@fuse/components/sidebar/sidebar.service';
@Component({ @Component({
selector : 'fuse-theme-options', selector : 'fuse-theme-options',
@ -14,50 +16,113 @@ import { FuseNavigationService } from '@fuse/components/navigation/navigation.se
}) })
export class FuseThemeOptionsComponent implements OnInit, OnDestroy export class FuseThemeOptionsComponent implements OnInit, OnDestroy
{ {
@Input() navigation; fuseConfig: any;
@ViewChild('openButton') openButton; form: FormGroup;
@ViewChild('panel') panel;
@ViewChild('overlay') overlay: ElementRef;
public player: AnimationPlayer; @HostBinding('class.bar-closed')
config: any; barClosed: boolean;
onConfigChanged: Subscription; // Private
private _unsubscribeAll: Subject<any>;
@HostBinding('class.bar-closed') barClosed: boolean;
/**
* Constructor
*
* @param {FormBuilder} _formBuilder
* @param {FuseConfigService} _fuseConfigService
* @param {FuseNavigationService} _fuseNavigationService
* @param {FuseSidebarService} _fuseSidebarService
* @param {Renderer2} _renderer
*/
constructor( constructor(
private animationBuilder: AnimationBuilder, private _formBuilder: FormBuilder,
private fuseConfig: FuseConfigService, private _fuseConfigService: FuseConfigService,
private navigationService: FuseNavigationService, private _fuseNavigationService: FuseNavigationService,
private renderer: Renderer2 private _fuseSidebarService: FuseSidebarService,
private _renderer: Renderer2
) )
{ {
// Set the defaults
this.barClosed = true; this.barClosed = true;
this.onConfigChanged = // Set the private defaults
this.fuseConfig.onConfigChanged this._unsubscribeAll = new Subject();
.subscribe(
(newConfig) => {
this.config = newConfig;
}
);
} }
ngOnInit() // -----------------------------------------------------------------------------------------------------
// @ Lifecycle hooks
// -----------------------------------------------------------------------------------------------------
/**
* On init
*/
ngOnInit(): void
{ {
this.renderer.listen(this.overlay.nativeElement, 'click', () => { // Build the config form
this.closeBar(); // noinspection TypeScriptValidateTypes
this.form = this._formBuilder.group({
layout : this._formBuilder.group({
style : new FormControl(),
width : new FormControl(),
navbar : this._formBuilder.group({
hidden : new FormControl(),
position : new FormControl(),
folded : new FormControl(),
background: new FormControl()
}),
toolbar: this._formBuilder.group({
hidden : new FormControl(),
position : new FormControl(),
background: new FormControl()
}),
footer : this._formBuilder.group({
hidden : new FormControl(),
position : new FormControl(),
background: new FormControl()
})
}),
customScrollbars: new FormControl()
}); });
// Get the nav model and add customize nav item // Subscribe to the config changes
// that opens the bar programmatically this._fuseConfigService.config
const nav: any = this.navigation; .pipe(takeUntil(this._unsubscribeAll))
.subscribe((config) => {
nav.push({ // Update the stored config
this.fuseConfig = config;
// Set the config form values without emitting an event
// so that we don't end up with an infinite loop
this.form.setValue(config, {emitEvent: false});
});
// Subscribe to the specific form value changes (layout.style)
this.form.get('layout.style').valueChanges
.pipe(takeUntil(this._unsubscribeAll))
.subscribe((value) => {
// Reset the form values based on the
// selected layout style
this._resetFormValues(value);
});
// Subscribe to the form value changes
this.form.valueChanges
.pipe(takeUntil(this._unsubscribeAll))
.subscribe((config) => {
// Update the config
this._fuseConfigService.config = config;
});
// Add customize nav item that opens the bar programmatically
const customFunctionNavItem = {
'id' : 'custom-function', 'id' : 'custom-function',
'title' : 'Custom Function', 'title' : 'Custom Function',
'type' : 'group', 'type' : 'group',
'icon' : 'settings',
'children': [ 'children': [
{ {
'id' : 'customize', 'id' : 'customize',
@ -65,50 +130,169 @@ export class FuseThemeOptionsComponent implements OnInit, OnDestroy
'type' : 'item', 'type' : 'item',
'icon' : 'settings', 'icon' : 'settings',
'function': () => { 'function': () => {
this.openBar(); this.toggleSidebarOpen('themeOptionsPanel');
} }
} }
] ]
}); };
this._fuseNavigationService.addNavigationItem(customFunctionNavItem, 'end');
} }
ngOnDestroy() /**
* On destroy
*/
ngOnDestroy(): void
{ {
this.onConfigChanged.unsubscribe(); // Unsubscribe from all subscriptions
this._unsubscribeAll.next();
this._unsubscribeAll.complete();
// Remove the custom function menu
this._fuseNavigationService.removeNavigationItem('custom-function');
} }
onSettingsChange() // -----------------------------------------------------------------------------------------------------
// @ Private methods
// -----------------------------------------------------------------------------------------------------
/**
* Reset the form values based on the
* selected layout style
*
* @param value
* @private
*/
private _resetFormValues(value): void
{ {
this.fuseConfig.setConfig(this.config); switch ( value )
{
// Vertical Layout #1
case 'vertical-layout-1':
{
this.form.patchValue({
layout: {
width : 'fullwidth',
navbar : {
hidden : false,
position : 'left',
folded : false,
background: 'mat-fuse-dark-700-bg'
},
toolbar: {
hidden : false,
position : 'below-static',
background: 'mat-white-500-bg'
},
footer : {
hidden : false,
position : 'below-static',
background: 'mat-fuse-dark-900-bg'
}
}
});
break;
}
// Vertical Layout #2
case 'vertical-layout-2':
{
this.form.patchValue({
layout: {
width : 'fullwidth',
navbar : {
hidden : false,
position : 'left',
folded : false,
background: 'mat-fuse-dark-700-bg'
},
toolbar: {
hidden : false,
position : 'below',
background: 'mat-white-500-bg'
},
footer : {
hidden : false,
position : 'below',
background: 'mat-fuse-dark-900-bg'
}
}
});
break;
}
// Vertical Layout #3
case 'vertical-layout-3':
{
this.form.patchValue({
layout: {
width : 'fullwidth',
navbar : {
hidden : false,
position : 'left',
folded : false,
background: 'mat-fuse-dark-700-bg'
},
toolbar: {
hidden : false,
position : 'above-static',
background: 'mat-white-500-bg'
},
footer : {
hidden : false,
position : 'above-static',
background: 'mat-fuse-dark-900-bg'
}
}
});
break;
}
// Horizontal Layout #1
case 'horizontal-layout-1':
{
this.form.patchValue({
layout: {
width : 'fullwidth',
navbar : {
hidden : false,
position : 'top',
folded : false,
background: 'mat-fuse-dark-700-bg'
},
toolbar: {
hidden : false,
position : 'above',
background: 'mat-white-500-bg'
},
footer : {
hidden : false,
position : 'above-fixed',
background: 'mat-fuse-dark-900-bg'
}
}
});
break;
}
}
} }
closeBar() // -----------------------------------------------------------------------------------------------------
// @ Public methods
// -----------------------------------------------------------------------------------------------------
/**
* Toggle sidebar open
*
* @param key
*/
toggleSidebarOpen(key): void
{ {
this.player = this._fuseSidebarService.getSidebar(key).toggleOpen();
this.animationBuilder
.build([
style({transform: 'translate3d(0,0,0)'}),
animate('400ms ease', style({transform: 'translate3d(100%,0,0)'}))
]).create(this.panel.nativeElement);
this.player.play();
this.player.onDone(() => {
this.barClosed = true;
});
} }
openBar()
{
this.barClosed = false;
this.player =
this.animationBuilder
.build([
style({transform: 'translate3d(100%,0,0)'}),
animate('400ms ease', style({transform: 'translate3d(0,0,0)'}))
]).create(this.panel.nativeElement);
this.player.play();
}
} }

View File

@ -1,10 +1,13 @@
import { NgModule } from '@angular/core'; import { NgModule } from '@angular/core';
import { CommonModule } from '@angular/common'; import { CommonModule } from '@angular/common';
import { FormsModule } from '@angular/forms'; import { FormsModule, ReactiveFormsModule } from '@angular/forms';
import { FlexLayoutModule } from '@angular/flex-layout'; import { FlexLayoutModule } from '@angular/flex-layout';
import { MatButtonModule, MatDividerModule, MatFormFieldModule, MatIconModule, MatOptionModule, MatRadioModule, MatSelectModule, MatSlideToggleModule } from '@angular/material'; import { MatButtonModule, MatDividerModule, MatFormFieldModule, MatIconModule, MatOptionModule, MatRadioModule, MatSelectModule, MatSlideToggleModule } from '@angular/material';
import { FuseDirectivesModule } from '@fuse/directives/directives';
import { FuseSidebarModule } from '@fuse/components/sidebar/sidebar.module';
import { FuseMaterialColorPickerModule } from '@fuse/components/material-color-picker/material-color-picker.module'; import { FuseMaterialColorPickerModule } from '@fuse/components/material-color-picker/material-color-picker.module';
import { FuseThemeOptionsComponent } from '@fuse/components/theme-options/theme-options.component'; import { FuseThemeOptionsComponent } from '@fuse/components/theme-options/theme-options.component';
@NgModule({ @NgModule({
@ -14,6 +17,7 @@ import { FuseThemeOptionsComponent } from '@fuse/components/theme-options/theme-
imports : [ imports : [
CommonModule, CommonModule,
FormsModule, FormsModule,
ReactiveFormsModule,
FlexLayoutModule, FlexLayoutModule,
@ -26,7 +30,9 @@ import { FuseThemeOptionsComponent } from '@fuse/components/theme-options/theme-
MatSelectModule, MatSelectModule,
MatSlideToggleModule, MatSlideToggleModule,
FuseMaterialColorPickerModule FuseDirectivesModule,
FuseMaterialColorPickerModule,
FuseSidebarModule
], ],
exports : [ exports : [
FuseThemeOptionsComponent FuseThemeOptionsComponent

View File

@ -5,7 +5,14 @@ import { Directive, ElementRef } from '@angular/core';
}) })
export class FuseWidgetToggleDirective export class FuseWidgetToggleDirective
{ {
constructor(public el: ElementRef) /**
* Constructor
*
* @param {ElementRef} elementRef
*/
constructor(
public elementRef: ElementRef
)
{ {
} }
} }

View File

@ -10,19 +10,38 @@ import { FuseWidgetToggleDirective } from './widget-toggle.directive';
export class FuseWidgetComponent implements AfterContentInit export class FuseWidgetComponent implements AfterContentInit
{ {
@HostBinding('class.flipped') flipped = false; @HostBinding('class.flipped')
@ContentChildren(FuseWidgetToggleDirective, {descendants: true}) toggleButtons: QueryList<FuseWidgetToggleDirective>; flipped = false;
constructor(private el: ElementRef, private renderer: Renderer2) @ContentChildren(FuseWidgetToggleDirective, {descendants: true})
toggleButtons: QueryList<FuseWidgetToggleDirective>;
/**
* Constructor
*
* @param {ElementRef} _elementRef
* @param {Renderer2} _renderer
*/
constructor(
private _elementRef: ElementRef,
private _renderer: Renderer2
)
{ {
} }
ngAfterContentInit() // -----------------------------------------------------------------------------------------------------
{ // @ Lifecycle hooks
setTimeout(() => { // -----------------------------------------------------------------------------------------------------
/**
* After content init
*/
ngAfterContentInit(): void
{
// Listen for the flip button click
setTimeout(() => {
this.toggleButtons.forEach(flipButton => { this.toggleButtons.forEach(flipButton => {
this.renderer.listen(flipButton.el.nativeElement, 'click', (event) => { this._renderer.listen(flipButton.elementRef.nativeElement, 'click', (event) => {
event.preventDefault(); event.preventDefault();
event.stopPropagation(); event.stopPropagation();
this.toggle(); this.toggle();
@ -31,7 +50,14 @@ export class FuseWidgetComponent implements AfterContentInit
}); });
} }
toggle() // -----------------------------------------------------------------------------------------------------
// @ Public methods
// -----------------------------------------------------------------------------------------------------
/**
* Toggle the flipped status
*/
toggle(): void
{ {
this.flipped = !this.flipped; this.flipped = !this.flipped;
} }

View File

@ -1,12 +1,14 @@
import { NgModule } from '@angular/core'; import { NgModule } from '@angular/core';
import { FuseIfOnDomDirective } from '@fuse/directives/fuse-if-on-dom/fuse-if-on-dom.directive'; import { FuseIfOnDomDirective } from '@fuse/directives/fuse-if-on-dom/fuse-if-on-dom.directive';
import { FuseInnerScrollDirective } from '@fuse/directives/fuse-inner-scroll/fuse-inner-scroll.directive';
import { FusePerfectScrollbarDirective } from '@fuse/directives/fuse-perfect-scrollbar/fuse-perfect-scrollbar.directive'; import { FusePerfectScrollbarDirective } from '@fuse/directives/fuse-perfect-scrollbar/fuse-perfect-scrollbar.directive';
import { FuseMatSidenavHelperDirective, FuseMatSidenavTogglerDirective } from '@fuse/directives/fuse-mat-sidenav/fuse-mat-sidenav.directive'; import { FuseMatSidenavHelperDirective, FuseMatSidenavTogglerDirective } from '@fuse/directives/fuse-mat-sidenav/fuse-mat-sidenav.directive';
@NgModule({ @NgModule({
declarations: [ declarations: [
FuseIfOnDomDirective, FuseIfOnDomDirective,
FuseInnerScrollDirective,
FuseMatSidenavHelperDirective, FuseMatSidenavHelperDirective,
FuseMatSidenavTogglerDirective, FuseMatSidenavTogglerDirective,
FusePerfectScrollbarDirective FusePerfectScrollbarDirective
@ -14,6 +16,7 @@ import { FuseMatSidenavHelperDirective, FuseMatSidenavTogglerDirective } from '@
imports : [], imports : [],
exports : [ exports : [
FuseIfOnDomDirective, FuseIfOnDomDirective,
FuseInnerScrollDirective,
FuseMatSidenavHelperDirective, FuseMatSidenavHelperDirective,
FuseMatSidenavTogglerDirective, FuseMatSidenavTogglerDirective,
FusePerfectScrollbarDirective FusePerfectScrollbarDirective

View File

@ -5,28 +5,44 @@ import { AfterContentChecked, Directive, ElementRef, TemplateRef, ViewContainerR
}) })
export class FuseIfOnDomDirective implements AfterContentChecked export class FuseIfOnDomDirective implements AfterContentChecked
{ {
isCreated = false; isCreated: boolean;
/**
* Constructor
*
* @param {ElementRef} _elementRef
* @param {TemplateRef<any>} _templateRef
* @param {ViewContainerRef} _viewContainerRef
*/
constructor( constructor(
private templateRef: TemplateRef<any>, private _elementRef: ElementRef,
private viewContainer: ViewContainerRef, private _templateRef: TemplateRef<any>,
private element: ElementRef private _viewContainerRef: ViewContainerRef
) )
{ {
// Set the defaults
this.isCreated = false;
} }
ngAfterContentChecked() // -----------------------------------------------------------------------------------------------------
// @ Lifecycle hooks
// -----------------------------------------------------------------------------------------------------
/**
* After content checked
*/
ngAfterContentChecked(): void
{ {
if ( document.body.contains(this.element.nativeElement) && !this.isCreated ) if ( document.body.contains(this._elementRef.nativeElement) && !this.isCreated )
{ {
setTimeout(() => { setTimeout(() => {
this.viewContainer.createEmbeddedView(this.templateRef); this._viewContainerRef.createEmbeddedView(this._templateRef);
}, 300); }, 300);
this.isCreated = true; this.isCreated = true;
} }
else if ( this.isCreated && !document.body.contains(this.element.nativeElement) ) else if ( this.isCreated && !document.body.contains(this._elementRef.nativeElement) )
{ {
this.viewContainer.clear(); this._viewContainerRef.clear();
this.isCreated = false; this.isCreated = false;
} }
} }

View File

@ -0,0 +1,115 @@
import { Directive, ElementRef, OnDestroy, OnInit, Renderer2 } from '@angular/core';
import { Subject } from 'rxjs';
import { takeUntil } from 'rxjs/operators';
import { FuseMatchMediaService } from '@fuse/services/match-media.service';
@Directive({
selector: '.inner-scroll'
})
export class FuseInnerScrollDirective implements OnInit, OnDestroy
{
// Private
private _parent: any;
private _grandParent: any;
private _unsubscribeAll: Subject<any>;
/**
* Constructor
*
* @param {ElementRef} _elementRef
* @param {FuseMatchMediaService} _fuseMediaMatchService
* @param {Renderer2} _renderer
*/
constructor(
private _elementRef: ElementRef,
private _fuseMediaMatchService: FuseMatchMediaService,
private _renderer: Renderer2
)
{
// Set the private defaults
this._unsubscribeAll = new Subject();
}
// -----------------------------------------------------------------------------------------------------
// @ Lifecycle hooks
// -----------------------------------------------------------------------------------------------------
/**
* On init
*/
ngOnInit(): void
{
// Get the parent
this._parent = this._renderer.parentNode(this._elementRef.nativeElement);
// Return, if there is no parent
if ( !this._parent )
{
return;
}
// Get the grand parent
this._grandParent = this._renderer.parentNode(this._parent);
// Register to the media query changes
this._fuseMediaMatchService.onMediaChange
.pipe(takeUntil(this._unsubscribeAll))
.subscribe((alias) => {
if ( alias === 'xs' )
{
this._removeClass();
}
else
{
this._addClass();
}
});
}
/**
* On destroy
*/
ngOnDestroy(): void
{
// Return, if there is no parent
if ( !this._parent )
{
return;
}
// Remove the class
this._removeClass();
// Unsubscribe from all subscriptions
this._unsubscribeAll.next();
this._unsubscribeAll.complete();
}
// -----------------------------------------------------------------------------------------------------
// @ Private methods
// -----------------------------------------------------------------------------------------------------
/**
* Add the class name
*
* @private
*/
private _addClass(): void
{
// Add the inner-scroll class
this._renderer.addClass(this._grandParent, 'inner-scroll');
}
/**
* Remove the class name
* @private
*/
private _removeClass(): void
{
// Remove the inner-scroll class
this._renderer.removeClass(this._grandParent, 'inner-scroll');
}
}

View File

@ -1,7 +1,8 @@
import { Directive, Input, OnInit, HostListener, OnDestroy, HostBinding } from '@angular/core'; import { Directive, Input, OnInit, HostListener, OnDestroy, HostBinding } from '@angular/core';
import { MatSidenav } from '@angular/material'; import { MatSidenav } from '@angular/material';
import { ObservableMedia } from '@angular/flex-layout'; import { ObservableMedia } from '@angular/flex-layout';
import { Subscription } from 'rxjs'; import { Subject } from 'rxjs';
import { takeUntil } from 'rxjs/operators';
import { FuseMatchMediaService } from '@fuse/services/match-media.service'; import { FuseMatchMediaService } from '@fuse/services/match-media.service';
import { FuseMatSidenavHelperService } from '@fuse/directives/fuse-mat-sidenav/fuse-mat-sidenav.service'; import { FuseMatSidenavHelperService } from '@fuse/directives/fuse-mat-sidenav/fuse-mat-sidenav.service';
@ -11,56 +12,91 @@ import { FuseMatSidenavHelperService } from '@fuse/directives/fuse-mat-sidenav/f
}) })
export class FuseMatSidenavHelperDirective implements OnInit, OnDestroy export class FuseMatSidenavHelperDirective implements OnInit, OnDestroy
{ {
matchMediaSubscription: Subscription; @HostBinding('class.mat-is-locked-open')
@HostBinding('class.mat-is-locked-open') isLockedOpen = true; isLockedOpen: boolean;
@Input('fuseMatSidenavHelper') id: string;
@Input('mat-is-locked-open') matIsLockedOpenBreakpoint: string;
@Input('fuseMatSidenavHelper')
id: string;
@Input('mat-is-locked-open')
matIsLockedOpenBreakpoint: string;
// Private
private _unsubscribeAll: Subject<any>;
/**
* Constructor
*
* @param {FuseMatchMediaService} _fuseMatchMediaService
* @param {FuseMatSidenavHelperService} _fuseMatSidenavHelperService
* @param {MatSidenav} _matSidenav
* @param {ObservableMedia} _observableMedia
*/
constructor( constructor(
private fuseMatSidenavService: FuseMatSidenavHelperService, private _fuseMatchMediaService: FuseMatchMediaService,
private fuseMatchMedia: FuseMatchMediaService, private _fuseMatSidenavHelperService: FuseMatSidenavHelperService,
private observableMedia: ObservableMedia, private _matSidenav: MatSidenav,
private matSidenav: MatSidenav private _observableMedia: ObservableMedia
) )
{ {
// Set the defaults
this.isLockedOpen = true;
// Set the private defaults
this._unsubscribeAll = new Subject();
} }
ngOnInit() // -----------------------------------------------------------------------------------------------------
{ // @ Lifecycle hooks
this.fuseMatSidenavService.setSidenav(this.id, this.matSidenav); // -----------------------------------------------------------------------------------------------------
if ( this.observableMedia.isActive(this.matIsLockedOpenBreakpoint) ) /**
* On init
*/
ngOnInit(): void
{
// Register the sidenav to the service
this._fuseMatSidenavHelperService.setSidenav(this.id, this._matSidenav);
if ( this._observableMedia.isActive(this.matIsLockedOpenBreakpoint) )
{ {
this.isLockedOpen = true; this.isLockedOpen = true;
this.matSidenav.mode = 'side'; this._matSidenav.mode = 'side';
this.matSidenav.toggle(true); this._matSidenav.toggle(true);
} }
else else
{ {
this.isLockedOpen = false; this.isLockedOpen = false;
this.matSidenav.mode = 'over'; this._matSidenav.mode = 'over';
this.matSidenav.toggle(false); this._matSidenav.toggle(false);
} }
this.matchMediaSubscription = this.fuseMatchMedia.onMediaChange.subscribe(() => { this._fuseMatchMediaService.onMediaChange
if ( this.observableMedia.isActive(this.matIsLockedOpenBreakpoint) ) .pipe(takeUntil(this._unsubscribeAll))
{ .subscribe(() => {
this.isLockedOpen = true; if ( this._observableMedia.isActive(this.matIsLockedOpenBreakpoint) )
this.matSidenav.mode = 'side'; {
this.matSidenav.toggle(true); this.isLockedOpen = true;
} this._matSidenav.mode = 'side';
else this._matSidenav.toggle(true);
{ }
this.isLockedOpen = false; else
this.matSidenav.mode = 'over'; {
this.matSidenav.toggle(false); this.isLockedOpen = false;
} this._matSidenav.mode = 'over';
}); this._matSidenav.toggle(false);
}
});
} }
ngOnDestroy() /**
* On destroy
*/
ngOnDestroy(): void
{ {
this.matchMediaSubscription.unsubscribe(); // Unsubscribe from all subscriptions
this._unsubscribeAll.next();
this._unsubscribeAll.complete();
} }
} }
@ -69,15 +105,29 @@ export class FuseMatSidenavHelperDirective implements OnInit, OnDestroy
}) })
export class FuseMatSidenavTogglerDirective export class FuseMatSidenavTogglerDirective
{ {
@Input('fuseMatSidenavToggler') id; @Input('fuseMatSidenavToggler')
id;
constructor(private fuseMatSidenavService: FuseMatSidenavHelperService) /**
* Constructor
*
* @param {FuseMatSidenavHelperService} _fuseMatSidenavHelperService
*/
constructor(
private _fuseMatSidenavHelperService: FuseMatSidenavHelperService)
{ {
} }
// -----------------------------------------------------------------------------------------------------
// @ Public methods
// -----------------------------------------------------------------------------------------------------
/**
* On click
*/
@HostListener('click') @HostListener('click')
onClick() onClick()
{ {
this.fuseMatSidenavService.getSidenav(this.id).toggle(); this._fuseMatSidenavHelperService.getSidenav(this.id).toggle();
} }
} }

View File

@ -6,17 +6,36 @@ export class FuseMatSidenavHelperService
{ {
sidenavInstances: MatSidenav[]; sidenavInstances: MatSidenav[];
/**
* Constructor
*/
constructor() constructor()
{ {
this.sidenavInstances = []; this.sidenavInstances = [];
} }
setSidenav(id, instance) // -----------------------------------------------------------------------------------------------------
// @ Accessors
// -----------------------------------------------------------------------------------------------------
/**
* Set sidenav
*
* @param id
* @param instance
*/
setSidenav(id, instance): void
{ {
this.sidenavInstances[id] = instance; this.sidenavInstances[id] = instance;
} }
getSidenav(id) /**
* Get sidenav
*
* @param id
* @returns {any}
*/
getSidenav(id): any
{ {
return this.sidenavInstances[id]; return this.sidenavInstances[id];
} }

View File

@ -1,72 +1,247 @@
import { AfterViewInit, Directive, ElementRef, HostListener, OnDestroy, OnInit } from '@angular/core'; import { AfterViewInit, Directive, ElementRef, HostListener, Input, OnDestroy } from '@angular/core';
import { NavigationEnd, Router } from '@angular/router';
import { Platform } from '@angular/cdk/platform'; import { Platform } from '@angular/cdk/platform';
import { Subscription } from 'rxjs'; import { Subject } from 'rxjs';
import { filter, takeUntil } from 'rxjs/operators';
import PerfectScrollbar from 'perfect-scrollbar'; import PerfectScrollbar from 'perfect-scrollbar';
import * as _ from 'lodash';
import { FuseConfigService } from '@fuse/services/config.service'; import { FuseConfigService } from '@fuse/services/config.service';
@Directive({ @Directive({
selector: '[fusePerfectScrollbar]' selector: '[fusePerfectScrollbar]'
}) })
export class FusePerfectScrollbarDirective implements OnInit, AfterViewInit, OnDestroy export class FusePerfectScrollbarDirective implements AfterViewInit, OnDestroy
{ {
onConfigChanged: Subscription; isInitialized: boolean;
isDisableCustomScrollbars = false; isMobile: boolean;
isMobile = false;
isInitialized = true;
ps: PerfectScrollbar; ps: PerfectScrollbar;
// Private
private _enabled: boolean | '';
private _debouncedUpdate: any;
private _options: any;
private _unsubscribeAll: Subject<any>;
/**
* Constructor
*
* @param {ElementRef} elementRef
* @param {FuseConfigService} _fuseConfigService
* @param {Platform} _platform
* @param {Router} _router
*/
constructor( constructor(
public element: ElementRef, public elementRef: ElementRef,
private fuseConfig: FuseConfigService, private _fuseConfigService: FuseConfigService,
private platform: Platform private _platform: Platform,
private _router: Router
) )
{ {
// Set the defaults
this.isInitialized = false;
this.isMobile = false;
// Set the private defaults
this._enabled = false;
this._debouncedUpdate = _.debounce(this.update, 150);
this._options = {
updateOnRouteChange: false
};
this._unsubscribeAll = new Subject();
} }
ngOnInit() // -----------------------------------------------------------------------------------------------------
{ // @ Accessors
this.onConfigChanged = // -----------------------------------------------------------------------------------------------------
this.fuseConfig.onConfigChanged.subscribe(
(settings) => {
this.isDisableCustomScrollbars = !settings.customScrollbars;
}
);
if ( this.platform.ANDROID || this.platform.IOS ) /**
* Perfect Scrollbar options
*
* @param value
*/
@Input()
set fusePerfectScrollbarOptions(value)
{
// Merge the options
this._options = _.merge({}, this._options, value);
}
get fusePerfectScrollbarOptions(): any
{
// Return the options
return this._options;
}
/**
* Is enabled
*
* @param {boolean | ""} value
*/
@Input('fusePerfectScrollbar')
set enabled(value: boolean | '')
{
// If nothing is provided with the directive (empty string),
// we will take that as a true
if ( value === '' )
{ {
this.isMobile = true; value = true;
} }
}
ngAfterViewInit() // Return, if both values are the same
{ if ( this.enabled === value )
if ( this.isMobile || this.isDisableCustomScrollbars )
{ {
this.isInitialized = false;
return; return;
} }
// Store the value
this._enabled = value;
// If enabled...
if ( this.enabled )
{
// Init the directive
this._init();
}
else
{
// Otherwise destroy it
this._destroy();
}
}
get enabled(): boolean | ''
{
// Return the enabled status
return this._enabled;
}
// -----------------------------------------------------------------------------------------------------
// @ Lifecycle hooks
// -----------------------------------------------------------------------------------------------------
/**
* After view init
*/
ngAfterViewInit(): void
{
// Check if scrollbars enabled or not from the main config
this._fuseConfigService.config
.pipe(takeUntil(this._unsubscribeAll))
.subscribe(
(settings) => {
this.enabled = settings.customScrollbars;
}
);
// Scroll to the top on every route change
if ( this.fusePerfectScrollbarOptions.updateOnRouteChange )
{
this._router.events
.pipe(
takeUntil(this._unsubscribeAll),
filter(event => event instanceof NavigationEnd)
)
.subscribe(() => {
setTimeout(() => {
this.scrollToTop();
this.update();
}, 0);
});
}
}
/**
* On destroy
*/
ngOnDestroy(): void
{
this._destroy();
// Unsubscribe from all subscriptions
this._unsubscribeAll.next();
this._unsubscribeAll.complete();
}
// -----------------------------------------------------------------------------------------------------
// @ Private methods
// -----------------------------------------------------------------------------------------------------
/**
* Initialize
*
* @private
*/
_init(): void
{
// Return, if already initialized
if ( this.isInitialized )
{
return;
}
// Check if is mobile
if ( this._platform.ANDROID || this._platform.IOS )
{
this.isMobile = true;
}
// Return if it's mobile
if ( this.isMobile )
{
// Return...
return;
}
// Set as initialized
this.isInitialized = true;
// Initialize the perfect-scrollbar // Initialize the perfect-scrollbar
this.ps = new PerfectScrollbar(this.element.nativeElement, { this.ps = new PerfectScrollbar(this.elementRef.nativeElement, {
wheelPropagation: true ...this.fusePerfectScrollbarOptions
}); });
} }
ngOnDestroy() /**
* Destroy
*
* @private
*/
_destroy(): void
{ {
if ( !this.isInitialized || !this.ps ) if ( !this.isInitialized || !this.ps )
{ {
return; return;
} }
this.onConfigChanged.unsubscribe();
// Destroy the perfect-scrollbar // Destroy the perfect-scrollbar
this.ps.destroy(); this.ps.destroy();
// Clean up
this.ps = null;
this.isInitialized = false;
} }
/**
* Update scrollbars on window resize
*
* @private
*/
@HostListener('window:resize')
_updateOnResize(): void
{
this._debouncedUpdate();
}
// -----------------------------------------------------------------------------------------------------
// @ Public methods
// -----------------------------------------------------------------------------------------------------
/**
* Document click
*
* @param {Event} event
*/
@HostListener('document:click', ['$event']) @HostListener('document:click', ['$event'])
documentClick(event: Event): void documentClick(event: Event): void
{ {
@ -82,7 +257,10 @@ export class FusePerfectScrollbarDirective implements OnInit, AfterViewInit, OnD
this.ps.update(); this.ps.update();
} }
update() /**
* Update the scrollbar
*/
update(): void
{ {
if ( !this.isInitialized ) if ( !this.isInitialized )
{ {
@ -93,62 +271,108 @@ export class FusePerfectScrollbarDirective implements OnInit, AfterViewInit, OnD
this.ps.update(); this.ps.update();
} }
destroy() /**
* Destroy the scrollbar
*/
destroy(): void
{ {
this.ngOnDestroy(); this.ngOnDestroy();
} }
scrollToX(x: number, speed?: number) /**
* Scroll to X
*
* @param {number} x
* @param {number} speed
*/
scrollToX(x: number, speed?: number): void
{ {
this.animateScrolling('scrollLeft', x, speed); this.animateScrolling('scrollLeft', x, speed);
} }
scrollToY(y: number, speed?: number) /**
* Scroll to Y
*
* @param {number} y
* @param {number} speed
*/
scrollToY(y: number, speed?: number): void
{ {
this.animateScrolling('scrollTop', y, speed); this.animateScrolling('scrollTop', y, speed);
} }
scrollToTop(offset?: number, speed?: number) /**
* Scroll to top
*
* @param {number} offset
* @param {number} speed
*/
scrollToTop(offset?: number, speed?: number): void
{ {
this.animateScrolling('scrollTop', (offset || 0), speed); this.animateScrolling('scrollTop', (offset || 0), speed);
} }
scrollToLeft(offset?: number, speed?: number) /**
* Scroll to left
*
* @param {number} offset
* @param {number} speed
*/
scrollToLeft(offset?: number, speed?: number): void
{ {
this.animateScrolling('scrollLeft', (offset || 0), speed); this.animateScrolling('scrollLeft', (offset || 0), speed);
} }
scrollToRight(offset?: number, speed?: number) /**
* Scroll to right
*
* @param {number} offset
* @param {number} speed
*/
scrollToRight(offset?: number, speed?: number): void
{ {
const width = this.element.nativeElement.scrollWidth; const width = this.elementRef.nativeElement.scrollWidth;
this.animateScrolling('scrollLeft', width - (offset || 0), speed); this.animateScrolling('scrollLeft', width - (offset || 0), speed);
} }
scrollToBottom(offset?: number, speed?: number) /**
* Scroll to bottom
*
* @param {number} offset
* @param {number} speed
*/
scrollToBottom(offset?: number, speed?: number): void
{ {
const height = this.element.nativeElement.scrollHeight; const height = this.elementRef.nativeElement.scrollHeight;
this.animateScrolling('scrollTop', height - (offset || 0), speed); this.animateScrolling('scrollTop', height - (offset || 0), speed);
} }
animateScrolling(target: string, value: number, speed?: number) /**
* Animate scrolling
*
* @param {string} target
* @param {number} value
* @param {number} speed
*/
animateScrolling(target: string, value: number, speed?: number): void
{ {
if ( !speed ) if ( !speed )
{ {
this.element.nativeElement[target] = value; this.elementRef.nativeElement[target] = value;
// PS has weird event sending order, this is a workaround for that // PS has weird event sending order, this is a workaround for that
this.update(); this.update();
this.update(); this.update();
} }
else if ( value !== this.element.nativeElement[target] ) else if ( value !== this.elementRef.nativeElement[target] )
{ {
let newValue = 0; let newValue = 0;
let scrollCount = 0; let scrollCount = 0;
let oldTimestamp = performance.now(); let oldTimestamp = performance.now();
let oldValue = this.element.nativeElement[target]; let oldValue = this.elementRef.nativeElement[target];
const cosParameter = (oldValue - value) / 2; const cosParameter = (oldValue - value) / 2;
@ -158,11 +382,11 @@ export class FusePerfectScrollbarDirective implements OnInit, AfterViewInit, OnD
newValue = Math.round(value + cosParameter + cosParameter * Math.cos(scrollCount)); newValue = Math.round(value + cosParameter + cosParameter * Math.cos(scrollCount));
// Only continue animation if scroll position has not changed // Only continue animation if scroll position has not changed
if ( this.element.nativeElement[target] === oldValue ) if ( this.elementRef.nativeElement[target] === oldValue )
{ {
if ( scrollCount >= Math.PI ) if ( scrollCount >= Math.PI )
{ {
this.element.nativeElement[target] = value; this.elementRef.nativeElement[target] = value;
// PS has weird event sending order, this is a workaround for that // PS has weird event sending order, this is a workaround for that
this.update(); this.update();
@ -171,7 +395,7 @@ export class FusePerfectScrollbarDirective implements OnInit, AfterViewInit, OnD
} }
else else
{ {
this.element.nativeElement[target] = oldValue = newValue; this.elementRef.nativeElement[target] = oldValue = newValue;
oldTimestamp = newTimestamp; oldTimestamp = newTimestamp;

View File

@ -3,7 +3,14 @@ import { Pipe, PipeTransform } from '@angular/core';
@Pipe({name: 'camelCaseToDash'}) @Pipe({name: 'camelCaseToDash'})
export class CamelCaseToDashPipe implements PipeTransform export class CamelCaseToDashPipe implements PipeTransform
{ {
transform(value: string, args: any[] = []) /**
* Transform
*
* @param {string} value
* @param {any[]} args
* @returns {string}
*/
transform(value: string, args: any[] = []): string
{ {
return value ? String(value).replace(/([A-Z])/g, (g) => `-${g[0].toLowerCase()}`) : ''; return value ? String(value).replace(/([A-Z])/g, (g) => `-${g[0].toLowerCase()}`) : '';
} }

View File

@ -4,6 +4,14 @@ import { FuseUtils } from '@fuse/utils';
@Pipe({name: 'filter'}) @Pipe({name: 'filter'})
export class FilterPipe implements PipeTransform export class FilterPipe implements PipeTransform
{ {
/**
* Transform
*
* @param {any[]} mainArr
* @param {string} searchText
* @param {string} property
* @returns {any}
*/
transform(mainArr: any[], searchText: string, property: string): any transform(mainArr: any[], searchText: string, property: string): any
{ {
return FuseUtils.filterArrayByString(mainArr, searchText); return FuseUtils.filterArrayByString(mainArr, searchText);

View File

@ -6,6 +6,14 @@ import { Pipe, PipeTransform } from '@angular/core';
}) })
export class GetByIdPipe implements PipeTransform export class GetByIdPipe implements PipeTransform
{ {
/**
* Transform
*
* @param {any[]} value
* @param {number} id
* @param {string} property
* @returns {any}
*/
transform(value: any[], id: number, property: string): any transform(value: any[], id: number, property: string): any
{ {
const foundItem = value.find(item => { const foundItem = value.find(item => {

View File

@ -3,7 +3,14 @@ import { Pipe, PipeTransform } from '@angular/core';
@Pipe({name: 'htmlToPlaintext'}) @Pipe({name: 'htmlToPlaintext'})
export class HtmlToPlaintextPipe implements PipeTransform export class HtmlToPlaintextPipe implements PipeTransform
{ {
transform(value: string, args: any[] = []) /**
* Transform
*
* @param {string} value
* @param {any[]} args
* @returns {string}
*/
transform(value: string, args: any[] = []): string
{ {
return value ? String(value).replace(/<[^>]+>/gm, '') : ''; return value ? String(value).replace(/<[^>]+>/gm, '') : '';
} }

View File

@ -3,6 +3,13 @@ import { Pipe, PipeTransform } from '@angular/core';
@Pipe({name: 'keys'}) @Pipe({name: 'keys'})
export class KeysPipe implements PipeTransform export class KeysPipe implements PipeTransform
{ {
/**
* Transform
*
* @param value
* @param {string[]} args
* @returns {any}
*/
transform(value: any, args: string[]): any transform(value: any, args: string[]): any
{ {
const keys: any[] = []; const keys: any[] = [];

View File

@ -30,6 +30,7 @@
@import "partials/material"; @import "partials/material";
@import "partials/angular-material-fix"; @import "partials/angular-material-fix";
@import "partials/typography"; @import "partials/typography";
@import "partials/docs";
@import "partials/page-layouts"; @import "partials/page-layouts";
@import "partials/cards"; @import "partials/cards";
@import "partials/navigation"; @import "partials/navigation";

View File

@ -85,3 +85,8 @@ mat-chip {
min-height: 0 !important; min-height: 0 !important;
} }
} }
// Fix: Mat-card-image requires a bigger width than 100%
.mat-card-image {
max-width: none !important;
}

View File

@ -227,6 +227,35 @@ $matColorHues: 50, 100, 200, 300, 400, 500, 600, 700, 800, 900, A100, A200, A400
} }
} }
@mixin generateFuseColorClasses($primary, $accent, $warn) {
$colorMap: (
primary: $primary,
accent: $accent,
warn: $warn
);
// Generate the color classes...
@each $name, $map in $colorMap {
@each $hue in $matColorHues {
$color: map-get($map, $hue);
$contrastColor: map-get(map-get($map, 'contrast'), $hue);
@if ($color != null and $contrastColor != null) {
@include generateColorClasses($name, $color, $contrastColor, '-#{$hue}');
// Run the generator one more time for default values (500)
@if ($hue == 500) {
@include generateColorClasses($name, $color, $contrastColor, '');
}
}
}
}
}
// Generate the color classes... // Generate the color classes...
@each $colorName, $colorMap in $matColorsMap { @each $colorName, $colorMap in $matColorsMap {

View File

@ -0,0 +1,42 @@
.docs {
font-size: 16px;
> .content {
max-width: 980px;
> .main-title {
&:first-child {
margin-top: 0;
}
}
}
.main-title {
display: flex;
margin-top: 72px;
font-size: 24px;
}
.section-title {
display: inline-flex;
font-size: 18px;
margin-top: 24px;
border-bottom: 1px solid #F44336;
color: #F44336;
}
ol,
ul {
padding-left: 24px;
li {
margin-bottom: 12px;
line-height: 1.7;
}
}
p {
line-height: 1.7;
}
}

View File

@ -1,6 +1,13 @@
html,
body { body {
display: flex;
> mat-sidenav-container { flex: 1 0 auto;
height: 100%; width: 100%;
} height: 100%;
max-height: 100%;
min-height: 100%;
margin: 0;
padding: 0;
overflow: hidden;
background: #F5F5F5;
} }

View File

@ -58,7 +58,7 @@
transition: opacity 0.2s ease-in-out 0.1s; transition: opacity 0.2s ease-in-out 0.1s;
margin-left: 8px; margin-left: 8px;
+ .collapse-arrow { + .collapsable-arrow {
margin-left: 8px; margin-left: 8px;
} }
} }
@ -72,14 +72,9 @@
} }
&.active { &.active {
background-color: mat-color($accent);
.mat-ripple-element { .nav-link-icon {
background-color: mat-color($accent, default-contrast, 0.1); opacity: 1;
}
&, .nav-link-icon {
color: mat-color($accent, default-contrast);
} }
.nav-link-badge { .nav-link-badge {
@ -90,10 +85,11 @@
.nav-link-icon { .nav-link-icon {
margin-right: 16px; margin-right: 16px;
opacity: 0.7;
} }
.nav-link-icon, .nav-link-icon,
.collapse-arrow { .collapsable-arrow {
font-size: 16px; font-size: 16px;
width: 16px; width: 16px;
height: 16px; height: 16px;
@ -103,7 +99,7 @@
} }
} }
&.nav-collapse { &.nav-collapsable {
display: block; display: block;
> .children { > .children {
@ -135,7 +131,7 @@
> .nav-item { > .nav-item {
&.nav-collapse { &.nav-collapsable {
background: transparent; background: transparent;
transition: background 200ms ease-in-out; transition: background 200ms ease-in-out;
@ -149,7 +145,7 @@
> .group-items { > .group-items {
> .nav-collapse { > .nav-collapsable {
background: transparent; background: transparent;
transition: background 200ms ease-in-out; transition: background 200ms ease-in-out;
@ -176,7 +172,7 @@
.nav-item { .nav-item {
&.nav-collapse { &.nav-collapsable {
position: relative; position: relative;
.children { .children {
@ -205,13 +201,13 @@
height: 56px; height: 56px;
} }
&.nav-collapse { &.nav-collapsable {
position: relative; position: relative;
> .nav-link { > .nav-link {
height: 56px; height: 56px;
.collapse-arrow { .collapsable-arrow {
display: none; display: none;
} }
} }

View File

@ -10,13 +10,9 @@ $header-height-sm: 100px !default;
$carded-header-height-without-toolbar: $carded-header-height - $carded-toolbar-height; $carded-header-height-without-toolbar: $carded-header-height - $carded-toolbar-height;
$carded-header-height-without-toolbar-sm: $carded-header-height-sm - $carded-toolbar-height; $carded-header-height-without-toolbar-sm: $carded-header-height-sm - $carded-toolbar-height;
// Top bg image
$top-bg-image: url('assets/images/backgrounds/header-bg.png');
.page-layout { .page-layout {
position: relative; position: relative;
overflow-x: hidden; overflow: hidden;
overflow-y: auto;
// Carded layout // Carded layout
&.carded { &.carded {
@ -27,15 +23,13 @@ $top-bg-image: url('assets/images/backgrounds/header-bg.png');
min-width: 100%; min-width: 100%;
// Top bg // Top bg
.top-bg { > .top-bg {
position: absolute; position: absolute;
z-index: 1; z-index: 1;
top: 0; top: 0;
right: 0; right: 0;
left: 0; left: 0;
height: $carded-header-height; height: $carded-header-height;
background-image: $top-bg-image;
background-size: cover;
@include media-breakpoint-down('sm') { @include media-breakpoint-down('sm') {
height: $carded-header-height-sm; height: $carded-header-height-sm;
@ -45,48 +39,39 @@ $top-bg-image: url('assets/images/backgrounds/header-bg.png');
// Fullwidth // Fullwidth
&.fullwidth { &.fullwidth {
// Single scroll
&.single-scroll {
> .center {
flex: 1 0 auto;
max-height: none;
}
}
// Center // Center
> .center { > .center {
display: flex; display: flex;
flex-direction: column; flex-direction: column;
flex: 1; flex: 1 0 auto;
position: relative; position: relative;
z-index: 2; z-index: 2;
padding: 0 32px; padding: 0 32px;
width: 100%; width: 100%;
min-width: 100%; min-width: 0;
max-width: 100%; max-width: 100%;
max-height: 100%; max-height: 100%;
.header { > .header {
height: $carded-header-height-without-toolbar; height: $carded-header-height-without-toolbar !important;
min-height: $carded-header-height-without-toolbar; min-height: $carded-header-height-without-toolbar !important;
max-height: $carded-header-height-without-toolbar; max-height: $carded-header-height-without-toolbar !important;
@include media-breakpoint-down('sm') { @include media-breakpoint-down('sm') {
height: $carded-header-height-without-toolbar-sm; height: $carded-header-height-without-toolbar-sm !important;
min-height: $carded-header-height-without-toolbar-sm; min-height: $carded-header-height-without-toolbar-sm !important;
max-height: $carded-header-height-without-toolbar-sm; max-height: $carded-header-height-without-toolbar-sm !important;
} }
} }
.content-card { > .content-card {
display: flex; display: flex;
flex-direction: column; flex-direction: column;
flex: 1; flex: 1 0 auto;
overflow: hidden; overflow: hidden;
@include mat-elevation(7); @include mat-elevation(7);
.toolbar { > .toolbar {
display: flex; display: flex;
justify-content: flex-start; justify-content: flex-start;
align-items: center; align-items: center;
@ -97,128 +82,7 @@ $top-bg-image: url('assets/images/backgrounds/header-bg.png');
} }
> .content { > .content {
display: flex; flex: 1 0 auto;
flex: 1;
overflow: auto;
}
}
}
}
// Left sidenav - Right sidenav
&.left-sidenav,
&.right-sidenav {
// Single scroll
&.single-scroll {
> mat-sidenav-container {
flex: 1 0 auto;
}
}
> mat-sidenav-container {
display: flex;
flex: 1;
background: none;
z-index: 2;
width: 100%;
.sidenav {
display: flex;
flex-direction: column;
flex: 1;
width: 240px;
min-width: 240px;
max-width: 240px;
height: auto;
z-index: 4;
overflow-y: hidden;
@include mat-elevation(7);
&.mat-is-locked-open {
background: none;
box-shadow: none;
}
.header {
height: $carded-header-height;
min-height: $carded-header-height;
max-height: $carded-header-height;
@include media-breakpoint-down('sm') {
height: $carded-header-height-sm;
min-height: $carded-header-height-sm;
max-height: $carded-header-height-sm;
}
}
.content {
background: transparent;
overflow: auto;
}
}
> .mat-sidenav-content,
> .mat-drawer-content {
display: flex;
flex: 1;
height: auto;
overflow: visible;
// Center
.center {
display: flex;
flex-direction: column;
flex: 1;
position: relative;
z-index: 3;
margin-left: 32px;
margin-right: 32px;
.header {
display: flex;
height: $carded-header-height-without-toolbar;
min-height: $carded-header-height-without-toolbar;
max-height: $carded-header-height-without-toolbar;
@include media-breakpoint-down('sm') {
height: $carded-header-height-without-toolbar-sm;
min-height: $carded-header-height-without-toolbar-sm;
max-height: $carded-header-height-without-toolbar-sm;
}
}
.content-card {
display: flex;
flex-direction: column;
flex: 1;
overflow: hidden;
@include mat-elevation(7);
.toolbar {
display: flex;
justify-content: flex-start;
align-items: center;
flex: 1;
border-bottom: 1px solid rgba(0, 0, 0, 0.12);
height: $carded-toolbar-height;
min-height: $carded-toolbar-height;
max-height: $carded-toolbar-height;
.sidenav-toggle {
margin: 0 8px 0 0 !important;
padding: 0 !important;
border-radius: 0;
}
}
.content {
display: flex;
flex: 1;
overflow: auto;
}
}
} }
} }
} }
@ -226,51 +90,82 @@ $top-bg-image: url('assets/images/backgrounds/header-bg.png');
// Tabbed // Tabbed
&.tabbed { &.tabbed {
> mat-sidenav-container { > .center {
width: 100%;
min-width: 0;
> .mat-sidenav-content, > .header {
> .mat-drawer-content { flex: 1 1 auto;
width: calc(100% - 240px); }
min-width: 0;
.center { > .content-card {
width: calc(100% - 32px);
min-width: 0;
@include media-breakpoint-down('md') { > .content {
width: calc(100% - 64px); display: flex;
}
.header { > .mat-tab-group {
flex: 1; overflow: hidden;
}
.content-card { .mat-tab-header {
.content { .mat-tab-label {
height: 64px;
}
}
.mat-tab-group { .mat-tab-body {
overflow: hidden;
.mat-tab-body-content {
overflow: hidden; overflow: hidden;
.mat-tab-header { .tab-content {
position: relative;
.mat-tab-label { width: 100%;
height: 64px; height: 100%;
}
} }
}
}
}
}
}
}
}
.mat-tab-body { // Inner scroll
overflow: hidden; &.inner-scroll {
flex: 1 1 auto;
.mat-tab-body-content { > .center {
overflow: hidden; flex: 1 1 auto;
.tab-content { > .content-card {
position: relative; flex: 1 1 auto;
width: 100%;
height: 100%; > .content {
overflow: auto; overflow: auto;
} flex: 1 1 auto;
}
}
}
// Tabbed
&.tabbed {
> .center {
> .content-card {
> .content {
> .mat-tab-group {
.mat-tab-body {
.mat-tab-body-content {
.tab-content {
overflow: auto;
} }
} }
} }
@ -282,21 +177,212 @@ $top-bg-image: url('assets/images/backgrounds/header-bg.png');
} }
} }
// Left sidenav // Left / Right sidebar
&.left-sidenav { &.left-sidebar,
&.right-sidebar {
flex-direction: row;
// Sidenav // Sidebar
> mat-sidenav-container { > .sidebar {
display: flex;
flex-direction: column;
flex: 1 1 auto;
width: 240px;
min-width: 240px;
max-width: 240px;
height: auto;
overflow: hidden;
@include mat-elevation(7);
.sidenav { &.locked-open {
background: none;
box-shadow: none;
&.mat-is-locked-open { + .center {
z-index: 1001;
}
~ .mat-sidenav-content, &.left-positioned {
~ .mat-drawer-content {
.center { + .center {
margin-left: 0; margin-left: 0;
}
}
&.right-positioned {
+ .center {
margin-right: 0;
}
}
}
.header {
height: $carded-header-height;
min-height: $carded-header-height;
max-height: $carded-header-height;
@include media-breakpoint-down('sm') {
height: $carded-header-height-sm;
min-height: $carded-header-height-sm;
max-height: $carded-header-height-sm;
}
}
.content {
background: transparent;
flex: 1 1 auto;
}
}
// Center
> .center {
display: flex;
flex-direction: column;
flex: 1 1 auto;
position: relative;
z-index: 3;
margin-left: 32px;
margin-right: 32px;
min-width: 0;
> .header {
display: flex;
height: $carded-header-height-without-toolbar;
min-height: $carded-header-height-without-toolbar;
max-height: $carded-header-height-without-toolbar;
@include media-breakpoint-down('sm') {
height: $carded-header-height-without-toolbar-sm;
min-height: $carded-header-height-without-toolbar-sm;
max-height: $carded-header-height-without-toolbar-sm;
}
}
> .content-card {
display: flex;
flex-direction: column;
flex: 1 1 auto;
overflow: hidden;
@include mat-elevation(7);
> .toolbar {
display: flex;
justify-content: flex-start;
align-items: center;
flex: 1 1 auto;
border-bottom: 1px solid rgba(0, 0, 0, 0.12);
height: $carded-toolbar-height;
min-height: $carded-toolbar-height;
max-height: $carded-toolbar-height;
.sidebar-toggle {
margin: 0 8px 0 0 !important;
padding: 0 !important;
border-radius: 0;
}
}
> .content {
flex: 1 1 auto;
}
}
}
// Tabbed
&.tabbed {
> .center {
width: calc(100% - 32px);
min-width: 0;
@include media-breakpoint-down('md') {
width: calc(100% - 64px);
}
> .header {
flex: 1 1 auto;
}
> .content-card {
> .content {
display: flex;
> .mat-tab-group {
overflow: hidden;
.mat-tab-header {
.mat-tab-label {
height: 64px;
}
}
.mat-tab-body {
overflow: hidden;
.mat-tab-body-content {
overflow: hidden;
.tab-content {
position: relative;
width: 100%;
height: 100%;
}
}
}
}
}
}
}
}
// Inner scroll
&.inner-scroll {
flex: 1 1 auto;
> .sidebar {
.content {
overflow: auto;
}
}
> .center {
flex: 1 1 auto;
> .content-card {
flex: 1 1 auto;
> .content {
flex: 1 1 auto;
overflow: auto;
}
}
}
// Tabbed
&.tabbed {
> .center {
> .content-card {
> .content {
> .mat-tab-group {
.mat-tab-body {
.mat-tab-body-content {
.tab-content {
overflow: auto;
}
}
}
}
} }
} }
} }
@ -304,26 +390,15 @@ $top-bg-image: url('assets/images/backgrounds/header-bg.png');
} }
} }
// Right sidenav // Right sidebar specific
&.right-sidenav { &.right-sidebar {
// Sidenav > .sidebar {
> mat-sidenav-container { order: 2;
}
.sidenav { > .center {
order: 999; order: 1;
&.mat-is-locked-open {
~ .mat-sidenav-content,
~ .mat-drawer-content {
.center {
margin-right: 0;
}
}
}
}
} }
} }
} }
@ -336,20 +411,17 @@ $top-bg-image: url('assets/images/backgrounds/header-bg.png');
width: 100%; width: 100%;
min-width: 100%; min-width: 100%;
// Top bg
> .header {
background-image: $top-bg-image;
background-size: cover;
}
// Fullwidth // Fullwidth
&.fullwidth { &.fullwidth {
overflow: auto;
> .content {
flex: 1 1 auto;
min-width: 0;
}
} }
&.fullwidth, &.fullwidth,
&.inner-sidenav { &.inner-sidebar {
min-height: 100%;
> .header { > .header {
height: $header-height; height: $header-height;
@ -358,109 +430,138 @@ $top-bg-image: url('assets/images/backgrounds/header-bg.png');
} }
} }
// Left sidenav - Right sidenav // Left / Right sidebar
&.left-sidenav, &.left-sidebar,
&.right-sidenav { &.right-sidebar {
flex-direction: row;
// Single scroll // Sidebar
&.single-scroll { > .sidebar {
width: 240px;
min-width: 240px;
max-width: 240px;
overflow: hidden;
@include mat-elevation(7);
> mat-sidenav-container { &.locked-open {
flex: 1 0 auto; background: none;
box-shadow: none;
> .mat-sidenav-content, + .center {
> .mat-drawer-content { z-index: 1001;
flex: 1 0 auto;
max-height: none;
} }
}
}
// Inner Sidenav &.left-positioned {
&.inner-sidenav {
> mat-sidenav-container { + .center {
flex: 1; margin-left: 0;
.sidenav {
.sidenav-content {
height: 100%;
} }
} }
> .mat-sidenav-content, &.right-positioned {
> .mat-drawer-content {
display: flex;
height: auto;
.center { + .center {
flex: 1; margin-right: 0;
min-height: 100%;
@include mat-elevation(0);
.content {
display: flex;
flex: 1 0 auto;
}
} }
} }
} }
.content {
flex: 1 1 auto;
}
} }
> mat-sidenav-container { // Center
> .center {
position: relative;
display: flex; display: flex;
flex-direction: column; flex-direction: column;
flex: 1; flex: 1 1 auto;
background: none; z-index: 3;
z-index: 2; min-width: 0;
width: 100%; @include mat-elevation(7);
.sidenav { > .header {
width: 240px; height: $header-height;
min-width: 240px; min-height: $header-height;
max-width: 240px; max-height: $header-height;
z-index: 51; }
@include mat-elevation(7);
&.mat-is-locked-open { > .content {
width: 220px; flex: 1 0 auto;
min-width: 220px; }
max-width: 220px; }
box-shadow: none;
background: transparent;
}
.sidenav-content { // Inner scroll
height: 100%; &.inner-scroll {
flex: 1 1 auto;
> .sidebar {
.content {
overflow: auto;
} }
} }
> .mat-sidenav-content, > .center {
> .mat-drawer-content { flex: 1 1 auto;
overflow: auto;
}
}
// Inner Sidebar
&.inner-sidebar {
flex-direction: column;
overflow: hidden;
height: 100%;
> .content {
display: flex; display: flex;
flex: 1; min-height: 0;
height: auto;
overflow: visible;
max-height: 100%;
.header { > .sidebar {
height: $header-height;
min-height: $header-height;
max-height: $header-height;
background-image: $top-bg-image;
}
.center { &.locked-open {
display: flex; background: none;
flex-direction: column; box-shadow: none;
flex: 1; }
overflow: auto;
@include mat-elevation(7);
.content { .content {
overflow: auto;
} }
} }
> .center {
flex: 1 1 auto;
overflow: auto;
}
}
}
}
// Right sidebar specific
&.right-sidebar {
> .sidebar {
order: 2;
}
> .center {
order: 1;
}
// Inner sidebar
&.inner-sidebar {
> .content {
> .sidebar {
order: 2;
}
> .center {
order: 1;
}
} }
} }
} }
@ -469,7 +570,7 @@ $top-bg-image: url('assets/images/backgrounds/header-bg.png');
&.tabbed { &.tabbed {
min-height: 100%; min-height: 100%;
.header { > .header {
height: $header-height; height: $header-height;
min-height: $header-height; min-height: $header-height;
max-height: $header-height; max-height: $header-height;
@ -477,7 +578,7 @@ $top-bg-image: url('assets/images/backgrounds/header-bg.png');
> .content { > .content {
.mat-tab-group { > .mat-tab-group {
.mat-tab-labels { .mat-tab-labels {
padding: 0 24px; padding: 0 24px;
@ -495,50 +596,6 @@ $top-bg-image: url('assets/images/backgrounds/header-bg.png');
@include media-breakpoint-down('xs') { @include media-breakpoint-down('xs') {
// Activate single-scroll
&.carded {
&.fullwidth {
> .center {
flex: 1 0 auto;
max-height: none;
}
}
&.left-sidenav,
&.right-sidenav {
> mat-sidenav-container {
flex: 1 0 auto;
}
}
}
&.simple {
&.fullwidth {
> .content {
flex: 1 0 auto;
}
}
&.left-sidenav,
&.right-sidenav {
> mat-sidenav-container {
flex: 1 0 auto !important;
> .mat-sidenav-content,
> .mat-drawer-content {
flex: 1 0 auto;
}
}
}
}
// End - Activate single-scroll
// Smaller margins // Smaller margins
&.carded { &.carded {
@ -549,18 +606,11 @@ $top-bg-image: url('assets/images/backgrounds/header-bg.png');
} }
} }
&.left-sidenav, &.left-sidebar,
&.right-sidenav { &.right-sidebar {
> mat-sidenav-container { > .center {
margin: 0 16px;
> .mat-sidenav-content,
> .mat-drawer-content {
.center {
margin: 0 16px;
}
}
} }
} }
} }

View File

@ -34,7 +34,7 @@
} }
/* General styles */ /* General styles */
fuse-root { app {
fuse-navbar-vertical, fuse-navbar-vertical,
fuse-navbar-horizontal, fuse-navbar-horizontal,

View File

@ -275,3 +275,118 @@ strong {
.text-nowrap { .text-nowrap {
white-space: nowrap; white-space: nowrap;
} }
// Changelog
.changelog {
.entry {
background: white;
margin-bottom: 24px;
padding: 24px 32px;
@include mat-elevation(2);
> .title {
display: flex;
align-items: center;
margin-bottom: 24px;
.version {
font-size: 24px;
}
.date {
margin-left: 8px;
font-size: 17px;
opacity: 0.54;
}
}
.groups {
div {
margin-bottom: 32px;
&:last-child {
margin-bottom: 0;
}
}
.title {
display: inline-flex;
font-size: 13px;
color: white;
letter-spacing: 0.015em;
line-height: 1;
padding: 5px 8px;
border-radius: 2px;
}
.breaking-changes {
.title {
background: #F44336;
}
}
.new {
.title {
background: #43A047;
}
}
.improved {
.title {
background: #673AB7;
}
}
.fixed {
.title {
background: #2196F3;
}
}
ul {
padding-left: 24px;
li {
margin-bottom: 6px;
letter-spacing: 0.015em;
}
}
}
}
}
// Message boxes
.message-box {
padding: 16px;
background: #607D8B;
border-left: 6px solid #37474F;
color: rgba(255, 255, 255, 1);
&.error {
background: #EF5350;
border-left-color: #B71C1C;
}
&.warning {
background: #FFECB3;
border-left-color: #FFC107;
color: rgba(0, 0, 0, 0.87);
}
&.success {
background: #4CAF50;
border-left-color: #2E7D32;
}
&.info {
background: #B3E5FC;
border-left-color: #03A9F4;
color: rgba(0, 0, 0, 0.87);
}
}

View File

@ -104,9 +104,9 @@ $code-color-attr-name: #B65611 !default;
// whitespace management // whitespace management
white-space: pre; // fallback white-space: pre; // fallback
white-space: pre-wrap; //white-space: pre-wrap;
word-break: break-all; //word-break: break-all;
word-wrap: break-word; //word-wrap: break-word;
font-family: $code-font-family; font-family: $code-font-family;
font-size: $code-font-size; font-size: $code-font-size;

View File

@ -1,93 +1,155 @@
import { Inject, Injectable, InjectionToken } from '@angular/core'; import { Inject, Injectable, InjectionToken } from '@angular/core';
import { NavigationEnd, NavigationStart, Router } from '@angular/router'; import { NavigationStart, Router } from '@angular/router';
import { Platform } from '@angular/cdk/platform'; import { Platform } from '@angular/cdk/platform';
import { BehaviorSubject } from 'rxjs'; import { BehaviorSubject, Observable } from 'rxjs';
import { filter } from 'rxjs/operators';
import * as _ from 'lodash'; import * as _ from 'lodash';
// Create the injection token for the custom config // Create the injection token for the custom settings
export const FUSE_CONFIG = new InjectionToken('fuseCustomConfig'); export const FUSE_CONFIG = new InjectionToken('fuseCustomConfig');
@Injectable() @Injectable()
export class FuseConfigService export class FuseConfigService
{ {
config: any; // Private
defaultConfig: any; private _configSubject: BehaviorSubject<any>;
isSetConfigRan = false; private readonly _defaultConfig: any;
onConfigChanged: BehaviorSubject<any>;
/** /**
* Constructor * Constructor
* *
* @param router * @param {Platform} _platform
* @param platform * @param {Router} _router
* @param config * @param _config
*/ */
constructor( constructor(
private router: Router, private _platform: Platform,
public platform: Platform, private _router: Router,
@Inject(FUSE_CONFIG) config @Inject(FUSE_CONFIG) private _config
) )
{ {
// Set the default config from the user provided one (forRoot) // Set the default config from the user provided config (from forRoot)
this.defaultConfig = config; this._defaultConfig = _config;
/** // Initialize the service
* Disable Custom Scrollbars if Browser is Mobile this._init();
*/ }
if ( this.platform.ANDROID || this.platform.IOS )
{
this.defaultConfig.customScrollbars = false;
}
// Set the config from the default config // -----------------------------------------------------------------------------------------------------
this.config = _.cloneDeep(this.defaultConfig); // @ Accessors
// -----------------------------------------------------------------------------------------------------
// Reload the default settings for the /**
// layout on every navigation start * Set and get the config
router.events.subscribe( */
(event) => { set config(value)
{
// Get the value from the behavior subject
let config = this._configSubject.getValue();
if ( event instanceof NavigationStart ) // Merge the new config
{ config = _.merge({}, config, value);
this.isSetConfigRan = false;
}
if ( event instanceof NavigationEnd ) // Notify the observers
{ this._configSubject.next(config);
if ( this.isSetConfigRan ) }
{
return;
}
this.setConfig({ get config(): any | Observable<any>
layout: this.defaultConfig.layout {
} return this._configSubject.asObservable();
);
}
}
);
// Create the behavior subject
this.onConfigChanged = new BehaviorSubject(this.config);
} }
/** /**
* Set the new config from given object * Get default config
* *
* @param config * @returns {any}
*/ */
setConfig(config): void get defaultConfig(): any
{ {
// Set the SetConfigRan true return this._defaultConfig;
this.isSetConfigRan = true; }
// Merge the config // -----------------------------------------------------------------------------------------------------
this.config = _.merge({}, this.config, config); // @ Private methods
// -----------------------------------------------------------------------------------------------------
// Trigger the event /**
this.onConfigChanged.next(this.config); * Initialize
*
* @private
*/
private _init(): void
{
/**
* Disable custom scrollbars if browser is mobile
*/
if ( this._platform.ANDROID || this._platform.IOS )
{
this._defaultConfig.customScrollbars = false;
}
// Set the config from the default config
this._configSubject = new BehaviorSubject(_.cloneDeep(this._defaultConfig));
// Reload the default config on every navigation start if
// the current config is different from the default one
this._router.events
.pipe(filter(event => event instanceof NavigationStart))
.subscribe(() => {
if ( !_.isEqual(this._configSubject.getValue(), this._defaultConfig) )
{
// Clone the default config
const config = _.cloneDeep(this._defaultConfig);
// Set the config
this._configSubject.next(config);
}
});
}
// -----------------------------------------------------------------------------------------------------
// @ Public methods
// -----------------------------------------------------------------------------------------------------
/**
* Set config
*
* @param value
* @param {{emitEvent: boolean}} opts
*/
setConfig(value, opts = {emitEvent: true}): void
{
// Get the value from the behavior subject
let config = this._configSubject.getValue();
// Merge the new config
config = _.merge({}, config, value);
// If emitEvent option is true...
if ( opts.emitEvent === true )
{
// Notify the observers
this._configSubject.next(config);
}
}
/**
* Get config
*
* @returns {Observable<any>}
*/
getConfig(): Observable<any>
{
return this._configSubject.asObservable();
}
/**
* Reset to the default config
*/
resetToDefaults(): void
{
// Set the config from the default config
this._configSubject.next(_.cloneDeep(this._defaultConfig));
} }
} }

View File

@ -11,16 +11,20 @@ import { Injectable } from '@angular/core';
@Injectable() @Injectable()
export class FuseCopierService export class FuseCopierService
{ {
private textarea: HTMLTextAreaElement; private textarea: HTMLTextAreaElement;
/** Copy the text value to the clipboard. */ /**
* Copy the text value to the clipboard
*
* @param {string} text
* @returns {boolean}
*/
copyText(text: string): boolean copyText(text: string): boolean
{ {
this.createTextareaAndSelect(text); this.createTextareaAndSelect(text);
const copySuccessful = document.execCommand('copy'); const copySuccessful = document.execCommand('copy');
this.removeFake(); this._removeFake();
return copySuccessful; return copySuccessful;
} }
@ -28,8 +32,10 @@ export class FuseCopierService
/** /**
* Creates a hidden textarea element, sets its value from `text` property, * Creates a hidden textarea element, sets its value from `text` property,
* and makes a selection on it. * and makes a selection on it.
*
* @param {string} text
*/ */
private createTextareaAndSelect(text: string) private createTextareaAndSelect(text: string): void
{ {
// Create a fake element to hold the contents to copy // Create a fake element to hold the contents to copy
this.textarea = document.createElement('textarea'); this.textarea = document.createElement('textarea');
@ -53,8 +59,12 @@ export class FuseCopierService
this.textarea.setSelectionRange(0, this.textarea.value.length); this.textarea.setSelectionRange(0, this.textarea.value.length);
} }
/** Remove the text area from the DOM. */ /**
private removeFake() * Remove the text area from the DOM
*
* @private
*/
private _removeFake(): void
{ {
if ( this.textarea ) if ( this.textarea )
{ {

View File

@ -8,16 +8,42 @@ export class FuseMatchMediaService
activeMediaQuery: string; activeMediaQuery: string;
onMediaChange: BehaviorSubject<string> = new BehaviorSubject<string>(''); onMediaChange: BehaviorSubject<string> = new BehaviorSubject<string>('');
constructor(private observableMedia: ObservableMedia) /**
* Constructor
*
* @param {ObservableMedia} _observableMedia
*/
constructor(
private _observableMedia: ObservableMedia
)
{ {
// Set the defaults
this.activeMediaQuery = ''; this.activeMediaQuery = '';
this.observableMedia.subscribe((change: MediaChange) => { // Initialize
if ( this.activeMediaQuery !== change.mqAlias ) this._init();
{
this.activeMediaQuery = change.mqAlias;
this.onMediaChange.next(change.mqAlias);
}
});
} }
// -----------------------------------------------------------------------------------------------------
// @ Private methods
// -----------------------------------------------------------------------------------------------------
/**
* Initialize
*
* @private
*/
private _init(): void
{
this._observableMedia
.subscribe((change: MediaChange) => {
if ( this.activeMediaQuery !== change.mqAlias )
{
this.activeMediaQuery = change.mqAlias;
this.onMediaChange.next(change.mqAlias);
}
});
}
} }

View File

@ -6,23 +6,45 @@ import { NavigationEnd, Router } from '@angular/router';
@Injectable() @Injectable()
export class FuseSplashScreenService export class FuseSplashScreenService
{ {
splashScreenEl; splashScreenEl: any;
public player: AnimationPlayer; player: AnimationPlayer;
/**
* Constructor
*
* @param {AnimationBuilder} _animationBuilder
* @param _document
* @param {Router} _router
*/
constructor( constructor(
private animationBuilder: AnimationBuilder, private _animationBuilder: AnimationBuilder,
@Inject(DOCUMENT) private document: any, @Inject(DOCUMENT) private _document: any,
private router: Router private _router: Router
) )
{
// Initialize
this._init();
}
// -----------------------------------------------------------------------------------------------------
// @ Private methods
// -----------------------------------------------------------------------------------------------------
/**
* Initialize
*
* @private
*/
private _init(): void
{ {
// Get the splash screen element // Get the splash screen element
this.splashScreenEl = this.document.body.querySelector('#fuse-splash-screen'); this.splashScreenEl = this._document.body.querySelector('#fuse-splash-screen');
// If the splash screen element exists... // If the splash screen element exists...
if ( this.splashScreenEl ) if ( this.splashScreenEl )
{ {
// Hide it on the first NavigationEnd event // Hide it on the first NavigationEnd event
const hideOnLoad = this.router.events.subscribe((event) => { const hideOnLoad = this._router.events.subscribe((event) => {
if ( event instanceof NavigationEnd ) if ( event instanceof NavigationEnd )
{ {
setTimeout(() => { setTimeout(() => {
@ -38,10 +60,17 @@ export class FuseSplashScreenService
} }
} }
show() // -----------------------------------------------------------------------------------------------------
// @ Public methods
// -----------------------------------------------------------------------------------------------------
/**
* Show the splash screen
*/
show(): void
{ {
this.player = this.player =
this.animationBuilder this._animationBuilder
.build([ .build([
style({ style({
opacity: '0', opacity: '0',
@ -55,10 +84,13 @@ export class FuseSplashScreenService
}, 0); }, 0);
} }
hide() /**
* Hide the splash screen
*/
hide(): void
{ {
this.player = this.player =
this.animationBuilder this._animationBuilder
.build([ .build([
style({opacity: '1'}), style({opacity: '1'}),
animate('400ms ease', style({ animate('400ms ease', style({

View File

@ -10,18 +10,34 @@ export interface Locale
@Injectable() @Injectable()
export class FuseTranslationLoaderService export class FuseTranslationLoaderService
{ {
constructor(private translate: TranslateService) /**
* Constructor
*
* @param {TranslateService} _translateService
*/
constructor(
private _translateService: TranslateService
)
{ {
} }
public loadTranslations(...args: Locale[]): void // -----------------------------------------------------------------------------------------------------
// @ Public methods
// -----------------------------------------------------------------------------------------------------
/**
* Load translations
*
* @param {Locale} args
*/
loadTranslations(...args: Locale[]): void
{ {
const locales = [...args]; const locales = [...args];
locales.forEach((locale) => { locales.forEach((locale) => {
// use setTranslation() with the third argument set to true // use setTranslation() with the third argument set to true
// to append translations instead of replacing them // to append translations instead of replacing them
this.translate.setTranslation(locale.lang, locale.data, true); this._translateService.setTranslation(locale.lang, locale.data, true);
}); });
} }
} }

View File

@ -0,0 +1,24 @@
export interface FuseConfig
{
layout: {
style: string,
width: 'fullwidth' | 'boxed',
navbar: {
hidden: boolean,
folded: boolean,
position: 'left' | 'right' | 'top',
background: string
},
toolbar: {
hidden: boolean,
position: 'above' | 'above-static' | 'above-fixed' | 'below' | 'below-static' | 'below-fixed',
background: string
}
footer: {
hidden: boolean,
position: 'above' | 'above-static' | 'above-fixed' | 'below' | 'below-static' | 'below-fixed',
background: string
}
};
customScrollbars: boolean;
}

View File

@ -0,0 +1,26 @@
export interface FuseNavigationItem
{
id: string;
title: string;
type: 'item' | 'group' | 'collapsable';
translate?: string;
icon?: string;
hidden?: boolean;
url?: string;
exactMatch?: boolean;
externalUrl?: boolean;
openInNewTab?: boolean;
function?: any;
badge?: {
title?: string;
translate?: string;
bg?: string;
fg?: string;
};
children?: FuseNavigationItem[];
}
export interface FuseNavigation extends FuseNavigationItem
{
children?: FuseNavigationItem[];
}

2
src/@fuse/types/index.ts Normal file
View File

@ -0,0 +1,2 @@
export * from './fuse-config';
export * from './fuse-navigation';

View File

@ -1,6 +1,13 @@
export class FuseUtils export class FuseUtils
{ {
public static filterArrayByString(mainArr, searchText) /**
* Filter array by string
*
* @param mainArr
* @param searchText
* @returns {any}
*/
public static filterArrayByString(mainArr, searchText): any
{ {
if ( searchText === '' ) if ( searchText === '' )
{ {
@ -14,7 +21,14 @@ export class FuseUtils
}); });
} }
public static searchInObj(itemObj, searchText) /**
* Search in object
*
* @param itemObj
* @param searchText
* @returns {boolean}
*/
public static searchInObj(itemObj, searchText): boolean
{ {
for ( const prop in itemObj ) for ( const prop in itemObj )
{ {
@ -51,7 +65,14 @@ export class FuseUtils
} }
} }
public static searchInArray(arr, searchText) /**
* Search in array
*
* @param arr
* @param searchText
* @returns {boolean}
*/
public static searchInArray(arr, searchText): boolean
{ {
for ( const value of arr ) for ( const value of arr )
{ {
@ -73,14 +94,26 @@ export class FuseUtils
} }
} }
public static searchInString(value, searchText) /**
* Search in string
*
* @param value
* @param searchText
* @returns {any}
*/
public static searchInString(value, searchText): any
{ {
return value.toLowerCase().includes(searchText); return value.toLowerCase().includes(searchText);
} }
public static generateGUID() /**
* Generate a unique GUID
*
* @returns {string}
*/
public static generateGUID(): string
{ {
function S4() function S4(): string
{ {
return Math.floor((1 + Math.random()) * 0x10000) return Math.floor((1 + Math.random()) * 0x10000)
.toString(16) .toString(16)
@ -90,7 +123,13 @@ export class FuseUtils
return S4() + S4(); return S4() + S4();
} }
public static toggleInArray(item, array) /**
* Toggle in array
*
* @param item
* @param array
*/
public static toggleInArray(item, array): void
{ {
if ( array.indexOf(item) === -1 ) if ( array.indexOf(item) === -1 )
{ {
@ -102,7 +141,13 @@ export class FuseUtils
} }
} }
public static handleize(text) /**
* Handleize
*
* @param text
* @returns {string}
*/
public static handleize(text): string
{ {
return text.toString().toLowerCase() return text.toString().toLowerCase()
.replace(/\s+/g, '-') // Replace spaces with - .replace(/\s+/g, '-') // Replace spaces with -

View File

@ -1 +1,29 @@
<fuse-main></fuse-main> <!-- VERTICAL LAYOUT 1 -->
<ng-container *ngIf="fuseConfig.layout.style === 'vertical-layout-1'">
<vertical-layout-1></vertical-layout-1>
</ng-container>
<!-- VERTICAL LAYOUT 2 -->
<ng-container *ngIf="fuseConfig.layout.style === 'vertical-layout-2'">
<vertical-layout-2></vertical-layout-2>
</ng-container>
<!-- VERTICAL LAYOUT 3 -->
<ng-container *ngIf="fuseConfig.layout.style === 'vertical-layout-3'">
<vertical-layout-3></vertical-layout-3>
</ng-container>
<!-- HORIZONTAL LAYOUT 1 -->
<ng-container *ngIf="fuseConfig.layout.style === 'horizontal-layout-1'">
<horizontal-layout-1></horizontal-layout-1>
</ng-container>
<!-- THEME OPTIONS PANEL -->
<button mat-icon-button class="mat-primary-bg mat-elevation-z2 theme-options-button"
(click)="toggleSidebarOpen('themeOptionsPanel')">
<mat-icon>settings</mat-icon>
</button>
<fuse-sidebar name="themeOptionsPanel" class="theme-options-sidebar" position="right" [invisibleOverlay]="true">
<fuse-theme-options></fuse-theme-options>
</fuse-sidebar>

View File

@ -0,0 +1,38 @@
:host {
position: relative;
display: flex;
flex: 1 1 auto;
width: 100%;
height: 100%;
min-width: 0;
.theme-options-button {
position: absolute;
top: 160px;
right: 0;
width: 48px;
height: 48px;
line-height: 48px;
text-align: center;
cursor: pointer;
border-radius: 0;
margin: 0;
pointer-events: auto;
opacity: .75;
z-index: 998;
mat-icon {
animation: rotating 3s linear infinite;
}
&:hover {
opacity: 1;
}
}
.theme-options-sidebar {
width: 360px;
min-width: 360px;
max-width: 360px;
}
}

View File

@ -1,42 +1,113 @@
import { Component } from '@angular/core'; import { Component, OnDestroy, OnInit } from '@angular/core';
import { TranslateService } from '@ngx-translate/core'; import { TranslateService } from '@ngx-translate/core';
import { Subject } from 'rxjs';
import { takeUntil } from 'rxjs/operators';
import { FuseConfigService } from '@fuse/services/config.service';
import { FuseNavigationService } from '@fuse/components/navigation/navigation.service';
import { FuseSidebarService } from '@fuse/components/sidebar/sidebar.service';
import { FuseSplashScreenService } from '@fuse/services/splash-screen.service'; import { FuseSplashScreenService } from '@fuse/services/splash-screen.service';
import { FuseTranslationLoaderService } from '@fuse/services/translation-loader.service'; import { FuseTranslationLoaderService } from '@fuse/services/translation-loader.service';
import { FuseNavigationService } from '@fuse/components/navigation/navigation.service';
import { locale as navigationEnglish } from './navigation/i18n/en'; import { navigation } from 'app/navigation/navigation';
import { locale as navigationTurkish } from './navigation/i18n/tr'; import { locale as navigationEnglish } from 'app/navigation/i18n/en';
import { locale as navigationTurkish } from 'app/navigation/i18n/tr';
@Component({ @Component({
selector : 'fuse-root', selector : 'app',
templateUrl: './app.component.html', templateUrl: './app.component.html',
styleUrls : ['./app.component.scss'] styleUrls : ['./app.component.scss']
}) })
export class AppComponent export class AppComponent implements OnInit, OnDestroy
{ {
navigation: any;
fuseConfig: any;
// Private
private _unsubscribeAll: Subject<any>;
/**
* Constructor
*
* @param {FuseConfigService} _fuseConfigService
* @param {FuseNavigationService} _fuseNavigationService
* @param {FuseSidebarService} _fuseSidebarService
* @param {FuseSplashScreenService} _fuseSplashScreenService
* @param {FuseTranslationLoaderService} _fuseTranslationLoaderService
* @param {TranslateService} _translateService
*/
constructor( constructor(
private translate: TranslateService, private _fuseConfigService: FuseConfigService,
private fuseNavigationService: FuseNavigationService, private _fuseNavigationService: FuseNavigationService,
private fuseSplashScreen: FuseSplashScreenService, private _fuseSidebarService: FuseSidebarService,
private fuseTranslationLoader: FuseTranslationLoaderService private _fuseSplashScreenService: FuseSplashScreenService,
private _fuseTranslationLoaderService: FuseTranslationLoaderService,
private _translateService: TranslateService
) )
{ {
// Get default navigation
this.navigation = navigation;
// Register the navigation to the service
this._fuseNavigationService.register('main', this.navigation);
// Set the main navigation as our current navigation
this._fuseNavigationService.setCurrentNavigation('main');
// Add languages // Add languages
this.translate.addLangs(['en', 'tr']); this._translateService.addLangs(['en', 'tr']);
// Set the default language // Set the default language
this.translate.setDefaultLang('en'); this._translateService.setDefaultLang('en');
// Give Angular some time to finish loading navigation before // Set the navigation translations
// loading navigation translations this._fuseTranslationLoaderService.loadTranslations(navigationEnglish, navigationTurkish);
setTimeout(() => {
// Set the navigation translations // Use a language
this.fuseTranslationLoader.loadTranslations(navigationEnglish, navigationTurkish); this._translateService.use('en');
// Use a language // Set the private defaults
this.translate.use('en'); this._unsubscribeAll = new Subject();
}); }
// -----------------------------------------------------------------------------------------------------
// @ Lifecycle hooks
// -----------------------------------------------------------------------------------------------------
/**
* On init
*/
ngOnInit(): void
{
// Subscribe to config changes
this._fuseConfigService.config
.pipe(takeUntil(this._unsubscribeAll))
.subscribe((config) => {
this.fuseConfig = config;
});
}
/**
* On destroy
*/
ngOnDestroy(): void
{
// Unsubscribe from all subscriptions
this._unsubscribeAll.next();
this._unsubscribeAll.complete();
}
// -----------------------------------------------------------------------------------------------------
// @ Public methods
// -----------------------------------------------------------------------------------------------------
/**
* Toggle sidebar open
*
* @param key
*/
toggleSidebarOpen(key): void
{
this._fuseSidebarService.getSidebar(key).toggleOpen();
} }
} }

View File

@ -3,44 +3,43 @@ import { BrowserModule } from '@angular/platform-browser';
import { HttpClientModule } from '@angular/common/http'; import { HttpClientModule } from '@angular/common/http';
import { BrowserAnimationsModule } from '@angular/platform-browser/animations'; import { BrowserAnimationsModule } from '@angular/platform-browser/animations';
import { RouterModule, Routes } from '@angular/router'; import { RouterModule, Routes } from '@angular/router';
import { MatMomentDateModule } from '@angular/material-moment-adapter';
import { MatButtonModule, MatIconModule } from '@angular/material';
import { InMemoryWebApiModule } from 'angular-in-memory-web-api'; import { InMemoryWebApiModule } from 'angular-in-memory-web-api';
import { TranslateModule } from '@ngx-translate/core'; import { TranslateModule } from '@ngx-translate/core';
import 'hammerjs'; import 'hammerjs';
import { FuseModule } from '@fuse/fuse.module'; import { FuseModule } from '@fuse/fuse.module';
import { FuseSharedModule } from '@fuse/shared.module'; import { FuseSharedModule } from '@fuse/shared.module';
import { FuseSidebarModule, FuseThemeOptionsModule } from '@fuse/components';
import { fuseConfig } from './fuse-config'; import { fuseConfig } from 'app/fuse-config';
import { AppComponent } from './app.component'; import { FakeDbService } from 'app/fake-db/fake-db.service';
import { FuseFakeDbService } from './fuse-fake-db/fuse-fake-db.service'; import { AppComponent } from 'app/app.component';
import { FuseMainModule } from './main/main.module'; import { AppStoreModule } from 'app/store/store.module';
import { AppStoreModule } from './store/store.module'; import { LayoutModule } from 'app/layout/layout.module';
const appRoutes: Routes = [ const appRoutes: Routes = [
{ {
path : 'apps', path : 'apps',
loadChildren: './main/content/apps/apps.module#FuseAppsModule' loadChildren: './main/apps/apps.module#AppsModule'
}, },
{ {
path : 'pages', path : 'pages',
loadChildren: './main/content/pages/pages.module#FusePagesModule' loadChildren: './main/pages/pages.module#PagesModule'
}, },
{ {
path : 'ui', path : 'ui',
loadChildren: './main/content/ui/ui.module#FuseUIModule' loadChildren: './main/ui/ui.module#UIModule'
}, },
{ {
path : 'services', path : 'documentation',
loadChildren: './main/content/services/services.module#FuseServicesModule' loadChildren: './main/documentation/documentation.module#DocumentationModule'
}, },
{ {
path : 'components', path : 'angular-material-elements',
loadChildren: './main/content/components/components.module#FuseComponentsModule' loadChildren: './main/angular-material-elements/angular-material-elements.module#AngularMaterialElementsModule'
},
{
path : 'components-third-party',
loadChildren: './main/content/components-third-party/components-third-party.module#FuseComponentsThirdPartyModule'
}, },
{ {
path : '**', path : '**',
@ -59,17 +58,27 @@ const appRoutes: Routes = [
RouterModule.forRoot(appRoutes), RouterModule.forRoot(appRoutes),
TranslateModule.forRoot(), TranslateModule.forRoot(),
InMemoryWebApiModule.forRoot(FuseFakeDbService, { InMemoryWebApiModule.forRoot(FakeDbService, {
delay : 0, delay : 0,
passThruUnknownUrl: true passThruUnknownUrl: true
}), }),
// Fuse Main and Shared modules // Material moment date module
MatMomentDateModule,
// Material
MatButtonModule,
MatIconModule,
// Fuse modules
FuseModule.forRoot(fuseConfig), FuseModule.forRoot(fuseConfig),
FuseSharedModule, FuseSharedModule,
FuseSidebarModule,
FuseThemeOptionsModule,
AppStoreModule, // App modules
FuseMainModule LayoutModule,
AppStoreModule
], ],
bootstrap : [ bootstrap : [
AppComponent AppComponent

View File

@ -1,27 +1,27 @@
import { InMemoryDbService } from 'angular-in-memory-web-api'; import { InMemoryDbService } from 'angular-in-memory-web-api';
import { ProjectDashboardDb } from './dashboard-project'; import { ProjectDashboardDb } from 'app/fake-db/dashboard-project';
import { AnalyticsDashboardDb } from './dashboard-analytics'; import { AnalyticsDashboardDb } from 'app/fake-db/dashboard-analytics';
import { CalendarFakeDb } from './calendar'; import { CalendarFakeDb } from 'app/fake-db/calendar';
import { ECommerceFakeDb } from './e-commerce'; import { ECommerceFakeDb } from 'app/fake-db/e-commerce';
import { AcademyFakeDb } from './academy'; import { AcademyFakeDb } from 'app/fake-db/academy';
import { MailFakeDb } from './mail'; import { MailFakeDb } from 'app/fake-db/mail';
import { ChatFakeDb } from './chat'; import { ChatFakeDb } from 'app/fake-db/chat';
import { FileManagerFakeDb } from './file-manager'; import { FileManagerFakeDb } from 'app/fake-db/file-manager';
import { ContactsFakeDb } from './contacts'; import { ContactsFakeDb } from 'app/fake-db/contacts';
import { TodoFakeDb } from './todo'; import { TodoFakeDb } from 'app/fake-db/todo';
import { ScrumboardFakeDb } from './scrumboard'; import { ScrumboardFakeDb } from 'app/fake-db/scrumboard';
import { InvoiceFakeDb } from './invoice'; import { InvoiceFakeDb } from 'app/fake-db/invoice';
import { ProfileFakeDb } from './profile'; import { ProfileFakeDb } from 'app/fake-db/profile';
import { SearchFakeDb } from './search'; import { SearchFakeDb } from 'app/fake-db/search';
import { FaqFakeDb } from './faq'; import { FaqFakeDb } from 'app/fake-db/faq';
import { KnowledgeBaseFakeDb } from './knowledge-base'; import { KnowledgeBaseFakeDb } from 'app/fake-db/knowledge-base';
import { IconsFakeDb } from './icons'; import { IconsFakeDb } from 'app/fake-db/icons';
import { QuickPanelFakeDb } from './quick-panel'; import { QuickPanelFakeDb } from 'app/fake-db/quick-panel';
export class FuseFakeDbService implements InMemoryDbService export class FakeDbService implements InMemoryDbService
{ {
createDb() createDb(): any
{ {
return { return {
// Dashboards // Dashboards

Some files were not shown because too many files have changed in this diff Show More