mirror of https://github.com/oxen-io/session-ios
				
				
				
			
			You cannot select more than 25 topics
			Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
		
		
		
		
		
			
		
			
				
	
	
		
			403 lines
		
	
	
		
			16 KiB
		
	
	
	
		
			Swift
		
	
			
		
		
	
	
			403 lines
		
	
	
		
			16 KiB
		
	
	
	
		
			Swift
		
	
// Copyright © 2022 Rangeproof Pty Ltd. All rights reserved.
 | 
						|
 | 
						|
import UIKit
 | 
						|
 | 
						|
// MARK: - Enums
 | 
						|
 | 
						|
public protocol ConstraintUtilitiesEdge {}
 | 
						|
 | 
						|
public extension UIView {
 | 
						|
    enum HorizontalEdge: ConstraintUtilitiesEdge { case left, leading, right, trailing }
 | 
						|
    enum VerticalEdge: ConstraintUtilitiesEdge { case top, bottom }
 | 
						|
    enum HorizontalMargin: ConstraintUtilitiesEdge { case left, leading, right, trailing }
 | 
						|
    enum VerticalMargin: ConstraintUtilitiesEdge { case top, bottom }
 | 
						|
    enum Direction { case horizontal, vertical }
 | 
						|
    enum VerticalDirection { case vertical }
 | 
						|
    enum HorizontalDirection { case horizontal }
 | 
						|
    enum Dimension { case width, height }
 | 
						|
}
 | 
						|
 | 
						|
// MARK: - Anchorable
 | 
						|
 | 
						|
public protocol Anchorable {
 | 
						|
    func anchor(from edge: UIView.HorizontalEdge) -> NSLayoutXAxisAnchor
 | 
						|
    func anchor(from edge: UIView.VerticalEdge) -> NSLayoutYAxisAnchor
 | 
						|
}
 | 
						|
 | 
						|
extension UIView: Anchorable {
 | 
						|
    public func anchor(from edge: UIView.HorizontalEdge) -> NSLayoutXAxisAnchor {
 | 
						|
        switch edge {
 | 
						|
            case .left: return leftAnchor
 | 
						|
            case .leading: return leadingAnchor
 | 
						|
            case .right: return rightAnchor
 | 
						|
            case .trailing: return trailingAnchor
 | 
						|
        }
 | 
						|
    }
 | 
						|
    
 | 
						|
    public func anchor(from edge: UIView.VerticalEdge) -> NSLayoutYAxisAnchor {
 | 
						|
        switch edge {
 | 
						|
            case .top: return topAnchor
 | 
						|
            case .bottom: return bottomAnchor
 | 
						|
        }
 | 
						|
    }
 | 
						|
    
 | 
						|
    public func attribute(from edge: UIView.HorizontalEdge) -> NSLayoutConstraint.Attribute {
 | 
						|
        switch edge {
 | 
						|
            case .left: return .left
 | 
						|
            case .leading: return .leading
 | 
						|
            case .right: return .right
 | 
						|
            case .trailing: return .trailing
 | 
						|
        }
 | 
						|
    }
 | 
						|
    
 | 
						|
    public func attribute(from edge: UIView.HorizontalMargin) -> NSLayoutConstraint.Attribute {
 | 
						|
        switch edge {
 | 
						|
            case .left: return .leftMargin
 | 
						|
            case .leading: return .leadingMargin
 | 
						|
            case .right: return .rightMargin
 | 
						|
            case .trailing: return .trailingMargin
 | 
						|
        }
 | 
						|
    }
 | 
						|
    
 | 
						|
    public func attribute(from edge: UIView.VerticalEdge) -> NSLayoutConstraint.Attribute {
 | 
						|
        switch edge {
 | 
						|
            case .top: return .top
 | 
						|
            case .bottom: return .bottom
 | 
						|
        }
 | 
						|
    }
 | 
						|
    
