Angular Material sidenav 指令在 app.component 中不起作用
Angular Material sidenav directive does not work in app.component
我有一个带有 Material sidenav 的应用程序,sidenav 显示了一个包含 links 的列表。其中之一 link 有一个指令,如果用户权限不符合预期的权限,则将其隐藏。如果我在我的 header 组件中使用列表,一切正常,但在我的应用程序组件中,指令在应用程序重新加载之前不起作用。我该如何解决?
app.component.ts
import { MatSidenav } from "@angular/material";
import { ViewChild, ViewEncapsulation, OnChanges } from "@angular/core";
import { Component, AfterViewInit, HostBinding, OnInit } from "@angular/core";
import {
Router,
NavigationStart,
NavigationEnd,
NavigationCancel
} from "@angular/router";
import { OverlayContainer } from "@angular/cdk/overlay";
import { CustomizeService } from "./services/shared-services/customize.service";
import { FocusMonitor } from "@angular/cdk/a11y";
import { SidenavService } from "./services/layout-services/sidenav.service";
@Component({
selector: "app-root",
templateUrl: "./app.component.html",
styleUrls: ["./app.component.css"]
})
export class AppComponent implements AfterViewInit, OnInit {
loading;
customTheme;
// theme: string;
@HostBinding("class") componentCssClass;
@ViewChild("sidenav") public sidenav: MatSidenav;
constructor(
private router: Router,
private sidenavService: SidenavService,
private focusMonitor: FocusMonitor,
public overlayContainer: OverlayContainer,
private customizeService: CustomizeService
) {
this.loading = true;
}
ngOnInit() {
this.getTheme();
this.sidenavService.setSidenav(this.sidenav);
}
ngAfterViewInit() {
this.focusMonitor.stopMonitoring(document.getElementById("clienti"));
this.router.events.subscribe(event => {
if (event instanceof NavigationStart) {
this.loading = true;
} else if (
event instanceof NavigationEnd ||
event instanceof NavigationCancel
) {
this.loading = false;
}
});
}
getTheme() {
this.customizeService.getTheme().subscribe(theme => {
this.overlayContainer.getContainerElement().classList.add(theme);
this.componentCssClass = theme || this.getLogoFromLocalStorage();
});
// this.sendTheme(theme);
}
getLogoFromLocalStorage() {
return localStorage.getItem("customStyle");
}
closeSidenav() {
this.sidenavService.close();
}
}
app.component.html
<div color="primary">
<mat-sidenav-container class="example-container" (backdropClick)="closeSidenav()">
<mat-sidenav #sidenav (keydown.escape)="closeSidenav()" disableClose>
<mat-list>
<mat-list-item (click)="closeSidenav()"><button id="clienti" mat-button [routerLink]="['/clienti']"
routerLinkActive="mat-accent">Clienti</button></mat-list-item>
<mat-list-item (click)="closeSidenav()"><button mat-button [routerLink]="['/utenti']" routerLinkActive="mat-accent">Utenti</button></mat-list-item>
<!-- <mat-list-item (click)="closeSidenav()"><butto [routerLink]="['/matricole']" routerLinkActive="mat-accent">Matricole</mat-list-item> -->
<mat-list-item (click)="closeSidenav()"><button mat-button [routerLink]="['/sks']" routerLinkActive="mat-accent">Sks</button></mat-list-item>
<mat-list-item (click)="closeSidenav()" [appCheckPermissions]="9">
<button mat-button [routerLink]="['/packs']" routerLinkActive="mat-accent">Pacchetti</button>
</mat-list-item>
<mat-list-item (click)="closeSidenav()"><button mat-button [routerLink]="['/pc']" routerLinkActive="mat-accent">Pc</button></mat-list-item>
<mat-list-item (click)="closeSidenav()"><button mat-button [routerLink]="['/rinnovi']" routerLinkActive="mat-accent">Rinnovi</button></mat-list-item>
</mat-list>
</mat-sidenav>
<mat-sidenav-content>
<app-header></app-header>
<div class="main" color="primary">
<div [hidden]="!loading" class="loader">
<app-progress-spinner></app-progress-spinner>
</div>
<div [hidden]="loading" class="router-output">
<router-outlet></router-outlet>
</div>
</div>
<app-footer></app-footer>
</mat-sidenav-content>
</mat-sidenav-container>
</div>
check-permission.directive.ts
import { Directive, ElementRef, Renderer2, HostListener, HostBinding, Input, OnInit, TemplateRef, ViewContainerRef } from '@angular/core';
import { DataService } from '../services/shared-services/data.service';
@Directive({
selector: '[appCheckPermissions]'
})
export class CheckPermissionsDirective {
permissions: number[] = [];
constructor(
private data: DataService,
private element: ElementRef,
private renderer: Renderer2
) {
this.getPerms();
}
@Input() set appCheckPermissions(perm: number) {
if (this.permissions.includes(perm)) {
// console.log('ok');
} else {
// console.log('non autorizzato');
this.renderer.setStyle(this.element.nativeElement, 'display', 'none');
}
}
getPerms() {
this.data.getPermissionsFromToken().subscribe(permsArr => {
this.permissions = permsArr;
});
}
}
编辑:permissions
和_perm
的日志
刷新前登录:
刷新后:
编辑 2
感谢 Sunil Singh 帮助我找到了问题:当我注销然后再次登录时 permissions
数据不可用,app.component
需要 appCheckPermission
指令不是 re-initialized,因此在浏览器刷新之前它会使用旧的权限数据。
解决方法是 service-based,我不知道如何使用指令修复(Sunil Singh 方法不起作用)。所以我写了一个 permissionService
:
permissions = [];
constructor(private data: DataService) {
this.getPerms();
}
getPerms() {
this.data.getPermissionsFromToken().subscribe(permsArr => {
this.permissions = permsArr;
});
}
isPacksManagerEnable() {
return this.permissions.includes(9);
}
单击菜单按钮时会触发 getPerms()
方法,然后在 app.component
中注入服务并在模板上调用 isPacksManagerEnable()
:
<mat-list-item (click)="closeSidenav()" *ngIf="permsService.isPacksManagerEnable()">
<button mat-button [routerLink]="['/packs']" routerLinkActive="mat-accent">Pacchetti</button>
</mat-list-item>
希望对遇到同样问题的人有所帮助
Issue
指令中的 permissions
和 appCheckPermissions
有问题。您正在检查当时不可用的异步数据 permissions
的权限。加载 permissions
后,您不会触发负责隐藏和显示 link.
的 setter appCheckPermissions
Fix
它有简单的修复,您需要确保每当 permissions
发生变化时,它应该触发权限检查。
修改后的版本
检查-permission.directive.ts
import { Directive, ElementRef, Renderer2, HostListener, HostBinding, Input, OnInit, TemplateRef, ViewContainerRef, AfterViewInit } from '@angular/core';
import { DataService } from '../services/shared-services/data.service';
@Directive({
selector: '[appCheckPermissions]'
})
export class CheckPermissionsDirective implements AfterViewInit {
permissions: number[] = [];
constructor(
private data: DataService,
private element: ElementRef,
private renderer: Renderer2
) {
//this.getPerms();
}
private _perm;
@Input() set appCheckPermissions(perm: number) {
this._perm = perm;
this.doPermissionCheck();
}
doPermissionCheck(){
if (this.permissions.includes(this._perm)) {
this.renderer.setStyle(this.element.nativeElement, 'display', 'block');
// console.log('ok');
} else {
// console.log('non autorizzato');
this.renderer.setStyle(this.element.nativeElement, 'display', 'none');
}
}
ngAfterViewInit() {
this.data.getPermissionsFromToken().subscribe(permsArr => {
this.permissions = permsArr;
this.doPermissionCheck();
});
}
}
Note : Code is written directly in Whosebug editor
so there could be typo or syntactical errors. Please fix yourself.
我有一个带有 Material sidenav 的应用程序,sidenav 显示了一个包含 links 的列表。其中之一 link 有一个指令,如果用户权限不符合预期的权限,则将其隐藏。如果我在我的 header 组件中使用列表,一切正常,但在我的应用程序组件中,指令在应用程序重新加载之前不起作用。我该如何解决?
app.component.ts
import { MatSidenav } from "@angular/material";
import { ViewChild, ViewEncapsulation, OnChanges } from "@angular/core";
import { Component, AfterViewInit, HostBinding, OnInit } from "@angular/core";
import {
Router,
NavigationStart,
NavigationEnd,
NavigationCancel
} from "@angular/router";
import { OverlayContainer } from "@angular/cdk/overlay";
import { CustomizeService } from "./services/shared-services/customize.service";
import { FocusMonitor } from "@angular/cdk/a11y";
import { SidenavService } from "./services/layout-services/sidenav.service";
@Component({
selector: "app-root",
templateUrl: "./app.component.html",
styleUrls: ["./app.component.css"]
})
export class AppComponent implements AfterViewInit, OnInit {
loading;
customTheme;
// theme: string;
@HostBinding("class") componentCssClass;
@ViewChild("sidenav") public sidenav: MatSidenav;
constructor(
private router: Router,
private sidenavService: SidenavService,
private focusMonitor: FocusMonitor,
public overlayContainer: OverlayContainer,
private customizeService: CustomizeService
) {
this.loading = true;
}
ngOnInit() {
this.getTheme();
this.sidenavService.setSidenav(this.sidenav);
}
ngAfterViewInit() {
this.focusMonitor.stopMonitoring(document.getElementById("clienti"));
this.router.events.subscribe(event => {
if (event instanceof NavigationStart) {
this.loading = true;
} else if (
event instanceof NavigationEnd ||
event instanceof NavigationCancel
) {
this.loading = false;
}
});
}
getTheme() {
this.customizeService.getTheme().subscribe(theme => {
this.overlayContainer.getContainerElement().classList.add(theme);
this.componentCssClass = theme || this.getLogoFromLocalStorage();
});
// this.sendTheme(theme);
}
getLogoFromLocalStorage() {
return localStorage.getItem("customStyle");
}
closeSidenav() {
this.sidenavService.close();
}
}
app.component.html
<div color="primary">
<mat-sidenav-container class="example-container" (backdropClick)="closeSidenav()">
<mat-sidenav #sidenav (keydown.escape)="closeSidenav()" disableClose>
<mat-list>
<mat-list-item (click)="closeSidenav()"><button id="clienti" mat-button [routerLink]="['/clienti']"
routerLinkActive="mat-accent">Clienti</button></mat-list-item>
<mat-list-item (click)="closeSidenav()"><button mat-button [routerLink]="['/utenti']" routerLinkActive="mat-accent">Utenti</button></mat-list-item>
<!-- <mat-list-item (click)="closeSidenav()"><butto [routerLink]="['/matricole']" routerLinkActive="mat-accent">Matricole</mat-list-item> -->
<mat-list-item (click)="closeSidenav()"><button mat-button [routerLink]="['/sks']" routerLinkActive="mat-accent">Sks</button></mat-list-item>
<mat-list-item (click)="closeSidenav()" [appCheckPermissions]="9">
<button mat-button [routerLink]="['/packs']" routerLinkActive="mat-accent">Pacchetti</button>
</mat-list-item>
<mat-list-item (click)="closeSidenav()"><button mat-button [routerLink]="['/pc']" routerLinkActive="mat-accent">Pc</button></mat-list-item>
<mat-list-item (click)="closeSidenav()"><button mat-button [routerLink]="['/rinnovi']" routerLinkActive="mat-accent">Rinnovi</button></mat-list-item>
</mat-list>
</mat-sidenav>
<mat-sidenav-content>
<app-header></app-header>
<div class="main" color="primary">
<div [hidden]="!loading" class="loader">
<app-progress-spinner></app-progress-spinner>
</div>
<div [hidden]="loading" class="router-output">
<router-outlet></router-outlet>
</div>
</div>
<app-footer></app-footer>
</mat-sidenav-content>
</mat-sidenav-container>
</div>
check-permission.directive.ts
import { Directive, ElementRef, Renderer2, HostListener, HostBinding, Input, OnInit, TemplateRef, ViewContainerRef } from '@angular/core';
import { DataService } from '../services/shared-services/data.service';
@Directive({
selector: '[appCheckPermissions]'
})
export class CheckPermissionsDirective {
permissions: number[] = [];
constructor(
private data: DataService,
private element: ElementRef,
private renderer: Renderer2
) {
this.getPerms();
}
@Input() set appCheckPermissions(perm: number) {
if (this.permissions.includes(perm)) {
// console.log('ok');
} else {
// console.log('non autorizzato');
this.renderer.setStyle(this.element.nativeElement, 'display', 'none');
}
}
getPerms() {
this.data.getPermissionsFromToken().subscribe(permsArr => {
this.permissions = permsArr;
});
}
}
编辑:permissions
和_perm
刷新前登录:
刷新后:
编辑 2
感谢 Sunil Singh 帮助我找到了问题:当我注销然后再次登录时 permissions
数据不可用,app.component
需要 appCheckPermission
指令不是 re-initialized,因此在浏览器刷新之前它会使用旧的权限数据。
解决方法是 service-based,我不知道如何使用指令修复(Sunil Singh 方法不起作用)。所以我写了一个 permissionService
:
permissions = [];
constructor(private data: DataService) {
this.getPerms();
}
getPerms() {
this.data.getPermissionsFromToken().subscribe(permsArr => {
this.permissions = permsArr;
});
}
isPacksManagerEnable() {
return this.permissions.includes(9);
}
单击菜单按钮时会触发 getPerms()
方法,然后在 app.component
中注入服务并在模板上调用 isPacksManagerEnable()
:
<mat-list-item (click)="closeSidenav()" *ngIf="permsService.isPacksManagerEnable()">
<button mat-button [routerLink]="['/packs']" routerLinkActive="mat-accent">Pacchetti</button>
</mat-list-item>
希望对遇到同样问题的人有所帮助
Issue
指令中的 permissions
和 appCheckPermissions
有问题。您正在检查当时不可用的异步数据 permissions
的权限。加载 permissions
后,您不会触发负责隐藏和显示 link.
appCheckPermissions
Fix
它有简单的修复,您需要确保每当 permissions
发生变化时,它应该触发权限检查。
修改后的版本
检查-permission.directive.ts
import { Directive, ElementRef, Renderer2, HostListener, HostBinding, Input, OnInit, TemplateRef, ViewContainerRef, AfterViewInit } from '@angular/core';
import { DataService } from '../services/shared-services/data.service';
@Directive({
selector: '[appCheckPermissions]'
})
export class CheckPermissionsDirective implements AfterViewInit {
permissions: number[] = [];
constructor(
private data: DataService,
private element: ElementRef,
private renderer: Renderer2
) {
//this.getPerms();
}
private _perm;
@Input() set appCheckPermissions(perm: number) {
this._perm = perm;
this.doPermissionCheck();
}
doPermissionCheck(){
if (this.permissions.includes(this._perm)) {
this.renderer.setStyle(this.element.nativeElement, 'display', 'block');
// console.log('ok');
} else {
// console.log('non autorizzato');
this.renderer.setStyle(this.element.nativeElement, 'display', 'none');
}
}
ngAfterViewInit() {
this.data.getPermissionsFromToken().subscribe(permsArr => {
this.permissions = permsArr;
this.doPermissionCheck();
});
}
}
Note : Code is written directly in
Whosebug editor
so there could be typo or syntactical errors. Please fix yourself.