iOS中实现图片自适应拉伸效果的方法

发布时间 - 2026-01-11 00:17:14    点击率:

前言

在Android中实现图片的拉伸特别特别简单,甚至不用写一行代码,直接使用.9图片进行划线即可。但是iOS就没这么简单了,比如对于下面的一张图片(原始尺寸:200*103):

我们不做任何处理,直接将它用作按钮的背景图片:

//
// ViewController.m
// ChatBgTest
//
// Created by 李峰峰 on 2017/1/23.
// Copyright © 2017年 李峰峰. All rights reserved.
//
 
#import "ViewController.h"
 
@interface ViewController ()
 
@end
 
@implementation ViewController
 
- (void)viewDidLoad {
 [super viewDidLoad];
 [self addBtn];
}
 
-(void)addBtn{
 // 创建一个按钮
 UIButton *btn = [UIButton buttonWithType:UIButtonTypeCustom];
 // 设置按钮的frame
 btn.frame = CGRectMake(50, 300, 300, 103);
 // 加载图片
 UIImage *image = [UIImage imageNamed:@"chat_bg"];
 // 设置按钮的背景图片
 [btn setBackgroundImage:image forState:UIControlStateNormal];
 // 将按钮添加到控制器的view
 [self.view addSubview:btn];
}
 
@end

运行效果如下:

可以看到图片被明显拉伸,显示效果较差。今天我们研究内容就是图片自适应拉伸。

图片自适应拉伸

1、iOS5之前

iOS中有个叫端盖(end cap)的概念,用来指定图片中的哪一部分不用拉伸,如下图:设置topCapHeight、leftCapWidth、bottomCapHeight、lerightCapWidth,图中的黑色区域就是图片拉伸的范围,也就是说边上的不会被拉伸。

使用UIImage的下面这个方法,可以通过设置端盖宽度返回一个经过拉伸处理的UIImage对象:

- (UIImage *)stretchableImageWithLeftCapWidth:(NSInteger)leftCapWidth topCapHeight:(NSInteger)topCapHeight; 

这个方法只有2个参数,leftCapWidth代表左端盖宽度,topCapHeight代表上端盖高度。系统会自动计算出右端盖宽度rightCapWidth和底端盖高度bottomCapHeight,代码如下:

/**
 第一种拉伸方式(iOS5之前)
 */
-(void)stretchTest1{
 
 // 创建一个按钮
 UIButton *btn = [UIButton buttonWithType:UIButtonTypeCustom];
 // 设置按钮的frame
 btn.frame = CGRectMake(50, 300, 300, 103);
 // 加载图片
 UIImage *image = [UIImage imageNamed:@"chat_bg"];
 
 // 设置左边端盖宽度
 NSInteger leftCapWidth = image.size.width * 0.5f;
 // 设置上边端盖高度
 NSInteger topCapHeight = image.size.height * 0.5f;
  
 UIImage *newImage = [image stretchableImageWithLeftCapWidth:leftCapWidth topCapHeight:topCapHeight];
 
 // 设置按钮的背景图片
 [btn setBackgroundImage:newImage forState:UIControlStateNormal];
 // 将按钮添加到控制器的view
 [self.view addSubview:btn];
}

这样一来,其实我们图片的可拉伸范围只有1 * 1,所以再怎么拉伸都不会影响图片的外观,运行效果如下:

现在再看一下效果是不是好多了。

2、iOS5

在iOS 5.0中,UIImage又有一个新方法可以处理图片的拉伸问题:

- (UIImage *)resizableImageWithCapInsets:(UIEdgeInsets)capInsets 
typedef struct UIEdgeInsets {
 CGFloat top, left, bottom, right; 
 // specify amount to inset (positive) for each of the edges. values can be negative to 'outset'
} UIEdgeInsets;

