在上层软件编程中,往往须要提供一个月历控件让用户选择对应日期或者用此月历控件来强调特定的一天。
MFC的 Month Calendar Control 控件自系统升级到 Windows 7 之后,对于日期控件所能做的操作变少了,不能对指定的日期的背景颜色进行更改。在网上查了非常久之后发现是操作系统画图风格的关系。
在这样的情况下。
仅仅能通过加粗特定日期的字体粗细来凸显该日期的不同(至少本人仅仅想到这个办法)。
首先拉入 Month Calendar Control 控件,为控件加入一个变量叫做 monthcontrol。将控件的 Day States 和 No Trailing Dates 属性置为 True。
在project的初始化代码中增加例如以下代码:
int nCount;
SYSTEMTIME timeFrom;
SYSTEMTIME timeUntil;
nCount = monthcontrol.GetMonthRange(&timeFrom, &timeUntil, GMR_DAYSTATE);//返回月历显示的月份数,将控件的No Trailing Dates属性设为True之后,一个页面仅仅显示一个月份
if (nCount == 1)
{
LPMONTHDAYSTATE pDayState;//日期状态结构体
pDayState = new MONTHDAYSTATE[nCount];
memset(pDayState, 0, sizeof(MONTHDAYSTATE)* nCount);
pDayState[0] |= 1 << (10 - 1);//设置本月的10日要加粗
VERIFY(monthcontrol.SetDayState(nCount, pDayState));//设置本月的10日加粗
delete[]pDayState;
}
这时候编译运行project就会看到月历的10号被加粗了,可是一旦点击月份切换,其它月份的页面就会有非常多莫名其妙的被加粗的日期。本人採取的办法是在控件的MCN_VIEWCHANGE控件事件中加入响应函数,让一个计时器启动,例如以下:
timerun = true;//计时器启动标志
SetTimer(0, 10, 0);//计时器启动
加入对话框的 WM_TIMER 消息
int nCount;
SYSTEMTIME timeFrom;
SYSTEMTIME timeUntil;
nCount = monthcontrol.GetMonthRange(&timeFrom, &timeUntil, GMR_DAYSTATE);//返回月历显示的月份数,将控件的No Trailing Dates属性设为True之后,一个页面仅仅显示一个月份
if (nCount == 1)
{
LPMONTHDAYSTATE pDayState;//日期状态结构体
pDayState = new MONTHDAYSTATE[nCount];
memset(pDayState, 0, sizeof(MONTHDAYSTATE)* nCount);
pDayState[0] |= 1 << (10 - 1);//设置本月的10日要加粗
VERIFY(monthcontrol.SetDayState(nCount, pDayState));//设置本月的10日加粗
delete[]pDayState;
if (timerun == true)
{
timerun = false;
KillTimer(0);
}
}
在控件的MCN_SELCHANGE控件事件中也加入响应函数,例如以下:
int nCount;
SYSTEMTIME timeFrom;
SYSTEMTIME timeUntil;
nCount = monthcontrol.GetMonthRange(&timeFrom, &timeUntil, GMR_DAYSTATE);//返回月历显示的月份数,将控件的No Trailing Dates属性设为True之后,一个页面仅仅显示一个月份
if (nCount == 1)
{
LPMONTHDAYSTATE pDayState;//日期状态结构体
pDayState = new MONTHDAYSTATE[nCount];
memset(pDayState, 0, sizeof(MONTHDAYSTATE)* nCount);
pDayState[0] |= 1 << (10 - 1);//设置本月的10日要加粗
VERIFY(monthcontrol.SetDayState(nCount, pDayState));//设置本月的10日加粗
delete[]pDayState;
}
到此。全部月份页面中的10号就都加粗了,而不影响别的日期。假设稍加留意会发现上面代码的反复率很高。为看起来简洁。能够用一个函数来封装一下。这样能大量缩减代码。
假设结合对于目录和文件的操作。能够实现搜索全部盘符指定目录中指定后缀文件。然后将月历中跟文件的改动日期同样的日期加粗的功能。代码例如以下:
void CDataTestDlg::selectfile()
{
WCHAR Drive[MAX_PATH] = { 0 };
int i = 0;
GetLogicalDriveStrings(MAX_PATH, Drive);
CString myDrive;
CString strDBPath;
bool dayset[33] = { 0 };
int datlong = 0;
int nCount;
int month = 0;
while (Drive[i - 1] != '\0' || Drive[i] != '\0')
{
myDrive = L"";
myDrive = Drive[i];
myDrive = myDrive + L":\\";
strDBPath = myDrive + L"h264视频";//搜索全部盘符以下的名为“h264视频”的目录
HTREEITEM hItem, hSubItem;
if (PathFileExists(strDBPath))//推断目录是否存在
{
CFileFind finder;
vector<CString> allfile;
vector<CString> allfile1;
CString mydrivewholefilepath;
mydrivewholefilepath = strDBPath + L"\\*.*";
bool bworking = finder.FindFile(mydrivewholefilepath);
while (bworking)
{
bworking = finder.FindNextFile();
allfile.push_back(finder.GetFileName());//全部文件。目录的名称
}
for (int i = 0; i<allfile.size(); i++)
{
int flag = allfile[i].Find(L".h264");//找目录以下后缀为“.h264”的文件
if (flag >= 0)
{
CString myfilepath;
myfilepath = strDBPath + L"\\" + allfile[i];
//获得指定路径文件的改动日期
SYSTEMTIME time;
CStdioFile file;
file.Open(myfilepath, CFile::modeRead);
FILETIME file_time;
GetFileTime(file.m_hFile, NULL, NULL, &file_time);
FileTimeToSystemTime(&file_time, &time);
SYSTEMTIME timeFrom;
SYSTEMTIME timeUntil;
nCount = monthcontrol.GetMonthRange(&timeFrom, &timeUntil, GMR_DAYSTATE);
if (timeFrom.wYear == time.wYear && timeFrom.wMonth == time.wMonth)
{
month = timeFrom.wMonth;
datlong = timeUntil.wDay - timeFrom.wDay;
dayset[time.wDay] = true;
}
file.Close();
}
}
finder.Close();
}
i += 4;
}
if (nCount == 1)
{
LPMONTHDAYSTATE pDayState;//日期状态结构体
pDayState = new MONTHDAYSTATE[nCount];
memset(pDayState, 0, sizeof(MONTHDAYSTATE)* nCount);
for (int i = 0; i <= datlong + 1; i++)
{
if (dayset[i] == true)
{
pDayState[0] |= 1 << (i - 1);
}
}
VERIFY(monthcontrol.SetDayState(nCount, pDayState));
delete[]pDayState;
}
}