Layout 布局组件快速搭建

news2025/1/10 1:51:30

文章目录

  • 设置主题样式变量
  • 封装公共布局组件
    • 封装 Logo 组件
    • 封装 Menu 菜单组件
    • 封装 Breadcrumb 面包屑组件
    • 封装 TabBar 标签栏组件
    • 封装 Main 内容区组件
    • 封装 Footer 底部组件
    • 封装 Theme 主题组件
  • 经典布局
  • 水平布局
  • 响应式布局
  • 搭建 Layout 布局组件
  • 添加 Layout 路由配置
  • 启动项目

设置主题样式变量

  1. 创建主题变量文件:src/assets/styles/theme.scss

  2. 添加页面布局组件用到的变量:

    :root {
      --os-layout-aside-width: 224px; // 侧边栏宽度
      --os-layout-logo-height: 56px; // 头部高度
      --os-layout-header-height: 56px; // 头部高度
      --os-layout-header-menu-width: 300px; // 头部菜单宽度
      --os-layout-footer-height: 40px; // 底部高度
      --os-layout-tab-height: 40px; // 底部高度
    }
    
  3. 设置全局滚动条样式:

    /* scroll bar */
    ::-webkit-scrollbar {
        width: 6px;
        height: 6px;
    }
    
    ::-webkit-scrollbar-thumb {
        background-color: var(--el-border-color-darker);
        border-radius: 20px;
    }
    
  4. 在统一管理和维护项目的样式文件src/assets/styles/index.scss中导入主题样式文件:

    @import "theme.scss";
    

注:统一管理和维护项目的样式文件src/assets/styles/index.scss需在mian.ts中导入,注意导入顺序。

封装公共布局组件

封装 Logo 组件

  1. 创建 Logo 组件:src/layouts/components/Logo/index.vue

  2. 通过import.meta.env获取.env文件中定义的环境变量,环境变量使用详情跳转认识和使用 Vite 环境变量配置:

    <script setup lang="ts">
    const app_title = import.meta.env.VITE_APP_TITLE
    </script>
    
  3. Logo 组件内容,添加 Logo 图标和项目标题:

    <template>
      <!-- logo 容器 -->
      <div class="logo-container">
        <!--  logo 图标 -->
        <svg-icon width="32px" height="32px" name="logo" />
    
        <!-- 应用标题,只在中等及以上屏幕尺寸上可见 -->
        <span class="logo-title hidden md:block">{{ app_title }}</span>
      </div>
    </template>
    
  4. 添加样式:

    <style scoped lang="postcss">
    .logo-container {
      @apply flex items-center justify-center;
      height: var(--os-layout-logo-height);
      width: var(--os-layout-aside-width);
    
      .logo-title {
        @apply text-lg font-bold subpixel-antialiased;
        @apply bg-clip-text text-transparent bg-gradient-to-r from-pink-500 to-violet-500;
      }
    }
    </style>
    

封装 Menu 菜单组件

  1. 创建 Menu 菜单组件:src/layouts/components/Menu/index.vue

  2. el-menu 组件中使用 props 来控制 mode 属性,可以实现菜单的不同显示模式(如 verticalhorizontal):

    <script setup lang="ts">
    const props = defineProps<{
      mode?: 'vertical' | 'horizontal'
    }>()
    </script>
    
  3. 使用 ElementPlus 的 el-menu 菜单组件搭建垂直菜单:

    <template>
      <!-- 菜单容器 -->
      <div class="menu-container">
        <!-- Element UI 菜单组件 -->
        <el-menu
          class="os-menu"
          :mode="props.mode"
        >
          <!-- 一级菜单 -->
          <el-sub-menu index="1">
            <!-- 一级菜单标题 -->
            <template #title>
              <span>Navigator One</span>
            </template>
              
            <!-- 二级菜单 -->
            <el-menu-item index="1-1">item one</el-menu-item>
            <el-menu-item index="1-2">item two</el-menu-item>
          </el-sub-menu>
            
          <!-- 一级菜单 -->
          <el-menu-item index="2">
            <span>Navigator Two</span>
          </el-menu-item>
        </el-menu>
      </div>
    </template>
    
  4. 添加样式:

    <style scoped lang="postcss">
    .menu-container{
      background-color: var(--el-menu-bg-color);
    
      .os-menu{
        @apply border-0;
    
        &.el-menu--horizontal{
          height: calc(var(--os-layout-logo-height) - 1px);
        }
      }
    }
    </style>
    

封装 Breadcrumb 面包屑组件

  1. 创建 src/layouts/components/Breadcrumb/index.vue

  2. 使用 ElementPlus el-breadcrumb 面包屑组件:

    <template>
      <el-breadcrumb separator="/" class="breadcrumb-container">
        <el-breadcrumb-item :to="{ path: '/' }">homepage</el-breadcrumb-item>
        <el-breadcrumb-item>management</el-breadcrumb-item>
        <el-breadcrumb-item>list</el-breadcrumb-item>
        <el-breadcrumb-item>detail</el-breadcrumb-item>
      </el-breadcrumb>
    </template>
    
  3. 添加样式:

    <style scoped lang="postcss">
    .breadcrumb-container {
      display: flex;
      flex-wrap: nowrap; /* 防止换行 */
      overflow: hidden; /* 横向滚动溢出处理 */
      padding-right: 12px;
    }
    </style>
    

