最近本身动手实现了一个像网易云鲸云特效的自定义view,这是实现效果。java
主要分为2个方面:git
图片剪切,主要利用ImageView的setOutlineProvider
这个方法,github
roundImage.setOutlineProvider(new ViewOutlineProvider() {
@Override
public void getOutline(View view, Outline outline) {
int width = roundImage.getWidth();
int height = roundImage.getHeight();
outline.setOval(0, 0, width, height);
}
});
roundImage.setClipToOutline(true);
复制代码
而后经过属性动画来旋转canvas
//属性动画让roundImage旋转起来
ObjectAnimator objectAnimator = ObjectAnimator.ofFloat(roundImage, "rotation", 0, 360);
objectAnimator.setDuration(15000);
objectAnimator.setRepeatMode(ValueAnimator.RESTART);
objectAnimator.setInterpolator(new LinearInterpolator());
objectAnimator.setRepeatCount(-1);
objectAnimator.start();
复制代码
关于ObjectAnimator可参考这篇文章bash
主要是在 DiffuseView 这个类下面逐步画圈,而后延时刷新。ide
首先须要在mMaxRadiusWidth
和mMinRadiusWidth
这两个参数来取得画圈半径的最小值和最大值,其中mMaxRadiusWidth
能够在onMeasure
中获取post
在onMeasure中得到宽高动画
mMaxRadiusWidth = Math.min(getMeasuredHeight() / 2, getMeasuredWidth() / 2);
复制代码
ps:注意,使用getWidth
和getHeight
方法可能会出现问题ui
而后用这两个参数表明圆圈半径,和对应透明度
private List<Integer> mAlphas = new ArrayList<>();
private List<Integer> mWidths = new ArrayList<>();
复制代码
可能须要:如何将dp转换成pixels
float twelveDp = TypedValue.applyDimension( TypedValue.COMPLEX_UNIT_DIP, 12,
mContext.getResources().getDisplayMetrics() );
复制代码
而后就在onDraw
中画圆,每画一次更新mAlphas
和mWidths
中的值,同时须要处理好扩散圆的增长和删除
for (int i = 0; i < mAlphas.size(); i++) {
Integer alpha = mAlphas.get(i);
mPaint.setAlpha(alpha);
Integer width = mWidths.get(i);
canvas.drawCircle(getWidth() / 2, getHeight() / 2, width, mPaint);
if (alpha >= 2 && width <= mMaxRadiusWidth) {
mAlphas.set(i, alpha - 2);
mWidths.set(i, width + 1);
} else if (width <= mMinRadiusWidth) {
mWidths.set(i, width + 1);
}
}
复制代码
而后延时刷新
if (mIsDiffuse) {
postInvalidateDelayed(30);
}
复制代码
注意:经过设置mIsDiffuse参数来控制扩散圆动画的开启和关闭
在activity中这样开启动画:
diffuseView.start();
复制代码
这是个人github地址,若是有什么问题和建议,欢迎经过issue提问,同时欢迎star,我会不按期地更新本身写的自定义view在github上
本文还参考了: