【Vue3+Vite+TS】7.0 组件五:通知菜单

必备UI组件

将用到的组件:

Badge 徽章
Popover 弹出框/气泡卡片
Tabs 标签页
Avatar 头像
Tag 标签
Scrollbar 滚动条

组件设计

新建src\components\baseline\notification\index.ts

import { App } from 'vue'
import Notification from './src/index.vue'

export { Notification }

//组件可通过use的形式使用
export default {
    Notification,
    install(app: App) {
        app.component('bs-notification', Notification)
    },
}

新建src\components\baseline\notification\src\index.vue


<template>
    <el-badge
        style="cursor: pointer"
        :value="value"
        :max="max"
        v-bind="$attrs"
        :is-dot="isDot"
    >
        <component :is="icon"></component>
    </el-badge>
</template>
<script lang="ts" setup>
const props = defineProps({
    //说明:图标
    icon: {
        type: String,
        default: 'el-icon-bell',
    },
    //说明:通知数量
    value: {
        type: [String, Number],
        default: 0,
    },
    color: {
        type: String,
        default: 'el-icon-bell',
    },
    //徽章最大值
    max: {
        type: Number,
    },
    //是否显示小圆点
    isDot: {
        type: Boolean,
        default: false,
    },
})
</script>
<style lang="scss" scoped></style>

新建src\views\baseline\notification\index.vue

<template>
    <div class="bs-wrapper">
        <bs-notification value="50"></bs-notification>
        <br />
        <br />
        <bs-notification value="50" max="30"></bs-notification>
        <br />
        <br />
        <bs-notification value="50" :is-dot="true"></bs-notification>
        <br />
        <br />
        <bs-notification value="50" icon="el-icon-chat-round"></bs-notification>
    </div>
</template>
<script lang="ts" setup></script>
<style lang="scss" scoped>
.bs-wrapper {
    .flex {
        display: flex;
    }
    div {
        margin-right: 0.1rem;
    }
}
</style>

修改src\router\index.ts

/*
 * @Author: bobokaka
 * @Date: 2021-12-19 11:26:38
 * @LastEditTime: 2021-12-24 23:34:35
 * @LastEditors: Please set LastEditors
 * @Description: 路由
 * @FilePath: \vue3-element-ui-baseline\src\router\index.ts
 */
import { createRouter, createWebHistory, RouteRecordRaw } from 'vue-router'

import Home from '../views/Home/index.vue'
import Container from '../components/baseline/container/src/index.vue'

const routes: RouteRecordRaw[] = [
    {
        path: '/',
        component: Container,
        children: [
            {
                path: '/',
                component: Home,
            },
            {
                path: '/chooseIcon',
                component: () => import('../views/baseline/chooseIcon/index.vue'),
            },
            {
                path: '/chooseArea',
                component: () => import('../views/baseline/chooseArea/index.vue'),
            },
            {
                path: '/trend',
                component: () => import('../views/baseline/trend/index.vue'),
            },
            {
                path: '/notification',
                component: () => import('../views/baseline/notification/index.vue'),
            },
        ],
    },
]

const router = createRouter({
    routes,
    history: createWebHistory(),
})
export default router

打开网页http://localhost:8080/notification

image.png

实现点击打开一个列表组件

修改src\views\baseline\notification\index.vue

<template>
    <div class="bs-wrapper">
        <!-- <bs-notification value="50"></bs-notification>
        <br />
        <br />
        <bs-notification value="50" max="30"></bs-notification>
        <br />
        <br />
        <bs-notification value="50" :is-dot="true"></bs-notification>
        <br />
        <br />
        <bs-notification value="50" icon="el-icon-chat-round"></bs-notification> -->
        <bs-notification value="50"></bs-notification>
    </div>
</template>
<script lang="ts" setup></script>
<style lang="scss" scoped>
.bs-wrapper {
    .flex {
        display: flex;
    }
    div {
        margin-right: 0.1rem;
    }
}
</style>

修改src\components\baseline\notification\index.ts

