vue组件化开发、vue组件封装使用步骤、scoped实现组件的私有样式、vue组件通信、父传子、子传父

vue组件化开发

组件是可复用的 Vue 实例, 封装标签, 样式和JS代码

vue组件分类:

  1. 页面组件
  2. 页面下的功能组件

组件化开发 :一个页面(.vue)可能有一个或多个组件(.vue)组成完整的页面功能

● 封装的思想,把页面上 可重用的部分 封装为 组件,从而方便项目的 开发 和 维护

一个页面, 可以拆分成一个个组件,一个组件就是一个整体, 每个组件可以有自己独立的 结构(template) 样式(style) 和 行为(script) (html, css和js)
在这里插入图片描述

小结

● 现代前端开发均会使用组件化的开发思路
● 组件化开发有利于解决代码重复、冗余等问题

vue组件-封装使用

为啥要封装组件

  1. 复用。一次封装,多次使用
  2. 代码整理,方便维护。

步骤

  1. 定义组件
  2. 注册组件
  3. 使用组件

在这里插入图片描述

案例

定义一个名为MyCom的组件,并在App.vue中使用它

目录
├── App.vue  # 在App.vue内部,导入并使用组件
└── MyCom.vue
  1. 创建组件: MyCom.vue 组件名开头大写驼峰(推荐)
  2. 引入并注册组件
// 局部注册组件

// 进入到当前组件内部
// 1. 导入组件
import 组件名 from './组件文件.vue'

// 2. 局部注册
export default {
   components: {
     组件名: 组件名
   }
}
  1. 使用组件。在当前页面中,当做标签来使用。
    注意:
    组件名不能与现有的html标签名一致。

小结

● 每一个组件都是封闭的。它有自己的template, script,style
● 组件之间可以相互引用使用。

vue组件-用scoped实现组件的私有样式

目标

解决多个组件样式名相同, 冲突问题

问题说明

默认组件style 中定义的样式是全局=》存在相同名字覆盖的情况
在这里插入图片描述

解决方案

局部样式:在style标签上加上scoped属性

<stype scoped>
  h2 {} // 样式只会在当前组件内生效
</style>

原理

● 在style上加入scoped属性, 就会在此组件的标签上加上一个随机生成的data-v开头的属性
● 而且必须是当前组件的元素或者子组件的根元素, 才会有这个自定义属性
在这里插入图片描述
在这里插入图片描述

小结

style上加scoped, 组件内的样式只在当前vue组件生效;相反,样式就是全局的

vue组件-/deep/深度作用选择符

问题导入

当父子组件都使用了scoped的情况下,如何在父组件中控制子组件的样式?

解决方案

父组件的选择器 /deep/ .子组件的选择器

父组件:

<template>
  <div class="box">
    <h1 class="red">父组件</h1>
    <hr />
+    <Child />
  </div>
</template>

<script>
import Child from '@/components/Child'
export default {
  components: {
    Child,
  },
}
</script>

<style scoped>
.red {
  color: blue;
}
+ .box /deep/ h2 {
+  color: lawngreen;
+ }
</style>

子组件:

<template>
  <div>
     <h2>子组件</h2>
     <p class="red">
      <span>123</span>
    </p>
  </div>
</template>

<script>
export default {

}
</script>

<style scoped>
.red {
  color: red;
}
</style>

小结

父组件中控制子组件元素或类名,覆盖样式=》需要在前边加上 /deep/

注意⚠️:默认子组件的根元素,会带上父组件的data-v-hash属性,所以可以直接控制

vue组件通信

背景

  1. 一个页面有多个组件构成
  2. 每个组件之间的数据是相互独立的

问题: 如何在组件之间做通讯?

因为每个组件的变量和值都是独立的=》 如果想获取对方页面中定义的变量应该怎么做?

在这里插入图片描述
组件通信先暂时关注父传子(数据从父组件传递给子组件), 子传父(数据从子组件传递给父组件)
在这里插入图片描述
● 父: 使用其他组件的vue文件

● 子: 被引入到这个vue文件的组件(嵌入)

vue组件通信_父传子

父子组件

如果一个组件A在组件B中被导入使用,称组件B是父组件,组件A是子组件

格式

在这里插入图片描述

示例代码

父组件 : 在子组件的标签中添加 :自定义属性名="值"

<template>
<div style="border:1px solid #ccc; margin:5px;padding:5px">
  <h1>父组件</h1>
  <!-- 1. 父传。自定义属性 -->
  <MyCom :abc="userName" :list="hobby"/>
  </div>
</template>

<script>
  // 导入->注册->使用
  import MyCom from './MyCom.vue'
  export default {
    data(){
      return {
        userName: '小花',
        hobby: ['vue','react']
      }
    },
    components: { MyCom }
  }
</script>

<style>
  
</style>

