123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212 |
- // CalendarConvert.cpp: implementation of the CCalendarConvert class.
- // Download by http://www.codefans.net
- //////////////////////////////////////////////////////////////////////
- #include "stdafx.h"
-
- #include "CalendarConvert.h"
- #ifdef _DEBUG
- #undef THIS_FILE
- static char THIS_FILE[]=__FILE__;
- #define new DEBUG_NEW
- #endif
- //////////////////////////////////////////////////////////////////////
- // Construction/Destruction
- //////////////////////////////////////////////////////////////////////
- CCalendarConvert::CCalendarConvert()
- {
- }
- CCalendarConvert::~CCalendarConvert()
- {
- }
- //
- // 得到表示农历通用表示的字串
- //
- CString CCalendarConvert::GetLunarString ( IN PSYSTEMTIME pSt, BOOL bEmbolism )
- {
- TCHAR szNongli[30], szNongliDay[10],szShuXiang[10];
- /*天干名称*/
- const char *cTianGan[] = {"甲","乙","丙","丁","戊","己","庚","辛","壬","癸"};
- /*地支名称*/
- const char *cDiZhi[] = {"子","丑","寅","卯","辰","巳","午", "未","申","酉","戌","亥"};
- /*属相名称*/
- const char *cShuXiang[] = {"鼠","牛","虎","兔","龙","蛇","马","羊","猴","鸡","狗","猪"};
- /*农历日期名*/
- const char *cDayName[] = { "*","初一","初二","初三","初四","初五",
- "初六","初七","初八","初九","初十",
- "十一","十二","十三","十四","十五",
- "十六","十七","十八","十九","二十",
- "廿一","廿二","廿三","廿四","廿五",
- "廿六","廿七","廿八","廿九","三十"};
- /*农历月份名*/
- const char *cMonName[] = {"*","正","二","三","四","五","六", "七","八","九","十","十一","腊"};
- /*--生成农历天干、地支、属相 ==> wNongli--*/
- int nShuXiang = ((pSt->wYear - 4) % 60) % 12;
- if ( nShuXiang < 0 || nShuXiang >= sizeof(cShuXiang)/sizeof(cShuXiang[0]) )
- return "Error Year";
- wsprintf(szShuXiang,"%s",cShuXiang[nShuXiang]);
- int nTianGan = ((pSt->wYear - 4) % 60) % 10;
- if ( nTianGan < 0 || nTianGan >= sizeof(cTianGan)/sizeof(cTianGan[0]) )
- return "Error Year";
- int nDiZhi = ((pSt->wYear - 4) % 60) % 12;
- if ( nDiZhi < 0 || nDiZhi >= sizeof(cDiZhi)/sizeof(cDiZhi[0]) )
- return "Error Year";
- wsprintf(szNongli,"%s(%s%s)年",szShuXiang,cTianGan[nTianGan],cDiZhi[nDiZhi]);
-
- /*--生成农历月、日 ==> wNongliDay--*/
- if ( pSt->wMonth < 0 || pSt->wMonth >= sizeof(cMonName)/sizeof(cMonName[0]) )
- return "Error Month";
- if ( bEmbolism )
- {
- wsprintf(szNongliDay,"闰%s",cMonName[pSt->wMonth]);
- }
- else
- {
- strcpy(szNongliDay,cMonName[pSt->wMonth]);
- }
-
- strcat(szNongliDay,"月");
- if ( pSt->wDay < 0 || pSt->wDay >= sizeof(cDayName)/sizeof(cDayName[0]) )
- return "Error Day";
- strcat(szNongliDay,cDayName[pSt->wDay]);
- return strcat(szNongli,szNongliDay);
- }
- //
- // 公历转农历
- //
- BOOL CCalendarConvert::Gregorian2Lunar ( IN PSYSTEMTIME pSt, OUT PSYSTEMTIME pDst, OUT BOOL *pEmbolism )
- {
- if ( !pSt || !pDst || !pEmbolism ) return FALSE;
- SYSTEMTIME Dst = {0};
- /*公历每月前面的天数*/
- const int wMonthAdd[12] = {0,31,59,90,120,151,181,212,243,273,304,334};
- /*农历数据*/
- const int wNongliData[100] = {2635,333387,1701,1748,267701,694,2391,133423,1175,396438
- ,3402,3749,331177,1453,694,201326,2350,465197,3221,3402
- ,400202,2901,1386,267611,605,2349,137515,2709,464533,1738
- ,2901,330421,1242,2651,199255,1323,529706,3733,1706,398762
- ,2741,1206,267438,2647,1318,204070,3477,461653,1386,2413
- ,330077,1197,2637,268877,3365,531109,2900,2922,398042,2395
- ,1179,267415,2635,661067,1701,1748,398772,2742,2391,330031
- ,1175,1611,200010,3749,527717,1452,2742,332397,2350,3222
- ,268949,3402,3493,133973,1386,464219,605,2349,334123,2709
- ,2890,267946,2773,592565,1210,2651,395863,1323,2707,265877};
- static int wCurYear,wCurMonth,wCurDay;
- static int nTheDate,nIsEnd,m,k,n,i,nBit;
- /*---取当前公历年、月、日---*/
- wCurYear = pSt->wYear;
- wCurMonth = pSt->wMonth;
- wCurDay = pSt->wDay;
- /*---计算到初始时间1921年2月8日的天数:1921-2-8(正月初一)---*/
- nTheDate = (wCurYear - 1921) * 365 + (wCurYear - 1921) / 4 + wCurDay + wMonthAdd[wCurMonth - 1] - 38;
- if ( nTheDate < 0 ) return FALSE;
- if((!(wCurYear % 4)) && (wCurMonth > 2))
- nTheDate = nTheDate + 1;
-
- /*--计算农历天干、地支、月、日---*/
- nIsEnd = 0;
- m = 0;
- while(nIsEnd != 1)
- {
- if(wNongliData[m] < 4095)
- k = 11;
- else
- k = 12;
- n = k;
- while(n>=0)
- {
- //获取wNongliData(m)的第n个二进制位的值
- nBit = wNongliData[m];
- for(i=1;i<n+1;i++)
- nBit = nBit/2;
-
- nBit = nBit % 2;
-
- if (nTheDate <= (29 + nBit))
- {
- nIsEnd = 1;
- break;
- }
-
- nTheDate = nTheDate - 29 - nBit;
- n = n - 1;
- }
- if(nIsEnd)
- break;
- m = m + 1;
- }
- wCurYear = 1921 + m;
- wCurMonth = k - n + 1;
- wCurDay = nTheDate;
- if ( wCurDay < 0 ) return FALSE;
- if (k == 12)
- {
- if (wCurMonth == wNongliData[m] / 65536 + 1)
- wCurMonth = 1 - wCurMonth;
- else if (wCurMonth > wNongliData[m] / 65536 + 1)
- wCurMonth = wCurMonth - 1;
- }
-
- Dst.wYear = (USHORT)wCurYear;
- if (wCurMonth < 1)
- {
- Dst.wMonth = -wCurMonth;
- *pEmbolism = TRUE;
- }
- else
- {
- Dst.wMonth = wCurMonth;
- *pEmbolism = FALSE;
- }
- Dst.wDay = wCurDay;
- memcpy ( pDst, &Dst, sizeof(SYSTEMTIME) );
- return TRUE;
- }
- //
- // 农历转公历
- //
- BOOL CCalendarConvert::Lunar2Gregorian ( IN PSYSTEMTIME pSt, IN BOOL bEmbolism, OUT PSYSTEMTIME pDst )
- {
- if ( !pSt || !pDst ) return FALSE;
- COleDateTimeSpan Span(1,0,0,0);
- COleDateTime tStart ( *pSt );
- for ( int i=0; i<100; i++ )
- {
- SYSTEMTIME Src = {0};
- SYSTEMTIME Dst = {0};
- BOOL bDstEmbolism = FALSE;
- if ( !tStart.GetAsSystemTime ( Src ) )
- return FALSE;
- if ( !Gregorian2Lunar ( &Src, &Dst, &bDstEmbolism ) )
- return FALSE;
- if ( Dst.wYear==pSt->wYear && Dst.wMonth==pSt->wMonth && Dst.wDay==pSt->wDay && bEmbolism==bDstEmbolism )
- {
- memcpy ( pDst, &Src, sizeof(SYSTEMTIME) );
- return TRUE;
- }
- tStart+=Span;
- }
- return FALSE;
- }
- COleDateTime CCalendarConvert::ConvertSysTime2OleDateTime(PSYSTEMTIME pSt)
- {
- ASSERT ( pSt );
- COleDateTime t ( *pSt );
- return t;
- }
|