<template>
    <el-popover
        placement="bottom"
        title="Title"
        :width="200"
        trigger="click"
        content="this is content, this is content, this is content"
    >
        <template #reference>
            <el-badge
                style="cursor: pointer"
                :value="value"
                :max="max"
                v-bind="$attrs"
                :is-dot="isDot"
            >
                <component :is="icon"></component>
            </el-badge>
        </template>
    </el-popover>
</template>
<script lang="ts" setup>
const props = defineProps({
    //说明:图标
    icon: {
        type: String,
        default: 'el-icon-bell',
    },
    //说明:通知数量
    value: {
        type: [String, Number],
        default: 0,
    },
    color: {
        type: String,
        default: 'el-icon-bell',
    },
    //徽章最大值
    max: {
        type: Number,
    },
    //是否显示小圆点
    isDot: {
        type: Boolean,
        default: false,
    },
})
</script>
<style lang="scss" scoped></style>
image.png

从官网复制一个过来,点击图标的效果如上,这里我们不需要标题,删掉如下代码:

        title="Title"

整体修改后如下:

<template>
    <el-popover placement="bottom" :width="300" trigger="hover">
        <template #default>
            <slot></slot>
        </template>
        <template #reference>
            <el-badge
                style="cursor: pointer"
                :value="value"
                :max="max"
                v-bind="$attrs"
                :is-dot="isDot"
            >
                <component :is="icon"></component>
            </el-badge>
        </template>
    </el-popover>
</template>

修改src\views\baseline\notification\index.vue

<template>
    <div class="bs-wrapper">
        <!-- <bs-notification value="50"></bs-notification>
        <br />
        <br />
        <bs-notification value="50" max="30"></bs-notification>
        <br />
        <br />
        <bs-notification value="50" :is-dot="true"></bs-notification>
        <br />
        <br />
        <bs-notification value="50" icon="el-icon-chat-round"></bs-notification> -->
        <bs-notification value="50">
            <template #default>这是一个标题</template>
        </bs-notification>
    </div>
</template>
<script lang="ts" setup></script>
<style lang="scss" scoped>
.bs-wrapper {
    .flex {
        display: flex;
    }
    div {
        margin-right: 0.1rem;
    }
}
</style>
image.png

菜单组件封装

新建src\components\baseline\list\src\index.vue

<template>List</template>
<script lang="ts" setup></script>
<style lang="scss" scoped></style>

新建src\components\baseline\list\index.ts

import { App } from 'vue'
import List from './src/index.vue'

export { List }

//组件可通过use的形式使用
export default {
  List,
    install(app: App) {
        app.component('bs-list', List)
    },
}

修改src\views\baseline\notification\index.vue

import { App } from 'vue'
import ChooseArea from './chooseArea'
import ChooseIcon from './chooseIcon'
import Container from './container'
import Trend from './trend'
import Notification from './notification'
import List from './list'
const components = [
    ChooseArea,
    ChooseIcon,
    Container,
    Trend,
    Notification,
    List,
]
export { ChooseArea, ChooseIcon, Container, Trend, Notification, List }

//组件可通过use的形式使用
export default {
    install(app: App) {
        components.map(item => {
            app.use(item)
        })
    },
    ChooseArea,
    ChooseIcon,
    Container,
    Trend,
    Notification,
    List,
}

修改src\components\baseline\notification\src\index.vue

<template>
    <div class="bs-wrapper">
        <!-- <bs-notification value="50"></bs-notification>
        <br />
        <br />
        <bs-notification value="50" max="30"></bs-notification>
        <br />
        <br />
        <bs-notification value="50" :is-dot="true"></bs-notification>
        <br />
        <br />
        <bs-notification value="50" icon="el-icon-chat-round"></bs-notification> -->
        <bs-notification value="50">
            <template #default><bs-list></bs-list></template>
        </bs-notification>
    </div>
</template>
......

运行效果如下:


image.png

列表组件

这里需要构建一个即将出场的列表组件。
新建src\components\baseline\list\src\index.vue

