Ensure search highlight position when scheduled runnable starts
Search highlight function includes two steps: Scroll list to target position first, then notifyItemChanged to it. We use a Handler.postDelay to implement this. However, when scheduled runnable starts, the original target position could have changed due to preference list update, calling recyclerview's methods after that will be easy to cause an exception. This CL ensures highlight position every time before calling recyclerView update, which also contribute to origin fix of RecyclerView IllegalArgumentException to a certain extent. Test: atest, also test some search results, and see the correct behavior Fixes: 246411107 Change-Id: Ifa758ce3718b047138079246cdfce99fdf66d5b2
This commit is contained in:
@@ -141,6 +141,8 @@ public class HighlightablePreferenceGroupAdapter extends PreferenceGroupAdapter
|
||||
return;
|
||||
}
|
||||
|
||||
// Highlight request accepted
|
||||
mHighlightRequested = true;
|
||||
// Collapse app bar after 300 milliseconds.
|
||||
if (appBarLayout != null) {
|
||||
root.postDelayed(() -> {
|
||||
@@ -152,17 +154,37 @@ public class HighlightablePreferenceGroupAdapter extends PreferenceGroupAdapter
|
||||
recyclerView.setItemAnimator(null);
|
||||
// Scroll to correct position after 600 milliseconds.
|
||||
root.postDelayed(() -> {
|
||||
mHighlightRequested = true;
|
||||
recyclerView.smoothScrollToPosition(position);
|
||||
mHighlightPosition = position;
|
||||
if (ensureHighlightPosition()) {
|
||||
recyclerView.smoothScrollToPosition(mHighlightPosition);
|
||||
}
|
||||
}, DELAY_HIGHLIGHT_DURATION_MILLIS);
|
||||
|
||||
// Highlight preference after 900 milliseconds.
|
||||
root.postDelayed(() -> {
|
||||
notifyItemChanged(position);
|
||||
if (ensureHighlightPosition()) {
|
||||
notifyItemChanged(mHighlightPosition);
|
||||
}
|
||||
}, DELAY_COLLAPSE_DURATION_MILLIS + DELAY_HIGHLIGHT_DURATION_MILLIS);
|
||||
}
|
||||
|
||||
/**
|
||||
* Make sure we highlight the real-wanted position in case of preference position already
|
||||
* changed when the delay time comes.
|
||||
*/
|
||||
private boolean ensureHighlightPosition() {
|
||||
if (TextUtils.isEmpty(mHighlightKey)) {
|
||||
return false;
|
||||
}
|
||||
final int position = getPreferenceAdapterPosition(mHighlightKey);
|
||||
final boolean allowHighlight = position >= 0;
|
||||
if (allowHighlight && mHighlightPosition != position) {
|
||||
Log.w(TAG, "EnsureHighlight: position has changed since last highlight request");
|
||||
// Make sure RecyclerView always uses latest correct position to avoid exceptions.
|
||||
mHighlightPosition = position;
|
||||
}
|
||||
return allowHighlight;
|
||||
}
|
||||
|
||||
public boolean isHighlightRequested() {
|
||||
return mHighlightRequested;
|
||||
}
|
||||
|
Reference in New Issue
Block a user