Vue.js2.0快速入门笔记

时间:2022-08-16 05:10:18

vue.js 解耦视图与数据,可复用的组件,前端路由,状态管理,虚拟DOM。

MVVM模式:当View(视图层)变化时,会自动更新ViewModel(视图模型),View与ViewModel之间双向绑定。

【Vue.js使用第一个程序】

<div id="app">
  <p>{{ message }}</p>
</div>
<script>
new Vue({
  el: '#app',
  data: {
    message: 'Hello Vue.js!'
  },
  created: function(){
    console.log(this.message); //Hello Vue.js!
  },
  mounted: function(){
    console.log(this.$el);
  },
  beforeDestroy: function(){

  },
  filters: {
    filterA:function(v){
      return v;
    }
  },
  methods: {
    close: function(){

    }
  },
  computde: {
    msg: function(){
      return this.message;
    }
  }
})
</script>


【第1-3章,Vue介绍、数据绑定、计算属性】

*el:挂载DOM对象。
*data:绑定数据集。
*created:实例创建完成后调用,但尚未挂载,$el还不可用。
*mounted:el挂载到实例上后调用,一般第一个业务逻辑在这里开始。
*beforeDestroy:实例销毁之前调用。
*filters:过滤器,支持在{{}}插值的尾部添加一个管道符“|”,经常用于格式化文本,
过滤器也可以串联多个,也可以传递参数,接收第一个参数默认是数据本身,
串联{{message|filterA|filterB}},接收参数{{message|filterA('arg1','arg2')}}。
过滤器应当用于简单的文本转换,如要实现复杂的数据变化,应使用“计算属性”。
*methods:声明所有绑定事件监听器实现的方法,每个方法以函数形式声明,
实例方法内可以通过this直接调用其他申明的方法,外部也可以通过声明实例名调用。
*computde:计算属性,复杂的逻辑都可以在这里实现,最终返回结果,
计算属性还可以返回多个Vue实例的数据,其中一个数据变化,计算属性会重新执行。
计算属性包含get和set函数,可以通过set函数进行修改数据值,一般set用的少。
计算属性可以依赖其他计算属性,也可以依赖其他实例中的数据。
计算属性缓存,所依赖的数据发生变化时,它才会重新取值。


2.1.3 {{}}:输出data中的属性值,里面还可以使用javascript表达式、三元运算等,vue.js只支持单个表达式,不支持语句和流控制,
另外,在表达式中不能使用用户自定义的全局变量,只能使用Vue自带的全局变量,如Math和Date等。

v-html:与{{}}不同的是,可以输出html,双大括号是解析后的纯文本。
这里要注意,如果使用v-html输出后,有可能导致XSS攻击,所以要在服务端对用户提交的内容进行处理。

2.2 v-:指令,vue.js中指令前缀带有“v-”,比如v-if、v-html、v-pre等,
指令主要职责就是当其表达式的值改变时,将某些行为应用到DOM上。

v-pre: 即可跳过当前元素和子元素的编译过程,可以将{{}}直接文本形式显示出来。

v-bind:动态更新HTML元素上的属性,比如id、class等,v-bind:id="id"、v-bind:class="class"。

v-on:用来绑定事件监听器,这样我们就可以做一些交互了,v-on:click="onclose",
onclose这些方法都写在Vue实例的methods属性内,表达式除了方法名,也可以直接是一个内联语句,
v-on:click="show=false",show是data中属性。

2.3 语法糖:
“v-bind:” 可以用:代替
“v-on” 可以用@代替

【第4章,关于样式class和style绑定】

4.2.1 绑定class对象:<div class="h12" :class="{'active':isActive,'error':isError}"></div>
以上设置对象可以动态切换class,当数据isActive和isError变化时,对应的class类名也会更新。

4.2.2 绑定class数组:<div :class="['active','error']"></div>
active和error直接依赖于数据,数组中也可以用三元表达式,数组中也可以包含对象。

