Mixonomer-iOS/Mixonomer/Views/LoginScreen.swift

208 lines
8.0 KiB
Swift

//
// LoginScreen.swift
// Mixonomer
//
// Created by Andy Pack on 17/02/2020.
// Copyright © 2020 Sarsoo. All rights reserved.
//
import SwiftUI
import ToastUI
import KeychainAccess
import SwiftyJSON
enum ScreenMode {
case None
case Login
case Register
}
struct LoginScreen: View {
@EnvironmentObject var liveUser: LiveUser
@State private var username: String = ""
@State private var password: String = ""
@State private var password2: String = ""
@State private var showingToast = false
@State private var toastText = ""
@State public var screenMode = ScreenMode.None
var body: some View {
VStack {
Image("MusicToolsLogo")
.resizable()
.frame(width: 200.0, height: 200.0, alignment: .trailing)
.cornerRadius(18)
.shadow(radius: /*@START_MENU_TOKEN@*/10/*@END_MENU_TOKEN@*/)
.padding(.bottom, 20)
Text("Mixonomer")
.font(.largeTitle)
.padding(.bottom, 20)
if screenMode != .None {
TextField("Username", text: $username)
.textFieldStyle(.roundedBorder)
SecureField("Password", text: $password)
.textFieldStyle(.roundedBorder)
}
if screenMode == .Register {
SecureField("Password Again", text: $password2)
.textFieldStyle(.roundedBorder)
.padding(.bottom, 20)
}
if screenMode == .None {
HStack {
Button(action: {
screenMode = .Login
}) {
Text("Log In")
}
.buttonStyle(.bordered)
.padding(.trailing, 20.0)
Button(action: {
screenMode = .Register
}) {
Text("Register")
}.buttonStyle(.bordered)
}
}
if screenMode == .Login {
Button(action: {
let keychain = Keychain(service: "xyz.sarsoo.music.login")
keychain["username"] = username
let api = AuthApi.token(username: username, password: password, expiry: 604800)
RequestBuilder.buildRequest(apiRequest: api)
.validate()
.responseJSON { response in
switch response.response?.statusCode {
case 200, 201:
guard let data = response.data else {
fatalError("error getting token")
}
guard let json = try? JSON(data: data) else {
fatalError("error parsing reponse")
}
let token = json["token"].stringValue
keychain["jwt"] = token
self.liveUser.loggedIn = true
case _:
keychain["username"] = nil
keychain["jwt"] = nil
toastText = "Login Failed"
showingToast = true
}
}
}) {
Text("Log In")
}
.buttonStyle(.bordered)
.padding(.top, 20)
}
if screenMode == .Register {
Button(action: {
let keychain = Keychain(service: "xyz.sarsoo.music.login")
keychain["username"] = username
let api = AuthApi.register(username: username, password: password, password2: password2)
RequestBuilder.buildRequest(apiRequest: api)
.validate()
.responseJSON { response in
switch response.response?.statusCode {
case 200, 201:
let token_api = AuthApi.token(username: username, password: password, expiry: 604800)
RequestBuilder.buildRequest(apiRequest: token_api)
.validate()
.responseJSON { response in
switch response.response?.statusCode {
case 200, 201:
guard let data = response.data else {
fatalError("error getting token")
}
guard let json = try? JSON(data: data) else {
fatalError("error parsing reponse")
}
let token = json["token"].stringValue
keychain["jwt"] = token
self.liveUser.loggedIn = true
case _:
keychain["username"] = nil
keychain["jwt"] = nil
toastText = "Token Generation Failed"
showingToast = true
}
}
// TODO: add handling for 400, password dont match
// TODO: add handling for 409, conflict
case _:
keychain["username"] = nil
keychain["jwt"] = nil
toastText = "Register Failed"
showingToast = true
}
}
}) {
Text("Register")
}
.buttonStyle(.bordered)
}
}
.toast(isPresented: $showingToast, dismissAfter: 3.0){
ToastView(toastText)
.toastViewStyle(.failure)
}
.toastDimmedBackground(false)
.padding()
}
}
struct LoginScreen_Previews: PreviewProvider {
static var previews: some View {
Group{
LoginScreen(screenMode: .None)
.environmentObject(LiveUser(playlists: [], tags: [], username: "user", loggedIn: false))
LoginScreen(screenMode: .Login)
.environmentObject(LiveUser(playlists: [], tags: [], username: "user", loggedIn: false))
LoginScreen(screenMode: .Register)
.environmentObject(LiveUser(playlists: [], tags: [], username: "user", loggedIn: false))
}
}
}