blob: d0ec9d7e089ee1af0de9a1beee5820914c077b31 [file] [log] [blame]
/*
* Copyright (C) 2018 The Android Open Source Project
*
* 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 com.android.launcher3.views;
import static androidx.dynamicanimation.animation.SpringForce.DAMPING_RATIO_MEDIUM_BOUNCY;
import static androidx.dynamicanimation.animation.SpringForce.STIFFNESS_LOW;
import static androidx.dynamicanimation.animation.SpringForce.STIFFNESS_MEDIUM;
import android.content.Context;
import android.graphics.Canvas;
import android.util.AttributeSet;
import android.util.SparseBooleanArray;
import android.view.View;
import android.widget.EdgeEffect;
import android.widget.RelativeLayout;
import androidx.annotation.NonNull;
import androidx.dynamicanimation.animation.DynamicAnimation;
import androidx.dynamicanimation.animation.FloatPropertyCompat;
import androidx.dynamicanimation.animation.SpringAnimation;
import androidx.dynamicanimation.animation.SpringForce;
import androidx.recyclerview.widget.RecyclerView;
import androidx.recyclerview.widget.RecyclerView.EdgeEffectFactory;
public class SpringRelativeLayout extends RelativeLayout {
private static final float STIFFNESS = (STIFFNESS_MEDIUM + STIFFNESS_LOW) / 2;
private static final float DAMPING_RATIO = DAMPING_RATIO_MEDIUM_BOUNCY;
private static final float VELOCITY_MULTIPLIER = 0.3f;
private static final FloatPropertyCompat<SpringRelativeLayout> DAMPED_SCROLL =
new FloatPropertyCompat<SpringRelativeLayout>("value") {
@Override
public float getValue(SpringRelativeLayout object) {
return object.mDampedScrollShift;
}
@Override
public void setValue(SpringRelativeLayout object, float value) {
object.setDampedScrollShift(value);
}
};
protected final SparseBooleanArray mSpringViews = new SparseBooleanArray();
private final SpringAnimation mSpring;
private float mDampedScrollShift = 0;
private SpringEdgeEffect mActiveEdge;
public SpringRelativeLayout(Context context) {
this(context, null);
}
public SpringRelativeLayout(Context context, AttributeSet attrs) {
this(context, attrs, 0);
}
public SpringRelativeLayout(Context context, AttributeSet attrs, int defStyleAttr) {
super(context, attrs, defStyleAttr);
mSpring = new SpringAnimation(this, DAMPED_SCROLL, 0);
mSpring.setSpring(new SpringForce(0)
.setStiffness(STIFFNESS)
.setDampingRatio(DAMPING_RATIO));
}
public void addSpringView(int id) {
mSpringViews.put(id, true);
}
public void removeSpringView(int id) {
mSpringViews.delete(id);
invalidate();
}
/**
* Used to clip the canvas when drawing child views during overscroll.
*/
public int getCanvasClipTopForOverscroll() {
return 0;
}
@Override
protected boolean drawChild(Canvas canvas, View child, long drawingTime) {
if (mDampedScrollShift != 0 && mSpringViews.get(child.getId())) {
int saveCount = canvas.save();
canvas.clipRect(0, getCanvasClipTopForOverscroll(), getWidth(), getHeight());
canvas.translate(0, mDampedScrollShift);
boolean result = super.drawChild(canvas, child, drawingTime);
canvas.restoreToCount(saveCount);
return result;
}
return super.drawChild(canvas, child, drawingTime);
}
private void setActiveEdge(SpringEdgeEffect edge) {
if (mActiveEdge != edge && mActiveEdge != null) {
mActiveEdge.mDistance = 0;
}
mActiveEdge = edge;
}
protected void setDampedScrollShift(float shift) {
if (shift != mDampedScrollShift) {
mDampedScrollShift = shift;
invalidate();
}
}
private void finishScrollWithVelocity(float velocity) {
mSpring.setStartVelocity(velocity);
mSpring.setStartValue(mDampedScrollShift);
mSpring.start();
}
protected void finishWithShiftAndVelocity(float shift, float velocity,
DynamicAnimation.OnAnimationEndListener listener) {
setDampedScrollShift(shift);
mSpring.addEndListener(listener);
finishScrollWithVelocity(velocity);
}
public EdgeEffectFactory createEdgeEffectFactory() {
return new SpringEdgeEffectFactory();
}
private class SpringEdgeEffectFactory extends EdgeEffectFactory {
@NonNull @Override
protected EdgeEffect createEdgeEffect(RecyclerView view, int direction) {
switch (direction) {
case DIRECTION_TOP:
return new SpringEdgeEffect(getContext(), +VELOCITY_MULTIPLIER);
case DIRECTION_BOTTOM:
return new SpringEdgeEffect(getContext(), -VELOCITY_MULTIPLIER);
}
return super.createEdgeEffect(view, direction);
}
}
private class SpringEdgeEffect extends EdgeEffect {
private final float mVelocityMultiplier;
private float mDistance;
public SpringEdgeEffect(Context context, float velocityMultiplier) {
super(context);
mVelocityMultiplier = velocityMultiplier;
}
@Override
public boolean draw(Canvas canvas) {
return false;
}
@Override
public void onAbsorb(int velocity) {
finishScrollWithVelocity(velocity * mVelocityMultiplier);
}
@Override
public void onPull(float deltaDistance, float displacement) {
setActiveEdge(this);
mDistance += deltaDistance * (mVelocityMultiplier / 3f);
setDampedScrollShift(mDistance * getHeight());
}
@Override
public void onRelease() {
mDistance = 0;
finishScrollWithVelocity(0);
}
}
}