tabs: add blue dot to home view
Changelog-Added: Add blue dot notification to home tab Changelog-Fixed: Clicking tabs now clear blue dots immediately Signed-off-by: William Casarin <jb55@jb55.com>
This commit is contained in:
@@ -174,7 +174,7 @@ struct ContentView: View {
|
|||||||
.navigationViewStyle(.stack)
|
.navigationViewStyle(.stack)
|
||||||
}
|
}
|
||||||
|
|
||||||
TabBar(new_notifications: $home.new_notifications, selected: $selected_timeline, action: switch_timeline)
|
TabBar(new_events: $home.new_events, selected: $selected_timeline, action: switch_timeline)
|
||||||
}
|
}
|
||||||
.onAppear() {
|
.onAppear() {
|
||||||
self.connect()
|
self.connect()
|
||||||
@@ -305,9 +305,6 @@ struct ContentView: View {
|
|||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
if (timeline != .notifications && self.selected_timeline == .notifications) || timeline == .notifications {
|
|
||||||
home.new_notifications = false
|
|
||||||
}
|
|
||||||
self.selected_timeline = timeline
|
self.selected_timeline = timeline
|
||||||
NotificationCenter.default.post(name: .switched_timeline, object: timeline)
|
NotificationCenter.default.post(name: .switched_timeline, object: timeline)
|
||||||
//self.selected_timeline = timeline
|
//self.selected_timeline = timeline
|
||||||
@@ -413,9 +410,10 @@ struct LastNotification {
|
|||||||
let created_at: Int64
|
let created_at: Int64
|
||||||
}
|
}
|
||||||
|
|
||||||
func get_last_notified() -> LastNotification? {
|
func get_last_event(_ timeline: Timeline) -> LastNotification? {
|
||||||
let last = UserDefaults.standard.string(forKey: "last_notification")
|
let str = timeline.rawValue
|
||||||
let last_created = UserDefaults.standard.string(forKey: "last_notification_time")
|
let last = UserDefaults.standard.string(forKey: "last_\(str)")
|
||||||
|
let last_created = UserDefaults.standard.string(forKey: "last_\(str)_time")
|
||||||
.flatMap { Int64($0) }
|
.flatMap { Int64($0) }
|
||||||
|
|
||||||
return last.flatMap { id in
|
return last.flatMap { id in
|
||||||
@@ -425,9 +423,10 @@ func get_last_notified() -> LastNotification? {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func save_last_notified(_ ev: NostrEvent) {
|
func save_last_event(_ ev: NostrEvent, timeline: Timeline) {
|
||||||
UserDefaults.standard.set(ev.id, forKey: "last_notification")
|
let str = timeline.rawValue
|
||||||
UserDefaults.standard.set(String(ev.created_at), forKey: "last_notification_time")
|
UserDefaults.standard.set(ev.id, forKey: "last_\(str)")
|
||||||
|
UserDefaults.standard.set(String(ev.created_at), forKey: "last_\(str)_time")
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|||||||
@@ -7,6 +7,27 @@
|
|||||||
|
|
||||||
import Foundation
|
import Foundation
|
||||||
|
|
||||||
|
struct NewEventsBits {
|
||||||
|
let bits: Int
|
||||||
|
|
||||||
|
init() {
|
||||||
|
bits = 0
|
||||||
|
}
|
||||||
|
|
||||||
|
init (prev: NewEventsBits, setting: Timeline) {
|
||||||
|
self.bits = prev.bits | timeline_bit(setting)
|
||||||
|
}
|
||||||
|
|
||||||
|
init (prev: NewEventsBits, unsetting: Timeline) {
|
||||||
|
self.bits = prev.bits & ~timeline_bit(unsetting)
|
||||||
|
}
|
||||||
|
|
||||||
|
func is_set(_ timeline: Timeline) -> Bool {
|
||||||
|
let notification_bit = timeline_bit(timeline)
|
||||||
|
return (bits & notification_bit) == notification_bit
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
class HomeModel: ObservableObject {
|
class HomeModel: ObservableObject {
|
||||||
var damus_state: DamusState
|
var damus_state: DamusState
|
||||||
@@ -18,9 +39,10 @@ class HomeModel: ObservableObject {
|
|||||||
let home_subid = UUID().description
|
let home_subid = UUID().description
|
||||||
let contacts_subid = UUID().description
|
let contacts_subid = UUID().description
|
||||||
let notifications_subid = UUID().description
|
let notifications_subid = UUID().description
|
||||||
|
let dms_subid = UUID().description
|
||||||
let init_subid = UUID().description
|
let init_subid = UUID().description
|
||||||
|
|
||||||
@Published var new_notifications: Bool = false
|
@Published var new_events: NewEventsBits = NewEventsBits()
|
||||||
@Published var notifications: [NostrEvent] = []
|
@Published var notifications: [NostrEvent] = []
|
||||||
@Published var events: [NostrEvent] = []
|
@Published var events: [NostrEvent] = []
|
||||||
@Published var loading: Bool = false
|
@Published var loading: Bool = false
|
||||||
@@ -261,21 +283,28 @@ class HomeModel: ObservableObject {
|
|||||||
return m[kind]
|
return m[kind]
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func handle_last_event(ev: NostrEvent, timeline: Timeline) {
|
||||||
|
let last_ev = get_last_event(timeline)
|
||||||
|
|
||||||
|
if last_ev == nil || last_ev!.created_at < ev.created_at {
|
||||||
|
save_last_event(ev, timeline: timeline)
|
||||||
|
new_events = NewEventsBits(prev: new_events, setting: timeline)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
func handle_notification(ev: NostrEvent) {
|
func handle_notification(ev: NostrEvent) {
|
||||||
if !insert_uniq_sorted_event(events: ¬ifications, new_ev: ev, cmp: { $0.created_at > $1.created_at }) {
|
if !insert_uniq_sorted_event(events: ¬ifications, new_ev: ev, cmp: { $0.created_at > $1.created_at }) {
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
let last_notified = get_last_notified()
|
handle_last_event(ev: ev, timeline: .notifications)
|
||||||
|
|
||||||
if last_notified == nil || last_notified!.created_at < ev.created_at {
|
|
||||||
save_last_notified(ev)
|
|
||||||
new_notifications = true
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
func insert_home_event(_ ev: NostrEvent) -> Bool {
|
func insert_home_event(_ ev: NostrEvent) -> Bool {
|
||||||
let ok = insert_uniq_sorted_event(events: &self.events, new_ev: ev, cmp: { $0.created_at > $1.created_at })
|
let ok = insert_uniq_sorted_event(events: &self.events, new_ev: ev, cmp: { $0.created_at > $1.created_at })
|
||||||
|
if ok {
|
||||||
|
handle_last_event(ev: ev, timeline: .home)
|
||||||
|
}
|
||||||
return ok
|
return ok
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -17,38 +17,12 @@ enum Timeline: String, CustomStringConvertible {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func timeline_bit(_ timeline: Timeline) -> Int {
|
||||||
struct MainTabView: View {
|
switch timeline {
|
||||||
var body: some View {
|
case .home: return 1 << 0
|
||||||
Text(/*@START_MENU_TOKEN@*/"Hello, World!"/*@END_MENU_TOKEN@*/)
|
case .notifications: return 1 << 1
|
||||||
}
|
case .search: return 1 << 2
|
||||||
}
|
case .dms: return 1 << 3
|
||||||
|
|
||||||
struct MainTabView_Previews: PreviewProvider {
|
|
||||||
static var previews: some View {
|
|
||||||
MainTabView()
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
struct NotificationsTab: View {
|
|
||||||
@Binding var new_notifications: Bool
|
|
||||||
@Binding var selected: Timeline?
|
|
||||||
|
|
||||||
let action: (Timeline) -> ()
|
|
||||||
|
|
||||||
var body: some View {
|
|
||||||
ZStack(alignment: .center) {
|
|
||||||
TabButton(timeline: .notifications, img: "bell", selected: $selected, action: action)
|
|
||||||
|
|
||||||
if new_notifications {
|
|
||||||
Circle()
|
|
||||||
.size(CGSize(width: 8, height: 8))
|
|
||||||
.frame(width: 10, height: 10, alignment: .topTrailing)
|
|
||||||
.alignmentGuide(VerticalAlignment.center) { a in a.height + 2.0 }
|
|
||||||
.alignmentGuide(HorizontalAlignment.center) { a in a.width - 12.0 }
|
|
||||||
.foregroundColor(.accentColor)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -58,11 +32,30 @@ struct TabButton: View {
|
|||||||
let img: String
|
let img: String
|
||||||
|
|
||||||
@Binding var selected: Timeline?
|
@Binding var selected: Timeline?
|
||||||
|
@Binding var new_events: NewEventsBits
|
||||||
|
|
||||||
let action: (Timeline) -> ()
|
let action: (Timeline) -> ()
|
||||||
|
|
||||||
var body: some View {
|
var body: some View {
|
||||||
Button(action: {action(timeline)}) {
|
ZStack(alignment: .center) {
|
||||||
|
Tab
|
||||||
|
|
||||||
|
if new_events.is_set(timeline) {
|
||||||
|
Circle()
|
||||||
|
.size(CGSize(width: 8, height: 8))
|
||||||
|
.frame(width: 10, height: 10, alignment: .topTrailing)
|
||||||
|
.alignmentGuide(VerticalAlignment.center) { a in a.height + 2.0 }
|
||||||
|
.alignmentGuide(HorizontalAlignment.center) { a in a.width - 12.0 }
|
||||||
|
.foregroundColor(.accentColor)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
var Tab: some View {
|
||||||
|
Button(action: {
|
||||||
|
action(timeline)
|
||||||
|
new_events = NewEventsBits(prev: new_events, unsetting: timeline)
|
||||||
|
}) {
|
||||||
Label("", systemImage: selected == timeline ? "\(img).fill" : img)
|
Label("", systemImage: selected == timeline ? "\(img).fill" : img)
|
||||||
.contentShape(Rectangle())
|
.contentShape(Rectangle())
|
||||||
.frame(maxWidth: .infinity, minHeight: 30.0)
|
.frame(maxWidth: .infinity, minHeight: 30.0)
|
||||||
@@ -73,7 +66,7 @@ struct TabButton: View {
|
|||||||
|
|
||||||
|
|
||||||
struct TabBar: View {
|
struct TabBar: View {
|
||||||
@Binding var new_notifications: Bool
|
@Binding var new_events: NewEventsBits
|
||||||
@Binding var selected: Timeline?
|
@Binding var selected: Timeline?
|
||||||
|
|
||||||
let action: (Timeline) -> ()
|
let action: (Timeline) -> ()
|
||||||
@@ -82,9 +75,9 @@ struct TabBar: View {
|
|||||||
VStack {
|
VStack {
|
||||||
Divider()
|
Divider()
|
||||||
HStack {
|
HStack {
|
||||||
TabButton(timeline: .home, img: "house", selected: $selected, action: action)
|
TabButton(timeline: .home, img: "house", selected: $selected, new_events: $new_events, action: action)
|
||||||
TabButton(timeline: .search, img: "magnifyingglass.circle", selected: $selected, action: action)
|
TabButton(timeline: .search, img: "magnifyingglass.circle", selected: $selected, new_events: $new_events, action: action)
|
||||||
NotificationsTab(new_notifications: $new_notifications, selected: $selected, action: action)
|
TabButton(timeline: .notifications, img: "bell", selected: $selected, new_events: $new_events, action: action)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
Reference in New Issue
Block a user