Vue + ElementUI 后台管理系统项目心得(二)

此篇是关于 Vue 的总结。Js 的总结请见前一篇:Vue + ElementUI 后台管理系统项目心得(一)

话说过完年回来,仿佛一夜之间身边的人都在谈论 Vue,相关的组件也如雨后春笋一般涌现出来,比如大名鼎鼎的 Element UIiView 还有不久前刚发布的 ATUI,这些组件库虽然实现方式和接口格式有所不同,但大都界面美观,体验流畅,可以说和 Vue 框架提供的良好性能是密不可分的。

Vue 是一个上手简单,却富有内涵的框架,既可以像使用 jQuery 一样简单地通过 <script> 标签引入,也支持通过打包工具构建。

// 一个简单的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 函数。

类似的还有 data 属性,我们通常使用这样的格式来定义 data

data() {
    return {
        commonData,
        layout,
        isShow: true,
        isLoading: false
    };
}

事实上这只是一个配置选项,编译时 Vue 会根据我们传入的选项来生成真正的 data 实例,其中的每个属性都绑定了 gettersetter 方法,具备响应式特性。这就是为什么需要显式的定义 data 属性。

默认情况下 data 内的属性会被 Vue 实例代理(Vue 实例的同名属性指向 data 对象的属性),如果某些属性不想被 Vue 代理可以在属性名称前面加上 '$''_' ,这样就只能通过 Vue.$data._someProp 来访问。

在实例化一个 Vue 对象时会经历那些阶段呢?通俗的说会有两个阶段:一是生成 data,二是挂载 template,也就是 MVVM 中的 VM(view-model)。

生成 data 被称作 create,挂载 template 被称作 mount,两个阶段的前后分别对应着两组生命周期节点,也就是 beforeCreatecreated 以及 beforeMoundmounded

有生则有灭,销毁响应式属性,解除绑定事件的过程被称作 destroy,前后也分别对应着 beforeDestroydestroyed 节点。

响应式属性的变化会触发视图的更新,这个更新的过程被称作 update,前后自然也对应着 beforeUpdateupdated 两个节点。数据与视图绑定是 VM 框架提供的核心功能,也正是所谓的数据驱动,数据驱动和事件驱动的差异有点类似于查询和中断的区别。

组件是一个封闭的环境,除非通过生命周期节点来探测,不然操作内部数据的进度是无法获知的,换句话说我们从组件外部访问 data 数据时,无法确定它是更新前还是更新后的,尤其在通过异步方式请求后端数据时,这个问题就更加明显。因此官方不建议通过 $parent$children$refs 等方式强行获取数据,而是建议在数据状态可控时通过事件触发来回传数据($emit、$on)。

常用的模板语法

  • 大胡子语法(双大括号)可以方便的在 HTML 元素内部(innerHTML)插入文本。
<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-ifv-elsev-else-ifv-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>

(未完待续...)

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 203,324评论 5 476
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 85,303评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 150,192评论 0 337
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,555评论 1 273
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,569评论 5 365
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,566评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,927评论 3 395
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,583评论 0 257
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,827评论 1 297
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,590评论 2 320
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,669评论 1 329
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,365评论 4 318
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,941评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,928评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,159评论 1 259
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 42,880评论 2 349
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,399评论 2 342

推荐阅读更多精彩内容

  • 这篇笔记主要包含 Vue 2 不同于 Vue 1 或者特有的内容,还有我对于 Vue 1.0 印象不深的内容。关于...
    云之外阅读 5,044评论 0 29
  • 1.安装 可以简单地在页面引入Vue.js作为独立版本,Vue即被注册为全局变量,可以在页面使用了。 如果希望搭建...
    Awey阅读 10,982评论 4 129
  • Vue 实例 属性和方法 每个 Vue 实例都会代理其 data 对象里所有的属性:var data = { a:...
    云之外阅读 2,198评论 0 6
  • 转载 :OpenDiggawesome-github-vue 是由OpenDigg整理并维护的Vue相关开源项目库...
    果汁密码阅读 23,083评论 8 124
  • 最近不知道什么情况,眼看公公的病情好转,今天又接到爸爸的电话说妈妈被闯红灯的电瓶车撞伤,小腿骨折!我的天,听着电话...
    思齐_yang阅读 193评论 2 0