Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

🔀 :: [#84] 음악 신청 ShareExtension 지원 #150

Merged
merged 5 commits into from
Aug 8, 2023
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
22 changes: 21 additions & 1 deletion Projects/App/Project.swift
Original file line number Diff line number Diff line change
Expand Up @@ -50,9 +50,29 @@ let targets: [Target] = [
.core(target: .KeyValueStore),
.core(target: .Networking),
.core(target: .Database),
.core(target: .Timer)
.core(target: .Timer),
.target(name: "\(env.name)ShareExtension")
],
settings: .settings(base: env.baseSetting)
),
.init(
name: "\(env.name)ShareExtension",
platform: .iOS,
product: .appExtension,
bundleId: "\(env.organizationName).\(env.name).share",
deploymentTarget: env.deploymentTarget,
infoPlist: .file(path: "ShareExtension/Support/Info.plist"),
sources: ["ShareExtension/Sources/**"],
resources: ["ShareExtension/Resources/**"],
entitlements: "Support/Dotori.entitlements",
dependencies: [
.domain(target: .MusicDomain),
.userInterface(target: .DesignSystem),
.userInterface(target: .Localization),
.core(target: .JwtStore),
.core(target: .KeyValueStore),
.core(target: .Networking)
]
)
]

Expand Down
27 changes: 27 additions & 0 deletions Projects/App/ShareExtension/Resources/MainInterface.storyboard
Original file line number Diff line number Diff line change
@@ -0,0 +1,27 @@
<?xml version="1.0" encoding="UTF-8"?>
<document type="com.apple.InterfaceBuilder3.CocoaTouch.Storyboard.XIB" version="3.0" toolsVersion="21701" targetRuntime="iOS.CocoaTouch" propertyAccessControl="none" useAutolayout="YES" useTraitCollections="YES" useSafeAreas="YES" colorMatched="YES" initialViewController="j1y-V4-xli">
<device id="retina6_12" orientation="portrait" appearance="light"/>
<dependencies>
<deployment identifier="iOS"/>
<plugIn identifier="com.apple.InterfaceBuilder.IBCocoaTouchPlugin" version="21678"/>
<capability name="Safe area layout guides" minToolsVersion="9.0"/>
<capability name="documents saved in the Xcode 8 format" minToolsVersion="8.0"/>
</dependencies>
<scenes>
<!--Dotori Share View Controller-->
<scene sceneID="ceB-am-kn3">
<objects>
<viewController id="j1y-V4-xli" customClass="DotoriShareViewController" customModule="DotoriShareExtension" customModuleProvider="target" sceneMemberID="viewController">
<view key="view" opaque="NO" contentMode="scaleToFill" id="wbc-yd-nQP">
<rect key="frame" x="0.0" y="0.0" width="393" height="852"/>
<autoresizingMask key="autoresizingMask" widthSizable="YES" heightSizable="YES"/>
<viewLayoutGuide key="safeArea" id="1Xd-am-t49"/>
<color key="backgroundColor" red="0.0" green="0.0" blue="0.0" alpha="0.0" colorSpace="custom" customColorSpace="sRGB"/>
</view>
</viewController>
<placeholder placeholderIdentifier="IBFirstResponder" id="CEy-Cv-SGf" userLabel="First Responder" sceneMemberID="firstResponder"/>
</objects>
<point key="canvasLocation" x="138.1679389312977" y="-2.1126760563380285"/>
</scene>
</scenes>
</document>
205 changes: 205 additions & 0 deletions Projects/App/ShareExtension/Sources/DotoriShareViewController.swift
Original file line number Diff line number Diff line change
@@ -0,0 +1,205 @@
import Combine
import Configure
import DesignSystem
import Foundation
import LinkPresentation
import Localization
import MSGLayout
import MusicDomainInterface
import Social
import UIKit
import UniformTypeIdentifiers

final class DotoriShareViewController: UIViewController {
private let contentView = UIView()
.set(\.backgroundColor, .dotori(.background(.card)))
.set(\.cornerRadius, 10)
.then {
$0.layer.maskedCorners = [.layerMinXMinYCorner, .layerMaxXMinYCorner]
}
private let proposeMusicLabel = DotoriLabel(L10n.ProposeMusic.proposeMusicTitle, font: .subtitle1)
private let cancelButton = DotoriTextButton(L10n.Global.cancelButtonTitle, textColor: .neutral(.n20), font: .body2)
private let thumbnailImageView = UIImageView()
.set(\.contentMode, .scaleAspectFill)
.set(\.cornerRadius, 4)
.set(\.clipsToBounds, true)
private let imageActivityIndicator = UIActivityIndicatorView(style: .medium)
private let titleLabel = DotoriLabel(font: .smalltitle)
.set(\.numberOfLines, 0)
private let proposeButton = DotoriButton(text: L10n.ProposeMusic.proposeButtonTitle)
.set(\.contentEdgeInsets, .vertical(16))
private let proposeMusicUseCase: any ProposeMusicUseCase

private var shareURL: URL?
private var subscription = Set<AnyCancellable>()

required init?(coder: NSCoder) {
self.proposeMusicUseCase = MusicContainer.shared.container.resolve(ProposeMusicUseCase.self)!
super.init(coder: coder)
}

override func viewDidLoad() {
super.viewDidLoad()
addView()
setLayout()
bindAction()
bindExtensionInput()
}
}

private extension DotoriShareViewController {
func addView() {
view.addSubviews {
contentView
}
thumbnailImageView.addSubviews {
imageActivityIndicator
}
}

func setLayout() {
MSGLayout.buildLayout {
contentView.layout
.horizontal(.toSuperview())
.bottom(.toSuperview())
.height(268)

imageActivityIndicator.layout
.center(.toSuperview())
}

MSGLayout.stackedLayout(self.contentView) {
VStackView(spacing: 8) {
HStackView {
proposeMusicLabel

SpacerView()

cancelButton
}
.alignment(.lastBaseline)

HStackView(spacing: 8) {
thumbnailImageView
.width(96)
.height(72)

titleLabel
}
.distribution(.fill)
.alignment(.center)
.height(88)
.margin(.horizontal(8))
.set(\.backgroundColor, .dotori(.neutral(.n50)))
.set(\.cornerRadius, 8)

VStackView {
proposeButton
}
.margin(.init(top: 16, left: 0, bottom: 8, right: 0))
}
.margin(.all(16))
}
}

func bindAction() {
cancelButton.tapPublisher
.sink(with: self, receiveValue: { owner, _ in
owner.hideExtension { _ in
owner.extensionContext?.cancelRequest(withError: NSError(domain: "Share Canceled", code: 1))
}
})
.store(in: &subscription)

proposeButton.tapPublisher
.sink(with: self, receiveValue: { owner, _ in
guard let url = owner.shareURL else { return }
Task {
do {
try await owner.proposeMusicUseCase(url: url.absoluteString)
owner.hideExtension { _ in
owner.extensionContext?.completeRequest(returningItems: nil)
}
} catch {
owner.hideExtension { _ in
owner.extensionContext?.cancelRequest(
withError: NSError(domain: "Jwt Token is expired", code: 6)
)
}
}
}
})
.store(in: &subscription)
}

func hideExtension(completion: @escaping (Bool) -> Void) {
UIView.animate(withDuration: 0.3, animations: {
self.view.transform = CGAffineTransform(
translationX: 0,
y: self.view.frame.size.height
)
}, completion: completion)
}

func bindExtensionInput() {
guard let extensionInput = extensionContext?.inputItems as? [NSExtensionItem] else {
self.extensionContext?.cancelRequest(withError: NSError(domain: "Invalid Inputs", code: 2))
return
}
for input in extensionInput where input.attachments?.isEmpty == false {
let itemProviders = input.attachments ?? []

for itemProvider in itemProviders where itemProvider.hasItemConformingToTypeIdentifier("public.url") {
itemProvider.loadItem(forTypeIdentifier: "public.url", options: nil) { [weak self] item, error in
guard let self, let url = item as? URL, error == nil else {
self?.hideExtension { _ in
self?.extensionContext?.cancelRequest(
withError: NSError(domain: "Invalid URL Input", code: 3)
)
}
return
}
self.shareURL = url

DispatchQueue.global(qos: .userInteractive).async {
self.bindYoutubeThumbnail(url: url)
}
}
}
}
}

func bindYoutubeThumbnail(url: URL) {
DispatchQueue.main.async {
self.imageActivityIndicator.startAnimating()
}
let provider = LPMetadataProvider()
provider.startFetchingMetadata(for: url) { [weak owner = self] metadata, error in
guard let owner, let metadata, error == nil else {
owner?.hideExtension { _ in
owner?.extensionContext?.cancelRequest(
withError: NSError(domain: "Invalid Youtube Metadata", code: 4)
)
}
return
}

metadata.imageProvider?.loadObject(ofClass: UIImage.self) { image, error in
guard let image = image as? UIImage, error == nil else {
owner.hideExtension { _ in
owner.extensionContext?.cancelRequest(
withError: NSError(domain: "Invalid Youtube Thumbnail Image", code: 5)
)
}
return
}

DispatchQueue.main.async {
self.thumbnailImageView.image = image
self.titleLabel.text = metadata.title
self.imageActivityIndicator.stopAnimating()
}
}
}
}
}
21 changes: 21 additions & 0 deletions Projects/App/ShareExtension/Sources/MusicContainer.swift
Original file line number Diff line number Diff line change
@@ -0,0 +1,21 @@
import Swinject
import MusicDomain
import JwtStore
import Networking
import KeyValueStore

