[转自]http://marshal.easymorse.com/archives/3703
在ios开发中,肯定会碰到需要截取部分图片的情况。
最终的效果类似这样:


先看最原始的示例,显示完整的图片
写了个最简单的读取图片并显示的代码,打算以此为开始,逐渐实现截取部分图片的功能。
代码主要是,在控制器代码中:
- (void)loadView {
[[UIApplication sharedApplication] setStatusBarHidden:YES withAnimation: UIStatusBarAnimationSlide];
UIImage *image=[UIImage imageNamed:@"1.jpg"];
UIImageView *contentView = [[UIImageView alloc] initWithFrame:[[UIScreen mainScreen] applicationFrame]];
[contentView setImage:image];
self.view=[[UIView alloc] initWithFrame:[[UIScreen mainScreen] applicationFrame]];
[self.view addSubview:contentView];
}
另外,应该有一个名为1.jpg的768×1024的图片(我这里是iPad)。
截取整个图片
可以认为截取整个图片是截取部分图片的一个特例。对ios不熟嘛,因此打算很谨慎的推进。截取整个图片可以减少中间的复杂性。
根据API,摸索着写了一个示例,效果出乎意料:

代码:
- (void)loadView {
[[UIApplication sharedApplication] setStatusBarHidden:YES withAnimation: UIStatusBarAnimationSlide];
UIImage *image=[UIImage imageNamed:@"1.jpg"];
UIImageView *contentView = [[UIImageView alloc] initWithFrame:[[UIScreen mainScreen] applicationFrame]];
//[contentView setImage:image];
CGRect rect = CGRectMake(0, 0, 768, 1024);//创建矩形框
UIGraphicsBeginImageContext(rect.size);//根据size大小创建一个基于位图的图形上下文
CGContextRef currentContext = UIGraphicsGetCurrentContext();//获取当前quartz 2d绘图环境
CGContextClipToRect( currentContext, rect);//设置当前绘图环境到矩形框
CGContextDrawImage(currentContext, rect, image.CGImage);//绘图
UIImage *cropped = UIGraphicsGetImageFromCurrentImageContext();//获得图片
UIGraphicsEndImageContext();//从当前堆栈中删除quartz 2d绘图环境
contentView.image=cropped;
self.view=[[UIView alloc] initWithFrame:[[UIScreen mainScreen] applicationFrame]];
[self.view addSubview:contentView];
[cropped release];
}
这个代码说明了两点:
- 好的方面:说明我的代码起作用了,确实截取了所需的图形
- 坏的方面:图形是颠倒的,而且是镜像的。
问题应该出在坐标系上。下面画了一个quartz 2d的坐标系,坐标原点在左下角:

因此以这个坐标系取图形,就会有转向180°的效果。
其实如果是对图片的缩放,而不是剪切部分图片内容,这样写就可以了:
- (void)loadView {
[[UIApplication sharedApplication] setStatusBarHidden:YES withAnimation: UIStatusBarAnimationSlide];
UIImage *image=[UIImage imageNamed:@"1.jpg"];
//[contentView setImage:image];
CGRect rect = CGRectMake(0, 0, 384, 512);//创建矩形框
UIGraphicsBeginImageContext(rect.size);//根据size大小创建一个基于位图的图形上下文
CGContextRef currentContext = UIGraphicsGetCurrentContext();//获取当前quartz 2d绘图环境
CGContextClipToRect(currentContext, rect);//设置当前绘图环境到矩形框
//CGContextRotateCTM(currentContext, 50);
//CGContextDrawImage(currentContext, rect, image.CGImage);//绘图
[image drawInRect:rect];
UIImage *cropped = UIGraphicsGetImageFromCurrentImageContext();//获得图片
UIGraphicsEndImageContext();//从当前堆栈中删除quartz 2d绘图环境
UIImageView *contentView = [[UIImageView alloc] initWithFrame:rect];
contentView.image=cropped;
self.view=[[UIView alloc] initWithFrame:[[UIScreen mainScreen] applicationFrame]];
[self.view addSubview:contentView];
[cropped release];
}
效果类似这样:

这个方法可以帮助我们在后续开发中实现缩略图。但是不符合现在的需求。
于是想了下面的基本思路:

