我正在使用以下Spannable
和TextView
,我已经让它为每个角色制作动画,但我想为每个单词制作动画,我该如何实现?
寻找alpha并翻译每个单词(从每个单词的位置底部开始)。
通过翻译我的意思是翻译(移动)动画。
来源:
private static class FadeyTextView extends TextView
{
private Interpolator mInterpolator;
private long mStart, mDurationPerLetter;
private boolean mAnimating = false;
private SpannableString mFadeyText;
private CharSequence mText;
public FadeyTextView(Context context)
{
super(context);
initView();
}
public FadeyTextView(Context context, AttributeSet attrs)
{
super(context, attrs);
initView();
}
public FadeyTextView(Context context, AttributeSet attrs, int defStyle)
{
super(context, attrs, defStyle);
initView();
}
private void initView()
{
// Set defaults
mInterpolator = new DecelerateInterpolator();
mDurationPerLetter = 250;
}
public void setInterpolator(Interpolator interpolator)
{
mInterpolator = interpolator;
}
public void setDurationPerLetter(long durationPerLetter)
{
mDurationPerLetter = durationPerLetter;
}
@Override
public void setText(CharSequence text, BufferType type)
{
mText = text;
mFadeyText = new SpannableString(text);
FadeyLetterSpan[] letters = mFadeyText.getSpans(0, mFadeyText.length(), FadeyLetterSpan.class);
for(FadeyLetterSpan letter : letters){
mFadeyText.removeSpan(letter);
}
final int length = mFadeyText.length();
for(int i = 0; i < length; i++){
mFadeyText.setSpan(new FadeyLetterSpan(), i, i + 1, Spanned.SPAN_INCLUSIVE_EXCLUSIVE);
}
super.setText(mFadeyText, BufferType.SPANNABLE);
mAnimating = true;
mStart = AnimationUtils.currentAnimationTimeMillis();
ViewCompat.postInvalidateOnAnimation(this);
}
@Override
@CapturedViewProperty
public CharSequence getText()
{
return mText;
}
public boolean isAnimating()
{
return mAnimating;
}
@Override
protected void onDraw(Canvas canvas)
{
super.onDraw(canvas);
if(mAnimating){
long mDelta = AnimationUtils.currentAnimationTimeMillis() - mStart;
FadeyLetterSpan[] letters = mFadeyText.getSpans(0, mFadeyText.length(), FadeyLetterSpan.class);
final int length = letters.length;
for(int i = 0; i < length; i++){
FadeyLetterSpan letter = letters[i];
float delta = (float)Math.max(Math.min((mDelta - (i * mDurationPerLetter)), mDurationPerLetter), 0);
letter.setAlpha(mInterpolator.getInterpolation(delta / (float)mDurationPerLetter));
}
if(mDelta < mDurationPerLetter * length){
ViewCompat.postInvalidateOnAnimation(this);
}else{
mAnimating = false;
}
}
}
private class FadeyLetterSpan extends CharacterStyle implements UpdateAppearance
{
private float mAlpha = 0.0f;
public void setAlpha(float alpha)
{
mAlpha = Math.max(Math.min(alpha, 1.0f), 0.0f);
}
@Override
public void updateDrawState(TextPaint tp)
{
int color = ((int)(0xFF * mAlpha) << 24) | (tp.getColor() & 0x00FFFFFF);
tp.setColor(color);
}
}
所以我现在让它为每个单词设置动画而不是每个字符,我这样做的方法是使用每个单词的包装器来保存其开始和结束索引值:
private static class IndexWrapper {
private int mStart;
private int mEnd;
public IndexWrapper(int start, int end) {
this.mStart = start;
this.mEnd = end;
}
public int getStart() {
return mStart;
}
public int getEnd() {
return mEnd;
}
@Override
public int hashCode() {
final int prime = 31;
int result = 1;
result = prime * result + mEnd;
result = prime * result + mStart;
return result;
}
@Override
public boolean equals(Object obj) {
if (this == obj)
return true;
if (obj == null)
return false;
if (getClass() != obj.getClass())
return false;
IndexWrapper other = (IndexWrapper) obj;
if (mEnd != other.mEnd)
return false;
if (mStart != other.mStart)
return false;
return true;
}
}
然后在此函数中调用此函数以生成List<IndexWrapper> indices
:
private List<IndexWrapper> getAllIndexes(CharSequence text) {
// First convert to a string and return an empty array if no words
String s = text.toString();
if (TextUtils.isEmpty(s))
return null;
// Get the number of words
String[] words = text.toString().split("\\s+");
// For each word we need to find the indices and create FadeWordSpans for them
List<IndexWrapper> indices = new ArrayList<>();
for (String word : words) {
// Get our sub index values for each word
List<IndexWrapper> subIndices = findIndexesForEachWord(s, word);
// Add it to the list of indices
indices.addAll(subIndices);
}
// Return our list of all index wrappers
return indices;
}
private List<IndexWrapper> findIndexesForEachWord(String wholeString, String word) {
String regex = "\\b" + word + "\\b";
Pattern pattern = Pattern.compile(regex);
Matcher matcher = pattern.matcher(wholeString);
List<IndexWrapper> indices = new ArrayList<>();
while (matcher.find()) {
// Get start and end indexes
int start = matcher.start();
int end = matcher.end();
// Create a new index wrapper and add it to the our list
indices.add(new IndexWrapper(start, end));
}
return indices;
}
我们拥有List<IndexWrapper> indices
后,我们就可以在setText(CharSequence text, BufferType type)
函数中调用它,如下所示:
@Override
public void setText(CharSequence text, BufferType type) {
// Go ahead and set our text
this.mText = text;
// We need to initialize our SpannableString
this.mFadeText = new SpannableString(text);
// Go ahead and set our characters styles
FadeWordSpan[] letters = mFadeText.getSpans(0, mFadeText.length(), FadeWordSpan.class);
// For each word remove the span
for (FadeWordSpan fadeWordSpan : letters)
mFadeText.removeSpan(fadeWordSpan);
List<IndexWrapper> indices = getAllIndexes(mFadeText.toString());
final int length = indices != null ? indices.size() : 0;
for (int i = 0; i < length; i++) {
// Get index
IndexWrapper index = indices.get(i);
int start = index.getStart();
int end = index.getEnd();
mFadeText.setSpan(new FadeWordSpan(), start, end, Spanned.SPAN_INCLUSIVE_EXCLUSIVE);
}
super.setText(mFadeText, BufferType.SPANNABLE);
// Set animating to true
this.mIsAnimating = true;
// Set our start time and redraw on animation
mStart = AnimationUtils.currentAnimationTimeMillis();
ViewCompat.postInvalidateOnAnimation(this);
}
然后我们可以在onDraw(Canvas canvas)
函数中为每个单词设置动画,如下所示:
// If the view is animating (meaning we set the text) go ahead and draw the animations
if (mIsAnimating) {
long delta = AnimationUtils.currentAnimationTimeMillis() - mStart;
// Initialize our word span
FadeWordSpan[] words = mFadeText.getSpans(0, mFadeText.length(), FadeWordSpan.class);
// Get our length
final int length = words.length;
// For each letter animate in the word
for (int i = 0; i < length; i++){
// Get our word
FadeWordSpan word = words[i];
// Calculate our delta
float dx = (float) Math.max(Math.min((delta - (i * mDurationPerWord)), mDurationPerWord), 0);
// Set our interpolation alpha
word.setAlpha(mInterpolator.getInterpolation(dx / (float) mDurationPerWord));
}
// If our delta is less than the duration per word times the length then we need to keep re-drawing
if (delta < mDurationPerWord * length)
ViewCompat.postInvalidateOnAnimation(this);
else
mIsAnimating = false;
}
FadeWordSpan
与FadeyLetterSpan
相同。