I'm trying to replace items in my RecyclerView when someone searches for a word. A list of GIFs is sent to my callback method, and I'm looping through those objects, and wanting to replace each one in the RecyclerView. However, when I try to do this, I get an IndexOutOfBoundsException if I call ArrayList.set()
in my adapter.
I get why it's happening...the ArrayList in my adapter has zero items in it. My question is why? What am I doing wrong?
My activity code:
package hidden.package;
// Imports hidden
public class GifSearchActivity extends AppCompatActivity {
@BindView(R.id.gifSearchBar)
FloatingSearchView searchGifs;
@BindView(R.id.gifSearchResults)
RecyclerView searchResults;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_gifsearch);
ButterKnife.bind(this);
final GIFResultsAdapter adapter = new GIFResultsAdapter();
final SharedPreferences prefs = PreferenceManager
.getDefaultSharedPreferences(GifSearchActivity.this);
final GIFInterface i = new GIFInterface();
i.setSearchListener(new GIFInterface.GIFSearchListener() {
@Override
public void onSearchResultsRetrieved(final GIFSearchResults gifSearchResults) {
runOnUiThread(new Runnable() {
@Override
public void run() {
Log.d("GAB", "Results Retrieved...");
int count = 0;
for (GIF gif : gifSearchResults.getGIFs()) {
adapter.replaceItem(gif, count);
count++;
}
}
});
}
});
searchResults.setLayoutManager(new GridLayoutManager(getBaseContext(), 1, GridLayoutManager.VERTICAL, false));
searchResults.setItemAnimator(new DefaultItemAnimator());
searchResults.setAdapter(adapter);
searchGifs.setOnQueryChangeListener(new FloatingSearchView.OnQueryChangeListener() {
@Override
public void onSearchTextChanged(String oldQuery, String newQuery) {
i.searchForGif(prefs.getString("token", ""), newQuery);
}
});
}
}
My Adapter code:
package hidden.package;
// Imports hidden
@SuppressWarnings("deprecation")
public class GifResultsAdapter extends RecyclerView.Adapter<GifResultsAdapter.GifsViewHolder> {
private ArrayList<GIF> gifs = new ArrayList<>();
private Context con;
public class GifsViewHolder extends RecyclerView.ViewHolder {
ImageView gif;
ProgressBar gifProgressBar;
public GifsViewHolder(View itemView) {
super(itemView);
gif = (ImageView) itemView.findViewById(R.id.gif);
gifProgressBar = (ProgressBar) itemView.findViewById(R.id.gifProgressBar);
}
}
public void replaceItem(GIF gif, int position) {
gifs.set(position, gif);
notifyItemChanged(position);
}
@Override
public GifsViewHolder onCreateViewHolder(ViewGroup parent, int viewType) {
Log.d("GAB", "ViewHolder Created");
View itemView = LayoutInflater.from(parent.getContext())
.inflate(R.layout.util_gif_row, parent, false);
this.con = parent.getContext();
return new GifsViewHolder(itemView);
}
@Override
public void onBindViewHolder(final GifsViewHolder holder, int position) {
Log.d("GAB", "ViewHolder bound");
final GIF gif = gifs.get(position);
Glide.with(con)
.load(gif.getUrl())
.override(700, 700)
.fitCenter()
.placeholder(new ColorDrawable(AndroidHelper.darkenColor(con.getResources().getColor(R.color.colorAccent), 0.95f)))
.into(holder.gif);
}
@Override
public int getItemCount() {
return gifs.size();
}
}
Edit: Here's the exception...
02-25 00:41:46.371 29205-29205/ai.gab.android E/UncaughtException: java.lang.IndexOutOfBoundsException: Index: 0, Size: 0
at java.util.ArrayList.set(ArrayList.java:427)
at ai.gab.android.ui.adapter.GIFResultsAdapter.replaceItem(GIFResultsAdapter.java:59)
at ai.gab.android.ui.activity.GifSearchActivity$1$1.run(GifSearchActivity.java:68)
at android.os.Handler.handleCallback(Handler.java:751)
at android.os.Handler.dispatchMessage(Handler.java:95)
at android.os.Looper.loop(Looper.java:154)
at android.app.ActivityThread.main(ActivityThread.java:6119)
at java.lang.reflect.Method.invoke(Native Method)
at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:886)
at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:776)
A set of fresh eyes would be greatly appreciated!