《电子尺》V1.02程序开发实例

王朝vc·作者佚名  2006-01-17
窄屏简体版  字體: |||超大  

《电子尺》V1.02程序开发实例

作者:电脑驿站 hcl

下载本文源程序

程序功能

有时在制作网页或一些多媒体时,需要插入一些自制的图片和flash动画,在制作之前一定需要先确定图片的高和宽,用这个软件就可以轻松的量出你所需要的高和宽。

总体介绍

程序在开始测量时要锁定整个屏幕,包括任务栏等。原先计划利用钩子(Hook)来截取所有的鼠标消息,实现屏幕的锁定。但是无论使用WH_MOUSE或WH_GETMESSGAE都无法完全截获所有消息。所以我就利用了一个占据整个屏幕的透明窗口来实现。虽然是透明的窗口,但是一旦窗口创建以后,实际屏幕的更新就不会再对窗口中显示的内容进行影响了。

在开始测量时,有一个跨这个屏幕的大十字随着鼠标移动,来辅助定位。在单击第一个点后,会出现一个小的红十字来做标记,如下图所示:

代码分析

首先要创建一个透明的窗口,因此我从CWnd继承了一个类Target。在Target类中自定义了一个创建透明窗口的方法:

void Target::CreateTarget(LPCTSTR lpTitle, CWnd* pWnd)

{

//取得屏幕的高和宽,用于创建跨整个屏幕的窗口

cxScreen=::GetSystemMetrics(SM_CXSCREEN);

cyScreen=::GetSystemMetrics(SM_CYSCREEN);

//用CWnd::CreateEx创建一个透明的窗口,WS_EX_TOPMOST使窗口总是在最顶层

CreateEx(WS_EX_TOPMOST,

AfxRegisterWndClass(0,AfxGetApp()-LoadStandardCursor(IDC_ARROW)),

"Target", WS_POPUP, 0, 0, cxScreen, cyScreen,

NULL, NULL, NULL );

//pDC用于开始测量时绘制辅助标志

pDC=GetDC();

//bSecond用于标识是否已经点击了一次

bSecond=FALSE;

//pWndParent保存父窗口的指针

pWndParent=pWnd;

//创建一个MemDC临时存放整个屏幕的画面,用于刷新屏幕

MemDC.CreateCompatibleDC(pDC);

CBitmap Bitmap;

Bitmap.CreateCompatibleBitmap(pDC,cxScreen,cyScreen);

//确定MemDC的大小

MemDC.SelectObject(&Bitmap);

//将这个屏幕的都存入MemDC

MemDC.BitBlt(0,0,cxScreen,cyScreen,pDC,0,0,SRCCOPY);

//将临时图片删除

::DeleteObject(Bitmap.m_hObject);

}

重载Target类的鼠标移动的消息处理函数,使鼠标移动时,有一个十字跟随移动,并且在已经点击了第一个点以后,会有一条链接起点与终点的线。

void Target::OnMouseMove(UINT nFlags, CPoint point)

{

//首先将MemDC中的图片复制到当前窗口,将原有的辅助线都掩盖掉

pDC-BitBlt(0,0,cxScreen,cyScreen,&MemDC,0,0,SRCCOPY);

//画一个新的十字

pDC-MoveTo(0,point.y);

pDC-LineTo(cxScreen,point.y);

pDC-MoveTo(point.x,0);

pDC-LineTo(point.x,cyScreen);

//如果已经点击过一次,再画一条从起点到终点的辅助线

if(bSecond)

{

pDC-MoveTo(startPos.x,startPos.y);

pDC-LineTo(point.x,point.y);

}

CWnd::OnMouseMove(nFlags, point);

}

重载鼠标左击的消息处理函数,两次单击后向父窗口发送一个自定义的消息WM_ENDCLICK

void Target::OnLButtonDown(UINT nFlags, CPoint point)