 | 
						|
    public func attribute(from edge: UIView.VerticalMargin) -> NSLayoutConstraint.Attribute {
 | 
						|
        switch edge {
 | 
						|
            case .top: return .topMargin
 | 
						|
            case .bottom: return .bottomMargin
 | 
						|
        }
 | 
						|
    }
 | 
						|
}
 | 
						|
 | 
						|
extension UILayoutGuide: Anchorable {
 | 
						|
    public func anchor(from edge: UIView.HorizontalEdge) -> NSLayoutXAxisAnchor {
 | 
						|
        switch edge {
 | 
						|
            case .left: return leftAnchor
 | 
						|
            case .leading: return leadingAnchor
 | 
						|
            case .right: return rightAnchor
 | 
						|
            case .trailing: return trailingAnchor
 | 
						|
        }
 | 
						|
    }
 | 
						|
    
 | 
						|
    public func anchor(from edge: UIView.VerticalEdge) -> NSLayoutYAxisAnchor {
 | 
						|
        switch edge {
 | 
						|
            case .top: return topAnchor
 | 
						|
            case .bottom: return bottomAnchor
 | 
						|
        }
 | 
						|
    }
 | 
						|
}
 | 
						|
 | 
						|
public extension NSLayoutConstraint {
 | 
						|
    @discardableResult
 | 
						|
    func setting(isActive: Bool) -> NSLayoutConstraint {
 | 
						|
        self.isActive = isActive
 | 
						|
        return self
 | 
						|
    }
 | 
						|
    
 | 
						|
    @discardableResult
 | 
						|
    func setting(priority: UILayoutPriority) -> NSLayoutConstraint {
 | 
						|
        self.priority = priority
 | 
						|
        return self
 | 
						|
    }
 | 
						|
}
 | 
						|
 | 
						|
public extension Anchorable {
 | 
						|
    @discardableResult
 | 
						|
    func pin(_ constraineeEdge: UIView.HorizontalEdge, to constrainerEdge: UIView.HorizontalEdge, of anchorable: Anchorable, withInset inset: CGFloat = 0) -> NSLayoutConstraint {
 | 
						|
        (self as? UIView)?.translatesAutoresizingMaskIntoConstraints = false
 | 
						|
        
 | 
						|
        return anchor(from: constraineeEdge)
 | 
						|
            .constraint(
 | 
						|
                equalTo: anchorable.anchor(from: constrainerEdge),
 | 
						|
                constant: inset
 | 
						|
            )
 | 
						|
            .setting(isActive: true)
 | 
						|
    }
 | 
						|
    
 | 
						|
    @discardableResult
 | 
						|
    func pin(_ constraineeEdge: UIView.HorizontalEdge, greaterThanOrEqualTo constrainerEdge: UIView.HorizontalEdge, of anchorable: Anchorable, withInset inset: CGFloat = 0) -> NSLayoutConstraint {
 | 
						|
        (self as? UIView)?.translatesAutoresizingMaskIntoConstraints = false
 | 
						|
        
 | 
						|
        return anchor(from: constraineeEdge)
 | 
						|
            .constraint(
 | 
						|
                greaterThanOrEqualTo: anchorable.anchor(from: constrainerEdge),
 | 
						|
                constant: inset
 | 
						|
            )
 | 
						|
            .setting(isActive: true)
 | 
						|
    }
 | 
						|
    
 | 
						|
    @discardableResult
 | 
						|
    func pin(_ constraineeEdge: UIView.HorizontalEdge, lessThanOrEqualTo constrainerEdge: UIView.HorizontalEdge, of anchorable: Anchorable, withInset inset: CGFloat = 0) -> NSLayoutConstraint {
 | 
						|
        (self as? UIView)?.translatesAutoresizingMaskIntoConstraints = false
 | 
						|
        
 | 
						|
        return anchor(from: constraineeEdge)
 | 
						|
            .constraint(
 | 
						|
                lessThanOrEqualTo: anchorable.anchor(from: constrainerEdge),
 | 
						|
                constant: inset
 | 
						|
            )
 | 
						|
            .setting(isActive: true)
 | 
						|
    }
 | 
						|
    
