IOS(Swift)

[IOS/Swift] ์ฝ˜ํ…์ธ  ๊ฒ€์ƒ‰ ์•ฑ ๊ตฌํ˜„ - 4. ๋กœ๊ทธ์ธ UI ๊ตฌํ˜„

Tempo 2022. 4. 4. 22:00

Swift UI - 4. ๋กœ๊ทธ์ธ UI ํ™”๋ฉด ๊ตฌํ˜„

์ด์ „์— ๊ตฌํ˜„ํ•œ ํšŒ์›๊ฐ€์ž…์œผ๋กœ ํšŒ์›์ด ๋“ฑ๋ก๋˜์—ˆ๋‹ค(DRF๋กœ ํšŒ์›๊ฐ€์ž… API๋ฅผ ๊ตฌํ˜„) ์ด์ œ ๋“ฑ๋ก๋œ ํšŒ์›์œผ๋กœ ๋กœ๊ทธ์ธ์„ ์ง„ํ–‰ํ•˜๊ณ  ํ‚ค์ฒด์ธ์„ ์ด์šฉํ•ด์„œ Access Token ๋ฐ Refresh Token์„ ์ €์žฅํ•˜์—ฌ ์•ฑ ์‚ฌ์šฉ ์ค‘ ๋กœ๊ทธ์ธ ์ƒํƒœ๋ฅผ ์œ ์ง€ํ•  ์ˆ˜ ์žˆ๋„๋ก ํ•ด๋ณด์ž.

//
//  LoginView.swift
//  mbtiPlayground
//
//  Created by ๋ฐ•์ข…ํ›„ on 2022/03/16.
//

import SwiftUI

struct LoginView: View {
    
    @State private var email: String = ""
    @State private var password: String = ""
    @State private var loginStatus: Bool = false // TODO env ๋ณ€์ˆ˜๋กœ ์„ ์–ธ ํ›„ ํ† ํฐ ๊ณ„์† ํ™•์ธ
    @State private var userAccessToken: String = ""
    
    
    var body: some View {
        
        NavigationView{
            if self.loginStatus != false {
                ProfileDetail(userEmail: self.email, userAccessToken: self.userAccessToken)
            } else {
                VStack{
                    Text("Sign In")
                        .font(.title)
                        .fontWeight(.medium)
                        .padding()
                        .foregroundColor(.black)
                    HStack{
                        Image(systemName: "person.fill")
                            .resizable()
                            .scaledToFit()
                            .frame(width: 30, height: 30)
                            .padding(.bottom)
                        
                        TextField("๋‹‰๋„ค์ž„์„ ์ž…๋ ฅํ•˜์„ธ์š”.", text: $email)
                            .frame(width: 300, height: 10)
                            .padding()
                            .background(Color(.systemGray6))
                            .cornerRadius(5.0)
                            .padding(.bottom, 20)
                            
                    }
                    HStack{
                        Image(systemName: "lock")
                            .resizable()
                            .scaledToFit()
                            .frame(width: 30, height: 30)
                            .padding(.bottom)
                            
                        SecureField("๋น„๋ฐ€๋ฒˆํ˜ธ๋ฅผ ์ž…๋ ฅํ•˜์„ธ์š”", text: $password)
                            .frame(width: 300, height: 10)
                            .padding()
                            .background(Color(.systemGray6))
                            .cornerRadius(5.0)
                            .padding(.bottom, 20)
                    }
                    
                    HStack{
                        Button(action: {
                            print(self.email + self.password)
                            
                            let rft = readItemKeyChain(userId: self.email)
                            if rft != nil {
                                UserDefaults.standard.set(rft, forKey: self.email)
                            }else{
                                sendPostRequest("<http://localhost:8000/auth/login>", parameters: ["username": self.email, "password": self.password]){
                                    responseObject, error in guard let _ = responseObject, error == nil else {
                                        print(error ?? "Unknown error")
                                        return
                                    }
                                    self.loginStatus = true
                                    
                                    if let rftToken = responseObject{
                                        let rft = rftToken["refresh"] as? String
                                        self.userAccessToken = rftToken["access"] as? String ?? ""
                                        setItemKeyChain(userId: self.email, rft: rft!)
                                        UserDefaults.standard.set(rft, forKey: self.email)
                                    }
                                }
                            }
                        }){
                            Text("๋กœ๊ทธ์ธ")
                                .frame(width: 80, height: 10)
                                .font(.headline)
                                .foregroundColor(.white)
                                .padding()
                                .background(Color(.systemBlue))
                                .cornerRadius(10)
                                
                        }
                        .padding()
                        
                        NavigationLink(destination: SignUpView()){
                            Text("ํšŒ์›๊ฐ€์ž…")
                                .frame(width: 80, height: 10)
                                .font(.headline)
                                .foregroundColor(.white)
                                .padding()
                                .background(Color(.systemBlue))
                                .cornerRadius(10)
                        }
                        
                    }
                    
                }
                .padding(.all, 30)
                .navigationBarTitle("", displayMode: .inline)
                .navigationBarHidden(true)
            }
        }
    }
}

