最近本身動手實現了一個像網易雲鯨雲特效的自定義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上
本文還參考了: