Fond d’écran iPhone SE 2022 dans SwiftUI
La semaine dernière, j’ai écrit un tutoriel sur la création du nouveau fond d’écran Apple iPhone SE à l’aide de Photoshop. C’était un petit tutoriel vraiment amusant et j’ai aussi vu un autre designer faire le même effet dans Figma. C’est incroyable de voir tant de personnes talentueuses ainsi que des outils incroyables à notre disposition. Avec la même image en tête, j’ai décidé d’essayer de la recréer en code en utilisant SwiftUI et cela s’avère être un exercice amusant avec un très beau résultat.
Remarque : il y a des bandes sur le dégradé. Mes connaissances sont limitées sur SwiftUI mais je suis sûr qu’il existe une solution. Il y a aussi un bug bizarre dans mon animation, mais ce n’est pas vraiment nécessaire. Voici donc ce que j’ai fait.
Aperçu vidéo
Étape 1
Commencez avec la vue rectangulaire simple. J’ai appelé ma structure ColorBar
struct ColorBar: View { var body: some View { Rectangle() } }
Étape 2
Ensuite, j’ai créé 5 variables différentes pour les dégradés et appliqué au remplissage de mon rectangle. Les première et dernière couleurs que j’ai utilisées 0 pour l’opacité. Assurez-vous également que startPoint est .top et endPoint est .bottom.
struct ColorBar: View { var g1:Color = Color(red: 95 / 255, green: 71 / 255, blue: 202 / 255) var g2:Color = Color(red: 255 / 255, green: 175 / 255, blue: 196 / 255) var g3:Color = Color(red: 230 / 255, green: 210 / 255, blue: 173 / 255) var g4:Color = Color(red: 244 / 255, green: 160 / 255, blue: 18 / 255) var g5:Color = Color(red: 228 / 255, green:0 / 255, blue: 0 / 255) var body: some View { Rectangle().fill( LinearGradient(gradient: Gradient(colors: [g1.opacity(0),g1,g2,g3,g4,g5,g5.opacity(0)]), startPoint: .top, endPoint: .bottom)) } }
Étape 3
Tout mettre ensemble. J’ai d’abord tout fait manuellement, c’est-à-dire que j’ai ajouté 9 instances de ma barre de couleurs et modifié la valeur de leur hauteur à l’aide du modificateur de cadre. J’ai créé une simple variable appelée multiplicateur à utiliser comme base pour la différence de hauteur.
struct ManualBars: View{ var multiplier:CGFloat = 60 var body: some View{ GeometryReader { geometry in HStack(alignment: .center, spacing:0){ ColorBar().frame(height: geometry.size.height - (multiplier * 1)).zIndex(5) ColorBar().frame(height: geometry.size.height - (multiplier * 2)).zIndex(4) ColorBar().frame(height: geometry.size.height - (multiplier * 3)).zIndex(3) ColorBar().frame(height: geometry.size.height - (multiplier * 4)).zIndex(2) ColorBar().frame(height: geometry.size.height - (multiplier * 5)) ColorBar().frame(height: geometry.size.height - (multiplier * 4)).zIndex(2) ColorBar().frame(height: geometry.size.height - (multiplier * 3)).zIndex(3) ColorBar().frame(height: geometry.size.height - (multiplier * 2)).zIndex(4) ColorBar().frame(height: geometry.size.height - (multiplier * 1)).zIndex(5) } } } }
Conclusion
Après cela, je passe un peu plus de temps à essayer d’automatiser le processus avec ForEach. J’ai toujours du mal avec le concept mais voici le code final. Ah ! J’ai également animé les barres lorsque l’application s’ouvre ainsi que lorsque vous appuyez dessus.
// // ContentView.swift // SEBG // // Created by Fabio Sasso on 3/12/22. // import SwiftUI struct ContentView: View { var multiplier:CGFloat = 60 var steps:Int = 9 var half:Int = 5 var gStart:Color = Color(red: 29 / 255, green: 6 / 255, blue: 74 / 255) var gEnd:Color = Color(red: 54 / 255, green: 4 / 255, blue: 2 / 255) @State var aniBar: Bool = false var body: some View { GeometryReader { geometry in ZStack{ HStack(alignment: .center, spacing:0){ ForEach((1...steps), id: .self) { i in if(i <= half){ ColorBar().frame(height: self.aniBar ? geometry.size.height - (multiplier * CGFloat(i)) : geometry.size.height).zIndex(Double(half - (i-half))).shadow(color: Color.black.opacity(self.aniBar ? 0.1 : 0),radius: 5 ).animation(Animation.easeInOut(duration:4)) //geometry.size.height - (multiplier * CGFloat(i)) } else{ ColorBar().frame(height: self.aniBar ? geometry.size.height - (multiplier * CGFloat(half - (i-half))) : geometry.size.height).zIndex(Double(i)).shadow(color: Color.black.opacity(self.aniBar ? 0.1 : 0),radius: 5 ).animation(Animation.easeInOut(duration:4)) } } }.frame(height: geometry.size.height) }.background(LinearGradient(gradient: Gradient(colors: [gStart,gEnd]), startPoint: .top, endPoint: .bottom)) }.edgesIgnoringSafeArea(.all) .onAppear{ self.aniBar.toggle() } .onTapGesture{ self.aniBar.toggle() } } } struct ContentView_Previews: PreviewProvider { static var previews: some View { ContentView() } } struct ColorBar: View { var g1:Color = Color(red: 95 / 255, green: 71 / 255, blue: 202 / 255) var g2:Color = Color(red: 255 / 255, green: 175 / 255, blue: 196 / 255) var g3:Color = Color(red: 230 / 255, green: 210 / 255, blue: 173 / 255) var g4:Color = Color(red: 244 / 255, green: 160 / 255, blue: 18 / 255) var g5:Color = Color(red: 228 / 255, green:0 / 255, blue: 0 / 255) var body: some View { Rectangle().fill( LinearGradient(gradient: Gradient(colors: [g1.opacity(0),g1,g2,g3,g4,g5,g5.opacity(0)]), startPoint: .top, endPoint: .bottom)) } }
Vous pouvez également le télécharger via GitHub ici et essayez également sur votre iPad avec Playgrounds.