封装 TabBar 标签栏组件

  1. 创建 src/layouts/components/TabBar/index.vue

  2. 生成 TabBar 标签数据:

    <script setup lang="ts">
    const tabs = Array(1).fill({
      label: 'About',
      name: 'about',
      icon: 'About',
      path: '/about'
    })
    </script>
    
  3. 使用 ElementPlus el-tabs 标签组件,Scrollbar 组件替换浏览器原生滚动条:

    <template>
      <!-- Tab Bar -->
      <div class="tab-bar-container">
        <div class="tab-bar-box">
          <el-scrollbar>
            <el-tabs
              tab-position="top"
              type="card"
              closable
            >
              <el-tab-pane
                v-for="(item, index) in tabs"
                :key="index"
                :name="item.name"
              >
                <template #label>
                    <span class="tab-title">
                      <svg-icon :name="item.icon" />
                      <span>{{ item.label }}</span>
                    </span>
                </template>
              </el-tab-pane>
            </el-tabs>
          </el-scrollbar>
        </div>
        <div class="more-button">
          <el-dropdown trigger="click">
            <span>
              <svg-icon name="More" />
            </span>
            <template #dropdown>
              <el-dropdown-menu>
                <el-dropdown-item>关闭</el-dropdown-item>
              </el-dropdown-menu>
            </template>
          </el-dropdown>
        </div>
      </div>
    </template>
    
  4. 添加样式:

    <style scoped lang="scss">
    .tab-bar-container {
      display: flex;
      align-items: center;
      gap: 8px;
      border-bottom: 1px solid var(--el-border-color-light);
    
      .tab-bar-box {
        display: flex;
        flex-grow: 1;
        margin-left: 8px;
        width: calc(100% - 60px);
    
        .tab-title{
          display: flex;
          align-items: center;
          gap: 8px;
        }
      }
    
      .more-button {
        display: flex;
        align-items: center;
        justify-content: center;
        width: var(--os-layout-tab-height);
        height: var(--os-layout-tab-height);
        border-left: 1px solid var(--el-border-color-light);
    
      }
    }
    
    :deep(.el-tabs) {
      .el-tabs__header {
        height: var(--os-layout-tab-height);
        padding: 0;
        margin: 0;
        border-bottom: none;
    
        .el-tabs__nav-wrap {
          .el-tabs__nav-scroll {
            .el-tabs__nav {
              border: none;
    
              .el-tabs__item {
                height: calc(var(--os-layout-tab-height) - 2px);
                line-height: calc(var(--os-layout-tab-height) - 2px);
                border: none;
                position: relative; // 确保父元素是相对定位的
    
                &::before {
                  content: '';
                  position: absolute;
                  bottom: 0;
                  width: 100%;
                  height: 2px;
                  background-color: transparent;
                }
    
                &.is-active {
                  color: var(--el-color-primary);
                  fill: var(--el-color-primary);
    
                  &::before {
                    background-color: var(--el-color-primary); // 改变背景色以显示下划线
                  }
                }
              }
            }
          }
        }
      }
    }
    </style>
    

封装 Main 内容区组件

  1. 创建 src/layouts/components/Main/index.vue

    <script setup lang="ts">
    
    </script>
    
    <template>
      <div>Main</div>
    </template>
    
    <style scoped lang="scss">
    
    </style>
    
  2. 后续补充内容…

封装 Footer 底部组件

  1. 创建 src/layouts/components/Footer/index.vue

  2. 通过import.meta.env获取.env文件中定义的环境变量

    <script setup lang="ts">
    const app_title = import.meta.env.VITE_APP_TITLE
    </script>
    
  3. 添加内容:

    <template>
      <div class="footer-container">
        <!-- 左侧:公司名称或标志 -->
        <div class="flex-content">
          <svg-icon name="logo" />
          <span class="logo-title">{{ app_title }}</span>
        </div>
        <!-- 中间:导航链接 -->
        <div class="flex-content">
          <a href="about">关于我们</a>
          <a href="#contact">联系我们</a>
          <a href="#privacy-policy">隐私政策</a>
          <a href="#terms-of-service">服务条款</a>
        </div>
        <!-- 右侧:版权信息 -->
        <div class="text-right">&copy; 2024 Octopus. 保留所有权利.</div>
      </div>
    </template>
    
  4. 添加样式:

    <style scoped lang="postcss">
    .footer-container{
      @apply flex items-center w-full;
      @apply select-none text-xs;
      height: var(--os-layout-footer-height);
      @apply justify-center lg:justify-between;
    
      .flex-content{
        @apply items-center gap-2;
        @apply flex justify-center gap-1 text-gray-400;
        @apply hidden lg:flex;
    
        .logo-title {
          @apply font-bold subpixel-antialiased text-sm;
          @apply bg-clip-text text-transparent bg-gradient-to-r from-pink-500 to-violet-500;
        }
      }
    }
    </style>
    