当:class的表达式过长或逻辑复杂时,还可以绑定一个计算属性和方法,表达式用[]括起来优先级。

例:return ['btn',{['btn-'+this.size]: this.size!=='',['btn-disabled']: this.disabled}]
当数据this.size不为空时,会应用‘btn-size12’,当数据this.disabled为true时,会应用‘btn-disabled’。

备注:使用计算属性给元素动态设置类名,在业务中经常用到,尤其是在写复用的组件时,应该尽可能的优先使用计算属性。

4.2.3 组件上绑定class:<my-component :class="{'active':isActive}"></my-component>
这种语法仅适用于自定义组件的最外层是一个根元素,否则会无效,需要给子元素设置类名时,应当使用组件props来传递。

4.3 绑定style内联样式:语法跟:class类似,<div :style="{'color':color,'fontSize':fontSize+'px'}"></div>
为了便于阅读和维护,一般写在data或computde里,应用多个样式对象时,可以使用数组语法。

【第5章,内置指令】
5.1.1 v-cloak:不需要表达式,它会在vue实例结束编译时从绑定的HTML元素上移除,经常和css的display:none配合使用。
在一般的情况下,v-cloak是一个解决初始化慢导致页面闪动的最佳实践,但使用webpack和vue-router之后,项目的HTML结构
只有一个空的div元素,剩余的内容都是由路由去挂载不同的组件来完成的,所以不再需要v-cloak。
备注:解决刚加载页面,但Vue未执行完成时,页面上会出现{{...}}显示问题。

v-once:也是一个不需要表达式的指令,作用是定义它的元素或组件只渲染一次,包括元素或组件的所有子节点。
首次渲染后,不再随着数据的变化重新渲染,将被视为静态内容。

5.2.1 v-if、v-else-if、v-else:与javascript的条件语句类似。
在渲染元素时,处于效率考虑,会尽可能的复用已有的元素,当切换后dom之前输入的值不变。
如果不希望这么做,可以使用key属性,唯一key值来控制是否要复用。

5.2.2 v-show:用法与v-if基本一致,只不过v-show是改变元素的css属性display,元素隐藏。
v-show不能在<template>上使用,频繁切换场景中适合使用。

v-for:列表渲染指令,循环显示的时候使用,需要配合in来使用,数组和对象都可以遍历。

<ul>
  <li v-for="book in books">{{book.name}}</li>
</ul>

v-for的表达式支持一个可选参数作为当前项的索引:

<ul>
  <li v-for="(book,index) in books">{{index}}-{{book.name}}</li>
</ul>

5.3.1 与v-if一样,v-for也可以用在内置标签<template>上,将多个元素进行渲染。
遍历对象属性时,有两个可选参数,分别是键名和索引。

5.3.2 数组更新:当我们修改数组时,视图也会立即更。
Vue包含了一组观察数组非变异的方法:
push()、pop()、shift()、unshift()、splice()、sort()、reverse()、filter()、concat()、slice()
它们返回的是一个新数组,在使用这些非变异的方法时,可以用新数组来替换原数组,相同的元素不会重新渲染。

app.books = app.books.filter(function(item){
  return item.name.match(/JavaScript/);
});

关于通过索引修改更新数据,app.books[3]={...} 视图不会被更新,可以使用vue内置的set方法。

Vue.set(app.books,3,{name:'abc',author:'jack'});

如果是在组件化的方式,默认是没有导入Vue的,这是可以使用$set。
关于修改数组长度,app.books.length=1,视图也不会更新,可以使用splice来解决。
app.books.splice(1);

5.3.3 过滤与排序。
当你不想改变原数组,想通过一个数组的副本来做过滤或排序显示时,可以使用计算属性来返回过滤或排序后的数组。

5.4 方法与事件
@click调用的方法名后面可以不写(),vue提供了一个特殊变量$event,用于访问原生DOM事件。

5.4.2 修饰符。
Vue支持以下修饰符:.stop、.prevent、.capture、.self、once。修饰符可以串联使用。


