//
//  LiveRoomCreateViewModel.swift
//  SodaLive
//
//  Created by klaus on 2023/08/14.
//

import UIKit
import Moya
import Combine

enum SelectedMenu: Int {
    case MENU_1 = 0
    case MENU_2 = 1
    case MENU_3 = 2
}

final class LiveRoomCreateViewModel: ObservableObject {
    enum TimeSettingMode: String {
        case NOW, RESERVATION
    }
    
    enum LiveRoomType: String, Codable {
        case OPEN, PRIVATE
    }
    
    let prices = [0, 100, 300, 500, 1000, 2000]
    
    @Published var isLoading = false
    @Published var tags: [String] = []
    @Published var title: String = ""
    @Published var content: String = "" {
        didSet {
            if content.count > 1000 {
                content = String(content.prefix(1000))
            }
        }
    }
    
    @Published var coverImage: UIImage? = nil
    @Published var coverImageUrl: String? = nil
    @Published var numberOfPeople = ""
    @Published var timeSettingMode: TimeSettingMode = .NOW
    @Published var reservationDateString: String = Date().convertDateFormat(dateFormat: "yyyy.MM.dd")
    @Published var reservationTimeString: String = Date().convertDateFormat(dateFormat: "a hh : mm")
    
    @Published var roomType: LiveRoomType = .OPEN {
        didSet {
            if roomType != .PRIVATE {
                password = ""
            }
        }
    }
    
    @Published var password = "" {
        didSet {
            if password.count > 6 {
                password = String(password.prefix(6))
            }
        }
    }
    
    @Published var isAdult = false
    @Published var priceString = "0" {
        didSet {
            if priceString.count > 5 {
                priceString = String(priceString.prefix(5))
            } else {
                if let price = Int(priceString) {
                    self.price = price
                } else {
                    self.price = 0
                }
            }
        }
    }
    @Published var price = 0
    
    @Published var errorMessage = ""
    @Published var isShowPopup = false
    @Published var isShowGetRecentInfoButton = true
    
    
    private var menuId = 0
    @Published var menu = ""
    @Published var menuList = [GetMenuPresetResponse]()
    
    @Published var isActivateMenu = false
    @Published var selectedMenu: SelectedMenu? = nil
    
    @Published var isAvailableJoinCreator = true
    
    private let repository = LiveRepository()
    private var subscription = Set<AnyCancellable>()
    
    var reservationDate = Date() {
        willSet {
            reservationDateString = newValue.convertDateFormat(dateFormat: "yyyy.MM.dd")
        }
    }
    
    var reservationTime = Date() {
        willSet {
            reservationTimeString = newValue.convertDateFormat(dateFormat: "a hh : mm")
        }
    }
    
    var coverImagePath: String? = nil
    
    let placeholder = "라이브 공지를 입력하세요"
    
