Android Клиент Fusedlocationprovider не работает с APN Sim, работает с Wifi или обычным inte rnet или Offline - PullRequest
4 голосов
/ 14 января 2020

Android Клиент Fusedlocationprovider не работает с APN Sim (что означает частный inte rnet, не все могут получить доступ), он работает с Wifi или обычным inte rnet (что означает global mobile inte rnet), и я также пытаюсь Диспетчер местоположений (служба воспроизведения не включает эти устройства).

Класс активности

    import android.Manifest;
import android.annotation.SuppressLint;
import android.app.Activity;
import android.app.Dialog;
import android.content.Context;
import android.content.DialogInterface;
import android.content.Intent;
import android.content.pm.PackageManager;
import android.location.Criteria;
import android.location.Location;
//import android.location.LocationListener;
import android.location.LocationManager;

import android.location.LocationProvider;
import android.os.PowerManager;
import android.support.annotation.NonNull;
import android.support.annotation.Nullable;
import android.support.v4.app.ActivityCompat;
import android.support.v7.app.AppCompatActivity;
import android.os.Bundle;
import android.util.Log;
import android.widget.Button;
import android.widget.TextView;
import android.widget.Toast;

import com.google.android.gms.common.GoogleApiAvailability;
import com.google.android.gms.location.FusedLocationProviderClient;
import com.google.android.gms.location.LocationAvailability;
import com.google.android.gms.location.LocationCallback;
import com.google.android.gms.location.LocationListener;
import com.google.android.gms.location.LocationRequest;
import com.google.android.gms.location.LocationResult;
import com.google.android.gms.location.LocationServices;
import com.google.android.gms.tasks.OnCanceledListener;
import com.google.android.gms.tasks.OnCompleteListener;
import com.google.android.gms.tasks.OnFailureListener;
import com.google.android.gms.tasks.OnSuccessListener;
import com.google.android.gms.tasks.SuccessContinuation;
import com.google.android.gms.tasks.Task;


import com.google.android.gms.common.ConnectionResult;
import com.google.android.gms.common.GooglePlayServicesUtil;

import java.util.Locale;

public class LocationActivity extends AppCompatActivity implements LocationListener {


    private FusedLocationProviderClient mFusedLocationClient;

    private double wayLatitude = 0.0, wayLongitude = 0.0;
    private LocationRequest locationRequest;
    private LocationCallback locationCallback;
    private Button btnLocation;
    private TextView txtLocation;
    private Button btnContinueLocation;
    private TextView txtContinueLocation;
    private StringBuilder stringBuilder;
    private boolean isContinue = false;
    private int LOCATION_REQUEST = 1000;

