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

<!--
*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>