Навигация по ячейкам - PullRequest
       9

Навигация по ячейкам

0 голосов
/ 01 июля 2018

Дорогие коллеги, пожалуйста, помогите мне. Я начал изучать Swift 2 месяца назад, и это будет мое первое приложение, если я смогу закончить ... Прежде всего, я не использую раскадровку. В моем приложении в главном окне есть наборы представлений. При касании изображения в ячейках, например изображения кошки в ячейке кошки, должен открыться файл Cat.swift. Но я пока не могу это реализовать. И точно так же, если вы нажмете на изображение собаки, файл Dog.swift должен быть просмотрен. Вот код.

это главный экран при открытии приложения:

import UIKit

class FeaturedAnimalsController: UICollectionViewController, UICollectionViewDelegateFlowLayout {

    var animalCategories: [AnimalCategory]?

    private let cellId = "cellId"

    override func viewDidLoad() {
        super.viewDidLoad()

        animalCategories = AnimalCategory.sampleAnimalCategories()

        collectionView?.backgroundColor = .white
        collectionView?.register(CategoryCell.self, forCellWithReuseIdentifier: cellId)
    }

    func showAnimalDetailForAnimal(animal: Animal) {
        let layout = UICollectionViewFlowLayout()
        let animalDetailController = AnimalDetailController(collectionViewLayout: layout)
        animalDetailController.animal = animal
        navigationController?.pushViewController(animalDetailController, animated: true)
    }


    override func collectionView(_ collectionView: UICollectionView, numberOfItemsInSection section: Int) -> Int {
        if let count = animalCategories?.count {
            return count
        }
        return 0
    }

    override func collectionView(_ collectionView: UICollectionView, cellForItemAt indexPath: IndexPath) -> UICollectionViewCell {

        let cell = collectionView.dequeueReusableCell(withReuseIdentifier: cellId, for: indexPath) as! CategoryCell

        cell.featuredAnimalsController = self

        cell.animalCategory = animalCategories?[indexPath.item]

        return cell
    }

    func collectionView(_ collectionView: UICollectionView, layout collectionViewLayout: UICollectionViewLayout, sizeForItemAt indexPath: IndexPath) -> CGSize {
        return CGSize(width: view.frame.width, height: 230)
    }

    override func didReceiveMemoryWarning() {
        super.didReceiveMemoryWarning()
        // Dispose of any resources that can be recreated.
    } 
}

Это CategoryCell:

import UIKit

class CategoryCell: UICollectionViewCell, UICollectionViewDataSource, UICollectionViewDelegate, UICollectionViewDelegateFlowLayout {

    var featuredAnimalsController: FeaturedAnimalsController?

    var animalCategory: AnimalCategory? {
        didSet {
            if let name = animalCategory?.name {
                nameLabel.text = name
            }
        }
    }

    private let cellId = "animalCellId"

    func collectionView(_ collectionView: UICollectionView, numberOfItemsInSection section: Int) -> Int {
        if let count = animalCategory?.animals?.count {
            return count
        }
        return 0
    }


    func collectionView(_ collectionView: UICollectionView, cellForItemAt indexPath: IndexPath) -> UICollectionViewCell {

        let cell = collectionView.dequeueReusableCell(withReuseIdentifier: cellId, for: indexPath) as! AnimalCell
        cell.animal = animalCategory?.animals?[indexPath.item]
        return cell
    }

    func collectionView(_ collectionView: UICollectionView, layout collectionViewLayout: UICollectionViewLayout, sizeForItemAt indexPath: IndexPath) -> CGSize {
        return CGSize(width: 100, height: frame.height - 32)
    }

    func collectionView(_ collectionView: UICollectionView, layout collectionViewLayout: UICollectionViewLayout, insetForSectionAt section: Int) -> UIEdgeInsets {
        return UIEdgeInsets(top: 0, left: 14, bottom: 0, right: 14)
    }

    func collectionView(_ collectionView: UICollectionView, didSelectItemAt indexPath: IndexPath) {
        if let animal = animalCategory?.animals![indexPath.item] {
            featuredAnimalsController?.showAnimalDetailForAnimal(animal: animal)
        }
    }

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