    GPSTracker gps;

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_location);
        this.txtContinueLocation = (TextView) findViewById(R.id.txtContinueLocation);
        this.btnContinueLocation = (Button) findViewById(R.id.btnContinueLocation);
        this.txtLocation = (TextView) findViewById(R.id.txtLocation);
        this.btnLocation = (Button) findViewById(R.id.btnLocation);


        try {


            locationRequest = LocationRequest.create();
            locationRequest.setPriority(LocationRequest.PRIORITY_HIGH_ACCURACY);
            locationRequest.setInterval(10 * 1000); // 10 seconds
            locationRequest.setFastestInterval(5 * 1000); // 5 seconds
            locationCallback = new LocationCallback() {

                @Override
                public void onLocationAvailability(LocationAvailability locationAvailability) {
                    super.onLocationAvailability(locationAvailability);
                    try {
                        if (!locationAvailability.isLocationAvailable()) {
                        GoogleApiAvailability api = GoogleApiAvailability.getInstance();
                        Task<Void> task = api.makeGooglePlayServicesAvailable(LocationActivity.this);
                        task.addOnCompleteListener(LocationActivity.this, new OnCompleteListener<Void>() {
                            @Override
                            public void onComplete(@NonNull Task<Void> task) {
                              //  getLocation();
                              //  mFusedLocationClient.removeLocationUpdates(this);
                                gps = new GPSTracker(LocationActivity.this);

                                // Check if GPS enabled
                                if (gps.canGetLocation()) {

                                    wayLatitude = gps.getLatitude();
                                    wayLongitude = gps.getLongitude();

                                    // \n is for new line
                                    Toast.makeText(getApplicationContext(), "Your Location is - \nLat: " + wayLatitude + "\nLong: " + wayLongitude, Toast.LENGTH_LONG).show();
                                } else {
                                    Toast.makeText(getApplicationContext(), "No location", Toast.LENGTH_LONG).show();
                                }
                            }
                        });

                        task.addOnFailureListener(LocationActivity.this, new OnFailureListener() {
                            @Override
                            public void onFailure(@NonNull Exception e) {
                                Toast.makeText(LocationActivity.this, "Playservice failure", Toast.LENGTH_LONG).show();
                            }
                        });
                    }
                               //init();


                   //     }
                    } catch (SecurityException e) {

                    }
                }

                @Override
                public void onLocationResult(LocationResult locationResult) {
                    if (locationResult == null) {
                        return;
                    }
                    for (Location location : locationResult.getLocations()) {
                        if (location != null) {
                            wayLatitude = location.getLatitude();
                            wayLongitude = location.getLongitude();
//                        if (!isContinue) {
//                            txtLocation.setText(String.format(Locale.US, "%s - %s", wayLatitude, wayLongitude));
//                        } else {
//                            stringBuilder.append(wayLatitude);
//                            stringBuilder.append("-");
//                            stringBuilder.append(wayLongitude);
//                            stringBuilder.append("\n\n");
//                            txtContinueLocation.setText(stringBuilder.toString());
                            Toast.makeText(LocationActivity.this, wayLatitude + " _ " + wayLongitude, Toast.LENGTH_LONG).show();
                            // }
//                        if (mFusedLocationClient != null) {
//                            mFusedLocationClient.removeLocationUpdates(locationCallback);
//                        }
                        }
                    }
                }

            };


            btnLocation.setOnClickListener(v -> {
                // checkGooglePlayServices();

                getLocation();
            });

            btnContinueLocation.setOnClickListener(v -> {
                //  checkGooglePlayServices();
//                PowerManager pm = (PowerManager) getSystemService(Context.POWER_SERVICE);
//                boolean isPowerSaveMode = pm.isPowerSaveMode();
//                Log.e("TAG", "Mode " + isPowerSaveMode);
//
                //  init();
                getLocation();
                //  _getLocation();
            });
            //   getLocation();
            mFusedLocationClient = LocationServices.getFusedLocationProviderClient(this);
        } catch (Exception e) {
            Log.e("TAG", e.getMessage());
        }

    }

    private void getLocation() {
        if (ActivityCompat.checkSelfPermission(this, Manifest.permission.ACCESS_FINE_LOCATION) != PackageManager.PERMISSION_GRANTED
                && ActivityCompat.checkSelfPermission(this, Manifest.permission.ACCESS_COARSE_LOCATION) != PackageManager.PERMISSION_GRANTED) {
            ActivityCompat.requestPermissions(this, new String[]{Manifest.permission.ACCESS_FINE_LOCATION, Manifest.permission.ACCESS_COARSE_LOCATION},
                    LOCATION_REQUEST);

        } else {
            getLatLang();

            // init();
        }
    }

    public void getLatLang() {
        try {

            mFusedLocationClient.requestLocationUpdates(locationRequest, locationCallback, null);
            mFusedLocationClient.getLastLocation().addOnSuccessListener(this, new OnSuccessListener<Location>() {
                @Override
                public void onSuccess(Location location) {

                    if (location != null) {
                        wayLatitude = location.getLatitude();
                        wayLongitude = location.getLongitude();
                        Toast.makeText(LocationActivity.this, wayLatitude + " _ " + wayLongitude, Toast.LENGTH_LONG).show();
                    } else {
                        Toast.makeText(LocationActivity.this, "Location not get Fused client", Toast.LENGTH_LONG).show();
                    }
                }
            });
            mFusedLocationClient.getLastLocation().addOnFailureListener(this, new OnFailureListener() {
                @Override
                public void onFailure(@NonNull Exception e) {
                    wayLatitude = 0.0;
                    wayLongitude = 0.0;
                    Toast.makeText(LocationActivity.this, e.getMessage(), Toast.LENGTH_LONG).show();

                }
            });
            mFusedLocationClient.getLastLocation().addOnCanceledListener(this, new OnCanceledListener() {
                @Override
                public void onCanceled() {
                    Log.e("TAG", "loc cancel");
                }
            });
        } catch (Exception e) {
            Log.e("TAG", "loc exce" + e.getMessage());
        }
    }

    @SuppressLint("MissingPermission")
    @Override
    public void onRequestPermissionsResult(int requestCode, @NonNull String[] permissions, @NonNull int[] grantResults) {
        super.onRequestPermissionsResult(requestCode, permissions, grantResults);
        switch (requestCode) {
            case 1000: {
                // If request is cancelled, the result arrays are empty.
                if (grantResults.length > 0
                        && grantResults[0] == PackageManager.PERMISSION_GRANTED) {

                    getLatLang();
                    //   init();
                } else {
                    Toast.makeText(this, "Permission denied", Toast.LENGTH_SHORT).show();
                }
                break;
            }
        }
    }

    @Override
    protected void onActivityResult(int requestCode, int resultCode, Intent data) {
        super.onActivityResult(requestCode, resultCode, data);
        if (resultCode == Activity.RESULT_OK) {

        }
    }

    @Override
    protected void onDestroy() {
        super.onDestroy();
        //  mFusedLocationClient.removeLocationUpdates(locationCallback);
//        if(locMgr!=null)
//            locMgr.removeUpdates();
    }

    @Override
    public void onLocationChanged(Location location) {
        if (location != null) {
            wayLatitude = location.getLatitude();
            wayLongitude = location.getLongitude();
            Toast.makeText(LocationActivity.this, wayLatitude + " _ " + wayLongitude + "  CHANGE", Toast.LENGTH_LONG).show();
        } else {
            Toast.makeText(LocationActivity.this, "Location not get Fused client chnage locATION", Toast.LENGTH_LONG).show();
        }
    }
    }

ПОМОЩЬ - если местоположение не получится, мы пробуем всех трех провайдеров, если кто-либо из провайдеров дает результат, тогда мы пропустите других провайдеров.

public class GPSTracker extends Service implements LocationListener {
    private final Context mContext;
    // flag for GPS status
    boolean isGPSEnabled = false;

    boolean isPassive = false;
    // flag for network status
    boolean isNetworkEnabled = false;
    // flag for GPS status
    boolean canGetLocation = false;
    Location location; // location
    double latitude; // latitude
    double longitude; // longitude
    // The minimum distance to change Updates in meters
    private static final long MIN_DISTANCE_CHANGE_FOR_UPDATES = 1; // 10 meters
    // The minimum time between updates in milliseconds
    private static final long MIN_TIME_BW_UPDATES = 1000 * 1; // 1 minute
    // Declaring a Location Manager
    protected LocationManager locationManager;

    public GPSTracker(Context context) {
        this.mContext = context;
        getLocation();
    }

    @SuppressLint("MissingPermission")
    public Location getLocation() {
        try {
            locationManager = (LocationManager) mContext.getSystemService(LOCATION_SERVICE);
            // getting GPS status
            isGPSEnabled = locationManager.isProviderEnabled(LocationManager.GPS_PROVIDER);
            // getting network status
            isNetworkEnabled = locationManager
                    .isProviderEnabled(LocationManager.NETWORK_PROVIDER);
            isPassive = locationManager
                    .isProviderEnabled(LocationManager.PASSIVE_PROVIDER);

            if (!isGPSEnabled && !isNetworkEnabled && !isPassive) {
                // no network provider is enabled
            } else {
                this.canGetLocation = true;
                // First get location from Network Provider

                if (isPassive) {
                    // if (location == null) {
                    locationManager.requestLocationUpdates(
                            LocationManager.PASSIVE_PROVIDER,
                            MIN_TIME_BW_UPDATES,
                            MIN_DISTANCE_CHANGE_FOR_UPDATES, this);
                    Toast.makeText(mContext, "Location Try Passive Provider", Toast.LENGTH_SHORT);
                    Log.e("Passive Enabled", "Passive Enabled");
                    if (locationManager != null) {
                        location = locationManager
                                .getLastKnownLocation(LocationManager.PASSIVE_PROVIDER);

                        if (location != null) {
                            latitude = location.getLatitude();
                            longitude = location.getLongitude();
                        } else {
                            Toast.makeText(mContext, "Location Not get Passive Provider", Toast.LENGTH_SHORT);
                        }
                    }
                }
                // }.

                // if GPS Enabled get lat/long using GPS Services
                if (isGPSEnabled) {
                    if (location == null) {
                        locationManager.requestLocationUpdates(
                                LocationManager.GPS_PROVIDER,
                                MIN_TIME_BW_UPDATES,
                                MIN_DISTANCE_CHANGE_FOR_UPDATES, this);
                        Toast.makeText(mContext, "Location Try GPS Provider", Toast.LENGTH_SHORT);
                        Log.e("GPS Enabled", "GPS Enabled");
                        if (locationManager != null) {
                            location = locationManager
                                    .getLastKnownLocation(LocationManager.GPS_PROVIDER);

                            if (location != null) {
                                latitude = location.getLatitude();
                                longitude = location.getLongitude();
                                Log.e("TAG", "GPS " + latitude + " vjhg " + longitude);
                            } else {
                                Toast.makeText(mContext, "Location Not get GPS Provider", Toast.LENGTH_SHORT);
                            }
                        }
                    }
                }

                if (isNetworkEnabled) {
                    if (location == null) {
                        locationManager.requestLocationUpdates(
                                LocationManager.NETWORK_PROVIDER,
                                MIN_TIME_BW_UPDATES,
                                MIN_DISTANCE_CHANGE_FOR_UPDATES, this);

                        Log.e("Network", "Network Enabled");
                        Toast.makeText(mContext, "Location Try Network Provider", Toast.LENGTH_SHORT);
                        if (locationManager != null) {
                            location = locationManager
                                    .getLastKnownLocation(LocationManager.NETWORK_PROVIDER);

                            if (location != null) {
                                latitude = location.getLatitude();
                                longitude = location.getLongitude();
                            } else {
                                Toast.makeText(mContext, "Location Not get Network Provider", Toast.LENGTH_SHORT);
                            }
                        }
                    }
                }


            }

        } catch (Exception e) {
            e.printStackTrace();
        }

        return location;
    }

