⭐ 欢迎来到虫虫下载站! | 📦 资源下载 📁 资源专辑 ℹ️ 关于我们
⭐ 虫虫下载站

📄 calendarconvert.cpp

📁 mytoolpad 源码
💻 CPP
字号:
#include "stdafx.h"
#include "CalendarConvert.h"


/***************    本类调用规则    ******************
******************************************************

1.	使用下面的函数来制作日历结构
	MakeCalendar(CALENDAR *m_cal,int y,int m,int d);//填充日期结构的年月日

2.	使用本函数来检验日历是否合法
	BOOL GongDataIsValid(CALENDAR m_date);		   //判断公历日期是否在界内
	BOOL NongDataIsValid(CALENDAR m_date);		   //判断农历日期是否在界内

3.   将合法日历进行公农历相互转换
	CALENDAR ConvertToGongLi(CALENDAR m_nongli);	   //将农历日期转化为公历日期
	CALENDAR ConvertToNongLi(CALENDAR m_gongli);	   //将公历日期转化为农历日期

4.	其他辅助函数:获取星期;获取公历日期之间天数;获取农历日期之间天数
	int  GetWeekInfo(CALENDAR m_gongli);		   //根据公历日期计算星期
	int  GetGongDays(CALENDAR m_start,CALENDAR m_end);//获取公历两个日期之间的天数
	int  GetNongDays(CALENDAR m_start,CALENDAR m_end);//获取农历两个日期之间的天数
	int  CompareTwoDate(CALENDAR m_fir,CALENDAR m_sec);//比较两个日期的大小
	CALENDAR GetCurGongDate();					//将当前公历日期合成为结构
	CString GetGanZhi(int m_nongyear);				//获取农历年份的干支名称

******************************************************
***************    调用规则结束    ******************/





