Bootstrap

Vue实训---5-路由搭建

回顾之前的代码

我们在my-vue-project\src\router\index.js中的代码如下:

// 什么是路由?路由就是路径和组件的对应关系
// 1.引入vue-router
import { createRouter, createWebHashHistory } from 'vue-router'

// 2.定义路由
const routes = [
    {
        path: '/',    // 路由路径
        name: 'main', // 路由名称
        component: () => import('@/views/Main.vue') // 路由组件
    },
]

// 创建路由实例
const router = createRouter({
    // 设置路由模式, hash模式
    history: createWebHashHistory(),
    routes, // 注入路由配置
})

// 3.导出路由
export default router 

表示我们定义了一个url路径:'/'(即默认路径/)与Main.vue组件的对应关系,然后当项目启动后就默认进到默认路径 '/' 中,即会显示Main.vue路由。Mian.vue组件具体显示在App.vue的路由视图<router-view></router-view>标签的位置。

1.新建组件

在src/views/文件夹下新建Home.vue、Mall.vue、User.vue、Page1.vue、Page2.vue组件,组件内容如下:

<template>
    <div>
        home组件(如果是其他组件的话名字修改一下)
    </div>
</template>

<script setup>

</script>

<style lang="less" scoped></style>

2.修改路由配置

my-vue-project\src\router\index.js

// 什么是路由?路由就是url地址和组件的对应关系

// 1.引入vue-router
import { createRouter, createWebHashHistory } from "vue-router";

// 2.定义路由
const routes = [
  {
    path: "/",
    name: "main",
    component: () => import("@/views/Main.vue"),
    redirect: "/home", //路由重定向
    children: [
      {
        path: "home",
        name: "home",
        component: () => import("@/views/Home.vue"),
      },
      {
        path: "user",
        name: "user",
        component: () => import("@/views/User.vue"),
      },
      {
        path: "mall",
        name: "mall",
        component: () => import("@/views/Mall.vue"),
      },
      {
        path: "page1",
        name: "page1",
        component: () => import("@/views/Page1.vue"),
      },
      {
        path: "page2",
        name: "page2",
        component: () => import("@/views/Page2.vue"),
      },
    ],
  },
];

// 创建路由实例
const router = createRouter({
  // 设置路由模式, hash模式
  history: createWebHashHistory(),
  routes, // 注入路由配置
});

// 3.导出路由
export default router;

3.为Mian.vue的子路由设置路由视图

在Main.vue中的el-main标签内部添加<router-view></router-view>路由视图,表示其子路由会显示在这个位置。

-- 在地址栏输入url地址验证

默认地址页面http://localhost:5173/#/ 会重定向到 http://localhost:5173/#/home,显示

地址页面http://localhost:5173/#/user,显示:

很棒!

5.配置左侧菜单栏与路由的联动

实现点击左侧菜单栏,右侧显示菜单栏项对应的组件。(即我们上一步实现的是地址栏URL与右侧组件的一一对应,现在实现左侧菜单栏与右侧组件的对应)

在CommonAside.vue组件中,找到标签el-menu-item(一共有两个),添加点击事件@click="handleMenu(传参)"

<!-- 循环遍历菜单栏 -->

<!-- 1.没有子菜单的遍历 -->

<el-menu-item @click="handleMenu(item)" v-for="item in noChildren" :key="item.path" :index="item.path">

<!-- 循环遍历子菜单 -->
<el-menu-item @click="handleMenu(child)" v-for="child in item.children" :key="child.path" :index="child.path">
    <!-- 动态获取icon图标 -->
    <component :is="child.icon" class="icons"></component>
    <!-- 显示文字内容 -->
    <span>{{ child.label }}</span>
</el-menu-item> 

继续在script中添加代码:

// 路由

import { useRouter, useRoute } from 'vue-router';

const router = useRouter()

// handleMenu方法的实现

const handleMenu = (item) => {

    router.push(item.path)

}

router.push() 是 Vue Router 提供的一个方法,用于在单页面应用(SPA)中进行路由路径跳转。该方法会改变浏览器的历史记录,并加载对应路径的组件,而不会刷新页面。

 即CommonAside.vue修改后的完整代码为:

