手把手教你Android自定义动画(新手向)

先来看最终效果图

LineAnimation.gif

整个效果图包括了

Share Element Transition
CollapsingToolbarLayout
AppbarLayout
自定义的动画效果

先来分析自定义动画,用两个TextView来分别显示Title(动物世界)和SubTitle(春天到了...)。SubTitle的动画需要自己手动来画,可以直接沿着TextView的周长来画线。
我们需要重写View的onDraw方法,然后在方法体里来画我们的图像,然后通过调用invalidate来刷新View让画面动起来。
线框的绘画其实就是由5条线组合在一起,这里用我用bottom,left,right,topLeft,topRight来代表这5条线,我们简单先给topLeft定义一个确定的长度length 等于TextView的宽度1/4。
为了让这些白线有一定的粗细,我们用canvas.drawRect来画线而不是用drawLine,设定线的宽度stokeWidth。


所以线框的完全体就是这样

length = getWidth()/4;
stokeWidth = 2;
@Override
protected void onDraw(Canvas canvas) {
        super.onDraw(canvas);
        //bottom
        canvas.drawRect(0, getHeight() - stokeWidth, getWidth(), getHeight(), paint);
        //left
        canvas.drawRect(0, 0, stokeWidth, getHeight(), paint);
        //right
        canvas.drawRect(getWidth() - stokeWidth, 0, getWidth(), getHeight(), paint);
        //topLeft
        canvas.drawRect(0, 0, length, stokeWidth, paint);
        //topRight
        canvas.drawRect(getWidth() - length, 0, getWidth(), stokeWidth, paint);
}

然后实现动画:
计算线框的总长度(或者说绘制完成时的长度) maxRound
动画过程中线框长度 currentRound
设置绘制比例值 percent
通过比较currentRound的值画出对应帧的线框图像

    @Override
    protected void onDraw(Canvas canvas) {
        super.onDraw(canvas);
        float currentRound = maxRound * percent;
        if (currentRound <= stokeWidth) {
            return;
        }
        //currentRound长度小于bottom时,只画出底部线条
        if (currentRound <= getWidth()) {
            canvas.drawRect((getWidth() - currentRound) / 2, getHeight() - stokeWidth, (getWidth() + currentRound) / 2, getHeight(), paint);
        //blr=bottom+left+right,当currentRound小于底边和左右两边加起来的时候,画出对应的线条
        } else if (currentRound <= blr) {
            canvas.drawRect(0, getHeight() - stokeWidth, getWidth(), getHeight(), paint);
            float y = getHeight() - (currentRound - getWidth()) / 2;
            canvas.drawRect(0, y, stokeWidth, getHeight(), paint);
            canvas.drawRect(getWidth() - stokeWidth, y, getWidth(), getHeight(), paint);
        } else {
            //bottom
            canvas.drawRect(0, getHeight() - stokeWidth, getWidth(), getHeight(), paint);
            //left
            canvas.drawRect(0, 0, stokeWidth, getHeight(), paint);
            //right
            canvas.drawRect(getWidth() - stokeWidth, 0, getWidth(), getHeight(), paint);
            //blr为bottom,left,right加起来的长度
            float r = (currentRound - blr) / 2;
            //topLeft
            canvas.drawRect(0, 0, r, stokeWidth, paint);
            //topRight
            canvas.drawRect(getWidth() - r, 0, getWidth(), stokeWidth, paint);
        }
    }
    //通过属性动画来画出帧动画
    public void animate(float f) {
        ValueAnimator valueAnimator = new ValueAnimator();
        valueAnimator.setFloatValues(percent, f);
        valueAnimator.setDuration(500);
        valueAnimator.addUpdateListener(this);
        valueAnimator.start();
    }
    //监听属性动画回调改变绘画的线框百分比percent
    @Override
    public void onAnimationUpdate(ValueAnimator animation) {
        percent= (float) animation.getAnimatedValue();
        setTextColor(Color.argb((int) (percent * 255), 255, 255, 255));
        invalidate();
    }

