Migrate TagGroup to SwiftData
This commit is contained in:
parent
527d982dce
commit
4870b202d6
|
@ -31,9 +31,15 @@ extension View {
|
|||
case let .conversationDetail(conversation):
|
||||
ConversationDetailView(conversation: conversation)
|
||||
case let .hashTag(tag, accountId):
|
||||
TimelineView(timeline: .constant(.hashtag(tag: tag, accountId: accountId)), scrollToTopSignal: .constant(0), canFilterTimeline: false)
|
||||
TimelineView(timeline: .constant(.hashtag(tag: tag, accountId: accountId)),
|
||||
selectedTagGroup: .constant(nil),
|
||||
scrollToTopSignal: .constant(0),
|
||||
canFilterTimeline: false)
|
||||
case let .list(list):
|
||||
TimelineView(timeline: .constant(.list(list: list)), scrollToTopSignal: .constant(0), canFilterTimeline: false)
|
||||
TimelineView(timeline: .constant(.list(list: list)),
|
||||
selectedTagGroup: .constant(nil),
|
||||
scrollToTopSignal: .constant(0),
|
||||
canFilterTimeline: false)
|
||||
case let .following(id):
|
||||
AccountsListView(mode: .following(accountId: id))
|
||||
case let .followers(id):
|
||||
|
@ -45,7 +51,10 @@ extension View {
|
|||
case let .accountsList(accounts):
|
||||
AccountsListView(mode: .accountsList(accounts: accounts))
|
||||
case .trendingTimeline:
|
||||
TimelineView(timeline: .constant(.trending), scrollToTopSignal: .constant(0), canFilterTimeline: false)
|
||||
TimelineView(timeline: .constant(.trending),
|
||||
selectedTagGroup: .constant(nil),
|
||||
scrollToTopSignal: .constant(0),
|
||||
canFilterTimeline: false)
|
||||
case let .tagsList(tags):
|
||||
TagsListView(tags: tags)
|
||||
}
|
||||
|
@ -125,6 +134,7 @@ extension View {
|
|||
modelContainer(for: [
|
||||
Draft.self,
|
||||
LocalTimeline.self,
|
||||
TagGroup.self
|
||||
])
|
||||
}
|
||||
}
|
||||
|
|
|
@ -31,6 +31,7 @@ struct SettingsTabs: View {
|
|||
@Binding var popToRootTab: Tab
|
||||
|
||||
@Query(sort: \LocalTimeline.creationDate, order: .reverse) var localTimelines: [LocalTimeline]
|
||||
@Query(sort: \TagGroup.creationDate, order: .reverse) var tagGroups: [TagGroup]
|
||||
|
||||
var body: some View {
|
||||
NavigationStack(path: $routerPath.path) {
|
||||
|
@ -274,20 +275,17 @@ struct SettingsTabs: View {
|
|||
|
||||
private var tagGroupsView: some View {
|
||||
Form {
|
||||
ForEach(preferences.tagGroups, id: \.self) { group in
|
||||
Label(group.title, systemImage: group.sfSymbolName)
|
||||
ForEach(tagGroups) { group in
|
||||
Label(group.title, systemImage: group.symbolName)
|
||||
.onTapGesture {
|
||||
routerPath.presentedSheet = .editTagGroup(tagGroup: group, onSaved: nil)
|
||||
}
|
||||
}
|
||||
.onDelete { indexes in
|
||||
if let index = indexes.first {
|
||||
_ = preferences.tagGroups.remove(at: index)
|
||||
context.delete(tagGroups[index])
|
||||
}
|
||||
}
|
||||
.onMove { source, destination in
|
||||
preferences.tagGroups.move(fromOffsets: source, toOffset: destination)
|
||||
}
|
||||
.listRowBackground(theme.primaryBackgroundColor)
|
||||
|
||||
Button {
|
||||
|
|
|
@ -10,8 +10,8 @@ import SwiftUI
|
|||
@MainActor
|
||||
struct EditTagGroupView: View {
|
||||
@Environment(\.dismiss) private var dismiss
|
||||
@Environment(\.modelContext) private var context
|
||||
|
||||
@Environment(UserPreferences.self) private var preferences
|
||||
@Environment(Theme.self) private var theme
|
||||
|
||||
@State private var title: String = ""
|
||||
|
@ -22,7 +22,7 @@ struct EditTagGroupView: View {
|
|||
|
||||
private var editingTagGroup: TagGroup?
|
||||
private var onSaved: ((TagGroup) -> Void)?
|
||||
|
||||
|
||||
private var canSave: Bool {
|
||||
!title.isEmpty &&
|
||||
// At least have 2 tags, one main and one additional.
|
||||
|
@ -70,7 +70,7 @@ struct EditTagGroupView: View {
|
|||
focusedField = .title
|
||||
if let editingTagGroup {
|
||||
title = editingTagGroup.title
|
||||
sfSymbolName = editingTagGroup.sfSymbolName
|
||||
sfSymbolName = editingTagGroup.symbolName
|
||||
tags = editingTagGroup.tags
|
||||
}
|
||||
}
|
||||
|
@ -162,25 +162,20 @@ struct EditTagGroupView: View {
|
|||
}
|
||||
|
||||
private func save() {
|
||||
var toSave = tags
|
||||
let main = toSave.removeFirst()
|
||||
|
||||
let tagGroup: TagGroup = .init(
|
||||
title: title.trimmingCharacters(in: .whitespaces),
|
||||
sfSymbolName: sfSymbolName,
|
||||
main: main,
|
||||
additional: toSave
|
||||
)
|
||||
if let editingTagGroup,
|
||||
let index = preferences.tagGroups.firstIndex(of: editingTagGroup)
|
||||
{
|
||||
preferences.tagGroups[index] = tagGroup
|
||||
if let editingTagGroup {
|
||||
editingTagGroup.title = title
|
||||
editingTagGroup.symbolName = sfSymbolName
|
||||
editingTagGroup.tags = tags
|
||||
onSaved?(editingTagGroup)
|
||||
} else {
|
||||
preferences.tagGroups.append(tagGroup)
|
||||
let tagGroup = TagGroup(title: title.trimmingCharacters(in: .whitespacesAndNewlines),
|
||||
symbolName: sfSymbolName,
|
||||
tags: tags)
|
||||
context.insert(tagGroup)
|
||||
onSaved?(tagGroup)
|
||||
}
|
||||
|
||||
dismiss()
|
||||
onSaved?(tagGroup)
|
||||
}
|
||||
|
||||
@ViewBuilder
|
||||
|
|
|
@ -22,12 +22,15 @@ struct TimelineTab: View {
|
|||
|
||||
@State private var didAppear: Bool = false
|
||||
@State private var timeline: TimelineFilter = .home
|
||||
@State private var selectedTagGroup: TagGroup?
|
||||
@State private var scrollToTopSignal: Int = 0
|
||||
|
||||
@Query(sort: \LocalTimeline.creationDate, order: .reverse) var localTimelines: [LocalTimeline]
|
||||
|
||||
@Query(sort: \TagGroup.creationDate, order: .reverse) var tagGroups: [TagGroup]
|
||||
@AppStorage("remote_local_timeline") var legacyLocalTimelines: [String] = []
|
||||
@AppStorage("last_timeline_filter") var lastTimelineFilter: TimelineFilter = .home
|
||||
@AppStorage("tag_groups") var legacyTagGroups: [LegacyTagGroup] = []
|
||||
|
||||
@AppStorage("last_timeline_filter") var lastTimelineFilter: TimelineFilter = .home
|
||||
|
||||
private let canFilterTimeline: Bool
|
||||
|
||||
|
@ -39,7 +42,10 @@ struct TimelineTab: View {
|
|||
|
||||
var body: some View {
|
||||
NavigationStack(path: $routerPath.path) {
|
||||
TimelineView(timeline: $timeline, scrollToTopSignal: $scrollToTopSignal, canFilterTimeline: canFilterTimeline)
|
||||
TimelineView(timeline: $timeline,
|
||||
selectedTagGroup: $selectedTagGroup,
|
||||
scrollToTopSignal: $scrollToTopSignal,
|
||||
canFilterTimeline: canFilterTimeline)
|
||||
.withAppRouter()
|
||||
.withSheetDestinations(sheetDestinations: $routerPath.presentedSheet)
|
||||
.toolbar {
|
||||
|
@ -50,6 +56,7 @@ struct TimelineTab: View {
|
|||
}
|
||||
.onAppear {
|
||||
migrateUserPreferencesTimeline()
|
||||
migrateUserPreferencesTagGroups()
|
||||
routerPath.client = client
|
||||
if !didAppear, canFilterTimeline {
|
||||
didAppear = true
|
||||
|
@ -153,12 +160,13 @@ struct TimelineTab: View {
|
|||
}
|
||||
|
||||
Menu("timeline.filter.tag-groups") {
|
||||
ForEach(preferences.tagGroups, id: \.self) { group in
|
||||
ForEach(tagGroups) { group in
|
||||
Button {
|
||||
timeline = .tagGroup(group)
|
||||
selectedTagGroup = group
|
||||
timeline = .tagGroup(title: group.title, tags: group.tags)
|
||||
} label: {
|
||||
VStack {
|
||||
let icon = group.sfSymbolName.isEmpty ? "number" : group.sfSymbolName
|
||||
let icon = group.symbolName.isEmpty ? "number" : group.symbolName
|
||||
Label(group.title, systemImage: icon)
|
||||
}
|
||||
}
|
||||
|
@ -249,4 +257,11 @@ struct TimelineTab: View {
|
|||
}
|
||||
legacyLocalTimelines = []
|
||||
}
|
||||
|
||||
func migrateUserPreferencesTagGroups() {
|
||||
for group in legacyTagGroups {
|
||||
context.insert(TagGroup(title: group.title, symbolName: group.sfSymbolName, tags: group.tags))
|
||||
}
|
||||
legacyTagGroups = []
|
||||
}
|
||||
}
|
||||
|
|
|
@ -7,7 +7,6 @@ import SwiftUI
|
|||
@MainActor
|
||||
@Observable public class UserPreferences {
|
||||
class Storage {
|
||||
@AppStorage("tag_groups") public var tagGroups: [TagGroup] = []
|
||||
@AppStorage("preferred_browser") public var preferredBrowser: PreferredBrowser = .inAppSafari
|
||||
@AppStorage("show_translate_button_inline") public var showTranslateButton: Bool = true
|
||||
@AppStorage("is_open_ai_enabled") public var isOpenAIEnabled: Bool = true
|
||||
|
@ -62,11 +61,6 @@ import SwiftUI
|
|||
|
||||
private var client: Client?
|
||||
|
||||
public var tagGroups: [TagGroup] {
|
||||
didSet {
|
||||
storage.tagGroups = tagGroups
|
||||
}
|
||||
}
|
||||
public var preferredBrowser: PreferredBrowser {
|
||||
didSet {
|
||||
storage.preferredBrowser = preferredBrowser
|
||||
|
@ -365,7 +359,6 @@ import SwiftUI
|
|||
}
|
||||
|
||||
private init() {
|
||||
tagGroups = storage.tagGroups
|
||||
preferredBrowser = storage.preferredBrowser
|
||||
showTranslateButton = storage.showTranslateButton
|
||||
isOpenAIEnabled = storage.isOpenAIEnabled
|
||||
|
|
|
@ -0,0 +1,28 @@
|
|||
import SwiftData
|
||||
import SwiftUI
|
||||
import Foundation
|
||||
|
||||
@Model public class TagGroup: Equatable {
|
||||
public var title: String
|
||||
public var symbolName: String
|
||||
public var tags: [String]
|
||||
public var creationDate: Date
|
||||
|
||||
public init(title: String, symbolName: String, tags: [String]) {
|
||||
self.title = title
|
||||
self.symbolName = symbolName
|
||||
self.tags = tags
|
||||
self.creationDate = Date()
|
||||
}
|
||||
}
|
||||
|
||||
public struct LegacyTagGroup: Codable, Equatable, Hashable {
|
||||
public let title: String
|
||||
public let sfSymbolName: String
|
||||
public let main: String
|
||||
public let additional: [String]
|
||||
|
||||
public var tags: [String] {
|
||||
[main] + additional
|
||||
}
|
||||
}
|
|
@ -63,20 +63,3 @@ extension Tag: Sendable {}
|
|||
extension Tag.History: Sendable {}
|
||||
extension FeaturedTag: Sendable {}
|
||||
|
||||
public struct TagGroup: Codable, Equatable, Hashable {
|
||||
public init(title: String, sfSymbolName: String, main: String, additional: [String]) {
|
||||
self.title = title
|
||||
self.sfSymbolName = sfSymbolName
|
||||
self.main = main
|
||||
self.additional = additional
|
||||
}
|
||||
|
||||
public let title: String
|
||||
public let sfSymbolName: String
|
||||
public let main: String
|
||||
public let additional: [String]
|
||||
|
||||
public var tags: [String] {
|
||||
[main] + additional
|
||||
}
|
||||
}
|
||||
|
|
|
@ -32,7 +32,7 @@ public enum RemoteTimelineFilter: String, CaseIterable, Hashable, Equatable {
|
|||
public enum TimelineFilter: Hashable, Equatable {
|
||||
case home, local, federated, trending
|
||||
case hashtag(tag: String, accountId: String?)
|
||||
case tagGroup(TagGroup)
|
||||
case tagGroup(title: String, tags: [String])
|
||||
case list(list: Models.List)
|
||||
case remoteLocal(server: String, filter: RemoteTimelineFilter)
|
||||
case latest
|
||||
|
@ -73,8 +73,8 @@ public enum TimelineFilter: Hashable, Equatable {
|
|||
"Home"
|
||||
case let .hashtag(tag, _):
|
||||
"#\(tag)"
|
||||
case let .tagGroup(group):
|
||||
group.title
|
||||
case let .tagGroup(title, _):
|
||||
title
|
||||
case let .list(list):
|
||||
list.title
|
||||
case let .remoteLocal(server, _):
|
||||
|
@ -96,8 +96,8 @@ public enum TimelineFilter: Hashable, Equatable {
|
|||
"timeline.home"
|
||||
case let .hashtag(tag, _):
|
||||
"#\(tag)"
|
||||
case let .tagGroup(group):
|
||||
LocalizedStringKey(group.title) // ?? not sure since this can't be localized.
|
||||
case let .tagGroup(title, _):
|
||||
LocalizedStringKey(title) // ?? not sure since this can't be localized.
|
||||
case let .list(list):
|
||||
LocalizedStringKey(list.title)
|
||||
case let .remoteLocal(server, _):
|
||||
|
@ -128,29 +128,31 @@ public enum TimelineFilter: Hashable, Equatable {
|
|||
|
||||
public func endpoint(sinceId: String?, maxId: String?, minId: String?, offset: Int?) -> Endpoint {
|
||||
switch self {
|
||||
case .federated: Timelines.pub(sinceId: sinceId, maxId: maxId, minId: minId, local: false)
|
||||
case .local: Timelines.pub(sinceId: sinceId, maxId: maxId, minId: minId, local: true)
|
||||
case .federated: return Timelines.pub(sinceId: sinceId, maxId: maxId, minId: minId, local: false)
|
||||
case .local: return Timelines.pub(sinceId: sinceId, maxId: maxId, minId: minId, local: true)
|
||||
case let .remoteLocal(_, filter):
|
||||
switch filter {
|
||||
case .local:
|
||||
Timelines.pub(sinceId: sinceId, maxId: maxId, minId: minId, local: true)
|
||||
return Timelines.pub(sinceId: sinceId, maxId: maxId, minId: minId, local: true)
|
||||
case .federated:
|
||||
Timelines.pub(sinceId: sinceId, maxId: maxId, minId: minId, local: false)
|
||||
return Timelines.pub(sinceId: sinceId, maxId: maxId, minId: minId, local: false)
|
||||
case .trending:
|
||||
Trends.statuses(offset: offset)
|
||||
return Trends.statuses(offset: offset)
|
||||
}
|
||||
case .latest: Timelines.home(sinceId: nil, maxId: nil, minId: nil)
|
||||
case .home: Timelines.home(sinceId: sinceId, maxId: maxId, minId: minId)
|
||||
case .trending: Trends.statuses(offset: offset)
|
||||
case let .list(list): Timelines.list(listId: list.id, sinceId: sinceId, maxId: maxId, minId: minId)
|
||||
case .latest: return Timelines.home(sinceId: nil, maxId: nil, minId: nil)
|
||||
case .home: return Timelines.home(sinceId: sinceId, maxId: maxId, minId: minId)
|
||||
case .trending: return Trends.statuses(offset: offset)
|
||||
case let .list(list): return Timelines.list(listId: list.id, sinceId: sinceId, maxId: maxId, minId: minId)
|
||||
case let .hashtag(tag, accountId):
|
||||
if let accountId {
|
||||
Accounts.statuses(id: accountId, sinceId: nil, tag: tag, onlyMedia: nil, excludeReplies: nil, pinned: nil)
|
||||
return Accounts.statuses(id: accountId, sinceId: nil, tag: tag, onlyMedia: nil, excludeReplies: nil, pinned: nil)
|
||||
} else {
|
||||
Timelines.hashtag(tag: tag, additional: nil, maxId: maxId)
|
||||
return Timelines.hashtag(tag: tag, additional: nil, maxId: maxId)
|
||||
}
|
||||
case let .tagGroup(group):
|
||||
Timelines.hashtag(tag: group.main, additional: group.additional, maxId: maxId)
|
||||
case let .tagGroup(_, tags):
|
||||
var tags = tags
|
||||
tags.removeFirst()
|
||||
return Timelines.hashtag(tag: tags.first ?? "", additional: tags, maxId: maxId)
|
||||
}
|
||||
}
|
||||
}
|
||||
|
@ -189,8 +191,13 @@ extension TimelineFilter: Codable {
|
|||
accountId: accountId
|
||||
)
|
||||
case .tagGroup:
|
||||
let group = try container.decode(TagGroup.self, forKey: .tagGroup)
|
||||
self = .tagGroup(group)
|
||||
var nestedContainer = try container.nestedUnkeyedContainer(forKey: .hashtag)
|
||||
let title = try nestedContainer.decode(String.self)
|
||||
let tags = try nestedContainer.decode([String].self)
|
||||
self = .tagGroup(
|
||||
title: title,
|
||||
tags: tags
|
||||
)
|
||||
case .list:
|
||||
let list = try container.decode(
|
||||
Models.List.self,
|
||||
|
@ -232,8 +239,10 @@ extension TimelineFilter: Codable {
|
|||
var nestedContainer = container.nestedUnkeyedContainer(forKey: .hashtag)
|
||||
try nestedContainer.encode(tag)
|
||||
try nestedContainer.encode(accountId)
|
||||
case let .tagGroup(group):
|
||||
try container.encode(group, forKey: .tagGroup)
|
||||
case let .tagGroup(title, tags):
|
||||
var nestedContainer = container.nestedUnkeyedContainer(forKey: .tagGroup)
|
||||
try nestedContainer.encode(title)
|
||||
try nestedContainer.encode(tags)
|
||||
case let .list(list):
|
||||
try container.encode(list, forKey: .list)
|
||||
case let .remoteLocal(server, filter):
|
||||
|
|
|
@ -27,11 +27,15 @@ public struct TimelineView: View {
|
|||
@State private var collectionView: UICollectionView?
|
||||
|
||||
@Binding var timeline: TimelineFilter
|
||||
@Binding var selectedTagGroup: TagGroup?
|
||||
@Binding var scrollToTopSignal: Int
|
||||
private let canFilterTimeline: Bool
|
||||
|
||||
public init(timeline: Binding<TimelineFilter>, scrollToTopSignal: Binding<Int>, canFilterTimeline: Bool) {
|
||||
public init(timeline: Binding<TimelineFilter>,
|
||||
selectedTagGroup: Binding<TagGroup?>,
|
||||
scrollToTopSignal: Binding<Int>, canFilterTimeline: Bool) {
|
||||
_timeline = timeline
|
||||
_selectedTagGroup = selectedTagGroup
|
||||
_scrollToTopSignal = scrollToTopSignal
|
||||
self.canFilterTimeline = canFilterTimeline
|
||||
}
|
||||
|
@ -40,13 +44,9 @@ public struct TimelineView: View {
|
|||
ScrollViewReader { proxy in
|
||||
ZStack(alignment: .top) {
|
||||
List {
|
||||
if viewModel.tagGroup != nil {
|
||||
tagGroupHeaderView
|
||||
} else if viewModel.tag == nil {
|
||||
scrollToTopView
|
||||
} else {
|
||||
tagHeaderView
|
||||
}
|
||||
scrollToTopView
|
||||
tagGroupHeaderView
|
||||
tagHeaderView
|
||||
switch viewModel.timeline {
|
||||
case .remoteLocal:
|
||||
StatusesListView(fetcher: viewModel, client: client, routerPath: routerPath, isRemote: true)
|
||||
|
@ -211,7 +211,7 @@ public struct TimelineView: View {
|
|||
|
||||
@ViewBuilder
|
||||
private var tagGroupHeaderView: some View {
|
||||
if let group = viewModel.tagGroup {
|
||||
if let group = selectedTagGroup {
|
||||
headerView {
|
||||
HStack {
|
||||
ScrollView(.horizontal) {
|
||||
|
@ -230,7 +230,7 @@ public struct TimelineView: View {
|
|||
.scrollIndicators(.hidden)
|
||||
Button("status.action.edit") {
|
||||
routerPath.presentedSheet = .editTagGroup(tagGroup: group, onSaved: { group in
|
||||
viewModel.timeline = .tagGroup(group)
|
||||
viewModel.timeline = .tagGroup(title: group.title, tags: group.tags)
|
||||
})
|
||||
}
|
||||
.buttonStyle(.bordered)
|
||||
|
|
|
@ -42,13 +42,6 @@ import SwiftUI
|
|||
|
||||
var tag: Tag?
|
||||
|
||||
var tagGroup: TagGroup? {
|
||||
if case let .tagGroup(group) = timeline {
|
||||
return group
|
||||
}
|
||||
return nil
|
||||
}
|
||||
|
||||
// Internal source of truth for a timeline.
|
||||
private var datasource = TimelineDatasource()
|
||||
private let cache = TimelineCache()
|
||||
|
|
Loading…
Reference in New Issue