3

init(frame:)Apple 的教程描述了和init?(coder:)as之间的区别

您通常通过以下两种方式之一创建视图:以编程方式初始化视图,或允许情节提要加载视图。每种方法都有一个相应的初始化程序: init(frame:)用于以编程方式初始化视图和 init?(coder:)从情节提要加载视图。您将需要在自定义控件中实现这两种方法。在设计应用程序时,Interface Builder 会在您将视图添加到画布时以编程方式将其实例化。在运行时,您的应用会从情节提要中加载视图。

我对“以编程方式初始化”和“由情节提要加载”的描述感到非常困惑。假设我有一个UIView被调用的子类MyView,“以编程方式初始化”是否意味着我编写代码来添加一个实例,MyView例如:

override func viewDidLoad() {      
        super.viewDidLoad()
        let myView = MyView()  // init(frame:) get invoked here??
}

虽然init?(coder:)Main.storyboard我从对象库中拖动一个UIView然后在身份检查器中将其类设置为MyView

此外,在我的 xcode 项目中,这两种方法最终得到了不同的模拟器布局和Main.storyboard相同的代码: 在此处输入图像描述

import UIKit

@IBDesignable
class RecordView: UIView {

    @IBInspectable
    var borderColor: UIColor = UIColor.clear {
        didSet {
            self.layer.borderColor = borderColor.cgColor
        }
    }

    @IBInspectable
    var borderWidth: CGFloat = 20 {
        didSet {
            layer.borderWidth = borderWidth
        }
    }

    @IBInspectable
    var cornerRadius: CGFloat = 100 {
        didSet {
            layer.cornerRadius = cornerRadius
        }
    }

    private var fillView = UIView()

    required init?(coder aDecoder: NSCoder) {
        super.init(coder: aDecoder)
        setupFillView()
    }

    override init(frame: CGRect) {
        super.init(frame: frame)
        setupFillView()
    }



    private func setupFillView() {
        let radius = (self.cornerRadius - self.borderWidth) * 0.95
        fillView.frame = CGRect(origin: CGPoint.zero, size: CGSize(width: radius * 2, height: radius * 2))
        fillView.center = CGPoint(x: self.bounds.midX, y: self.bounds.midY)
        fillView.layer.cornerRadius = radius
        fillView.backgroundColor = UIColor.red
        self.addSubview(fillView)
    }

    override func layoutSubviews() {
        super.layoutSubviews()
    }

    func didClick() {
        UIView.animate(withDuration: 1.0, animations: {
            self.fillView.transform = CGAffineTransform(scaleX: 0.6, y: 0.6)
        }) { (true) in
            print()
        }
    }
}

为什么他们的行为不同?(我从对象库中拖动一个UIView并将其类设置为 RecordView)

4

2 回答 2

2

我对“以编程方式初始化”和“由情节提要加载”的描述感到非常困惑。

基于对象的编程是关于类和实例的。您需要创建一个类的实例。使用 Xcode,有两种截然不同的方法来获取类的实例:

  • 您的代码创建实例

  • 您加载一个笔尖(例如故事板中的视图控制器视图),笔尖加载过程创建实例并将其交给您

在这两种情况下调用的初始化程序是不同的。如果您的代码创建了一个 UIView 实例,那么您必须调用的指定初始化程序是init(frame:). 但是如果 nib 创建视图,则 nib 加载过程调用的指定初始化程序是init(coder:).

因此,如果你有一个 UIView 子类并且你想覆盖初始化器,你必须考虑将调用哪个初始化器(基于视图实例的创建方式)。

于 2017-12-03T16:43:45.627 回答
1

init?(coder:)首先,您在和之间的划分init(frame:)基本上是正确的。前者在实际运行应用程序时实例化情节提要场景时使用,但后者在您使用let foo = RecordView()或以编程方式实例化它时使用let bar = RecordView(frame: ...)。此外,在 IB 中init(frame:)预览@IBDesignable视图时使用。

center其次,关于您的问题,我建议您fillViewsetupFillView. 问题是当init被调用时,你通常不知道bounds最终会是什么。您应该设置centerin layoutSubviews,每次视图更改大小时都会调用它。

class RecordView: UIView {  // this is the black circle with a white border

    private var fillView = UIView() // this is the inner red circle

    required init?(coder aDecoder: NSCoder) {
        super.init(coder: aDecoder)
        setupFillView()
    }

    override init(frame: CGRect = .zero) {
        super.init(frame: frame)
        setupFillView()
    }

    private func setupFillView() {
        fillView.backgroundColor = .red
        self.addSubview(fillView)
    }

    override func layoutSubviews() {
        super.layoutSubviews()

        let radius = (cornerRadius - borderWidth) * 0.95       // these are not defined in this snippet, but I simply assume you omitted them for the sake of brevity?
        fillView.frame = CGRect(origin: .zero, size: CGSize(width: radius * 2, height: radius * 2))
        fillView.layer.cornerRadius = radius
        fillView.center = CGPoint(x: bounds.midX, y: bounds.midY)
    }
}
于 2017-12-03T16:38:37.667 回答