SwiftUI实现循环滚动动画过程
SSwwiiffttUUII

SwiftUI实现循环滚动动画过程

在《汇率仓库》应用的开发过程中,想要实现一个汇率外币国旗和名称滚动的动画,例如下图中的底部两层国旗和名称向左滚动。

实现过程

首先把国旗的列表使用数组表示出来:

let countrys = [["AED","AUD","CAD","CHF","DKK","EUR","GBP","HKD","HUF","JPY","KRW","MOP"],["MXN","MYR","NOK","NZD","PLN","RUB","SAR","SEK","SGD","THB","TRY","USD","ZAR"]]

国旗列表数组内有两个不同的货币符号数组内容,用于区分上下两层动画。

将国旗和货币名称列表,使用HStack实现出来:

HStack {
    ForEach(countrys[item],id: \.self) { country in
        Image("\(country)")
            .resizable()
            .scaledToFit()
            .frame(width: 100)
        Text(LocalizedStringKey(country))
            .font(.footnote)
            .fixedSize()
    }
}
.frame(width: width,height: 30)

在这段代码中,Image需要设置固定的frame尺寸,Text需要设置fixedSize,否则HStack可能导致图片和文字被压缩的问题

因为需要显示出两层国旗列表,我在外层再添加一个ForEach遍历:

ForEach(0..<2) { item in
    HStack {
        ForEach(countrys[item],id: \.self) { country in
            Image("\(country)")
                .resizable()
                .scaledToFit()
                .frame(width: 100)
            Text(LocalizedStringKey(country))
                .font(.footnote)
                .fixedSize()
        }
    }
    .frame(width: width,height: 30)
    if item == 0 {
        Spacer().frame(height: 30)
    }
}

通过 ForEach(0..<2) 实现双层的效果。

接下来是自动滚动动画的实现,这里主要依赖的是offset设置视图的偏移量,模仿实际的滚动效果。

HStack {
    ForEach(countrys[item],id: \.self) { country in
        Image("\(country)")
            .resizable()
            .scaledToFit()
            .frame(width: 100)
        Text(LocalizedStringKey(country))
            .font(.footnote)
            .fixedSize()
    }
}
.offset(-100)
.frame(width: width,height: 30)

例如,我设置HStack的offset为x轴偏移-100,显示的实际效果就是向左偏移100的距离。

为了管理两层滚动视图的offset,需要先设置一个固定的offsets偏移变量。

@State private var offsets: [CGFloat] = [-730, -950]

这里需要根据HStack的宽度设置,例如我设置的偏移量分别为:-730和-950。

为什么设置-730和-950呢?是因为这两层HStack在设置为-730和-950以后,HStack基本都是偏移到末尾的状态,并且两个国旗列表之间会有一个偏差的间隙。

在实际应用中,具体的偏移量还需要自己根据视图位置进行设置,也可以使用geometryReader计算宽度并设置偏移量。

在ForEach遍历中,使用offsets[item]绑定对应的偏移变量。

ForEach(0..<2) { item in
    HStack {
        ForEach(countrys[item],id: \.self) { country in
            Image("\(country)")
                .resizable()
                .scaledToFit()
                .frame(width: 100)
            Text(LocalizedStringKey(country))
                .font(.footnote)
                .fixedSize()
        }
    }
    .offset(x:offsets[item])
    .frame(width: width,height: 30)
    if item == 0 {
        Spacer().frame(height: 30)
    }
}

设置offset绑定对应的offsets数组,这样就可以设置offsets数组中的每个值,使得绑定的视图进行偏移。

设置滚动动画

最后,新增一个 startAnimation 方法,通过withAnimationAnimation以及repeatForever属性实现平滑的动画效果。

在withAnimation中,将两个offsets数组值设置为0,这样,就会通过动画的效果,将实现两层国旗向左滚动的效果。

func startAnimation() {
    withAnimation(Animation.linear(duration: animationDuration).repeatForever(autoreverses:false)) {
        offsets[0] = 0
        offsets[1] = 0
    }
}

其中duration设置的animationDuration是一个@State变量,用于设置动画的持续时间:

let animationDuration: Double = 60

在国旗视图代码上,设置onAppear闭包调用startAnimation方法。

