2

我有一个存储特定极限运动信息的数据模型。此数据显示在 tableView 中,可以点击其中的单元格(极限运动技巧)以查看 Details VC 中的所有详细信息。这些细节的一部分是可选的“相关”部分,可以定义为模型的一部分。

示例 - 表格单元格可能显示“Kickflip”(滑板技巧)。我可以将“Ollie”和“360-flip”(另外两个滑板技巧)与 Kickflip 模型的相关部分相关联。

我希望这两个相关的术语(Ollie 和 360-flip)可点击,这样用户就可以阅读 Ollie 或 360-flip 的定义,而不是阅读“Kickflip”详细信息。

我的问题是如何获取“相关”项目以将数据刷新到特定术语。

这是我的模型:

struct ExtremeSportsData {
    static func getAllSportsTerms() -> [ExtremeSportsTermsWithSectionHeaders] {
        return [
            ExtremeSportsTermsWithSectionHeaders(sectionName: "A", sectionObjects: [
                ExtremeSportsTermsModel(term: "Acid Drop", definition: "Lorem ipsum dolor sit amet, consectetur adipiscing elit.", relatedTerms: nil),
                ExtremeSportsTermsModel(term: "Alpha Flip", definition: "Lorem ipsum dolor sit amet, consectetur adipiscing elit.", relatedTerms: nil),
                ExtremeSportsTermsModel(term: "Axle Stall", definition: "Lorem ipsum dolor sit amet, consectetur adipiscing elit.", relatedTerms: ["Ollie"])
                ]),
            ExtremeSportsTermsWithSectionHeaders(sectionName: "O", sectionObjects: [
                ExtremeSportsTermsModel(term: "Ollie", definition: "Lorem ipsum dolor sit amet, consectetur adipiscing elit.", relatedTerms: ["Shuvit"]),
                ExtremeSportsTermsModel(term: "Overturn", definition: "Lorem ipsum dolor sit amet, consectetur adipiscing elit.", relatedTerms: ["Acid Drop", "Alpha Flip"])
                ]),
            ExtremeSportsTermsWithSectionHeaders(sectionName: "S", sectionObjects: [
                ExtremeSportsTermsModel(term: "Shuvit", definition: "Lorem ipsum dolor sit amet, consectetur adipiscing elit.", relatedTerms: ["Ollie", "Axle Stall", "Overturn"])
                ])
        ]
    }
}

一旦点击了一个单元格,DetailVC 上的信息就会显示在一个集合视图中。我希望相关术语(如果有的话!)将数据更新为该特定术语。正如您在我上面的数据中看到的那样——如果用户点击 Ollie,他们会将 Shuvit 作为相关术语。可以单击以阅读 Shuvit 定义和信息。

我对 UI 的思考过程是使用 Contains(String),但是如果数据中存在拼写错误,此解决方案将不起作用..(并且我下面的代码并不完美)

func collectionView(_ collectionView: UICollectionView, didSelectItemAt indexPath: IndexPath) {
    let specificRelatedTerm = receivedSportTerm!.relatedTerms![indexPath.item]
    if receivedSportTerm!.term.contains(specificRelatedTerm) {
        print("Yes - Related")
    }
    else {
        print("No - Not related")
    }
}

如何设置数据之间的“关系”,以便我可以点击 UI 中的按钮以根据关系更新信息?

我很乐意澄清上述任何内容......有些事情告诉我,我并没有尽我所能,但感谢您提供任何帮助。

用户界面

4

1 回答 1

1

我认为某种类喜欢RepositoryHolder会适合这个选项。基本思想 - 你拥有Repository包含所有ExtremeSportsTermsModels 和相关项目的每个ExtremeSportsTermsModelfuncs 可以使用数据进行操作。

final class RelatedFlipsRepository {

    enum Action {
        case add
        case remove
    }

    private var relations: [String: Set<String>] = [:]
    var flips: [ExtremeSportsTermsModel] = []

    init(flips: [ExtremeSportsTermsModel]) {
        self.flips = flips
    }

    // MARK: - Public

    func addFlip(_ flip: ExtremeSportsTermsModel, relatedTo related: [ExtremeSportsTermsModel] = []) {
        if !flips.contains(where: { $0.term == flip.term }) {
            flips.append(flip)
        }

        configureRelated(related, flip: flip, action: .add)
    }

    func getRelated(to flip: ExtremeSportsTermsModel) -> [ExtremeSportsTermsModel] {
        guard let values = relations[flip.term], !values.isEmpty else { return [] }
        return flips.filter({ values.contains($0.term) })
    }

    func configureRelated(_ related: [ExtremeSportsTermsModel], flip: ExtremeSportsTermsModel, action: Action) {
        switch action {
        case .add:
            addRelated(related, to: flip)
            related.forEach({ addRelated([flip], to: $0) })

        case .remove:
            removeRelated(related, from: flip)
            related.forEach({ removeRelated([flip], from: $0) })
        }

    }

    // MARK: - Private

    private func addRelated(_ related: [ExtremeSportsTermsModel], to flip: ExtremeSportsTermsModel) {
        if !related.isEmpty {
            let relatedValues = related.map({ $0.term })
            if let values = relations[flip.term] {
                relations[flip.term] = values.union(relatedValues)
            } else {
                relations[flip.term] = Set(relatedValues)
            }
        }
    }

    private func removeRelated(_ related: [ExtremeSportsTermsModel], from flip: ExtremeSportsTermsModel) {
        guard let values = relations[flip.term], !related.isEmpty else { return }
        relations[flip.term] = values.subtracting(related.map({ $0.term }))
    }

}

用法:

let ollie = ExtremeSportsTermsModel(term: "Ollie")
let shuvit = ExtremeSportsTermsModel(term: "Shuvit")
let acidDrop = ExtremeSportsTermsModel(term: "Acid Drop")

let repository = RelatedFlipsRepository(flips: [ollie, shuvit, acidDrop])

repository.configureRelated([acidDrop, shuvit], flip: ollie, action: .add)
print(repository.getRelated(to: ollie)) // [Shuvit, Acid Drop]
print(repository.getRelated(to: acidDrop)) // [Ollie]

repository.configureRelated([acidDrop], flip: ollie, action: .remove)
print(repository.getRelated(to: ollie)) // [Shuvit]
print(repository.getRelated(to: acidDrop)) // []  

如您所见Repository,有两个属性relationsflips(没有人说您必须将所有数据保存在一个属性中,对吗?)。relations具有[String: Set<String>]ExtremeSportsTermsModel'sterm值作为“主键”的类型,它并不是很好。在大多数情况下,有一个let id: Int价值或接近它的东西,也许你也应该考虑一下。

这个工作流可以帮助你处理两个方向的相关数据,但是如果你想要更具体Graph的解决方案,你可以使用数据结构,但它并不容易实现和支持。

PS复制/粘贴file


didSelectItemAt问题。

我认为您的策略应该是 1) 抓取 tapped relatedItem,2) 将relatedItemitem设置为ViewController可以重新加载自己的值,3) 并调用适当的 func 来重新加载 UI。

func collectionView(_ collectionView: UICollectionView, didSelectItemAt indexPath: IndexPath) {
    let relatedItem = repository.getRelated(to: receivedSportItem!)[indexPath.row]
    // set `relatedItem` to you `item`
    // call `reloadUI` func
}
于 2018-01-16T07:38:06.217 回答