Android实现文字上下滚动效果
作者:Ideaqjjl 发布时间:2023-02-02 07:40:00
关于Android实现文字上下滚动这个功能,我目前有两种方法实现:
一个是在TextView 中加上翻转的动画效果,然后设置循环滚动;一种是改写ViewPager 的滚动方向,使它从下到上进行滚动,并设置循环滚动;
首先介绍第一种方法:
实现思路:自定义TextView,在TextView中加上从下到上滚动的动画效果,然后设置循环播放;
创建一个AutoTextVieW使之继承TextView,然后在onDraw方法中调用getHeight()方法获取textview当前的高度。
在接下来的动画翻转效果中,根据这个高度设置TextView上下滚动的距离。下面是动画实现的方法:
/**
* 向上脱离屏幕的动画效果
*/
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);
}
/**
* 从屏幕下面向上的动画效果
*/
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);
}
接下来实现ObjectAnimator的监听事件,在onAnimationEnd 调用setText方法,在动画没结束一次更新文字,并且继续执行动画效果
@Override
public void onAnimationEnd(Animator animator) {
super.setText(mText);
if (mAnimOver == null) {
animationOver();
}
mAnimOver.start();
}
然后调用一个可以设置循环滚动的类,这里可以使用ScheduledExecutorService,也可以使用 Timer几设置计时滚动,在更新UI的时候,调用Handler方法更新;
因为采用Timer执行定时任务时只创建一个线程,所以这里建议采用ScheduledExecutorService;
/**
* 获取数据并设置滚动播放
* @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();
}
}
}
定时刷新频率较高,容易产生内存泄漏,这里采用弱引用避免这个情况发生
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)
{
/**
* 设置当前文字
*/
String text = textList.get(index);
index++;
if (index > textList.size() - 1) {
index = 0;
}
textView.setAutoText(text);
}
}
}
到此第一种方法介绍完毕。
第二种方法实现的原理和轮播图的原理类似,轮播图一般是左右横向滚动,这里需要把ViewPager改成上下滑动,关于上下滑动的viewpager,可以在给github上找到;
其次轮播图中播放的是图片,把图片换成文字即可;
然后同样调用Timer或者ScheduledExecutorService使ViewPager自行滚动;
以下是代码:
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方法实现文字滚动
*
* @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();
}
}
}
/**
* 开始滚动
*/
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;
}
/**
* 设置Adapter
* @param adapter
*/
public void setAdapter(PagerAdapter adapter){
mVerticalViewPager.setAdapter(adapter);
mAdapter = adapter;
dataChanged();
}
private void dataChanged(){
autoPlay();
}
}
来源:https://blog.csdn.net/Ideaqjjl/article/details/51132153
猜你喜欢
- 一、什么是 websocket 接口使用 websocket 建立长连接,服务端和客户端可以互相通信,服务端只要有数据更新,就可以主动推给客
- Swagger2配置(解决404报错)在spring boot项目中配置Swagger2,配置好了但是访问确实404,SwaggerConf
- 数组的长度是固定的,无法适应数据变化的需求。为了解决这个问题,Java提供了另一个容器 java.util.ArrayList集合类,让我们
- 对象创建的几种方法:使用new关键字使用clone方法反射机制反序列化以上四种都可以产生java对象1,3都会明确的显式的调用构造函数2是在
- 目录前言1、什么叫循环依赖呢2、具体出现循环依赖的代码逻辑3、解决循环依赖的代码实现总结前言本文基于springboot版本2.5.1 &n
- 前言Intellij IDEA 2017.2.2版本针对Springboot设置了一些特性,本篇文章给大家简单介绍一下如何使用这些特性。Ru
- 引言最近在工作中结合线程池使用 InheritableThreadLocal 出现了获取线程变量“错误&rdqu
- DataHub 类似于传统大数据解决方案中 Kafka 的角色,提供了一个数据队列功能。DataHub 除了供了一个缓冲的队列作用。同时由于
- javaWeb 四大域对象1)和属性相关的方法Object getAttribute(String name) 获取指定的属性En
- java 多线程的几种实现方法总结1.多线程有几种实现方法?同步有几种实现方法?多线程有两种实现方法,分别是继承Thread类与实现Runn
- public class Wrapper { public static void main
- 1 问题手写一个程序,完成List集合对象的逆序遍历2 方法创建List接口的多态对象向创建好list集合添加元素使用hasPrevious
- MyBatis介绍MyBatis本是apache的一个开源项目iBatis, 2010年这个项目由apache software
- 注意事项阿里云的dataworks提供了OpenApi, 需要是企业版或旗舰版才能够调用,也就是付费项目。这里测试主要是调用拉取datawo
- 职责链模式简介及UML职责链也叫责任链,他是一种行为型模式,它为请求创建了一个接收请求者对象的链,并将请求沿着这条链传递到目标对象去处理。该
- 一、堆参数设置-XX:+PrintGC 使用这个参数,虚拟机启动后,只要遇到GC就会打印日志-XX:+UseSerialGC 配置串行回收器
- 先贴代码,后面做一些简单说明:public static string sendPostHttpRequest_2(string url,
- 我们在java中处理字符串的时候,一般会选择String,在python中同样也是作用于字符串。那么我们今天延伸一下它的用法,只使用Stri
- spring mvc url匹配禁用后缀访问在spring mvc中默认 访问url 加任意后缀名都能访问比如:你想访问 /login ,但
- 前言我们在项目的开发中,难免会遇到各种可预知的、不可预知的异常需要处理。每个过程都单独处理异常,系统的代码耦合度高,工作量大且不好统一,维护