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

这篇具有很好参考价值的文章主要介绍了Vue3 + Element Plus 实现动态标签页及右键菜单。希望对大家有所帮助。如果存在错误或未考虑完全的地方,请大家不吝赐教,您也可以点击"举报违法"按钮提交疑问。

目录

先上图

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

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

每个右键项对应的功能 

控制每次只显示一个右键

完整代码


先上图

        只有首页的情况

Vue3 + Element Plus 实现动态标签页及右键菜单,vue3,elementui,前端,javascript

        多个tab页的情况

Vue3 + Element Plus 实现动态标签页及右键菜单,vue3,elementui,前端,javascript

 使用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>

 文章来源地址https://www.toymoban.com/news/detail-723174.html

到了这里,关于Vue3 + Element Plus 实现动态标签页及右键菜单的文章就介绍完了。如果您还想了解更多内容,请在右上角搜索TOY模板网以前的文章或继续浏览下面的相关文章,希望大家以后多多支持TOY模板网!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处: 如若内容造成侵权/违法违规/事实不符,请点击违法举报进行投诉反馈,一经查实,立即删除!

领支付宝红包 赞助服务器费用

相关文章

  • vue3 element-plus动态菜单及动态图标

    引入element-plus 注册图标组件 动态引入图标代码 完整代码 路由如下

    2024年01月18日
    浏览(31)
  • Vue3中动态绑定:disabled element-plus使用方法

    @change=\\\"whetherFlag($event)\\\"  根据value值判断是否禁用 :disabled=\\\"isShow\\\" 初始值为禁用状态 const isShow = refboolean(true);  根据value的值判断是否禁用  

    2024年01月25日
    浏览(41)
  • 【VUE】ElementPlus之动态主题色调切换(Vue3 + Element Plus+Scss + Pinia)

    关于ElementPlus的基础主题色自定义可以参阅《【VUE】ElementPlus之自定义主题样式和命名空间》 有了上面基础的了解,我们知道 ElementPlus 的主题色调是基于 CSS3变量 特性进行全局控制的, 那么接下来我们也基于 CSS3变量 来实现主题色调的 动态切换 效果; 主要控制的色调类型有

    2024年02月08日
    浏览(27)
  • Vue3-element-plus表格中动态加载数据后再进行列排序

    直接上代码 1、表格定义 2、js 参考: https://element-plus.org/zh-CN/component/table.html#table-%E5%B1%9E%E6%80%A7 https://www.cnblogs.com/onesea/p/15702253.html

    2024年02月21日
    浏览(30)
  • vue3 element-plus 实现图片预览

    element-plus下有这么一个组件 el-image-viewer /,但是这个组件是没写在文档上面的,像普通组件一样使用即可 可以通过点击按钮实现图片预览,而非el-image组件只能通过点击图片实现预览 2.1封装组件 2.3组件使用 在需要使用的地方引入,然后使用即可,这不是重点,每个人使用的

    2024年02月15日
    浏览(44)
  • vue3+element Plus实现表格前端分页

    每一处都写了注释,还是很容易看懂的  

    2024年02月11日
    浏览(42)
  • Vue3+Element plus+pageHelper实现分页

    安装element plus 引入 修改main.js: 后端 pom.xml: controller: service: 前端 vue:

    2024年02月13日
    浏览(40)
  • Vue3+element-plus实现后台管理系统

     环境:node.js软件 、Vs code、vite、elemnt-plus、windicss(样式框架)     1、首先,使用npm 命令构建项目( vscode安装的插件 vscode中文显示插件   2、高亮提示插件volar   3、vue 3 sni 代码提示) 快速上手 | Vue.js    a. npm -v 查看node.js 版本    b.  npm  config get registry   查看注册镜像是

    2024年02月09日
    浏览(39)
  • 前端vue3+element plus 分页table排序功能实现

    我有如下所示的一个table,数据data是一个computed计算属性,一般情况下筛选使用element的sortable属性就可以了,可查看Element - The world\\\'s most popular Vue UI framework 但我的 table 是一个分页的 table ,当我使用 sortable 时发现,它只是对当前页的数据进行排序,但这并不是我需要的,我的

    2024年02月14日
    浏览(37)
  • Vue3+element plus+sortablejs实现table列表拖拽

    1、安装 2、引入 3、使用 表格表头必须加 row-key ,否则会出现排序错乱 完整代码 table.vue

    2024年02月07日
    浏览(37)

觉得文章有用就打赏一下文章作者

支付宝扫一扫打赏

博客赞助

微信扫一扫打赏

请作者喝杯咖啡吧~博客赞助

支付宝扫一扫领取红包,优惠每天领

二维码1

领取红包

二维码2

领红包