487 lines
18 KiB
Swift
487 lines
18 KiB
Swift
//
|
|
// ChatRoomViewModel.swift
|
|
// SodaLive
|
|
//
|
|
// Created by klaus on 9/2/25.
|
|
//
|
|
|
|
import Foundation
|
|
import Combine
|
|
import Moya
|
|
|
|
final class ChatRoomViewModel: ObservableObject {
|
|
// MARK: - Published State
|
|
@Published var isResetting: Bool = false
|
|
@Published var isLoading: Bool = false
|
|
@Published var errorMessage: String = ""
|
|
@Published var isShowPopup = false
|
|
|
|
@Published var chatRoomBgImageId: Int = 0
|
|
@Published private(set) var characterId: Int64 = 0
|
|
@Published private(set) var characterProfileUrl: String = ""
|
|
@Published private(set) var characterName: String = I18n.Chat.Room.defaultCharacterName
|
|
@Published private(set) var characterType: CharacterType = .Character
|
|
@Published private(set) var chatRoomBgImageUrl: String? = nil
|
|
@Published private(set) var roomId: Int = 0 {
|
|
didSet {
|
|
isHideNotice = UserDefaults.standard.bool(forKey: noticeUserDefaultsKey())
|
|
}
|
|
}
|
|
|
|
@Published private(set) var totalRemaining: Int = 0
|
|
@Published private(set) var showQuotaNoticeView: Bool = false
|
|
|
|
@Published private(set) var showSendingMessage: Bool = false
|
|
|
|
// MARK: - Message State
|
|
@Published var messageText: String = ""
|
|
@Published private(set) var messages: [ServerChatMessage] = []
|
|
|
|
@Published var selectedMessage: ServerChatMessage? = nil
|
|
@Published var selectedMessageIndex: Int = -1
|
|
|
|
@Published var isShowImageViewer = false
|
|
@Published var selectedImageIndex: Int = 0
|
|
|
|
@Published var isHideBg = false {
|
|
didSet {
|
|
UserDefaults.standard.set(isHideBg, forKey: bgHideKey())
|
|
}
|
|
}
|
|
@Published var isShowingChatSettingsView = false
|
|
@Published var isShowingChangeBgView = false
|
|
@Published var isShowingChatResetConfirmDialog = false
|
|
@Published var isHideNotice = false {
|
|
didSet {
|
|
UserDefaults.standard.set(isHideNotice, forKey: noticeUserDefaultsKey())
|
|
}
|
|
}
|
|
|
|
var ownedImageUrls: [String] {
|
|
return messages
|
|
.filter { $0.hasAccess }
|
|
.filter { $0.messageType.lowercased() == "image" && $0.imageUrl != nil && !$0.imageUrl.isNullOrBlank() }
|
|
.map { $0.imageUrl! }
|
|
}
|
|
|
|
// MARK: - Private
|
|
private let userRepository = UserRepository()
|
|
private let repository = ChatRoomRepository()
|
|
private var subscription = Set<AnyCancellable>()
|
|
|
|
private var hasMoreMessages: Bool = true
|
|
private var nextCursor: Int64? = nil
|
|
|
|
// MARK: - Actions
|
|
func sendMessage() {
|
|
guard !messageText.trimmingCharacters(in: .whitespacesAndNewlines).isEmpty else {
|
|
return
|
|
}
|
|
|
|
if showSendingMessage {
|
|
return
|
|
}
|
|
|
|
let message = messageText.trimmingCharacters(in: .whitespacesAndNewlines)
|
|
messageText = ""
|
|
|
|
let nowMs = Int64(Date().timeIntervalSince1970 * 1000)
|
|
|
|
messages.append(
|
|
ServerChatMessage(
|
|
messageId: 0 - nowMs,
|
|
message: message,
|
|
profileImageUrl: "",
|
|
mine: true,
|
|
createdAt: nowMs,
|
|
messageType: "TEXT",
|
|
imageUrl: nil,
|
|
price: nil,
|
|
hasAccess: true
|
|
)
|
|
)
|
|
|
|
showSendingMessage = true
|
|
repository.sendMessage(roomId: roomId, message: message)
|
|
.receive(on: DispatchQueue.main)
|
|
.sink { [weak self] completion in
|
|
guard let self = self else { return }
|
|
switch completion {
|
|
case .finished:
|
|
DEBUG_LOG("finish")
|
|
case .failure(let error):
|
|
self.showSendingMessage = false // 실패 시 복구
|
|
self.errorMessage = I18n.Common.commonError
|
|
self.isShowPopup = true
|
|
ERROR_LOG(error.localizedDescription)
|
|
}
|
|
} receiveValue: { [weak self] response in
|
|
guard let self = self else { return }
|
|
let responseData = response.data
|
|
do {
|
|
let jsonDecoder = JSONDecoder()
|
|
let decoded = try jsonDecoder.decode(ApiResponse<SendChatMessageResponse>.self, from: responseData)
|
|
if let data = decoded.data, decoded.success {
|
|
self.messages.append(contentsOf: data.messages)
|
|
self.updateQuota(totalRemaining: data.totalRemaining)
|
|
} else {
|
|
self.errorMessage = decoded.message ?? I18n.Common.commonError
|
|
self.isShowPopup = true
|
|
}
|
|
self.showSendingMessage = false // 성공 시 종료
|
|
} catch {
|
|
self.showSendingMessage = false
|
|
self.errorMessage = I18n.Common.commonError
|
|
self.isShowPopup = true
|
|
}
|
|
}
|
|
.store(in: &subscription)
|
|
}
|
|
|
|
func enterRoom(roomId: Int) {
|
|
isLoading = true
|
|
self.roomId = roomId
|
|
self.isHideBg = UserDefaults.standard.bool(forKey: bgHideKey())
|
|
self.chatRoomBgImageId = getSavedBackgroundImageId() ?? 0
|
|
|
|
repository.enterChatRoom(
|
|
roomId: roomId,
|
|
characterImageId: self.chatRoomBgImageId
|
|
)
|
|
.receive(on: DispatchQueue.main)
|
|
.sink { result in
|
|
switch result {
|
|
case .finished:
|
|
DEBUG_LOG("finish")
|
|
case .failure(let error):
|
|
ERROR_LOG(error.localizedDescription)
|
|
}
|
|
} receiveValue: { [weak self] response in
|
|
let responseData = response.data
|
|
|
|
do {
|
|
let jsonDecoder = JSONDecoder()
|
|
let decoded = try jsonDecoder.decode(ApiResponse<ChatRoomEnterResponse>.self, from: responseData)
|
|
|
|
if let data = decoded.data, decoded.success {
|
|
self?.characterId = data.character.characterId
|
|
self?.characterName = data.character.name
|
|
self?.characterType = data.character.characterType
|
|
self?.characterProfileUrl = data.character.profileImageUrl
|
|
|
|
self?.chatRoomBgImageUrl = data.bgImageUrl ?? data.character.profileImageUrl
|
|
self?.messages.insert(contentsOf: data.messages.sorted(), at: 0)
|
|
|
|
self?.hasMoreMessages = data.hasMoreMessages
|
|
self?.nextCursor = data.messages.last?.messageId
|
|
|
|
self?.updateQuota(totalRemaining: data.totalRemaining)
|
|
} else {
|
|
if let message = decoded.message {
|
|
self?.errorMessage = message
|
|
} else {
|
|
self?.errorMessage = I18n.Common.commonError
|
|
}
|
|
|
|
self?.isShowPopup = true
|
|
}
|
|
|
|
self?.isLoading = false
|
|
} catch {
|
|
self?.isLoading = false
|
|
self?.errorMessage = I18n.Common.commonError
|
|
self?.isShowPopup = true
|
|
}
|
|
}
|
|
.store(in: &subscription)
|
|
}
|
|
|
|
func getMemberInfo() {
|
|
userRepository.getMemberInfo()
|
|
.receive(on: DispatchQueue.main)
|
|
.sink { result in
|
|
switch result {
|
|
case .finished:
|
|
DEBUG_LOG("finish")
|
|
case .failure(let error):
|
|
ERROR_LOG(error.localizedDescription)
|
|
}
|
|
} receiveValue: { response in
|
|
let responseData = response.data
|
|
|
|
do {
|
|
let jsonDecoder = JSONDecoder()
|
|
let decoded = try jsonDecoder.decode(ApiResponse<GetMemberInfoResponse>.self, from: responseData)
|
|
|
|
if let data = decoded.data, decoded.success {
|
|
UserDefaults.set(data.can, forKey: .can)
|
|
UserDefaults.set(data.point, forKey: .point)
|
|
}
|
|
} catch {
|
|
print(error)
|
|
}
|
|
}
|
|
.store(in: &subscription)
|
|
}
|
|
|
|
func purchaseChatMessage() {
|
|
guard let selectedMessage = selectedMessage else {
|
|
return
|
|
}
|
|
|
|
isLoading = true
|
|
|
|
repository.purchaseMessage(roomId: roomId, messageId: selectedMessage.messageId)
|
|
.receive(on: DispatchQueue.main)
|
|
.sink { result in
|
|
switch result {
|
|
case .finished:
|
|
DEBUG_LOG("finish")
|
|
case .failure(let error):
|
|
ERROR_LOG(error.localizedDescription)
|
|
}
|
|
} receiveValue: { [weak self] response in
|
|
let responseData = response.data
|
|
|
|
do {
|
|
let jsonDecoder = JSONDecoder()
|
|
let decoded = try jsonDecoder.decode(ApiResponse<ServerChatMessage>.self, from: responseData)
|
|
|
|
if let data = decoded.data, decoded.success {
|
|
self?.messages.insert(data, at: self?.selectedMessageIndex ?? 0)
|
|
self?.messages.remove(at: (self?.selectedMessageIndex ?? 0) + 1)
|
|
|
|
self?.selectedMessage = nil
|
|
self?.selectedMessageIndex = -1
|
|
} else {
|
|
if let message = decoded.message {
|
|
self?.errorMessage = message
|
|
} else {
|
|
self?.errorMessage = I18n.Common.commonError
|
|
}
|
|
|
|
self?.isShowPopup = true
|
|
}
|
|
|
|
self?.isLoading = false
|
|
} catch {
|
|
self?.isLoading = false
|
|
self?.errorMessage = I18n.Common.commonError
|
|
self?.isShowPopup = true
|
|
}
|
|
}
|
|
.store(in: &subscription)
|
|
}
|
|
|
|
func purchaseChatQuota(canOption: ChatRoomQuotaCanOption) {
|
|
purchaseChatQuota(chargeType: .can, canOption: canOption)
|
|
}
|
|
|
|
func showRewardedAdForChatQuota() {
|
|
_Concurrency.Task {
|
|
await YandexRewardedAdManager.shared.showAdIfAvailable(for: .chatRoomQuota) { [weak self] in
|
|
self?.purchaseChatQuota(chargeType: .ad, canOption: nil)
|
|
}
|
|
}
|
|
}
|
|
|
|
private func purchaseChatQuota(
|
|
chargeType: ChatRoomQuotaChargeType,
|
|
canOption: ChatRoomQuotaCanOption?
|
|
) {
|
|
isLoading = true
|
|
|
|
repository.purchaseChatQuota(roomId: roomId, chargeType: chargeType, canOption: canOption)
|
|
.receive(on: DispatchQueue.main)
|
|
.sink { result in
|
|
switch result {
|
|
case .finished:
|
|
DEBUG_LOG("finish")
|
|
case .failure(let error):
|
|
ERROR_LOG(error.localizedDescription)
|
|
}
|
|
} receiveValue: { [weak self] response in
|
|
let responseData = response.data
|
|
|
|
do {
|
|
let jsonDecoder = JSONDecoder()
|
|
let decoded = try jsonDecoder.decode(ApiResponse<ChatQuotaStatusResponse>.self, from: responseData)
|
|
|
|
if let data = decoded.data, decoded.success {
|
|
self?.updateQuota(totalRemaining: data.totalRemaining)
|
|
|
|
if let canOption {
|
|
let can = UserDefaults.int(forKey: .can)
|
|
UserDefaults.set(can - canOption.needCan, forKey: .can)
|
|
}
|
|
} else {
|
|
if let message = decoded.message {
|
|
self?.errorMessage = message
|
|
} else {
|
|
self?.errorMessage = I18n.Common.commonError
|
|
}
|
|
|
|
self?.isShowPopup = true
|
|
}
|
|
|
|
self?.isLoading = false
|
|
} catch {
|
|
self?.isLoading = false
|
|
self?.errorMessage = I18n.Common.commonError
|
|
self?.isShowPopup = true
|
|
}
|
|
}
|
|
.store(in: &subscription)
|
|
}
|
|
|
|
func resetChatRoom() {
|
|
isResetting = true
|
|
|
|
repository.resetChatRoom(roomId: roomId)
|
|
.receive(on: DispatchQueue.main)
|
|
.sink { result in
|
|
switch result {
|
|
case .finished:
|
|
DEBUG_LOG("finish")
|
|
case .failure(let error):
|
|
ERROR_LOG(error.localizedDescription)
|
|
}
|
|
} receiveValue: { [weak self] response in
|
|
let responseData = response.data
|
|
self?.isResetting = false
|
|
|
|
do {
|
|
let jsonDecoder = JSONDecoder()
|
|
let decoded = try jsonDecoder.decode(ApiResponse<CreateChatRoomResponse>.self, from: responseData)
|
|
|
|
if let data = decoded.data, decoded.success {
|
|
self?.resetData()
|
|
self?.getMemberInfo()
|
|
self?.enterRoom(roomId: data.chatRoomId)
|
|
} else {
|
|
if let message = decoded.message {
|
|
self?.errorMessage = message
|
|
} else {
|
|
self?.errorMessage = I18n.Common.commonError
|
|
}
|
|
|
|
self?.isShowPopup = true
|
|
}
|
|
} catch {
|
|
ERROR_LOG(String(describing: error))
|
|
self?.errorMessage = I18n.Common.commonError
|
|
self?.isShowPopup = true
|
|
}
|
|
}
|
|
.store(in: &subscription)
|
|
}
|
|
|
|
func showImageViewer(_ imageUrl: String?) {
|
|
if let imageUrl = imageUrl {
|
|
selectedImageIndex = ownedImageUrls.firstIndex(of: imageUrl) ?? 0
|
|
isShowImageViewer = true
|
|
}
|
|
}
|
|
|
|
func setBackgroundImage(imageItem: CharacterImageListItemResponse) {
|
|
UserDefaults.standard.set(imageItem.id, forKey: bgImageIdKey())
|
|
chatRoomBgImageUrl = imageItem.imageUrl
|
|
chatRoomBgImageId = imageItem.id
|
|
}
|
|
|
|
private func noticeUserDefaultsKey() -> String {
|
|
return "chat_notice_hidden_room_\(roomId)"
|
|
}
|
|
|
|
private func resetData() {
|
|
characterProfileUrl = ""
|
|
characterName = I18n.Chat.Room.defaultCharacterName
|
|
characterType = .Character
|
|
chatRoomBgImageUrl = nil
|
|
roomId = 0
|
|
|
|
totalRemaining = 0
|
|
showQuotaNoticeView = false
|
|
|
|
showSendingMessage = false
|
|
|
|
messageText = ""
|
|
messages = []
|
|
selectedMessage = nil
|
|
selectedMessageIndex = -1
|
|
isShowImageViewer = false
|
|
selectedImageIndex = 0
|
|
|
|
isShowingChatSettingsView = false
|
|
isShowingChangeBgView = false
|
|
isShowingChatResetConfirmDialog = false
|
|
}
|
|
|
|
private func checkQuotaStatus() {
|
|
isLoading = true
|
|
|
|
repository.getChatQuotaStatus(roomId: roomId)
|
|
.sink { result in
|
|
switch result {
|
|
case .finished:
|
|
DEBUG_LOG("finish")
|
|
case .failure(let error):
|
|
ERROR_LOG(error.localizedDescription)
|
|
}
|
|
} receiveValue: { [weak self] response in
|
|
let responseData = response.data
|
|
|
|
do {
|
|
let jsonDecoder = JSONDecoder()
|
|
let decoded = try jsonDecoder.decode(ApiResponse<ChatQuotaStatusResponse>.self, from: responseData)
|
|
|
|
if let data = decoded.data, decoded.success {
|
|
self?.updateQuota(totalRemaining: data.totalRemaining)
|
|
} else {
|
|
if let message = decoded.message {
|
|
self?.errorMessage = message
|
|
} else {
|
|
self?.errorMessage = I18n.Common.commonError
|
|
}
|
|
|
|
self?.isShowPopup = true
|
|
}
|
|
|
|
self?.isLoading = false
|
|
} catch {
|
|
self?.isLoading = false
|
|
self?.errorMessage = I18n.Common.commonError
|
|
self?.isShowPopup = true
|
|
}
|
|
}
|
|
.store(in: &subscription)
|
|
}
|
|
|
|
private func updateQuota(totalRemaining: Int) {
|
|
self.totalRemaining = totalRemaining
|
|
showQuotaNoticeView = totalRemaining <= 0
|
|
prepareRewardedAdIfNeeded(totalRemaining: totalRemaining)
|
|
}
|
|
|
|
private func prepareRewardedAdIfNeeded(totalRemaining: Int) {
|
|
guard totalRemaining <= 1 else { return }
|
|
|
|
_Concurrency.Task {
|
|
await YandexRewardedAdManager.shared.preloadAd(for: .chatRoomQuota)
|
|
}
|
|
}
|
|
|
|
private func getSavedBackgroundImageId() -> Int? {
|
|
let imageId = UserDefaults.standard.integer(forKey: bgImageIdKey())
|
|
return imageId > 0 ? imageId : nil
|
|
}
|
|
|
|
private func bgImageIdKey() -> String {
|
|
return "chat_bg_image_id_room_\(roomId)"
|
|
}
|
|
|
|
private func bgHideKey() -> String {
|
|
return "chat_bg_hide_room_\(roomId)"
|
|
}
|
|
}
|