Обновление данных о местоположении на экране Android

Я хочу обновить данные о местоположении на экране Android. У меня есть LocationListener и LocationManager в LiczydloGPS.class:

package com.zoltrix.gpsallinfo;

import android.app.AlertDialog;
import android.app.Service;
import android.content.Context;
import android.content.DialogInterface;
import android.content.Intent;
import android.location.Location;
import android.location.LocationListener;
import android.location.LocationManager;
import android.os.Bundle;
import android.os.IBinder;
import android.provider.Settings;
import android.util.Log;

public class LiczydloGPS extends Service implements LocationListener {

    private final Context mContext;

    // flag for GPS status
    boolean isGPSEnabled = false;

    // flag for network status
    boolean isNetworkEnabled = false;

    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 = 10; // 10 meters

    // The minimum time between updates in milliseconds
    private static final long MIN_TIME_BW_UPDATES = 1000 * 60 * 1; // 1 minute

    // Declaring a Location Manager
    protected LocationManager locationManager;

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

    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);

            if (!isGPSEnabled && !isNetworkEnabled) {
                // no network provider is enabled
            } else {
                this.canGetLocation = true;
                // First get location from Network Provider
                if (isNetworkEnabled) {
                    locationManager.requestLocationUpdates(
                            LocationManager.NETWORK_PROVIDER,
                            MIN_TIME_BW_UPDATES,
                            MIN_DISTANCE_CHANGE_FOR_UPDATES, this);
                    Log.d("Network", "Network");
                    if (locationManager != null) {
                        location = locationManager
                                .getLastKnownLocation(LocationManager.NETWORK_PROVIDER);
                        if (location != null) {
                            latitude = location.getLatitude();
                            longitude = location.getLongitude();
                        }
                    }
                }
                // 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);
                        Log.d("GPS Enabled", "GPS Enabled");
                        if (locationManager != null) {
                            location = locationManager
                                    .getLastKnownLocation(LocationManager.GPS_PROVIDER);
                            if (location != null) {
                                latitude = location.getLatitude();
                                longitude = location.getLongitude();
                            }
                        }
                    }
                }
            }

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

        return location;
    }

    @Override
    public void onLocationChanged(Location location) {

        if(location!=null)
        {
        MainActivity main2 = new MainActivity();

        main2.update(location.getLatitude(), location.getLongitude());
        }
        Log.d("onLocationChanged", "idze do maina");
    }

    @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;
    }

    /**
     * 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 if best network provider
     * 
     * @return boolean
     * */
    public boolean canGetLocation() {
        return this.canGetLocation;
    }

    /**
     * Function to show settings alert dialog
     * */
    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?");

        // Setting Icon to Dialog
        // alertDialog.setIcon(R.drawable.delete);

        // 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();
    }
    /**
     * Stop using GPS listener
     * Calling this function will stop using GPS in your app
     * */
    public void stopUsingGPS(){
        if(locationManager != null){
            locationManager.removeUpdates(LiczydloGPS.this);
        }       
    }
}

В MainActivity я сделал метод update, который вызывается в LiczydloGPS.class в onLocationChanged:

    package com.zoltrix.gpsallinfo;

import com.zoltrix.gpsallinfo.R.id;

import android.os.Bundle;
import android.app.Activity;
import android.content.Intent;
import android.util.Log;
import android.view.Menu;
import android.view.MenuItem;
import android.widget.TextView;
import android.widget.Toast;

public class MainActivity extends Activity {

    double lat, lon;
    TextView latData, longData;

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        latData = (TextView) findViewById(R.id.latData);
        longData = (TextView) findViewById(R.id.longData);

        LiczydloGPS gps = new LiczydloGPS(this);

        if (gps.canGetLocation() != true) {
            gps.showSettingsAlert();

        } else {

            lat = gps.getLatitude(); // returns latitude
            lon = gps.getLongitude();
            latData.setText(String.valueOf(lat));
            longData.setText(String.valueOf(lon));

        }

    }

    @Override
    public boolean onCreateOptionsMenu(Menu menu) {
        // Inflate the menu; this adds items to the action bar if it is present.
        getMenuInflater().inflate(R.menu.main, menu);

        return true;
    }

    @Override
    public boolean onOptionsItemSelected(MenuItem item) {
        // TODO Auto-generated method stub
        switch (item.getItemId()) {

        case id.oprogramie:

            Intent intent = new Intent(this, Oprogramie.class);
            startActivity(intent);
        case id.action_settings:
            Intent intent2 = new Intent(this, Ustawienia.class);
            startActivity(intent2);
        }

        return super.onOptionsItemSelected(item);
    }

    public void update(double lat1, double lon1) {
        Log.d("onLocationChanged", "jest w mainie!");
         latData.setText(lat1);
         longData.setText(lon1);

        Log.d("onLocationChanged", String.valueOf(lat1));
        //Toast.makeText(getApplicationContext(), "TOAST!!!!!!!!!!!!!!!", Toast.LENGTH_SHORT).show();

    }



}

Но когда вы хотите изменить данные GPS в DDMS, приложение убивается. ЖУРНАЛЫ:

