详解vue组件三大核心概念

前言

本文主要介绍属性、事件和插槽这三个vue基础概念、使用方法及其容易被忽略的一些重要细节。若是你阅读别人写的组件,也能够从这三个部分展开,它们能够帮助你快速了解一个组件的全部功能。html

clipboard.png

本文的代码请猛戳github博客,纸上得来终觉浅,你们动手多敲敲代码!前端

1、属性

1.自定义属性props

prop 定义了这个组件有哪些可配置的属性,组件的核心功能也都是它来肯定的。写通用组件时,props 最好用对象的写法,这样能够针对每一个属性设置类型、默认值或自定义校验属性的值,这点在组件开发中很重要,然而不少人却忽视,直接使用 props 的数组用法,这样的组件每每是不严谨的。vue

// 父组件
 <props name='属性'
           :type='type'
           :is-visible="false"
           :on-change="handlePropChange"
           :list=[22,33,44]
           title="属性Demo"
           class="test1"
           :class="['test2']"
           :style="{ marginTop: '20px' }" //注意:style 的优先级是要高于 style
           style="margin-top: 10px">
  </props>
// 子组件
  props: {
    name: String,
    type: {
   //从父级传入的 type,它的值必须是指定的 'success', 'warning', 'danger'中的一个,若是传入这三个之外的值,都会抛出一条警告
      validator: (value) => {
        return ['success', 'warning', 'danger'].includes(value)
      }
    },
    onChange: {
    //对于接收的数据,能够是各类数据类型,一样也能够传递一个函数
      type: Function,
      default: () => { }
    },
    isVisible: {
      type: Boolean,
      default: false
    },
    list: {
      type: Array,
      // 对象或数组默认值必须从一个工厂函数获取
      default: () => []
    }
  }

从上面的例中,能够得出props 能够显示定义一个或一个以上的数据,对于接收的数据,能够是各类数据类型,一样也能够传递一个函数。经过通常属性实现父向子通讯;经过函数属性实现子向父通讯git

2.inheritAttrs

这是2.4.0 新增的一个API,默认状况下父做用域的不被认做 props 的特性绑定将会“回退”且做为普通的 HTML 特性应用在子组件的根元素上。可经过设置 inheritAttrs 为 false,这些默认行为将会被去掉。注意:这个选项不影响 class 和 style 绑定
上个例中,title属性没有在子组件中props中声明,就会默认挂在子组件的根元素上,以下图所示:github

clipboard.png

3. data与props区别

  • 相同点

二者选项里均可以存放各类类型的数据,当行为操做改变时,全部行为操做所用到和模板所渲染的数据同时都会发生同步变化。segmentfault

  • 不一样点

data 被称之为动态数据,在各自实例中,在任何状况下,咱们均可以随意改变它的数据类型和数据结构,不会被任何环境所影响。api

props 被称之为静态数据,在各自实例中,一旦在初始化被定义好类型时,基于 Vue 是单向数据流,在数据传递时始终不能改变它的数据类型,并且不容许在子组件中直接操做 传递过来的props数据,而是须要经过别的手段,改变传递源中的数据。至于如何改变,咱们接下去详细介绍:数组

4.单向数据流

这个概念出如今组件通讯。props的数据都是经过父组件或者更高层级的组件数据或者字面量的方式进行传递的,不容许直接操做改变各自实例中的props数据,而是须要经过别的手段,改变传递源中的数据。那若是有时候咱们想修改传递过来的prop,有哪些办法呢?数据结构

  • 方法1:过渡到 data 选项中

在子组件的 data 中拷贝一份 prop,data 是能够修改的架构

export default {
  props: {
    type: String
  },
  data () {
    return {
      currentType: this.type
    }
  }
}

在 data 选项里经过 currentType接收 props中type数据,至关于对 currentType= type进行一个赋值操做,不只拿到了 currentType的数据,并且也能够改变 currentType数据。

  • 方法2:利用计算属性
export default {
  props: {
    type: String
  },
  computed: {
    normalizedType: function () {
      return this.type.toUpperCase();
    }
  }
}

以上两种方法虽能够在子组件间接修改props的值,但若是子组件想修改数据而且同步更新到父组件,却无济于事。在一些状况下,咱们可能会须要对一个 prop 进行『双向绑定』,此时就推荐如下这两种方法:

  • 方法3:使用.sync
// 父组件
<template>
  <div class="hello">
    <div>
      <p>父组件msg:{{ msg }}</p>
      <p>父组件数组:{{ arr }}</p>
    </div>
    <button @click="show = true">打开model框</button>
    <br />
    <demo :show.sync="show" :msg.sync="msg" :arr="arr"></demo>
  </div>
</template>

<script>
import Demo from "./demo.vue";
export default {
  name: "Hello",
  components: {
    Demo
  },
  data() {
    return {
      show: false,
      msg: "模拟一个model框",
      arr: [1, 2, 3]
    };
  }
};
</script>
// 子组件
<template>
  <div v-if="show" class="border">
    <div>子组件msg:{{ msg }}</div>
    <div>子组件数组:{{ arr }}</div>
    <button @click="closeModel">关闭model框</button>
    <button @click="$emit('update:msg', '浪里行舟')">
      改变文字
    </button>
    <button @click="arr.push('前端工匠')">改变数组</button> 
  </div>
</template>
<script>
export default {
  props: {
    msg: {
      type: String
    },
    show: {
      type: Boolean
    },
    arr: {
      type: Array //在子组件中改变传递过来数组将会影响到父组件的状态
    }
  },
  methods: {
    closeModel() {
      this.$emit("update:show", false);
    }
  }
};

clipboard.png

父组件向子组件 props 里传递了 msg 和 show 两个值,都用了.sync 修饰符,进行双向绑定。
不过.sync 虽好,但也有限制,好比:

1)不能和表达式一块儿使用(如 v-bind:title.sync="doc.title + '!'" 是无效的);
2)不能用在字面量对象上(如 v-bind.sync="{ title: doc.title }" 是没法正常工做的)。

  • 方法4:将父组件中的数据包装成对象传递给子组件

这是由于在 JavaScript 中对象和数组是经过引用传入的,因此对于一个数组或对象类型的 prop 来讲,在子组件中改变这个对象或数组自己将会影响到父组件的状态。好比上例中在子组件中修改父组件传递过来的数组arr,从而改变父组件的状态。

5.向子组件中传递数据时加和不加 v-bind?

对于字面量语法和动态语法,初学者可能在父组件模板中向子组件中传递数据时到底加和不加 v-bind 会感受迷惑。

v-bind:msg = 'msg'

这是经过 v-bind 进行传递数据而且传递的数据并非一个字面量,双引号里的解析的是一个表达式,一样也能够是实例上定义的数据和方法(其实就是引用一个变量)。

msg='浪里行舟'

这种在没有 v-bind 的模式下只能传递一个字面量,这个字面量只限于 String 类量,字符串类型。那若是想经过字面量进行数据传递时,若是想传递非String类型,必须props名前要加上v-bind,内部经过实例寻找,若是实例方没有此属性和方法,则默认为对应的数据类型。

:msg='11111' //Number 
:msg='true' //Bootlean 
:msg='()=>{console.log(1)}' //Function
:msg='{a:1}' //Object

2、事件

1.事件驱动与数据驱动

用原生JavaScript事件驱动一般是这样的流程:

  • 先经过特定的选择器查找到须要操做的节点 -> 给节点添加相应的事件监听
  • 而后用户执行某事件(点击,输入,后退等等) -> 调用 JavaScript 来修改节点

这种模式对业务来讲是没有什么问题,可是从开发成本和效率来讲会比较不理想,特别是在业务系统愈来愈庞大的时候。另外一方面,找节点和修改节点这件事,效率自己就很低,所以出现了数据驱动模式。

Vue的一个核心思想是数据驱动。所谓数据驱动,是指视图是由数据驱动生成的,咱们对视图的修改,不会直接操做 DOM,而是经过修改数据,其流程以下:

用户执行某个操做 -> 反馈到 VM 处理(能够致使 Model 变更) -> VM 层改变,经过绑定关系直接更新页面对应位置的数据

能够简单地理解:数据驱动不是操做节点的,而是经过虚拟的抽象数据层来直接更新页面。主要就是由于这一点,数据驱动框架才得以有较快的运行速度(由于不须要去折腾节点),而且能够应用到大型项目。

2.修饰符事件

Vue事件分为普通事件和修饰符事件,这里咱们主要介绍修饰符事件。

Vue 提供了大量的修饰符封装了这些过滤和判断,让开发者少写代码,把时间都投入的业务、逻辑上,只须要经过一个修饰符去调用。咱们先来思考这样问题:怎样给这个自定义组件 custom-component 绑定一个原生的 click 事件?

<custom-component>组件内容</custom-component>

若是你的回答是<custom-component @click="xxx">,那就错了。这里的 @click 是自定义事件 click,并非原生事件 click。绑定原生的 click 是这样的:

<custom-component @click.native="xxx">组件内容</custom-component>

实际开发过程当中离不开事件修饰符,常见事件修饰符有如下这些:

  • 表单修饰符

1).lazy

