逐行超详细讲解:Vue3 + TS + Ant Design Vue 全局头部组件(小白也能看懂)
大家好~ 今天给各位小白带来一篇「保姆级」代码讲解,针对 Vue3 + TypeScript + Ant Design Vue 开发的全局头部组件(GlobalHeader.vue),全程逐行、逐标签、逐语法拆解,不跳步、不省略、不堆砌专业黑话,哪怕你是刚入门前端,跟着看也能彻底看懂每一行代码的作用,还能直接复制复用!
先说明下:这篇讲解的核心是「搞懂每一行是什么、为什么这么写、来自哪里」,重点解决小白最困惑的「标签/组件不知道哪来的」「属性看不懂」「样式为什么这么写」三个问题,全程干货,建议收藏慢慢看~
一、组件整体介绍(先有全局认知)
我们今天讲解的 GlobalHeader.vue,是前端项目中最常用的「全局顶部导航栏」—— 也就是说,项目中所有页面都会共用这个组件,相当于网站的“门面”,主要包含 4 个核心部分:
- 左侧:Logo 图片 + 网站标题(User Center)
- 中间:导航菜单(主页、用户登录、用户注册、用户管理、编程导航)
- 右侧:登录按钮
- 整体:深色渐变风格,支持手机端响应式适配(电脑、手机都能正常显示)
用到的技术栈(小白不用怕,后面逐一提及):Vue3 + TypeScript + setup 语法糖 + Ant Design Vue 组件库 + CSS 渐变/动画 + Vue Router 路由
先放完整代码(方便大家对照讲解查看),后面我们逐行拆解:
<template>
<div class="global-header">
<div class="logo">
<img src="@/assets/logo.png" alt="logo" class="logo-img" />
<span class="logo-text">User Center</span>
</div>
<div class="nav-container">
<a-menu
v-model:selectedKeys="current"
mode="horizontal"
:items="items"
class="nav-menu"
:theme="'dark'"
@click="handleMenuClick"
/>
</div>
<div class="user-info">
<a-button type="primary" ghost> Login </a-button>
</div>
</div>
</template>
<script lang="ts" setup>
import { h, ref, watch } from "vue";
import { useRouter, useRoute } from "vue-router";
import {
HomeOutlined,
UserOutlined,
} from "@ant-design/icons-vue";
import { MenuProps } from "ant-design-vue";
const router = useRouter();
const route = useRoute();
const current = ref<string[]>([route.path]);
watch(
() => route.path,
(newPath) => {
current.value = [newPath];
}
);
const handleMenuClick = ({ key }: { key: string }) => {
router.push(key);
};
const items = ref<MenuProps["items"]>([
{
key: "/",
icon: () => h(HomeOutlined),
label: "主页",
title: "主页",
},
{
key: "/user/login",
label: "用户登录",
title: "用户登录",
},
{
key: "/user/register",
label: "用户注册",
title: "用户注册",
},
{
key: "/admin/userManage",
label: "用户管理",
title: "用户管理",
},
{
key: "/yupi",
label: "编程导航",
title: "编程导航",
},
]);
</script>
<style scoped>
.global-header {
display: flex;
align-items: center;
justify-content: flex-start;
width: 100%;
height: 100%;
padding: 0 16px;
background: linear-gradient(135deg, #1a1a2e 0%, #16213e 100%);
box-shadow: 0 2px 12px rgba(0, 0, 0, 0.15);
}
.logo {
display: flex;
align-items: center;
margin-right: 24px;
cursor: pointer;
transition: transform 0.3s ease;
}
.logo:hover {
transform: scale(1.02);
}
.logo-img {
width: 36px;
height: 36px;
margin-right: 14px;
border-radius: 8px;
box-shadow: 0 2px 8px rgba(0, 0, 0, 0.2);
transition: transform 0.3s ease, box-shadow 0.3s ease;
}
.logo:hover .logo-img {
transform: rotate(5deg);
box-shadow: 0 4px 12px rgba(0, 0, 0, 0.3);
}
.logo-text {
font-size: 22px;
font-weight: 700;
color: #fff;
letter-spacing: 1.5px;
text-shadow: 0 2px 4px rgba(0, 0, 0, 0.2);
background: linear-gradient(135deg, #ffffff 0%, #e0f7fa 100%);
-webkit-background-clip: text;
-webkit-text-fill-color: transparent;
background-clip: text;
}
.nav-container {
flex: 0 0 auto;
display: flex;
justify-content: flex-start;
}
.nav-menu {
border-bottom: none;
background: transparent;
font-size: 15px;
font-weight: 500;
}
.nav-menu :deep(.ant-menu-item) {
border-radius: 6px;
margin: 0 2px;
padding: 0 16px;
transition: all 0.3s ease;
color: rgba(255, 255, 255, 0.85) !important;
}
.nav-menu :deep(.ant-menu-item:hover) {
background: rgba(255, 255, 255, 0.1);
transform: translateY(-2px);
color: #fff !important;
}
.nav-menu :deep(.ant-menu-item-selected) {
background: rgba(255, 255, 255, 0.15) !important;
font-weight: 600;
color: #fff !important;
}
.user-info {
display: flex;
align-items: center;
gap: 16px;
}
.user-info {
margin-left: auto;
display: flex;
align-items: center;
gap: 12px;
}
.user-info :deep(.ant-btn) {
border-radius: 20px;
padding: 0 20px;
height: 34px;
font-weight: 500;
transition: all 0.3s ease;
border-width: 2px;
background: #fff;
color: #1a1a2e;
border-color: #fff;
}
.user-info :deep(.ant-btn:hover) {
transform: translateY(-2px);
box-shadow: 0 4px 12px rgba(255, 255, 255, 0.3);
background: #f0f0f0;
border-color: #f0f0f0;
}
/* 响应式设计 */
@media (max-width: 768px) {
.global-header {
padding: 0 12px;
}
.logo {
margin-right: 16px;
}
.logo-text {
font-size: 16px;
}
.logo-img {
width: 28px;
height: 28px;
}
.nav-menu :deep(.ant-menu-item) {
padding: 0 10px;
font-size: 13px;
}
.user-info :deep(.ant-btn) {
padding: 0 14px;
height: 30px;
font-size: 13px;
}
}
</style>