CImage通过WinApi的SetWorldTransform来实现图片旋转

SetWorldTransform的功能是旋转画布,这样产生的效果就是图像旋转。因此,在旋转画布之前,要把要旋转的图像的位置和大小准备好,这样旋转之后,才能使图像正好出现在显示区域内。这需要计算两个关键参数,图像的左上角坐标和旋转中心坐标。因为是固定大小旋转,因此我们将中心设定在图像的显示中心。这样需要计算选中图像的高和宽。

如下图:

具体实现方法如下:

cpp 复制代码
void ImageRotation(CImage* dst, const CImage* src, double angle)
{
	// 计算弧度
	angle = angle * PI / 180;

	// 获取图像宽度和高度
	int width = src->GetWidth();
	int height = src->GetHeight();

	// 计算旋转后的图像大小,并调整目标图像尺寸
	int newWidth = static_cast<int>(abs(cos(angle)) * width + abs(sin(angle)) * height);
	int newHeight = static_cast<int>(abs(sin(angle)) * width + abs(cos(angle)) * height);

	dst->Create(newWidth, newHeight, src->GetBPP());

	CPoint centerPt;
	CRect rect;
	rect.SetRect(0, 0, dst->GetWidth(), dst->GetHeight());
	centerPt.x = (rect.left + rect.right) / 2;
	centerPt.y = (rect.top + rect.bottom) / 2;

	// 获取源图像和目标图像的设备上下文对象
	CImageDC hdcSource(*src);
	CImageDC hdcDest(*dst);

	// 设置图形模式
	SetGraphicsMode(hdcDest, GM_ADVANCED);

	// 保存旋转数据的结构体
	XFORM xform;
	xform.eM11 = static_cast<FLOAT>(cos(angle));
	xform.eM12 = static_cast<FLOAT>(sin(angle));
	xform.eM21 = static_cast<FLOAT>(-sin(angle));
	xform.eM22 = static_cast<FLOAT>(cos(angle));
	xform.eDx = (float)(centerPt.x - cos(angle)*centerPt.x + sin(angle)*centerPt.y);
	xform.eDy = (float)(centerPt.y - cos(angle)*centerPt.y - sin(angle)*centerPt.x);

	int nx, ny;
	nx = newWidth / 2 - width / 2;
	ny = newHeight / 2 - height / 2;

	// 进行旋转操作
	SetWorldTransform(hdcDest, &xform);
	CDC* pSrcDC = CDC::FromHandle(hdcSource);
	CDC* pDstDC = CDC::FromHandle(hdcDest);
	pDstDC->StretchBlt(nx, ny, src->GetWidth(), src->GetHeight(), pSrcDC, 0, 0, src->GetWidth(), src->GetHeight(), SRCCOPY);
}
相关推荐
Paddi9301 小时前
Codeforces Round 1004 (Div. 1) C. Bitwise Slides
c++·算法
流星白龙5 小时前
【C++】36.C++IO流
开发语言·c++
靡不有初1116 小时前
CCF-CSP第31次认证第二题——坐标变换(其二)【NA!前缀和思想的细节,输出为0的常见原因】
c++·学习·ccfcsp
YH_DevJourney9 小时前
Linux-C/C++《C/7、字符串处理》(字符串输入/输出、C 库中提供的字符串处理函数、正则表达式等)
linux·c语言·c++
和光同尘@9 小时前
1011. A+B和C (15)-PAT乙级真题
c语言·开发语言·数据结构·c++·算法
AomanHao9 小时前
图像质量评价指标-UCIQE-UIQM
图像处理·人工智能·计算机视觉·评价指标
xianwu54310 小时前
反向代理模块kd
开发语言·网络·数据库·c++·mysql
鄃鳕10 小时前
C++11 thread
开发语言·c++
wen__xvn11 小时前
每日一题洛谷P5738 【深基7.例4】歌唱比赛c++
java·c++·算法
C#Thread11 小时前
机器视觉--图像的运算(乘法)
图像处理·人工智能·计算机视觉