        setupViews()
    }

    required init?(coder aDecoder: NSCoder) {
        fatalError("init(coder:) has not been implemented")
    }

    let nameLabel: UILabel = {
        let label = UILabel()
        label.text = "Cardiology"
        label.font = UIFont.systemFont(ofSize: 16)
        //label.numberOfLines = 1  --> DEFAULT
        label.translatesAutoresizingMaskIntoConstraints = false
        return label
    }()

    let animalsCollectionView: UICollectionView = {
        let layout = UICollectionViewFlowLayout()
        layout.scrollDirection = .horizontal
        let collectionView = UICollectionView(frame: .zero, collectionViewLayout: layout)
        collectionView.backgroundColor = .clear
        collectionView.translatesAutoresizingMaskIntoConstraints = false
        return collectionView
    }()

    func setupViews() {
        backgroundColor = .clear

        animalsCollectionView.dataSource = self
        animalsCollectionView.delegate = self

        animalsCollectionView.register(AnimalCell.self, forCellWithReuseIdentifier: cellId)


        addSubview(nameLabel)
        addSubview(animalsCollectionView)

        addConstraints(NSLayoutConstraint.constraints(withVisualFormat: "H:|-14-[v0]|", options: NSLayoutFormatOptions(), metrics: nil, views: ["v0": nameLabel]))
        addConstraints(NSLayoutConstraint.constraints(withVisualFormat: "H:|[v0]|", options: NSLayoutFormatOptions(), metrics: nil, views: ["v0": animalsCollectionView]))
        addConstraints(NSLayoutConstraint.constraints(withVisualFormat: "V:|[nameLabel(30)][v0]|", options: NSLayoutFormatOptions(), metrics: nil, views: ["v0": animalsCollectionView, "nameLabel": nameLabel]))

    }


}

class AnimalCell: UICollectionViewCell {

    var animal: Animal? {
        didSet {
            if let name = animal?.name {
                nameLabel.text = name
            }
            categoryLabel.text = animal?.category

            if let imageName = animal?.imageName {
                imageView.image = UIImage(named: imageName)
            }
        }
    }

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

        setupViews()
    }

    required init?(coder aDecoder: NSCoder) {
        fatalError("init(coder:) has not been implemented")
    }

    func setupViews() {
        backgroundColor = .clear

        addSubview(categoryLabel)
        addSubview(imageView)
        addSubview(nameLabel)

        categoryLabel.frame = CGRect(x: 0, y: frame.width + 38, width: frame.width, height: 20)
        imageView.frame = CGRect(x: 0, y: 0, width: frame.width, height: frame.width)
        nameLabel.frame = CGRect(x: 0, y: frame.width + 2, width: frame.width, height: 40)

    }

    let imageView: UIImageView = {
        let iv = UIImageView()
        //iv.image = UIImage(named: "cat")
        iv.contentMode = .scaleAspectFill
        iv.translatesAutoresizingMaskIntoConstraints = false
        iv.layer.cornerRadius = 16
        iv.layer.masksToBounds = true
        return iv
    }()

    let nameLabel: UILabel = {
        let label = UILabel()
        label.text = ""
        label.font = UIFont.systemFont(ofSize: 14)
        label.numberOfLines = 2
        return label
    }()

    let categoryLabel: UILabel = {
        let label = UILabel()
        label.text = ""
        label.font = UIFont.systemFont(ofSize: 13)
        label.numberOfLines = 2
        return label
    }()  
}

Это модель категории животных:

import UIKit

class AnimalCategory: NSObject {

    var name: String?
    var animals: [Animal]?

    static func sampleAnimalCategories() -> [AnimalCategory] {


        let catFamilyCategory = AnimalCategory()
        catFamilyCategory.name = "Cat Family"
        var catFamilyAnimals = [Animal]()

        let catAnimal = Animal()
        catAnimal.name = "Cat"
        catAnimal.imageName = "cat"
        catAnimal.category = "Lovely"
        //catAnimal.dvcName = Cat.self()
        catFamilyAnimals.append(catAnimal)

        catFamilyCategory.animals = catFamilyAnimals



        let dogFamilyCategory = AnimalCategory()
        dogFamilyCategory.name = "Dog Family"
        var dogFamilyAnimals = [Animal]()

        let dogAnimal = Animal()
        dogAnimal.name = "Dog"
        dogAnimal.imageName = "dog"
        dogAnimal.category = "Friendly"
        //dogAnimal.dvcName = Dog.self
        dogFamilyAnimals.append(dogAnimal)

        dogFamilyCategory.animals = dogFamilyAnimals


        return [catFamilyCategory, dogFamilyCategory]
    }
}

Это модель животного:

import UIKit

class Animal: NSObject {

    //var id: NSNumber?
    var name: String?
    var category: String?
    var imageName: String?
    //var dvc: AnimalDetailController?

}

