Bugfixes, simplifications

This commit is contained in:
2026-05-16 13:58:38 +02:00
parent db92487782
commit 10e08435a7
6 changed files with 150 additions and 162 deletions
+1 -1
View File
@@ -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)?) {}
}