【第6章,表单与v-model】
6.1 v-model:完成表单类控件的数据双向绑定,如input、select等。
<input type="text" v-model="message" placeholder="输入...">
备注:v-model也是一个特殊的语法糖,使用@input来替代v-model,可以实时更新。

单选按钮radio:分单独使用和组合使用,单独使用时,不需要绑定v-model,使用v-bind绑定一个布尔类型,
组合使用时需要v-model和value来配合使用,v-model绑定数据为选中的value值,字符串类型。

<input type="radio" :checked="picked">
{data:{picked:true}}
<input type="radio" v-model="picked" value="html" id="html"><label for="html">HTML</label>
<input type="radio" v-model="picked" value="css" id="css"><label for="css">CSS</label>
{data:{picked:'html'}}

复选框checked:复选框也分单选使用和组合使用,v-model都绑定到同一个数据,单选时值为布尔值,
组合使用时,需要value值,自动push到绑定的数组中,数据类型为字符串数组。

<input type="checkbox" v-model="checked" id="checked">
{data:{checked:false}}
<input type="checkbox" v-model="checked" value="html" id="html"><label for="html">HTML</label>
<input type="checkbox" v-model="checked" value="css" id="css"><label for="css">CSS</label>
{data:{checked:['html','css']}}

选择列表select:下拉选择也分单选和多选两种方式,单选时绑定数据为字符串,多选时为字符串数组形式,
<option>是备选项,如果包含value属性,v-model会优先取value值,没有则取text值,multiple属性支持多选。

<select v-model="selected" multiple>
<option>html</option>
<option value="js">JavaScript</option>
{data:{selected:['html','js']}}

在业务中<option>经常用v-for动态输出,value和text也是用v-bind来动态输出的。

<option v-for="option in options" :value="option.value"> {{option.text}} </option>

6.2 绑定值:单选按钮、复选框和选择列表在单独使用或单选的模式下,v-model绑定的值是一个静态字符串或布尔值,
但在业务中,有时需要绑定一个动态的数据,这时可以用v-bind来实现。

<input type="radio" v-model="picked" :value="value">
{data:{picked:false,value:123}} //选中后,picked值为123
<input type="checkbox" v-model="toggle" :true-value="value1" :false-value="value2">
{data:{toggle:false,value1:'a',value2:'b'}} //选中时toggle为a,未选中时toggle为b
<select v-model="selected">
<option :value="{number:123}">123</option> //当选中时,app.selected是一个Object,app.selected.number===123
</select>

6.3 修饰符:与事件的修饰符类似,v-model也有修饰符,用于控制数据同步的时机。
v-model.lazy:在输入框中,v-model默认是在input事件中同步输入框的数据,使用.lazy会转变为change事件中同步。
v-model.number:可以将输入转换为Number类型,v-model默认输入的数字其实也是字符串类型。
v-model.trim:可以自动过滤输入的首位空格。


【第7章,组件详解】
7.1.2 组件用法,组件需要注册后才能使用,分全局注册和局部注册,全局注册后,任何Vue实例都可以使用。

Vue.component("my-component"),{
 template: '<div>这里是组件内容</div>'
})

使用组件:<my-component></my-component> //my-component就是注册的组件自定义标签名称。
在Vue实例中,使用components选项可以局部注册组件,组件也可以嵌套组件。
<table>内无法直接使用组件,可以使用is特殊指令挂载组件,除了table还有ul、ol、select。

<div id="app">
  <table>
    <tbody is="my-component"></tbody>
  </table>
</div> 

注册组件,除了template选项外,还可以有data、computed、methods等选项。

Vue.component('my-component',{
  template:'<div>{{message}}</div>',
  data: function(){
    return { message: '组件内容' };  //这里data必须是函数用return返回
  }
});

备注:如果data中引用了外部对象,那么这个对象就是共享的,所有修改同步,如果想复用组件,必须在内部返回一个新的对象。

