Browse Source

wot: show friend icons is some views

easier to detect if someone is trying to fake us out

Changelog-Added: Friend icons next to names on some views. Check is friend. Arrows are friend-of-friends
Signed-off-by: William Casarin <jb55@jb55.com>
profile-edit
William Casarin 2 years ago
parent
commit
22cad4b072
  1. 4
      damus/ContentView.swift
  2. 2
      damus/Views/ChatView.swift
  3. 2
      damus/Views/DMChatView.swift
  4. 2
      damus/Views/DirectMessagesView.swift
  5. 2
      damus/Views/EventDetailView.swift
  6. 14
      damus/Views/EventView.swift
  7. 2
      damus/Views/FollowingView.swift
  8. 37
      damus/Views/ProfileName.swift
  9. 9
      damus/Views/ProfileView.swift
  10. 2
      damus/Views/ReplyView.swift
  11. 2
      damus/Views/SearchHomeView.swift
  12. 2
      damus/Views/SearchView.swift
  13. 6
      damus/Views/TimelineView.swift

4
damus/ContentView.swift

@ -90,7 +90,7 @@ struct ContentView: View {
var PostingTimelineView: some View { var PostingTimelineView: some View {
ZStack { ZStack {
if let damus = self.damus_state { if let damus = self.damus_state {
TimelineView(events: $home.events, loading: $home.loading, damus: damus) TimelineView(events: $home.events, loading: $home.loading, damus: damus, show_friend_icon: false)
} }
if privkey != nil { if privkey != nil {
PostButtonContainer { PostButtonContainer {
@ -119,7 +119,7 @@ struct ContentView: View {
PostingTimelineView PostingTimelineView
case .notifications: case .notifications:
TimelineView(events: $home.notifications, loading: $home.loading, damus: damus) TimelineView(events: $home.notifications, loading: $home.loading, damus: damus, show_friend_icon: true)
.navigationTitle("Notifications") .navigationTitle("Notifications")
case .dms: case .dms:

2
damus/Views/ChatView.swift

@ -85,7 +85,7 @@ struct ChatView: View {
VStack(alignment: .leading) { VStack(alignment: .leading) {
if just_started { if just_started {
HStack { HStack {
ProfileName(pubkey: event.pubkey, profile: damus_state.profiles.lookup(id: event.pubkey)) ProfileName(pubkey: event.pubkey, profile: damus_state.profiles.lookup(id: event.pubkey), contacts: damus_state.contacts, show_friend_confirmed: true)
.foregroundColor(colorScheme == .dark ? id_to_color(event.pubkey) : Color.black) .foregroundColor(colorScheme == .dark ? id_to_color(event.pubkey) : Color.black)
//.shadow(color: Color.black, radius: 2) //.shadow(color: Color.black, radius: 2)
Text("\(format_relative_time(event.created_at))") Text("\(format_relative_time(event.created_at))")

2
damus/Views/DMChatView.swift

@ -39,7 +39,7 @@ struct DMChatView: View {
HStack { HStack {
ProfilePicView(pubkey: pubkey, size: 24, highlight: .none, image_cache: damus_state.image_cache, profiles: damus_state.profiles) ProfilePicView(pubkey: pubkey, size: 24, highlight: .none, image_cache: damus_state.image_cache, profiles: damus_state.profiles)
ProfileName(pubkey: pubkey, profile: profile) ProfileName(pubkey: pubkey, profile: profile, contacts: damus_state.contacts, show_friend_confirmed: true)
} }
} }
.buttonStyle(PlainButtonStyle()) .buttonStyle(PlainButtonStyle())

2
damus/Views/DirectMessagesView.swift

@ -27,7 +27,7 @@ struct DirectMessagesView: View {
let chat = DMChatView(damus_state: damus_state, pubkey: tup.0) let chat = DMChatView(damus_state: damus_state, pubkey: tup.0)
.environmentObject(tup.1) .environmentObject(tup.1)
NavigationLink(destination: chat) { NavigationLink(destination: chat) {
EventView(damus: damus_state, event: ev, pubkey: tup.0) EventView(damus: damus_state, event: ev, pubkey: tup.0, show_friend_icon: true)
} }
.buttonStyle(PlainButtonStyle()) .buttonStyle(PlainButtonStyle())
} else { } else {

2
damus/Views/EventDetailView.swift

@ -71,7 +71,7 @@ struct EventDetailView: View {
toggle_thread_view() toggle_thread_view()
} }
case .event(let ev, let highlight): case .event(let ev, let highlight):
EventView(event: ev, highlight: highlight, has_action_bar: true, damus: damus) EventView(event: ev, highlight: highlight, has_action_bar: true, damus: damus, show_friend_icon: true)
.onTapGesture { .onTapGesture {
if thread.initial_event.id == ev.id { if thread.initial_event.id == ev.id {
toggle_thread_view() toggle_thread_view()

14
damus/Views/EventView.swift

@ -42,31 +42,35 @@ struct EventView: View {
let has_action_bar: Bool let has_action_bar: Bool
let damus: DamusState let damus: DamusState
let pubkey: String let pubkey: String
let show_friend_icon: Bool
@EnvironmentObject var action_bar: ActionBarModel @EnvironmentObject var action_bar: ActionBarModel
init(event: NostrEvent, highlight: Highlight, has_action_bar: Bool, damus: DamusState) { init(event: NostrEvent, highlight: Highlight, has_action_bar: Bool, damus: DamusState, show_friend_icon: Bool) {
self.event = event self.event = event
self.highlight = highlight self.highlight = highlight
self.has_action_bar = has_action_bar self.has_action_bar = has_action_bar
self.damus = damus self.damus = damus
self.pubkey = event.pubkey self.pubkey = event.pubkey
self.show_friend_icon = show_friend_icon
} }
init(damus: DamusState, event: NostrEvent) { init(damus: DamusState, event: NostrEvent, show_friend_icon: Bool) {
self.event = event self.event = event
self.highlight = .none self.highlight = .none
self.has_action_bar = false self.has_action_bar = false
self.damus = damus self.damus = damus
self.pubkey = event.pubkey self.pubkey = event.pubkey
self.show_friend_icon = show_friend_icon
} }
init(damus: DamusState, event: NostrEvent, pubkey: String) { init(damus: DamusState, event: NostrEvent, pubkey: String, show_friend_icon: Bool) {
self.event = event self.event = event
self.highlight = .none self.highlight = .none
self.has_action_bar = false self.has_action_bar = false
self.damus = damus self.damus = damus
self.pubkey = pubkey self.pubkey = pubkey
self.show_friend_icon = show_friend_icon
} }
var body: some View { var body: some View {
@ -81,7 +85,7 @@ struct EventView: View {
HStack { HStack {
Label("", systemImage: "arrow.2.squarepath") Label("", systemImage: "arrow.2.squarepath")
.foregroundColor(Color.gray) .foregroundColor(Color.gray)
ProfileName(pubkey: event.pubkey, profile: damus.profiles.lookup(id: event.pubkey)) ProfileName(pubkey: event.pubkey, profile: damus.profiles.lookup(id: event.pubkey), contacts: damus.contacts, show_friend_confirmed: show_friend_icon)
.foregroundColor(Color.gray) .foregroundColor(Color.gray)
Text(" Boosted") Text(" Boosted")
.foregroundColor(Color.gray) .foregroundColor(Color.gray)
@ -113,7 +117,7 @@ struct EventView: View {
VStack(alignment: .leading) { VStack(alignment: .leading) {
HStack(alignment: .center) { HStack(alignment: .center) {
ProfileName(pubkey: pubkey, profile: profile) ProfileName(pubkey: pubkey, profile: profile, contacts: damus.contacts, show_friend_confirmed: show_friend_icon)
Text("\(format_relative_time(event.created_at))") Text("\(format_relative_time(event.created_at))")
.foregroundColor(.gray) .foregroundColor(.gray)
} }

2
damus/Views/FollowingView.swift

@ -22,7 +22,7 @@ struct FollowUserView: View {
VStack(alignment: .leading) { VStack(alignment: .leading) {
let profile = damus_state.profiles.lookup(id: target.pubkey) let profile = damus_state.profiles.lookup(id: target.pubkey)
ProfileName(pubkey: target.pubkey, profile: profile) ProfileName(pubkey: target.pubkey, profile: profile, contacts: damus_state.contacts, show_friend_confirmed: false)
if let about = profile.flatMap { $0.about } { if let about = profile.flatMap { $0.about } {
Text(about) Text(about)
} }

37
damus/Views/ProfileName.swift

@ -10,6 +10,7 @@ import SwiftUI
struct ProfileFullName: View { struct ProfileFullName: View {
let pubkey: String let pubkey: String
let profile: Profile? let profile: Profile?
let contacts: Contacts
@State var display_name: String? @State var display_name: String?
@ -18,11 +19,11 @@ struct ProfileFullName: View {
if let real_name = profile?.display_name { if let real_name = profile?.display_name {
Text(real_name) Text(real_name)
.bold() .bold()
ProfileName(pubkey: pubkey, profile: profile, prefix: "@") ProfileName(pubkey: pubkey, profile: profile, prefix: "@", contacts: contacts, show_friend_confirmed: true)
.font(.footnote) .font(.footnote)
.foregroundColor(.gray) .foregroundColor(.gray)
} else { } else {
ProfileName(pubkey: pubkey, profile: profile) ProfileName(pubkey: pubkey, profile: profile, contacts: contacts, show_friend_confirmed: true)
} }
} }
} }
@ -31,20 +32,43 @@ struct ProfileFullName: View {
struct ProfileName: View { struct ProfileName: View {
let pubkey: String let pubkey: String
let profile: Profile? let profile: Profile?
let contacts: Contacts
let prefix: String let prefix: String
let show_friend_confirmed: Bool
@State var display_name: String? @State var display_name: String?
init(pubkey: String, profile: Profile?) { init(pubkey: String, profile: Profile?, contacts: Contacts, show_friend_confirmed: Bool) {
self.pubkey = pubkey self.pubkey = pubkey
self.profile = profile self.profile = profile
self.prefix = "" self.prefix = ""
self.contacts = contacts
self.show_friend_confirmed = show_friend_confirmed
} }
init(pubkey: String, profile: Profile?, prefix: String) { init(pubkey: String, profile: Profile?, prefix: String, contacts: Contacts, show_friend_confirmed: Bool) {
self.pubkey = pubkey self.pubkey = pubkey
self.profile = profile self.profile = profile
self.prefix = prefix self.prefix = prefix
self.contacts = contacts
self.show_friend_confirmed = show_friend_confirmed
}
var friend_icon: String? {
if !show_friend_confirmed {
return nil
}
if self.contacts.is_friend(self.pubkey) {
return "person.fill.checkmark"
}
if self.contacts.is_friend_of_friend(self.pubkey) {
return "person.fill.and.arrow.left.and.arrow.right"
}
return nil
} }
var body: some View { var body: some View {
@ -52,6 +76,11 @@ struct ProfileName: View {
Text(prefix + String(display_name ?? Profile.displayName(profile: profile, pubkey: pubkey))) Text(prefix + String(display_name ?? Profile.displayName(profile: profile, pubkey: pubkey)))
//.foregroundColor(hex_to_rgb(pubkey)) //.foregroundColor(hex_to_rgb(pubkey))
.fontWeight(prefix == "@" ? .none : .bold) .fontWeight(prefix == "@" ? .none : .bold)
if let frend = friend_icon {
Label("", systemImage: frend)
.foregroundColor(.gray)
.font(.footnote)
}
} }
.onReceive(handle_notify(.profile_updated)) { notif in .onReceive(handle_notify(.profile_updated)) { notif in
let update = notif.object as! ProfileUpdate let update = notif.object as! ProfileUpdate

9
damus/Views/ProfileView.swift

@ -48,6 +48,7 @@ func follow_btn_enabled_state(_ fs: FollowState) -> Bool {
struct ProfileNameView: View { struct ProfileNameView: View {
let pubkey: String let pubkey: String
let profile: Profile? let profile: Profile?
let contacts: Contacts
var body: some View { var body: some View {
Group { Group {
@ -55,12 +56,12 @@ struct ProfileNameView: View {
VStack(alignment: .leading) { VStack(alignment: .leading) {
Text(real_name) Text(real_name)
.font(.title) .font(.title)
ProfileName(pubkey: pubkey, profile: profile, prefix: "@") ProfileName(pubkey: pubkey, profile: profile, prefix: "@", contacts: contacts, show_friend_confirmed: true)
.font(.callout) .font(.callout)
.foregroundColor(.gray) .foregroundColor(.gray)
} }
} else { } else {
ProfileName(pubkey: pubkey, profile: profile) ProfileName(pubkey: pubkey, profile: profile, contacts: contacts, show_friend_confirmed: true)
} }
} }
} }
@ -94,7 +95,7 @@ struct ProfileView: View {
HStack(alignment: .center) { HStack(alignment: .center) {
ProfilePicView(pubkey: profile.pubkey, size: PFP_SIZE, highlight: .custom(Color.black, 2), image_cache: damus_state.image_cache, profiles: damus_state.profiles) ProfilePicView(pubkey: profile.pubkey, size: PFP_SIZE, highlight: .custom(Color.black, 2), image_cache: damus_state.image_cache, profiles: damus_state.profiles)
ProfileNameView(pubkey: profile.pubkey, profile: data) ProfileNameView(pubkey: profile.pubkey, profile: data, contacts: damus_state.contacts)
Spacer() Spacer()
@ -146,7 +147,7 @@ struct ProfileView: View {
Divider() Divider()
InnerTimelineView(events: $profile.events, damus: damus_state) InnerTimelineView(events: $profile.events, damus: damus_state, show_friend_icon: false)
} }
.frame(maxHeight: .infinity, alignment: .topLeading) .frame(maxHeight: .infinity, alignment: .topLeading)
} }

2
damus/Views/ReplyView.swift

@ -33,7 +33,7 @@ struct ReplyView: View {
.foregroundColor(.gray) .foregroundColor(.gray)
.font(.footnote) .font(.footnote)
} }
EventView(event: replying_to, highlight: .none, has_action_bar: false, damus: damus) EventView(event: replying_to, highlight: .none, has_action_bar: false, damus: damus, show_friend_icon: true)
PostView(references: gather_reply_ids(our_pubkey: damus.pubkey, from: replying_to)) PostView(references: gather_reply_ids(our_pubkey: damus.pubkey, from: replying_to))
} }
.padding() .padding()

2
damus/Views/SearchHomeView.swift

@ -39,7 +39,7 @@ struct SearchHomeView: View {
} }
var GlobalContent: some View { var GlobalContent: some View {
TimelineView(events: $model.events, loading: $model.loading, damus: damus_state) TimelineView(events: $model.events, loading: $model.loading, damus: damus_state, show_friend_icon: true)
} }
var SearchContent: some View { var SearchContent: some View {

2
damus/Views/SearchView.swift

@ -13,7 +13,7 @@ struct SearchView: View {
@Environment(\.dismiss) var dismiss @Environment(\.dismiss) var dismiss
var body: some View { var body: some View {
TimelineView(events: $search.events, loading: $search.loading, damus: appstate) TimelineView(events: $search.events, loading: $search.loading, damus: appstate, show_friend_icon: true)
.navigationBarTitle(describe_search(search.search)) .navigationBarTitle(describe_search(search.search))
.padding([.leading, .trailing], 6) .padding([.leading, .trailing], 6)
.onReceive(handle_notify(.switched_timeline)) { obj in .onReceive(handle_notify(.switched_timeline)) { obj in

6
damus/Views/TimelineView.swift

@ -15,6 +15,7 @@ enum TimelineAction {
struct InnerTimelineView: View { struct InnerTimelineView: View {
@Binding var events: [NostrEvent] @Binding var events: [NostrEvent]
let damus: DamusState let damus: DamusState
let show_friend_icon: Bool
var body: some View { var body: some View {
LazyVStack { LazyVStack {
@ -22,7 +23,7 @@ struct InnerTimelineView: View {
let tv = ThreadView(thread: ThreadModel(event: ev, pool: damus.pool, privkey: damus.keypair.privkey), damus: damus, is_chatroom: has_hashtag(ev.tags, hashtag: "chat")) let tv = ThreadView(thread: ThreadModel(event: ev, pool: damus.pool, privkey: damus.keypair.privkey), damus: damus, is_chatroom: has_hashtag(ev.tags, hashtag: "chat"))
NavigationLink(destination: tv) { NavigationLink(destination: tv) {
EventView(event: ev, highlight: .none, has_action_bar: true, damus: damus) EventView(event: ev, highlight: .none, has_action_bar: true, damus: damus, show_friend_icon: show_friend_icon)
} }
.isDetailLink(true) .isDetailLink(true)
.buttonStyle(PlainButtonStyle()) .buttonStyle(PlainButtonStyle())
@ -36,6 +37,7 @@ struct TimelineView: View {
@Binding var loading: Bool @Binding var loading: Bool
let damus: DamusState let damus: DamusState
let show_friend_icon: Bool
var body: some View { var body: some View {
MainContent MainContent
@ -48,7 +50,7 @@ struct TimelineView: View {
ProgressView() ProgressView()
.progressViewStyle(.circular) .progressViewStyle(.circular)
} }
InnerTimelineView(events: $events, damus: damus) InnerTimelineView(events: $events, damus: damus, show_friend_icon: show_friend_icon)
} }
.onReceive(NotificationCenter.default.publisher(for: .scroll_to_top)) { _ in .onReceive(NotificationCenter.default.publisher(for: .scroll_to_top)) { _ in
guard let event = events.first else { guard let event = events.first else {

Loading…
Cancel
Save