英昌盛
(吉林師范大學(xué)計(jì)算機(jī)學(xué)院,吉林四平 136000)
?
基于MFC的半透明窗口設(shè)計(jì)與實(shí)現(xiàn)
英昌盛
(吉林師范大學(xué)計(jì)算機(jī)學(xué)院,吉林四平 136000)
在對工業(yè)檢測結(jié)果進(jìn)行實(shí)時(shí)顯示過程中,用戶常常需要對檢測的某些數(shù)據(jù)點(diǎn)進(jìn)行觀察和分析,并根據(jù)分析的結(jié)果進(jìn)行相應(yīng)處理。在不影響顯示檢測結(jié)果的同時(shí),檢測軟件還需要實(shí)時(shí)提供被選定的檢測點(diǎn)信息,這些信息不應(yīng)該占用屏幕上固定區(qū)域,而應(yīng)該隨被選定點(diǎn)的變化而浮動(dòng)顯示。本文基于MFC設(shè)計(jì)了跟隨鼠標(biāo)位置浮動(dòng)顯示選定檢測點(diǎn)信息的半透明提示窗口,使軟件更人性化的同時(shí)提高了用戶處理檢測結(jié)果的效率。
實(shí)時(shí);選定;浮動(dòng);半透明
隨著信息化的推進(jìn),光電檢測發(fā)揮著愈來愈重要的作用。光電檢測系統(tǒng)由前端檢測和后端顯示處理兩部分構(gòu)成。檢測部分主要負(fù)責(zé)待檢測內(nèi)容,經(jīng)由光學(xué)系統(tǒng)、光電轉(zhuǎn)換系統(tǒng),將光信號(hào)轉(zhuǎn)換成電信號(hào);然后在系統(tǒng)內(nèi)部經(jīng)放大、模/數(shù)轉(zhuǎn)換之后形成數(shù)字化信號(hào),從而構(gòu)成一維或二維的數(shù)字信號(hào);顯示處理部分則負(fù)責(zé)處理經(jīng)電纜傳遞過來的數(shù)字信號(hào),并以圖形或圖像的形式將檢測結(jié)果呈現(xiàn)給用戶。為了方便用戶處理,并給用戶更好的體驗(yàn),需要將檢測結(jié)果以簡潔明要的形式表現(xiàn)出來,同時(shí)將用戶關(guān)心的信息及時(shí)提供給用戶。一維數(shù)字信號(hào)通常以曲線形式呈現(xiàn),而二維信號(hào)則通常以灰度圖像形式呈現(xiàn)。用戶關(guān)心的通常是某一檢測點(diǎn)的數(shù)據(jù)值或灰度值,可以使用半透明浮動(dòng)窗口的形式動(dòng)態(tài)向用戶呈現(xiàn)所選擇的數(shù)據(jù)點(diǎn)的信息。
1.1 顯示結(jié)果分析
檢測軟件的顯示界面通常如圖1所示。其由框架區(qū)域、顯示檢測結(jié)果對應(yīng)的曲線區(qū)域和浮動(dòng)顯示的動(dòng)態(tài)數(shù)據(jù)區(qū)域等幾個(gè)部分構(gòu)成。
圖1 檢測軟件的通常界面
框架區(qū)域是整個(gè)軟件的顯示容器;檢測結(jié)果對應(yīng)的曲線區(qū)域由坐標(biāo)系統(tǒng)和數(shù)據(jù)曲線兩部分構(gòu)成,若檢測結(jié)果為二維信號(hào)則應(yīng)顯示圖像的灰度信息;浮動(dòng)顯示數(shù)據(jù)區(qū)提供用戶選定的檢測點(diǎn)相關(guān)信息,通常跟隨鼠標(biāo)浮動(dòng)實(shí)時(shí)顯示,同時(shí)為不遮住其下方的曲線及其它內(nèi)容,需以半透明的方式呈現(xiàn)。
根據(jù)檢測軟件常用界面及要求,可以設(shè)計(jì)帶浮動(dòng)信息提示窗口的檢測軟件布局[1],如圖2所示。其由主對話框、顯示曲線的靜態(tài)文本區(qū)域和顯示浮動(dòng)區(qū)域的子對話框構(gòu)成。
圖2 檢測軟件的界面構(gòu)成
圖3 半透明提示窗口的布局
1.2 實(shí)現(xiàn)原理
對于框架,可以使用MFC中的對話框來實(shí)現(xiàn);對于檢測結(jié)果顯示區(qū)域,則需要將MFC中的靜態(tài)文本采用子類化技術(shù)來完成;對于浮動(dòng)區(qū)域,則仍需要使用對話框來實(shí)現(xiàn)。
檢測結(jié)果的可視化分為兩部分:擴(kuò)展CStatic類形成一個(gè)能夠自我重繪的子類,在該子類中完成坐標(biāo)系及相應(yīng)檢測數(shù)據(jù)的繪制工作;MFC中的CStatic控件與該擴(kuò)展子類進(jìn)行關(guān)聯(lián)以實(shí)現(xiàn)子類化,從而完成檢測結(jié)果的顯示及繪制。
對于浮動(dòng)窗口則需要解決兩個(gè)問題:一是半透明;一是實(shí)時(shí)浮動(dòng)顯示相應(yīng)內(nèi)容。對于半透明,可以使用Windows提供的SetLayeredWindowAttributes函數(shù),借助層次化來實(shí)現(xiàn)。對于數(shù)據(jù)的動(dòng)態(tài)顯示及窗口浮動(dòng),則需要建立主對話框與浮動(dòng)對話框之間的數(shù)據(jù)關(guān)聯(lián)及消息響應(yīng)來實(shí)現(xiàn)[2]。
創(chuàng)建基于對話框的MFC應(yīng)用程序,在對話框中刪除原有控件,然后添加一個(gè)靜態(tài)文本控件,同時(shí)添加一個(gè)新的對話框,并將其設(shè)置為無邊框風(fēng)格。
2.1 檢測數(shù)據(jù)顯示實(shí)現(xiàn)
將已經(jīng)編寫好的自定義數(shù)據(jù)顯示類CDataShow類添加到項(xiàng)目中,并在相應(yīng)的文件中添加對其頭文件的引用。CDataShow類主要完成與主對話框類進(jìn)行數(shù)據(jù)交換、繪制坐標(biāo)系統(tǒng)以及曲線的實(shí)時(shí)繪制等工作,同時(shí)還需要與半透明提示窗口時(shí)行消息傳遞。
2.2 半透明窗口實(shí)現(xiàn)原理
在MFC中,可以借助user32.dll提供的SetLayeredWindowAttributes函數(shù)來實(shí)現(xiàn)層次化,通過層次化來創(chuàng)建半透明窗口。用于實(shí)現(xiàn)半透明窗口的對話框不能使用邊框,而且應(yīng)該添加一個(gè)靜態(tài)文本控件用于顯示數(shù)據(jù)點(diǎn)信息,其布局如圖3所示。
2.3 半透明窗口實(shí)現(xiàn)
以子對話框?yàn)榛A(chǔ),創(chuàng)建一個(gè)新的對話框類CMyTextDlg,同時(shí)需要對該類添加重寫虛函數(shù)OnInitDialog。在對話框初始化函數(shù)中,使用SetWindowLong函數(shù)設(shè)置子對話框窗口風(fēng)格[3]為WS_EX_LAYERED(0x00080000);設(shè)置好該風(fēng)格之后,就可以調(diào)用SetLayeredWindowAttributes函數(shù)來透明化窗口。透明化窗口之后,還需要將其設(shè)置為頂層窗體,同時(shí)取消其在任務(wù)欄中的圖標(biāo)顯示。
BOOL CMyTextDlg::OnInitDialog()
{
……
SetWindowLong(this->GetSafeHwnd(),GWL_EXSTYLE,GetWindowLong(this->GetSafeHwnd(),GWL_EXSTYLE)|WS_EX_LAYERED);
……
ModifyStyleEx(WS_EX_APPWINDOW,WS_EX_TOOLWINDOW,SWP_FRAMECHANGED);
::SetWindowPos(m_hWnd,HWND_TOPMOST,0,0,0,0,SWP_NOSIZE|SWP_NOMOVE);
……
}
為在提示窗口中動(dòng)態(tài)顯示提示信息,需要為CMyTextDlg類添加自定義公共函數(shù)SetText。
voidCMyTextDlg::SetText(CStringStr)
{
SetDlgItemText(IDC_INFO,Str);
}
在主對話框中使用子對話框作為提示窗口時(shí),需要在其頭文件中添加對子對話框類的引用,同時(shí)定義與之對應(yīng)的對象指針和標(biāo)志量。
CMyTextDlg *m_pDlgAlpha;
BOOL m_bIsCap;
因用戶感興趣的數(shù)據(jù)均處于檢測結(jié)果的繪制區(qū)域,當(dāng)鼠標(biāo)不在該區(qū)域內(nèi)時(shí)不應(yīng)顯示提示窗口。所以需要捕獲主對話框的鼠標(biāo)按下及移動(dòng)事件,并添加相應(yīng)的消息響應(yīng)函數(shù)。
voidCmyTestDlg::OnLButtonDown(UINT nFlags,CPoint point)
{
m_bIsCap=TRUE;
if(!m_pDlgAlpha)
{
m_pDlgAlpha=new CMyTextDlg;
m_pDlgAlpha->Create(CMyTextDlg::IDD,CWnd::FromHandle(::GetDesktopWindow()));
}
CDialog::OnLButtonDown(nFlags, point);
}
voidCmyTestDlg::OnLButtonUp(UINT nFlags, CPoint point)
{
if(m_pDlgAlpha)
{
m_pDlgAlpha->ShowWindow(SW_HIDE);
deletem_pDlgAlpha;m_pDlgAlpha=NULL;
}
m_bIsCap=FALSE;
CDialog::OnLButtonUp(nFlags, point);
}
voidCmyTestDlg::OnMouseMove(UINT nFlags, CPoint point)
{
……
CStringstr=_T("");
str.Format(_T("坐標(biāo):(%d,%d)"),pnt.x,pnt.y);
ShowTips(color,str);
……
}
為主對話框添加公共函數(shù)void ShowTips(CStringStr),通過子對話框指針設(shè)置其中的提示文本,并根據(jù)文本的高度及寬度動(dòng)態(tài)調(diào)整了對話框的大小。
voidCmyTestDlg::ShowTips(CStringStr )
{
……
HWND DeskHwnd = ::GetDesktopWindow(); //取得桌面句柄
HDC DeskDC= ::GetWindowDC(DeskHwnd);
CDC *pDC=CDC::FromHandle(DeskDC);
CSizesSize=pDC->GetTextExtent(Str);//獲得文字的寬和高
m_pDlgAlpha->MoveWindow(pnt.x+10,pnt.y,sSize.cx,sSize.cy);
m_pDlgAlpha->SetText(Str);
m_pDlgAlpha->ShowWindow(SW_SHOW);
ReleaseDC(pDC);
::ReleaseDC(m_hWnd,DeskDC);
}
經(jīng)實(shí)際測試,檢測軟件在顯示檢測結(jié)果曲線的同時(shí),能夠?qū)崟r(shí)浮動(dòng)顯示用戶選定的檢測數(shù)據(jù)點(diǎn)信息,改善并提高了用戶的使用體驗(yàn)。
[1]聶斐,殷興輝.基于MFC的實(shí)時(shí)數(shù)據(jù)動(dòng)態(tài)顯示界面設(shè)計(jì)[J].電子設(shè)計(jì)工程,2013(10):136-138.
[2]楊剛.基于MFC用戶界面設(shè)計(jì)主、子對話框數(shù)據(jù)的傳遞[J].機(jī)電產(chǎn)品開發(fā)與創(chuàng)新,2005(6):98-99.
[3]孫鑫,余安萍.Visual C++深入詳解[M].北京:電子工業(yè)出版社,2006.
Designing and Implementation of Semitransparent Window Based on MFC
YING Chang-sheng
(Jilin Normal University, Siping Jilin 136000, China)
In the duration of displaying the detected industrial result, users often need to analyze some detected points and do some corresponding process according to the analysis. Detecting software should also provide information of selected detecting points, as well as showing the detecting result. The information should be displayed in a floating region and should not occupy display region. Our software displays the information of selected points in a semitransparent floating window based on MFC, which providing a kindly user interface.
real time; selected; floating; semitransparent
2016-03-29
英昌盛(1979- ),男,講師,從事算法與圖像處理研究。
TP312
A
2095-7602(2016)08-0038-04