【问题标题】:Using a Data Model to 'relate' to other models (create relationships)使用数据模型与其他模型“关联”(创建关系)
【发布时间】:2018-06-24 18:59:45
【问题描述】:

我有一个存储特定极限运动信息的数据模型。此数据显示在 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 中的按钮以根据关系更新信息?

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

【问题讨论】:

    标签: ios swift model-view-controller uicollectionview related-content


    【解决方案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&lt;String&gt;] 类型,它将 ExtremeSportsTermsModelterm 值作为“主键”,这并不是很好。在大多数情况下,有一个let id: Int 值或接近它的值,也许你也应该考虑一下。

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

    附:复制/粘贴file


    didSelectItemAt 问题。

    我认为你的策略应该是 1) 抓住relatedItem,2) 将 relatedItem 设置为 item 值,ViewController 可以通过该值重新加载自己,3) 并调用适当的函数来重新加载 UI。

    func collectionView(_ collectionView: UICollectionView, didSelectItemAt indexPath: IndexPath) {
        let relatedItem = repository.getRelated(to: receivedSportItem!)[indexPath.row]
        // set `relatedItem` to you `item`
        // call `reloadUI` func
    }
    

    【讨论】:

    • 我要等到以后才能尝试这个,只是想先给你一个先发制人的感谢你的帮助!我稍后会接触基地:)
    • 我的 UI 现在是如何设置的!我稍微修改了您上面的代码,但是关系在那里,并且相关数据正在正确更新。我将如何处理点击“相关”CV 单元格来更新 UI?这就是我在原始帖子中被困在“didSelectItemAt”中的地方。此外,我在主要的 VC viewDidLoad 方法中定义了我的所有术语......我猜我应该将数据保存在它自己的类中?
    • 嗨,@Joe。我有点坚持你的问题。也许您可以分享一些屏幕截图或 github 代码(左右)。这将非常有助于理解此时的问题。
    • 当然!我在原始问题中添加了屏幕截图。所有数据(术语标题、术语定义、相关术语)都正确显示在 Detail VC 中。我只希望相关术语单元格可点击,因此它们会根据点击的术语重新加载数据。 (相关术语是集合视图。)我在 didSelectItemAt 中尝试了上述问题,但是,这似乎不是一个好方法。
    猜你喜欢
    • 1970-01-01
    • 2021-11-14
    • 1970-01-01
    • 1970-01-01
    • 2020-03-30
    • 2021-04-06
    • 2020-06-13
    • 1970-01-01
    • 2011-12-24
    相关资源
    最近更新 更多