콘텐츠 메인
- 라이브 다시듣기 탭 UI 페이지 생성
This commit is contained in:
		| @@ -56,6 +56,9 @@ enum ContentApi { | |||||||
|      |      | ||||||
|     case getContentMainAsmr |     case getContentMainAsmr | ||||||
|     case getPopularAsmrContentByCreator(creatorId: Int) |     case getPopularAsmrContentByCreator(creatorId: Int) | ||||||
|  |      | ||||||
|  |     case getContentMainReplay | ||||||
|  |     case getPopularReplayContentByCreator(creatorId: Int) | ||||||
| } | } | ||||||
|  |  | ||||||
| extension ContentApi: TargetType { | extension ContentApi: TargetType { | ||||||
| @@ -190,6 +193,12 @@ extension ContentApi: TargetType { | |||||||
|              |              | ||||||
|         case .getPopularAsmrContentByCreator: |         case .getPopularAsmrContentByCreator: | ||||||
|             return "/v2/audio-content/main/asmr/popular-content-by-creator" |             return "/v2/audio-content/main/asmr/popular-content-by-creator" | ||||||
|  |              | ||||||
|  |         case .getContentMainReplay: | ||||||
|  |             return "/v2/audio-content/main/replay" | ||||||
|  |              | ||||||
|  |         case .getPopularReplayContentByCreator: | ||||||
|  |             return "/v2/audio-content/main/replay/popular-content-by-creator" | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
|      |      | ||||||
| @@ -206,7 +215,7 @@ extension ContentApi: TargetType { | |||||||
|              |              | ||||||
|         case .getContentMainHome, .getPopularContentByCreator, .getContentMainSeries, .getRecommendSeriesListByGenre, .getRecommendSeriesByCreator, .getContentMainContent, |         case .getContentMainHome, .getPopularContentByCreator, .getContentMainSeries, .getRecommendSeriesListByGenre, .getRecommendSeriesByCreator, .getContentMainContent, | ||||||
|                 .getContentMainNewContentOfTheme, .getDailyContentRanking, .getRecommendContentByTag, .getContentMainAlarm, .getContentMainAlarmAll, |                 .getContentMainNewContentOfTheme, .getDailyContentRanking, .getRecommendContentByTag, .getContentMainAlarm, .getContentMainAlarmAll, | ||||||
|                 .getContentMainAsmr, .getPopularAsmrContentByCreator: |                 .getContentMainAsmr, .getPopularAsmrContentByCreator, .getContentMainReplay, .getPopularReplayContentByCreator: | ||||||
|             return .get |             return .get | ||||||
|              |              | ||||||
|         case .likeContent, .modifyAudioContent, .modifyComment, .unpinContent: |         case .likeContent, .modifyAudioContent, .modifyComment, .unpinContent: | ||||||
| @@ -363,7 +372,7 @@ extension ContentApi: TargetType { | |||||||
|              |              | ||||||
|             return .requestParameters(parameters: parameters, encoding: URLEncoding.queryString) |             return .requestParameters(parameters: parameters, encoding: URLEncoding.queryString) | ||||||
|              |              | ||||||
|         case .getContentMainHome, .getContentMainSeries, .getContentMainContent, .getContentMainAlarm, .getContentMainAsmr: |         case .getContentMainHome, .getContentMainSeries, .getContentMainContent, .getContentMainAlarm, .getContentMainAsmr, .getContentMainReplay: | ||||||
|             return .requestPlain |             return .requestPlain | ||||||
|              |              | ||||||
|         case .getRecommendSeriesListByGenre(let genreId): |         case .getRecommendSeriesListByGenre(let genreId): | ||||||
| @@ -403,7 +412,7 @@ extension ContentApi: TargetType { | |||||||
|              |              | ||||||
|             return .requestParameters(parameters: parameters, encoding: URLEncoding.queryString) |             return .requestParameters(parameters: parameters, encoding: URLEncoding.queryString) | ||||||
|              |              | ||||||
|         case .getPopularAsmrContentByCreator(let creatorId): |         case .getPopularAsmrContentByCreator(let creatorId), .getPopularReplayContentByCreator(let creatorId): | ||||||
|             let parameters = ["creatorId": creatorId] |             let parameters = ["creatorId": creatorId] | ||||||
|             return .requestParameters(parameters: parameters, encoding: URLEncoding.queryString) |             return .requestParameters(parameters: parameters, encoding: URLEncoding.queryString) | ||||||
|         } |         } | ||||||
|   | |||||||
| @@ -19,6 +19,6 @@ final class ContentMainTabAsmrRepository { | |||||||
|     } |     } | ||||||
|      |      | ||||||
|     func getPopularContentByCreator(creatorId: Int) -> AnyPublisher<Response, MoyaError> { |     func getPopularContentByCreator(creatorId: Int) -> AnyPublisher<Response, MoyaError> { | ||||||
|         return api.requestPublisher(.getPopularContentByCreator(creatorId: creatorId)) |         return api.requestPublisher(.getPopularAsmrContentByCreator(creatorId: creatorId)) | ||||||
|     } |     } | ||||||
| } | } | ||||||
|   | |||||||
| @@ -67,6 +67,40 @@ final class ContentMainTabAsmrViewModel: ObservableObject { | |||||||
|     } |     } | ||||||
|      |      | ||||||
|     func getPopularContentByCreator(creatorId: Int) { |     func getPopularContentByCreator(creatorId: Int) { | ||||||
|          |         isLoading = true | ||||||
|  |         repository.getPopularContentByCreator(creatorId: creatorId) | ||||||
|  |             .sink { result in | ||||||
|  |                 switch result { | ||||||
|  |                 case .finished: | ||||||
|  |                     DEBUG_LOG("finish") | ||||||
|  |                 case .failure(let error): | ||||||
|  |                     ERROR_LOG(error.localizedDescription) | ||||||
|  |                 } | ||||||
|  |             } receiveValue: { [unowned self] response in | ||||||
|  |                 let responseData = response.data | ||||||
|  |                  | ||||||
|  |                 do { | ||||||
|  |                     let jsonDecoder = JSONDecoder() | ||||||
|  |                     let decoded = try jsonDecoder.decode(ApiResponse<[GetAudioContentRankingItem]>.self, from: responseData) | ||||||
|  |                      | ||||||
|  |                     if let data = decoded.data, decoded.success { | ||||||
|  |                         self.salesCountRankContentList = data | ||||||
|  |                     } else { | ||||||
|  |                         if let message = decoded.message { | ||||||
|  |                             self.errorMessage = message | ||||||
|  |                         } else { | ||||||
|  |                             self.errorMessage = "다시 시도해 주세요.\n계속 같은 문제가 발생할 경우 고객센터로 문의 주시기 바랍니다." | ||||||
|  |                         } | ||||||
|  |                          | ||||||
|  |                         self.isShowPopup = true | ||||||
|  |                     } | ||||||
|  |                 } catch { | ||||||
|  |                     self.errorMessage = "다시 시도해 주세요.\n계속 같은 문제가 발생할 경우 고객센터로 문의 주시기 바랍니다." | ||||||
|  |                     self.isShowPopup = true | ||||||
|  |                 } | ||||||
|  |                  | ||||||
|  |                 self.isLoading = false | ||||||
|  |             } | ||||||
|  |             .store(in: &subscription) | ||||||
|     } |     } | ||||||
| } | } | ||||||
|   | |||||||
| @@ -0,0 +1,24 @@ | |||||||
|  | // | ||||||
|  | //  ContentMainTabReplayRepository.swift | ||||||
|  | //  SodaLive | ||||||
|  | // | ||||||
|  | //  Created by klaus on 2/22/25. | ||||||
|  | // | ||||||
|  |  | ||||||
|  | import Foundation | ||||||
|  | import CombineMoya | ||||||
|  | import Combine | ||||||
|  | import Moya | ||||||
|  |  | ||||||
|  | final class ContentMainTabReplayRepository { | ||||||
|  |      | ||||||
|  |     private let api = MoyaProvider<ContentApi>() | ||||||
|  |      | ||||||
|  |     func getContentMainReplay() -> AnyPublisher<Response, MoyaError> { | ||||||
|  |         return api.requestPublisher(.getContentMainReplay) | ||||||
|  |     } | ||||||
|  |      | ||||||
|  |     func getPopularContentByCreator(creatorId: Int) -> AnyPublisher<Response, MoyaError> { | ||||||
|  |         return api.requestPublisher(.getPopularReplayContentByCreator(creatorId: creatorId)) | ||||||
|  |     } | ||||||
|  | } | ||||||
| @@ -8,8 +8,70 @@ | |||||||
| import SwiftUI | import SwiftUI | ||||||
|  |  | ||||||
| struct ContentMainTabReplayView: View { | struct ContentMainTabReplayView: View { | ||||||
|  |      | ||||||
|  |     @StateObject var viewModel = ContentMainTabReplayViewModel() | ||||||
|  |      | ||||||
|     var body: some View { |     var body: some View { | ||||||
|         Text(/*@START_MENU_TOKEN@*/"Hello, World!"/*@END_MENU_TOKEN@*/) |         BaseView(isLoading: $viewModel.isLoading) { | ||||||
|  |             ScrollView(.vertical, showsIndicators: false) { | ||||||
|  |                 VStack(spacing: 0) { | ||||||
|  |                     if !viewModel.bannerList.isEmpty { | ||||||
|  |                         ContentMainBannerViewV2(bannerList: viewModel.bannerList) | ||||||
|  |                             .padding(.horizontal, 13.3) | ||||||
|  |                     } | ||||||
|  |                      | ||||||
|  |                     if !viewModel.newReplayContentList.isEmpty { | ||||||
|  |                         ContentMainNewContentViewV2( | ||||||
|  |                             title: "새로운 라이브 다시듣기", | ||||||
|  |                             onClickMore: {}, | ||||||
|  |                             themeList: [], | ||||||
|  |                             contentList: viewModel.newReplayContentList | ||||||
|  |                         ) { _ in } | ||||||
|  |                         .padding(.top, 30) | ||||||
|  |                     } | ||||||
|  |                      | ||||||
|  |                     if !viewModel.creatorList.isEmpty { | ||||||
|  |                         ContentByChannelView( | ||||||
|  |                             title: "채널별 라이브 다시듣기", | ||||||
|  |                             creatorList: viewModel.creatorList, | ||||||
|  |                             contentList: viewModel.salesCountRankContentList, | ||||||
|  |                             onClickCreator: { | ||||||
|  |                                 viewModel.getPopularContentByCreator(creatorId: $0) | ||||||
|  |                             } | ||||||
|  |                         ) | ||||||
|  |                         .padding(.top, 30) | ||||||
|  |                     } | ||||||
|  |                      | ||||||
|  |                     if !viewModel.eventBannerList.isEmpty { | ||||||
|  |                         SectionEventBannerView(items: viewModel.eventBannerList) | ||||||
|  |                             .padding(.top, 30) | ||||||
|  |                     } | ||||||
|  |                      | ||||||
|  |                     if !viewModel.curationList.isEmpty { | ||||||
|  |                         ContentMainCurationViewV2(curationList: viewModel.curationList) | ||||||
|  |                         .padding(.top, 30) | ||||||
|  |                     } | ||||||
|  |                 } | ||||||
|  |                 .onAppear { | ||||||
|  |                     viewModel.fetchData() | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |         .popup(isPresented: $viewModel.isShowPopup, type: .toast, position: .bottom, autohideIn: 2) { | ||||||
|  |             HStack { | ||||||
|  |                 Spacer() | ||||||
|  |                 Text(viewModel.errorMessage) | ||||||
|  |                     .padding(.vertical, 13.3) | ||||||
|  |                     .frame(width: screenSize().width - 66.7, alignment: .center) | ||||||
|  |                     .font(.custom(Font.medium.rawValue, size: 12)) | ||||||
|  |                     .background(Color.button) | ||||||
|  |                     .foregroundColor(Color.white) | ||||||
|  |                     .multilineTextAlignment(.leading) | ||||||
|  |                     .cornerRadius(20) | ||||||
|  |                     .padding(.bottom, 66.7) | ||||||
|  |                 Spacer() | ||||||
|  |             } | ||||||
|  |         } | ||||||
|     } |     } | ||||||
| } | } | ||||||
|  |  | ||||||
|   | |||||||
| @@ -6,9 +6,101 @@ | |||||||
| // | // | ||||||
|  |  | ||||||
| import Foundation | import Foundation | ||||||
|  | import Combine | ||||||
|  |  | ||||||
| final class ContentMainTabReplayViewModel: ObservableObject { | final class ContentMainTabReplayViewModel: ObservableObject { | ||||||
|  |     private let repository = ContentMainTabReplayRepository() | ||||||
|  |     private var subscription = Set<AnyCancellable>() | ||||||
|  |      | ||||||
|     @Published var errorMessage = "" |     @Published var errorMessage = "" | ||||||
|     @Published var isShowPopup = false |     @Published var isShowPopup = false | ||||||
|     @Published var isLoading = false |     @Published var isLoading = false | ||||||
|  |      | ||||||
|  |     @Published var bannerList: [GetAudioContentBannerResponse] = [] | ||||||
|  |     @Published var newReplayContentList: [GetAudioContentMainItem] = [] | ||||||
|  |     @Published var creatorList: [ContentCreatorResponse] = [] | ||||||
|  |     @Published var salesCountRankContentList: [GetAudioContentRankingItem] = [] | ||||||
|  |     @Published var eventBannerList: [EventItem] = [] | ||||||
|  |     @Published var curationList: [GetContentCurationResponse] = [] | ||||||
|  |      | ||||||
|  |     func fetchData() { | ||||||
|  |         isLoading = true | ||||||
|  |         repository.getContentMainReplay() | ||||||
|  |             .sink { result in | ||||||
|  |                 switch result { | ||||||
|  |                 case .finished: | ||||||
|  |                     DEBUG_LOG("finish") | ||||||
|  |                 case .failure(let error): | ||||||
|  |                     ERROR_LOG(error.localizedDescription) | ||||||
|  |                 } | ||||||
|  |             } receiveValue: { [unowned self] response in | ||||||
|  |                 let responseData = response.data | ||||||
|  |                  | ||||||
|  |                 do { | ||||||
|  |                     let jsonDecoder = JSONDecoder() | ||||||
|  |                     let decoded = try jsonDecoder.decode(ApiResponse<GetContentMainTabReplayResponse>.self, from: responseData) | ||||||
|  |                      | ||||||
|  |                     if let data = decoded.data, decoded.success { | ||||||
|  |                         self.bannerList = data.contentBannerList | ||||||
|  |                         self.newReplayContentList = data.newLiveReplayContentList | ||||||
|  |                         self.creatorList = data.creatorList | ||||||
|  |                         self.salesCountRankContentList = data.salesCountRankContentList | ||||||
|  |                         self.eventBannerList = data.eventBannerList.eventList | ||||||
|  |                         self.curationList = data.curationList | ||||||
|  |                     } else { | ||||||
|  |                         if let message = decoded.message { | ||||||
|  |                             self.errorMessage = message | ||||||
|  |                         } else { | ||||||
|  |                             self.errorMessage = "다시 시도해 주세요.\n계속 같은 문제가 발생할 경우 고객센터로 문의 주시기 바랍니다." | ||||||
|  |                         } | ||||||
|  |                          | ||||||
|  |                         self.isShowPopup = true | ||||||
|  |                     } | ||||||
|  |                 } catch { | ||||||
|  |                     self.errorMessage = "다시 시도해 주세요.\n계속 같은 문제가 발생할 경우 고객센터로 문의 주시기 바랍니다." | ||||||
|  |                     self.isShowPopup = true | ||||||
|  |                 } | ||||||
|  |                  | ||||||
|  |                 self.isLoading = false | ||||||
|  |             } | ||||||
|  |             .store(in: &subscription) | ||||||
|  |     } | ||||||
|  |      | ||||||
|  |     func getPopularContentByCreator(creatorId: Int) { | ||||||
|  |         isLoading = true | ||||||
|  |         repository.getPopularContentByCreator(creatorId: creatorId) | ||||||
|  |             .sink { result in | ||||||
|  |                 switch result { | ||||||
|  |                 case .finished: | ||||||
|  |                     DEBUG_LOG("finish") | ||||||
|  |                 case .failure(let error): | ||||||
|  |                     ERROR_LOG(error.localizedDescription) | ||||||
|  |                 } | ||||||
|  |             } receiveValue: { [unowned self] response in | ||||||
|  |                 let responseData = response.data | ||||||
|  |                  | ||||||
|  |                 do { | ||||||
|  |                     let jsonDecoder = JSONDecoder() | ||||||
|  |                     let decoded = try jsonDecoder.decode(ApiResponse<[GetAudioContentRankingItem]>.self, from: responseData) | ||||||
|  |                      | ||||||
|  |                     if let data = decoded.data, decoded.success { | ||||||
|  |                         self.salesCountRankContentList = data | ||||||
|  |                     } else { | ||||||
|  |                         if let message = decoded.message { | ||||||
|  |                             self.errorMessage = message | ||||||
|  |                         } else { | ||||||
|  |                             self.errorMessage = "다시 시도해 주세요.\n계속 같은 문제가 발생할 경우 고객센터로 문의 주시기 바랍니다." | ||||||
|  |                         } | ||||||
|  |                          | ||||||
|  |                         self.isShowPopup = true | ||||||
|  |                     } | ||||||
|  |                 } catch { | ||||||
|  |                     self.errorMessage = "다시 시도해 주세요.\n계속 같은 문제가 발생할 경우 고객센터로 문의 주시기 바랍니다." | ||||||
|  |                     self.isShowPopup = true | ||||||
|  |                 } | ||||||
|  |                  | ||||||
|  |                 self.isLoading = false | ||||||
|  |             } | ||||||
|  |             .store(in: &subscription) | ||||||
|  |     } | ||||||
| } | } | ||||||
|   | |||||||
| @@ -0,0 +1,15 @@ | |||||||
|  | // | ||||||
|  | //  GetContentMainTabReplayResponse.swift | ||||||
|  | //  SodaLive | ||||||
|  | // | ||||||
|  | //  Created by klaus on 2/22/25. | ||||||
|  | // | ||||||
|  |  | ||||||
|  | struct GetContentMainTabReplayResponse: Decodable { | ||||||
|  |     let contentBannerList: [GetAudioContentBannerResponse] | ||||||
|  |     let newLiveReplayContentList: [GetAudioContentMainItem] | ||||||
|  |     let creatorList: [ContentCreatorResponse] | ||||||
|  |     let salesCountRankContentList: [GetAudioContentRankingItem] | ||||||
|  |     let eventBannerList: GetEventResponse | ||||||
|  |     let curationList: [GetContentCurationResponse] | ||||||
|  | } | ||||||
		Reference in New Issue
	
	Block a user
	 Yu Sung
					Yu Sung