mirror of
https://github.com/immich-app/immich.git
synced 2026-06-05 21:02:58 +08:00
Some checks failed
CLI Build / CLI Publish (push) Has been cancelled
CodeQL / Analyze (javascript) (push) Has been cancelled
CodeQL / Analyze (python) (push) Has been cancelled
Docker / pre-job (push) Has been cancelled
Docs build / pre-job (push) Has been cancelled
Static Code Analysis / pre-job (push) Has been cancelled
Static Code Analysis / zizmor (push) Has been cancelled
Test / pre-job (push) Has been cancelled
Test / ShellCheck (push) Has been cancelled
Test / OpenAPI Clients (push) Has been cancelled
Test / SQL Schema Checks (push) Has been cancelled
CLI Build / Docker (push) Has been cancelled
Docker / Re-Tag ML () (push) Has been cancelled
Docker / Re-Tag ML (-armnn) (push) Has been cancelled
Docker / Re-Tag ML (-cuda) (push) Has been cancelled
Docker / Re-Tag ML (-openvino) (push) Has been cancelled
Docker / Re-Tag ML (-rknn) (push) Has been cancelled
Docker / Re-Tag ML (-rocm) (push) Has been cancelled
Docker / Re-Tag Server () (push) Has been cancelled
Docker / Build and Push ML (armnn, linux/arm64, -armnn) (push) Has been cancelled
Docker / Build and Push ML (cpu, ) (push) Has been cancelled
Docker / Build and Push ML (cuda, linux/amd64, -cuda) (push) Has been cancelled
Docker / Build and Push ML (openvino, linux/amd64, -openvino) (push) Has been cancelled
Docker / Build and Push ML (rknn, linux/arm64, -rknn) (push) Has been cancelled
Docker / Build and Push ML (rocm, linux/amd64, {"linux/amd64": "mich"}, -rocm) (push) Has been cancelled
Docker / Build and Push Server (push) Has been cancelled
Docker / Docker Build & Push Server Success (push) Has been cancelled
Docker / Docker Build & Push ML Success (push) Has been cancelled
Docs build / Docs Build (push) Has been cancelled
Static Code Analysis / Run Dart Code Analysis (push) Has been cancelled
Test / Test & Lint Server (push) Has been cancelled
Test / Unit Test CLI (push) Has been cancelled
Test / Unit Test CLI (Windows) (push) Has been cancelled
Test / Lint Web (push) Has been cancelled
Test / Test Web (push) Has been cancelled
Test / Test i18n (push) Has been cancelled
Test / End-to-End Lint (push) Has been cancelled
Test / Medium Tests (Server) (push) Has been cancelled
Test / End-to-End Tests (Server & CLI) (ubuntu-24.04-arm) (push) Has been cancelled
Test / End-to-End Tests (Server & CLI) (ubuntu-latest) (push) Has been cancelled
Test / End-to-End Tests (Web) (ubuntu-24.04-arm) (push) Has been cancelled
Test / End-to-End Tests (Web) (ubuntu-latest) (push) Has been cancelled
Test / End-to-End Tests Success (push) Has been cancelled
Test / Unit Test Mobile (push) Has been cancelled
Test / Unit Test ML (push) Has been cancelled
Test / .github Files Formatting (push) Has been cancelled
* improvements to error handling, ability to select "Favorites" as a virtual album, fix widgets not showing image when tinting homescreen * dont include isFavorite all the time * remove check for if the album exists this will never run because we default to Album.NONE and its impossible to distinguish between no album selected and album DNE (we dont know what the store ID is, only what iOS gives)
149 lines
3.5 KiB
Swift
149 lines
3.5 KiB
Swift
import SwiftUI
|
|
import WidgetKit
|
|
|
|
typealias EntryMetadata = ImageEntry.Metadata
|
|
|
|
struct ImageEntry: TimelineEntry {
|
|
let date: Date
|
|
var image: UIImage?
|
|
var metadata: Metadata = Metadata()
|
|
|
|
struct Metadata: Codable {
|
|
var subtitle: String? = nil
|
|
var error: WidgetError? = nil
|
|
var deepLink: URL? = nil
|
|
}
|
|
|
|
static func build(
|
|
api: ImmichAPI,
|
|
asset: Asset,
|
|
dateOffset: Int,
|
|
subtitle: String? = nil
|
|
)
|
|
async throws -> Self
|
|
{
|
|
let entryDate = Calendar.current.date(
|
|
byAdding: .minute,
|
|
value: dateOffset * 20,
|
|
to: Date.now
|
|
)!
|
|
let image = try await api.fetchImage(asset: asset)
|
|
|
|
return Self(
|
|
date: entryDate,
|
|
image: image,
|
|
metadata: EntryMetadata(
|
|
subtitle: subtitle,
|
|
deepLink: asset.deepLink
|
|
)
|
|
)
|
|
}
|
|
|
|
func cache(for key: String) throws {
|
|
if let containerURL = FileManager.default.containerURL(
|
|
forSecurityApplicationGroupIdentifier: IMMICH_SHARE_GROUP
|
|
) {
|
|
let imageURL = containerURL.appendingPathComponent("\(key)_image.png")
|
|
let metadataURL = containerURL.appendingPathComponent(
|
|
"\(key)_metadata.json"
|
|
)
|
|
|
|
// build metadata JSON
|
|
let entryMetadata = try JSONEncoder().encode(self.metadata)
|
|
|
|
// write to disk
|
|
try self.image?.pngData()?.write(to: imageURL, options: .atomic)
|
|
try entryMetadata.write(to: metadataURL, options: .atomic)
|
|
}
|
|
}
|
|
|
|
static func loadCached(for key: String, at date: Date = Date.now)
|
|
-> ImageEntry?
|
|
{
|
|
if let containerURL = FileManager.default.containerURL(
|
|
forSecurityApplicationGroupIdentifier: IMMICH_SHARE_GROUP
|
|
) {
|
|
let imageURL = containerURL.appendingPathComponent("\(key)_image.png")
|
|
let metadataURL = containerURL.appendingPathComponent(
|
|
"\(key)_metadata.json"
|
|
)
|
|
|
|
guard let imageData = try? Data(contentsOf: imageURL),
|
|
let metadataJSON = try? Data(contentsOf: metadataURL),
|
|
let decodedMetadata = try? JSONDecoder().decode(
|
|
Metadata.self,
|
|
from: metadataJSON
|
|
)
|
|
else {
|
|
return nil
|
|
}
|
|
|
|
return ImageEntry(
|
|
date: date,
|
|
image: UIImage(data: imageData),
|
|
metadata: decodedMetadata
|
|
)
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
static func handleError(
|
|
for key: String,
|
|
error: WidgetError = .fetchFailed
|
|
) -> Timeline<ImageEntry> {
|
|
var timelineEntry = ImageEntry(
|
|
date: Date.now,
|
|
image: nil,
|
|
metadata: EntryMetadata(error: error)
|
|
)
|
|
|
|
// use cache if generic failed error
|
|
// we want to show the other errors to the user since without intervention,
|
|
// it will never succeed
|
|
if error == .fetchFailed, let cachedEntry = ImageEntry.loadCached(for: key)
|
|
{
|
|
timelineEntry = cachedEntry
|
|
}
|
|
|
|
return Timeline(entries: [timelineEntry], policy: .atEnd)
|
|
}
|
|
|
|
}
|
|
|
|
func generateRandomEntries(
|
|
api: ImmichAPI,
|
|
now: Date,
|
|
count: Int,
|
|
filter: SearchFilter = Album.NONE.filter,
|
|
subtitle: String? = nil
|
|
)
|
|
async throws -> [ImageEntry]
|
|
{
|
|
|
|
var entries: [ImageEntry] = []
|
|
|
|
let randomAssets = try await api.fetchSearchResults(with: filter)
|
|
|
|
await withTaskGroup(of: ImageEntry?.self) { group in
|
|
for (dateOffset, asset) in randomAssets.enumerated() {
|
|
group.addTask {
|
|
return try? await ImageEntry.build(
|
|
api: api,
|
|
asset: asset,
|
|
dateOffset: dateOffset,
|
|
subtitle: subtitle
|
|
)
|
|
}
|
|
}
|
|
|
|
for await result in group {
|
|
if let entry = result {
|
|
entries.append(entry)
|
|
}
|
|
}
|
|
}
|
|
|
|
return entries
|
|
}
|