android仿微信表情雨下落效果的實(shí)現(xiàn)方法
前言
眾所周知,微信聊天中我們輸入一些關(guān)鍵詞會(huì)有表情雨下落,比如輸入「生日快樂(lè)」「么么噠」會(huì)有相應(yīng)的蛋糕、親吻的表情雨下落,今天就來(lái)完成這個(gè)表情雨下落的效果。下面話不多說(shuō)了,來(lái)一起看看詳細(xì)的介紹吧
效果圖
先來(lái)看下效果,真·狗頭雨·落!

實(shí)現(xiàn)代碼
確認(rèn)表情的模型,定義屬性
public class ItemEmoje {
//坐標(biāo)
public int x;
public int y;
// 橫向偏移
public int offsetX;
//縱向偏移
public int offsetY;
//縮放
public float scale;
//圖片資源
public Bitmap bitmap;
}
自定義RainView 表情下落視圖,初始化變量。
public class RainView extends View {
private Paint paint;
//圖片處理
private Matrix matrix;
private Random random;
//判斷是否運(yùn)行的,默認(rèn)沒(méi)有
private boolean isRun;
//表情包集合
private List<ItemEmoje> bitmapList;
//表情圖片
private int imgResId = R.mipmap.dog;
public RainView(Context context) {
this(context, null);
}
public RainView(Context context, @Nullable AttributeSet attrs) {
this(context, attrs, 0);
}
public RainView(Context context, @Nullable AttributeSet attrs, int defStyleAttr) {
super(context, attrs, defStyleAttr);
init();
}
private void init() {
paint = new Paint();
paint.setAntiAlias(true);
paint.setFilterBitmap(true);
paint.setDither(true);
matrix = new Matrix();
random = new Random();
bitmapList = new ArrayList<>();
}
}
初始化表情雨數(shù)據(jù),確認(rèn)每個(gè)表情的起始位置,下落過(guò)程中橫向、縱向的偏移,以及縮放大小。
private void initData() {
for (int i = 0; i < 20; i++) {
ItemEmoje itemEmoje = new ItemEmoje();
itemEmoje.bitmap = BitmapFactory.decodeResource(getResources(), imgResId);
//起始橫坐標(biāo)在[100,getWidth()-100) 之間
itemEmoje.x = random.nextInt(getWidth() - 200) + 100;
//起始縱坐標(biāo)在(-getHeight(),0] 之間,即一開(kāi)始位于屏幕上方以外
itemEmoje.y = -random.nextInt(getHeight());
//橫向偏移[-2,2) ,即左右搖擺區(qū)間
itemEmoje.offsetX = random.nextInt(4) - 2;
//縱向固定下落12
itemEmoje.offsetY = 12;
//縮放比例[0.8,1.2) 之間
itemEmoje.scale = (float) (random.nextInt(40) + 80) / 100f;
bitmapList.add(itemEmoje);
}
}
下落過(guò)程通過(guò) onDraw進(jìn)行繪制,不斷的計(jì)算橫縱坐標(biāo),達(dá)到下落效果。
@Override
protected void onDraw(Canvas canvas) {
super.onDraw(canvas);
if (isRun) {
//用于判斷表情下落結(jié)束,結(jié)束即不再進(jìn)行重繪
boolean isInScreen = false;
for (int i = 0; i < bitmapList.size(); i++) {
matrix.reset();
//縮放
matrix.setScale(bitmapList.get(i).scale, bitmapList.get(i).scale);
//下落過(guò)程坐標(biāo)
bitmapList.get(i).x = bitmapList.get(i).x + bitmapList.get(i).offsetX;
bitmapList.get(i).y = bitmapList.get(i).y + bitmapList.get(i).offsetY;
if (bitmapList.get(i).y <= getHeight()) {//當(dāng)表情仍在視圖內(nèi),則繼續(xù)重繪
isInScreen = true;
}
//位移
matrix.postTranslate(bitmapList.get(i).x, bitmapList.get(i).y);
canvas.drawBitmap(bitmapList.get(i).bitmap, matrix, paint);
}
if (isInScreen) {
postInvalidate();
}else {
release();
}
}
}
/**
*釋放資源
*/
private void release(){
if(bitmapList != null && bitmapList.size()>0){
for(ItemEmoje itemEmoje : bitmapList){
if(!itemEmoje.bitmap.isRecycled()){
itemEmoje.bitmap.recycle();
}
}
bitmapList.clear();
}
}
提供start() 方法觸發(fā)。
public void start(boolean isRun) {
this.isRun = isRun;
initData();
postInvalidate();
}
布局文件
<?xml version="1.0" encoding="utf-8"?> <RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android" android:layout_width="match_parent" android:layout_height="match_parent"> <com.rain.RainView android:id="@+id/testView" android:layout_width="match_parent" android:layout_height="match_parent" /> <Button android:id="@+id/btn_dog" android:layout_width="wrap_content" android:layout_height="wrap_content" android:layout_alignParentBottom="true" android:text="真·狗頭雨·落!" /> <Button android:id="@+id/btn_cake" android:layout_width="wrap_content" android:layout_height="wrap_content" android:layout_alignParentBottom="true" android:layout_marginLeft="10dp" android:layout_toRightOf="@+id/btn_dog" android:text="蛋糕雨" /> </RelativeLayout>
activity 點(diǎn)擊事件觸發(fā)
btnCake.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View v) {
//蛋糕圖片
rainView.setImgResId(R.mipmap.cake);
rainView.start(true);
}
});
btnDog.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View v) {
//狗頭圖片
rainView.setImgResId(R.mipmap.dog);
rainView.start(true);
}
});
github地址:https://github.com/taixiang/rain_emoji (本地下載)
總結(jié)
以上就是這篇文章的全部?jī)?nèi)容了,希望本文的內(nèi)容對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,如果有疑問(wèn)大家可以留言交流,謝謝大家對(duì)腳本之家的支持。
- Android高仿微信表情輸入與鍵盤(pán)輸入詳解
- android高仿微信表情輸入與鍵盤(pán)輸入代碼(詳細(xì)實(shí)現(xiàn)分析)
- 基于Android開(kāi)發(fā)支持表情的實(shí)現(xiàn)詳解
- Android開(kāi)發(fā)技巧之像QQ一樣輸入文字和表情圖像
- Android編程實(shí)現(xiàn)QQ表情的發(fā)送和接收完整實(shí)例(附源碼)
- Android編程開(kāi)發(fā)實(shí)現(xiàn)TextView顯示表情圖像和文字的方法
- 完整的Android表情功能處理方案
- Android編程開(kāi)發(fā)之EditText實(shí)現(xiàn)輸入QQ表情圖像的方法
- Android自帶emoji表情的使用方法詳解
相關(guān)文章
Android利用貝塞爾曲線繪制動(dòng)畫(huà)的示例代碼
本篇就借由動(dòng)畫(huà)驅(qū)動(dòng)貝塞爾曲線繪制看看動(dòng)起來(lái)的貝塞爾曲線什么效果。文中的示例代碼講解詳細(xì),感興趣的小伙伴可以跟隨小編一起學(xué)習(xí)一下2022-05-05
Flutter中數(shù)據(jù)存儲(chǔ)的四種方式小結(jié)
在 Flutter 中,存儲(chǔ)是指用于本地和遠(yuǎn)程存儲(chǔ)和管理數(shù)據(jù)的機(jī)制,本給大家介紹了Flutter中不同存儲(chǔ)選項(xiàng)的概述和示例,通過(guò)代碼示例講解的非常詳細(xì),具有一定的參考價(jià)值,需要的朋友可以參考下2023-11-11
Android使用ViewFlipper和GestrueDetector共同實(shí)現(xiàn)滑屏效果實(shí)例
這篇文章主要介紹了Android使用ViewFlipper和GestrueDetector共同實(shí)現(xiàn)滑屏效果,結(jié)合完整實(shí)例形式分析了ViewFlipper和GestrueDetector控件實(shí)現(xiàn)滑屏功能的布局與相關(guān)操作技巧,需要的朋友可以參考下2017-02-02
ionic監(jiān)聽(tīng)android返回鍵實(shí)現(xiàn)“再按一次退出”功能
本篇文章主要介紹了ionic監(jiān)聽(tīng)android返回鍵實(shí)現(xiàn)“再按一次退出”功能,非常具有實(shí)用價(jià)值,需要的朋友可以參考下2018-02-02
Android自定義View實(shí)現(xiàn)仿GitHub的提交活躍表格
這篇文章主要介紹了Android自定義View實(shí)現(xiàn)仿GitHub的提交活躍表格,非常不錯(cuò),具有參考借鑒價(jià)值,需要的的朋友參考下吧2017-01-01
android4.0混淆XmlPullParser報(bào)錯(cuò)原因分析解決
今天,用android4.0在proguard-project.txt中加入 -libraryjars libs/ksoap2-android-assembly-2.6.0-jar-with-dependencies.jar這句話后,混淆時(shí)報(bào)上面的錯(cuò)誤,下面與大家分享下具體的解決方法2013-06-06
Android?APP瘦身shrinkResources使用問(wèn)題詳解
這篇文章主要為大家介紹了Android?APP瘦身shrinkResources使用問(wèn)題詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2022-11-11
Android自定義View之RadioGroup實(shí)現(xiàn)跨多行顯示
這篇文章主要介紹了Android自定義View之RadioGroup實(shí)現(xiàn)跨多行顯示,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2019-11-11
Android開(kāi)發(fā)之使用ExifInterface獲取拍照后的圖片屬性
這篇文章主要介紹了Android開(kāi)發(fā)之使用ExifInterface獲取拍照后的圖片屬性,較為詳細(xì)的分析了ExifInterface類操作圖片的具體使用技巧,需要的朋友可以參考下2016-01-01