 | 
						|
    @discardableResult
 | 
						|
    func pin(_ constraineeEdge: UIView.VerticalEdge, to constrainerEdge: UIView.VerticalEdge, of anchorable: Anchorable, withInset inset: CGFloat = 0) -> NSLayoutConstraint {
 | 
						|
        (self as? UIView)?.translatesAutoresizingMaskIntoConstraints = false
 | 
						|
        
 | 
						|
        return anchor(from: constraineeEdge)
 | 
						|
            .constraint(
 | 
						|
                equalTo: anchorable.anchor(from: constrainerEdge),
 | 
						|
                constant: inset
 | 
						|
            )
 | 
						|
            .setting(isActive: true)
 | 
						|
    }
 | 
						|
    
 | 
						|
    @discardableResult
 | 
						|
    func pin(_ constraineeEdge: UIView.VerticalEdge, greaterThanOrEqualTo constrainerEdge: UIView.VerticalEdge, of anchorable: Anchorable, withInset inset: CGFloat = 0) -> NSLayoutConstraint {
 | 
						|
        (self as? UIView)?.translatesAutoresizingMaskIntoConstraints = false
 | 
						|
        
 | 
						|
        return anchor(from: constraineeEdge)
 | 
						|
            .constraint(
 | 
						|
                greaterThanOrEqualTo: anchorable.anchor(from: constrainerEdge),
 | 
						|
                constant: inset
 | 
						|
            )
 | 
						|
            .setting(isActive: true)
 | 
						|
    }
 | 
						|
    
 | 
						|
    @discardableResult
 | 
						|
    func pin(_ constraineeEdge: UIView.VerticalEdge, lessThanOrEqualTo constrainerEdge: UIView.VerticalEdge, of anchorable: Anchorable, withInset inset: CGFloat = 0) -> NSLayoutConstraint {
 | 
						|
        (self as? UIView)?.translatesAutoresizingMaskIntoConstraints = false
 | 
						|
        
 | 
						|
        return anchor(from: constraineeEdge)
 | 
						|
            .constraint(
 | 
						|
                lessThanOrEqualTo: anchorable.anchor(from: constrainerEdge),
 | 
						|
                constant: inset
 | 
						|
            )
 | 
						|
            .setting(isActive: true)
 | 
						|
    }
 | 
						|
}
 | 
						|
 | 
						|
// MARK: - View extensions
 | 
						|
 | 
						|
public extension UIView {
 | 
						|
    func pin(_ edges: [ConstraintUtilitiesEdge], to view: UIView) {
 | 
						|
        edges.forEach {
 | 
						|
            switch $0 {
 | 
						|
                case let edge as HorizontalEdge: pin(edge, to: edge, of: view)
 | 
						|
                case let edge as VerticalEdge: pin(edge, to: edge, of: view)
 | 
						|
                default: break
 | 
						|
            }
 | 
						|
        }
 | 
						|
    }
 | 
						|
    
 | 
						|
    func pin(to view: UIView) {
 | 
						|
        [ HorizontalEdge.leading, HorizontalEdge.trailing ].forEach { pin($0, to: $0, of: view) }
 | 
						|
        [ VerticalEdge.top, VerticalEdge.bottom ].forEach { pin($0, to: $0, of: view) }
 | 
						|
    }
 | 
						|
    
 | 
						|
    func pin(to view: UIView, withInset inset: CGFloat) {
 | 
						|
        pin(.leading, to: .leading, of: view, withInset: inset)
 | 
						|
        pin(.top, to: .top, of: view, withInset: inset)
 | 
						|
        view.pin(.trailing, to: .trailing, of: self, withInset: inset)
 | 
						|
        view.pin(.bottom, to: .bottom, of: self, withInset: inset)
 | 
						|
    }
 | 
						|
    
