Repository
https://github.com/hitenkmr/ConstraintMaker
What is ConstraintMaker?
ConstraintMaker allows you to bind views programmatically. It's a light-weight class which wraps AutoLayout with nice and clean syntax. ConstraintMaker provides its own NSLayoutConstraint resulting in code that is more concise and readable.
For Example clone the repo or just download and go to DemoProject and run it, After running the project you will find how easy it is to manage views programmatically with ConstraintMaker.swift class.
Difficulty with Autolayout
Suppose that we have to bind a view from the 4 ends to the superview with 50 margins, then we have to write a huge line of code.
let view1 = UIView.init()
view1.translatesAutoresizingMaskIntoConstraints = false
view1.backgroundColor = UIColor.red
superView.addSubview(view1)
superView.addConstraints([
NSLayoutConstraint.init(item: view1,
attribute: NSLayoutAttribute.leading,
relatedBy: .equal,
toItem: self,
attribute: NSLayoutAttribute.leading,
multiplier: 1.0,
constant: 20),
NSLayoutConstraint.init(item: view1,
attribute: NSLayoutAttribute.trailing,
relatedBy: .equal,
toItem: self,
attribute: NSLayoutAttribute.trailing,
multiplier: 1.0,
constant: 20),
NSLayoutConstraint.init(item: view1,
attribute: NSLayoutAttribute.top,
relatedBy: .equal,
toItem: self,
attribute: NSLayoutAttribute.top,
multiplier: 1.0,
constant: 20),
NSLayoutConstraint.init(item: view1,
attribute: NSLayoutAttribute.bottom,
relatedBy: .equal,
toItem: self,
attribute: NSLayoutAttribute.bottom,
multiplier: 1.0,
constant: 20)
])
But using ConstraintMaker with just few lines of code you can have a workaround to make it possible.
view1.prepareForNewConstraints { (v) in
//give leading space from superview
v?.setLeadingSpaceFromSuperView(leadingSpace: 50)
//give trailing space from superview
v?.setTrailingSpaceFromSuperView(trailingSpace: -50)
//give top space from superview
v?.setTopSpaceFromSuperView(topSpace: 50)
//give bottom space from superview
v?.setBottomSpaceFromSuperView(bottomSpace: -50)
}
What's in ConstraintMaker.swift class
We have an extension to the UIView which contains functions that can be very useful while wrapping the view in another view. Autolayout Engine is smart and does all the work for you but adding a view on its superview programmatically takes huge lines of code that we have mentioned above, but using this class one can easily make use of it with few lines of code.
func prepareForNewConstraints(block : ((UIView?) -> ())!)
{
self.translatesAutoresizingMaskIntoConstraints = false
block(self)
}
If you want to use Auto Layout to dynamically calculate the size and position of your view, you must set 'translatesAutoresizingMaskIntoConstraints' property to false. This method is used to bind the view and with a callback it prepares view magically.
var topSpaceConstraint: NSLayoutConstraint? {
get {
return objc_getAssociatedObject(self, &Keys.TopSpaceConstraint) as? NSLayoutConstraint
}
set (newValue) {
objc_setAssociatedObject(self, &Keys.TopSpaceConstraint, nil,
objc_AssociationPolicy.OBJC_ASSOCIATION_RETAIN_NONATOMIC)
objc_setAssociatedObject(self, &Keys.TopSpaceConstraint, newValue,
objc_AssociationPolicy.OBJC_ASSOCIATION_RETAIN_NONATOMIC)
}
}
var bottomSpaceConstraint: NSLayoutConstraint? {
get {
return objc_getAssociatedObject(self, &Keys.BottomSpaceConstraint) as? NSLayoutConstraint
}
set (newValue) {
objc_setAssociatedObject(self, &Keys.BottomSpaceConstraint, nil,
objc_AssociationPolicy.OBJC_ASSOCIATION_RETAIN_NONATOMIC)
objc_setAssociatedObject(self, &Keys.BottomSpaceConstraint, newValue,
objc_AssociationPolicy.OBJC_ASSOCIATION_RETAIN_NONATOMIC)
}
}
var leadingSpaceConstraint: NSLayoutConstraint? {
get {
return objc_getAssociatedObject(self, &Keys.LeadingSpaceConstraint) as? NSLayoutConstraint
}
set (newValue) {
objc_setAssociatedObject(self, &Keys.LeadingSpaceConstraint, nil,
objc_AssociationPolicy.OBJC_ASSOCIATION_RETAIN_NONATOMIC)
objc_setAssociatedObject(self, &Keys.LeadingSpaceConstraint, newValue,
objc_AssociationPolicy.OBJC_ASSOCIATION_RETAIN_NONATOMIC)
}
}
var trailingSpaceConstraint: NSLayoutConstraint? {
get {
return objc_getAssociatedObject(self, &Keys.TrailingSpaceConstraint) as? NSLayoutConstraint
}
set (newValue) {
objc_setAssociatedObject(self, &Keys.TrailingSpaceConstraint, nil,
objc_AssociationPolicy.OBJC_ASSOCIATION_RETAIN_NONATOMIC)
objc_setAssociatedObject(self, &Keys.TrailingSpaceConstraint, newValue,
objc_AssociationPolicy.OBJC_ASSOCIATION_RETAIN_NONATOMIC)
}
}
var heightConstraint: NSLayoutConstraint? {
get {
return objc_getAssociatedObject(self, &Keys.HeightConstraint) as? NSLayoutConstraint
}
set (newValue) {
objc_setAssociatedObject(self, &Keys.HeightConstraint, nil,
objc_AssociationPolicy.OBJC_ASSOCIATION_RETAIN_NONATOMIC)
objc_setAssociatedObject(self, &Keys.HeightConstraint, newValue,
objc_AssociationPolicy.OBJC_ASSOCIATION_RETAIN_NONATOMIC)
}
}
var widthConstraint: NSLayoutConstraint? {
get {
return objc_getAssociatedObject(self, &Keys.WidthConstraint) as? NSLayoutConstraint
}
set (newValue) {
objc_setAssociatedObject(self, &Keys.WidthConstraint, nil,
objc_AssociationPolicy.OBJC_ASSOCIATION_RETAIN_NONATOMIC)
objc_setAssociatedObject(self, &Keys.WidthConstraint, newValue,
objc_AssociationPolicy.OBJC_ASSOCIATION_RETAIN_NONATOMIC)
}
}
var verticalSpaceConstraint: NSLayoutConstraint? {
get {
return objc_getAssociatedObject(self, &Keys.verticalSpaceConstraint) as? NSLayoutConstraint
}
set (newValue) {
objc_setAssociatedObject(self, &Keys.verticalSpaceConstraint, nil,
objc_AssociationPolicy.OBJC_ASSOCIATION_RETAIN_NONATOMIC)
objc_setAssociatedObject(self, &Keys.verticalSpaceConstraint, newValue,
objc_AssociationPolicy.OBJC_ASSOCIATION_RETAIN_NONATOMIC)
}
}
While setting the constraints on views you can get them at runtime also with Computed properties containing the getters and setters. While setting the specific constraint constant we have user association policy retain-nonatomic which keeps the strong reference to the object.
OBJC_ASSOCIATION_RETAIN_NONATOMIC Specifies a strong reference to the associated object, and that the association is not made atomically.
How to use the class
Just go to the repository and download the ConstraintMaker.swift class and drag and drop it into the project directory.
Suppose that we have to add a view called redView to the ViewController view programmatically so that it have 50 margin from left, right and top and height of 200. So just with few lines of code, this can be achieved.
let redView = UIView.init()
redView.backgroundColor = UIColor.red
self.view.addSubview(redView)
//If you want to use Auto Layout to dynamically calculate the size and position of your view, you must set 'translatesAutoresizingMaskIntoConstraints' property to false
redView.prepareForNewConstraints { (v) in
//give leading space from superview
v?.setLeadingSpaceFromSuperView(leadingSpace: 50)
//give trailing space from superview
v?.setTrailingSpaceFromSuperView(trailingSpace: -50)
//give top space from superview
v?.setTopSpaceFromSuperView(topSpace: 50)
//give bottom space from superview
v?.pinHeightConstraint(constant: 200)
}
Now suppose that we have to add another view called greenView, to make it center horizontally to the superview, vertical space from greenView of 20, width 100 and height 100.
let greenView = UIView.init()
greenView.backgroundColor = UIColor.green
self.view.addSubview(greenView)
//If you want to use Auto Layout to dynamically calculate the size and position of your view, you must set 'translatesAutoresizingMaskIntoConstraints' property to false
greenView.prepareForNewConstraints { (v) in
//center Horizontally view to superview
v?.centerHorizontallyWithView(view: self.view)
//give vertical space from nearest neighbour
v?.setVerticalSpaceFrom(toView: redView, constant: 20)
//pin width
v?.pinWidthConstraint(constant: 100)
//pin height
v?.pinHeightConstraint(constant: 100)
}
I was asked by @amosbastian to review the code as I have some knowledge about iOS development.
I've found the following problems with your code:
In spite of the issues I've mentioned the library itself seems to work correctly. I don't believe this project will be deemed useful by fellow Swift programmers. I hope to be proven wrong, though :)
Every contribution to free open source software is welcome though, and I myself would love to see more projects from you. Keep coding!
PS. Keep that in mind this is not an official review of your code. It's just my opinion of your code. It may influence the utopian review, but you will still, probably, be rewarded. Thank you for contributing to FOSS.
Downvoting a post can decrease pending rewards and make it less visible. Common reasons:
Submit
Hey @nepeta
Here's a tip for your valuable feedback! @Utopian-io loves and incentivises informative comments.
Contributing on Utopian
Learn how to contribute on our website.
Want to chat? Join us on Discord https://discord.gg/h52nFrV.
Vote for Utopian Witness!
Downvoting a post can decrease pending rewards and make it less visible. Common reasons:
Submit
Thanks for the appreciation, though there are some issues which I have to focus on. From my next contribution I will try my best to include all things in a proper way as mentioned above and will try to include comments in the code to make the developers know the exact meaning of each bit of code.
Downvoting a post can decrease pending rewards and make it less visible. Common reasons:
Submit
Isn't that easier to use VFL
Downvoting a post can decrease pending rewards and make it less visible. Common reasons:
Submit
So I made this particular class so that i need not write that much of code.
Downvoting a post can decrease pending rewards and make it less visible. Common reasons:
Submit
Ofcourse it's a matter of choice, but you'll need to import that class everytime, you need to keep up with updates. I'm not sure if it's efficiently work on phone rotations. And by you mean lesser code you'll need to memorize.
And if you start to work with others they'll suffer from it IMO.
Downvoting a post can decrease pending rewards and make it less visible. Common reasons:
Submit
You just have to drag the class into the project and no need to import the class everytime as swift classes are globally available throughout the project, In swift, you only import module eg. UIKit. This class also has full support for rotation as well. Hope it helps.
Downvoting a post can decrease pending rewards and make it less visible. Common reasons:
Submit
Hi, thanks for the contribution! I don't really have much to add to @nepeta's amazing comment, so I hope you take her feedback into account for future contributions.
Need help? Write a ticket on https://support.utopian.io/.
Chat with us on Discord.
[utopian-moderator]
Downvoting a post can decrease pending rewards and make it less visible. Common reasons:
Submit
Hey @hitenkmr
Thanks for contributing on Utopian.
We’re already looking forward to your next contribution!
Contributing on Utopian
Learn how to contribute on our website or by watching this tutorial on Youtube.
Want to chat? Join us on Discord https://discord.gg/h52nFrV.
Vote for Utopian Witness!
Downvoting a post can decrease pending rewards and make it less visible. Common reasons:
Submit