最近处理了一个标签页与父子页面联动的问题,在开发一个后台管理项目时,前端框架是在另一个项目的基础上改进使用的,原有项目的业务整体比较简单,通过不同导航页进入不同的标签页后,查询显示结果即可。而在新项目中,还需要处理另一种业务情况:从父页面进入子页面时的标签页处理,例如查询列表页跳转进详情页/编辑页。此时才发现这个框架的坑,而偏偏还已经开发了一段时间,换框架是没得可能只能硬着头皮改造。

  这里的主要问题是:标签页是与路由绑定的,在进入子页面时是需要打开新的标签页?子页面的返回行为又该如何处理。在各种讨论妥协后,最终选择方案就是:子页面打开时新建标签页且相同组件只能创建一个,父页面关闭时连带着子页面一起关闭。

一、业务背景

  作为一个 Angular 后台管理项目,布局结构是传统的 顶部-侧边布局-通栏 (见下图),拥有顶部导航及侧边栏,而由于是 SPA 单页应用的缘故,页面地址是记录在地址后的 # 后面。Tab 栏作为已打开页面的切换入口,与路由进行了关联。

页面结构

1. 标签页与路由复用

  当标签页与路由联动后,在处理父子页面上出现了一些问题,由于标签页是与地址直接匹配的,所以如果在父页面进行诸如查看详情之类的操作时,子页此时有两种处理方式:一是顶替覆盖掉当前父页面所对应的标签页,另一则是独立新建一个标签页 (当前框架设计没法使用子路由 🎈)

  对于标签页来说,标签的切换本质还是路由跳转,在切换过程中就涉及到路由复用的处理,通过实现 RouteReuseStrategy 完成自定义的复用策略。简单来说就是在 路由离开时记录、路由切换时恢复、标签页删除时销毁 。一般来说是用当前路由的地址作为 Key ,通过这个 Key 进行操作,也就是:

路由复用策略(部分)
/** 当路由离开时会触发。 */
public store(route: ActivatedRouteSnapshot, handle: DetachedRouteHandle): void {
if (SimpleReuseStrategy.waitDelete && SimpleReuseStrategy.waitDelete === this.getRouteUrl(route)) {
SimpleReuseStrategy.waitDelete = null;
return;
}
SimpleReuseStrategy.handlers[this.getRouteUrl(route)] = handle;
}

// 根据路由包装地址
private getRouteUrl(route: ActivatedRouteSnapshot) {
// 这句代码可以获取当前路由的组件名componentName,但生成环境(打包)将组建名缩写成随机单个字母,
// 所以需要手动通过route.routeConfig.data['componentName']去获取在路由上自定义的组件名
let componentShortName = (route.routeConfig.loadChildren
|| route.routeConfig.component.toString().split('(')[0].split(' ')[1]);
if (route.routeConfig.data && route.routeConfig.data['componentName']) {
componentShortName = route.routeConfig.data['componentName'];
}

const url = route['_routerState'].url.replace(/\//g, '_');
return url.substring(0, url.indexOf('?') === -1 ? url.length : url.indexOf('?')) + '_' + componentShortName;
}

  由于标签页切换的本质就是路由跳转,所以标签页记录的这个地址就相当重要了,路由与标签页是一对一的。

2. 标签页的事件处理

  对于标签组件,需要实现的核心功能便是:增删查。通过对 Router 事件进行过滤我们可以拿到定义在 RouterModule 里的路由信息。

router.events
this.router.events.pipe(
filter(event => event instanceof NavigationEnd),
map(() => this.activatedRoute),
map(route => {
while (route.firstChild) {
route = route.firstChild;
}
return route;
}),
filter(route => route.outlet === 'primary'),
mergeMap(route => route.data)
).subscribe(
const menu = {...event}; // 这里的 menu 就是我们需要的信息
}
);

  将所有的 menu 记录到 this.menuList 中,也就得知当前打开了多少标签页,以及他们的标题、地址等信息。

二、父子页面处理

  简单来说该问题可以视为 是否新建子页面的标签页 ,进一步的则是是否保留父页面标签。我们已经知道路由离开时会按照地址存进快照中,所以还需要处理恢复快照的逻辑,并且无论是快照、标签页、还是路由跳转都与地址息息相关,以地址为唯一值。

1. 子页面不新建标签页

  如果不新建标签页,那么就得修改原父页面中存储的字段(狸猫换太子? 替掉父标签相应内容)。首先是决定替换哪些字段,标题用以显示在标签页中自然首选,那么剩下的就是组件名和地址了:已知 标签页通过地址作唯一判断,快照通过地址和组件名作唯一判断。所以替换地址的行为就相当于关闭了父页面标签,从子页面进行返回操作约等于新建一个父页面。这个方案导致 父页面标签丢失。并且当子页面未关闭时,可以通过侧栏菜单打开父页面,此时父子页面同时出现,违背初衷了。

  而如果只替换标题,逻辑上父标签主要信息未动,只是显示在标签上的名称改变了。但是,由于当前标签页还是父页面地址,在进行标签页切换后,回到此标签时恢复的是父页面的内容,而不是子页面内容,同时在快照缓存上,为了避免内存泄漏,所有的子页面都不能记录快照。原则上说只要用户不切换页面,这个方式还算是可行。我最初就是使用的这套,因为简单所以框架改动不大。可是、可是客户爸爸它不认啊, 需要当前标签页显示什么回来后还是什么,没辙只能苦逼研究研究改框架了。

