vue-router学习三:封装组件,封装TabBar组件。

简介: 这篇文章是关于如何使用Vue.js和vue-router封装一个自定义的TabBar组件的教程。

前言

一、封装思路

1. 效果展示

在这里插入图片描述

2. 实现思路

  1. 如果在下方有一个单独的TabBar组件,你如何封装

    • 自定义TabBar组件,在APP中使用
    • 让TabBar出于底部,并且设置相关的样式
  2. TabBar中显示的内容由外界决定

    • 定义插槽
    • flex布局平分TabBar
  3. 自定义TabBarItem,可以传入 图片和文字

    • 定义TabBarItem,并且定义两个插槽:图片、文字。
    • 给两个插槽外层包装div,用于设置样式。
    • 填充插槽,实现底部TabBar的效果
  4. 传入 高亮图片

    • 定义另外一个插槽,插入active-icon的数据
    • 定义一个变量isActive,通过v-show来决定是否显示对应的icon
  5. TabBarItem绑定路由数据

    • 安装路由:npm install vue-router —save
    • 完成router/index.js的内容,以及创建对应的组件
    • main.js中注册router
    • APP中加入<router-view>组件
  6. 点击item跳转到对应路由,并且动态决定isActive

    • 监听item的点击,通过this.$router.replace()替换路由路径
    • 通过this.$route.path.indexOf(this.link) !== -1来判断是否是active
  7. 动态计算active样式

    • 封装新的计算属性:this.isActive ? {'color': 'red'} : {}

二、代码实现

1. 新建项目

  1. 使用 vue init webpack 02-tabbar
  2. 安装的时候记得选择安装vue-router
  3. 项目目录结构如下:
    在这里插入图片描述

2. 创建组件Home/Category/Cart/Profie

新建目录view,用于存放各个页面的组件。
在这里插入图片描述

3. 配置路由规则

在这里插入图片描述

三、封装TarBar

1. 先封装TabBarItem.vue

<template>
  <div class="tab-bar-item" @click="itemClick">
    <div v-if="!isActive">
      <slot name="item-icon"></slot>
    </div>
    <div v-else>
      <slot name="item-icon-active"></slot>
    </div>
    <!-- :class="{active: isActive}" -->
    <div :style="activeStyle">
      <slot name="item-text"></slot>
    </div>
    <!--    <img src="../../assets/img/tabbar/home.svg" alt="">-->
    <!--    <div>首页</div>-->
  </div>
</template>

<script>
  export default {
    name: "TabBarItem",
    // 传过来的数据
    props: {
      path: String,
      activeColor: {
        type: String,
        default: 'red'
      }
    },
    data() {
      return {
        // isActive: true,
      }
    },
    computed: {
      isActive() {
        // this.$route.path 获取的事当前路由的 路径
        return this.$route.path.indexOf(this.path) !== -1
      },
      activeStyle() {
        return this.isActive ? {color: this.activeColor} : {}
      }
    },
    methods: {
      itemClick() {
        this.$router.replace(
          {path: this.path},
          onComplete => {
          },
          onAbort => {
          }
        )
        // console.log('itemClick')
      }
    }
  }
</script>

<style scoped>
  .tab-bar-item {
    flex: 1;
    text-align: center;
    height: 49px;
    font-size: 14px;
  }

  .tab-bar-item img {
    width: 24px;
    height: 24px;
    margin-top: 3px;
    vertical-align: middle;
    margin-bottom: 2px;
  }

</style>

2. TabBar.vue

<template>
  <div id="tab-bar">
    <slot></slot>
  </div>
</template>

<script>
  export default {
    name: "TabBar"
  }
</script>

<style scoped>
  @import "../../assets/css/base.css";

  #tab-bar{
    display: flex;
    background-color: #f7f7f7;

    /*放到下面*/
    position: fixed;
    left: 0;
    right: 0;
    bottom: 0;

    /*添加阴影*/
    box-shadow: 0 -3px 1px rgba(100, 100, 100, .1);
  }

</style>

3. MainTabBar.vue

<template>
  <TabBar>
    <tab-bar-item path="/home" activeColor="blue">
      <img slot="item-icon" src="../assets/img/tabbar/home.svg">
      <img slot="item-icon-active" src="../assets/img/tabbar/home_active.svg">
      <div slot="item-text">首页</div>
    </tab-bar-item>
    <tab-bar-item path="/category" activeColor="blue">
      <img slot="item-icon" src="../assets/img/tabbar/category.svg">
      <img slot="item-icon-active" src="../assets/img/tabbar/category_active.svg">
      <div slot="item-text">分类</div>
    </tab-bar-item>
    <tab-bar-item path="/cart" activeColor="blue">
      <img slot="item-icon" src="../assets/img/tabbar/shopcart.svg">
      <img slot="item-icon-active" src="../assets/img/tabbar/shopcart_active.svg">
      <div slot="item-text">购物车</div>
    </tab-bar-item>
    <tab-bar-item path="/profile" activeColor="blue">
      <img slot="item-icon" src="../assets/img/tabbar/profile.svg">
      <img slot="item-icon-active" src="../assets/img/tabbar/profile_active.svg">
      <div slot="item-text">我的</div>
    </tab-bar-item>
  </TabBar>
</template>

<script>
  import TabBar from "../components/tabbar/TabBar";
  import TabBarItem from "../components/tabbar/TabBarItem";

  export default {
    name: "MainTabBar",
    components: {
      TabBar,
      TabBarItem
    }
  }
</script>

<style scoped>

</style>

4. APP.vue 使用

<template>
  <div>
    <MainTabBar/>
    <router-view/>
  </div>
</template>

<script>
  import MainTabBar from "./components/MainTabBar";

export default {
  name: 'App',
  components: {
    MainTabBar
  }
}
</script>

<style>

</style>

5. 用到的地儿

  1. router
  2. slot 插槽
  3. 父传子 传参数
相关文章
|
JavaScript 开发者
【Vue3 组件封装】vue3 轮播图组件封装
【Vue3 组件封装】vue3 轮播图组件封装
175 0
|
8月前
|
JavaScript
vue封装面包屑组件
vue封装面包屑组件
52 0
|
8月前
vue3封装面包屑
vue3封装面包屑
155 0
|
JavaScript 前端开发 开发者
vue封装单文件弹窗组件
  VUE想要实现一个普通弹窗,想必对于大部分前端开发者来说实现起来都是非常简单一件事情,但是如果说要封装某一个具体需求效果,可能就难倒不少同学 ,由此可见 从实现到封装这个过程是非常考验个人能力的!
148 0
|
JavaScript 前端开发
Vue之函数式弹窗组件的封装原理
Vue之函数式弹窗组件的封装原理
659 0
【Vue3】面包屑组件封装
【Vue3】面包屑组件封装
103 0
Vue3简易路由菜单组件封装
Vue3简易路由菜单组件封装
162 0
|
JavaScript
Vue3自定义封装组件,父子组件传值
Vue3自定义封装组件,父子组件传值
|
前端开发 数据格式
#yyds干货盘点 React+antDesign封装一个tab组件(类组件) (1)
#yyds干货盘点 React+antDesign封装一个tab组件(类组件)
125 0
|
数据格式
#yyds干货盘点 React+antDesign封装一个tab组件(类组件) (2)
#yyds干货盘点 React+antDesign封装一个tab组件(类组件)
136 0