From a131bfbc8ec9bb68467ef67ff72824ecf0239e07 Mon Sep 17 00:00:00 2001 From: Pinar Olguc Date: Fri, 3 May 2024 12:50:31 +0300 Subject: [PATCH] Fix: UILabel placeholder animations doesn't work in ProfileViewController (#241) * Account icon should come from local * Fix the issue where label animations get interrupted easily --- .../ProfileActivityIndicator.swift | 3 ++- .../ProfileView/BaseProfileView.swift | 2 +- .../AccountButtonsPlaceholderDisplayer.swift | 3 ++- .../BackgroundColorPlaceholderDisplayer.swift | 3 ++- .../LabelPlaceholderDisplayer.swift | 17 ++++++++++++++++- .../Placeholder/PlaceholderDisplaying.swift | 6 ++++-- 6 files changed, 27 insertions(+), 7 deletions(-) diff --git a/Sources/GravatarUI/ProfileView/ActivityIndicator/ProfileActivityIndicator.swift b/Sources/GravatarUI/ProfileView/ActivityIndicator/ProfileActivityIndicator.swift index 8023b2db..a5f0c992 100644 --- a/Sources/GravatarUI/ProfileView/ActivityIndicator/ProfileActivityIndicator.swift +++ b/Sources/GravatarUI/ProfileView/ActivityIndicator/ProfileActivityIndicator.swift @@ -20,7 +20,7 @@ class ProfilePlaceholderActivityIndicator: ProfileActivityIndicator { guard placeholderDisplayer.isShowing else { return } shouldStopAnimating = false self.placeholderDisplayer.elements?.forEach { element in - element.prepareForAnimation() + element.animationWillBegin() } doLoadingAnimation(index: 0, animatingColors: baseView.placeholderColors.loadingAnimationColors) } @@ -31,6 +31,7 @@ class ProfilePlaceholderActivityIndicator: ProfileActivityIndicator { animator?.stopAnimation(true) } self.placeholderDisplayer.elements?.forEach { element in + element.animationDidEnd() if placeholderDisplayer.isShowing { element.refreshColor() } else { diff --git a/Sources/GravatarUI/ProfileView/BaseProfileView.swift b/Sources/GravatarUI/ProfileView/BaseProfileView.swift index e74dc1fe..e00f16db 100644 --- a/Sources/GravatarUI/ProfileView/BaseProfileView.swift +++ b/Sources/GravatarUI/ProfileView/BaseProfileView.swift @@ -297,7 +297,7 @@ open class BaseProfileView: UIView, UIContentView { } func createAccountIconView(model: AccountModel) -> UIView { - let button: UIControl = if UIImage(named: model.shortname) != nil { + let button: UIControl = if UIImage(localName: model.shortname) != nil { createAccountButton(model: model) } else if let iconURL = model.iconURL { // If we have the iconURL try downloading the icon createRemoteSVGButton(url: iconURL) diff --git a/Sources/GravatarUI/ProfileView/Placeholder/PlaceholderDisplayers/AccountButtonsPlaceholderDisplayer.swift b/Sources/GravatarUI/ProfileView/Placeholder/PlaceholderDisplayers/AccountButtonsPlaceholderDisplayer.swift index 073b3ba6..c64078db 100644 --- a/Sources/GravatarUI/ProfileView/Placeholder/PlaceholderDisplayers/AccountButtonsPlaceholderDisplayer.swift +++ b/Sources/GravatarUI/ProfileView/Placeholder/PlaceholderDisplayers/AccountButtonsPlaceholderDisplayer.swift @@ -63,5 +63,6 @@ class AccountButtonsPlaceholderDisplayer: PlaceholderDisplaying { } } - func prepareForAnimation() {} + func animationWillBegin() {} + func animationDidEnd() {} } diff --git a/Sources/GravatarUI/ProfileView/Placeholder/PlaceholderDisplayers/BackgroundColorPlaceholderDisplayer.swift b/Sources/GravatarUI/ProfileView/Placeholder/PlaceholderDisplayers/BackgroundColorPlaceholderDisplayer.swift index c78758db..58330548 100644 --- a/Sources/GravatarUI/ProfileView/Placeholder/PlaceholderDisplayers/BackgroundColorPlaceholderDisplayer.swift +++ b/Sources/GravatarUI/ProfileView/Placeholder/PlaceholderDisplayers/BackgroundColorPlaceholderDisplayer.swift @@ -41,5 +41,6 @@ class BackgroundColorPlaceholderDisplayer: PlaceholderDisplaying { baseView.layer.backgroundColor = newColor?.cgColor } - func prepareForAnimation() {} + func animationWillBegin() {} + func animationDidEnd() {} } diff --git a/Sources/GravatarUI/ProfileView/Placeholder/PlaceholderDisplayers/LabelPlaceholderDisplayer.swift b/Sources/GravatarUI/ProfileView/Placeholder/PlaceholderDisplayers/LabelPlaceholderDisplayer.swift index 13d247b5..52c2ade8 100644 --- a/Sources/GravatarUI/ProfileView/Placeholder/PlaceholderDisplayers/LabelPlaceholderDisplayer.swift +++ b/Sources/GravatarUI/ProfileView/Placeholder/PlaceholderDisplayers/LabelPlaceholderDisplayer.swift @@ -3,8 +3,23 @@ import UIKit /// A ``PlaceholderDisplaying`` implementation for a UILabel. @MainActor class LabelPlaceholderDisplayer: RectangularPlaceholderDisplayer { - override func prepareForAnimation() { + var isAnimating: Bool = false + + override func animationDidEnd() { + isAnimating = false + } + + override func animationWillBegin() { // If UILabel's backgroundColor is set, the animation won't be visible. So we need to clear it. This is only needed for UILabel so far. set(viewColor: .clear) + isAnimating = true + } + + override func set(viewColor newColor: UIColor?) { + if !isAnimating { + // If UILabel's backgroundColor is set, the animation won't be visible. + // So prevent setting it if there's an animation in progress. + super.set(viewColor: newColor) + } } } diff --git a/Sources/GravatarUI/ProfileView/Placeholder/PlaceholderDisplaying.swift b/Sources/GravatarUI/ProfileView/Placeholder/PlaceholderDisplaying.swift index 01b62eac..bda0e23c 100644 --- a/Sources/GravatarUI/ProfileView/Placeholder/PlaceholderDisplaying.swift +++ b/Sources/GravatarUI/ProfileView/Placeholder/PlaceholderDisplaying.swift @@ -16,8 +16,10 @@ public protocol PlaceholderDisplaying { func set(layerColor newColor: UIColor?) /// Sets the `backgroundColor` of the underlying view element. func set(viewColor newColor: UIColor?) - /// Prepares for color animations. - func prepareForAnimation() + /// Informs the element before the animations begin. + func animationWillBegin() + /// Informs the element after the animations end. + func animationDidEnd() /// Refreshes the color. `backgroundColor` is set to `placeholderColor` and`layer.backgroundColor` to nil. func refreshColor() }