 | 
						|
    @discardableResult
 | 
						|
    func pin(_ constraineeEdge: UIView.HorizontalEdge, toMargin constrainerMargin: UIView.HorizontalMargin, of constrainerView: UIView, withInset inset: CGFloat = 0) -> NSLayoutConstraint {
 | 
						|
        translatesAutoresizingMaskIntoConstraints = false
 | 
						|
        
 | 
						|
        return NSLayoutConstraint(
 | 
						|
            item: self,
 | 
						|
            attribute: attribute(from: constraineeEdge),
 | 
						|
            relatedBy: .equal,
 | 
						|
            toItem: constrainerView,
 | 
						|
            attribute: constrainerView.attribute(from: constrainerMargin),
 | 
						|
            multiplier: 1,
 | 
						|
            constant: inset
 | 
						|
        )
 | 
						|
        .setting(isActive: true)
 | 
						|
    }
 | 
						|
    
 | 
						|
    @discardableResult
 | 
						|
    func pin(_ constraineeEdge: UIView.VerticalEdge, toMargin constrainerMargin: UIView.VerticalMargin, of constrainerView: UIView, withInset inset: CGFloat = 0) -> NSLayoutConstraint {
 | 
						|
        translatesAutoresizingMaskIntoConstraints = false
 | 
						|
        
 | 
						|
        return NSLayoutConstraint(
 | 
						|
            item: self,
 | 
						|
            attribute: attribute(from: constraineeEdge),
 | 
						|
            relatedBy: .equal,
 | 
						|
            toItem: constrainerView,
 | 
						|
            attribute: constrainerView.attribute(from: constrainerMargin),
 | 
						|
            multiplier: 1,
 | 
						|
            constant: inset
 | 
						|
        )
 | 
						|
        .setting(isActive: true)
 | 
						|
    }
 | 
						|
    
 | 
						|
    func pin(toMarginsOf view: UIView) {
 | 
						|
        pin(.top, toMargin: .top, of: view)
 | 
						|
        pin(.leading, toMargin: .leading, of: view)
 | 
						|
        pin(.trailing, toMargin: .trailing, of: view)
 | 
						|
        pin(.bottom, toMargin: .bottom, of: view)
 | 
						|
    }
 | 
						|
    
 | 
						|
    @discardableResult
 | 
						|
    func center(_ direction: Direction, in view: UIView, withInset inset: CGFloat = 0) -> NSLayoutConstraint {
 | 
						|
        translatesAutoresizingMaskIntoConstraints = false
 | 
						|
        let constraint: NSLayoutConstraint = {
 | 
						|
            switch direction {
 | 
						|
            case .horizontal: return centerXAnchor.constraint(equalTo: view.centerXAnchor, constant: inset)
 | 
						|
            case .vertical: return centerYAnchor.constraint(equalTo: view.centerYAnchor, constant: inset)
 | 
						|
            }
 | 
						|
        }()
 | 
						|
        constraint.isActive = true
 | 
						|
        return constraint
 | 
						|
    }
 | 
						|
    
 | 
						|
    func center(in view: UIView) {
 | 
						|
        center(.horizontal, in: view)
 | 
						|
        center(.vertical, in: view)
 | 
						|
    }
 | 
						|
    
 | 
						|
    @discardableResult
 | 
						|
    func center(_ direction: VerticalDirection, against constrainerEdge: UIView.VerticalEdge, of anchorable: Anchorable, withInset inset: CGFloat = 0) -> NSLayoutConstraint {
 | 
						|
        translatesAutoresizingMaskIntoConstraints = false
 | 
						|
        
 | 
						|
        return centerYAnchor
 | 
						|
            .constraint(equalTo: anchorable.anchor(from: constrainerEdge), constant: inset)
 | 
						|
            .setting(isActive: true)
 | 
						|
    }
 | 
						|
    
