2015-11-02 17:40:41 -08:00
|
|
|
/**
|
|
|
|
* Modified version of
|
|
|
|
* https://github.com/AndroidDeveloperLB/LollipopContactsRecyclerViewFastScroller
|
|
|
|
*
|
|
|
|
* Their license:
|
|
|
|
*
|
|
|
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
|
|
|
* you may not use this file except in compliance with the License.
|
|
|
|
* You may obtain a copy of the License at
|
|
|
|
*
|
|
|
|
* http://www.apache.org/licenses/LICENSE-2.0
|
|
|
|
*
|
|
|
|
* Unless required by applicable law or agreed to in writing, software
|
|
|
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
|
|
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
|
|
* See the License for the specific language governing permissions and
|
|
|
|
* limitations under the License.
|
|
|
|
*/
|
|
|
|
package org.thoughtcrime.securesms.components;
|
|
|
|
|
|
|
|
import android.animation.Animator;
|
|
|
|
import android.animation.AnimatorListenerAdapter;
|
|
|
|
import android.animation.ObjectAnimator;
|
|
|
|
import android.annotation.TargetApi;
|
|
|
|
import android.content.Context;
|
|
|
|
import android.os.Build.VERSION;
|
2020-08-19 10:06:26 +10:00
|
|
|
import androidx.annotation.NonNull;
|
|
|
|
import androidx.annotation.Nullable;
|
|
|
|
import androidx.recyclerview.widget.LinearLayoutManager;
|
|
|
|
import androidx.recyclerview.widget.RecyclerView;
|
2015-11-02 17:40:41 -08:00
|
|
|
import android.util.AttributeSet;
|
|
|
|
import android.view.MotionEvent;
|
|
|
|
import android.view.View;
|
|
|
|
import android.view.ViewTreeObserver;
|
|
|
|
import android.widget.LinearLayout;
|
|
|
|
import android.widget.TextView;
|
|
|
|
|
2019-07-24 12:30:23 +10:00
|
|
|
import network.loki.messenger.R;
|
2015-11-02 17:40:41 -08:00
|
|
|
import org.thoughtcrime.securesms.util.Util;
|
|
|
|
import org.thoughtcrime.securesms.util.ViewUtil;
|
|
|
|
|
|
|
|
public class RecyclerViewFastScroller extends LinearLayout {
|
|
|
|
private static final int BUBBLE_ANIMATION_DURATION = 100;
|
|
|
|
private static final int TRACK_SNAP_RANGE = 5;
|
|
|
|
|
2015-11-23 11:33:51 -08:00
|
|
|
private @NonNull TextView bubble;
|
|
|
|
private @NonNull View handle;
|
|
|
|
private @Nullable RecyclerView recyclerView;
|
|
|
|
|
2015-11-02 17:40:41 -08:00
|
|
|
private int height;
|
|
|
|
private ObjectAnimator currentAnimator;
|
|
|
|
|
|
|
|
private final RecyclerView.OnScrollListener onScrollListener = new RecyclerView.OnScrollListener() {
|
|
|
|
@Override
|
2019-05-22 13:51:56 -03:00
|
|
|
public void onScrolled(@NonNull final RecyclerView recyclerView, final int dx, final int dy) {
|
2015-11-23 11:33:51 -08:00
|
|
|
if (handle.isSelected()) return;
|
|
|
|
final int offset = recyclerView.computeVerticalScrollOffset();
|
|
|
|
final int range = recyclerView.computeVerticalScrollRange();
|
|
|
|
final int extent = recyclerView.computeVerticalScrollExtent();
|
|
|
|
final int offsetRange = Math.max(range - extent, 1);
|
|
|
|
setBubbleAndHandlePosition((float) Util.clamp(offset, 0, offsetRange) / offsetRange);
|
2015-11-02 17:40:41 -08:00
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
public interface FastScrollAdapter {
|
|
|
|
CharSequence getBubbleText(int pos);
|
|
|
|
}
|
|
|
|
|
|
|
|
public RecyclerViewFastScroller(final Context context) {
|
|
|
|
this(context, null);
|
|
|
|
}
|
|
|
|
|
|
|
|
public RecyclerViewFastScroller(final Context context, final AttributeSet attrs) {
|
|
|
|
super(context, attrs);
|
|
|
|
setOrientation(HORIZONTAL);
|
|
|
|
setClipChildren(false);
|
2016-05-22 12:12:56 +02:00
|
|
|
setScrollContainer(true);
|
2015-11-02 17:40:41 -08:00
|
|
|
inflate(context, R.layout.recycler_view_fast_scroller, this);
|
|
|
|
bubble = ViewUtil.findById(this, R.id.fastscroller_bubble);
|
|
|
|
handle = ViewUtil.findById(this, R.id.fastscroller_handle);
|
|
|
|
}
|
|
|
|
|
|
|
|
@Override
|
|
|
|
protected void onSizeChanged(int w, int h, int oldw, int oldh) {
|
|
|
|
super.onSizeChanged(w, h, oldw, oldh);
|
|
|
|
height = h;
|
|
|
|
}
|
|
|
|
|
|
|
|
@Override
|
|
|
|
@TargetApi(11)
|
|
|
|
public boolean onTouchEvent(@NonNull MotionEvent event) {
|
|
|
|
final int action = event.getAction();
|
|
|
|
switch (action) {
|
|
|
|
case MotionEvent.ACTION_DOWN:
|
|
|
|
if (event.getX() < ViewUtil.getX(handle) - handle.getPaddingLeft() ||
|
|
|
|
event.getY() < ViewUtil.getY(handle) - handle.getPaddingTop() ||
|
|
|
|
event.getY() > ViewUtil.getY(handle) + handle.getHeight() + handle.getPaddingBottom())
|
|
|
|
{
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
if (currentAnimator != null) {
|
|
|
|
currentAnimator.cancel();
|
|
|
|
}
|
|
|
|
if (bubble.getVisibility() != VISIBLE) {
|
|
|
|
showBubble();
|
|
|
|
}
|
|
|
|
handle.setSelected(true);
|
|
|
|
case MotionEvent.ACTION_MOVE:
|
|
|
|
final float y = event.getY();
|
2015-11-16 16:57:51 -08:00
|
|
|
setBubbleAndHandlePosition(y / height);
|
2015-11-02 17:40:41 -08:00
|
|
|
setRecyclerViewPosition(y);
|
|
|
|
return true;
|
|
|
|
case MotionEvent.ACTION_UP:
|
|
|
|
case MotionEvent.ACTION_CANCEL:
|
|
|
|
handle.setSelected(false);
|
|
|
|
hideBubble();
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
return super.onTouchEvent(event);
|
|
|
|
}
|
|
|
|
|
2015-11-23 11:33:51 -08:00
|
|
|
public void setRecyclerView(final @NonNull RecyclerView recyclerView) {
|
|
|
|
if (this.recyclerView != null) {
|
|
|
|
this.recyclerView.removeOnScrollListener(onScrollListener);
|
|
|
|
}
|
2015-11-02 17:40:41 -08:00
|
|
|
this.recyclerView = recyclerView;
|
|
|
|
recyclerView.addOnScrollListener(onScrollListener);
|
|
|
|
recyclerView.getViewTreeObserver().addOnPreDrawListener(new ViewTreeObserver.OnPreDrawListener() {
|
|
|
|
@Override
|
|
|
|
public boolean onPreDraw() {
|
|
|
|
recyclerView.getViewTreeObserver().removeOnPreDrawListener(this);
|
2015-11-23 11:33:51 -08:00
|
|
|
if (handle.isSelected()) return true;
|
2015-11-02 17:40:41 -08:00
|
|
|
final int verticalScrollOffset = recyclerView.computeVerticalScrollOffset();
|
|
|
|
final int verticalScrollRange = recyclerView.computeVerticalScrollRange();
|
|
|
|
float proportion = (float)verticalScrollOffset / ((float)verticalScrollRange - height);
|
|
|
|
setBubbleAndHandlePosition(height * proportion);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
@Override
|
|
|
|
protected void onDetachedFromWindow() {
|
|
|
|
super.onDetachedFromWindow();
|
|
|
|
if (recyclerView != null)
|
|
|
|
recyclerView.removeOnScrollListener(onScrollListener);
|
|
|
|
}
|
|
|
|
|
|
|
|
private void setRecyclerViewPosition(float y) {
|
|
|
|
if (recyclerView != null) {
|
|
|
|
final int itemCount = recyclerView.getAdapter().getItemCount();
|
|
|
|
float proportion;
|
2015-11-16 16:57:51 -08:00
|
|
|
if (ViewUtil.getY(handle) == 0) {
|
2015-11-02 17:40:41 -08:00
|
|
|
proportion = 0f;
|
2015-11-16 16:57:51 -08:00
|
|
|
} else if (ViewUtil.getY(handle) + handle.getHeight() >= height - TRACK_SNAP_RANGE) {
|
2015-11-02 17:40:41 -08:00
|
|
|
proportion = 1f;
|
2015-11-16 16:57:51 -08:00
|
|
|
} else {
|
|
|
|
proportion = y / (float)height;
|
|
|
|
}
|
|
|
|
|
2015-11-02 17:40:41 -08:00
|
|
|
final int targetPos = Util.clamp((int)(proportion * (float)itemCount), 0, itemCount - 1);
|
|
|
|
((LinearLayoutManager) recyclerView.getLayoutManager()).scrollToPositionWithOffset(targetPos, 0);
|
|
|
|
final CharSequence bubbleText = ((FastScrollAdapter) recyclerView.getAdapter()).getBubbleText(targetPos);
|
2015-11-23 11:33:51 -08:00
|
|
|
bubble.setText(bubbleText);
|
2015-11-02 17:40:41 -08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
private void setBubbleAndHandlePosition(float y) {
|
|
|
|
final int handleHeight = handle.getHeight();
|
|
|
|
final int bubbleHeight = bubble.getHeight();
|
2015-11-16 16:57:51 -08:00
|
|
|
final int handleY = Util.clamp((int)((height - handleHeight) * y), 0, height - handleHeight);
|
|
|
|
ViewUtil.setY(handle, handleY);
|
|
|
|
ViewUtil.setY(bubble, Util.clamp(handleY - bubbleHeight - bubble.getPaddingBottom() + handleHeight,
|
|
|
|
0,
|
|
|
|
height - bubbleHeight));
|
2015-11-02 17:40:41 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
@TargetApi(11)
|
|
|
|
private void showBubble() {
|
|
|
|
bubble.setVisibility(VISIBLE);
|
|
|
|
if (VERSION.SDK_INT >= 11) {
|
|
|
|
if (currentAnimator != null) currentAnimator.cancel();
|
|
|
|
currentAnimator = ObjectAnimator.ofFloat(bubble, "alpha", 0f, 1f).setDuration(BUBBLE_ANIMATION_DURATION);
|
|
|
|
currentAnimator.start();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
@TargetApi(11)
|
|
|
|
private void hideBubble() {
|
|
|
|
if (VERSION.SDK_INT >= 11) {
|
|
|
|
if (currentAnimator != null) currentAnimator.cancel();
|
|
|
|
currentAnimator = ObjectAnimator.ofFloat(bubble, "alpha", 1f, 0f).setDuration(BUBBLE_ANIMATION_DURATION);
|
|
|
|
currentAnimator.addListener(new AnimatorListenerAdapter() {
|
|
|
|
@Override
|
|
|
|
public void onAnimationEnd(Animator animation) {
|
|
|
|
super.onAnimationEnd(animation);
|
|
|
|
bubble.setVisibility(INVISIBLE);
|
|
|
|
currentAnimator = null;
|
|
|
|
}
|
|
|
|
|
|
|
|
@Override
|
|
|
|
public void onAnimationCancel(Animator animation) {
|
|
|
|
super.onAnimationCancel(animation);
|
|
|
|
bubble.setVisibility(INVISIBLE);
|
|
|
|
currentAnimator = null;
|
|
|
|
}
|
|
|
|
});
|
|
|
|
currentAnimator.start();
|
|
|
|
} else {
|
|
|
|
bubble.setVisibility(INVISIBLE);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|