安卓桌面动画

anim

场景:点击桌面图标,图标放大,然后圆形进度旋转一圈结束,时间可控~~~

先理一下,启动"桌面动画"app,第一次启动在launcher上创建一个"点我点我"快捷图标,点击这个图标,动画开始!

1.创建桌面图标

 public class App extends Application {
    private static Context context;
    public static Context getContext() {
        return context;
    }
    public void onCreate() {
        super.onCreate();
        context = this;
        if (Config.getIsFirstLaunch()) {
            Intent intent = new Intent();
            intent.setClass(this, LauncherActivity.class);
            intent.setAction("com.snow.action.start");
            ShortcutUtils.buildShortcut("点我点我", R.drawable.widget, intent, this);
            Config.setIsFirstLaunch(false);
        }
    }
}

创建的时候用了一个异步线程,交给ScheduledThreadPoolExecutor执行器,首先要判断一下图标是否存在,创建过程:
给launcher发消息

    private static boolean addShortcut(String name, int iconId, Context context, Intent intent) {
        Intent intent1 = new Intent("com.android.launcher.action.INSTALL_SHORTCUT");
        intent1.putExtra("duplicate", false);
        intent1.putExtra("android.intent.extra.shortcut.NAME", name);
        intent1.putExtra("android.intent.extra.shortcut.ICON", BitmapFactory.decodeResource(context.getResources(),
                iconId));
        intent1.putExtra("android.intent.extra.shortcut.ICON_RESOURCE", Intent.ShortcutIconResource.fromContext(
                context, iconId));
        intent1.putExtra("android.intent.extra.shortcut.INTENT", intent);
        intent.addFlags(Intent.FLAG_ACTIVITY_NO_ANIMATION);//65536
        context.sendBroadcast(intent1);
        return true;
    }

一条权限:

<uses-permission android:name="com.android.launcher.permission.INSTALL_SHORTCUT" />

好啦,点击图标,LauncherActivity启动,这个Activity是singleInstance单例模式,然后关键一点,主题配置成透明样式

    <style name="Transparent" parent="@style/Theme.AppCompat">
        <item name="android:windowBackground">@android:color/transparent</item>
        <item name="android:windowNoTitle">true</item>
        <item name="android:windowIsTranslucent">true</item>
        <item name="android:windowAnimationStyle">@null</item>
        <item name="android:actionBarStyle">@null</item>
        <item name="actionBarStyle">@null</item>
    </style>

接着看下布局

 <?xml version="1.0" encoding="utf-8"?>
<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
    android:id="@+id/root"
    android:layout_width="match_parent"
    android:layout_height="match_parent">


    <com.snow.cc.views.CircleAnimView
        android:id="@+id/rocket_view"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:background="@drawable/widget" />

</RelativeLayout>

没错就是一个相对布局,加一个自定义view.
widget是她的背景图,也就是火箭

背景

问题来了,既然是相对布局,而动画效果,是背景图刚好在桌面icon的正上方(Z轴上来说),感觉就是桌面图标自己在变化,其实不然,只是公用了一个图片而已,那么怎么调整rocket_view的位置呢?

Rect rect = intent.getSourceBounds();//获取icon坐标信息
这里,桌面启动的时候,Launcher会通过setSourceBounds方法,设置图标的坐标信息并通过Intent发送出去,原来Launcher这么会玩...

有了Rect就可以调整rocket_view的位置了,考虑StatusBar的高度

    private RelativeLayout.LayoutParams computeAnimationIconLayoutParams(View icon, Rect rect) {
        RelativeLayout.LayoutParams layoutParams = (RelativeLayout.LayoutParams) icon.getLayoutParams();
        layoutParams.topMargin = rect.top - this.getStatusBarHeight();
        layoutParams.leftMargin = rect.left;
        layoutParams.width = rect.right - rect.left;
        layoutParams.height = layoutParams.width;
        return layoutParams;
    }

好了,进入动画知识了

rocketView.startAnimation(360, new Animation.AnimationListener() {
     @Override
     public void onAnimationEnd(Animation animation) {
                    finish();//动画结束,自己退出
                    overridePendingTransition(0, 0);
     }
     ...
 });

360度刚好一圈,结束了,使命就完成了额~~~

自定义CircleAnimView继承自View

