Realização do efeito de rolagem do Android para cima e para baixo (letreiro)

Há um requisito para o produto rolar o texto para cima e para baixo. A primeira coisa que me vem à mente é usar a animação de atributos para alternar entre dois TextViews. Eu li um monte de materiais na Internet e a maioria deles é TextSwitch. é realmente melancólico escrever um blog sem postar as renderizações. Não sei o efeito específico, se há um problema com a primeira troca, se há um problema com a última troca para a segunda, se a animação é suave, etc. Não consigo ver nada. Então, de acordo com a primeira ideia, encontrei uma abordagem semelhante, mas parece ser um problema, então eu a modifiquei. A renderização é a seguinte: (A taxa de quadros do GIF é um pouco baixa, não muito suave e funciona bem na máquina real)

 

A ideia é que os dois TextViews usem animação de atributo para alternar para cima e para baixo, atrasando o encadeamento de envio para acionar a próxima rolagem.

código mostrado abaixo:

1.ScrrollTextView.java

import android.animation.ObjectAnimator;
import android.content.Context;
import android.os.Handler;
import android.support.annotation.Nullable;
import android.util.AttributeSet;
import android.view.LayoutInflater;
import android.view.View;
import android.widget.LinearLayout;
import android.widget.TextView;

import com.lanjinger.choiassociatedpress.R;

import java.util.List;

/**
 * 上下滚动的 textView
 */
public class ScrollTextView extends LinearLayout {
    private TextView mBannerTV1;
    private TextView mBannerTV2;
    private Handler handler;
    private boolean isShow = false;
    private int startY1, endY1, startY2, endY2;
    private Runnable runnable;
    private List<String> list;
    private int position = 0;
    private int offsetY = 100;
    private boolean hasPostRunnable = false;

    public ScrollTextView(Context context) {
        this(context, null);
    }

    public ScrollTextView(Context context, @Nullable AttributeSet attrs) {
        this(context, attrs, 0);
    }

    public ScrollTextView(Context context, @Nullable AttributeSet attrs, int defStyleAttr) {
        super(context, attrs, defStyleAttr);

        View view = LayoutInflater.from(context).inflate(R.layout.widget_scroll_text_layout, this);
        mBannerTV1 = view.findViewById(R.id.tv_banner1);
        mBannerTV2 = view.findViewById(R.id.tv_banner2);
        handler = new Handler();
        runnable = new Runnable() {
            @Override
            public void run() {
                isShow = !isShow;
                if (position == list.size() - 1) {
                    position = 0;
                }

                if (isShow) {
                    mBannerTV1.setText(list.get(position++));
                    mBannerTV2.setText(list.get(position));
                } else {
                    mBannerTV2.setText(list.get(position++));
                    mBannerTV1.setText(list.get(position));
                }

                startY1 = isShow ? 0 : offsetY;
                endY1 = isShow ? -offsetY : 0;
                ObjectAnimator.ofFloat(mBannerTV1, "translationY", startY1, endY1).setDuration(300).start();

                startY2 = isShow ? offsetY : 0;
                endY2 = isShow ? 0 : -offsetY;
                ObjectAnimator.ofFloat(mBannerTV2, "translationY", startY2, endY2).setDuration(300).start();

                handler.postDelayed(runnable, 3000);
            }
        };
    }

    public List<String> getList() {
        return list;
    }

    public void setList(List<String> list) {
        this.list = list;

        //处理最后一条数据切换到第一条数据 太快的问题
        if (list.size() > 1) {
            list.add(list.get(0));
        }
    }

    public void startScroll() {
        mBannerTV1.setText(list.get(0));
        if (list.size() > 1) {
            if(!hasPostRunnable) {
                hasPostRunnable = true;
                //处理第一次进入 第一条数据切换第二条 太快的问题
                handler.postDelayed(runnable,3000);
            }
        } else {
            //只有一条数据不进行滚动
            hasPostRunnable = false;
//            mBannerTV1.setText(list.get(0));
        }
    }

    public void stopScroll() {
        handler.removeCallbacks(runnable);
        hasPostRunnable = false;
    }


}

Pontos a serem observados: 1. Para ativar e desativar a rolagem, ambos precisam ser controlados fora do controle. Se você precisar controlá-lo dentro do controle,

onAttachedToWindow Inicie o thread aqui, preste atenção para determinar se a lista está vazia.

2. Quando os dados da lista forem maiores do que 1, adicionarei manualmente os primeiros dados no final. Para evitar, o último é alterado para um porque a rolagem é muito rápida (ou seja, o problema do efeito dinâmico não óbvio ), Eu não julguei aqui., Porque deve haver dados passados, então Yan Jin pode adicionar um espaço em branco.


2.widget_scroll_text_layout.xml

<?xml version="1.0" encoding="utf-8"?>
<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
    android:layout_width="match_parent"
    android:layout_height="wrap_content"
    android:orientation="horizontal">

    <TextView
        android:id="@+id/tv_banner1"
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:layout_centerVertical="true"
        android:ellipsize="end"
        android:singleLine="true"
        android:textColor="@color/skin_common_title"
        android:textSize="12sp" />

    <TextView
        android:id="@+id/tv_banner2"
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:layout_centerVertical="true"
        android:ellipsize="end"
        android:singleLine="true"
        android:textColor="@color/skin_common_title"
        android:textSize="12sp" />

</RelativeLayout>

3. Chamada específica

        ScrollTextView marqueeText = headView.findViewById(R.id.xxxx);

        List<String> demographicsList = new ArrayList<>();

        demographicsList.add("今日测试股票 上市");
        demographicsList.add("今日科伦药业 中国人保 可申购");
        demographicsList.add("今日中国平安 上市");

        marqueeText.setList(demographicsList);
        marqueeText.startScroll();

Nota: Trate corretamente o problema do ciclo de vida. Quando chamar starScroll () e stopScroll (), isso depende do ciclo de vida específico;


rabo:

Eu sinto que você pode definir qualquer animação que quiser desta forma, e você pode personalizar a animação de atributo de acordo com suas necessidades, e não há necessidade de novo TextView () extra; um olhar simples, parece que TextSwitch é um pouco mais problemático.

Depois que a tarefa de hoje terminar, estudarei TextSwitch e compararei as vantagens e desvantagens dos dois;

 

Acho que você gosta

Origin blog.csdn.net/android_freshman/article/details/84105637
Recomendado
Clasificación