这样,需要一个能旋转和向下移动的API。ios提供了C++界面的函数调用:
- CGContextRotateCTM,实现角度的转换
- CGContextTranslateCTM,可以重新设置坐标系原点,平移坐标系和移动图片是等效的
代码:
- (void)loadView {
[[UIApplication sharedApplication] setStatusBarHidden:YES withAnimation: UIStatusBarAnimationSlide];
UIImage *image=[UIImage imageNamed:@"1.jpg"];
//[contentView setImage:image];
CGRect rect = CGRectMake(0, 0, 384, 512);//创建矩形框
UIGraphicsBeginImageContext(rect.size);//根据size大小创建一个基于位图的图形上下文
CGContextRef currentContext = UIGraphicsGetCurrentContext();//获取当前quartz 2d绘图环境
CGContextClipToRect(currentContext, rect);//设置当前绘图环境到矩形框
CGContextRotateCTM(currentContext, M_PI);
CGContextTranslateCTM(currentContext, -rect.size.width, -rect.size.height);
CGContextDrawImage(currentContext, rect, image.CGImage);//绘图
//[image drawInRect:rect];
UIImage *cropped = UIGraphicsGetImageFromCurrentImageContext();//获得图片
UIGraphicsEndImageContext();//从当前堆栈中删除quartz 2d绘图环境
UIImageView *contentView = [[UIImageView alloc] initWithFrame:rect];
contentView.image=cropped;
self.view=[[UIView alloc] initWithFrame:[[UIScreen mainScreen] applicationFrame]];
[self.view addSubview:contentView];
[cropped release];
}

这个结果还有缺陷,可以看到图片是正立的了,但是图片反转了,是个镜像。
解决办法也有,不过不是操作图片了,而是操作图片所在的视图。思路是把视图看作一个位图的矩阵,对它做矩阵变换运算,使视图做镜像反转。写法很简单:
- (void)loadView {
[[UIApplication sharedApplication] setStatusBarHidden:YES withAnimation: UIStatusBarAnimationSlide];
UIImage *image=[UIImage imageNamed:@"1.jpg"];
//[contentView setImage:image];
CGRect rect = CGRectMake(0, 0, 384, 512);//创建矩形框
UIGraphicsBeginImageContext(rect.size);//根据size大小创建一个基于位图的图形上下文
CGContextRef currentContext = UIGraphicsGetCurrentContext();//获取当前quartz 2d绘图环境
CGContextClipToRect(currentContext, rect);//设置当前绘图环境到矩形框
CGContextRotateCTM(currentContext, M_PI);
CGContextTranslateCTM(currentContext, -rect.size.width, -rect.size.height);
//CGContextTranslateCTM(currentContext,0.0,200.0);
CGContextDrawImage(currentContext, rect, image.CGImage);//绘图
//[image drawInRect:rect];
UIImage *cropped = UIGraphicsGetImageFromCurrentImageContext();//获得图片
UIGraphicsEndImageContext();//从当前堆栈中删除quartz 2d绘图环境
UIImageView *contentView = [[UIImageView alloc] initWithFrame:rect];
contentView.image=cropped;
contentView.transform = CGAffineTransformIdentity;
contentView.transform = CGAffineTransformMakeScale(-1.0, 1.0);
self.view=[[UIView alloc] initWithFrame:[[UIScreen mainScreen] applicationFrame]];
[self.view addSubview:contentView];
[cropped release];
}
这里的转换因子,一个是针对x轴的,一个是针对y轴的。终于可以产生这样的效果了:

这里参考了这个文档:
http://macdevcenter.com/pub/a/mac/2004/11/02/quartz.html
虽然是很古老的文章了,但是说的很清楚。另外,方法名称已经发生变化,需要注意。
截取部分图片
截取部分图片,比如:

截取左边人像部分。
实现后的代码,效果是这样的:

