Skip to content

Instantly share code, notes, and snippets.

@SintraWorks
Last active May 2, 2022 13:46
Show Gist options
  • Star 5 You must be signed in to star a gist
  • Fork 1 You must be signed in to fork a gist
  • Save SintraWorks/d6d692a09d9a340517529d139894f6c9 to your computer and use it in GitHub Desktop.
Save SintraWorks/d6d692a09d9a340517529d139894f6c9 to your computer and use it in GitHub Desktop.
A scrolling capable drop-in replacement for UIStackView.
//
// ScrollingStackView.swift
//
// Created by Antonio Nunes on 05/08/2018.
// Copyright © 2018 SintraWorks. All rights reserved.
//
// Permission is hereby granted, free of charge, to any person obtaining a copy
// of this software and associated documentation files (the "Software"), to deal
// in the Software without restriction, including without limitation the rights
// to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
// copies of the Software, and to permit persons to whom the Software is furnished
// to do so, subject to the following conditions:
//
// The above copyright notice and this permission notice shall be included in
// all copies or substantial portions of the Software.
//
// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
// IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
// FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
// AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
// LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
// OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
// THE SOFTWARE.
import UIKit
class ScrollingStackView: UIScrollView {
private var stackView = UIStackView()
private var axisConstraint: NSLayoutConstraint?
required init?(coder aDecoder: NSCoder) {
super.init(coder: aDecoder)
commonInit(axis: .vertical)
}
override init(frame: CGRect) {
super.init(frame: frame)
commonInit(axis: .vertical)
}
init(axis: NSLayoutConstraint.Axis = .vertical) {
super.init(frame: CGRect.zero)
commonInit(axis: axis)
}
convenience init(arrangedSubviews: [UIView], axis: NSLayoutConstraint.Axis = .vertical) {
self.init(axis: axis)
for subview in arrangedSubviews {
stackView.addArrangedSubview(subview)
}
}
private func commonInit(axis: NSLayoutConstraint.Axis) {
embed(stackView)
self.axis = axis
}
}
// MARK: - Pass-throughs to UIStackView
// MARK: Managing arranged subviews
extension ScrollingStackView {
func addArrangedSubview(_ view: UIView) {
stackView.addArrangedSubview(view)
}
var arrangedSubviews: [UIView] {
return stackView.arrangedSubviews
}
func insertArrangedSubview(_ view: UIView, at stackIndex: Int) {
stackView.insertArrangedSubview(view, at: stackIndex)
}
func removeArrangedSubview(_ view: UIView) {
stackView.removeArrangedSubview(view)
}
}
// MARK: Configuring the layout
extension ScrollingStackView {
var alignment: UIStackView.Alignment {
get { return stackView.alignment }
set { stackView.alignment = newValue }
}
var axis: NSLayoutConstraint.Axis {
get { return stackView.axis }
set {
axisConstraint?.isActive = false // deactivate existing constraint, if any
switch newValue as NSLayoutConstraint.Axis {
case .vertical:
axisConstraint = stackView.widthAnchor.constraint(equalTo: self.widthAnchor)
case .horizontal:
axisConstraint = stackView.heightAnchor.constraint(equalTo: self.heightAnchor)
}
axisConstraint?.isActive = true // activate new constraint
stackView.axis = newValue
}
}
var isBaselineRelativeArrangement: Bool {
get { return stackView.isBaselineRelativeArrangement }
set { stackView.isBaselineRelativeArrangement = newValue }
}
var distribution: UIStackView.Distribution {
get { return stackView.distribution }
set { stackView.distribution = newValue }
}
var isLayoutMarginsRelativeArrangement: Bool {
get { return stackView.isLayoutMarginsRelativeArrangement }
set { stackView.isLayoutMarginsRelativeArrangement = newValue }
}
var spacing: CGFloat {
get { return stackView.spacing }
set { stackView.spacing = newValue }
}
}
// MARK: Adding space between items
@available(iOS 11.0, *)
extension ScrollingStackView {
func customSpacing(after arrangedSubview: UIView) -> CGFloat {
return stackView.customSpacing(after: arrangedSubview)
}
func setCustomSpacing(_ spacing: CGFloat, after arrangedSubview: UIView) {
stackView.setCustomSpacing(spacing, after: arrangedSubview)
}
class var spacingUseDefault: CGFloat {
return UIStackView.spacingUseDefault
}
class var spacingUseSystem: CGFloat {
return UIStackView.spacingUseSystem
}
}
// MARK: - UIView overrides
extension ScrollingStackView {
override var layoutMargins: UIEdgeInsets {
get { return stackView.layoutMargins }
set { stackView.layoutMargins = newValue }
}
override var directionalLayoutMargins: NSDirectionalEdgeInsets {
get { return stackView.directionalLayoutMargins }
set { stackView.directionalLayoutMargins = newValue }
}
}
extension UIView {
func embed(_ child: UIView) {
child.translatesAutoresizingMaskIntoConstraints = false
addSubview(child)
let constraints = [
child.leftAnchor.constraint(equalTo: self.leftAnchor),
child.rightAnchor.constraint(equalTo: self.rightAnchor),
child.topAnchor.constraint(equalTo: self.topAnchor),
child.bottomAnchor.constraint(equalTo: self.bottomAnchor),
]
NSLayoutConstraint.activate(constraints)
}
}
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment