Пользовательский интерфейс обновляется, когда мое приложение теряет фокус и восстанавливает фокус. Странная проблема

Я хочу обновить свой textView в aynctask, но он вообще не обновляется. Я не могу найти свою ошибку.

Вот мой код:

ПРИМЕЧАНИЕ. Пользовательский интерфейс обновляется, когда мое приложение теряет фокус и возвращает его. Странная проблема

public class LocateGeopoint extends AsyncTask<LatLng, String, List<Address>> {

    Geocoder geocoder;
    private TextView tv;

    public LocateGeopoint(View v) {
        this.geocoder = new Geocoder(v.getContext());
        this.tv = (TextView) v.findViewById(R.id.textView1);
    }

    @Override
    protected void onPostExecute(List<Address> addresses) {
        super.onPostExecute(addresses);
        if(addresses!=null){
            String city = addresses.get(0).getAddressLine(0);
            String country = addresses.get(0).getAddressLine(1);
            System.out.println(city+"  "+country);

            tv.setText(city);

        }
    }

    @Override
    protected List<Address> doInBackground(LatLng... params) {
        // TODO Auto-generated method stub
        double lattitude = params[0].latitude;
        double longitude = params[0].longitude;

        List<Address> addresses = null;
        if (lattitude != 0 || longitude != 0) {

            try {
                addresses = geocoder.getFromLocation(lattitude, longitude, 1);
            } catch (IOException e) {
                e.printStackTrace();
            }
        }
        return addresses;
    }

}

XML-файл (раздутый)

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
    android:id="@+id/parent_layout"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    android:orientation="horizontal" >

    <ImageView
        android:id="@+id/imageView1"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:src="@drawable/ic_launcher" />

    <FrameLayout
        android:id="@+id/view_container"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content" >

        <TextView
            android:id="@+id/textView1"
            android:layout_width="wrap_content"
            android:layout_height="wrap_content"
            android:focusable="true"
           android:text="Loading Address.." />

    </FrameLayout>

Вот как я вызываю Asyntask из основного действия

googleMap.setInfoWindowAdapter(new InfoWindowAdapter() {

            @Override
            public View getInfoWindow(Marker arg0) {
                return null;
            }

            @SuppressWarnings("unchecked")
            @Override
            public View getInfoContents(Marker marker) {
                LayoutInflater inflater = (LayoutInflater) MapsDemo.this
                        .getSystemService(Context.LAYOUT_INFLATER_SERVICE);
                View v =  inflater.inflate(R.layout.custom_info, null);
                new LocateGeopoint(v).execute(marker.getPosition());
                return v;
            }
        });

Когда я отлаживаю приложение, оно обновляется, но обычно это не так.


person Gaurav Arora    schedule 01.04.2013    source источник
comment
Вы получили значение города?   -  person Parag Chauhan    schedule 01.04.2013
comment
Может быть, geocoder.getFromLocation(lattitude, longitude, 1); просто возвращает null?   -  person Michał Z.    schedule 03.04.2013
comment
@MichałZ. Он никогда не возвращает ноль. Пожалуйста, внимательно прочитайте заявление. Текст не обновляется. Когда я блокирую экран и снова открываю его, - ТЕПЕРЬ ОБНОВЛЕНО   -  person Gaurav Arora    schedule 03.04.2013
comment
Вы получаете какие-либо сообщения в LogCat?   -  person Shade    schedule 09.04.2013
comment
Нет, я не получаю сообщений о представлении в logcat.   -  person Gaurav Arora    schedule 09.04.2013


Ответы (4)


Добавьте кэш в свой InfoWindowAdapter код и измените его getInfoContents() метод, чтобы проверить его:

    googleMap.setInfoWindowAdapter(new InfoWindowAdapter() {
        LruCache<Marker, View> cache = new LruCache<Marker, View>(8); // added

        // no change to getInfoWindow()

        @SuppressWarnings("unchecked")
        @Override
        public View getInfoContents(Marker marker) {
            View v = cache.get(marker);
            if (v == null) {
                LayoutInflater inflater = (LayoutInflater) MapsDemo.this
                        .getSystemService(Context.LAYOUT_INFLATER_SERVICE);
                v =  inflater.inflate(R.layout.custom_info, null);
                new LocateGeopoint(v, marker).execute(marker.getPosition());
                cache.put(marker, v);
            }
            return v;
        }
    });

