Dynamiczne obliczanie odległości między dwoma punktami. Przebyty dystans. Android Studio

0

Hej, piszę aplikację która obliczy przebyty dystans. Mam kod który wybiera bieżącą lokalizację co 10 sekund ale nie wiem jak zapisać starą lokalizację, aby zaimplementować (
do czterech punktów):

public double getDistance(double lat1, double lon1, double lat2, double lon2) 
{ 
    double latA = Math.toRadians(lat1); 
    double lonA = Math.toRadians(lon1);
    double latB = Math.toRadians(lat2); 
    double lonB = Math.toRadians(lon2);
    double cosAng = (Math.cos(latA) * Math.cos(latB) * Math.cos(lonB-lonA)) +
        (Math.sin(latA) * Math.sin(latB));
    double ang = Math.acos(cosAng);
    double dist = ang *6371;
    return dist;
}

Miałem pomysł żeby co 10 sekund brało lokalizacje i po 5 sekundach w tej samej funkcji brało kolejną wartość i wtedy liczyło dystans z tych czterech, i dodało to do zmiennej "globalnej" distance ale nie mogę zrobić opóźnienia w funkcji na te kilka sekund bo cała funkcja się zatrzymuje i nie zbiera tych pierwszych wartości.
Dopiero uczę się javy, więc przepraszam za moje niezrozumienie. Od razu wybrałem się na "głęboką wodę", żeby zrealizować jeden projekt w mojej głowie.
Kod poniżej:

import androidx.annotation.NonNull;
import androidx.appcompat.app.AppCompatActivity;
import androidx.core.app.ActivityCompat;
import androidx.core.content.ContextCompat;

import android.Manifest;
import android.content.pm.PackageManager;
import android.os.Bundle;
import android.os.Looper;
import android.view.View;
import android.widget.ProgressBar;
import android.widget.TextView;
import android.widget.Toast;

import com.google.android.gms.location.LocationCallback;
import com.google.android.gms.location.LocationRequest;
import com.google.android.gms.location.LocationResult;
import com.google.android.gms.location.LocationServices;

import android.os.Handler;


public class MainActivity extends AppCompatActivity {

    private static final int REQUEST_CODE_LOCATION_PERMISSION = 1;
    private TextView textLatLong, distance;
    private ProgressBar progressBar;


    Handler handler = new Handler();
    Runnable runnable;
    int delay = 10000;

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);

        textLatLong = findViewById(R.id.textLatLong);
        distance = findViewById(R.id.distance);
        progressBar = findViewById(R.id.progressBar);

        findViewById(R.id.buttonGetCurrentLocation).setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View view) {
                if (ContextCompat.checkSelfPermission(
                        getApplicationContext(), Manifest.permission.ACCESS_FINE_LOCATION
                ) != PackageManager.PERMISSION_GRANTED) {
                    ActivityCompat.requestPermissions(
                            MainActivity.this,
                            new String[]{Manifest.permission.ACCESS_FINE_LOCATION},
                            REQUEST_CODE_LOCATION_PERMISSION
                    );
                } else {

                    getCurrentLocation();

                }
            }
        });
    }

    @Override
    public void onRequestPermissionsResult(int requestCode, @NonNull String[] permissions, @NonNull int[] grantResults) {
        super.onRequestPermissionsResult(requestCode, permissions, grantResults);
        if (requestCode == REQUEST_CODE_LOCATION_PERMISSION && grantResults.length > 0) {
            if (grantResults[0] == PackageManager.PERMISSION_GRANTED) {
                getCurrentLocation();
            } else {
                Toast.makeText(this, "Permission denied", Toast.LENGTH_SHORT).show();
            }
        }
    }


    @Override
    protected void onResume() {
        handler.postDelayed(runnable = new Runnable() {
            public void run() {
                handler.postDelayed(runnable, delay);
                getCurrentLocation();
            }
        }, delay);
        super.onResume();
    }

    @Override
    protected void onPause() {
        handler.removeCallbacks(runnable); //stop handler when activity not visible super.onPause();
    }






    private void getCurrentLocation() {

        progressBar.setVisibility(View.VISIBLE);

        LocationRequest locationRequest = new LocationRequest();
        locationRequest.setInterval(10000);
        locationRequest.setFastestInterval(3000);
        locationRequest.setPriority(LocationRequest.PRIORITY_HIGH_ACCURACY);

        if (ActivityCompat.checkSelfPermission(this, Manifest.permission.ACCESS_FINE_LOCATION) != PackageManager.PERMISSION_GRANTED && ActivityCompat.checkSelfPermission(this, Manifest.permission.ACCESS_COARSE_LOCATION) != PackageManager.PERMISSION_GRANTED) {
            // TODO: Consider calling
            //    ActivityCompat#requestPermissions
            // here to request the missing permissions, and then overriding
            //   public void onRequestPermissionsResult(int requestCode, String[] permissions,
            //                                          int[] grantResults)
            // to handle the case where the user grants the permission. See the documentation
            // for ActivityCompat#requestPermissions for more details.
            return;
        }
        LocationServices.getFusedLocationProviderClient(MainActivity.this)
                .requestLocationUpdates(locationRequest, new LocationCallback() {

                    @Override
                    public void onLocationResult(LocationResult locationResult) {
                        super.onLocationResult(locationResult);
                        LocationServices.getFusedLocationProviderClient(MainActivity.this)
                                .removeLocationUpdates(this);
                        if (locationResult != null && locationResult.getLocations().size() > 0) {
                            int latestLocationIndex = locationResult.getLocations().size() - 1;

                            double latitude = locationResult.getLocations().get(latestLocationIndex).getLatitude();
                            double longitude = locationResult.getLocations().get(latestLocationIndex).getLongitude();

                            textLatLong.setText(String.format("Latitude: %s\nLongitude: %s", latitude, longitude));

                        }
                        progressBar.setVisibility(View.GONE);

                    }
                }, Looper.getMainLooper());
    }
