swift - 将 subview Controller 放置在父 View 内部

标签 swift xcode layout

我试图将 subview Controller 放置在父 View Controller 中,但是我总是收到错误...

由于未捕获的异常“NSGenericException”而终止应用程序,原因:“无法激活带有 anchor 的约束,因为它们没有共同的祖先。约束或其 anchor 是否引用不同 View 层次结构中的项目?这是非法的。”

有人可以帮我吗?我不知道发生了什么。

父 View Controller

class MainView: UIViewController {

    let settingsIcon = UIButton(type: .custom)

    override func viewDidLoad() {
        super.viewDidLoad()

        view.backgroundColor = .white
        addQuickFactView()
        setupNavBar()
    }

    func setupNavBar() {
        navigationItem.title = "Home"
        navigationController?.navigationBar.barTintColor = .clear

        settingsIcon.setImage(UIImage(named: "SettingsIcon"), for: .normal)
        settingsIcon.addTarget(self, action: #selector(goToSettings), for: .touchUpInside)
        settingsIcon.heightAnchor.constraint(equalToConstant: 50).isActive = true
        settingsIcon.widthAnchor.constraint(equalTo: settingsIcon.heightAnchor).isActive = true
        let leftButton = UIBarButtonItem(customView: settingsIcon)
        self.navigationItem.leftBarButtonItem = leftButton
    }
    @objc func goToSettings() {
        navigationController?.pushViewController(Settings(), animated: true)
    }

    func addQuickFactView() {
        addChild(QuickFact())
        view.addSubview(QuickFact().view)
        QuickFact().didMove(toParent: self)
        QuickFact().view.translatesAutoresizingMaskIntoConstraints = false
        QuickFact().view.rightAnchor.constraint(equalTo: view.rightAnchor).isActive = true
        QuickFact().view.leftAnchor.constraint(equalTo: view.leftAnchor).isActive = true
        QuickFact().view.topAnchor.constraint(equalTo: view.topAnchor).isActive = true
        QuickFact().view.heightAnchor.constraint(equalToConstant: 200).isActive = true

    }
}

subview Controller

class QuickFact: UIViewController {

    var category = ""
    var factString = ""
    let header = UILabel()
    let fact = UITextView()

    let facts = [["Recycling cardboard only takes 75% of the energy required to make new cardboard.", "Over 90% of all products shipped in the U.S. are shipped in corrugated boxes, which makes up more than 400 billion square feet of cardboard.", "Around 80% of retailers and grocers recycle cardboard.", "70% of corrugated cardboard is recovered for recycling., Approximately 100 billion cardboard boxes are produced each year in the U.S.", "One ton of recycled cardboard saves 46 gallons of oil.", "One ton of recycled cardboard saves 9 cubic yards of landfill space."],
        ["Nearly half of the food in the U.S. goes to waste - approximately 3,000 pounds per second.","Only about 5% of food is diverted from landfill.","The U.S. produces approximately 34 million tons of food waste each year.","Food scraps make up almost 12% of municipal solid waste generated in the U.S.","In 2015, about 137.7 million tons of MSW were landfilled. Food was the largest component at about 22%."],
        ["2.5 million plastic bottles are thrown away every hour in America.","Recycling plastic takes 88% less energy than making it from raw materials.","Enough plastic is thrown away each year to circle the earth four times.","Only 23% of disposable water bottles are recycled.","Plastic bags can take up to 1,000 years to decompose.","Recycling one ton of plastic saves the equivalent of 1,000–2,000 gallons of gasoline.","Recycling plastic saves twice as much energy as burning it in an incinerator.","Styrofoam never decomposes.","The world produces more than 14 million tons of Polystyrene (plastic foam) each year.","Recycling one ton of plastic bottles saves the equivalent energy usage of a two person household for one year."],
        ["A modern glass bottle would take 4,000 years or more to decompose -- and even longer if it's in landfill.","Glass can be recycled and re-manufactured an infinite amount of times and never wear out.","More than 28 billion glass bottles and jars go to landfills every year. That's enough to fill two Empire State Buildings every three weeks.","A modern glass bottle would take 4,000 years or more to decompose − and even longer if it’s in landfill."],
        ["Americans use 85 million tons of paper per year which is about 680 pounds per person.","70% of the total waste in offices is paper waste.","Recycling one ton of paper saves 7,000 gallons of water.","The average office worker uses 10,000 sheets of paper per year.","American businesses use around 21 million tons of paper - with about 750,000 copies made every minute.","Each ton of recycled paper can save 17 mature trees.","Recycling a stack of newspaper just 3 feet high saves one tree.","Approximately 1 billion trees worth of paper are thrown away every year in the U.S."],
        ["The average person has the opportunity to recycle more than 25,000 cans in their life.","An aluminum can can be recycled and back on a grocery store shelf as a new can in as little as 60 days.","Aluminum can be recycled forever without any loss of quality.","Aluminum can be recycled using only 5% of the energy used to make the product from new.","Americans throw away 25 billion Styrofoam coffee cups every year.","Recycling a single aluminum can saves enough energy to power a TV for 3 hours."],
        ["About 11 million tons of textiles end up in U.S. landfills each year — an average of about 70 pounds per person.","In 2007, 1.8 million tons of e-waste ended up in landfills.","The average person generates 4.4 pounds of solid waste every day.","In 2014, The U.S. generated 258 million tons of municipal solid waste (MSW).","The EPA estimates that 75% of the American waste stream is recyclable, but we only recycle about 30% of it.","94% of the U.S. population has access to some type of recycling program.","Americans generate an additional 5 million tons of waste throughout the holidays.","Americans throw away enough trash in an average year to circle the earth 24 times.","Electronic waste totals approximately 2% of the waste stream in the U.S.","On average, it costs $30 per ton to recycle trash, $50 to send it to the landfill and $65 to $75 to incinerate it."]]