    func getRecentInfo() {
        isLoading = true
        
        repository.getRecentRoomInfo()
            .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
                
                DispatchQueue.global(qos: .background).async {
                    do {
                        let jsonDecoder = JSONDecoder()
                        let decoded = try jsonDecoder.decode(ApiResponse<GetRecentRoomInfoResponse>.self, from: responseData)
                        
                        if let data = decoded.data, decoded.success {
                            DispatchQueue.main.async {
                                self.title = data.title
                                self.content = data.notice
                                self.coverImageUrl = data.coverImageUrl
                                self.coverImagePath = data.coverImagePath
                                self.numberOfPeople = String(data.numberOfPeople)
                                
                                self.errorMessage = "최근데이터를 불러왔습니다."
                                self.isShowPopup = true
                                self.isShowGetRecentInfoButton = false
                            }
                        } else {
                            DispatchQueue.main.async {
                                if let message = decoded.message {
                                    self.errorMessage = message
                                } else {
                                    self.errorMessage = "최근데이터를 불러오지 못했습니다.\n다시 시도해 주세요."
                                }
                                
                                self.isShowPopup = true
                            }
                        }
                    } catch {
                        print(error)
                        DispatchQueue.main.async {
                            self.errorMessage = "최근데이터를 불러오지 못했습니다.\n다시 시도해 주세요."
                            self.isShowPopup = true
                        }
                    }
                }
            }
            .store(in: &subscription)
    }
    
    func createRoom(onSuccess: @escaping (CreateLiveRoomResponse) -> Void) {
        if !isLoading && validate() {
            isLoading = true
            
            var request = CreateLiveRoomRequest(
                title: title,
                content: content.trimmingCharacters(in: .whitespacesAndNewlines) != placeholder ? content : "",
                coverImageUrl: coverImagePath,
                tags: tags,
                numberOfPeople: Int(numberOfPeople)!,
                isAdult: isAdult,
                price: price,
                type: roomType,
                password: (roomType == .PRIVATE && !password.trimmingCharacters(in: .whitespaces).isEmpty) ? password : nil,
                menuPanId: isActivateMenu ? menuId : 0,
                menuPan: isActivateMenu ? menu : "",
                isActiveMenuPan: isActivateMenu,
                isAvailableJoinCreator: isAvailableJoinCreator
            )
            
            if timeSettingMode == .RESERVATION {
                request.beginDateTimeString = "\(reservationDate.convertDateFormat(dateFormat: "yyyy-MM-dd")) \(reservationTime.convertDateFormat(dateFormat: "HH:mm"))"
            }
            
            var multipartData = [MultipartFormData]()
            
            let encoder = JSONEncoder()
            encoder.outputFormatting = .withoutEscapingSlashes
            let jsonData = try? encoder.encode(request)
            
            if let jsonData = jsonData {
                
                if let coverImage = coverImage, let imageData = coverImage.jpegData(compressionQuality: 0.8) {
                    multipartData.append(
                        MultipartFormData(
                            provider: .data(imageData),
                            name: "coverImage",
                            fileName: "\(UUID().uuidString)_\(Date().timeIntervalSince1970 * 1000).jpg",
                            mimeType: "image/*")
                    )
                }
                
                multipartData.append(MultipartFormData(provider: .data(jsonData), name: "request"))
                
                repository
                    .createRoom(parameters: multipartData)
                    .sink { result in
                        switch result {
                        case .finished:
                            DEBUG_LOG("finish")
                        case .failure(let error):
                            ERROR_LOG(error.localizedDescription)
                        }
                    } receiveValue: { response in
                        self.isLoading = false
                        let responseData = response.data
                        
                        do {
                            let jsonDecoder = JSONDecoder()
                            let decoded = try jsonDecoder.decode(ApiResponse<CreateLiveRoomResponse>.self, from: responseData)
                            
                            if let data = decoded.data, decoded.success {
                                onSuccess(data)
                            } else {
                                if let message = decoded.message {
                                    self.errorMessage = message
                                } else {
                                    self.errorMessage = "다시 시도해 주세요.\n계속 같은 문제가 발생할 경우 고객센터로 문의 주시기 바랍니다."
                                }
                                
                                self.isShowPopup = true
                            }
                        } catch {
                            self.errorMessage = "다시 시도해 주세요.\n계속 같은 문제가 발생할 경우 고객센터로 문의 주시기 바랍니다."
                            self.isShowPopup = true
                        }
                    }
                    .store(in: &subscription)
            } else {
                self.errorMessage = "라이브을 만들지 못했습니다.\n다시 시도해 주세요.\n계속 같은 문제가 발생할 경우 고객센터로 문의 주시기 바랍니다."
                self.isShowPopup = true
                self.isLoading = false
            }
        }
    }
    
    func selectMenuPreset(selectedMenuPreset: SelectedMenu) {
        if menuList.isEmpty && (selectedMenuPreset == .MENU_2 || selectedMenuPreset == .MENU_3) {
            errorMessage = "메뉴 1을 먼저 설정하세요"
            isShowPopup = true
            return
        }
        
        if menuList.count == 1 && selectedMenuPreset == .MENU_3 {
            errorMessage = "메뉴 1과 메뉴 2를 먼저 설정하세요"
            isShowPopup = true
            return
        }
        
        if self.selectedMenu != selectedMenuPreset {
            self.selectedMenu = selectedMenuPreset
            
            if menuList.count > selectedMenuPreset.rawValue {
                let menu = menuList[selectedMenuPreset.rawValue]
                self.menu = menu.menu
                self.menuId = menu.id
            } else {
                self.menu = ""
                self.menuId = 0
            }
        }
    }
    
    func getAllMenuPreset() {
        isLoading = true
        
        repository.getAllMenuPreset(creatorId: UserDefaults.int(forKey: .userId))
            .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<[GetMenuPresetResponse]>.self, from: responseData)
                    
                    if let data = decoded.data, decoded.success {
                        self.menuList.removeAll()
                        self.menuList.append(contentsOf: data)
                        self.selectMenuPreset(selectedMenuPreset: .MENU_1)
                    } else {
                        if let message = decoded.message {
                            self.errorMessage = message
                        } else {
                            self.errorMessage = "다시 시도해 주세요.\n계속 같은 문제가 발생할 경우 고객센터로 문의 주시기 바랍니다."
                        }
                        
                        self.isShowPopup = true
                    }
                } catch {
                    self.errorMessage = "다시 시도해 주세요.\n계속 같은 문제가 발생할 경우 고객센터로 문의 주시기 바랍니다."
                    self.isShowPopup = true
                }
            }
            .store(in: &subscription)
    }
    
    private func validate() -> Bool {
        if coverImage == nil && coverImagePath == nil {
            self.errorMessage = "커버이미지를 선택해주세요."
            self.isShowPopup = true
            return false
        }
        
        if title.trimmingCharacters(in: .whitespaces).isEmpty {
            self.errorMessage = "제목을 입력해 주세요."
            self.isShowPopup = true
            return false
        }
        
        let notice = content.trimmingCharacters(in: .whitespacesAndNewlines) != placeholder ? content : ""
        if notice.isEmpty && notice.count < 5 {
            self.errorMessage = "공지를 5자 이상 입력해주세요."
            self.isShowPopup = true
            return false
        }
        
        guard let numberOfPeople = Int(numberOfPeople), (numberOfPeople >= 3 && numberOfPeople <= 999) else {
            self.errorMessage = "인원을 3~999명 사이로 입력해주세요."
            self.isShowPopup = true
            return false
        }
        
        if roomType == .PRIVATE && (password.trimmingCharacters(in: .whitespaces).isEmpty || password.count != 6) {
            self.errorMessage = "방 입장 비밀번호 6자리를 입력해 주세요."
            self.isShowPopup = true
            return false
        }
        
        return true
    }
}