CCalendarConvert::CCalendarConvert()
{
	m_minyear=1950;
	m_maxyear=2050;

	unsigned char data[100][3]=
	{
		{0x2f,0x6c,0xa0},{0x24,0xb5,0x50},{0xda,0x53,0x55},{0x2c,0x4d,0xa0},
		{0x21,0xa5,0xb0},{0x57,0x45,0x73},{0xaa,0x52,0xb0},{0x1e,0xa9,0xa8},
		{0x30,0xe9,0x50},{0x26,0x6a,0xa0},{0x9b,0xae,0xa6},{0x2d,0xab,0x50},
		{0x23,0x4b,0x60},{0x18,0xaa,0xe4},{0xab,0xa5,0x70},{0x20,0x52,0x60},
		{0x14,0xf2,0x63},{0x27,0xd9,0x50},{0x9d,0x5b,0x57},{0x2f,0x56,0xa0},

		{0x24,0x96,0xd0},{0x1a,0x4d,0xd5},{0xad,0x4a,0xd0},{0x21,0xa4,0xd0},
		{0x16,0xd4,0xd4},{0x29,0xd2,0x50},{0x9e,0xd5,0x58},{0x30,0xb5,0x40},
		{0x25,0xb6,0xa0},{0x5b,0x95,0xa6},{0xae,0x95,0xb0},{0x23,0x49,0xb0},
		{0x18,0xa9,0x74},{0x2b,0xa4,0xb0},{0xa0,0xb2,0x7a},{0x32,0x6a,0x50},
		{0x27,0x6d,0x40},{0x1c,0xaf,0x46},{0xaf,0xab,0x60},{0x24,0x95,0x70},

		{0x1a,0x4a,0xf5},{0x2d,0x49,0x70},{0xa2,0x64,0xb0},{0x16,0x74,0xa3},
		{0x28,0xea,0x50},{0x1e,0x6b,0x58},{0xb1,0x5a,0xc0},{0x25,0xab,0x60},
		{0x1b,0x96,0xd5},{0x2e,0x92,0xe0},{0xa3,0xc9,0x60},{0x17,0xd9,0x54},
		{0x2a,0xd4,0xa0},{0x1f,0xda,0x50},{0x95,0x75,0x52},{0x27,0x56,0xa0},
		{0x1c,0xab,0xb7},{0x30,0x25,0xd0},{0xa5,0x92,0xd0},{0x19,0xca,0xb5},

		{0x2c,0xa9,0x50},{0x21,0xb4,0xa0},{0x96,0xba,0xa4},{0x28,0xad,0x50},
		{0x1e,0x55,0xd9},{0x31,0x4b,0xa0},{0xa6,0xa5,0xb0},{0x5b,0x51,0x76},
		{0x2e,0x52,0xb0},{0x23,0xa9,0x30},{0x98,0x79,0x54},{0x2a,0x6a,0xa0},
		{0x1f,0xad,0x50},{0x15,0x5b,0x52},{0xa8,0x4b,0x60},{0x1c,0xa6,0xe6},
		{0x2f,0xa4,0xe0},{0x24,0xd2,0x60},{0x99,0xea,0x65},{0x2b,0xd5,0x30},

		{0x21,0x5a,0xa0},{0x16,0x76,0xa3},{0xa9,0x96,0xd0},{0x1e,0x4a,0xfb},
		{0x31,0x4a,0xd0},{0x26,0xa4,0xd0},{0xdb,0xd0,0xb6},{0x2d,0xd2,0x50},
		{0x22,0xd5,0x20},{0x17,0xdd,0x45},{0xaa,0xb5,0xa0},{0x1f,0x56,0xd0},
		{0x15,0x55,0xb2},{0x28,0x49,0xb0},{0x9d,0xa5,0x77},{0x2f,0xa4,0xb0},
		{0x24,0xaa,0x50},{0x59,0xb2,0x55},{0xac,0x6d,0x20},{0x20,0xad,0xa0}
	};
	CalendarData=new char*[100];
	for(int i=0;i<100;i++)
	{
		CalendarData[i]=new char[3];
		for(int j=0;j<3;j++)
			CalendarData[i][j]=data[i][j];
	}
}
CCalendarConvert::~CCalendarConvert()
{
	for(int i=0;i<100;i++)
		delete[] CalendarData[i];
	delete[] CalendarData;
}

//判断公历日期是否在界内
BOOL CCalendarConvert::GongDataIsValid(CALENDAR m_date)
{
	if(m_date.year>1950 && m_date.year<2050)
		return TRUE;
	else
	{
		if(m_date.year==1950)
		{
			if(m_date.month>2)
				return TRUE;
			else
			{
				if(m_date.month==2)
				{
					if(m_date.day>16)
						return TRUE;
				}
			}
		}
	}
	return FALSE;
}

//判断农历日期是否在界内
BOOL CCalendarConvert::NongDataIsValid(CALENDAR m_date)
{
	if(m_date.year>1949 && m_date.year<2049)
		return TRUE;
	else
	{
		if(m_date.year==2049)
		{
			if(m_date.month<12)
				return TRUE;
			else
			{
				if(m_date.month==12)
				{
					if(m_date.day<8)
						return TRUE;
				}
			}
		}
	}
	return FALSE;
}

//将农历日期转化为公历日期
CALENDAR CCalendarConvert::ConvertToGongLi(CALENDAR m_nongli)
{
	int days=DaysFromSpringDay(m_nongli); //计算农历当前日期到春节的天数
	days+=GetDaysFromStart(m_nongli.year);//加上春节到元旦的天数(以元旦为基准)
	int year=m_nongli.year;			   //默认年份相同
	int alldays=GetGongYearDays(m_nongli.year);//计算农历年的公历天数
	if(days>alldays)				   //公历年份超前
	{
		days-=alldays;
		year++;
	}
	CALENDAR result=CalGongDate(year,days);//根据以元旦为基准的天数来确定日期
	result.week=GetWeekInfo(result);
	return result;
}