11-11 09:22:27.626: D/GPS включен(2734): GPS включен 11-11 09:22:28.136: D/gralloc_goldfish(2734): обнаружен эмулятор без эмуляции графического процессора. 11-11 09:22:32.506: D/onLocationChanged(2734): шутка, мэни! 11-11 09:22:32.506: D/AndroidRuntime(2734): завершение работы виртуальной машины 11-11 09:22:32.506: W/dalvikvm(2734): threadid=1: поток завершается с необработанным исключением (group=0xb3aa2b90) 11 -11 09:22:32.516: E/AndroidRuntime(2734): НЕИСПРАВНОЕ ИСКЛЮЧЕНИЕ: main 11-11 09:22:32.516: E/AndroidRuntime(2734): Процесс: com.zoltrix.gpsallinfo, PID: 2734 11-11 09 :22:32.516: E/AndroidRuntime(2734): java.lang.NullPointerException 11-11 09:22:32.516: E/AndroidRuntime(2734): в com.zoltrix.gpsallinfo.MainActivity.update(MainActivity.java:69) 11-11 09:22:32.516: E/AndroidRuntime(2734): в com.zoltrix.gpsallinfo.LiczydloGPS.onLocationChanged(LiczydloGPS.java:113) 11-11 09:22:32.516: E/AndroidRuntime(2734): в android.location.LocationManager$ListenerTransport._handleMessage(LocationManager.java:279) 11-11 09:22:32.516: E/AndroidRuntime(2734): в android.location.LocationManager$ListenerTransport.access$000(LocationManager.java:208) 11-11 09:22:32.516: E/AndroidRuntime(2734): в идр oid.location.LocationManager$ListenerTransport$1.handleMessage(LocationManager.java:224) 11-11 09:22:32.516: E/AndroidRuntime(2734): в android.os.Handler.dispatchMessage(Handler.java:102) 11- 11 09:22:32.516: E/AndroidRuntime(2734): на android.os.Looper.loop(Looper.java:137) 11-11 09:22:32.516: E/AndroidRuntime(2734): на android.app. ActivityThread.main(ActivityThread.java:4998) 11-11 09:22:32.516: E/AndroidRuntime(2734): в java.lang.reflect.Method.invokeNative(собственный метод) 11-11 09:22:32.516: E /AndroidRuntime(2734): в java.lang.reflect.Method.invoke(Method.java:515) 11-11 09:22:32.516: E/AndroidRuntime(2734): в com.android.internal.os.ZygoteInit$ MethodAndArgsCaller.run(ZygoteInit.java:777) 11-11 09:22:32.516: E/AndroidRuntime(2734): в com.android.internal.os.ZygoteInit.main(ZygoteInit.java:593) 11-11 09: 22:32.516: E/AndroidRuntime(2734): в dalvik.system.NativeStart.main(собственный метод)

Заранее благодарим за помощь


person Bart    schedule 11.11.2013    source источник


Ответы (2)


Вам нужно получить свой lat и long insdie onLocationChanged() примерно так.

@Override
public void onLocationChanged(Location location) {
    if(location!=null)
    {
update(location.getLatitude(), location.getLongitude());
    }
}
person user755    schedule 11.11.2013
comment
У меня до сих пор NullPointerException :( , Обнаружил, что все идет хорошо, пока не обновится () изменить TextView latData longData . Почему я не могу изменить значение TextView в методе update ()? - person Bart; 11.11.2013
comment
Пробовал добавить простой тост в методе update(), и то уже выскакивала ошибка. - person Bart; 11.11.2013
comment
Вы уверены, что можете создать экземпляр активности? - person user755; 11.11.2013
comment
Я добавляю только Log.d(onLocationChanged, String.valueOf(lat1)); обновить(). и lat1 показывает только при запуске, когда я меняю данные GPS в DDMS, в логе не отображается lat1 и никаких ошибок. :/ - person Bart; 11.11.2013
comment
Я не вижу, где вы вызываете update() в своей деятельности, код в optionmenu(..) должен быть внутри oncreate().. Я понятия не имею, почему вы выбираете этот подход. - person user755; 11.11.2013
comment
Я вызываю update() в onLocationChanged и исправляю ошибку с oncreate() ранее. Обновляю основной пост. - person Bart; 11.11.2013
comment
попробуйте Log.d(..) внутри onLocationChanged, если вам это удалось, используйте намерение для передачи широты и долготы. Я не думаю, что вы могли бы создать экземпляр активности Android в службе. а также вы вызываете службу как вызов функции, что является плохой практикой. - person user755; 11.11.2013
comment
В таком случае, должен ли я делать все в одном классе? Нет ли способа, чтобы менеджер местоположений находился в другом классе и отправлял текущие данные GPS в TextView в MainActivity? - person Bart; 11.11.2013
comment
Нет, все, что вам нужно для реализации шаблона наблюдателя... это очень просто. погуглите много инфы найдете.. - person user755; 11.11.2013
comment
Спасибо за вашу помощь. Реализация шаблона наблюдателя займет у меня некоторое время. - person Bart; 11.11.2013
comment
Вот реализация интерфейса шаблона наблюдателя. Надеюсь, это сэкономит время :) tab-top" title="как вернуть данные в android locationlistener"> stackoverflow.com/questions/8508828/ - person user755; 11.11.2013

Вы получаете NullPointerException, потому что вы не инициализируете свой TextView в методе onCreate(). Вы инициализируете их на onCreateOptionsMenu(), что неверно,

Измените свой код, как показано ниже,

@Override
    protected void onCreate(Bundle savedInstanceState) 
    {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        latData = (TextView) findViewById(R.id.latData);
        longData = (TextView) findViewById(R.id.longData);
    }
person Vigbyor    schedule 11.11.2013
comment
Я исправил эту глупую ошибку, но у меня все еще есть NullPointerException :( , моя новая основная активность: pastebin.com/ydjCZ6Pz - person Bart; 11.11.2013
comment
Обнаружил, что все идет нормально до тех пор, пока не произойдет обновление () изменение TextView latData longData . Почему я не могу изменить значение TextView в методе update()? - person Bart; 11.11.2013