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-05
Android中Intent組件的入門(mén)學(xué)習(xí)心得
Intent組件雖然不是四大組件,但卻是連接四大組件的橋梁,學(xué)習(xí)好這個(gè)知識(shí),也非常的重要,下面這篇文章主要給大家介紹了關(guān)于Android中Intent組件的相關(guān)資料,需要的朋友可以參考下2021-12-12
android通過(guò)拼音搜索中文的功能實(shí)現(xiàn)代碼
這篇文章主要介紹了android通過(guò)拼音搜索中文的功能實(shí)現(xiàn)代碼,本文通過(guò)實(shí)例代碼給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2020-11-11
Popupwindow 的簡(jiǎn)單實(shí)用案例(顯示在控件下方)
下面小編就為大家?guī)?lái)一篇Popupwindow 的簡(jiǎn)單實(shí)用案例(顯示在控件下方)。小編覺(jué)得挺不錯(cuò)的,現(xiàn)在就分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧2017-04-04
android實(shí)現(xiàn)圓環(huán)倒計(jì)時(shí)控件
這篇文章主要為大家詳細(xì)介紹了android實(shí)現(xiàn)圓環(huán)倒計(jì)時(shí)控件,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2021-01-01
Android實(shí)現(xiàn)簡(jiǎn)單的城市列表功能
這篇文章主要為大家詳細(xì)介紹了Android實(shí)現(xiàn)簡(jiǎn)單的城市列表功能,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2017-09-09
Android開(kāi)發(fā)之Gradle?進(jìn)階Tasks深入了解
這篇文章主要為大家介紹了Android開(kāi)發(fā)之Gradle?進(jìn)階Tasks深入了解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2022-08-08
Android解析服務(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-06
Android 列表選擇框 Spinner詳解及實(shí)例
這篇文章主要介紹了Android 列表選擇框 Spinner詳解及實(shí)例的相關(guān)資料,需要的朋友可以參考下2017-06-06

