bfend
Version:
Admin template base on ng-zorro-antd
294 lines • 24.7 kB
JavaScript
/**
* @fileoverview added by tsickle
* @suppress {checkTypes,extraRequire,missingReturn,uselessCode} checked by tsc
*/
import { Subject, ReplaySubject } from 'rxjs';
import { distinct, filter, map, publishReplay, refCount, scan, tap } from 'rxjs/operators';
import { Injectable } from '@angular/core';
import { BfACLService } from './auth/acl.service';
import * as i0 from "@angular/core";
import * as i1 from "./auth/acl.service";
/**
* @record
*/
export function Menu() { }
if (false) {
/** @type {?} */
Menu.prototype.text;
/** @type {?|undefined} */
Menu.prototype.translate;
/** @type {?|undefined} */
Menu.prototype.group;
/** @type {?|undefined} */
Menu.prototype.link;
/** @type {?|undefined} */
Menu.prototype.externalLink;
/** @type {?|undefined} */
Menu.prototype.target;
/** @type {?|undefined} */
Menu.prototype.icon;
/** @type {?|undefined} */
Menu.prototype.badge;
/** @type {?|undefined} */
Menu.prototype.badge_dot;
/** @type {?|undefined} */
Menu.prototype.badge_status;
/** @type {?|undefined} */
Menu.prototype.hide;
/** @type {?|undefined} */
Menu.prototype.acl;
/** @type {?|undefined} */
Menu.prototype.children;
/** @type {?|undefined} */
Menu.prototype._type;
/** @type {?|undefined} */
Menu.prototype._selected;
/** @type {?|undefined} */
Menu.prototype._open;
/** @type {?|undefined} */
Menu.prototype._depth;
/* Skipping unhandled member: [key: string]: any;*/
}
export class BfMenuService {
/**
* @param {?} aclService
*/
constructor(aclService) {
this.aclService = aclService;
this.data = [];
this.updateSubject = new ReplaySubject();
this.setSubject = new Subject();
this.toggleOpenSubject = new Subject();
this.currentSubject = new Subject();
this.refreshSubject = new Subject();
// 最新的菜单数据
this.menus$ = this.updateSubject.pipe(scan((menus, mutation) => [...mutation(menus)], []), tap(menus => (this.data = menus)), publishReplay(1), refCount());
this.valid$ = this.menus$.pipe(map(menus => menus && menus.length > 0), filter(state => state), // select true
distinct());
// 设置整个菜单数据结构
this.setSubject.pipe(map(this.setMutation.bind(this)))
.subscribe(this.updateSubject);
// 菜单的打开或关闭
this.toggleOpenSubject.pipe(map(this.toggleOpenMutation.bind(this)))
.subscribe(this.updateSubject);
// 当前菜单,一般是和 URL 关联起来的
this.currentSubject.pipe(map(this.setCurrentMutation.bind(this)))
.subscribe(this.updateSubject);
this.refreshSubject.pipe(map(this.refreshMutation.bind(this)))
.subscribe(this.updateSubject);
// 用户权限改变时,重新计算菜单权限
this.aclService.rules$.subscribe(this.refreshSubject);
}
/**
* @param {?} menus
* @param {?} callback
* @return {?}
*/
walk(menus, callback) {
/** @type {?} */
const _walk = (list, parentMenu, depth) => {
for (const item of list) {
callback(item, parentMenu, depth);
if (item.children && item.children.length > 0) {
_walk(item.children, item, depth + 1);
}
else {
item.children = [];
}
}
};
_walk(menus, null, 0);
}
/**
* @param {?} menus
* @return {?}
*/
prune(menus) {
/** @type {?} */
const result = [];
for (const item of menus) {
if (item.hide) {
continue;
}
if (item.children && item.children.length > 0) {
item.children = this.prune(item.children);
if ((item._type === 3 || item.group) && item.children.length === 0) {
continue; // 父菜单和分组如果没有了孩子,自己也消失
}
}
result.push(item);
}
return result;
}
/**
* @param {?} menus
* @return {?}
*/
normalize(menus) {
/** @type {?} */
let i = 1;
this.walk(menus, (item, parent, depth) => {
item.__id = i++;
item.__parent = parent;
item._depth = depth;
// badge
if (item.badge) {
if (item.badge_dot !== true) {
item.badge_dot = false;
}
if (!item.badge_status) {
item.badge_status = 'error';
}
}
item.hide = item.acl && !this.aclService.can(item.acl);
item._type = item.externalLink ? 2 : 1;
if (item.children && item.children.length > 0) {
item._type = 3;
}
});
return this.prune(menus);
}
/**
* @param {?} items
* @return {?}
*/
set(items) {
this.setSubject.next(items);
}
/**
* @param {?} menu
* @return {?}
*/
setCurrent(menu) {
this.currentSubject.next(menu);
}
/**
* @param {?} url
* @return {?}
*/
getByUrl(url) {
/** @type {?} */
let findItem = null;
this.walk(this.data, item => {
item._open = false;
if (!item.link) {
return;
}
if (!findItem && new RegExp(`^${url}\\b`).test(item.link)) {
findItem = item;
}
});
return findItem;
}
/**
* @param {?} url
* @return {?}
*/
getPathByUrl(url) {
/** @type {?} */
let item = null;
this.walk(this.data, (i, parent, depth) => {
if (i.link === url) {
item = i;
}
});
/** @type {?} */
const ret = [];
if (!item) {
return ret;
}
do {
ret.splice(0, 0, item);
item = item.__parent;
} while (item);
return ret;
}
/**
* @param {?} item
* @return {?}
*/
toggleOpen(item) {
this.toggleOpenSubject.next(item);
}
/**
* @param {?} menus
* @return {?}
*/
setMutation(menus) {
return () => {
// 规范化,添加部分计算属性
return this.normalize(menus);
};
}
/**
* @param {?} menu
* @return {?}
*/
toggleOpenMutation(menu) {
return menus => {
this.walk(menus, m => {
if (m.__id !== menu.__id) {
m._open = false;
return;
}
menu._open = !menu._open;
});
return menus;
};
}
/**
* @param {?} menu
* @return {?}
*/
setCurrentMutation(menu) {
return menus => {
/** @type {?} */
let p = menu;
if (p) {
do {
p._open = true;
p = p.__parent;
} while (p);
}
return menus;
};
}
/**
* @return {?}
*/
refreshMutation() {
return menus => {
return this.normalize(menus);
};
}
}
BfMenuService.decorators = [
{ type: Injectable, args: [{
providedIn: 'root'
},] },
];
BfMenuService.ctorParameters = () => [
{ type: BfACLService }
];
/** @nocollapse */ BfMenuService.ngInjectableDef = i0.defineInjectable({ factory: function BfMenuService_Factory() { return new BfMenuService(i0.inject(i1.BfACLService)); }, token: BfMenuService, providedIn: "root" });
if (false) {
/** @type {?} */
BfMenuService.prototype.menus$;
/** @type {?} */
BfMenuService.prototype.data;
/** @type {?} */
BfMenuService.prototype.updateSubject;
/** @type {?} */
BfMenuService.prototype.setSubject;
/** @type {?} */
BfMenuService.prototype.toggleOpenSubject;
/** @type {?} */
BfMenuService.prototype.currentSubject;
/** @type {?} */
BfMenuService.prototype.refreshSubject;
/** @type {?} */
BfMenuService.prototype.valid$;
/** @type {?} */
BfMenuService.prototype.aclService;
}
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"menu.service.js","sourceRoot":"ng://bfend/","sources":["lib/menu.service.ts"],"names":[],"mappings":"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","sourcesContent":["import { Observable, Subject, ReplaySubject } from 'rxjs';\nimport { distinct, filter, map, publishReplay, refCount, scan, tap } from 'rxjs/operators';\nimport { Injectable } from '@angular/core';\nimport { BfACLService, ACLType } from './auth/acl.service';\n\nexport interface Menu {\n  // 文本\n  text: string;\n  // i18n主键\n  translate?: string;\n  // 是否菜单组\n  group?: boolean;\n  // angular 链接\n  link?: string;\n  // 外部链接\n  externalLink?: string;\n  // 链接 target\n  target?: '_blank' | '_self' | '_parent' | '_top';\n  // 图标\n  icon?: string;\n  // 徽标数，展示的数字\n  badge?: number;\n  // 徽标数，显示小红点\n  badge_dot?: boolean;\n  // 徽标数，设置 Badge 颜色 （默认：error）\n  badge_status?: string;\n  // 是否隐藏\n  hide?: boolean;\n  // ACL配置\n  acl?: string | string[] | ACLType;\n  // 二级菜单\n  children?: Menu[];\n  // 菜单类型，无须指定由 Service 自动识别\n  // 1：链接\n  // 2：外部链接\n  // 3：链接（子菜单）\n  _type?: number;\n  // 是否选中\n  _selected?: boolean;\n  // 是否打开\n  _open?: boolean;\n  _depth?: number;\n\n  [key: string]: any;\n}\n\ntype Mutation = (menus: Menu[]) => Menu[];\n\n@Injectable({\n  providedIn: 'root'\n})\nexport class BfMenuService {\n  menus$: Observable<Menu[]>;\n\n  private data: Menu[] = [];\n\n  private updateSubject = new ReplaySubject<Mutation>();\n\n  private setSubject = new Subject<Menu[]>();\n\n  private toggleOpenSubject = new Subject<Menu>();\n\n  private currentSubject = new Subject<Menu>();\n\n  private refreshSubject = new Subject();\n\n  valid$: Observable<boolean>;\n\n  constructor(private aclService: BfACLService) {\n    // 最新的菜单数据\n    this.menus$ = this.updateSubject.pipe(\n      scan((menus: Menu[], mutation: Mutation) => [...mutation(menus)], []),\n      tap(menus => (this.data = menus)),\n      publishReplay(1),\n      refCount()\n    );\n\n    this.valid$ = this.menus$.pipe(\n      map(menus => menus && menus.length > 0),\n      filter(state => state), // select true\n      distinct()\n    );\n\n    // 设置整个菜单数据结构\n    this.setSubject.pipe(map(this.setMutation.bind(this)))\n      .subscribe(this.updateSubject);\n\n    // 菜单的打开或关闭\n    this.toggleOpenSubject.pipe(map(this.toggleOpenMutation.bind(this)))\n      .subscribe(this.updateSubject);\n\n    // 当前菜单，一般是和 URL 关联起来的\n    this.currentSubject.pipe(map(this.setCurrentMutation.bind(this)))\n      .subscribe(this.updateSubject);\n\n    this.refreshSubject.pipe(map(this.refreshMutation.bind(this)))\n      .subscribe(this.updateSubject);\n\n    // 用户权限改变时，重新计算菜单权限\n    this.aclService.rules$.subscribe(this.refreshSubject);\n  }\n\n  private walk(menus, callback: (item: Menu, parentMenu: Menu, depth?: number) => void) {\n    const _walk = (list: Menu[], parentMenu: Menu, depth: number) => {\n      for (const item of list) {\n        callback(item, parentMenu, depth);\n        if (item.children && item.children.length > 0) {\n          _walk(item.children, item, depth + 1);\n        } else {\n          item.children = [];\n        }\n      }\n    };\n\n    _walk(menus, null, 0);\n  }\n\n  private prune(menus: Menu[]): Menu[] {\n    const result: Menu[] = [];\n\n    for (const item of menus) {\n      if (item.hide) {\n        continue;\n      }\n\n      if (item.children && item.children.length > 0) {\n        item.children = this.prune(item.children);\n        if ((item._type === 3 || item.group) && item.children.length === 0) {\n          continue; // 父菜单和分组如果没有了孩子，自己也消失\n        }\n      }\n\n      result.push(item);\n    }\n\n    return result;\n  }\n\n  private normalize(menus: Menu[]): Menu[] {\n    let i = 1;\n\n    this.walk(menus, (item, parent, depth) => {\n      item.__id = i++;\n      item.__parent = parent;\n      item._depth = depth;\n\n      // badge\n      if (item.badge) {\n        if (item.badge_dot !== true) {\n          item.badge_dot = false;\n        }\n        if (!item.badge_status) {\n          item.badge_status = 'error';\n        }\n      }\n\n      item.hide = item.acl && !this.aclService.can(item.acl);\n\n      item._type = item.externalLink ? 2 : 1;\n      if (item.children && item.children.length > 0) {\n        item._type = 3;\n      }\n    });\n\n    return this.prune(menus);\n  }\n\n  set(items: Menu[]) {\n    this.setSubject.next(items);\n  }\n\n  setCurrent(menu: Menu) {\n    this.currentSubject.next(menu);\n  }\n\n  getByUrl(url) {\n    let findItem: Menu = null;\n    this.walk(this.data, item => {\n      item._open = false;\n      if (!item.link) {\n        return;\n      }\n      if (!findItem && new RegExp(`^${url}\\\\b`).test(item.link)) {\n        findItem = item;\n      }\n    });\n\n    return findItem;\n  }\n\n  getPathByUrl(url) {\n    let item: Menu = null;\n    this.walk(this.data, (i, parent, depth) => {\n      if (i.link === url) {\n        item = i;\n      }\n    });\n\n    const ret: Menu[] = [];\n    if (!item) {\n      return ret;\n    }\n\n    do {\n      ret.splice(0, 0, item);\n      item = item.__parent;\n    } while (item);\n\n    return ret;\n  }\n\n  toggleOpen(item: Menu) {\n    this.toggleOpenSubject.next(item);\n  }\n\n  private setMutation(menus: Menu[]): Mutation {\n    return () => {\n      // 规范化，添加部分计算属性\n      return this.normalize(menus);\n    };\n  }\n\n  private toggleOpenMutation(menu: Menu): Mutation {\n    return menus => {\n      this.walk(menus, m => {\n        if (m.__id !== menu.__id) {\n          m._open = false;\n          return;\n        }\n\n        menu._open = !menu._open;\n      });\n\n      return menus;\n    };\n  }\n\n  private setCurrentMutation(menu: Menu): Mutation {\n    return menus => {\n      let p = menu;\n\n      if (p) {\n        do {\n          p._open = true;\n          p = p.__parent;\n        } while (p);\n      }\n\n      return menus;\n    };\n  }\n\n  private refreshMutation(): Mutation {\n    return menus => {\n      return this.normalize(menus);\n    };\n  }\n}\n"]}