 | 
						|
    @discardableResult
 | 
						|
    func center(_ direction: HorizontalDirection, against constrainerEdge: UIView.HorizontalEdge, of anchorable: Anchorable, withInset inset: CGFloat = 0) -> NSLayoutConstraint {
 | 
						|
        translatesAutoresizingMaskIntoConstraints = false
 | 
						|
        
 | 
						|
        return centerXAnchor
 | 
						|
            .constraint(equalTo: anchorable.anchor(from: constrainerEdge), constant: inset)
 | 
						|
            .setting(isActive: true)
 | 
						|
    }
 | 
						|
    
 | 
						|
    @discardableResult
 | 
						|
    func set(_ dimension: Dimension, to size: CGFloat) -> NSLayoutConstraint {
 | 
						|
        translatesAutoresizingMaskIntoConstraints = false
 | 
						|
        let constraint: NSLayoutConstraint = {
 | 
						|
            switch dimension {
 | 
						|
            case .width: return widthAnchor.constraint(equalToConstant: size)
 | 
						|
            case .height: return heightAnchor.constraint(equalToConstant: size)
 | 
						|
            }
 | 
						|
        }()
 | 
						|
        constraint.isActive = true
 | 
						|
        return constraint
 | 
						|
    }
 | 
						|
    
 | 
						|
    @discardableResult
 | 
						|
    func set(_ dimension: Dimension, to otherDimension: Dimension, of view: UIView, withOffset offset: CGFloat = 0, multiplier: CGFloat = 1) -> NSLayoutConstraint {
 | 
						|
        translatesAutoresizingMaskIntoConstraints = false
 | 
						|
        let otherAnchor: NSLayoutDimension = {
 | 
						|
            switch otherDimension {
 | 
						|
                case .width: return view.widthAnchor
 | 
						|
                case .height: return view.heightAnchor
 | 
						|
            }
 | 
						|
        }()
 | 
						|
        let constraint: NSLayoutConstraint = {
 | 
						|
            switch dimension {
 | 
						|
                case .width: return widthAnchor.constraint(equalTo: otherAnchor, multiplier: multiplier, constant: offset)
 | 
						|
                case .height: return heightAnchor.constraint(equalTo: otherAnchor, multiplier: multiplier, constant: offset)
 | 
						|
            }
 | 
						|
        }()
 | 
						|
        constraint.isActive = true
 | 
						|
        return constraint
 | 
						|
    }
 | 
						|
    
 | 
						|
    @discardableResult
 | 
						|
    func set(_ dimension: Dimension, greaterThanOrEqualTo size: CGFloat) -> NSLayoutConstraint {
 | 
						|
        translatesAutoresizingMaskIntoConstraints = false
 | 
						|
        let constraint: NSLayoutConstraint = {
 | 
						|
            switch dimension {
 | 
						|
            case .width: return widthAnchor.constraint(greaterThanOrEqualToConstant: size)
 | 
						|
            case .height: return heightAnchor.constraint(greaterThanOrEqualToConstant: size)
 | 
						|
            }
 | 
						|
        }()
 | 
						|
        constraint.isActive = true
 | 
						|
        return constraint
 | 
						|
    }
 | 
						|
    
 | 
						|
    @discardableResult
 | 
						|
    func set(_ dimension: Dimension, greaterThanOrEqualTo otherDimension: Dimension, of view: UIView, withOffset offset: CGFloat = 0, multiplier: CGFloat = 1) -> NSLayoutConstraint {
 | 
						|
        translatesAutoresizingMaskIntoConstraints = false
 | 
						|
        let otherAnchor: NSLayoutDimension = {
 | 
						|
            switch otherDimension {
 | 
						|
                case .width: return view.widthAnchor
 | 
						|
                case .height: return view.heightAnchor
 | 
						|
            }
 | 
						|
        }()
 | 
						|
        let constraint: NSLayoutConstraint = {
 | 
						|
            switch dimension {
 | 
						|
            case .width: return widthAnchor.constraint(greaterThanOrEqualTo: otherAnchor, multiplier: multiplier, constant: offset)
 | 
						|
            case .height: return heightAnchor.constraint(greaterThanOrEqualTo: otherAnchor, multiplier: multiplier, constant: offset)
 | 
						|
            }
 | 
						|
        }()
 | 
						|
        constraint.isActive = true
 | 
						|
        return constraint
 | 
						|
    }
 | 
						|
    
