콘텐츠 메인
- ASMR 탭 UI 페이지 생성
This commit is contained in:
		| @@ -53,6 +53,9 @@ enum ContentApi { | |||||||
|      |      | ||||||
|     case getContentMainAlarm |     case getContentMainAlarm | ||||||
|     case getContentMainAlarmAll(theme: String, page: Int, size: Int) |     case getContentMainAlarmAll(theme: String, page: Int, size: Int) | ||||||
|  |      | ||||||
|  |     case getContentMainAsmr | ||||||
|  |     case getPopularAsmrContentByCreator(creatorId: Int) | ||||||
| } | } | ||||||
|  |  | ||||||
| extension ContentApi: TargetType { | extension ContentApi: TargetType { | ||||||
| @@ -181,6 +184,12 @@ extension ContentApi: TargetType { | |||||||
|              |              | ||||||
|         case .getContentMainAlarmAll: |         case .getContentMainAlarmAll: | ||||||
|             return "/v2/audio-content/main/alarm/all" |             return "/v2/audio-content/main/alarm/all" | ||||||
|  |              | ||||||
|  |         case .getContentMainAsmr: | ||||||
|  |             return "/v2/audio-content/main/asmr" | ||||||
|  |              | ||||||
|  |         case .getPopularAsmrContentByCreator: | ||||||
|  |             return "/v2/audio-content/main/asmr/popular-content-by-creator" | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
|      |      | ||||||
| @@ -196,7 +205,8 @@ extension ContentApi: TargetType { | |||||||
|             return .get |             return .get | ||||||
|              |              | ||||||
|         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: | ||||||
|             return .get |             return .get | ||||||
|              |              | ||||||
|         case .likeContent, .modifyAudioContent, .modifyComment, .unpinContent: |         case .likeContent, .modifyAudioContent, .modifyComment, .unpinContent: | ||||||
| @@ -353,7 +363,7 @@ extension ContentApi: TargetType { | |||||||
|              |              | ||||||
|             return .requestParameters(parameters: parameters, encoding: URLEncoding.queryString) |             return .requestParameters(parameters: parameters, encoding: URLEncoding.queryString) | ||||||
|              |              | ||||||
|         case .getContentMainHome, .getContentMainSeries, .getContentMainContent, .getContentMainAlarm: |         case .getContentMainHome, .getContentMainSeries, .getContentMainContent, .getContentMainAlarm, .getContentMainAsmr: | ||||||
|             return .requestPlain |             return .requestPlain | ||||||
|              |              | ||||||
|         case .getRecommendSeriesListByGenre(let genreId): |         case .getRecommendSeriesListByGenre(let genreId): | ||||||
| @@ -392,6 +402,10 @@ extension ContentApi: TargetType { | |||||||
|             ] as [String : Any] |             ] as [String : Any] | ||||||
|              |              | ||||||
|             return .requestParameters(parameters: parameters, encoding: URLEncoding.queryString) |             return .requestParameters(parameters: parameters, encoding: URLEncoding.queryString) | ||||||
|  |              | ||||||
|  |         case .getPopularAsmrContentByCreator(let creatorId): | ||||||
|  |             let parameters = ["creatorId": creatorId] | ||||||
|  |             return .requestParameters(parameters: parameters, encoding: URLEncoding.queryString) | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
|      |      | ||||||
|   | |||||||
| @@ -0,0 +1,24 @@ | |||||||
|  | // | ||||||
|  | //  ContentMainTabAsmrRepository.swift | ||||||
|  | //  SodaLive | ||||||
|  | // | ||||||
|  | //  Created by klaus on 2/22/25. | ||||||
|  | // | ||||||
|  |  | ||||||
|  | import Foundation | ||||||
|  | import CombineMoya | ||||||
|  | import Combine | ||||||
|  | import Moya | ||||||
|  |  | ||||||
|  | final class ContentMainTabAsmrRepository { | ||||||
|  |      | ||||||
|  |     private let api = MoyaProvider<ContentApi>() | ||||||
|  |      | ||||||
|  |     func getContentMainAsmr() -> AnyPublisher<Response, MoyaError> { | ||||||
|  |         return api.requestPublisher(.getContentMainAsmr) | ||||||
|  |     } | ||||||
|  |      | ||||||
|  |     func getPopularContentByCreator(creatorId: Int) -> AnyPublisher<Response, MoyaError> { | ||||||
|  |         return api.requestPublisher(.getPopularContentByCreator(creatorId: creatorId)) | ||||||
|  |     } | ||||||
|  | } | ||||||
| @@ -8,8 +8,70 @@ | |||||||
| import SwiftUI | import SwiftUI | ||||||
|  |  | ||||||
| struct ContentMainTabAsmrView: View { | struct ContentMainTabAsmrView: View { | ||||||
|  |      | ||||||
|  |     @StateObject var viewModel = ContentMainTabAsmrViewModel() | ||||||
|  |      | ||||||
|     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.newAsmrContentList.isEmpty { | ||||||
|  |                         ContentMainNewContentViewV2( | ||||||
|  |                             title: "새로운 ASMR", | ||||||
|  |                             onClickMore: {}, | ||||||
|  |                             themeList: [], | ||||||
|  |                             contentList: viewModel.newAsmrContentList | ||||||
|  |                         ) { _ in } | ||||||
|  |                         .padding(.top, 30) | ||||||
|  |                     } | ||||||
|  |                      | ||||||
|  |                     if !viewModel.creatorList.isEmpty { | ||||||
|  |                         ContentByChannelView( | ||||||
|  |                             title: "채널별 추천 ASMR", | ||||||
|  |                             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,67 @@ | |||||||
| // | // | ||||||
|  |  | ||||||
| import Foundation | import Foundation | ||||||
|  | import Combine | ||||||
|  |  | ||||||
| final class ContentMainTabAsmrViewModel: ObservableObject { | final class ContentMainTabAsmrViewModel: ObservableObject { | ||||||
|  |     private let repository = ContentMainTabAsmrRepository() | ||||||
|  |     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 newAsmrContentList: [GetAudioContentMainItem] = [] | ||||||
|  |     @Published var creatorList: [ContentCreatorResponse] = [] | ||||||
|  |     @Published var salesCountRankContentList: [GetAudioContentRankingItem] = [] | ||||||
|  |     @Published var eventBannerList: [EventItem] = [] | ||||||
|  |     @Published var curationList: [GetContentCurationResponse] = [] | ||||||
|  |      | ||||||
|  |     func fetchData() { | ||||||
|  |         isLoading = true | ||||||
|  |         repository.getContentMainAsmr() | ||||||
|  |             .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<GetContentMainTabAsmrResponse>.self, from: responseData) | ||||||
|  |                      | ||||||
|  |                     if let data = decoded.data, decoded.success { | ||||||
|  |                         self.bannerList = data.contentBannerList | ||||||
|  |                         self.newAsmrContentList = data.newAsmrContentList | ||||||
|  |                         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) { | ||||||
|  |          | ||||||
|  |     } | ||||||
| } | } | ||||||
|   | |||||||
| @@ -0,0 +1,15 @@ | |||||||
|  | // | ||||||
|  | //  GetContentMainTabAsmrResponse.swift | ||||||
|  | //  SodaLive | ||||||
|  | // | ||||||
|  | //  Created by klaus on 2/22/25. | ||||||
|  | // | ||||||
|  |  | ||||||
|  | struct GetContentMainTabAsmrResponse: Decodable { | ||||||
|  |     let contentBannerList: [GetAudioContentBannerResponse] | ||||||
|  |     let newAsmrContentList: [GetAudioContentMainItem] | ||||||
|  |     let creatorList: [ContentCreatorResponse] | ||||||
|  |     let salesCountRankContentList: [GetAudioContentRankingItem] | ||||||
|  |     let eventBannerList: GetEventResponse | ||||||
|  |     let curationList: [GetContentCurationResponse] | ||||||
|  | } | ||||||
| @@ -22,23 +22,24 @@ struct ContentMainNewContentViewV2: View { | |||||||
|             HStack(spacing: 0) { |             HStack(spacing: 0) { | ||||||
|                 Text(title) |                 Text(title) | ||||||
|                     .font(.custom(Font.bold.rawValue, size: 18.3)) |                     .font(.custom(Font.bold.rawValue, size: 18.3)) | ||||||
|                     .foregroundColor(Color(hex: "eeeeee")) |                     .foregroundColor(.grayee) | ||||||
|                  |                  | ||||||
|                 Spacer() |                 Spacer() | ||||||
|                  |                  | ||||||
|                 Image("ic_forward") |                 Image("ic_forward") | ||||||
|                     .resizable() |                     .resizable() | ||||||
|                     .frame(width: 20, height: 20) |                     .frame(width: 20, height: 20) | ||||||
|                     .onTapGesture { |                     .onTapGesture { onClickMore() } | ||||||
|                     } |  | ||||||
|             } |             } | ||||||
|             .padding(.horizontal, 13.3) |             .padding(.horizontal, 13.3) | ||||||
|              |              | ||||||
|  |             if !themeList.isEmpty { | ||||||
|                 ContentMainContentThemeView( |                 ContentMainContentThemeView( | ||||||
|                     themeList: themeList, |                     themeList: themeList, | ||||||
|                     selectTheme: selectTheme, |                     selectTheme: selectTheme, | ||||||
|                     selectedTheme: $selectedTheme |                     selectedTheme: $selectedTheme | ||||||
|                 ) |                 ) | ||||||
|  |             } | ||||||
|              |              | ||||||
|             ScrollView(.horizontal, showsIndicators: false) { |             ScrollView(.horizontal, showsIndicators: false) { | ||||||
|                 HStack(spacing: 13.3) { |                 HStack(spacing: 13.3) { | ||||||
| @@ -50,9 +51,11 @@ struct ContentMainNewContentViewV2: View { | |||||||
|             } |             } | ||||||
|         } |         } | ||||||
|         .onAppear { |         .onAppear { | ||||||
|  |             if !themeList.isEmpty { | ||||||
|                 selectedTheme = themeList[0] |                 selectedTheme = themeList[0] | ||||||
|             } |             } | ||||||
|         } |         } | ||||||
|  |     } | ||||||
| } | } | ||||||
|  |  | ||||||
| #Preview { | #Preview { | ||||||
|   | |||||||
		Reference in New Issue
	
	Block a user
	 Yu Sung
					Yu Sung