封装 Theme 主题组件

  1. 创建 src/stores/modules/theme/types.ts 定义主题类型:

    export type LayoutMode = 'classic' | 'horizontal' | 'responsive'
    // 主题配置
    export interface ThemeConfig {
      showThemeConfig: boolean,
      darkThemeEnabled: boolean,
      grayThemeEnabled: boolean,
      layoutMode: LayoutMode,
      themePrimaryColor: string,
      predefineColors: string[]
    }
    
  2. 创建 src/stores/modules/theme/index.ts 仓库存储主题状态:

    import { defineStore } from 'pinia'
    import { reactive, watch } from 'vue'
    import { ThemeConfig } from './types'
    import { ElMessage } from 'element-plus'
    import { getDarkColor, getLightColor } from '@/utils/Color'
    
    // theme store
    export const useThemeStore = defineStore(
      'theme',
      () => {
        const themeConfig = reactive<ThemeConfig>({
          showThemeConfig: false,
          darkThemeEnabled: false,
          grayThemeEnabled: false,
          layoutMode: 'classic',
          themePrimaryColor: '#409eff',
          predefineColors: ['#409eff']
        })
    
        // 切换显示状态的方法
        const toggleThemeConfig = () => {
          themeConfig.showThemeConfig = !themeConfig.showThemeConfig
        }
    
        // 黑暗主题切换
        watch(() => themeConfig.darkThemeEnabled, () => {
          const html = document.documentElement as HTMLElement
          if (themeConfig.darkThemeEnabled) {
            html.setAttribute('class', 'dark')
          } else {
            html.removeAttribute('class')
          }
          changeThemeColor(themeConfig.themePrimaryColor)
        })
    
        watch(() => themeConfig.grayThemeEnabled,() => {
          if (themeConfig.grayThemeEnabled) {
            document.documentElement.dataset.theme = 'gray'
          } else {
            document.documentElement.dataset.theme = ''
          }
        })
    
        // 添加预定义颜色
        function addPredefineColor(color:string) {
          console.log('color', color)
          const predefineColors = themeConfig.predefineColors
          // 查找元素的索引
          const index = predefineColors.indexOf(color)
    
          if (index !== -1) {
            // 使用 splice 方法删除该元素
            predefineColors.splice(index, 1)
          }
          if (themeConfig.predefineColors.length === 10) {
            themeConfig.predefineColors.pop() // 删除队列中的第一个元素
          }
          themeConfig.predefineColors.unshift(color)
        }
    
        // 修改主题颜色
        const changeThemeColor = (color:string ) => {
          if (!color) {
            color = '#409eff'
            ElMessage({ type: 'success', message: '主题颜色已重置' })
          }
          addPredefineColor(color)
          // 计算主题颜色变化
          document.documentElement.style.setProperty('--el-color-primary', color)
          document.documentElement.style.setProperty(
            '--el-color-primary-dark-2',
            themeConfig.darkThemeEnabled ? `${getLightColor(color, 0.2)}` : `${getDarkColor(color, 0.3)}`
          )
          for (let i = 1; i <= 9; i++) {
            const primaryColor = themeConfig.darkThemeEnabled ? `${getDarkColor(color, i / 10)}` : `${getLightColor(color, i / 10)}`
            document.documentElement.style.setProperty(`--el-color-primary-light-${i}`, primaryColor)
          }
        }
    
        return {
          themeConfig,
          toggleThemeConfig,
          changeThemeColor
        }
      },
      {
        persist: true
      })
    
  3. 创建 src/layouts/components/Theme/index.vue 主题组件并引入主题仓库:

    <script setup lang="ts">
    import { useThemeStore } from '@/stores/modules/theme'
    import { onMounted } from 'vue'
    
    const { themeConfig, changeThemeColor } = useThemeStore()
    
    onMounted(() => {
      // 在组件挂载到 DOM 后更新主题颜色
      changeThemeColor(themeConfig.themePrimaryColor)
    })
    </script>
    
  4. 使用 el-drawer 抽屉组件进行主题布局:

    <template>
      <div>
        <el-drawer v-model="themeConfig.showThemeConfig" size="300">
          <template #header="{ titleId }">
            <span :id="titleId" class="theme-header"> 主题配置 </span>
          </template>
    
          <el-scrollbar>
            <div class="mr-4">
              <el-divider>布局样式</el-divider>
              <div class="layout-box">
                <el-tooltip effect="dark" content="经典布局" placement="top" :show-after="200">
                  <div
                    :class="['layout-item layout-classic', { 'is-active': themeConfig.layoutMode == 'classic' }]"
                    @click="themeConfig.layoutMode='classic'"
                  >
                    <div class="layout-dark"></div>
                    <div class="layout-container">
                      <div class="layout-light"></div>
                      <div class="layout-content"></div>
                    </div>
                    <svg-icon class="select-layout" name="About" v-if="themeConfig.layoutMode == 'classic'" />
                  </div>
                </el-tooltip>
                <el-tooltip
                  effect="dark"
                  content="横向布局"
                  placement="top"
                  :show-after="200"
                >
                  <div
                    :class="['layout-item layout-transverse', { 'is-active': themeConfig.layoutMode == 'horizontal' }]"
                    @click="themeConfig.layoutMode='horizontal'"
                  >
                    <div class="layout-dark"></div>
                    <div class="layout-content"></div>
                    <svg-icon class="select-layout" name="About" v-if=" themeConfig.layoutMode == 'horizontal'" />
                  </div>
                </el-tooltip>
              </div>
    
              <el-divider>主题风格</el-divider>
              <div class="switch-container">
                <div class="switch-box">
                  <div class="switch-title">
                    <span>主题颜色</span>
                    <el-tooltip content="主题颜色" placement="top">
                      <svg-icon class="size-4" name="About"></svg-icon>
                    </el-tooltip>
                  </div>
                  <el-color-picker
                    color-format="hex"
                    v-model="themeConfig.themePrimaryColor"
                    :predefine="themeConfig.predefineColors"
                    @change="changeThemeColor"
                  />
                </div>
                <div class="switch-box">
                  <div class="switch-title">
                    <span>暗黑主题</span>
                    <el-tooltip content="暗黑主题" placement="top">
                      <svg-icon class="size-4" name="About"></svg-icon>
                    </el-tooltip>
                  </div>
                  <el-switch v-model="themeConfig.darkThemeEnabled" />
                </div>
                <div class="switch-box">
                  <div class="switch-title">
                    <span>灰色模式</span>
                    <el-tooltip content="灰色模式" placement="top">
                      <svg-icon class="size-4" name="About"></svg-icon>
                    </el-tooltip>
                  </div>
                  <el-switch v-model="themeConfig.grayThemeEnabled" />
                </div>
              </div>
    
              <el-divider>界面设置</el-divider>
              <div class="switch-container">
                <div class="switch-box">
                  <div class="switch-title">
                    <span>theme</span>
                  </div>
                  <el-switch inline-prompt />
                </div>
              </div>
            </div>
          </el-scrollbar>
        </el-drawer>
      </div>
    </template>
    
  5. 为组件添加样式:

    <style scoped lang="postcss">
    .theme-header {
      @apply text-base font-bold flex items-center;
    }
    
    .switch-container {
      @apply flex-grow space-y-2;
    }
    
    .switch-box {
      @apply flex justify-between items-center w-full;
    }
    
    .switch-title {
      @apply flex gap-2 items-center;
    }
    
    :deep(.el-drawer__header) {
      @apply px-5 py-0 h-12 border-b-2 border-solid m-0;
      color: var(--el-text-color-regular);
      border-color: var(--el-border-color-light);
    }
    
    :deep(.el-drawer__title) {
      font-size: 20px;
    }
    
    :deep(.el-drawer__body) {
      @apply py-0 pr-0;
    }
    
    .layout-box {
      position: relative;
      display: flex;
      flex-wrap: wrap;
      justify-content: space-between;
      padding: 15px 15px 0;
    
      .layout-item {
        position: relative;
        box-sizing: border-box;
        width: 100px;
        height: 70px;
        padding: 6px;
        cursor: pointer;
        border-radius: 5px;
        box-shadow: 0 0 5px 1px var(--el-border-color-dark);
        transition: all 0.2s;
    
        .layout-dark {
          background-color: var(--el-color-primary);
          border-radius: 3px;
        }
    
        .layout-light {
          background-color: var(--el-color-primary-light-5);
          border-radius: 3px;
        }
    
        .layout-content {
          background-color: var(--el-color-primary-light-8);
          border: 1px dashed var(--el-color-primary);
          border-radius: 3px;
        }
    
        .select-layout {
          position: absolute;
          right: 10px;
          bottom: 10px;
          fill: var(--el-color-primary);
          transition: all 0.2s;
        }
    
        &:hover {
          box-shadow: 0 0 5px 1px var(--el-text-color-secondary);
        }
      }
    
      .is-active {
        box-shadow: 0 0 0 2px var(--el-color-primary) !important;
      }
    
      .layout-classic {
        display: flex;
        justify-content: space-between;
        margin-bottom: 20px;
    
        .layout-dark {
          width: 20%;
        }
    
        .layout-container {
          display: flex;
          flex-direction: column;
          justify-content: space-between;
          width: 72%;
    
          .layout-light {
            height: 20%;
          }
    
          .layout-content {
            height: 67%;
          }
        }
      }
    
      .layout-transverse {
        display: flex;
        flex-direction: column;
        justify-content: space-between;
        margin-bottom: 15px;
    
        .layout-dark {
          height: 20%;
        }
    
        .layout-content {
          height: 67%;
        }
      }
    }
    </style>
    

