小程序日,周,月,季度,自定义tab选择组件

效果图:
2.gif
应用场景

考虑到某些报表需要根据某天,某周,某月,某季度,或者自定义查询展示;然后每种方式的日期的选择方式各种各样,这里根据我们的需求写了一种针对这些的选择器;

编写日,周,月,季度,自定义tab组件

编写tab组件,组件名tab-cycle
tab-cycle.js关键代码,提供tab_datas和cur_tab_id两个属性,tab_datas属性只要是提供给外部重新自定义tab内容,提供了一个默认值defaut_tab_datas,外面不重新赋值就是用默认的,cur_tab_id属性表示当前选择的是哪个tab;监听每个tab的点击事件,然后执行更新tab_datas的值和cur_tab_id,
tab-cycle.js

"use strict";

Component({
   properties: {
       tab_datas: {
           type: Array,
           value:[
               { id: "day", title: "日", isSelect: true },
               { id: "weeks", title: "周", isSelect: false },
               { id: "month", title: "月", isSelect: false },
               { id: "quarter", title: "季度", isSelect: false },
               { id: "custom", title: "自定义", isSelect: false }
           ],
           observer: "onItemsChange"
       },
       cur_tab_id: {
           type: String,
           value: "",
           observer: 'onSelectIndexChange'
       }
   },


   data: {
       defaut_tab_datas: [
           { id: "day", title: "日", isSelect: true },
           { id: "weeks", title: "周", isSelect: false },
           { id: "month", title: "月", isSelect: false },
           { id: "quarter", title: "季度", isSelect: false },
           { id: "custom", title: "自定义", isSelect: false }
       ]
   },


   methods: {
       onTabItemClick: function onTabItemClick(e) {
           var id = e.currentTarget.dataset.tabs.id;
           this.updateData(id);
           this.triggerEvent('tabclick', { id: e.currentTarget.dataset.tabs.id });
       },
       onSelectIndexChange: function onSelectIndexChange() {
           var id = this.data.cur_tab_id;
           this.updateData(id);
       },
       updateData: function updateData(id) {
           for (var i = 0; i < this.data.tab_datas.length; i++) {
               if (id == this.data.tab_datas[i].id) {
                   this.data.tab_datas[i].isSelect = true;
                   this.data.cur_tab_id = this.data.tab_datas[i].id;
               } else {
                   this.data.tab_datas[i].isSelect = false;
               }
           }
           this.onItemsChange();
       },
       onItemsChange: function onItemsChange() {
           this.setData({
               tab_datas: this.data.tab_datas == null || this.data.tab_datas.length == 0 ? this.data.defaut_tab_datas : this.data.tab_datas
           });
       }
   }
});
编写日,周,月,季度,自定义选择器组件

编写选择器组件,组件名calendar;该组件给外部提供了dateType属性,该属性存在五个值分别为日:day,周:weeks,月:month,季度:quarter,自定义:custom,通过传不一样的值,将会展示其对应的选择器,比如,传day,展示出来的是可以通过点击上一天,下一天来选择日期;传weeks展示出来的是可以通过点击上一周,下一周来选择日期,;传month展示出来的是可以通过点击上一月,下一月来选择日期;传quarter展示出来的是可以通过点击上一季度,下一季度来选择日期;传custom展示出来的是可以通过点击选择开始时间,选择结束时间,然后点击查询来选择时间;然后每次点击或者查询都会回调onCallbackDate事件,会附带以下参数

 callbackDate: {
            curDate: "2018-11-11",
            startDate: "请选择开始时间",
            endDate: "选择结束时间"
 },

calendar.js

"use strict";

