Conditional onTapGesture in SwiftUI
Asked Answered
C

4

5

I have a navigation link and I need a different behavior when its label (MyView) is tapped depending on the edit mode (or any other condition):

  1. If we are not in edit mode, I want to trigger the navigation link and show the DetailView with the selected model.
  2. If we are in edit mode, I don't want to trigger the navigation link and show an EditingView in a modal sheet instead.

Here's a way to implement this that I came up with:

NavigationLink(tag: model, selection: $displayedItem) {
    DetailView(model: model)
} label: {
    if editMode == .active {
        MyView()
            .onTapGesture {
                editingModel = model
            }
    } else {
        MyView()
    }
}
.sheet(item: $editingModel) { model in
    EditingView(model: model)
}

The problem with this approach is that the views in the if- and the else-branch have not the same type (due to the onTapGesture modifier) and SwiftUI doesn't recognize them as the same view. Thus, animations cannot be interpolated and don't work properly. Also, MyView always loses its state each time editMode is toggled.

(Here's a great explanation from Chris Eidhof on why that happens: https://www.objc.io/blog/2021/08/24/conditional-view-modifiers/)

So I went ahead and moved the if-statement inside the onTapGesture modifier as follows so that I don't have two different MyViews:

NavigationLink(tag: model, selection: $displayedItem) {
    DetailView(model: model)
} label: {
    MyView()
        .onTapGesture {
            if editMode == .active { // moved
                editingModel = model
            }                        // moved
        }
    }
}
.sheet(item: $editingModel) { model in
    EditingView(model: model)
}

Problem with this is that now requirement #1 doesn't work anymore: The onTapGesture completely swallow the tap gesture and thus the navigation link is never trigged to show the DetailView. Makes sense.

Now my question is:

How can I get the desired behavior without any of these downsides?

Cabrera answered 2/4, 2022 at 18:8 Comment(0)
T
10

In short, you want to change:

if editMode == .active {
    MyView()
        .onTapGesture {
            editingModel = model
        }
} else {
    MyView()
}

Into:

MyView()
    .allowsHitTesting(editMode == .active)
    .onTapGesture {
        editingModel = model
    }

This fixes the issue because now the onTapGesture is only triggered when it can actually listen to touches. It can only trigger when editMode == .active, because otherwise the hit testing is disabled.


Full example:

struct ContentView: View {
    @State private var displayedItem: String?
    @State private var editingModel: EditingModel?
    @State private var editMode: EditMode = .inactive

    var body: some View {
        NavigationView {
            List {
                Button("Edit mode: \(editMode == .active ? "active" : "inactive")") {
                    if editMode == .active {
                        editMode = .inactive
                    } else {
                        editMode = .active
                    }
                }

                NavigationLink(tag: "model", selection: $displayedItem) {
                    Text("DetailView")
                } label: {
                    if editMode == .active {
                        MyView()
                            .onTapGesture {
                                editingModel = EditingModel(tag: "model")
                            }
                    } else {
                        MyView()
                    }
                }
                .sheet(item: $editingModel) { model in
                    Text("EditingView: \(model.tag)")
                }
            }
        }
    }
}
struct MyView: View {
    var body: some View {
        Text("MyView")
            .frame(maxWidth: .infinity, alignment: .leading)
            .contentShape(Rectangle())
    }
}
struct EditingModel: Identifiable {
    var id: String { tag }
    let tag: String
}

Changing the inner label bit to:

MyView()
    .allowsHitTesting(editMode == .active)
    .onTapGesture {
        editingModel = EditingModel(tag: "model")
    }
Truditrudie answered 2/4, 2022 at 18:37 Comment(1)
Great answer! Straightforward and simple. Thanks! One could go ahead and wrap the hit-test modifier and the tap gesture modifier into a single modifier to make it more expressive of what's happening.Cabrera
C
3

Building on George's suggestion, I ended up with the following custom modifier:

extension View {
    func onTapGestureIf(_ condition: Bool, closure: @escaping () -> ()) -> some View {
        self.allowsHitTesting(condition)
            .onTapGesture {
                closure()
            }
    }
}

Works like a charm and this way, I don't have to specify the same condition editMode == .active twice (single source of truth). Highly recommended. 😉

MyView()
    .onTapGestureIf(editMode == .active) {
        editingCounter = counter
    }
Cabrera answered 2/4, 2022 at 19:4 Comment(1)
Beware though, in case you need to also support other gestures, for example a longPress on the same View, this method will disable the other gestures. A better option is to extend the View with a @ViewBuilder func that returns some View, either self.onTapGesture { someAction } or just self.Chaperon
C
1

Use Group to wrap onTapGesture. That's it.

extension View {
  public func onTapGesture(isOn: Bool, count: Int = 1, perform action: @escaping () -> Void) -> some View {
    Group {
      if isOn {
        self.onTapGesture(count: count, perform: action)
      } else {
        self
      }
    }
  }
}
Cornerwise answered 11/7, 2022 at 8:18 Comment(0)
A
0

you can use custom modifier.

struct ContentView: View {
@State private var displayedItem: String?
@State private var editingModel: EditingModel?
@State private var editMode: EditMode = .inactive
var body: some View {
    NavigationView {
        List {
            Button("Edit mode: \(editMode == .active ? "active" : "inactive")") {
                if editMode == .active {
                    editMode = .inactive
                } else {
                    editMode = .active
                }
            }
            
            NavigationLink(tag: "model", selection: $displayedItem) {
                Text("DetailView")
            } label: {
                MyView()
                    .modifier(conditionalTapGesture(editMode: $editMode, editingModel: $editingModel))
            }
            .sheet(item: $editingModel) { model in
                Text("EditingView: \(model.tag)")
            }
        }
    }
}

}

struct conditionalTapGesture : ViewModifier {
    @Binding var editMode: EditMode
    @Binding var editingModel: EditingModel?
    @ViewBuilder func body(content: Content) -> some View {
        if editMode == .active {
            content.onTapGesture {
                editingModel = EditingModel(tag: "model")
            }
        }
    }
}
Aleta answered 14/12, 2022 at 7:28 Comment(0)

© 2022 - 2025 — McMap. All rights reserved.