<template>List</template>
<script lang="ts" setup>
import { PropType } from 'vue'
import { ActionOptions, listOptions } from './type'
const props = defineProps({
    list: {
        type: Array as PropType<listOptions[]>,
        required: true,
    },
    //操作的内容
    actions: {
        type: Array as PropType<ActionOptions[]>,
        default: [],
    },
})
</script>
<style lang="scss" scoped></style>

新建src\components\baseline\list\src\type.ts

export interface ListItem {
    //头像
    avatar?: string
    //标题
    title?: string
    //描述
    desc: string
    //时间
    time?: string
    //标签内容
    tag?: string
    tagType?: 'success' | 'info' | 'warning' | 'danger'
}
/**
 * 列表
 */
export interface listOptions {
    title: string
    content: ListItem[]
}
export interface ActionOptions {
    text: string
    icon?: string
}

新建src\components\baseline\list\index.ts

import { App } from 'vue'
import List from './src/index.vue'

export { List }

//组件可通过use的形式使用
export default {
  List,
    install(app: App) {
        app.component('bs-list', List)
    },
}

初始化,完毕,开始使用改造:
准备数据,新建src\views\baseline\notification\data.ts

/*
 * @Author: your name
 * @Date: 2021-12-28 18:45:30
 * @LastEditTime: 2021-12-29 01:03:45
 * @LastEditors: Please set LastEditors
 * @Description: 模拟数据
 * @FilePath: \vue3-element-ui-baseline\src\views\baseline\notification\data.ts
 */
export const list = [
    {
        title: '通知',
        content: [
            {
                title: '鲁迅回复了你的邮件',
                time: '2021-12-24 17:47:54',
                avatar: 'https://upload-images.jianshu.io/upload_images/16102290-ce12da7b12204094.png?imageMogr2/auto-orient/strip%7CimageView2/2/w/1240',
            },
            {
                title: '刘备邀请你参加会议',
                time: '2021-12-28 9:15:27',
                avatar: 'https://www.baidu.com',
            },
            {
                title: '诸葛孔明已批准了你的休假申请',
                time: '2021-12-27 16:47:33',
                avatar: 'https://www.baidu.com',
            },
        ],
    },
    {
        title: '关注',
        content: [
            {
                title: '鲁迅 评论了你',
                desc: '愿中国青年都摆脱冷气,只是向上走,不必听自暴自弃者的话。',
                time: '2021-12-24 17:47:54',
                avatar: 'https://www.baidu.com',
            },
            {
                title: '刘备 评论了你',
                desc: '惟贤惟德,能服于人!',
                time: '2021-12-28 9:15:27',
                avatar: 'https://www.baidu.com',
            },
            {
                title: '诸葛孔明 评论了你',
                desc: '恢弘志士之气,不宜妄自菲薄。',
                time: '2021-12-27 16:47:33',
                avatar: 'https://www.baidu.com',
            },
        ],
    },
    {
        title: '代办',
        content: [
            {
                title: '鲁迅发表文章,待审核',
                desc: '需要在2021-12-30 18:00:00前审核',
                avatar: 'https://www.baidu.com',
                tag: '未开始',
                tagType: '',
            },
            {
                title: '刘备出兵计划发起',
                desc: '刘备提交于2021-12-27 16:04:00 需要在2021-12-30 18:00:00前审核',
                time: '2021-12-28 9:15:27',
                avatar: 'https://www.baidu.com',
                tag: '即将过期',
                tagType: 'danger',
            },
            {
                title: '诸葛孔明科目二考试',
                desc: '需要在2021-12-29 18:00:00前审核',
                time: '2021-12-27 16:47:33',
                avatar: 'https://www.baidu.com',
                tag: '已过期',
                tagType: 'warning',
            },
            {
                title: '鲁迅发表文章,待审核',
                desc: '需要在2021-12-30 18:00:00前审核',
                avatar: 'https://www.baidu.com',
                tag: '未开始',
                tagType: '',
            },
            {
                title: '刘备出兵计划发起',
                desc: '刘备提交于2021-12-27 16:04:00 需要在2021-12-30 18:00:00前审核',
                time: '2021-12-28 9:15:27',
                avatar: 'https://www.baidu.com',
                tag: '即将过期',
                tagType: 'danger',
            },
            {
                title: '诸葛孔明科目二考试',
                desc: '需要在2021-12-29 18:00:00前审核',
                time: '2021-12-27 16:47:33',
                avatar: 'https://www.baidu.com',
                tag: '已过期',
                tagType: 'warning',
            },
        ],
    },
]
export const actions = [
    {
        text: '清空代办',
        icon: 'el-icon-delete',
    },
    {
        text: '查看更多',
        icon: 'el-icon-edit',
    },
]

