2021-12-24 23:41:18 -08:00
|
|
|
|
//
|
2025-10-05 17:51:18 -07:00
|
|
|
|
// UserMessageList.swift
|
|
|
|
|
|
// MeshtasticApple
|
2022-01-01 08:03:46 -08:00
|
|
|
|
//
|
2025-10-05 17:51:18 -07:00
|
|
|
|
// Created by Garth Vander Houwen on 12/24/21.
|
2022-01-01 08:03:46 -08:00
|
|
|
|
//
|
|
|
|
|
|
|
|
|
|
|
|
import SwiftUI
|
|
|
|
|
|
import CoreData
|
2024-06-03 02:17:55 -07:00
|
|
|
|
import OSLog
|
2025-10-05 17:51:18 -07:00
|
|
|
|
import MeshtasticProtobufs // Added to ensure RoutingError is accessible if needed
|
2022-01-01 08:03:46 -08:00
|
|
|
|
|
2022-11-05 08:26:27 -07:00
|
|
|
|
struct UserMessageList: View {
|
2024-07-10 21:17:14 -05:00
|
|
|
|
@EnvironmentObject var appState: AppState
|
Transports Interface to Support TCP for all Platforms and Serial on Mac (#1341)
* Initial implementation of transports
* Initial LogRadio implementation
* Fixes for Settings view (caused by debug commenting)
* Refinement of the object and actor model
* Connect view text and tab updates
* Fix mac catalyst and tests
* Warning and logging clean-up
* In progress commit
* Serial Transport and Reconnect draft work
* Serial transport and reconnection draft work
* Quick fix for BLE - still more work to do
* interim commit
* More in progress changes
* Minor improvements
* Pretty good initial implementation
* Bump version beyond the app store
* Fix for disconnection swipeAction
* Tweaks to TCPConnection implementation
* Retry for NONCE_ONLY_DB
* Revert json string change
* Simplified some of the API + "Anti-discovery"
* Tweaks for devices leaving the discovery process
* Bump version
* iOS26 Tweaks
* Tweaks and bug fixes
* Add link with slash sf symbol
* update symbol image on connect view
* BLE disconnect handling
* Log privacy attributes
* Onboarding and minor fixes.
* change database to nodes, add emoji to tcp logs
* Error handling improvements
* More logging emojis
* Suppressed unnecessary errors on disconnect
* Heartbeat emoji
* Add bluetooth symbol
* add privacy attributes to [TCP] logs, add custom bluetooth logo
* Improve routing logs
* Emoji for connect logs
* Heartbeat emoji
* Add CBCentralManagerScanOptionAllowDuplicatesKey options to central for bluetooth
* fix nav errors by switching from observableobject to state
* Update connection indicator icon
* fix for BLE disconnects
* Connection process fixes
* More fixes/tweaks to connection process
* Strict concurrency
* Fix some warnings, remove wifi warning
* delete stale keys
* interim commit
* Update privacy for log, fix wrong space
* fix a couple of linting items
* Switch to targeted
* interim commit
* BLE Signal strenth on connect view
* Remove BLE RSSI from long press menu
* Modem lights
* minor spacing tweak
* Additional BLE logging and a scanning fix.
* Discovery and BLE RSSI improvements
* Background suspension
* Update isConnected to enable UI during db load
* update protobufs
* Replace config if statements with switches, Fix unknown module config logging, make dark mode modem circle stroke color white so they are visible
* Additional logging cleanup
* hast
* Set unmessagable to true if the longname has the unmessagable emoji
* Connect error handling improvements
* Admin popup list icon and activity lights updates
* Revert use of .toolbar back to .navigationBarItems
* More public logging
* Better BLE error handling
* Node DB progress meter
* minor tweak to activity light interaction timing
* Fix comment linting, remove stale keys
* Remove stale keys
* Easy linting fixes
* Two more simple linting fixes
* clean up meshtasticapp
* More public logging
* Replay config
* Logging
* Fix for unselected node on Settings
* Tweak to progress meter based on device idiom
* Update protos
* Session replay redaction of messages
* Serial fix for old devices, and a let statement
* Mask text too
* Fix typo
* BLE poweredOff is now an auto-reconnectable error
* Update logging
* Fix for peerRemovedPairingInformation
* Logging for BLE peripheral:didUpdateValueFor errors.
* Fix for inconsistent swipe disconnect behavior
* periperal:didUpdateValueFor error handling
* Fix for BLEConnection continuation guarding
* BLEConnection actor deadlock on disconnect
* Heartbeat nonce
* Fix for swipe disconnect and task cancellation
* Fix for swipe actions not honoring .disabled()
* Tell BLETransport when BLEConnection is cancelled
* Update navigation logging
* Logging updates
* Bump version to 2.7.0
* Organize into folders and heartbeat stuff
* Minor improvements to manual TCP connection
* Auto-connect toggle
* Possible BLE bug, still waiting to see in logs
* Concurrency tweaks
* Concurrency improvements
* requestDeviceMetadata fix. fixes remote admin
* Minor typo fixes
* "All" button for log filters: category and level
* More robust continuation handling for BLE
* @FetchRequest based ChannelMessageList
* Update info.plist and device hardware file
* Move auto connect toggle to app settings and debug mode, tint properly with the accent color
* Add label to auto connect toggle
* Update log for node info received from ourselves over the mesh
* Remove unused scrollViewProxy
* Update Meshtastic/Views/Onboarding/DeviceOnboarding.swift
Co-authored-by: Copilot <175728472+Copilot@users.noreply.github.com>
* Update target for connect view
* Properly Set datadog environment
* Comment out ble manager
* Adjust cyclomatic complexity thresholds in .swiftlint.yml
* Linting fixes, delete ble manager
* Make session replay debug only
---------
Co-authored-by: jake-b <jake-b@users.noreply.github.com>
Co-authored-by: jake <jake@jakes-Mac-mini.local>
Co-authored-by: Copilot <175728472+Copilot@users.noreply.github.com>
2025-08-27 08:09:02 -07:00
|
|
|
|
@EnvironmentObject var accessoryManager: AccessoryManager
|
2025-10-17 18:16:00 -07:00
|
|
|
|
@Environment(\.scenePhase) var scenePhase
|
2024-07-10 21:17:14 -05:00
|
|
|
|
@Environment(\.managedObjectContext) var context
|
2024-02-05 23:31:54 -07:00
|
|
|
|
@FocusState var messageFieldFocused: Bool
|
2023-09-04 06:31:38 -07:00
|
|
|
|
@ObservedObject var user: UserEntity
|
2022-01-01 22:55:25 -08:00
|
|
|
|
@State private var replyMessageId: Int64 = 0
|
2025-05-02 16:54:16 -07:00
|
|
|
|
@State private var messageToHighlight: Int64 = 0
|
2025-10-05 17:51:18 -07:00
|
|
|
|
@State private var redrawTapbacksTrigger = UUID()
|
|
|
|
|
|
@AppStorage("preferredPeripheralNum") private var preferredPeripheralNum = -1
|
2025-10-17 18:16:00 -07:00
|
|
|
|
@FetchRequest private var allPrivateMessages: FetchedResults<MessageEntity>
|
|
|
|
|
|
|
|
|
|
|
|
init(user: UserEntity) {
|
|
|
|
|
|
self.user = user
|
|
|
|
|
|
|
|
|
|
|
|
// Configure fetch request here
|
|
|
|
|
|
let request: NSFetchRequest<MessageEntity> = user.messageFetchRequest
|
|
|
|
|
|
_allPrivateMessages = FetchRequest(fetchRequest: request)
|
2025-10-05 17:51:18 -07:00
|
|
|
|
}
|
2025-10-17 18:16:00 -07:00
|
|
|
|
|
2025-10-05 17:51:18 -07:00
|
|
|
|
func handleInteractionComplete() {
|
|
|
|
|
|
markMessagesAsRead()
|
|
|
|
|
|
redrawTapbacksTrigger = UUID()
|
|
|
|
|
|
}
|
2025-10-17 18:16:00 -07:00
|
|
|
|
|
2025-10-05 17:51:18 -07:00
|
|
|
|
func markMessagesAsRead() {
|
|
|
|
|
|
do {
|
|
|
|
|
|
for unreadMessage in allPrivateMessages.filter({ !$0.read }) {
|
|
|
|
|
|
unreadMessage.read = true
|
|
|
|
|
|
}
|
|
|
|
|
|
try context.save()
|
|
|
|
|
|
Logger.data.info("📖 [App] All unread direct messages marked as read for user \(user.num, privacy: .public).")
|
2025-10-17 18:16:00 -07:00
|
|
|
|
|
|
|
|
|
|
if let connectedPeripheralNum = accessoryManager.activeDeviceNum,
|
|
|
|
|
|
let connectedNode = getNodeInfo(id: connectedPeripheralNum, context: context),
|
|
|
|
|
|
let connectedUser = connectedNode.user {
|
|
|
|
|
|
appState.unreadDirectMessages = connectedUser.unreadMessages(context: context, skipLastMessageCheck: true) // skipLastMessageCheck=true because we don't update lastMessage on our own connected node
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2025-10-05 17:51:18 -07:00
|
|
|
|
context.refresh(user, mergeChanges: true)
|
|
|
|
|
|
} catch {
|
|
|
|
|
|
Logger.data.error("Failed to read direct messages: \(error.localizedDescription, privacy: .public)")
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
2025-10-17 18:16:00 -07:00
|
|
|
|
|
|
|
|
|
|
private func routerIsShowingThisUser() -> Bool {
|
2025-10-28 14:34:39 -07:00
|
|
|
|
guard appState.router.navigationState.selectedTab == .messages else { return false }
|
2025-10-17 18:16:00 -07:00
|
|
|
|
return scenePhase == .active
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2022-12-24 22:06:28 -08:00
|
|
|
|
var body: some View {
|
2025-10-17 18:16:00 -07:00
|
|
|
|
// Cast user.messageList to an array for easier indexing and ForEach.
|
|
|
|
|
|
let messages: [MessageEntity] = Array(allPrivateMessages)
|
|
|
|
|
|
|
|
|
|
|
|
// Precompute previous message
|
|
|
|
|
|
let previousByID: [Int64: MessageEntity?] = {
|
|
|
|
|
|
var dict = [Int64: MessageEntity?]()
|
|
|
|
|
|
var prev: MessageEntity?
|
|
|
|
|
|
for m in messages { dict[m.messageId] = prev; prev = m }
|
|
|
|
|
|
return dict
|
|
|
|
|
|
}()
|
|
|
|
|
|
|
2023-09-16 08:48:36 -07:00
|
|
|
|
VStack {
|
2022-01-01 15:45:00 -08:00
|
|
|
|
ScrollViewReader { scrollView in
|
2025-10-05 17:51:18 -07:00
|
|
|
|
ScrollView {
|
|
|
|
|
|
LazyVStack {
|
2025-10-17 18:16:00 -07:00
|
|
|
|
ForEach(messages, id: \.messageId) { message in
|
|
|
|
|
|
let previousMessage: MessageEntity? = previousByID[message.messageId] ?? nil
|
2025-10-05 17:51:18 -07:00
|
|
|
|
|
|
|
|
|
|
UserMessageRow(
|
|
|
|
|
|
message: message,
|
2025-10-17 18:16:00 -07:00
|
|
|
|
allMessages: messages,
|
2025-10-05 17:51:18 -07:00
|
|
|
|
previousMessage: previousMessage,
|
|
|
|
|
|
preferredPeripheralNum: preferredPeripheralNum,
|
|
|
|
|
|
user: user,
|
|
|
|
|
|
replyMessageId: $replyMessageId,
|
|
|
|
|
|
messageFieldFocused: $messageFieldFocused,
|
|
|
|
|
|
messageToHighlight: $messageToHighlight,
|
|
|
|
|
|
scrollView: scrollView,
|
|
|
|
|
|
onInteractionComplete: handleInteractionComplete
|
|
|
|
|
|
)
|
|
|
|
|
|
.onAppear {
|
2025-10-10 14:07:36 -07:00
|
|
|
|
// Only mark as read if the app is in the foreground
|
|
|
|
|
|
if !message.read && UIApplication.shared.applicationState == .active {
|
|
|
|
|
|
message.read = true
|
|
|
|
|
|
LocalNotificationManager().cancelNotificationForMessageId(message.messageId)
|
|
|
|
|
|
// Race condition, sometimes the app doesn't update unread count if we run this too early
|
|
|
|
|
|
// So, run it in the main queue after everything saves and stabilizes
|
|
|
|
|
|
DispatchQueue.main.async {
|
|
|
|
|
|
markMessagesAsRead()
|
|
|
|
|
|
scrollView.scrollTo("bottomAnchor", anchor: .bottom)
|
|
|
|
|
|
}
|
2025-04-02 17:44:01 -07:00
|
|
|
|
}
|
|
|
|
|
|
}
|
2025-10-17 18:16:00 -07:00
|
|
|
|
|
2025-04-24 18:11:58 -07:00
|
|
|
|
}
|
2025-10-05 17:51:18 -07:00
|
|
|
|
// Invisible spacer to detect reaching bottom
|
|
|
|
|
|
Color.clear
|
|
|
|
|
|
.frame(height: 1)
|
|
|
|
|
|
.id("bottomAnchor")
|
2025-02-06 20:01:26 -05:00
|
|
|
|
}
|
2025-10-05 17:51:18 -07:00
|
|
|
|
}
|
|
|
|
|
|
.defaultScrollAnchor(.bottom)
|
|
|
|
|
|
.defaultScrollAnchorTopAlignment()
|
|
|
|
|
|
.defaultScrollAnchorBottomSizeChanges()
|
|
|
|
|
|
.scrollDismissesKeyboard(.immediately)
|
|
|
|
|
|
.onChange(of: messageFieldFocused) {
|
|
|
|
|
|
if messageFieldFocused {
|
|
|
|
|
|
DispatchQueue.main.asyncAfter(deadline: .now() + 0.1) {
|
|
|
|
|
|
scrollView.scrollTo("bottomAnchor", anchor: .bottom)
|
2025-04-24 18:11:58 -07:00
|
|
|
|
}
|
2022-10-17 18:52:49 -07:00
|
|
|
|
}
|
2024-10-05 15:50:57 -07:00
|
|
|
|
}
|
2022-01-01 15:45:00 -08:00
|
|
|
|
}
|
2024-02-05 23:31:54 -07:00
|
|
|
|
TextMessageField(
|
2024-02-17 13:26:09 -07:00
|
|
|
|
destination: .user(user),
|
2024-02-05 23:31:54 -07:00
|
|
|
|
replyMessageId: $replyMessageId,
|
|
|
|
|
|
isFocused: $messageFieldFocused
|
2025-09-08 20:19:29 -07:00
|
|
|
|
)
|
2022-01-01 15:45:00 -08:00
|
|
|
|
}
|
2025-09-08 20:19:29 -07:00
|
|
|
|
.navigationBarTitleDisplayMode(.inline)
|
2022-01-01 08:03:46 -08:00
|
|
|
|
.toolbar {
|
2025-08-13 21:09:11 -07:00
|
|
|
|
if !user.keyMatch {
|
|
|
|
|
|
ToolbarItem(placement: .bottomBar) {
|
|
|
|
|
|
VStack {
|
|
|
|
|
|
HStack {
|
|
|
|
|
|
Image(systemName: "key.slash.fill")
|
|
|
|
|
|
.symbolRenderingMode(.multicolor)
|
|
|
|
|
|
.foregroundStyle(.red)
|
|
|
|
|
|
.font(.caption2)
|
|
|
|
|
|
Text("There is an issue with this contact's public key.")
|
|
|
|
|
|
.foregroundStyle(.secondary)
|
|
|
|
|
|
.font(.caption2)
|
|
|
|
|
|
}
|
|
|
|
|
|
Link(destination: URL(string: "meshtastic:///nodes?nodenum=\(user.num)")!) {
|
|
|
|
|
|
Text("Details...")
|
|
|
|
|
|
.font(.caption2)
|
|
|
|
|
|
.offset(y: -15)
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
.offset(y: -15)
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
2022-01-01 08:03:46 -08:00
|
|
|
|
ToolbarItem(placement: .principal) {
|
|
|
|
|
|
HStack {
|
2023-09-02 18:02:51 -07:00
|
|
|
|
CircleText(text: user.shortName ?? "?", color: Color(UIColor(hex: UInt32(user.num))), circleSize: 44)
|
2025-10-05 17:51:18 -07:00
|
|
|
|
Text(user.longName ?? "Unknown").font(.headline)
|
2022-01-01 08:03:46 -08:00
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
ToolbarItem(placement: .navigationBarTrailing) {
|
|
|
|
|
|
ZStack {
|
|
|
|
|
|
ConnectedDevice(
|
Transports Interface to Support TCP for all Platforms and Serial on Mac (#1341)
* Initial implementation of transports
* Initial LogRadio implementation
* Fixes for Settings view (caused by debug commenting)
* Refinement of the object and actor model
* Connect view text and tab updates
* Fix mac catalyst and tests
* Warning and logging clean-up
* In progress commit
* Serial Transport and Reconnect draft work
* Serial transport and reconnection draft work
* Quick fix for BLE - still more work to do
* interim commit
* More in progress changes
* Minor improvements
* Pretty good initial implementation
* Bump version beyond the app store
* Fix for disconnection swipeAction
* Tweaks to TCPConnection implementation
* Retry for NONCE_ONLY_DB
* Revert json string change
* Simplified some of the API + "Anti-discovery"
* Tweaks for devices leaving the discovery process
* Bump version
* iOS26 Tweaks
* Tweaks and bug fixes
* Add link with slash sf symbol
* update symbol image on connect view
* BLE disconnect handling
* Log privacy attributes
* Onboarding and minor fixes.
* change database to nodes, add emoji to tcp logs
* Error handling improvements
* More logging emojis
* Suppressed unnecessary errors on disconnect
* Heartbeat emoji
* Add bluetooth symbol
* add privacy attributes to [TCP] logs, add custom bluetooth logo
* Improve routing logs
* Emoji for connect logs
* Heartbeat emoji
* Add CBCentralManagerScanOptionAllowDuplicatesKey options to central for bluetooth
* fix nav errors by switching from observableobject to state
* Update connection indicator icon
* fix for BLE disconnects
* Connection process fixes
* More fixes/tweaks to connection process
* Strict concurrency
* Fix some warnings, remove wifi warning
* delete stale keys
* interim commit
* Update privacy for log, fix wrong space
* fix a couple of linting items
* Switch to targeted
* interim commit
* BLE Signal strenth on connect view
* Remove BLE RSSI from long press menu
* Modem lights
* minor spacing tweak
* Additional BLE logging and a scanning fix.
* Discovery and BLE RSSI improvements
* Background suspension
* Update isConnected to enable UI during db load
* update protobufs
* Replace config if statements with switches, Fix unknown module config logging, make dark mode modem circle stroke color white so they are visible
* Additional logging cleanup
* hast
* Set unmessagable to true if the longname has the unmessagable emoji
* Connect error handling improvements
* Admin popup list icon and activity lights updates
* Revert use of .toolbar back to .navigationBarItems
* More public logging
* Better BLE error handling
* Node DB progress meter
* minor tweak to activity light interaction timing
* Fix comment linting, remove stale keys
* Remove stale keys
* Easy linting fixes
* Two more simple linting fixes
* clean up meshtasticapp
* More public logging
* Replay config
* Logging
* Fix for unselected node on Settings
* Tweak to progress meter based on device idiom
* Update protos
* Session replay redaction of messages
* Serial fix for old devices, and a let statement
* Mask text too
* Fix typo
* BLE poweredOff is now an auto-reconnectable error
* Update logging
* Fix for peerRemovedPairingInformation
* Logging for BLE peripheral:didUpdateValueFor errors.
* Fix for inconsistent swipe disconnect behavior
* periperal:didUpdateValueFor error handling
* Fix for BLEConnection continuation guarding
* BLEConnection actor deadlock on disconnect
* Heartbeat nonce
* Fix for swipe disconnect and task cancellation
* Fix for swipe actions not honoring .disabled()
* Tell BLETransport when BLEConnection is cancelled
* Update navigation logging
* Logging updates
* Bump version to 2.7.0
* Organize into folders and heartbeat stuff
* Minor improvements to manual TCP connection
* Auto-connect toggle
* Possible BLE bug, still waiting to see in logs
* Concurrency tweaks
* Concurrency improvements
* requestDeviceMetadata fix. fixes remote admin
* Minor typo fixes
* "All" button for log filters: category and level
* More robust continuation handling for BLE
* @FetchRequest based ChannelMessageList
* Update info.plist and device hardware file
* Move auto connect toggle to app settings and debug mode, tint properly with the accent color
* Add label to auto connect toggle
* Update log for node info received from ourselves over the mesh
* Remove unused scrollViewProxy
* Update Meshtastic/Views/Onboarding/DeviceOnboarding.swift
Co-authored-by: Copilot <175728472+Copilot@users.noreply.github.com>
* Update target for connect view
* Properly Set datadog environment
* Comment out ble manager
* Adjust cyclomatic complexity thresholds in .swiftlint.yml
* Linting fixes, delete ble manager
* Make session replay debug only
---------
Co-authored-by: jake-b <jake-b@users.noreply.github.com>
Co-authored-by: jake <jake@jakes-Mac-mini.local>
Co-authored-by: Copilot <175728472+Copilot@users.noreply.github.com>
2025-08-27 08:09:02 -07:00
|
|
|
|
deviceConnected: accessoryManager.isConnected,
|
|
|
|
|
|
name: accessoryManager.activeConnection?.device.shortName ?? "?")
|
2022-01-01 08:03:46 -08:00
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
2022-12-24 22:06:28 -08:00
|
|
|
|
}
|
2022-01-01 08:03:46 -08:00
|
|
|
|
}
|