经典布局

  1. 创建经典页面布局组件:src/layouts/LayoutClassic/index.vue

  2. 导入布局组件:

    <script setup lang="ts">
    import Footer from '@/layouts/components/Footer/index.vue'; // 引入页脚组件
    import Menu from '@/layouts/components/Menu/index.vue'; // 引入菜单组件
    import Logo from '@/layouts/components/Logo/index.vue'; // 引入Logo组件
    import Main from '@/layouts/components/Main/index.vue'; // 引入主要内容组件
    import TabBar from '@/layouts/components/TabBar/index.vue'; // 引入标签栏组件
    import Breadcrumb from '@/layouts/components/Breadcrumb/index.vue'; // 引入面包屑导航组件
    
    import { useThemeStore } from '@/stores/modules/theme' // 引入主题仓库
    const { toggleThemeConfig } = useThemeStore()
    
  3. 使用 ElementPlus 提供的 Container 布局容器快速搭建页面的基本结构:

  4. 在组件中使用布局组件划分出页面结构:侧边菜单栏、顶部、Tab 标签栏、主体和底部:

    <el-container>:外层容器。 当子元素中包含 <el-header><el-footer> 时,全部子元素会垂直上下排列, 否则会水平左右排列。

    <el-header>:顶栏容器。

    <el-aside>:侧边栏容器。

    <el-main>:主要区域容器。

    <el-footer>:底栏容器。

    <template>
      <div class="layout-container">
        <el-container>
          <!-- 左侧边栏 -->
          <el-aside>
            <!-- Logo 组件 -->
            <Logo class="os-logo"/>
            <!-- 菜单组件 -->
            <Menu class="menu-classic" />
          </el-aside>
    
          <el-container>
            <!-- 页面顶部区域 -->
            <el-header>
              <!-- 头部内容区域 -->
              <div class="header-content">
                <!-- 折叠按钮 -->
                <svg-icon class="fold-expand-button" name="Fold" />
                <!-- 面包屑导航 -->
                <Breadcrumb />
              </div>
    
              <!-- 头部菜单区域 -->
              <div class="header-menu">
                <!-- 语言切换下拉菜单 -->
                <el-dropdown :hide-on-click="false">
                  <svg-icon name="Earth" />
                  <template #dropdown>
                    <el-dropdown-menu>
                      <el-dropdown-item>简体中文</el-dropdown-item>
                    </el-dropdown-menu>
                  </template>
                </el-dropdown>
    
                <!-- 搜索按钮 -->
                <svg-icon class="cursor-pointer" name="Search" />
    
                <!-- 主题配置按钮 -->
                <el-tooltip content="主题配置" placement="bottom">
                  <svg-icon class="cursor-pointer" name="Theme" @click="toggleThemeConfig" />
                </el-tooltip>
    
                <!-- 锁屏按钮 -->
                <el-tooltip content="锁屏" placement="bottom">
                  <svg-icon class="cursor-pointer" name="Lock" />
                </el-tooltip>
    
                <!-- 全屏按钮 -->
                <el-tooltip content="全屏" placement="bottom">
                  <svg-icon class="cursor-pointer" name="FullScreen" />
                </el-tooltip>
    
                <!-- 用户信息下拉菜单 -->
                <el-dropdown>
                  <div class="flex items-center gap-1">
                    <el-tag type="primary">Admin</el-tag>
                    <el-avatar src="https://cube.elemecdn.com/0/88/03b0d39583f48206768a7534e55bcpng.png" />
                  </div>
                  <template #dropdown>
                    <el-dropdown-menu>
                      <el-dropdown-item command="profile">
                        <template #default>
                          <div class="flex items-center gap-1">
                            <svg-icon name="User" />
                            <span>个人中心</span>
                          </div>
                        </template>
                      </el-dropdown-item>
                      <el-dropdown-item command="logout">
                        <template #default>
                          <div class="flex items-center gap-1">
                            <svg-icon name="Logout" />
                            <span>退出登录</span>
                          </div>
                        </template>
                      </el-dropdown-item>
                    </el-dropdown-menu>
                  </template>
                </el-dropdown>
              </div>
    
            </el-header>
    
            <!-- 标签栏 -->
            <tab-bar class="tab-container" />
    
            <!-- 主内容区域 -->
            <el-main>
              <Main />
            </el-main>
    
            <!-- 底部页脚 -->
            <el-footer>
              <Footer />
            </el-footer>
          </el-container>
        </el-container>
      </div>
    </template>
    
  5. 为组件添加样式:

    <style scoped lang="postcss">
    .layout-container {
      @apply w-screen h-screen;
    
      .el-aside {
        @apply h-full;
        @apply border-r border-solid overflow-hidden;
        border-color: var(--el-border-color-light);
        width: var(--os-layout-aside-width);
    
        .os-logo {
          @apply overflow-hidden pl-2;
        }
    
        .menu-classic {
          height: calc(100vh - var(--os-layout-logo-height));
        }
      }
    
      .el-container {
        @apply h-full w-full;
    
        .tab-container {
          height: var(--os-layout-tab-height);
        }
    
        .el-header {
          @apply flex items-center justify-between;
          @apply border-b border-solid overflow-hidden;
          border-color: var(--el-border-color-light);
          height: var(--os-layout-header-height);
    
          .header-content {
            @apply flex items-center gap-5;
    
            .fold-expand-button {
              @apply cursor-pointer;
              @apply size-6;
            }
          }
    
          .header-menu {
            @apply flex items-center justify-between gap-3;
          }
        }
    
        .el-main {
          @apply h-full p-3;
          background-color: var(--el-bg-color-page);
        }
    
        .el-footer {
          height: var(--os-layout-footer-height);
        }
      }
    }
    </style>
    

水平布局

  1. 创建经典页面布局组件:src/layouts/LayoutClassic/index.vue

  2. 导入布局组件:

    <script setup lang="ts">
    import Footer from '@/layouts/components/Footer/index.vue'; // 引入页脚组件
    import Menu from '@/layouts/components/Menu/index.vue'; // 引入菜单组件
    import Logo from '@/layouts/components/Logo/index.vue'; // 引入Logo组件
    import Main from '@/layouts/components/Main/index.vue'; // 引入主要内容组件
    import TabBar from '@/layouts/components/TabBar/index.vue'; // 引入标签栏组件
    
    import { useThemeStore } from '@/stores/modules/theme' // 引入主题仓库
    const { toggleThemeConfig } = useThemeStore()
    </script>
    
  3. 在组件中使用 Element UI 的布局组件划分出页面结构:顶部、Tab 标签栏、主体和底部:

    <template>
      <div class="layout-container">
        <el-container>
          <el-container>
            <el-header>
              <div class="flex">
                <Logo class="os-logo" />
                <Menu mode="horizontal" class="menu-horizontal " />
              </div>
              <div class="header-menu">
                <!-- 语言转换 -->
                <el-dropdown :hide-on-click="false">
                  <svg-icon name="Earth" />
                  <template #dropdown>
                    <el-dropdown-menu>
                      <el-dropdown-item>简体中文</el-dropdown-item>
                    </el-dropdown-menu>
                  </template>
                </el-dropdown>
    
                <!-- 搜索按钮 -->
                <svg-icon class="cursor-pointer" name="Search" />
    
                <!-- 皮肤按钮 -->
                <el-tooltip content="主题配置" placement="bottom">
                  <svg-icon class="cursor-pointer" name="Theme" @click="toggleThemeConfig"/>
                </el-tooltip>
    
                <!-- 锁屏按钮 -->
                <el-tooltip content="锁屏" placement="bottom">
                  <svg-icon class="cursor-pointer" name="Lock" />
                </el-tooltip>
    
                <!-- 全屏按钮 -->
                <el-tooltip content="全屏" placement="bottom">
                  <svg-icon class="cursor-pointer" name="FullScreen" />
                </el-tooltip>
    
                <!-- 用户信息 -->
                <el-dropdown>
                  <div class="flex items-center gap-1">
                    <el-tag type="primary">Admin</el-tag>
                    <el-avatar src="https://cube.elemecdn.com/0/88/03b0d39583f48206768a7534e55bcpng.png" />
                  </div>
                  <template #dropdown>
                    <el-dropdown-menu>
                      <el-dropdown-item command="profile">
                        <template #default>
                          <div class="flex items-center gap-1">
                            <svg-icon name="User" />
                            <span>个人中心</span>
                          </div>
                        </template>
                      </el-dropdown-item>
                      <el-dropdown-item command="logout">
                        <template #default>
                          <div class="flex items-center gap-1">
                            <svg-icon name="Logout" />
                            <span>退出登录</span>
                          </div>
                        </template>
                      </el-dropdown-item>
                    </el-dropdown-menu>
                  </template>
                </el-dropdown>
              </div>
            </el-header>
            <tab-bar class="tab-container" />
            <el-main>
              <Main />
            </el-main>
            <el-footer>
              <Footer />
            </el-footer>
          </el-container>
        </el-container>
      </div>
    </template>
    
  4. 添加样式:

    <style scoped lang="postcss">
    .layout-container {
      @apply w-screen h-screen;
    
      .el-aside {
        @apply bg-amber-300 h-full;
        width: var(--os-layout-aside-width);
      }
    
      .el-container {
        @apply h-full w-full;
    
        .tab-container {
          height: var(--os-layout-tab-height);
        }
    
        .el-header {
          @apply w-full pl-2 pr-2 flex justify-between;
          height: var(--os-layout-header-height);
          @apply border-b border-solid overflow-hidden;
          border-color: var(--el-border-color-light);
    
          .menu-horizontal{
            width: calc(100vw - var(--os-layout-aside-width) - var(--os-layout-header-menu-width));
          }
    
          .header-menu{
            @apply flex items-center justify-between gap-3;
          }
        }
    
        .el-main {
          @apply h-full p-3;
          background-color:var(--el-bg-color-page);
        }
    
        .el-footer {
          height: var(--os-layout-footer-height);
        }
      }
    }
    </style>
    

