Vue3+Element Plus实现动态标签页以及右键菜单功能

 更新时间:2023年09月22日 11:42:52   作者:日进斗识  
这篇文章主要给大家介绍了关于Vue3+Element Plus实现动态标签页以及右键菜单功能的相关资料,Vue 3和Element Plus提供了一种简单的方法来实现侧边菜单栏与标签页之间的联动,文中通过代码介绍的非常详细,需要的朋友可以参考下

先上图

只有首页的情况

多个tab页的情况

使用el-dropdown绑定右键菜单,为每个tab页绑定一个右键

<el-tabs v-model="activeTab" @tab-click="clickTab" type="border-card" class="demo-tabs" closable @edit="removeTab">
        <el-tab-pane v-for="(item, index) in tabs" :key="item.path" :label="item.name" :name="item.path">
            <template #label>
                <el-dropdown :trigger="['contextmenu']" ref="dropdownRef" :id="item.name"
                    @visible-change="handleChange($event, item.name)">
                    {{ item.name }}
                    <template #dropdown>
                        <el-dropdown-menu>
                            <el-dropdown-item @click="reload(item)" :disabled="handDisabled('reload', item, index)">
                                <el-icon>
                                    <Refresh />
                                </el-icon>重新刷新
                            </el-dropdown-item>
                            <el-dropdown-item v-if="item.path != '/welcome'" @click="closeMy(item)"
                                :disabled="handDisabled('closeMy', item, index)">
                                <el-icon>
                                    <Close />
                                </el-icon>关闭当前</el-dropdown-item>
                            <el-dropdown-item v-if="tabs.length > 1" @click="closeOther(item)"
                                :disabled="handDisabled('closeOther', item, index)">
                                <el-icon>
                                    <Remove />
                                </el-icon>关闭其他</el-dropdown-item>
                            <el-dropdown-item v-if="tabs.length > 1" @click="closeAll(item)"
                                :disabled="handDisabled('closeAll', item, index)">
                                <el-icon>
                                    <Minus />
                                </el-icon>关闭全部</el-dropdown-item>
                        </el-dropdown-menu>
                    </template>
                </el-dropdown>
            </template>
        </el-tab-pane>
        <!--路由占位符-->
        <router-view></router-view>
    </el-tabs>

右键菜单生效后控制每个下拉项的禁用与显示(每一项代表一个功能)

const handDisabled = (action: string, data: any, index: any) => {
    if (action == 'reload') {
        return route.path != data.path
    }
    if (action == 'closeMy') {
        return route.path != data.path
    }
    if (action == 'closeOther') {
        return route.path != data.path
    }
    return false
}

每个右键项对应的功能 

const reload = (item: any) => {
    router.go(0)
}
const closeMy = (item: any) => {
    removeTab(item.path)
}
const closeOther = (item: any) => {
    const welcome = { name: "欢迎界面", path: "/welcome" }
    console.info(item)
    tabsSt.setTabs([welcome])
    const { name, path } = item
    let oldTabs: any = []
    tabs.value.forEach((element: any) => {
        oldTabs.push(element.path)
    });
    if (!oldTabs.includes(path)) {
        tabs.value.push({ name: name as any, path })
    }
}
const closeAll = (item: any) => {
    const welcome = { name: "欢迎界面", path: "/welcome", }
    tabsSt.setTabs([welcome])
    router.push('/welcome')
}

控制每次只显示一个右键

const dropdownRef = ref()
const handleChange = (visible: boolean, name: string) => {
    if (!visible) return
    dropdownRef.value.forEach((item: { id: string; handleClose: () => void }) => {
        if (item.id === name) return
        item.handleClose()
    })
}

完整代码