<template>
    <!-- 首先最外层是Main.vue中注释的<el-aside width="200px" class="el-aside">Aside</el-aside>,我们将侧边栏改成我们自定义的组件,通过common-aside引入我们自己写的组件 -->
    <el-aside :width="width" class="el-aside">
        <el-menu background-color="#545c64" text-color="#fff" :collapse="isCollapse" :collapse-transition="false">
            <!-- 写菜单栏的大标题————通用后台管理系统 -->
            <h3 v-show="!isCollapse">通用后台管理系统</h3>
            <h3 v-show="isCollapse">后台</h3>
            <!-- 循环遍历菜单栏 -->
            <!-- 1.没有子菜单的遍历 -->
            <el-menu-item @click="handleMenu(item)" v-for="item in noChildren" :key="item.path" :index="item.path">
                <!-- <el-icon><icon-menu /></el-icon> -->
                <!-- 动态获取icon图标 -->
                <component :is="item.icon" class="icons"></component>
                <!-- 显示文字内容 -->
                <span>{{ item.label }}</span>
            </el-menu-item>
            <!-- 2.有子菜单的遍历 -->
            <el-sub-menu v-for="item in hasChildren" :key="item.path" :index="item.path">
                <template #title>
                    <!-- 动态获取icon图标 -->
                    <component :is="item.icon" class="icons"></component>
                    <!-- 显示文字内容 -->
                    <span>{{ item.label }}</span>
                </template>
                <!-- 分组 -->
                <el-menu-item-group>
                    <!-- 循环遍历子菜单 -->
                    <el-menu-item @click="handleMenu(child)" v-for="child in item.children" :key="child.path"
                        :index="child.path">
                        <!-- 动态获取icon图标 -->
                        <component :is="child.icon" class="icons"></component>
                        <!-- 显示文字内容 -->
                        <span>{{ child.label }}</span>
                    </el-menu-item>
                </el-menu-item-group>
            </el-sub-menu>
        </el-menu>
    </el-aside>
</template>

<script setup>
import { ref, computed } from 'vue'
import { storeToRefs } from 'pinia';
import { useAllDateStore } from '@/stores/index';
// 获取 store 实例
const store = useAllDateStore();
// 使用 storeToRefs 解构响应式属性
const { isCollapse } = storeToRefs(store);
// 根据 isCollapse 来计算宽度
const width = computed(() => isCollapse.value ? '64px' : '200px')

const list = ref([
    {
        path: '/home',
        name: 'home',
        label: '首页',
        icon: 'house',
        url: 'Home'
    },
    {
        path: '/mall',
        name: 'mall',
        label: '商品管理',
        icon: 'video-play',
        url: 'Mall'
    },
    {
        path: '/user',
        name: 'user',
        label: '用户管理',
        icon: 'user',
        url: 'User'
    },
    {
        path: 'other',
        label: '其他',
        icon: 'location',
        children: [
            {
                path: '/page1',
                name: 'page1',
                label: '页面1',
                icon: 'setting',
                url: 'Page1'
            },
            {
                path: '/page2',
                name: 'page2',
                label: '页面2',
                icon: 'setting',
                url: 'Page2'
            }
        ]
    }
])
const noChildren = computed(() => list.value.filter(item => !item.children))
const hasChildren = computed(() => list.value.filter(item => item.children))

// 路由
import { useRouter, useRoute } from 'vue-router';
const router = useRouter()
// handleMenu方法的实现
const handleMenu = (item) => {
    router.push(item.path)
}
</script>

<style lang="less" scoped>
.icons {
    width: 18px;
    height: 18px;
    margin-right: 5px;
}

.el-menu {
    border-right: none;

    h3 {
        line-height: 48px;
        color: #fff;
        text-align: center;
    }
}

.el-aside {
    height: 100%;
    background-color: #545c64;
}
</style>
--点击左侧菜单栏验证

点击“商品管理”:

点击“页面1”:

成功!

项目源代码:

通过百度网盘分享的文件:实现路由
链接:https://pan.baidu.com/s/1WaOLqig2MV-x-nv1M-6bMA?pwd=6666 
提取码:6666

;