3

Nie rób żadnego sleepa ani delaya, po prostu miej w tej klasie pola previousLat, previousLon i jak dostajesz nowy odczyt z GPSu, to wyliczasz co tam sobie chcesz i wpisujesz nowe wartości do tych zmiennych. W ten sposób zawsze (poza pierwszym odczytem) masz wartości poprzednie i wartości aktualne.
Odnosze wrażenie że twój problem polega na tym, ze nie umiesz podstaw programowania, a zabierasz sie za pisanie złożonej aplikacji na trudną platformę jaka jest android.

0

@Shalom: Zgadza się, ogólnie potrzebne mi to jest do aplikacji typu speedometer, bibliotekę z prędkościomierzem pobrałem, gotową z githuba bo nigdzie nie znalazłem jak zrobić to samemu / nauczyć się, skróty do map, latarkę, zrobiłem, w sumie już przez to coś się nauczyłem, jestem na studiach i po technikum. Niby byłem na kursie javy na android studio ale mało co nauczyli. Stwierdziłem że najpierw spróbuje zrealizować pomysł a potem zacznę naukę javy od podstaw powoli, ( w sumie od następnego roku na studiach też mi się pojawi ) i wtedy wkroczę w androida. Niby od d**y strony, ale ten sposób dowiedział bym się co, po co jest i bym miał szerszy wgląd na to wszystko. Miałem w głowie pomysły z tymi klasami ale ciężko mi się to implementuje przez te nowe rzeczy... Bardzo ciężko. Wiem że nic nie wiem. Dzięki w sumie pomogłeś mi tym wpisem bo cały czas kombinowałem nie potrzebnie z tymi opóźnieniami.

0

Udało mi się, faktycznie rozwiązanie było banalnie proste, tak na marginesie, bo nigdzie nie mogę znaleźć, wynik z powyższej funkcji getDistance jest w jakiej jednostce (mile/metry)?

4

Przecież to ty napisałeś/ukradłeś tą funkcje to chyba powinieneś sam wiedzieć :D

Na szczęście to poziom szkoły podstawowej więc można to łatwo wywnioskować:

  1. Stała 6371 to średni promień ziemi w kilometrach
  2. Wzór angle * R to nic innego jak wzór na długość łuku (masz kolo o promieniu R i chcesz wiedzieć jak długi jest łuk wyznaczony przez kąt alfa)

Z tego wynika że to co dostajesz jest w kilometrach

To jest niestety smutne, że ludzie niby uczą się pewnych rzeczy w szkole (np. wyznaczania długości łuku na matematyce), uczą się pewnych faktów (np. promień ziemi na geografii) ale zupełnie nie potrafią tej wiedzy zastosować w praktyce.

1 użytkowników online, w tym zalogowanych: 0, gości: 1