Files
sodalive-ios/SodaLive/Sources/MyPage/Profile/ProfileUpdateViewModel.swift

355 lines
13 KiB
Swift

//
// ProfileUpdateViewModel.swift
// SodaLive
//
// Created by klaus on 2023/08/19.
//
import UIKit
import Combine
import Moya
final class ProfileUpdateViewModel: ObservableObject {
private let repository = UserRepository()
private var subscription = Set<AnyCancellable>()
@Published var isLoading = false
@Published var errorMessage = ""
@Published var isShowPopup = false
@Published var email = ""
@Published var nickname = ""
@Published var youtubeUrl = ""
@Published var instagramUrl = ""
@Published var kakaoOpenChatUrl = ""
@Published var fancimmUrl = ""
@Published var xUrl = ""
@Published var gender: Gender = .NONE
@Published var introduce = ""
@Published var currentPassword = ""
@Published var newPassword = ""
@Published var newPasswordConfirm = ""
@Published var tags = [String]()
@Published var insertTags = [String]()
@Published var removeTags = [String]()
var profileImage: UIImage? = nil {
didSet {
if let _ = profileImage {
updateProfileImage()
}
}
}
var profileResponse: GetProfileResponse?
var refresh: () -> Void = {}
let placeholder = I18n.ProfileUpdate.introductionPlaceholder
func getMyProfile() {
isLoading = true
repository.getMyProfile()
.sink { result in
switch result {
case .finished:
DEBUG_LOG("finish")
case .failure(let error):
ERROR_LOG(error.localizedDescription)
}
} receiveValue: { [unowned self] response in
self.isLoading = false
let responseData = response.data
do {
let jsonDecoder = JSONDecoder()
let decoded = try jsonDecoder.decode(ApiResponse<GetProfileResponse>.self, from: responseData)
if let data = decoded.data, decoded.success {
self.profileResponse = data
self.email = data.email
self.nickname = data.nickname
self.youtubeUrl = data.youtubeUrl ?? ""
self.instagramUrl = data.instagramUrl ?? ""
self.kakaoOpenChatUrl = data.kakaoOpenChatUrl ?? ""
self.fancimmUrl = data.fancimmUrl ?? ""
self.xUrl = preferredXUrl(xUrl: data.xUrl, xurl: data.xurl)
self.introduce = data.introduce
self.gender = data.gender
self.tags.append(contentsOf: data.tags)
UserDefaults.set(data.nickname, forKey: .nickname)
} else {
if let message = decoded.message {
self.errorMessage = message
} else {
self.errorMessage = I18n.Common.commonError
}
self.isShowPopup = true
}
} catch {
self.errorMessage = I18n.Common.commonError
self.isShowPopup = true
}
}
.store(in: &subscription)
}
func updateProfile() {
let originalYouTubeUrl = profileResponse!.youtubeUrl ?? ""
let originalInstagramUrl = profileResponse!.instagramUrl ?? ""
let originalKakaoOpenChatUrl = profileResponse!.kakaoOpenChatUrl ?? ""
let originalFancimmUrl = profileResponse!.fancimmUrl ?? ""
let originalXUrl = preferredXUrl(xUrl: profileResponse!.xUrl, xurl: profileResponse!.xurl)
if profileResponse!.nickname != nickname ||
originalYouTubeUrl != youtubeUrl ||
originalInstagramUrl != instagramUrl ||
originalKakaoOpenChatUrl != kakaoOpenChatUrl ||
originalFancimmUrl != fancimmUrl ||
originalXUrl != xUrl ||
profileResponse!.gender != gender ||
profileResponse!.introduce != introduce ||
!insertTags.isEmpty ||
!removeTags.isEmpty {
let request = ProfileUpdateRequest(
email: profileResponse!.email,
nickname: profileResponse!.nickname != nickname ? nickname : nil,
gender: profileResponse!.gender != gender ? gender : nil,
introduce: profileResponse!.introduce != introduce && introduce.trimmingCharacters(in: .whitespacesAndNewlines) != placeholder ? introduce : nil,
youtubeUrl: originalYouTubeUrl != youtubeUrl ? youtubeUrl : nil,
instagramUrl: originalInstagramUrl != instagramUrl ? instagramUrl : nil,
fancimmUrl: originalFancimmUrl != fancimmUrl ? fancimmUrl : nil,
xUrl: originalXUrl != xUrl ? xUrl : nil,
kakaoOpenChatUrl: originalKakaoOpenChatUrl != kakaoOpenChatUrl ? kakaoOpenChatUrl : nil,
insertTags: !insertTags.isEmpty ? insertTags : nil,
removeTags: !removeTags.isEmpty ? removeTags : nil
)
isLoading = true
repository.profileUpdate(request: request)
.sink { result in
switch result {
case .finished:
DEBUG_LOG("finish")
case .failure(let error):
ERROR_LOG(error.localizedDescription)
}
} receiveValue: { [unowned self] response in
self.isLoading = false
let responseData = response.data
do {
let jsonDecoder = JSONDecoder()
let decoded = try jsonDecoder.decode(ApiResponse<GetProfileResponse>.self, from: responseData)
if let _ = decoded.data, decoded.success {
self.refresh()
self.errorMessage = I18n.ProfileUpdate.profileUpdated
self.isShowPopup = true
DispatchQueue.main.asyncAfter(deadline: .now() + 1) {
AppState.shared.back()
}
} else {
if let message = decoded.message {
self.errorMessage = message
} else {
self.errorMessage = I18n.Common.commonError
}
self.isShowPopup = true
}
} catch {
self.errorMessage = I18n.Common.commonError
self.isShowPopup = true
}
}
.store(in: &subscription)
} else {
AppState.shared.back()
}
}
func updatePassword() {
if currentPassword.trimmingCharacters(in: .whitespaces).isEmpty {
errorMessage = I18n.ProfileUpdate.passwordCurrentRequired
isShowPopup = true
return
}
if newPassword.trimmingCharacters(in: .whitespaces).isEmpty {
errorMessage = I18n.ProfileUpdate.passwordNewRequired
isShowPopup = true
return
}
if newPassword != newPasswordConfirm {
errorMessage = I18n.ProfileUpdate.passwordNotMatch
isShowPopup = true
return
}
if !validatePassword() {
errorMessage = I18n.ProfileUpdate.passwordRuleHint
isShowPopup = true
return
}
let request = ProfileUpdateRequest(
email: UserDefaults.string(forKey: .email),
password: currentPassword,
modifyPassword: newPassword
)
isLoading = true
repository.profileUpdate(request: request)
.sink { result in
switch result {
case .finished:
DEBUG_LOG("finish")
case .failure(let error):
ERROR_LOG(error.localizedDescription)
}
} receiveValue: { [unowned self] response in
self.isLoading = false
let responseData = response.data
do {
let jsonDecoder = JSONDecoder()
let decoded = try jsonDecoder.decode(ApiResponse<GetProfileResponse>.self, from: responseData)
if let _ = decoded.data, decoded.success {
self.errorMessage = I18n.ProfileUpdate.passwordUpdated
self.isShowPopup = true
self.currentPassword = ""
self.newPassword = ""
self.newPasswordConfirm = ""
DispatchQueue.main.asyncAfter(deadline: .now() + 1) {
AppState.shared.back()
}
} else {
if let message = decoded.message {
self.errorMessage = message
} else {
self.errorMessage = I18n.Common.commonError
}
self.isShowPopup = true
}
} catch {
self.errorMessage = I18n.Common.commonError
self.isShowPopup = true
}
}
.store(in: &subscription)
}
func removeTag(tag: String) {
if let index = tags.firstIndex(of: tag) {
tags.remove(at: index)
}
if (insertTags.contains(tag)) {
if let index = insertTags.firstIndex(of: tag) {
insertTags.remove(at: index)
}
} else {
removeTags.append(tag)
}
}
func addTag(tag: String) {
tags.append(tag)
if (removeTags.contains(tag)) {
if let index = removeTags.firstIndex(of: tag) {
removeTags.remove(at: index)
}
} else {
insertTags.append(tag)
}
}
private func updateProfileImage() {
isLoading = false
if let profileImage = profileImage, let imageData = profileImage.jpegData(compressionQuality: 0.8) {
repository
.profileImageUpdate(parameter: MultipartFormData(
provider: .data(imageData),
name: "image",
fileName: "\(UUID().uuidString)_\(Date().timeIntervalSince1970 * 1000).jpg",
mimeType: "image/*"
))
.sink { result in
switch result {
case .finished:
DEBUG_LOG("finish")
case .failure(let error):
ERROR_LOG(error.localizedDescription)
}
} receiveValue: { [unowned self] response in
self.isLoading = false
let responseData = response.data
do {
let jsonDecoder = JSONDecoder()
let decoded = try jsonDecoder.decode(ApiResponse<String>.self, from: responseData)
if let profileImageUrl = decoded.data, decoded.success {
UserDefaults.set(profileImageUrl, forKey: .profileImage)
self.refresh()
self.getMyProfile()
} else {
if let message = decoded.message {
self.errorMessage = message
} else {
self.errorMessage = I18n.Common.commonError
}
self.isShowPopup = true
}
} catch {
self.errorMessage = I18n.Common.commonError
self.isShowPopup = true
}
}
.store(in: &subscription)
} else {
self.errorMessage = I18n.ProfileUpdate.profileImageUpdateFailed
self.isShowPopup = true
self.isLoading = false
}
}
private func validatePassword() -> Bool {
let passwordRegEx = "^(?=.*[A-Za-z])(?=.*\\d)[A-Za-z\\d$@!%*#?&]{8,}$"
let predicate = NSPredicate(format:"SELF MATCHES %@", passwordRegEx)
return predicate.evaluate(with: newPassword)
}
private func preferredXUrl(xUrl: String?, xurl: String?) -> String {
if let xUrl,
!xUrl.trimmingCharacters(in: .whitespacesAndNewlines).isEmpty {
return xUrl
}
if let xurl,
!xurl.trimmingCharacters(in: .whitespacesAndNewlines).isEmpty {
return xurl
}
return ""
}
}