此篇是关于 Vue 的总结。Js 的总结请见前一篇:Vue + ElementUI 后台管理系统项目心得(一)javascript
话说过完年回来,仿佛一晚上之间身边的人都在谈论 Vue,相关的组件也如雨后春笋通常涌现出来,好比大名鼎鼎的 Element UI
、iView
还有不久前刚发布的 ATUI
,这些组件库虽然实现方式和接口格式有所不一样,但大都界面美观,体验流畅,能够说和 Vue 框架提供的良好性能是密不可分的。css
Vue 是一个上手简单,却富有内涵的框架,既能够像使用 jQuery
同样简单地经过 <script>
标签引入,也支持经过打包工具构建。html
// 一个简单的title.vue模板文件,同时具有了Web三要素:结构层、表现层和行为层。
<template>
<div class="box">
<span class="title-text">title</span>
</div>
</template>
<script> export default { name: 'TitleBox', }; </script>
<style> .title-text{ font-size:18px } </style>复制代码
扩展名为 vue
的文件会被打包工具编译成标准 js
文件以供浏览器执行,在编译时 <template>
模板的内容会被转换为 render
函数,换句话说 <template>
的内容只是一个配置参数,Vue 经过这种相似于 HTML 的结构来生成渲染方法,最后经过执行渲染方法来初始化一个真正的页面。固然对于复杂的需求咱们也能够不用模板,直接编写 render
函数。vue
相似的还有 data
属性,咱们一般使用这样的格式来定义 data
。java
data() {
return {
commonData,
layout,
isShow: true,
isLoading: false
};
}复制代码
事实上这只是一个配置选项,编译时 Vue 会根据咱们传入的选项来生成真正的 data
实例,其中的每一个属性都绑定了 getter
、setter
方法,具有响应式特性。这就是为何须要显式的定义 data
属性。后端
默认状况下 data
内的属性会被 Vue 实例代理(Vue 实例的同名属性指向 data
对象的属性),若是某些属性不想被 Vue 代理能够在属性名称前面加上 '$'
或 '_'
,这样就只能经过 Vue.$data._someProp
来访问。数组
在实例化一个 Vue 对象时会经历那些阶段呢?通俗的说会有两个阶段:一是生成 data
,二是挂载 template
,也就是 MVVM 中的 VM(view-model)。浏览器
生成 data
被称做 create
,挂载 template
被称做 mount
,两个阶段的先后分别对应着两组生命周期节点,也就是 beforeCreate
、created
以及 beforeMound
、mounded
。框架
有生则有灭,销毁响应式属性,解除绑定事件的过程被称做 destroy
,先后也分别对应着 beforeDestroy
和 destroyed
节点。异步
响应式属性的变化会触发视图的更新,这个更新的过程被称做 update
,先后天然也对应着 beforeUpdate
、updated
两个节点。数据与视图绑定是 VM 框架提供的核心功能,也正是所谓的数据驱动,数据驱动和事件驱动的差别有点相似于查询和中断的区别。
组件是一个封闭的环境,除非经过生命周期节点来探测,否则操做内部数据的进度是没法获知的,换句话说咱们从组件外部访问 data
数据时,没法肯定它是更新前仍是更新后的,尤为在经过异步方式请求后端数据时,这个问题就更加明显。所以官方不建议经过 $parent
、$children
或 $refs
等方式强行获取数据,而是建议在数据状态可控时经过事件触发来回传数据($emit、$on
)。
<span> {{ status ? '有效' : '无效' }} </span>复制代码
<span prop="text"></span>
的格式书写,若是须要为属性绑定变量或表达式则需使用 v-bind
指令:<span :prop="value"></span>
,没错,冒号 ':'
正是 v-bind
指令的缩写,等同于 <span v-bind:prop="value"></span>
,有时咱们须要绑定数值(Number
),而非内容为数字的字符串,v-bind
指令刚好能够区分二者。// 后端发送的数据
requiredData: {cityId: 3}
// 模板绑定的数据
<ex-select v-model="cityId">
<ex-option :value="1">北京</option>
<ex-option :value="2">上海</option>
<ex-option :value="3">广州</option>
// 若是不加 ':' 会由于 3 !== '3'而找不到匹配项
</ex-select>复制代码
v-on
,能够用 '@'
表示,好比 <button @click="handleClick"></button>
,指令较多时,缩写可使代码更清晰。<ex-input v-model="userName" :placeholder="姓名" :disabled="isDisabled" :readonly="isReadonly" @focus="handleFocus" @input="handleInput" @change="handleChange">
</ex-input>复制代码
v-if
、v-else
、v-else-if
和 v-for
。// 判断单行或多行文本框
<input type="text" v-if="inputType==='singleLine'"></input>
<textarea v-else="inputType==='multiLine'"></textarea>
// v-for 除了能够生成列表,还能够用于经过 JSON 动态生成模板
let formConfig = [{
type: 'singleLine',
name: 'userName'
},{
type: 'singleLine',
name: 'birthday'
},{
type: 'multiLine',
name: 'comment'
}]
<label v-for="item in formConfig">
{{ item.name }}
<ex-input inputType="item.type"></ex-input> </label>复制代码
data() {
return {
updateTime: 1490526754
}
}
computed: {
timeString: {
get () {
// Js默认以毫秒为单位,须要进行转换
let timeStamp = new Date(this.updateTime * 1000)
return timeStamp.toLocaleString()
},
set (val) {
let timeStamp = Math.round(Date.getTime() / 1000)
this.updateTime = val || timeStamp
}
}
}复制代码
data
中的数组?JSON
数组:<ex-table
data="dataSource"
></ex-table> ... data () { return { dataSource: [{ id: 1, name: lily, age: 21 },{ id: 2, name: terry, age: 23 },{ id: 3, name: dill, age: 26 }] } }复制代码
咱们修改了其中的某一项,并经过下面的方式更新到数组中:
let modifiedRow = {id: 2, name: frank, age: 20}
// 视图未发生更新
this.dataSource[1] = modifiedRow复制代码
咱们会发现,表格中显示的数据并无改变,这是由于直接对数组中的元素赋值不会触发Vue的响应式更新,为了解决 Js 语言特性的限制,Vue 提供了一种变通的方法,可使用 push()
、pop()
、shift()
等函数来修改数组。
// 视图响应式更新
dataSource.splice(1, 1, modifiedRow)复制代码
除此以外咱们也能够经过 Vue.set()
函数修改数组,好比:
Vue.set(dataSource, 1, modifiedRow)复制代码
v-model
指令为表单元素绑定数据表单元素不但能够显示数据也能够修改数据,咱们固然能够经过 :value="data"
来显示数据,而且经过 @input="data=$event.target.value"
来修改数据,不过 Vue 为咱们提供了一个更简洁的语法糖,直接使用 v-model="data"
便可同时实现二者的功能。
其实,对于任何自定义组件只要咱们实现了 value
属性和 input
事件均可以使用 v-model
指令进行数据绑定。若是实现了 change
事件也能够配合 v-model.lazy
来调用。
// ex-input.vue
<input
type="text"
v-if="inputType==='singleLine'"
:value="editValue"
@input="handelInput"
></input>
<textarea
v-else="inputType==='multiLine'"
:value="editValue"
@input="handelInput"
></textarea>
...
data() {
return {
// 获取表单初始值
editValue: this.value
}
},
prop: {
// 定义value属性
value:[String, Number]
},
watch: {
// 观察并同步value属性
value(val) {
this.editValue = val
}
},
methods: {
handelInput (event) {
this.editValue = event.target.value
// 实现input事件
this.$emit('input', this.editValue)
}
}
...
// 二者是等价的,显然 v-model 更简洁一些
<ex-input v-model="data"></ex-input>
<ex-input :value="data" @input="data=arguments[0]"></ex-input>复制代码
(未完待续...)