SwiftUI Text clips on resize
Asked Answered
F

5

12

In the following sample, tapping the 'Expand' button causes the text '39' to clip as it resizes in transition to '40'. The real context of this use is a text label that reflects the value of a picker in an animated diagram. I'd like for the text to only take up the space that it requires, but without clipping as it animates between values.

Disabling animations for the label isn't an option, as the label's position in my diagram also animates.

Reproduction:

import SwiftUI

struct TextClipping: View {
    enum ExpansionState {
        case expanded
        case contracted

        mutating func toggle() {
            switch self {
            case .expanded:
                self = .contracted
            case .contracted:
                self = .expanded
            }
        }
    }

    @State var expansion: ExpansionState = .contracted

    var text: String {
        switch expansion {
        case .expanded:
            return "40"
        case .contracted:
            return "39"
        }
    }

    var body: some View {
        VStack(spacing: 16) {
            Text(text)
                .font(.system(.title, design: .rounded))
                .fontWeight(.bold)
                .foregroundColor(.black)

            Button(self.expansion == .contracted ? "Expand" : "Contract") {
                withAnimation {
                    self.expansion.toggle()
                }
            }
        }
    }
}

struct TextClipping_Previews: PreviewProvider {
    static var previews: some View {
        TextClipping()
    }
}

Any ideas for how I might allow the text of the label to change fluidly, without clipping?

Fess answered 20/10, 2019 at 19:11 Comment(1)
Not reproducible with Xcode 13 / iOS 15. Could you attach gif with demo of what's wrong or provide code with more visible reproduce?Imagination
A
4

So, I don't think it was entirely clear by your question, but I assume the "clipping" you were referring to was the 39 and 40 moving slightly left and right as you animate between them. In a parent view, this might clip, but I don't see it here. First things first, you don't need to make your own mutating struct to modify a variable in a view, this is the whole point of @State variables, and this whole issue can be solved with a simple bool. All of this code is essentially the same, except for one key line: the .id line. This tells SwiftUI that the views are not the same. Before adding it, it thought they were the same view, but the text was changing, causing the offsets to look weird. If this still doesn't answer your question let me know but I cant imagine what else you'd be referring to, as you did not include a visual example and I'm unable to reproduce the "clipping" effect you're referring to.

struct TextClipping: View {
    @State private var expanded: Bool = false
    var body: some View {
        let text: String = expanded ? "40" : "39"
        VStack(spacing: 16) {
             Text(text)
                .font(.system(.title, design: .rounded))
                .fontWeight(.bold)
                .foregroundColor(.black)
                // Opacity is the default animation, but can be replaced by another if you feel like it
                //.transition(.opacity)
                .id("label\(text)")

            Button(expanded ? "Contract" : "Expand") {
                withAnimation {
                    expanded.toggle()
                }
            }
        }
    }
}

struct TextClipping_Previews: PreviewProvider {
    static var previews: some View {
        TextClipping()
    }
}
Addam answered 2/10, 2021 at 17:12 Comment(1)
This should be the accepted answer, simply adding an id to Text fixes the problem. Thank you!Luehrmann
F
1

Resolved by applying

.minimumScaleFactor(0.1)

to the Text.

Fess answered 27/3, 2020 at 0:38 Comment(1)
This doesn't work (at least on iOS 15 with a custom font)Scleroma
N
1

You may try adding this modifier to text

.fixedSize()
Narco answered 28/9, 2021 at 17:18 Comment(0)
H
1

To disable the size animation without disabling the offset animation, I use a UIHostingController embedded in another UIViewController: this allows me to update the size without animation, allowing the SwiftUI to animate the offset.

p.s. Technically, you could convert this code to a modifier, but I think embedding the view in a container should be clear, because in some cases it can significantly affect the layout.

struct DisableSizeAnimationContainer<Content: View>: View {
    let content: Content
    
    init(@ViewBuilder _ content: () -> Content) {
        self.content = content()
    }
    
    var body: some View {
        ContainerRepreseentable {
            content
        }.fixedSize()
    }
}

private struct ContainerRepreseentable<Content: View>: UIViewControllerRepresentable {
    let content: Content
    
    init(@ViewBuilder _ content: () -> Content) {
        self.content = content()
    }
    
    func makeUIViewController(context: Context) -> UIHostingControllerContainer<Content> {
        UIHostingControllerContainer(rootView: content)
    }
    
    func updateUIViewController(_ uiViewController: UIHostingControllerContainer<Content>, context: Context) {
        uiViewController.rootView = content
    }
}

private class UIHostingControllerContainer<RootView: View>: UIViewController {
    private let hostingController: UIHostingController<RootView>
    
    init(rootView: RootView) {
        hostingController = UIHostingController(rootView: rootView)
        super.init(nibName: nil, bundle: nil)
    }
    
    var rootView: RootView {
        get { hostingController.rootView }
        set {
            hostingController.rootView = newValue
            preferredContentSize = hostingController.view.intrinsicContentSize
            
            view.frame = .init(origin: .zero, size: preferredContentSize)
            hostingController.view.frame = view.frame
        }
    }
    
    required dynamic init?(coder aDecoder: NSCoder) {
        fatalError("init(coder:) has not been implemented")
    }
    
    override func viewDidLoad() {
        super.viewDidLoad()

        addChild(hostingController)
        view.addSubview(hostingController.view)
    }
}

Usage:

Button(action: {
    withAnimation {
        expansion.toggle()
    }
}) {
    DisableSizeAnimationContainer {
        Text(expansion ? "Small text" : "Significantly larger text")
    }
}.offset(x: expansion ? 100 : 0)

Result:

enter image description here

Hygiene answered 29/9, 2021 at 7:57 Comment(0)
E
0

To prevent clipping you can use an overlay like this:

struct TextClipping: View {
    @State private var expanded: Bool = false
    var body: some View {
        let text: String = expanded ? "40" : "39"
        VStack(spacing: 16) {
            Text("40")
                .fontWeight(.bold)
                .font(.system(.title, design: .rounded))
                .hidden()
                .overlay(
                    Text(text).fontWeight(.bold).font(.system(.title, design: .rounded))
                ).id("label\(text)")
            
            Button(expanded ? "Contract" : "Expand") {
                withAnimation {
                    expanded.toggle()
                }
            }
        }
    }
}
Emalia answered 5/10, 2021 at 12:11 Comment(0)

© 2022 - 2025 — McMap. All rights reserved.