子组件 : props: [‘自定义的属性名’']

<template>
  <div style="border:1px solid #ccc; margin:5px;padding:5px">
    <h2>子组件</h2>
    <!-- 使用 -->
    {{abc}}
    <p>
      {{list[0]}}
    </p>
    <button @click="fn">打印</button>
  </div>
</template>

<script>
export default {
  // 2.子接
  props: ['abc', 'list'],
  methods: {
    fn(){
      console.log(this, this.abc)
      
    }
  }
}
</script>

<style>

</style>

在这里插入图片描述
props属性名建议都小写,因为标签里的属性只能小写/把变量驼峰转成-连接

vue组件通信_父向子-循环复用

目标

对子组件使用v-for循环,把数据循环分别传入给组件内显示

<template>
  <div>
    <MyProduct v-for="obj in list"
    :title="obj.proname" 
    :price="obj.proprice" 
    :info="obj.info"
    :key="obj.id"></MyProduct>
  </div>
</template>

<script>
import MyProduct from "./components/MyProduct_13.1";
export default {
  data() {
    return {
      list: [
        { id: 1, proname: "超级好吃的棒棒糖", proprice: 18.8, info: '开业大酬宾, 全场8折' },
        { id: 2, proname: "超级好吃的大鸡腿", proprice: 34.2, info: '好吃不腻, 快来买啊' },
        { id: 3, proname: "超级无敌的冰激凌", proprice: 14.2, info: '炎热的夏天, 来个冰激凌了' },
      ],
    };
  },
  components: {
    MyProduct,
  },
};
</script>

<style>
</style>

vue单向数据流-不要修改props

在vue中需要遵循单向数据流原则

  1. 在父传子的前提下,父组件的数据发生会通知子组件自动更新
  2. 子组件内部,不能直接修改父组件传递过来的props => props是只读的
    在这里插入图片描述
    父组件
<template>
  <div style="border:1px solid #ccc; margin:5px;padding:5px">
    <h1>31-vue单向数据流-父组件</h1>
    <MyCom
      :name="name"
      :hobby="hobby"/>
      <button @click="fn">改数据</button>
  </div>
</template>

<script>
// 导入
import MyCom from './MyCom.vue'
export default {
  data(){
    return {
      name: '小花',
      hobby:['vue', 'react']
    }
  },
  components: { MyCom },
  methods: {
    fn(){
      this.name = '小花花'
      this.hobby.push('小程序')
    }
  }
}
</script>

<style>

</style>

子组件

<template>
  <div style="border:1px solid #ccc; margin:5px;padding:5px">
    <h2>子组件</h2>
    <p>
      name: {{name}}
    </p>
    <p>
      hobby: {{hobby}}
    </p>
    <button @click="fn">修改props</button>
  </div>
</template>

<script>
export default {
  props: ['name', 'hobby'],
  methods: {
    fn(){
      // 直接去修改props ===> 改了父组件传来的数据
      // 这里打破 单向数据流的规则,vue能捕获到错误
      // this.name = '小花花'

      // 这里打破 单向数据流的规则,vue能不能捕获到错误
      // hobby是引用数据类型,push并没有修改 数组的地址  但是赋值会修改地址
      this.hobby.push('小程序')
    }
  }
}
</script>

<style>

</style>

子组件的v-model绑定的值如果是父传子的,会导致修改props
在这里插入图片描述
特殊说明

说明:父组件传给子组件的是一个对象,子组件修改对象的属性,是不会报错的,对象是引用类型, 互相更新;但不能改变引用地址

小结

props的值不能重新赋值, 但是引用类型可以子改父,最好不要改

vue组件通信_子传父

子传父是指:从子组件内部把数据传出来给父组件使用或者修改父组件数据
在这里插入图片描述

语法

● 父组件中:< 子组件 @自定义事件名1=“父methods函数1” @自定义事件名2=“父methods函数2” />
● 子: this.$emit(“自定义事件名1”, 传值1) —> 执行父methods里函数代码

在这里插入图片描述
父组件

<template>
  <div style="border:1px solid #ccc; margin:5px;padding:5px">
    <h1>32-子传父</h1>

    <!-- 1. 添加事件监听 -->
    <!-- 当子组件发生了abc事件要执行fn函数 -->
    <MyCom @abc="fn"/>
  </div>
</template>

<script>
// 导入
import MyCom from './MyCom.vue'
export default {
  components: { MyCom },
  methods: {
    fn(obj){
      console.log('fn-子组件发生了abc事件',obj)
    }
  }
}
</script>

<style>

</style>

子组件

<template>
  <div style="border:1px solid #ccc; margin:5px;padding:5px">
    <h2>子组件</h2>
    <button @click="fn">触发abc事件</button>
  </div>
</template>

<script>
export default {
  methods: {
    fn(){
      console.log('子组件click')
      // 2. 触发abc事件
      this.$emit('abc',{name:'小花'})
    }
  }
}
</script>

在这里插入图片描述

小结

自定义事件 + $emit

拓展-全局注册

全局入口在main.js, 在new Vue之上注册

语法:

import Vue from 'vue'
import 组件对象 from 'vue文件路径'

Vue.component("组件名", 组件对象)

main.js

import Vue from 'vue'
import App from './App.vue'
+ import Pannel from './components/Pannel' // 引入组件文件对象

+ Vue.component("PannelCom", Pannel) // 组件名开头大写驼峰(推荐) => 全局注册一个组件

Vue.config.productionTip = false

new Vue({
  render: h => h(App),
}).$mount('#app')

全局注册PannelCom组件名后, 就可以当做标签在任意template里用

单双标签都可以, 运行后, 会把这个自定义标签当做组件解析, 使用组件里封装的标签替换到这个位置

在页面中使用:

<template>
  <div id="app">
    <h3>案例:折叠面板</h3>
    <PannelCom></PannelCom>
    	<!-- or -->
    <PannelCom />
  </div>
</template>

总结

  • 组件分类=》1.页面组件 2. 页面下功能 (.vue格式)
  • 组件化开发是什么 =》一个页面由多个.vue文件组成,完成一个完整的页面效果
  • 组件创建和复用 =》1. 全局 (main.js) 2. 局部
  • 掌握组件通信的主要方式
    a. 父传子 =》父组件:提供数据 =》:传递数据名字=“变量” | 子组件:接收props:[‘传递数据名字’]
    b. 子传父(单向数据流)=》父组件:提供自定义事件=》@语义化事件名=“callback” | 子组件:通知父组件修改 this.$emit(‘语义化事件名’, data,data2)
评论 3
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值