这个方法只接收一个UIEdgeInsets类型的参数,可以通过设置UIEdgeInsets中的CGFloat top, left, bottom, right就是用来设置上端盖、左端盖、下端盖、右端盖的尺寸(逆时针方向)。

/**
 第二种拉伸方式(iOS5)
 */
-(void)stretchTest2{
 
 // 创建一个按钮
 UIButton *btn = [UIButton buttonWithType:UIButtonTypeCustom];
 // 设置按钮的frame
 btn.frame = CGRectMake(50, 300, 300, 103);
 // 加载图片
 UIImage *image = [UIImage imageNamed:@"chat_bg"];
 
 // 设置端盖的值
 CGFloat top = image.size.height * 0.5;
 CGFloat left = image.size.width * 0.5;
 CGFloat bottom = image.size.height * 0.5;
 CGFloat right = image.size.width * 0.5;
 
 UIEdgeInsets edgeInsets = UIEdgeInsetsMake(top, left, bottom, right);
 
 // 拉伸图片
 UIImage *newImage = [image resizableImageWithCapInsets:edgeInsets];
 
 // 设置按钮的背景图片
 [btn setBackgroundImage:newImage forState:UIControlStateNormal];
 // 将按钮添加到控制器的view
 [self.view addSubview:btn];
}

运行效果与第一种一样,就不再截图了。

3、iOS6

在iOS6.0中,UIImage又提供了一个方法处理图片拉伸:

- (UIImage *)resizableImageWithCapInsets:(UIEdgeInsets)capInsets resizingMode:(UIImageResizingMode)resizingMode 

相比iOS5中的方法多了一个resizingMode参数:

typedef NS_ENUM(NSInteger, UIImageResizingMode) {
 UIImageResizingModeTile, // 平铺模式,通过重复显示UIEdgeInsets指定的矩形区域来填充图片
 UIImageResizingModeStretch, // 拉伸模式,通过拉伸UIEdgeInsets指定的矩形区域来填充图片
};

具体实现代码如下:

/**
 第三种拉伸方式(iOS6)
 */
-(void)stretchTest3{
 
 // 创建一个按钮
 UIButton *btn = [UIButton buttonWithType:UIButtonTypeCustom];
 // 设置按钮的frame
 btn.frame = CGRectMake(50, 300, 300, 103);
 // 加载图片
 UIImage *image = [UIImage imageNamed:@"chat_bg"];
 
 // 设置端盖的值
 CGFloat top = image.size.height * 0.5;
 CGFloat left = image.size.width * 0.5;
 CGFloat bottom = image.size.height * 0.5;
 CGFloat right = image.size.width * 0.5;
 
 // 设置端盖的值
 UIEdgeInsets edgeInsets = UIEdgeInsetsMake(top, left, bottom, right);
 // 设置拉伸的模式
 UIImageResizingMode mode = UIImageResizingModeStretch;
 
 // 拉伸图片
 UIImage *newImage = [image resizableImageWithCapInsets:edgeInsets resizingMode:mode];
 
 // 设置按钮的背景图片
 [btn setBackgroundImage:newImage forState:UIControlStateNormal];
 // 将按钮添加到控制器的view
 [self.view addSubview:btn];
}

运行效果与第一种一样,就不再截图了。

总结

以上就是这篇文章的全部内容了,希望本文的内容对给位iOS开发者们能带来一定的帮助,如果有疑问大家可以留言交流,谢谢大家对的支持。


# ios  # 图片自适应  # ios图片自适应大小  # ios按钮图片自适应  # iOS使用UICollectionView实现列表头部拉伸效果  # iOS tableview实现顶部拉伸效果  # iOS实现头部拉伸效果  # iOS tableView实现头部拉伸并改变导航条渐变色  # iOS图片实现可拉伸不变形的处理操作  # iOS图片拉伸的4种方法  # iOS应用开发中图片的拉伸问题解决方案  # iOS图片拉伸小技巧  # iOS 解决按钮背景图片拉伸问题(推荐)  # iOS tableView实现顶部图片拉伸效果  # 创建一个  # 左端  # 加载  # 第一种  # 可以通过  # 自适应  # 有个  # 右端  # 平铺  # 又有  # 就没  # 可以看到  # 不做  # 再看  # 这篇文章  # 将它  # 谢谢大家  # 这么简单  # 第二种  # 图中 