Измените код в своем классе LocateGeopoint, добавив ссылку на Marker и вызвав showInfoWindow() после обновления вашего TextView.

public class LocateGeopoint extends AsyncTask<LatLng, String, List<Address>> {

    // :
    Marker marker; // added

    public LocateGeopoint(View v, Marker marker) {
        // :
        this.marker = marker; // added
    }

    @Override
    protected void onPostExecute(List<Address> addresses) {
        super.onPostExecute(addresses);
        if(addresses!=null){
            // :
            tv.setText(city);
            marker.showInfoWindow(); // added
        }
    }

    // no change to doInBackground()
}

ПРИМЕЧАНИЕ. Этот ответ на самом деле является просто быстрой и грязной реализацией (я на самом деле не тестировал его, но я думаю, что он должен работать), основанный на ответе MaciejGórski на этот вопрос, пожалуйста, прочитайте упомянутый пост в этом ответе для более подробной информации: )

person Joe    schedule 09.04.2013
comment
Это хорошая реализация, и она сработала для меня. Спасибо @Joe - person Gaurav Arora; 19.04.2013
comment
Но одна проблема, с которой я сталкиваюсь, заключается в том, что я получаю предыдущие значения для всех кликов... кеш не очищается - person Gaurav Arora; 19.04.2013

Может быть, попробуйте вызвать tv.invalidate(), надеюсь, это сработает.

если это не сработало

попробуйте создать переменную вне onCreate, например текст TextView; а затем внутри onCreate поместите: text = (TextView) findViewById(R.id.textView2);

а затем просто поместите text.setText(city); внутри метода onPostExecute.

Посмотрите, работает ли это.

person flexdroid    schedule 03.04.2013
comment
Моя асинтаска — это отдельный класс, а класс активности — отдельный. - person Gaurav Arora; 03.04.2013
comment
Добавьте ur Asynctask в сам класс активности. - person flexdroid; 03.04.2013
comment
Почему я должен добавить сам класс активности? - person Gaurav Arora; 03.04.2013
comment
Чтобы получить доступ к экземпляру TextView в самом классе активности и установить текст, как я указал в своем ответе - person flexdroid; 03.04.2013
comment
Разве мы не можем использовать вот так? Вы первый раз смотрите? - person Gaurav Arora; 03.04.2013
comment
Извините, я не понял вас .. И могу ли я узнать, какую версию Android SDK вы используете? - person flexdroid; 03.04.2013
comment
давайте продолжим это обсуждение в чате - person flexdroid; 04.04.2013

Как указано в документации, возвращаемый вид не является просмотром в реальном времени и не может быть обновлен после возврата из getInfoContents.

Попробуйте выполнить этот ответ: Динамическое содержимое в Maps V2 InfoWindow

person MaciejGórski    schedule 03.04.2013

Попробуй это:

runOnUiThread(new Runnable() {
                public void run() { 

                        try {
                            Thread.sleep(25);
                        } catch (InterruptedException e) {
                            e.printStackTrace();
                        }
                        tv.setText(city);
                }
            });

При ближайшем рассмотрении вашего кода:

@Override
protected void onPostExecute(List<Address> addresses) {
    super.onPostExecute(addresses);
    if(addresses!=null){
        String city = addresses.get(0).getAddressLine(0);
        String country = addresses.get(0).getAddressLine(1);
        System.out.println(city+"  "+country);

        tv.setText(city);

    }
}

Вы переопределили метод onPostExecute(), зачем называть его суперфункцией? Пожалуйста, удалите super.onPostExecute(addresses);.

person Karthik Balakrishnan    schedule 09.04.2013