feat: lots of fixes
This commit is contained in:
parent
5d617131ee
commit
48476f6fe4
@ -1,4 +1,5 @@
|
|||||||
<manifest xmlns:android="http://schemas.android.com/apk/res/android">
|
<manifest xmlns:android="http://schemas.android.com/apk/res/android">
|
||||||
|
<uses-permission android:name="android.permission.INTERNET" />
|
||||||
<application
|
<application
|
||||||
android:label="ot_viewer_app"
|
android:label="ot_viewer_app"
|
||||||
android:name="${applicationName}"
|
android:name="${applicationName}"
|
||||||
|
1
devtools_options.yaml
Normal file
1
devtools_options.yaml
Normal file
@ -0,0 +1 @@
|
|||||||
|
extensions:
|
27
lib/global_location_store.dart
Normal file
27
lib/global_location_store.dart
Normal file
@ -0,0 +1,27 @@
|
|||||||
|
|
||||||
|
|
||||||
|
import 'package:bloc/bloc.dart';
|
||||||
|
import 'package:fast_immutable_collections/fast_immutable_collections.dart';
|
||||||
|
import 'package:ot_viewer_app/owntracks_api.dart';
|
||||||
|
|
||||||
|
class GlobalLocationStoreState {
|
||||||
|
final IMap<(String, String), Point> locations;
|
||||||
|
|
||||||
|
GlobalLocationStoreState({required this.locations});
|
||||||
|
|
||||||
|
@override
|
||||||
|
String toString() => 'GlobalLocationStoreState(locations: $locations)';
|
||||||
|
}
|
||||||
|
|
||||||
|
// Define the Cubit
|
||||||
|
class GlobalLocationStoreCubit extends Cubit<GlobalLocationStoreState> {
|
||||||
|
GlobalLocationStoreCubit() : super(GlobalLocationStoreState(locations: IMap(const {})));
|
||||||
|
|
||||||
|
// Function to update the points
|
||||||
|
void updatePoint(String user, String device, Point point) {
|
||||||
|
// Create a new map with the updated point
|
||||||
|
final updatedLocations = state.locations.add((user, device), point);
|
||||||
|
// Emit the new state
|
||||||
|
emit(GlobalLocationStoreState(locations: updatedLocations));
|
||||||
|
}
|
||||||
|
}
|
@ -2,9 +2,11 @@ import 'package:flutter/foundation.dart';
|
|||||||
import 'package:flutter/material.dart';
|
import 'package:flutter/material.dart';
|
||||||
import 'package:flutter_bloc/flutter_bloc.dart';
|
import 'package:flutter_bloc/flutter_bloc.dart';
|
||||||
import 'package:hydrated_bloc/hydrated_bloc.dart';
|
import 'package:hydrated_bloc/hydrated_bloc.dart';
|
||||||
|
import 'package:ot_viewer_app/global_location_store.dart';
|
||||||
import 'package:ot_viewer_app/settings_page.dart';
|
import 'package:ot_viewer_app/settings_page.dart';
|
||||||
import 'package:path_provider/path_provider.dart';
|
import 'package:path_provider/path_provider.dart';
|
||||||
import 'map_page.dart';
|
import 'map_page.dart';
|
||||||
|
import 'package:get_it/get_it.dart';
|
||||||
import 'owntracks_api.dart';
|
import 'owntracks_api.dart';
|
||||||
|
|
||||||
Future<void> main() async {
|
Future<void> main() async {
|
||||||
@ -16,6 +18,8 @@ Future<void> main() async {
|
|||||||
: await getApplicationDocumentsDirectory(),
|
: await getApplicationDocumentsDirectory(),
|
||||||
);
|
);
|
||||||
|
|
||||||
|
GetIt.I
|
||||||
|
.registerSingleton<GlobalLocationStoreCubit>(GlobalLocationStoreCubit());
|
||||||
|
|
||||||
runApp(MyApp());
|
runApp(MyApp());
|
||||||
}
|
}
|
||||||
@ -28,7 +32,7 @@ class MyApp extends StatelessWidget {
|
|||||||
theme: ThemeData.dark(),
|
theme: ThemeData.dark(),
|
||||||
home: Scaffold(
|
home: Scaffold(
|
||||||
appBar: AppBar(title: const Text('OwnTrakcs Data Viewer')),
|
appBar: AppBar(title: const Text('OwnTrakcs Data Viewer')),
|
||||||
body: MainPage(),
|
body: const MainPage(),
|
||||||
),
|
),
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
@ -45,7 +49,7 @@ class _MainPageState extends State<MainPage> {
|
|||||||
int _currentIndex = 0;
|
int _currentIndex = 0;
|
||||||
final List<Widget> _pages = [
|
final List<Widget> _pages = [
|
||||||
const MapPage(),
|
const MapPage(),
|
||||||
SettingsPage(), // Assume this is your settings page widget
|
const SettingsPage(), // Assume this is your settings page widget
|
||||||
];
|
];
|
||||||
|
|
||||||
void _onItemTapped(int index) {
|
void _onItemTapped(int index) {
|
||||||
|
@ -1,5 +1,6 @@
|
|||||||
import 'package:bloc/bloc.dart';
|
import 'package:bloc/bloc.dart';
|
||||||
import 'package:dio_cache_interceptor_hive_store/dio_cache_interceptor_hive_store.dart';
|
import 'package:dio_cache_interceptor_hive_store/dio_cache_interceptor_hive_store.dart';
|
||||||
|
import 'package:fast_immutable_collections/fast_immutable_collections.dart';
|
||||||
import 'package:flutter/cupertino.dart';
|
import 'package:flutter/cupertino.dart';
|
||||||
import 'package:flutter/material.dart';
|
import 'package:flutter/material.dart';
|
||||||
import 'package:flutter/widgets.dart';
|
import 'package:flutter/widgets.dart';
|
||||||
@ -7,8 +8,11 @@ import 'package:flutter_bloc/flutter_bloc.dart';
|
|||||||
import 'package:flutter_map/flutter_map.dart';
|
import 'package:flutter_map/flutter_map.dart';
|
||||||
import 'package:flutter_map_cache/flutter_map_cache.dart';
|
import 'package:flutter_map_cache/flutter_map_cache.dart';
|
||||||
import 'package:flutter_map_compass/flutter_map_compass.dart';
|
import 'package:flutter_map_compass/flutter_map_compass.dart';
|
||||||
|
import 'package:flutter_map_math/flutter_geo_math.dart';
|
||||||
|
import 'package:get_it/get_it.dart';
|
||||||
import 'package:intl/intl.dart';
|
import 'package:intl/intl.dart';
|
||||||
import 'package:latlong2/latlong.dart';
|
import 'package:latlong2/latlong.dart';
|
||||||
|
import 'package:ot_viewer_app/global_location_store.dart';
|
||||||
import 'package:ot_viewer_app/settings_page.dart';
|
import 'package:ot_viewer_app/settings_page.dart';
|
||||||
import 'package:ot_viewer_app/user_path_bloc.dart';
|
import 'package:ot_viewer_app/user_path_bloc.dart';
|
||||||
import 'package:ot_viewer_app/util.dart';
|
import 'package:ot_viewer_app/util.dart';
|
||||||
@ -58,15 +62,14 @@ class _MapPageState extends State<MapPage> {
|
|||||||
return BlocProvider(
|
return BlocProvider(
|
||||||
create: (context) {
|
create: (context) {
|
||||||
final cubit = LocationSubscribeCubit();
|
final cubit = LocationSubscribeCubit();
|
||||||
final settings_cubit = context.read<SettingsCubit>();
|
final settingsCubit = context.read<SettingsCubit>();
|
||||||
cubit.subscribe(settings_cubit.state);
|
cubit.subscribe(settingsCubit.state);
|
||||||
settings_cubit.stream.listen((settings) => cubit.subscribe(settings));
|
settingsCubit.stream.listen((settings) => cubit.subscribe(settings));
|
||||||
return cubit;
|
return cubit;
|
||||||
},
|
},
|
||||||
child: FutureBuilder<String>(
|
child: FutureBuilder<String>(
|
||||||
future: getPath(),
|
future: getPath(),
|
||||||
builder: (something, temp_path) =>
|
builder: (something, tempPath) => BlocBuilder<SettingsCubit, SettingsState>(
|
||||||
BlocBuilder<SettingsCubit, SettingsState>(
|
|
||||||
builder: (context, state) {
|
builder: (context, state) {
|
||||||
return FlutterMap(
|
return FlutterMap(
|
||||||
options: const MapOptions(
|
options: const MapOptions(
|
||||||
@ -78,20 +81,16 @@ class _MapPageState extends State<MapPage> {
|
|||||||
urlTemplate: 'https://tile.openstreetmap.org/{z}/{x}/{y}.png',
|
urlTemplate: 'https://tile.openstreetmap.org/{z}/{x}/{y}.png',
|
||||||
tileProvider: CachedTileProvider(
|
tileProvider: CachedTileProvider(
|
||||||
maxStale: const Duration(days: 30),
|
maxStale: const Duration(days: 30),
|
||||||
store: HiveCacheStore(temp_path.data,
|
store: HiveCacheStore(tempPath.data, hiveBoxName: 'HiveCacheStore')),
|
||||||
hiveBoxName: 'HiveCacheStore')),
|
|
||||||
),
|
),
|
||||||
...state.activeDevices
|
...state.activeDevices.map((id) => UserPath(device: id, settings: state)),
|
||||||
.map((id) => UserPath(device: id, settings: state)),
|
const MapCompass.cupertino(rotationDuration: Duration(milliseconds: 600)),
|
||||||
const MapCompass.cupertino(
|
|
||||||
rotationDuration: Duration(milliseconds: 600)),
|
|
||||||
// CurrentLocationLayer(), TODO: add permission
|
// CurrentLocationLayer(), TODO: add permission
|
||||||
RichAttributionWidget(
|
RichAttributionWidget(
|
||||||
attributions: [
|
attributions: [
|
||||||
TextSourceAttribution(
|
TextSourceAttribution(
|
||||||
'OpenStreetMap contributors',
|
'OpenStreetMap contributors',
|
||||||
onTap: () =>
|
onTap: () => (Uri.parse('https://openstreetmap.org/copyright')),
|
||||||
(Uri.parse('https://openstreetmap.org/copyright')),
|
|
||||||
),
|
),
|
||||||
],
|
],
|
||||||
),
|
),
|
||||||
@ -124,14 +123,12 @@ class _UserPathState extends State<UserPath> {
|
|||||||
return bloc;
|
return bloc;
|
||||||
},
|
},
|
||||||
child: BlocListener<LocationSubscribeCubit, LocationUpdateState>(
|
child: BlocListener<LocationSubscribeCubit, LocationUpdateState>(
|
||||||
|
listenWhen: (prevState, state) => state is LocationUpdateReceived,
|
||||||
listener: (context, state) {
|
listener: (context, state) {
|
||||||
UserPathBloc userPathBloc = context.read<UserPathBloc>();
|
UserPathBloc userPathBloc = context.read<UserPathBloc>();
|
||||||
if (state
|
if (state case LocationUpdateReceived(:final position, :final deviceId)) {
|
||||||
case LocationUpdateReceived(:final position, :final deviceId)) {
|
|
||||||
if (userPathBloc.deviceId == state.deviceId) {
|
if (userPathBloc.deviceId == state.deviceId) {
|
||||||
context
|
context.read<UserPathBloc>().add(UserPathLiveSubscriptionUpdate(position));
|
||||||
.read<UserPathBloc>()
|
|
||||||
.add(UserPathLiveSubscriptionUpdate(position));
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
@ -140,10 +137,10 @@ class _UserPathState extends State<UserPath> {
|
|||||||
print("rebuild");
|
print("rebuild");
|
||||||
final _istate = state as MainUserPathState;
|
final _istate = state as MainUserPathState;
|
||||||
// make markers
|
// make markers
|
||||||
final List<Marker> _markers = [];
|
final List<Marker> markers = [];
|
||||||
|
|
||||||
if (state.livePoints.isNotEmpty) {
|
if (state.livePoints.isNotEmpty) {
|
||||||
_markers.add(Marker(
|
markers.add(Marker(
|
||||||
width: 500,
|
width: 500,
|
||||||
height: 100,
|
height: 100,
|
||||||
point: state.livePoints.last.asLatLng,
|
point: state.livePoints.last.asLatLng,
|
||||||
@ -151,25 +148,23 @@ class _UserPathState extends State<UserPath> {
|
|||||||
mainAxisAlignment: MainAxisAlignment.end,
|
mainAxisAlignment: MainAxisAlignment.end,
|
||||||
children: [
|
children: [
|
||||||
GestureDetector(
|
GestureDetector(
|
||||||
onTap: () => showUserLocationModalBottomSheet(
|
onTap: () => showUserLocationModalBottomSheet(context, widget.device),
|
||||||
context, widget.device
|
|
||||||
),
|
|
||||||
child: Container(
|
child: Container(
|
||||||
decoration: BoxDecoration(
|
decoration: BoxDecoration(
|
||||||
color: Colors.black.withOpacity(0.85),
|
color: Colors.black.withOpacity(0.85),
|
||||||
// border: Border.all(color: Colors.lightBlue, width: 2), TODO: add border
|
// border: Border.all(color: Colors.lightBlue, width: 2), TODO: add border
|
||||||
borderRadius: BorderRadius.circular(10),
|
borderRadius: BorderRadius.circular(10),
|
||||||
boxShadow: [
|
boxShadow: const [
|
||||||
// BoxShadow(color: Colors.black, blurRadius: 4)
|
// BoxShadow(color: Colors.black, blurRadius: 4)
|
||||||
]),
|
]),
|
||||||
padding: EdgeInsets.all(8),
|
padding: const EdgeInsets.all(8),
|
||||||
child: Text(
|
child: Text(
|
||||||
"${widget.device.$1}:${widget.device.$2}",
|
"${widget.device.$1}:${widget.device.$2}",
|
||||||
softWrap: false,
|
softWrap: false,
|
||||||
),
|
),
|
||||||
),
|
),
|
||||||
),
|
),
|
||||||
Icon(
|
const Icon(
|
||||||
Icons.location_history,
|
Icons.location_history,
|
||||||
size: 32,
|
size: 32,
|
||||||
)
|
)
|
||||||
@ -184,8 +179,7 @@ class _UserPathState extends State<UserPath> {
|
|||||||
List<List<LatLng>> segments = [];
|
List<List<LatLng>> segments = [];
|
||||||
List<Color> colors = [];
|
List<Color> colors = [];
|
||||||
if (state.initialPoints.isNotEmpty) {
|
if (state.initialPoints.isNotEmpty) {
|
||||||
final allPoints =
|
final allPoints = state.initialPoints.map((e) => LatLng(e.lat, e.lon)).toList();
|
||||||
state.initialPoints.map((e) => LatLng(e.lat, e.lon)).toList();
|
|
||||||
final segmentCount = math.min(100, allPoints.length);
|
final segmentCount = math.min(100, allPoints.length);
|
||||||
final pointsPerSegment = (allPoints.length / segmentCount).ceil();
|
final pointsPerSegment = (allPoints.length / segmentCount).ceil();
|
||||||
|
|
||||||
@ -208,9 +202,8 @@ class _UserPathState extends State<UserPath> {
|
|||||||
Polyline(
|
Polyline(
|
||||||
points: segments[i],
|
points: segments[i],
|
||||||
strokeWidth: 4.0,
|
strokeWidth: 4.0,
|
||||||
color: colors[i].withOpacity(
|
color: colors[i]
|
||||||
(math.pow(i, 2) / math.pow(segments.length, 2)) * 0.7 +
|
.withOpacity((math.pow(i, 2) / math.pow(segments.length, 2)) * 0.7 + 0.3), // Fading effect
|
||||||
0.3), // Fading effect
|
|
||||||
),
|
),
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
@ -234,14 +227,12 @@ class _UserPathState extends State<UserPath> {
|
|||||||
),
|
),
|
||||||
PolylineLayer(polylines: [
|
PolylineLayer(polylines: [
|
||||||
Polyline(
|
Polyline(
|
||||||
points: state.livePoints
|
points: state.livePoints.map((e) => LatLng(e.lat, e.lon)).toList(),
|
||||||
.map((e) => LatLng(e.lat, e.lon))
|
|
||||||
.toList(),
|
|
||||||
strokeWidth: 4.0,
|
strokeWidth: 4.0,
|
||||||
color: Colors.blue.shade200,
|
color: Colors.blue.shade200,
|
||||||
),
|
),
|
||||||
]),
|
]),
|
||||||
MarkerLayer(markers: _markers)
|
MarkerLayer(markers: markers)
|
||||||
],
|
],
|
||||||
);
|
);
|
||||||
},
|
},
|
||||||
@ -256,43 +247,40 @@ showUserLocationModalBottomSheet(BuildContext context, (String, String) user) {
|
|||||||
context: context,
|
context: context,
|
||||||
builder: (bsContext) {
|
builder: (bsContext) {
|
||||||
return Container(
|
return Container(
|
||||||
height: MediaQuery.of(bsContext).size.height * 0.26,
|
height: MediaQuery.of(bsContext).size.height * 0.5,
|
||||||
width: MediaQuery.of(bsContext).size.width,
|
width: MediaQuery.of(bsContext).size.width,
|
||||||
decoration: BoxDecoration(
|
decoration: const BoxDecoration(
|
||||||
color: Colors.black,
|
color: Colors.black,
|
||||||
// border: Border.all(color: Colors.blueAccent, width: 2),
|
|
||||||
borderRadius: BorderRadius.only(topLeft: Radius.circular(30), topRight: Radius.circular(30)),
|
borderRadius: BorderRadius.only(topLeft: Radius.circular(30), topRight: Radius.circular(30)),
|
||||||
),
|
),
|
||||||
padding: EdgeInsets.all(32),
|
padding: EdgeInsets.all(32),
|
||||||
child: Column(
|
child: Column(
|
||||||
|
mainAxisSize: MainAxisSize.min,
|
||||||
children: [
|
children: [
|
||||||
Text(
|
Flexible(
|
||||||
|
// Wrap non-grid content in Flexible to manage space dynamically
|
||||||
|
child: Text(
|
||||||
'${user.$1}:${user.$2}',
|
'${user.$1}:${user.$2}',
|
||||||
style: const TextStyle(
|
style: const TextStyle(
|
||||||
fontSize: 24,
|
fontSize: 24,
|
||||||
fontWeight: FontWeight.bold,
|
fontWeight: FontWeight.bold,
|
||||||
),
|
),
|
||||||
),
|
),
|
||||||
SizedBox(height: 16,),
|
),
|
||||||
Builder(
|
const SizedBox(height: 16),
|
||||||
builder: (sheetContext) {
|
StreamBuilder<UserPathState>(
|
||||||
final state = context.watch<UserPathBloc>().state
|
stream: context.read<UserPathBloc>().stream,
|
||||||
as MainUserPathState;
|
builder: (sheetContext, state) {
|
||||||
|
final istate =
|
||||||
// get user's current location
|
state.data as MainUserPathState? ?? context.read<UserPathBloc>().state as MainUserPathState;
|
||||||
// final _istate = state as MainUserPathState;
|
if (istate.livePoints.isEmpty) {
|
||||||
|
return Text("Couldn't find ${user.$1}:${user.$2}'s Location");
|
||||||
if (state.livePoints.isEmpty) {
|
|
||||||
return Text(
|
|
||||||
"Couldn't find ${user.$1}:${user.$2}'s Location");
|
|
||||||
}
|
}
|
||||||
|
final curLocation = istate.livePoints.last;
|
||||||
final curLocation = state.livePoints.last;
|
return Flexible(
|
||||||
|
// Use Flexible for dynamic content
|
||||||
// MapController.of(sheetContext).camera.pointToLatLng(
|
child: Column(
|
||||||
// math.Point(curLocation.lat, curLocation.lon));
|
mainAxisSize: MainAxisSize.min,
|
||||||
|
|
||||||
return Column(
|
|
||||||
children: [
|
children: [
|
||||||
Container(
|
Container(
|
||||||
decoration: BoxDecoration(
|
decoration: BoxDecoration(
|
||||||
@ -300,36 +288,126 @@ showUserLocationModalBottomSheet(BuildContext context, (String, String) user) {
|
|||||||
border: Border.all(color: Colors.orange, width: 2),
|
border: Border.all(color: Colors.orange, width: 2),
|
||||||
borderRadius: BorderRadius.circular(10),
|
borderRadius: BorderRadius.circular(10),
|
||||||
),
|
),
|
||||||
padding: EdgeInsets.all(16),
|
padding: const EdgeInsets.all(16),
|
||||||
width: double.infinity,
|
width: double.infinity,
|
||||||
child: Column(
|
child: Column(
|
||||||
children: [
|
children: [
|
||||||
Text(
|
Text("(${curLocation.lat.toStringAsFixed(4)}, ${curLocation.lon.toStringAsFixed(4)})"),
|
||||||
"(${curLocation.lat.toStringAsFixed(4)}, ${curLocation.lon.toStringAsFixed(4)})"),
|
Text(DateFormat('dd.MM.yyyy - kk:mm:ss').format(curLocation.timestamp)),
|
||||||
Text(DateFormat('dd.MM.yyyy - kk:mm:ss')
|
|
||||||
.format(curLocation.timestamp)),
|
|
||||||
StreamBuilder(
|
StreamBuilder(
|
||||||
// rebuild every second for that ticking effect
|
stream: Stream.periodic(const Duration(seconds: 1)),
|
||||||
// not hyper efficient, but it's only a text
|
|
||||||
stream: Stream.periodic(
|
|
||||||
const Duration(seconds: 1)),
|
|
||||||
builder: (context, _) {
|
builder: (context, _) {
|
||||||
return Text(
|
return Text("${formatDuration(DateTime.now().difference(curLocation.timestamp))} ago");
|
||||||
"${formatDuration(DateTime.now().difference(curLocation.timestamp))} ago");
|
},
|
||||||
}),
|
),
|
||||||
],
|
],
|
||||||
),
|
),
|
||||||
|
),
|
||||||
|
const SizedBox(
|
||||||
|
height: 16,
|
||||||
|
),
|
||||||
|
Expanded(
|
||||||
|
// Ensure GridView.builder is within an Expanded widget
|
||||||
|
child: BlocBuilder<GlobalLocationStoreCubit, GlobalLocationStoreState>(
|
||||||
|
bloc: GetIt.I.get<GlobalLocationStoreCubit>(),
|
||||||
|
builder: (sheetContext, state) {
|
||||||
|
// get map camera
|
||||||
|
final mapController = MapController.of(context);
|
||||||
|
|
||||||
|
final mapRotation = mapController.camera.rotation;
|
||||||
|
|
||||||
|
List<MapEntry<(String, String), Point>> locations = state.locations
|
||||||
|
.remove(user) // remove this
|
||||||
|
.entries // get entries into a list, and sort alphabetically
|
||||||
|
.toList()
|
||||||
|
..sort((a, b) => "${a.key.$1}:${a.key.$2}".compareTo("${b.key.$1}:${b.key.$2}"));
|
||||||
|
if (locations.isEmpty) {
|
||||||
|
return const SizedBox();
|
||||||
|
}
|
||||||
|
return GridView.builder(
|
||||||
|
gridDelegate: const SliverGridDelegateWithFixedCrossAxisCount(
|
||||||
|
crossAxisCount: 2,
|
||||||
|
crossAxisSpacing: 4.0,
|
||||||
|
mainAxisSpacing: 4.0,
|
||||||
|
childAspectRatio: 2.8,
|
||||||
|
),
|
||||||
|
itemCount: locations.length,
|
||||||
|
itemBuilder: (BuildContext context, int index) {
|
||||||
|
// calculate distance and bearing
|
||||||
|
double distance = distanceBetween(curLocation.lat, curLocation.lon,
|
||||||
|
locations[index].value.lat, locations[index].value.lon, "meters");
|
||||||
|
|
||||||
|
double bearing = (bearingBetween(
|
||||||
|
curLocation.lat,
|
||||||
|
curLocation.lon,
|
||||||
|
locations[index].value.lat,
|
||||||
|
locations[index].value.lon,
|
||||||
|
) +
|
||||||
|
mapRotation) %
|
||||||
|
360;
|
||||||
|
|
||||||
|
print(distance);
|
||||||
|
print(bearing);
|
||||||
|
|
||||||
|
return Center(
|
||||||
|
child: Container(
|
||||||
|
margin: const EdgeInsets.all(8),
|
||||||
|
decoration: BoxDecoration(
|
||||||
|
color: Colors.black,
|
||||||
|
border: Border.all(color: Colors.pink, width: 2),
|
||||||
|
borderRadius: BorderRadius.circular(10),
|
||||||
|
),
|
||||||
|
padding: EdgeInsets.all(16),
|
||||||
|
child: Column(
|
||||||
|
children: [
|
||||||
|
Row(
|
||||||
|
children: [
|
||||||
|
Text(
|
||||||
|
"${locations[index].key.$1}:${locations[index].key.$2}",
|
||||||
|
style: const TextStyle(fontWeight: FontWeight.bold),
|
||||||
|
),
|
||||||
|
Spacer(),
|
||||||
|
Text(formatDistance(distance ~/ 1)),
|
||||||
|
Transform.rotate(
|
||||||
|
angle: bearing * (math.pi / 180),
|
||||||
|
child: Icon(
|
||||||
|
Icons.arrow_upward,
|
||||||
|
color: Colors.blue,
|
||||||
|
),
|
||||||
)
|
)
|
||||||
],
|
],
|
||||||
|
),
|
||||||
|
Row(
|
||||||
|
children: [
|
||||||
|
StreamBuilder(
|
||||||
|
stream: Stream.periodic(const Duration(seconds: 1)),
|
||||||
|
builder: (context, _) {
|
||||||
|
return Text(
|
||||||
|
"${formatDuration(DateTime.now().difference(locations[index].value.timestamp))} ago",
|
||||||
|
style: const TextStyle(fontSize: 12)
|
||||||
|
);
|
||||||
|
},
|
||||||
|
),
|
||||||
|
Spacer(),
|
||||||
|
],
|
||||||
|
),
|
||||||
|
],
|
||||||
|
),
|
||||||
|
),
|
||||||
|
);
|
||||||
|
},
|
||||||
|
);
|
||||||
|
},
|
||||||
|
),
|
||||||
|
),
|
||||||
|
],
|
||||||
|
),
|
||||||
);
|
);
|
||||||
},
|
},
|
||||||
),
|
),
|
||||||
],
|
],
|
||||||
));
|
),
|
||||||
// title with icon, user name and device id
|
);
|
||||||
|
},
|
||||||
// location as a lat, long (truncated to 4 comma spaces)
|
);
|
||||||
// time since last location update
|
|
||||||
// location to other users in a grid (2 rows)
|
|
||||||
});
|
|
||||||
}
|
}
|
||||||
|
@ -118,7 +118,7 @@ class OwntracksApi {
|
|||||||
}
|
}
|
||||||
|
|
||||||
// Method to create and return a WebSocket connection
|
// Method to create and return a WebSocket connection
|
||||||
Future<WebSocketClient> createWebSocketConnection({
|
Future<Result<WebSocketClient>> createWebSocketConnection({
|
||||||
required String wsPath,
|
required String wsPath,
|
||||||
required void Function(Object message) onMessage,
|
required void Function(Object message) onMessage,
|
||||||
required void Function(WebSocketClientState stateChange) onStateChange,
|
required void Function(WebSocketClientState stateChange) onStateChange,
|
||||||
@ -149,10 +149,15 @@ class OwntracksApi {
|
|||||||
client.stateChanges.listen(onStateChange);
|
client.stateChanges.listen(onStateChange);
|
||||||
|
|
||||||
// Connect to the WebSocket server
|
// Connect to the WebSocket server
|
||||||
|
try {
|
||||||
await client.connect("${baseUrl.replaceFirst('http', 'ws')}/ws/$wsPath");
|
await client.connect("${baseUrl.replaceFirst('http', 'ws')}/ws/$wsPath");
|
||||||
|
} catch (e) {
|
||||||
|
await client.disconnect();
|
||||||
|
return bail("WebSocket connection to path $wsPath was unsuccessful: $e");
|
||||||
|
}
|
||||||
|
|
||||||
// Return the connected client
|
// Return the connected client
|
||||||
return client;
|
return Ok(client);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -163,6 +168,11 @@ class Point {
|
|||||||
|
|
||||||
Point({required this.lat, required this.lon, required this.timestamp});
|
Point({required this.lat, required this.lon, required this.timestamp});
|
||||||
|
|
||||||
|
@override
|
||||||
|
String toString() {
|
||||||
|
return 'Point{lat: $lat, lon: $lon, timestamp: $timestamp}';
|
||||||
|
}
|
||||||
|
|
||||||
factory Point.fromJson(Map<String, dynamic> json) {
|
factory Point.fromJson(Map<String, dynamic> json) {
|
||||||
return Point(
|
return Point(
|
||||||
lat: json['lat'],
|
lat: json['lat'],
|
||||||
|
@ -3,6 +3,8 @@ import 'dart:convert';
|
|||||||
|
|
||||||
import 'package:fast_immutable_collections/fast_immutable_collections.dart';
|
import 'package:fast_immutable_collections/fast_immutable_collections.dart';
|
||||||
import 'package:flutter/foundation.dart';
|
import 'package:flutter/foundation.dart';
|
||||||
|
import 'package:get_it/get_it.dart';
|
||||||
|
import 'package:ot_viewer_app/global_location_store.dart';
|
||||||
import 'package:rust_core/option.dart';
|
import 'package:rust_core/option.dart';
|
||||||
import 'package:anyhow/anyhow.dart';
|
import 'package:anyhow/anyhow.dart';
|
||||||
import 'package:bloc/bloc.dart';
|
import 'package:bloc/bloc.dart';
|
||||||
@ -72,6 +74,16 @@ class UserPathBloc extends Bloc<UserPathEvent, UserPathState> {
|
|||||||
|
|
||||||
@override
|
@override
|
||||||
void onTransition(Transition<UserPathEvent, UserPathState> transition) {
|
void onTransition(Transition<UserPathEvent, UserPathState> transition) {
|
||||||
|
super.onTransition(transition);
|
||||||
|
|
||||||
|
if (transition.nextState is MainUserPathState) {
|
||||||
|
// add current location to global location thingy
|
||||||
|
final pt = (transition.nextState as MainUserPathState).livePoints.lastOrNull;
|
||||||
|
if (pt != null) {
|
||||||
|
GetIt.I.get<GlobalLocationStoreCubit>().updatePoint(deviceId.$1, deviceId.$2, pt);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
print("upb $deviceId: $transition");
|
print("upb $deviceId: $transition");
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -1,3 +1,5 @@
|
|||||||
|
import 'dart:math';
|
||||||
|
|
||||||
String formatDuration(Duration duration) {
|
String formatDuration(Duration duration) {
|
||||||
final days = duration.inDays;
|
final days = duration.inDays;
|
||||||
final hours = duration.inHours.remainder(24);
|
final hours = duration.inHours.remainder(24);
|
||||||
@ -16,3 +18,81 @@ String formatDuration(Duration duration) {
|
|||||||
return '$seconds second${plural(seconds)}';
|
return '$seconds second${plural(seconds)}';
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
String formatDistance(int distanceInMeters) {
|
||||||
|
if (distanceInMeters < 1000) {
|
||||||
|
// If the distance is less than 1 kilometer, display it in meters.
|
||||||
|
return '${distanceInMeters}m';
|
||||||
|
} else {
|
||||||
|
// If the distance is 1 kilometer or more, display it in kilometers and meters.
|
||||||
|
final kilometers = distanceInMeters ~/ 1000; // Integer division to get whole kilometers.
|
||||||
|
final meters = distanceInMeters % 1000; // Remainder to get the remaining meters.
|
||||||
|
if (meters == 0) {
|
||||||
|
// If there are no remaining meters, display only kilometers.
|
||||||
|
return '${kilometers}km';
|
||||||
|
} else {
|
||||||
|
// If there are remaining meters, display both kilometers and meters.
|
||||||
|
return '${kilometers}km ${meters}m';
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
double degreesToRadians(double degrees) {
|
||||||
|
return degrees * (pi / 180);
|
||||||
|
}
|
||||||
|
|
||||||
|
double radiansToDegrees(double radians) {
|
||||||
|
return radians * (180 / pi);
|
||||||
|
}
|
||||||
|
|
||||||
|
double bearingBetween(double lat1, double lon1, double lat2, double lon2) {
|
||||||
|
var dLon = degreesToRadians(lon2 - lon1);
|
||||||
|
var y = sin(dLon) * cos(degreesToRadians(lat2));
|
||||||
|
var x = cos(degreesToRadians(lat1)) * sin(degreesToRadians(lat2)) -
|
||||||
|
sin(degreesToRadians(lat1)) * cos(degreesToRadians(lat2)) * cos(dLon);
|
||||||
|
var angle = atan2(y, x);
|
||||||
|
return (radiansToDegrees(angle) + 360) % 360;
|
||||||
|
}
|
||||||
|
|
||||||
|
double distanceBetween(
|
||||||
|
double lat1, double lon1, double lat2, double lon2, String unit) {
|
||||||
|
const earthRadius = 6371; // in km
|
||||||
|
// assuming earth is a perfect sphere(it's not)
|
||||||
|
|
||||||
|
// Convert degrees to radians
|
||||||
|
final lat1Rad = degreesToRadians(lat1);
|
||||||
|
final lon1Rad = degreesToRadians(lon1);
|
||||||
|
final lat2Rad = degreesToRadians(lat2);
|
||||||
|
final lon2Rad = degreesToRadians(lon2);
|
||||||
|
|
||||||
|
final dLat = lat2Rad - lat1Rad;
|
||||||
|
final dLon = lon2Rad - lon1Rad;
|
||||||
|
|
||||||
|
// Haversine formula
|
||||||
|
final a = pow(sin(dLat / 2), 2) +
|
||||||
|
cos(lat1Rad) * cos(lat2Rad) * pow(sin(dLon / 2), 2);
|
||||||
|
final c = 2 * atan2(sqrt(a), sqrt(1 - a));
|
||||||
|
|
||||||
|
final distance = earthRadius * c;
|
||||||
|
|
||||||
|
return toRequestedUnit(unit, distance);
|
||||||
|
|
||||||
|
// return distance; // in km
|
||||||
|
}
|
||||||
|
|
||||||
|
double toRequestedUnit(String unit, double distanceInKm) {
|
||||||
|
switch (unit) {
|
||||||
|
case 'kilometers':
|
||||||
|
return distanceInKm;
|
||||||
|
case 'meters':
|
||||||
|
return distanceInKm * 1000;
|
||||||
|
case 'miles':
|
||||||
|
return (distanceInKm * 1000) / 1609.344;
|
||||||
|
case 'yards':
|
||||||
|
return distanceInKm * 1093.61;
|
||||||
|
case '':
|
||||||
|
return distanceInKm;
|
||||||
|
}
|
||||||
|
return distanceInKm;
|
||||||
|
}
|
||||||
|
@ -24,20 +24,33 @@ class LocationUpdateReceived extends LocationUpdateState {
|
|||||||
|
|
||||||
class LocationSubscribeCubit extends Cubit<LocationUpdateState> {
|
class LocationSubscribeCubit extends Cubit<LocationUpdateState> {
|
||||||
Option<WebSocketClient> _wsClient = None;
|
Option<WebSocketClient> _wsClient = None;
|
||||||
|
String url = '';
|
||||||
|
String username = '';
|
||||||
|
String pass = '';
|
||||||
|
|
||||||
LocationSubscribeCubit() : super(LocationUpdateUnconnected());
|
LocationSubscribeCubit() : super(LocationUpdateUnconnected());
|
||||||
|
|
||||||
subscribe(SettingsState settings) async {
|
subscribe(SettingsState settings) async {
|
||||||
|
|
||||||
|
// check if resubscribe is necessary (different URL)
|
||||||
|
if (settings.url == url && settings.username == username && settings.password == pass) {
|
||||||
|
return;
|
||||||
|
} else {
|
||||||
|
url = settings.url;
|
||||||
|
username = settings.username;
|
||||||
|
pass = settings.password;
|
||||||
|
}
|
||||||
|
|
||||||
|
await _wsConnectionEstablish();
|
||||||
|
}
|
||||||
|
|
||||||
|
Future<void> _wsConnectionEstablish() async {
|
||||||
if (_wsClient.isSome()) {
|
if (_wsClient.isSome()) {
|
||||||
await _wsClient.unwrap().close();
|
await _wsClient.unwrap().close();
|
||||||
_wsClient = None;
|
_wsClient = None;
|
||||||
}
|
}
|
||||||
|
|
||||||
var ws = await OwntracksApi(
|
var ws = await OwntracksApi(baseUrl: url, username: username, pass: pass)
|
||||||
baseUrl: settings.url,
|
|
||||||
username: settings.username,
|
|
||||||
pass: settings.password)
|
|
||||||
.createWebSocketConnection(
|
.createWebSocketConnection(
|
||||||
wsPath: 'last',
|
wsPath: 'last',
|
||||||
onMessage: (msg) {
|
onMessage: (msg) {
|
||||||
@ -59,12 +72,15 @@ class LocationSubscribeCubit extends Cubit<LocationUpdateState> {
|
|||||||
// build device_id
|
// build device_id
|
||||||
final deviceId = (topic[1], topic[2]);
|
final deviceId = (topic[1], topic[2]);
|
||||||
|
|
||||||
|
print(map);
|
||||||
|
|
||||||
// build point
|
// build point
|
||||||
final p = Point(
|
final p = Point(
|
||||||
lat: map['lat'] as double,
|
lat: map['lat'] as double,
|
||||||
lon: map['lon'] as double,
|
lon: map['lon'] as double,
|
||||||
timestamp:
|
timestamp: DateTime.fromMillisecondsSinceEpoch((map['tst'] as int) * 1000));
|
||||||
DateTime.fromMillisecondsSinceEpoch(map['tst'] as int));
|
|
||||||
|
print(p);
|
||||||
|
|
||||||
emit(LocationUpdateReceived(p, deviceId));
|
emit(LocationUpdateReceived(p, deviceId));
|
||||||
}
|
}
|
||||||
@ -74,22 +90,25 @@ class LocationSubscribeCubit extends Cubit<LocationUpdateState> {
|
|||||||
}
|
}
|
||||||
},
|
},
|
||||||
onStateChange: (sc) {
|
onStateChange: (sc) {
|
||||||
if (sc case WebSocketClientState$Open(:final url)) {
|
switch (sc) {
|
||||||
|
case WebSocketClientState$Open(:final url):
|
||||||
_wsClient.map((wsc) => wsc.add('LAST'));
|
_wsClient.map((wsc) => wsc.add('LAST'));
|
||||||
|
emit(LocationUpdateConnected());
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
emit(LocationUpdateUnconnected());
|
||||||
|
break;
|
||||||
}
|
}
|
||||||
print(sc);
|
print(sc);
|
||||||
},
|
},
|
||||||
);
|
);
|
||||||
|
|
||||||
_wsClient = Some(ws);
|
_wsClient = ws.expect("Estabilshing Websocket Conenction failed").toOption();
|
||||||
emit(LocationUpdateConnected());
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
@override
|
@override
|
||||||
void onChange(Change<LocationUpdateState> change) {
|
void onChange(Change<LocationUpdateState> change) {
|
||||||
print('loc_sub_cubit change: $change');
|
print('loc_sub_cubit change: $change');
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
@override
|
@override
|
||||||
|
@ -208,6 +208,14 @@ packages:
|
|||||||
description: flutter
|
description: flutter
|
||||||
source: sdk
|
source: sdk
|
||||||
version: "0.0.0"
|
version: "0.0.0"
|
||||||
|
get_it:
|
||||||
|
dependency: "direct main"
|
||||||
|
description:
|
||||||
|
name: get_it
|
||||||
|
sha256: e6017ce7fdeaf218dc51a100344d8cb70134b80e28b760f8bb23c242437bafd7
|
||||||
|
url: "https://pub.dev"
|
||||||
|
source: hosted
|
||||||
|
version: "7.6.7"
|
||||||
hive:
|
hive:
|
||||||
dependency: transitive
|
dependency: transitive
|
||||||
description:
|
description:
|
||||||
|
@ -54,6 +54,7 @@ dependencies:
|
|||||||
dio_cache_interceptor_hive_store: ^3.2.2
|
dio_cache_interceptor_hive_store: ^3.2.2
|
||||||
flutter_map_math: ^0.1.7
|
flutter_map_math: ^0.1.7
|
||||||
intl: ^0.19.0
|
intl: ^0.19.0
|
||||||
|
get_it: ^7.6.7
|
||||||
|
|
||||||
dev_dependencies:
|
dev_dependencies:
|
||||||
flutter_test:
|
flutter_test:
|
||||||
|
Loading…
Reference in New Issue
Block a user