场景:运行项目,登录后跳到首页,新开一个标签页并访问首页,状态是未登录状态
理想结果:
1、无论打开多少个标签页,访问同一个项目,都是同一个登录状态;
2、切换账号后其余标签跟着切换;
3、更换头像后,其他标签页跟着切换;
4、退出登录其余页面跟着退出;
不多说,上代码:
在main.js添加存储的监听
// 全局监听sessionStorage
app.config.globalProperties.$addStorageEvent = function (key, data) {
// 创建一个StorageEvent事件
var newStorageEvent = document.createEvent('StorageEvent');
const storage = {
setItem: function (k, val) {
sessionStorage.setItem(k, val)
// 初始化创建的事件
newStorageEvent.initStorageEvent('setItem', false, false, k, null, val, null, null);
// 派发对象
window.dispatchEvent(newStorageEvent);
}
}
storage.setItem(key, data);
// 如果sessionStorage设置为null 代表退出登录
if (!data) {
sessionStorage.clear();
localStorage.setItem('sessionStorage', "");
localStorage.removeItem('sessionStorage');
} else {
if (sessionStorage.getItem('userInfo') && JSON.parse(sessionStorage.getItem('userInfo'))) {
localStorage.setItem('sessionStorage', sessionStorage.getItem('userInfo'));
localStorage.removeItem('sessionStorage');
}
}
}
// 该事件是核心
window.addEventListener('storage', function (event) {
if (event.key == 'getSessionStorage') {
// 已存在的标签页会收到这个事件
if (sessionStorage.getItem('userInfo') && JSON.parse(sessionStorage.getItem('userInfo'))) {
localStorage.setItem('sessionStorage', sessionStorage.getItem('userInfo'));
localStorage.removeItem('sessionStorage');
}
} else if (event.key == 'sessionStorage') {
// 新开启的标签页会收到这个事件
if (event.newValue && event.newValue != '' && event.newValue != 'null') {
sessionStorage.setItem('userInfo', event.newValue);
// 派发事件
var newStorageEvent = document.createEvent('StorageEvent');
newStorageEvent.initStorageEvent('setItem', false, false, '', null, '', null, null);
window.dispatchEvent(newStorageEvent);
}
if(event.newValue == ''){
var newStorageEvents = document.createEvent('StorageEvent');
newStorageEvents.initStorageEvent('setItem', false, false, '', null, '', null, null);
window.dispatchEvent(newStorageEvents);
sessionStorage.clear();
}
}
});
home.vue 登录时将信息写入session
import { Login } from "@http/index.js";
mounted() {
// 判断地址栏上是否带了ticket和存储上有没有token
if (
window.location.href.indexOf("ticket") > -1 &&
!this.$utils.getToken()
) {
this.service = window.location.href.slice(
0,
window.location.href.indexOf("ticket") - 1
);
this.ticket = window.location.href.slice(
window.location.href.indexOf("ticket") + 7
);
this.getUserInfo();
}
},
methods: {
getUserInfo() {
Login({ service: this.service, ticket: this.ticket })
.then((res) => {
if (res.code === "0") {
// 触发监听sessionStorage事件
this.$addStorageEvent("userInfo", JSON.stringify(res.result));
// 触发通过localStorage的事件,将userInfo写入session
localStorage.setItem("getSessionStorage", Date.now());
} else {
this.$ElMessage.error(res.message);
}
})
.catch((err) => {
console.log(err);
});
},
},
head.vue 头部监听session变动(userInfo包含头像地址,head.vue主要就是监听头像的变动)
<el-avatar size="small" :src="userInfo.avatar ? userInfo.avatar : ''" @click="personalDrawer = !personalDrawer" style="margin-left:28px; cursor:pointer;">el-avatar>
mounted() {
// 监听到session的变化
window.addEventListener("setItem", () => {
this.$nextTick(() => {
this.userInfo = this.$utils.getUserInfo();
});
});
},