struct LoginView_Previews: PreviewProvider {
    static var previews: some View {
        LoginView()
    }
}

์šฐ์„  21๋ฒˆ ์งธ ์ค„์—์„œ ์œ ์ €์˜ ๋กœ๊ทธ์ธ ์ƒํƒœ์— ๋”ฐ๋ผ ํ”„๋กœํ•„ ์ƒํƒœ(๋กœ๊ทธ์ธ ํ›„ ์œ ์ €์˜ ์ •๋ณด๋ฅผ ๋ณด์—ฌ์ค„ ํŽ˜์ด์ง€)๋ฅผ ๋ณด์—ฌ์ค„์ง€ ์•„๋‹ˆ๋ฉด ๋กœ๊ทธ์ธ ํ™”๋ฉด์„ ๋ณด์—ฌ์ค„์ง€์— ๋Œ€ํ•ด ์กฐ๊ฑด๋ฌธ์„ ๋„ฃ์—ˆ๋‹ค.

UI์—์„œ๋Š” ๋‹‰๋„ค์ž„๊ณผ ๋น„๋ฐ€๋ฒˆํ˜ธ๋ฅผ ์ž…๋ ฅํ•˜์—ฌ ๋กœ๊ทธ์ธํ•˜๋Š” ํ™”๋ฉด์„ ๊ตฌํ˜„ํ•˜์˜€๋‹ค. (๋กœ๊ทธ์ธ ์‹œ ์ด๋ฉ”์ผ ๋˜๋Š” ๋‹‰๋„ค์ž„์„ ์‚ฌ์šฉํ•˜๋„๋ก ๊ตฌํ˜„ํ•  ์ˆ˜ ์žˆ๋Š”๋ฐ, ํ˜„์žฌ DRF์—์„œ๋Š” ๋‹‰๋„ค์ž„๋งŒ ์‚ฌ์šฉํ•˜์—ฌ ๋กœ๊ทธ์ธํ•˜๋„๋ก ์„ค์ •ํ•œ ์ƒํƒœ)

์—ฌ๊ธฐ์„œ ๋กœ๊ทธ์ธ ์„ฑ๊ณต ํ›„ ํ† ํฐ์„ ๋ฐœ๊ธ‰๋ฐ›๋Š” ๊ณผ์ •์—์„œ ๋ฆฌํ”„๋ ˆ์‹œ ํ† ํฐ์„ ํ‚ค์ฒด์ธ์— ์ €์žฅํ•˜๊ณ  ๋‚˜์ค‘์— ์œ ์ €๊ฐ€ ๋‹ค์‹œ ์ ‘์†ํ–ˆ์„ ๋•Œ์—๋„ ๋‹ค์‹œ ๋กœ๊ทธ์ธ ํ•˜์ง€ ์•Š๋„๋ก ๊ตฌํ˜„ํ•˜์˜€๋‹ค.

setItemKeyChain(userId: self.email, rft: rft!)
UserDefaults.standard.set(rft, forKey: self.email)

setItemKeyChain ํ•จ์ˆ˜๋Š” ๋กœ๊ทธ์ธํ•œ ์œ ์ €์˜ ๋ฆฌํ”„๋ ˆ์‹œ ํ† ํฐ ์ •๋ณด๋ฅผ ์ €์žฅํ•˜๊ธฐ ์œ„ํ•ด ์ž‘์„ฑํ•˜์˜€๋‹ค.

func setItemKeyChain(userId: String, rft: String){
    let previousQuery: [CFString: Any] = [kSecClass: kSecClassIdentity, kSecAttrAccount: userId]
    
    let updateQuery: [CFString: Any] = [kSecValueData: rft]
    let status = SecItemUpdate(previousQuery as CFDictionary, updateQuery as CFDictionary)
    
    if status == errSecSuccess {
        print("Update!")
    } else {
        print("Update fail")
    }
}

์ถ”๊ฐ€์ ์œผ๋กœ ์œ ์ €์˜ ๋ฐ์ดํ„ฐ๋ฅผ ์ €์žฅํ•˜๊ธฐ ์œ„ํ•ด UserDefaults ๋„ ์‚ฌ์šฉํ•˜์˜€๋‹ค. UserDefaults ๋Š” Redis ์‚ฌ์šฉ๊ณผ ์œ ์‚ฌํ•˜๋‹ค

์œ„ ๋‘ ์ฝ”๋“œ๋ฅผ ํ†ตํ•ด ์‚ฌ์šฉ์ž๊ฐ€ ๋กœ๊ทธ์ธ ํ›„ ๋ฐœ๊ธ‰๋ฐ›์€ ์•ก์„ธ์Šค ํ† ํฐ ๋ฐ ๋ฆฌํ”„๋ ˆ์‹œ ํ† ํฐ์„ ์‚ฌ์šฉ์ž์˜ ๊ธฐ๊ธฐ์— ์ €์žฅํ•˜์—ฌ ๋กœ๊ทธ์•„์›ƒ ์—†์ด ๊ณ„์† ์‚ฌ์šฉํ•  ์ˆ˜ ์žˆ๋‹ค.

๋ฐ˜์‘ํ˜•