RecyclerView не показывает данные из JSON - PullRequest
0 голосов
/ 14 октября 2018

Я пытаюсь отобразить данные, собранные из JSON, предоставляемые AirVisual API, в RecyclerView.Однако я сталкиваюсь с проблемой, когда данные из JSON не будут отображаться в RecyclerView, тогда как, если я сам добавлю к ним данные (через ArrayList), данные будут отображаться в RecyclerView.

Вот данные JSON, которые я анализирую из API:

{
  "status": "success",
  "data": {
    "city": "New York",
    "state": "New York",
    "country": "USA",
    "location": {
      "type": "Point",
      "coordinates": [
        -73.928596,
        40.694401
      ]
    },
    "current": {
      "weather": {
        "ts": "2018-09-12T23:00:00.000Z",
        "hu": 93,
        "ic": "10d",
        "pr": 1024,
        "tp": 23,
        "wd": 102,
        "ws": 2.31
      },
      "pollution": {
        "ts": "2018-09-12T22:00:00.000Z",
        "aqius": 18,
        "mainus": "p2",
        "aqicn": 6,
        "maincn": "p2"
      }
    }
  }
} 

Я подозреваю, что данные из JSON вообще не были добавлены в ArrayList, так как размер журналаArrayList вернул 2, то есть 2 объекта, которые я вручную добавил в список.

Для класса Station для хранения данных из JSON я использовал jsonschema2pojo.org для генерациинеобходимые классы для десериализации Gson (не уверен, что это правильный / оптимальный способ использования данных из JSON).

Вот MainActivity.java

public class MainActivity extends AppCompatActivity {
    private static final String url = "http://api.airvisual.com/v2/city?city=New%20York&state=New%20York&country=USA&key={{API_KEY}}";
    private RecyclerView recyclerView;
    private Adapter adapter;

    private List<Station> stationList;

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

        RecyclerView recyclerView = (RecyclerView) findViewById(R.id.recyclerView);
        recyclerView.setHasFixedSize(true);
        recyclerView.setLayoutManager(new LinearLayoutManager(this));

        stationList = new ArrayList<>();

        loadRecyclerViewData();

        //Information for these objects are in the code
        //but I decided not to include it in here
        //because these are just for testing
        Station station1 = new Station();
        Station station2 = new Station();

        stationList.add(station1);
        stationList.add(station2);

        adapter = new Adapter(this, stationList);

        recyclerView.setAdapter(adapter);

        //Returns 2 in the console
        Log.d("List Size", "" + stationList.size());

Теперь вот loadRecyclerViewData function:

    private void loadRecyclerViewData() {
    RequestQueue requestQueue = Volley.newRequestQueue(this);
    // Request a string response from the provided URL.
    StringRequest stringRequest = new StringRequest(Request.Method.GET, url,
            new Response.Listener<String>() {
                @Override
                public void onResponse(String response) {
                    // Using Gson to turn JSON to Java object of Station
                    GsonBuilder gsonBuilder = new GsonBuilder();
                    Gson gson = gsonBuilder.create();
                    Station station = gson.fromJson(response, Station.class);
                    stationList.add(station);
                    Log.d("API RESPONSE", response);
                }
            }, new Response.ErrorListener() {
        @Override
        public void onErrorResponse(VolleyError error) {
            Log.d("VOLLEY ERROR", error.toString());
        }
    });
// Add the request to the RequestQueue.
    requestQueue.add(stringRequest);
}

Класс адаптера

public class Adapter extends RecyclerView.Adapter<Adapter.ViewHolder> {
    private Context context;
    private List<Station> stationList;

    public Adapter(Context context, List<Station> stationList) {
        this.context = context;
        this.stationList = stationList;
    }

    @Override
    public ViewHolder onCreateViewHolder(ViewGroup parent, int viewType) {
        LayoutInflater layoutInflater = LayoutInflater.from(parent.getContext());
        View view = layoutInflater.inflate(R.layout.layout_listitem, null);
        return new ViewHolder(view);
    }

    @Override
    public void onBindViewHolder(ViewHolder holder, int position) {
        Station station = stationList.get(position);
        holder.textViewTitle.setText(station.getData().getCity());
        List<Double> coordinates = station.getData().getLocation().getCoordinates();
        String listString = coordinates.stream().map(Object::toString)
                .collect(Collectors.joining(", "));
        holder.textViewShortDesc.setText(listString);
        int aqius = station.getData().getCurrent().getPollution().getAqius();
        holder.textViewRating.setText(aqius + "");
        holder.textViewPrice.setText(rankAQIUS(aqius));
//        holder.imageView.setImageDrawable(context.getResources().getDrawable());
    }

    public String rankAQIUS(int aqius) {
        if (aqius >= 0 && aqius <= 50) {
            return "Good";
        } else if (aqius >= 51 && aqius <= 100) {
            return "Moderate";
        } else if (aqius >= 101 && aqius <= 150) {
            return "Unhealthy for Sensitive Groups";
        } else if (aqius >= 151 && aqius <= 200) {
            return "Unhealthy";
        } else if (aqius >= 201 && aqius <= 300) {
            return "Very Unhealthy";
        } else if (aqius >= 301) {
            return "Hazardous";
        }
        return "ERROR";
    }

