本教程將介紹avalon的高級特性
- date過濾器的使用, 這個與angular的date過濾器的使用方法差不多,{{xxx|date("yyyy MM d")}}
- $watch監聽,相當於把監控屬性變成一種事件,當它變化時,就執行它綁定的回調。vm.$watch("firstName", function(){alert("我是回調")})
- 不監聽屬性或方法的定義,就是$開頭,或放在$skipArray數組中。
- if綁定,類似於knockout的if綁定,根據求值表達式的情況決定把當前元素插入DOM樹。ms-if=expr
HTML結構
<div ms-controller="datepicker">
<div class="ui-datepicker">
<div class="ui-datepicker-header">
<a href="###" title="Prev" class="ui-datepicker-month ui-datepicker-prev" ms-click="prevMonth">
<
</a>
<a href="###"title="Next" class="ui-datepicker-month ui-datepicker-next" ms-click="nextMonth">
>
</a >
<div class="ui-datepicker-title" >
<select ms-each-month="$months" ms-if="changeMonth" ms-model="currentMonth" >
<option value="{{month}}" ms-selected="currentMonth === month">{{month+1}}月</option>
</select>
<select ms-each-year="candidateYears" ms-if="changeYear" ms-model="currentYear" >
<option value="{{year}}" ms-selected="currentYear === year">{{year}}年</option>
</select>
{{title}}
</div>
</div>
<table ms-click="select">
<thead>
<tr ms-each-date="$weeks">
<th>{{date}}</th>
</tr>
</thead>
<tbody ms-each-week="candidateWeeks">
<tr ms-each-day="week" >
<td><span ms-visible="day" ms-class-selected="isSelected" ms-class-disabled="day && day[3]" ms-class-today="isToday" >{{ day[2] }}</span></td>
</tr>
</tbody>
</table>
</div>
<p><input type="radio" ms-model="changeMonth" />可選擇月份</br/>
<input type="radio" ms-model="changeYear" />可選擇年份</p>
<div class="font">{{selectedDate | date("yyyy-MM-d")}}</div>
</div>
JS
avalon.ready(function() {
var model = avalon.define("datepicker", function(vm) {
//配置
vm.changeYear = false
vm.changeMonth = false
vm.minDate = new Date(2013, 4, 25)
//當前時間
vm.current = new Date;
vm.currentMonth = vm.current.getMonth();
vm.currentYear = vm.current.getFullYear();
vm.selectedDate = new Date;
//顯示頂部的年份與月份
vm.title = {
get: function() {
var format = "";
if (!this.changeYear && this.changeMonth) {
format = "yyyy年";
} else if (this.changeYear && !this.changeMonth) {
format = "MMMM";
} else if (!this.changeYear && !this.changeMonth) {
format = "MMMM yyyy年";
}
return format && avalon.filters.date(this.current, format);
}
};
//星期顯示
vm.$weeks = "日一二三四五六".split("");
//月份下拉菜單
vm.$months = [0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11]
//當月的日期
function getWeeks() {
var cur = vm.current;
var year = cur.getFullYear();
var month = cur.getMonth();
var date = cur.getDate()
cur.setMonth(month + 1);
cur.setDate(0);
var num = cur.getDate();
var dates = avalon.range(1, num + 1);
dates = dates.map(function(d) {
var timestamp = new Date(year, month + 1, d) - 0
var disabled = false;
if (vm.minDate) {
disabled = timestamp < vm.minDate - 0;
}
if (disabled && vm.maxDate) {
disabled = timestamp > vm.maxDate - 0;
}
return [ year, month, d, disabled];
});
cur.setMonth(month);
cur.setDate(1);
var before = cur.getDay();
var shim = new Array(before);//如果當月的第一天不是星期天,需要補空格
dates = shim.concat(dates);
cur.setDate(date);//還原
var ret = [];
while (dates.length) {//每行七個分組
ret.push(dates.splice(0, 7));
}
return ret;
}
vm.candidateWeeks = getWeeks();
//取得當年的前后20年
function getYears() {
var y = vm.currentYear;
return avalon.range(y - 10, y + 10);
}
vm.candidateYears = getYears();
//點擊事件
vm.nextMonth = function() {
var date = vm.current;
var m = date.getMonth();
vm.$fire("currentMonth", m + 1)
};
//點擊事件
vm.prevMonth = function() {
var date = vm.current;
var m = date.getMonth();
vm.$fire("currentMonth", m - 1);
};
//偵聽
vm.$watch("currentMonth", function(month) {
var date = vm.current;
date.setMonth(month);
vm.currentMonth = month;
vm.candidateWeeks = getWeeks();
vm.title = date - 0;
});
vm.$watch("currentYear", function(year) {
var date = vm.current;
date.setFullYear(year);
vm.currentYear = year;
vm.$fire("currentMonth", date.getMonth());
});
//高亮當前選中的日期
vm.select = function(e) {
var el = e.target;
if (el.tagName === "SPAN" && el.parentNode.tagName === "TD" && !/disabled/.test(el.className)) {
vm.selected = el.innerHTML;
var d = el.$scope.day.slice(0, 3);
vm.selectedDate = new Date(d[0],d[1],d[2]);
}
};
vm.selected = "";
vm.isSelected = function() {
return this.innerHTML === vm.selected;
};
//高亮今天的日期
var today = new Date;
today = [ today.getFullYear(), today.getMonth(), today.getDate() ].join();
vm.isToday = function() {
var day = this.$scope && this.$scope.day;
return day && day.slice(0, 3).join() === today;
};
});
avalon.scan();
});
這就完成了,如果要用jQuery實現相似的功能,起碼要千行。主要原因是MVVM的核心是數據。數據是底層、是心臟,數據變了,作為表層的UI就會跟着變,將數據展現給用戶;如果用戶修改了UI元素上的值,相當於透過UI元素直接修改了底層的數據;數據處於核心地位,UI處於從屬地位。DOM操作全部隱藏在UI的綁定中(ms-html,就是調用innerHTML操作,ms-on-click進行事件綁定, ms-class-*,就是調用增刪類名,ms-visible控制了元素的顯隱,ms-if,ms-each進行流程控制處理某個區域的所有元素的生成移除……),由框架自動維護,我們只需要關注業務邏輯!