Component({
    properties: {
        dateType: {
            type: String,
            value: "",//日:day,周:weeks,月:month,季度:quarter,自定义:custom
            observer: 'onDateTypeChange'
        }
    },

    data: {
        preEnable: true,
        nextEable: false,
        queryEable: false,
        callbackDate: {
            curDate: "2018-11-11",
            startDate: "请选择开始时间",
            endDate: "选择结束时间"
        },
        content: "2018-11-11",
        preText: "上一日",
        nextText: "下一日",
        date: null,
        startDatePickerValue: ['', '', ''],
        startDatePickerIsShow: false,
        endDatePickerValue: ['', '', ''],
        endDatePickerIsShow: false,

    },

    created: function created() {
        this.data.date = new Date();
    },



    methods: {
        /**
        * 选择开始日期
        * @param {*} e 
        */
        onClickSelectStart: function onClickSelectStart(e) {
            this.setData({
                startDatePickerIsShow: true,
            });
        },
        /**
        * 选择结束日期
        * @param {*} e 
        */
        onClickSelectEnd: function onClickSelectEnd(e) {
            this.setData({
                endDatePickerIsShow: true,
            });
        },
        /**
         * 确认选择时间
         * @param {*} e 
         */
        onClickSureDatePicker: function onClickCancelDatePicker(e) {
            console.log('datePickerOnSureClick');
            console.log(e);
            let updateData = {};
            let callbackDate = this.data.callbackDate;
            if (e.currentTarget.id == 'start_date_picker') {
                updateData.startDatePickerValue = e.detail.value;
                updateData.startDatePickerIsShow = false;
                callbackDate.startDate = e.detail.value[0] + "-" + e.detail.value[1] + "-" + e.detail.value[2];
            } else if (e.currentTarget.id == 'end_date_picker') {
                updateData.endDatePickerValue = e.detail.value;
                updateData.endDatePickerIsShow = false;
                callbackDate.endDate = e.detail.value[0] + "-" + e.detail.value[1] + "-" + e.detail.value[2];
            }
            if (callbackDate.startDate != '请选择开始时间' && callbackDate.endDate != '选择结束时间') {
                updateData.queryEable = true;
            } else {
                updateData.queryEable = false;
            }
            updateData.callbackDate = callbackDate;
            this.setData(updateData);
        },
        /**
        * 取消选择时间
        * @param {*} e 
        */
        onClickCancelDatePicker: function onClickCancelDatePicker(event) {
            console.log('datePickerOnCancelClick');
            console.log(event);
            this.setData({
                startDatePickerIsShow: false,
                endDatePickerIsShow: false
            });
        },
        /**
         * 查询
         * @param {*} e 
         */
        onClickQuery: function onClickQuery(e) {
            if (!this.data.queryEable) {
                return;
            }
            if (!(this.data.endDatePickerValue[0] >= this.data.startDatePickerValue[0] && this.data.endDatePickerValue[1] >= this.data.startDatePickerValue[1] && this.data.endDatePickerValue[2] >= this.data.startDatePickerValue[2])) {
                wx.showToast({
                    title: '结束时间不能大于开始时间,请重新选择时间!',
                    duration: 2000,
                    icon:'none',
                    mask:true
                })
            }

            this.triggerEvent('onCallbackDate', { date: this.data.callbackDate });
        },
        /**
        * 上个日期
        * @param {*} e 
        */
        onClickPre: function onClickPre(e) {
            if (!this.data.preEnable) {
                return;
            }
            console.log(e);
            let updateData = this.getUpdateData(this.data.date, this.data.dateType, 'pre');
            this.setData(updateData);
            this.triggerEvent('onCallbackDate', { date: this.data.callbackDate });
        },
        /**
         * 下个日期
         * @param {*} e 
         */
        onClickNext: function onClickNext(e) {
            if (!this.data.nextEable) {
                return;
            }
            console.log(e);
            let updateData = this.getUpdateData(this.data.date, this.data.dateType, 'next');
            this.setData(updateData);
            this.triggerEvent('onCallbackDate', { date: this.data.callbackDate });
        },

        /**
         * 获取更新数据
         * @param {*} date 
         * @param {*} dateType 
         */
        getUpdateData: function getUpdateData(date, dateType, direction) {
            let updateData = {};
            let content = "";
            let year = date.getFullYear();
            let month = date.getMonth() + 1;
            let day = date.getDate();
            let weeksDay = date.getDay();
            let nextEable = true;
            let callbackDate = this.data.callbackDate;
            switch (dateType) {
                case 'day': {
                    if (direction == 'pre') {
                        date.setDate(date.getDate() - 1);
                    } else if (direction == 'next') {
                        date.setDate(date.getDate() + 1);
                    }
                    if (this.checkNextEableStatus(new Date(), date)) {
                        nextEable = false;
                    }
                    year = date.getFullYear();
                    month = date.getMonth() + 1;
                    day = date.getDate();
                    content = year + "-" + month + "-" + day;
                    callbackDate.curDate = content;
                    callbackDate.startDate = content;
                    callbackDate.endDate = content;
                    break;
                }
                case 'weeks': {
                    //一周有七天,所有点上一周就减去七天,下一周就加上七天
                    if (direction == 'pre') {
                        date.setDate(date.getDate() - 7);
                    } else if (direction == 'next') {
                        date.setDate(date.getDate() + 7);
                    }
                    if (this.checkNextEableStatus(new Date(), date)) {
                        nextEable = false;
                    }

                    //获取当天在这一周是属于第几天,然后计算这一周的起始天日期和结束的日期
                    weeksDay = date.getDay();

                    let starDate = new Date(date.getTime());
                    starDate.setDate(date.getDate() - weeksDay);
                    let startYear = starDate.getFullYear();
                    let startMonth = starDate.getMonth() + 1;
                    let startDay = starDate.getDate();

                    let endDate = new Date(date.getTime());
                    if (nextEable) {
                        endDate.setDate(date.getDate() + (6 - weeksDay));
                    }
                    let endYear = endDate.getFullYear();
                    let endMonth = endDate.getMonth() + 1;
                    let endDay = endDate.getDate();

                    content = startYear + "-" + startMonth + "-" + startDay + ' 至 ' + endYear + "-" + endMonth + "-" + endDay;
                    callbackDate.curDate = content;
                    callbackDate.startDate = startYear + "-" + startMonth + "-" + startDay;
                    callbackDate.endDate = endYear + "-" + endMonth + "-" + endDay;
                    break;
                }
                case 'month': {
                    if (direction == 'pre') {
                        date.setMonth(date.getMonth() - 1);
                    } else if (direction == 'next') {
                        date.setMonth(date.getMonth() + 1);
                    }
                    if (this.checkNextEableStatus(new Date(), date)) {
                        nextEable = false;
                    }

                    let starDate = new Date(date.getTime());
                    //计算当月的第一天日期
                    starDate.setDate(1);
                    let startYear = starDate.getFullYear();
                    let startMonth = starDate.getMonth() + 1;
                    let startDay = starDate.getDate();

                    let endDate = new Date(date.getTime());

                    if (nextEable) {
                        //计算当月的最后一天的日期,通过设置日期加一个月,然后减去一天可以得到当月的最后一天的日期
                        endDate.setMonth(date.getMonth() + 1);
                        endDate.setDate(0);
                    }
                    let endYear = endDate.getFullYear();
                    let endMonth = endDate.getMonth() + 1;
                    let endDay = endDate.getDate();

                    content = startYear + "-" + startMonth + "-" + startDay + ' 至 ' + endYear + "-" + endMonth + "-" + endDay;
                    callbackDate.curDate = content;
                    callbackDate.startDate = startYear + "-" + startMonth + "-" + startDay;
                    callbackDate.endDate = endYear + "-" + endMonth + "-" + endDay;
                    break;
                }
                case 'quarter': {
                    //一季度有三个月,所有点上一季度就减去三个月,下一季度就加上3个月
                    if (direction == 'pre') {
                        date.setMonth(date.getMonth() - 3);
                    } else if (direction == 'next') {
                        date.setMonth(date.getMonth() + 3);
                    }
                    if (this.checkNextEableStatus(new Date(), date)) {
                        nextEable = false;
                    }

                    //获取当前月份除以3,根据向下取整和求余来计算当前季度和季度的起始日期
                    let starDate = new Date(date.getTime());
                    let startMonth = starDate.getMonth() + 1;
                    let startqQuarter = Math.floor(startMonth / 3);
                    let startQuarterMonth = startMonth % 3;
                    //当当前月份是该季度的最后一个月,求开始月份应该减去2个月,当当前月份不是该季度的最后一个月,求开始月份应该减去求余数然后加一
                    if (startQuarterMonth == 0) {
                        starDate.setMonth(starDate.getMonth() - 2);
                    } else {
                        startqQuarter++;
                        starDate.setMonth(starDate.getMonth() + 1 - startQuarterMonth);
                    }

                    starDate.setDate(1);
                    let startYear = starDate.getFullYear();
                    startMonth = starDate.getMonth() + 1;
                    let startDay = starDate.getDate();

                    let endDate = new Date(date.getTime());
                    let endMonth = endDate.getMonth() + 1;
                    let endQuarterMonth = endMonth % 3;
                    if (nextEable) {
                        //当当前月份是该季度的最后一个月,求结束月份应该加上一个月然后设置日期减去0,当当前月份不是该季度的最后一个月,求结束月份应该减去求余数然后加4再然后设置日期减去0
                        if (endQuarterMonth == 0) {
                            endDate.setMonth(endDate.getMonth() + 1);
                        } else {
                            endDate.setMonth(endDate.getMonth() + 4 - endQuarterMonth);
                        }
                        endDate.setDate(0);
                    }
                    let endYear = endDate.getFullYear();
                    endMonth = endDate.getMonth() + 1;
                    let endDay = endDate.getDate();

                    content = startYear + "年第" + startqQuarter + "季度";
                    callbackDate.curDate = content;
                    callbackDate.startDate = startYear + "-" + startMonth + "-" + startDay;
                    callbackDate.endDate = endYear + "-" + endMonth + "-" + endDay;
                    break;
                }
                case 'custom': {

                    break;
                }
                default: {

                    break;
                }
            }
            updateData.content = content;
            updateData.nextEable = nextEable;
            updateData.callbackDate = callbackDate;
            return updateData;
        },


        /**
         * 比较两个时间是否相等
         * @param {*} e 
         */
        checkNextEableStatus: function checkNextEableStatus(date1, date2) {
            if (date1.getFullYear() == date2.getFullYear() && date1.getMonth() == date2.getMonth() && date1.getDate() == date2.getDate()) {
                return true;
            } else {
                return false;
            }
        },

        /**
         * 选择日期类型变化
         * @param {*} e 
         */
        onDateTypeChange: function onDateTypeChange(e) {
            let updateData = {}
            let date = this.data.date;
            updateData = this.getInitUpdateData(date, this.data.dateType, updateData);
            switch (this.data.dateType) {
                case 'day': {
                    updateData.preText = "上一日";
                    updateData.nextText = "下一日";
                    break;
                }
                case 'weeks': {
                    updateData.preText = "上一周";
                    updateData.nextText = "下一周";
                    break;
                }
                case 'month': {
                    updateData.preText = "上一月";
                    updateData.nextText = "下一月";
                    break;
                }
                case 'quarter': {
                    updateData.preText = "上一季度";
                    updateData.nextText = "下一季度";
                    break;
                }
                case 'custom': {

                    break;
                }
                default: {

                    break;
                }
            }
            this.setData(updateData);
            this.triggerEvent('onCallbackDate', { date: this.data.callbackDate });
        },

        getInitUpdateData: function getInitUpdateData(date, dateType, updateData) {
            let year = date.getFullYear();
            let month = date.getMonth() + 1;
            let day = date.getDate();
            let weeksDay = date.getDay();
            updateData = Object.assign(updateData, this.getUpdateData(date, dateType, 'nromal'));

            return updateData;
        },
    }
});

代码详细地址:https://github.com/fuxingkai/frankui-weapp

你可能感兴趣的:(小程序日,周,月,季度,自定义tab选择组件)