响应式布局

  1. 创建src/layouts/LayoutResponsive/index.vue响应式布局组件

  2. 添加公共组件:

    <script setup lang="ts">
    import Logo from '@/layouts/components/Logo/index.vue'
    import Main from '@/layouts/components/Main/index.vue'
    import Menu from '@/layouts/components/Menu/index.vue'
    import { ref } from 'vue'
    
    import { useThemeStore } from '@/stores/modules/theme'
    const { themeConfig } = useThemeStore()
    
    const navDrawer = ref(false)
    const menuDrawer = ref(false)
    const translateShow = ref(false)
    </script>
    
  3. 页面布局:

    <template>
      <div class="layout-responsive">
        <el-container>
          <el-header class="flex items-center justify-between">
            <div class="h-full flex items-center gap-2">
              <svg-icon width="30px" height="30px" name="menu" @click="menuDrawer = !menuDrawer" />
              <Logo />
            </div>
            <div class="header-menu">
              <!-- 搜索按钮 -->
              <svg-icon width="28px" height="28px" class="cursor-pointer" name="Search" />
    
              <!-- 用户信息 -->
              <el-avatar src="https://cube.elemecdn.com/0/88/03b0d39583f48206768a7534e55bcpng.png"
                         @click="navDrawer = !navDrawer" />
    
              <!-- 导航信息 -->  
              <el-drawer v-model="navDrawer" size="100%">
                <template #header>
                  <Logo />
                </template>
                <div class="nav-container ">
                  <a href="#">个人中心</a>
                  <el-divider />
                  <div>
                      <span class="flex items-center justify-between" @click="translateShow = !translateShow">
                        翻译
                        <svg-icon name="Earth"></svg-icon>
                      </span>
                    
                    <!-- 语言列表 -->
                    <transition name="fade">
                      <div v-show="translateShow" id="language-list" class="language-list mt-2">
                        <ul>
                          <li class="p-2">英语</li>
                          <li class="p-2">中文</li>
                          <li class="p-2">西班牙语</li>
                        </ul>
                      </div>
                    </transition>
                  </div>
                  <el-divider />
                  <div class="nav-theme-switch">
                    <span>暗黑主题</span>
                    <el-switch v-model="themeConfig.darkThemeEnabled" size="large" />
                  </div>
                  <el-divider />
                  <span>退出登录</span>
                </div>
              </el-drawer>
            </div>
          </el-header>
          <el-main>
            <el-backtop target=".el-main" />
            <Main />
          </el-main>
        </el-container>
    
        <!-- 菜单组件 -->
        <el-drawer
          v-model="menuDrawer"
          direction="ltr"
          size="320px"
        >
          <Menu />
        </el-drawer>
      </div>
    </template>
    
  4. 添加样式:

    <style scoped lang="postcss">
    .layout-responsive {
      @apply w-screen h-screen;
    
      .el-container {
        @apply h-full w-full;
    
        .el-header {
          @apply w-full flex pl-2;
          height: var(--os-layout-header-height);
          @apply border-b border-solid overflow-hidden;
          border-color: var(--el-border-color-light);
    
          .header-menu {
            @apply flex items-center justify-between gap-3;
          }
        }
    
        .tab-container {
          background-color: var(--el-bg-color);
    
        }
    
        .el-main {
          @apply h-full;
          background-color: var(--el-bg-color-page);
        }
      }
    }
    
    :deep(.el-drawer__header) {
      @apply px-3 py-0 h-14 border-solid m-2;
      color: var(--el-text-color-regular);
      border-color: var(--el-border-color-light);
    }
    
    :deep(.el-drawer__title) {
      font-size: 20px;
    }
    
    :deep(.el-drawer__body) {
      @apply py-0 px-8;
    }
    
    :deep(.el-drawer__close-btn) {
      @apply text-3xl;
    }
    
    .nav-container {
      margin: 0 auto;
      padding: 24px 0 96px;
      max-width: 18rem;
    }
    
    .nav-theme-switch {
      @apply flex items-center justify-between rounded-md;
    }
    
    /* 添加动画效果 */
    .fade-enter-active {
      transition: opacity 0.5s ease;
    }
    
    .fade-leave-active {
      transition: opacity 0.2s ease;
    }
    
    .fade-enter-from, .fade-leave-to {
      opacity: 0;
    }
    
    .logo-title {
      @apply text-lg font-bold subpixel-antialiased;
      @apply bg-clip-text text-transparent bg-gradient-to-r from-pink-500 to-violet-500;
    }
    </style>
    