//将公历日期转化为农历日期
CALENDAR CCalendarConvert::ConvertToNongLi(CALENDAR m_gongli)
{
	int days=DaysFromNewYear(m_gongli);    //公历日期到元旦的天数
	int alldays=GetDaysFromStart(m_gongli.year);//春节到元旦的天数

	int year=m_gongli.year;			    //默认农历年与公历年相同
	if(days<=alldays)				    //农历年滞后
	{
		year--;
		days+=GetGongYearDays(year);	    //再加上公历年的整年天数
	}
	days-=GetDaysFromStart(year);	         //减去春节到元旦天数得到以春节为基准的天数
	CALENDAR result=CalNongDate(year,days);//根据农历年和以春节为基准的天数计算公历日期
	result.week=GetWeekInfo(m_gongli);
	return result;
}

//根据公历日期计算星期
int  CCalendarConvert::GetWeekInfo(CALENDAR m_gongli)
{
	int left=0;
	for(int i=m_minyear;i<m_gongli.year;i++)//1950年元旦为星期六
	{
		left+=YearIsRunNian(i)?2:1;
		left%=7;
	}
	for(i=1;i<m_gongli.month;i++)
	{
		left+=GetGongMonthDays(m_gongli.year,i);
		left%=7;
	}
	left+=(m_gongli.day+6);
	left%=7;
	return left;
}

//当前公历日期后N天的公历日期
CALENDAR CCalendarConvert::GetDateAfterDays(CALENDAR m_first,int m_days)
{
	int m_firstdays=DaysFromNewYear(m_first)+m_days;
	return CalGongDate(m_first.year,m_firstdays);
}

//根据给定年份并且以元旦为基准的天数来确定公历日期
CALENDAR CCalendarConvert::CalGongDate(int year,int days)
{
	CALENDAR result;
	int caldays=0;
	result.year=year;

	for(int i=1;i<13;i++)
	{
		caldays+=GetGongMonthDays(year,i);
		if(caldays>=days)
		{
			caldays-=GetGongMonthDays(year,i);
			result.month=i;
			result.day=days-caldays;
			break;
		}
	}
	return result;
}

//根据农历年和以春节为基准的天数计算公历日期
CALENDAR CCalendarConvert::CalNongDate(int year,int days)
{
	CALENDAR result;
	int caldays=0;
	result.year=year;
	result.isrunyue=FALSE;

	for(int i=1;i<13;i++)
	{
		caldays+=GetNongMonthDays(year,i);		//计算春节以来月天数累加和
		if(caldays>=days)					//天数超过总天数,月份已到
		{
			caldays-=GetNongMonthDays(year,i);	//减去多算的一个月
			result.month=i;
			result.day=days-caldays;
			result.isrunyue=FALSE;
			break;
		}
		else
		{
			if(GetNongRunYue(year)==i)         //考虑闰月
			{
				caldays+=GetNongRunYueDays(year);
				if(caldays>=days)
				{
					caldays-=GetNongRunYueDays(year);
					result.month=i;
					result.day=days-caldays;
					result.isrunyue=TRUE;
					break;
				}
			}
		}
	}
	return result;
}

//填充日期结构的年月日
void CCalendarConvert::MakeCalendar(CALENDAR *m_cal,int y,int m,int d)
{
	m_cal->year=y;
	m_cal->month=m;
	m_cal->day=d;
}

//获取公历月的天数
int  CCalendarConvert::GetGongMonthDays(int year,int month)
{
	int ret=30;
	if(month==2)
		ret=YearIsRunNian(year)?29:28;
	else
		ret=GongMonthIsLarge(month)?31:30;
	return ret;
}

//获取农历月的天数
int  CCalendarConvert::GetNongMonthDays(int year,int month)
{
	int days=NongMonthIsLarge(year,month)?30:29;
	return days;
}