如何实现的呢,这时候才发现,其实根本不需要上面那些转换,如果不使用quartz 2d的话,截取部分图片这么简单:
- (void)loadView {
[[UIApplication sharedApplication] setStatusBarHidden:YES withAnimation: UIStatusBarAnimationSlide];
UIImage *image=[UIImage imageNamed:@"1.jpg"];
CGRect rect = CGRectMake(60, 80, 331, 353);//创建矩形框
UIImageView *contentView = [[UIImageView alloc] initWithFrame:rect];
contentView.image=[UIImage imageWithCGImage:CGImageCreateWithImageInRect([image CGImage], rect)];
self.view=[[UIView alloc] initWithFrame:[[UIScreen mainScreen] applicationFrame]];
[self.view addSubview:contentView];
[image release];
}
虽然编写代码的过程是曲折的,但是摸到很多有用的东西,都是以后要用到的。
分享到:
相关推荐
QQ截图工具是一款非常实用的应用,尤其对于经常需要截取屏幕图像的用户来说,它提供了方便快捷的方式。在本文中,我们将深入探讨“仿QQ截图工具源码”这一主题,特别是关于增加线条粗细功能的实现。 首先,我们需要...
单击OK,单击Tools中的Hyperlink按钮,将鼠标指移到一要素上并单击以显示其超链接。 28.Arcmap中的SQL语言 当查询ArcInfo coverages, shape文件, INFO表以及dBASE表时,SQL表达式中的字段名必须用双引号扩起。如:...
2025年上海市公交路线及站点矢量shp数据.zip
大学时数电实验的资料,仅用于复习和学习参考
2025年DeepSeek与AI幻觉研究报告.pdf
基于STM32的无刷直流电机有_无传感器调速系统代码与原理图.pdf
基于Stm32硬件浮点运算芯片的6us一轮代码全手写:讲解代码流程,包含有感FOC速度环、电流环、位置环的注释详解.pdf
门户元素-日历日程.zip
以下是一个人工智能相关资源在实践工作中的使用案例: ### 案例:京东方工业互联网中的AI应用 - **资源**:京东方利用人工智能技术结合工业互联网平台,实现了生产过程的智能化管理和优化。
1、文件说明: Centos8操作系统usermode-gtk-1.113-2.el8.rpm以及相关依赖,全打包为一个tar.gz压缩包 2、安装指令: #Step1、解压 tar -zxvf usermode-gtk-1.113-2.el8.tar.gz #Step2、进入解压后的目录,执行安装 sudo rpm -ivh *.rpm
基于SVM的人民币面值识别系统的Matlab GUI实现.pdf
基于STM32的画图板功能详解:包含原理图、PCB、BOM表格及Kiel工程源码.pdf
皓拓拼版,错误1解决办法
内容概要:本文深入阐述了面向对象编程(OOP)的关键要素,如对象、类、继承和多态,并探讨了OOP的重要设计原则,包括单一职责、开放封闭、里氏替换、依赖倒置、接口隔离等。详细解读了依赖倒置的设计思路,通过实际代码案例展示了如何利用面向对象的思想改善系统结构,增强了程序灵活性与扩展性。接着介绍了统一建模语言(UML)及其图表的应用范围,特别是UML的不同类型的图如何帮助理解和设计软件系统架构。进一步讲解了三大类经典设计模式:创建型模式(如工厂方法、抽象工厂、生成器、原型、单例),结构型模式(适配器、桥接、组合等)和行为型模式(职责链、命令、迭代器、中介者等)。每种模式均有实例代码演示,便于初学者快速掌握。最后结合具体的工程应用场景分析,引导学习者灵活运用设计模式解决问题。 适用人群:具有初步编程经验,特别是对 Java 编程语言有所了解的技术爱好者和初级开发者。这部分群体正在寻求加深理解面向对象的概念和技巧,掌握设计模式的应用。 使用场景及目标:适用于正在设计小型到中型规模项目的个人和团队。无论是希望构建稳健的应用框架,还是希望通过合理的模块划分降低系统的复杂度,或是提高代码的可维护性和可
PHP编程语言实践指南
基于STM32的水质_浊度检测仪设计与实现:详细设计说明书+原理图PCB工程+源码工程.pdf
2025年江苏宿迁公交路线及站点矢量shp数据.zip
阿里巴巴的销售管理分析(50页)
基于STM32的物联网智能家居系统:实时监控与智能调节.pdf