搭建 Layout 布局组件

  1. 创建 Layout 布局组件 src/layouts/index.vue

  2. 导入经典布局组件 LayoutClassic 和水平布局组件 LayoutHorizontal,创建一个 layoutMode 对象来将布局模式字符串映射到相应的组件,使用 computed 函数根据 mode 动态选择布局模式。

    <script setup lang="ts">
    import { type Component, computed, onMounted, onUnmounted, ref, watch } from 'vue'
    import LayoutClassic from '@/layouts/LayoutClassic/index.vue'
    import LayoutHorizontal from '@/layouts/LayoutHorizontal/index.vue'
    import LayoutResponsive from '@/layouts/LayoutResponsive/index.vue'
    import Theme from '@/layouts/components/Theme/index.vue'
    
    import { useThemeStore } from '@/stores/modules/theme'
    const { themeConfig } = useThemeStore()
    
    // 定义布局模式与对应的组件类型
    const layoutMode: Record<string, Component> = {
      classic: LayoutClassic,   // 经典布局
      horizontal: LayoutHorizontal,  // 横向布局
      responsive: LayoutResponsive  // 响应式布局
    }
    
    // 布局模式
    const layMode = ref(themeConfig.layoutMode)
    
    // 监听主题配置模式
    watch(() => themeConfig.layoutMode, (newMode) => {
      layMode.value = newMode
    })
    
    // 计算属性,根据具体逻辑返回所选布局的组件
    const mode = computed(() => layoutMode[layMode.value])
    
    // 响应式布局,切换布局模式
    const updateMode = () => {
      if (window.innerWidth <= 768) {
        layMode.value = 'responsive'
      } else {
        layMode.value = themeConfig.layoutMode
      }
    }
    
    onMounted(() => {
      // 初始更新
      updateMode()
      window.addEventListener('resize', updateMode)
    })
    
    onUnmounted(() => {
      window.removeEventListener('resize', updateMode)
    })
    </script>
    
    <template>
      <!-- 根据 mode 的值渲染不同的布局样式 -->
      <component :is="mode" />
      <Theme />
    </template>
    
    <style scoped lang="postcss">
    </style>
    

添加 Layout 路由配置

  1. src/App.vue 中添加代码,当访问路由路径时,Vue Router 会根据路由配置将对应的组件渲染到 <router-view> 中,实现单页应用的页面切换效果:

    <template>
      <router-view />
    </template>
    
  2. src/router/index.ts 中添加路由:

    const router = createRouter({
      routes: [
        {
          path: '/',
          name: 'layout',
          component: () => import('@/layouts/index.vue')
        }
      ]
    })
    

启动项目

  1. 终端执行命令启动项目:

    pnpm run dev
    
  2. 浏览器访问:http://localhost:8080/,默认为经典布局样式:

    image-20240805095808149

  3. src/layouts/index.vue 中切换为水平布局样式,查看页面:

    const mode = computed(() => layoutMode['horizontal'])
    

    image-20240805095901012

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/2043387.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

关于Idea中的debug模式只能执行一次的问题

希望文章能给到你启发和灵感&#xff5e; 如果觉得文章对你有帮助的话&#xff0c;点赞 关注 收藏 支持一下博主吧&#xff5e; 阅读指南 开篇说明一、基础环境说明1.1 硬件环境1.2 软件环境 二、为什么debug模式只有生效一次三、补充说明其他调试功能四、最后 开篇说明 记录一…

设计模式学习优质网站分享:refactoring.guru

地址 英文版地址&#xff1a;https://refactoring.guru/design-patterns 中文版地址&#xff1a;https://refactoringguru.cn/design-patterns 介绍 这个网站是专门学习 设计模式 和 软件重构 的网站 整体来说并不花哨&#xff0c;但我觉得他最大的优点就是&#xff1a; 概…

PyTorch 基础学习(2)- 张量 Tensors

PyTorch张量简介 张量是数学和计算机科学中的一个基本概念&#xff0c;用于表示多维数据&#xff0c;是AI世界中一切事物的表示和抽象。可以将张量视为一个扩展了标量、向量和矩阵的通用数据结构。以下是对张量的详细解释&#xff1a; 张量的定义 标量&#xff08;0阶张量&am…

Assembly(七)实验环境搭建

本篇文章将讲解在win11环境下的王爽老师的汇编语言的环境搭建 首先凑齐这些文件: 随后安装好Dosbox,去官网下载就好 打开箭头所指文件 找到文件最后部分 [autoexec] # Lines in this section will be run at startup. # You can put your MOUNT lines here. MOUNT C D:\Debug …

快速搭建Vue_cli以及ElementUI简单项目学生管理系统雏形

为了帮助大家快速搭建Vue_cli脚手架还有ElementUI的简单项目,今天我给大家提供方法. 因为这个搭建这个项目步骤繁多,容易忘记,所以给大家提供这个资料希望可以帮助到你们. 废话不多说开始搭建项目: 搭建Vue_cli项目 首先点开HBuilder左上角的文件点击新建,点击项目,选择vue项…

2024年人工智能固态硬盘采购容量预计超过45 EB

