Android實(shí)現(xiàn)自定義的彈幕效果
一、效果圖
先來看看效果圖吧~~
二、實(shí)現(xiàn)原理方案
1、自定義ViewGroup-XCDanmuView
,繼承RelativeLayout
來實(shí)現(xiàn),當(dāng)然也可以繼承其他三大布局類哈
2、初始化若干個(gè)TextView
(彈幕的item View
,這里以TextView
為例,當(dāng)然也可以其他了~),然后通過addView
添加到自定義View
中
3、通過addView
添加到XCDanmuView
中,位置在坐標(biāo),為了實(shí)現(xiàn) 從屏幕外移動(dòng)進(jìn)來的效果
我們還需要修改添加進(jìn)來TextView
的位置,以從右向左移動(dòng)方向來說,addView
后必須將該TextView
的位置設(shè)置到右邊的屏幕外
這樣我們采用的方法,是在onLayout()
方法中對childView
進(jìn)行layout
重新布局設(shè)置位置
4、隨機(jī)沖左側(cè)或右側(cè)出來彈幕itemView
,移動(dòng)采用屬性動(dòng)畫來實(shí)現(xiàn)平移,從屏幕的一端移動(dòng)到另一端,當(dāng)動(dòng)畫結(jié)束后,就將
該child從XCDanmuView
中remove
掉。并重新new
一個(gè)彈幕itemView
,并addView
到XCDanmuView
中,并開始動(dòng)畫移動(dòng)
5、本自定義彈幕View
支持從左到右和從右到左兩個(gè)方向,支持自定義設(shè)置屏幕彈幕最多顯示個(gè)數(shù)。
三、自定義彈幕效果XCDanmuView的具體實(shí)現(xiàn)
1、初始化需要用到的數(shù)據(jù)變量
private int mWidth; private int mScreenWidth; private List<View> mChildList; private boolean mIsWorking = false; private Context mContext; private int mMaxShowNum = 15; private int mRowNum = 4; private int[] mSpeeds = { 3000,4000,5000,6000 }; private int mDelayDuration = 500; private int[] mBgResIds = { R.drawable.bg_danmu0, R.drawable.bg_danmu1, R.drawable.bg_danmu2, R.drawable.bg_danmu3 }; private int[] mRowPos = { 150,140,160,150 }; private Random mRandom; private String[] mStrContents; public static enum XCDirection{ FROM_RIGHT_TO_LEFT, FORM_LEFT_TO_RIGHT } public enum XCAction{ SHOW,HIDE } private XCDirection mDirection = XCDirection.FROM_RIGHT_TO_LEFT; private void init() { mScreenWidth = getScreenWidth(); mChildList = new ArrayList<>(); mRandom = new Random(); }
2、初始化若干個(gè)彈幕item view
public void initDanmuItemViews(String[] strContents){ mStrContents = strContents; for(int i = 0; i < mMaxShowNum; i ++){ int index = mRandom.nextInt(100) % strContents.length; createDanmuView(i,strContents[index],false); } }
3、創(chuàng)建彈幕item view 并addView到XCDanmuView中
public void createDanmuView(int index,String content,boolean reset){ final TextView textView = new TextView(mContext); textView.setTextColor(Color.WHITE); int r = mRandom.nextInt(100) % mRowNum; textView.setBackgroundResource(mBgResIds[r]); textView.setText(content +"_"+ (index+1)); RelativeLayout.LayoutParams lp = new LayoutParams(LayoutParams.WRAP_CONTENT, RelativeLayout.LayoutParams.WRAP_CONTENT); int row = mRandom.nextInt(100) % mRowNum; while(row == lastRow){ row = mRandom.nextInt(100)% mRowNum; } int pos = mRandom.nextInt(100)% mRowNum; lp.topMargin = row * mRowPos[pos]; lastRow = row; textView.setLayoutParams(lp); textView.setPadding(40, 2, 40, 2); this.addView(textView); if(reset){ mChildList.set(index,textView); }else{ mChildList.add(index,textView); } textView.setClickable(true); textView.setOnClickListener(new OnClickListener() { @Override public void onClick(View view) { Toast toast = Toast.makeText(mContext, textView.getText(), Toast.LENGTH_SHORT); toast.setGravity(Gravity.TOP,0,50); toast.show(); } }); }
4、重新設(shè)置childView的初始位置到屏幕之外
@Override protected void onLayout(boolean changed, int l, int t, int r, int b) { super.onLayout(changed, l, t, r, b); int childCount = this.getChildCount(); for(int i=0;i<childCount;i++){ View view = getChildAt(i); RelativeLayout.LayoutParams lp = (LayoutParams) view.getLayoutParams(); if(lp.leftMargin <= 0){ if(mDirection == XCDirection.FORM_LEFT_TO_RIGHT){ view.layout(-view.getMeasuredWidth(), lp.topMargin, 0,lp.topMargin + view.getMeasuredHeight()); }else{ view.layout(mScreenWidth,lp.topMargin,mScreenWidth+view.getMeasuredWidth(), lp.topMargin+view.getMeasuredHeight()); } }else{ continue; } } }
5、彈幕item view的移動(dòng)效果
private Handler mHandler = new Handler() { @Override public void handleMessage(final Message msg) { super.handleMessage(msg); final int pos = msg.what; ViewPropertyAnimator animator; if(mDirection == XCDirection.FROM_RIGHT_TO_LEFT){ animator = mChildList.get(msg.what).animate() .translationXBy(-(mScreenWidth + mChildList.get(msg.what).getWidth())); }else{ animator = mChildList.get(msg.what).animate() .translationXBy(mScreenWidth + mChildList.get(msg.what).getWidth()); } Random random = new Random(System.currentTimeMillis()); int index = random.nextInt(100) % mSpeeds.length; animator.setDuration(mSpeeds[index]); animator.setInterpolator(new LinearInterpolator()); animator.setListener(new Animator.AnimatorListener() { @Override public void onAnimationStart(Animator animator) { } @Override public void onAnimationEnd(Animator animator) { XCDanmuView.this.removeView(mChildList.get(pos)); int index = mRandom.nextInt(100) % mStrContents.length; createDanmuView(pos, mStrContents[index], true); mHandler.sendEmptyMessageDelayed(pos, mDelayDuration); Log.v("czm", "size=" + mChildList.size()); } @Override public void onAnimationCancel(Animator animator) { } @Override public void onAnimationRepeat(Animator animator) { } }); animator.start(); } };
6、開啟彈幕效果和關(guān)閉彈幕效果以及對于的動(dòng)畫效果
boolean isFirst = true; public void start(){ switchAnimation(XCAction.SHOW); if(isFirst){ for(int i =0;i< mChildList.size();i++){ mHandler.sendEmptyMessageDelayed(i,i * mDelayDuration); } isFirst = false; } mIsWorking = true; } public void hide(){ switchAnimation(XCAction.HIDE); mIsWorking =false; } public void stop(){ this.setVisibility(View.GONE); for(int i =0;i< mChildList.size();i++){ mChildList.get(i).clearAnimation(); mHandler.removeMessages(i); } mIsWorking =false; } private void switchAnimation(final XCAction action){ AlphaAnimation animation; if(action == XCAction.HIDE){ animation = new AlphaAnimation(1.0f,0.0f); animation.setDuration(400); }else{ animation = new AlphaAnimation(0.0f,1.0f); animation.setDuration(1000); } XCDanmuView.this.startAnimation(animation); animation.setAnimationListener(new Animation.AnimationListener() { @Override public void onAnimationStart(Animation animation) { } @Override public void onAnimationEnd(Animation animation) { if(action == XCAction.HIDE){ XCDanmuView.this.setVisibility(View.GONE); }else{ XCDanmuView.this.setVisibility(View.VISIBLE); } } @Override public void onAnimationRepeat(Animation animation) { } }); }
四、如何使用該自定義側(cè)滑View控件
使用該自定義View非常簡單,控件默認(rèn)效果從右向左,如果需要修改方向?yàn)閺淖蟮接?,只需設(shè)置下方向即可
public class MainActivity extends Activity { private XCDanmuView mDanmuView; private List<View> mViewList; private String[] mStrItems = { "搜狗","百度", "騰訊","360", "阿里巴巴","搜狐", "網(wǎng)易","新浪", "搜狗-上網(wǎng)從搜狗開始","百度一下,你就知道", "必應(yīng)搜索-有求必應(yīng)","好搜-用好搜,特順手", "Android-谷歌","IOS-蘋果", "Windows-微軟","Linux" }; @Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.activity_main); initDanmuView(); initListener(); } private void initListener() { findViewById(R.id.btn).setOnClickListener(new View.OnClickListener() { @Override public void onClick(View view) { if (mDanmuView.isWorking()) { mDanmuView.hide(); ((Button) view).setText("開啟彈幕"); } else { mDanmuView.start(); ((Button) view).setText("關(guān)閉彈幕"); } } }); } private void initDanmuView() { mDanmuView = (XCDanmuView)findViewById(R.id.danmu); mDanmuView.initDanmuItemViews(mStrItems); } }
五、總結(jié)
以上就是在Android中實(shí)現(xiàn)自定義彈幕效果的全部內(nèi)容個(gè),希望本文的內(nèi)容對大家開發(fā)Android的時(shí)候能有所幫助。如果有疑問可以留言交流。
相關(guān)文章
Android實(shí)現(xiàn)滑動(dòng)折疊Header全流程詳解
這篇文章主要介紹了Android實(shí)現(xiàn)滑動(dòng)折疊Header,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)吧2022-11-11Android onClick方法與setOnClickListener方法對比
這篇文章主要介紹了Android onClick方法與setOnClickListener方法對比的相關(guān)資料,這兩個(gè)方法都是點(diǎn)擊事件處理函數(shù)的方法,它們之間到底有什么區(qū)別呢,下面就給大家說下,需要的朋友可以參考下2016-12-12Android中封裝SDK時(shí)常用的注解總結(jié)
這篇文章主要給大家總結(jié)了在Android中封裝SDK時(shí)常用的注解的相關(guān)資料,文中介紹的非常詳細(xì),對大家具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面來一起看看吧。2017-05-05Android編程使用AlarmManager設(shè)置鬧鐘的方法
這篇文章主要介紹了Android編程使用AlarmManager設(shè)置鬧鐘的方法,結(jié)合具體實(shí)例分析了Android基于AlarmManager實(shí)現(xiàn)鬧鐘功能的設(shè)置、取消、顯示等相關(guān)操作技巧,需要的朋友可以參考下2017-03-03Android Support Annotations資料整理
這篇文章主要介紹了Android Support Annotations資料整理的相關(guān)資料,需要的朋友可以參考下2017-05-05Android實(shí)現(xiàn)中軸旋轉(zhuǎn)特效 Android制作別樣的圖片瀏覽器
這篇文章主要為大家詳細(xì)介紹了Android實(shí)現(xiàn)中軸旋轉(zhuǎn)特效,制作別樣的圖片瀏覽器,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2017-11-11