修改src\views\baseline\notification\index.vue

<template>
    <div class="bs-wrapper">
        <!-- <bs-notification value="50"></bs-notification>
        <br />
        <br />
        <bs-notification value="50" max="30"></bs-notification>
        <br />
        <br />
        <bs-notification value="50" :is-dot="true"></bs-notification>
        <br />
        <br />
        <bs-notification value="50" icon="el-icon-chat-round"></bs-notification> -->
        <bs-notification value="50">
            <template #default>
                <bs-list :list="list" :actions="actions"></bs-list>
            </template>
        </bs-notification>
    </div>
</template>
<script lang="ts" setup>
import { list, actions } from './data'
</script>
<style lang="scss" scoped>
.bs-wrapper {
    .flex {
        display: flex;
    }
    div {
        margin-right: 0.1rem;
    }
}
</style>

修改src\components\baseline\list\src\index.vue

<!--
 * @Author: your name
 * @Date: 2021-12-23 00:07:25
 * @LastEditTime: 2021-12-29 01:50:17
 * @LastEditors: Please set LastEditors
 * @Description: 打开koroFileHeader查看配置 进行设置: https://github.com/OBKoro1/koro1FileHeader/wiki/%E9%85%8D%E7%BD%AE
 * @FilePath: \vue3-element-ui-baseline\src\components\baseline\trend\src\index.vue
-->
<template>
    <div class="bs-wrapper">
        <el-tabs class="tabs">
            <el-tab-pane
                v-for="(item, index) in list"
                v-model="activeName"
                :key="index"
                :label="item.title"
                :name="index + ''"
            >
                <div class="list">
                    <el-scrollbar max-height="3rem">
                        <!-- 待办信息 -->
                        <div
                            class="list__box"
                            v-for="(item2, index2) in item.content"
                            :key="index2"
                        >
                            <!-- 头像 -->
                            <div class="list__box__avatar">
                                <el-avatar
                                    size="small"
                                    :src="item2.avatar"
                                ></el-avatar>
                            </div>
                            <div class="list__box__content">
                                <div
                                    v-if="item2.title"
                                    class="list__box__content__title"
                                >
                                    <!-- 主题和报警 -->
                                    <div class="title">{{ item2.title }}</div>
                                    <el-tag
                                        class="tag"
                                        size="mini"
                                        v-if="item2.tag"
                                        :type="item2.tagType"
                                    >
                                        {{ item2.tag }}
                                    </el-tag>
                                </div>
                                <!-- 说明 -->
                                <div
                                    v-if="item2.desc"
                                    class="list__box__content__desc"
                                >
                                    {{ item2.desc }}
                                </div>
                                <!-- 时间 -->
                                <div
                                    v-if="item2.time"
                                    class="list__box__content__time"
                                >
                                    {{ item2.time }}
                                </div>
                            </div>
                        </div>
                    </el-scrollbar>
                </div>

                <div class="actions">
                    <div
                        class="actions__content"
                        :class="{ border: actionIndex + 1 !== actions.length }"
                        v-for="(actionItem, actionIndex) in actions"
                        :key="index"
                    >
                        <div
                            v-if="actionItem.icon"
                            class="actions__content__icon"
                        >
                            <component :is="actionItem.icon"></component>
                        </div>
                        <div class="actions__content__text">
                            {{ actionItem.text }}
                        </div>
                    </div>
                </div>
            </el-tab-pane>
        </el-tabs>
    </div>