//获取农历月的天数
int  CCalendarConvert::GetNongMonthDays(int year,int month,BOOL m_run)
{
	int days=0;
	if(m_run)
		days=GetNongRunYueDays(year);
	else
		days=GetNongMonthDays(year,month);
	return days;
}

//获取公历年的天数
int  CCalendarConvert::GetGongYearDays(int year)
{
	int days=YearIsRunNian(year)?366:365;
	return days;
}

//获取农历年的天数
int  CCalendarConvert::GetNongYearDays(int year)
{
	int days=0;
	for(int i=1;i<13;i++)
		days+=GetNongMonthDays(year,i);
	days+=GetNongRunYueDays(year);
	return days;
}

//获取农历年闰月天数
int  CCalendarConvert::GetNongRunYueDays(int year)
{
	if(GetNongRunYue(year)==0)
		return 0;
	int days=RunYueIsLarge(year)?30:29;
	return days;
}

//获取农历闰月
int  CCalendarConvert::GetNongRunYue(int year)
{
	return (CalendarData[year-m_minyear][2] & 0x0f);
}

//公历日期到元旦的天数
int  CCalendarConvert::DaysFromNewYear(CALENDAR m_day)
{
	int days=0;
	for(int i=1;i<m_day.month;i++)
		days+=GetGongMonthDays(m_day.year,i);
	days+=m_day.day;
	return days;
}

//农历日期到春节的天数
int  CCalendarConvert::DaysFromSpringDay(CALENDAR m_day)
{
	int days=0;
	int month=GetNongRunYue(m_day.year);//处理闰月
	if(month<m_day.month)			 //闰月小于日期月份
		days+=GetNongRunYueDays(m_day.year);
	else
	{
		if((month==m_day.month) && m_day.isrunyue)//日期月份是闰月
			days+=GetNongRunYueDays(m_day.year);
	}
	for(int i=1;i<m_day.month;i++)
		days+=GetNongMonthDays(m_day.year,i);
	days+=m_day.day;
	return days;
}

//公历年是否闰年
BOOL CCalendarConvert::YearIsRunNian(int year)
{
	return (CalendarData[year-m_minyear][0] & 0x80);
}


//判断闰月是否为大月
BOOL CCalendarConvert::RunYueIsLarge(int year)
{
	return (CalendarData[year-m_minyear][0]&0x40);
}


//计算2的N次方
int  CCalendarConvert::Cal2N(int n)
{
	int ret=1;
	for(int i=0;i<n;i++)
		ret*=2;
	return ret;
}

//判断农历年的月份是否为大月
BOOL CCalendarConvert::NongMonthIsLarge(int year,int month)
{
	BOOL ret=FALSE;
	if(month<9)
	{
		if(CalendarData[year-m_minyear][1] & Cal2N(8-month))
			ret=TRUE;
	}
	else
	{
		unsigned char ch=Cal2N(12-month);
		ch<<=4;
		if(CalendarData[year-m_minyear][2] & ch)
			ret=TRUE;
	}
	return ret;
}

//判断公历月份是否为大月(二月除外)
BOOL CCalendarConvert::GongMonthIsLarge(int month)
{
	BOOL ret=FALSE;
	if(month<8)
	{
		if(month%2)
			ret=TRUE;
	}
	else
	{
		if(!(month%2))
			ret=TRUE;
	}
	return ret;
}

//农历大年初一到元旦的天数
int  CCalendarConvert::GetDaysFromStart(int year)
{
	return (CalendarData[year-m_minyear][0]&0x3f);
}


/*********暂时没有用到的成员函数***********/
//判断是否是闰月
BOOL CCalendarConvert::MonthIsRunYue(int year,int month)
{
	return (month==GetNongRunYue(year));
}

//获取公历两个日期之间的天数
int  CCalendarConvert::GetGongDays(CALENDAR m_start,CALENDAR m_end)
{
	int days=0;
	for(int i=m_start.year;i<m_end.year;i++)
		days+=GetGongYearDays(i);
	days-=DaysFromNewYear(m_start);
	days+=DaysFromNewYear(m_end);
	return days;
}

