• 02 Vue3 UI Framework 顶部边栏


    顶部边栏比较简单,而且首页和文档页都需要,所以我们先从顶部边栏做起

    前文回顾点击 这里 返回阅读列表点击 这里

    初始化

    首先,在 components 文件夹下,创建一个 vue 组件,命名为 Topnav.vue ,然后快速创建 vue 模板,代码如下:

    点击查看代码
    <template>
      <div>
        
      </div>
    </template>
    <script lang="ts">
    export default {
      
    }
    </script>
    <style lang="scss" scoped>
      
    </style>
    

    页面结构分析

    分析顶部边栏的结构,可以得出如下表格信息:

    功能 位置 默认状态 小于 500 px 时状态
    首页跳转入口 左侧 可见 居中
    文档页跳转入口 右侧 可见 不可见
    展开菜单按钮 左侧 不可见 可见

    页面结构代码如下:

    点击查看代码
    <template>
        <div class="topnav">
            <router-link to="/">
                首页
            </router-link>
            <router-link to="/document">
                文档页
            </router-link>
            <button>
                菜单
            </button>
        </div>
    </template>
    

    考虑到后续顶部菜单项扩展方便,这里最好做成一个列表

    点击查看代码
    <template>
        <div class="topnav">
            <router-link to="/" class="logo">
                首页
            </router-link>
            <ul class="menu">
                <li>
                    <router-link to="/document">
                        文档页
                    </router-link>
                </li>
            </ul>
            <button @click="toggleMenu">
                菜单
            </button>
        </div>
    </template>
    

    功能分析

    需要的控制元素

    顶边边栏中的弹出菜单按键,可以在引入它的组件中,被设置是否可见,那么应当有一个 Boolean 类型的变量来控制可见,且需要提供一个方法,控制菜单本体是否可见,但是,现在又犯难了——菜单本体是属于文档页的,而不是属于顶边栏的。如何跨组件控制呢 ?

    回顾需求分析可得,通过弹出菜单按键,以及视口宽度,共同控制菜单是否可见,其中,视口宽度显然是一个全局属性,那么理应在 App.vue 中控制,于是我们在 App.vue 中定义其是否可见,并通过 provide/inject API 暴露给子组件,相关代码如下:

    点击查看代码
    <script lang="ts">
    import { provide, ref } from "vue";
    export default {
      name: "App",
      setup() {
        const width = ref(document.documentElement.clientWidth);
        const menuVisible = ref(width.value > 500 ? true : false);
        window.onresize = () => {
          width.value = document.documentElement.clientWidth;
          if (width.value > 500) {
            menuVisible.value = true;
          } else {
            menuVisible.value = false;
          }
        };
    
        provide("menuVisible", menuVisible);
      },
    };
    </script>
    

    初始化时,可以根据视口宽度决定顶边栏的弹出菜单按键是否可见,并监听视口大小变化,根据视口宽度自动更新控制变量。

    编写顶部边栏核心代码

    Topnav.vue 代码如下:

    点击查看代码
    <script lang="ts">
    import { inject, Ref } from "vue";
    export default {
      props: {
        toggleMenuButtonVisible: {
          type: Boolean,
          default: false,
        },
      },
      setup() {
        const menuVisible = inject<Ref<boolean>>("menuVisible");
        const toggleMenu = () => {
          menuVisible.value = !menuVisible.value;
        };
        return {
          toggleMenu,
        };
      },
    };
    </script>
    

    注意一下

    此处的 inject 需要注明变量的类型。因为 menuVisible 声明的时候是 ref(true|false),所以其类型为 Ref<boolean>

    层叠样式表

    尽量优化用户体验,我们这里做个简单的样式美化,底部采用渐变线的分割效果:

    点击查看代码
    <style lang="scss" scoped>
    $color: #8c6fef;
    .topnav {
      color: $color;
      display: flex;
      background: linear-gradient(
        180deg,
        rgba(255, 255, 255, 1) 0%,
        rgba(255, 255, 255, 1) 97%,
        #8c6fef 97%,
        #8c6fef 100%
      );
      padding: 0 32px;
      position: absolute;
      top: 0;
      left: 0;
       100%;
      z-index: 20;
      justify-content: center;
      align-items: center;
      > .logo {
        max- 6em;
        margin-right: auto;
        > svg {
           80px;
          height: 80px;
        }
        > img {
          height: 80px;
        }
      }
      > .menu {
        display: flex;
        white-space: nowrap;
        flex-wrap: nowrap;
        > li {
          margin: 0 1em;
          > a {
            > svg {
               32px;
              height: 32px;
            }
            > img {
              height: 80px;
            }
          }
        }
      }
      > .toggleAside {
         32px;
        height: 32px;
        position: absolute;
        left: 16px;
        top: 50%;
        transform: translateY(-50%);
        display: none;
        cursor: pointer;
      }
      @media (max- 500px) {
        > .menu {
          display: none;
        }
        > .logo {
          margin: 0 auto;
        }
        > .toggleAside {
          display: inline-block;
        }
      }
    }
    img {
      padding: 6px 0;
    }
    </style>
    

    至此,我们的顶部边栏组件已经完成了,接下来,我们把这个组件引入到 App.vue 中看下实际运行效果吧!

    点击查看代码
    <template>
      <div class="app">
        <router-view />
      </div>
    </template>
    
    <script lang="ts">
    import { provide, ref } from "vue";
    export default {
      name: "App",
      setup() {
        const width = ref(document.documentElement.clientWidth);
        const menuVisible = ref(width.value > 500 ? true : false);
        window.onresize = () => {
          width.value = document.documentElement.clientWidth;
          if (width.value > 500) {
            menuVisible.value = true;
          } else {
            menuVisible.value = false;
          }
        };
    
        provide("menuVisible", menuVisible);
      },
    };
    </script>
    <style lang="scss" scoped>
    $max- 1200px;
    .app {
      max- $max-width;
      margin-left: calc(50vw - 600px);
      position: relative;
      @media (max- $max-width) {
        margin-left: 0;
      }
    }
    </style>
    

    效果图

    image-20211211092512943

    项目地址

    GitHub: https://github.com/JeremyWu917/jeremy-ui

    官网地址

    JeremyUI: https://ui.jeremywu.top

    感谢阅读 ☕

  • 相关阅读:
    [FAQ] GitHub 开启二次验证之后,如何通过 https clone 项目 ?
    [FAQ] GoLand 需要手动开启代码补全吗 ?
    [FAQ] 夏玉米 按规则查询域名靠谱吗 ?
    [FAQ] Error: com.mysql.jdbc.Driver not loaded. :jdbc_driver_library
    [php-src] Php内核的有趣高频宏
    [php-src] Php扩展开发的琐碎注意点、细节
    [ELK] Docker 运行 Elastic Stack 支持 TLS 的两种简单方式
    [Contract] Solidity 生成随机数方案
    [MySQL] 导入数据库和表的两种方式
    [ELK] 生产环境中 Elasticsearch 的重要配置项
  • 原文地址:https://www.cnblogs.com/jeremywucnblog/p/15674600.html
Copyright © 2020-2023  润新知