使用avalon MVVM框架編寫日歷控件


本教程將介紹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();
            });

                
日歷
< >
{{title}}
{{date}}
{{ day[2] }}
可選擇月份
可選擇年份

{{selectedDate | date("yyyy-MM-d")}}

這就完成了,如果要用jQuery實現相似的功能,起碼要千行。主要原因是MVVM的核心是數據。數據是底層、是心臟,數據變了,作為表層的UI就會跟着變,將數據展現給用戶;如果用戶修改了UI元素上的值,相當於透過UI元素直接修改了底層的數據;數據處於核心地位,UI處於從屬地位。DOM操作全部隱藏在UI的綁定中(ms-html,就是調用innerHTML操作,ms-on-click進行事件綁定, ms-class-*,就是調用增刪類名,ms-visible控制了元素的顯隱,ms-if,ms-each進行流程控制處理某個區域的所有元素的生成移除……),由框架自動維護,我們只需要關注業務邏輯!


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM