mirror of
https://github.com/X1a0He/Adobe-Downloader.git
synced 2025-11-25 11:18:53 +08:00
518 lines
18 KiB
Swift
518 lines
18 KiB
Swift
//
|
|
// Adobe Downloader
|
|
//
|
|
// Created by X1a0He on 2024/10/30.
|
|
//
|
|
|
|
import SwiftUI
|
|
|
|
class IconCache {
|
|
static let shared = IconCache()
|
|
private var cache = NSCache<NSString, NSImage>()
|
|
|
|
func getIcon(for url: String) -> NSImage? {
|
|
cache.object(forKey: url as NSString)
|
|
}
|
|
|
|
func setIcon(_ image: NSImage, for url: String) {
|
|
cache.setObject(image, forKey: url as NSString)
|
|
}
|
|
}
|
|
|
|
class AppCardViewModel: ObservableObject {
|
|
@Published var iconImage: NSImage?
|
|
@Published var showError = false
|
|
@Published var errorMessage = ""
|
|
@Published var showVersionPicker = false
|
|
@Published var selectedVersion = ""
|
|
@Published var showLanguagePicker = false
|
|
@Published var selectedLanguage = ""
|
|
@Published var showExistingFileAlert = false
|
|
@Published var existingFilePath: URL?
|
|
@Published var pendingVersion = ""
|
|
@Published var pendingLanguage = ""
|
|
@Published var showRedownloadConfirm = false
|
|
|
|
let sap: Sap
|
|
weak var networkManager: NetworkManager?
|
|
|
|
@Published var isDownloading = false
|
|
private let userDefaults = UserDefaults.standard
|
|
|
|
var useDefaultDirectory: Bool {
|
|
get { userDefaults.bool(forKey: "useDefaultDirectory") }
|
|
}
|
|
|
|
var defaultDirectory: String {
|
|
get { userDefaults.string(forKey: "defaultDirectory") ?? "" }
|
|
}
|
|
|
|
init(sap: Sap, networkManager: NetworkManager?) {
|
|
self.sap = sap
|
|
self.networkManager = networkManager
|
|
loadIcon()
|
|
|
|
NotificationCenter.default.addObserver(
|
|
self,
|
|
selector: #selector(updateDownloadingStatus),
|
|
name: NSNotification.Name("UpdateDownloadStatus"),
|
|
object: nil
|
|
)
|
|
}
|
|
|
|
deinit {
|
|
NotificationCenter.default.removeObserver(self)
|
|
}
|
|
|
|
@objc func updateDownloadingStatus() {
|
|
Task { @MainActor in
|
|
isDownloading = networkManager?.downloadTasks.contains { task in
|
|
return task.sapCode == sap.sapCode && task.status.isActive
|
|
} ?? false
|
|
}
|
|
}
|
|
|
|
func getDestinationURL(version: String, language: String, useDefaultDirectory: Bool, defaultDirectory: String) async throws -> URL {
|
|
let platform = sap.versions[version]?.apPlatform ?? "unknown"
|
|
let installerName = sap.sapCode == "APRO"
|
|
? "Adobe Downloader \(sap.sapCode)_\(version)_\(platform).dmg"
|
|
: "Adobe Downloader \(sap.sapCode)_\(version)-\(language)-\(platform)"
|
|
|
|
if useDefaultDirectory && !defaultDirectory.isEmpty {
|
|
return URL(fileURLWithPath: defaultDirectory)
|
|
.appendingPathComponent(installerName)
|
|
}
|
|
|
|
return try await withCheckedThrowingContinuation { continuation in
|
|
DispatchQueue.main.async {
|
|
let panel = NSOpenPanel()
|
|
panel.title = "选择保存位置"
|
|
panel.canCreateDirectories = true
|
|
panel.canChooseDirectories = true
|
|
panel.canChooseFiles = false
|
|
panel.directoryURL = FileManager.default.urls(for: .downloadsDirectory, in: .userDomainMask).first
|
|
|
|
if panel.runModal() == .OK, let selectedURL = panel.url {
|
|
continuation.resume(returning: selectedURL.appendingPathComponent(installerName))
|
|
} else {
|
|
continuation.resume(throwing: NetworkError.cancelled)
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
func handleError(_ error: Error) {
|
|
Task { @MainActor in
|
|
if case NetworkError.cancelled = error { return }
|
|
errorMessage = error.localizedDescription
|
|
showError = true
|
|
}
|
|
}
|
|
|
|
func loadIcon() {
|
|
if let bestIcon = sap.getBestIcon(),
|
|
let iconURL = URL(string: bestIcon.url) {
|
|
|
|
if let cachedImage = IconCache.shared.getIcon(for: bestIcon.url) {
|
|
self.iconImage = cachedImage
|
|
return
|
|
}
|
|
|
|
Task {
|
|
do {
|
|
var request = URLRequest(url: iconURL)
|
|
request.timeoutInterval = 10
|
|
|
|
let (data, response) = try await URLSession.shared.data(for: request)
|
|
|
|
guard let httpResponse = response as? HTTPURLResponse,
|
|
(200...299).contains(httpResponse.statusCode) else {
|
|
throw URLError(.badServerResponse)
|
|
}
|
|
|
|
await MainActor.run {
|
|
if let image = NSImage(data: data) {
|
|
IconCache.shared.setIcon(image, for: bestIcon.url)
|
|
self.iconImage = image
|
|
}
|
|
}
|
|
} catch {
|
|
await MainActor.run {
|
|
if let localImage = NSImage(named: sap.sapCode) {
|
|
self.iconImage = localImage
|
|
}
|
|
}
|
|
}
|
|
}
|
|
} else {
|
|
if let localImage = NSImage(named: sap.sapCode) {
|
|
self.iconImage = localImage
|
|
}
|
|
}
|
|
}
|
|
|
|
func handleDownloadRequest(_ version: String, useDefaultLanguage: Bool, defaultLanguage: String) async {
|
|
await MainActor.run {
|
|
if useDefaultLanguage {
|
|
Task {
|
|
await checkAndStartDownload(version: version, language: defaultLanguage)
|
|
}
|
|
} else {
|
|
selectedVersion = version
|
|
showLanguagePicker = true
|
|
}
|
|
}
|
|
}
|
|
|
|
func checkAndStartDownload(version: String, language: String) async {
|
|
if let networkManager = networkManager {
|
|
if let existingPath = await networkManager.isVersionDownloaded(sap: sap, version: version, language: language) {
|
|
await MainActor.run {
|
|
existingFilePath = existingPath
|
|
pendingVersion = version
|
|
pendingLanguage = language
|
|
showExistingFileAlert = true
|
|
}
|
|
} else {
|
|
startDownload(version, language)
|
|
}
|
|
}
|
|
}
|
|
|
|
func startDownload(_ version: String, _ language: String) {
|
|
Task {
|
|
do {
|
|
let destinationURL = try await getDestinationURL(
|
|
version: version,
|
|
language: language,
|
|
useDefaultDirectory: useDefaultDirectory,
|
|
defaultDirectory: defaultDirectory
|
|
)
|
|
|
|
try await networkManager?.startDownload(
|
|
sap: sap,
|
|
selectedVersion: version,
|
|
language: language,
|
|
destinationURL: destinationURL
|
|
)
|
|
} catch {
|
|
handleError(error)
|
|
}
|
|
}
|
|
}
|
|
|
|
func createCompletedTask(_ path: URL) async {
|
|
guard let networkManager = networkManager,
|
|
let productInfo = sap.versions[pendingVersion] else { return }
|
|
|
|
let existingTask = await networkManager.downloadTasks.first { task in
|
|
return task.sapCode == sap.sapCode &&
|
|
task.version == pendingVersion &&
|
|
task.language == pendingLanguage &&
|
|
task.directory == path
|
|
}
|
|
|
|
if existingTask != nil {
|
|
return
|
|
}
|
|
|
|
var productsToDownload: [ProductsToDownload] = []
|
|
let mainProduct = ProductsToDownload(
|
|
sapCode: sap.sapCode,
|
|
version: pendingVersion,
|
|
buildGuid: productInfo.buildGuid
|
|
)
|
|
productsToDownload.append(mainProduct)
|
|
|
|
for dependency in productInfo.dependencies {
|
|
if let dependencyVersions = await networkManager.saps[dependency.sapCode]?.versions {
|
|
let sortedVersions = dependencyVersions.sorted { first, second in
|
|
first.value.productVersion.compare(second.value.productVersion, options: .numeric) == .orderedDescending
|
|
}
|
|
|
|
var buildGuid = ""
|
|
for (_, versionInfo) in sortedVersions where versionInfo.baseVersion == dependency.version {
|
|
if await networkManager.allowedPlatform.contains(versionInfo.apPlatform) {
|
|
buildGuid = versionInfo.buildGuid
|
|
break
|
|
}
|
|
}
|
|
|
|
if !buildGuid.isEmpty {
|
|
let dependencyProduct = ProductsToDownload(
|
|
sapCode: dependency.sapCode,
|
|
version: dependency.version,
|
|
buildGuid: buildGuid
|
|
)
|
|
productsToDownload.append(dependencyProduct)
|
|
}
|
|
}
|
|
}
|
|
|
|
let completedTask = NewDownloadTask(
|
|
sapCode: sap.sapCode,
|
|
version: pendingVersion,
|
|
language: pendingLanguage,
|
|
displayName: sap.displayName,
|
|
directory: path,
|
|
productsToDownload: productsToDownload,
|
|
retryCount: 0,
|
|
createAt: Date(),
|
|
totalStatus: .completed(DownloadStatus.CompletionInfo(
|
|
timestamp: Date(),
|
|
totalTime: 0,
|
|
totalSize: 0
|
|
)),
|
|
totalProgress: 1.0,
|
|
totalDownloadedSize: 0,
|
|
totalSize: 0,
|
|
totalSpeed: 0
|
|
)
|
|
|
|
await MainActor.run {
|
|
networkManager.downloadTasks.append(completedTask)
|
|
networkManager.objectWillChange.send()
|
|
}
|
|
}
|
|
|
|
var dependenciesCount: Int {
|
|
if let firstVersion = sap.versions.first?.value {
|
|
return firstVersion.dependencies.count
|
|
}
|
|
return 0
|
|
}
|
|
}
|
|
|
|
struct AppCardView: View {
|
|
@StateObject private var viewModel: AppCardViewModel
|
|
@EnvironmentObject private var networkManager: NetworkManager
|
|
@AppStorage("useDefaultLanguage") private var useDefaultLanguage = true
|
|
@AppStorage("defaultLanguage") private var defaultLanguage: String = "zh_CN"
|
|
|
|
init(sap: Sap) {
|
|
_viewModel = StateObject(wrappedValue: AppCardViewModel(sap: sap, networkManager: nil))
|
|
}
|
|
|
|
var body: some View {
|
|
CardContent(
|
|
sap: viewModel.sap,
|
|
iconImage: viewModel.iconImage,
|
|
loadIcon: viewModel.loadIcon,
|
|
dependenciesCount: viewModel.dependenciesCount,
|
|
isDownloading: viewModel.isDownloading,
|
|
showVersionPicker: $viewModel.showVersionPicker
|
|
)
|
|
.padding()
|
|
.frame(width: 250, height: 200)
|
|
.background(RoundedRectangle(cornerRadius: 10).fill(Color.black.opacity(0.05)))
|
|
.overlay(
|
|
RoundedRectangle(cornerRadius: 10)
|
|
.stroke(Color.gray.opacity(0.1), lineWidth: 2)
|
|
)
|
|
.modifier(AlertModifier(viewModel: viewModel, confirmRedownload: true))
|
|
.sheet(isPresented: $viewModel.showVersionPicker) {
|
|
VersionPickerView(sap: viewModel.sap) { version in
|
|
Task {
|
|
await viewModel.handleDownloadRequest(version, useDefaultLanguage: useDefaultLanguage, defaultLanguage: defaultLanguage)
|
|
}
|
|
}
|
|
.environmentObject(networkManager)
|
|
}
|
|
.sheet(isPresented: $viewModel.showLanguagePicker) {
|
|
LanguagePickerView(languages: AppStatics.supportedLanguages) { language in
|
|
Task {
|
|
await viewModel.checkAndStartDownload(version: viewModel.selectedVersion, language: language)
|
|
}
|
|
}
|
|
}
|
|
.onAppear {
|
|
viewModel.networkManager = networkManager
|
|
viewModel.updateDownloadingStatus()
|
|
}
|
|
}
|
|
}
|
|
|
|
private struct CardContent: View {
|
|
let sap: Sap
|
|
let iconImage: NSImage?
|
|
let loadIcon: () -> Void
|
|
let dependenciesCount: Int
|
|
let isDownloading: Bool
|
|
@Binding var showVersionPicker: Bool
|
|
|
|
var body: some View {
|
|
VStack {
|
|
IconView(iconImage: iconImage, loadIcon: loadIcon)
|
|
ProductInfoView(sap: sap, dependenciesCount: dependenciesCount)
|
|
Spacer()
|
|
DownloadButton(
|
|
isDownloading: isDownloading,
|
|
showVersionPicker: $showVersionPicker
|
|
)
|
|
}
|
|
}
|
|
}
|
|
|
|
private extension View {
|
|
func applyModifiers(viewModel: AppCardViewModel) -> some View {
|
|
self.modifier(AlertModifier(viewModel: viewModel, confirmRedownload: true))
|
|
}
|
|
}
|
|
|
|
private struct IconView: View {
|
|
let iconImage: NSImage?
|
|
let loadIcon: () -> Void
|
|
|
|
var body: some View {
|
|
Group {
|
|
if let iconImage = iconImage {
|
|
Image(nsImage: iconImage)
|
|
.resizable()
|
|
.interpolation(.high)
|
|
.scaledToFit()
|
|
} else {
|
|
Image(systemName: "app.fill")
|
|
.resizable()
|
|
.scaledToFit()
|
|
.foregroundColor(.secondary)
|
|
}
|
|
}
|
|
.frame(width: 64, height: 64)
|
|
.onAppear(perform: loadIcon)
|
|
}
|
|
}
|
|
|
|
private struct ProductInfoView: View {
|
|
let sap: Sap
|
|
let dependenciesCount: Int
|
|
|
|
var body: some View {
|
|
VStack {
|
|
Text(sap.displayName)
|
|
.font(.system(size: 16))
|
|
.fontWeight(.bold)
|
|
.lineLimit(2)
|
|
.multilineTextAlignment(.center)
|
|
|
|
HStack(spacing: 4) {
|
|
Text("可用版本: \(sap.versions.count)")
|
|
Text("|")
|
|
Text("依赖包: \(dependenciesCount)")
|
|
}
|
|
.font(.caption)
|
|
.foregroundColor(.secondary)
|
|
.frame(height: 20)
|
|
}
|
|
}
|
|
}
|
|
|
|
private struct DownloadButton: View {
|
|
let isDownloading: Bool
|
|
@Binding var showVersionPicker: Bool
|
|
|
|
var body: some View {
|
|
Button(action: { showVersionPicker = true }) {
|
|
Label(isDownloading ? "下载中" : "下载",
|
|
systemImage: isDownloading ? "hourglass.circle.fill" : "arrow.down.circle")
|
|
.font(.system(size: 14))
|
|
.frame(minWidth: 0, maxWidth: .infinity)
|
|
.frame(height: 32)
|
|
}
|
|
.buttonStyle(.borderedProminent)
|
|
.tint(isDownloading ? .gray : .blue)
|
|
.disabled(isDownloading)
|
|
}
|
|
}
|
|
|
|
#Preview {
|
|
let networkManager = NetworkManager()
|
|
let sap = Sap(
|
|
hidden: false,
|
|
displayName: "Photoshop",
|
|
sapCode: "PHSP",
|
|
versions: [
|
|
"25.0.0": Sap.Versions(
|
|
sapCode: "PHSP",
|
|
baseVersion: "25.0.0",
|
|
productVersion: "25.0.0",
|
|
apPlatform: "macuniversal",
|
|
dependencies: [
|
|
Sap.Versions.Dependencies(sapCode: "ACR", version: "9.6"),
|
|
Sap.Versions.Dependencies(sapCode: "COCM", version: "1.0"),
|
|
Sap.Versions.Dependencies(sapCode: "COSY", version: "2.4.1")
|
|
],
|
|
buildGuid: ""
|
|
)
|
|
],
|
|
icons: [
|
|
Sap.ProductIcon(
|
|
size: "192x192",
|
|
url: "https://ffc-static-cdn.oobesaas.adobe.com/icons/PHSP/25.0.0/192x192.png"
|
|
)
|
|
]
|
|
)
|
|
|
|
return AppCardView(sap: sap)
|
|
.environmentObject(networkManager)
|
|
}
|
|
|
|
struct AlertModifier: ViewModifier {
|
|
@ObservedObject var viewModel: AppCardViewModel
|
|
let confirmRedownload: Bool
|
|
|
|
func body(content: Content) -> some View {
|
|
content
|
|
.sheet(isPresented: $viewModel.showExistingFileAlert) {
|
|
if let path = viewModel.existingFilePath {
|
|
ExistingFileAlertView(
|
|
path: path,
|
|
onUseExisting: {
|
|
viewModel.showExistingFileAlert = false
|
|
if !viewModel.pendingVersion.isEmpty && !viewModel.pendingLanguage.isEmpty {
|
|
Task {
|
|
await viewModel.createCompletedTask(path)
|
|
}
|
|
}
|
|
},
|
|
onRedownload: {
|
|
viewModel.showExistingFileAlert = false
|
|
if !viewModel.pendingVersion.isEmpty && !viewModel.pendingLanguage.isEmpty {
|
|
if confirmRedownload {
|
|
viewModel.showRedownloadConfirm = true
|
|
} else {
|
|
viewModel.startDownload(viewModel.pendingVersion, viewModel.pendingLanguage)
|
|
}
|
|
}
|
|
},
|
|
onCancel: {
|
|
viewModel.showExistingFileAlert = false
|
|
},
|
|
iconImage: viewModel.iconImage
|
|
)
|
|
.background(Color.black.opacity(0.3))
|
|
.ignoresSafeArea()
|
|
}
|
|
}
|
|
.alert("确认重新下载", isPresented: $viewModel.showRedownloadConfirm) {
|
|
Button("取消", role: .cancel) { }
|
|
Button("确认") {
|
|
if !viewModel.pendingVersion.isEmpty && !viewModel.pendingLanguage.isEmpty {
|
|
viewModel.startDownload(viewModel.pendingVersion, viewModel.pendingLanguage)
|
|
}
|
|
}
|
|
} message: {
|
|
Text("是否确认重新下载?这将覆盖现有的安装程序。")
|
|
}
|
|
.alert("下载错误", isPresented: $viewModel.showError) {
|
|
Button("确定", role: .cancel) { }
|
|
Button("重试") {
|
|
if !viewModel.selectedVersion.isEmpty {
|
|
viewModel.startDownload(viewModel.selectedVersion, viewModel.selectedLanguage)
|
|
}
|
|
}
|
|
} message: {
|
|
Text(viewModel.errorMessage)
|
|
}
|
|
}
|
|
}
|