Android實(shí)現(xiàn)文字上下滾動(dòng)效果
關(guān)于Android實(shí)現(xiàn)文字上下滾動(dòng)這個(gè)功能,我目前有兩種方法實(shí)現(xiàn):
一個(gè)是在TextView 中加上翻轉(zhuǎn)的動(dòng)畫(huà)效果,然后設(shè)置循環(huán)滾動(dòng);一種是改寫(xiě)ViewPager 的滾動(dòng)方向,使它從下到上進(jìn)行滾動(dòng),并設(shè)置循環(huán)滾動(dòng);
首先介紹第一種方法:
實(shí)現(xiàn)思路:自定義TextView,在TextView中加上從下到上滾動(dòng)的動(dòng)畫(huà)效果,然后設(shè)置循環(huán)播放;
創(chuàng)建一個(gè)AutoTextVieW使之繼承TextView,然后在onDraw方法中調(diào)用getHeight()方法獲取textview當(dāng)前的高度。
在接下來(lái)的動(dòng)畫(huà)翻轉(zhuǎn)效果中,根據(jù)這個(gè)高度設(shè)置TextView上下滾動(dòng)的距離。下面是動(dòng)畫(huà)實(shí)現(xiàn)的方法:
/** * 向上脫離屏幕的動(dòng)畫(huà)效果 */ private void animationStart() { ObjectAnimator translate = ObjectAnimator.ofFloat(this, "translationY", 0, -height); ObjectAnimator alpha = ObjectAnimator.ofFloat(this, "alpha", 1f, 0f); mAnimStart = new AnimatorSet(); mAnimStart.play(translate).with(alpha); mAnimStart.setDuration(DURATION); mAnimStart.addListener(this); } /** * 從屏幕下面向上的動(dòng)畫(huà)效果 */ public void animationOver() { ObjectAnimator translate = ObjectAnimator.ofFloat(this, "translationY", height, 0); ObjectAnimator alpha = ObjectAnimator.ofFloat(this, "alpha", 0f, 1f); mAnimOver = new AnimatorSet(); mAnimOver.play(translate).with(alpha); mAnimOver.setDuration(DURATION); }
接下來(lái)實(shí)現(xiàn)ObjectAnimator的監(jiān)聽(tīng)事件,在onAnimationEnd 調(diào)用setText方法,在動(dòng)畫(huà)沒(méi)結(jié)束一次更新文字,并且繼續(xù)執(zhí)行動(dòng)畫(huà)效果
@Override public void onAnimationEnd(Animator animator) { super.setText(mText); if (mAnimOver == null) { animationOver(); } mAnimOver.start(); }
然后調(diào)用一個(gè)可以設(shè)置循環(huán)滾動(dòng)的類,這里可以使用ScheduledExecutorService,也可以使用 Timer幾設(shè)置計(jì)時(shí)滾動(dòng),在更新UI的時(shí)候,調(diào)用Handler方法更新;
因?yàn)椴捎肨imer執(zhí)行定時(shí)任務(wù)時(shí)只創(chuàng)建一個(gè)線程,所以這里建議采用ScheduledExecutorService;
/** * 獲取數(shù)據(jù)并設(shè)置滾動(dòng)播放 * @param textView * @param list * @param autoPlayTime */ public void getTextData(final IdeaAutoTextview textView, List<String> list, int autoPlayTime) { this.textView = textView; this.textList = list; if (autoPlayTime != 0) { scheduledExecutorService = Executors.newSingleThreadScheduledExecutor(); scheduledExecutorService.scheduleWithFixedDelay(new WeakTimerTask(this), autoPlayTime, autoPlayTime, TimeUnit.SECONDS); } } private TimeTaskHandler mHandler = new TimeTaskHandler(this); private static class WeakTimerTask extends TimerTask { private WeakReference<IdeaAutoTextview> autoTextReference; public WeakTimerTask(IdeaAutoTextview mautoText) { this.autoTextReference = new WeakReference<>(mautoText); } @Override public void run() { IdeaAutoTextview autoText = autoTextReference.get(); if (autoText != null) { if (autoText.isShown()) { autoText.mHandler.sendEmptyMessage(0); } } else { cancel(); } } }
定時(shí)刷新頻率較高,容易產(chǎn)生內(nèi)存泄漏,這里采用弱引用避免這個(gè)情況發(fā)生
private final class TimeTaskHandler extends Handler { private WeakReference<IdeaAutoTextview> autoTextReference; public TimeTaskHandler(IdeaAutoTextview autoText) { this.autoTextReference = new WeakReference<>(autoText); } @Override public void handleMessage(Message msg) { IdeaAutoTextview autoText = autoTextReference.get(); if (autoText!=null) { /** * 設(shè)置當(dāng)前文字 */ String text = textList.get(index); index++; if (index > textList.size() - 1) { index = 0; } textView.setAutoText(text); } } }
到此第一種方法介紹完畢。
第二種方法實(shí)現(xiàn)的原理和輪播圖的原理類似,輪播圖一般是左右橫向滾動(dòng),這里需要把ViewPager改成上下滑動(dòng),關(guān)于上下滑動(dòng)的viewpager,可以在給github上找到;
其次輪播圖中播放的是圖片,把圖片換成文字即可;
然后同樣調(diào)用Timer或者ScheduledExecutorService使ViewPager自行滾動(dòng);
以下是代碼:
package com.idea.idea.viewutils; import android.content.Context; import android.os.Handler; import android.os.Message; import android.support.v4.view.PagerAdapter; import android.support.v4.view.ViewPager; import android.util.AttributeSet; import android.widget.RelativeLayout; import java.lang.ref.WeakReference; import java.util.TimerTask; import java.util.concurrent.Executors; import java.util.concurrent.ScheduledExecutorService; import java.util.concurrent.TimeUnit; /** * todo:修改ViewPager方法實(shí)現(xiàn)文字滾動(dòng) * * @author: Create by qjj * @email: gxuqjj@163.com */ public class AutoViewpager extends RelativeLayout{ private VerticalViewPager mVerticalViewPager; private PagerAdapter mAdapter; private int autoPlayTime; private ScheduledExecutorService scheduledExecutorService; public AutoViewpager(Context context){ this(context,null); } public AutoViewpager(Context context, AttributeSet attrs) { this(context, attrs, 0); } public AutoViewpager(Context context, AttributeSet attrs, int defStyle) { super(context, attrs, defStyle); initView(); } /** * 初始化view */ private void initView(){ if(mVerticalViewPager!=null){ removeView(mVerticalViewPager); } mVerticalViewPager = new VerticalViewPager(getContext()); mVerticalViewPager.setLayoutParams(new LayoutParams(LayoutParams.MATCH_PARENT, LayoutParams.MATCH_PARENT)); addView(mVerticalViewPager); } private final static class TimeTaskHandler extends Handler { private WeakReference<AutoViewpager> mRollPagerViewWeakReference; public TimeTaskHandler(AutoViewpager autoViewpager) { this.mRollPagerViewWeakReference = new WeakReference<>(autoViewpager); } @Override public void handleMessage(Message msg) { AutoViewpager autoViewpager = mRollPagerViewWeakReference.get(); int cur = autoViewpager.getViewPager().getCurrentItem()+1; if(cur>= autoViewpager.mAdapter.getCount()){ cur=0; } autoViewpager.getViewPager().setCurrentItem(cur); } } private TimeTaskHandler mHandler = new TimeTaskHandler(this); private static class WeakTimerTask extends TimerTask { private WeakReference<AutoViewpager> mRollPagerViewWeakReference; public WeakTimerTask(AutoViewpager mAutoViewpager) { this.mRollPagerViewWeakReference = new WeakReference<>(mAutoViewpager); } @Override public void run() { AutoViewpager autoViewpager = mRollPagerViewWeakReference.get(); if (autoViewpager !=null){ if(autoViewpager.isShown()){ autoViewpager.mHandler.sendEmptyMessage(0); } }else{ cancel(); } } } /** * 開(kāi)始滾動(dòng) */ private void autoPlay(){ if(autoPlayTime<=0||mAdapter == null||mAdapter.getCount()<=1){ return; } scheduledExecutorService = Executors.newSingleThreadScheduledExecutor(); scheduledExecutorService.scheduleWithFixedDelay(new WeakTimerTask(this), autoPlayTime, autoPlayTime, TimeUnit.SECONDS); } public void setAutoTime(int autoPlayTime){ this.autoPlayTime = autoPlayTime; autoPlay(); } /** * viewpager * @return */ public ViewPager getViewPager() { return mVerticalViewPager; } /** * 設(shè)置Adapter * @param adapter */ public void setAdapter(PagerAdapter adapter){ mVerticalViewPager.setAdapter(adapter); mAdapter = adapter; dataChanged(); } private void dataChanged(){ autoPlay(); } }
以上就是本文的全部?jī)?nèi)容,希望對(duì)大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
相關(guān)文章
Android實(shí)現(xiàn)手機(jī)多點(diǎn)觸摸畫(huà)圓
這篇文章主要為大家詳細(xì)介紹了Android實(shí)現(xiàn)手機(jī)多點(diǎn)觸摸畫(huà)圓,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2022-05-05Android中Intent組件的入門(mén)學(xué)習(xí)心得
Intent組件雖然不是四大組件,但卻是連接四大組件的橋梁,學(xué)習(xí)好這個(gè)知識(shí),也非常的重要,下面這篇文章主要給大家介紹了關(guān)于Android中Intent組件的相關(guān)資料,需要的朋友可以參考下2021-12-12android通過(guò)拼音搜索中文的功能實(shí)現(xiàn)代碼
這篇文章主要介紹了android通過(guò)拼音搜索中文的功能實(shí)現(xiàn)代碼,本文通過(guò)實(shí)例代碼給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2020-11-11Popupwindow 的簡(jiǎn)單實(shí)用案例(顯示在控件下方)
下面小編就為大家?guī)?lái)一篇Popupwindow 的簡(jiǎn)單實(shí)用案例(顯示在控件下方)。小編覺(jué)得挺不錯(cuò)的,現(xiàn)在就分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧2017-04-04android實(shí)現(xiàn)圓環(huán)倒計(jì)時(shí)控件
這篇文章主要為大家詳細(xì)介紹了android實(shí)現(xiàn)圓環(huán)倒計(jì)時(shí)控件,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2021-01-01Android實(shí)現(xiàn)簡(jiǎn)單的城市列表功能
這篇文章主要為大家詳細(xì)介紹了Android實(shí)現(xiàn)簡(jiǎn)單的城市列表功能,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2017-09-09Android開(kāi)發(fā)之Gradle?進(jìn)階Tasks深入了解
這篇文章主要為大家介紹了Android開(kāi)發(fā)之Gradle?進(jìn)階Tasks深入了解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2022-08-08Android解析服務(wù)器端發(fā)來(lái)的xml數(shù)據(jù)示例
Android跟服務(wù)器交互數(shù)據(jù),有時(shí)數(shù)據(jù)量大時(shí),就需要以xml形式的交互數(shù)據(jù),下面與大家分享下使用XmlPullParser來(lái)解析xml數(shù)據(jù),感興趣的朋友可以參考下哈2013-06-06Android 列表選擇框 Spinner詳解及實(shí)例
這篇文章主要介紹了Android 列表選擇框 Spinner詳解及實(shí)例的相關(guān)資料,需要的朋友可以參考下2017-06-06