不新建子页面

2. 子页面新建标签页

  在最初的设想中是不限制子页面打开的数量,可以通过父页面跳转进不同内容下的相同子页面(组件),程序实现上没什么难度。但在业务上存在个问题:有些子页面是拥有回到父页面的功能(例如子页面新增后提交)。按照直觉来说,返回自然是回到进入子页面前的父页面状态,但是由于子页面不限制,所以 可能出现某一子页面还未关闭,父页面修改了内容打开了另一个子页面,这两个子页面是通过不同的查询条件筛选出的,但快照只能记录/返回最后一次的父页面信息,这近似于丢失了进入子页面前的父页面状态,回到原点了。

  所以 同一个父页面下的子页面只能存在一个,前文已知标签与地址是一对一的,而子页面的地址中一定会携带参数,对标签来说虽然他们都拥有相同的组件但是地址上它们就是不同。那么我们干脆简单点,如果已经存在了一个子页面,那么在新的子页面打开前直接删除原子页面的标签和快照,然后再打开新的子页面,这解决了多个子标签的问题。另一方面由于地址转义的缘故,我们不能拿完整的地址去当作快照的 Key ,子页面快照所存储的 Key 是剔除参数后的地址

  最后就是父子页面的删除关系了,由于得满足子页面返回时是返回进入子页面前的父页面状态这一条件,那就不能出现只剩下子标签的情况。举个栗子:父页面是一个查询页,通过一些条件筛选出了结果,然后进入到某个结果的详情页,此时切换标签到父页面,并且删除了父页面标签,子页面还存在,此时子页面的返回就相当于父页面重新打开。所以为了避免业务上出现问题,最简单的处理还是 父页面关闭时,父页面下所有子页面都关闭

三、业务实现

  理论分析后就是代码上的实现了,这里考虑了子页面的子页面跳转情况,在对他们的路径设置时额外添加了三个参数:isChild, parComponent ,commonComponent ,他们的关系如下:

组件关系

  父页面不关心也不需要知道自己有多少个子页面,所有子页面需要通过 isChild 标记自己是子页面;同时子页面需要通过 parComponent 记录自己上一个页面的组件名;除了父页面的直系子页面外,其它层级的子页面需要通过 commonComponent 记录最顶层父页面的组件名(或者叫祖先,嚯嚯嚯大笑)。

  在创建首层子页面时,只要发现当前标签页中 存在记录了祖先是当前页面的子页面 或者 父页面是当前页面的子页面 ,就提示是否先关闭已经打开的子页面(此处不关心他们的层级,也就是当前可能存在第三级的子页面,但是只要父页面打开第一级子页面时就关闭第三级子页面标签和第三级页面的快照)。

  在关闭父页面时,寻找可能的子页面和子孙页面,然后发出提示弹窗需要连带一起关闭。当标签页只存在一个时不允许关闭,当标签页只存在父页面和它的子(孙)页面时,不允许关闭父页面。

1. 标签页的处理

  在上面的设计中父页面是一个唯一的值,带领着它的子页面,所以 父页面的路径存储一定不能携带参数 ,但是 子页面的路径存储必须得携带参数。我们在 TabComponent 监听了路由切换事件,并通过它记录路由信息,这部分处理就在事件监听之中:

路由事件的处理(部分)
let menuUrl = menu.url, sameTabChild = false, childComponent = null;
this.menuList.forEach(item => {
// 如果是子页面标签
if (item.isChild && !sameTabChild) {
// 相同的子页面组件名 不同的地址
if (item.componentName === menu.componentName && item.url !== menuUrl) {
sameTabChild = true;
childComponent = item;
}
// 当前页是子页面 子页面的子页面
if (menu.isChild && item.parComponent === menu.componentName ) {
sameTabChild = true;
childComponent = item;
}
}
});

// 即将打开的页面为相同的子页面
if (sameTabChild) {
const indexOfChild = this.menuList.findIndex(p => p.componentName === childComponent.componentName);
this.menuList.splice(indexOfChild, 1); // 删除原【子页面】记录值
} else if (menu.isChild === undefined) {
// 为了确保唯一性,剔除可能存在的参数
const indexParam = menu.url.indexOf('?');
menuUrl = menu.url.substring(0, indexParam === -1 ? menu.url.length : indexParam);
}

  前文已经说了切换本质是路由跳转,而拿到的地址是网页 Url ,所以得先提取出参数部分转换为对象才能跳转。

标签页的切换
nzSelectChange(event) {
this.currentIndex = event.index;
const menu = this.menuList[this.currentIndex], temp = menu.url.indexOf('?');
const params = temp === -1 ? null : this.getUrlParams(menu.url.substring(temp + 1, menu.url.length));
const url = temp === -1 ? menu.url : menu.url.substring(0, temp === -1 ? menu.url.length : temp);

this.router.navigate([url], {
queryParams: params, // 存放可能的参数
skipLocationChange: true
})
}

