前段時間接觸到一個庫叫作Vue.js, 我的感受很棒,因此整理了一篇博文作個介紹。
Vue讀音/vju:/,和view相似。是一個數據驅動的web界面庫。Vue.js只聚焦於視圖層,能夠很容易的和其餘庫整合。代碼壓縮後只有24kb。
能夠去這裏下載。本身整理了一個Vue.js的demo,https://github.com/chenhao-ch/demo-vuejavascript
如下代碼是Vue.js最簡單的例子, 當 input
中的內容變化時,p
節點的內容會跟着變化。html
<!-- html --> <div id="demo"> <p>{{message}}</p> <input v-model="message"> </div>
new Vue({ el: '#demo', data: { message: 'Hello Vue.js!' } })
數據綁定就是指將js中的變量自動更新到html中。以下代碼, message的默認值是「Hello Vue.js!」, 那麼當頁面啓動時,html中的默認值會被設置成「Hello Vue.js」前端
<!-- html --> <div id="demo"> <p>{{message}}</p> <input v-model="message"> </div>
new Vue({ el: '#demo', data: { message: 'Hello Vue.js!' } })
若是要輸出原生的html,能夠使用三個大括號來實現vue
<p>{{{messageHtml}}}</p>
也能夠作表達式的綁定java
<div>{{length - 1}}</div> <div>{{isShow ? 'block' : 'none'}}</div>
表達式後面能夠添加過濾器,對輸出的數據進行過濾。git
<div>{{ message | capitalize }}</div>
Vue.js運行本身定義過濾器。好比:github
Vue.filter('wrap', function (value, begin, end) { return begin + value + end; })
<!-- 'vue' => 'before vue after' --> <span>{{ message | wrap 'before' 'after' }}</span>
指令是特殊的帶有前綴 v-
的特性。當表達式的值發生變化時,響應應用特定的行爲到DOM。web
<!-- 當greeting爲true時,才顯示p節點 --> <p v-if="greeting">hello</p> <!-- 綁定href屬性爲js中url的值 --> <a v-bind:href="url"></a> <!-- 綁定事件,btnClick是js方法 --> <button v-on:click="btnClick"></button>
bind
, on
指令能夠進行縮寫json
<a v-bind:href="url"></a> <a :href="url"></a> <button v-on:click="btnClick"></button> <button @click="btnClick"></button>
Vue.directive('demo', { bind: function () { // 準備工做 // 例如,添加事件處理器或只須要運行一次的高耗任務 }, update: function (newValue, oldValue) { // 值更新時的工做 // 也會以初始值爲參數調用一次 }, unbind: function () { // 清理工做 // 例如,刪除 bind() 添加的事件監聽器 } })
Vue.js支持對js對象進行判斷(if), 循環(for)輸出。相似於前端模板。api
<!-- 判斷,若是ok爲true,則顯示yes, 若是爲false, 顯示no --> <h1 v-if="ok">Yes</h1> <h1 v-else>No</h1> <!-- 相似於v-if, v-if是是否加節點, v-show是display是否爲none --> <h1 v-show="ok">Hello!</h1> <!-- 循環, 對items屬性進行循環。 track-by指item的是否重複,對重複的進行服用 --> <!-- 循環中, $index表示數組第n個元素; $key表示對象的key --> <ul id="example-1"> <li v-for="item in items" track-by="_uid"> {{ $index }} : {{ item.message }} </li> </ul>
var example1 = new Vue({ el: '#example-1', data: { items: [ {_uid: '1', message: 'Foo' }, {_uid: '2', message: 'Bar' } ] } })
樣式也能夠根據js中的變量來動態肯定。
<!-- isA 爲true時, class多一個class-a --> <div class="static" v-bind:class="{ 'class-a': isA, 'class-b': isB }"></div> <!-- classA, classB 兩個變量的值設置成class --> <div v-bind:class="[classA, classB]"> <!-- 綁定style, 自動添加前綴,styleObject裏面是style鍵值對 --> <div v-bind:style="styleObject"></div>
能夠使用 v-on
指令來監聽DOM事件。
<div id="example-2"> <button v-on:click="say('hi', $event)">Say Hi</button> <button v-on:click="say('what', $event)">Say What</button> </div>
new Vue({ el: '#example-2', methods: { say: function (msg, event) { alert(msg); event.preventDefault(); } } })
常見的阻止冒泡,禁止默認行爲等event方法能夠經過修飾符來快速處理。
<!-- 禁止冒泡 --> <a v-on:click.stop='do'></a> <!-- 禁止冒泡和默認行爲 --> <a @click.stop.prevent="do"></a>
對特殊按鍵生效也能夠使用修飾符
<!-- keyCode是13的時候出發。 --> <input v-on:keyup.13="submit" /> <input v-on:keyup.enter="submit" /> <!-- 支持的鍵名有: enter,tab,delete,esc,space,up,down,left,right -->
組件系統是 Vue.js 另外一個重要概念,由於它提供了一種抽象,讓咱們能夠用獨立可複用的小組件來構建大型應用。
經過Vue.extend()
來定義一個組件,Vue.component()
來註冊組件。
<div id="box"> <tree></tree> </div>
// 定義 var Tree = Vue.extend({ template: '<div>This is a tree!</div>' }); // 註冊 Vue.component('tree', Tree); // 開始渲染 new Vue({ el: '#box' }); // 定義,註冊能夠合併成一步。下面的代碼和上面同樣 Vue.component('tree', { template: '<div>This is a tree!</div>' }); new Vue({ el: '#box' });
渲染結果爲:
<div id="box"> <div>This is a tree!</div> </div>
還能夠進行局部註冊
var Child = Vue.extend({ /* ... */ }) var Parent = Vue.extend({ template: '...', components: { 'my-component': Child } })
props用於父組件向子組件傳遞數據。
Vue.component('child', { props: ['childMsg'], // prop 能夠用在模板內 // 能夠用 `this.msg` 設置 template: '<span>{{ childMsg }}</span>' });
<child child-msg="hello"></child>
動態props, 當父組件的數據變化時,須要通知子組件跟着變化。
<input v-model="parentMsg" /> <child v-bind:child-msg="parentMsg"></child>
當父組件數據變化時,能夠經過props來通知子組件,子組件狀態變化時,能夠利用事件的冒泡來通知父組件。
子組件能夠用 this.$parent
訪問它的父組件。父組件有一個數組 this.$children
,包含它全部的子元素。
例子:
<!-- 子組件模板 --> <template id="child-template"> <input v-model="msg"> <button v-on:click="notify">Dispatch Event</button> </template> <!-- 父組件模板 --> <div id="events-example"> <p>Messages: {{ messages | json }}</p> <child></child> </div>
// 註冊子組件 // 將當前消息派發出去 Vue.component('child', { template: '#child-template', data: function () { return { msg: 'hello' } }, methods: { notify: function () { if (this.msg.trim()) { this.$dispatch('child-msg', this.msg) // 觸發child-msg事件 this.msg = '' } } } }) // 啓動父組件 // 將收到消息時將事件推入一個數組 var parent = new Vue({ el: '#events-example', data: { messages: [] }, // 在建立實例時 `events` 選項簡單地調用 `$on` events: { 'child-msg': function (msg) { // 監聽到 child-msg事件 // 事件回調內的 `this` 自動綁定到註冊它的實例上 this.messages.push(msg) // messages改變自動修改html內容 } } })
上面這種寫法child-msg事件觸發後,會冒泡到父組件,並執行父組件的child-msg
events。
可是上面的的執行流程不夠直觀,能夠在html中經過指定on事件來實現event的監聽。下面是所有代碼:
<template id="child-template"> <input v-model="msg"> <button v-on:click="notify">Dispatch Event</button> </template> <!-- 父組件模板 --> <div id="events-example"> <p>Messages: {{ messages | json }}</p> <!-- 當child-msg觸發時, 執行父組件的handleIt方法。 --> <child v-on:child-msg="handleIt"></child> </div> <script> // 註冊子組件 // 將當前消息派發出去 Vue.component('child', { template: '#child-template', data: function () { return { msg: 'hello' } }, methods: { notify: function () { if (this.msg.trim()) { this.$dispatch('child-msg', this.msg) // 觸發child-msg事件 this.msg = '' } } } }); // 啓動父組件 // 將收到消息時將事件推入一個數組 var parent = new Vue({ el: '#events-example', data: { messages: [] }, methods: { handleIt: function(msg) { this.messages.push(msg); } } }) </script>
在典型的 Vue.js 項目中,咱們會把界面拆分爲多個小組件,每一個組件在同一地方封裝它的 CSS 樣式,模板和 JavaScript 定義,這麼作比較好。如上所述,使用 Webpack 或 Browserify 以及合適的源碼轉換器,咱們能夠這樣寫組件:
固然也能夠使用預處理器,