final class MusicContainer {
static let shared = MusicContainer()

let container = Container()
var assembler: Assembler?

init() {
assembler = Assembler([
KeyValueStoreAssembly(),
JwtStoreAssembly(),
NetworkingAssembly(),
MusicDomainAssembly()
], container: container)
}
}
10 changes: 10 additions & 0 deletions Projects/App/ShareExtension/Support/DotoriShare.entitlements
Original file line number Diff line number Diff line change
@@ -0,0 +1,10 @@
<?xml version="1.0" encoding="UTF-8"?>
<!DOCTYPE plist PUBLIC "-//Apple//DTD PLIST 1.0//EN" "http://www.apple.com/DTDs/PropertyList-1.0.dtd">
<plist version="1.0">
<dict>
<key>keychain-access-groups</key>
<array>
<string>$(AppIdentifierPrefix)com.msg.Dotori.keychainGroup</string>
</array>
</dict>
</plist>
45 changes: 45 additions & 0 deletions Projects/App/ShareExtension/Support/Info.plist
Original file line number Diff line number Diff line change
@@ -0,0 +1,45 @@
<?xml version="1.0" encoding="UTF-8"?>
<!DOCTYPE plist PUBLIC "-//Apple//DTD PLIST 1.0//EN" "http://www.apple.com/DTDs/PropertyList-1.0.dtd">
<plist version="1.0">
<dict>
<key>AppIdentifierPrefix</key>
<string>$(AppIdentifierPrefix)</string>
<key>CFBundleDevelopmentRegion</key>
<string>$(DEVELOPMENT_LANGUAGE)</string>
<key>CFBundleDisplayName</key>
<string>DotoriShareExtension</string>
<key>CFBundleExecutable</key>
<string>$(EXECUTABLE_NAME)</string>
<key>CFBundleIdentifier</key>
<string>$(PRODUCT_BUNDLE_IDENTIFIER).shareextension</string>
<key>CFBundleInfoDictionaryVersion</key>
<string>6.0</string>
<key>CFBundleName</key>
<string>$(PRODUCT_NAME)</string>
<key>CFBundlePackageType</key>
<string>$(PRODUCT_BUNDLE_PACKAGE_TYPE)</string>
<key>CFBundleShortVersionString</key>
<string>1.0</string>
<key>CFBundleVersion</key>
<string>1.0</string>
<key>NSExtension</key>
<dict>
<key>NSExtensionAttributes</key>
<dict>
<key>NSExtensionActivationRule</key>
<dict>
<key>NSExtensionActivationDictionaryVersion</key>
<integer>1</integer>
<key>NSExtensionActivationSupportsText</key>
<true/>
<key>NSExtensionActivationSupportsWebURLWithMaxCount</key>
<string>1</string>
</dict>
</dict>
<key>NSExtensionMainStoryboard</key>
<string>MainInterface</string>
<key>NSExtensionPointIdentifier</key>
<string>com.apple.share-services</string>
</dict>
</dict>
</plist>
4 changes: 4 additions & 0 deletions Projects/App/Support/Dotori.entitlements
Original file line number Diff line number Diff line change
Expand Up @@ -6,5 +6,9 @@
<array>
<string>group.msg.dotori</string>
</array>
<key>keychain-access-groups</key>
<array>
<string>$(AppIdentifierPrefix)com.msg.Dotori.keychainGroup</string>
</array>
</dict>
</plist>
2 changes: 2 additions & 0 deletions Projects/App/Support/Info.plist
Original file line number Diff line number Diff line change
Expand Up @@ -2,6 +2,8 @@
<!DOCTYPE plist PUBLIC "-//Apple//DTD PLIST 1.0//EN" "http://www.apple.com/DTDs/PropertyList-1.0.dtd">
<plist version="1.0">
<dict>
<key>AppIdentifierPrefix</key>
<string>$(AppIdentifierPrefix)</string>
<key>CFBundleDisplayName</key>
<string>도토리</string>
<key>CFBundleDevelopmentRegion</key>
Expand Down
Loading