//获取农历两个日期之间的天数
int  CCalendarConvert::GetNongDays(CALENDAR m_start,CALENDAR m_end)
{
	int days=0;
	for(int i=m_start.year;i<m_end.year;i++)
		days+=GetNongYearDays(i);
	days-=DaysFromSpringDay(m_start);
	days+=DaysFromSpringDay(m_end);
	return days;
}

//将当前公历日期合成为结构
CALENDAR CCalendarConvert::GetCurGongDate()
{
	CALENDAR m_cal;
	SYSTEMTIME m_time;
	GetLocalTime(&m_time);
	m_cal.year=(int)m_time.wYear;
	m_cal.month=(int)m_time.wMonth;
	m_cal.day=(int)m_time.wDay;
	m_cal.week=(int)m_time.wDayOfWeek;
	return m_cal;
}

//比较两个日期的大小
int  CCalendarConvert::CompareTwoDate(CALENDAR m_fir,CALENDAR m_sec)
{
	int m_state=0;

	if(m_fir.year>m_sec.year)
		m_state=1;
	else
	{
		if(m_fir.year<m_sec.year)
			m_state=-1;
		else
		{
			if(m_fir.month>m_sec.month)
				m_state=1;
			else
			{
				if(m_fir.month<m_sec.month)
					m_state=-1;
				else
				{
					if(m_fir.day>m_sec.day)
						m_state=1;
					else
					{
						if(m_fir.day<m_sec.day)
							m_state=-1;
						else
							m_state=0;
					}
				}
			}
		}
	}

	return m_state;
}

//获取60年中的第N年的天干地支名称
CString CCalendarConvert::GetNNameIn60(int index)
{
	char ShengXiao[25]="鼠牛虎兔龙蛇马羊猴鸡狗猪";
	char TianGan[21]="甲乙丙丁戊己庚辛壬癸";
	char DiZhi[25]="子丑寅卯辰巳午未申酉戌亥";
	char buffer[20];
	memset(buffer,0,20);
	strcpy(buffer,"农历");
	
	int m_cur=0,m_this=0;
	int tian=0,di=0;
	for(int i=0;i<60;i++)
	{
		tian=i%10;
		di=i%12;
		if(m_this==index)
		{
			strncpy(&buffer[4],&TianGan[tian*2],2);
			strncpy(&buffer[6],&DiZhi[di*2],2);
			strcat(buffer,"年,");
			strncpy(&buffer[12],&ShengXiao[di*2],2);
			strcat(buffer,"年");
		}
		m_this++;
	}
	return CString(buffer);
}


//获取农历年份的干支名称
CString CCalendarConvert::GetGanZhi(int m_nongyear)
{	   
	int m_index=(m_nongyear-1924)%60;
	return GetNNameIn60(m_index);
}

//将农历年的月份注入组合框
void CCalendarConvert::NongMonthToList(int m_nongyear,CComboBox & m_list)
{
	m_list.ResetContent();
	int m_run=GetNongRunYue(m_nongyear);
	char m_buffer[12][7]={"一月","二月","三月","四月","五月","六月","七月","八月","九月","十月","十一月","十二月"};
	if(m_run==0)
	{
		for(int i=0;i<12;i++)
			m_list.AddString(m_buffer[i]);
	}
	else	
	{
		for(int i=0;i<m_run;i++)
			m_list.AddString(m_buffer[i]);
		char m_name[10]="闰";
		strcat(m_name,m_buffer[i-1]);
		m_list.AddString(m_name);
		for(i=m_run;i<12;i++)
			m_list.AddString(m_buffer[i]);
	}
}

⌨️ 快捷键说明

复制代码 Ctrl + C
搜索代码 Ctrl + F
全屏模式 F11
切换主题 Ctrl + Shift + D
显示快捷键 ?
增大字号 Ctrl + =
减小字号 Ctrl + -