Это контроллер подробностей животных и пример cat.swift:

import UIKit

class AnimalDetailController: UICollectionViewController, UICollectionViewDelegateFlowLayout {

    var animal: Animal? {
        didSet {
            navigationItem.title = animal?.name
        }
    }
}

class Cat: AnimalDetailController {
    override func viewDidLoad() {
        super.viewDidLoad()

        view.backgroundColor = .brown
    }
}

После некоторых изменений, благодаря @ Paulw11, приведен код для представления конкретного подробного контроллера вида для каждого животного:

func showAnimalDetailFor(_ animal: Animal) {
        let layout = UICollectionViewFlowLayout()

        var dvc = animal.dvc
        dvc = AnimalDetailController(collectionViewLayout: layout)
        self.show(dvc, sender: self)
    }

Вот модель категории животных и животных:

struct Animal {

    let name: String
    let category: String
    let imageName: String
    let dvc: AnimalDetailController

}

struct AnimalCategory {

    var name: String
    private var _animals = [Animal]()

    var animals: [Animal] {
        get {
            return self._animals
        }
    }

    init(name: String) {
        self.name = name
    }

    static var sampleAnimalCategories: [AnimalCategory] = {

        var dogFamily = AnimalCategory(name: "Dog Family")
        dogFamily.addAnimal(Animal(name: "Dog", category: "Friendly", imageName: "dog", dvc: DogDetailViewController()))

        var catFamily = AnimalCategory(name: "Cat Family")
        catFamily.addAnimal(Animal(name: "Cat", category: "Lovely", imageName: "cat", dvc: CatDetailViewController()))

        let categories = [dogFamily,catFamily]
        return categories
    }()

    mutating func addAnimal(_ animal: Animal) {
        self._animals.append(animal)
    }

}

Контроллер класса животных и деталей:

class AnimalDetailController: UICollectionViewController, UICollectionViewDelegateFlowLayout {


    var animal: Animal? {
        didSet {
            navigationItem.title = animal?.name
        }
    }
}

Я не буду передавать данные между контроллерами представления. Я планирую сделать конкретные быстрые файлы для каждого животного в клетке, а затем, когда вы нажмете на иконку, я хочу, чтобы эти быстрые файлы появились. И, конечно, я использую контроллер Nav и устанавливаю его в делегате приложения.

Пожалуйста, помогите мне. Спасибо всем.

1 Ответ

0 голосов
/ 01 июля 2018

В целом, я бы предостерег от тесной связи ваших данных и вашего кода, но вы можете делать то, что вы хотите, сохраняя ссылку на класс контроллера представления в вашей модели данных; Вы почти правильно сделали это в своем комментарии; вам просто не нужно () после self.

Лучше всего использовать структуры, если вам не нужна изменчивость, поэтому ваши объекты Animal и AnimalCategory могут быть структурами. Вы также можете избежать всех этих опций.

struct Animal {
    let name: String
    let category: String
    let imageName: String
    let dvc: AnimalDetailController.Type
}

struct AnimalCategory {

    var name: String

    private var _animals = [Animal]()

    var animals: [Animal] {
        get {
            return self._animals
        }
    }

    init(name: String) {
        self.name = name
    }

    static var sampleAnimalCategories: [AnimalCategory] = {

        var dogs = AnimalCategory(name: "Dog Family")
        dogs.addAnimal(Animal(name: "Dog", category: "Friendly", imageName: "dog", dvc: DogDetailViewController.self))

        var cats = AnimalCategory(name: "Cat Family")
        cats.addAnimal(Animal(name: "Cat", category: "Lovely", imageName: "cat", dvc: CatDetailViewController.self))

        let categories = [dogs,cats]
        return categories
    }()

    mutating func addAnimal(_ animal: Animal) {
        self._animals.append(animal)
    }

}

Тогда ваш showAnimalDetailForAnimal будет выглядеть примерно так:

func showAnimalDetailFor(_ animal: Animal) {
    let dvc = animal.dvc.init() 
    self.show(dvc, sender: self)
}

Существует также много способов очистки, которые вы можете выполнить в связи с использованием дополнительных принадлежностей. Например, animals и animalCategories могут быть неявно развернутыми необязательными.

Вы не сможете иметь свой AnimalDetailViewController подкласс UICollectionViewController, так как вы не можете инициализировать с макетом. Вам нужно будет создать подкласс UIViewController и добавить представление коллекции в ваш viewDidLoad

...