<template>
    <el-tabs v-model="activeTab" @tab-click="clickTab" type="border-card" class="demo-tabs" closable @edit="removeTab">
        <el-tab-pane v-for="(item, index) in tabs" :key="item.path" :label="item.name" :name="item.path">
            <template #label>
                <el-dropdown :trigger="['contextmenu']" ref="dropdownRef" :id="item.name"
                    @visible-change="handleChange($event, item.name)">
                    {{ item.name }}
                    <template #dropdown>
                        <el-dropdown-menu>
                            <el-dropdown-item @click="reload(item)" :disabled="handDisabled('reload', item, index)">
                                <el-icon>
                                    <Refresh />
                                </el-icon>重新刷新
                            </el-dropdown-item>
                            <el-dropdown-item v-if="item.path != '/welcome'" @click="closeMy(item)"
                                :disabled="handDisabled('closeMy', item, index)">
                                <el-icon>
                                    <Close />
                                </el-icon>关闭当前</el-dropdown-item>
                            <el-dropdown-item v-if="tabs.length > 1" @click="closeOther(item)"
                                :disabled="handDisabled('closeOther', item, index)">
                                <el-icon>
                                    <Remove />
                                </el-icon>关闭其他</el-dropdown-item>
                            <el-dropdown-item v-if="tabs.length > 1" @click="closeAll(item)"
                                :disabled="handDisabled('closeAll', item, index)">
                                <el-icon>
                                    <Minus />
                                </el-icon>关闭全部</el-dropdown-item>
                        </el-dropdown-menu>
                    </template>
                </el-dropdown>
            </template>
        </el-tab-pane>
        <!--路由占位符-->
        <router-view></router-view>
    </el-tabs>
</template>
<script lang="ts" setup true>
import { tabsStore } from '@/pinia/tabs';
import { homeStore } from '@/pinia/home';
import { useRoute, useRouter } from 'vue-router';
import { computed, onMounted, reactive, ref, toRef, watch } from 'vue';
import { Close, Minus, Refresh } from '@element-plus/icons-vue';
const router = useRouter();
const route = useRoute();
const tabsSt = tabsStore();
const { active, menus } = homeStore();
//tabs默认选项卡
const activeTab = ref(active)
const tabs = computed(() => {
    console.info("....初始化tabs")
    if (tabsSt.getTabs.length == 1) {
        const welcome = { name: "欢迎界面", path: "/welcome", }
        tabsSt.setTabs([welcome])
    }
    return tabsSt.getTabs
})
//设置tabs选项卡
const setActive = () => {
    activeTab.value = route.path;
}
const removeTab = (targetName: any) => {
    if (targetName === '/welcome') {
        return
    }
    const tablist = tabs.value
    let activeName = activeTab.value;
    if (activeName === targetName) {
        tablist.forEach((tab: any, index: any) => {
            if (tab.path === targetName) {
                const nextTab = tablist[index + 1] || tablist[index - 1]
                if (nextTab) {
                    activeName = nextTab.path
                }
            }
        })
    }
    activeTab.value = activeName
    tabsSt.setTabs(tablist.filter((tab: any) => tab.path !== targetName))
    router.push({ path: activeName })
}
const clickTab = (tab: any) => {
    const { props } = tab
    router.push({ path: props.name })
}
const reload = (item: any) => {
    router.go(0)
}
const closeMy = (item: any) => {
    removeTab(item.path)
}
const closeOther = (item: any) => {
    const welcome = { name: "欢迎界面", path: "/welcome" }
    console.info(item)
    tabsSt.setTabs([welcome])
    const { name, path } = item
    let oldTabs: any = []
    tabs.value.forEach((element: any) => {
        oldTabs.push(element.path)
    });
    if (!oldTabs.includes(path)) {
        tabs.value.push({ name: name as any, path })
    }
}
const closeAll = (item: any) => {
    const welcome = { name: "欢迎界面", path: "/welcome", }
    tabsSt.setTabs([welcome])
    router.push('/welcome')
}
const dropdownRef = ref()
const handleChange = (visible: boolean, name: string) => {
    if (!visible) return
    dropdownRef.value.forEach((item: { id: string; handleClose: () => void }) => {
        if (item.id === name) return
        item.handleClose()
    })
}
const handDisabled = (action: string, data: any, index: any) => {
    if (action == 'reload') {
        return route.path != data.path
    }
    if (action == 'closeMy') {
        return route.path != data.path
    }
    if (action == 'closeOther') {
        return route.path != data.path
    }
    return false
}
const addTab = () => {
    const { name, path } = route
    let oldTabs: any = []
    tabs.value.forEach((element: any) => {
        oldTabs.push(element.path)
    });
    if (!oldTabs.includes(path)) {
        tabs.value.push({ name: name as any, path })
    }
}
watch(() => route.path, () => {
    setActive();
    addTab();
})
//vuex刷新数据丢失
// const beforeRefresh = () => {
//     window.addEventListener('beforeunload', () => {
//         sessionStorage.setItem("tabsView", JSON.stringify(tabsSt.getTabs))
//     })
//     let tabSession = sessionStorage.getItem("tabsView")
//     if (tabSession) {
//         let oldTabs = JSON.parse(tabSession);
//         if (oldTabs.length > 0) {
//             tabsSt.setTabs(oldTabs)
//         }
//     }
// }
const beforeRefresh = () => {
    window.addEventListener('beforeunload', () => {
        console.info("beforeunload...")
    })
}
onMounted(() => {
    beforeRefresh();
    setActive();
})
</script>
<style lang="scss">
.demo-tabs {
    // min-height: 100vh;
    border: none;
    border-radius: 5px;
}
.el-tabs--border-card>.el-tabs__content {
    padding: 10px;
}
.el-tabs__item {
    font-size: 8px;
    font-weight: 1;
    padding: 0 8px;
}
</style>

