@@ -43,15 +43,18 @@ struct ContentView: View {
|
|||||||
@State var timeline: Timeline = .friends
|
@State var timeline: Timeline = .friends
|
||||||
@State var pool: RelayPool? = nil
|
@State var pool: RelayPool? = nil
|
||||||
|
|
||||||
|
let sub_id = UUID().description
|
||||||
let pubkey = "32e1827635450ebb3c5a7d12c1f8e7b2b514439ac10a67eef3d9fd9c5c68e245"
|
let pubkey = "32e1827635450ebb3c5a7d12c1f8e7b2b514439ac10a67eef3d9fd9c5c68e245"
|
||||||
|
|
||||||
var MainContent: some View {
|
func MainContent(pool: RelayPool) -> some View {
|
||||||
ScrollView {
|
ScrollView {
|
||||||
ForEach(events, id: \.id) { ev in
|
ForEach(self.events, id: \.id) { (ev: NostrEvent) in
|
||||||
if ev.is_local && timeline == .debug || (timeline == .global && !ev.is_local) || (timeline == .friends && is_friend(ev.pubkey)) {
|
if ev.is_local && timeline == .debug || (timeline == .global && !ev.is_local) || (timeline == .friends && is_friend(ev.pubkey)) {
|
||||||
let profile: Profile? = profiles[ev.pubkey]?.profile
|
let profile: Profile? = profiles[ev.pubkey]?.profile
|
||||||
NavigationLink(destination: EventDetailView(event: ev, profile: profile).navigationBarTitle("Note")) {
|
let evdet = EventDetailView(event: ev, pool: pool, profiles: profiles)
|
||||||
EventView(event: ev, profile: profile)
|
.navigationBarTitle("Note")
|
||||||
|
NavigationLink(destination: evdet) {
|
||||||
|
EventView(event: ev, profile: profile, highlighted: false)
|
||||||
}
|
}
|
||||||
.buttonStyle(PlainButtonStyle())
|
.buttonStyle(PlainButtonStyle())
|
||||||
}
|
}
|
||||||
@@ -93,8 +96,10 @@ struct ContentView: View {
|
|||||||
VStack {
|
VStack {
|
||||||
TopBar(selected: self.timeline)
|
TopBar(selected: self.timeline)
|
||||||
ZStack {
|
ZStack {
|
||||||
MainContent
|
if let pool = self.pool {
|
||||||
.padding()
|
MainContent(pool: pool)
|
||||||
|
.padding()
|
||||||
|
}
|
||||||
PostButtonContainer
|
PostButtonContainer
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -137,13 +142,15 @@ struct ContentView: View {
|
|||||||
}
|
}
|
||||||
|
|
||||||
func connect() {
|
func connect() {
|
||||||
let pool = RelayPool(handle_event: handle_event)
|
let pool = RelayPool()
|
||||||
|
|
||||||
add_relay(pool, "nostr-relay.wlvs.space")
|
add_relay(pool, "wss://nostr.onsats.org")
|
||||||
add_relay(pool, "nostr.bitcoiner.social")
|
add_relay(pool, "nostr.bitcoiner.social")
|
||||||
add_relay(pool, "nostr-relay.freeberty.net")
|
add_relay(pool, "nostr-relay.freeberty.net")
|
||||||
add_relay(pool, "nostr-relay.untethr.me")
|
add_relay(pool, "nostr-relay.untethr.me")
|
||||||
|
|
||||||
|
pool.register_handler(sub_id: sub_id, handler: handle_event)
|
||||||
|
|
||||||
self.pool = pool
|
self.pool = pool
|
||||||
pool.connect()
|
pool.connect()
|
||||||
}
|
}
|
||||||
@@ -194,8 +201,6 @@ struct ContentView: View {
|
|||||||
|
|
||||||
let filters = [since_filter, profile_filter, contacts_filter]
|
let filters = [since_filter, profile_filter, contacts_filter]
|
||||||
print("connected to \(relay_id), refreshing from \(since)")
|
print("connected to \(relay_id), refreshing from \(since)")
|
||||||
let sub_id = UUID().description
|
|
||||||
print("subscribing to \(sub_id)")
|
|
||||||
self.pool?.send(.subscribe(.init(filters: filters, sub_id: sub_id)))
|
self.pool?.send(.subscribe(.init(filters: filters, sub_id: sub_id)))
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -211,6 +216,11 @@ struct ContentView: View {
|
|||||||
switch ev {
|
switch ev {
|
||||||
case .connected:
|
case .connected:
|
||||||
send_filters(relay_id: relay_id)
|
send_filters(relay_id: relay_id)
|
||||||
|
case .error(let merr):
|
||||||
|
let desc = merr.debugDescription
|
||||||
|
if desc.contains("Software caused connection abort") {
|
||||||
|
self.pool?.connect(to: [relay_id])
|
||||||
|
}
|
||||||
case .disconnected:
|
case .disconnected:
|
||||||
self.pool?.connect(to: [relay_id])
|
self.pool?.connect(to: [relay_id])
|
||||||
case .cancelled:
|
case .cancelled:
|
||||||
@@ -227,7 +237,12 @@ struct ContentView: View {
|
|||||||
|
|
||||||
case .nostr_event(let ev):
|
case .nostr_event(let ev):
|
||||||
switch ev {
|
switch ev {
|
||||||
case .event(_, let ev):
|
case .event(let sub_id, let ev):
|
||||||
|
if sub_id != self.sub_id {
|
||||||
|
// TODO: other views like threads might have their own sub ids, so ignore those events... or should we?
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
if self.loading {
|
if self.loading {
|
||||||
self.loading = false
|
self.loading = false
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -19,6 +19,19 @@ struct KeyEvent {
|
|||||||
let relay_url: String
|
let relay_url: String
|
||||||
}
|
}
|
||||||
|
|
||||||
|
struct ReferencedId {
|
||||||
|
let ref_id: String
|
||||||
|
let relay_id: String?
|
||||||
|
}
|
||||||
|
|
||||||
|
struct EventId: Identifiable, CustomStringConvertible {
|
||||||
|
let id: String
|
||||||
|
|
||||||
|
var description: String {
|
||||||
|
id
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
class NostrEvent: Codable, Identifiable {
|
class NostrEvent: Codable, Identifiable {
|
||||||
var id: String
|
var id: String
|
||||||
var sig: String
|
var sig: String
|
||||||
@@ -39,6 +52,26 @@ class NostrEvent: Codable, Identifiable {
|
|||||||
case id, sig, tags, pubkey, created_at, kind, content
|
case id, sig, tags, pubkey, created_at, kind, content
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private func get_referenced_ids(key: String) -> [ReferencedId] {
|
||||||
|
return tags.reduce(into: []) { (acc, tag) in
|
||||||
|
if tag.count >= 2 && tag[0] == key {
|
||||||
|
var relay_id: String? = nil
|
||||||
|
if tag.count >= 3 {
|
||||||
|
relay_id = tag[2]
|
||||||
|
}
|
||||||
|
acc.append(ReferencedId(ref_id: tag[1], relay_id: relay_id))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public var referenced_ids: [ReferencedId] {
|
||||||
|
return get_referenced_ids(key: "e")
|
||||||
|
}
|
||||||
|
|
||||||
|
public var referenced_pubkeys: [ReferencedId] {
|
||||||
|
return get_referenced_ids(key: "p")
|
||||||
|
}
|
||||||
|
|
||||||
/// Make a local event
|
/// Make a local event
|
||||||
public static func local(content: String, pubkey: String) -> NostrEvent {
|
public static func local(content: String, pubkey: String) -> NostrEvent {
|
||||||
let ev = NostrEvent(content: content, pubkey: pubkey)
|
let ev = NostrEvent(content: content, pubkey: pubkey)
|
||||||
|
|||||||
@@ -14,5 +14,6 @@ struct NostrSubscribe {
|
|||||||
|
|
||||||
enum NostrRequest {
|
enum NostrRequest {
|
||||||
case subscribe(NostrSubscribe)
|
case subscribe(NostrSubscribe)
|
||||||
|
case unsubscribe(String)
|
||||||
case event(NostrEvent)
|
case event(NostrEvent)
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -14,13 +14,17 @@ struct RelayInfo {
|
|||||||
static let rw = RelayInfo(read: true, write: true)
|
static let rw = RelayInfo(read: true, write: true)
|
||||||
}
|
}
|
||||||
|
|
||||||
struct Relay: Identifiable {
|
struct RelayDescriptor {
|
||||||
let url: URL
|
let url: URL
|
||||||
let info: RelayInfo
|
let info: RelayInfo
|
||||||
|
}
|
||||||
|
|
||||||
|
struct Relay: Identifiable {
|
||||||
|
let descriptor: RelayDescriptor
|
||||||
let connection: RelayConnection
|
let connection: RelayConnection
|
||||||
|
|
||||||
var id: String {
|
var id: String {
|
||||||
return get_relay_id(url)
|
return get_relay_id(descriptor.url)
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -40,6 +40,8 @@ class RelayConnection: WebSocketDelegate {
|
|||||||
print("failed to encode nostr req: \(req)")
|
print("failed to encode nostr req: \(req)")
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
print("req: \(req)")
|
||||||
|
|
||||||
socket.write(string: req)
|
socket.write(string: req)
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -75,6 +77,8 @@ func make_nostr_req(_ req: NostrRequest) -> String? {
|
|||||||
switch req {
|
switch req {
|
||||||
case .subscribe(let sub):
|
case .subscribe(let sub):
|
||||||
return make_nostr_subscription_req(sub.filters, sub_id: sub.sub_id)
|
return make_nostr_subscription_req(sub.filters, sub_id: sub.sub_id)
|
||||||
|
case .unsubscribe(let sub_id):
|
||||||
|
return make_nostr_unsubscribe_req(sub_id)
|
||||||
case .event(let ev):
|
case .event(let ev):
|
||||||
return make_nostr_push_event(ev: ev)
|
return make_nostr_push_event(ev: ev)
|
||||||
}
|
}
|
||||||
@@ -89,6 +93,10 @@ func make_nostr_push_event(ev: NostrEvent) -> String? {
|
|||||||
return encoded
|
return encoded
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func make_nostr_unsubscribe_req(_ sub_id: String) -> String? {
|
||||||
|
return "[\"CLOSE\",\"\(sub_id)\"]"
|
||||||
|
}
|
||||||
|
|
||||||
func make_nostr_subscription_req(_ filters: [NostrFilter], sub_id: String) -> String? {
|
func make_nostr_subscription_req(_ filters: [NostrFilter], sub_id: String) -> String? {
|
||||||
let encoder = JSONEncoder()
|
let encoder = JSONEncoder()
|
||||||
var req = "[\"REQ\",\"\(sub_id)\""
|
var req = "[\"REQ\",\"\(sub_id)\""
|
||||||
@@ -101,7 +109,6 @@ func make_nostr_subscription_req(_ filters: [NostrFilter], sub_id: String) -> St
|
|||||||
req += filter_json_str
|
req += filter_json_str
|
||||||
}
|
}
|
||||||
req += "]"
|
req += "]"
|
||||||
print("req: \(req)")
|
|
||||||
return req
|
return req
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -7,12 +7,41 @@
|
|||||||
|
|
||||||
import Foundation
|
import Foundation
|
||||||
|
|
||||||
|
struct SubscriptionId: Identifiable, CustomStringConvertible {
|
||||||
|
let id: String
|
||||||
|
|
||||||
|
var description: String {
|
||||||
|
id
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
struct RelayId: Identifiable, CustomStringConvertible {
|
||||||
|
let id: String
|
||||||
|
|
||||||
|
var description: String {
|
||||||
|
id
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
struct RelayHandler {
|
||||||
|
let sub_id: String
|
||||||
|
let callback: (String, NostrConnectionEvent) -> ()
|
||||||
|
}
|
||||||
|
|
||||||
class RelayPool {
|
class RelayPool {
|
||||||
var relays: [Relay] = []
|
var relays: [Relay] = []
|
||||||
let custom_handle_event: (String, NostrConnectionEvent) -> ()
|
var handlers: [RelayHandler] = []
|
||||||
|
|
||||||
init(handle_event: @escaping (String, NostrConnectionEvent) -> ()) {
|
var descriptors: [RelayDescriptor] {
|
||||||
self.custom_handle_event = handle_event
|
relays.map { $0.descriptor }
|
||||||
|
}
|
||||||
|
|
||||||
|
func remove_handler(sub_id: String) {
|
||||||
|
handlers = handlers.filter { $0.sub_id != sub_id }
|
||||||
|
}
|
||||||
|
|
||||||
|
func register_handler(sub_id: String, handler: @escaping (String, NostrConnectionEvent) -> ()) {
|
||||||
|
self.handlers.append(RelayHandler(sub_id: sub_id, callback: handler))
|
||||||
}
|
}
|
||||||
|
|
||||||
func add_relay(_ url: URL, info: RelayInfo) throws {
|
func add_relay(_ url: URL, info: RelayInfo) throws {
|
||||||
@@ -23,7 +52,8 @@ class RelayPool {
|
|||||||
let conn = RelayConnection(url: url) { event in
|
let conn = RelayConnection(url: url) { event in
|
||||||
self.handle_event(relay_id: relay_id, event: event)
|
self.handle_event(relay_id: relay_id, event: event)
|
||||||
}
|
}
|
||||||
let relay = Relay(url: url, info: info, connection: conn)
|
let descriptor = RelayDescriptor(url: url, info: info)
|
||||||
|
let relay = Relay(descriptor: descriptor, connection: conn)
|
||||||
self.relays.append(relay)
|
self.relays.append(relay)
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -34,6 +64,13 @@ class RelayPool {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func disconnect(to: [String]? = nil) {
|
||||||
|
let relays = to.map{ get_relays($0) } ?? self.relays
|
||||||
|
for relay in relays {
|
||||||
|
relay.connection.disconnect()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
func send(_ req: NostrRequest, to: [String]? = nil) {
|
func send(_ req: NostrRequest, to: [String]? = nil) {
|
||||||
let relays = to.map{ get_relays($0) } ?? self.relays
|
let relays = to.map{ get_relays($0) } ?? self.relays
|
||||||
|
|
||||||
@@ -68,7 +105,9 @@ class RelayPool {
|
|||||||
|
|
||||||
func handle_event(relay_id: String, event: NostrConnectionEvent) {
|
func handle_event(relay_id: String, event: NostrConnectionEvent) {
|
||||||
// handle reconnect logic, etc?
|
// handle reconnect logic, etc?
|
||||||
custom_handle_event(relay_id, event)
|
for handler in handlers {
|
||||||
|
handler.callback(relay_id, event)
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -9,12 +9,62 @@ import SwiftUI
|
|||||||
|
|
||||||
struct EventDetailView: View {
|
struct EventDetailView: View {
|
||||||
let event: NostrEvent
|
let event: NostrEvent
|
||||||
let profile: Profile?
|
|
||||||
|
|
||||||
var body: some View {
|
let sub_id = UUID().description
|
||||||
|
|
||||||
|
@State var events: [NostrEvent] = []
|
||||||
|
@State var has_event: [String: ()] = [:]
|
||||||
|
|
||||||
|
let pool: RelayPool
|
||||||
|
let profiles: [String: TimestampedProfile]
|
||||||
|
|
||||||
|
func unsubscribe_to_thread() {
|
||||||
|
print("unsubscribing from thread \(event.id) with sub_id \(sub_id)")
|
||||||
|
self.pool.send(.unsubscribe(sub_id))
|
||||||
|
self.pool.remove_handler(sub_id: sub_id)
|
||||||
|
}
|
||||||
|
|
||||||
|
func subscribe_to_thread() {
|
||||||
|
var ref_events = NostrFilter.filter_text
|
||||||
|
var events = NostrFilter.filter_text
|
||||||
|
|
||||||
|
// TODO: add referenced relays
|
||||||
|
ref_events.referenced_ids = event.referenced_ids.map { $0.ref_id }
|
||||||
|
ref_events.referenced_ids!.append(event.id)
|
||||||
|
|
||||||
|
events.ids = ref_events.referenced_ids!
|
||||||
|
|
||||||
|
print("subscribing to thread \(event.id) with sub_id \(sub_id)")
|
||||||
|
pool.register_handler(sub_id: sub_id, handler: handle_event)
|
||||||
|
pool.send(.subscribe(.init(filters: [ref_events, events], sub_id: sub_id)))
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
func handle_event(relay_id: String, ev: NostrConnectionEvent) {
|
||||||
|
switch ev {
|
||||||
|
case .ws_event:
|
||||||
|
break
|
||||||
|
case .nostr_event(let res):
|
||||||
|
switch res {
|
||||||
|
case .event(let sub_id, let ev):
|
||||||
|
if sub_id != self.sub_id || self.has_event[ev.id] != nil {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
self.add_event(ev)
|
||||||
|
|
||||||
|
case .notice(_):
|
||||||
|
// TODO: handle notices in threads?
|
||||||
|
break
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
var NoteBody: some View {
|
||||||
HStack {
|
HStack {
|
||||||
|
let profile = profiles[event.pubkey]?.profile
|
||||||
|
|
||||||
VStack {
|
VStack {
|
||||||
ProfilePicView(picture: profile?.picture, size: 64)
|
ProfilePicView(picture: profile?.picture, size: 64, highlighted: false)
|
||||||
|
|
||||||
Spacer()
|
Spacer()
|
||||||
}
|
}
|
||||||
@@ -30,20 +80,50 @@ struct EventDetailView: View {
|
|||||||
Text(event.content)
|
Text(event.content)
|
||||||
.frame(maxWidth: .infinity, alignment: .leading)
|
.frame(maxWidth: .infinity, alignment: .leading)
|
||||||
|
|
||||||
Divider()
|
|
||||||
.padding([.bottom], 10)
|
|
||||||
|
|
||||||
EventActionBar(event: event)
|
EventActionBar(event: event)
|
||||||
|
|
||||||
Spacer()
|
Divider()
|
||||||
|
.padding([.bottom], 10)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
var body: some View {
|
||||||
|
ScrollView {
|
||||||
|
ForEach(events, id: \.id) { ev in
|
||||||
|
let evdet = EventDetailView(event: ev, pool: pool, profiles: profiles)
|
||||||
|
.navigationBarTitle("Note")
|
||||||
|
NavigationLink(destination: evdet) {
|
||||||
|
EventView(event: ev, profile: self.profiles[ev.pubkey]?.profile, highlighted: ev.id == event.id)
|
||||||
|
}
|
||||||
|
.buttonStyle(PlainButtonStyle())
|
||||||
|
//EventView(event: ev, profile: self.profiles[ev.pubkey]?.profile, highlighted: ev.id == event.id)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
.padding()
|
.padding()
|
||||||
|
.onDisappear() {
|
||||||
|
unsubscribe_to_thread()
|
||||||
|
}
|
||||||
|
.onAppear() {
|
||||||
|
self.add_event(event)
|
||||||
|
subscribe_to_thread()
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
func add_event(_ ev: NostrEvent) {
|
||||||
|
if self.has_event[ev.id] == nil {
|
||||||
|
self.has_event[ev.id] = ()
|
||||||
|
self.events.append(ev)
|
||||||
|
self.events = self.events.sorted { $0.created_at < $1.created_at }
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
struct EventDetailView_Previews: PreviewProvider {
|
struct EventDetailView_Previews: PreviewProvider {
|
||||||
static var previews: some View {
|
static var previews: some View {
|
||||||
EventDetailView(event: NostrEvent(content: "Hello", pubkey: "Guy"), profile: nil)
|
EventDetailView(event: NostrEvent(content: "Hello", pubkey: "Guy"), profile: nil)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
*/
|
||||||
|
|||||||
@@ -12,11 +12,12 @@ import CachedAsyncImage
|
|||||||
struct EventView: View {
|
struct EventView: View {
|
||||||
let event: NostrEvent
|
let event: NostrEvent
|
||||||
let profile: Profile?
|
let profile: Profile?
|
||||||
|
let highlighted: Bool
|
||||||
|
|
||||||
var body: some View {
|
var body: some View {
|
||||||
HStack {
|
HStack {
|
||||||
VStack {
|
VStack {
|
||||||
ProfilePicView(picture: profile?.picture, size: 64)
|
ProfilePicView(picture: profile?.picture, size: 64, highlighted: highlighted)
|
||||||
|
|
||||||
Spacer()
|
Spacer()
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -10,8 +10,5 @@ import SwiftUI
|
|||||||
func ProfileName(pubkey: String, profile: Profile?) -> some View {
|
func ProfileName(pubkey: String, profile: Profile?) -> some View {
|
||||||
Text(String(profile?.name ?? String(pubkey.prefix(16))))
|
Text(String(profile?.name ?? String(pubkey.prefix(16))))
|
||||||
.bold()
|
.bold()
|
||||||
.onTapGesture {
|
|
||||||
UIPasteboard.general.string = pubkey
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -14,6 +14,7 @@ let CORNER_RADIUS: CGFloat = 32
|
|||||||
struct ProfilePicView: View {
|
struct ProfilePicView: View {
|
||||||
let picture: String?
|
let picture: String?
|
||||||
let size: CGFloat
|
let size: CGFloat
|
||||||
|
let highlighted: Bool
|
||||||
|
|
||||||
var body: some View {
|
var body: some View {
|
||||||
if let pic = picture.flatMap({ URL(string: $0) }) {
|
if let pic = picture.flatMap({ URL(string: $0) }) {
|
||||||
@@ -23,17 +24,21 @@ struct ProfilePicView: View {
|
|||||||
Color.purple.opacity(0.1)
|
Color.purple.opacity(0.1)
|
||||||
}
|
}
|
||||||
.frame(width: PFP_SIZE, height: PFP_SIZE)
|
.frame(width: PFP_SIZE, height: PFP_SIZE)
|
||||||
.cornerRadius(CORNER_RADIUS)
|
.clipShape(Circle())
|
||||||
|
.overlay(Circle().stroke(highlighted ? Color.red : Color.black, lineWidth: highlighted ? 4 : 0))
|
||||||
|
.padding(2)
|
||||||
} else {
|
} else {
|
||||||
Color.purple.opacity(0.1)
|
Color.purple.opacity(0.1)
|
||||||
.frame(width: PFP_SIZE, height: PFP_SIZE)
|
.frame(width: PFP_SIZE, height: PFP_SIZE)
|
||||||
.cornerRadius(CORNER_RADIUS)
|
.cornerRadius(CORNER_RADIUS)
|
||||||
|
.overlay(Circle().stroke(highlighted ? Color.red : Color.black, lineWidth: highlighted ? 4 : 0))
|
||||||
|
.padding(2)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
struct ProfilePicView_Previews: PreviewProvider {
|
struct ProfilePicView_Previews: PreviewProvider {
|
||||||
static var previews: some View {
|
static var previews: some View {
|
||||||
ProfilePicView(picture: "http://cdn.jb55.com/img/red-me.jpg", size: 64)
|
ProfilePicView(picture: "http://cdn.jb55.com/img/red-me.jpg", size: 64, highlighted: false)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
Reference in New Issue
Block a user