</template>
<script lang="ts" setup>
import { ref, PropType } from 'vue'
import { ActionOptions, listOptions } from './type'
const props = defineProps({
    list: {
        type: Array as PropType<listOptions[]>,
        required: true,
    },
    //操作的内容
    actions: {
        type: Array as PropType<ActionOptions[]>,
        default: [],
    },
})
const activeName = ref('0')
// console.log('props.list', props.list)
// console.log('props.actions', props.actions)
</script>
<style lang="scss" scoped>
.bs-wrapper {
    width: 100%;
    height: 100%;
    margin: 0;
    padding: 0;
}
.tabs {
    //让tabs值占满格分布并横向摆放
    ::v-deep(.el-tabs__nav) {
        width: 100%;
        display: flex;
    }
    ::v-deep(.el-tabs__item) {
        flex: 1;
        text-align: center;
    }
    .list {
        padding-bottom: 0.04rem;
        &__box {
            display: flex;
            align-items: center;
            padding: 0.12rem 0.2rem;
            cursor: pointer; //鼠标悬浮变小手
            &:hover {
                background-color: #e6f6ff;
            }
            &__avatar {
                flex: 1;
            }
            &__content {
                flex: 3;
                &__title {
                    display: flex;
                    align-items: center; //垂直方向居中
                    justify-content: space-between; //水平方向两边排列
                    .title {
                    }
                    .tag {
                    }
                }
                &__desc,
                &__time {
                    font-size: 0.12rem;
                    color: #999;
                    margin-top: 0.04rem;
                }
            }
        }
    }

    .actions {
        height: 0.5rem;
        display: flex;
        align-items: center;
        &__content {
            height: 0.5rem;
            cursor: pointer; //鼠标悬浮变小手
            flex: 1;
            display: flex;
            align-items: center;
            justify-content: center; //水平方向两边排列
            border-top: 2px solid #eee;

            // &:hover {
            //     background-color: #e6f6ff;
            // }
            &__icon {
                margin-right: 0.08rem;
                position: relative;
                top: 0.02rem;
            }
            &__text {
            }
        }
    }
}
.border {
    border-right: 2px solid #eee;
}
</style>

修改src\style\ui.scss

//修改组件库内部的样式
//1.需要自定义一个类名空间
//2.浏览器中调试样式
//3.调试好的类名放在这个类名中
//4.在App.vue里面引入这个文件
//5.在组件内需要改样式的元素的父元素加上这个类名
.bk--choose-icon-dialog-body-height {
    .el-dialog__body {
        height: 5rem;
        overflow-y: scroll;
        overflow-x: auto;
    }
}
//去掉el-tabs盒子默认的padding
.el-popper {
    padding: 0 !important;
}

效果如下:

image.png

image.png

list组件完善

修改src\components\baseline\list\src\index.vue

<!--
 * @Author: your name
 * @Date: 2021-12-23 00:07:25
 * @LastEditTime: 2021-12-29 02:17:41
 * @LastEditors: Please set LastEditors
 * @Description: 打开koroFileHeader查看配置 进行设置: https://github.com/OBKoro1/koro1FileHeader/wiki/%E9%85%8D%E7%BD%AE
 * @FilePath: \vue3-element-ui-baseline\src\components\baseline\trend\src\index.vue