总结

到此这篇关于Vue3+Element Plus实现动态标签页以及右键菜单功能的文章就介绍到这了,更多相关Vue3动态标签页及右键菜单内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!

相关文章

  • 如何理解Vue的render函数的具体用法

    如何理解Vue的render函数的具体用法

    本篇文章主要介绍了如何理解Vue的render函数的具体用法,小编觉得挺不错的,现在分享给大家,也给大家做个参考。一起跟随小编过来看看吧
    2017-08-08
  • vue实现页面上传文件夹压缩后传给服务器的操作

    vue实现页面上传文件夹压缩后传给服务器的操作

    这篇文章主要介绍了vue实现页面上传文件夹压缩后传给服务器的操作,本文通过示例代码给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友可以参考下
    2023-09-09
  • vue实现局部刷新的实现示例

    vue实现局部刷新的实现示例

    这篇文章主要介绍了vue实现局部刷新的实现示例,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2019-04-04
  • Vue3 的ref和reactive的用法和区别示例解析

    Vue3 的ref和reactive的用法和区别示例解析

    ref和reactive是Vue3中用来实现数据响应式的API,一般情况下,ref定义基本数据类型,reactive定义引用数据类型,本文给大家介绍Vue3 的ref和reactive的用法和区别,感兴趣的朋友一起看看吧
    2023-10-10
  • Vue Steam同款登录验证数字输入框功能

    Vue Steam同款登录验证数字输入框功能

    这篇文章主要介绍了Vue Steam同款登录验证数字输入框功能,本文通过实例代码给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友参考下吧
    2024-03-03
  • vue-cli配置文件——config篇

    vue-cli配置文件——config篇

    这篇文章主要介绍了vue-cli中的webpack是如何配置的,本篇文章主要是分析vue中关于config文件夹中的相关代码,config的文件结构,感兴趣的朋友参考下本文
    2018-01-01
  • vue+vux实现移动端文件上传样式

    vue+vux实现移动端文件上传样式

    这篇文章主要介绍了vue+vux实现移动端文件上传样式,样式使用的是vux的cell组件,需要的朋友可以参考下
    2017-07-07
  • vue-cli 3如何使用vue-bootstrap-datetimepicker日期插件

    vue-cli 3如何使用vue-bootstrap-datetimepicker日期插件

    这篇文章主要介绍了vue-cli 3如何使用vue-bootstrap-datetimepicker日期插件,帮助大家更好的理解和学习使用vue框架,感兴趣的朋友可以了解下
    2021-02-02
  • vue.js父子组件通信动态绑定的实例

    vue.js父子组件通信动态绑定的实例

    今天小编就为大家分享一篇vue.js父子组件通信动态绑定的实例,具有很好的参考价值,希望对大家有所帮助。一起跟随小编过来看看吧
    2018-09-09
  • vue实现分页组件

    vue实现分页组件

    这篇文章主要为大家详细介绍了vue实现分页组件的具体代码,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2018-12-12

最新评论