Compare commits
14 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
| ee4fd6d059 | |||
| aba992d8a2 | |||
| 061e64fe98 | |||
| 7e035f1414 | |||
| e8a6e4464a | |||
| e986cccdd6 | |||
| cb94aa042a | |||
| 1293176f89 | |||
| 057f4a4809 | |||
| a3b2deec21 | |||
| 6c7ceac1a6 | |||
| a46674844c | |||
| ff9636b75b | |||
| 1be6b0681f |
1
.gitignore
vendored
1
.gitignore
vendored
@ -69,6 +69,7 @@ unlinked_spec.ds
|
|||||||
**/android/**/GeneratedPluginRegistrant.java
|
**/android/**/GeneratedPluginRegistrant.java
|
||||||
**/android/key.properties
|
**/android/key.properties
|
||||||
*.jks
|
*.jks
|
||||||
|
**/android/app/.cxx/
|
||||||
|
|
||||||
# iOS/XCode related
|
# iOS/XCode related
|
||||||
**/ios/**/*.mode1v3
|
**/ios/**/*.mode1v3
|
||||||
|
|||||||
5
.vscode/settings.json
vendored
5
.vscode/settings.json
vendored
@ -1,4 +1,7 @@
|
|||||||
{
|
{
|
||||||
"java.configuration.updateBuildConfiguration": "interactive",
|
"java.configuration.updateBuildConfiguration": "interactive",
|
||||||
"cmake.sourceDirectory": "${workspaceFolder}/linux/flutter"
|
"cmake.sourceDirectory": "${workspaceFolder}/linux/flutter",
|
||||||
|
"editor.wordBasedSuggestions": "off",
|
||||||
|
"editor.tabCompletion": "onlySnippets",
|
||||||
|
"editor.selectionHighlight": false
|
||||||
}
|
}
|
||||||
@ -1,52 +1,34 @@
|
|||||||
def localProperties = new Properties()
|
plugins {
|
||||||
def localPropertiesFile = rootProject.file('local.properties')
|
id "com.android.application"
|
||||||
if (localPropertiesFile.exists()) {
|
id "kotlin-android"
|
||||||
localPropertiesFile.withReader('UTF-8') { reader ->
|
id "dev.flutter.flutter-gradle-plugin"
|
||||||
localProperties.load(reader)
|
id "com.google.gms.google-services"
|
||||||
}
|
id "com.google.firebase.crashlytics"
|
||||||
}
|
}
|
||||||
|
|
||||||
def flutterRoot = localProperties.getProperty('flutter.sdk')
|
def keystoreProperties = new Properties()
|
||||||
if (flutterRoot == null) {
|
def keystorePropertiesFile = rootProject.file('key.properties')
|
||||||
throw new GradleException("Flutter SDK not found. Define location with flutter.sdk in the local.properties file.")
|
if (keystorePropertiesFile.exists()) {
|
||||||
}
|
|
||||||
|
|
||||||
def flutterVersionCode = localProperties.getProperty('flutter.versionCode')
|
|
||||||
if (flutterVersionCode == null) {
|
|
||||||
flutterVersionCode = '1'
|
|
||||||
}
|
|
||||||
|
|
||||||
def flutterVersionName = localProperties.getProperty('flutter.versionName')
|
|
||||||
if (flutterVersionName == null) {
|
|
||||||
flutterVersionName = '1.0'
|
|
||||||
}
|
|
||||||
|
|
||||||
apply plugin: 'com.android.application'
|
|
||||||
apply plugin: 'kotlin-android'
|
|
||||||
apply from: "$flutterRoot/packages/flutter_tools/gradle/flutter.gradle"
|
|
||||||
apply plugin: 'com.google.gms.google-services'
|
|
||||||
|
|
||||||
def keystoreProperties=new Properties()
|
|
||||||
def keystorePropertiesFile=rootProject.file('key.properties')
|
|
||||||
if(keystorePropertiesFile.exists()){
|
|
||||||
keystoreProperties.load(new FileInputStream(keystorePropertiesFile))
|
keystoreProperties.load(new FileInputStream(keystorePropertiesFile))
|
||||||
}
|
}
|
||||||
|
|
||||||
android {
|
android {
|
||||||
compileSdkVersion 33
|
namespace = "hu.app_dev.terepi_seged"
|
||||||
ndkVersion "25.1.8937393"
|
compileSdk 35
|
||||||
|
// ndkVersion "25.1.8937393"
|
||||||
|
// ndkVersion flutter.ndkVersion
|
||||||
|
ndkVersion "27.0.12077973"
|
||||||
|
|
||||||
compileOptions {
|
compileOptions {
|
||||||
sourceCompatibility JavaVersion.VERSION_1_8
|
// sourceCompatibility JavaVersion.VERSION_1_8
|
||||||
targetCompatibility JavaVersion.VERSION_1_8
|
// targetCompatibility JavaVersion.VERSION_1_8
|
||||||
|
sourceCompatibility JavaVersion.VERSION_17
|
||||||
|
targetCompatibility JavaVersion.VERSION_17
|
||||||
}
|
}
|
||||||
|
|
||||||
kotlinOptions {
|
kotlinOptions {
|
||||||
jvmTarget = '1.8'
|
// jvmTarget = JavaVersion.VERSION_1_8
|
||||||
}
|
jvmTarget = JavaVersion.VERSION_17
|
||||||
|
|
||||||
sourceSets {
|
|
||||||
main.java.srcDirs += 'src/main/kotlin'
|
|
||||||
}
|
}
|
||||||
|
|
||||||
defaultConfig {
|
defaultConfig {
|
||||||
@ -55,10 +37,12 @@ android {
|
|||||||
// You can update the following values to match your application needs.
|
// You can update the following values to match your application needs.
|
||||||
// For more information, see: https://docs.flutter.dev/deployment/android#reviewing-the-build-configuration.
|
// For more information, see: https://docs.flutter.dev/deployment/android#reviewing-the-build-configuration.
|
||||||
minSdkVersion 23
|
minSdkVersion 23
|
||||||
targetSdkVersion 33
|
targetSdkVersion 35
|
||||||
versionCode flutterVersionCode.toInteger()
|
versionCode flutter.versionCode
|
||||||
versionName flutterVersionName
|
versionName flutter.versionName
|
||||||
multiDexEnabled true
|
|
||||||
|
// archivesBaseName = "terepi_seged-${versionName}-${new Date().format('yyyyMMdd-HHmm')}"
|
||||||
|
// multiDexEnabled true
|
||||||
}
|
}
|
||||||
|
|
||||||
signingConfigs {
|
signingConfigs {
|
||||||
@ -83,8 +67,7 @@ flutter {
|
|||||||
source '../..'
|
source '../..'
|
||||||
}
|
}
|
||||||
|
|
||||||
dependencies {
|
// dependencies {
|
||||||
implementation "org.jetbrains.kotlin:kotlin-stdlib-jdk7:$kotlin_version"
|
// implementation platform('com.google.firebase:firebase-bom:31.3.0')
|
||||||
implementation platform('com.google.firebase:firebase-bom:31.3.0')
|
// implementation 'com.google.firebase:firebase-analytics-ktx'
|
||||||
implementation 'com.google.firebase:firebase-analytics-ktx'
|
// }
|
||||||
}
|
|
||||||
|
|||||||
@ -1,17 +1,3 @@
|
|||||||
buildscript {
|
|
||||||
ext.kotlin_version = '1.8.10'
|
|
||||||
repositories {
|
|
||||||
google()
|
|
||||||
mavenCentral()
|
|
||||||
}
|
|
||||||
|
|
||||||
dependencies {
|
|
||||||
classpath 'com.android.tools.build:gradle:7.1.2'
|
|
||||||
classpath "org.jetbrains.kotlin:kotlin-gradle-plugin:$kotlin_version"
|
|
||||||
classpath 'com.google.gms:google-services:4.3.15'
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
allprojects {
|
allprojects {
|
||||||
repositories {
|
repositories {
|
||||||
google()
|
google()
|
||||||
@ -25,6 +11,18 @@ allprojects {
|
|||||||
}
|
}
|
||||||
|
|
||||||
rootProject.buildDir = '../build'
|
rootProject.buildDir = '../build'
|
||||||
|
subprojects {
|
||||||
|
afterEvaluate { project ->
|
||||||
|
if (project.hasProperty('android')) {
|
||||||
|
project.android {
|
||||||
|
if (namespace == null) {
|
||||||
|
namespace project.group
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
subprojects {
|
subprojects {
|
||||||
project.buildDir = "${rootProject.buildDir}/${project.name}"
|
project.buildDir = "${rootProject.buildDir}/${project.name}"
|
||||||
}
|
}
|
||||||
|
|||||||
@ -1,6 +1,7 @@
|
|||||||
#Fri Jun 23 08:50:38 CEST 2017
|
|
||||||
distributionBase=GRADLE_USER_HOME
|
distributionBase=GRADLE_USER_HOME
|
||||||
distributionPath=wrapper/dists
|
distributionPath=wrapper/dists
|
||||||
|
distributionUrl=https\://services.gradle.org/distributions/gradle-8.13-bin.zip
|
||||||
|
networkTimeout=10000
|
||||||
|
validateDistributionUrl=true
|
||||||
zipStoreBase=GRADLE_USER_HOME
|
zipStoreBase=GRADLE_USER_HOME
|
||||||
zipStorePath=wrapper/dists
|
zipStorePath=wrapper/dists
|
||||||
distributionUrl=https\://services.gradle.org/distributions/gradle-7.4-all.zip
|
|
||||||
|
|||||||
@ -1,11 +1,27 @@
|
|||||||
include ':app'
|
pluginManagement {
|
||||||
|
def flutterSdkPath = {
|
||||||
|
def properties = new Properties()
|
||||||
|
file("local.properties").withInputStream { properties.load(it) }
|
||||||
|
def flutterSdkPath = properties.getProperty("flutter.sdk")
|
||||||
|
assert flutterSdkPath != null, "flutter.sdk not set in local.properties"
|
||||||
|
return flutterSdkPath
|
||||||
|
}()
|
||||||
|
|
||||||
def localPropertiesFile = new File(rootProject.projectDir, "local.properties")
|
includeBuild("$flutterSdkPath/packages/flutter_tools/gradle")
|
||||||
def properties = new Properties()
|
|
||||||
|
|
||||||
assert localPropertiesFile.exists()
|
repositories {
|
||||||
localPropertiesFile.withReader("UTF-8") { reader -> properties.load(reader) }
|
google()
|
||||||
|
mavenCentral()
|
||||||
|
gradlePluginPortal()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
def flutterSdkPath = properties.getProperty("flutter.sdk")
|
plugins {
|
||||||
assert flutterSdkPath != null, "flutter.sdk not set in local.properties"
|
id "dev.flutter.flutter-plugin-loader" version "1.0.0"
|
||||||
apply from: "$flutterSdkPath/packages/flutter_tools/gradle/app_plugin_loader.gradle"
|
id "com.android.application" version "8.7.1" apply false
|
||||||
|
id "org.jetbrains.kotlin.android" version "2.2.0" apply false
|
||||||
|
id "com.google.gms.google-services" version "4.4.0" apply false
|
||||||
|
id "com.google.firebase.crashlytics" version "2.9.9" apply false
|
||||||
|
}
|
||||||
|
|
||||||
|
include ":app"
|
||||||
3
devtools_options.yaml
Normal file
3
devtools_options.yaml
Normal file
@ -0,0 +1,3 @@
|
|||||||
|
description: This file stores settings for Dart & Flutter DevTools.
|
||||||
|
documentation: https://docs.flutter.dev/tools/devtools/extensions#configure-extension-enablement-states
|
||||||
|
extensions:
|
||||||
414
lib/controls/geojson_parser.dart
Normal file
414
lib/controls/geojson_parser.dart
Normal file
@ -0,0 +1,414 @@
|
|||||||
|
import 'dart:convert';
|
||||||
|
import 'package:flutter/material.dart';
|
||||||
|
|
||||||
|
import 'package:flutter_map/flutter_map.dart';
|
||||||
|
import 'package:latlong2/latlong.dart';
|
||||||
|
|
||||||
|
typedef MarkerCreationCallback = Marker Function(
|
||||||
|
LatLng point, Map<String, dynamic> properties);
|
||||||
|
typedef CircleMarkerCreationCallback = CircleMarker Function(
|
||||||
|
LatLng point, Map<String, dynamic> properties);
|
||||||
|
typedef PolylineCreationCallback = Polyline Function(
|
||||||
|
List<LatLng> points, Map<String, dynamic> properties);
|
||||||
|
typedef PolygonCreationCallback = Polygon Function(List<LatLng> points,
|
||||||
|
List<List<LatLng>>? holePointsList, Map<String, dynamic> properties);
|
||||||
|
typedef FilterFunction = bool Function(Map<String, dynamic> properties);
|
||||||
|
|
||||||
|
/// GeoJsonParser parses the GeoJson and fills three lists of parsed objects
|
||||||
|
/// which are defined in flutter_map package
|
||||||
|
/// - list of [Marker]s
|
||||||
|
/// - list of [CircleMarker]s
|
||||||
|
/// - list of [Polyline]s
|
||||||
|
/// - list of [Polygon]s
|
||||||
|
///
|
||||||
|
/// One should pass these lists when creating adequate layers in flutter_map.
|
||||||
|
/// For details see example.
|
||||||
|
///
|
||||||
|
/// Currently GeoJson parser supports only FeatureCollection and not GeometryCollection.
|
||||||
|
/// See the GeoJson Format specification at: https://www.rfc-editor.org/rfc/rfc7946
|
||||||
|
///
|
||||||
|
/// For creation of [Marker], [Polyline], [CircleMarker] and [Polygon] objects the default callback functions
|
||||||
|
/// are provided which are used in case when no user-defined callback function is provided.
|
||||||
|
/// To fully customize the [Marker], [Polyline], [CircleMarker] and [Polygon] creation one has to write his own
|
||||||
|
/// callback functions. As a template the default callback functions can be used.
|
||||||
|
///
|
||||||
|
class GeoJsonParser {
|
||||||
|
/// list of [Marker] objects created as result of parsing
|
||||||
|
final List<Marker> markers = [];
|
||||||
|
|
||||||
|
/// list of [Polyline] objects created as result of parsing
|
||||||
|
final List<Polyline> polylines = [];
|
||||||
|
|
||||||
|
/// list of [Polygon] objects created as result of parsing
|
||||||
|
final List<Polygon> polygons = [];
|
||||||
|
|
||||||
|
/// list of [CircleMarker] objects created as result of parsing
|
||||||
|
final List<CircleMarker> circles = [];
|
||||||
|
|
||||||
|
/// user defined callback function that creates a [Marker] object
|
||||||
|
MarkerCreationCallback? markerCreationCallback;
|
||||||
|
|
||||||
|
/// user defined callback function that creates a [Polyline] object
|
||||||
|
PolylineCreationCallback? polyLineCreationCallback;
|
||||||
|
|
||||||
|
/// user defined callback function that creates a [Polygon] object
|
||||||
|
PolygonCreationCallback? polygonCreationCallback;
|
||||||
|
|
||||||
|
/// user defined callback function that creates a [Polygon] object
|
||||||
|
CircleMarkerCreationCallback? circleMarkerCreationCallback;
|
||||||
|
|
||||||
|
/// default [Marker] color
|
||||||
|
Color? defaultMarkerColor;
|
||||||
|
|
||||||
|
/// default [Marker] icon
|
||||||
|
IconData? defaultMarkerIcon;
|
||||||
|
|
||||||
|
/// default [Polyline] color
|
||||||
|
Color? defaultPolylineColor;
|
||||||
|
|
||||||
|
/// default [Polyline] stroke
|
||||||
|
double? defaultPolylineStroke;
|
||||||
|
|
||||||
|
/// default [Polygon] border color
|
||||||
|
Color? defaultPolygonBorderColor;
|
||||||
|
|
||||||
|
/// default [Polygon] fill color
|
||||||
|
Color? defaultPolygonFillColor;
|
||||||
|
|
||||||
|
/// default [Polygon] border stroke
|
||||||
|
double? defaultPolygonBorderStroke;
|
||||||
|
|
||||||
|
/// default flag if [Polygon] is filled (default is true)
|
||||||
|
bool? defaultPolygonIsFilled;
|
||||||
|
|
||||||
|
/// default [CircleMarker] border color
|
||||||
|
Color? defaultCircleMarkerColor;
|
||||||
|
|
||||||
|
/// default [CircleMarker] border stroke
|
||||||
|
Color? defaultCircleMarkerBorderColor;
|
||||||
|
|
||||||
|
/// default flag if [CircleMarker] is filled (default is true)
|
||||||
|
bool? defaultCircleMarkerIsFilled;
|
||||||
|
|
||||||
|
/// user defined callback function called when the [Marker] is tapped
|
||||||
|
void Function(Map<String, dynamic>)? onMarkerTapCallback;
|
||||||
|
|
||||||
|
/// user defined callback function called when the [CircleMarker] is tapped
|
||||||
|
void Function(Map<String, dynamic>)? onCircleMarkerTapCallback;
|
||||||
|
|
||||||
|
/// user defined callback function called during parse for filtering
|
||||||
|
FilterFunction? filterFunction;
|
||||||
|
|
||||||
|
/// default constructor - all parameters are optional and can be set later with setters
|
||||||
|
GeoJsonParser({
|
||||||
|
this.markerCreationCallback,
|
||||||
|
this.polyLineCreationCallback,
|
||||||
|
this.polygonCreationCallback,
|
||||||
|
this.circleMarkerCreationCallback,
|
||||||
|
this.filterFunction,
|
||||||
|
this.defaultMarkerColor,
|
||||||
|
this.defaultMarkerIcon,
|
||||||
|
this.onMarkerTapCallback,
|
||||||
|
this.defaultPolylineColor,
|
||||||
|
this.defaultPolylineStroke,
|
||||||
|
this.defaultPolygonBorderColor,
|
||||||
|
this.defaultPolygonFillColor,
|
||||||
|
this.defaultPolygonBorderStroke,
|
||||||
|
this.defaultPolygonIsFilled,
|
||||||
|
this.defaultCircleMarkerColor,
|
||||||
|
this.defaultCircleMarkerBorderColor,
|
||||||
|
this.defaultCircleMarkerIsFilled,
|
||||||
|
this.onCircleMarkerTapCallback,
|
||||||
|
});
|
||||||
|
|
||||||
|
/// parse GeJson in [String] format
|
||||||
|
void parseGeoJsonAsString(String g) {
|
||||||
|
return parseGeoJson(jsonDecode(g) as Map<String, dynamic>);
|
||||||
|
}
|
||||||
|
|
||||||
|
/// set default [Marker] color
|
||||||
|
set setDefaultMarkerColor(Color color) {
|
||||||
|
defaultMarkerColor = color;
|
||||||
|
}
|
||||||
|
|
||||||
|
/// set default [Marker] icon
|
||||||
|
set setDefaultMarkerIcon(IconData ic) {
|
||||||
|
defaultMarkerIcon = ic;
|
||||||
|
}
|
||||||
|
|
||||||
|
/// set default [Marker] tap callback function
|
||||||
|
void setDefaultMarkerTapCallback(
|
||||||
|
Function(Map<String, dynamic> f) onTapFunction) {
|
||||||
|
onMarkerTapCallback = onTapFunction;
|
||||||
|
}
|
||||||
|
|
||||||
|
/// set default [CircleMarker] color
|
||||||
|
set setDefaultCircleMarkerColor(Color color) {
|
||||||
|
defaultCircleMarkerColor = color;
|
||||||
|
}
|
||||||
|
|
||||||
|
/// set default [CircleMarker] tap callback function
|
||||||
|
void setDefaultCircleMarkerTapCallback(
|
||||||
|
Function(Map<String, dynamic> f) onTapFunction) {
|
||||||
|
onCircleMarkerTapCallback = onTapFunction;
|
||||||
|
}
|
||||||
|
|
||||||
|
/// set default [Polyline] color
|
||||||
|
set setDefaultPolylineColor(Color color) {
|
||||||
|
defaultPolylineColor = color;
|
||||||
|
}
|
||||||
|
|
||||||
|
/// set default [Polyline] stroke
|
||||||
|
set setDefaultPolylineStroke(double stroke) {
|
||||||
|
defaultPolylineStroke = stroke;
|
||||||
|
}
|
||||||
|
|
||||||
|
/// set default [Polygon] fill color
|
||||||
|
set setDefaultPolygonFillColor(Color color) {
|
||||||
|
defaultPolygonFillColor = color;
|
||||||
|
}
|
||||||
|
|
||||||
|
/// set default [Polygon] border stroke
|
||||||
|
set setDefaultPolygonBorderStroke(double stroke) {
|
||||||
|
defaultPolygonBorderStroke = stroke;
|
||||||
|
}
|
||||||
|
|
||||||
|
/// set default [Polygon] border color
|
||||||
|
set setDefaultPolygonBorderColorStroke(Color color) {
|
||||||
|
defaultPolygonBorderColor = color;
|
||||||
|
}
|
||||||
|
|
||||||
|
/// set default [Polygon] setting whether polygon is filled
|
||||||
|
set setDefaultPolygonIsFilled(bool filled) {
|
||||||
|
defaultPolygonIsFilled = filled;
|
||||||
|
}
|
||||||
|
|
||||||
|
/// main GeoJson parsing function
|
||||||
|
void parseGeoJson(Map<String, dynamic> g) {
|
||||||
|
// set default values if they are not specified by constructor
|
||||||
|
markerCreationCallback ??= createDefaultMarker;
|
||||||
|
circleMarkerCreationCallback ??= createDefaultCircleMarker;
|
||||||
|
polyLineCreationCallback ??= createDefaultPolyline;
|
||||||
|
polygonCreationCallback ??= createDefaultPolygon;
|
||||||
|
filterFunction ??= defaultFilterFunction;
|
||||||
|
defaultMarkerColor ??= Colors.red.withOpacity(0.8);
|
||||||
|
defaultMarkerIcon ??= Icons.location_pin;
|
||||||
|
defaultPolylineColor ??= Colors.blue.withOpacity(0.8);
|
||||||
|
defaultPolylineStroke ??= 3.0;
|
||||||
|
defaultPolygonBorderColor ??= Colors.black.withOpacity(0.8);
|
||||||
|
defaultPolygonFillColor ??= Colors.black.withOpacity(0.1);
|
||||||
|
defaultPolygonIsFilled ??= true;
|
||||||
|
defaultPolygonBorderStroke ??= 1.0;
|
||||||
|
defaultCircleMarkerColor ??= Colors.blue.withOpacity(0.25);
|
||||||
|
defaultCircleMarkerBorderColor ??= Colors.black.withOpacity(0.8);
|
||||||
|
defaultCircleMarkerIsFilled ??= true;
|
||||||
|
|
||||||
|
// loop through the GeoJson Map and parse it
|
||||||
|
for (Map f in g['features'] as List) {
|
||||||
|
String geometryType = f['geometry']['type'].toString();
|
||||||
|
// check if this spatial object passes the filter function
|
||||||
|
if (!filterFunction!(f['properties'] as Map<String, dynamic>)) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
switch (geometryType) {
|
||||||
|
case 'Point':
|
||||||
|
{
|
||||||
|
markers.add(
|
||||||
|
markerCreationCallback!(
|
||||||
|
LatLng(f['geometry']['coordinates'][1] as double,
|
||||||
|
f['geometry']['coordinates'][0] as double),
|
||||||
|
f['properties'] as Map<String, dynamic>),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
case 'Circle':
|
||||||
|
{
|
||||||
|
circles.add(
|
||||||
|
circleMarkerCreationCallback!(
|
||||||
|
LatLng(f['geometry']['coordinates'][1] as double,
|
||||||
|
f['geometry']['coordinates'][0] as double),
|
||||||
|
f['properties'] as Map<String, dynamic>),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
case 'MultiPoint':
|
||||||
|
{
|
||||||
|
for (final point in f['geometry']['coordinates'] as List) {
|
||||||
|
markers.add(
|
||||||
|
markerCreationCallback!(
|
||||||
|
LatLng(point[1] as double, point[0] as double),
|
||||||
|
f['properties'] as Map<String, dynamic>),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
case 'LineString':
|
||||||
|
{
|
||||||
|
final List<LatLng> lineString = [];
|
||||||
|
for (final coords in f['geometry']['coordinates'] as List) {
|
||||||
|
lineString.add(LatLng(coords[1] as double, coords[0] as double));
|
||||||
|
}
|
||||||
|
polylines.add(polyLineCreationCallback!(
|
||||||
|
lineString, f['properties'] as Map<String, dynamic>));
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
case 'MultiLineString':
|
||||||
|
{
|
||||||
|
for (final line in f['geometry']['coordinates'] as List) {
|
||||||
|
final List<LatLng> lineString = [];
|
||||||
|
for (final coords in line as List) {
|
||||||
|
lineString
|
||||||
|
.add(LatLng(coords[1] as double, coords[0] as double));
|
||||||
|
}
|
||||||
|
polylines.add(polyLineCreationCallback!(
|
||||||
|
lineString, f['properties'] as Map<String, dynamic>));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
case 'Polygon':
|
||||||
|
{
|
||||||
|
final List<LatLng> outerRing = [];
|
||||||
|
final List<List<LatLng>> holesList = [];
|
||||||
|
int pathIndex = 0;
|
||||||
|
for (final path in f['geometry']['coordinates'] as List) {
|
||||||
|
final List<LatLng> hole = [];
|
||||||
|
for (final coords in path as List<dynamic>) {
|
||||||
|
if (pathIndex == 0) {
|
||||||
|
// add to polygon's outer ring
|
||||||
|
outerRing
|
||||||
|
.add(LatLng(coords[1] as double, coords[0] as double));
|
||||||
|
} else {
|
||||||
|
// add it to current hole
|
||||||
|
hole.add(LatLng(coords[1] as double, coords[0] as double));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (pathIndex > 0) {
|
||||||
|
// add hole to the polygon's list of holes
|
||||||
|
holesList.add(hole);
|
||||||
|
}
|
||||||
|
pathIndex++;
|
||||||
|
}
|
||||||
|
polygons.add(polygonCreationCallback!(
|
||||||
|
outerRing, holesList, f['properties'] as Map<String, dynamic>));
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
case 'MultiPolygon':
|
||||||
|
{
|
||||||
|
for (final polygon in f['geometry']['coordinates'] as List) {
|
||||||
|
final List<LatLng> outerRing = [];
|
||||||
|
final List<List<LatLng>> holesList = [];
|
||||||
|
int pathIndex = 0;
|
||||||
|
for (final path in polygon as List) {
|
||||||
|
List<LatLng> hole = [];
|
||||||
|
for (final coords in path as List<dynamic>) {
|
||||||
|
if (pathIndex == 0) {
|
||||||
|
// add to polygon's outer ring
|
||||||
|
outerRing
|
||||||
|
.add(LatLng(coords[1] as double, coords[0] as double));
|
||||||
|
} else {
|
||||||
|
// add it to a hole
|
||||||
|
hole.add(LatLng(coords[1] as double, coords[0] as double));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (pathIndex > 0) {
|
||||||
|
// add to polygon's list of holes
|
||||||
|
holesList.add(hole);
|
||||||
|
}
|
||||||
|
pathIndex++;
|
||||||
|
}
|
||||||
|
polygons.add(polygonCreationCallback!(outerRing, holesList,
|
||||||
|
f['properties'] as Map<String, dynamic>));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
/// default function for creating tappable [Marker]
|
||||||
|
Widget defaultTappableMarker(Map<String, dynamic> properties,
|
||||||
|
void Function(Map<String, dynamic>) onMarkerTap) {
|
||||||
|
return MouseRegion(
|
||||||
|
cursor: SystemMouseCursors.click,
|
||||||
|
child: GestureDetector(
|
||||||
|
onTap: () {
|
||||||
|
onMarkerTap(properties);
|
||||||
|
},
|
||||||
|
child: Icon(defaultMarkerIcon, color: defaultMarkerColor),
|
||||||
|
),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
/// default callback function for creating [Marker]
|
||||||
|
Marker createDefaultMarker(LatLng point, Map<String, dynamic> properties) {
|
||||||
|
return Marker(
|
||||||
|
point: point,
|
||||||
|
child: defaultTappableMarker(properties, markerTapped),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
// /// default callback function for creating [Marker]
|
||||||
|
// Marker createDefaultMarker(LatLng point, Map<String, dynamic> properties) {
|
||||||
|
// return Marker(
|
||||||
|
// point: point,
|
||||||
|
// child: MouseRegion(
|
||||||
|
// cursor: SystemMouseCursors.click,
|
||||||
|
// child: GestureDetector(
|
||||||
|
// onTap: () {
|
||||||
|
// markerTapped(properties);
|
||||||
|
// },
|
||||||
|
// child: Icon(defaultMarkerIcon, color: defaultMarkerColor),
|
||||||
|
// ),
|
||||||
|
// ),
|
||||||
|
// width: 60,
|
||||||
|
// height: 60,
|
||||||
|
// );
|
||||||
|
// }
|
||||||
|
|
||||||
|
/// default callback function for creating [Polygon]
|
||||||
|
CircleMarker createDefaultCircleMarker(
|
||||||
|
LatLng point, Map<String, dynamic> properties) {
|
||||||
|
return CircleMarker(
|
||||||
|
point: point,
|
||||||
|
radius: properties["radius"].toDouble(),
|
||||||
|
useRadiusInMeter: true,
|
||||||
|
color: defaultCircleMarkerColor!,
|
||||||
|
borderColor: defaultCircleMarkerBorderColor!,
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
/// default callback function for creating [Polyline]
|
||||||
|
Polyline createDefaultPolyline(
|
||||||
|
List<LatLng> points, Map<String, dynamic> properties) {
|
||||||
|
return Polyline(
|
||||||
|
points: points,
|
||||||
|
color: defaultPolylineColor!,
|
||||||
|
strokeWidth: defaultPolylineStroke!);
|
||||||
|
}
|
||||||
|
|
||||||
|
/// default callback function for creating [Polygon]
|
||||||
|
Polygon createDefaultPolygon(List<LatLng> outerRing,
|
||||||
|
List<List<LatLng>>? holesList, Map<String, dynamic> properties) {
|
||||||
|
return Polygon(
|
||||||
|
points: outerRing,
|
||||||
|
holePointsList: holesList,
|
||||||
|
borderColor: defaultPolygonBorderColor!,
|
||||||
|
color: defaultPolygonFillColor!,
|
||||||
|
// isFilled: defaultPolygonIsFilled!,
|
||||||
|
borderStrokeWidth: defaultPolygonBorderStroke!,
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
/// the default filter function returns always true - therefore no filtering
|
||||||
|
bool defaultFilterFunction(Map<String, dynamic> properties) {
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
/// default callback function called when tappable [Marker] is tapped
|
||||||
|
void markerTapped(Map<String, dynamic> map) {
|
||||||
|
if (onMarkerTapCallback != null) {
|
||||||
|
onMarkerTapCallback!(map);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
@ -11,7 +11,7 @@ Future<void> main() async {
|
|||||||
await Supabase.initialize(
|
await Supabase.initialize(
|
||||||
url: 'https://supa.app-dev.hu',
|
url: 'https://supa.app-dev.hu',
|
||||||
anonKey:
|
anonKey:
|
||||||
'eyJhbGciOiJIUzI1NiIsInR5cCI6IkpXVCJ9.ewogICJyb2xlIjogImFub24iLAogICJpc3MiOiAic3VwYWJhc2UiLAogICJpYXQiOiAxNzExMDYyMDAwLAogICJleHAiOiAxODY4ODI4NDAwCn0.XWtP3eEysZDxXjaHHUZyyhw0n4YZo_xWUMWS5ajBcbI');
|
'eyJhbGciOiJIUzI1NiIsInR5cCI6IkpXVCJ9.ewogICJyb2xlIjogImFub24iLAogICJpc3MiOiAic3VwYWJhc2UiLAogICJpYXQiOiAxNzQwMjY1MjAwLAogICJleHAiOiAxODk4MDMxNjAwCn0.4cMVfAnBLxne1lq0fm94rgtXLBJdRx-0f-E4Jd_jFwI');
|
||||||
|
|
||||||
runApp(const MyApp());
|
runApp(const MyApp());
|
||||||
}
|
}
|
||||||
|
|||||||
9
lib/pages/field_trip/bindings/field_trip_bindings.dart
Normal file
9
lib/pages/field_trip/bindings/field_trip_bindings.dart
Normal file
@ -0,0 +1,9 @@
|
|||||||
|
import 'package:get/get.dart';
|
||||||
|
import 'package:terepi_seged/pages/field_trip/presentations/controllers/field_trip_controller.dart';
|
||||||
|
|
||||||
|
class FieldTripBinding extends Bindings {
|
||||||
|
@override
|
||||||
|
void dependencies() {
|
||||||
|
Get.lazyPut(() => FieldTripController());
|
||||||
|
}
|
||||||
|
}
|
||||||
@ -0,0 +1,162 @@
|
|||||||
|
import 'package:flutter/material.dart';
|
||||||
|
import 'package:flutter_map/flutter_map.dart';
|
||||||
|
import 'package:flutter_map_polygon_editor/polygon_editor/polygon_editor_controller.dart';
|
||||||
|
import 'package:get/get.dart';
|
||||||
|
import 'package:latlong2/latlong.dart';
|
||||||
|
|
||||||
|
class FieldTripController extends GetxController {
|
||||||
|
RxBool mapIsInitialized = false.obs;
|
||||||
|
RxBool mapInEditorMode = false.obs;
|
||||||
|
RxBool mapInPointEditorMode = false.obs;
|
||||||
|
|
||||||
|
final pointNotes = <Marker>[].obs;
|
||||||
|
final polylineNotes = <Polyline<Object>>[].obs;
|
||||||
|
final polygonNotes = <Polygon<Object>>[].obs;
|
||||||
|
|
||||||
|
late final MapController mapController;
|
||||||
|
late final MapOptions mapOptions;
|
||||||
|
late final PolygonEditorController polygonEditorController;
|
||||||
|
|
||||||
|
final PolygonLabelPlacementCalculator _labelPlacementCalculator =
|
||||||
|
const PolygonLabelPlacementCalculator.centroid();
|
||||||
|
|
||||||
|
@override
|
||||||
|
void onInit() async {
|
||||||
|
// TODO: implement onInit
|
||||||
|
super.onInit();
|
||||||
|
|
||||||
|
mapOptions = MapOptions(
|
||||||
|
onLongPress: (tapPosition, point) {
|
||||||
|
if (mapInEditorMode.value) {
|
||||||
|
polygonEditorController.addPoint(point);
|
||||||
|
}
|
||||||
|
if (mapInPointEditorMode.value) {
|
||||||
|
savePointNote(point: point);
|
||||||
|
}
|
||||||
|
},
|
||||||
|
keepAlive: true,
|
||||||
|
);
|
||||||
|
|
||||||
|
polylineNotes.add(
|
||||||
|
Polyline(
|
||||||
|
points: [
|
||||||
|
const LatLng(47.65, 19.00),
|
||||||
|
const LatLng(47.64, 19.02),
|
||||||
|
const LatLng(47.69, 19.22),
|
||||||
|
],
|
||||||
|
strokeWidth: 8,
|
||||||
|
color: const Color.fromARGB(255, 227, 238, 71),
|
||||||
|
hitValue: (
|
||||||
|
title: 'Purple Line',
|
||||||
|
subtitle: 'Nothing really special here...',
|
||||||
|
),
|
||||||
|
),
|
||||||
|
);
|
||||||
|
|
||||||
|
polygonEditorController =
|
||||||
|
PolygonEditorController(mode: PolygonEditorMode.polygon);
|
||||||
|
|
||||||
|
mapController = MapController();
|
||||||
|
|
||||||
|
mapIsInitialized.value = true;
|
||||||
|
}
|
||||||
|
|
||||||
|
@override
|
||||||
|
void dispose() {
|
||||||
|
// TODO: implement dispose
|
||||||
|
polygonEditorController.dispose();
|
||||||
|
super.dispose();
|
||||||
|
}
|
||||||
|
|
||||||
|
void cancelEditorController() {
|
||||||
|
polygonEditorController.clear();
|
||||||
|
mapInEditorMode.value = false;
|
||||||
|
}
|
||||||
|
|
||||||
|
void startPolygonEdition() {
|
||||||
|
polygonEditorController.clear();
|
||||||
|
polygonEditorController.setMode(PolygonEditorMode.polygon);
|
||||||
|
mapInEditorMode.value = true;
|
||||||
|
}
|
||||||
|
|
||||||
|
void startPolylineEdition() {
|
||||||
|
polygonEditorController.clear();
|
||||||
|
polygonEditorController.setMode(PolygonEditorMode.line);
|
||||||
|
mapInEditorMode.value = true;
|
||||||
|
}
|
||||||
|
|
||||||
|
void startPointEdition() {
|
||||||
|
mapInPointEditorMode.value = true;
|
||||||
|
}
|
||||||
|
|
||||||
|
void saveNote() {
|
||||||
|
if (polygonEditorController.mode == PolygonEditorMode.line) {
|
||||||
|
print("Points number in line: ${polygonEditorController.points.length}");
|
||||||
|
print(
|
||||||
|
"1. point coords: ${polygonEditorController.points[0].latitude} - ${polygonEditorController.points[0].longitude}");
|
||||||
|
|
||||||
|
Polyline polyline = Polyline(
|
||||||
|
points: polygonEditorController.points,
|
||||||
|
color: Colors.red,
|
||||||
|
strokeWidth: 8,
|
||||||
|
// hitValue: (
|
||||||
|
// title: 'Purple Line',
|
||||||
|
// subtitle: 'Nothing really special here...',
|
||||||
|
// ),
|
||||||
|
);
|
||||||
|
polylineNotes.add(polyline);
|
||||||
|
// polylineNotes.refresh();
|
||||||
|
|
||||||
|
print("Points number in polylineNotes: ${polylineNotes.length}");
|
||||||
|
print(
|
||||||
|
"1. point coords of polyline: ${polyline.points[0].latitude} - ${polyline.points[0].longitude}");
|
||||||
|
|
||||||
|
polygonEditorController.clear();
|
||||||
|
mapInEditorMode.value = false;
|
||||||
|
}
|
||||||
|
if (polygonEditorController.mode == PolygonEditorMode.polygon) {
|
||||||
|
print(
|
||||||
|
"Points number in polygon: ${polygonEditorController.points.length}");
|
||||||
|
|
||||||
|
Polygon polygon = Polygon(
|
||||||
|
points: polygonEditorController.points,
|
||||||
|
color: Colors.purple,
|
||||||
|
borderColor: Colors.yellow,
|
||||||
|
borderStrokeWidth: 4,
|
||||||
|
label: 'Label!',
|
||||||
|
labelPlacementCalculator: _labelPlacementCalculator,
|
||||||
|
// hitValue: (
|
||||||
|
// title: 'Basic Filled Polygon',
|
||||||
|
// subtitle: 'Nothing really special here...',
|
||||||
|
);
|
||||||
|
|
||||||
|
polygonNotes.add(polygon);
|
||||||
|
polygonNotes.refresh();
|
||||||
|
update();
|
||||||
|
|
||||||
|
print("Points number in polygonNotes: ${polygonNotes.length}");
|
||||||
|
|
||||||
|
polygonEditorController.clear();
|
||||||
|
mapInEditorMode.value = false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void savePointNote({required LatLng point}) {
|
||||||
|
Marker marker = Marker(
|
||||||
|
point: point,
|
||||||
|
width: 15.0,
|
||||||
|
height: 15.0,
|
||||||
|
child: Container(
|
||||||
|
width: 15.0,
|
||||||
|
height: 15.0,
|
||||||
|
decoration: BoxDecoration(
|
||||||
|
color: Colors.amber[700],
|
||||||
|
shape: BoxShape.circle,
|
||||||
|
border: Border.all(width: 1.0, color: Colors.black)),
|
||||||
|
));
|
||||||
|
pointNotes.add(marker);
|
||||||
|
pointNotes.refresh();
|
||||||
|
update();
|
||||||
|
mapInPointEditorMode.value = false;
|
||||||
|
}
|
||||||
|
}
|
||||||
113
lib/pages/field_trip/presentations/views/fiels_trip_view.dart
Normal file
113
lib/pages/field_trip/presentations/views/fiels_trip_view.dart
Normal file
@ -0,0 +1,113 @@
|
|||||||
|
import 'package:flutter_map/flutter_map.dart';
|
||||||
|
import 'package:flutter_map_location_marker/flutter_map_location_marker.dart';
|
||||||
|
import 'package:flutter_map_polygon_editor/polygon_editor.dart';
|
||||||
|
import 'package:get/get.dart';
|
||||||
|
import 'package:latlong2/latlong.dart';
|
||||||
|
import 'package:terepi_seged/pages/field_trip/presentations/controllers/field_trip_controller.dart';
|
||||||
|
import 'package:flutter/material.dart';
|
||||||
|
|
||||||
|
class FieldTripView extends GetView<FieldTripController> {
|
||||||
|
const FieldTripView({Key? key}) : super(key: key);
|
||||||
|
|
||||||
|
@override
|
||||||
|
Widget build(BuildContext context) {
|
||||||
|
return Scaffold(
|
||||||
|
resizeToAvoidBottomInset: false,
|
||||||
|
extendBody: true,
|
||||||
|
appBar: AppBar(
|
||||||
|
title: const Column(
|
||||||
|
crossAxisAlignment: CrossAxisAlignment.start,
|
||||||
|
children: [
|
||||||
|
Text('Terepbejárás'),
|
||||||
|
],
|
||||||
|
)),
|
||||||
|
body: Column(
|
||||||
|
children: [
|
||||||
|
Expanded(
|
||||||
|
child: Stack(children: [
|
||||||
|
Obx(() => controller.mapIsInitialized.value
|
||||||
|
? FlutterMap(
|
||||||
|
mapController: controller.mapController,
|
||||||
|
options: controller.mapOptions,
|
||||||
|
children: [
|
||||||
|
TileLayer(
|
||||||
|
urlTemplate:
|
||||||
|
'http://{s}.google.com/vt/lyrs=s,h&x={x}&y={y}&z={z}',
|
||||||
|
subdomains: const ['mt0', 'mt1', 'mt2', 'mt3'],
|
||||||
|
maxNativeZoom: 18,
|
||||||
|
),
|
||||||
|
CurrentLocationLayer(
|
||||||
|
alignPositionOnUpdate: AlignOnUpdate.once,
|
||||||
|
),
|
||||||
|
MarkerLayer(markers: controller.pointNotes),
|
||||||
|
PolylineLayer(
|
||||||
|
polylines: controller.polylineNotes,
|
||||||
|
),
|
||||||
|
PolygonLayer(
|
||||||
|
polygons: controller.polygonNotes,
|
||||||
|
useAltRendering: true,
|
||||||
|
),
|
||||||
|
// PolylineLayer(polylines: [
|
||||||
|
// Polyline(
|
||||||
|
// points: [
|
||||||
|
// const LatLng(47.65, 19.00),
|
||||||
|
// const LatLng(47.64, 19.02),
|
||||||
|
// const LatLng(47.69, 19.22),
|
||||||
|
// ],
|
||||||
|
// strokeWidth: 8,
|
||||||
|
// color: const Color(0xFF60399E),
|
||||||
|
// hitValue: (
|
||||||
|
// title: 'Purple Line',
|
||||||
|
// subtitle: 'Nothing really special here...',
|
||||||
|
// ),
|
||||||
|
// ),
|
||||||
|
// ]),
|
||||||
|
PolygonEditor(
|
||||||
|
controller: controller.polygonEditorController,
|
||||||
|
throttleDuration: Duration.zero)
|
||||||
|
],
|
||||||
|
)
|
||||||
|
: const Center(child: CircularProgressIndicator())),
|
||||||
|
Positioned(
|
||||||
|
bottom: 3,
|
||||||
|
right: 3,
|
||||||
|
child: Row(
|
||||||
|
children: [
|
||||||
|
IconButton.filled(
|
||||||
|
iconSize: 32,
|
||||||
|
onPressed: () {
|
||||||
|
controller.startPointEdition();
|
||||||
|
},
|
||||||
|
icon: const Icon(Icons.flag)),
|
||||||
|
IconButton.filled(
|
||||||
|
iconSize: 32,
|
||||||
|
onPressed: () {
|
||||||
|
controller.startPolylineEdition();
|
||||||
|
},
|
||||||
|
icon: const Icon(Icons.polyline)),
|
||||||
|
IconButton.filled(
|
||||||
|
iconSize: 32,
|
||||||
|
onPressed: () {
|
||||||
|
controller.startPolygonEdition();
|
||||||
|
},
|
||||||
|
icon: const Icon(Icons.border_outer)),
|
||||||
|
IconButton.filled(
|
||||||
|
iconSize: 32,
|
||||||
|
onPressed: () {
|
||||||
|
controller.saveNote();
|
||||||
|
},
|
||||||
|
icon: const Icon(Icons.done)),
|
||||||
|
IconButton.filled(
|
||||||
|
iconSize: 32,
|
||||||
|
onPressed: () {
|
||||||
|
controller.cancelEditorController();
|
||||||
|
},
|
||||||
|
icon: const Icon(Icons.cancel))
|
||||||
|
],
|
||||||
|
))
|
||||||
|
]))
|
||||||
|
],
|
||||||
|
),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
@ -55,16 +55,44 @@ class HomeView extends GetView<HomeViewController> {
|
|||||||
// child: const Text('RTCM teszt'))
|
// child: const Text('RTCM teszt'))
|
||||||
],
|
],
|
||||||
),
|
),
|
||||||
|
Padding(
|
||||||
|
padding: const EdgeInsets.symmetric(vertical: 10.0),
|
||||||
|
child: Row(
|
||||||
|
mainAxisAlignment: MainAxisAlignment.spaceEvenly,
|
||||||
|
children: [
|
||||||
|
TextButton(
|
||||||
|
onPressed: () async {
|
||||||
|
Get.toNamed("/field_trip");
|
||||||
|
},
|
||||||
|
child: Text('Field trip')),
|
||||||
|
// TextButton(
|
||||||
|
// onPressed: () async {
|
||||||
|
// Get.toNamed("/tracking");
|
||||||
|
// },
|
||||||
|
// child: Text('Tracking')),
|
||||||
|
TextButton(
|
||||||
|
onPressed: () async {
|
||||||
|
Get.toNamed("/map");
|
||||||
|
},
|
||||||
|
child: Text('Kitűzés'))
|
||||||
|
])),
|
||||||
Padding(
|
Padding(
|
||||||
padding: const EdgeInsets.only(top: 10.0),
|
padding: const EdgeInsets.only(top: 10.0),
|
||||||
child: Row(
|
child: Row(
|
||||||
mainAxisAlignment: MainAxisAlignment.spaceEvenly,
|
mainAxisAlignment: MainAxisAlignment.spaceEvenly,
|
||||||
children: [
|
children: [
|
||||||
|
// BigButtonWidget(
|
||||||
|
// iconData: Icons.flag,
|
||||||
|
// label: "Kitűzés",
|
||||||
|
// onPressed: () async {
|
||||||
|
// Get.toNamed('/map');
|
||||||
|
// },
|
||||||
|
// ),
|
||||||
BigButtonWidget(
|
BigButtonWidget(
|
||||||
iconData: Icons.flag,
|
iconData: Icons.flag,
|
||||||
label: "Kitűzés",
|
label: "Bemérés",
|
||||||
onPressed: () async {
|
onPressed: () async {
|
||||||
Get.toNamed('/map');
|
Get.toNamed('/map_survey');
|
||||||
},
|
},
|
||||||
),
|
),
|
||||||
BigButtonWidget(
|
BigButtonWidget(
|
||||||
|
|||||||
@ -9,7 +9,7 @@ import 'package:flutter/material.dart';
|
|||||||
import 'package:flutter/services.dart';
|
import 'package:flutter/services.dart';
|
||||||
import 'package:flutter_bluetooth_serial/flutter_bluetooth_serial.dart';
|
import 'package:flutter_bluetooth_serial/flutter_bluetooth_serial.dart';
|
||||||
import 'package:flutter_map/flutter_map.dart';
|
import 'package:flutter_map/flutter_map.dart';
|
||||||
import 'package:flutter_map_geojson/flutter_map_geojson.dart';
|
// import 'package:flutter_map_geojson/flutter_map_geojson.dart';
|
||||||
import 'package:flutter_map_polywidget/flutter_map_polywidget.dart';
|
import 'package:flutter_map_polywidget/flutter_map_polywidget.dart';
|
||||||
import 'package:get/get.dart';
|
import 'package:get/get.dart';
|
||||||
import 'package:intl/intl.dart';
|
import 'package:intl/intl.dart';
|
||||||
@ -19,6 +19,8 @@ import 'package:nmea/nmea.dart';
|
|||||||
import 'package:path_provider/path_provider.dart';
|
import 'package:path_provider/path_provider.dart';
|
||||||
import 'package:permission_handler/permission_handler.dart'
|
import 'package:permission_handler/permission_handler.dart'
|
||||||
as permission_handler;
|
as permission_handler;
|
||||||
|
import 'package:share_plus/share_plus.dart';
|
||||||
|
import 'package:supabase_flutter/supabase_flutter.dart';
|
||||||
import 'package:terepi_seged/eov/convert_coordinate.dart';
|
import 'package:terepi_seged/eov/convert_coordinate.dart';
|
||||||
import 'package:terepi_seged/eov/eov.dart';
|
import 'package:terepi_seged/eov/eov.dart';
|
||||||
import 'package:terepi_seged/gnss_sentences/gngga.dart';
|
import 'package:terepi_seged/gnss_sentences/gngga.dart';
|
||||||
@ -27,12 +29,14 @@ import 'package:terepi_seged/gnss_sentences/gnrmc.dart';
|
|||||||
import 'package:terepi_seged/models/point_to_measure.dart';
|
import 'package:terepi_seged/models/point_to_measure.dart';
|
||||||
import 'package:terepi_seged/models/point_with_description_model.dart';
|
import 'package:terepi_seged/models/point_with_description_model.dart';
|
||||||
import 'package:proj4dart/proj4dart.dart' as proj4;
|
import 'package:proj4dart/proj4dart.dart' as proj4;
|
||||||
|
import 'package:shared_preferences/shared_preferences.dart';
|
||||||
|
import 'package:terepi_seged/pages/map/presentation/views/measured_points_table_dialog.dart';
|
||||||
|
|
||||||
class MapViewController extends GetxController {
|
class MapViewController extends GetxController {
|
||||||
// String gpsAddress = "E8:31:CD:14:8B:B2";
|
// String gpsAddress = "E8:31:CD:14:8B:B2";
|
||||||
// String gpsAddress = "98:CD:AC:62:FF:4E";
|
// String gpsAddress = "98:CD:AC:62:FF:4E";
|
||||||
RxString gpsAddress = "98:CD:AC:62:FF:36".obs;
|
RxString gpsAddress = "98:CD:AC:62:FF:36".obs;
|
||||||
String gpsName = "TiGNSS Rover-FF4E";
|
RxString gpsName = "TiGNSS Rover-FF4E".obs;
|
||||||
// String gpsName = "TiGNSS Rover-8BB2";
|
// String gpsName = "TiGNSS Rover-8BB2";
|
||||||
static const double maxZoomValue = 25.0;
|
static const double maxZoomValue = 25.0;
|
||||||
Rx<bool> gpsIsConnected = false.obs;
|
Rx<bool> gpsIsConnected = false.obs;
|
||||||
@ -57,6 +61,8 @@ class MapViewController extends GetxController {
|
|||||||
DateTime.now().add(const Duration(seconds: -30));
|
DateTime.now().add(const Duration(seconds: -30));
|
||||||
|
|
||||||
NumberFormat formatEov = NumberFormat("##0,000.0", "hu-HU");
|
NumberFormat formatEov = NumberFormat("##0,000.0", "hu-HU");
|
||||||
|
NumberFormat formatEovZ = NumberFormat("###0.0", "hu-HU");
|
||||||
|
NumberFormat formatAltitudeError = NumberFormat("####0.000", "hu-HU");
|
||||||
NumberFormat formatEovForFile = NumberFormat("#####0.0", "hu-HU");
|
NumberFormat formatEovForFile = NumberFormat("#####0.0", "hu-HU");
|
||||||
NumberFormat formatWgs84Sec = NumberFormat('00.000', 'hu-HU');
|
NumberFormat formatWgs84Sec = NumberFormat('00.000', 'hu-HU');
|
||||||
|
|
||||||
@ -65,6 +71,7 @@ class MapViewController extends GetxController {
|
|||||||
RxDouble gpsLongitude = 0.0.obs;
|
RxDouble gpsLongitude = 0.0.obs;
|
||||||
RxString gpsLongitudeDirection = "".obs;
|
RxString gpsLongitudeDirection = "".obs;
|
||||||
RxDouble gpsAltitude = 0.0.obs;
|
RxDouble gpsAltitude = 0.0.obs;
|
||||||
|
RxDouble gpsGeoidSeparation = 0.0.obs;
|
||||||
RxInt gpsQuality = 0.obs;
|
RxInt gpsQuality = 0.obs;
|
||||||
RxDouble gpsLatitudeError = 0.0.obs;
|
RxDouble gpsLatitudeError = 0.0.obs;
|
||||||
RxDouble gpsLongitudeError = 0.0.obs;
|
RxDouble gpsLongitudeError = 0.0.obs;
|
||||||
@ -103,8 +110,8 @@ class MapViewController extends GetxController {
|
|||||||
int pointId = 1;
|
int pointId = 1;
|
||||||
String pointIdPrefix = "";
|
String pointIdPrefix = "";
|
||||||
String pointIdPostfix = "";
|
String pointIdPostfix = "";
|
||||||
String ntripUserName = "";
|
RxString ntripUserName = "".obs;
|
||||||
String ntripPassword = "";
|
RxString ntripPassword = "".obs;
|
||||||
|
|
||||||
Rx<bool> pointMeasuringDirectionForward = true.obs;
|
Rx<bool> pointMeasuringDirectionForward = true.obs;
|
||||||
|
|
||||||
@ -115,7 +122,8 @@ class MapViewController extends GetxController {
|
|||||||
late proj4.Projection eovProj, wgsProj;
|
late proj4.Projection eovProj, wgsProj;
|
||||||
RxBool mapIsInitialized = false.obs;
|
RxBool mapIsInitialized = false.obs;
|
||||||
|
|
||||||
GeoJsonParser parser = GeoJsonParser(defaultMarkerColor: Colors.yellow);
|
// GeoJsonParser parser =
|
||||||
|
// GeoJsonParser(defaultMarkerColor: const Color.fromARGB(255, 85, 34, 49));
|
||||||
|
|
||||||
final CollectionReference _measuredPoints =
|
final CollectionReference _measuredPoints =
|
||||||
FirebaseFirestore.instance.collection('measuredPoints');
|
FirebaseFirestore.instance.collection('measuredPoints');
|
||||||
@ -123,6 +131,14 @@ class MapViewController extends GetxController {
|
|||||||
// late SMIBool gpsTrigger;
|
// late SMIBool gpsTrigger;
|
||||||
// late StateMachineController riveGpsIconController;
|
// late StateMachineController riveGpsIconController;
|
||||||
|
|
||||||
|
late SharedPreferences prefs;
|
||||||
|
Rx<bool> isShowPassword = false.obs;
|
||||||
|
final passwordFieldFocusNode = FocusNode();
|
||||||
|
|
||||||
|
late AuthResponse authResponse;
|
||||||
|
late Session? session;
|
||||||
|
late User? user;
|
||||||
|
|
||||||
@override
|
@override
|
||||||
void onInit() async {
|
void onInit() async {
|
||||||
super.onInit();
|
super.onInit();
|
||||||
@ -144,6 +160,26 @@ class MapViewController extends GetxController {
|
|||||||
..registerTalkerSentence("RMC", (line) => Gnrmc(raw: line))
|
..registerTalkerSentence("RMC", (line) => Gnrmc(raw: line))
|
||||||
..registerTalkerSentence("GST", (line) => Gngst(raw: line));
|
..registerTalkerSentence("GST", (line) => Gngst(raw: line));
|
||||||
|
|
||||||
|
prefs = await SharedPreferences.getInstance();
|
||||||
|
|
||||||
|
authResponse = await Supabase.instance.client.auth
|
||||||
|
.signInWithPassword(email: 'test.elek.1@email.hu', password: 'demo');
|
||||||
|
session = authResponse.session;
|
||||||
|
user = authResponse.user;
|
||||||
|
|
||||||
|
Supabase.instance.client
|
||||||
|
.channel('public:TerepiSeged_Receiver')
|
||||||
|
.onPostgresChanges(
|
||||||
|
event: PostgresChangeEvent.update,
|
||||||
|
schema: 'public',
|
||||||
|
table: 'TerepiSeged_Receiver',
|
||||||
|
callback: (playload) {
|
||||||
|
var id = playload.newRecord['pointNumber'] as int;
|
||||||
|
print('Change received: ${id}');
|
||||||
|
updatePointStatus(id);
|
||||||
|
})
|
||||||
|
.subscribe();
|
||||||
|
|
||||||
mapController = MapController();
|
mapController = MapController();
|
||||||
|
|
||||||
// riveGpsIconController = RiveUtils.getRiveController(Artboard(),
|
// riveGpsIconController = RiveUtils.getRiveController(Artboard(),
|
||||||
@ -154,7 +190,7 @@ class MapViewController extends GetxController {
|
|||||||
}
|
}
|
||||||
|
|
||||||
@override
|
@override
|
||||||
void onClose() {
|
void onClose() async {
|
||||||
super.onClose();
|
super.onClose();
|
||||||
FlutterBluetoothSerial.instance.setPairingRequestHandler(null);
|
FlutterBluetoothSerial.instance.setPairingRequestHandler(null);
|
||||||
if (gpsIsConnected.value) {
|
if (gpsIsConnected.value) {
|
||||||
@ -169,6 +205,10 @@ class MapViewController extends GetxController {
|
|||||||
pointPostfixController.dispose();
|
pointPostfixController.dispose();
|
||||||
ntripUsernameController.dispose();
|
ntripUsernameController.dispose();
|
||||||
ntripPasswordController.dispose();
|
ntripPasswordController.dispose();
|
||||||
|
|
||||||
|
await Supabase.instance.client
|
||||||
|
.channel('public:TerepiSeged_Receiver')
|
||||||
|
.unsubscribe();
|
||||||
}
|
}
|
||||||
|
|
||||||
@override
|
@override
|
||||||
@ -200,6 +240,30 @@ class MapViewController extends GetxController {
|
|||||||
} else {
|
} else {
|
||||||
print("No external storage permission");
|
print("No external storage permission");
|
||||||
}
|
}
|
||||||
|
if (prefs.containsKey('gpsAddress')) {
|
||||||
|
var address = prefs.getString('gpsAddress');
|
||||||
|
if (address != null) {
|
||||||
|
gpsAddress.value = address;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (prefs.containsKey('gpsName')) {
|
||||||
|
var name = prefs.getString('gpsName');
|
||||||
|
if (name != null) {
|
||||||
|
gpsName.value = name;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (prefs.containsKey('ntripUserName')) {
|
||||||
|
var userName = prefs.getString('ntripUserName');
|
||||||
|
if (userName != null) {
|
||||||
|
ntripUserName.value = userName;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (prefs.containsKey('ntripPassword')) {
|
||||||
|
var password = prefs.getString('ntripPassword');
|
||||||
|
if (password != null) {
|
||||||
|
ntripPassword.value = password;
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
directory = await getExternalStorageDirectory();
|
directory = await getExternalStorageDirectory();
|
||||||
@ -228,7 +292,7 @@ class MapViewController extends GetxController {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
gpsHeightController.text = '1.7';
|
gpsHeightController.text = '1.8';
|
||||||
}
|
}
|
||||||
|
|
||||||
void _getInitialLocation() async {
|
void _getInitialLocation() async {
|
||||||
@ -302,7 +366,9 @@ class MapViewController extends GetxController {
|
|||||||
header += "User-Agent: SharpGps iter.dk\r\n";
|
header += "User-Agent: SharpGps iter.dk\r\n";
|
||||||
header += "Accept: */*\r\nConnection: close\r\n";
|
header += "Accept: */*\r\nConnection: close\r\n";
|
||||||
// header += "Authorization: Basic ${_toBase64("info@mail.app-dev.hu:")}\r\n";
|
// header += "Authorization: Basic ${_toBase64("info@mail.app-dev.hu:")}\r\n";
|
||||||
header += "Authorization: Basic ${_toBase64("elgi08:Laszl0stef1")}\r\n";
|
//header += "Authorization: Basic ${_toBase64("elgi08:Laszl0stef1")}\r\n";
|
||||||
|
header +=
|
||||||
|
"Authorization: Basic ${_toBase64("${ntripUserName.value}:${ntripPassword.value}")}\r\n";
|
||||||
// header += "Host:rtk2go.com:2101\r\n";
|
// header += "Host:rtk2go.com:2101\r\n";
|
||||||
header += "Host:gnssnet.hu:2101\r\n";
|
header += "Host:gnssnet.hu:2101\r\n";
|
||||||
header += "Ntrip-Vesrsion:Ntrip/2.0\r\n";
|
header += "Ntrip-Vesrsion:Ntrip/2.0\r\n";
|
||||||
@ -412,6 +478,7 @@ class MapViewController extends GetxController {
|
|||||||
gpsLongitude.value = sentence.longitude;
|
gpsLongitude.value = sentence.longitude;
|
||||||
gpsLongitudeDirection.value = sentence.longitudeDirection;
|
gpsLongitudeDirection.value = sentence.longitudeDirection;
|
||||||
gpsAltitude.value = sentence.altitudeAboveMeanSeaLevel;
|
gpsAltitude.value = sentence.altitudeAboveMeanSeaLevel;
|
||||||
|
gpsGeoidSeparation.value = sentence.geoidSeparation;
|
||||||
gpsQuality.value = sentence.gpsQualityIndicator;
|
gpsQuality.value = sentence.gpsQualityIndicator;
|
||||||
eov.value = ConvertCoordinate.ConvertWgsToEov(
|
eov.value = ConvertCoordinate.ConvertWgsToEov(
|
||||||
gpsLatitude.value, gpsLongitude.value);
|
gpsLatitude.value, gpsLongitude.value);
|
||||||
@ -441,7 +508,7 @@ class MapViewController extends GetxController {
|
|||||||
(DateTime.now()
|
(DateTime.now()
|
||||||
.difference(lastSendTimeGgaMessage)
|
.difference(lastSendTimeGgaMessage)
|
||||||
.inSeconds >=
|
.inSeconds >=
|
||||||
30)) {
|
5)) {
|
||||||
sendGgaMessage(lastGgaMessage);
|
sendGgaMessage(lastGgaMessage);
|
||||||
print("Send GGA message: $lastGgaMessage");
|
print("Send GGA message: $lastGgaMessage");
|
||||||
ggaSenDataPacketNumber.value++;
|
ggaSenDataPacketNumber.value++;
|
||||||
@ -723,6 +790,31 @@ class MapViewController extends GetxController {
|
|||||||
"gpsHeight": gpsHeightController.text
|
"gpsHeight": gpsHeightController.text
|
||||||
});
|
});
|
||||||
|
|
||||||
|
await Supabase.instance.client
|
||||||
|
.from('TerepiSeged_MeasuredPoints')
|
||||||
|
.insert({
|
||||||
|
'pointNumber': pointId,
|
||||||
|
'gnssNumber': gpsName.value,
|
||||||
|
'latitude': gpsLatitude.value,
|
||||||
|
'longitude': gpsLongitude.value,
|
||||||
|
'altitude': gpsAltitude.value,
|
||||||
|
'heightOfGeoid': gpsGeoidSeparation.value,
|
||||||
|
'eovX': eov.value.X,
|
||||||
|
'eovY': eov.value.Y,
|
||||||
|
'poleHeight': double.tryParse(gpsHeightController.text),
|
||||||
|
'horizontalError':
|
||||||
|
max(gpsLatitudeError.value, gpsLongitudeError.value),
|
||||||
|
'verticalError': gpsAltitudeError.value,
|
||||||
|
'description': pointDescriptionController.text,
|
||||||
|
'isDeleted': false,
|
||||||
|
'projectId': 2
|
||||||
|
});
|
||||||
|
|
||||||
|
await Supabase.instance.client
|
||||||
|
.from('TerepiSeged_Receiver')
|
||||||
|
.update({'isMeasured': true}).eq(
|
||||||
|
'pointNumber', pointId);
|
||||||
|
|
||||||
if (pointMeasuringDirectionForward.isTrue) {
|
if (pointMeasuringDirectionForward.isTrue) {
|
||||||
if (pointsToMeasureSelectedValue.value <
|
if (pointsToMeasureSelectedValue.value <
|
||||||
pointsToMeasure.length - 1) {
|
pointsToMeasure.length - 1) {
|
||||||
@ -853,6 +945,79 @@ class MapViewController extends GetxController {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void readPointsFromSupa() async {
|
||||||
|
// final AuthResponse res = await Supabase.instance.client.auth
|
||||||
|
// .signInWithPassword(email: 'test.elek.1@email.hu', password: 'demo');
|
||||||
|
// final Session? session = res.session;
|
||||||
|
// final User? user = res.user;
|
||||||
|
|
||||||
|
final data =
|
||||||
|
await Supabase.instance.client.from('TerepiSeged_Receiver').select();
|
||||||
|
print('supa count ->${data.length}');
|
||||||
|
|
||||||
|
for (int i = 0; i < data.length; i++) {
|
||||||
|
var items = data[i];
|
||||||
|
if (items.length >= 3) {
|
||||||
|
var id = items['pointNumber'];
|
||||||
|
num numEovX = items['eovX'];
|
||||||
|
var coordX = numEovX.toDouble();
|
||||||
|
num numEovY = items['eovY'];
|
||||||
|
var coordY = numEovY.toDouble();
|
||||||
|
|
||||||
|
if (id != null && coordX != null && coordY != null) {
|
||||||
|
pointsToMeasure
|
||||||
|
.add(PointToMeasure(id: id, coordX: coordX, coordY: coordY));
|
||||||
|
|
||||||
|
var eovCoord = proj4.Point(x: coordX, y: coordY);
|
||||||
|
var wgsCoord = eovProj.transform(wgsProj, eovCoord);
|
||||||
|
|
||||||
|
print("Lat -> ${wgsCoord.x}, Long -> ${wgsCoord.y}");
|
||||||
|
|
||||||
|
pointsToMeasureMarker.add(Marker(
|
||||||
|
point: LatLng(wgsCoord.y, wgsCoord.x),
|
||||||
|
width: 15.0,
|
||||||
|
height: 15.0,
|
||||||
|
child: Container(
|
||||||
|
width: 15.0,
|
||||||
|
height: 15.0,
|
||||||
|
decoration: BoxDecoration(
|
||||||
|
color: Colors.purple,
|
||||||
|
shape: BoxShape.circle,
|
||||||
|
border: Border.all(width: 1.0, color: Colors.black)),
|
||||||
|
)));
|
||||||
|
|
||||||
|
pointsToMeasureLabel.add(PolyWidget(
|
||||||
|
center: LatLng(wgsCoord.y + 0.0000075, wgsCoord.x + 0.0000075),
|
||||||
|
widthInMeters: 3,
|
||||||
|
heightInMeters: 3,
|
||||||
|
// constraints: const BoxConstraints(
|
||||||
|
// minWidth: 250,
|
||||||
|
// maxWidth: 350,
|
||||||
|
// ),
|
||||||
|
child: FittedBox(
|
||||||
|
child: Padding(
|
||||||
|
padding: const EdgeInsets.all(4.0),
|
||||||
|
child: Text(
|
||||||
|
' $id ',
|
||||||
|
style: const TextStyle(
|
||||||
|
fontWeight: FontWeight.bold,
|
||||||
|
color: Colors.yellow,
|
||||||
|
fontSize: 12.0),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
)));
|
||||||
|
|
||||||
|
pointsToMeasureDropDownMenuItem.add(DropdownMenuItem<int>(
|
||||||
|
value: i,
|
||||||
|
child: Text('$id'),
|
||||||
|
));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
pointsToMeasureSelectedValue.value = 0;
|
||||||
|
print('Converted points number -> ${pointsToMeasure.length}');
|
||||||
|
}
|
||||||
|
|
||||||
void pointsToMeasureSelectedValueChanged(int value) {
|
void pointsToMeasureSelectedValueChanged(int value) {
|
||||||
pointsToMeasureSelectedValue.value = value;
|
pointsToMeasureSelectedValue.value = value;
|
||||||
print('Selected point -> ${pointsToMeasureSelectedValue.value}');
|
print('Selected point -> ${pointsToMeasureSelectedValue.value}');
|
||||||
@ -887,4 +1052,101 @@ class MapViewController extends GetxController {
|
|||||||
void showAddPointDialog() {
|
void showAddPointDialog() {
|
||||||
onBottomNavigationBarTap(0);
|
onBottomNavigationBarTap(0);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void saveGpsAddress(String address) {
|
||||||
|
prefs.setString('gpsAddress', address);
|
||||||
|
}
|
||||||
|
|
||||||
|
void saveGpsName(String name) {
|
||||||
|
prefs.setString('gpsName', name);
|
||||||
|
}
|
||||||
|
|
||||||
|
void saveNtripUserName(String username) {
|
||||||
|
prefs.setString('ntripUserName', username);
|
||||||
|
}
|
||||||
|
|
||||||
|
void saveNtripPassword(String password) {
|
||||||
|
prefs.setString('ntripPassword', password);
|
||||||
|
}
|
||||||
|
|
||||||
|
void updatePointStatus(int pointId) {}
|
||||||
|
|
||||||
|
void toggleShowPassword() {
|
||||||
|
isShowPassword.value = !isShowPassword.value;
|
||||||
|
if (passwordFieldFocusNode.hasPrimaryFocus) return;
|
||||||
|
passwordFieldFocusNode.canRequestFocus = false;
|
||||||
|
}
|
||||||
|
|
||||||
|
void showMeasuredPointsTableDialog() {
|
||||||
|
Get.to(() => MeasuredPointsTableDialog(), transition: Transition.fadeIn);
|
||||||
|
}
|
||||||
|
|
||||||
|
Future<List> readMeasuredPoints() async {
|
||||||
|
var response = await Supabase.instance.client
|
||||||
|
.from('TerepiSeged_MeasuredPoints')
|
||||||
|
.select()
|
||||||
|
.eq('projectId', 2)
|
||||||
|
.order('created_at');
|
||||||
|
|
||||||
|
print(response);
|
||||||
|
|
||||||
|
return response;
|
||||||
|
}
|
||||||
|
|
||||||
|
void SaveMeasuredPointsToFile() async {
|
||||||
|
// var pointsDirectory = await getExternalStorageDirectory();
|
||||||
|
var pointsDirectory = await getApplicationDocumentsDirectory();
|
||||||
|
print(directory!.path);
|
||||||
|
// String newPath = '';
|
||||||
|
// List<String> folders = directory!.path.split("/");
|
||||||
|
// for (int i = 1; i < folders.length; i++) {
|
||||||
|
// String folder = folders[i];
|
||||||
|
// if (folder != "Android") {
|
||||||
|
// newPath += "/" + folder;
|
||||||
|
// } else {
|
||||||
|
// break;
|
||||||
|
// }
|
||||||
|
// }
|
||||||
|
// newPath = newPath + "/TerepisSegedApp";
|
||||||
|
// directory = Directory(newPath);
|
||||||
|
if (!await pointsDirectory!.exists()) {
|
||||||
|
await pointsDirectory.create(recursive: true);
|
||||||
|
}
|
||||||
|
var oldMeasuredPointsFile = File("${directory!.path}/measuredsPoints.csv");
|
||||||
|
|
||||||
|
if (await oldMeasuredPointsFile.exists()) {
|
||||||
|
await oldMeasuredPointsFile.delete();
|
||||||
|
}
|
||||||
|
|
||||||
|
var measuredPointsFile =
|
||||||
|
await File("${directory!.path}/measuredsPoints.csv").create();
|
||||||
|
|
||||||
|
if (await pointsDirectory.exists()) {
|
||||||
|
if (await measuredPointsFile.exists()) {
|
||||||
|
await measuredPointsFile.writeAsString(
|
||||||
|
"Id;DateTime;Description;EovX;EovY;Altitude;Hor.Err;Vert.Err\r\n");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
var data = await readMeasuredPoints();
|
||||||
|
|
||||||
|
data.forEach((d) {
|
||||||
|
measuredPointsFile.writeAsStringSync(
|
||||||
|
"${d['id']};${d['created_at']};${d['description']};${formatEov.format(d['eovY'])};${formatEov.format(d['eovX'])};${formatEovZ.format(d['altitude'] - d['poleHeight'])};${formatAltitudeError.format(d['horizontalError'])};${formatAltitudeError.format(d['verticalError'])}\r\n",
|
||||||
|
flush: true,
|
||||||
|
mode: FileMode.append,
|
||||||
|
encoding: utf8);
|
||||||
|
});
|
||||||
|
|
||||||
|
print('Number of data: ${data.length}');
|
||||||
|
|
||||||
|
final params = ShareParams(
|
||||||
|
text: "Mérési eredmények",
|
||||||
|
// files: [XFile('$measuredPointsFile')],
|
||||||
|
files: [XFile("${directory!.path}/measuredsPoints.csv")],
|
||||||
|
subject: 'Mérési eredmények',
|
||||||
|
title: 'Mérési eredmények');
|
||||||
|
|
||||||
|
final result = await SharePlus.instance.share(params);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@ -30,7 +30,7 @@ class MapView extends GetView<MapViewController> {
|
|||||||
children: [
|
children: [
|
||||||
Text('Térkép'),
|
Text('Térkép'),
|
||||||
Text(
|
Text(
|
||||||
"Budapest",
|
"",
|
||||||
style: TextStyle(fontSize: 12.0),
|
style: TextStyle(fontSize: 12.0),
|
||||||
)
|
)
|
||||||
],
|
],
|
||||||
@ -89,8 +89,27 @@ class MapView extends GetView<MapViewController> {
|
|||||||
Padding(
|
Padding(
|
||||||
padding: const EdgeInsets.only(right: 20.0),
|
padding: const EdgeInsets.only(right: 20.0),
|
||||||
child: GestureDetector(
|
child: GestureDetector(
|
||||||
onTap: () => Get.to(() => SettingsDialog(),
|
onTap: () {
|
||||||
transition: Transition.downToUp),
|
controller.readPointsFromSupa();
|
||||||
|
},
|
||||||
|
child: const Icon(Icons.cloud_download,
|
||||||
|
size: 26.0, color: Colors.blue),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
Padding(
|
||||||
|
padding: const EdgeInsets.only(right: 20.0),
|
||||||
|
child: GestureDetector(
|
||||||
|
onTap: () {
|
||||||
|
if (controller.ntripUserName.value.isNotEmpty) {
|
||||||
|
controller.ntripUsernameController.text =
|
||||||
|
controller.ntripUserName.value;
|
||||||
|
}
|
||||||
|
if (controller.ntripPassword.value.isNotEmpty) {
|
||||||
|
controller.ntripPasswordController.text =
|
||||||
|
controller.ntripPassword.value;
|
||||||
|
}
|
||||||
|
Get.to(() => SettingsDialog(), transition: Transition.downToUp);
|
||||||
|
},
|
||||||
child: const Icon(
|
child: const Icon(
|
||||||
Icons.more_vert,
|
Icons.more_vert,
|
||||||
size: 26.0,
|
size: 26.0,
|
||||||
@ -224,10 +243,11 @@ class MapView extends GetView<MapViewController> {
|
|||||||
? FlutterMap(
|
? FlutterMap(
|
||||||
mapController: controller.mapController,
|
mapController: controller.mapController,
|
||||||
options: MapOptions(
|
options: MapOptions(
|
||||||
center: LatLng(controller.currentLatitude.value,
|
initialCenter: LatLng(
|
||||||
|
controller.currentLatitude.value,
|
||||||
controller.currentLongitude.value),
|
controller.currentLongitude.value),
|
||||||
maxZoom: 25,
|
maxZoom: 25,
|
||||||
zoom: controller.currentZoom.value,
|
initialZoom: controller.currentZoom.value,
|
||||||
),
|
),
|
||||||
children: [
|
children: [
|
||||||
TileLayer(
|
TileLayer(
|
||||||
@ -243,7 +263,7 @@ class MapView extends GetView<MapViewController> {
|
|||||||
),
|
),
|
||||||
MarkerLayer(
|
MarkerLayer(
|
||||||
markers: controller.currentLocationMarker),
|
markers: controller.currentLocationMarker),
|
||||||
MarkerLayer(markers: controller.parser.markers),
|
// MarkerLayer(markers: controller.parser.markers),
|
||||||
MarkerLayer(
|
MarkerLayer(
|
||||||
markers: controller.pointsToMeasureMarker),
|
markers: controller.pointsToMeasureMarker),
|
||||||
// PolylineLayer(
|
// PolylineLayer(
|
||||||
@ -438,14 +458,7 @@ class MapView extends GetView<MapViewController> {
|
|||||||
FloatingActionButton(
|
FloatingActionButton(
|
||||||
onPressed: () {
|
onPressed: () {
|
||||||
// controller.isMapMoveToCenter();
|
// controller.isMapMoveToCenter();
|
||||||
// controller.addMeasuredPoint();
|
controller.showMeasuredPointsTableDialog();
|
||||||
ScaffoldMessenger.of(context).showSnackBar(const SnackBar(
|
|
||||||
content: Text(
|
|
||||||
"Fejlesztlés alatt",
|
|
||||||
style: TextStyle(fontWeight: FontWeight.bold),
|
|
||||||
),
|
|
||||||
backgroundColor: Colors.black54,
|
|
||||||
));
|
|
||||||
},
|
},
|
||||||
heroTag: 'Database test',
|
heroTag: 'Database test',
|
||||||
tooltip: 'Pont bemérése',
|
tooltip: 'Pont bemérése',
|
||||||
|
|||||||
@ -0,0 +1,124 @@
|
|||||||
|
import 'package:flutter/material.dart';
|
||||||
|
import 'package:get/get.dart';
|
||||||
|
import 'package:terepi_seged/pages/map/presentation/controllers/map_controller.dart';
|
||||||
|
|
||||||
|
class MeasuredPointsTableDialog extends StatelessWidget {
|
||||||
|
final controller = Get.find<MapViewController>();
|
||||||
|
MeasuredPointsTableDialog({super.key});
|
||||||
|
|
||||||
|
Widget build(BuildContext context) {
|
||||||
|
return Scaffold(
|
||||||
|
body: Padding(
|
||||||
|
padding: const EdgeInsets.only(top: 20.0),
|
||||||
|
child: Column(
|
||||||
|
children: [
|
||||||
|
Padding(
|
||||||
|
padding: const EdgeInsets.all(8.0),
|
||||||
|
child: Row(
|
||||||
|
mainAxisAlignment: MainAxisAlignment.spaceBetween,
|
||||||
|
children: [
|
||||||
|
Row(children: [
|
||||||
|
IconButton(
|
||||||
|
onPressed: () {
|
||||||
|
Get.back();
|
||||||
|
},
|
||||||
|
icon: const Icon(Icons.close)),
|
||||||
|
SizedBox(
|
||||||
|
width: 10,
|
||||||
|
),
|
||||||
|
IconButton(
|
||||||
|
onPressed: () {
|
||||||
|
controller.SaveMeasuredPointsToFile();
|
||||||
|
},
|
||||||
|
icon: const Icon(Icons.save)),
|
||||||
|
]),
|
||||||
|
TextButton(
|
||||||
|
style: ButtonStyle(
|
||||||
|
overlayColor:
|
||||||
|
MaterialStateProperty.all(Colors.transparent)),
|
||||||
|
onPressed: () {
|
||||||
|
Get.back();
|
||||||
|
},
|
||||||
|
child: const Text(
|
||||||
|
'Bezár',
|
||||||
|
style: TextStyle(color: Colors.blue, fontSize: 14.0),
|
||||||
|
))
|
||||||
|
],
|
||||||
|
),
|
||||||
|
),
|
||||||
|
const Padding(
|
||||||
|
padding: EdgeInsets.symmetric(horizontal: 40.0),
|
||||||
|
child: Text(
|
||||||
|
'Bemért pontok',
|
||||||
|
style: TextStyle(fontSize: 20.0, fontWeight: FontWeight.bold),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
const SizedBox(height: 5),
|
||||||
|
Expanded(
|
||||||
|
child: FutureBuilder<List<dynamic>>(
|
||||||
|
future: controller.readMeasuredPoints(),
|
||||||
|
builder: (context, snapshot) {
|
||||||
|
if (snapshot.connectionState == ConnectionState.waiting) {
|
||||||
|
return Center(child: CircularProgressIndicator());
|
||||||
|
}
|
||||||
|
|
||||||
|
if (snapshot.hasError) {
|
||||||
|
return Center(
|
||||||
|
child: Text(
|
||||||
|
snapshot.error.toString(),
|
||||||
|
),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
if (!snapshot.hasData) {
|
||||||
|
return const Center(
|
||||||
|
child: Text("No Data available.\n Create new Data"));
|
||||||
|
}
|
||||||
|
// print(snapshot.data);
|
||||||
|
// return const Center(child: Text("Data available."));
|
||||||
|
return ListView.builder(
|
||||||
|
itemCount: snapshot.data!.length,
|
||||||
|
shrinkWrap: true,
|
||||||
|
itemBuilder: (context, int index) {
|
||||||
|
var data = snapshot.data![index];
|
||||||
|
print("snapshot data:");
|
||||||
|
print(data);
|
||||||
|
return ListTile(
|
||||||
|
leading: CircleAvatar(
|
||||||
|
backgroundColor: const Color(0xff764abc),
|
||||||
|
child: Text((index + 1).toString())),
|
||||||
|
title: Text(data['pointNumber'].toString(),
|
||||||
|
style: TextStyle(fontWeight: FontWeight.bold)),
|
||||||
|
subtitle: Column(
|
||||||
|
crossAxisAlignment: CrossAxisAlignment.start,
|
||||||
|
children: [
|
||||||
|
Text(data['description'],
|
||||||
|
style: TextStyle(
|
||||||
|
fontStyle: FontStyle.italic,
|
||||||
|
color: Colors.grey.shade700)),
|
||||||
|
Text(
|
||||||
|
"EovX: ${controller.formatEov.format(data['eovY'])} - EovY: ${controller.formatEov.format(data['eovX'])}",
|
||||||
|
style: TextStyle(
|
||||||
|
fontStyle: FontStyle.italic,
|
||||||
|
color: Colors.grey.shade400)),
|
||||||
|
Text(
|
||||||
|
"EovZ: ${controller.formatEovZ.format(data['altitude'] - data['poleHeight'])} (m)",
|
||||||
|
style: TextStyle(
|
||||||
|
fontStyle: FontStyle.italic,
|
||||||
|
color: Colors.grey.shade400)),
|
||||||
|
Text(
|
||||||
|
"H.hiba: ${controller.formatAltitudeError.format(data['horizontalError'])} (m) - V.hiba: ${controller.formatAltitudeError.format(data['verticalError'])} (m)",
|
||||||
|
style: TextStyle(
|
||||||
|
fontStyle: FontStyle.italic,
|
||||||
|
color: Colors.grey.shade400)),
|
||||||
|
],
|
||||||
|
));
|
||||||
|
});
|
||||||
|
},
|
||||||
|
),
|
||||||
|
)
|
||||||
|
],
|
||||||
|
),
|
||||||
|
),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
@ -6,7 +6,6 @@ class SettingsDialog extends StatelessWidget {
|
|||||||
final controller = Get.find<MapViewController>();
|
final controller = Get.find<MapViewController>();
|
||||||
SettingsDialog({super.key});
|
SettingsDialog({super.key});
|
||||||
|
|
||||||
@override
|
|
||||||
Widget build(BuildContext context) {
|
Widget build(BuildContext context) {
|
||||||
return Scaffold(
|
return Scaffold(
|
||||||
body: ListView(
|
body: ListView(
|
||||||
@ -18,6 +17,19 @@ class SettingsDialog extends StatelessWidget {
|
|||||||
children: [
|
children: [
|
||||||
IconButton(
|
IconButton(
|
||||||
onPressed: () {
|
onPressed: () {
|
||||||
|
if (controller.ntripUsernameController.text.isNotEmpty) {
|
||||||
|
controller.ntripUserName.value =
|
||||||
|
controller.ntripUsernameController.text;
|
||||||
|
controller.saveNtripUserName(
|
||||||
|
controller.ntripUsernameController.text);
|
||||||
|
if (controller
|
||||||
|
.ntripPasswordController.text.isNotEmpty) {
|
||||||
|
controller.ntripPassword.value =
|
||||||
|
controller.ntripPasswordController.text;
|
||||||
|
controller.saveNtripPassword(
|
||||||
|
controller.ntripPasswordController.text);
|
||||||
|
}
|
||||||
|
}
|
||||||
Get.back();
|
Get.back();
|
||||||
},
|
},
|
||||||
icon: const Icon(Icons.close)),
|
icon: const Icon(Icons.close)),
|
||||||
@ -25,7 +37,22 @@ class SettingsDialog extends StatelessWidget {
|
|||||||
style: ButtonStyle(
|
style: ButtonStyle(
|
||||||
overlayColor:
|
overlayColor:
|
||||||
MaterialStateProperty.all(Colors.transparent)),
|
MaterialStateProperty.all(Colors.transparent)),
|
||||||
onPressed: () {},
|
onPressed: () {
|
||||||
|
if (controller.ntripUsernameController.text.isNotEmpty) {
|
||||||
|
controller.ntripUserName.value =
|
||||||
|
controller.ntripUsernameController.text;
|
||||||
|
controller.saveNtripUserName(
|
||||||
|
controller.ntripUsernameController.text);
|
||||||
|
if (controller
|
||||||
|
.ntripPasswordController.text.isNotEmpty) {
|
||||||
|
controller.ntripPassword.value =
|
||||||
|
controller.ntripPasswordController.text;
|
||||||
|
controller.saveNtripPassword(
|
||||||
|
controller.ntripPasswordController.text);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
Get.back();
|
||||||
|
},
|
||||||
child: const Text(
|
child: const Text(
|
||||||
'Bezár',
|
'Bezár',
|
||||||
style: TextStyle(color: Colors.blue, fontSize: 14.0),
|
style: TextStyle(color: Colors.blue, fontSize: 14.0),
|
||||||
@ -48,12 +75,95 @@ class SettingsDialog extends StatelessWidget {
|
|||||||
),
|
),
|
||||||
),
|
),
|
||||||
Obx(() => Column(children: [
|
Obx(() => Column(children: [
|
||||||
|
RadioListTile(
|
||||||
|
title: Text('TiGNSS Rover-BE6A'),
|
||||||
|
value: '10:06:1C:97:BE:6A',
|
||||||
|
groupValue: controller.gpsAddress.value,
|
||||||
|
onChanged: (value) {
|
||||||
|
controller.gpsAddress.value = value!;
|
||||||
|
controller.gpsName.value = 'TiGNSS Rover-BE6A';
|
||||||
|
controller.saveGpsAddress(value);
|
||||||
|
controller.saveGpsName('TiGNSS Rover-BE6A');
|
||||||
|
}),
|
||||||
|
RadioListTile(
|
||||||
|
title: Text('TiGNSS Rover-1DC6'),
|
||||||
|
value: 'E8:31:CD:16:1D:C6',
|
||||||
|
groupValue: controller.gpsAddress.value,
|
||||||
|
onChanged: (value) {
|
||||||
|
controller.gpsAddress.value = value!;
|
||||||
|
controller.gpsName.value = 'TiGNSS Rover-1DC6';
|
||||||
|
controller.saveGpsAddress(value);
|
||||||
|
controller.saveGpsName('TiGNSS Rover-1DC6');
|
||||||
|
}),
|
||||||
|
RadioListTile(
|
||||||
|
title: Text('TiGNSS Rover-9C3A'),
|
||||||
|
value: '08:3A:8D:14:9C:3A',
|
||||||
|
groupValue: controller.gpsAddress.value,
|
||||||
|
onChanged: (value) {
|
||||||
|
controller.gpsAddress.value = value!;
|
||||||
|
controller.gpsName.value = 'TiGNSS Rover-9C3A';
|
||||||
|
controller.saveGpsAddress(value);
|
||||||
|
controller.saveGpsName('TiGNSS Rover-9C3A');
|
||||||
|
}),
|
||||||
|
RadioListTile(
|
||||||
|
title: Text('TiGNSS Rover-72C2'),
|
||||||
|
value: '10:06:1C:97:72:C2',
|
||||||
|
groupValue: controller.gpsAddress.value,
|
||||||
|
onChanged: (value) {
|
||||||
|
controller.gpsAddress.value = value!;
|
||||||
|
controller.gpsName.value = 'TiGNSS Rover-72C2';
|
||||||
|
controller.saveGpsAddress(value);
|
||||||
|
controller.saveGpsName('TiGNSS Rover-72C2');
|
||||||
|
}),
|
||||||
|
RadioListTile(
|
||||||
|
title: Text('TiGNSS Rover-FE16'),
|
||||||
|
value: '10:06:1C:9F:FE:16',
|
||||||
|
groupValue: controller.gpsAddress.value,
|
||||||
|
onChanged: (value) {
|
||||||
|
controller.gpsAddress.value = value!;
|
||||||
|
controller.gpsName.value = 'TiGNSS Rover-FE16';
|
||||||
|
controller.saveGpsAddress(value);
|
||||||
|
controller.saveGpsName('TiGNSS Rover-FE16');
|
||||||
|
}),
|
||||||
|
RadioListTile(
|
||||||
|
title: Text('TiGNSS Rover-3B0A'),
|
||||||
|
value: '10:C6:1C:9E:3B:0A',
|
||||||
|
groupValue: controller.gpsAddress.value,
|
||||||
|
onChanged: (value) {
|
||||||
|
controller.gpsAddress.value = value!;
|
||||||
|
controller.gpsName.value = 'TiGNSS Rover-3B0A';
|
||||||
|
controller.saveGpsAddress(value);
|
||||||
|
controller.saveGpsName('TiGNSS Rover-3B0A');
|
||||||
|
}),
|
||||||
|
RadioListTile(
|
||||||
|
title: Text('TiGNSS Rover-7FEA'),
|
||||||
|
value: '10:06:1C:9C:7F:EA',
|
||||||
|
groupValue: controller.gpsAddress.value,
|
||||||
|
onChanged: (value) {
|
||||||
|
controller.gpsAddress.value = value!;
|
||||||
|
controller.gpsName.value = 'TiGNSS Rover-7FEA';
|
||||||
|
controller.saveGpsAddress(value);
|
||||||
|
controller.saveGpsName('TiGNSS Rover-7FEA');
|
||||||
|
}),
|
||||||
|
RadioListTile(
|
||||||
|
title: Text('TiGNSS Rover-A39E'),
|
||||||
|
value: '10:06:1C:97:A3:9E',
|
||||||
|
groupValue: controller.gpsAddress.value,
|
||||||
|
onChanged: (value) {
|
||||||
|
controller.gpsAddress.value = value!;
|
||||||
|
controller.gpsName.value = 'TiGNSS Rover-A39E';
|
||||||
|
controller.saveGpsAddress(value);
|
||||||
|
controller.saveGpsName('TiGNSS Rover-A39E');
|
||||||
|
}),
|
||||||
RadioListTile(
|
RadioListTile(
|
||||||
title: Text('TiGNSS Rover-FF4E'),
|
title: Text('TiGNSS Rover-FF4E'),
|
||||||
value: '98:CD:AC:62:FF:4E',
|
value: '98:CD:AC:62:FF:4E',
|
||||||
groupValue: controller.gpsAddress.value,
|
groupValue: controller.gpsAddress.value,
|
||||||
onChanged: (value) {
|
onChanged: (value) {
|
||||||
controller.gpsAddress.value = value!;
|
controller.gpsAddress.value = value!;
|
||||||
|
controller.gpsName.value = 'TiGNSS Rover-FF4E';
|
||||||
|
controller.saveGpsAddress(value);
|
||||||
|
controller.saveGpsName('TiGNSS Rover-FF4E');
|
||||||
}),
|
}),
|
||||||
RadioListTile(
|
RadioListTile(
|
||||||
title: Text('TiGNSS Rover-8BB2'),
|
title: Text('TiGNSS Rover-8BB2'),
|
||||||
@ -61,7 +171,9 @@ class SettingsDialog extends StatelessWidget {
|
|||||||
groupValue: controller.gpsAddress.value,
|
groupValue: controller.gpsAddress.value,
|
||||||
onChanged: (value) {
|
onChanged: (value) {
|
||||||
controller.gpsAddress.value = value!;
|
controller.gpsAddress.value = value!;
|
||||||
;
|
controller.gpsName.value = 'TiGNSS Rover-8BB2';
|
||||||
|
controller.saveGpsAddress(value);
|
||||||
|
controller.saveGpsName('TiGNSS Rover-8BB2');
|
||||||
}),
|
}),
|
||||||
RadioListTile(
|
RadioListTile(
|
||||||
title: Text('TiGNSS Rover-FF36'),
|
title: Text('TiGNSS Rover-FF36'),
|
||||||
@ -69,6 +181,9 @@ class SettingsDialog extends StatelessWidget {
|
|||||||
groupValue: controller.gpsAddress.value,
|
groupValue: controller.gpsAddress.value,
|
||||||
onChanged: (value) {
|
onChanged: (value) {
|
||||||
controller.gpsAddress.value = value!;
|
controller.gpsAddress.value = value!;
|
||||||
|
controller.gpsName.value = 'TiGNSS Rover-FF36';
|
||||||
|
controller.saveGpsAddress(value);
|
||||||
|
controller.saveGpsName('TiGNSS Rover-FF36');
|
||||||
})
|
})
|
||||||
])),
|
])),
|
||||||
const Padding(
|
const Padding(
|
||||||
@ -215,26 +330,54 @@ class SettingsDialog extends StatelessWidget {
|
|||||||
height: 40,
|
height: 40,
|
||||||
child: TextField(
|
child: TextField(
|
||||||
controller: controller.ntripUsernameController,
|
controller: controller.ntripUsernameController,
|
||||||
decoration: const InputDecoration(
|
enableSuggestions: false,
|
||||||
border: OutlineInputBorder(),
|
autocorrect: false,
|
||||||
|
decoration: InputDecoration(
|
||||||
|
floatingLabelBehavior: FloatingLabelBehavior.never,
|
||||||
|
isDense: true,
|
||||||
|
filled: true,
|
||||||
|
fillColor: Colors.grey.shade300,
|
||||||
|
border: OutlineInputBorder(
|
||||||
|
borderSide: BorderSide.none,
|
||||||
|
borderRadius: BorderRadius.circular(12)),
|
||||||
labelText: 'Felhasználónév',
|
labelText: 'Felhasználónév',
|
||||||
icon: Icon(Icons.account_circle_rounded)),
|
prefixIcon: Icon(Icons.account_circle_rounded)),
|
||||||
),
|
),
|
||||||
),
|
),
|
||||||
const SizedBox(height: 10),
|
const SizedBox(height: 10),
|
||||||
SizedBox(
|
Obx(
|
||||||
height: 40,
|
() => SizedBox(
|
||||||
child: TextField(
|
height: 40,
|
||||||
obscureText: true,
|
child: TextField(
|
||||||
enableSuggestions: false,
|
keyboardType: TextInputType.visiblePassword,
|
||||||
autocorrect: false,
|
obscureText: !controller.isShowPassword.value,
|
||||||
controller: controller.ntripPasswordController,
|
focusNode: controller.passwordFieldFocusNode,
|
||||||
decoration: const InputDecoration(
|
enableSuggestions: false,
|
||||||
border: OutlineInputBorder(),
|
autocorrect: false,
|
||||||
labelText: 'Jelszó',
|
controller: controller.ntripPasswordController,
|
||||||
icon: Icon(
|
decoration: InputDecoration(
|
||||||
Icons.lock,
|
floatingLabelBehavior: FloatingLabelBehavior.never,
|
||||||
)),
|
isDense: true,
|
||||||
|
filled: true,
|
||||||
|
fillColor: Colors.grey.shade300,
|
||||||
|
border: OutlineInputBorder(
|
||||||
|
borderSide: BorderSide.none,
|
||||||
|
borderRadius: BorderRadius.circular(12)),
|
||||||
|
labelText: 'Jelszó',
|
||||||
|
prefixIcon: Icon(
|
||||||
|
Icons.lock_rounded,
|
||||||
|
size: 24,
|
||||||
|
),
|
||||||
|
suffixIcon: Padding(
|
||||||
|
padding: const EdgeInsets.fromLTRB(0, 0, 4, 0),
|
||||||
|
child: GestureDetector(
|
||||||
|
onTap: controller.toggleShowPassword,
|
||||||
|
child: Icon(
|
||||||
|
controller.isShowPassword.value
|
||||||
|
? Icons.visibility_rounded
|
||||||
|
: Icons.visibility_off_rounded,
|
||||||
|
size: 24)))),
|
||||||
|
),
|
||||||
),
|
),
|
||||||
),
|
),
|
||||||
const SizedBox(height: 20)
|
const SizedBox(height: 20)
|
||||||
|
|||||||
9
lib/pages/map_survey/bindings/map_survey_bindings.dart
Normal file
9
lib/pages/map_survey/bindings/map_survey_bindings.dart
Normal file
@ -0,0 +1,9 @@
|
|||||||
|
import 'package:get/get.dart';
|
||||||
|
import 'package:terepi_seged/pages/map_survey/presentations/controllers/map_survey_controller.dart';
|
||||||
|
|
||||||
|
class MapSurveyBinding extends Bindings {
|
||||||
|
@override
|
||||||
|
void dependencies() {
|
||||||
|
Get.lazyPut(() => MapSurveyController());
|
||||||
|
}
|
||||||
|
}
|
||||||
@ -0,0 +1,876 @@
|
|||||||
|
import 'dart:async';
|
||||||
|
import 'dart:convert';
|
||||||
|
import 'dart:io';
|
||||||
|
import 'dart:math';
|
||||||
|
|
||||||
|
import 'package:cloud_firestore/cloud_firestore.dart';
|
||||||
|
import 'package:file_picker/file_picker.dart';
|
||||||
|
import 'package:flutter/material.dart';
|
||||||
|
import 'package:flutter/services.dart';
|
||||||
|
import 'package:flutter_bluetooth_serial/flutter_bluetooth_serial.dart';
|
||||||
|
import 'package:flutter_map/flutter_map.dart';
|
||||||
|
// import 'package:flutter_map_geojson/flutter_map_geojson.dart';
|
||||||
|
import 'package:flutter_map_polywidget/flutter_map_polywidget.dart';
|
||||||
|
import 'package:get/get.dart';
|
||||||
|
import 'package:intl/intl.dart';
|
||||||
|
import 'package:location/location.dart';
|
||||||
|
import 'package:latlong2/latlong.dart';
|
||||||
|
import 'package:nmea/nmea.dart';
|
||||||
|
import 'package:path_provider/path_provider.dart';
|
||||||
|
import 'package:permission_handler/permission_handler.dart'
|
||||||
|
as permission_handler;
|
||||||
|
import 'package:supabase_flutter/supabase_flutter.dart';
|
||||||
|
import 'package:terepi_seged/eov/convert_coordinate.dart';
|
||||||
|
import 'package:terepi_seged/eov/eov.dart';
|
||||||
|
import 'package:terepi_seged/gnss_sentences/gngga.dart';
|
||||||
|
import 'package:terepi_seged/gnss_sentences/gngst.dart';
|
||||||
|
import 'package:terepi_seged/gnss_sentences/gnrmc.dart';
|
||||||
|
import 'package:terepi_seged/models/point_to_measure.dart';
|
||||||
|
import 'package:terepi_seged/models/point_with_description_model.dart';
|
||||||
|
import 'package:proj4dart/proj4dart.dart' as proj4;
|
||||||
|
import 'package:shared_preferences/shared_preferences.dart';
|
||||||
|
import 'package:terepi_seged/pages/map_survey/presentations/views/measured_points_table_dialog.dart';
|
||||||
|
|
||||||
|
class MapSurveyController extends GetxController {
|
||||||
|
// String gpsAddress = "E8:31:CD:14:8B:B2";
|
||||||
|
// String gpsAddress = "98:CD:AC:62:FF:4E";
|
||||||
|
RxString gpsAddress = "98:CD:AC:62:FF:36".obs;
|
||||||
|
RxString gpsName = "TiGNSS Rover-FF4E".obs;
|
||||||
|
// String gpsName = "TiGNSS Rover-8BB2";
|
||||||
|
static const double maxZoomValue = 25.0;
|
||||||
|
Rx<bool> gpsIsConnected = false.obs;
|
||||||
|
RxBool ntripIsConnected = false.obs;
|
||||||
|
RxInt ntripDataPacketNumbers = 0.obs;
|
||||||
|
RxInt ggaSenDataPacketNumber = 0.obs;
|
||||||
|
RxString ggaSendLastTimeStr = "".obs;
|
||||||
|
String _messageBuffer = "";
|
||||||
|
late BluetoothConnection connection;
|
||||||
|
late Socket socket;
|
||||||
|
late StreamSubscription socketStreamSubscription;
|
||||||
|
RxInt gpsReceivedData = 0.obs;
|
||||||
|
RxBool hasGpsValidData = false.obs;
|
||||||
|
RxBool isMapMoveToCenter = true.obs;
|
||||||
|
DateTime lastGpsRefreshTime = DateTime.now();
|
||||||
|
String utcOfPositionFix = "";
|
||||||
|
String utcDateOfPositionFix = "";
|
||||||
|
RxInt ntripReceivedData = 0.obs;
|
||||||
|
final NmeaDecoder nmeaDecoder = NmeaDecoder();
|
||||||
|
String lastGgaMessage = '';
|
||||||
|
DateTime lastSendTimeGgaMessage =
|
||||||
|
DateTime.now().add(const Duration(seconds: -30));
|
||||||
|
|
||||||
|
NumberFormat formatEov = NumberFormat("##0,000.0", "hu-HU");
|
||||||
|
NumberFormat formatEovZ = NumberFormat("###0.0", "hu-HU");
|
||||||
|
NumberFormat formatAltitudeError = NumberFormat("####0.000", "hu-HU");
|
||||||
|
NumberFormat formatEovForFile = NumberFormat("#####0.0", "hu-HU");
|
||||||
|
NumberFormat formatWgs84Sec = NumberFormat('00.000', 'hu-HU');
|
||||||
|
|
||||||
|
RxDouble gpsLatitude = 0.0.obs;
|
||||||
|
RxString gpsLatitudeDirection = "".obs;
|
||||||
|
RxDouble gpsLongitude = 0.0.obs;
|
||||||
|
RxString gpsLongitudeDirection = "".obs;
|
||||||
|
RxDouble gpsAltitude = 0.0.obs;
|
||||||
|
RxDouble gpsGeoidSeparation = 0.0.obs;
|
||||||
|
RxInt gpsQuality = 0.obs;
|
||||||
|
RxDouble gpsLatitudeError = 0.0.obs;
|
||||||
|
RxDouble gpsLongitudeError = 0.0.obs;
|
||||||
|
RxDouble gpsAltitudeError = 0.0.obs;
|
||||||
|
Rx<DateTime> gpsDateTime = DateTime(2000).obs;
|
||||||
|
Rx<Eov> eov = Eov(0, 0).obs;
|
||||||
|
RxInt latDegree = 0.obs;
|
||||||
|
RxInt latMin = 0.obs;
|
||||||
|
RxDouble latSec = 0.0.obs;
|
||||||
|
RxInt longDegree = 0.obs;
|
||||||
|
RxInt longMin = 0.obs;
|
||||||
|
RxDouble longSec = 0.0.obs;
|
||||||
|
|
||||||
|
RxDouble currentLongitude = 0.0.obs;
|
||||||
|
RxDouble currentLatitude = 0.0.obs;
|
||||||
|
RxDouble currentZoom = 12.0.obs;
|
||||||
|
late final MapController mapController;
|
||||||
|
final currentLocationMarker = <Marker>[];
|
||||||
|
final pointNotesMarker = <Marker>[];
|
||||||
|
|
||||||
|
List<PointToMeasure> pointsToMeasure = <PointToMeasure>[];
|
||||||
|
final pointsToMeasureMarker = <Marker>[];
|
||||||
|
final pointsToMeasureLabel = <PolyWidget>[];
|
||||||
|
|
||||||
|
TextEditingController pointIdController = TextEditingController();
|
||||||
|
TextEditingController pointDescriptionController = TextEditingController();
|
||||||
|
TextEditingController gpsHeightController = TextEditingController();
|
||||||
|
TextEditingController pointPrefixController = TextEditingController();
|
||||||
|
TextEditingController pointPostfixController = TextEditingController();
|
||||||
|
TextEditingController ntripUsernameController = TextEditingController();
|
||||||
|
TextEditingController ntripPasswordController = TextEditingController();
|
||||||
|
|
||||||
|
int pointId = 1;
|
||||||
|
String pointIdPrefix = "";
|
||||||
|
String pointIdPostfix = "";
|
||||||
|
RxString ntripUserName = "".obs;
|
||||||
|
RxString ntripPassword = "".obs;
|
||||||
|
|
||||||
|
Rx<bool> pointMeasuringDirectionForward = true.obs;
|
||||||
|
|
||||||
|
Rx<bool> isShowPassword = false.obs;
|
||||||
|
final passwordFieldFocusNode = FocusNode();
|
||||||
|
|
||||||
|
List? measuredPoints;
|
||||||
|
|
||||||
|
List<PointWithDescription> pointWithDescriptionList = [];
|
||||||
|
late Directory? directory;
|
||||||
|
late File dataFile;
|
||||||
|
|
||||||
|
late proj4.Projection eovProj, wgsProj;
|
||||||
|
RxBool mapIsInitialized = false.obs;
|
||||||
|
|
||||||
|
// GeoJsonParser parser =
|
||||||
|
// GeoJsonParser(defaultMarkerColor: const Color.fromARGB(255, 85, 34, 49));
|
||||||
|
|
||||||
|
final CollectionReference _measuredPoints =
|
||||||
|
FirebaseFirestore.instance.collection('measuredPoints');
|
||||||
|
|
||||||
|
// late SMIBool gpsTrigger;
|
||||||
|
// late StateMachineController riveGpsIconController;
|
||||||
|
|
||||||
|
late SharedPreferences prefs;
|
||||||
|
|
||||||
|
late AuthResponse authResponse;
|
||||||
|
late Session? session;
|
||||||
|
late User? user;
|
||||||
|
|
||||||
|
@override
|
||||||
|
void onInit() async {
|
||||||
|
super.onInit();
|
||||||
|
final bytes = (await rootBundle.load('assets/Grids/etrs2eov_notowgs.gsb'))
|
||||||
|
.buffer
|
||||||
|
.asUint8List();
|
||||||
|
proj4.Projection.nadgrid('Etrs2Eov', bytes);
|
||||||
|
// var def =
|
||||||
|
// '+proj=somerc +lat_0=47.14439372222222 +lon_0=19.04857177777778 +k_0=0.99993 +x_0=650000 +y_0=200000 +ellps=GRS67 +towgs84=52.17,-71.82,-14.9,0,0,0,0 +units=m +nadgrids=@ignorable,Etrs2Eov,null +no_defs';
|
||||||
|
var def =
|
||||||
|
'+proj=somerc +lat_0=47.14439372222222 +lon_0=19.04857177777778 +k_0=0.99993 +x_0=650000 +y_0=200000 +ellps=GRS67 +towgs84=52.17,-71.82,-14.9,0,0,0,0 +units=m +nadgrids=Etrs2Eov +no_defs';
|
||||||
|
|
||||||
|
// Named Projection signature, later find it from anywhere via Projection.get('EPSG:23700')
|
||||||
|
eovProj = proj4.Projection.add('EPSG:23700', def);
|
||||||
|
wgsProj = proj4.Projection.WGS84;
|
||||||
|
//Loading, Success, Error handle with 1 line of code
|
||||||
|
nmeaDecoder
|
||||||
|
..registerTalkerSentence("GGA", (line) => Gngga(raw: line))
|
||||||
|
..registerTalkerSentence("RMC", (line) => Gnrmc(raw: line))
|
||||||
|
..registerTalkerSentence("GST", (line) => Gngst(raw: line));
|
||||||
|
|
||||||
|
prefs = await SharedPreferences.getInstance();
|
||||||
|
|
||||||
|
authResponse = await Supabase.instance.client.auth
|
||||||
|
.signInWithPassword(email: 'test.elek.1@email.hu', password: 'demo');
|
||||||
|
session = authResponse.session;
|
||||||
|
user = authResponse.user;
|
||||||
|
|
||||||
|
Supabase.instance.client
|
||||||
|
.channel('public:TerepiSeged_Receiver')
|
||||||
|
.onPostgresChanges(
|
||||||
|
event: PostgresChangeEvent.update,
|
||||||
|
schema: 'public',
|
||||||
|
table: 'TerepiSeged_Receiver',
|
||||||
|
callback: (playload) {
|
||||||
|
var id = playload.newRecord['pointNumber'] as int;
|
||||||
|
print('Change received: ${id}');
|
||||||
|
updatePointStatus(id);
|
||||||
|
})
|
||||||
|
.subscribe();
|
||||||
|
|
||||||
|
mapController = MapController();
|
||||||
|
|
||||||
|
// riveGpsIconController = RiveUtils.getRiveController(Artboard(),
|
||||||
|
// stateMachineName: "gps_Interactivity");
|
||||||
|
// gpsTrigger = riveGpsIconController.findSMI("active");
|
||||||
|
|
||||||
|
mapIsInitialized.value = true;
|
||||||
|
}
|
||||||
|
|
||||||
|
@override
|
||||||
|
void onClose() async {
|
||||||
|
super.onClose();
|
||||||
|
FlutterBluetoothSerial.instance.setPairingRequestHandler(null);
|
||||||
|
if (gpsIsConnected.value) {
|
||||||
|
connection.close();
|
||||||
|
// connection = null;
|
||||||
|
print("BluetoothTestController dispose ....");
|
||||||
|
}
|
||||||
|
pointDescriptionController.dispose();
|
||||||
|
pointIdController.dispose();
|
||||||
|
gpsHeightController.dispose();
|
||||||
|
pointPrefixController.dispose();
|
||||||
|
pointPostfixController.dispose();
|
||||||
|
ntripUsernameController.dispose();
|
||||||
|
ntripPasswordController.dispose();
|
||||||
|
|
||||||
|
await Supabase.instance.client
|
||||||
|
.channel('public:TerepiSeged_Receiver')
|
||||||
|
.unsubscribe();
|
||||||
|
}
|
||||||
|
|
||||||
|
@override
|
||||||
|
void onReady() async {
|
||||||
|
super.onReady();
|
||||||
|
|
||||||
|
_getInitialLocation();
|
||||||
|
|
||||||
|
// String data = await rootBundle.loadString('assets/Files/transzekt.geojson');
|
||||||
|
// parser.parseGeoJsonAsString(data);
|
||||||
|
|
||||||
|
if (await permission_handler.Permission.storage.isGranted) {
|
||||||
|
print("Storage permission is ok ...");
|
||||||
|
} else {
|
||||||
|
var result = await permission_handler.Permission.storage.request();
|
||||||
|
if (result == permission_handler.PermissionStatus.granted) {
|
||||||
|
print("Storage permission request is ok ....");
|
||||||
|
} else {
|
||||||
|
print("No storage permission");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (await permission_handler.Permission.manageExternalStorage.isGranted) {
|
||||||
|
print("External storage permission is ok ...");
|
||||||
|
} else {
|
||||||
|
var result =
|
||||||
|
await permission_handler.Permission.manageExternalStorage.request();
|
||||||
|
if (result == permission_handler.PermissionStatus.granted) {
|
||||||
|
print("External storage permission request is ok ....");
|
||||||
|
} else {
|
||||||
|
print("No external storage permission");
|
||||||
|
}
|
||||||
|
if (prefs.containsKey('gpsAddress')) {
|
||||||
|
var address = prefs.getString('gpsAddress');
|
||||||
|
if (address != null) {
|
||||||
|
gpsAddress.value = address;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (prefs.containsKey('gpsName')) {
|
||||||
|
var name = prefs.getString('gpsName');
|
||||||
|
if (name != null) {
|
||||||
|
gpsName.value = name;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (prefs.containsKey('ntripUserName')) {
|
||||||
|
var userName = prefs.getString('ntripUserName');
|
||||||
|
if (userName != null) {
|
||||||
|
ntripUserName.value = userName;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (prefs.containsKey('ntripPassword')) {
|
||||||
|
var password = prefs.getString('ntripPassword');
|
||||||
|
if (password != null) {
|
||||||
|
ntripPassword.value = password;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
directory = await getExternalStorageDirectory();
|
||||||
|
print(directory!.path);
|
||||||
|
// String newPath = '';
|
||||||
|
// List<String> folders = directory!.path.split("/");
|
||||||
|
// for (int i = 1; i < folders.length; i++) {
|
||||||
|
// String folder = folders[i];
|
||||||
|
// if (folder != "Android") {
|
||||||
|
// newPath += "/" + folder;
|
||||||
|
// } else {
|
||||||
|
// break;
|
||||||
|
// }
|
||||||
|
// }
|
||||||
|
// newPath = newPath + "/TerepisSegedApp";
|
||||||
|
// directory = Directory(newPath);
|
||||||
|
if (!await directory!.exists()) {
|
||||||
|
await directory!.create(recursive: true);
|
||||||
|
}
|
||||||
|
dataFile = File("${directory!.path}/data.txt");
|
||||||
|
|
||||||
|
if (await directory!.exists()) {
|
||||||
|
if (!await dataFile.exists()) {
|
||||||
|
dataFile.writeAsString(
|
||||||
|
"Id;DateTime;Description;EovX;EovY;Latitude;Longitude;Altitude;Hor.Err;Vert.Err\r\n");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
gpsHeightController.text = '1.8';
|
||||||
|
}
|
||||||
|
|
||||||
|
void _getInitialLocation() async {
|
||||||
|
bool servicedEnabled;
|
||||||
|
PermissionStatus permissionGranted;
|
||||||
|
LocationData locationData;
|
||||||
|
Location location = Location();
|
||||||
|
|
||||||
|
servicedEnabled = await location.serviceEnabled();
|
||||||
|
if (!servicedEnabled) {
|
||||||
|
servicedEnabled = await location.requestService();
|
||||||
|
if (!servicedEnabled) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
permissionGranted = await location.hasPermission();
|
||||||
|
if (permissionGranted == PermissionStatus.denied) {
|
||||||
|
permissionGranted = await location.requestPermission();
|
||||||
|
if (permissionGranted != PermissionStatus.granted) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
locationData = await location.getLocation();
|
||||||
|
currentLatitude.value = locationData.latitude ?? 0.0;
|
||||||
|
currentLongitude.value = locationData.longitude ?? 0.0;
|
||||||
|
print("Current location initialized -> $currentLatitude $currentLongitude");
|
||||||
|
mapController.move(LatLng(currentLatitude.value, currentLongitude.value),
|
||||||
|
currentZoom.value);
|
||||||
|
_updateCurrentLocationMarker();
|
||||||
|
}
|
||||||
|
|
||||||
|
void connectToGps() {
|
||||||
|
if (gpsIsConnected == false) {
|
||||||
|
BluetoothConnection.toAddress(gpsAddress.value).then((value) {
|
||||||
|
connection = value;
|
||||||
|
gpsIsConnected.value = true;
|
||||||
|
print("GPS is connected ...");
|
||||||
|
|
||||||
|
connection.input!.listen(_onDataReceived);
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void disconnectFromGps() {
|
||||||
|
if (gpsIsConnected.value) {
|
||||||
|
connection.close();
|
||||||
|
gpsIsConnected.value = false;
|
||||||
|
print("GPS is disconnected ....");
|
||||||
|
}
|
||||||
|
if (ntripIsConnected.value) {
|
||||||
|
disconnectFromNtripServer();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void connectToNtripServer() async {
|
||||||
|
// socket = await Socket.connect(InternetAddress('3.23.52.207'), 2101,
|
||||||
|
// timeout: const Duration(seconds: 5));
|
||||||
|
socket = await Socket.connect(InternetAddress('84.206.45.44'), 2101,
|
||||||
|
timeout: const Duration(seconds: 5));
|
||||||
|
socket.asBroadcastStream;
|
||||||
|
ntripIsConnected.value = true;
|
||||||
|
|
||||||
|
socket.encoding = ascii;
|
||||||
|
print("Connected to ntrip server ....");
|
||||||
|
|
||||||
|
// String header = "GET /KOVARIK HTTP/1.1\r\n";
|
||||||
|
String header = "GET /SGO_RTK3.2 HTTP/1.1\r\n";
|
||||||
|
|
||||||
|
header += "User-Agent: SharpGps iter.dk\r\n";
|
||||||
|
header += "Accept: */*\r\nConnection: close\r\n";
|
||||||
|
// header += "Authorization: Basic ${_toBase64("info@mail.app-dev.hu:")}\r\n";
|
||||||
|
//header += "Authorization: Basic ${_toBase64("elgi08:Laszl0stef1")}\r\n";
|
||||||
|
header +=
|
||||||
|
"Authorization: Basic ${_toBase64("${ntripUserName.value}:${ntripPassword.value}")}\r\n";
|
||||||
|
// header += "Host:rtk2go.com:2101\r\n";
|
||||||
|
header += "Host:gnssnet.hu:2101\r\n";
|
||||||
|
header += "Ntrip-Vesrsion:Ntrip/2.0\r\n";
|
||||||
|
header += "\r\n";
|
||||||
|
|
||||||
|
// listen for responses from the server
|
||||||
|
socketStreamSubscription = socket.listen(
|
||||||
|
// handle data from the server
|
||||||
|
(Uint8List data) {
|
||||||
|
// "ICY 200 OK" - first response
|
||||||
|
final serverResponse = String.fromCharCodes(data);
|
||||||
|
ntripReceivedData.value = data.length;
|
||||||
|
ntripDataPacketNumbers.value++;
|
||||||
|
if (gpsIsConnected.value) {
|
||||||
|
if (data.length > 14) {
|
||||||
|
connection.output.add(data);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
print('Server: $ntripReceivedData');
|
||||||
|
},
|
||||||
|
|
||||||
|
// handle errors
|
||||||
|
onError: (error) {
|
||||||
|
print(error);
|
||||||
|
socket.destroy();
|
||||||
|
},
|
||||||
|
|
||||||
|
// handle server ending connection
|
||||||
|
onDone: () async {
|
||||||
|
print('Server left.');
|
||||||
|
socketStreamSubscription.cancel();
|
||||||
|
await socket.flush();
|
||||||
|
ntripIsConnected.value = false;
|
||||||
|
ntripReceivedData.value = 0;
|
||||||
|
socket.destroy();
|
||||||
|
},
|
||||||
|
);
|
||||||
|
|
||||||
|
socket.add(_convertStringToUint8List(header));
|
||||||
|
|
||||||
|
// sendGgaMessage(lastGgaMessage);
|
||||||
|
}
|
||||||
|
|
||||||
|
void disconnectFromNtripServer() async {
|
||||||
|
socketStreamSubscription.cancel();
|
||||||
|
await socket.flush();
|
||||||
|
socket.close();
|
||||||
|
socket.destroy();
|
||||||
|
ntripReceivedData.value = 0;
|
||||||
|
ntripIsConnected.value = false;
|
||||||
|
print("Disconnect from ntrip server....");
|
||||||
|
}
|
||||||
|
|
||||||
|
void _onDataReceived(Uint8List data) {
|
||||||
|
String sentence = "";
|
||||||
|
|
||||||
|
print("Bluetooth received -> ${data.length} byte(s)");
|
||||||
|
String dataString = String.fromCharCodes(data);
|
||||||
|
int index = data.indexOf(13);
|
||||||
|
|
||||||
|
if (~index != 0) {
|
||||||
|
sentence = _messageBuffer + dataString.substring(0, index);
|
||||||
|
_messageBuffer = dataString.substring(index);
|
||||||
|
} else {
|
||||||
|
_messageBuffer = _messageBuffer + dataString;
|
||||||
|
}
|
||||||
|
|
||||||
|
// print("Message ($count): $sentence");
|
||||||
|
_processGnssMessage(sentence);
|
||||||
|
}
|
||||||
|
|
||||||
|
String _toBase64(String str) {
|
||||||
|
final bytes = ascii.encode(str);
|
||||||
|
final base64Str = base64.encode(bytes);
|
||||||
|
|
||||||
|
return base64Str;
|
||||||
|
}
|
||||||
|
|
||||||
|
Uint8List _convertStringToUint8List(String str) {
|
||||||
|
final List<int> codeUnits = str.codeUnits;
|
||||||
|
final Uint8List unit8List = Uint8List.fromList(codeUnits);
|
||||||
|
return unit8List;
|
||||||
|
}
|
||||||
|
|
||||||
|
void _processGnssMessage(String message) {
|
||||||
|
LineSplitter lineSplitter = const LineSplitter();
|
||||||
|
List<String> lines = lineSplitter.convert(message);
|
||||||
|
|
||||||
|
if (lines.isEmpty) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
for (String line in lines) {
|
||||||
|
if (line.trim().isEmpty) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
if (line.startsWith("\$GNGGA")) {
|
||||||
|
final sentence = nmeaDecoder.decode(line);
|
||||||
|
if (sentence!.valid && sentence is Gngga) {
|
||||||
|
hasGpsValidData.value = sentence.gpsQualityIndicator > 0;
|
||||||
|
if (hasGpsValidData.value) {
|
||||||
|
if (DateTime.now().difference(lastGpsRefreshTime).inSeconds >= 0) {
|
||||||
|
lastGpsRefreshTime = DateTime.now();
|
||||||
|
utcOfPositionFix = sentence.utcOfPositionFix;
|
||||||
|
gpsLatitude.value = sentence.latitude;
|
||||||
|
gpsLatitudeDirection.value = sentence.latitudeDirection;
|
||||||
|
gpsLongitude.value = sentence.longitude;
|
||||||
|
gpsLongitudeDirection.value = sentence.longitudeDirection;
|
||||||
|
gpsAltitude.value = sentence.altitudeAboveMeanSeaLevel;
|
||||||
|
gpsGeoidSeparation.value = sentence.geoidSeparation;
|
||||||
|
gpsQuality.value = sentence.gpsQualityIndicator;
|
||||||
|
eov.value = ConvertCoordinate.ConvertWgsToEov(
|
||||||
|
gpsLatitude.value, gpsLongitude.value);
|
||||||
|
|
||||||
|
latDegree.value = ConvertCoordinate.toDegree(gpsLatitude.value);
|
||||||
|
latMin.value = ConvertCoordinate.toMinute(gpsLatitude.value);
|
||||||
|
latSec.value = ConvertCoordinate.toSecond(gpsLatitude.value);
|
||||||
|
longDegree.value = ConvertCoordinate.toDegree(gpsLongitude.value);
|
||||||
|
longMin.value = ConvertCoordinate.toMinute(gpsLongitude.value);
|
||||||
|
longSec.value = ConvertCoordinate.toSecond(gpsLongitude.value);
|
||||||
|
currentLatitude.value = gpsLatitude.value;
|
||||||
|
currentLongitude.value = gpsLongitude.value;
|
||||||
|
_updateCurrentLocationMarker();
|
||||||
|
lastGgaMessage = line;
|
||||||
|
if (ntripIsConnected.value &&
|
||||||
|
(DateTime.now()
|
||||||
|
.difference(lastSendTimeGgaMessage)
|
||||||
|
.inSeconds >=
|
||||||
|
5)) {
|
||||||
|
sendGgaMessage(lastGgaMessage);
|
||||||
|
print("Send GGA message: $lastGgaMessage");
|
||||||
|
ggaSenDataPacketNumber.value++;
|
||||||
|
ggaSendLastTimeStr.value = utcOfPositionFix;
|
||||||
|
lastSendTimeGgaMessage = DateTime.now();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (line.startsWith("\$GNGST") && hasGpsValidData.value) {
|
||||||
|
final sentence = nmeaDecoder.decode(line);
|
||||||
|
if (sentence!.valid && sentence is Gngst) {
|
||||||
|
gpsLatitudeError.value = sentence.latitudeError;
|
||||||
|
gpsLongitudeError.value = sentence.longitudeError;
|
||||||
|
gpsAltitudeError.value = sentence.heightError;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (line.startsWith("\$GNRMC") && hasGpsValidData.value) {
|
||||||
|
final sentence = nmeaDecoder.decode(line);
|
||||||
|
if (sentence!.valid && sentence is Gnrmc) {
|
||||||
|
utcDateOfPositionFix = sentence.date;
|
||||||
|
if (utcDateOfPositionFix.isNotEmpty && utcOfPositionFix.isNotEmpty) {
|
||||||
|
gpsDateTime.value = DateTime(
|
||||||
|
2000 +
|
||||||
|
int.parse(
|
||||||
|
"${utcDateOfPositionFix[4]}${utcDateOfPositionFix[5]}"),
|
||||||
|
int.parse(
|
||||||
|
"${utcDateOfPositionFix[2]}${utcDateOfPositionFix[3]}"),
|
||||||
|
int.parse(
|
||||||
|
"${utcDateOfPositionFix[0]}${utcDateOfPositionFix[1]}"),
|
||||||
|
int.parse("${utcOfPositionFix[0]}${utcOfPositionFix[1]}"),
|
||||||
|
int.parse("${utcOfPositionFix[2]}${utcOfPositionFix[3]}"),
|
||||||
|
int.parse("${utcOfPositionFix[4]}${utcOfPositionFix[5]}"));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
String getGpsQualityIndicator({required int quality}) {
|
||||||
|
String qualityStr = "Invalid";
|
||||||
|
switch (quality) {
|
||||||
|
case 0:
|
||||||
|
{
|
||||||
|
qualityStr = "Invalid";
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
case 1:
|
||||||
|
{
|
||||||
|
qualityStr = "Standard GPS (2D/3D)";
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
case 2:
|
||||||
|
{
|
||||||
|
qualityStr = "Differential GPS";
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
case 4:
|
||||||
|
{
|
||||||
|
qualityStr = "RTK Fix";
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
case 5:
|
||||||
|
{
|
||||||
|
qualityStr = "RTK Float";
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
case 6:
|
||||||
|
{
|
||||||
|
qualityStr = "Estimated (DR) Fix";
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
{
|
||||||
|
qualityStr = "Invalid (-1)";
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return qualityStr;
|
||||||
|
}
|
||||||
|
|
||||||
|
void mapZoomOut() {
|
||||||
|
double cLat, cLong;
|
||||||
|
if (currentZoom.value > 0) {
|
||||||
|
currentZoom.value = currentZoom.value - 1;
|
||||||
|
cLat = isMapMoveToCenter.value
|
||||||
|
? currentLatitude.value
|
||||||
|
: mapController.camera.center.latitude;
|
||||||
|
cLong = isMapMoveToCenter.value
|
||||||
|
? currentLongitude.value
|
||||||
|
: mapController.camera.center.longitude;
|
||||||
|
mapController.move(LatLng(cLat, cLong), currentZoom.value);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void mapZoomIn() {
|
||||||
|
double cLat, cLong;
|
||||||
|
if (currentZoom.value < maxZoomValue) {
|
||||||
|
currentZoom.value++;
|
||||||
|
cLat = isMapMoveToCenter.value
|
||||||
|
? currentLatitude.value
|
||||||
|
: mapController.camera.center.latitude;
|
||||||
|
cLong = isMapMoveToCenter.value
|
||||||
|
? currentLongitude.value
|
||||||
|
: mapController.camera.center.longitude;
|
||||||
|
mapController.move(LatLng(cLat, cLong), currentZoom.value);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void _updateCurrentLocationMarker() {
|
||||||
|
currentLocationMarker.clear();
|
||||||
|
currentLocationMarker.add(Marker(
|
||||||
|
point: LatLng(currentLatitude.value, currentLongitude.value),
|
||||||
|
width: 15.0,
|
||||||
|
height: 15.0,
|
||||||
|
child: Container(
|
||||||
|
width: 15.0,
|
||||||
|
height: 15.0,
|
||||||
|
decoration: BoxDecoration(
|
||||||
|
color: getCurrentLocationMarkerColor(quality: gpsQuality.value),
|
||||||
|
shape: BoxShape.circle,
|
||||||
|
border: Border.all(width: 1.5, color: Colors.white)),
|
||||||
|
)));
|
||||||
|
if (isMapMoveToCenter.value) {
|
||||||
|
mapController.move(LatLng(currentLatitude.value, currentLongitude.value),
|
||||||
|
currentZoom.value);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
Color getCurrentLocationMarkerColor({required int quality}) {
|
||||||
|
Color qualityStr = Colors.black;
|
||||||
|
switch (quality) {
|
||||||
|
case 0:
|
||||||
|
{
|
||||||
|
qualityStr = Colors.black;
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
case 1:
|
||||||
|
{
|
||||||
|
qualityStr = Colors.red;
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
case 2:
|
||||||
|
{
|
||||||
|
qualityStr = Colors.blue;
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
case 4:
|
||||||
|
{
|
||||||
|
qualityStr = Colors.green;
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
case 5:
|
||||||
|
{
|
||||||
|
qualityStr = Colors.orange;
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
case 6:
|
||||||
|
{
|
||||||
|
qualityStr = Colors.yellow;
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
{
|
||||||
|
qualityStr = Colors.white;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return qualityStr;
|
||||||
|
}
|
||||||
|
|
||||||
|
void onBottomNavigationBarTap(int index) async {
|
||||||
|
bool noMorePoint = false;
|
||||||
|
print("OnBottomNavTap -> $index");
|
||||||
|
if (index == 0) {
|
||||||
|
pointIdController.text = pointId.toString();
|
||||||
|
pointDescriptionController.text = "";
|
||||||
|
Get.dialog(
|
||||||
|
AlertDialog(
|
||||||
|
title: const Text("Pont rögzítése"),
|
||||||
|
content: Column(
|
||||||
|
mainAxisSize: MainAxisSize.min,
|
||||||
|
children: [
|
||||||
|
TextField(
|
||||||
|
controller: pointIdController,
|
||||||
|
autofocus: true,
|
||||||
|
keyboardType: TextInputType.number,
|
||||||
|
decoration: const InputDecoration(
|
||||||
|
border: OutlineInputBorder(), labelText: 'Azonosító'),
|
||||||
|
),
|
||||||
|
const SizedBox(
|
||||||
|
height: 20.0,
|
||||||
|
),
|
||||||
|
TextField(
|
||||||
|
controller: pointDescriptionController,
|
||||||
|
decoration: const InputDecoration(
|
||||||
|
border: OutlineInputBorder(), labelText: 'Leírás'),
|
||||||
|
),
|
||||||
|
const SizedBox(
|
||||||
|
height: 20,
|
||||||
|
),
|
||||||
|
TextField(
|
||||||
|
controller: gpsHeightController,
|
||||||
|
autofocus: true,
|
||||||
|
keyboardType: TextInputType.number,
|
||||||
|
decoration: const InputDecoration(
|
||||||
|
border: OutlineInputBorder(), labelText: 'GPS magasság'),
|
||||||
|
),
|
||||||
|
],
|
||||||
|
),
|
||||||
|
actions: <Widget>[
|
||||||
|
Row(
|
||||||
|
mainAxisAlignment: MainAxisAlignment.spaceEvenly,
|
||||||
|
children: [
|
||||||
|
OutlinedButton(
|
||||||
|
style: OutlinedButton.styleFrom(
|
||||||
|
minimumSize: const Size(120.0, 40.0)),
|
||||||
|
child: const Text(
|
||||||
|
"Mégsem",
|
||||||
|
style: TextStyle(color: Colors.red),
|
||||||
|
),
|
||||||
|
onPressed: () {
|
||||||
|
Get.back();
|
||||||
|
},
|
||||||
|
),
|
||||||
|
OutlinedButton(
|
||||||
|
style: OutlinedButton.styleFrom(
|
||||||
|
minimumSize: const Size(120.0, 40.0)),
|
||||||
|
child: const Text(
|
||||||
|
"Ment",
|
||||||
|
style: TextStyle(
|
||||||
|
color: Colors.green, fontWeight: FontWeight.bold),
|
||||||
|
),
|
||||||
|
onPressed: () async {
|
||||||
|
pointId = int.parse(pointIdController.text);
|
||||||
|
pointWithDescriptionList.add(PointWithDescription(
|
||||||
|
pointId,
|
||||||
|
gpsDateTime.value,
|
||||||
|
pointDescriptionController.text,
|
||||||
|
eov.value.Y,
|
||||||
|
eov.value.X,
|
||||||
|
gpsLatitude.value,
|
||||||
|
gpsLongitude.value,
|
||||||
|
max(gpsLatitudeError.value, gpsLongitudeError.value),
|
||||||
|
gpsAltitudeError.value));
|
||||||
|
print(
|
||||||
|
"pointWithDescriptionList -> ${pointWithDescriptionList.length}");
|
||||||
|
|
||||||
|
pointNotesMarker.add(Marker(
|
||||||
|
point: LatLng(gpsLatitude.value, gpsLongitude.value),
|
||||||
|
width: 15.0,
|
||||||
|
height: 15.0,
|
||||||
|
child: Container(
|
||||||
|
width: 15.0,
|
||||||
|
height: 15.0,
|
||||||
|
decoration: BoxDecoration(
|
||||||
|
color: Colors.amber[700],
|
||||||
|
shape: BoxShape.circle,
|
||||||
|
border:
|
||||||
|
Border.all(width: 1.0, color: Colors.black)),
|
||||||
|
)));
|
||||||
|
await dataFile.writeAsString(
|
||||||
|
"$pointId;$gpsDateTime;${pointDescriptionController.text};${formatEovForFile.format(eov.value.Y)};${formatEovForFile.format(eov.value.X)};$gpsLatitude;$gpsLongitude;$gpsAltitude;${max(gpsLatitudeError.value, gpsLongitudeError.value)};$gpsAltitudeError;${gpsHeightController.text}\r\n",
|
||||||
|
mode: FileMode.append);
|
||||||
|
|
||||||
|
_measuredPoints.add({
|
||||||
|
"id": pointId,
|
||||||
|
"dateAndTime": gpsDateTime.value,
|
||||||
|
"latitude": gpsLatitude.value,
|
||||||
|
"longitude": gpsLongitude.value,
|
||||||
|
"altitude": gpsAltitude.value,
|
||||||
|
"eovY": formatEovForFile.format(eov.value.Y),
|
||||||
|
"eovX": formatEovForFile.format(eov.value.X),
|
||||||
|
"description": pointDescriptionController.text,
|
||||||
|
"horizontalError":
|
||||||
|
max(gpsLatitudeError.value, gpsLongitudeError.value),
|
||||||
|
"verticalError": gpsAltitudeError.value,
|
||||||
|
"gpsHeight": gpsHeightController.text
|
||||||
|
});
|
||||||
|
|
||||||
|
await Supabase.instance.client
|
||||||
|
.from('TerepiSeged_MeasuredPoints')
|
||||||
|
.insert({
|
||||||
|
'pointNumber': pointId,
|
||||||
|
'gnssNumber': gpsName.value,
|
||||||
|
'latitude': gpsLatitude.value,
|
||||||
|
'longitude': gpsLongitude.value,
|
||||||
|
'altitude': gpsAltitude.value,
|
||||||
|
'heightOfGeoid': gpsGeoidSeparation.value,
|
||||||
|
'eovX': eov.value.X,
|
||||||
|
'eovY': eov.value.Y,
|
||||||
|
'poleHeight': double.tryParse(gpsHeightController.text),
|
||||||
|
'horizontalError':
|
||||||
|
max(gpsLatitudeError.value, gpsLongitudeError.value),
|
||||||
|
'verticalError': gpsAltitudeError.value,
|
||||||
|
'description': pointDescriptionController.text,
|
||||||
|
'isDeleted': false,
|
||||||
|
'projectId': 2
|
||||||
|
});
|
||||||
|
|
||||||
|
await Supabase.instance.client
|
||||||
|
.from('TerepiSeged_Receiver')
|
||||||
|
.update({'isMeasured': true}).eq(
|
||||||
|
'pointNumber', pointId);
|
||||||
|
|
||||||
|
pointId++;
|
||||||
|
|
||||||
|
Get.back();
|
||||||
|
},
|
||||||
|
),
|
||||||
|
],
|
||||||
|
)
|
||||||
|
],
|
||||||
|
),
|
||||||
|
barrierDismissible: false,
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void sendGgaMessage(String ggaMessage) {
|
||||||
|
if (ntripIsConnected.value && ggaMessage.isNotEmpty) {
|
||||||
|
socket.add(_convertStringToUint8List("$ggaMessage\r\n"));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void setIsMapMoveToCenter() {
|
||||||
|
isMapMoveToCenter.value = !isMapMoveToCenter.value;
|
||||||
|
// onBottomNavigationBarTap(0);
|
||||||
|
}
|
||||||
|
|
||||||
|
void addMeasuredPoint() {
|
||||||
|
_measuredPoints.add({"id": 4001, "latitude": 46.3455, "longitude": 19.652});
|
||||||
|
}
|
||||||
|
|
||||||
|
void showAddPointDialog() {
|
||||||
|
onBottomNavigationBarTap(0);
|
||||||
|
}
|
||||||
|
|
||||||
|
void saveGpsAddress(String address) {
|
||||||
|
prefs.setString('gpsAddress', address);
|
||||||
|
}
|
||||||
|
|
||||||
|
void saveGpsName(String name) {
|
||||||
|
prefs.setString('gpsName', name);
|
||||||
|
}
|
||||||
|
|
||||||
|
void saveNtripUserName(String username) {
|
||||||
|
prefs.setString('ntripUserName', username);
|
||||||
|
}
|
||||||
|
|
||||||
|
void saveNtripPassword(String password) {
|
||||||
|
prefs.setString('ntripPassword', password);
|
||||||
|
}
|
||||||
|
|
||||||
|
void updatePointStatus(int pointId) {}
|
||||||
|
|
||||||
|
void toggleShowPassword() {
|
||||||
|
isShowPassword.value = !isShowPassword.value;
|
||||||
|
if (passwordFieldFocusNode.hasPrimaryFocus) return;
|
||||||
|
passwordFieldFocusNode.canRequestFocus = false;
|
||||||
|
}
|
||||||
|
|
||||||
|
void showMeasuredPointsTableDialog() {
|
||||||
|
Get.to(() => MeasuredPointsTableDialog(), transition: Transition.fadeIn);
|
||||||
|
}
|
||||||
|
|
||||||
|
Future<List> readMeasuredPoints() async {
|
||||||
|
var response = await Supabase.instance.client
|
||||||
|
.from('TerepiSeged_MeasuredPoints')
|
||||||
|
.select()
|
||||||
|
.eq('projectId', 2)
|
||||||
|
.order('created_at');
|
||||||
|
|
||||||
|
print(response);
|
||||||
|
|
||||||
|
return response;
|
||||||
|
}
|
||||||
|
}
|
||||||
@ -0,0 +1,97 @@
|
|||||||
|
import 'package:flutter/material.dart';
|
||||||
|
import 'package:get/get.dart';
|
||||||
|
import 'package:terepi_seged/pages/map_survey/presentations/controllers/map_survey_controller.dart';
|
||||||
|
|
||||||
|
class MapAddPointDialog extends GetView<MapSurveyController> {
|
||||||
|
const MapAddPointDialog({Key? key}) : super(key: key);
|
||||||
|
|
||||||
|
@override
|
||||||
|
Widget build(BuildContext context) {
|
||||||
|
return Scaffold(
|
||||||
|
body: SafeArea(
|
||||||
|
child: Padding(
|
||||||
|
padding: const EdgeInsets.symmetric(horizontal: 24, vertical: 16),
|
||||||
|
child: Column(
|
||||||
|
mainAxisSize: MainAxisSize.min,
|
||||||
|
children: [
|
||||||
|
TextField(
|
||||||
|
controller: controller.pointIdController,
|
||||||
|
autofocus: true,
|
||||||
|
keyboardType: TextInputType.number,
|
||||||
|
decoration: const InputDecoration(
|
||||||
|
border: OutlineInputBorder(), labelText: 'Azonosító'),
|
||||||
|
),
|
||||||
|
const SizedBox(
|
||||||
|
height: 20.0,
|
||||||
|
),
|
||||||
|
TextField(
|
||||||
|
controller: controller.pointDescriptionController,
|
||||||
|
decoration: const InputDecoration(
|
||||||
|
border: OutlineInputBorder(), labelText: 'Leírás'),
|
||||||
|
),
|
||||||
|
Row(
|
||||||
|
mainAxisAlignment: MainAxisAlignment.spaceEvenly,
|
||||||
|
children: [
|
||||||
|
OutlinedButton(
|
||||||
|
style: OutlinedButton.styleFrom(
|
||||||
|
minimumSize: const Size(120.0, 40.0)),
|
||||||
|
child: const Text(
|
||||||
|
"Mégsem",
|
||||||
|
style: TextStyle(color: Colors.red),
|
||||||
|
),
|
||||||
|
onPressed: () {
|
||||||
|
Get.back();
|
||||||
|
},
|
||||||
|
),
|
||||||
|
OutlinedButton(
|
||||||
|
style: OutlinedButton.styleFrom(
|
||||||
|
minimumSize: const Size(120.0, 40.0)),
|
||||||
|
child: const Text(
|
||||||
|
"Ment",
|
||||||
|
style: TextStyle(
|
||||||
|
color: Colors.green, fontWeight: FontWeight.bold),
|
||||||
|
),
|
||||||
|
onPressed: () async {
|
||||||
|
// pointId = int.parse(pointIdController.text);
|
||||||
|
// pointWithDescriptionList.add(PointWithDescription(
|
||||||
|
// pointId,
|
||||||
|
// gpsDateTime.value,
|
||||||
|
// pointDescriptionController.text,
|
||||||
|
// eov.value.Y,
|
||||||
|
// eov.value.X,
|
||||||
|
// gpsLatitude.value,
|
||||||
|
// gpsLongitude.value,
|
||||||
|
// max(gpsLatitudeError.value, gpsLongitudeError.value),
|
||||||
|
// gpsAltitudeError.value));
|
||||||
|
// print(
|
||||||
|
// "pointWithDescriptionList -> ${pointWithDescriptionList.length}");
|
||||||
|
// pointNotesMarker.add(Marker(
|
||||||
|
// point: LatLng(gpsLatitude.value, gpsLongitude.value),
|
||||||
|
// width: 15.0,
|
||||||
|
// height: 15.0,
|
||||||
|
// builder: (ctx) => Container(
|
||||||
|
// width: 15.0,
|
||||||
|
// height: 15.0,
|
||||||
|
// decoration: BoxDecoration(
|
||||||
|
// color: Colors.purple,
|
||||||
|
// shape: BoxShape.circle,
|
||||||
|
// border: Border.all(
|
||||||
|
// width: 1.0, color: Colors.black)),
|
||||||
|
// )));
|
||||||
|
// await dataFile.writeAsString(
|
||||||
|
// "$pointId;$gpsDateTime;${pointDescriptionController.text};${formatEovForFile.format(eov.value.Y)};${formatEovForFile.format(eov.value.X)};$gpsLatitude;$gpsLongitude;$gpsAltitude;${max(gpsLatitudeError.value, gpsLongitudeError.value)};$gpsAltitudeError\r\n",
|
||||||
|
// mode: FileMode.append);
|
||||||
|
|
||||||
|
// pointId++;
|
||||||
|
|
||||||
|
Get.back();
|
||||||
|
},
|
||||||
|
),
|
||||||
|
],
|
||||||
|
)
|
||||||
|
],
|
||||||
|
),
|
||||||
|
)),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
423
lib/pages/map_survey/presentations/views/mapsurvey_view.dart
Normal file
423
lib/pages/map_survey/presentations/views/mapsurvey_view.dart
Normal file
@ -0,0 +1,423 @@
|
|||||||
|
import 'dart:io';
|
||||||
|
import 'dart:math';
|
||||||
|
|
||||||
|
import 'package:file_picker/file_picker.dart';
|
||||||
|
import 'package:flutter/material.dart';
|
||||||
|
import 'package:flutter_map_polywidget/flutter_map_polywidget.dart';
|
||||||
|
import 'package:get/get.dart';
|
||||||
|
import 'package:flutter_map/flutter_map.dart';
|
||||||
|
import 'package:latlong2/latlong.dart';
|
||||||
|
import 'package:rive/rive.dart';
|
||||||
|
import 'package:terepi_seged/pages/map_survey/presentations/controllers/map_survey_controller.dart';
|
||||||
|
import 'package:terepi_seged/pages/map_survey/presentations/views/settings_dialog.dart';
|
||||||
|
import 'package:terepi_seged/utils/rive_utils.dart';
|
||||||
|
|
||||||
|
import 'map_add_point_dialog.dart';
|
||||||
|
|
||||||
|
class MapSurveyView extends GetView<MapSurveyController> {
|
||||||
|
const MapSurveyView({Key? key}) : super(key: key);
|
||||||
|
|
||||||
|
@override
|
||||||
|
Widget build(BuildContext context) {
|
||||||
|
late SMIBool gpsTrigger;
|
||||||
|
|
||||||
|
return Scaffold(
|
||||||
|
resizeToAvoidBottomInset: false,
|
||||||
|
extendBody: true,
|
||||||
|
appBar: AppBar(
|
||||||
|
title: const Column(
|
||||||
|
crossAxisAlignment: CrossAxisAlignment.start,
|
||||||
|
children: [
|
||||||
|
Text('Térkép'),
|
||||||
|
Text(
|
||||||
|
"",
|
||||||
|
style: TextStyle(fontSize: 12.0),
|
||||||
|
)
|
||||||
|
],
|
||||||
|
),
|
||||||
|
actions: [
|
||||||
|
Padding(
|
||||||
|
padding: const EdgeInsets.only(right: 20.0),
|
||||||
|
child: Obx(
|
||||||
|
() => GestureDetector(
|
||||||
|
onTap: () {
|
||||||
|
if (controller.gpsIsConnected.value) {
|
||||||
|
controller.disconnectFromGps();
|
||||||
|
} else {
|
||||||
|
controller.connectToGps();
|
||||||
|
}
|
||||||
|
},
|
||||||
|
child: Icon(Icons.gps_fixed,
|
||||||
|
size: 26.0,
|
||||||
|
color: controller.gpsIsConnected.value
|
||||||
|
? Colors.green
|
||||||
|
: Colors.red),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
Padding(
|
||||||
|
padding: const EdgeInsets.only(right: 20.0),
|
||||||
|
child: Obx(
|
||||||
|
() => GestureDetector(
|
||||||
|
onTap: () {
|
||||||
|
if (controller.ntripIsConnected.value) {
|
||||||
|
controller.disconnectFromNtripServer();
|
||||||
|
} else {
|
||||||
|
controller.connectToNtripServer();
|
||||||
|
}
|
||||||
|
},
|
||||||
|
child: Icon(
|
||||||
|
Icons.assistant,
|
||||||
|
size: 26.0,
|
||||||
|
color: controller.ntripIsConnected.value
|
||||||
|
? Colors.green
|
||||||
|
: Colors.red,
|
||||||
|
),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
Padding(
|
||||||
|
padding: const EdgeInsets.only(right: 20.0),
|
||||||
|
child: GestureDetector(
|
||||||
|
onTap: () {
|
||||||
|
if (controller.ntripUserName.value.isNotEmpty) {
|
||||||
|
controller.ntripUsernameController.text =
|
||||||
|
controller.ntripUserName.value;
|
||||||
|
}
|
||||||
|
if (controller.ntripPassword.value.isNotEmpty) {
|
||||||
|
controller.ntripPasswordController.text =
|
||||||
|
controller.ntripPassword.value;
|
||||||
|
}
|
||||||
|
Get.to(() => SettingsDialog(), transition: Transition.downToUp);
|
||||||
|
},
|
||||||
|
child: const Icon(
|
||||||
|
Icons.more_vert,
|
||||||
|
size: 26.0,
|
||||||
|
),
|
||||||
|
),
|
||||||
|
)
|
||||||
|
],
|
||||||
|
),
|
||||||
|
body: Column(
|
||||||
|
children: [
|
||||||
|
Obx(
|
||||||
|
() => controller.gpsIsConnected.value
|
||||||
|
? Padding(
|
||||||
|
padding: const EdgeInsets.all(4.0),
|
||||||
|
child: Row(
|
||||||
|
mainAxisAlignment: MainAxisAlignment.start,
|
||||||
|
crossAxisAlignment: CrossAxisAlignment.start,
|
||||||
|
children: [
|
||||||
|
Container(
|
||||||
|
width: 170.0,
|
||||||
|
// height: 160.0,
|
||||||
|
decoration: BoxDecoration(
|
||||||
|
border: Border.all(
|
||||||
|
width: 2.0,
|
||||||
|
color: const Color.fromARGB(
|
||||||
|
130, 255, 255, 255)),
|
||||||
|
color:
|
||||||
|
const Color.fromARGB(130, 255, 255, 255)),
|
||||||
|
child: Column(
|
||||||
|
mainAxisAlignment: MainAxisAlignment.start,
|
||||||
|
crossAxisAlignment: CrossAxisAlignment.start,
|
||||||
|
children: [
|
||||||
|
Text(
|
||||||
|
"${controller.gpsDateTime} (UTC)",
|
||||||
|
style: const TextStyle(fontSize: 10.0),
|
||||||
|
),
|
||||||
|
const Divider(
|
||||||
|
height: 5.0,
|
||||||
|
color: Colors.black,
|
||||||
|
),
|
||||||
|
Text(
|
||||||
|
"EovX: ${controller.formatEov.format(controller.eov.value.Y)}"),
|
||||||
|
Text(
|
||||||
|
"EovY: ${controller.formatEov.format(controller.eov.value.X)}"),
|
||||||
|
Text("Alt: ${controller.gpsAltitude} (m)"),
|
||||||
|
const Divider(
|
||||||
|
height: 5.0,
|
||||||
|
color: Colors.black,
|
||||||
|
),
|
||||||
|
Text(
|
||||||
|
"Hor. error: ${max(controller.gpsLatitudeError.value, controller.gpsLongitudeError.value)} (m)",
|
||||||
|
style: const TextStyle(fontSize: 14.0),
|
||||||
|
),
|
||||||
|
Text(
|
||||||
|
"Vert. error: ${controller.gpsAltitudeError} (m)",
|
||||||
|
style: const TextStyle(fontSize: 14.0),
|
||||||
|
),
|
||||||
|
const Divider(
|
||||||
|
height: 5.0,
|
||||||
|
color: Colors.black,
|
||||||
|
),
|
||||||
|
Text(
|
||||||
|
"GPS quality -> ${controller.getGpsQualityIndicator(quality: controller.gpsQuality.value)} ",
|
||||||
|
style: const TextStyle(fontSize: 8.0),
|
||||||
|
),
|
||||||
|
controller.ntripIsConnected.value
|
||||||
|
? Column(
|
||||||
|
crossAxisAlignment:
|
||||||
|
CrossAxisAlignment.start,
|
||||||
|
children: [
|
||||||
|
const Divider(
|
||||||
|
height: 5.0,
|
||||||
|
color: Colors.black,
|
||||||
|
),
|
||||||
|
Text(
|
||||||
|
"Ntrip ${controller.ntripReceivedData} byte(s) (${controller.ntripDataPacketNumbers})",
|
||||||
|
style:
|
||||||
|
const TextStyle(fontSize: 10.0),
|
||||||
|
),
|
||||||
|
Text(
|
||||||
|
"GGA last send: ${controller.ggaSendLastTimeStr} (${controller.ggaSenDataPacketNumber})",
|
||||||
|
style:
|
||||||
|
const TextStyle(fontSize: 10.0),
|
||||||
|
),
|
||||||
|
],
|
||||||
|
)
|
||||||
|
: const SizedBox(
|
||||||
|
width: 0.0,
|
||||||
|
height: 0.0,
|
||||||
|
)
|
||||||
|
],
|
||||||
|
)),
|
||||||
|
],
|
||||||
|
),
|
||||||
|
)
|
||||||
|
: const SizedBox(),
|
||||||
|
),
|
||||||
|
Expanded(
|
||||||
|
child: Stack(
|
||||||
|
children: [
|
||||||
|
Obx(
|
||||||
|
() => controller.mapIsInitialized.value
|
||||||
|
? FlutterMap(
|
||||||
|
mapController: controller.mapController,
|
||||||
|
options: MapOptions(
|
||||||
|
initialCenter: LatLng(
|
||||||
|
controller.currentLatitude.value,
|
||||||
|
controller.currentLongitude.value),
|
||||||
|
maxZoom: 25,
|
||||||
|
initialZoom: controller.currentZoom.value,
|
||||||
|
),
|
||||||
|
children: [
|
||||||
|
TileLayer(
|
||||||
|
urlTemplate:
|
||||||
|
'http://{s}.google.com/vt/lyrs=s,h&x={x}&y={y}&z={z}',
|
||||||
|
subdomains: const ['mt0', 'mt1', 'mt2', 'mt3'],
|
||||||
|
maxNativeZoom: 18,
|
||||||
|
// urlTemplate: 'https://tile.openstreetmap.org/{z}/{x}/{y}.png',
|
||||||
|
// userAgentPackageName: 'hu.app_dev.terepi_seged',
|
||||||
|
),
|
||||||
|
MarkerLayer(
|
||||||
|
markers: controller.pointNotesMarker,
|
||||||
|
),
|
||||||
|
MarkerLayer(
|
||||||
|
markers: controller.currentLocationMarker),
|
||||||
|
// MarkerLayer(markers: controller.parser.markers),
|
||||||
|
MarkerLayer(
|
||||||
|
markers: controller.pointsToMeasureMarker),
|
||||||
|
// PolylineLayer(
|
||||||
|
// polylines: controller.parser.polylines),
|
||||||
|
PolyWidgetLayer(
|
||||||
|
polyWidgets: controller.pointsToMeasureLabel)
|
||||||
|
],
|
||||||
|
)
|
||||||
|
: const Center(child: CircularProgressIndicator()),
|
||||||
|
),
|
||||||
|
// Obx(
|
||||||
|
// () => controller.gpsIsConnected.value
|
||||||
|
// ? Positioned(
|
||||||
|
// left: 4.0,
|
||||||
|
// top: 4.0,
|
||||||
|
// child: Container(
|
||||||
|
// width: 170.0,
|
||||||
|
// // height: 160.0,
|
||||||
|
// decoration: BoxDecoration(
|
||||||
|
// border: Border.all(
|
||||||
|
// width: 2.0,
|
||||||
|
// color: const Color.fromARGB(
|
||||||
|
// 130, 255, 255, 255)),
|
||||||
|
// color:
|
||||||
|
// const Color.fromARGB(130, 255, 255, 255)),
|
||||||
|
// child: Column(
|
||||||
|
// mainAxisAlignment: MainAxisAlignment.start,
|
||||||
|
// crossAxisAlignment: CrossAxisAlignment.start,
|
||||||
|
// children: [
|
||||||
|
// Text(
|
||||||
|
// "${controller.gpsDateTime} (UTC)",
|
||||||
|
// style: const TextStyle(fontSize: 10.0),
|
||||||
|
// ),
|
||||||
|
// const Divider(
|
||||||
|
// height: 5.0,
|
||||||
|
// color: Colors.black,
|
||||||
|
// ),
|
||||||
|
// Text(
|
||||||
|
// "EovX: ${controller.formatEov.format(controller.eov.value.Y)}"),
|
||||||
|
// Text(
|
||||||
|
// "EovY: ${controller.formatEov.format(controller.eov.value.X)}"),
|
||||||
|
// Text("Alt: ${controller.gpsAltitude} (m)"),
|
||||||
|
// const Divider(
|
||||||
|
// height: 5.0,
|
||||||
|
// color: Colors.black,
|
||||||
|
// ),
|
||||||
|
// Text(
|
||||||
|
// "Hor. error: ${max(controller.gpsLatitudeError.value, controller.gpsLongitudeError.value)} (m)",
|
||||||
|
// style: const TextStyle(fontSize: 14.0),
|
||||||
|
// ),
|
||||||
|
// Text(
|
||||||
|
// "Vert. error: ${controller.gpsAltitudeError} (m)",
|
||||||
|
// style: const TextStyle(fontSize: 14.0),
|
||||||
|
// ),
|
||||||
|
// const Divider(
|
||||||
|
// height: 5.0,
|
||||||
|
// color: Colors.black,
|
||||||
|
// ),
|
||||||
|
// Text(
|
||||||
|
// "GPS quality -> ${controller.getGpsQualityIndicator(quality: controller.gpsQuality.value)} ",
|
||||||
|
// style: const TextStyle(fontSize: 8.0),
|
||||||
|
// ),
|
||||||
|
// controller.ntripIsConnected.value
|
||||||
|
// ? Column(
|
||||||
|
// crossAxisAlignment:
|
||||||
|
// CrossAxisAlignment.start,
|
||||||
|
// children: [
|
||||||
|
// const Divider(
|
||||||
|
// height: 5.0,
|
||||||
|
// color: Colors.black,
|
||||||
|
// ),
|
||||||
|
// Text(
|
||||||
|
// "Ntrip ${controller.ntripReceivedData} byte(s) (${controller.ntripDataPacketNumbers})",
|
||||||
|
// style: const TextStyle(
|
||||||
|
// fontSize: 10.0),
|
||||||
|
// ),
|
||||||
|
// Text(
|
||||||
|
// "GGA last send: ${controller.ggaSendLastTimeStr} (${controller.ggaSenDataPacketNumber})",
|
||||||
|
// style: const TextStyle(
|
||||||
|
// fontSize: 10.0),
|
||||||
|
// ),
|
||||||
|
// ],
|
||||||
|
// )
|
||||||
|
// : const SizedBox(
|
||||||
|
// width: 0.0,
|
||||||
|
// height: 0.0,
|
||||||
|
// )
|
||||||
|
// ],
|
||||||
|
// )),
|
||||||
|
// )
|
||||||
|
// : const Positioned(
|
||||||
|
// child: SizedBox(
|
||||||
|
// width: 0.0,
|
||||||
|
// height: 0.0,
|
||||||
|
// )),
|
||||||
|
// )
|
||||||
|
],
|
||||||
|
),
|
||||||
|
),
|
||||||
|
],
|
||||||
|
),
|
||||||
|
// bottomNavigationBar: Container(
|
||||||
|
// padding: const EdgeInsets.all(12),
|
||||||
|
// margin: const EdgeInsets.symmetric(horizontal: 24),
|
||||||
|
// decoration: BoxDecoration(
|
||||||
|
// color: Colors.black.withOpacity(0.5),
|
||||||
|
// borderRadius: const BorderRadius.all(Radius.circular(24))),
|
||||||
|
// child: Row(
|
||||||
|
// children: [
|
||||||
|
// GestureDetector(
|
||||||
|
// onTap: () {
|
||||||
|
// gpsTrigger.change(true);
|
||||||
|
// Future.delayed(const Duration(seconds: 1), () {
|
||||||
|
// gpsTrigger.change(false);
|
||||||
|
// });
|
||||||
|
// // controller.onBottomNavigationBarTap(0);
|
||||||
|
// Get.to(() => const MapAddPointDialog(),
|
||||||
|
// fullscreenDialog: true,
|
||||||
|
// transition: Transition.downToUp,
|
||||||
|
// duration: const Duration(milliseconds: 600));
|
||||||
|
// },
|
||||||
|
// child: SizedBox(
|
||||||
|
// height: 36,
|
||||||
|
// width: 36,
|
||||||
|
// child: RiveAnimation.asset(
|
||||||
|
// "assets/RiveAssets/travel_icons_pack.riv",
|
||||||
|
// artboard: "GPS", onInit: (artboard) {
|
||||||
|
// StateMachineController controllerRive =
|
||||||
|
// RiveUtils.getRiveController(artboard,
|
||||||
|
// stateMachineName: "gps_interactivity");
|
||||||
|
// gpsTrigger = controllerRive.findSMI("active") as SMIBool;
|
||||||
|
// }),
|
||||||
|
// ),
|
||||||
|
// )
|
||||||
|
// ],
|
||||||
|
// ),
|
||||||
|
// ),
|
||||||
|
floatingActionButtonLocation: FloatingActionButtonLocation.startFloat,
|
||||||
|
floatingActionButton: Wrap(
|
||||||
|
direction: Axis.horizontal,
|
||||||
|
children: [
|
||||||
|
FloatingActionButton(
|
||||||
|
onPressed: () {
|
||||||
|
controller.mapZoomOut();
|
||||||
|
},
|
||||||
|
heroTag: 'ZoomOut',
|
||||||
|
tooltip: 'Zoom out',
|
||||||
|
child: const Icon(Icons.remove_circle_outline_rounded),
|
||||||
|
),
|
||||||
|
const SizedBox(
|
||||||
|
width: 15,
|
||||||
|
),
|
||||||
|
FloatingActionButton(
|
||||||
|
onPressed: () {
|
||||||
|
controller.mapZoomIn();
|
||||||
|
},
|
||||||
|
heroTag: 'ZoomIn',
|
||||||
|
tooltip: 'Zoom in',
|
||||||
|
child: const Icon(Icons.add_circle_outline_rounded),
|
||||||
|
),
|
||||||
|
const SizedBox(width: 15.0),
|
||||||
|
FloatingActionButton(
|
||||||
|
onPressed: () {
|
||||||
|
// controller.isMapMoveToCenter();
|
||||||
|
controller.setIsMapMoveToCenter();
|
||||||
|
},
|
||||||
|
heroTag: 'Center map',
|
||||||
|
tooltip: 'Center map',
|
||||||
|
child: Obx(
|
||||||
|
() => Icon(Icons.center_focus_strong,
|
||||||
|
color: controller.isMapMoveToCenter.value
|
||||||
|
? Colors.green
|
||||||
|
: Colors.red),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
const SizedBox(
|
||||||
|
width: 15.0,
|
||||||
|
),
|
||||||
|
FloatingActionButton(
|
||||||
|
onPressed: () {
|
||||||
|
// controller.addMeasuredPoint();
|
||||||
|
// controller.onBottomNavigationBarTap(0);
|
||||||
|
controller.showAddPointDialog();
|
||||||
|
},
|
||||||
|
heroTag: 'Pont bemérése',
|
||||||
|
tooltip: 'Pont bemérése',
|
||||||
|
child: const Icon(Icons.flag_sharp),
|
||||||
|
),
|
||||||
|
const SizedBox(
|
||||||
|
width: 15.0,
|
||||||
|
),
|
||||||
|
FloatingActionButton(
|
||||||
|
onPressed: () {
|
||||||
|
// controller.isMapMoveToCenter();
|
||||||
|
controller.showMeasuredPointsTableDialog();
|
||||||
|
},
|
||||||
|
heroTag: 'Database test',
|
||||||
|
tooltip: 'Pont bemérése',
|
||||||
|
child: const Icon(Icons.dataset),
|
||||||
|
),
|
||||||
|
],
|
||||||
|
),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
@ -0,0 +1,114 @@
|
|||||||
|
import 'package:flutter/material.dart';
|
||||||
|
import 'package:get/get.dart';
|
||||||
|
import 'package:terepi_seged/pages/map_survey/presentations/controllers/map_survey_controller.dart';
|
||||||
|
|
||||||
|
class MeasuredPointsTableDialog extends StatelessWidget {
|
||||||
|
final controller = Get.find<MapSurveyController>();
|
||||||
|
MeasuredPointsTableDialog({super.key});
|
||||||
|
|
||||||
|
Widget build(BuildContext context) {
|
||||||
|
return Scaffold(
|
||||||
|
body: Padding(
|
||||||
|
padding: const EdgeInsets.only(top: 20.0),
|
||||||
|
child: Column(
|
||||||
|
children: [
|
||||||
|
Padding(
|
||||||
|
padding: const EdgeInsets.all(8.0),
|
||||||
|
child: Row(
|
||||||
|
mainAxisAlignment: MainAxisAlignment.spaceBetween,
|
||||||
|
children: [
|
||||||
|
IconButton(
|
||||||
|
onPressed: () {
|
||||||
|
Get.back();
|
||||||
|
},
|
||||||
|
icon: const Icon(Icons.close)),
|
||||||
|
TextButton(
|
||||||
|
style: ButtonStyle(
|
||||||
|
overlayColor:
|
||||||
|
MaterialStateProperty.all(Colors.transparent)),
|
||||||
|
onPressed: () {
|
||||||
|
Get.back();
|
||||||
|
},
|
||||||
|
child: const Text(
|
||||||
|
'Bezár',
|
||||||
|
style: TextStyle(color: Colors.blue, fontSize: 14.0),
|
||||||
|
))
|
||||||
|
],
|
||||||
|
),
|
||||||
|
),
|
||||||
|
const Padding(
|
||||||
|
padding: EdgeInsets.symmetric(horizontal: 40.0),
|
||||||
|
child: Text(
|
||||||
|
'Bemért pontok',
|
||||||
|
style: TextStyle(fontSize: 20.0, fontWeight: FontWeight.bold),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
const SizedBox(height: 5),
|
||||||
|
Expanded(
|
||||||
|
child: FutureBuilder<List<dynamic>>(
|
||||||
|
future: controller.readMeasuredPoints(),
|
||||||
|
builder: (context, snapshot) {
|
||||||
|
if (snapshot.connectionState == ConnectionState.waiting) {
|
||||||
|
return Center(child: CircularProgressIndicator());
|
||||||
|
}
|
||||||
|
|
||||||
|
if (snapshot.hasError) {
|
||||||
|
return Center(
|
||||||
|
child: Text(
|
||||||
|
snapshot.error.toString(),
|
||||||
|
),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
if (!snapshot.hasData) {
|
||||||
|
return const Center(
|
||||||
|
child: Text("No Data available.\n Create new Data"));
|
||||||
|
}
|
||||||
|
// print(snapshot.data);
|
||||||
|
// return const Center(child: Text("Data available."));
|
||||||
|
return ListView.builder(
|
||||||
|
itemCount: snapshot.data!.length,
|
||||||
|
shrinkWrap: true,
|
||||||
|
itemBuilder: (context, int index) {
|
||||||
|
var data = snapshot.data![index];
|
||||||
|
print("snapshot data:");
|
||||||
|
print(data);
|
||||||
|
return ListTile(
|
||||||
|
leading: CircleAvatar(
|
||||||
|
backgroundColor: const Color(0xff764abc),
|
||||||
|
child: Text((index + 1).toString())),
|
||||||
|
title: Text(data['pointNumber'].toString(),
|
||||||
|
style: TextStyle(fontWeight: FontWeight.bold)),
|
||||||
|
subtitle: Column(
|
||||||
|
crossAxisAlignment: CrossAxisAlignment.start,
|
||||||
|
children: [
|
||||||
|
Text(data['description'],
|
||||||
|
style: TextStyle(
|
||||||
|
fontStyle: FontStyle.italic,
|
||||||
|
color: Colors.grey.shade700)),
|
||||||
|
Text(
|
||||||
|
"EovX: ${controller.formatEov.format(data['eovY'])} - EovY: ${controller.formatEov.format(data['eovX'])}",
|
||||||
|
style: TextStyle(
|
||||||
|
fontStyle: FontStyle.italic,
|
||||||
|
color: Colors.grey.shade400)),
|
||||||
|
Text(
|
||||||
|
"EovZ: ${controller.formatEovZ.format(data['altitude'] - data['poleHeight'])} (m)",
|
||||||
|
style: TextStyle(
|
||||||
|
fontStyle: FontStyle.italic,
|
||||||
|
color: Colors.grey.shade400)),
|
||||||
|
Text(
|
||||||
|
"H.hiba: ${controller.formatAltitudeError.format(data['horizontalError'])} (m) - V.hiba: ${controller.formatAltitudeError.format(data['verticalError'])} (m)",
|
||||||
|
style: TextStyle(
|
||||||
|
fontStyle: FontStyle.italic,
|
||||||
|
color: Colors.grey.shade400)),
|
||||||
|
],
|
||||||
|
));
|
||||||
|
});
|
||||||
|
},
|
||||||
|
),
|
||||||
|
)
|
||||||
|
],
|
||||||
|
),
|
||||||
|
),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
390
lib/pages/map_survey/presentations/views/settings_dialog.dart
Normal file
390
lib/pages/map_survey/presentations/views/settings_dialog.dart
Normal file
@ -0,0 +1,390 @@
|
|||||||
|
import 'package:flutter/material.dart';
|
||||||
|
import 'package:get/get.dart';
|
||||||
|
import 'package:terepi_seged/pages/map_survey/presentations/controllers/map_survey_controller.dart';
|
||||||
|
|
||||||
|
class SettingsDialog extends StatelessWidget {
|
||||||
|
final controller = Get.find<MapSurveyController>();
|
||||||
|
SettingsDialog({super.key});
|
||||||
|
|
||||||
|
Widget build(BuildContext context) {
|
||||||
|
return Scaffold(
|
||||||
|
body: ListView(
|
||||||
|
children: [
|
||||||
|
Padding(
|
||||||
|
padding: const EdgeInsets.all(8.0),
|
||||||
|
child: Row(
|
||||||
|
mainAxisAlignment: MainAxisAlignment.spaceBetween,
|
||||||
|
children: [
|
||||||
|
IconButton(
|
||||||
|
onPressed: () {
|
||||||
|
if (controller.ntripUsernameController.text.isNotEmpty) {
|
||||||
|
controller.ntripUserName.value =
|
||||||
|
controller.ntripUsernameController.text;
|
||||||
|
controller.saveNtripUserName(
|
||||||
|
controller.ntripUsernameController.text);
|
||||||
|
if (controller
|
||||||
|
.ntripPasswordController.text.isNotEmpty) {
|
||||||
|
controller.ntripPassword.value =
|
||||||
|
controller.ntripPasswordController.text;
|
||||||
|
controller.saveNtripPassword(
|
||||||
|
controller.ntripPasswordController.text);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
Get.back();
|
||||||
|
},
|
||||||
|
icon: const Icon(Icons.close)),
|
||||||
|
TextButton(
|
||||||
|
style: ButtonStyle(
|
||||||
|
overlayColor:
|
||||||
|
MaterialStateProperty.all(Colors.transparent)),
|
||||||
|
onPressed: () {
|
||||||
|
if (controller.ntripUsernameController.text.isNotEmpty) {
|
||||||
|
controller.ntripUserName.value =
|
||||||
|
controller.ntripUsernameController.text;
|
||||||
|
controller.saveNtripUserName(
|
||||||
|
controller.ntripUsernameController.text);
|
||||||
|
if (controller
|
||||||
|
.ntripPasswordController.text.isNotEmpty) {
|
||||||
|
controller.ntripPassword.value =
|
||||||
|
controller.ntripPasswordController.text;
|
||||||
|
controller.saveNtripPassword(
|
||||||
|
controller.ntripPasswordController.text);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
Get.back();
|
||||||
|
},
|
||||||
|
child: const Text(
|
||||||
|
'Bezár',
|
||||||
|
style: TextStyle(color: Colors.blue, fontSize: 14.0),
|
||||||
|
))
|
||||||
|
],
|
||||||
|
),
|
||||||
|
),
|
||||||
|
const Padding(
|
||||||
|
padding: EdgeInsets.symmetric(horizontal: 40.0),
|
||||||
|
child: Text(
|
||||||
|
'Beállítások',
|
||||||
|
style: TextStyle(fontSize: 20.0, fontWeight: FontWeight.bold),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
const Padding(
|
||||||
|
padding: EdgeInsets.symmetric(horizontal: 40.0),
|
||||||
|
child: Text(
|
||||||
|
'GPS vevő',
|
||||||
|
style: TextStyle(fontSize: 14.0, fontWeight: FontWeight.w500),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
Obx(() => Column(children: [
|
||||||
|
RadioListTile(
|
||||||
|
title: Text('TiGNSS Rover-BE6A'),
|
||||||
|
value: '10:06:1C:97:BE:6A',
|
||||||
|
groupValue: controller.gpsAddress.value,
|
||||||
|
onChanged: (value) {
|
||||||
|
controller.gpsAddress.value = value!;
|
||||||
|
controller.gpsName.value = 'TiGNSS Rover-BE6A';
|
||||||
|
controller.saveGpsAddress(value);
|
||||||
|
controller.saveGpsName('TiGNSS Rover-BE6A');
|
||||||
|
}),
|
||||||
|
RadioListTile(
|
||||||
|
title: Text('TiGNSS Rover-1DC6'),
|
||||||
|
value: 'E8:31:CD:16:1D:C6',
|
||||||
|
groupValue: controller.gpsAddress.value,
|
||||||
|
onChanged: (value) {
|
||||||
|
controller.gpsAddress.value = value!;
|
||||||
|
controller.gpsName.value = 'TiGNSS Rover-1DC6';
|
||||||
|
controller.saveGpsAddress(value);
|
||||||
|
controller.saveGpsName('TiGNSS Rover-1DC6');
|
||||||
|
}),
|
||||||
|
RadioListTile(
|
||||||
|
title: Text('TiGNSS Rover-9C3A'),
|
||||||
|
value: '08:3A:8D:14:9C:3A',
|
||||||
|
groupValue: controller.gpsAddress.value,
|
||||||
|
onChanged: (value) {
|
||||||
|
controller.gpsAddress.value = value!;
|
||||||
|
controller.gpsName.value = 'TiGNSS Rover-9C3A';
|
||||||
|
controller.saveGpsAddress(value);
|
||||||
|
controller.saveGpsName('TiGNSS Rover-9C3A');
|
||||||
|
}),
|
||||||
|
RadioListTile(
|
||||||
|
title: Text('TiGNSS Rover-72C2'),
|
||||||
|
value: '10:06:1C:97:72:C2',
|
||||||
|
groupValue: controller.gpsAddress.value,
|
||||||
|
onChanged: (value) {
|
||||||
|
controller.gpsAddress.value = value!;
|
||||||
|
controller.gpsName.value = 'TiGNSS Rover-72C2';
|
||||||
|
controller.saveGpsAddress(value);
|
||||||
|
controller.saveGpsName('TiGNSS Rover-72C2');
|
||||||
|
}),
|
||||||
|
RadioListTile(
|
||||||
|
title: Text('TiGNSS Rover-FE16'),
|
||||||
|
value: '10:06:1C:9F:FE:16',
|
||||||
|
groupValue: controller.gpsAddress.value,
|
||||||
|
onChanged: (value) {
|
||||||
|
controller.gpsAddress.value = value!;
|
||||||
|
controller.gpsName.value = 'TiGNSS Rover-FE16';
|
||||||
|
controller.saveGpsAddress(value);
|
||||||
|
controller.saveGpsName('TiGNSS Rover-FE16');
|
||||||
|
}),
|
||||||
|
RadioListTile(
|
||||||
|
title: Text('TiGNSS Rover-3B0A'),
|
||||||
|
value: '10:C6:1C:9E:3B:0A',
|
||||||
|
groupValue: controller.gpsAddress.value,
|
||||||
|
onChanged: (value) {
|
||||||
|
controller.gpsAddress.value = value!;
|
||||||
|
controller.gpsName.value = 'TiGNSS Rover-3B0A';
|
||||||
|
controller.saveGpsAddress(value);
|
||||||
|
controller.saveGpsName('TiGNSS Rover-3B0A');
|
||||||
|
}),
|
||||||
|
RadioListTile(
|
||||||
|
title: Text('TiGNSS Rover-7FEA'),
|
||||||
|
value: '10:06:1C:9C:7F:EA',
|
||||||
|
groupValue: controller.gpsAddress.value,
|
||||||
|
onChanged: (value) {
|
||||||
|
controller.gpsAddress.value = value!;
|
||||||
|
controller.gpsName.value = 'TiGNSS Rover-7FEA';
|
||||||
|
controller.saveGpsAddress(value);
|
||||||
|
controller.saveGpsName('TiGNSS Rover-7FEA');
|
||||||
|
}),
|
||||||
|
RadioListTile(
|
||||||
|
title: Text('TiGNSS Rover-A39E'),
|
||||||
|
value: '10:06:1C:97:A3:9E',
|
||||||
|
groupValue: controller.gpsAddress.value,
|
||||||
|
onChanged: (value) {
|
||||||
|
controller.gpsAddress.value = value!;
|
||||||
|
controller.gpsName.value = 'TiGNSS Rover-A39E';
|
||||||
|
controller.saveGpsAddress(value);
|
||||||
|
controller.saveGpsName('TiGNSS Rover-A39E');
|
||||||
|
}),
|
||||||
|
RadioListTile(
|
||||||
|
title: Text('TiGNSS Rover-FF4E'),
|
||||||
|
value: '98:CD:AC:62:FF:4E',
|
||||||
|
groupValue: controller.gpsAddress.value,
|
||||||
|
onChanged: (value) {
|
||||||
|
controller.gpsAddress.value = value!;
|
||||||
|
controller.gpsName.value = 'TiGNSS Rover-FF4E';
|
||||||
|
controller.saveGpsAddress(value);
|
||||||
|
controller.saveGpsName('TiGNSS Rover-FF4E');
|
||||||
|
}),
|
||||||
|
RadioListTile(
|
||||||
|
title: Text('TiGNSS Rover-8BB2'),
|
||||||
|
value: 'E8:31:CD:14:8B:B2',
|
||||||
|
groupValue: controller.gpsAddress.value,
|
||||||
|
onChanged: (value) {
|
||||||
|
controller.gpsAddress.value = value!;
|
||||||
|
controller.gpsName.value = 'TiGNSS Rover-8BB2';
|
||||||
|
controller.saveGpsAddress(value);
|
||||||
|
controller.saveGpsName('TiGNSS Rover-8BB2');
|
||||||
|
}),
|
||||||
|
RadioListTile(
|
||||||
|
title: Text('TiGNSS Rover-FF36'),
|
||||||
|
value: '98:CD:AC:62:FF:36',
|
||||||
|
groupValue: controller.gpsAddress.value,
|
||||||
|
onChanged: (value) {
|
||||||
|
controller.gpsAddress.value = value!;
|
||||||
|
controller.gpsName.value = 'TiGNSS Rover-FF36';
|
||||||
|
controller.saveGpsAddress(value);
|
||||||
|
controller.saveGpsName('TiGNSS Rover-FF36');
|
||||||
|
})
|
||||||
|
])),
|
||||||
|
const Padding(
|
||||||
|
padding: EdgeInsets.symmetric(horizontal: 20),
|
||||||
|
child: Divider(
|
||||||
|
height: 5,
|
||||||
|
),
|
||||||
|
),
|
||||||
|
// const SizedBox(height: 5),
|
||||||
|
// const Padding(
|
||||||
|
// padding: EdgeInsets.symmetric(horizontal: 40.0),
|
||||||
|
// child: Text(
|
||||||
|
// 'Bemért pont azonosítója:',
|
||||||
|
// style: TextStyle(fontSize: 14.0, fontWeight: FontWeight.w500),
|
||||||
|
// ),
|
||||||
|
// ),
|
||||||
|
// const SizedBox(height: 4),
|
||||||
|
// Padding(
|
||||||
|
// padding: const EdgeInsets.symmetric(horizontal: 40),
|
||||||
|
// child: Row(
|
||||||
|
// children: [
|
||||||
|
// Flexible(
|
||||||
|
// child: TextField(
|
||||||
|
// controller: controller.pointPrefixController,
|
||||||
|
// decoration: const InputDecoration(
|
||||||
|
// border: OutlineInputBorder(), labelText: 'Előtag'),
|
||||||
|
// ),
|
||||||
|
// ),
|
||||||
|
// const Padding(
|
||||||
|
// padding: EdgeInsets.symmetric(horizontal: 10.0),
|
||||||
|
// child: Text(
|
||||||
|
// 'Eredeti azonosító',
|
||||||
|
// style: TextStyle(fontSize: 12.0),
|
||||||
|
// ),
|
||||||
|
// ),
|
||||||
|
// Flexible(
|
||||||
|
// child: TextField(
|
||||||
|
// controller: controller.pointPostfixController,
|
||||||
|
// decoration: const InputDecoration(
|
||||||
|
// border: OutlineInputBorder(), labelText: 'Utótag'),
|
||||||
|
// ),
|
||||||
|
// ),
|
||||||
|
// ],
|
||||||
|
// ),
|
||||||
|
// ),
|
||||||
|
const SizedBox(height: 5),
|
||||||
|
const Padding(
|
||||||
|
padding: EdgeInsets.symmetric(horizontal: 20.0),
|
||||||
|
child: Text(
|
||||||
|
'Mérés iránya:',
|
||||||
|
style: TextStyle(fontSize: 14.0, fontWeight: FontWeight.w500),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
Obx(
|
||||||
|
() => Column(children: [
|
||||||
|
RadioListTile<bool>(
|
||||||
|
title: Text('Növekvő'),
|
||||||
|
value: true,
|
||||||
|
groupValue: controller.pointMeasuringDirectionForward.value,
|
||||||
|
onChanged: (value) {
|
||||||
|
controller.pointMeasuringDirectionForward.value = value!;
|
||||||
|
}),
|
||||||
|
RadioListTile<bool>(
|
||||||
|
title: Text('Csökkenő'),
|
||||||
|
value: false,
|
||||||
|
groupValue: controller.pointMeasuringDirectionForward.value,
|
||||||
|
onChanged: (value) {
|
||||||
|
controller.pointMeasuringDirectionForward.value = value!;
|
||||||
|
})
|
||||||
|
]),
|
||||||
|
),
|
||||||
|
const Padding(
|
||||||
|
padding: EdgeInsets.symmetric(horizontal: 20),
|
||||||
|
child: Divider(
|
||||||
|
height: 5,
|
||||||
|
),
|
||||||
|
),
|
||||||
|
const SizedBox(height: 5),
|
||||||
|
const Padding(
|
||||||
|
padding: EdgeInsets.symmetric(horizontal: 20.0),
|
||||||
|
child: Text(
|
||||||
|
'Kitűzött pont azonostója:',
|
||||||
|
style: TextStyle(fontSize: 14.0, fontWeight: FontWeight.w500),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
const SizedBox(height: 4),
|
||||||
|
Padding(
|
||||||
|
padding: const EdgeInsets.symmetric(horizontal: 20),
|
||||||
|
child: SizedBox(
|
||||||
|
height: 32,
|
||||||
|
width: double.infinity,
|
||||||
|
child: Row(
|
||||||
|
children: [
|
||||||
|
Flexible(
|
||||||
|
child: TextField(
|
||||||
|
controller: controller.pointPrefixController,
|
||||||
|
decoration: const InputDecoration(
|
||||||
|
border: OutlineInputBorder(), labelText: 'Előtag'),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
const Padding(
|
||||||
|
padding: EdgeInsets.symmetric(horizontal: 10.0),
|
||||||
|
child: Text(
|
||||||
|
'Eredeti azonosító',
|
||||||
|
style: TextStyle(fontSize: 12.0),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
Flexible(
|
||||||
|
child: TextField(
|
||||||
|
controller: controller.pointPostfixController,
|
||||||
|
decoration: const InputDecoration(
|
||||||
|
border: OutlineInputBorder(), labelText: 'Utótag'),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
],
|
||||||
|
),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
const Padding(
|
||||||
|
padding: EdgeInsets.only(left: 20, top: 8),
|
||||||
|
child: Text(
|
||||||
|
'Például: előtag: 6, utótag: -bp, eredeti azonositó: 1002 -> 61002-bp',
|
||||||
|
style: TextStyle(fontSize: 10, fontWeight: FontWeight.w400))),
|
||||||
|
const SizedBox(height: 15),
|
||||||
|
const Padding(
|
||||||
|
padding: EdgeInsets.symmetric(horizontal: 20),
|
||||||
|
child: Divider(
|
||||||
|
height: 5,
|
||||||
|
),
|
||||||
|
),
|
||||||
|
const SizedBox(height: 5),
|
||||||
|
const Padding(
|
||||||
|
padding: EdgeInsets.symmetric(horizontal: 20.0),
|
||||||
|
child: Text(
|
||||||
|
'Ntrip szolgáltatás:',
|
||||||
|
style: TextStyle(fontSize: 14.0, fontWeight: FontWeight.w500),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
const SizedBox(height: 5),
|
||||||
|
Padding(
|
||||||
|
padding: const EdgeInsets.symmetric(horizontal: 20.0),
|
||||||
|
child: Column(children: [
|
||||||
|
SizedBox(
|
||||||
|
height: 40,
|
||||||
|
child: TextField(
|
||||||
|
controller: controller.ntripUsernameController,
|
||||||
|
enableSuggestions: false,
|
||||||
|
autocorrect: false,
|
||||||
|
decoration: InputDecoration(
|
||||||
|
floatingLabelBehavior: FloatingLabelBehavior.never,
|
||||||
|
isDense: true,
|
||||||
|
filled: true,
|
||||||
|
fillColor: Colors.grey.shade300,
|
||||||
|
border: OutlineInputBorder(
|
||||||
|
borderSide: BorderSide.none,
|
||||||
|
borderRadius: BorderRadius.circular(12)),
|
||||||
|
labelText: 'Felhasználónév',
|
||||||
|
prefixIcon: Icon(Icons.account_circle_rounded)),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
const SizedBox(height: 10),
|
||||||
|
Obx(
|
||||||
|
() => SizedBox(
|
||||||
|
height: 40,
|
||||||
|
child: TextField(
|
||||||
|
keyboardType: TextInputType.visiblePassword,
|
||||||
|
obscureText: controller.isShowPassword.value,
|
||||||
|
focusNode: controller.passwordFieldFocusNode,
|
||||||
|
enableSuggestions: false,
|
||||||
|
autocorrect: false,
|
||||||
|
controller: controller.ntripPasswordController,
|
||||||
|
decoration: InputDecoration(
|
||||||
|
floatingLabelBehavior: FloatingLabelBehavior.never,
|
||||||
|
isDense: true,
|
||||||
|
filled: true,
|
||||||
|
fillColor: Colors.grey.shade300,
|
||||||
|
border: OutlineInputBorder(
|
||||||
|
borderSide: BorderSide.none,
|
||||||
|
borderRadius: BorderRadius.circular(12)),
|
||||||
|
labelText: 'Jelszó',
|
||||||
|
prefixIcon: Icon(
|
||||||
|
Icons.lock_rounded,
|
||||||
|
size: 24,
|
||||||
|
),
|
||||||
|
suffixIcon: Padding(
|
||||||
|
padding: const EdgeInsets.fromLTRB(0, 0, 4, 0),
|
||||||
|
child: GestureDetector(
|
||||||
|
onTap: controller.toggleShowPassword,
|
||||||
|
child: Icon(
|
||||||
|
controller.isShowPassword.value
|
||||||
|
? Icons.visibility_rounded
|
||||||
|
: Icons.visibility_off_rounded,
|
||||||
|
size: 24)))),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
const SizedBox(height: 20)
|
||||||
|
]),
|
||||||
|
),
|
||||||
|
],
|
||||||
|
),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
@ -9,9 +9,10 @@ import 'package:flutter/material.dart';
|
|||||||
import 'package:flutter/services.dart';
|
import 'package:flutter/services.dart';
|
||||||
import 'package:flutter_bluetooth_serial/flutter_bluetooth_serial.dart';
|
import 'package:flutter_bluetooth_serial/flutter_bluetooth_serial.dart';
|
||||||
import 'package:flutter_map/flutter_map.dart';
|
import 'package:flutter_map/flutter_map.dart';
|
||||||
import 'package:flutter_map_geojson/flutter_map_geojson.dart';
|
// import 'package:flutter_map_geojson/flutter_map_geojson.dart';
|
||||||
import 'package:flutter_map_polywidget/flutter_map_polywidget.dart';
|
import 'package:flutter_map_polywidget/flutter_map_polywidget.dart';
|
||||||
import 'package:get/get.dart';
|
import 'package:get/get.dart';
|
||||||
|
import 'package:get/get_connect/http/src/utils/utils.dart';
|
||||||
import 'package:intl/date_time_patterns.dart';
|
import 'package:intl/date_time_patterns.dart';
|
||||||
import 'package:intl/intl.dart';
|
import 'package:intl/intl.dart';
|
||||||
import 'package:location/location.dart';
|
import 'package:location/location.dart';
|
||||||
@ -20,6 +21,8 @@ import 'package:nmea/nmea.dart';
|
|||||||
import 'package:path_provider/path_provider.dart';
|
import 'package:path_provider/path_provider.dart';
|
||||||
import 'package:permission_handler/permission_handler.dart'
|
import 'package:permission_handler/permission_handler.dart'
|
||||||
as permission_handler;
|
as permission_handler;
|
||||||
|
import 'package:supabase_flutter/supabase_flutter.dart';
|
||||||
|
import 'package:terepi_seged/controls/geojson_parser.dart';
|
||||||
import 'package:terepi_seged/eov/convert_coordinate.dart';
|
import 'package:terepi_seged/eov/convert_coordinate.dart';
|
||||||
import 'package:terepi_seged/eov/eov.dart';
|
import 'package:terepi_seged/eov/eov.dart';
|
||||||
import 'package:terepi_seged/gnss_sentences/gngga.dart';
|
import 'package:terepi_seged/gnss_sentences/gngga.dart';
|
||||||
@ -28,12 +31,13 @@ import 'package:terepi_seged/gnss_sentences/gnrmc.dart';
|
|||||||
import 'package:terepi_seged/models/point_to_measure.dart';
|
import 'package:terepi_seged/models/point_to_measure.dart';
|
||||||
import 'package:terepi_seged/models/point_with_description_model.dart';
|
import 'package:terepi_seged/models/point_with_description_model.dart';
|
||||||
import 'package:proj4dart/proj4dart.dart' as proj4;
|
import 'package:proj4dart/proj4dart.dart' as proj4;
|
||||||
|
import 'package:shared_preferences/shared_preferences.dart';
|
||||||
|
|
||||||
class NavigationViewController extends GetxController {
|
class NavigationViewController extends GetxController {
|
||||||
// String gpsAddress = "E8:31:CD:14:8B:B2";
|
// String gpsAddress = "E8:31:CD:14:8B:B2";
|
||||||
// String gpsAddress = "98:CD:AC:62:FF:4E";
|
// String gpsAddress = "98:CD:AC:62:FF:4E";
|
||||||
RxString gpsAddress = "98:CD:AC:62:FF:36".obs;
|
RxString gpsAddress = "98:CD:AC:62:FF:36".obs;
|
||||||
String gpsName = "TiGNSS Rover-FF4E";
|
RxString gpsName = "TiGNSS Rover-FF4E".obs;
|
||||||
// String gpsName = "TiGNSS Rover-8BB2";
|
// String gpsName = "TiGNSS Rover-8BB2";
|
||||||
static const double maxZoomValue = 25.0;
|
static const double maxZoomValue = 25.0;
|
||||||
Rx<bool> gpsIsConnected = false.obs;
|
Rx<bool> gpsIsConnected = false.obs;
|
||||||
@ -118,10 +122,20 @@ class NavigationViewController extends GetxController {
|
|||||||
final CollectionReference _vibratorTracker =
|
final CollectionReference _vibratorTracker =
|
||||||
FirebaseFirestore.instance.collection('vibratorTracker');
|
FirebaseFirestore.instance.collection('vibratorTracker');
|
||||||
DateTime lastGpsDataSaveTime = DateTime(2000, 1, 1, 0, 0, 0);
|
DateTime lastGpsDataSaveTime = DateTime(2000, 1, 1, 0, 0, 0);
|
||||||
int vehicleNumber = 2;
|
RxInt vehicleNumber = 5.obs;
|
||||||
late Location internalGpsLogger;
|
late Location internalGpsLogger;
|
||||||
Location internalGpsLocation = Location();
|
Location internalGpsLocation = Location();
|
||||||
late StreamSubscription<LocationData> internalGpsLocationSubscription;
|
late StreamSubscription<LocationData>? internalGpsLocationSubscription;
|
||||||
|
|
||||||
|
late List<Polyline<Object>> pathLayer = [];
|
||||||
|
|
||||||
|
RxString ntripUserName = "".obs;
|
||||||
|
RxString ntripPassword = "".obs;
|
||||||
|
late SharedPreferences prefs;
|
||||||
|
|
||||||
|
late AuthResponse authResponse;
|
||||||
|
late Session? session;
|
||||||
|
late User? user;
|
||||||
|
|
||||||
// late SMIBool gpsTrigger;
|
// late SMIBool gpsTrigger;
|
||||||
// late StateMachineController riveGpsIconController;
|
// late StateMachineController riveGpsIconController;
|
||||||
@ -146,13 +160,18 @@ class NavigationViewController extends GetxController {
|
|||||||
..registerTalkerSentence("GST", (line) => Gngst(raw: line));
|
..registerTalkerSentence("GST", (line) => Gngst(raw: line));
|
||||||
|
|
||||||
mapController = MapController();
|
mapController = MapController();
|
||||||
|
prefs = await SharedPreferences.getInstance();
|
||||||
|
|
||||||
|
authResponse = await Supabase.instance.client.auth
|
||||||
|
.signInWithPassword(email: 'test.elek.1@email.hu', password: 'demo');
|
||||||
|
session = authResponse.session;
|
||||||
|
user = authResponse.user;
|
||||||
|
|
||||||
// riveGpsIconController = RiveUtils.getRiveController(Artboard(),
|
// riveGpsIconController = RiveUtils.getRiveController(Artboard(),
|
||||||
// stateMachineName: "gps_Interactivity");
|
// stateMachineName: "gps_Interactivity");
|
||||||
// gpsTrigger = riveGpsIconController.findSMI("active");
|
// gpsTrigger = riveGpsIconController.findSMI("active");
|
||||||
|
|
||||||
mapIsInitialized.value = true;
|
mapIsInitialized.value = true;
|
||||||
vehicleNumberController.text = vehicleNumber.toString();
|
|
||||||
}
|
}
|
||||||
|
|
||||||
@override
|
@override
|
||||||
@ -175,7 +194,7 @@ class NavigationViewController extends GetxController {
|
|||||||
// internalGpsLogTimer.cancel();
|
// internalGpsLogTimer.cancel();
|
||||||
// }
|
// }
|
||||||
|
|
||||||
internalGpsLocationSubscription.cancel();
|
internalGpsLocationSubscription!.cancel();
|
||||||
}
|
}
|
||||||
|
|
||||||
@override
|
@override
|
||||||
@ -184,8 +203,8 @@ class NavigationViewController extends GetxController {
|
|||||||
|
|
||||||
_getInitialLocation();
|
_getInitialLocation();
|
||||||
|
|
||||||
String data = await rootBundle.loadString('assets/Files/kozmuvek.geojson');
|
// String data = await rootBundle.loadString('assets/Files/kozmuvek.geojson');
|
||||||
parser.parseGeoJsonAsString(data);
|
// parser.parseGeoJsonAsString(data);
|
||||||
|
|
||||||
if (await permission_handler.Permission.storage.isGranted) {
|
if (await permission_handler.Permission.storage.isGranted) {
|
||||||
print("Storage permission is ok ...");
|
print("Storage permission is ok ...");
|
||||||
@ -207,6 +226,37 @@ class NavigationViewController extends GetxController {
|
|||||||
} else {
|
} else {
|
||||||
print("No external storage permission");
|
print("No external storage permission");
|
||||||
}
|
}
|
||||||
|
if (prefs.containsKey('gpsAddress')) {
|
||||||
|
var address = prefs.getString('gpsAddress');
|
||||||
|
if (address != null) {
|
||||||
|
gpsAddress.value = address;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (prefs.containsKey('gpsName')) {
|
||||||
|
var name = prefs.getString('gpsName');
|
||||||
|
if (name != null) {
|
||||||
|
gpsName.value = name;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (prefs.containsKey('ntripUserName')) {
|
||||||
|
var userName = prefs.getString('ntripUserName');
|
||||||
|
if (userName != null) {
|
||||||
|
ntripUserName.value = userName;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (prefs.containsKey('ntripPassword')) {
|
||||||
|
var password = prefs.getString('ntripPassword');
|
||||||
|
if (password != null) {
|
||||||
|
ntripPassword.value = password;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (prefs.containsKey('vehicleNumber')) {
|
||||||
|
var vehicleNum = prefs.getInt('vehicleNumber');
|
||||||
|
if (vehicleNum != null) {
|
||||||
|
vehicleNumber.value = vehicleNum;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
vehicleNumberController.text = vehicleNumber.toString();
|
||||||
}
|
}
|
||||||
|
|
||||||
directory = await getExternalStorageDirectory();
|
directory = await getExternalStorageDirectory();
|
||||||
@ -256,7 +306,7 @@ class NavigationViewController extends GetxController {
|
|||||||
await internalGpsLocation.changeSettings(interval: 10000);
|
await internalGpsLocation.changeSettings(interval: 10000);
|
||||||
|
|
||||||
internalGpsLocationSubscription =
|
internalGpsLocationSubscription =
|
||||||
internalGpsLogger.onLocationChanged.listen((LocationData value) {
|
internalGpsLogger.onLocationChanged.listen((LocationData value) async {
|
||||||
double? longitude;
|
double? longitude;
|
||||||
double? latitude;
|
double? latitude;
|
||||||
double? accuracy;
|
double? accuracy;
|
||||||
@ -276,16 +326,29 @@ class NavigationViewController extends GetxController {
|
|||||||
internalGpsFile.writeAsString(
|
internalGpsFile.writeAsString(
|
||||||
"$vehicleNumber;$time;$latitude;$longitude;$altitude;$accuracy\r\n");
|
"$vehicleNumber;$time;$latitude;$longitude;$altitude;$accuracy\r\n");
|
||||||
|
|
||||||
_vibratorTracker.add({
|
// _vibratorTracker.add({
|
||||||
"gpsType": 0,
|
// "gpsType": 0,
|
||||||
"vibratorNumber": vehicleNumber,
|
// "vibratorNumber": vehicleNumber.value,
|
||||||
"gpsDateTime": time,
|
// "gpsDateTime": time,
|
||||||
"latitude": latitude,
|
// "latitude": latitude,
|
||||||
"longitude": longitude,
|
// "longitude": longitude,
|
||||||
"altitude": altitude,
|
// "altitude": altitude,
|
||||||
"horizontalError": accuracy,
|
// "horizontalError": accuracy,
|
||||||
"speed": speed,
|
// "speed": speed,
|
||||||
"createdAt": DateTime.timestamp()
|
// "createdAt": DateTime.timestamp()
|
||||||
|
// });
|
||||||
|
|
||||||
|
await Supabase.instance.client
|
||||||
|
.from('TerepiSeged_VibratorTracker')
|
||||||
|
.insert({
|
||||||
|
'gpsType': 0,
|
||||||
|
'vibratorNumber': vehicleNumber.value,
|
||||||
|
'gpsDateTime': time.toIso8601String(),
|
||||||
|
'latitude': gpsLatitude.value,
|
||||||
|
'longitude': gpsLongitude.value,
|
||||||
|
'altitude': gpsAltitude.value,
|
||||||
|
'horizontalError': accuracy,
|
||||||
|
'speed': speed
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
@ -581,14 +644,17 @@ class NavigationViewController extends GetxController {
|
|||||||
|
|
||||||
if (DateTime.now().difference(lastGpsDataSaveTime).inSeconds >= 10 &&
|
if (DateTime.now().difference(lastGpsDataSaveTime).inSeconds >= 10 &&
|
||||||
gpsQuality.value > 0) {
|
gpsQuality.value > 0) {
|
||||||
|
lastGpsDataSaveTime = DateTime.now();
|
||||||
|
|
||||||
_savePositionToDatabase();
|
_savePositionToDatabase();
|
||||||
|
print("Save to databases: ${lastGpsDataSaveTime}");
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
void _savePositionToDatabase() {
|
void _savePositionToDatabase() async {
|
||||||
_vibratorTracker.add({
|
_vibratorTracker.add({
|
||||||
"gpsType": 1,
|
"gpsType": 1,
|
||||||
"vibratorNumber": vehicleNumber,
|
"vibratorNumber": vehicleNumber.value,
|
||||||
"gpsQuality": gpsQuality.value,
|
"gpsQuality": gpsQuality.value,
|
||||||
"gpsDateTime": gpsDateTime.value,
|
"gpsDateTime": gpsDateTime.value,
|
||||||
"latitude": gpsLatitude.value,
|
"latitude": gpsLatitude.value,
|
||||||
@ -601,7 +667,25 @@ class NavigationViewController extends GetxController {
|
|||||||
"antennaHeight": double.parse(gpsHeightController.text),
|
"antennaHeight": double.parse(gpsHeightController.text),
|
||||||
"createdAt": DateTime.timestamp()
|
"createdAt": DateTime.timestamp()
|
||||||
});
|
});
|
||||||
lastGpsDataSaveTime = DateTime.now();
|
|
||||||
|
final res = await Supabase.instance.client
|
||||||
|
.from('TerepiSeged_VibratorTracker')
|
||||||
|
.insert({
|
||||||
|
"gpsType": 1,
|
||||||
|
"vibratorNumber": vehicleNumber.value,
|
||||||
|
"gpsQuality": gpsQuality.value,
|
||||||
|
"gpsDateTime": gpsDateTime.value.toIso8601String(),
|
||||||
|
"latitude": gpsLatitude.value,
|
||||||
|
"longitude": gpsLongitude.value,
|
||||||
|
"altitude": gpsAltitude.value,
|
||||||
|
"eovY": eov.value.Y,
|
||||||
|
"eovX": eov.value.X,
|
||||||
|
"horizontalError": max(gpsLatitudeError.value, gpsLongitudeError.value),
|
||||||
|
"verticalError": gpsAltitudeError.value,
|
||||||
|
// "heightOfGeoid": gpsGeoidSeparation.value,
|
||||||
|
"poleHeight": double.parse(gpsHeightController.text)
|
||||||
|
}).select();
|
||||||
|
print(res);
|
||||||
}
|
}
|
||||||
|
|
||||||
String getGpsQualityIndicator({required int quality}) {
|
String getGpsQualityIndicator({required int quality}) {
|
||||||
@ -896,15 +980,15 @@ class NavigationViewController extends GetxController {
|
|||||||
width: 15.0,
|
width: 15.0,
|
||||||
height: 15.0,
|
height: 15.0,
|
||||||
decoration: BoxDecoration(
|
decoration: BoxDecoration(
|
||||||
color: Colors.purple,
|
color: Colors.yellow,
|
||||||
shape: BoxShape.circle,
|
shape: BoxShape.circle,
|
||||||
border: Border.all(width: 1.0, color: Colors.black)),
|
border: Border.all(width: 1.0, color: Colors.black)),
|
||||||
)));
|
)));
|
||||||
|
|
||||||
pointsToMeasureLabel.add(PolyWidget(
|
pointsToMeasureLabel.add(PolyWidget(
|
||||||
center: LatLng(wgsCoord.y + 0.0000075, wgsCoord.x + 0.0000075),
|
center: LatLng(wgsCoord.y + 0.0000275, wgsCoord.x + 0.0000275),
|
||||||
widthInMeters: 3,
|
widthInMeters: 28,
|
||||||
heightInMeters: 3,
|
heightInMeters: 28,
|
||||||
// constraints: const BoxConstraints(
|
// constraints: const BoxConstraints(
|
||||||
// minWidth: 250,
|
// minWidth: 250,
|
||||||
// maxWidth: 350,
|
// maxWidth: 350,
|
||||||
@ -934,6 +1018,24 @@ class NavigationViewController extends GetxController {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void ReadPathFromFile() async {
|
||||||
|
File? file;
|
||||||
|
FilePickerResult? result = await FilePicker.platform.pickFiles();
|
||||||
|
|
||||||
|
if (result != null) {
|
||||||
|
file = File(result.files.single.path!);
|
||||||
|
} else {
|
||||||
|
print("No file selected");
|
||||||
|
}
|
||||||
|
if (await file!.exists()) {
|
||||||
|
String data = await file.readAsString();
|
||||||
|
parser.defaultPolylineColor = Colors.orangeAccent;
|
||||||
|
parser.defaultPolylineStroke = 5.0;
|
||||||
|
parser.parseGeoJsonAsString(data);
|
||||||
|
pathLayer = parser.polylines;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
void pointsToMeasureSelectedValueChanged(int value) {
|
void pointsToMeasureSelectedValueChanged(int value) {
|
||||||
pointsToMeasureSelectedValue.value = value;
|
pointsToMeasureSelectedValue.value = value;
|
||||||
print('Selected point -> ${pointsToMeasureSelectedValue.value}');
|
print('Selected point -> ${pointsToMeasureSelectedValue.value}');
|
||||||
@ -964,4 +1066,24 @@ class NavigationViewController extends GetxController {
|
|||||||
void addMeasuredPoint() {
|
void addMeasuredPoint() {
|
||||||
// _measuredPoints.add({"id": 4001, "latitude": 46.3455, "longitude": 19.652});
|
// _measuredPoints.add({"id": 4001, "latitude": 46.3455, "longitude": 19.652});
|
||||||
}
|
}
|
||||||
|
void saveGpsAddress(String address) {
|
||||||
|
prefs.setString('gpsAddress', address);
|
||||||
|
}
|
||||||
|
|
||||||
|
void saveGpsName(String name) {
|
||||||
|
prefs.setString('gpsName', name);
|
||||||
|
}
|
||||||
|
|
||||||
|
void saveNtripUserName(String username) {
|
||||||
|
prefs.setString('ntripUserName', username);
|
||||||
|
}
|
||||||
|
|
||||||
|
void saveNtripPassword(String password) {
|
||||||
|
prefs.setString('ntripPassword', password);
|
||||||
|
}
|
||||||
|
|
||||||
|
void saveVehicleNumber(int vehicleNumber) {
|
||||||
|
prefs.setInt('vehicleNumber', vehicleNumber);
|
||||||
|
print("Save vehicleNumber to prefs");
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@ -27,10 +27,10 @@ class NavigationView extends GetView<NavigationViewController> {
|
|||||||
crossAxisAlignment: CrossAxisAlignment.start,
|
crossAxisAlignment: CrossAxisAlignment.start,
|
||||||
children: [
|
children: [
|
||||||
Text('Navigáció'),
|
Text('Navigáció'),
|
||||||
Text(
|
// Text(
|
||||||
"Mérés: Budapest-4",
|
// "Mérés: Zalaegerszeg",
|
||||||
style: TextStyle(fontSize: 12.0),
|
// style: TextStyle(fontSize: 12.0),
|
||||||
)
|
// )
|
||||||
],
|
],
|
||||||
),
|
),
|
||||||
actions: [
|
actions: [
|
||||||
@ -84,6 +84,16 @@ class NavigationView extends GetView<NavigationViewController> {
|
|||||||
const Icon(Icons.file_open, size: 26.0, color: Colors.blue),
|
const Icon(Icons.file_open, size: 26.0, color: Colors.blue),
|
||||||
),
|
),
|
||||||
),
|
),
|
||||||
|
Padding(
|
||||||
|
padding: const EdgeInsets.only(right: 20.0),
|
||||||
|
child: GestureDetector(
|
||||||
|
onTap: () {
|
||||||
|
controller.ReadPathFromFile();
|
||||||
|
},
|
||||||
|
child: const Icon(Icons.line_weight_rounded,
|
||||||
|
size: 26.0, color: Colors.blue),
|
||||||
|
),
|
||||||
|
),
|
||||||
Padding(
|
Padding(
|
||||||
padding: const EdgeInsets.only(right: 20.0),
|
padding: const EdgeInsets.only(right: 20.0),
|
||||||
child: GestureDetector(
|
child: GestureDetector(
|
||||||
@ -222,10 +232,11 @@ class NavigationView extends GetView<NavigationViewController> {
|
|||||||
? FlutterMap(
|
? FlutterMap(
|
||||||
mapController: controller.mapController,
|
mapController: controller.mapController,
|
||||||
options: MapOptions(
|
options: MapOptions(
|
||||||
center: LatLng(controller.currentLatitude.value,
|
initialCenter: LatLng(
|
||||||
|
controller.currentLatitude.value,
|
||||||
controller.currentLongitude.value),
|
controller.currentLongitude.value),
|
||||||
maxZoom: 25,
|
maxZoom: 25,
|
||||||
zoom: controller.currentZoom.value,
|
initialZoom: controller.currentZoom.value,
|
||||||
),
|
),
|
||||||
children: [
|
children: [
|
||||||
TileLayer(
|
TileLayer(
|
||||||
@ -241,13 +252,14 @@ class NavigationView extends GetView<NavigationViewController> {
|
|||||||
),
|
),
|
||||||
MarkerLayer(
|
MarkerLayer(
|
||||||
markers: controller.currentLocationMarker),
|
markers: controller.currentLocationMarker),
|
||||||
MarkerLayer(markers: controller.parser.markers),
|
// MarkerLayer(markers: controller.parser.markers),
|
||||||
|
PolylineLayer(polylines: controller.pathLayer),
|
||||||
MarkerLayer(
|
MarkerLayer(
|
||||||
markers: controller.pointsToMeasureMarker),
|
markers: controller.pointsToMeasureMarker),
|
||||||
// PolylineLayer(
|
PolylineLayer(
|
||||||
// polylines: controller.parser.polylines),
|
polylines: controller.parser.polylines),
|
||||||
PolyWidgetLayer(
|
PolyWidgetLayer(
|
||||||
polyWidgets: controller.pointsToMeasureLabel)
|
polyWidgets: controller.pointsToMeasureLabel),
|
||||||
],
|
],
|
||||||
)
|
)
|
||||||
: const Center(child: CircularProgressIndicator()),
|
: const Center(child: CircularProgressIndicator()),
|
||||||
|
|||||||
@ -1,13 +1,13 @@
|
|||||||
import 'package:flutter/material.dart';
|
import 'package:flutter/material.dart';
|
||||||
import 'package:flutter/services.dart';
|
import 'package:flutter/services.dart';
|
||||||
import 'package:get/get.dart';
|
import 'package:get/get.dart';
|
||||||
|
import 'package:terepi_seged/pages/map/presentation/controllers/map_controller.dart';
|
||||||
import 'package:terepi_seged/pages/navigation/presentation/controllers/navigation_controller.dart';
|
import 'package:terepi_seged/pages/navigation/presentation/controllers/navigation_controller.dart';
|
||||||
|
|
||||||
class SettingsDialog extends StatelessWidget {
|
class SettingsDialog extends StatelessWidget {
|
||||||
final controller = Get.find<NavigationViewController>();
|
final controller = Get.find<NavigationViewController>();
|
||||||
SettingsDialog({super.key});
|
SettingsDialog({super.key});
|
||||||
|
|
||||||
@override
|
|
||||||
Widget build(BuildContext context) {
|
Widget build(BuildContext context) {
|
||||||
return Scaffold(
|
return Scaffold(
|
||||||
body: ListView(
|
body: ListView(
|
||||||
@ -19,8 +19,24 @@ class SettingsDialog extends StatelessWidget {
|
|||||||
children: [
|
children: [
|
||||||
IconButton(
|
IconButton(
|
||||||
onPressed: () {
|
onPressed: () {
|
||||||
controller.vehicleNumber =
|
if (controller.ntripUsernameController.text.isNotEmpty) {
|
||||||
int.parse(controller.vehicleNumberController.text);
|
controller.ntripUserName.value =
|
||||||
|
controller.ntripUsernameController.text;
|
||||||
|
controller.saveNtripUserName(
|
||||||
|
controller.ntripUsernameController.text);
|
||||||
|
}
|
||||||
|
if (controller.ntripPasswordController.text.isNotEmpty) {
|
||||||
|
controller.ntripPassword.value =
|
||||||
|
controller.ntripPasswordController.text;
|
||||||
|
controller.saveNtripPassword(
|
||||||
|
controller.ntripPasswordController.text);
|
||||||
|
}
|
||||||
|
if (controller.vehicleNumberController.text.isNotEmpty) {
|
||||||
|
controller.vehicleNumber.value =
|
||||||
|
int.parse(controller.vehicleNumberController.text);
|
||||||
|
controller.saveVehicleNumber(
|
||||||
|
int.parse(controller.vehicleNumberController.text));
|
||||||
|
}
|
||||||
Get.back();
|
Get.back();
|
||||||
},
|
},
|
||||||
icon: const Icon(Icons.close)),
|
icon: const Icon(Icons.close)),
|
||||||
@ -28,7 +44,29 @@ class SettingsDialog extends StatelessWidget {
|
|||||||
style: ButtonStyle(
|
style: ButtonStyle(
|
||||||
overlayColor:
|
overlayColor:
|
||||||
MaterialStateProperty.all(Colors.transparent)),
|
MaterialStateProperty.all(Colors.transparent)),
|
||||||
onPressed: () {},
|
onPressed: () {
|
||||||
|
if (controller.ntripUsernameController.text.isNotEmpty) {
|
||||||
|
controller.ntripUserName.value =
|
||||||
|
controller.ntripUsernameController.text;
|
||||||
|
controller.saveNtripUserName(
|
||||||
|
controller.ntripUsernameController.text);
|
||||||
|
}
|
||||||
|
if (controller.ntripPasswordController.text.isNotEmpty) {
|
||||||
|
controller.ntripPassword.value =
|
||||||
|
controller.ntripPasswordController.text;
|
||||||
|
controller.saveNtripPassword(
|
||||||
|
controller.ntripPasswordController.text);
|
||||||
|
}
|
||||||
|
print(
|
||||||
|
"vehicleNumberController: ${controller.vehicleNumberController.text}");
|
||||||
|
if (controller.vehicleNumberController.text.isNotEmpty) {
|
||||||
|
controller.vehicleNumber.value =
|
||||||
|
int.parse(controller.vehicleNumberController.text);
|
||||||
|
controller.saveVehicleNumber(
|
||||||
|
int.parse(controller.vehicleNumberController.text));
|
||||||
|
}
|
||||||
|
Get.back();
|
||||||
|
},
|
||||||
child: const Text(
|
child: const Text(
|
||||||
'Bezár',
|
'Bezár',
|
||||||
style: TextStyle(color: Colors.blue, fontSize: 14.0),
|
style: TextStyle(color: Colors.blue, fontSize: 14.0),
|
||||||
@ -51,12 +89,95 @@ class SettingsDialog extends StatelessWidget {
|
|||||||
),
|
),
|
||||||
),
|
),
|
||||||
Obx(() => Column(children: [
|
Obx(() => Column(children: [
|
||||||
|
RadioListTile(
|
||||||
|
title: Text('TiGNSS Rover-BE6A'),
|
||||||
|
value: '10:06:1C:97:BE:6A',
|
||||||
|
groupValue: controller.gpsAddress.value,
|
||||||
|
onChanged: (value) {
|
||||||
|
controller.gpsAddress.value = value!;
|
||||||
|
controller.gpsName.value = 'TiGNSS Rover-BE6A';
|
||||||
|
controller.saveGpsAddress(value);
|
||||||
|
controller.saveGpsName('TiGNSS Rover-BE6A');
|
||||||
|
}),
|
||||||
|
RadioListTile(
|
||||||
|
title: Text('TiGNSS Rover-1DC6'),
|
||||||
|
value: 'E8:31:CD:16:1D:C6',
|
||||||
|
groupValue: controller.gpsAddress.value,
|
||||||
|
onChanged: (value) {
|
||||||
|
controller.gpsAddress.value = value!;
|
||||||
|
controller.gpsName.value = 'TiGNSS Rover-1DC6';
|
||||||
|
controller.saveGpsAddress(value);
|
||||||
|
controller.saveGpsName('TiGNSS Rover-1DC6');
|
||||||
|
}),
|
||||||
|
RadioListTile(
|
||||||
|
title: Text('TiGNSS Rover-9C3A'),
|
||||||
|
value: '08:3A:8D:14:9C:3A',
|
||||||
|
groupValue: controller.gpsAddress.value,
|
||||||
|
onChanged: (value) {
|
||||||
|
controller.gpsAddress.value = value!;
|
||||||
|
controller.gpsName.value = 'TiGNSS Rover-9C3A';
|
||||||
|
controller.saveGpsAddress(value);
|
||||||
|
controller.saveGpsName('TiGNSS Rover-9C3A');
|
||||||
|
}),
|
||||||
|
RadioListTile(
|
||||||
|
title: Text('TiGNSS Rover-72C2'),
|
||||||
|
value: '10:06:1C:97:72:C2',
|
||||||
|
groupValue: controller.gpsAddress.value,
|
||||||
|
onChanged: (value) {
|
||||||
|
controller.gpsAddress.value = value!;
|
||||||
|
controller.gpsName.value = 'TiGNSS Rover-72C2';
|
||||||
|
controller.saveGpsAddress(value);
|
||||||
|
controller.saveGpsName('TiGNSS Rover-72C2');
|
||||||
|
}),
|
||||||
|
RadioListTile(
|
||||||
|
title: Text('TiGNSS Rover-FE16'),
|
||||||
|
value: '10:06:1C:9F:FE:16',
|
||||||
|
groupValue: controller.gpsAddress.value,
|
||||||
|
onChanged: (value) {
|
||||||
|
controller.gpsAddress.value = value!;
|
||||||
|
controller.gpsName.value = 'TiGNSS Rover-FE16';
|
||||||
|
controller.saveGpsAddress(value);
|
||||||
|
controller.saveGpsName('TiGNSS Rover-FE16');
|
||||||
|
}),
|
||||||
|
RadioListTile(
|
||||||
|
title: Text('TiGNSS Rover-3B0A'),
|
||||||
|
value: '10:C6:1C:9E:3B:0A',
|
||||||
|
groupValue: controller.gpsAddress.value,
|
||||||
|
onChanged: (value) {
|
||||||
|
controller.gpsAddress.value = value!;
|
||||||
|
controller.gpsName.value = 'TiGNSS Rover-3B0A';
|
||||||
|
controller.saveGpsAddress(value);
|
||||||
|
controller.saveGpsName('TiGNSS Rover-3B0A');
|
||||||
|
}),
|
||||||
|
RadioListTile(
|
||||||
|
title: Text('TiGNSS Rover-7FEA'),
|
||||||
|
value: '10:06:1C:9C:7F:EA',
|
||||||
|
groupValue: controller.gpsAddress.value,
|
||||||
|
onChanged: (value) {
|
||||||
|
controller.gpsAddress.value = value!;
|
||||||
|
controller.gpsName.value = 'TiGNSS Rover-7FEA';
|
||||||
|
controller.saveGpsAddress(value);
|
||||||
|
controller.saveGpsName('TiGNSS Rover-7FEA');
|
||||||
|
}),
|
||||||
|
RadioListTile(
|
||||||
|
title: Text('TiGNSS Rover-A39E'),
|
||||||
|
value: '10:06:1C:97:A3:9E',
|
||||||
|
groupValue: controller.gpsAddress.value,
|
||||||
|
onChanged: (value) {
|
||||||
|
controller.gpsAddress.value = value!;
|
||||||
|
controller.gpsName.value = 'TiGNSS Rover-A39E';
|
||||||
|
controller.saveGpsAddress(value);
|
||||||
|
controller.saveGpsName('TiGNSS Rover-A39E');
|
||||||
|
}),
|
||||||
RadioListTile(
|
RadioListTile(
|
||||||
title: Text('TiGNSS Rover-FF4E'),
|
title: Text('TiGNSS Rover-FF4E'),
|
||||||
value: '98:CD:AC:62:FF:4E',
|
value: '98:CD:AC:62:FF:4E',
|
||||||
groupValue: controller.gpsAddress.value,
|
groupValue: controller.gpsAddress.value,
|
||||||
onChanged: (value) {
|
onChanged: (value) {
|
||||||
controller.gpsAddress.value = value!;
|
controller.gpsAddress.value = value!;
|
||||||
|
controller.gpsName.value = 'TiGNSS Rover-FF4E';
|
||||||
|
controller.saveGpsAddress(value);
|
||||||
|
controller.saveGpsName('TiGNSS Rover-FF4E');
|
||||||
}),
|
}),
|
||||||
RadioListTile(
|
RadioListTile(
|
||||||
title: Text('TiGNSS Rover-8BB2'),
|
title: Text('TiGNSS Rover-8BB2'),
|
||||||
@ -64,7 +185,9 @@ class SettingsDialog extends StatelessWidget {
|
|||||||
groupValue: controller.gpsAddress.value,
|
groupValue: controller.gpsAddress.value,
|
||||||
onChanged: (value) {
|
onChanged: (value) {
|
||||||
controller.gpsAddress.value = value!;
|
controller.gpsAddress.value = value!;
|
||||||
;
|
controller.gpsName.value = 'TiGNSS Rover-8BB2';
|
||||||
|
controller.saveGpsAddress(value);
|
||||||
|
controller.saveGpsName('TiGNSS Rover-8BB2');
|
||||||
}),
|
}),
|
||||||
RadioListTile(
|
RadioListTile(
|
||||||
title: Text('TiGNSS Rover-FF36'),
|
title: Text('TiGNSS Rover-FF36'),
|
||||||
@ -72,6 +195,9 @@ class SettingsDialog extends StatelessWidget {
|
|||||||
groupValue: controller.gpsAddress.value,
|
groupValue: controller.gpsAddress.value,
|
||||||
onChanged: (value) {
|
onChanged: (value) {
|
||||||
controller.gpsAddress.value = value!;
|
controller.gpsAddress.value = value!;
|
||||||
|
controller.gpsName.value = 'TiGNSS Rover-FF36';
|
||||||
|
controller.saveGpsAddress(value);
|
||||||
|
controller.saveGpsName('TiGNSS Rover-FF36');
|
||||||
})
|
})
|
||||||
])),
|
])),
|
||||||
const Padding(
|
const Padding(
|
||||||
@ -173,15 +299,7 @@ class SettingsDialog extends StatelessWidget {
|
|||||||
)),
|
)),
|
||||||
),
|
),
|
||||||
),
|
),
|
||||||
|
|
||||||
const SizedBox(height: 5),
|
const SizedBox(height: 5),
|
||||||
const Padding(
|
|
||||||
padding: EdgeInsets.symmetric(horizontal: 20),
|
|
||||||
child: Divider(
|
|
||||||
height: 5,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
const SizedBox(height: 4),
|
|
||||||
const Padding(
|
const Padding(
|
||||||
padding: EdgeInsets.symmetric(horizontal: 20.0),
|
padding: EdgeInsets.symmetric(horizontal: 20.0),
|
||||||
child: Text(
|
child: Text(
|
||||||
|
|||||||
9
lib/pages/tracking/bindings/tracking_bindings.dart
Normal file
9
lib/pages/tracking/bindings/tracking_bindings.dart
Normal file
@ -0,0 +1,9 @@
|
|||||||
|
import 'package:get/get.dart';
|
||||||
|
import 'package:terepi_seged/pages/tracking/presentation/controllers/tracking_controller.dart';
|
||||||
|
|
||||||
|
class TrackingBinding extends Bindings {
|
||||||
|
@override
|
||||||
|
void dependencies() {
|
||||||
|
Get.lazyPut(() => TrackingController());
|
||||||
|
}
|
||||||
|
}
|
||||||
@ -0,0 +1,3 @@
|
|||||||
|
import 'package:get/get.dart';
|
||||||
|
|
||||||
|
class TrackingController extends GetxController {}
|
||||||
13
lib/pages/tracking/presentation/views/tracking_view.dart
Normal file
13
lib/pages/tracking/presentation/views/tracking_view.dart
Normal file
@ -0,0 +1,13 @@
|
|||||||
|
import 'package:get/get.dart';
|
||||||
|
import 'package:flutter/material.dart';
|
||||||
|
|
||||||
|
import '../controllers/tracking_controller.dart';
|
||||||
|
|
||||||
|
class TrackingView extends GetView<TrackingController> {
|
||||||
|
const TrackingView({Key? key}) : super(key: key);
|
||||||
|
|
||||||
|
@override
|
||||||
|
Widget build(BuildContext context) {
|
||||||
|
return Container();
|
||||||
|
}
|
||||||
|
}
|
||||||
@ -1,11 +1,15 @@
|
|||||||
import 'package:get/get.dart';
|
import 'package:get/get.dart';
|
||||||
import 'package:terepi_seged/pages/bleutooth/bindings/bluetooth_bindings.dart';
|
import 'package:terepi_seged/pages/bleutooth/bindings/bluetooth_bindings.dart';
|
||||||
import 'package:terepi_seged/pages/bleutooth/presentation/views/bluetooth_test_view.dart';
|
import 'package:terepi_seged/pages/bleutooth/presentation/views/bluetooth_test_view.dart';
|
||||||
|
import 'package:terepi_seged/pages/field_trip/bindings/field_trip_bindings.dart';
|
||||||
|
import 'package:terepi_seged/pages/field_trip/presentations/views/fiels_trip_view.dart';
|
||||||
import 'package:terepi_seged/pages/home/bindings/home_bindings.dart';
|
import 'package:terepi_seged/pages/home/bindings/home_bindings.dart';
|
||||||
import 'package:terepi_seged/pages/home/presentation/views/home_view.dart';
|
import 'package:terepi_seged/pages/home/presentation/views/home_view.dart';
|
||||||
import 'package:terepi_seged/pages/map/bindings/map_bindings.dart';
|
import 'package:terepi_seged/pages/map/bindings/map_bindings.dart';
|
||||||
import 'package:terepi_seged/pages/map/presentation/views/map_add_point_dialog.dart';
|
import 'package:terepi_seged/pages/map/presentation/views/map_add_point_dialog.dart';
|
||||||
import 'package:terepi_seged/pages/map/presentation/views/map_view.dart';
|
import 'package:terepi_seged/pages/map/presentation/views/map_view.dart';
|
||||||
|
import 'package:terepi_seged/pages/map_survey/bindings/map_survey_bindings.dart';
|
||||||
|
import 'package:terepi_seged/pages/map_survey/presentations/views/mapsurvey_view.dart';
|
||||||
import 'package:terepi_seged/pages/measured_data/bindings/measured_data_bindings.dart';
|
import 'package:terepi_seged/pages/measured_data/bindings/measured_data_bindings.dart';
|
||||||
import 'package:terepi_seged/pages/measured_data/presentation/views/measured_data_view.dart';
|
import 'package:terepi_seged/pages/measured_data/presentation/views/measured_data_view.dart';
|
||||||
import 'package:terepi_seged/pages/navigation/bindings/navigation_bindings.dart';
|
import 'package:terepi_seged/pages/navigation/bindings/navigation_bindings.dart';
|
||||||
@ -18,6 +22,8 @@ import 'package:terepi_seged/pages/socket_test/bindings/socket_test_bindings.dar
|
|||||||
import 'package:terepi_seged/pages/socket_test/presentation/views/socket_test_view.dart';
|
import 'package:terepi_seged/pages/socket_test/presentation/views/socket_test_view.dart';
|
||||||
import 'package:terepi_seged/pages/start/bindings/start_page_bindings.dart';
|
import 'package:terepi_seged/pages/start/bindings/start_page_bindings.dart';
|
||||||
import 'package:terepi_seged/pages/start/presentation/views/start_page.dart';
|
import 'package:terepi_seged/pages/start/presentation/views/start_page.dart';
|
||||||
|
import 'package:terepi_seged/pages/tracking/bindings/tracking_bindings.dart';
|
||||||
|
import 'package:terepi_seged/pages/tracking/presentation/views/tracking_view.dart';
|
||||||
|
|
||||||
import '../pages/map_test/bindings/map_test_bindings.dart';
|
import '../pages/map_test/bindings/map_test_bindings.dart';
|
||||||
import '../pages/map_test/presentation/views/map_test_view.dart';
|
import '../pages/map_test/presentation/views/map_test_view.dart';
|
||||||
@ -73,6 +79,18 @@ class AppPages {
|
|||||||
GetPage(
|
GetPage(
|
||||||
name: Routes.MEASUREDDATA,
|
name: Routes.MEASUREDDATA,
|
||||||
binding: MeasuredDataBinding(),
|
binding: MeasuredDataBinding(),
|
||||||
page: () => const MeasuredDataView())
|
page: () => const MeasuredDataView()),
|
||||||
|
GetPage(
|
||||||
|
name: Routes.MAPSURVEY,
|
||||||
|
binding: MapSurveyBinding(),
|
||||||
|
page: () => const MapSurveyView()),
|
||||||
|
GetPage(
|
||||||
|
name: Routes.FIELDTRIP,
|
||||||
|
binding: FieldTripBinding(),
|
||||||
|
page: () => const FieldTripView()),
|
||||||
|
GetPage(
|
||||||
|
name: Routes.TRACKING,
|
||||||
|
binding: TrackingBinding(),
|
||||||
|
page: () => const TrackingView())
|
||||||
];
|
];
|
||||||
}
|
}
|
||||||
|
|||||||
@ -14,6 +14,9 @@ abstract class Routes {
|
|||||||
static const MAPTEST = '/map_test';
|
static const MAPTEST = '/map_test';
|
||||||
static const PROPERTYLIST = '/property_list';
|
static const PROPERTYLIST = '/property_list';
|
||||||
static const MEASUREDDATA = '/measured_data';
|
static const MEASUREDDATA = '/measured_data';
|
||||||
|
static const MAPSURVEY = '/map_survey';
|
||||||
|
static const FIELDTRIP = '/field_trip';
|
||||||
|
static const TRACKING = '/tracking';
|
||||||
|
|
||||||
static const MAPADDPOINTDIALOG = "/map_add_point_dialog";
|
static const MAPADDPOINTDIALOG = "/map_add_point_dialog";
|
||||||
}
|
}
|
||||||
|
|||||||
73
pubspec.yaml
73
pubspec.yaml
@ -27,43 +27,47 @@ environment:
|
|||||||
# the latest version available on pub.dev. To see which dependencies have newer
|
# the latest version available on pub.dev. To see which dependencies have newer
|
||||||
# versions available, run `flutter pub outdated`.
|
# versions available, run `flutter pub outdated`.
|
||||||
dependencies:
|
dependencies:
|
||||||
cupertino_icons: ^1.0.5
|
cupertino_icons: ^1.0.8
|
||||||
flutter_map: ^6.0.1
|
flutter_map: ^8.2.2
|
||||||
|
flutter_map_polygon_editor: ^0.1.2
|
||||||
flutter_bluetooth_serial: ^0.4.0
|
flutter_bluetooth_serial: ^0.4.0
|
||||||
get: ^4.6.5
|
flutter_map_geojson2: ^1.0.2
|
||||||
latlong2: ^0.9.0
|
get: ^4.7.2
|
||||||
nmea: ^3.2.0
|
latlong2: ^0.9.1
|
||||||
intl: ^0.17.0
|
nmea: ^3.3.2
|
||||||
location: ^4.4.0
|
intl: ^0.20.2
|
||||||
permission_handler: ^10.2.0
|
location: ^8.0.0
|
||||||
flutter_map_location_marker: ^8.0.0
|
permission_handler: ^11.4.0
|
||||||
path_provider: ^2.0.14
|
flutter_map_location_marker: ^10.1.0
|
||||||
rive: ^0.11.3
|
path_provider: ^2.1.5
|
||||||
firebase_core: ^2.23.0
|
rive: ^0.13.20
|
||||||
firebase_auth: ^4.4.0
|
firebase_core: ^3.12.0
|
||||||
cloud_firestore: ^4.13.2
|
firebase_auth: ^5.5.0
|
||||||
firebase_storage: ^11.5.2
|
cloud_firestore: ^5.6.4
|
||||||
google_sign_in: ^6.1.0
|
firebase_storage: ^12.4.3
|
||||||
flutter_secure_storage: ^8.0.0
|
google_sign_in: ^6.2.2
|
||||||
googleapis: ^11.4.0
|
flutter_secure_storage: ^9.2.4
|
||||||
googleapis_auth: ^1.4.1
|
googleapis: ^13.2.0
|
||||||
file_picker: ^6.1.1
|
googleapis_auth: ^1.6.0
|
||||||
|
file_picker: ^9.0.0
|
||||||
# file_picker: ^5.0.0
|
# file_picker: ^5.0.0
|
||||||
sqflite: ^2.2.6
|
sqflite: ^2.4.2
|
||||||
google_fonts: ^4.0.3
|
google_fonts: ^6.2.1
|
||||||
# flutter_iconpicker: ^3.2.2
|
# flutter_iconpicker: ^3.2.2
|
||||||
animated_text_kit: ^4.2.2
|
animated_text_kit: ^4.2.3
|
||||||
package_info_plus: ^4.2.0
|
package_info_plus: ^8.2.1
|
||||||
flutter_map_geojson: ^1.0.1
|
# flutter_map_geojson: ^1.0.8
|
||||||
# geojson: ^1.0.0
|
# geojson: ^1.0.0
|
||||||
syncfusion_flutter_datagrid: ^23.2.4
|
syncfusion_flutter_datagrid: ^28.2.6
|
||||||
syncfusion_flutter_maps: ^23.2.4
|
syncfusion_flutter_maps: ^28.2.6
|
||||||
proj4dart: ^2.1.0
|
proj4dart: ^2.1.0
|
||||||
flutter_map_polywidget: ^1.0.1
|
flutter_map_polywidget: ^1.0.1
|
||||||
connectivity_plus: ^5.0.2
|
connectivity_plus: ^6.1.3
|
||||||
photo_view: ^0.14.0
|
photo_view: ^0.15.0
|
||||||
widget_zoom: ^0.0.3
|
widget_zoom: ^0.0.4
|
||||||
supabase_flutter: ^2.3.4
|
supabase_flutter: ^2.10.2
|
||||||
|
appwrite: ^20.0.0
|
||||||
|
share_plus: ^12.0.1
|
||||||
|
|
||||||
flutter:
|
flutter:
|
||||||
sdk: flutter
|
sdk: flutter
|
||||||
@ -75,11 +79,14 @@ dev_dependencies:
|
|||||||
# activated in the `analysis_options.yaml` file located at the root of your
|
# activated in the `analysis_options.yaml` file located at the root of your
|
||||||
# package. See that file for information about deactivating specific lint
|
# package. See that file for information about deactivating specific lint
|
||||||
# rules and activating additional ones.
|
# rules and activating additional ones.
|
||||||
flutter_native_splash: ^2.2.19
|
flutter_native_splash: ^2.4.4
|
||||||
flutter_lints: ^2.0.0
|
flutter_lints: ^5.0.0
|
||||||
flutter_test:
|
flutter_test:
|
||||||
sdk: flutter
|
sdk: flutter
|
||||||
|
|
||||||
|
dependency_overrides:
|
||||||
|
flutter_web_auth_2: 4.1.0
|
||||||
|
|
||||||
flutter_native_splash:
|
flutter_native_splash:
|
||||||
android: true
|
android: true
|
||||||
ios: true
|
ios: true
|
||||||
|
|||||||
Loading…
Reference in New Issue
Block a user