在默认状况下,v-model 在每次 input 事件触发后将输入框的值与数据进行同步 。你能够添加 lazy 修饰符,从而转变为使用 change事件进行同步。适用于输入完全部内容后,光标离开才更新视图的场景。

2).trim

若是要自动过滤用户输入的首尾空白字符,能够给 v-model 添加 trim 修饰符:

<input v-model.trim="msg">

这个修饰符能够过滤掉输入完密码不当心多敲了一下空格的场景。须要注意的是,它只能过滤首尾的空格!首尾,中间的是不会过滤的。

3).number

若是想自动将用户的输入值转为数值类型,能够给 v-model 添加 number 修饰符:

<input v-model.number="value" type="text" />

clipboard.png

从上面例子,能够获得若是你先输入数字,那它就会限制你输入的只能是数字。若是你先输入字符串,那它就至关于没有加.number

  • 事件修饰符
<!-- 阻止单击事件继续传播 -->
<a v-on:click.stop="doThis"></a>

<!-- 提交事件再也不重载页面 -->
<form v-on:submit.prevent="onSubmit"></form>

<!-- 修饰符能够串联 -->
<a v-on:click.stop.prevent="doThat"></a>

3、插槽

插槽分为普通插槽和做用域插槽,其实二者很相似,只不过做用域插槽能够接受子组件传递过来的参数。

1.做用域插槽

咱们不妨经过一个todolist的例子来了解做用域插槽。若是当item选中后,文字变为黄色(以下图所示),该如何实现呢?

clipboard.png

// 父组件
<template>
  <div class="toList">
    <input v-model="info" type="text" /> <button @click="addItem">添加</button>
    <ul>
      <TodoItem v-for="(item, index) in listData" :key="index">
        <template v-slot:item="itemProps"> // 这是个具名插槽
        // 其中itemProps的值就是子组件传递过来的对象
          <span
            :style="{
              fontSize: '20px',
              color: itemProps.checked ? 'yellow' : 'blue'
            }"
            >{{ item }}</span
          >
        </template>
      </TodoItem>
    </ul>
  </div>
</template>
<script>
import TodoItem from "./TodoItem";
export default {
  components: {
    TodoItem
  },
  data() {
    return {
      info: "",
      listData: []
    };
  },
  methods: {
    addItem() {
      this.listData.push(this.info);
      this.info = "";
    }
  }
};
</script>
// 子组件
<template>
  <div>
    <li class="item">
      <input v-model="checked" type="checkbox" />
      <slot name="item" :checked="checked"></slot> // 将checked的值传递给父组件
    </li>
  </div>
</template>
<script>
export default {
  data() {
    return {
      checked: false
    };
  }
};
</script>

值得注意:v-bind:style 的对象语法十分直观——看着很是像 CSS,但实际上是一个 JavaScript 对象。CSS 属性名能够用驼峰式 (camelCase) 或短横线分隔 (kebab-case,记得用引号括起来) 来命名。

2.v-slot新语法

在 2.6.0 中,咱们为具名插槽和做用域插槽引入了一个新的统一的语法 (即 v-slot 指令)。它取代了 slot 和 slot-scope 。咱们来思考个问题:相同名称的插槽是合并仍是替换

  • Vue2.5版本,普通插槽合并、做用域插槽替换
  • Vue2.6版本,都是替换(见下面例子)

咱们经过一个例子介绍下Vue2.6版本默认插槽、具名插槽和做用域插槽的新语法:

// 父组件
<template>
  <div class="helloSlot">
    <h2>2.6 新语法</h2>
    <SlotDemo>
      <p>默认插槽:default slot</p>
      <template v-slot:title>
        <p>具名插槽:title slot1</p>
        <p>具名插槽:title slot2</p>
      </template>
      <template v-slot:title>
        <p>new具名插槽:title slot1</p>
        <p>new具名插槽:title slot2</p>
      </template>
      <template v-slot:item="props">
        <p>做用域插槽:item slot-scope {{ props }}</p>
      </template>
    </SlotDemo>
  </div>
</template>
<script>
import Slot from "./slot";
export default {
  components: {
    SlotDemo: Slot
  }
};
</script>
// 子组件
<template>
  <div>
    <slot />
    <slot name="title" />
    <slot name="item" :propData="propData" />  // propData这个属性名能够任意取
  </div>
</template>
<script>
export default {
  data() {
    return {
      propData: {
        value: "浪里行舟"
      }
    };
  }
};
</script>

slot

给你们推荐一个好用的BUG监控工具Fundebug,欢迎免费试用!

欢迎关注公众号:前端工匠,你的成长咱们一块儿见证!
image

参考文章