{//如果第一次按左击

if(bSecond==FALSE)

{

//记录按下的坐标

startPos=point;

//改变标记

bSecond=TRUE;

//恢复原来的屏幕

pDC-BitBlt(0,0,cxScreen,cyScreen,&MemDC,0,0,SRCCOPY);

//定义一个红色的笔

CPen pen(PS_SOLID,2,RGB(255,0,0));

CPen* pOldPen;

//选入红色的笔,并且记录原来的笔

pOldPen=pDC-SelectObject(&pen);

//画一个红色的标记

pDC-MoveTo(point.x-10,point.y);

pDC-LineTo(point.x+10,point.y);

pDC-MoveTo(point.x,point.y-10);

pDC-LineTo(point.x,point.y+10);

//装入原来的笔,用于在其他辅助线

pDC-SelectObject(pOldPen);

//删除红色的笔

::DeleteObject(pen.m_hObject);

//将带有红色标记的屏幕图片保存到MemDC中

MemDC.BitBlt(0,0,cxScreen,cyScreen,pDC,0,0,SRCCOPY);

}

else//如果第二次单击

{

//记录终点坐标

endPos=point;

//调用计算长度的函数

Calculate();

//将当前DC和临时内存DC删除

pDC-DeleteDC();

MemDC.DeleteDC();

//撤销窗口

DestroyWindow();

//向父窗口发送一个自定义的WM_ENDCLICK

pWndParent-PostMessage(WM_ENDCLICK);

}

CWnd::OnLButtonDown(nFlags, point);

}

重载鼠标右击的消息处理函数,测量时按下右键就取消测量,向父窗口发送WM_CANCELCLICK自定义消息。

void Target::OnRButtonUp(UINT nFlags, CPoint point)

{

CWnd::OnRButtonUp(nFlags, point);

//删除不用的DC

pDC-DeleteDC();

MemDC.DeleteDC();

DestroyWindow();

pWndParent-PostMessage(WM_CANCELCLICK);

}

Target类中的Calculate方法用于计算不同单位的长度。不同的单位主要取决于SetMapMode的参数。

void Target::Calculate()

{

//x、y用于记录两点坐标差

double x,y;

x=endPos.x-startPos.x;

y=endPos.y-startPos.y;

//默认情况下,计算的是像素的单位

iLen_p=(int)sqrt(x*x + y*y);

CDC* pDC;

pDC=GetDC();

//将当前窗口的映射模式改为MM_LOMETRIC,这样逻辑坐标的单位为0.1mm

pDC-SetMapMode(MM_LOMETRIC);

POINT tmpStart=startPos;

POINT tmpEnd=endPos;

//将设备坐标改为逻辑坐标

pDC-DPtoLP(&tmpStart);

pDC-DPtoLP(&tmpEnd);

x=tmpEnd.x-tmpStart.x;

y=tmpEnd.y-tmpStart.y;

dLen_m=sqrt(x*x + y*y);

//映射模式改为MM_LOENGLISH的话,逻辑坐标的单位为0.01inch

pDC-SetMapMode(MM_LOENGLISH);

tmpStart=startPos;

tmpEnd=endPos;

pDC-DPtoLP(&tmpStart);

pDC-DPtoLP(&tmpEnd);

x=tmpEnd.x-tmpStart.x;

y=tmpEnd.y-tmpStart.y;

dLen_i=sqrt(x*x + y*y);

ReleaseDC(pDC);

}

这样整个Target类就定义完了,接着再主对话框中调用该类,首先将Target类的对象作为自己的一个成员变量。在主对话框类中最重要的就是接受两个自定义的消息WM_ENDCLICK和WM_CANCELCLICK,分别表示结束测量和取消测量。

具体程序详见源程序,这里就不再说明了。

 
 
 
免责声明:本文为网络用户发布,其观点仅代表作者个人观点,与本站无关,本站仅提供信息存储服务。文中陈述内容未经本站证实,其真实性、完整性、及时性本站不作任何保证或承诺,请读者仅作参考,并请自行核实相关内容。
 
 
© 2005- 王朝網路 版權所有 導航