本文主要是介绍CImage通过WinApi的SetWorldTransform来实现图片旋转,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!
SetWorldTransform的功能是旋转画布,这样产生的效果就是图像旋转。因此,在旋转画布之前,要把要旋转的图像的位置和大小准备好,这样旋转之后,才能使图像正好出现在显示区域内。这需要计算两个关键参数,图像的左上角坐标和旋转中心坐标。因为是固定大小旋转,因此我们将中心设定在图像的显示中心。这样需要计算选中图像的高和宽。
如下图:
具体实现方法如下:
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);if (!dst->IsNull()){dst->Destroy();}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);
}
该方法优点是速度快,比我之前写的双循环遍历像素点,对图片进行旋转要快得多。
我用来测试的jpg图片的分辨率是3456*4806,之前的遍历像素点方法用时是1700多毫秒,而上面的方法用时是600多毫秒,可见效果很明显了。
PS:我一开始用SetWorldTransform,旋转后,图像无法定位到正确的地方,还好借鉴了一个大佬的文章,连接在下方。
图片浏览器开发日志-05(显示速度)_setworldtransform 后显示慢-CSDN博客
这篇关于CImage通过WinApi的SetWorldTransform来实现图片旋转的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!