diff --git a/Projects/App/Sources/Application/AppComponent+Artist.swift b/Projects/App/Sources/Application/AppComponent+Artist.swift index c6ff090c8..1c7c96532 100644 --- a/Projects/App/Sources/Application/AppComponent+Artist.swift +++ b/Projects/App/Sources/Application/AppComponent+Artist.swift @@ -56,6 +56,12 @@ public extension AppComponent { } } + var findArtistIDUseCase: any FindArtistIDUseCase { + shared { + FindArtistIDUseCaseImpl(artistRepository: artistRepository) + } + } + // MARK: Artist Detail > Artist Music var artistMusicComponent: ArtistMusicComponent { ArtistMusicComponent(parent: self) diff --git a/Projects/Domains/ArtistDomain/Interface/DataSource/RemoteArtistDataSource.swift b/Projects/Domains/ArtistDomain/Interface/DataSource/RemoteArtistDataSource.swift index 13b9d9204..b1bb8fc5a 100644 --- a/Projects/Domains/ArtistDomain/Interface/DataSource/RemoteArtistDataSource.swift +++ b/Projects/Domains/ArtistDomain/Interface/DataSource/RemoteArtistDataSource.swift @@ -7,4 +7,5 @@ public protocol RemoteArtistDataSource { func fetchArtistSongList(id: String, sort: ArtistSongSortType, page: Int) -> Single<[ArtistSongListEntity]> func fetchArtistSubscriptionStatus(id: String) -> Single func subscriptionArtist(id: String, on: Bool) -> Completable + func findArtistID(name: String) -> Single } diff --git a/Projects/Domains/ArtistDomain/Interface/Repository/ArtistRepository.swift b/Projects/Domains/ArtistDomain/Interface/Repository/ArtistRepository.swift index 45d84abd2..d3aca2a5c 100644 --- a/Projects/Domains/ArtistDomain/Interface/Repository/ArtistRepository.swift +++ b/Projects/Domains/ArtistDomain/Interface/Repository/ArtistRepository.swift @@ -7,4 +7,5 @@ public protocol ArtistRepository { func fetchArtistSongList(id: String, sort: ArtistSongSortType, page: Int) -> Single<[ArtistSongListEntity]> func fetchArtistSubscriptionStatus(id: String) -> Single func subscriptionArtist(id: String, on: Bool) -> Completable + func findArtistID(name: String) -> Single } diff --git a/Projects/Domains/ArtistDomain/Interface/UseCase/FindArtistIDUseCase.swift b/Projects/Domains/ArtistDomain/Interface/UseCase/FindArtistIDUseCase.swift new file mode 100644 index 000000000..b2ce2a44e --- /dev/null +++ b/Projects/Domains/ArtistDomain/Interface/UseCase/FindArtistIDUseCase.swift @@ -0,0 +1,5 @@ +import RxSwift + +public protocol FindArtistIDUseCase { + func execute(name: String) -> Single +} diff --git a/Projects/Domains/ArtistDomain/Sources/API/ArtistAPI.swift b/Projects/Domains/ArtistDomain/Sources/API/ArtistAPI.swift index 7ce3569a1..4f0b9b52b 100644 --- a/Projects/Domains/ArtistDomain/Sources/API/ArtistAPI.swift +++ b/Projects/Domains/ArtistDomain/Sources/API/ArtistAPI.swift @@ -10,6 +10,7 @@ public enum ArtistAPI { case fetchArtistSongList(id: String, sort: ArtistSongSortType, page: Int) case fetchSubscriptionStatus(id: String) case subscriptionArtist(id: String, on: Bool) + case findArtistID(name: String) } extension ArtistAPI: WMAPI { @@ -29,6 +30,8 @@ extension ArtistAPI: WMAPI { return "/\(id)/subscription" case let .subscriptionArtist(id, _): return "/\(id)/subscription" + case let .findArtistID(name): + return "/find" } } @@ -37,7 +40,8 @@ extension ArtistAPI: WMAPI { case .fetchArtistList, .fetchArtistDetail, .fetchArtistSongList, - .fetchSubscriptionStatus: + .fetchSubscriptionStatus, + .findArtistID: return .get case let .subscriptionArtist(_, on): return on ? .post : .delete @@ -59,12 +63,19 @@ extension ArtistAPI: WMAPI { ], encoding: URLEncoding.queryString ) + case let .findArtistID(name): + return .requestParameters( + parameters: [ + "name": name + ], + encoding: URLEncoding.queryString + ) } } public var jwtTokenType: JwtTokenType { switch self { - case .fetchArtistList, .fetchArtistDetail, .fetchArtistSongList: + case .fetchArtistList, .fetchArtistDetail, .fetchArtistSongList, .findArtistID: return .none case .fetchSubscriptionStatus, .subscriptionArtist: return .accessToken diff --git a/Projects/Domains/ArtistDomain/Sources/DataSource/RemoteArtistDataSourceImpl.swift b/Projects/Domains/ArtistDomain/Sources/DataSource/RemoteArtistDataSourceImpl.swift index 36c18ffe4..1d0f65c6f 100644 --- a/Projects/Domains/ArtistDomain/Sources/DataSource/RemoteArtistDataSourceImpl.swift +++ b/Projects/Domains/ArtistDomain/Sources/DataSource/RemoteArtistDataSourceImpl.swift @@ -32,4 +32,10 @@ public final class RemoteArtistDataSourceImpl: BaseRemoteDataSource, request(.subscriptionArtist(id: id, on: on)) .asCompletable() } + + public func findArtistID(name: String) -> Single { + request(.findArtistID(name: name)) + .map(FindArtistIDResponseDTO.self) + .map { $0.id } + } } diff --git a/Projects/Domains/ArtistDomain/Sources/Repository/ArtistRepositoryImpl.swift b/Projects/Domains/ArtistDomain/Sources/Repository/ArtistRepositoryImpl.swift index 80b82c2ce..a0d9ec2d6 100644 --- a/Projects/Domains/ArtistDomain/Sources/Repository/ArtistRepositoryImpl.swift +++ b/Projects/Domains/ArtistDomain/Sources/Repository/ArtistRepositoryImpl.swift @@ -29,4 +29,8 @@ public final class ArtistRepositoryImpl: ArtistRepository { public func subscriptionArtist(id: String, on: Bool) -> Completable { remoteArtistDataSource.subscriptionArtist(id: id, on: on) } + + public func findArtistID(name: String) -> Single { + remoteArtistDataSource.findArtistID(name: name) + } } diff --git a/Projects/Domains/ArtistDomain/Sources/ResponseDTO/FindArtistIDResponseDTO.swift b/Projects/Domains/ArtistDomain/Sources/ResponseDTO/FindArtistIDResponseDTO.swift new file mode 100644 index 000000000..8657cf15e --- /dev/null +++ b/Projects/Domains/ArtistDomain/Sources/ResponseDTO/FindArtistIDResponseDTO.swift @@ -0,0 +1,5 @@ +import Foundation + +struct FindArtistIDResponseDTO: Decodable { + let id: String +} diff --git a/Projects/Domains/ArtistDomain/Sources/UseCase/FindArtistIDUseCaseImpl.swift b/Projects/Domains/ArtistDomain/Sources/UseCase/FindArtistIDUseCaseImpl.swift new file mode 100644 index 000000000..163385f45 --- /dev/null +++ b/Projects/Domains/ArtistDomain/Sources/UseCase/FindArtistIDUseCaseImpl.swift @@ -0,0 +1,16 @@ +import ArtistDomainInterface +import RxSwift + +public struct FindArtistIDUseCaseImpl: FindArtistIDUseCase { + private let artistRepository: any ArtistRepository + + public init( + artistRepository: any ArtistRepository + ) { + self.artistRepository = artistRepository + } + + public func execute(name: String) -> Single { + return artistRepository.findArtistID(name: name) + } +} diff --git a/Projects/Domains/ArtistDomain/Testing/UseCase/FindArtistIDUseCaseSpy.swift b/Projects/Domains/ArtistDomain/Testing/UseCase/FindArtistIDUseCaseSpy.swift new file mode 100644 index 000000000..7a37e5016 --- /dev/null +++ b/Projects/Domains/ArtistDomain/Testing/UseCase/FindArtistIDUseCaseSpy.swift @@ -0,0 +1,11 @@ +import ArtistDomainInterface +import RxSwift + +public final class FindArtistIDUseCaseSpy: FindArtistIDUseCase { + public private(set) var callCount = 0 + public var handler: ((String) -> Single) = { _ in .never() } + public func execute(name: String) -> Single { + callCount += 1 + return handler(name) + } +} diff --git a/Projects/Features/MusicDetailFeature/Demo/Sources/AppDelegate.swift b/Projects/Features/MusicDetailFeature/Demo/Sources/AppDelegate.swift index 944f0a847..ebbcf1b78 100644 --- a/Projects/Features/MusicDetailFeature/Demo/Sources/AppDelegate.swift +++ b/Projects/Features/MusicDetailFeature/Demo/Sources/AppDelegate.swift @@ -1,3 +1,5 @@ +@testable import ArtistDomainTesting +import ArtistFeatureInterface import BaseFeature import BaseFeatureInterface import Inject @@ -48,6 +50,11 @@ final class AppDelegate: UIResponder, UIApplicationDelegate { .just(.init(status: "", likes: 0)) } + let findArtistIDUseCase = FindArtistIDUseCaseSpy() + findArtistIDUseCase.handler = { _ in + .just("fgSXAKsq-Vo") + } + let reactor = MusicDetailReactor( songIDs: [ "fgSXAKsq-Vo", @@ -58,7 +65,8 @@ final class AppDelegate: UIResponder, UIApplicationDelegate { selectedID: "DPEtmqvaKqY", fetchSongUseCase: fetchSongUseCase, addLikeSongUseCase: addLikeSongUseCase, - cancelLikeSongUseCase: cancelLikeSongUseCase + cancelLikeSongUseCase: cancelLikeSongUseCase, + findArtistIDUseCase: findArtistIDUseCase ) let viewController = Inject.ViewControllerHost( UINavigationController( @@ -70,6 +78,7 @@ final class AppDelegate: UIResponder, UIApplicationDelegate { containSongsFactory: DummyContainSongsFactory(), textPopupFactory: DummyTextPopupFactory(), karaokeFactory: DummyKaraokeFactory(), + artistDetailFactory: DummyArtistDetailFactory(), playlistPresenterGlobalState: DummyPlaylistPresenterGlobalState() ) ) @@ -129,3 +138,9 @@ final class DummyKaraokeFactory: KaraokeFactory { UIViewController() } } + +final class DummyArtistDetailFactory: ArtistDetailFactory { + func makeView(artistID: String) -> UIViewController { + UIViewController() + } +} diff --git a/Projects/Features/MusicDetailFeature/Project.swift b/Projects/Features/MusicDetailFeature/Project.swift index 38a0896e7..6f5af8b8e 100644 --- a/Projects/Features/MusicDetailFeature/Project.swift +++ b/Projects/Features/MusicDetailFeature/Project.swift @@ -12,18 +12,22 @@ let project = Project.module( .feature(target: .LyricHighlightingFeature, type: .interface), .feature(target: .SongCreditFeature, type: .interface), .feature(target: .SignInFeature, type: .interface), + .feature(target: .ArtistFeature, type: .interface), .domain(target: .SongsDomain, type: .interface), - .domain(target: .LikeDomain, type: .interface) + .domain(target: .LikeDomain, type: .interface), + .domain(target: .ArtistDomain, type: .interface) ]), .tests(module: .feature(.MusicDetailFeature), dependencies: [ .feature(target: .MusicDetailFeature), .domain(target: .SongsDomain, type: .testing), - .domain(target: .LikeDomain, type: .testing) + .domain(target: .LikeDomain, type: .testing), + .domain(target: .ArtistDomain, type: .testing) ]), .demo(module: .feature(.MusicDetailFeature), dependencies: [ .feature(target: .MusicDetailFeature), .domain(target: .SongsDomain, type: .testing), - .domain(target: .LikeDomain, type: .testing) + .domain(target: .LikeDomain, type: .testing), + .domain(target: .ArtistDomain, type: .testing) ]) ] ) diff --git a/Projects/Features/MusicDetailFeature/Sources/MusicDetail/Component/MusicDetailComponent.swift b/Projects/Features/MusicDetailFeature/Sources/MusicDetail/Component/MusicDetailComponent.swift index a7fce96d3..55bfb6210 100644 --- a/Projects/Features/MusicDetailFeature/Sources/MusicDetail/Component/MusicDetailComponent.swift +++ b/Projects/Features/MusicDetailFeature/Sources/MusicDetail/Component/MusicDetailComponent.swift @@ -1,3 +1,5 @@ +import ArtistDomainInterface +import ArtistFeatureInterface import BaseFeature import BaseFeatureInterface import LikeDomainInterface @@ -17,9 +19,11 @@ public protocol MusicDetailDependency: Dependency { var containSongsFactory: any ContainSongsFactory { get } var karaokeFactory: any KaraokeFactory { get } var textPopupFactory: any TextPopupFactory { get } + var artistDetailFactory: any ArtistDetailFactory { get } var playlistPresenterGlobalState: any PlayListPresenterGlobalStateProtocol { get } var addLikeSongUseCase: any AddLikeSongUseCase { get } var cancelLikeSongUseCase: any CancelLikeSongUseCase { get } + var findArtistIDUseCase: any FindArtistIDUseCase { get } } public final class MusicDetailComponent: Component, MusicDetailFactory { @@ -29,7 +33,8 @@ public final class MusicDetailComponent: Component, Music selectedID: selectedID, fetchSongUseCase: dependency.fetchSongUseCase, addLikeSongUseCase: dependency.addLikeSongUseCase, - cancelLikeSongUseCase: dependency.cancelLikeSongUseCase + cancelLikeSongUseCase: dependency.cancelLikeSongUseCase, + findArtistIDUseCase: dependency.findArtistIDUseCase ) let viewController = MusicDetailViewController( @@ -40,6 +45,7 @@ public final class MusicDetailComponent: Component, Music containSongsFactory: dependency.containSongsFactory, textPopupFactory: dependency.textPopupFactory, karaokeFactory: dependency.karaokeFactory, + artistDetailFactory: dependency.artistDetailFactory, playlistPresenterGlobalState: dependency.playlistPresenterGlobalState ) diff --git a/Projects/Features/MusicDetailFeature/Sources/MusicDetail/MusicDetailReactor.swift b/Projects/Features/MusicDetailFeature/Sources/MusicDetail/MusicDetailReactor.swift index 9bda66e2e..8b3f9bc6e 100644 --- a/Projects/Features/MusicDetailFeature/Sources/MusicDetail/MusicDetailReactor.swift +++ b/Projects/Features/MusicDetailFeature/Sources/MusicDetail/MusicDetailReactor.swift @@ -1,3 +1,4 @@ +import ArtistDomainInterface import BaseFeature import Foundation import Kingfisher @@ -30,6 +31,7 @@ final class MusicDetailReactor: Reactor { case musicPickButtonDidTap case playListButtonDidTap case dismissButtonDidTap + case didTapArtistLabel } enum Mutation { @@ -48,6 +50,7 @@ final class MusicDetailReactor: Reactor { case textPopup(text: String, completion: () -> Void) case signin case karaoke(ky: Int?, tj: Int?) + case artist(artistID: String) } struct State { @@ -71,6 +74,7 @@ final class MusicDetailReactor: Reactor { private let fetchSongUseCase: any FetchSongUseCase private let addLikeSongUseCase: any AddLikeSongUseCase private let cancelLikeSongUseCase: any CancelLikeSongUseCase + private let findArtistIDUseCase: any FindArtistIDUseCase private var shouldRefreshLikeList = false private var pendingLikeRequests: [String: LikeRequest] = [:] @@ -83,7 +87,8 @@ final class MusicDetailReactor: Reactor { selectedID: String, fetchSongUseCase: any FetchSongUseCase, addLikeSongUseCase: any AddLikeSongUseCase, - cancelLikeSongUseCase: any CancelLikeSongUseCase + cancelLikeSongUseCase: any CancelLikeSongUseCase, + findArtistIDUseCase: any FindArtistIDUseCase ) { let selectedIndex = songIDs.firstIndex(of: selectedID) ?? 0 self.initialState = .init( @@ -94,6 +99,7 @@ final class MusicDetailReactor: Reactor { self.fetchSongUseCase = fetchSongUseCase self.addLikeSongUseCase = addLikeSongUseCase self.cancelLikeSongUseCase = cancelLikeSongUseCase + self.findArtistIDUseCase = findArtistIDUseCase let urls = [ songIDs[safe: selectedIndex - 1], @@ -133,6 +139,8 @@ final class MusicDetailReactor: Reactor { return playListButtonDidTap() case .dismissButtonDidTap: return navigateMutation(navigate: .dismiss) + case .didTapArtistLabel: + return didTapArtistLabel() } } @@ -416,6 +424,25 @@ private extension MusicDetailReactor { LogManager.analytics(log) return navigateMutation(navigate: .playlist(id: song.videoID)) } + + func didTapArtistLabel() -> Observable { + guard let selectedSong = currentState.selectedSong else { return .empty() } + let artists = selectedSong.artistString.components(separatedBy: ",") + + if artists.count == 1, let artistName = artists.first { + return findArtistIDUseCase.execute(name: artistName) + .asObservable() + .flatMap { + return Observable.concat( + .just(Mutation.navigate(.artist(artistID: $0))), + .just(.navigate(nil)) + ) + } + .catchAndReturn(.navigate(.credit(id: selectedSong.videoID))) + } else { + return navigateMutation(navigate: .credit(id: selectedSong.videoID)) + } + } } // MARK: - Private Methods diff --git a/Projects/Features/MusicDetailFeature/Sources/MusicDetail/MusicDetailView.swift b/Projects/Features/MusicDetailFeature/Sources/MusicDetail/MusicDetailView.swift index f995728e2..1441a4af1 100644 --- a/Projects/Features/MusicDetailFeature/Sources/MusicDetail/MusicDetailView.swift +++ b/Projects/Features/MusicDetailFeature/Sources/MusicDetail/MusicDetailView.swift @@ -32,6 +32,7 @@ private protocol MusicDetailActionProtocol { var playlistButtonDidTap: Observable { get } var creditButtonDidTap: Observable { get } var dismissButtonDidTap: Observable { get } + var didTapArtistLabel: Observable { get } } final class MusicDetailView: UIView { @@ -256,4 +257,5 @@ extension Reactive: MusicDetailActionProtocol where Base: MusicDetailView { var playlistButtonDidTap: Observable { base.musicToolbarView.rx.playlistButtonDidTap } var creditButtonDidTap: Observable { base.creditButton.rx.tap.asObservable() } var dismissButtonDidTap: Observable { base.dismissButton.rx.tap.asObservable() } + var didTapArtistLabel: Observable { base.musicControlView.rx.didTapArtistLabel } } diff --git a/Projects/Features/MusicDetailFeature/Sources/MusicDetail/MusicDetailViewController.swift b/Projects/Features/MusicDetailFeature/Sources/MusicDetail/MusicDetailViewController.swift index 7cefcb4c3..3fe2d8490 100644 --- a/Projects/Features/MusicDetailFeature/Sources/MusicDetail/MusicDetailViewController.swift +++ b/Projects/Features/MusicDetailFeature/Sources/MusicDetail/MusicDetailViewController.swift @@ -1,3 +1,4 @@ +import ArtistFeatureInterface import BaseFeature import BaseFeatureInterface import DesignSystem @@ -20,6 +21,7 @@ final class MusicDetailViewController: BaseReactorViewController { get } var singingRoomButtonDidTap: Observable { get } var lyricsButtonDidTap: Observable { get } + var didTapArtistLabel: Observable { get } } final class MusicControlView: UIView { @@ -28,7 +30,7 @@ final class MusicControlView: UIView { textColor: DesignSystemAsset.BlueGrayColor.gray25.color, font: .t4(weight: .medium) ) - private let artistLabel: WMFlowLabel = WMFlowLabel( + fileprivate let artistLabel: WMFlowLabel = WMFlowLabel( text: "", textColor: DesignSystemAsset.BlueGrayColor.gray100.color.withAlphaComponent(0.6), font: .t5(weight: .medium) @@ -214,4 +216,11 @@ extension Reactive: MusicControlActionProtocol where Base: MusicControlView { var lyricsButtonDidTap: Observable { base.lyricsRoomButton.rx.tap.asObservable() } + + var didTapArtistLabel: Observable { + base.artistLabel.rx.tapGesture() + .when(.recognized) + .map { _ in () } + .asObservable() + } } diff --git a/Projects/Modules/Utility/Sources/Utils/UniquedSequence.swift b/Projects/Modules/Utility/Sources/Utils/UniquedSequence.swift index 1616cc018..9e2ac759d 100644 --- a/Projects/Modules/Utility/Sources/Utils/UniquedSequence.swift +++ b/Projects/Modules/Utility/Sources/Utils/UniquedSequence.swift @@ -52,3 +52,9 @@ extension UniquedSequence: Sequence { } extension UniquedSequence: LazySequenceProtocol where Base: LazySequenceProtocol {} + +public extension UniquedSequence { + func toArray() -> [Base.Element] { + Array(self) + } +}