You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
575 lines
29 KiB
575 lines
29 KiB
<!--
|
|
*calendarData.isShow:日历组件显示隐藏
|
|
*calendarData.yearStyle:设置年样式
|
|
*calendarData.monthStyle:设置月样式
|
|
*calendarData.dayStyle:设置日样式
|
|
*returnDate:已选日期回调方法,返回结果{time:'已选日期',timeType:'对应选择模块',isShow:'为了处理组件显示隐藏回调uniapp的坑'}
|
|
:status="checkStatus"
|
|
-->
|
|
<template>
|
|
<view :status="checkStatus" v-if="calendarData.isShow" class="lotus-calendar-wrap">
|
|
<view :class="calendarData.isShow?'lotus-calendar':'lotus-calendar lotus-calendar-out'">
|
|
<view class="lotus-calendar-title">选择日期</view>
|
|
<view class="lotus-calendar-cur-bg-month">{{curMonth+1}}</view>
|
|
<view class="lotus-calendar-cur-date">
|
|
<view class="lotus-calendar-center">
|
|
<view class="lotus-calendar-month">
|
|
<view class="lotus-calendar-prev" @tap="clickPrevMonth">←</view>
|
|
<view :style="calendarData.monthStyle" @tap="setStauts('showMonthFlag',true);" class="lotus-calendar-cur-text">{{showCurMonth}}月</view>
|
|
<view class="lotus-calendar-next" @tap="clickNextMonth">→</view>
|
|
</view>
|
|
<view class="lotus-calendar-year">
|
|
<view class="lotus-calendar-prev" @tap="clickPrevYear">←</view>
|
|
<view :style="calendarData.yearStyle" @tap="setStauts('showYearFlag',true);" class="lotus-calendar-cur-text">{{curYear}}</view>
|
|
<view class="lotus-calendar-next" @tap="clickNextYear">→</view>
|
|
</view>
|
|
</view>
|
|
</view>
|
|
<view class="lotus-calendar-week">
|
|
<view class="lotus-calendar-week-text lotus-calendar-week-range" v-for="(item,index) in weekText" :key="index">
|
|
{{item}}
|
|
</view>
|
|
</view>
|
|
<view class="lotus-calendar-days">
|
|
<view class="lotus-calendar-days-text" v-for="(item,index) in totalDaysArr" :key="index" @tap="clickTargetTime(item,index)">
|
|
<view :class="item.flag?'lotus-calendar-days-act':'lotus-calendar-days-gray'" v-if="item.type == 0">{{item.day}}</view>
|
|
<view :class="item.flag?'lotus-calendar-days-act':''" v-if="item.type == 1">{{item.day}}</view>
|
|
<view :style="calendarData.dayStyle" :class="item.flag?'lotus-calendar-days-act':''" v-if="item.type == 2">{{item.day}}</view>
|
|
</view>
|
|
</view>
|
|
<view class="lotus-calendar-result">
|
|
<view class="lotus-calendar-result-time" v-text="selectTime"></view>
|
|
<view class="lotus-calendar-result-btn" @tap="confirmFn">确定</view>
|
|
</view>
|
|
</view>
|
|
<!--月份选择-->
|
|
<view v-if="showMonthFlag" class="lotus-calendar-months">
|
|
<!-- <view class="lotus-calendar-months-cancel">
|
|
<image @tap="setStauts('showMonthFlag',false);" class="lotus-calendar-months-cancel-icon" src="" mode="aspectFit"></image>
|
|
</view> -->
|
|
<view class="lotus-calendar-months-box">
|
|
<view v-for="(item,index) in monthArray" :key="index" :style="item.flag&&calendarData.monthStyle" class="lotus-calendar-months-text">
|
|
<text @tap="setAct(item,'month');" v-text="item.monthText"></text>
|
|
</view>
|
|
</view>
|
|
</view>
|
|
<!--年份份选择-->
|
|
<view v-if="showYearFlag" class="lotus-calendar-months">
|
|
<view class="lotus-calendar-year lotus-calendar-year2">
|
|
<view class="lotus-calendar-prev" @tap="yearRangeChanage(-10);">←</view>
|
|
<view :style="calendarData.yearStyle" @tap="setStauts('showYearFlag',true);" class="lotus-calendar-cur-text">{{curYear-10}}-{{curYear}}</view>
|
|
<view class="lotus-calendar-next" @tap="yearRangeChanage(+10);">→</view>
|
|
<!-- <view class="lotus-calendar-months-cancel lotus-calendar-months-cancel2">
|
|
<image @tap="setStauts('showYearFlag',false);" class="lotus-calendar-months-cancel-icon" src="" mode="aspectFit"></image>
|
|
</view> -->
|
|
</view>
|
|
<view style="padding-top:20rpx;" class="lotus-calendar-months-box">
|
|
<view v-for="(item,index) in yearArray" :key="index" :style="item.flag&&calendarData.monthStyle" class="lotus-calendar-months-text lotus-calendar-months-text2">
|
|
<text @tap="setAct(item,'year');" v-text="item.y"></text>
|
|
</view>
|
|
</view>
|
|
</view>
|
|
<view v-if="calendarData.isShow" @tap="clickShowCalendar" class="lotus-calendar-mask"></view>
|
|
</view>
|
|
</template>
|
|
<style lang="less">
|
|
@import './Winglau14-lotusCalendar.css';
|
|
</style>
|
|
<script>
|
|
export default {
|
|
name: 'lotus-calendar',
|
|
props: ['calendarData'],
|
|
data () {
|
|
return {
|
|
isShow: true,
|
|
weekText: ['一', '二', '三', '四', '五', '六', '日'],
|
|
aMonth: [31, 28, 31, 30, 31, 30, 31, 31, 30, 31, 30, 31],
|
|
showMonthFlag:false,
|
|
fullFlag:false,
|
|
monthArray:[{
|
|
monthText:'01月',
|
|
showCurMonth:'1',
|
|
flag:false
|
|
},{
|
|
monthText:'02月',
|
|
showCurMonth:'2',
|
|
flag:false
|
|
},{
|
|
monthText:'03月',
|
|
showCurMonth:'3',
|
|
flag:false
|
|
},{
|
|
monthText:'04月',
|
|
showCurMonth:'4',
|
|
flag:false
|
|
},{
|
|
monthText:'05月',
|
|
showCurMonth:'5',
|
|
flag:false
|
|
},{
|
|
monthText:'06月',
|
|
showCurMonth:'6',
|
|
flag:false
|
|
},{
|
|
monthText:'07月',
|
|
showCurMonth:'7',
|
|
flag:false
|
|
},{
|
|
monthText:'08月',
|
|
showCurMonth:'8',
|
|
flag:false
|
|
},{
|
|
monthText:'09月',
|
|
showCurMonth:'9',
|
|
flag:true
|
|
},{
|
|
monthText:'10月',
|
|
showCurMonth:'10',
|
|
flag:false
|
|
},{
|
|
monthText:'11月',
|
|
showCurMonth:'11',
|
|
flag:false
|
|
},{
|
|
monthText:'12月',
|
|
showCurMonth:'12',
|
|
flag:false
|
|
}],
|
|
yearArray:[],
|
|
showYearFlag:false,
|
|
curMonthDays: 0,//当前月天数
|
|
preMonthDays: 0,//上一个月天数
|
|
nextMonthDays: 0,//下一个月天数
|
|
totalDaysArr: [],//日历总天数
|
|
curYear: 0,//当前年份
|
|
curMonth: 0,//当前月份
|
|
curDate: 0,//当前日份
|
|
showCurMonth: 0,//显示日历中月份
|
|
choseIndex: 0,
|
|
choseCurTime: null,//每个日期obj有day/month/year
|
|
prevYear: '',//上一年
|
|
time:null,
|
|
returnType:'',
|
|
sTime:'',
|
|
eTime:'',
|
|
systemMonth:(new Date().getMonth()),//系统当前月份
|
|
parentDate:'',
|
|
actFlag:false,
|
|
savePreMonth:0,
|
|
savePreYear:0,
|
|
startYear:0,
|
|
endYear:0,
|
|
selectTime: ''
|
|
}
|
|
},
|
|
components: {},
|
|
computed:{
|
|
checkStatus(){
|
|
const _this = this;
|
|
const t = _this.calendarData.isShow;
|
|
_this.fullFlag = true;
|
|
if(t&&!_this.actFlag){
|
|
//获取父组件已选的时间值
|
|
_this.parentDate = _this.calendarData.choseTime.split('-');
|
|
_this.selectTime = _this.calendarData.choseTime;
|
|
_this.totalDaysArr = [];
|
|
//记录父组件的年份与月份的值,用于年月日比对上当前日期高亮
|
|
_this.savePreMonth = _this.parentDate[1]*1-1;
|
|
_this.savePreYear = _this.parentDate[0]*1;
|
|
//显示日历组件并生成对应数据
|
|
_this.show(_this.parentDate[0]*1,_this.parentDate[1]*1-1,_this.parentDate[2]*1);
|
|
}
|
|
return t;
|
|
}
|
|
},
|
|
methods: {
|
|
//点击上一月
|
|
clickPrevMonth(){
|
|
//console.log('prev'+this.curMonth);
|
|
this.totalDaysArr = [];
|
|
this.actFlag = true;
|
|
//缓存当前切换月份,用于当前日期高亮判断
|
|
const tMonth = this.curMonth;
|
|
this.savePreMonth = tMonth;
|
|
//判断点击到一月进入一年设置月份为12月
|
|
if (this.curMonth <= 0) {
|
|
this.curMonth = 11;
|
|
this.curYear--;
|
|
} else {
|
|
this.curMonth--;
|
|
}
|
|
//判断当前切换的月份是否与父组件传的月份一致,用于当前日期高亮
|
|
const pMonth = this.calendarData.choseTime.split('-')[1]*1-1;
|
|
if(this.curMonth === pMonth){
|
|
this.savePreMonth = pMonth;
|
|
}
|
|
//生成日历数据
|
|
this.createCalendarData(this.curYear, this.curMonth, this.curDate);
|
|
},
|
|
//点击下一月
|
|
clickNextMonth(){
|
|
this.totalDaysArr = [];
|
|
//缓存当前切换月份,用于当前日期高亮判断
|
|
const tMonth = this.curMonth;
|
|
this.savePreMonth = tMonth;
|
|
this.curMonth++;
|
|
this.actFlag = true;
|
|
//超过12月进入下一年判断设置月份为1月
|
|
if (this.curMonth >= 12) {
|
|
this.curMonth = 0;
|
|
this.curYear++;
|
|
}
|
|
//判断当前切换的月份是否与父组件传的月份一致,用于当前日期高亮
|
|
const pMonth = this.calendarData.choseTime.split('-')[1]*1-1;
|
|
if(this.curMonth === pMonth){
|
|
this.savePreMonth = pMonth;
|
|
}
|
|
//生成日历数据
|
|
this.createCalendarData(this.curYear, this.curMonth, this.curDate);
|
|
},
|
|
//点击上一年
|
|
clickPrevYear(){
|
|
this.prevYear = '';
|
|
this.totalDaysArr = [];
|
|
this.actFlag = true;
|
|
//缓存当前切换年份,用于当前日期高亮判断
|
|
const tYear = this.curYear;
|
|
this.savePreYear = tYear;
|
|
this.curYear--;
|
|
//判断当前切换的年份是否与父组件传的年份一致,用于当前日期高亮
|
|
const pYear = this.calendarData.choseTime.split('-')[0]*1;
|
|
if(this.curYear === pYear){
|
|
this.savePreYear = pYear;
|
|
}
|
|
//生成日历数据
|
|
this.createCalendarData(this.curYear, this.curMonth, this.curDate);
|
|
},
|
|
//点击下一年
|
|
clickNextYear(){
|
|
this.prevYear = '';
|
|
this.totalDaysArr = [];
|
|
//缓存当前切换年份,用于当前日期高亮判断
|
|
const tYear = this.curYear;
|
|
this.savePreYear = tYear;
|
|
this.curYear++;
|
|
this.actFlag = true;
|
|
//判断当前切换的年份是否与父组件传的年份一致,用于当前日期高亮
|
|
const pYear = this.calendarData.choseTime.split('-')[0]*1;
|
|
if(this.curYear === pYear){
|
|
this.savePreYear = pYear;
|
|
}
|
|
//生成日历数据
|
|
this.createCalendarData(this.curYear, this.curMonth, this.curDate);
|
|
},
|
|
//选中日期
|
|
clickTargetTime(curTime){
|
|
let objTime = curTime;
|
|
|
|
//历史日期不允许选择
|
|
let month=objTime.month+1<10?'0'+(objTime.month+1):objTime.month+1;
|
|
let day=objTime.day<10?'0'+objTime.day:objTime.day;
|
|
let rq=objTime.year+'-'+(month)+'-'+day;
|
|
let date=new Date(rq);
|
|
let date2=new Date(this.calendarData.Disablerq);
|
|
if(date.getTime()<date2.getTime()){
|
|
return;
|
|
}
|
|
//历史日期不允许选择
|
|
|
|
//遍历选中时间是否与当前时间一致,一致flag = true or flag = false;
|
|
this.totalDaysArr.map((item) => {
|
|
item.day === objTime.day && item.month === objTime.month ? item.flag = true : item.flag = false;
|
|
});
|
|
//为了不影响objTime这个数据源,定义一个临时obj
|
|
let tempObj = {};
|
|
//处理选中的时间
|
|
for (let i in objTime) {
|
|
if (i === 'day') {
|
|
tempObj.day = objTime[i];
|
|
tempObj.day = tempObj.day < 10 ? `0${tempObj.day}` : tempObj.day;
|
|
}
|
|
if (i === 'month') {
|
|
//选中时间跨入上一年12月份
|
|
tempObj.month = objTime[i];
|
|
if (objTime[i] < 0) {
|
|
tempObj.month = 12;
|
|
} else {
|
|
tempObj.month = tempObj.month + 1 < 10 ? `0${tempObj.month + 1}` : tempObj.month + 1;
|
|
}
|
|
}
|
|
if(i === 'year'){
|
|
tempObj.year = objTime[i];
|
|
}
|
|
}
|
|
|
|
this.choseCurTime = tempObj;
|
|
|
|
//判断选择时间跨入了下一年换算当前月份
|
|
if (this.choseCurTime.month > 12) {
|
|
this.choseCurTime.month = this.choseCurTime.month - 12 < 10 ? `0${this.choseCurTime.month - 12}` : this.choseCurTime.month - 12;
|
|
}
|
|
this.calendarData.isShow = false;
|
|
//用于computed计算标识不然函数会再次执行
|
|
this.actFlag = false;
|
|
this.calendarData.choseTime = `${this.choseCurTime.year}-${this.choseCurTime.month}-${this.choseCurTime.day}`;
|
|
this.selectTime = this.calendarData.choseTime;
|
|
//传值给父组件
|
|
// this.$emit('returnDate',{time:`${this.choseCurTime.year}-${this.choseCurTime.month}-${this.choseCurTime.day}`,timeType:this.calendarData.type,isShow:false});
|
|
},
|
|
//显示与隐藏日历
|
|
clickShowCalendar(){
|
|
this.showMonthFlag = false;
|
|
this.showYearFlag = false;
|
|
this.actFlag = false;
|
|
this.$emit('closeCalendar',{
|
|
time: this.selectTime,
|
|
isShow: false
|
|
});
|
|
},
|
|
//显示
|
|
show(curYear,curMonth,curDate) {
|
|
this.returnType = this.calendarData.type;
|
|
this.totalDaysArr = [];
|
|
const objDate = new Date();
|
|
//当前年份
|
|
this.curYear = curYear||objDate.getFullYear();
|
|
//当前月份
|
|
if(curMonth === 0 || curMonth){
|
|
this.curMonth = curMonth;
|
|
}else{
|
|
this.curMonth = objDate.getMonth();
|
|
}
|
|
//当前几号
|
|
this.curDate = curDate||objDate.getDate();
|
|
//生成对应年份区间列表数据
|
|
this.crateYearRange();
|
|
//生成日历数据
|
|
this.createCalendarData(this.curYear, this.curMonth, this.curDate);
|
|
},
|
|
//方法调用
|
|
lotusCalendar() {
|
|
this.show();
|
|
},
|
|
//是否为闰年
|
|
isLeapYear(year) {
|
|
return ((year % 4 === 0 && year % 100 !== 0) || year % 400 === 0);
|
|
},
|
|
//获取当前月份天数
|
|
getCurMonthDays(year, month) {
|
|
//console.log(year, month);
|
|
if (this.isLeapYear(year)) {
|
|
//判断当前月份是2月
|
|
if (this.curMonth === 1&&this.curMonth === month) {
|
|
return this.aMonth[1] = 29;
|
|
}else {
|
|
return this.aMonth[month];
|
|
}
|
|
} else {
|
|
//不是闰年重置2月天数为28
|
|
this.aMonth[1] = 28;
|
|
return this.aMonth[month];
|
|
}
|
|
},
|
|
//自动补0
|
|
autoPatchZero(val) {
|
|
return val < 10 ? `0${val}` : val;
|
|
},
|
|
//生成日历数据 curYear年份 curMonth月份 curDate号数
|
|
createCalendarData(curYear, curMonth, curDate) {
|
|
//换算当前月份自动补全 val < 10 ? `0${val}` : val;
|
|
this.showCurMonth = curMonth;
|
|
this.showCurMonth = this.autoPatchZero(this.showCurMonth + 1);
|
|
let preYear = '';
|
|
//当前月份1号星期几
|
|
let iWeek = new Date(curYear, curMonth, 1).getDay();
|
|
if (iWeek === 0) {
|
|
iWeek = 7;
|
|
}
|
|
if (curMonth === 0) {
|
|
const t = 12;
|
|
preYear = curYear;
|
|
preYear -= 1;
|
|
this.prevYear = preYear;
|
|
//上一个月份共多少天
|
|
this.preMonthDays = this.getCurMonthDays(preYear, t - 1);
|
|
} else {
|
|
//上一个月份共多少天
|
|
this.preMonthDays = this.getCurMonthDays(curYear, curMonth - 1);
|
|
}
|
|
//当前月份共多少天
|
|
this.curMonthDays = this.getCurMonthDays(curYear, curMonth);
|
|
//下一个月份共多少天
|
|
this.nextMonthDays = this.getCurMonthDays(curYear, curMonth + 1);
|
|
//上一个月剩下几天
|
|
for (let i = iWeek - 2; i >= 0; i--) {
|
|
const obj = {
|
|
day: this.preMonthDays - i,
|
|
className: 'gray',
|
|
type: 0,
|
|
flag: false,
|
|
month: curMonth - 1,
|
|
year: preYear ? preYear : curYear
|
|
};
|
|
this.totalDaysArr.push(obj);
|
|
}
|
|
|
|
//当前月份天数
|
|
for (let i = 1; i <= this.curMonthDays; i++) {
|
|
const obj = {
|
|
day: i,
|
|
type: 1,
|
|
flag: false,
|
|
className: '',
|
|
month: curMonth,
|
|
year: curYear
|
|
};
|
|
if (i === curDate && curMonth === this.savePreMonth&& curYear === this.savePreYear) {
|
|
obj.flag = true;
|
|
obj.className = 'act';
|
|
obj.type = 2;
|
|
}
|
|
//历史日期不允许选择
|
|
let month=curMonth+1<10?'0'+(curMonth+1):curMonth+1;
|
|
let day=i<10?'0'+i:i;
|
|
let rq=curYear+'-'+(month)+'-'+day;
|
|
let date=new Date(rq);
|
|
let date2=new Date(this.calendarData.Disablerq);
|
|
if(date.getTime()<date2.getTime()){
|
|
obj.flag = false;
|
|
obj.className = 'gray';
|
|
obj.type = 0;
|
|
}
|
|
//历史日期不允许选择
|
|
this.totalDaysArr.push(obj);
|
|
}
|
|
|
|
//下一个月开始的几天
|
|
const leaveLength = 42 - this.totalDaysArr.length;
|
|
for (let i = 1; i <= leaveLength; i++) {
|
|
const obj = {
|
|
day: i,
|
|
className: '',
|
|
type: 1,
|
|
flag: false,
|
|
month: curMonth + 1,
|
|
year: curYear
|
|
};
|
|
//点击月份进入了下一年
|
|
if(obj.month >= 12){
|
|
obj.year += 1
|
|
}
|
|
//历史日期不允许选择
|
|
let month=curMonth+2<10?'0'+(curMonth+2):curMonth+2;
|
|
let day=obj.day<10?'0'+obj.day:obj.day;
|
|
let rq=obj.year+'-'+(month)+'-'+day;
|
|
let date=new Date(rq);
|
|
let date2=new Date(this.calendarData.Disablerq);
|
|
if(date.getTime()<date2.getTime()){
|
|
obj.flag = false;
|
|
obj.className = 'gray';
|
|
obj.type = 0;
|
|
}
|
|
//历史日期不允许选择
|
|
this.totalDaysArr.push(obj);
|
|
}
|
|
},
|
|
//月份/年份设置高亮
|
|
setAct(obj,type){
|
|
if(type === "month"){
|
|
this.monthArray.map((item,index)=>{
|
|
item.flag = false;
|
|
});
|
|
this.curMonth = obj.showCurMonth*1-1;
|
|
this.setStauts('showMonthFlag',false);
|
|
}else{
|
|
this.yearArray.map((item,index)=>{
|
|
item.flag = false;
|
|
});
|
|
this.setStauts('showYearFlag',false);
|
|
this.curYear = obj.y;
|
|
}
|
|
obj.flag = true;
|
|
this.totalDaysArr = [];
|
|
this.createCalendarData(this.curYear,this.curMonth,this.curDate);
|
|
this.selectTime = `${this.curYear}-${this.curMonth+1<10?'0'+(this.curMonth+1):(this.curMonth+1)}-${this.curDate<10?'0'+this.curDate:this.curDate}`
|
|
this.actFlag = true;
|
|
this.fullFlag = true;
|
|
},
|
|
//更改状态
|
|
setStauts(type,obj){
|
|
//月份选中
|
|
if(type === "showMonthFlag" && obj){
|
|
this.monthArray.map((item,index)=>{
|
|
item.flag = false;
|
|
if(item.showCurMonth*1-1 === this.curMonth){
|
|
item.flag = true;
|
|
}
|
|
});
|
|
}else{
|
|
//年份选中
|
|
this.yearArray.map((item,index)=>{
|
|
item.flag = false;
|
|
if(item.y === this.curYear){
|
|
item.flag = true;
|
|
}
|
|
});
|
|
}
|
|
this[type] = obj;
|
|
this.fullFlag = false;
|
|
},
|
|
//年份区间数据创建
|
|
crateYearRange(year){
|
|
this.yearArray = [];
|
|
this.startYear = this.curYear-15;
|
|
this.endYear = this.curYear+5;
|
|
this.actFlag = true;
|
|
for(let i = this.startYear+1;i<this.startYear+15;i++){
|
|
let sObj = {
|
|
y:i,
|
|
flag:false
|
|
}
|
|
//判断切换年份是否与之前年份一致&高亮
|
|
if(i === this.savePreYear){
|
|
sObj.flag = true;
|
|
}
|
|
this.yearArray.push(sObj);
|
|
}
|
|
for(let j = this.curYear;j<=this.curYear+5;j++){
|
|
let eObj = {
|
|
y:j,
|
|
flag:false
|
|
}
|
|
//判断切换年份是否与之前年份一致&高亮
|
|
if(j === this.savePreYear){
|
|
eObj.flag = true;
|
|
}
|
|
this.yearArray.push(eObj);
|
|
}
|
|
},
|
|
//年份区间变化
|
|
yearRangeChanage(type){
|
|
this.curYear += type;
|
|
this.crateYearRange();
|
|
},
|
|
// 点击确定按钮
|
|
confirmFn(){
|
|
this.showMonthFlag = false;
|
|
this.showYearFlag = false;
|
|
this.actFlag = false;
|
|
let option = {
|
|
isShow: false
|
|
}
|
|
if(this.choseCurTime){
|
|
option.time = `${this.choseCurTime.year}-${this.choseCurTime.month}-${this.choseCurTime.day}`;
|
|
option.timeType = this.calendarData.type?this.calendarData.type:'';
|
|
}
|
|
if(this.selectTime){
|
|
option.time = this.selectTime;
|
|
}
|
|
this.$emit('returnDate',{
|
|
...option,
|
|
});
|
|
}
|
|
}
|
|
}
|
|
</script>
|