0:00
/
0:00

💡 SwiftUI FYI #8 - Coffee Tracker Animation

Cool fill animation and coffee tracker using a wave effect in #SwiftUI

Inspired by Hacking with Swift, we create a wave effect to mimic the filling of coffee on our screen. We display a coffee mug indicator with a mesmerizing continuous liquid animation and a counter.

Upon the addition of a new cup of coffee, we initiate the coffee fill animation on our screen.

🚨 Full code below and also on Github

🤝 Find more about me on saidmarouf.com. I’m also on XThreads, and LinkedIn

Thanks for reading Your Weekly SwiftUI FYI 💡! Subscribe for free to receive new posts and support my work.


import SwiftUI

struct CoffeeTrackerView: View {
    
    @State private var coffeeCount = 0
    @State private var isAddingCoffee = false
    
    var body: some View {
        
        ZStack {
            if isAddingCoffee {
                AnimatedFillView(completion: {  
                   withAnimation(.easeOut(duration: 0.2)) {
                    isAddingCoffee.toggle()
                    coffeeCount += 1
                }
                })
                .frame(maxWidth: .infinity, maxHeight: .infinity)
                .ignoresSafeArea()
            }
            
            VStack {
                Spacer()
                
                ZStack {
                    MugIndicatorView(fillPercentage: 60)
                        .frame(width: 140, height: 140)
                    
                    Text("\(coffeeCount)")
                        .font(.system(size: 50))
                        .fontWeight(.heavy)
                        .contentTransition(.numericText())
                        .offset(x: -10, y: 16)
                }
                
                Button {
                    withAnimation { isAddingCoffee.toggle() }
                } label: {
                    Label("Caffeine++", systemImage: "heart.fill")
                        .symbolRenderingMode(.multicolor)
                        .font(.title).bold()
                        .padding(.horizontal, 16)
                }
                .buttonStyle(.borderedProminent)
                .tint(.white)
                .foregroundStyle(.black)
                .padding()
                
                Spacer()
            }
        }
    }
}

#Preview {
    CoffeeTrackerView()
}

struct AnimatedFillView: View {
    
    @State private var percent: Double = 0.0
    @State private var waveOffset = Angle(degrees: 0)
    @State private var waveOffset2 = Angle(degrees: 180)
    @State private var waveOffset3 = Angle(degrees: 360)
    
    let completion: (() -> Void)?
    
    private let timer = Timer.publish(every: 0.015, 
                                      on: .main, 
                                      in: .common)
                        .autoconnect()
    private let coffeeColor = Color(red: 111/255.0,
                                    green: 78/255.0,
                                    blue: 55/255.0)
    
    var body: some View {
        // 3 layers of waves
        ZStack {
            Wave(offset: Angle(degrees: waveOffset.degrees), 
                 percent: percent)
                .fill(coffeeColor)
            
            Wave(offset: Angle(degrees: waveOffset2.degrees), 
                 percent: percent)
                .fill(coffeeColor.opacity(0.6))
            
            Wave(offset: Angle(degrees: waveOffset3.degrees), 
                 percent: percent)
                .fill(coffeeColor.opacity(0.3))
        }
        .onReceive(timer) { _ in
            // 120 instead of 100 to account for the wave dips, 
               and let the animation fill the whole screen
            if percent == 120 {
                timer.upstream.connect().cancel()
                completion?()
            }
            percent += 1.0
        }
        .onAppear {
            //Start wave animation
            withAnimation(.linear(duration: 0.5)
                .repeatForever(autoreverses: false)) {
                    self.waveOffset = Angle(degrees: 360)
                    self.waveOffset2 = Angle(degrees: 180)
                    self.waveOffset3 = Angle(degrees: 0)
                }
        }
    }
}

struct MugIndicatorView: View {
    
    @State var fillPercentage: Double = 0.0
    @State private var waveOffset = Angle(degrees: 0)
    
    let coffeeColor = Color(red: 111/255.0, green: 78/255.0, 
                            blue: 55/255.0)
    
    var body: some View {
        
        Wave(offset: Angle(degrees: waveOffset.degrees), 
             percent: fillPercentage)
            .fill(coffeeColor)
            .mask {
                Image(systemName: "mug.fill")
                    .resizable()
                    .aspectRatio(contentMode: .fit)
            }
            .overlay {
                Image(systemName: "mug")
                    .resizable()
                    .aspectRatio(contentMode: .fit)
            }
            .onAppear {
                withAnimation(Animation.linear(duration: 1)
                    .repeatForever(autoreverses: false)) {
                        self.waveOffset = Angle(degrees: 360)
                    }
            }
    }
}

// Inspired by Hacking with Swift
struct Wave: Shape {
    
    var offset: Angle
    var percent: Double
    
    var animatableData: AnimatablePair<Double, Double> {
        get { AnimatablePair(offset.degrees, percent) }
        set {
            offset = Angle(degrees: newValue.first)
            percent = newValue.second
        }
    }
    
    func path(in rect: CGRect) -> Path {
        var p = Path()
        
        let lowestWave = 0.1
        let highestWave = 1.00
        
        let newPercent = lowestWave + 
                         (highestWave - lowestWave) * (percent / 100)
        let waveHeight = 0.05 * rect.height
        let yOffset = CGFloat(1 - newPercent) * 
                      (rect.height - 4 * waveHeight) + 2 * waveHeight
        let startAngle = offset
        let endAngle = offset + Angle(degrees: 360)
        
        p.move(to: CGPoint(x: 0, y: yOffset + 
                             waveHeight * CGFloat(sin(offset.radians))))
        
        for angle in stride(from: startAngle.degrees, 
                            through: endAngle.degrees, by: 5) {
            let x = CGFloat((angle - startAngle.degrees) / 360) 
                    * rect.width
            let y = yOffset + waveHeight * 
                            CGFloat(sin(Angle(degrees: angle).radians))
            p.addLine(to: CGPoint(x: x, y: y))
        }
        
        p.addLine(to: CGPoint(x: rect.width, y: rect.height))
        p.addLine(to: CGPoint(x: 0, y: rect.height))
        p.closeSubpath()
        
        return p
    }
}