ConstraintBuilder.swift 34 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808
  1. //
  2. // Snap
  3. //
  4. // Copyright (c) 2011-2015 Masonry Team - https://github.com/Masonry
  5. //
  6. // Permission is hereby granted, free of charge, to any person obtaining a copy
  7. // of this software and associated documentation files (the "Software"), to deal
  8. // in the Software without restriction, including without limitation the rights
  9. // to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
  10. // copies of the Software, and to permit persons to whom the Software is
  11. // furnished to do so, subject to the following conditions:
  12. //
  13. // The above copyright notice and this permission notice shall be included in
  14. // all copies or substantial portions of the Software.
  15. //
  16. // THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
  17. // IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
  18. // FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
  19. // AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
  20. // LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
  21. // OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
  22. // THE SOFTWARE.
  23. #if os(iOS)
  24. import UIKit
  25. #else
  26. import AppKit
  27. #endif
  28. /**
  29. * ConstraintBuilderFinalizable is a protocol that allows getting a finalized constraint
  30. */
  31. public protocol ConstraintBuilderFinalizable: class {
  32. var constraint: Constraint { get }
  33. }
  34. /**
  35. * ConstraintBuilderPriortizable is a protocol that allows a constraint to be prioritized
  36. */
  37. public protocol ConstraintBuilderPriortizable: ConstraintBuilderFinalizable {
  38. func priority(priority: Float) -> ConstraintBuilderFinalizable
  39. func priority(priority: Double) -> ConstraintBuilderFinalizable
  40. func priority(priority: CGFloat) -> ConstraintBuilderFinalizable
  41. func priority(priority: UInt) -> ConstraintBuilderFinalizable
  42. func priority(priority: Int) -> ConstraintBuilderFinalizable
  43. func priorityRequired() -> ConstraintBuilderFinalizable
  44. func priorityHigh() -> ConstraintBuilderFinalizable
  45. func priorityMedium() -> ConstraintBuilderFinalizable
  46. func priorityLow() -> ConstraintBuilderFinalizable
  47. }
  48. /**
  49. * ConstraintBuilderMultipliable is a protocol that allows a constraint to be multiplied
  50. */
  51. public protocol ConstraintBuilderMultipliable: ConstraintBuilderPriortizable {
  52. func multipliedBy(amount: Float) -> ConstraintBuilderPriortizable
  53. func multipliedBy(amount: Double) -> ConstraintBuilderPriortizable
  54. func multipliedBy(amount: CGFloat) -> ConstraintBuilderPriortizable
  55. func multipliedBy(amount: Int) -> ConstraintBuilderPriortizable
  56. func multipliedBy(amount: UInt) -> ConstraintBuilderPriortizable
  57. func dividedBy(amount: Float) -> ConstraintBuilderPriortizable
  58. func dividedBy(amount: Double) -> ConstraintBuilderPriortizable
  59. func dividedBy(amount: CGFloat) -> ConstraintBuilderPriortizable
  60. func dividedBy(amount: Int) -> ConstraintBuilderPriortizable
  61. func dividedBy(amount: UInt) -> ConstraintBuilderPriortizable
  62. }
  63. /**
  64. * ConstraintBuilderOffsetable is a protocol that allows a constraint to be offset / inset
  65. */
  66. public protocol ConstraintBuilderOffsetable: ConstraintBuilderMultipliable {
  67. func offset(amount: Float) -> ConstraintBuilderMultipliable
  68. func offset(amount: Double) -> ConstraintBuilderMultipliable
  69. func offset(amount: CGFloat) -> ConstraintBuilderMultipliable
  70. func offset(amount: Int) -> ConstraintBuilderMultipliable
  71. func offset(amount: UInt) -> ConstraintBuilderMultipliable
  72. func offset(amount: CGPoint) -> ConstraintBuilderMultipliable
  73. func offset(amount: CGSize) -> ConstraintBuilderMultipliable
  74. func offset(amount: EdgeInsets) -> ConstraintBuilderMultipliable
  75. func inset(amount: EdgeInsets) -> ConstraintBuilderMultipliable
  76. @availability(*, deprecated=0.10.0, renamed="inset")
  77. func insets(amount: EdgeInsets) -> ConstraintBuilderMultipliable
  78. }
  79. /**
  80. * ConstraintBuilderRelatable is a protocol that allows a constraint to be set related to another item/constant by equalTo, lessThanOrEqualTo or greaterThanOrEqualTo
  81. */
  82. public protocol ConstraintBuilderRelatable: class {
  83. func equalTo(other: ConstraintItem) -> ConstraintBuilderOffsetable
  84. func equalTo(other: View) -> ConstraintBuilderOffsetable
  85. #if os(iOS)
  86. func equalTo(other: UILayoutSupport) -> ConstraintBuilderOffsetable
  87. #endif
  88. func equalTo(other: Float) -> ConstraintBuilderMultipliable
  89. func equalTo(other: Double) -> ConstraintBuilderMultipliable
  90. func equalTo(other: CGFloat) -> ConstraintBuilderMultipliable
  91. func equalTo(other: Int) -> ConstraintBuilderMultipliable
  92. func equalTo(other: UInt) -> ConstraintBuilderMultipliable
  93. func equalTo(other: CGSize) -> ConstraintBuilderMultipliable
  94. func equalTo(other: CGPoint) -> ConstraintBuilderMultipliable
  95. func equalTo(other: EdgeInsets) -> ConstraintBuilderMultipliable
  96. func lessThanOrEqualTo(other: ConstraintItem) -> ConstraintBuilderOffsetable
  97. func lessThanOrEqualTo(other: View) -> ConstraintBuilderOffsetable
  98. #if os(iOS)
  99. func lessThanOrEqualTo(other: UILayoutSupport) -> ConstraintBuilderOffsetable
  100. #endif
  101. func lessThanOrEqualTo(other: Float) -> ConstraintBuilderMultipliable
  102. func lessThanOrEqualTo(other: Double) -> ConstraintBuilderMultipliable
  103. func lessThanOrEqualTo(other: CGFloat) -> ConstraintBuilderMultipliable
  104. func lessThanOrEqualTo(other: Int) -> ConstraintBuilderMultipliable
  105. func lessThanOrEqualTo(other: UInt) -> ConstraintBuilderMultipliable
  106. func lessThanOrEqualTo(other: CGSize) -> ConstraintBuilderMultipliable
  107. func lessThanOrEqualTo(other: CGPoint) -> ConstraintBuilderMultipliable
  108. func lessThanOrEqualTo(other: EdgeInsets) -> ConstraintBuilderMultipliable
  109. func greaterThanOrEqualTo(other: ConstraintItem) -> ConstraintBuilderOffsetable
  110. func greaterThanOrEqualTo(other: View) -> ConstraintBuilderOffsetable
  111. #if os(iOS)
  112. func greaterThanOrEqualTo(other: UILayoutSupport) -> ConstraintBuilderOffsetable
  113. #endif
  114. func greaterThanOrEqualTo(other: Float) -> ConstraintBuilderMultipliable
  115. func greaterThanOrEqualTo(other: Double) -> ConstraintBuilderMultipliable
  116. func greaterThanOrEqualTo(other: CGFloat) -> ConstraintBuilderMultipliable
  117. func greaterThanOrEqualTo(other: Int) -> ConstraintBuilderMultipliable
  118. func greaterThanOrEqualTo(other: UInt) -> ConstraintBuilderMultipliable
  119. func greaterThanOrEqualTo(other: CGSize) -> ConstraintBuilderMultipliable
  120. func greaterThanOrEqualTo(other: CGPoint) -> ConstraintBuilderMultipliable
  121. func greaterThanOrEqualTo(other: EdgeInsets) -> ConstraintBuilderMultipliable
  122. }
  123. /**
  124. * ConstraintBuilderExtendable is a protocol that allows a constraint to be extended
  125. */
  126. public protocol ConstraintBuilderExtendable: ConstraintBuilderRelatable {
  127. var left: ConstraintBuilderExtendable { get }
  128. var top: ConstraintBuilderExtendable { get }
  129. var bottom: ConstraintBuilderExtendable { get }
  130. var right: ConstraintBuilderExtendable { get }
  131. var leading: ConstraintBuilderExtendable { get }
  132. var trailing: ConstraintBuilderExtendable { get }
  133. var width: ConstraintBuilderExtendable { get }
  134. var height: ConstraintBuilderExtendable { get }
  135. var centerX: ConstraintBuilderExtendable { get }
  136. var centerY: ConstraintBuilderExtendable { get }
  137. var baseline: ConstraintBuilderExtendable { get }
  138. #if os(iOS)
  139. var firstBaseline: ConstraintBuilderExtendable { get }
  140. var leftMargin: ConstraintBuilderExtendable { get }
  141. var rightMargin: ConstraintBuilderExtendable { get }
  142. var topMargin: ConstraintBuilderExtendable { get }
  143. var bottomMargin: ConstraintBuilderExtendable { get }
  144. var leadingMargin: ConstraintBuilderExtendable { get }
  145. var trailingMargin: ConstraintBuilderExtendable { get }
  146. var centerXWithinMargins: ConstraintBuilderExtendable { get }
  147. var centerYWithinMargins: ConstraintBuilderExtendable { get }
  148. #endif
  149. }
  150. /**
  151. * ConstraintBuilder is a single item that defines all the properties for a single ConstraintMaker building chain
  152. */
  153. final internal class ConstraintBuilder: Constraint, ConstraintBuilderExtendable, ConstraintBuilderOffsetable, ConstraintBuilderFinalizable {
  154. var left: ConstraintBuilderExtendable { return self.addConstraint(ConstraintAttributes.Left) }
  155. var top: ConstraintBuilderExtendable { return self.addConstraint(ConstraintAttributes.Top) }
  156. var right: ConstraintBuilderExtendable { return self.addConstraint(ConstraintAttributes.Right) }
  157. var bottom: ConstraintBuilderExtendable { return self.addConstraint(ConstraintAttributes.Bottom) }
  158. var leading: ConstraintBuilderExtendable { return self.addConstraint(ConstraintAttributes.Leading) }
  159. var trailing: ConstraintBuilderExtendable { return self.addConstraint(ConstraintAttributes.Trailing) }
  160. var width: ConstraintBuilderExtendable { return self.addConstraint(ConstraintAttributes.Width) }
  161. var height: ConstraintBuilderExtendable { return self.addConstraint(ConstraintAttributes.Height) }
  162. var centerX: ConstraintBuilderExtendable { return self.addConstraint(ConstraintAttributes.CenterX) }
  163. var centerY: ConstraintBuilderExtendable { return self.addConstraint(ConstraintAttributes.CenterY) }
  164. var baseline: ConstraintBuilderExtendable { return self.addConstraint(ConstraintAttributes.Baseline) }
  165. #if os(iOS)
  166. var firstBaseline: ConstraintBuilderExtendable { return self.addConstraint(ConstraintAttributes.FirstBaseline) }
  167. var leftMargin: ConstraintBuilderExtendable { return self.addConstraint(ConstraintAttributes.LeftMargin) }
  168. var rightMargin: ConstraintBuilderExtendable { return self.addConstraint(ConstraintAttributes.RightMargin) }
  169. var topMargin: ConstraintBuilderExtendable { return self.addConstraint(ConstraintAttributes.TopMargin) }
  170. var bottomMargin: ConstraintBuilderExtendable { return self.addConstraint(ConstraintAttributes.BottomMargin) }
  171. var leadingMargin: ConstraintBuilderExtendable { return self.addConstraint(ConstraintAttributes.LeadingMargin) }
  172. var trailingMargin: ConstraintBuilderExtendable { return self.addConstraint(ConstraintAttributes.TrailingMargin) }
  173. var centerXWithinMargins: ConstraintBuilderExtendable { return self.addConstraint(ConstraintAttributes.CenterXWithinMargins) }
  174. var centerYWithinMargins: ConstraintBuilderExtendable { return self.addConstraint(ConstraintAttributes.CenterYWithinMargins) }
  175. #endif
  176. // MARK: initializer
  177. init(fromItem: ConstraintItem) {
  178. self.fromItem = fromItem
  179. self.toItem = ConstraintItem(object: nil, attributes: ConstraintAttributes.None)
  180. }
  181. // MARK: equalTo
  182. func equalTo(other: ConstraintItem) -> ConstraintBuilderOffsetable {
  183. return self.constrainTo(other, relation: .Equal)
  184. }
  185. func equalTo(other: View) -> ConstraintBuilderOffsetable {
  186. return self.constrainTo(other, relation: .Equal)
  187. }
  188. #if os(iOS)
  189. func equalTo(other: UILayoutSupport) -> ConstraintBuilderOffsetable {
  190. return self.constrainTo(other, relation: .Equal)
  191. }
  192. #endif
  193. func equalTo(other: Float) -> ConstraintBuilderMultipliable {
  194. return self.constrainTo(other, relation: .Equal)
  195. }
  196. func equalTo(other: Double) -> ConstraintBuilderMultipliable {
  197. return self.constrainTo(Float(other), relation: .Equal)
  198. }
  199. func equalTo(other: CGFloat) -> ConstraintBuilderMultipliable {
  200. return self.constrainTo(Float(other), relation: .Equal)
  201. }
  202. func equalTo(other: Int) -> ConstraintBuilderMultipliable {
  203. return self.constrainTo(Float(other), relation: .Equal)
  204. }
  205. func equalTo(other: UInt) -> ConstraintBuilderMultipliable {
  206. return self.constrainTo(Float(other), relation: .Equal)
  207. }
  208. func equalTo(other: CGSize) -> ConstraintBuilderMultipliable {
  209. return self.constrainTo(other, relation: .Equal)
  210. }
  211. func equalTo(other: CGPoint) -> ConstraintBuilderMultipliable {
  212. return self.constrainTo(other, relation: .Equal)
  213. }
  214. func equalTo(other: EdgeInsets) -> ConstraintBuilderMultipliable {
  215. return self.constrainTo(other, relation: .Equal)
  216. }
  217. // MARK: lessThanOrEqualTo
  218. func lessThanOrEqualTo(other: ConstraintItem) -> ConstraintBuilderOffsetable {
  219. return self.constrainTo(other, relation: .LessThanOrEqualTo)
  220. }
  221. func lessThanOrEqualTo(other: View) -> ConstraintBuilderOffsetable {
  222. return self.constrainTo(other, relation: .LessThanOrEqualTo)
  223. }
  224. #if os(iOS)
  225. func lessThanOrEqualTo(other: UILayoutSupport) -> ConstraintBuilderOffsetable {
  226. return self.constrainTo(other, relation: .LessThanOrEqualTo)
  227. }
  228. #endif
  229. func lessThanOrEqualTo(other: Float) -> ConstraintBuilderMultipliable {
  230. return self.constrainTo(other, relation: .LessThanOrEqualTo)
  231. }
  232. func lessThanOrEqualTo(other: Double) -> ConstraintBuilderMultipliable {
  233. return self.constrainTo(Float(other), relation: .LessThanOrEqualTo)
  234. }
  235. func lessThanOrEqualTo(other: CGFloat) -> ConstraintBuilderMultipliable {
  236. return self.constrainTo(Float(other), relation: .LessThanOrEqualTo)
  237. }
  238. func lessThanOrEqualTo(other: Int) -> ConstraintBuilderMultipliable {
  239. return self.constrainTo(Float(other), relation: .LessThanOrEqualTo)
  240. }
  241. func lessThanOrEqualTo(other: UInt) -> ConstraintBuilderMultipliable {
  242. return self.constrainTo(Float(other), relation: .LessThanOrEqualTo)
  243. }
  244. func lessThanOrEqualTo(other: CGSize) -> ConstraintBuilderMultipliable {
  245. return self.constrainTo(other, relation: .LessThanOrEqualTo)
  246. }
  247. func lessThanOrEqualTo(other: CGPoint) -> ConstraintBuilderMultipliable {
  248. return self.constrainTo(other, relation: .LessThanOrEqualTo)
  249. }
  250. func lessThanOrEqualTo(other: EdgeInsets) -> ConstraintBuilderMultipliable {
  251. return self.constrainTo(other, relation: .LessThanOrEqualTo)
  252. }
  253. // MARK: greaterThanOrEqualTo
  254. func greaterThanOrEqualTo(other: ConstraintItem) -> ConstraintBuilderOffsetable {
  255. return self.constrainTo(other, relation: .GreaterThanOrEqualTo)
  256. }
  257. func greaterThanOrEqualTo(other: View) -> ConstraintBuilderOffsetable {
  258. return self.constrainTo(other, relation: .GreaterThanOrEqualTo)
  259. }
  260. #if os(iOS)
  261. func greaterThanOrEqualTo(other: UILayoutSupport) -> ConstraintBuilderOffsetable {
  262. return self.constrainTo(other, relation: .GreaterThanOrEqualTo)
  263. }
  264. #endif
  265. func greaterThanOrEqualTo(other: Float) -> ConstraintBuilderMultipliable {
  266. return self.constrainTo(other, relation: .GreaterThanOrEqualTo)
  267. }
  268. func greaterThanOrEqualTo(other: Double) -> ConstraintBuilderMultipliable {
  269. return self.constrainTo(Float(other), relation: .GreaterThanOrEqualTo)
  270. }
  271. func greaterThanOrEqualTo(other: CGFloat) -> ConstraintBuilderMultipliable {
  272. return self.constrainTo(Float(other), relation: .GreaterThanOrEqualTo)
  273. }
  274. func greaterThanOrEqualTo(other: Int) -> ConstraintBuilderMultipliable {
  275. return self.constrainTo(Float(other), relation: .GreaterThanOrEqualTo)
  276. }
  277. func greaterThanOrEqualTo(other: UInt) -> ConstraintBuilderMultipliable {
  278. return self.constrainTo(Float(other), relation: .GreaterThanOrEqualTo)
  279. }
  280. func greaterThanOrEqualTo(other: CGSize) -> ConstraintBuilderMultipliable {
  281. return self.constrainTo(other, relation: .GreaterThanOrEqualTo)
  282. }
  283. func greaterThanOrEqualTo(other: CGPoint) -> ConstraintBuilderMultipliable {
  284. return self.constrainTo(other, relation: .GreaterThanOrEqualTo)
  285. }
  286. func greaterThanOrEqualTo(other: EdgeInsets) -> ConstraintBuilderMultipliable {
  287. return self.constrainTo(other, relation: .GreaterThanOrEqualTo)
  288. }
  289. // MARK: multiplier
  290. func multipliedBy(amount: Float) -> ConstraintBuilderPriortizable {
  291. self.multiplier = amount
  292. return self
  293. }
  294. func multipliedBy(amount: Double) -> ConstraintBuilderPriortizable {
  295. return self.multipliedBy(Float(amount))
  296. }
  297. func multipliedBy(amount: CGFloat) -> ConstraintBuilderPriortizable {
  298. return self.multipliedBy(Float(amount))
  299. }
  300. func multipliedBy(amount: Int) -> ConstraintBuilderPriortizable {
  301. return self.multipliedBy(Float(amount))
  302. }
  303. func multipliedBy(amount: UInt) -> ConstraintBuilderPriortizable {
  304. return self.multipliedBy(Float(amount))
  305. }
  306. func dividedBy(amount: Float) -> ConstraintBuilderPriortizable {
  307. self.multiplier = 1.0 / amount;
  308. return self
  309. }
  310. func dividedBy(amount: Double) -> ConstraintBuilderPriortizable {
  311. return self.dividedBy(Float(amount))
  312. }
  313. func dividedBy(amount: CGFloat) -> ConstraintBuilderPriortizable {
  314. return self.dividedBy(Float(amount))
  315. }
  316. func dividedBy(amount: Int) -> ConstraintBuilderPriortizable {
  317. return self.dividedBy(Float(amount))
  318. }
  319. func dividedBy(amount: UInt) -> ConstraintBuilderPriortizable {
  320. return self.dividedBy(Float(amount))
  321. }
  322. // MARK: priority
  323. func priority(priority: Float) -> ConstraintBuilderFinalizable {
  324. self.priority = priority
  325. return self
  326. }
  327. func priority(priority: Double) -> ConstraintBuilderFinalizable {
  328. return self.priority(Float(priority))
  329. }
  330. func priority(priority: CGFloat) -> ConstraintBuilderFinalizable {
  331. return self.priority(Float(priority))
  332. }
  333. func priority(priority: UInt) -> ConstraintBuilderFinalizable {
  334. return self.priority(Float(priority))
  335. }
  336. func priority(priority: Int) -> ConstraintBuilderFinalizable {
  337. return self.priority(Float(priority))
  338. }
  339. func priorityRequired() -> ConstraintBuilderFinalizable {
  340. return self.priority(1000.0)
  341. }
  342. func priorityHigh() -> ConstraintBuilderFinalizable {
  343. return self.priority(750.0)
  344. }
  345. func priorityMedium() -> ConstraintBuilderFinalizable {
  346. #if os(iOS)
  347. return self.priority(500.0)
  348. #else
  349. return self.priority(501.0)
  350. #endif
  351. }
  352. func priorityLow() -> ConstraintBuilderFinalizable {
  353. return self.priority(250.0)
  354. }
  355. // MARK: offset
  356. func offset(amount: Float) -> ConstraintBuilderMultipliable {
  357. self.constant = amount
  358. return self
  359. }
  360. func offset(amount: Double) -> ConstraintBuilderMultipliable {
  361. return self.offset(Float(amount))
  362. }
  363. func offset(amount: CGFloat) -> ConstraintBuilderMultipliable {
  364. return self.offset(Float(amount))
  365. }
  366. func offset(amount: Int) -> ConstraintBuilderMultipliable {
  367. return self.offset(Float(amount))
  368. }
  369. func offset(amount: UInt) -> ConstraintBuilderMultipliable {
  370. return self.offset(Float(amount))
  371. }
  372. func offset(amount: CGPoint) -> ConstraintBuilderMultipliable {
  373. self.constant = amount
  374. return self
  375. }
  376. func offset(amount: CGSize) -> ConstraintBuilderMultipliable {
  377. self.constant = amount
  378. return self
  379. }
  380. func offset(amount: EdgeInsets) -> ConstraintBuilderMultipliable {
  381. self.constant = amount
  382. return self
  383. }
  384. // MARK: inset
  385. func inset(amount: EdgeInsets) -> ConstraintBuilderMultipliable {
  386. self.constant = EdgeInsets(top: amount.top, left: amount.left, bottom: -amount.bottom, right: -amount.right)
  387. return self
  388. }
  389. func insets(amount: EdgeInsets) -> ConstraintBuilderMultipliable {
  390. return self.inset(amount)
  391. }
  392. // MARK: Constraint
  393. var constraint: Constraint {
  394. return self
  395. }
  396. // MARK: install / uninstall
  397. func install() -> [LayoutConstraint] {
  398. return self.installOnView(updateExisting: false)
  399. }
  400. func uninstall() {
  401. self.uninstallFromView()
  402. }
  403. func activate() {
  404. if NSLayoutConstraint.respondsToSelector("activateConstraints:") && self.installInfo != nil {
  405. let layoutConstraints = self.installInfo!.layoutConstraints.allObjects as! [LayoutConstraint]
  406. if layoutConstraints.count > 0 {
  407. NSLayoutConstraint.activateConstraints(layoutConstraints)
  408. }
  409. } else {
  410. self.install()
  411. }
  412. }
  413. func deactivate() {
  414. if NSLayoutConstraint.respondsToSelector("deactivateConstraints:") && self.installInfo != nil {
  415. let layoutConstraints = self.installInfo!.layoutConstraints.allObjects as! [LayoutConstraint]
  416. if layoutConstraints.count > 0 {
  417. NSLayoutConstraint.deactivateConstraints(layoutConstraints)
  418. }
  419. } else {
  420. self.uninstall()
  421. }
  422. }
  423. func installOnView(updateExisting: Bool = false) -> [LayoutConstraint] {
  424. var installOnView: View? = nil
  425. if self.toItem.view != nil {
  426. installOnView = ConstraintBuilder.closestCommonSuperviewFromView(self.fromItem.view, toView: self.toItem.view)
  427. if installOnView == nil {
  428. NSException(name: "Cannot Install Constraint", reason: "No common superview between views", userInfo: nil).raise()
  429. return []
  430. }
  431. } else {
  432. installOnView = self.fromItem.view?.superview
  433. if installOnView == nil {
  434. if self.fromItem.attributes == ConstraintAttributes.Width || self.fromItem.attributes == ConstraintAttributes.Height {
  435. installOnView = self.fromItem.view
  436. }
  437. if installOnView == nil {
  438. NSException(name: "Cannot Install Constraint", reason: "Missing superview", userInfo: nil).raise()
  439. return []
  440. }
  441. }
  442. }
  443. if let installedOnView = self.installInfo?.view {
  444. if installedOnView != installOnView {
  445. NSException(name: "Cannot Install Constraint", reason: "Already installed on different view.", userInfo: nil).raise()
  446. return []
  447. }
  448. return self.installInfo?.layoutConstraints.allObjects as? [LayoutConstraint] ?? []
  449. }
  450. var newLayoutConstraints = [LayoutConstraint]()
  451. let layoutFromAttributes = self.fromItem.attributes.layoutAttributes
  452. let layoutToAttributes = self.toItem.attributes.layoutAttributes
  453. // get layout from
  454. let layoutFrom: View? = self.fromItem.view
  455. // get layout relation
  456. let layoutRelation: NSLayoutRelation = (self.relation != nil) ? self.relation!.layoutRelation : .Equal
  457. for layoutFromAttribute in layoutFromAttributes {
  458. // get layout to attribute
  459. let layoutToAttribute = (layoutToAttributes.count > 0) ? layoutToAttributes[0] : layoutFromAttribute
  460. // get layout constant
  461. var layoutConstant: CGFloat = layoutToAttribute.snp_constantForValue(self.constant)
  462. // get layout to
  463. var layoutTo: View? = self.toItem.view
  464. if layoutTo == nil && layoutToAttribute != .Width && layoutToAttribute != .Height {
  465. layoutTo = installOnView
  466. }
  467. // create layout constraint
  468. let layoutConstraint = LayoutConstraint(
  469. item: layoutFrom!,
  470. attribute: layoutFromAttribute,
  471. relatedBy: layoutRelation,
  472. toItem: layoutTo,
  473. attribute: layoutToAttribute,
  474. multiplier: CGFloat(self.multiplier),
  475. constant: layoutConstant)
  476. // set priority
  477. layoutConstraint.priority = self.priority
  478. // set constraint
  479. layoutConstraint.snp_constraint = self
  480. newLayoutConstraints.append(layoutConstraint)
  481. }
  482. // special logic for updating
  483. if updateExisting {
  484. // get existing constraints for this view
  485. let existingLayoutConstraints = reverse(layoutFrom!.snp_installedLayoutConstraints)
  486. // array that will contain only new layout constraints to keep
  487. var newLayoutConstraintsToKeep = [LayoutConstraint]()
  488. // begin looping
  489. for layoutConstraint in newLayoutConstraints {
  490. // layout constraint that should be updated
  491. var updateLayoutConstraint: LayoutConstraint? = nil
  492. // loop through existing and check for match
  493. for existingLayoutConstraint in existingLayoutConstraints {
  494. if existingLayoutConstraint == layoutConstraint {
  495. updateLayoutConstraint = existingLayoutConstraint
  496. break
  497. }
  498. }
  499. // if we have existing one lets just update the constant
  500. if updateLayoutConstraint != nil {
  501. updateLayoutConstraint!.constant = layoutConstraint.constant
  502. }
  503. // otherwise add this layout constraint to new keep list
  504. else {
  505. newLayoutConstraintsToKeep.append(layoutConstraint)
  506. }
  507. }
  508. // set constraints to only new ones
  509. newLayoutConstraints = newLayoutConstraintsToKeep
  510. }
  511. // add constraints
  512. installOnView!.addConstraints(newLayoutConstraints)
  513. // set install info
  514. self.installInfo = ConstraintInstallInfo(view: installOnView, layoutConstraints: NSHashTable.weakObjectsHashTable())
  515. // store which layout constraints are installed for this constraint
  516. for layoutConstraint in newLayoutConstraints {
  517. self.installInfo!.layoutConstraints.addObject(layoutConstraint)
  518. }
  519. // store the layout constraints against the layout from view
  520. layoutFrom!.snp_installedLayoutConstraints += newLayoutConstraints
  521. // return the new constraints
  522. return newLayoutConstraints
  523. }
  524. func uninstallFromView() {
  525. if let installInfo = self.installInfo,
  526. let installedLayoutConstraints = installInfo.layoutConstraints.allObjects as? [LayoutConstraint] {
  527. if installedLayoutConstraints.count > 0 {
  528. if let installedOnView = installInfo.view {
  529. // remove the constraints from the UIView's storage
  530. installedOnView.removeConstraints(installedLayoutConstraints)
  531. }
  532. // remove the constraints from the from item view
  533. if let fromView = self.fromItem.view {
  534. fromView.snp_installedLayoutConstraints = fromView.snp_installedLayoutConstraints.filter {
  535. return !contains(installedLayoutConstraints, $0)
  536. }
  537. }
  538. }
  539. }
  540. self.installInfo = nil
  541. }
  542. // MARK: private
  543. private let fromItem: ConstraintItem
  544. private var toItem: ConstraintItem
  545. private var relation: ConstraintRelation?
  546. private var constant: Any = Float(0.0)
  547. private var multiplier: Float = 1.0
  548. private var priority: Float = 1000.0
  549. private var installInfo: ConstraintInstallInfo?
  550. private func addConstraint(attributes: ConstraintAttributes) -> ConstraintBuilder {
  551. if self.relation == nil {
  552. self.fromItem.attributes += attributes
  553. }
  554. return self
  555. }
  556. private func constrainTo(other: ConstraintItem, relation: ConstraintRelation) -> ConstraintBuilder {
  557. if other.attributes != ConstraintAttributes.None {
  558. let toLayoutAttributes = other.attributes.layoutAttributes
  559. if toLayoutAttributes.count > 1 {
  560. let fromLayoutAttributes = self.fromItem.attributes.layoutAttributes
  561. if toLayoutAttributes != fromLayoutAttributes {
  562. NSException(name: "Invalid Constraint", reason: "Cannot constrain to multiple non identical attributes", userInfo: nil).raise()
  563. return self
  564. }
  565. other.attributes = ConstraintAttributes.None
  566. }
  567. }
  568. self.toItem = other
  569. self.relation = relation
  570. return self
  571. }
  572. private func constrainTo(other: View, relation: ConstraintRelation) -> ConstraintBuilder {
  573. return constrainTo(ConstraintItem(object: other, attributes: ConstraintAttributes.None), relation: relation)
  574. }
  575. #if os(iOS)
  576. private func constrainTo(other: UILayoutSupport, relation: ConstraintRelation) -> ConstraintBuilder {
  577. return constrainTo(ConstraintItem(object: other, attributes: ConstraintAttributes.None), relation: relation)
  578. }
  579. #endif
  580. private func constrainTo(other: Float, relation: ConstraintRelation) -> ConstraintBuilder {
  581. self.constant = other
  582. return constrainTo(ConstraintItem(object: nil, attributes: ConstraintAttributes.None), relation: relation)
  583. }
  584. private func constrainTo(other: Double, relation: ConstraintRelation) -> ConstraintBuilder {
  585. self.constant = other
  586. return constrainTo(ConstraintItem(object: nil, attributes: ConstraintAttributes.None), relation: relation)
  587. }
  588. private func constrainTo(other: CGSize, relation: ConstraintRelation) -> ConstraintBuilder {
  589. self.constant = other
  590. return constrainTo(ConstraintItem(object: nil, attributes: ConstraintAttributes.None), relation: relation)
  591. }
  592. private func constrainTo(other: CGPoint, relation: ConstraintRelation) -> ConstraintBuilder {
  593. self.constant = other
  594. return constrainTo(ConstraintItem(object: nil, attributes: ConstraintAttributes.None), relation: relation)
  595. }
  596. private func constrainTo(other: EdgeInsets, relation: ConstraintRelation) -> ConstraintBuilder {
  597. self.constant = other
  598. return constrainTo(ConstraintItem(object: nil, attributes: ConstraintAttributes.None), relation: relation)
  599. }
  600. private class func closestCommonSuperviewFromView(fromView: View?, toView: View?) -> View? {
  601. var views = Set<View>()
  602. var fromView = fromView
  603. var toView = toView
  604. do {
  605. if let view = toView {
  606. if views.contains(view) {
  607. return view
  608. }
  609. views.insert(view)
  610. toView = view.superview
  611. }
  612. if let view = fromView {
  613. if views.contains(view) {
  614. return view
  615. }
  616. views.insert(view)
  617. fromView = view.superview
  618. }
  619. } while (fromView != nil || toView != nil)
  620. return nil
  621. }
  622. }
  623. private extension NSLayoutAttribute {
  624. private func snp_constantForValue(value: Any?) -> CGFloat {
  625. // Float
  626. if let float = value as? Float {
  627. return CGFloat(float)
  628. }
  629. // Double
  630. else if let double = value as? Double {
  631. return CGFloat(double)
  632. }
  633. // UInt
  634. else if let int = value as? Int {
  635. return CGFloat(int)
  636. }
  637. // Int
  638. else if let uint = value as? UInt {
  639. return CGFloat(uint)
  640. }
  641. // CGFloat
  642. else if let float = value as? CGFloat {
  643. return float
  644. }
  645. // CGSize
  646. else if let size = value as? CGSize {
  647. if self == .Width {
  648. return size.width
  649. } else if self == .Height {
  650. return size.height
  651. }
  652. }
  653. // CGPoint
  654. else if let point = value as? CGPoint {
  655. #if os(iOS)
  656. switch self {
  657. case .Left, .CenterX, .LeftMargin, .CenterXWithinMargins: return point.x
  658. case .Top, .CenterY, .TopMargin, .CenterYWithinMargins, .Baseline, .FirstBaseline: return point.y
  659. case .Right, .RightMargin: return point.x
  660. case .Bottom, .BottomMargin: return point.y
  661. case .Leading, .LeadingMargin: return point.x
  662. case .Trailing, .TrailingMargin: return point.x
  663. case .Width, .Height, .NotAnAttribute: return CGFloat(0)
  664. }
  665. #else
  666. switch self {
  667. case .Left, .CenterX: return point.x
  668. case .Top, .CenterY, .Baseline: return point.y
  669. case .Right: return point.x
  670. case .Bottom: return point.y
  671. case .Leading: return point.x
  672. case .Trailing: return point.x
  673. case .Width, .Height, .NotAnAttribute: return CGFloat(0)
  674. }
  675. #endif
  676. }
  677. // EdgeInsets
  678. else if let insets = value as? EdgeInsets {
  679. #if os(iOS)
  680. switch self {
  681. case .Left, .CenterX, .LeftMargin, .CenterXWithinMargins: return insets.left
  682. case .Top, .CenterY, .TopMargin, .CenterYWithinMargins, .Baseline, .FirstBaseline: return insets.top
  683. case .Right, .RightMargin: return insets.right
  684. case .Bottom, .BottomMargin: return insets.bottom
  685. case .Leading, .LeadingMargin: return (Config.interfaceLayoutDirection == .LeftToRight) ? insets.left : -insets.right
  686. case .Trailing, .TrailingMargin: return (Config.interfaceLayoutDirection == .LeftToRight) ? insets.right : -insets.left
  687. case .Width, .Height, .NotAnAttribute: return CGFloat(0)
  688. }
  689. #else
  690. switch self {
  691. case .Left, .CenterX: return insets.left
  692. case .Top, .CenterY, .Baseline: return insets.top
  693. case .Right: return insets.right
  694. case .Bottom: return insets.bottom
  695. case .Leading, .LeadingMargin: return (Config.interfaceLayoutDirection == .LeftToRight) ? insets.left : -insets.right
  696. case .Trailing, .TrailingMargin: return (Config.interfaceLayoutDirection == .LeftToRight) ? insets.right : -insets.left
  697. case .Width, .Height, .NotAnAttribute: return CGFloat(0)
  698. }
  699. #endif
  700. }
  701. return CGFloat(0);
  702. }
  703. }
  704. private struct ConstraintInstallInfo {
  705. weak var view: View? = nil
  706. let layoutConstraints: NSHashTable
  707. }
  708. internal func ==(left: ConstraintBuilder, right: ConstraintBuilder) -> Bool {
  709. return (left.fromItem == right.fromItem &&
  710. left.toItem == right.toItem &&
  711. left.relation == right.relation &&
  712. left.multiplier == right.multiplier &&
  713. left.priority == right.priority)
  714. }