自定义日历
![](https://img.haomeiwen.com/i5456155/f3fca8a3c004c150.png)
最近在做项目,遇到了日历需求,于是左思右想,决定自己写一下。本文在做日历有些取巧,算不上正统的思路,希望能为你打开一片新的天地。
需求
正如你所看,需要做出公历和农历,同时需要设置周六日默认橘红色,点击某个日期,出现橘色背景同时文字发生颜色变化,然后可以切换日期,这里只能逐月切换,有兴趣的朋友可以修改一下,很容易修改。
1> 对于日历,这里只需要知道每个月的第一天是周几即可,后面顺序排下来,然后知道这个月有几天,有了这两个数据就能算出这个月的公历。
// 获取该月的天数
- (NSInteger)getNumberOfDaysInMonthWithMonth:(NSInteger)number withYear:(NSInteger)year
{
NSDate * currentDate = [self getPriousorLaterDateFromDate:self.date withMonth:number withYear:year];
// 这个日期可以你自己给定
NSRange range = [self.calendar rangeOfUnit:NSCalendarUnitDay
inUnit: NSCalendarUnitMonth
forDate:currentDate];
return range.length;
}
/**
* 获取当月中所有天数是周几
*/
- (NSArray*) getAllDayWeeksWithCalendarWithMonth:(NSInteger)number withYear:(NSInteger)year
{
//一个月的总天数
NSUInteger dayCount = [self getNumberOfDaysInMonthWithMonth:number withYear:year];
NSDateFormatter * formatter = [[NSDateFormatter alloc] init];
[formatter setDateFormat:@"yyyy-MM"];
NSString * str = [formatter stringFromDate:[self getPriousorLaterDateFromDate:self.date withMonth:number withYear:year]];
[formatter setDateFormat:@"yyyy-MM-dd"];
NSMutableArray * allDayWeeksArray = [[NSMutableArray alloc] init];
for (NSInteger i = 1; i <= dayCount; i++) {
NSString * sr = [NSString stringWithFormat:@"%@-%ld",str,i];
NSDate *suDate = [formatter dateFromString:sr];
[allDayWeeksArray addObject:[self getweekDayWithDate:suDate]];
}
return allDayWeeksArray;
}
/**
* 获得某天的数据
*
* 获取指定的日期是星期几
*/
- (id) getweekDayWithDate:(NSDate *) date
{
NSDateComponents *comps = [self.calendar components:NSCalendarUnitWeekday fromDate:date];
// 1 是周日,2是周一 3.以此类推
return @([comps weekday]);
}
在这里,我封装了一个函数,方便获取每个月的date数据
//获取指定某个月的date数据
- (NSDate *)getPriousorLaterDateFromDate:(NSDate *)date withMonth:(NSInteger)month withYear:(NSInteger)year
{
NSDateComponents *comp = [[NSDateComponents alloc]init];
[comp setMonth:month];
[comp setYear:year];
NSDate *currentDate = [self.calendar dateFromComponents:comp];
return currentDate;
}
到此,基本上公历数据就可以了,然后通过UICollectionViewController布局日历即可。
2>说到布局,这里就在于我取巧的部分,我并没有获取到正统的公历,我只获取了每个月的第一天和天数。比如,2017年9月的第一天是周五,那么此时返回item的个数就应该是5+这个月的天数,日历都是从周日开始算的,然后算出35个单元格,第一个格是在indexPath.item = 5,然后这一天是周五,item = 5,是1号。
默认初始每个格就是item的值,那么第一个格为0,然后依次排到最后一个格是34,日历上没有0,那么这里需要得到item + 1这个数,35个格,从1排到35。现在要将周五上是item + 1,是6,这个数变成1,即减去5,也即是这个月前面从周日到这一天没有日期的天数,9月份有5天,然后得出,9月1号在周五,然后前面的值均小于1,然后做一步判断将item里面的值置为""即可。其余的月份都是如此,你可以自己试试别的月份。
3>农历部分在网上找到的,难度比较大,直接拿来用的。
先前直接计算出来的日期,这时候得到年月日,拼接起来算出公历日期。
// 字符串转date
-(NSDate* )theTargetStringConversionDate:(NSString *)str
{
//设置转换格式
NSDateFormatter *formatter = [[NSDateFormatter alloc] init] ;
[formatter setDateFormat:@"yyyy-MM-dd"];
//NSString转NSDate
NSDate *date=[formatter dateFromString:str];
return date;
}
然后,通过公历日期,得出农历日期,这部分难度稍大,有兴趣的小伙伴可以自行研究一下。
至此,核心部分讲述完毕,有需要的小伙伴可以下载demo研究。
Demo地址:https://github.com/mayuwen/Calendar