Bugfixes, simplifications
This commit is contained in:
@@ -14,7 +14,7 @@ struct AllNotesScreen: View {
|
|||||||
@State private var ownPeer: OwnPeer?
|
@State private var ownPeer: OwnPeer?
|
||||||
|
|
||||||
var isUsernameValid: Bool {
|
var isUsernameValid: Bool {
|
||||||
username.map(\.isEmpty) ?? false
|
!(username.map(\.isEmpty) ?? true)
|
||||||
}
|
}
|
||||||
|
|
||||||
var body: some View {
|
var body: some View {
|
||||||
|
|||||||
@@ -4,8 +4,6 @@ import Combine
|
|||||||
|
|
||||||
struct OwnPeer {
|
struct OwnPeer {
|
||||||
let peer: MCPeerID
|
let peer: MCPeerID
|
||||||
|
|
||||||
static var fallback: Self { Self(peer: .init(displayName: "fallback_user")) }
|
|
||||||
}
|
}
|
||||||
|
|
||||||
struct Peer: Identifiable {
|
struct Peer: Identifiable {
|
||||||
@@ -25,15 +23,15 @@ struct Peer: Identifiable {
|
|||||||
final class NoteEditingSessionServer: NSObject {
|
final class NoteEditingSessionServer: NSObject {
|
||||||
private let session: MCSession
|
private let session: MCSession
|
||||||
private let browser: MCNearbyServiceBrowser
|
private let browser: MCNearbyServiceBrowser
|
||||||
private let ownPeer: OwnPeer
|
private(set) var ownPeer: OwnPeer
|
||||||
|
|
||||||
var visiblePeers: [Peer] = []
|
var visiblePeers: [Peer] = []
|
||||||
let noteChangesEmitter = PassthroughSubject<String, Never>()
|
let noteChangesEmitter = PassthroughSubject<NoteMessage, Never>()
|
||||||
|
|
||||||
init(peer: OwnPeer) {
|
init(peer: OwnPeer) {
|
||||||
ownPeer = peer
|
ownPeer = peer
|
||||||
browser = .init(peer: peer.peer, serviceType: "peered")
|
browser = .init(peer: peer.peer, serviceType: "peered")
|
||||||
session = .init(peer: peer.peer)
|
session = .init(peer: peer.peer, securityIdentity: nil, encryptionPreference: .required)
|
||||||
super.init()
|
super.init()
|
||||||
browser.delegate = self
|
browser.delegate = self
|
||||||
session.delegate = self
|
session.delegate = self
|
||||||
@@ -56,26 +54,34 @@ final class NoteEditingSessionServer: NSObject {
|
|||||||
withContext: try! JSONEncoder().encode(note),
|
withContext: try! JSONEncoder().encode(note),
|
||||||
timeout: 5
|
timeout: 5
|
||||||
)
|
)
|
||||||
|
guard let idxToUpdate = visiblePeers.firstIndex(where: { $0.mcPeer == peer.mcPeer }) else { return }
|
||||||
let idxToUpdate = visiblePeers.firstIndex(where: { $0.mcPeer == peer.mcPeer })!
|
|
||||||
visiblePeers[idxToUpdate].state = .invitationPending
|
visiblePeers[idxToUpdate].state = .invitationPending
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func send(note: String, to peers: [MCPeerID]) {
|
||||||
|
let message = NoteMessage(senderID: ownPeer.peer.displayName, content: note)
|
||||||
|
guard !peers.isEmpty, let data = try? JSONEncoder().encode(message) else { return }
|
||||||
|
try? session.send(data, toPeers: peers, with: .reliable)
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
extension NoteEditingSessionServer: MCNearbyServiceBrowserDelegate {
|
extension NoteEditingSessionServer: MCNearbyServiceBrowserDelegate {
|
||||||
func browser(
|
func browser(
|
||||||
_ browser: MCNearbyServiceBrowser,
|
_ browser: MCNearbyServiceBrowser,
|
||||||
foundPeer peerID: MCPeerID,
|
foundPeer peerID: MCPeerID,
|
||||||
withDiscoveryInfo info: [String : String]?
|
withDiscoveryInfo info: [String: String]?
|
||||||
) {
|
) {
|
||||||
guard !visiblePeers.contains(where: { $0.mcPeer == peerID }) && peerID.displayName != ownPeer.peer.displayName else { return }
|
guard !visiblePeers.contains(where: { $0.mcPeer == peerID }) && peerID.displayName != ownPeer.peer.displayName else { return }
|
||||||
let newPeer = Peer(mcPeer: peerID, state: .available)
|
DispatchQueue.main.async {
|
||||||
visiblePeers.append(newPeer)
|
self.visiblePeers.append(Peer(mcPeer: peerID, state: .available))
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func browser(_ browser: MCNearbyServiceBrowser, lostPeer peerID: MCPeerID) {
|
func browser(_ browser: MCNearbyServiceBrowser, lostPeer peerID: MCPeerID) {
|
||||||
guard let peerIdx = visiblePeers.firstIndex(where: { $0.mcPeer == peerID }) else { return }
|
DispatchQueue.main.async {
|
||||||
visiblePeers.remove(at: peerIdx)
|
guard let peerIdx = self.visiblePeers.firstIndex(where: { $0.mcPeer == peerID }) else { return }
|
||||||
|
self.visiblePeers.remove(at: peerIdx)
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -85,39 +91,37 @@ extension NoteEditingSessionServer: MCSessionDelegate {
|
|||||||
peer peerID: MCPeerID,
|
peer peerID: MCPeerID,
|
||||||
didChange state: MCSessionState
|
didChange state: MCSessionState
|
||||||
) {
|
) {
|
||||||
|
DispatchQueue.main.async {
|
||||||
}
|
guard let idx = self.visiblePeers.firstIndex(where: { $0.mcPeer == peerID }) else { return }
|
||||||
|
switch state {
|
||||||
func session(
|
case .connected:
|
||||||
_ session: MCSession,
|
self.visiblePeers[idx].state = .joined
|
||||||
didReceive stream: InputStream,
|
case .notConnected:
|
||||||
withName streamName: String,
|
let currentState = self.visiblePeers[idx].state
|
||||||
fromPeer peerID: MCPeerID
|
if currentState == .invitationPending || currentState == .joined {
|
||||||
) {
|
self.visiblePeers[idx].state = .rejected
|
||||||
|
}
|
||||||
}
|
default:
|
||||||
|
break
|
||||||
func session(
|
}
|
||||||
_ session: MCSession,
|
}
|
||||||
didStartReceivingResourceWithName resourceName: String,
|
|
||||||
fromPeer peerID: MCPeerID,
|
|
||||||
with progress: Progress
|
|
||||||
) {
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
func session(
|
|
||||||
_ session: MCSession,
|
|
||||||
didFinishReceivingResourceWithName resourceName: String,
|
|
||||||
fromPeer peerID: MCPeerID,
|
|
||||||
at localURL: URL?,
|
|
||||||
withError error: (any Error)?
|
|
||||||
) {
|
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
func session(_ session: MCSession, didReceive data: Data, fromPeer peerID: MCPeerID) {
|
func session(_ session: MCSession, didReceive data: Data, fromPeer peerID: MCPeerID) {
|
||||||
guard let note = String(data: data, encoding: .utf8) else { fatalError() }
|
guard let message = try? JSONDecoder().decode(NoteMessage.self, from: data) else { return }
|
||||||
noteChangesEmitter.send(note)
|
|
||||||
|
// Broadcast to all other connected peers, preserving the original senderID
|
||||||
|
let otherPeers = session.connectedPeers.filter { $0 != peerID }
|
||||||
|
if !otherPeers.isEmpty {
|
||||||
|
try? session.send(data, toPeers: otherPeers, with: .reliable)
|
||||||
|
}
|
||||||
|
|
||||||
|
DispatchQueue.main.async {
|
||||||
|
self.noteChangesEmitter.send(message)
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func session(_ session: MCSession, didReceive stream: InputStream, withName streamName: String, fromPeer peerID: MCPeerID) {}
|
||||||
|
func session(_ session: MCSession, didStartReceivingResourceWithName resourceName: String, fromPeer peerID: MCPeerID, with progress: Progress) {}
|
||||||
|
func session(_ session: MCSession, didFinishReceivingResourceWithName resourceName: String, fromPeer peerID: MCPeerID, at localURL: URL?, withError error: (any Error)?) {}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -12,17 +12,18 @@ struct NoteEditorScreen: View {
|
|||||||
let note: Note
|
let note: Note
|
||||||
@State private var noteAdvertiser: NoteEditingSessionServer
|
@State private var noteAdvertiser: NoteEditingSessionServer
|
||||||
@State private var noteContent: String = ""
|
@State private var noteContent: String = ""
|
||||||
|
@State private var remoteNoteContent: String? = nil
|
||||||
@State private var timer = Timer.publish(every: 5, on: .current, in: .common)
|
@State private var timer = Timer.publish(every: 5, on: .current, in: .common)
|
||||||
.autoconnect()
|
.autoconnect()
|
||||||
@State private var showManageMembers = false
|
@State private var showManageMembers = false
|
||||||
|
|
||||||
init(note: Note, peer: OwnPeer) {
|
init(note: Note, peer: OwnPeer) {
|
||||||
self.note = note
|
self.note = note
|
||||||
self._noteAdvertiser = .init(initialValue: .init(peer: peer))
|
self._noteAdvertiser = .init(initialValue: .init(peer: peer))
|
||||||
}
|
}
|
||||||
|
|
||||||
var body: some View {
|
var body: some View {
|
||||||
TextEditor(text: $noteContent)
|
NoteTextEditor(text: $noteContent, remoteText: remoteNoteContent)
|
||||||
.toolbar {
|
.toolbar {
|
||||||
ToolbarItem(placement: .primaryAction) {
|
ToolbarItem(placement: .primaryAction) {
|
||||||
Button("Manage members") {
|
Button("Manage members") {
|
||||||
@@ -32,18 +33,29 @@ struct NoteEditorScreen: View {
|
|||||||
}
|
}
|
||||||
.sheet(isPresented: $showManageMembers) {
|
.sheet(isPresented: $showManageMembers) {
|
||||||
NavigationStack {
|
NavigationStack {
|
||||||
ManageMembersView(
|
ManageMembersScreen(
|
||||||
noteAdvertiser: noteAdvertiser,
|
noteAdvertiser: noteAdvertiser,
|
||||||
noteTitle: note.name,
|
noteTitle: note.name,
|
||||||
noteContent: $noteContent
|
noteContent: $noteContent
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
.onReceive(noteAdvertiser.noteChangesEmitter) { updatedNote in
|
.onReceive(noteAdvertiser.noteChangesEmitter) { message in
|
||||||
self.noteContent = updatedNote
|
guard message.senderID != noteAdvertiser.ownPeer.peer.displayName else { return }
|
||||||
|
remoteNoteContent = message.content
|
||||||
|
}
|
||||||
|
.task(id: noteContent) {
|
||||||
|
if noteContent == remoteNoteContent { return }
|
||||||
|
let connectedPeers = noteAdvertiser.visiblePeers
|
||||||
|
.filter { $0.state == .joined }
|
||||||
|
.map(\.mcPeer)
|
||||||
|
guard !connectedPeers.isEmpty else { return }
|
||||||
|
try? await Task.sleep(nanoseconds: 500_000_000)
|
||||||
|
guard !Task.isCancelled else { return }
|
||||||
|
noteAdvertiser.send(note: noteContent, to: connectedPeers)
|
||||||
}
|
}
|
||||||
.onAppear {
|
.onAppear {
|
||||||
noteContent = try! String(contentsOf: note.path, encoding: .utf8)
|
noteContent = (try? String(contentsOf: note.path, encoding: .utf8)) ?? ""
|
||||||
noteAdvertiser.startServer()
|
noteAdvertiser.startServer()
|
||||||
}
|
}
|
||||||
.onDisappear {
|
.onDisappear {
|
||||||
@@ -54,8 +66,8 @@ struct NoteEditorScreen: View {
|
|||||||
saveNote()
|
saveNote()
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func saveNote() {
|
func saveNote() {
|
||||||
try! noteContent.write(to: note.path, atomically: true, encoding: .utf8)
|
try? noteContent.write(to: note.path, atomically: true, encoding: .utf8)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -2,60 +2,15 @@ import SwiftUI
|
|||||||
|
|
||||||
struct NoteTextEditor: View {
|
struct NoteTextEditor: View {
|
||||||
@Binding var text: String
|
@Binding var text: String
|
||||||
let remoteText: String?
|
let remoteText: String?
|
||||||
@State private var selection: TextSelection? = nil
|
@State private var selection: TextSelection? = nil
|
||||||
|
|
||||||
var body: some View {
|
var body: some View {
|
||||||
TextEditor(text: $text, selection: $selection)
|
TextEditor(text: $text, selection: $selection)
|
||||||
.onChange(of: remoteText) { oldValue, newValue in
|
.onChange(of: remoteText) { oldValue, newValue in
|
||||||
guard let newValue, newValue != text else { return }
|
guard let newValue, newValue != text else { return }
|
||||||
|
// Apply remote text first
|
||||||
let previousRemote = oldValue ?? text
|
text = newValue
|
||||||
|
}
|
||||||
// Where the remote edit begins
|
}
|
||||||
let changeStart = commonPrefixUTF16Length(previousRemote, newValue)
|
|
||||||
|
|
||||||
// How many UTF-16 units were inserted (positive) or removed (negative)
|
|
||||||
let delta = newValue.utf16.count - previousRemote.utf16.count
|
|
||||||
|
|
||||||
// Apply remote text first
|
|
||||||
text = newValue
|
|
||||||
|
|
||||||
// Get current cursor offset
|
|
||||||
guard
|
|
||||||
let selection,
|
|
||||||
case .selection(let range) = selection.indices,
|
|
||||||
let cursorPos = range.lowerBound.samePosition(in: text.utf16)
|
|
||||||
else { return }
|
|
||||||
|
|
||||||
let cursorUTF16 = text.utf16.distance(
|
|
||||||
from: text.utf16.startIndex,
|
|
||||||
to: cursorPos
|
|
||||||
)
|
|
||||||
|
|
||||||
// Only shift if the change happened strictly before the cursor
|
|
||||||
guard changeStart < cursorUTF16 else { return }
|
|
||||||
|
|
||||||
let newCursorUTF16 = max(0, min(cursorUTF16 + delta, newValue.utf16.count))
|
|
||||||
let newUTF16 = newValue.utf16
|
|
||||||
let newUTF16Index = newUTF16.index(newUTF16.startIndex, offsetBy: newCursorUTF16)
|
|
||||||
guard let newIndex = newUTF16Index.samePosition(in: newValue) else { return }
|
|
||||||
|
|
||||||
self.selection = TextSelection(range: newIndex..<newIndex)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
private func commonPrefixUTF16Length(_ a: String, _ b: String) -> Int {
|
|
||||||
var count = 0
|
|
||||||
let aUTF16 = a.utf16
|
|
||||||
let bUTF16 = b.utf16
|
|
||||||
let minLen = min(aUTF16.count, bUTF16.count)
|
|
||||||
while count < minLen {
|
|
||||||
let aIdx = aUTF16.index(aUTF16.startIndex, offsetBy: count)
|
|
||||||
let bIdx = bUTF16.index(bUTF16.startIndex, offsetBy: count)
|
|
||||||
guard aUTF16[aIdx] == bUTF16[bIdx] else { break }
|
|
||||||
count += 1
|
|
||||||
}
|
|
||||||
return count
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -8,10 +8,10 @@ import SwiftUI
|
|||||||
|
|
||||||
struct SharedNoteEditor: View {
|
struct SharedNoteEditor: View {
|
||||||
@State var note: String?
|
@State var note: String?
|
||||||
@State var sendTask: Task<Void, Never>?
|
@State var remoteNote: String? = nil
|
||||||
@State var invitation: NoteInvitation
|
@State var invitation: NoteInvitation
|
||||||
@Bindable var noteClient: NoteEditingSessionClient
|
@Bindable var noteClient: NoteEditingSessionClient
|
||||||
|
|
||||||
init(
|
init(
|
||||||
invitation: NoteInvitation,
|
invitation: NoteInvitation,
|
||||||
noteClient: NoteEditingSessionClient
|
noteClient: NoteEditingSessionClient
|
||||||
@@ -19,27 +19,26 @@ struct SharedNoteEditor: View {
|
|||||||
self._invitation = .init(initialValue: invitation)
|
self._invitation = .init(initialValue: invitation)
|
||||||
self._noteClient = .init(noteClient)
|
self._noteClient = .init(noteClient)
|
||||||
}
|
}
|
||||||
|
|
||||||
var body: some View {
|
var body: some View {
|
||||||
ZStack {
|
ZStack {
|
||||||
if let note = Binding($note) {
|
if let note = Binding($note) {
|
||||||
TextEditor(text: note)
|
NoteTextEditor(text: note, remoteText: remoteNote)
|
||||||
} else {
|
} else {
|
||||||
ProgressView {
|
ProgressView {
|
||||||
Text("Fetching note...")
|
Text("Fetching note...")
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
.onChange(of: note) { _, newValue in
|
|
||||||
sendTask?.cancel()
|
|
||||||
sendTask = Task {
|
|
||||||
try? await Task.sleep(nanoseconds: 500000000)
|
|
||||||
guard let note else { return }
|
|
||||||
DispatchQueue.main.async {
|
|
||||||
noteClient.send(note: note, to: invitation.invitatorID)
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
.onReceive(noteClient.noteChangesEmitter) { message in
|
||||||
|
guard message.senderID != noteClient.ownPeer.displayName else { return }
|
||||||
|
remoteNote = message.content
|
||||||
|
}
|
||||||
|
.task(id: note) {
|
||||||
|
try? await Task.sleep(nanoseconds: 500_000_000)
|
||||||
|
guard !Task.isCancelled, let note else { return }
|
||||||
|
noteClient.send(note: note, to: invitation.invitatorID)
|
||||||
|
}
|
||||||
.onAppear {
|
.onAppear {
|
||||||
invitation.accept()
|
invitation.accept()
|
||||||
note = invitation.note.noteSnapshot
|
note = invitation.note.noteSnapshot
|
||||||
|
|||||||
@@ -1,5 +1,6 @@
|
|||||||
import MultipeerConnectivity
|
import MultipeerConnectivity
|
||||||
import Foundation
|
import Foundation
|
||||||
|
import Combine
|
||||||
|
|
||||||
struct NoteInvitation: Identifiable {
|
struct NoteInvitation: Identifiable {
|
||||||
struct NoteContent: Codable {
|
struct NoteContent: Codable {
|
||||||
@@ -8,18 +9,16 @@ struct NoteInvitation: Identifiable {
|
|||||||
}
|
}
|
||||||
|
|
||||||
var id: MCPeerID { invitatorID }
|
var id: MCPeerID { invitatorID }
|
||||||
let noteName: String
|
var noteName: String { note.title }
|
||||||
let invitatorID: MCPeerID
|
let invitatorID: MCPeerID
|
||||||
let note: NoteContent
|
let note: NoteContent
|
||||||
private var invitationHandler: ((Bool) -> Void)?
|
private var invitationHandler: ((Bool) -> Void)?
|
||||||
|
|
||||||
init(
|
init(
|
||||||
noteName: String,
|
|
||||||
invitatorID: MCPeerID,
|
invitatorID: MCPeerID,
|
||||||
note: NoteContent,
|
note: NoteContent,
|
||||||
invitationHandler: ((Bool) -> Void)? = nil
|
invitationHandler: ((Bool) -> Void)? = nil
|
||||||
) {
|
) {
|
||||||
self.noteName = noteName
|
|
||||||
self.invitatorID = invitatorID
|
self.invitatorID = invitatorID
|
||||||
self.note = note
|
self.note = note
|
||||||
self.invitationHandler = invitationHandler
|
self.invitationHandler = invitationHandler
|
||||||
@@ -40,13 +39,10 @@ struct NoteInvitation: Identifiable {
|
|||||||
final class NoteEditingSessionClient: NSObject {
|
final class NoteEditingSessionClient: NSObject {
|
||||||
private let session: MCSession
|
private let session: MCSession
|
||||||
private let advertiser: MCNearbyServiceAdvertiser
|
private let advertiser: MCNearbyServiceAdvertiser
|
||||||
private let ownPeer: MCPeerID
|
private(set) var ownPeer: MCPeerID
|
||||||
|
|
||||||
var invitations: [NoteInvitation] = [] {
|
var invitations: [NoteInvitation] = []
|
||||||
didSet {
|
let noteChangesEmitter = PassthroughSubject<NoteMessage, Never>()
|
||||||
print(invitations)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
init(peer: MCPeerID) {
|
init(peer: MCPeerID) {
|
||||||
ownPeer = peer
|
ownPeer = peer
|
||||||
@@ -62,22 +58,22 @@ final class NoteEditingSessionClient: NSObject {
|
|||||||
)
|
)
|
||||||
super.init()
|
super.init()
|
||||||
advertiser.delegate = self
|
advertiser.delegate = self
|
||||||
|
session.delegate = self
|
||||||
}
|
}
|
||||||
|
|
||||||
func startBrowsingForNotes() {
|
func startBrowsingForNotes() {
|
||||||
advertiser.startAdvertisingPeer()
|
advertiser.startAdvertisingPeer()
|
||||||
}
|
}
|
||||||
|
|
||||||
func stopBrowsingForNotes() {
|
func stopBrowsingForNotes() {
|
||||||
advertiser.stopAdvertisingPeer()
|
advertiser.stopAdvertisingPeer()
|
||||||
}
|
session.disconnect()
|
||||||
|
}
|
||||||
|
|
||||||
func send(note: String, to peer: MCPeerID) {
|
func send(note: String, to peer: MCPeerID) {
|
||||||
try! session.send(
|
let message = NoteMessage(senderID: ownPeer.displayName, content: note)
|
||||||
note.data(using: .utf8)!,
|
guard let data = try? JSONEncoder().encode(message) else { return }
|
||||||
toPeers: [peer],
|
try? session.send(data, toPeers: [peer], with: .reliable)
|
||||||
with: .reliable
|
|
||||||
)
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -91,23 +87,45 @@ extension NoteEditingSessionClient: MCNearbyServiceAdvertiserDelegate {
|
|||||||
guard
|
guard
|
||||||
let context,
|
let context,
|
||||||
let noteContent = try? JSONDecoder().decode(NoteInvitation.NoteContent.self, from: context)
|
let noteContent = try? JSONDecoder().decode(NoteInvitation.NoteContent.self, from: context)
|
||||||
else { fatalError() }
|
else { return }
|
||||||
|
|
||||||
invitations.append(
|
DispatchQueue.main.async {
|
||||||
.init(
|
self.invitations.append(
|
||||||
noteName: noteContent.title,
|
.init(
|
||||||
invitatorID: peerID,
|
invitatorID: peerID,
|
||||||
note: noteContent,
|
note: noteContent,
|
||||||
invitationHandler: { [weak self, invitationHandler] accepted in
|
invitationHandler: { [weak self, invitationHandler] accepted in
|
||||||
guard let self else { return }
|
guard let self else { return }
|
||||||
invitationHandler(accepted, self.session)
|
invitationHandler(accepted, self.session)
|
||||||
|
DispatchQueue.main.async {
|
||||||
DispatchQueue.main.async {
|
self.invitations.removeAll { $0.id == peerID }
|
||||||
guard let idx = self.invitations.firstIndex(where: { $0.id == peerID }) else { return }
|
}
|
||||||
self.invitations.remove(at: idx)
|
|
||||||
}
|
}
|
||||||
}
|
)
|
||||||
)
|
)
|
||||||
)
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
extension NoteEditingSessionClient: MCSessionDelegate {
|
||||||
|
func session(
|
||||||
|
_ session: MCSession,
|
||||||
|
peer peerID: MCPeerID,
|
||||||
|
didChange state: MCSessionState
|
||||||
|
) {}
|
||||||
|
|
||||||
|
func session(
|
||||||
|
_ session: MCSession,
|
||||||
|
didReceive data: Data,
|
||||||
|
fromPeer peerID: MCPeerID
|
||||||
|
) {
|
||||||
|
guard let message = try? JSONDecoder().decode(NoteMessage.self, from: data) else { return }
|
||||||
|
DispatchQueue.main.async {
|
||||||
|
self.noteChangesEmitter.send(message)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func session(_ session: MCSession, didReceive stream: InputStream, withName streamName: String, fromPeer peerID: MCPeerID) {}
|
||||||
|
func session(_ session: MCSession, didStartReceivingResourceWithName resourceName: String, fromPeer peerID: MCPeerID, with progress: Progress) {}
|
||||||
|
func session(_ session: MCSession, didFinishReceivingResourceWithName resourceName: String, fromPeer peerID: MCPeerID, at localURL: URL?, withError error: (any Error)?) {}
|
||||||
|
}
|
||||||
|
|||||||
Reference in New Issue
Block a user