Android – [SelfView] 自定义多动画效果文本显示器
PS: 滚动文本,不限长度;* 显示效果:* 1. 静态显示(左对齐、居中、右对齐)* 2. 向左滚动;(纵向居中)* 3. 向上滚动(一条一条);* 4. 向上滚动(整体);* * 可控接口:* 1. 字体大小、颜色;* 2. 背景颜色(可透明);* 3. 滚动速度(1-20);* 4. 修饰符(斜体、粗体、下划线、删除线);
效果:

1. 引用:
<com.nepalese.harinetest.player.VirgoScrollTextViewandroid:id="@+id/vst_left"android:layout_width="match_parent"android:layout_height="40dp"android:layout_margin="10dp"/>
private VirgoScrollTextView scrollTextLeft;
private final String CONT ="《将进酒》 -- 李白\n" +"君不见黄河之水天上来,奔流到海不复回。\n" +"君不见高堂明镜悲白发,朝如青丝暮成雪。\n" +"人生得意须尽欢,莫使金樽空对月。\n" +"天生我材必有用,千金散尽还复来。\n" +"烹羊宰牛且为乐,会须一饮三百杯。\n" +"岑夫子,丹丘生,将进酒,杯莫停。\n" +"与君歌一曲,请君为我倾耳听。\n" +"钟鼓馔玉不足贵,但愿长醉不愿醒。\n" +"古来圣贤皆寂寞,惟有饮者留其名。\n" +"陈王昔时宴平乐,斗酒十千恣欢谑。\n" +"主人何为言少钱,径须沽取对君酌。\n" +"五花马、千金裘,呼儿将出换美酒,与尔同销万古愁。";scrollTextLeft = findViewById(R.id.vst_left);
scrollTextLeft.setAnimMode(VirgoScrollTextView.MODE_SCROLL_LEFT);
scrollTextLeft.setTextCont(CONT);
scrollTextLeft.startRolling();
2. VirgoScrollTextView.java
package com.nepalese.harinetest.player;import android.content.Context;
import android.graphics.Canvas;
import android.graphics.Color;
import android.graphics.Paint;
import android.graphics.PixelFormat;
import android.graphics.PorterDuff;
import android.os.Build;
import android.os.Handler;
import android.os.HandlerThread;
import android.os.Looper;
import android.text.Html;
import android.text.Layout;
import android.text.TextPaint;
import android.text.TextUtils;
import android.util.AttributeSet;
import android.util.Log;
import android.view.SurfaceHolder;
import android.view.SurfaceView;import androidx.annotation.Nullable;import com.nepalese.harinetest.config.Constants;import java.io.Serializable;
import java.util.ArrayList;
import java.util.List;public class VirgoScrollTextView extends SurfaceView implements SurfaceHolder.Callback {private static final String TAG = "VirgoScrollTextView";private static final int MAX_SPEED = 20;private static final int MIN_SPEED = 1;public static final int MODE_STATIC_DRAW = 1;public static final int MODE_SCROLL_UP = 2;public static final int MODE_SCROLL_UP_ALL = 3;public static final int MODE_SCROLL_LEFT = 4;public static final int ALIGN_LEFT = 1;public static final int ALIGN_CENTER = 2;public static final int ALIGN_RIGHT = 3;protected static final String TEXT_SPACE = "\u0020";protected static final String TXT_ENTER = "\u3000\u3000";protected static final int DEF_BG = Color.TRANSPARENT;protected static final int DEF_TEXT_COLOR = Color.BLACK;private static final int DEF_SPEED = 10;protected static final float DEF_TEXT_SIZE = 30f;private static final float SPEED_RATIO = 0.30f;private static final long DELAY_FLASH = 40L;private Paint paint;private List<BaseText> textList;private SurfaceHolder surfaceHolder;private DrawRun runnable;private Handler drawHandler;protected int width, height;private String cont;private int animMode;private int alignMode;private int bgColr;private int pageIndex;private int padding;private float textSpeed;private float lineSpace;private float textSize;private float textHeight;private float startY;private float offset;private float allHeight;private boolean isHolderReady;public VirgoScrollTextView(Context context) {this(context, null);}public VirgoScrollTextView(Context context, @Nullable AttributeSet attrs) {this(context, attrs, 0);}public VirgoScrollTextView(Context context, @Nullable AttributeSet attrs, int defStyleAttr) {super(context, attrs, defStyleAttr);init();}private void init() {surfaceHolder = getHolder();surfaceHolder.addCallback(this);setZOrderOnTop(true);surfaceHolder.setFormat(PixelFormat.TRANSLUCENT);paint = new Paint();paint.setAntiAlias(true);paint.setStyle(Paint.Style.FILL);paint.setFakeBoldText(false);textList = new ArrayList<>();setDefault();}private void setDefault() {width = 0;height = 0;animMode = MODE_STATIC_DRAW;offset = 0f;pageIndex = 0;padding = 0;lineSpace = 1.3f;alignMode = ALIGN_CENTER;allHeight = 0;if (runnable != null) {runnable.isRolling = false;}isHolderReady = false;setBgColr(DEF_BG);setTextColor(DEF_TEXT_COLOR);setTextSize(DEF_TEXT_SIZE);setTextSpeed(DEF_SPEED);}@Overrideprotected void onSizeChanged(int w, int h, int oldw, int oldh) {super.onSizeChanged(w, h, oldw, oldh);if (w > 0 && h > 0) {initLayout(w, h);setTextCont(this.cont);}}private void initLayout(int width, int height) {this.width = width;this.height = height;this.allHeight = height;Log.d(TAG, "initLayout: " + width + " - " + height);}public void stopRolling() {if (drawHandler != null) {if (runnable != null) {runnable.isRolling = false;runnable = null;}Looper mLooper = drawHandler.getLooper();if (Build.VERSION.SDK_INT >= 18) {mLooper.quitSafely();} else {mLooper.quit();}drawHandler = null;}}public void startRolling() {if (!isHolderReady) {Log.w(TAG, "还没准备好");stopRolling();return;}if (textList == null || textList.isEmpty()) {Log.w(TAG, "文本为空");stopRolling();return;}if (animMode == MODE_STATIC_DRAW) {stopRolling();drawStatic();} else {if (drawHandler == null) {HandlerThread mHandlerThread = new HandlerThread("scheduleText_" + System.currentTimeMillis());mHandlerThread.start();drawHandler = new Handler(mHandlerThread.getLooper());}if (runnable != null && !runnable.isRolling) {runnable.isRolling = true;drawHandler.post(runnable);} else if (runnable == null) {runnable = new DrawRun();runnable.isRolling = true;drawHandler.post(runnable);}}}public void setTextCont(String cont) {if (TextUtils.isEmpty(cont)) {return;}if (width <= 0) {this.cont = cont;return;}stopRolling();textList.clear();textList.addAll(parseCont(cont));offset = 0f;pageIndex = 0;if (animMode == MODE_SCROLL_LEFT) {startY = height / 2f + textHeight / 3;} else {if (textList.size() > 0) {allHeight = textList.size() * (textHeight + lineSpace);} else {allHeight = height;}offset = -height;}}public void setBgColr(int bgColr) {this.bgColr = bgColr;}public void setTextColor(int textColor) {if (paint != null) {paint.setColor(textColor);}}public void setTextSpeed(int textSpeed) {if (textSpeed > MAX_SPEED || textSpeed < MIN_SPEED) {this.textSpeed = DEF_SPEED * SPEED_RATIO;} else {this.textSpeed = textSpeed * SPEED_RATIO;}}public void setTextSize(float textSize) {if (this.textSize == textSize) {return;}this.textSize = textSize;if (paint != null) {paint.setTextSize(textSize);}this.textHeight = getContHeight2();}public void setAnimMode(int animMode) {this.animMode = animMode;}public void setAlignMode(int alignMode) {this.alignMode = alignMode;}public void setPadding(int padding) {this.padding = padding;}public void setLineSpace(float lineSpace) {this.lineSpace = lineSpace;}public void setTextDec(String dec) {if (TextUtils.isEmpty(dec) || dec.equals(Constants.DEC.NONE)) {if (paint != null) {paint.setFlags(Paint.ANTI_ALIAS_FLAG);}return;}if (dec.equals(Constants.DEC.DELETE)) {if (paint != null) {paint.setFlags(Paint.STRIKE_THRU_TEXT_FLAG);}} else if (dec.equals(Constants.DEC.UNDERLINE)) {if (paint != null) {paint.setFlags(Paint.UNDERLINE_TEXT_FLAG);}}}public void setTextItalic(boolean isItalic) {if (paint != null) {paint.setTextSkewX(isItalic ? -0.25f : 0);}}public void setTextBold(boolean isBold) {if (paint != null) {paint.setFakeBoldText(isBold);}}public void release() {stopRolling();if (textList != null) {textList.clear();textList = null;}}public void reset() {stopRolling();textList.clear();setDefault();}private void drawStatic() {Log.d(TAG, "drawStatic: ");if (!textList.isEmpty() && surfaceHolder != null) {Canvas canvas = surfaceHolder.lockCanvas();canvas.drawColor(bgColr, PorterDuff.Mode.CLEAR);canvas.drawColor(bgColr);for (int i = 0; i < textList.size(); i++) {BaseText baseText = textList.get(i);float startX;switch (alignMode) {case ALIGN_LEFT:startX = padding;break;case ALIGN_RIGHT:startX = width - baseText.getLenght() - padding;break;case ALIGN_CENTER:default:startX = (width - baseText.getLenght()) / 2;break;}if (i > 0) {startY = padding + textHeight + i * (textHeight + lineSpace);} else {startY = padding + textHeight;}canvas.drawText(baseText.getCont(), startX, startY, paint);}surfaceHolder.unlockCanvasAndPost(canvas);}}private void drawHorizontal() {if (!textList.isEmpty() && surfaceHolder != null) {offset += textSpeed;int page = textList.size();BaseText baseText1, baseText2, baseText3;if (pageIndex - 2 < 0) {baseText1 = new BaseText("", 0);} else {baseText1 = textList.get((pageIndex - 2) % page);}if (pageIndex - 1 < 0) {baseText2 = new BaseText("", 0);} else {baseText2 = textList.get((pageIndex - 1) % page);}baseText3 = textList.get(pageIndex % page);String curText = baseText1.getCont() + baseText2.getCont() + baseText3.getCont();Canvas canvas = surfaceHolder.lockCanvas();if (canvas != null) {canvas.drawColor(bgColr, PorterDuff.Mode.CLEAR);canvas.drawColor(bgColr);canvas.drawText(curText, width - offset, startY, paint);surfaceHolder.unlockCanvasAndPost(canvas);}if (offset >= (baseText1.getLenght() + baseText2.getLenght() + baseText3.getLenght())) {pageIndex++;if (pageIndex >= page * 2) {pageIndex -= page;}offset -= baseText1.getLenght();}}}private void drawVertical() {if (!textList.isEmpty() && surfaceHolder != null) {offset += textSpeed;int page = textList.size();BaseText baseText1, baseText2;if (page < 2) {baseText1 = textList.get(0);baseText2 = textList.get(0);} else {baseText1 = textList.get((pageIndex) % page);baseText2 = textList.get((pageIndex + 1) % page);}Canvas canvas = surfaceHolder.lockCanvas();if (canvas != null) {canvas.drawColor(bgColr, PorterDuff.Mode.CLEAR);canvas.drawColor(bgColr);float startX;switch (alignMode) {case ALIGN_LEFT:startX = padding;break;case ALIGN_RIGHT:startX = width - baseText1.getLenght() - padding;break;case ALIGN_CENTER:default:startX = (width - baseText1.getLenght()) / 2;break;}startY = padding + textHeight - offset;canvas.drawText(baseText1.getCont(), startX, startY, paint);switch (alignMode) {case ALIGN_LEFT:startX = padding;break;case ALIGN_RIGHT:startX = width - baseText2.getLenght() - padding;break;case ALIGN_CENTER:default:startX = (width - baseText2.getLenght()) / 2;break;}startY = padding + textHeight + height - offset;canvas.drawText(baseText2.getCont(), startX, startY, paint);surfaceHolder.unlockCanvasAndPost(canvas);}if (offset >= height) {pageIndex++;if (pageIndex >= page * 2) {pageIndex -= page;}offset = 0;}}}private void drawVerticalAll() {if (!textList.isEmpty() && surfaceHolder != null) {offset += textSpeed;Canvas canvas = surfaceHolder.lockCanvas();canvas.drawColor(bgColr, PorterDuff.Mode.CLEAR);canvas.drawColor(bgColr);for (int i = 0; i < textList.size(); i++) {BaseText baseText = textList.get(i);float startX;switch (alignMode) {case ALIGN_LEFT:startX = padding;break;case ALIGN_RIGHT:startX = width - baseText.getLenght() - padding;break;case ALIGN_CENTER:default:startX = (width - baseText.getLenght()) / 2;break;}if (i > 0) {startY = padding + textHeight + i * (textHeight + lineSpace) - offset;} else {startY = padding + textHeight - offset;}canvas.drawText(baseText.getCont(), startX, startY, paint);}surfaceHolder.unlockCanvasAndPost(canvas);if (offset >= allHeight) {offset = -height;}}}private class DrawRun implements Runnable {private boolean isRolling;@Overridepublic void run() {while (isRolling) {if (textList == null || textList.isEmpty()) {isRolling = false;break;}try {if (animMode == MODE_SCROLL_LEFT) {drawHorizontal();} else if (animMode == MODE_SCROLL_UP) {drawVertical();} else if (animMode == MODE_SCROLL_UP_ALL) {drawVerticalAll();} else {isRolling = false;break;}} catch (Throwable e) {e.printStackTrace();}try {Thread.sleep(DELAY_FLASH);} catch (InterruptedException e) {
}}}}private List<BaseText> parseCont(String cont) {List<BaseText> tempList = new ArrayList<>();if (animMode == MODE_SCROLL_LEFT || animMode == MODE_SCROLL_UP) {if (animMode == MODE_SCROLL_UP) {cont = cont.replaceAll(" ", TEXT_SPACE).replaceAll("(\r\n|\n|\\\\n)", "").replaceAll("\\\\", "");} else {cont = cont.replaceAll(" ", TEXT_SPACE).replaceAll("(\r\n|\n|\\\\n)", TXT_ENTER).replaceAll("\\\\", "");}cont = cont.replaceAll(" ", TEXT_SPACE).replaceAll("(\r\n|\n|\\\\n)", TXT_ENTER).replaceAll("\\\\", "");try {if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.N) {cont = Html.fromHtml(cont, Html.FROM_HTML_MODE_LEGACY).toString();} else {cont = Html.fromHtml(cont).toString();}} catch (Throwable e) {}float allLen = getContWidth2(cont);Log.d(TAG, "控件宽: " + width + ", 文本总长度: " + allLen + ", 高度: " + textHeight + ", len: " + cont.length());if (animMode == MODE_SCROLL_LEFT) {if (allLen < width) {String baseCont = cont;int times = (int) (width / allLen) + 1;StringBuilder contBuilder = new StringBuilder(cont);for (int i = 0; i < times; i++) {contBuilder.append(TEXT_SPACE);contBuilder.append(baseCont);}cont = contBuilder.toString();allLen = getContWidth2(cont);}}BaseText baseText;int pLen = (int) Math.floor(width / allLen * cont.length());int start = 0;do {int end = start + pLen;String str;if (end > cont.length()) {int num = end - cont.length();if (num > pLen / 2) {num = num - pLen / 2;}end = cont.length();str = cont.substring(start, end);for (int i = 0; i < num; i++) {str = str.concat(TEXT_SPACE);}} else {str = cont.substring(start, end);}baseText = new BaseText();baseText.setCont(str);baseText.setLenght(getContWidth2(str));tempList.add(baseText);start = end;} while (start < cont.length());Log.d(TAG, "每页需文字长度: " + pLen + ", 分页数: " + tempList.size());} else {cont = cont.replaceAll(" ", TEXT_SPACE).replaceAll("\\\\n|\n", "_n").replaceAll("\\\\", "");int lineWidth = (int) (width - 2 * padding - 2 * textSize);if (cont.contains("_n")) {String[] arr = cont.split("_n");for (String line : arr) {BaseText baseText;float lineLen = getContWidth2(line);if (lineLen > lineWidth) {int pLen = (int) Math.floor(lineWidth / lineLen * line.length());int start = 0;do {int end = start + pLen;String str;if (end > line.length()) {int num = end - line.length();if (num > pLen / 2) {num = num - pLen / 2;}end = line.length();str = line.substring(start, end);for (int i = 0; i < num; i++) {str = str.concat(TEXT_SPACE);}} else {str = line.substring(start, end);}baseText = new BaseText();baseText.setCont(str);baseText.setLenght(getContWidth2(str));tempList.add(baseText);start = end;} while (start < line.length());} else {baseText = new BaseText();baseText.setCont(line);baseText.setLenght(lineLen);tempList.add(baseText);}}} else {float allLen = getContWidth2(cont);BaseText baseText;int pLen = (int) Math.floor(lineWidth / allLen * cont.length());int start = 0;do {int end = start + pLen;String str;if (end > cont.length()) {int num = end - cont.length();if (num > pLen / 2) {num = num - pLen / 2;}end = cont.length();str = cont.substring(start, end);for (int i = 0; i < num; i++) {str = str.concat(TEXT_SPACE);}} else {str = cont.substring(start, end);}baseText = new BaseText();baseText.setCont(str);baseText.setLenght(getContWidth2(str));tempList.add(baseText);start = end;} while (start < cont.length());}}return tempList;}private float getContWidth(String txt) {TextPaint temp = new TextPaint();temp.setTextAlign(Paint.Align.LEFT);temp.setStyle(Paint.Style.FILL);temp.setTextSize(textSize);return Layout.getDesiredWidth(txt, temp);}private float getContWidth2(String txt) {return paint.measureText(txt);}private float getContHeight() {Paint temp = new Paint();temp.setTextAlign(Paint.Align.CENTER);temp.setTextSize(textSize);Paint.FontMetrics fm = temp.getFontMetrics();return (float) Math.ceil(fm.descent - fm.ascent);}private float getContHeight2() {Paint.FontMetrics fm = paint.getFontMetrics();return (float) Math.ceil(fm.descent - fm.ascent);}@Overridepublic void surfaceCreated(SurfaceHolder holder) {Log.d(TAG, "surfaceCreated: ");isHolderReady = true;startRolling();}@Overridepublic void surfaceChanged(SurfaceHolder holder, int format, int width, int height) {}@Overridepublic void surfaceDestroyed(SurfaceHolder holder) {isHolderReady = false;stopRolling();}static class BaseText implements Serializable {private String cont;private float lenght;public BaseText() {}public BaseText(String cont, float lenght) {this.cont = cont;this.lenght = lenght;}public String getCont() {return cont;}public void setCont(String cont) {this.cont = cont;}public float getLenght() {return lenght;}public void setLenght(float lenght) {this.lenght = lenght;}}
}