逐行超详细讲解:Vue3 + TS + Ant Design Vue 全局头部组件(小白也能看懂)

4 阅读23分钟

逐行超详细讲解: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>

二、