根据TrendForce发布的最新市场报告&#xff0c;人工智能&#xff08;AI&#xff09;服务器客户在过去两个季度显著增加了对企业级固态硬盘&#xff08;SSD&#xff09;的订单。为了满足AI应用中不断增长的SSD需求&#xff0c;上游供应商正在加速工艺升级&#xff0c;并计划在20…

智慧交通物联网应用,5G路由器赋能高速道路监控数据传输

高速道路为了保障交通的高速、安全运行&#xff0c;沿线部署了控制设施、监视设施、情报设施、传输设施、显示设施及控制中心等。在传统的高速管理中&#xff0c;这些设施的传输设施多采用光纤线缆进行数据传输&#xff0c;但高速道路覆盖范围广、距离远&#xff0c;布线与施工…

韩顺平 集合

集合 一、体系结构图二、Collection2.1 Collection 接口和常用方法2.2 集合遍历2.2.1 迭代器2.2.2 增强for循环 三、List接口及其常用方法3.1 三种遍历方式3.2 ArrayList3.3 LinkedList 四 MAP4.1 hashmap 一、体系结构图 集合主要是两组 单列和双列集合 Collection接口有两个重…

第十五章:高级调度

本章内容包括&#xff1a; 使用节点污点和pod容忍度组织pod调度到特定节点将节点亲和性规则作为节点选择器的一种替代使用节点亲和性进行多个pod的共同调度使用节点非亲和性来分离多个pod Kubernetes允许你去影响pod被调度到哪个节点。起初&#xff0c;只能通过在pod规范⾥指定…

Linux安装Nginx后,无法解析Windows主机Hosts文件

问题展示&#xff1a; 配置好Linux的Nginx配置后&#xff0c;Windows同样配置好host&#xff0c;而通过浏览器只能用IP地址成功访问&#xff0c;而域名则不行 解决方法&#xff1a; 点击Windows图标&#xff0c;搜索记事本&#xff0c;选择以管理员身份运行&#xff0c;编辑…

php-xlswriter实现数据导出excel单元格合并,内容从指定行开始写

最终效果图&#xff1a; 代码&#xff1a; public function export_data() {$list $this->get_list_organ();$content [];$content[] []; // 第2行不设置内容&#xff0c;设置为空foreach ($list as $key > $value) {$content[] [$value[organ_name], $value[clas…

防火墙技术与地址转换

文章目录 前言一、四种区域二、实验拓扑图基础配置防火墙配置测试结果 前言 防火墙是计算机网络中的一种安全设备或软件功能&#xff0c;旨在监控和控制进出网络的网络流量。其核心目的是保护内部网络免受外部攻击或不必要的访问。防火墙通过设定一系列安全规则&#xff0c;允…

【iOS】UITableViewCell的重用问题解决方法

我自己在实验中对cell的重用总结如下&#xff1a; 非自定义Cell和非自定义cell的复用情况一样&#xff1a; 第一次加载创建tableView的时候&#xff0c;是屏幕上最多也显示几行cell就先创建几个cell&#xff0c;此时复用池里什么都没有开始下滑tableView&#xff0c;刚开始滑…

可视化编程-七巧低代码入门02

1.1.什么是可视化编程 非可视化编程是一种直接在集成开发环境中&#xff08;IDE&#xff09;编写代码的编程方式&#xff0c;这种编程方式要求开发人员具备深入的编程知识&#xff0c;开发效率相对较低&#xff0c;代码维护难度较大&#xff0c;容易出现错误&#xff0c;也需要…

最新的APS高级计划排程系统推动的MRP供应链计划是什么?

在当下“内卷”的市场环境下&#xff0c;制造业的订单需求从过去大批量标准品生产已经演变成小批量、多订单的非标订单生产&#xff0c;这对制造业的供应链提出了更高的要求。为了应对市场实现产销平衡&#xff0c;中大型的企业都开始重视供应链的建设工作&#xff0c;以应对企…

数字签名和CA数字证书的核心原理和作用

B站讲解视频&#xff0c;讲述HTTPS CA认证的整个行程过程与原理 https://www.bilibili.com/video/BV1mj421d7VE

[Qt][Qt 文件]详细讲解

目录 1.输入输出设备类2.文件读写类3.文件和目录信息类 1.输入输出设备类 在Qt中&#xff0c;⽂件读写的类为QFile&#xff0c;其⽗类为QFileDevice QFileDevice提供了⽂件交互操作的底层功能QFileDevice的⽗类是QIODevice&#xff0c;其⽗类为QObject QIODevice是Qt中所有I/O…

【数学建模备赛】Ep05:斯皮尔曼spearman相关系数

文章目录 一、前言&#x1f680;&#x1f680;&#x1f680;二、斯皮尔曼spearman相关系数&#xff1a;☀️☀️☀️1. 回顾皮尔逊相关系数2. 斯皮尔曼spearman相关系数3. 斯皮尔曼相关系数公式4. 另外一种斯皮尔曼相关系数定义5. matlab的用法5. matlab的用法 三、对斯皮尔曼相…

立仪光谱共焦传感器行业应用|透明胶水高度测量

01&#xff5c;检测需求&#xff1a;透明胶水高度测量 02&#xff5c;检测方式 根据客户要求及观察我们使用立仪科技D40A26XL镜头搭配E系列控制器进行测量 03&#xff5c;光谱共焦测量结果 经过测量可以得出胶水的高度为1076.406μm 04&#xff5c;光谱共焦侧头 D40A26XL侧头…

uniapp接口请求this.$request

代码示例&#xff1a; createPhoto(url) {this.$request({url: /emp/gallery-photo/create,//后端接口method: post,//请求方法header: {//请求头tenant-id: 1,},data: {//请求参数galleryId: this.albumId,empUserId: this.empUserId,"url": url,}}).then((res) &…