Apple有时候是非常有人性化的,他们的家的iphone居然还支持藏文,但有时候就是很坑的,简直毫无人了可言。
Apple有一个 UIDatePicker 这UI控件可以显示中国的农历,但是去获取现实的数据的时候却是给了一个NSDate类型的数据,然后你如果要现实农历还需要去转换,最坑的是UIDatePicker和UIPickerView还不是同一个东西,一个是继承UIControl, 一个是继承 UIView的,UIPickerView可以通过代理去获取,但是UIDatePicker就只能通过addTarget的方式去获取它的value。
这里就需要去转化了。
试过很多的方法,但是网上的很多人说有坑,说的是万年历的数据不全,或者是有错误,我这里尝试了很多套的万年历数据,最后用了一套能解决百分之九十九的日期转换。一般会出错的就是农历闰月的时间转化,闰月转阳历都说少一天,反复切换,日期就不会不断的变早。
我们先去创建一个header文件去放万年历的数据:
//
// QYJDataConfig.h
// QYJGYDateCenter
//
// Created by Isoftstone on 17/2/7.
// Copyright © 2017年 com.qyji.DateDemo. All rights reserved.
//
#ifndef QYJDataConfig_h
#define QYJDataConfig_h
#include <stdio.h>
#include <string.h>
#import "NSDate+FSExtension.h"
//最小年限
#define MIN_YEAR 1900
//最大年限 取决下面万年历转化的数据
#define MAX_YEAR 2099
#define lunarDays @[@"零",@"初一",@"初二",@"初三",@"初四",@"初五",@"初六",@"初七",@"初八",@"初九",@"初十",@"十一",@"十二",@"十三",@"十四",@"十五", @"十六",@"十七",@"十八",@"十九",@"廿", @"廿一",@"廿二", @"廿三", @"廿四", @"廿五", @"廿六", @"廿七", @"廿八", @"廿九", @"卅"]
#define lunarMonths @[@"零", @"正",@"二", @"三", @"四", @"五", @"六", @"七", @"八", @"九", @"十", @"冬", @"腊"]
static int DAYS_BEFORE_MONTH[13] = {0, 31, 59, 90, 120, 151, 181, 212, 243, 273, 304, 334, 365};
unsigned int lunarToSolar[200] = {
0x84B6BF,/*1900*/
0x04AE53, 0x0A5748, 0x5526BD, 0x0D2650, 0x0D9544, 0x46AAB9, 0x056A4D, 0x09AD42, 0x24AEB6, 0x04AE4A,/*1901-1910*/
0x6A4DBE, 0x0A4D52, 0x0D2546, 0x5D52BA, 0x0B544E, 0x0D6A43, 0x296D37, 0x095B4B, 0x749BC1, 0x049754,/*1911-1920*/
0x0A4B48, 0x5B25BC, 0x06A550, 0x06D445, 0x4ADAB8, 0x02B64D, 0x095742, 0x2497B7, 0x04974A, 0x664B3E,/*1921-1930*/
0x0D4A51, 0x0EA546, 0x56D4BA, 0x05AD4E, 0x02B644, 0x393738, 0x092E4B, 0x7C96BF, 0x0C9553, 0x0D4A48,/*1931-1940*/
0x6DA53B, 0x0B554F, 0x056A45, 0x4AADB9, 0x025D4D, 0x092D42, 0x2C95B6, 0x0A954A, 0x7B4ABD, 0x06CA51,/*1941-1950*/
0x0B5546, 0x555ABB, 0x04DA4E, 0x0A5B43, 0x352BB8, 0x052B4C, 0x8A953F, 0x0E9552, 0x06AA48, 0x6AD53C,/*1951-1960*/
0x0AB54F, 0x04B645, 0x4A5739, 0x0A574D, 0x052642, 0x3E9335, 0x0D9549, 0x75AABE, 0x056A51, 0x096D46,/*1961-1970*/
0x54AEBB, 0x04AD4F, 0x0A4D43, 0x4D26B7, 0x0D254B, 0x8D52BF, 0x0B5452, 0x0B6A47, 0x696D3C, 0x095B50,/*1971-1980*/
0x049B45, 0x4A4BB9, 0x0A4B4D, 0xAB25C2, 0x06A554, 0x06D449, 0x6ADA3D, 0x0AB651, 0x095746, 0x5497BB,/*1981-1990*/
0x04974F, 0x064B44, 0x36A537, 0x0EA54A, 0x86B2BF, 0x05AC53, 0x0AB647, 0x5936BC, 0x092E50, 0x0C9645,/*1991-2000*/
0x4D4AB8, 0x0D4A4C, 0x0DA541, 0x25AAB6, 0x056A49, 0x7AADBD, 0x025D52, 0x092D47, 0x5C95BA, 0x0A954E,/*2001-2010*/
0x0B4A43, 0x4B5537, 0x0AD54A, 0x955ABF, 0x04BA53, 0x0A5B48, 0x652BBC, 0x052B50, 0x0A9345, 0x474AB9,/*2011-2020*/
0x06AA4C, 0x0AD541, 0x24DAB6, 0x04B64A, 0x6a573D, 0x0A4E51, 0x0D2646, 0x5E933A, 0x0D534D, 0x05AA43,/*2021-2030*/
0x36B537, 0x096D4B, 0xB4AEBF, 0x04AD53, 0x0A4D48, 0x6D25BC, 0x0D254F, 0x0D5244, 0x5DAA38, 0x0B5A4C,/*2031-2040*/
0x056D41, 0x24ADB6, 0x049B4A, 0x7A4BBE, 0x0A4B51, 0x0AA546, 0x5B52BA, 0x06D24E, 0x0ADA42, 0x355B37,/*2041-2050*/
0x09374B, 0x8497C1, 0x049753, 0x064B48, 0x66A53C, 0x0EA54F, 0x06AA44, 0x4AB638, 0x0AAE4C, 0x092E42,/*2051-2060*/
0x3C9735, 0x0C9649, 0x7D4ABD, 0x0D4A51, 0x0DA545, 0x55AABA, 0x056A4E, 0x0A6D43, 0x452EB7, 0x052D4B,/*2061-2070*/
0x8A95BF, 0x0A9553, 0x0B4A47, 0x6B553B, 0x0AD54F, 0x055A45, 0x4A5D38, 0x0A5B4C, 0x052B42, 0x3A93B6,/*2071-2080*/
0x069349, 0x7729BD, 0x06AA51, 0x0AD546, 0x54DABA, 0x04B64E, 0x0A5743, 0x452738, 0x0D264A, 0x8E933E,/*2081-2090*/
0x0D5252, 0x0DAA47, 0x66B53B, 0x056D4F, 0x04AE45, 0x4A4EB9, 0x0A4D4C, 0x0D1541, 0x2D92B5 /*2091-2099*/
};
unsigned int solarTolunar[] = {
0x04bd8,0x04ae0,0x0a570,0x054d5,0x0d260,0x0d950,0x16554,0x056a0,0x09ad0,0x055d2,
0x04ae0,0x0a5b6,0x0a4d0,0x0d250,0x1d255,0x0b540,0x0d6a0,0x0ada2,0x095b0,0x14977,
0x04970,0x0a4b0,0x0b4b5,0x06a50,0x06d40,0x1ab54,0x02b60,0x09570,0x052f2,0x04970,
0x06566,0x0d4a0,0x0ea50,0x06e95,0x05ad0,0x02b60,0x186e3,0x092e0,0x1c8d7,0x0c950,
0x0d4a0,0x1d8a6,0x0b550,0x056a0,0x1a5b4,0x025d0,0x092d0,0x0d2b2,0x0a950,0x0b557,
0x06ca0,0x0b550,0x15355,0x04da0,0x0a5b0,0x14573,0x052b0,0x0a9a8,0x0e950,0x06aa0,
0x0aea6,0x0ab50,0x04b60,0x0aae4,0x0a570,0x05260,0x0f263,0x0d950,0x05b57,0x056a0,
0x096d0,0x04dd5,0x04ad0,0x0a4d0,0x0d4d4,0x0d250,0x0d558,0x0b540,0x0b6a0,0x195a6,
0x095b0,0x049b0,0x0a974,0x0a4b0,0x0b27a,0x06a50,0x06d40,0x0af46,0x0ab60,0x09570,
0x04af5,0x04970,0x064b0,0x074a3,0x0ea50,0x06b58,0x055c0,0x0ab60,0x096d5,0x092e0,
0x0c960,0x0d954,0x0d4a0,0x0da50,0x07552,0x056a0,0x0abb7,0x025d0,0x092d0,0x0cab5,
0x0a950,0x0b4a0,0x0baa4,0x0ad50,0x055d9,0x04ba0,0x0a5b0,0x15176,0x052b0,0x0a930,
0x07954,0x06aa0,0x0ad50,0x05b52,0x04b60,0x0a6e6,0x0a4e0,0x0d260,0x0ea65,0x0d530,
0x05aa0,0x076a3,0x096d0,0x04bd7,0x04ad0,0x0a4d0,0x1d0b6,0x0d250,0x0d520,0x0dd45,
0x0b5a0,0x056d0,0x055b2,0x049b0,0x0a577,0x0a4b0,0x0aa50,0x1b255,0x06d20,0x0ada0,
0x04b63,0x0937f,0x049f8,0x04970,0x064b0,0x068a6,0x0ea5f,0x06b20,0x0a6c4,0x0aaef,
0x092e0,0x0d2e3,0x0c960,0x0d557,0x0d4a0,0x0da50,0x05d55,0x056a0,0x0a6d0,0x055d4,
0x052d0,0x0a9b8,0x0a950,0x0b4a0,0x0b6a6,0x0ad50,0x055a0,0x0aba4,0x0a5b0,0x052b0,
0x0b273,0x06930,0x07337,0x06aa0,0x0ad50,0x04b55,0x04b6f,0x0a570,0x054e4,0x0d260,
0x0e968,0x0d520,0x0daa0,0x06aa6,0x056df,0x04ae0,0x0a9d4,0x0a4d0,0x0d150,0x0f252,
0x0d520};
#endif /* QYJDataConfig_h */
这里使用两套的数据源,分开来使用。
转换的代码:
阳历转阴历(公转农历)
#pragma mark - lunar To solar
// 传入农历转换成新历
+ (NSString *)toSolar:(id)lunar {
NSString *lunarStr;
if ([lunar isKindOfClass:[NSDate class]]) {
NSDateFormatter *formatter = [[NSDateFormatter alloc] init];
formatter.dateFormat = @"yyyy-MM-dd";
lunarStr = [formatter stringFromDate:lunar];
return lunarStr;
} else {
lunarStr = lunar;
}
NSCharacterSet *set = [NSCharacterSet characterSetWithCharactersInString:@"年月"];
NSArray *year_month_day = [lunarStr componentsSeparatedByCharactersInSet:set];
int year = [year_month_day.firstObject intValue];
NSString *monthStr = year_month_day[1];
int month = 0;
BOOL reserved = NO;
if ([monthStr rangeOfString:@"闰"].length > 0) {
//闰年
monthStr = [monthStr substringFromIndex:1];
month = (int)[lunarMonths indexOfObject:monthStr];
reserved = YES;
} else {
//平年
month = (int)[lunarMonths indexOfObject:monthStr];
reserved = NO;
}
int day = (int)[lunarDays indexOfObject:year_month_day[2]];
return [LunarToSolar lunarToSolarYear:year month:month monthDay:day leapMonth:reserved];
}
+ (NSString *)lunarToSolarYear:(int)year month:(int)month monthDay:(int)monthDay leapMonth:(BOOL)isLeapMonth {
int dayOffset;
int leapMonth;
int i;
if (year < MIN_YEAR || year > MAX_YEAR || month < 1 || month > 12
|| monthDay < 1 || monthDay > 30) {
return @"";
}
dayOffset = (lunarToSolar[year - MIN_YEAR] & 0x001F) - 1;
if (((lunarToSolar[year - MIN_YEAR] & 0x0060) >> 5) == 2)
dayOffset += 31;
for (i = 1; i < month; i++) {
if ((lunarToSolar[year - MIN_YEAR] & (0x80000 >> (i - 1))) == 0)
dayOffset += 29;
else
dayOffset += 30;
}
dayOffset += monthDay;
leapMonth = (lunarToSolar[year - MIN_YEAR] & 0xf00000) >> 20;
// 这一年有闰月
if (leapMonth != 0) {
if (month > leapMonth || (month == leapMonth && isLeapMonth)) {
if ((lunarToSolar[year - MIN_YEAR] & (0x80000 >> (month - 1))) == 0)
dayOffset += 29;
else
dayOffset += 30;
}
}
if (dayOffset > 366 || (year % 4 != 0 && dayOffset > 365)) {
year += 1;
if (year % 4 == 1)
dayOffset -= 366;
else
dayOffset -= 365;
}
int solarInfo[3];
for (i = 1; i < 13; i++) {
int iPos = DAYS_BEFORE_MONTH[i];
if (year % 4 == 0 && i > 2) {
iPos += 1;
}
if (year % 4 == 0 && i == 2 && iPos + 1 == dayOffset) {
solarInfo[1] = i;
solarInfo[2] = dayOffset - 31;
break;
}
if (iPos >= dayOffset) {
solarInfo[1] = i;
iPos = DAYS_BEFORE_MONTH[i - 1];
if (year % 4 == 0 && i > 2) {
iPos += 1;
}
if (dayOffset > iPos)
solarInfo[2] = dayOffset - iPos;
else if (dayOffset == iPos) {
if (year % 4 == 0 && i == 2)
solarInfo[2] = DAYS_BEFORE_MONTH[i] - DAYS_BEFORE_MONTH[i - 1] + 1;
else
solarInfo[2] = DAYS_BEFORE_MONTH[i] - DAYS_BEFORE_MONTH[i - 1];
} else
solarInfo[2] = dayOffset;
break;
}
}
solarInfo[0] = year;
int resultYear = year;
int resultMonth = solarInfo[1];
int resultDay = solarInfo[2];
NSString *result = [NSString stringWithFormat:@"%d-%02d-%02d", resultYear, resultMonth, resultDay];
return result;
}
阴历转阳历(农历转公历)
#pragma mark - solar To lunar
// 传入新历转换成农历
+ (NSString *)toLunar:(id)solar {
NSDate *solarDate = nil;
if ([solar isKindOfClass:[NSDate class]]) {
solarDate = solar;
} else {
NSDateFormatter *formatter = [[NSDateFormatter alloc] init];
formatter.dateFormat = @"yyyy-MM-dd";
solarDate = [formatter dateFromString:solar];
}
NSInteger year = solarDate.fs_year;
NSInteger month = solarDate.fs_month;
NSInteger day = solarDate.fs_day;
return [self solarToLunarCalendarThisDate:solarDate year:year month:month day:day];
}
+ (NSString *)solarToLunarCalendarThisDate:(NSDate *)thisdate year:(int)lunarYear month:(int)lunarMonth day:(int)lunarDay {
NSString *start = @"1900-01-31";
NSDateFormatter *dateFormatter = [[NSDateFormatter alloc] init];
[dateFormatter setDateFormat:@"yyyy-MM-dd"];
NSString *end = [dateFormatter stringFromDate:thisdate];
NSDateFormatter *f = [[NSDateFormatter alloc] init];
[f setDateFormat:@"yyyy-MM-dd"];
NSDate *startDate = [f dateFromString:start];
NSDate *endDate = [f dateFromString:end];
NSCalendar *gregorianCalendar = [[NSCalendar alloc] initWithCalendarIdentifier:NSGregorianCalendar];
NSDateComponents *components = [gregorianCalendar components:NSDayCalendarUnit fromDate:startDate toDate:endDate options:0];
int dayCyclical=(int)(([components day] + 30)/(86400/(3600*24)))+10;
int sumdays = (int)[components day];
int tempdays = 0;
//计算农历年
for (lunarYear = 1900; lunarYear < 2100 && sumdays > 0; lunarYear++) {
tempdays = [self LunarYearDays:lunarYear];
sumdays -= tempdays;
}
if (sumdays < 0) {
sumdays += tempdays;
lunarYear--;
}
//计算闰月
int doubleMonth = [self DoubleMonth:lunarYear];
BOOL isLeap = NO;
//计算农历月
for (lunarMonth = 1; lunarMonth < 13 && sumdays > 0; lunarMonth++) {
//闰月
if (doubleMonth > 0 && lunarMonth == (doubleMonth + 1) && isLeap == false) {
--lunarMonth;
isLeap = true;
tempdays = [self DoubleMonthDays:lunarYear];
} else {
tempdays = [self MonthDays:lunarYear:lunarMonth];
}
//解除闰月
if (isLeap == true && lunarMonth == (doubleMonth + 1)) {
isLeap = false;
}
sumdays -= tempdays;
}
//计算农历日
if (sumdays == 0 && doubleMonth > 0 && lunarMonth == doubleMonth + 1) {
if (isLeap) {
isLeap = false;
} else {
isLeap = true;
--lunarMonth;
}
}
if (sumdays < 0) {
sumdays += tempdays;
--lunarMonth;
}
lunarDay = sumdays + 1;
NSString *string = [NSString stringWithFormat:@"%d年", lunarYear];
if (isLeap) {
string = [string stringByAppendingString:[NSString stringWithFormat:@"闰%@月", lunarMonths[lunarMonth]]];
} else {
string = [string stringByAppendingString:[NSString stringWithFormat:@"%@月", lunarMonths[lunarMonth]]];
}
string = [string stringByAppendingString:[NSString stringWithFormat:@"%@", lunarDays[lunarDay]]];
return string;
}
+ (int)LunarYearDays:(int)y {
int i, sum = 348;
for (i = 0x8000; i > 0x8; i >>= 1)
{
if ((solarTolunar[y - 1900] & i) != 0)
sum += 1;
}
return (sum + [self DoubleMonthDays:y]);
}
+ (int)DoubleMonth:(int)y {
return (solarTolunar[y - 1900] & 0xf);
}
///返回农历年闰月的天数
+ (int)DoubleMonthDays:(int)y {
if ([self DoubleMonth:y] != 0)
return (((solarTolunar[y - 1900] & 0x10000) != 0) ? 30 : 29);
else
return (0);
}
///返回农历年月份的总天数
+ (int)MonthDays:(int)y :(int)m {
return (((solarTolunar[y - 1900] & (0x10000 >> m)) != 0) ? 30 : 29);
}
/*!
@method toLunar:
@abstract 公历转农历
@discussion 公历转农历
@param solar id
@result NSString
*/
+ (NSString *)toLunar:(id)solar;
/*!
@method toSolar:
@abstract 农历转公历
@discussion 农历转公历
@param lunar id
@result NSString
*/
+ (NSString *)toSolar:(id)lunar;
外部调用下这两个方法。