相关栏目: 【 网站优化151355 】 【 网络推广146373 】 【 网络技术251813 】 【 AI营销90571


相关推荐: Linux虚拟化技术教程_KVMQEMU虚拟机安装与调优  Laravel辅助函数有哪些_Laravel Helpers常用助手函数大全  Android GridView 滑动条设置一直显示状态(推荐)  Laravel如何使用Laravel Vite编译前端_Laravel10以上版本前端静态资源管理【教程】  制作无缝贴图网站有哪些,3dmax无缝贴图怎么调?  如何在Windows服务器上快速搭建网站?  邀请函制作网站有哪些,有没有做年会邀请函的网站啊?在线制作,模板很多的那种?  阿里云网站搭建费用解析:服务器价格与建站成本优化指南  Laravel Telescope怎么调试_使用Laravel Telescope进行应用监控与调试  如何解决hover在ie6中的兼容性问题  javascript基于原型链的继承及call和apply函数用法分析  浅述节点的创建及常见功能的实现  网页设计与网站制作内容,怎样注册网站?  头像制作网站在线观看,除了站酷,还有哪些比较好的设计网站?  PHP 实现电台节目表的智能时间匹配与今日/明日轮播逻辑  JS中对数组元素进行增删改移的方法总结  node.js报错:Cannot find module 'ejs'的解决办法  宙斯浏览器文件分类查看教程 快速筛选视频文档与图片方法  详解jQuery中基本的动画方法  Laravel集合Collection怎么用_Laravel集合常用函数详解  Laravel如何实现多语言支持_Laravel本地化与国际化(i18n)配置教程  装修招标网站设计制作流程,装修招标流程?  如何快速搭建高效香港服务器网站?  怎样使用JSON进行数据交换_它有什么限制  如何在橙子建站上传落地页?操作指南详解  如何用JavaScript实现文本编辑器_光标和选区怎么处理  C++时间戳转换成日期时间的步骤和示例代码  JavaScript如何实现音频处理_Web Audio API如何工作?  ChatGPT回答中断怎么办 引导AI继续输出完整内容的方法  Laravel队列由Redis驱动怎么配置_Laravel Redis队列使用教程  Python制作简易注册登录系统  html文件怎么打开证书错误_https协议的html打开提示不安全【指南】  Laravel中DTO是什么概念_在Laravel项目中使用数据传输对象(DTO)  电视网站制作tvbox接口,云海电视怎样自定义添加电视源?  如何在HTML表单中获取用户输入并结合JavaScript动态控制复利计算循环  Laravel怎么使用Session存储数据_Laravel会话管理与自定义驱动配置【详解】  高防服务器租用如何选择配置与防御等级?  如何注册花生壳免费域名并搭建个人网站?  Android okhttputils现在进度显示实例代码  成都品牌网站制作公司,成都营业执照年报网上怎么办理?  Laravel如何实现本地化和多语言支持?(i18n教程)  如何快速生成凡客建站的专业级图册?  rsync同步时出现rsync: failed to set times on “xxxx”: Operation not permitted  如何在阿里云虚拟主机上快速搭建个人网站?  如何在云服务器上快速搭建个人网站?  Laravel Seeder填充数据教程_Laravel模型工厂Factory使用  Win11怎么查看显卡温度 Win11任务管理器查看GPU温度【技巧】  如何在局域网内绑定自建网站域名?  laravel怎么为应用开启和关闭维护模式_laravel应用维护模式开启与关闭方法  详解jQuery中的事件