-->
<template>
    <div class="bs-wrapper">
        <el-tabs class="tabs">
            <el-tab-pane
                v-for="(item, index) in list"
                v-model="activeName"
                :key="index"
                :label="item.title"
                :name="index + ''"
            >
                <div class="list">
                    <el-scrollbar max-height="3rem">
                        <!-- 待办信息 -->
                        <div
                            class="list__box"
                            v-for="(item2, index2) in item.content"
                            :key="index2"
                            @click="handleChickList(item2, index2)"
                        >
                            <!-- 头像 -->
                            <div class="list__box__avatar">
                                <el-avatar
                                    size="small"
                                    :src="item2.avatar"
                                ></el-avatar>
                            </div>
                            <div class="list__box__content">
                                <div
                                    v-if="item2.title"
                                    class="list__box__content__title"
                                >
                                    <!-- 主题和报警 -->
                                    <div class="title">{{ item2.title }}</div>
                                    <el-tag
                                        class="tag"
                                        size="mini"
                                        v-if="item2.tag"
                                        :type="item2.tagType"
                                    >
                                        {{ item2.tag }}
                                    </el-tag>
                                </div>
                                <!-- 说明 -->
                                <div
                                    v-if="item2.desc"
                                    class="list__box__content__desc"
                                >
                                    {{ item2.desc }}
                                </div>
                                <!-- 时间 -->
                                <div
                                    v-if="item2.time"
                                    class="list__box__content__time"
                                >
                                    {{ item2.time }}
                                </div>
                            </div>
                        </div>
                    </el-scrollbar>
                </div>

                <div class="actions">
                    <div
                        class="actions__content"
                        :class="{ border: actionIndex + 1 !== actions.length }"
                        v-for="(actionItem, actionIndex) in actions"
                        :key="index"
                        @click="handleChickActions(actionItem, actionIndex)"
                    >
                        <div
                            v-if="actionItem.icon"
                            class="actions__content__icon"
                        >
                            <component :is="actionItem.icon"></component>
                        </div>
                        <div class="actions__content__text">
                            {{ actionItem.text }}
                        </div>
                    </div>
                </div>
            </el-tab-pane>
        </el-tabs>
    </div>
</template>
<script lang="ts" setup>
import { ref, PropType } from 'vue'
import { ActionOptions, listOptions, ListItem } from './type'
const props = defineProps({
    list: {
        type: Array as PropType<listOptions[]>,
        required: true,
    },
    //操作的内容
    actions: {
        type: Array as PropType<ActionOptions[]>,
        default: [],
    },
})

const activeName = ref('0')
const emits = defineEmits(['chickList', 'chickActions'])
const handleChickList = (item: ListItem, index: number) => {
    emits('chickList', { item, index })
}
const handleChickActions = (item: ActionOptions, index: number) => {
    emits('chickActions', { item, index })
}

// console.log('props.list', props.list)
// console.log('props.actions', props.actions)
</script>
<style lang="scss" scoped>
.bs-wrapper {
    width: 100%;
    height: 100%;
    margin: 0;
    padding: 0;
}
.tabs {
    //让tabs值占满格分布并横向摆放
    ::v-deep(.el-tabs__nav) {
        width: 100%;
        display: flex;
    }
    ::v-deep(.el-tabs__item) {
        flex: 1;
        text-align: center;
    }
    .list {
        padding-bottom: 0.04rem;
        &__box {
            display: flex;
            align-items: center;
            padding: 0.12rem 0.2rem;
            cursor: pointer; //鼠标悬浮变小手
            &:hover {
                background-color: #e6f6ff;
            }
            &__avatar {
                flex: 1;
            }
            &__content {
                flex: 3;
                &__title {
                    display: flex;
                    align-items: center; //垂直方向居中
                    justify-content: space-between; //水平方向两边排列
                    .title {
                    }
                    .tag {
                    }
                }
                &__desc,
                &__time {
                    font-size: 0.12rem;
                    color: #999;
                    margin-top: 0.04rem;
                }
            }
        }
    }

    .actions {
        height: 0.5rem;
        display: flex;
        align-items: center;
        &__content {
            height: 0.5rem;
            cursor: pointer; //鼠标悬浮变小手
            flex: 1;
            display: flex;
            align-items: center;
            justify-content: center; //水平方向两边排列
            border-top: 2px solid #eee;

            // &:hover {
            //     background-color: #e6f6ff;
            // }
            &__icon {
                margin-right: 0.08rem;
                position: relative;
                top: 0.02rem;
            }
            &__text {
            }
        }
    }
}
.border {
    border-right: 2px solid #eee;
}
</style>

修改src\views\baseline\notification\index.vue