ForEach(0..<2) { item in
    // 相关代码
}
.onAppear {
    startAnimation()
}

运行Xcode预览,实现循环滚动的动画效果。

补充知识:无缝循环

动画基本实现,但是循环滚动的效果还不好,因为当我把持续时间设置为5秒时,可以看到动画在执行完毕后,会卡顿并返回到原来的起始位置并重新播放动画。

如果想要实现无缝循环动画,就需要使用关键偏移量重制

无缝循环的原理

1、滚动内容超过视图宽度时,重置 offset

动画持续滚动时,offset 不断向负方向移动。

当 offset 变化量等于内容总宽度的一半时,瞬间重置 offset 回到起点,形成无缝效果。

2、视图复制两份,但逻辑上只需要滚动一个宽度

在滚动完全结束时瞬间回到初始位置。

使用Timer进行手动更新

使用Timer进行手动更新后,不用使用 withAnimation(Animation.linear().repeatForever()),而是使用 Timer 每隔一段时间更新 offsets。

首先声明一个timer变量:

@State private var timer: Timer?

将原来的startAnimation方法修改为:

func startAnimation() {
    let step: CGFloat = 3  // 每帧减少 3 像素
    let interval = 0.01 // 计算时间间隔
    
    timer = Timer.scheduledTimer(withTimeInterval: interval, repeats: true) { _ in
        for i in 0..<2 {
            withAnimation {
                offsets[i] += step
            }
            // 当偏移量超过最大值时,重置为初始位置,实现无缝循环
            if i == 0, offsets[i] >= 330 {
                offsets[i] = -1550
            } else if i == 1, offsets[i] >= 460 {
                offsets[i] = -1800
            }
        }
    }
}

startAnimation方法中的step设置每一次offset偏移的位置。Interval用于Timer的触发间隔,如果Interval为0.01,表示每0.01秒触发一次Timer代码。

在Timer中设置repeats为重复触发。

Timer每次触发,offsets都会递增step对应的偏移位置,比如上面代码中每0.01秒偏移3像素。

i==0表示第一个国旗滚动列表,当第一个列表的偏移大于330时,重置offset的位置。当第二个列表对应的offset大于460时,重置offset的位置。

这里的330和460是根据列表单独调整的,为了循环时跳转到对应的offset偏移设置的。

而-1550和1800也是我为了避免出现空白区域,再次调整的offsets偏移。

@State private var offsets: [CGFloat] = [-1550, -1800]

总结

实际上无缝循环的代码还是有一些问题,例如重置offset偏移时,还需要手动调整。如果想要实现非常流畅的无缝循环,应该还是需要GeometryReader计算整个HStack国旗列表的宽度,然后根据代码进行offset的偏移设置,而不是手动配置。

手动配置的结果一是不准,二是当应用需要本地化时,每种语言的间隔也不一样,因此这里的无缝循环只是作为一个引导,后续有更好的实现思路还是再补充。

以上就是全部的循环滚动动画的过程。

相关文章

1、HStack默认布局导致视图不显示的问题:https://fangjunyu.com/2025/03/31/hstack%e9%bb%98%e8%ae%a4%e5%b8%83%e5%b1%80%e5%af%bc%e8%87%b4%e8%a7%86%e5%9b%be%e4%b8%8d%e6%98%be%e7%a4%ba%e7%9a%84%e9%97%ae%e9%a2%98/

2、SwiftUI容器视图GeometryReader:https://fangjunyu.com/2024/12/15/swiftui%e5%ae%b9%e5%99%a8%e8%a7%86%e5%9b%begeometryreader/

3、SwiftUI动画animation:https://fangjunyu.com/2024/12/16/swiftui%e5%8a%a8%e7%94%bbanimation/

4、SwiftUI动画效果withAnimation:https://fangjunyu.com/2024/12/14/swiftui%e5%8a%a8%e7%94%bb%e6%95%88%e6%9e%9cwithanimation/

5、SwiftUI动画重复修饰符repeatForever:https://fangjunyu.com/2024/12/15/swiftui%e5%8a%a8%e7%94%bb%e9%87%8d%e5%a4%8d%e4%bf%ae%e9%a5%b0%e7%ac%a6repeatforever/