7.2.1 使用props传递数据,组件间进行通信,父组件的模板中包含子组件,父组件要正向的向子组件传递数据或参数,
子组件根据接收不同数据渲染不同的内容或执行操作,这种正向传递数据的过程就是通过props来实现的。

<my-component message='来自父组件的数据'>
Vue.component('my-component',{
  props:['message'], //props的值分两种,字符串数组或对象。
  template:'<div>{{message}}</div>'
});

备注:props中声明的数据来自父级,组件data函数return的数据是组件自己的数据。

驼峰命名:当使用DOM模板时,props名称要转为短横分割命名。
例如:

<my-component warning-text='来自父组件的数据'>
{props:['warningText'],template:'<div>{{warningText}}</div>'}

直接传递数字、布尔值、数组、对象和通过v-bind绑定传递值区别?

<my-component message='[1,2,3]'></my-component>  //长度为7
<my-component :message='[1,2,3]'></my-component> //长度为3
Vue.component('my-component',{
  props:['message'],
  template:'<div>{{message.length}}</div>'
});

7.2.2 单项数据流,业务中经常遇到两种需要改变prop的情况,一种是父组件传递初始值进来,
子组件将它作为初始值保存起来,在自己的作用域下可以随意使用和修改。

<my-component :init-count='1'></my-component>
Vue.component('my-component',{
  props: ['initCount'],
  template:'<div>{{ count }}</div>',
  data: function(){
    return {
      count: this.initCount
    }
  }
}

另一种情况是prop作为需要被转变的原始值传入,这种情况用计算属性就可以了。

<my-component :width='100'></my-component>
Vue.component('my-component',{
 props:['width'],
 template:'<div :style="style">组件内容</div>',
 computed:{
  style:function(){
   return {
    width: this.width + 'px'
   }
  }
 }
});

备注:在JavaScript中对象和数组是引用类型,指向同一个内存空间,所以props是对象和数组时,
在子组件内改变时会影响父组件的。

7.2.3 数据验证,props选项的值需要用对象,验证数据非法,会在控制台弹出警告。
验证的type类型包括:String、Number、Boolean、Object、Array、Function
type也可以是一个自定义构造器,使用instanceof检测。

Vue.component('my-component',{
  props:{
    //必须是数字类型
    propA: Number,
    //必须是字符串或数字类型
    propB: [String, Number],
    //布尔类型,默认值为true
    propC: {
      type: Boolean,
      default: true
    },
    //数字类型,必传参数
    propD: {
      type: Number,
      required: ture
    },
    //数组或对象类型,默认值必须是一个函数来返回
    propE: {
      type: Array,
      default: function(){
        return [];
      }
    },
    //自定义一个验证函数
    propF: {
      validator: function(value){
        return value > 10;
      }
    }
  }
});

7.3.1 组件通信自定义事件,当子组件需要向父组件传递数据时,就要用到自定义事件。
子组件用$emit()来触发事件,父组件用$on()来绑定监听子组件的事件。
父组件也可以直接在子组件的自定义标签上使用v-on来监听子组件触发的自定义事件。

<div id="app">
  <p>总数:{{ total }}</p>
  <my-component @increase="handleGetTotal" @reduce="handleGetTotal"></my-component>
</div>
Vue.component('my-component',{
  template: '\
  <div>\
    <button @click="handleIncrease">+1</button>\
    <button @click="handleReduce">-1</button>\
  </div>',
  data: function(){
    return {
      counter: 0
    }
  },
  methods:{
    handleIncrease: function(){
      counter++;
      this.$emit('increase',this.counter);
    },
    handleReduce: function(){
      counter--;
      this.$emit('reduce',this.counter);
    }
  }
});
var app = new Vue({
  el: '#app',
  data: {
    total: 0
  },
  methods: {
    handleGetTotal: function(total){
      this.total = total;
    }
  }
});

解析:increase和reduce是自定义事件标签,由子组件$emit来触发这个事件。
备注:v-on在组件上监听事件,可以使用.native修饰符表示监听的是一个原生事件。

7.3.2 使用v-model,在自定义组件上使用v-model指令。
v-model是语法糖,$emit()事件名是特殊的input。
v-model还可以用来创建自定义的表单输入组件,进行数据双向绑定。
双向绑定要满足两个要求:接收一个value属性,在有新的value时触发input事件。

<div id="app">
  <p>总数:{{ total }}</p>
  <my-component v-model="total"></my-component>
  <button @click="handleReduce">-1</button>
</div>
Vue.component('my-component',{
  props: ['value'],
  template: '<input :value="value" @input="updateValue">',
  methods: {
    updateValue: function(event){
      this.$emit('input',event.target.value);
    }
  }
});
var app = new Vue({
  el: '#app',
  data: {
    total: 0
  },
  methods: {
    handleReduce: function(){
      this.total--;
    }
  }
});

7.3.3 非父子组件通信,非父子组件一般有两种,兄弟组件和跨多级组件。
使用一个空的Vue实例作为*事件总线(bus),也就是一个中介,在实例初始化时让bus获取一次,
任何时间,任何组件就可以从中直接使用了。

<div id="app">
  {{ message }}
  <component-a></component-a>
</div>

var bus = new Vue();
Vue.component('component-a',{
  template: '<button @click="handleEvent">传递事件</button>',
  methods: {
    handleEvent: function(){
      bus.$emit('on-message','来自组件component-a的内容');
    }
  }
});

var app = new Vue({
  el: '#app',
  data: {
    message: ''
  },
  mounted: function(){
    var _this = this;
    //在实例初始化时,监听来自bus实例的事件
    bus.$on('on-message',function(msg){
      _this.message = msg;
    });
  }
});

父链:在子组件中,使用this.$parent可以直接访问该组件的父实例或组件,
this.$parent.message = '来自组件component-a的内容';
子组件索引:父组件也可以通过this.$children访问它所有的子组件。
在父组件模板中,子组件标签上使用ref指定一个名称,并在父组件内通过this.$refs来访问指定名称的子组件。

<component-a ref="comA"></component-a>
this.$refs.comA.message;

备注:除了*事件总线bus,父链和子组件索引也可以实现组件间通信,但业务中子组件尽可能避免依赖父组件。
$refs是非响应式的,直接访问子组件,避免在模板或计算属性中使用。

7.4.1 使用slot分发内容,什么是slot?
当需要让组件组合使用,混合父组件的内容与子组件的模板时,就会用到slot,这个过程叫做内容分发。
以<app>为例,它有两个特点:
<app>组件不知道它的挂载点会有什么内容,挂载点的内容是有<app>的父组件决定的。
<app>组件很可能有它自己的模板。
备注:props传递数据、events触发事件和slot内容分发就构成了Vue组件的3个API来源,
在复杂的组件也是由这3部分构成的。

7.4.2 作用域,父组件模板的内容是在父组件作用域内编译,子组件模板的内容是在子组件作用域内编译。

7.4.3 slot用法
单个Slot,在子组件内使用特殊的<slot>元素就可以为这个子组件开启一个slot插槽,
在父组件模板里插入在子组件标签内的所有内容将替代子组件的<slot>标签及它的内容。

<div id="app">
  <child-component>
    <p>分发的内容</p>
    <p>更多分发的内容</p>
  </child-component>
</div>
<script>
  Vue.component('child-component',{
    template:'\
    <div>\
      <slot>\
        <p>如果父组件没有插入内容,我将作为默认出现</p>\
      </slot>\
    </div>'
  });
  var app = new Vue({
    el:'#app'
  });
</script>
渲染结果为:
<div id="app">
  <div>
    <p>分发的内容</p>
    <p>更多分发的内容</p>
  </div>
</div>

注意:子组件<slot>内的备用内容,它的作用域是子组件本身。

具名Slot:给<slot>元素指定一个name后可以分发多个内容,具名Slot可以与单个Slot共存。

<div id="app">
  <child-component>
    <h2 slot="header">标题</h2>
    <p>正文内容</p>
    <p>更多的正文内容</p>
    <div slot="footer">底部信息</div>
  </child-component>
</div>
<script>
  Vue.component('child-component',{
    template:'\
    <div class="container">\
      <div class="header">\
        <slot name="header"></slot>\
      </div>\
      <div class="main">\
        <slot></slot>\
      </div>\
      <div class="footer">\
        <slot name="footer"></slot>\
      </div>\
    </div>'
  });
  var app = new Vue({
    el:'#app'
  });
</script>

备注:slot没有使用name特性,所有内容将作为默认slot出现,有name特性的指定位置出现。

7.4.4 作用域插槽,是一种特殊的slot,使用一个可以复用的模板替换已渲染元素。

<div id="app">
  <child-component>
    <template scope="props">
      <p>来自父组件的内容</p>
      <p>{{ props.msg }}</p>
    </template>
  </child-component>
</div>
<script>
  Vue.component('child-component',{
    template:'\
    <div class="container">\
      <slot msg="来自子组件的内容"></slot>\
    </div>'
  });
  var app=new Vue({
    el:'#app'
  });
</script>

备注:子组件<slot>中有类似props传递数据方式,声明参数msg数据传递给插槽,
父组件中使用了<template>元素,而且拥有一个scope特性来接收子组件插槽的数据。

例:作用域插槽更具代表性的用例是列表组件,允许组件自定义应该如何渲染列表每一项。

<div id="app">
  <my-list :books="books">
    <template slot="book" scope="props">
      <li>{{ props.bookname }}</li>
    </template>
  <my-list>
</div>
<script>
  Vue.component('my-list',{
    props:{
      books:{
        type:Array,
        default: function(){
          return [];
        }
      }
    },
    template:'\
    <ul>\
      <slot name="book" v-for="book in books" :book-name="book.name"><!--这里也可以写默认slot内容--></solt>\
    </ul>'
  });
  var app = new Vue({
    el:'#app',
    data:{
      books:[
        { name: '《Vue.js实战》'},
        { name: '《JavaScript语言精粹》'},
        { name: '《HTML5/CSS3基础》'}
      ]
    }
  });
</script>

备注:作用域插槽也可以是具名的Slot。

7.4.5 访问slot,用来访问被slot分发的内容的方法$slots。

mounted: function(){
  var header = this.$slots.header;
  var main = this.$slots.default;
  var footer = this.$slots.footer;
  console.log(footer);
  console.log(footer[0].elm.innerHTML);
}

备注:通过$slots可以访问某个具名slot,this.$slots.default包括了所有没有被包含在具名slot中的节点。
在用render函数创建组件时比较有用。

7.5.1 递归组件,组件在它的模板内可以递归地调用自己,只有给组件设置name的选项就可以了,
必须给一个条件来限制递归数量,否则会抛出异常。

7.5.2 内联模板,给组件标签使用inline-template特性,组件就会把它的内容当作模板,而不是把它内容分发。
备注:在父组件和子组件中声明的数据,都可以直接渲染,如果同名优先使用子组件数据,不建议使用内联模板。

7.5.3 动态组件,Vue.js提供了一个特殊的元素<component>用来动态挂载不同的组件,使用is特性来选择要挂载的组件。

<div id="app">
  <component :is="currentView"></component>
  <button @click="handleChangeView('A')">切换到A</button>
  <button @click="handleChangeView('B')">切换到B</button>
  <button @click="handleChangeView('C')">切换到C</button>
</div>
<script>
  var app = new Vue({
    el: '#app',
    components: [
      comA: {
        template: '<div>组件A</div>'
      },
      comB: {
        template: '<div>组件B</div>'
      },
      comC: {
        template: '<div>组件C</div>'
      }
    ],
    data:{
      currentView : 'comA'
    },
    methods:{
      handleChangeView: function(component){
        this.currentView = 'com'+ component;
      }
    }
  });
</script>

动态的改变currentView的值就可以动态挂载组件了,也可以直接绑定在组件对象上。

<div id="app">
  <component :is="currentView"></component>
</div>
<script>
  var Home = { template: '<p>Welcome home!</p>' }
  var app = new Vue({
    el: '#app',
    data: {
      currentView: Home
    }
  });
</script>

7.5.4 异步组件,Vue.js允许将组件定义为一个工厂函数,动态地解析组件,
Vue.js只在组件需要渲染时触发工厂函数,并且把结果缓存起来,用于后面的再次渲染。

7.6.1 $nextTick,当v-if="true"时div不会被立即创建出来,这时操作dom会报错,
$nextTick(function(){}),就是用来知道什么时候DOM更新完成的。

7.6.2 X-Templates,Vue提供了另外一种定义模板的方式,在<script>标签中使用text/x-template类型,
并且指定一个id,将这个id赋给template。

7.6.3 手动挂载实例,通常我们都是通过new Vue()形式创建的实例,在一些非常特殊的情况下,
我们需要动态地去创建Vue实例,Vue提供了Vue.extend和$mount两个方法来手动挂载一个实例。

<div id="mount-div"></div>
<script>
  var MyComponent = Vue.extend({
    template:'<div>Hello:{{ name }}</div>',
    data: function(){
       return: 'Aresn'
    }
  });
  new MyComponent().$mount('#mount-div');
</script>

8 自定义指令,Vue有许多内置指令,比如v-if、v-show等,这些丰富的内置指令能满足我们的绝大部分业务需求,
不过在需要一些特殊功能时,我们仍然希望对DOM进行底层操作,这是就要弄到自定义指令了。

8.1 基本用法,自定义指令的注册方法和组件很像,也分全局注册和局部注册。
比如注册一个v-focus的指令,用于在<input>、<textarea>元素初始化时自动获得焦点。
//全局注册:

Vue.directive('focus',{
  //指令选项
});

//局部注册:

var app = new Vue({
  el: '#app',
  directives: {
    focus: {
      //指令选项
    }
  }
});

*bind:只调用一次,指令第一次绑定到元素时调用,用这个钩子函数可以定义一个在绑定时执行一次的初始化动作。
*inserted:被绑定元素插入父节点时调用(父节点存在即可调用,不必存在于document中)。
*update:被绑定元素所在的模板更新时调用,而不论绑定值是否变化,通过比较更新前后的绑定值,可以忽略不必要的模板更新。
*componentUpdated:被绑定元素所在模板完成一次更新周期时调用。
*unbind:只调用一次,指令与元素解绑时调用。
示例:

<div id="app">
  <input type="text" v-focus>
</div>
<script>
  Vue.directive('focus',{
    inserted: function(el){
      el.focus(); //聚焦元素
    }
  });
  var app = new Vue({
    el: '#app'
  });
</script>

=======================================================================================

JavaScript(ES6) snippets:ES6的语法支持。
JavaScript Snippet Pack:JavaScript代码片段集合。
ESLint:最流行的代码检测插件。
Beatufy:最流行的格式化工具。
Live Server:开启本地开发时服务器。
Vetur:实现支持vue文件的代码高亮。
Beautify:自动格式化插件,F1,输入bea

Beautify配置:在工作目录下建立.jsbeautifyrc文件
{
   "brace_style": "none,preserve-inline",
   "indent_size": 2,
   "indent_char": " ",
   "jslint_happy": true,
   "unformatted": [""],
   "css": {
     "indent_size": 2
   }
 }
在VSCode的配置文件里添加 "editor.formatOnSave":true 即可实现保存时自动格式化

{
    "emmet.syntaxProfiles": {
        "vue-html": "html",
        "vue": "html"
    },
    "eslint.validate": ["javascript", "javascriptreact", "html", "vue"],
    "eslint.options": {
        "plugins": ["html"]
    },
    "editor.formatOnSave": true
}
================================================================================================