    override func viewDidLoad() {
        super.viewDidLoad()
        view.backgroundColor = .white
        setupBackgroundColorAndFactString()
        setupAndPositionHeader()
        setupAndPositionFact()
    }

    func setupBackgroundColorAndFactString() {
        let backgroundRandom = Int.random(in: 0...6)
        let factRandom = Int.random(in: 0...facts[backgroundRandom].count - 1)
        if backgroundRandom == 0 {
            view.backgroundColor = .gray
            category = "Cardboard"
            factString = facts[0][factRandom]
        }
        if backgroundRandom == 1 {
            view.backgroundColor = .green
            category = "Organics"
            factString = facts[1][factRandom]
        }
        if backgroundRandom == 2 {
            view.backgroundColor = .blue
            category = "Plastic"
            factString = facts[2][factRandom]
        }
        if backgroundRandom == 3 {
            view.backgroundColor = .cyan
            category = "Glass"
            factString = facts[3][factRandom]
        }
        if backgroundRandom == 4 {
            view.backgroundColor = .white
            category = "Paper"
            factString = facts[4][factRandom]
        }
        if backgroundRandom == 5 {
            view.backgroundColor = .lightGray
            category = "Metal"
            factString = facts[5][factRandom]
        }
        if backgroundRandom == 6 {
            view.backgroundColor = .orange
            category = "General"
            factString = facts[6][factRandom]
        }
    }

    func setupAndPositionHeader() {
        header.text = "Did You Know..."
        header.textColor = .white
        header.textAlignment = .left
        view.addSubview(fact)

        header.translatesAutoresizingMaskIntoConstraints = false
        header.rightAnchor.constraint(equalTo: view.rightAnchor).isActive = true
        header.leftAnchor.constraint(equalTo: view.leftAnchor).isActive = true
        header.topAnchor.constraint(equalTo: view.topAnchor).isActive = true
        header.heightAnchor.constraint(equalToConstant: 50).isActive = true
    }

    func setupAndPositionFact() {
        fact.text = category + "\n" + factString
        fact.textColor = .white
        fact.textAlignment = .left
        fact.isScrollEnabled = false
        view.addSubview(fact)

        fact.translatesAutoresizingMaskIntoConstraints = false
        fact.rightAnchor.constraint(equalTo: view.rightAnchor).isActive = true
        fact.leftAnchor.constraint(equalTo: view.leftAnchor).isActive = true
        fact.topAnchor.constraint(equalTo: header.bottomAnchor).isActive = true
        fact.bottomAnchor.constraint(equalTo: view.bottomAnchor).isActive = true
    }
}

最佳答案

您收到此错误是因为您将 fact 添加为 subview 而不是 header,因此由于 header 不在 View 中它与 View 层次结构中的其他 View 之间不能有约束:

view.addSubview(fact)
[...]
header.rightAnchor.constraint(equalTo: view.rightAnchor).isActive = true

将其更改为:

view.addSubview(header)
[...]
header.rightAnchor.constraint(equalTo: view.rightAnchor).isActive = true

此外,每次您说 QuickFact() 时,您都会创建一个 QuickFact View Controller 的全新实例。您需要调用此初始化程序一次并使用生成的实例执行您想要的操作。如果不这样做,您将得到与以前相同的错误,但在这一行:

QuickFact().view.rightAnchor.constraint(equalTo: view.rightAnchor).isActive = true

您需要将 addQuickFactView 函数更改为如下所示:

func addQuickFactView() {
    // Make a single instance of QuickFact and use it later on:
    let quickFactVC = QuickFact()

    addChild(quickFactVC)
    view.addSubview(quickFactVC.view)
    quickFactVC.didMove(toParent: self)
    quickFactVC.view.translatesAutoresizingMaskIntoConstraints = false
    quickFactVC.view.rightAnchor.constraint(equalTo: view.rightAnchor).isActive = true
    quickFactVC.view.leftAnchor.constraint(equalTo: view.leftAnchor).isActive = true
    quickFactVC.view.topAnchor.constraint(equalTo: view.topAnchor).isActive = true
    quickFactVC.view.heightAnchor.constraint(equalToConstant: 200).isActive = true
}

关于swift - 将 subview Controller 放置在父 View 内部,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/60458068/

相关文章:

linux - 在 Ubuntu Linux 的 Swift 3 中将字符串写入文件

xcode - 添加 Swift 文件后无法运行测试

php - APNS 不再适用于 PHP

objective-c - 在 objective-c 中编辑/更新文件后,NSFileSystemFileNumber 发生更改

ios - Objective-C : Auto Layout with existing Custom View changing bounds width and height to 0, 0

安卓错误 :You must supply layout_width attribute

ios - 如何在 ActionSheet 中使用 if/else 语句?

swift - 按下按钮(快速代码)时如何更改为约束前后?

algorithm - Spring Graph Algorithm w节点大小

html - 带侧边栏的居中内容(无标题)