    @Override
    public int getItemCount() {
        return stationList.size();
    }

    public class ViewHolder extends RecyclerView.ViewHolder {
        ImageView imageView;
        TextView textViewTitle, textViewShortDesc, textViewRating, textViewPrice;

        public ViewHolder(View itemView) {
            super(itemView);

            imageView = itemView.findViewById(R.id.imageView);
            textViewTitle = itemView.findViewById(R.id.textViewTitle);
            textViewShortDesc = itemView.findViewById(R.id.textViewShortDesc);
            textViewRating = itemView.findViewById(R.id.textViewRating);
            textViewPrice = itemView.findViewById(R.id.textViewPrice);

        }
    }
} 

В файле activity_main.xml у меня просто есть виджет RecyclerView

<android.support.v7.widget.RecyclerView
    android:id="@+id/recyclerView"
    android:layout_width="match_parent"
    android:layout_height="match_parent">
</android.support.v7.widget.RecyclerView> 

Наконец, для макета RecyclerView,У меня просто машинаdView in LinearLayout

<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
    android:layout_width="match_parent"
    android:layout_height="wrap_content"
    android:orientation="vertical">

    <android.support.v7.widget.CardView
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:layout_margin="5dp">

        <RelativeLayout
            android:layout_width="match_parent"
            android:layout_height="wrap_content"
            android:padding="8dp">

            <ImageView
                android:id="@+id/imageView"
                android:layout_width="120dp"
                android:layout_height="90dp"
                android:padding="4dp" />

            <TextView
                android:id="@+id/textViewTitle"
                android:layout_width="wrap_content"
                android:layout_height="wrap_content"
                android:layout_marginLeft="5dp"
                android:layout_toRightOf="@id/imageView"
                android:text="City, State, Country"
                android:textAppearance="@style/Base.TextAppearance.AppCompat.Small"
                android:textColor="#000000" />

            <TextView
                android:id="@+id/textViewShortDesc"
                android:layout_width="wrap_content"
                android:layout_height="wrap_content"
                android:layout_below="@id/textViewTitle"
                android:layout_marginLeft="5dp"
                android:layout_marginTop="5dp"
                android:layout_toRightOf="@id/imageView"
                android:text="Longitude, Latitude"
                android:textAppearance="@style/Base.TextAppearance.AppCompat.Small" />

            <TextView
                android:id="@+id/textViewRating"
                android:layout_width="wrap_content"
                android:layout_height="wrap_content"
                android:layout_below="@id/textViewShortDesc"
                android:layout_marginLeft="5dp"
                android:layout_marginTop="5dp"
                android:layout_toRightOf="@id/imageView"
                android:background="@color/colorPrimary"
                android:paddingLeft="15dp"
                android:paddingRight="15dp"
                android:text="AQI"
                android:textAppearance="@style/Base.TextAppearance.AppCompat.Small.Inverse"
                android:textStyle="bold" />

            <TextView
                android:id="@+id/textViewPrice"
                android:layout_width="wrap_content"
                android:layout_height="wrap_content"
                android:layout_below="@id/textViewRating"
                android:layout_marginLeft="5dp"
                android:layout_marginTop="5dp"
                android:layout_toRightOf="@id/imageView"
                android:text="Risk Level"
                android:textAppearance="@style/Base.TextAppearance.AppCompat.Large"
                android:textStyle="bold" />

        </RelativeLayout>

    </android.support.v7.widget.CardView>

</LinearLayout> 

Вот как выглядит приложение в данный момент, поскольку вы можете видеть, что в RecyclerView есть только 2 элемента, которые я добавил вручную.Если бы были добавлены данные из JSON, было бы 3 элемента.

Я пытался создать RecyclerView and Adapter в функции loadRecyclerViewData, но столкнулся с ошибками «Неизвестный класс: адаптер».

Помощь будет принята с благодарностью, спасибо!

Bugged RecyclerView screenshot

1 Ответ

0 голосов
/ 14 октября 2018
  private void loadRecyclerViewData() { RequestQueue requestQueue = Volley.newRequestQueue(this); // Request a string response from the provided URL. StringRequest stringRequest = new StringRequest(Request.Method.GET, url, new Response.Listener<String>() { @Override public void onResponse(String response) { // Using Gson to turn JSON to Java object of Station GsonBuilder gsonBuilder = new GsonBuilder(); Gson gson = gsonBuilder.create(); Station station = gson.fromJson(response, Station.class); stationList.add(station);
adapter.notifyDataSetChanged();
Log.d("API RESPONSE", response); } }, new Response.ErrorListener() { @Override public void onErrorResponse(VolleyError error) { Log.d("VOLLEY ERROR", error.toString()); } }); // Add the request to the RequestQueue. requestQueue.add(stringRequest); }

Вы должны вызвать notifyDataSetChanged () после добавления данных в набор данных.

...