这里扫描角从0度到360度,借助了动画api

   class OpenAnimation extends Animation {
        float sweepAngle;

        public OpenAnimation(int sweepAngle, long duration) {
            super();
            this.sweepAngle = ((float) sweepAngle);
            this.setDuration(duration);
            this.setInterpolator(new AccelerateDecelerateInterpolator());
        }

        @Override
        protected void applyTransformation(float interpolatedTime, Transformation t) {
            CircleAnimView.this.currentAngle = this.sweepAngle * interpolatedTime;
            CircleAnimView.this.invalidate();
            Log.e(TAG, "OpenAnimation :" + interpolatedTime + "  " + currentAngle);
            ...
        }
    }

在绘制Animation的过程中会反复的调用applyTransformation函数,每次调用参数interpolatedTime值都会变化,从0渐 变为1,当该参数为1时表明动画结束。通过参数Transformation 来获取变换的矩阵(matrix),通过改变矩阵就可以实现各种复杂的效果。

但是这里只关心interpolatedTime,初始话的时候sweepAngle扫描角度为360度,当然是可控的

rocketView.startAnimation(360, new Animation.AnimationListener() {...});

动画的插值器指定为AccelerateDecelerateInterpolator:

/** * An interpolator where the rate of change starts and ends slowly but * accelerates through the middle. */

在动画开始与结束的地方速率改变比较慢,在中间的时候加速

每次计算当前扫描角度,然后invalidate(),会引起View的
onDraw(Canvas canvas) 方法调用,那我们就在这个方法里根据当前弧度计算圆弧和小白点的位置:

    private void computePosition() {
        int w = ((int) ((((double) this.getWidth())) * RATE_CIRCLE));//进度环所在区域宽度
        int h = ((int) ((((double) this.getHeight())) * RATE_CIRCLE));//进度环所在区域高度
        int diameter = w > h ? h : w;//调整圆环的直径
        this.centerX = ((float) (this.getWidth() / 2));//中心点x
        this.centerY = ((float) (this.getHeight() / 2));//中心点y
        this.radius = ((float) (diameter / 2));//圆环的半径
        this.endPointRadius = ((int) ((((double) this.radius)) * RATE_RADIUS_ENDPOINT));//小白球的半径
        this.paintWidth = ((int) ((((double) this.radius)) * RATE_RADIUS_PAINT_WIDTH));
        this.left = this.centerX - this.radius;//圆环的左边距
        this.right = this.centerX + this.radius;//圆环的右边距
        this.top = this.centerY - this.radius;//圆环的顶边距
        this.bottom = this.centerY + this.radius;//圆环的底边距
    }

位置,尺寸都有了,就可以用画笔画了:
画圆弧进度

canvas.drawArc(new RectF(this.left, this.top, this.right, this.bottom), ZERO_ANGLE, this.currentAngle,
          false, this.progressPaint);

画小白球

     private void drawProgressBarPoint(Canvas canvas) {
        float angle = inOpeningAnimation ? currentAngle : endAngle;
        this.endX = (float) Math.sin(Math.toRadians(angle)) * radius + centerX;
        this.endY = (float) Math.sin(Math.toRadians(angle) - Math.PI / 2) * radius + centerY;
        canvas.drawCircle(endX, endY, ((float) endPointRadius), endPointPaint);//1.570796
    }

这里要把角度转换成对应的弧度值,靠Math.toRadians这个方法了
再回顾一下弧度和角度,我发现这些都还给数学老师了@~@


它们的关系可用下式表示和计算: 角(弧度)= 弧长/半径
圆的周长是半径的 2π倍,所以一个周角(360度)是 2π弧度。 半圆的长度是半径的 π倍,所以一个平角(180度)是 π弧度。

定了中心点和半径,有了画笔,就drawCircle啦!!!

项目点这里:
https://github.com/shonegg/DesktopCircleAnimation

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

推荐阅读更多精彩内容

  • Android 自定义View的各种姿势1 Activity的显示之ViewRootImpl详解 Activity...
    passiontim阅读 171,922评论 25 707
  • 发现 关注 消息 iOS 第三方库、插件、知名博客总结 作者大灰狼的小绵羊哥哥关注 2017.06.26 09:4...
    肇东周阅读 12,077评论 4 62
  • 何以解忧 唯有暴富 醉知酒浓 醒知梦空 人分两类 迷人乏味 说戏人笑 听戏人哭 玩就大方 爱就坦荡 生死看淡 不...
    故梦j阅读 757评论 17 9
  • 专项储备这个名词在听课过程中出现的一个名词,相对来说 自己并没有完全的理解到这个名词的真正含义,能够了解到的地方大...
    锦潇阅读 752评论 0 2
  • 读了黔上听香的《在最美的地方遇见你》,国内第一本旅游书信形式暖心故事。并不熟悉黔上听香,百度上搜索一下,是一个很书...
    空八一五阅读 263评论 0 0