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 {
|
2024-09-10 04:53:19 +00:00
|
|
|
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-09-10 04:53:19 +00:00
|
|
|
}
|
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
|
|
|
}
|