StretchBlt函数直接对图片进行放大,缩小,显示位置变换。
这个函数有两种形态一种全局函数是这样的: 
BOOL StretchBlt(HDC hdcDest, int nXOriginDest, int nYOriginDest, int nWidthDest, int nHeightDest, HDC hdcSrc, int nXOriginSrc, int nYOriginSrc, int nWidthSrc, int nHeightSrc, DWORD dwRop);

一种是CDC的成员函数是这样的:

BOOL StretchBlt( int x, int y, int nWidth, int nHeight, CDC* pSrcDC, int xSrc, int ySrc, int nSrcWidth, int nSrcHeight, DWORD dwRop );

在实现时有两点要注意

  1. 显示模式,使用SetStretchBltMode来控制像素在拉伸,缩小时的像素运算。在拉伸位图时一般不需要修改,在压缩位图显示时,默认值会出现颜色失真,这时可以使用HALFTONE模式。
  2. 在StrectchBlt中xSrc,ySrc为拷贝设备的逻辑坐标,在拷贝位图后在边缘是有一个逻辑单位的空白的,这点在放大图片对xSrc,ySrc进行设置容易被忽略。

目的描述

  1. 显示位图,鼠标点击图片,以所点击图片的位置为中心按照一定比例系数对图片显示进行放大缩小操作。
  2. 移动位图显示窗口,按下鼠标右键可以拖动位图的显示位置,对位图的显示区进行移动操作。

思路

  通过WM_LBUTTONDOWN或WM_LBUTTONCLICK来获取鼠标的动作的坐标,通常是CClientDC的逻辑坐标,如果是采用文档模式则一般CClientDC和显示窗口是一致的。如果采用PictureControl在窗口应用程序当中显示时,可以先获得PictureControl区域的屏幕坐标再转换成CClientDC的坐标。经过PicturenControl左上角坐标的计算,就可以得到鼠标点击位置在PictureControl区域中的逻辑坐标。再通过该坐标计算出在原设备区中的逻辑坐标,通过该逻辑坐标就可以计算出鼠标所在点在位图上的坐标。知道该位图上坐标后,就可以对原设备区进行拉伸,缩小,移动的操作了。

实现

放大,缩小

CStatic m_sPreview;//PictureControl 控件
CPoint m_CurClientPoint;//当前鼠标位置所在客户区的逻辑坐标
int m_iMultiple;//图像缩放显示倍数
enum PreviewType{enlarge,narrow,restore,move};
PreviewType m_nPrevewType;//命令模式
CPoint m_OriginSrcPoint;//显示图像右上角坐标
unsigned int iWidth,iHeight;//iWidth与iHeight是图片的横纵像素的数目
...
ClientToPictureControl(&m_sPreview,&m_CurClientPoint);
PictureControlToBmp(m_nWidthDest,m_nHeightDest,m_nWidthSrc,m_nHeightSrc,&m_OriginSrcPoint,&m_CurClientPoint);
m_nWidthSrc=iWidth/m_iMultiple;
m_nHeightSrc=iHeight/m_iMultiple;
Mutiply(m_nPreviewType,&m_OriginSrcPoint,&m_CurClientPoint,m_nWidthSrc,m_nHeightSrc);
m_nYOriginSrc=m_OriginSrcPoint.y;//
m_nXOriginSrc=m_OriginSrcPoint.x;//
函数ClientToPictureControl功能为得到该点再控件坐标系的。
void ClientToPictureControl(CStatic *picControl,CPoint *point)//得到picControl在客户区的逻辑位置
{
CWnd *pWnd;
pWnd=picControl;
CRect tmp;
pWnd->GetWindowRect(&tmp);//控件窗口位置
ScreenToClient(&tmp);
*point = *point - tmp.TopLeft();
}
函数PictureControlToBmp功能是为得到鼠标位置在图像坐标系下的像素坐标 
void PictureControlToBmp(int nWidthDest,int nHeightDest,int nWidthSrc,int nHeightSrc,
CPoint *Origin,CPoint *point)
{
  point->x = point->x * nWidthSrc / nWidthDest+ Origin->x;
  point->y= point->y * nHeightSrc / nHeightDest+ Origin->y;
}
函数Mutiply//是调整显示右上角坐标的。
void Mutiply(PreviewType type,CPoint *Origin,CPoint *point, int width,int height)
{
switch (type)
{
case CAlphaDlg::enlarge:
if(point->x-width/>)
{
if(m_pMyDevice->m_sFrInfo.iWidth > point->x+width/){
Origin->x=point->x-width/;}
else{
Origin->x=m_pMyDevice->m_sFrInfo.iWidth-width;
}
}
else
{
Origin->x=;
}
if(point->y-height/>)
{
if(m_pMyDevice->m_sFrInfo.iHeight> point->y+height/){
Origin->y=point->y-height/;}
else{
Origin->y=m_pMyDevice->m_sFrInfo.iHeight-height;
}
}
else
{
Origin->y=;
}
break;
case CAlphaDlg::narrow:
if(point->x-width/>)
{
if(m_pMyDevice->m_sFrInfo.iWidth > point->x+width/){
Origin->x=point->x-width/;}
else{
Origin->x=m_pMyDevice->m_sFrInfo.iWidth-width;
}
}
else
{
Origin->x=;
}
if(point->y-height/>)
{
if(m_pMyDevice->m_sFrInfo.iHeight> point->y+height/){
Origin->y=point->y-height/;}
else{
Origin->y=m_pMyDevice->m_sFrInfo.iHeight-height;
}
}
else
{
Origin->y=;
}
break;
case CAlphaDlg::restore:
Origin->x=;
Origin->y=;
break;
case CAlphaDlg::move:
break;
default:
break;
}
}

