Android實(shí)現(xiàn)文字上下滾動(dòng)效果
關(guān)于Android實(shí)現(xiàn)文字上下滾動(dòng)這個(gè)功能,我目前有兩種方法實(shí)現(xiàn):
一個(gè)是在TextView 中加上翻轉(zhuǎn)的動(dòng)畫效果,然后設(shè)置循環(huán)滾動(dòng);一種是改寫ViewPager 的滾動(dòng)方向,使它從下到上進(jìn)行滾動(dòng),并設(shè)置循環(huán)滾動(dòng);
首先介紹第一種方法:
實(shí)現(xiàn)思路:自定義TextView,在TextView中加上從下到上滾動(dòng)的動(dòng)畫效果,然后設(shè)置循環(huán)播放;
創(chuàng)建一個(gè)AutoTextVieW使之繼承TextView,然后在onDraw方法中調(diào)用getHeight()方法獲取textview當(dāng)前的高度。
在接下來的動(dòng)畫翻轉(zhuǎn)效果中,根據(jù)這個(gè)高度設(shè)置TextView上下滾動(dòng)的距離。下面是動(dòng)畫實(shí)現(xiàn)的方法:
/**
* 向上脫離屏幕的動(dòng)畫效果
*/
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)畫效果
*/
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);
}
接下來實(shí)現(xiàn)ObjectAnimator的監(jiān)聽事件,在onAnimationEnd 調(diào)用setText方法,在動(dòng)畫沒結(jié)束一次更新文字,并且繼續(xù)執(zhí)行動(dòng)畫效果
@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();
}
}
}
/**
* 開始滾動(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 仿微信demo——登錄功能實(shí)現(xiàn)(移動(dòng)端)
本篇文章主要介紹了微信小程序-閱讀小程序?qū)嵗╠emo),小編覺得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過來看看吧,希望能給你們提供幫助2021-06-06
Android 中 viewpager 滑動(dòng)指示器的實(shí)例代碼
本文通過實(shí)例代碼給大家介紹了android 中 viewpager 滑動(dòng)指示器,代碼簡(jiǎn)單易懂,非常不錯(cuò),具有參考借鑒價(jià)值,需要的朋友可以參考下2017-12-12
Android屬性動(dòng)畫實(shí)現(xiàn)布局的下拉展開效果
這篇文章主要為大家詳細(xì)介紹了Android屬性動(dòng)畫實(shí)現(xiàn)布局的下拉展開效果,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2016-07-07
條件數(shù)據(jù)庫Android:sqllite的簡(jiǎn)單使用
條件數(shù)據(jù)庫Android:sqllite的簡(jiǎn)單使用,需要的朋友可以參考一下2013-05-05
Notification與NotificationManager詳細(xì)介紹
在Android系統(tǒng)中,發(fā)一個(gè)狀態(tài)欄通知還是很方便的。下面我們就來看一下,怎么發(fā)送狀態(tài)欄通知,狀態(tài)欄通知又有哪些參數(shù)可以設(shè)置2012-11-11
Activity透明/半透明效果的設(shè)置transparent(兩種實(shí)現(xiàn)方法)
兩種方法實(shí)現(xiàn)Activity透明/半透明效果的設(shè)置,代碼思路很有調(diào)理,感興趣的朋友可以參考下,希望本文可以幫助到你2013-02-02
Android跳轉(zhuǎn)三方應(yīng)用實(shí)例代碼
大家好,本篇文章主要講的是Android跳轉(zhuǎn)三方應(yīng)用實(shí)例代碼,感興趣的同學(xué)趕快來看一看吧,對(duì)你有幫助的話記得收藏一下,方便下次瀏覽2021-12-12
android 解析json數(shù)據(jù)格式的方法
這篇文章主要介紹了android 解析json數(shù)據(jù)格式的方法,有需要的朋友可以參考一下2014-01-01
Android 自定義彈性ListView控件實(shí)例代碼(三種方法)
關(guān)于在Android中實(shí)現(xiàn)ListView的彈性效果,有很多不同的方法,網(wǎng)上一搜,也有很多,下面貼出在項(xiàng)目中經(jīng)常用到的兩種實(shí)現(xiàn)ListView彈性效果的方法(基本上拿來就可以用),需要的朋友參考下本段代碼2016-01-01