顺手给subTitle文字加个渐变,效果大概就是这样


pic2.gif

接下来是让动画和AppbarLayout的滑动事件关联起来
xml布局

<?xml version="1.0" encoding="utf-8"?>
<android.support.design.widget.CoordinatorLayout xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:app="http://schemas.android.com/apk/res-auto"
    xmlns:tools="http://schemas.android.com/tools"
    android:layout_width="match_parent"
    android:layout_height="match_parent">

    <android.support.design.widget.AppBarLayout
        android:id="@+id/app_bar"
        android:layout_width="match_parent"
        android:layout_height="150dp"
        app:elevation="0dp">

        <android.support.design.widget.CollapsingToolbarLayout
            android:id="@+id/collapsingToolbar"
            android:layout_width="match_parent"
            android:layout_height="match_parent"
            app:expandedTitleMarginEnd="64dp"
            app:expandedTitleMarginStart="48dp"
            app:layout_scrollFlags="scroll|exitUntilCollapsed|snap">

            <com.facebook.drawee.view.SimpleDraweeView
                android:id="@+id/bgImageView"
                android:layout_width="match_parent"
                android:layout_height="match_parent"
                app:layout_collapseMode="parallax"
                app:layout_collapseParallaxMultiplier="0.5"
                tools:background="#cccccc" />

            <View
                android:layout_width="match_parent"
                android:layout_height="match_parent"
                android:background="#55000000" />

            <android.support.v7.widget.Toolbar
                android:id="@+id/toolbar"
                android:layout_width="match_parent"
                android:layout_height="?attr/actionBarSize"
                app:contentInsetLeft="0dp"
                app:contentInsetStart="0dp"
                app:layout_collapseMode="pin" />

            <TextView
                android:id="@+id/title"
                android:layout_width="wrap_content"
                android:layout_height="wrap_content"
                android:layout_gravity="center_horizontal"
                android:layout_marginTop="48dp"
                android:maxLines="1"
                android:paddingLeft="8dp"
                android:paddingRight="8dp"
                android:textColor="@color/white"
                android:textSize="18dp"
                android:textStyle="bold"
                app:layout_collapseMode="parallax"
                tools:text="走进科学" />

        </android.support.design.widget.CollapsingToolbarLayout>

    </android.support.design.widget.AppBarLayout>

    <android.support.v7.widget.RecyclerView
        android:id="@+id/recyclerView"
        android:layout_width="match_parent"
        android:layout_height="match_parent"
        app:layout_behavior="@string/appbar_scrolling_view_behavior" />

    <etong.lineanimation.SubTitleView
        android:id="@+id/subTitle"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:layout_gravity="center_horizontal"
        android:layout_marginTop="58dp"
        android:gravity="center"
        android:maxLines="2"
        android:paddingBottom="12dp"
        android:paddingLeft="20dp"
        android:paddingRight="20dp"
        android:paddingTop="20dp"
        android:textColor="@color/white"
        android:textSize="14dp"
        tools:text="春天到了,又到了交配的季节。" />


    <Button
        android:id="@+id/button"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:text="show"
        android:layout_gravity="bottom"/>
</android.support.design.widget.CoordinatorLayout>

设置AppBarLayout监听
appBar.addOnOffsetChangedListener(this);
根据滑动距离来和最大滑动距离来计算线框绘制的百分比

    @Override
    public void onOffsetChanged(AppBarLayout appBarLayout, int verticalOffset) {
        if (!TextUtils.isEmpty(subTitle)) {
            float value = appBar.getHeight() - toolbar.getHeight() * 2;
            float p = (value + verticalOffset) / value;
            if (p < 0) {
                p = 0;
            }
            subTitleView.setPercent(p);
        }
   }

这里线框的位置是固定的,所以在appBarLayout滑动的时候为了避免线框出界,需要线框提前一点消失,所以本来最大滑动距离为
appBar.getHeight()-toolbar.getHeight()
改为
appBar.getHeight() - toolbar.getHeight() * 2
效果


