IceCubesApp/IceCubesAppWidgetsExtension/LatestPostsWidget/LatestPostsWidget.swift

124 lines
3.3 KiB
Swift
Raw Permalink Normal View History

2024-05-05 17:31:28 +00:00
import DesignSystem
import Models
2024-05-06 06:38:37 +00:00
import Network
import SwiftUI
2024-05-05 17:31:28 +00:00
import Timeline
2024-05-06 06:38:37 +00:00
import WidgetKit
2024-05-05 17:31:28 +00:00
struct LatestPostsWidgetProvider: AppIntentTimelineProvider {
2024-05-06 06:38:37 +00:00
func placeholder(in _: Context) -> PostsWidgetEntry {
2024-10-28 09:57:48 +00:00
.init(
date: Date(),
title: "Home",
statuses: [.placeholder()],
images: [:])
2024-05-05 17:31:28 +00:00
}
2024-05-06 06:38:37 +00:00
2024-10-28 09:57:48 +00:00
func snapshot(for configuration: LatestPostsWidgetConfiguration, in context: Context) async
-> PostsWidgetEntry
{
2024-05-05 17:31:28 +00:00
if let entry = await timeline(for: configuration, context: context).entries.first {
return entry
}
2024-10-28 09:57:48 +00:00
return .init(
date: Date(),
title: configuration.timeline?.timeline.title ?? "",
statuses: [],
images: [:])
2024-05-05 17:31:28 +00:00
}
2024-05-06 06:38:37 +00:00
2024-10-28 09:57:48 +00:00
func timeline(for configuration: LatestPostsWidgetConfiguration, in context: Context) async
-> Timeline<PostsWidgetEntry>
{
2024-05-05 17:31:28 +00:00
await timeline(for: configuration, context: context)
}
2024-05-06 06:38:37 +00:00
2024-10-28 09:57:48 +00:00
private func timeline(for configuration: LatestPostsWidgetConfiguration, context: Context) async
-> Timeline<PostsWidgetEntry>
{
2024-05-05 17:31:28 +00:00
do {
guard let timeline = configuration.timeline, let account = configuration.account else {
2024-10-28 09:57:48 +00:00
return Timeline(
entries: [
.init(
date: Date(),
title: "",
statuses: [],
images: [:])
],
policy: .atEnd)
}
2024-10-28 09:57:48 +00:00
let statuses = await loadStatuses(
for: timeline.timeline,
account: account,
widgetFamily: context.family)
2024-05-06 06:38:37 +00:00
let images = try await loadImages(urls: statuses.map { $0.account.avatar })
2024-10-28 09:57:48 +00:00
return Timeline(
entries: [
.init(
date: Date(),
title: timeline.timeline.title,
statuses: statuses,
images: images)
], policy: .atEnd)
2024-05-05 17:31:28 +00:00
} catch {
2024-10-28 09:57:48 +00:00
return Timeline(
entries: [
.init(
date: Date(),
title: configuration.timeline?.timeline.title ?? "",
statuses: [],
images: [:])
],
policy: .atEnd)
2024-05-05 17:31:28 +00:00
}
}
2024-05-06 06:38:37 +00:00
2024-05-05 17:31:28 +00:00
private func loadImages(urls: [URL]) async throws -> [URL: UIImage] {
try await withThrowingTaskGroup(of: (URL, UIImage?).self) { group in
for url in urls {
group.addTask {
let response = try await URLSession.shared.data(from: url)
return (url, UIImage(data: response.0))
}
}
2024-05-06 06:38:37 +00:00
2024-05-05 17:31:28 +00:00
var images: [URL: UIImage] = [:]
2024-05-06 06:38:37 +00:00
2024-05-05 17:31:28 +00:00
for try await (url, image) in group {
images[url] = image
}
2024-05-06 06:38:37 +00:00
2024-05-05 17:31:28 +00:00
return images
}
}
}
struct LatestPostsWidget: Widget {
let kind: String = "LatestPostsWidget"
2024-05-06 06:38:37 +00:00
2024-05-05 17:31:28 +00:00
var body: some WidgetConfiguration {
2024-10-28 09:57:48 +00:00
AppIntentConfiguration(
kind: kind,
intent: LatestPostsWidgetConfiguration.self,
provider: LatestPostsWidgetProvider()
) { entry in
2024-05-05 17:37:06 +00:00
PostsWidgetView(entry: entry)
2024-05-05 17:31:28 +00:00
.containerBackground(Color("WidgetBackground").gradient, for: .widget)
}
.configurationDisplayName("Latest posts")
.description("Show the latest post for the selected timeline")
.supportedFamilies([.systemSmall, .systemMedium, .systemLarge, .systemExtraLarge])
}
}
#Preview(as: .systemMedium) {
LatestPostsWidget()
} timeline: {
2024-10-28 09:57:48 +00:00
PostsWidgetEntry(
date: .now,
title: "Mastodon",
statuses: [.placeholder(), .placeholder(), .placeholder(), .placeholder()],
images: [:])
2024-05-05 17:31:28 +00:00
}