<!--
 * @Author: your name
 * @Date: 2021-12-23 00:07:25
 * @LastEditTime: 2021-12-29 02:16:52
 * @LastEditors: Please set LastEditors
 * @Description:组件五:通知菜单
 * @FilePath: \vue3-element-ui-baseline\src\components\baseline\trend\src\index.vue
-->
<template>
    <div class="bs-wrapper">
        <!-- <bs-notification value="50"></bs-notification>
        <br />
        <br />
        <bs-notification value="50" max="30"></bs-notification>
        <br />
        <br />
        <bs-notification value="50" :is-dot="true"></bs-notification>
        <br />
        <br />
        <bs-notification value="50" icon="el-icon-chat-round"></bs-notification> -->
        <bs-notification value="50">
            <template #default>
                <bs-list
                    :list="list"
                    :actions="actions"
                    @chickList="handleChickList"
                    @chickActions="handleChickActions"
                ></bs-list>
            </template>
        </bs-notification>
    </div>
</template>
<script lang="ts" setup>
import { list, actions } from './data'
const handleChickList = (val: any) => {
    console.log('handleChickList:', val)
}
const handleChickActions = (val: any) => {
    console.log('handleChickActions:', val)
}
</script>
<style lang="scss" scoped>
.bs-wrapper {
    .flex {
        display: flex;
    }
    div {
        margin-right: 0.1rem;
    }
}
</style>

这样就实现了点击响应.

如上代码为了兼容别名使用,修改vite.config.ts
增加如下代码:

    resolve: {
        alias: {
            '@': '/src',
            '@style': '/src/style',
            '@com': '/src/components',
            '@baseline': '/src/components/baseline',
            '@business': '/src/components/business',
        },
    },

完整代码如下:

import { defineConfig } from 'vite'
import vue from '@vitejs/plugin-vue'

// https://vitejs.dev/config/
export default defineConfig({
    plugins: [vue()],
    resolve: {
        alias: {
            '@': '/src',
            '@style': '/src/style',
            '@com': '/src/components',
            '@baseline': '/src/components/baseline',
            '@business': '/src/components/business',
        },
    },
    server: {
        port: 8080,
    },
})

修改tsconfig.json
增加如下代码:

     "types": ["vite/client"],
        // ++ 这里加上baseUrl 和 path即可 ++
        "baseUrl": "./",
        "paths": {
            // 根据别名配置相关路径
            "@/*": ["./src/*"],
            "@style/*": ["./src/style/*"],
            "@com/*": ["./src/components/*"],
            "@baseline/*": ["./src/components/baseline/*"],
            "@business/*": ["./src/components/business/*"]
        }

完整代码如下:

{
    "compilerOptions": {
        "target": "esnext",
        "useDefineForClassFields": true,
        "module": "esnext",
        "moduleResolution": "node",
        "strict": true,
        "jsx": "preserve",
        "sourceMap": true,
        "resolveJsonModule": true,
        "esModuleInterop": true,
        "forceConsistentCasingInFileNames": true,
        "importHelpers": true,
        "isolatedModules": true,
        "lib": ["esnext", "dom"],
        "types": ["vite/client"],
        // ++ 这里加上baseUrl 和 path即可 ++
        "baseUrl": "./",
        "paths": {
            // 根据别名配置相关路径
            "@/*": ["./src/*"],
            "@style/*": ["./src/style/*"],
            "@com/*": ["./src/components/*"],
            "@baseline/*": ["./src/components/baseline/*"],
            "@business/*": ["./src/components/business/*"]
        }
    },
    "include": ["src/**/*.ts", "src/**/*.d.ts", "src/**/*.tsx", "src/**/*.vue"]
}

结果如下:


image.png
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 215,923评论 6 498
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,154评论 3 392
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 161,775评论 0 351
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,960评论 1 290
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,976评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,972评论 1 295
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,893评论 3 416
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,709评论 0 271
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,159评论 1 308
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,400评论 2 331
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,552评论 1 346
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,265评论 5 341
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,876评论 3 325
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,528评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,701评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,552评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,451评论 2 352

推荐阅读更多精彩内容