以上,是图像放大缩小的步骤。

如果移动图像和上面类似,直接响应WM_MOUSEMOVE消息

void OnMouseMove(UINT nFlags, CPoint point)
{
if(m_bTool==TRUE&&m_nPreviewType==move)
{SetCursor(LoadCursor(NULL,IDC_SIZEALL));
CRect rect;
switch (nFlags)
{
case MK_LBUTTON:
if(m_CurClientPoint!=point)
{
if(m_nXOriginSrc+(-point.x+m_CurClientPoint.x)/m_iMultiple>)
{
if(m_nXOriginSrc+(-point.x+m_CurClientPoint.x)/m_iMultiple + m_nWidthSrc< m_pMyDevice->m_sFrInfo.iWidth)
m_nXOriginSrc=m_nXOriginSrc+(-point.x+m_CurClientPoint.x)/m_iMultiple;
}
else
{
m_nXOriginSrc=;
}
if(m_nYOriginSrc+(-point.y+m_CurClientPoint.y)/m_iMultiple>)
{
if(m_nYOriginSrc+(-point.y+m_CurClientPoint.y)/m_iMultiple + m_nHeightSrc< m_pMyDevice->m_sFrInfo.iHeight)
m_nYOriginSrc=m_nYOriginSrc+(-point.y+m_CurClientPoint.y)/m_iMultiple;
}
else
{
m_nYOriginSrc=;
}
m_CurClientPoint = point ;
}
m_sPreview.GetClientRect(&rect);
InvalidateRect(rect,);
break;
default:
break;
}
}

MFC中显示图像的放大、缩小、移动功能的更多相关文章

  1. JS中图片的放大缩小没反应

    这段代码无反应: 代码如下: <script type="text/javascript"> onload = function () { document.getEl ...

  2. 不使用CvvImage类来在MFC中显示图像

    /* * ===================================================================================== * * Filen ...

  3. canvas上画出坐标集合,并标记新坐标,背景支持放大缩小拖动功能

    写在前面:项目需求,用户上传一个区位的平面图片,用户可以在图片上添加新的相机位置,并且展示之前已绑定的相机坐标位置,图片支持放大缩小&拖动的功能.新增坐标,页面展示相对canvas定位,保存时 ...

  4. html5在微信中不允许放大缩小页面

    在头部添加 <meta name="viewport" content="width=device-width, initial-scale=1, maximum- ...

  5. MFC中实现定时执行与提醒功能(自编代码)

    具体实现代码如下:添加一个计时器:SetTimer(1,1000,NULL); 下面仅列举核心代码,详细步聚不作说明,效果如下所示: void CShowTimer::OnTimer(UINT_PTR ...

  6. h5在微信中不允许放大缩小页面

    在头部添加 <meta name="viewport" content="width=device-width, initial-scale=1, maximum- ...

  7. 鼠标滚轮图片放大缩小功能,使用layer弹框后不起作用

    今天在项目中遇到的一个问题:点击按钮使用layer弹框弹出一张图片,需要加一个鼠标滚轮放大缩小,图片也跟着放大缩小的功能.于是在网上找了一个demo. DEMO: <!DOCTYPE html ...

  8. Android多点触控技术,实现对图片的放大缩小平移,惯性滑动等功能

    首先推荐一下鸿洋大大的打造个性的图片预览与多点触控视频教程,这套教程教我们一步一步实现了多点触控实现对图片的平移和缩放的功能.这篇文章我将在鸿洋大大的基础之上做了一些扩展功能: 1.图片的惯性滑动 2 ...

  9. 自定义mousewheel事件,实现图片放大缩小功能实现

    本文是承接 上一篇的<自定义鼠标滚动事件>  的基础上实现的,建议大家先看一下上一篇的mousewheel的实现,再浏览下文: 上篇中我们介绍到: $element.mousewheel( ...

随机推荐

  1. 记一次ceph集群的严重故障

    问题:集群状态,坏了一个盘,pg状态好像有点问题[root@ceph-1 ~]# ceph -s    cluster 72f44b06-b8d3-44cc-bb8b-2048f5b4acfe     ...

  2. 洛谷P3168 [CQOI2015]任务查询系统 [主席树,差分]

    题目传送门 任务查询系统 题目描述 最近实验室正在为其管理的超级计算机编制一套任务管理系统,而你被安排完成其中的查询部分.超级计算机中的任务用三元组(Si,Ei,Pi)描述,(Si,Ei,Pi)表示任 ...

  3. angular4 监听input框输入值的改变

    angular中一般控件会有change事件,但是如果某些控件没有这个事件 我们如何监听值的变化呢? 对于双向绑定的值,当值改变后监听事件有如下写法: 1. 如果是ngModel可以用ngModelC ...

  4. Mybatis源码分析之参数处理

    Mybatis对参数的处理是值得推敲的,不然在使用的过程中对发生的一系列错误直接懵逼了. 以前遇到参数绑定相关的错误我就是直接给加@param注解,也稀里糊涂地解决了,但是后来遇到了一些问题推翻了我的 ...

  5. JSP中动态INCLUDE与静态INCLUDE的区别?

    动态INCLUDE用jsp:include动作实现 它总是会检查所含文件中的变化,适合用于包含动态页面,并且可以带参数 静态INCLUDE用include伪码实现,定不会检查所含文件的变化,适用于包含 ...

  6. 【BZOJ 3482】 3482: [COCI2013]hiperprostor (dij+凸包)

    3482: [COCI2013]hiperprostor Time Limit: 20 Sec  Memory Limit: 256 MBSubmit: 277  Solved: 81 Descrip ...

  7. luoguP3232 [HNOI2013]游走 贪心 + 概率期望 + 高斯消元

    首先,题目中的无向简单连通图代表着没有自环,重边... 总分的期望 = 每条边的期望之和...................每条边的期望又可以拆成$u \to v$的期望和$v \to u$的期望 ...

  8. Linux 自动化部署

    1.pexpect Pexpect 是 Don Libes 的 Expect 语言的一个 Python 实现,是一个用来启动子程序,并使用正则表达式对程序输出做出特定响应,以此实现与其自动交互的 Py ...

  9. l1和l2正则化的区别 - 面试错题集

    L0:计算非零个数,用于产生稀疏性,但是在实际研究中很少用,因为L0范数很难优化求解,是一个NP-hard问题,因此更多情况下我们是使用L1范数L1:计算绝对值之和,用以产生稀疏性,因为它是L0范式的 ...

  10. Codeforces Round #302 (Div. 2) B. Sea and Islands 构造

    B. Sea and Islands Time Limit: 20 Sec  Memory Limit: 256 MB 题目连接 http://codeforces.com/contest/544/p ...