如何使用SwiftUI框架实现删除和移动功能

时间:2019-06-07 06:55:28

标签: swiftui

无法删除列表视图中的项目并将其移至该行。

Apple在他们的培训视频中提供了这些方法,但不起作用

要删除:

array.remove(atOffsets: IndexSet)

移动:

array.move(fromOffsets: IndexSet, toOffset: Int)

这两种方法在Apple文档中均不可用。

Simulator Screen

3 个答案:

答案 0 :(得分:2)

删除元素

TLDR:

.onDelete{index in
                self.array.remove(at: index.first!)

        }

完整代码(可以复制粘贴):

import SwiftUI

struct MyTableView : View {

 @State var array=[1,2,3,4,5]

    var body: some View {

        List{
            ForEach(array){element in
                Text("\(element)")
                }
                .onDelete{index in
                    self.array.remove(at: index.first!)

            }
        }

    }
}

#if DEBUG
struct MyTableView_Previews : PreviewProvider {
    static var previews: some View {
        MyTableView()
    }
}
#endif

答案 1 :(得分:2)

这些方法是Swift 5.1标准库的一部分,Apple将会在以后的Xcode Beta中提供该方法。在此期间,您可以使用以下扩展名:

extension Array {
    mutating func remove(atOffsets offsets: IndexSet) {
        let suffixStart = halfStablePartition { index, _ in
            return offsets.contains(index)
        }
        removeSubrange(suffixStart...)
    }

    mutating func move(fromOffsets source: IndexSet, toOffset destination: Int) {
        let suffixStart = halfStablePartition { index, _ in
            return source.contains(index)
        }
        let suffix = self[suffixStart...]
        removeSubrange(suffixStart...)
        insert(contentsOf: suffix, at: destination)
    }

    mutating func halfStablePartition(isSuffixElement predicate: (Index, Element) -> Bool) -> Index {
        guard var i = firstIndex(where: predicate) else {
            return endIndex
        }

        var j = index(after: i)
        while j != endIndex {
            if !predicate(j, self[j]) {
                swapAt(i, j)
                formIndex(after: &i)
            }
            formIndex(after: &j)
        }
        return i
    }

    func firstIndex(where predicate: (Index, Element) -> Bool) -> Index? {
        for (index, element) in self.enumerated() {
            if predicate(index, element) {
                return index
            }
        }
        return nil
    }
}

答案 2 :(得分:1)

他们使用了Array的自定义扩展或尚未向公众提供的某些版本的Swift。我实现的解决方法是:

删除


func delete(at offsets: IndexSet) {
    if let first = offsets.first {
        store.rooms.remove(at: first)
    }
}

移动


func move(from source: IndexSet, to destination: Int) {
    if let first = source.first {
        store.rooms.swapAt(first, destination)
    }
}

移动功能有效,但是动画效果不如WWDC 2019上的视频动画。