6、Swift和Foundation框架创建和管理定时任务的Timer类:https://fangjunyu.com/2024/12/14/swift%e5%92%8cfoundation%e6%a1%86%e6%9e%b6%e5%88%9b%e5%bb%ba%e5%92%8c%e7%ae%a1%e7%90%86%e5%ae%9a%e6%97%b6%e4%bb%bb%e5%8a%a1%e7%9a%84timer%e7%b1%bb/

附录

视图代码

import SwiftUI

struct WelcomeView: View {
    @State private var offsets: [CGFloat] = [-1550, -1800]
    let animationDuration: Double = 5
    @Binding var ViewSteps: Int
    @Environment(\.colorScheme) var color
    let countrys = [
        ["AED","AUD","CAD","CHF","DKK","EUR","GBP","HKD","HUF","JPY","KRW","MOP"],
        ["MXN","MYR","NOK","NZD","PLN","RUB","SAR","SEK","SGD","THB","TRY","USD","ZAR"]
    ]
    
    @State private var timer: Timer?
    
    func startAnimation() {
        let step: CGFloat = 1  // 每帧减少 3 像素
        let interval = 0.01 // 计算时间间隔
        
        timer = Timer.scheduledTimer(withTimeInterval: interval, repeats: true) { _ in
            for i in 0..<2 {
                withAnimation {
                    offsets[i] += step
                }
                // 当偏移量超过最大值时,重置为初始位置,实现无缝循环
                if i == 0, offsets[i] >= 330 {
                    offsets[i] = -1550
                } else if i == 1, offsets[i] >= 460 {
                    offsets[i] = -1800
                }
            }
        }
    }
    
    var body: some View {
        GeometryReader { geo in
            let width = geo.frame(in: .global).width * 0.95
            VStack {
                Spacer().frame(height: 50)
                // 设置最大宽度
                VStack {
                    Text("Welcome")
                        .font(.largeTitle)
                        .fontWeight(.bold)
                    Spacer()
                        .frame(height: 30)
                    Text("ERdepot")
                        .fontWeight(.bold)
                        .padding(.vertical,8)
                        .padding(.horizontal,50)
                        .foregroundColor(color == .light ? .white : .black)
                        .background(color == .light ? .black : .white)
                        .cornerRadius(4)
                    Spacer().frame(height: 30)
                    Image("welcome")
                        .resizable()
                        .scaledToFit()
                        .frame(width: 240)
                    
                    Spacer().frame(height: 30)
                    // 介绍文字
                    VStack {
                        Text("We provide free exchange rate data.")
                        Spacer().frame(height: 20)
                        Text("Track historical foreign exchange rate trends.")
                    }
                    .multilineTextAlignment(.center)
                    .foregroundColor(.gray)
                    .font(.footnote)
                }
                .frame(maxWidth: width)
                Spacer().frame(height: 30)
                ForEach(0..<2) { item in
                    HStack {
                        ForEach(0..<2) { _ in
                            ForEach(countrys[item],id: \.self) { country in
                                Image("\(country)")
                                    .resizable()
                                    .scaledToFit()
                                    .frame(width: 100)
                                Text(LocalizedStringKey(country))
                                    .font(.footnote)
                                    .fixedSize()
                            }
                        }
                    }
                    .offset(x:offsets[item])
                    .frame(width: width,height: 30)
                    if item == 0 {
                        Spacer().frame(height: 30)
                    }
                }
                .onAppear {
                    startAnimation()
                }
                Spacer().frame(height: 30)
                // 设置最大宽度
                VStack {
                    Button(action: {
                        // 跳转到隐私视图
                        ViewSteps = 1
                    }, label: {
                        Text("Start")
                            .fontWeight(.bold)
                            .padding(.vertical,16)
                            .padding(.horizontal,80)
                            .foregroundColor(color == .light ? .white : .black)
                            .background(color == .light ? .black : .white)
                            .cornerRadius(6)
                    })
                }
                .frame(maxWidth: width)
                Spacer()
            }
            .frame(maxWidth: .infinity,maxHeight: .infinity)
        }
    }
}

#Preview {
    WelcomeView(ViewSteps: .constant(0))
}

如果您认为这篇文章给您带来了帮助,您可以在此通过支付宝或者微信打赏网站开放者。

发表回复

您的电子邮箱地址不会被公开。 必填项已用 * 标注