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. [转]基于Protel DXP软件的PCB高级编辑技巧大全

    来源:基于Protel DXP软件的PCB高级编辑技巧大全 一.放置坐标指示 放置坐标指示可以显示出PCB板上任何一点的坐标位置. 启用放置坐标的方法如下:从主菜单中执行命令 Place/Coordi ...

  2. bzoj 1452 二维树状数组

    #include<bits/stdc++.h> #define LL long long #define fi first #define se second #define mk mak ...

  3. Nodejs JSON.parse()无法解析ObjectID和ISODate的问题

    一个早上搞清楚了一个问题,关于Nodjes JSON.parse()方法只能解析字符串.布尔值.数字等,但不能解析ObjectID及ISODate的值 原因:<How to handle Obj ...

  4. 洛谷P3857 [TJOI2008]彩灯 [线性基]

    题目传送门 彩灯 题目描述 Peter女朋友的生日快到了,他亲自设计了一组彩灯,想给女朋友一个惊喜.已知一组彩灯是由一排N个独立的灯泡构成的,并且有M个开关控制它们.从数学的角度看,这一排彩灯的任何一 ...

  5. Python学习之argparse

    http://www.2cto.com/kf/201412/363654.html https://docs.python.org/3.4/howto/argparse.html# 一.简介: arg ...

  6. 快速配置webpack多入口脚手架

    背景 当我们基于vue开发单个项目时,我们会init一个vue-cli,但当我们想在其他项目里共用这套模板时,就需要重新init一个,或者clone过来,这非常不方便,而且当多人开发时,我们希望所有的 ...

  7. Shiro切入Spring的方式

    在springMVC中要使用shiro,一般都遵循下面的配置: applicationContext-shiro.xml <bean id="shiroFilter" cla ...

  8. Docker应用系列(四)| 部署java应用

    本示例基于Centos 7,假设目前使用的账号为release,拥有sudo权限. 由于Docker官方镜像下载较慢,可以开启阿里云的Docker镜像下载加速器,可参考此文进行配置. 主机上服务安装步 ...

  9. chomd 1+2+4

    2,把目录 /tmp/sco修改为可写可读可执行 chmod 777 /tmp/sco 要修改某目录下所有的文件夹属性为可写可读可执行 chmod 777 * 把文件夹名称用*来代替就可以了 要修改/ ...

  10. [BZOJ4591][SHOI2015]超能粒子炮·改(Lucas定理+数位DP)

    大组合数取模可以想到Lucas,考虑Lucas的意义,实际上是把数看成P进制计算. 于是问题变成求1~k的所有2333进制数上每一位数的组合数之积. 数位DP,f[i][0/1]表示从高到低第i位,这 ...