pic3.gif

接下来需要处理一下SubTtileView本身高宽和主标题Title所在TextView的宽度问题。

  • Title最大行数为一行,SubTitle最大行数为两行
  • SubTtileView本身的最大宽度也必须小于屏幕宽度,而且需要有最小的左右边距 padding
  • SubTitleView的topLeft和topRight必须留有一点的长度,才能形成一个完整的线框 minLength
  • Title文字过多的时候会占满屏幕,所以必须限制Title的最大宽度
    maxTitleWidth = screenWidth - padding * 2 - minLength * 2
  • SubTitleView的宽度必须大于Title的宽度,所以我们要根据Title的宽度手动计算SubTitleView的宽度
    minSubTitleViewWidth = titleWidth - minLenght * 2
  • ............
subtitleMargin = getResources().getDimensionPixelSize(R.dimen.subtitle_margin);
minLength = getResources().getDimensionPixelSize(R.dimen.min_length);
maxTitleWidth = ScreenUtils.screenWidth - subtitleMargin * 2 - minLength * 2;

    public void calculateSubTitle() {
        int titleWidth = titleTextView.getWidth();

        if (titleWidth > maxTitleWidth) {
            titleTextView.getLayoutParams().width = maxTitleWidth;
            subtitleView.getLayoutParams().width = maxTitleWidth + minLength * 2;
            subtitleView.setLength(minLength * 2);
            subtitleView.requestLayout();
            titleTextView.requestLayout();
        } else if (subtitleView.getWidth() > ScreenUtils.screenWidth - subtitleMargin * 2) {
            subtitleView.getLayoutParams().width = ScreenUtils.screenWidth - subtitleMargin * 2;
            subtitleView.setLength(subtitleView.getLayoutParams().width - titleWidth);
            subtitleView.requestLayout();
        } else if (subtitleView.getWidth() < titleWidth + minLength * 2) {
            subtitleView.getLayoutParams().width = titleWidth + minLength * 2;
            subtitleView.setLength(subtitleView.getLayoutParams().width - titleWidth);
            subtitleView.requestLayout();
        } else {
            subtitleView.setLength(subtitleView.getLayoutParams().width - titleWidth);
            subtitleView.requestLayout();
        }
    }

到此所有的自定义部分就写完了,剩下的过场动画由于是Android支持库的api就不多解释。
最后,源码链接 https://github.com/RoyWallace/LineAnimation
如果发现有错误的地方或者不明白之处欢迎加群讨论 qq群:295456349

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 204,293评论 6 478
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 85,604评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 150,958评论 0 337
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,729评论 1 277
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,719评论 5 366
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,630评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 38,000评论 3 397
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,665评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,909评论 1 299
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,646评论 2 321
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,726评论 1 330
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,400评论 4 321
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,986评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,959评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,197评论 1 260
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 44,996评论 2 349
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,481评论 2 342

推荐阅读更多精彩内容

  • 内容抽屉菜单ListViewWebViewSwitchButton按钮点赞按钮进度条TabLayout图标下拉刷新...
    皇小弟阅读 46,705评论 22 664
  • 最近做了一个Android UI相关开源项目库汇总,里面集合了OpenDigg 上的优质的Android开源项目库...
    OpenDigg阅读 17,160评论 6 223
  • Android 自定义View的各种姿势1 Activity的显示之ViewRootImpl详解 Activity...
    passiontim阅读 171,428评论 25 707
  • 小时候雪是一个懵懂无知的小女孩儿她轻柔,优雅有时,她很害怕害怕因为自己让整个多彩的世界迷失自我! 长大后 ...
    梦见梦里梦说梦阅读 242评论 0 0
  • 昨天家里停电。本来想着烛火下写字应该也挺有氛围,后来觉得桌上易燃物太多,遂作罢。 落花诗第二首: 飘飘荡荡复悠悠,...
    苏白杞阅读 515评论 0 2