With a List
one can simply use the .onDelete
modifier to remove the rows from the List. But how can we do the same in a ForEach
shown within a LazyVStack
. I am using SwipeCell to apply a drag gesture and showing a delete button but how do I actually remove from CoreData, specifically how do I access the IndexSet
in this case
LazyVStack {
ForEach(items) { item in
Text("\(item.name)")
.swipeCell(cellPosition: .right, leftSlot: nil, rightSlot: slot1)
.alert(isPresented: $showAlert){
Alert(title: Text("Delete Task?"), message: nil, primaryButton:.destructive(Text("Delete"), action: {
// HOW DO I DELETE HERE ?
dismissDestructiveDelayButton()
}),secondaryButton: .cancel({dismissDestructiveDelayButton()}))
}
}
}
private func deleteItems(offsets: IndexSet) {
withAnimation {
offsets.map { items[$0] }.forEach(viewContext.delete)
do {
try viewContext.save()
} catch {
let nsError = error as NSError
fatalError("Unresolved error \(nsError), \(nsError.userInfo)")
}
}
}
I've made a ViewModifier, that allows you to delete whatever view you want, with something similar to the Apple Animation.
When using, just call .onDelete { //perform deletion }
on any View.
struct Delete: ViewModifier {
let action: () -> Void
@State var offset: CGSize = .zero
@State var initialOffset: CGSize = .zero
@State var contentWidth: CGFloat = 0.0
@State var willDeleteIfReleased = false
func body(content: Content) -> some View {
content
.background(
GeometryReader { geometry in
ZStack {
Rectangle()
.foregroundColor(.red)
Image(systemName: "trash")
.foregroundColor(.white)
.font(.title2.bold())
.layoutPriority(-1)
}.frame(width: -offset.width)
.offset(x: geometry.size.width)
.onAppear {
contentWidth = geometry.size.width
}
.gesture(
TapGesture()
.onEnded {
delete()
}
)
}
)
.offset(x: offset.width, y: 0)
.gesture (
DragGesture()
.onChanged { gesture in
if gesture.translation.width + initialOffset.width <= 0 {
self.offset.width = gesture.translation.width + initialOffset.width
}
if self.offset.width < -deletionDistance && !willDeleteIfReleased {
hapticFeedback()
willDeleteIfReleased.toggle()
} else if offset.width > -deletionDistance && willDeleteIfReleased {
hapticFeedback()
willDeleteIfReleased.toggle()
}
}
.onEnded { _ in
if offset.width < -deletionDistance {
delete()
} else if offset.width < -halfDeletionDistance {
offset.width = -tappableDeletionWidth
initialOffset.width = -tappableDeletionWidth
} else {
offset = .zero
initialOffset = .zero
}
}
)
.animation(.interactiveSpring())
}
private func delete() {
offset.width = -contentWidth
action()
}
private func hapticFeedback() {
let generator = UIImpactFeedbackGenerator(style: .medium)
generator.impactOccurred()
}
//MARK: Constants
let deletionDistance = CGFloat(200)
let halfDeletionDistance = CGFloat(50)
let tappableDeletionWidth = CGFloat(100)
}
extension View {
func onDelete(perform action: @escaping () -> Void) -> some View {
self.modifier(Delete(action: action))
}
}
If you love us? You can donate to us via Paypal or buy me a coffee so we can maintain and grow! Thank you!
Donate Us With