iOS中UIView快速调整坐标和尺寸

此处代码由Swift3.1展示(兼容Swift4),推荐使用Swift,Objective-C版本后续会有更新, 相关的UIView扩展方法我已经上传至github中UIViewExtension(https://github.com/cba023/UIViewExtension),若要使用,请导入文件到您的项目。

开发环境

Mac OS 10.12+ / Xcode 8+ / Swift 3+(兼容Swift4)

支持环境

iOS 8+, iPhone & iPad

1. 系统初始化的布局及其缺点

在iOS开发中,UIView的布局通常是用frame来进行设置的,我们可以直接使用带frame参数的初始化方法。
如下,在视图控制器中添加一个UIView为view1,若要设置view1的布局距离屏幕上边20,屏幕的左边20,屏幕下边20,距离屏幕右边20。代码:

let view1 = UIView(frame: CGRect(x: 20.0, y: 20.0, width: UIScreen.main.bounds.size.width - 40.0, height: UIScreen.main.bounds.size.height - 40.0))

显然,对于不更改的情况还能较为方便的对frame进行赋值,不过要注意的是设置width的时候, view1的x坐标已经是20.0,既然右边也距离屏幕边沿20.0,那么它的宽度就只能设置成UIScreen.main.bounds.size.width - 40.0。纵向布局同理。

这样的布局有一定的缺点:

  • view宽高需要依照屏幕中的空间布局经过计算才能得到,所以造成不必要的麻烦。
  • 如果对frame中的单项属性进行修改,这里是不可以的。

2.解决方案

系统没有对frame中的属性提供set方法,所以此处我们需要利用自己的智慧来页面布局,提高可读性,提高编码效率。

扩展方法size

/// 尺寸
var size: CGSize {
  get {
    return self.frame.size
  }
  set(newValue) {
    self.frame.size = CGSize(width: newValue.width, height: newValue.height)
  }
}

width

/// 宽度
var width: CGFloat {
  get {
   return self.frame.size.width
  }
  set(newValue) {
   self.frame.size.width = newValue
  }
}

height

/// 高度
var height: CGFloat {
  get {
    return self.frame.size.height
  }
  set(newValue) {
    self.frame.size.height = newValue
  }
}

x

/// 横坐标
var x: CGFloat {
  get {
    return self.frame.minX
  }
  set(newValue) {
   self.frame = CGRect(x: newValue, y: y, width: width, height: height)
  }
}

y

/// 纵坐标
var y: CGFloat {
  get {
    return self.frame.minY
  }
  set(newValue) {
   self.frame = CGRect(x: x, y: newValue, width: width, height: height)
  }
}

right

/// 纵坐标
var y: CGFloat {
  get {
    return self.frame.minY
  }
  set(newValue) {
    self.frame = CGRect(x: x, y: newValue, width: width, height: height)
  }
}

bottom

/// 底端纵坐标
var bottom: CGFloat {
  get {
    return frame.origin.y + frame.size.height
  }
  set(newValue) {
    frame.origin.y = newValue - frame.size.height
  }
}

centerX

/// 中心横坐标
var centerX: CGFloat {
  get {
    return self.center.x
  }
  set(newValue) {
    center.x = newValue
  }
}

centerY

/// 中心纵坐标
var centerY: CGFloat {
  get {
    return center.y
  }
  set(newValue) {
    center.y = newValue
  }
}

origin

/// 原点
var origin: CGPoint {
  get {
    return self.origin
  }
  set(newValue) {
    frame.origin = newValue
  }
}

topRight

/// 右上角坐标
var topRight: CGPoint {
  get {
    return CGPoint(x: frame.origin.x + frame.size.width, y: frame.origin.y)
  }
  set(newValue) {
    frame.origin = CGPoint(x: newValue.x - width, y: newValue.y)
  }
}

bottomRight

/// 右下角坐标
var bottomRight: CGPoint {
  get {
    return CGPoint(x: frame.origin.x + frame.size.width, y: frame.origin.y + frame.size.height)
  }
  set(newValue) {
    frame.origin = CGPoint(x: newValue.x - width, y: newValue.y - height)
  }
}

bottomLeft

/// 左下角坐标
var bottomLeft: CGPoint {
  get {
    return CGPoint(x: frame.origin.x, y: frame.origin.y + frame.size.height)
  }
  set(newValue) {
    frame.origin = CGPoint(x: newValue.x, y: newValue.y - height)
  }
}

剪掉一个方向的部分距离。To cut off in one direction some distance

/// 获取UIView对象某个方向缩进指定距离后的方形区域
///
/// - Parameters:
///   - direction: 要缩进的方向
///   - distance: 缩进的距离
/// - Returns: 得到的区域
func cutRect(direction: Direction, distance: CGFloat) ->  CGRect {
  switch direction {
  case .top:
    return CGRect(x: 0, y: distance, width: self.width, height: self.height - distance)
  case .left:
    return CGRect(x: distance, y: 0, width: self.width - distance, height: self.height)
  case .right:
    return CGRect(x: 0, y: 0, width: self.width - distance, height: self.height)
  case .bottom:
    return CGRect(x: 0, y: 0, width: self.width, height: self.height - distance)
  }
}

3.使用

给UIView写入相关扩展

  • 设置view1横坐标为20.0
view1.x = 20.0;
  • 设置view1纵坐标为 30.0
view1.y = 30.0
  • 设置view1的宽度为 40.0
view1.width = 40.0
  • 设置view1的高度为 50.0
view1.height = 50.0
  • 设置view1的尺寸为宽60.0,宽70.0
view1.size = CGSize(width: 60.0, height: 70.0)
  • 设置view1的右边沿的横坐标位90.0
view1.right = 90.0
  • 设置view1的底边的纵坐标位110.0
view1.bottom = 110.0
  • 设置view1的中心横坐标为100.0
view1.bottom = 100.0
  • 设置view1的中心纵坐标为 110.0
view1.centerY = 110.0
  • 设置view1的原点坐标为(120,150)
view1.origin = CGPoint(x: 120.0, y:150.0)
  • 设置view1的右上角坐标为(130,140)
view1.topRight = CGPoint(x: 130.0, y:140.0)
  • 其他几个角落的坐标设置同理

  • 在实际设置UIView的坐标的时候,可能会遇见某一个方位向内缩进一段距离。比如, view1的顶部要比父视图往下缩进20.0, 其他方向与父视图重合,此时可以用如下方法:

view1.cutRect(direction: .up, distance: 20.0)

4.总结

  • 上述方法主要对UIView的frame相关的get和set方法封装成了UIView的对象方法,从而实现了快速的调整UIView的大小和坐标,
  1. 在使用相关方法的时候请先设置view的大小,在设置坐标,保证实际显示与设置的值对应;
  2. 在xib开发的时候,应当将相应的方法在 awakeFromNib() 方法中实现,已保证父类的坐标体系已经固定下来。

5. 致读者

相关的UIView扩展方法我已经上传至github中UIViewExtension(https://github.com/cba023/UIViewExtension),可以在那里直接star 或者fork 该项目,它可能会长期的帮助您高效地进行程序开发,当然也欢迎留言,有不足或者错误的地方可以随时指正,您的指导和建议是我前行路上新的动力!

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 202,529评论 5 475
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 85,015评论 2 379
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 149,409评论 0 335
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,385评论 1 273
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,387评论 5 364
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,466评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,880评论 3 395
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,528评论 0 256
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,727评论 1 295
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,528评论 2 319
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,602评论 1 329
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,302评论 4 318
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,873评论 3 306
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,890评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,132评论 1 259
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 42,777评论 2 349
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,310评论 2 342

推荐阅读更多精彩内容