    /**
     * Stop using GPS listener
     * Calling this function will stop using GPS in your app
     */
    public void stopUsingGPS() {
        if (locationManager != null) {
            locationManager.removeUpdates(GPSTracker.this);
        }
    }

    /**
     * Function to get latitude
     */
    public double getLatitude() {
        if (location != null) {
            latitude = location.getLatitude();
        }
        // return latitude
        return latitude;
    }

    /**
     * Function to get longitude
     */
    public double getLongitude() {
        if (location != null) {
            longitude = location.getLongitude();
        }
        // return longitude
        return longitude;
    }

    /**
     * Function to check GPS/wifi enabled
     *
     * @return boolean
     */
    public boolean canGetLocation() {
        return this.canGetLocation;
    }

    /**
     * Function to show settings alert dialog
     * On pressing Settings button will lauch Settings Options
     */
    public void showSettingsAlert() {
        AlertDialog.Builder alertDialog = new AlertDialog.Builder(mContext);
        // Setting Dialog Title
        alertDialog.setTitle("GPS is settings");
        // Setting Dialog Message
        alertDialog.setMessage("GPS is not enabled. Do you want to go to settings menu?");
        // On pressing Settings button
        alertDialog.setPositiveButton("Settings", new DialogInterface.OnClickListener() {
            public void onClick(DialogInterface dialog, int which) {
                Intent intent = new Intent(Settings.ACTION_LOCATION_SOURCE_SETTINGS);
                mContext.startActivity(intent);
            }
        });

        // on pressing cancel button
        alertDialog.setNegativeButton("Cancel", new DialogInterface.OnClickListener() {
            public void onClick(DialogInterface dialog, int which) {
                dialog.cancel();
            }
        });

        // Showing Alert Message
        alertDialog.show();
    }

    @Override
    public void onLocationChanged(Location location) {
        if (location != null) {
            latitude = location.getLatitude();
            longitude = location.getLongitude();
            Log.e("TAG", "Change " + latitude + " ``` " + longitude);
        }
    }

    @Override
    public void onProviderDisabled(String provider) {
    }

    @Override
    public void onProviderEnabled(String provider) {
    }

    @Override
    public void onStatusChanged(String provider, int status, Bundle extras) {
    }

    @Override
    public IBinder onBind(Intent arg0) {
        return null;
    }
}

Манифест

<uses-permission android:name="android.permission.INTERNET" />
    <uses-permission android:name="android.permission.ACCESS_COARSE_LOCATION" />
    <uses-permission android:name="android.permission.ACCESS_FINE_LOCATION" />
    <uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE" />
    <uses-permission android:name="android.permission.ACCESS_NETWORK_STATE" />
    <uses-permission android:name="android.permission.ACCESS_WIFI_STATE" />
    <uses-permission android:name="android.permission.CHANGE_NETWORK_STATE" />
    <uses-permission android:name="android.permission.CAMERA" />
    <uses-permission android:name="android.permission.READ_EXTERNAL_STORAGE" />

    <!--    <uses-permission android:name="android.permission.ACCESS_" />-->
    <uses-feature
        android:name="android.hardware.location.gps"
        android:required="false" />
    <uses-feature
        android:name="android.hardware.location.network"
        android:required="false" />

    <application
        android:allowBackup="false"
        android:icon="@mipmap/ic_launcher"
        android:label="@string/app_name"
        android:largeHeap="true"
        android:roundIcon="@mipmap/ic_launcher_round"
        android:screenOrientation="portrait"
        android:supportsRtl="true">

Это мой пример кода. Как только я подключил свой Wi-Fi, он работает, но моя точка зрения, мы хотим работать с APN sim.

1 Ответ

3 голосов
/ 20 января 2020

Ответ: Службы определения местоположения Google используют гибридные данные местоположения Google для более быстрого определения местоположения устройства.

От: Требуется ли у поставщика сетевых расположений inte rnet для определить местоположение? и видео Google IO 2019: Плавное и плавное расположение везде с новым FusedLocationProvider , который предоставляет подробную информацию о FusedLocationProvider.

GPS все еще работает, его медленнее получить расположение и работает лучше всего на открытом воздухе. AGPS (Assisted GPS) требуется мобильное соединение для определения местоположения GPS. Если вы хотите расположиться в помещении или в городской среде, вам понадобится внешняя служба, которая использует Гибридное позиционирование , которая использует Wi-Fi, Bluetooth, возможно сотовые вышки? но потребуется хотя бы некоторое частое подключение к внешней службе, поскольку база данных местоположений постоянно обновляется (т. е. меняются точки доступа WiFi, новое строительство может изменить шаблоны радиосвязи), поэтому такая же проблема, как и доступ к службам Google в частной сети.

Какие URL-адреса Google мне нужно добавить в белый список [для Служб Google]?

От: Порты FCM и ваш брандмауэр :

Для исходящих соединения, FCM не предоставляет определенные c IP-адреса, поскольку наш диапазон IP-адресов меняется слишком часто, и ваши правила брандмауэра могут устареть, что может повлиять на работу ваших пользователей. В идеале вы должны занести в белый список порты 5228-5230 без ограничений IP. Однако, если у вас должно быть ограничение IP, вы должны внести в белый список все IP-адреса в блоках IPv4 и IPv6, перечисленных в ASN Google 15169 . Это большой список, и вы должны планировать обновлять свои правила ежемесячно. Проблемы, вызванные ограничениями IP-адресов брандмауэра, часто являются прерывистыми и их трудно диагностировать.

Порты, которые нужно открыть для входящих сообщений:

5228

5229

5230

Порты для разрешения исходящих соединений:

Один из них (вариант № 1 предпочтителен):

  1. Нет IP ограничения
  2. Все IP-адреса, содержащиеся в блоках IP, перечисленных в ASN Google 15169 . Не забывайте обновлять его не реже одного раза в месяц.

Службы Google не являются Firebase Cloud Messaging и находятся в одной и той же постоянно меняющейся глобальной инфраструктуре, поэтому разумное совпадение без указания c документация, в которой указано иное.

Также, учитывая, что Карты Google не являются Службами определения местоположения Google, см. связанные с Google Maps ресурсы в этом вопросе: Использование API-интерфейса Google Maps за брандмауэром

Альтернативы

Другая альтернатива - не использовать Службы Google для вашего поставщика гибридного местоположения. Существуют сторонние поставщики, такие как HERE , которые, похоже, имеют собственную премиальную услугу Advanced Positioning by HERE . Вам нужно будет проверить, какой план подходит для вашего использования , и связаться с ними, чтобы узнать, какие IP-адреса / домены должны быть в белом списке, если они не включены в . Какие IP-адреса мне нужно разрешить? брандмауэр для доступа к geocoder.api.here.com?

Возможно, есть другие провайдеры, такие как Skyhook , чтобы выяснить, соответствуют ли они вашим потребностям. Я бы посмотрел, есть ли у провайдера APN Sim услуга. Или, если все устройства принадлежат одному производителю, проверьте, есть ли у производителя проприетарная (не Google) служба определения местоположения.

Невозможно выполнить ни одно из указанных выше действий

Если компания не в состоянии разрешить вышеуказанные опции, тогда приложение ограничено медленным исправлением GPS, когда GPS доступен. GPS бесплатный, но для ускорения определения местоположения или улучшения городского / внутреннего местоположения требуется какой-либо сторонний сервис, чтобы сообщить устройству, где оно находится.

...