mirror of
https://github.com/zjs81/meshcore-open.git
synced 2026-04-20 22:13:48 +00:00
- Implemented radio statistics features in multiple screens including chat, channels, and settings. - Added localization for new strings in Swedish, Ukrainian, and Chinese. - Introduced a setting to jump to the oldest unread message in chat and channels. - Enhanced path management and display for contacts and messages. - Updated app settings to include new boolean for jumping to the oldest unread message. - Improved battery indicator and radio stats display in the app bar. - Removed unused wakelock_plus dependency and updated plugin registrations.
225 lines
6.6 KiB
Dart
225 lines
6.6 KiB
Dart
import 'dart:typed_data';
|
|
import 'package:meshcore_open/utils/app_logger.dart';
|
|
|
|
import '../connector/meshcore_protocol.dart';
|
|
|
|
class Contact {
|
|
final Uint8List publicKey;
|
|
final String name;
|
|
final int type;
|
|
final int flags;
|
|
final int pathLength; // -1 = flood, 0+ = direct hops (from device)
|
|
final Uint8List path; // Path bytes from device
|
|
final int?
|
|
pathOverride; // User's path override: -1 = force flood, null = auto
|
|
final Uint8List? pathOverrideBytes; // User's path override bytes
|
|
final double? latitude;
|
|
final double? longitude;
|
|
final DateTime lastSeen;
|
|
final DateTime lastMessageAt;
|
|
final bool isActive;
|
|
final bool wasPulled;
|
|
final Uint8List? rawPacket;
|
|
|
|
Contact({
|
|
required this.publicKey,
|
|
required this.name,
|
|
required this.type,
|
|
this.flags = 0,
|
|
required this.pathLength,
|
|
required this.path,
|
|
this.pathOverride,
|
|
this.pathOverrideBytes,
|
|
this.latitude,
|
|
this.longitude,
|
|
required this.lastSeen,
|
|
DateTime? lastMessageAt,
|
|
this.isActive = true,
|
|
this.wasPulled = false,
|
|
this.rawPacket,
|
|
}) : lastMessageAt = lastMessageAt ?? lastSeen;
|
|
|
|
String get publicKeyHex => pubKeyToHex(publicKey);
|
|
|
|
String get typeLabel {
|
|
switch (type) {
|
|
case advTypeChat:
|
|
return 'Chat';
|
|
case advTypeRepeater:
|
|
return 'Repeater';
|
|
case advTypeRoom:
|
|
return 'Room';
|
|
case advTypeSensor:
|
|
return 'Sensor';
|
|
default:
|
|
return 'Unknown';
|
|
}
|
|
}
|
|
|
|
String get pathLabel {
|
|
if (pathOverride != null) {
|
|
if (pathOverride! < 0) return 'Flood (forced)';
|
|
if (pathOverride == 0) return 'Direct (forced)';
|
|
return '$pathOverride hops (forced)';
|
|
}
|
|
if (pathLength < 0) return 'Flood';
|
|
if (pathLength == 0) return 'Direct';
|
|
return '$pathLength hops';
|
|
}
|
|
|
|
bool get hasLocation {
|
|
const double epsilon = 1e-6;
|
|
final lat = latitude ?? 0.0;
|
|
final lon = longitude ?? 0.0;
|
|
return (lat.abs() > epsilon || lon.abs() > epsilon) &&
|
|
lat >= -90.0 &&
|
|
lat <= 90.0 &&
|
|
lon >= -180.0 &&
|
|
lon <= 180.0;
|
|
}
|
|
|
|
bool get isFavorite => (flags & contactFlagFavorite) != 0;
|
|
|
|
Contact copyWith({
|
|
Uint8List? publicKey,
|
|
String? name,
|
|
int? type,
|
|
int? flags,
|
|
int? pathLength,
|
|
Uint8List? path,
|
|
int? pathOverride,
|
|
Uint8List? pathOverrideBytes,
|
|
bool clearPathOverride = false,
|
|
double? latitude,
|
|
double? longitude,
|
|
DateTime? lastSeen,
|
|
DateTime? lastMessageAt,
|
|
bool? isActive,
|
|
Uint8List? rawPacket,
|
|
}) {
|
|
return Contact(
|
|
publicKey: publicKey ?? this.publicKey,
|
|
name: name ?? this.name,
|
|
type: type ?? this.type,
|
|
flags: flags ?? this.flags,
|
|
pathLength: pathLength ?? this.pathLength,
|
|
path: path ?? this.path,
|
|
pathOverride: clearPathOverride
|
|
? null
|
|
: (pathOverride ?? this.pathOverride),
|
|
pathOverrideBytes: clearPathOverride
|
|
? null
|
|
: (pathOverrideBytes ?? this.pathOverrideBytes),
|
|
latitude: latitude ?? this.latitude,
|
|
longitude: longitude ?? this.longitude,
|
|
lastSeen: lastSeen ?? this.lastSeen,
|
|
lastMessageAt: lastMessageAt ?? this.lastMessageAt,
|
|
isActive: isActive ?? this.isActive,
|
|
rawPacket: rawPacket ?? this.rawPacket,
|
|
);
|
|
}
|
|
|
|
/// Formats path bytes into comma-separated hex groups of [hashByteWidth] bytes.
|
|
String pathFormattedIdList(int hashByteWidth) {
|
|
final pathBytes = pathBytesForDisplay;
|
|
if (pathBytes.isEmpty) return '';
|
|
final w = hashByteWidth.clamp(1, 8);
|
|
final parts = <String>[];
|
|
for (int i = 0; i < pathBytes.length; i += w) {
|
|
final end = (i + w) <= pathBytes.length ? (i + w) : pathBytes.length;
|
|
final chunk = pathBytes.sublist(i, end);
|
|
parts.add(
|
|
chunk
|
|
.map((b) => b.toRadixString(16).padLeft(2, '0').toUpperCase())
|
|
.join(),
|
|
);
|
|
}
|
|
return parts.join(',');
|
|
}
|
|
|
|
/// Default grouping uses legacy single-byte hop hash width.
|
|
String get pathIdList => pathFormattedIdList(pathHashSize);
|
|
|
|
String get shortPubKeyHex {
|
|
return "<${publicKeyHex.substring(0, 8)}...${publicKeyHex.substring(publicKeyHex.length - 8)}>";
|
|
}
|
|
|
|
Uint8List get pathBytesForDisplay {
|
|
if (pathOverride != null) {
|
|
if (pathOverride! < 0) return Uint8List(0);
|
|
return pathOverrideBytes ?? Uint8List(0);
|
|
}
|
|
return path;
|
|
}
|
|
|
|
static Contact? fromFrame(Uint8List data) {
|
|
if (data.isEmpty) return null;
|
|
final reader = BufferReader(data);
|
|
try {
|
|
final respCode = reader.readByte();
|
|
if (respCode != respCodeContact && respCode != pushCodeNewAdvert) {
|
|
return null;
|
|
}
|
|
final pubKey = reader.readBytes(pubKeySize);
|
|
|
|
// Guard: reject contacts with zeroed or mostly-zeroed public keys
|
|
// (indicates corrupt flash storage on the firmware side)
|
|
final zeroCount = pubKey.where((b) => b == 0).length;
|
|
if (zeroCount > pubKeySize ~/ 2) return null;
|
|
|
|
final type = reader.readByte();
|
|
final flags = reader.readByte();
|
|
final pathLen = reader.readByte();
|
|
final safePathLen = pathLen > 0
|
|
? (pathLen > maxPathSize ? maxPathSize : pathLen)
|
|
: 0;
|
|
final pathBytes = reader.readBytes(maxPathSize).sublist(0, safePathLen);
|
|
final name = reader.readCStringGreedy(maxNameSize);
|
|
|
|
// Guard: reject contacts with non-printable names (corrupt flash data)
|
|
if (name.isNotEmpty &&
|
|
name.codeUnits.every((c) => c < 0x20 || c == 0xFFFD)) {
|
|
return null;
|
|
}
|
|
|
|
final lastMod = reader.readUInt32LE();
|
|
|
|
double? lat, lon;
|
|
if (reader.remaining >= 8) {
|
|
final latRaw = reader.readInt32LE();
|
|
final lonRaw = reader.readInt32LE();
|
|
if (latRaw != 0 || lonRaw != 0) {
|
|
lat = latRaw / 1e6;
|
|
lon = lonRaw / 1e6;
|
|
}
|
|
}
|
|
|
|
return Contact(
|
|
publicKey: pubKey,
|
|
name: name.isEmpty ? 'Unknown' : name,
|
|
type: type,
|
|
flags: flags,
|
|
pathLength: (pathLen == 0xFF || pathLen > maxPathSize) ? -1 : pathLen,
|
|
path: pathBytes,
|
|
latitude: lat,
|
|
longitude: lon,
|
|
lastSeen: DateTime.fromMillisecondsSinceEpoch(lastMod * 1000),
|
|
isActive: true,
|
|
rawPacket: null,
|
|
);
|
|
} catch (e) {
|
|
appLogger.error('Failed to parse contact frame: $e');
|
|
return null;
|
|
}
|
|
}
|
|
|
|
@override
|
|
bool operator ==(Object other) =>
|
|
other is Contact && publicKeyHex == other.publicKeyHex;
|
|
|
|
@override
|
|
int get hashCode => publicKeyHex.hashCode;
|
|
bool get teleBaseEnabled => (flags & contactFlagTeleBase) != 0;
|
|
bool get teleLocEnabled => (flags & contactFlagTeleLoc) != 0;
|
|
bool get teleEnvEnabled => (flags & contactFlagTeleEnv) != 0;
|
|
}
|