// Url 参数转 对象
getUrlParams(url) {
if (url === '') return null;
let hash, myJson = {}, hashes = url.slice(url.indexOf('?') + 1).split('&');
for (let i = 0; i < hashes.length; i++) {
hash = hashes[i].split('=');
myJson[hash[0]] = decodeURIComponent(hash[1]);
}
return myJson;
}

  大致的删除逻辑上文已经描述了,代码部分:

标签页的删除
// 关闭选项标签
closeUrl(menu) {
// 如果只有一个不可以关闭
if (this.menuList.length === 1) return;

// 如果当前只剩下父页面和对应子页面,则不可以关闭父页面
if (this.menuList.length === 2) {
if (this.menuList.some(item => item.isChild
&& (item.parComponent === menu.componentName || item.commonComponent === menu.componentName))) {
return;
}
}

// 当前关闭的是第几个路由
const index = this.menuList.findIndex(p => p.url === menu.url);
if (index === -1) { // 子页面的子页面返回子页面时造成的 (坏笑
return;
}

// 寻找可能存在的子页面
const childComponent = this.menuList.filter(item => this.menuList[index].isChild === undefined
&& item.parComponent === this.menuList[index].componentName) || [];

// 寻找可能存在的子页面的剩下子页面
this.menuList.filter(item => this.menuList[index].isChild === undefined // 本身必须是父页面
&& item.commonComponent === this.menuList[index].componentName // 祖先
&& item.parComponent !== this.menuList[index].componentName // 不是第一个子页面
).forEach(item => {
childComponent.push(item);
});

// 删除
if (childComponent.length === 0) {
this.deleteTabs(menu);
} else {
this.modal.confirm({
nzContent: '【' + childComponent[0].title + '】页将一并关闭,是否继续?',
nzCancelText: '否',
nzOkText: '是',
nzOkType: 'danger',
nzOnOk: () => {
childComponent.forEach(item => {
this.deleteTabs(menu, item);
})
}
});
}
}

// 执行 Tab 删除
deleteTabs(menu, childComponent = null) {
const index = this.menuList.findIndex(p => p.url === menu.url);
const indexMenuParam = menu.url.indexOf('?');
const path = menu.url.substring(0, indexMenuParam === -1 ? menu.url.length : indexMenuParam) + '_' + menu.componentName;
this.menuList.splice(index, 1); // 删除父
SimpleReuseStrategy.deleteRouteSnapshot(path); // 删除复用

// 删除子
if (childComponent) {
const indexOfChild = this.menuList.findIndex(p => p.componentName === childComponent.componentName);
const indexChildParam = childComponent.url.indexOf('?');
const childPath = childComponent.url.substring(0, indexChildParam === -1 ? childComponent.url.length : indexChildParam)
+ '_' + childComponent.componentName;
this.menuList.splice(indexOfChild, 1); // 删除子页面
SimpleReuseStrategy.deleteRouteSnapshot(childPath); // 删除子页面复用
}

this.settingsService.setMenuList(this.menuList); // 记录

// 如果当前删除的对象是当前选中的,那么需要跳转
if (this.currentUrl === menu.url) {
// 显示上一个选中
let menu = this.menuList[index - 1];
if (!menu) {// 如果上一个没有下一个选中
menu = this.menuList[index];
}
// 跳转路由
this.router.navigate([menu.url]);
}
}

2. 路由守卫

  我们需要在即将替换关闭子页面时,发出提示,这个行为交给 CanActivate 比较方便。

Angular 路由守卫
can-overwrite-tab.guard.ts
canActivate(
next: ActivatedRouteSnapshot,
state: RouterStateSnapshot): Observable<boolean | UrlTree> | Promise<boolean | UrlTree> {
return new Promise((resolve) => {
const url = state.url;
const nextData = next.data;
const menuList = this.settingsService.getMenuList() || [];
let flag = false, childComponent = null, msg = null, indexParam = 0;
menuList.forEach(item => {
if (!flag) {
// 判断 1
if (item.isChild && item.parComponent === nextData.parComponent && url !== item.url) {
flag = true;
childComponent = item;
indexParam = url.indexOf('?');
msg = '已打开【' + nextData.title + '】页,是否继续?';
}
// 判断 2
if (item.isChild && item.parComponent === nextData.componentName && url !== item.url) {
flag = true;
childComponent = item;
indexParam = item.url.indexOf('?');
msg = '已打开【' + item.title + '】页,是否继续?';
}
}
});

if (flag) {
this.modal.confirm({
nzContent: msg,
nzCancelText: '否',
nzOkText: '是',
nzOkType: 'danger',
nzOnOk: () => {
const childLength = childComponent.url.length;
const childPath = childComponent.url.substring(0, indexParam === -1 ? childLength : indexParam)
+ '_' + childComponent.componentName;
SimpleReuseStrategy.deleteRouteSnapshot(childPath); // 先一步删除快照
resolve(true);
},
nzOnCancel: () => {
resolve(false);
}
})
} else {
resolve(true);
}
});
}

评论