 | 
						|
    @discardableResult
 | 
						|
    func set(_ dimension: Dimension, lessThanOrEqualTo size: CGFloat) -> NSLayoutConstraint {
 | 
						|
        translatesAutoresizingMaskIntoConstraints = false
 | 
						|
        let constraint: NSLayoutConstraint = {
 | 
						|
            switch dimension {
 | 
						|
            case .width: return widthAnchor.constraint(lessThanOrEqualToConstant: size)
 | 
						|
            case .height: return heightAnchor.constraint(lessThanOrEqualToConstant: size)
 | 
						|
            }
 | 
						|
        }()
 | 
						|
        constraint.isActive = true
 | 
						|
        return constraint
 | 
						|
    }
 | 
						|
    
 | 
						|
    @discardableResult
 | 
						|
    func set(_ dimension: Dimension, lessThanOrEqualTo otherDimension: Dimension, of view: UIView, withOffset offset: CGFloat = 0, multiplier: CGFloat = 1) -> NSLayoutConstraint {
 | 
						|
        translatesAutoresizingMaskIntoConstraints = false
 | 
						|
        let otherAnchor: NSLayoutDimension = {
 | 
						|
            switch otherDimension {
 | 
						|
                case .width: return view.widthAnchor
 | 
						|
                case .height: return view.heightAnchor
 | 
						|
            }
 | 
						|
        }()
 | 
						|
        let constraint: NSLayoutConstraint = {
 | 
						|
            switch dimension {
 | 
						|
            case .width: return widthAnchor.constraint(lessThanOrEqualTo: otherAnchor, multiplier: multiplier, constant: offset)
 | 
						|
            case .height: return heightAnchor.constraint(lessThanOrEqualTo: otherAnchor, multiplier: multiplier, constant: offset)
 | 
						|
            }
 | 
						|
        }()
 | 
						|
        constraint.isActive = true
 | 
						|
        return constraint
 | 
						|
    }
 | 
						|
    
 | 
						|
    func setContentHugging(to priority: UILayoutPriority) {
 | 
						|
        setContentHuggingPriority(priority, for: .vertical)
 | 
						|
        setContentHuggingPriority(priority, for: .horizontal)
 | 
						|
    }
 | 
						|
    
 | 
						|
    func setContentHugging(_ direction: Direction, to priority: UILayoutPriority) {
 | 
						|
        switch direction {
 | 
						|
            case .vertical: setContentHuggingPriority(priority, for: .vertical)
 | 
						|
            case .horizontal: setContentHuggingPriority(priority, for: .horizontal)
 | 
						|
        }
 | 
						|
    }
 | 
						|
    
 | 
						|
    func setCompressionResistance(to priority: UILayoutPriority) {
 | 
						|
        setContentCompressionResistancePriority(priority, for: .vertical)
 | 
						|
        setContentCompressionResistancePriority(priority, for: .horizontal)
 | 
						|
    }
 | 
						|
    
 | 
						|
    func setCompressionResistance(_ direction: Direction, to priority: UILayoutPriority) {
 | 
						|
        switch direction {
 | 
						|
            case .vertical: setContentCompressionResistancePriority(priority, for: .vertical)
 | 
						|
            case .horizontal: setContentCompressionResistancePriority(priority, for: .horizontal)
 | 
						|
        }
 | 
						|
    }
 | 
						|
}
 |