小米平板卸载软件的时候,会有一个粉碎的效果,看起来很拉风,GitHub上有一个开源控件可以实现这个效果,我们一起来看看。先来看看效果图:

看起来不错吧,那我们今天就来详细说说ExplosionField的使用和它内部的实现原理吧。

1.获得一个ExplosionField

ExplosionField是GitHub上的一个开源项目,我们直接在GitHub上下载就可以了。

下载地址https://github.com/tyrantgit/ExplosionField

下载好之后我们直接将tyrantgit文件夹拷贝到我们的项目中就可以使用了,该文件夹的位置在:ExplosionField-master\ExplosionField-master\explosionfield\src\main\java中。

注意:使用ExplosionField要求我们的jre版本在1.7以上。

2.ExplosionField的使用

布局文件很简单,就是四个ImageView:

<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:tools="http://schemas.android.com/tools"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    android:orientation="vertical" >

    <LinearLayout
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:layout_weight="1"
        android:gravity="center"
        android:orientation="horizontal" >

        <ImageView
            android:id="@+id/iv1"
            android:layout_width="72dp"
            android:layout_height="72dp"
            android:layout_weight="1"
            android:onClick="onClick"
            android:src="@drawable/p1" />

        <ImageView
            android:id="@+id/iv2"
            android:layout_width="72dp"
            android:layout_height="72dp"
            android:layout_weight="1"
            android:onClick="onClick"
            android:src="@drawable/p2" />
    </LinearLayout>

    <LinearLayout
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:layout_weight="1"
        android:orientation="horizontal" >

        <ImageView
            android:id="@+id/iv3"
            android:layout_width="72dp"
            android:layout_height="72dp"
            android:layout_weight="1"
            android:onClick="onClick"
            android:src="@drawable/p3" />

        <ImageView
            android:id="@+id/iv4"
            android:layout_width="72dp"
            android:layout_height="72dp"
            android:layout_weight="1"
            android:onClick="onClick"
            android:src="@drawable/p4" />
    </LinearLayout>

</LinearLayout>

在MainActivity中,我们拿到一个ExplosionField的实例,然后通过执行它的explode方法就可以实现View的粉碎效果了。

public class MainActivity extends Activity {

	private ExplosionField explosionField;
	private ImageView iv1, iv2, iv3, iv4;

	@Override
	protected void onCreate(Bundle savedInstanceState) {
		super.onCreate(savedInstanceState);
		setContentView(R.layout.activity_main);
		iv1 = (ImageView) this.findViewById(R.id.iv1);
		iv2 = (ImageView) this.findViewById(R.id.iv2);
		iv3 = (ImageView) this.findViewById(R.id.iv3);
		iv4 = (ImageView) this.findViewById(R.id.iv4);
		explosionField = ExplosionField.attach2Window(this);
	}

	public void onClick(View v) {
		switch (v.getId()) {
		case R.id.iv1:
			explosionField.explode(iv1);
			iv1.setVisibility(View.INVISIBLE);
			break;
		case R.id.iv2:
			explosionField.explode(iv2);
			iv2.setVisibility(View.INVISIBLE);
			break;
		case R.id.iv3:
			explosionField.explode(iv3);
			iv3.setVisibility(View.INVISIBLE);
			break;
		case R.id.iv4:
			explosionField.explode(iv4);
			iv4.setVisibility(View.INVISIBLE);
			break;
		}
	}
}

使用起来是如此的方便,这里我要特别说一句,为什么粉碎完成之后还要将控件隐藏,这是因为View粉碎之后,我们就看不到了,但是实际上它还在页面上,如果这个View本身有点击事件,那么这个时候你点击它粉碎前的位置,点击事件还是能够响应,将控件隐藏之后就可以避免这个问题了。说了这么多,我们来看看这个效果究竟是怎么实现的。

3.源码解读

通过前面两步,我们已经知道,跟这个效果有关的就是三个类,那么我们这里就从explosionField = ExplosionField.attach2Window(this);这行代码所执行的attach2Window(this);方法看起:

	public static ExplosionField attach2Window(Activity activity) {
		ViewGroup rootView = (ViewGroup) activity
				.findViewById(Window.ID_ANDROID_CONTENT);
		ExplosionField explosionField = new ExplosionField(activity);
		rootView.addView(explosionField, new ViewGroup.LayoutParams(
				ViewGroup.LayoutParams.MATCH_PARENT,
				ViewGroup.LayoutParams.MATCH_PARENT));
		return explosionField;
	}

首先我们要知道,ExplosionField继承自View,它本身就是一个View,那么在attach2Window方法中,我们先拿到一个rootView,这个rootView其实就是我们setContentView(R.layout.activity_main);所设置的View,然后我们new一个ExplosionField并把它添加到rootView当中,并把它的宽高都设置为MATCH_PARENT,这样就可以保证View爆炸产生的碎片可以被绘制在整个Activity区域。

好了,拿到一个ExplosionField对象之后,我们就可以粉碎一个View了,粉碎View执行的方法是explode,这个方法需要我们把要粉碎的View作为一个参数传入进去,我们看看这个方法:

	public void explode(final View view) {
		Rect r = new Rect();
		view.getGlobalVisibleRect(r);
		int[] location = new int[2];
		getLocationOnScreen(location);
		r.offset(-location[0], -location[1]);
		r.inset(-mExpandInset[0], -mExpandInset[1]);
		int startDelay = 100;
		ValueAnimator animator = ValueAnimator.ofFloat(0f, 1f).setDuration(150);
		animator.addUpdateListener(new ValueAnimator.AnimatorUpdateListener() {

			Random random = new Random();

			@Override
			public void onAnimationUpdate(ValueAnimator animation) {
				view.setTranslationX((random.nextFloat() - 0.5f)
						* view.getWidth() * 0.05f);
				view.setTranslationY((random.nextFloat() - 0.5f)
						* view.getHeight() * 0.05f);

			}
		});
		animator.start();
		view.animate().setDuration(150).setStartDelay(startDelay).scaleX(0f)
				.scaleY(0f).alpha(0f).start();
		explode(Utils.createBitmapFromView(view), r, startDelay,
				ExplosionAnimator.DEFAULT_DURATION);
	}

这里的代码我们可以分为4部分来理解:

第一部分:2-7行

这一段代码我们主要是用来获取要粉碎View的一个可视区域,拿到这个Rect之后,我们再拿到ExplosionField在当前页面中的坐标,然后根据这个坐标对Rect进行平移,平移之后就可以让粉碎的效果在ExplosionField中显示,最后我们还要将Rect的区域扩大一下,默认情况下,横向扩大64dp,纵向扩大64dp,mExpandInset的初始化在ExplosionField的构造方法中调用,最终拿到的Rect我们会在第四部分使用。

第二部分:8-23行

这一段代码主要是是完成粉碎前的振动效果,一个View要被粉碎了,吓得它先打个哆嗦。这里的实现还是比较简单的,先是生成一个随机数,根据这个随机数算出View在X轴和Y轴平移的距离,反复的平移就形成了振动效果。

第三部分:24-25行

这一段代码是将View隐藏,隐藏的方式就是让scaleX和scaleY、alpha最终都变为0f。

第四部分:26-27行

这一段代码是粉碎View的核心代码,我们下面分析。
好了,这一段代码算是一个准备工作,下面我们看看上面第四部分提到的这个explode的重载方法,这里一共需要四个参数,第一个参数是一个Bitmap,这个Bitmap根据我们传入的View创建,第二个参数就是我们上面第一部分提到的那个Rect,第三个参数是一个启动延迟时间,第四个参数是动画默认的执行时间,四个参数中,后面两个都是固定值,第二个参数我们上文已经说过,这里我们主要看看第一个参数的获得:

    public static Bitmap createBitmapFromView(View view) {
        if (view instanceof ImageView) {
            Drawable drawable = ((ImageView) view).getDrawable();
            if (drawable != null && drawable instanceof BitmapDrawable) {
                return ((BitmapDrawable) drawable).getBitmap();
            }
        }
        view.clearFocus();
        Bitmap bitmap = createBitmapSafely(view.getWidth(),
                view.getHeight(), Bitmap.Config.ARGB_8888, 1);
        if (bitmap != null) {
            synchronized (sCanvas) {
                Canvas canvas = sCanvas;
                canvas.setBitmap(bitmap);
                view.draw(canvas);
                canvas.setBitmap(null);
            }
        }
        return bitmap;
    }

    public static Bitmap createBitmapSafely(int width, int height, Bitmap.Config config, int retryCount) {
        try {
            return Bitmap.createBitmap(width, height, config);
        } catch (OutOfMemoryError e) {
            e.printStackTrace();
            if (retryCount > 0) {
                System.gc();
                return createBitmapSafely(width, height, config, retryCount - 1);
            }
            return null;
        }
    }

根据我们传入的View,我们得到一个Bitmap,如果这个View本身就是一个ImageView,那么我们会比较容易获得一个Bitmap,如果这个View不是ImageView,那么我们根据这个View的大小创建一个空的Bitmap,在创建的过程中,如果发生了OOM,我们会尝试执行一次System.gc,然后重新创建,如果还创建失败,那么就会返回一个null。拿到Bitmap之后,我们把这个Bitmap设置为一个Canvas的底布,然后把View绘制在Canvas上,最后再把Canvas的底布设置为null,同时将Bitmap返回。

经过上面的步骤,我们就可以拿到一个根据我们要粉碎的View创建的bitmap,然后我们就来看看explode的重载方法:

	public void explode(Bitmap bitmap, Rect bound, long startDelay,
			long duration) {
		final ExplosionAnimator explosion = new ExplosionAnimator(this, bitmap,
				bound);
		explosion.addListener(new AnimatorListenerAdapter() {
			@Override
			public void onAnimationEnd(Animator animation) {
				mExplosions.remove(animation);
			}
		});
		explosion.setStartDelay(startDelay);
		explosion.setDuration(duration);
		mExplosions.add(explosion);
		explosion.start();
	}

到了这里,我们先来看看ExplosionAnimator这个类,这个类继承自ValueAnimator,我们来看看ExplosionField的构造方法:

	public ExplosionAnimator(View container, Bitmap bitmap, Rect bound) {
		mPaint = new Paint();
		mBound = new Rect(bound);
		int partLen = 15;
		mParticles = new Particle[partLen * partLen];
		Random random = new Random(System.currentTimeMillis());
		int w = bitmap.getWidth() / (partLen + 2);
		int h = bitmap.getHeight() / (partLen + 2);
		for (int i = 0; i < partLen; i++) {
			for (int j = 0; j < partLen; j++) {
				mParticles[(i * partLen) + j] = generateParticle(
						bitmap.getPixel((j + 1) * w, (i + 1) * h), random);
			}
		}
		mContainer = container;
		setFloatValues(0f, END_VALUE);
		setInterpolator(DEFAULT_INTERPOLATOR);
		setDuration(DEFAULT_DURATION);
	}

在这里我们看到了View被粉碎的方式,一个View一共被分为15×15个Particle,通过generateParticle方法可以获得这一个一个的Particle,每一个Particle的颜色就是bitmap上对应的颜色,这些Particle存在一个mParticles数组中,container就是我们传进来的ExplosionField,它赋给了mContainer。我们注意到,ExplosionField重写了start方法:

	@Override
	public void start() {
		super.start();
		mContainer.invalidate(mBound);
	}

在start方法中调用了ExplosionField的draw方法,那我们看看ExplosionField方法中的draw方法:

	@Override
	protected void onDraw(Canvas canvas) {
		super.onDraw(canvas);
		for (ExplosionAnimator explosion : mExplosions) {
			explosion.draw(canvas);
		}
	}

好啊,这里竟然又调用了ExplosionAnimator的draw方法,那我们再看看ExplosionAnimator中的draw方法:

	public boolean draw(Canvas canvas) {
		if (!isStarted()) {
			return false;
		}
		for (Particle particle : mParticles) {
			particle.advance((float) getAnimatedValue());
			if (particle.alpha > 0f) {
				mPaint.setColor(particle.color);
				mPaint.setAlpha((int) (Color.alpha(particle.color) * particle.alpha));
				canvas.drawCircle(particle.cx, particle.cy, particle.radius,
						mPaint);
			}
		}
		mContainer.invalidate();
		return true;
	}

在ExplosionAnimator的draw方法中,我们看到,在for循环中会将我们的15×15个Parcicle都画出来,然后又回去调用ExplosionField中的draw方法,如此循环往复互相之间不断调用,直到Particle的alpha属性为0时停止调用。

看到这里,我们对ExplosionField有了一个基本的了解,这个时候我们再回过头看ExplosionField中的explode方法,我们注意到有一个List集合专门用来存放ExplosionAnimator,这是由于我们的一个页面上可能同时有多个View被粉碎,每一个被粉碎的View对应一个单独的ExplosionAnimator,当该动画执行完了之后,就把它从List集合中移除。

关于粉碎View的事就说这么多。

Demo下载https://github.com/lenve/ExplosionFieldTest

详解ExplosionField的使用,实现View的粉碎效果的更多相关文章

  1. 详解iOS开发之自定义View

    iOS开发之自定义View是本文要将介绍的内容,iOS SDK中的View是UIView,我们可以很方便的自定义一个View.创建一个 Window-based Application程序,在其中添加 ...

  2. Unity 坐标 转换 详解 World世界坐标 Screen屏幕坐标 View视口坐标 GUI坐标 NGUI坐标 localPosition相对父级坐标

    在制作游戏中我们经常会遇到这样一个需求: 在人物模型的上面显示 名字.称号 一类的文字或者图片 如下图 人物模型属于是Camera1   UI Title信息属于NGUI Camera2 如下图 这时 ...

  3. SwipeListView 详解 实现微信,QQ等滑动删除效果

    转载请标明出处:http://blog.csdn.net/lmj623565791/article/details/28508769 今天看别人项目,看到别人使用了SwipeListView,Goog ...

  4. Dialog详解(包括进度条、PopupWindow、自定义view、自定义样式的对话框)

    Dialog详解(包括进度条.PopupWindow.自定义view.自定义样式的对话框)   Android中提供了多种对话框,在实际应用中我们可能会需要修改这些已有的对话框.本实例就是从实际出发, ...

  5. Android 通过Java代码生成创建界面。动态生成View,动态设置View属性。addRules详解

    废话不多说,本文将会层层深入给大家讲解如何动态的生成一个完整的界面. 本文内容: Java代码中动态生成View Java代码中动态设置View的位置,以及其他的属性 LayoutParams详解 一 ...

  6. view坐标_ _ Android应用坐标系统全面详解

    转:http://blog.csdn.net/yanbober/article/details/50419117 1 背景 去年有很多人私信告诉我让说说自定义控件,其实通观网络上的很多博客都在讲各种自 ...

  7. View绘制详解(五),draw方法细节详解之View的滚动/滑动问题

    关于View绘制系列的文章已经完成了四篇了,前面四篇文章主要带小伙伴们熟悉一下View的体系的整体框架.View的测量以及布局等过程,从本篇博客开始,我们就来看看View的绘制过程.View的绘制涉及 ...

  8. View绘制详解(四),谝一谝layout过程

    上篇博客我们介绍了View的测量过程,这只是View显示过程的第一步,第二步就是layout了,这个我们一般译作布局,其实就是在View测量完成之后根据View的大小,将其一个一个摆放在ViewGro ...

  9. View绘制详解(三),扒一扒View的测量过程

    所有东西都是难者不会,会者不难,Android开发中有很多小伙伴觉得自定义View和事件分发或者Binder机制等是难点,其实不然,如果静下心来花点时间把这几个技术点都研究一遍,你会发现其实这些东西都 ...

随机推荐

  1. Android业务组件化之现状分析与探讨

    前言: 从个人经历来说的话,从事APP开发这么多年来,所接触的APP的体积变得越来越大,业务的也变得越来越复杂,总来来说只有一句话:这是一个APP臃肿的时代!所以为了告别APP臃肿的时代,让我们进入一 ...

  2. Windows中使用OpenBLAS加速R语言计算速度

    在使用R的时候会发现R对CPU的利用率并不是很高,反正当我在使用R的时候,无论R做何种运算R的CPU利用率都只有百分子几,这就导致一旦计算量大的时候计算时间非常长,会给人一种错觉(R真的在计算吗?会不 ...

  3. 关于SharePoint 2013 UserProfile跨场的几点注意

    1.跨场中需要以下几个Service实例,没有这几个会遇到各种问题 2.发布场和消费场同时需要创建Host站点,否则消费场的SiteFeed无法使用. 3.跨场关注的问题请参考:http://www. ...

  4. javascript void运算符

    参考链接:http://www.cnblogs.com/ziyunfei/archive/2012/09/23/2698607.html语法: void expr 作用:计算表达式expr,并返回un ...

  5. 如何使用 vimdiff 来 git diff /svn diff

    #git 如何实现vimdiffgit config --global diff.tool vimdiff git config --global difftool.prompt false git ...

  6. android 蓝牙串口通讯使用简介

    需要的权限 <uses-permission android:name="android.permission.BLUETOOTH" />  <uses-perm ...

  7. android开发 ,对接支付宝,服务器(PHP)校验失败

    已备忘记,资料链接: http://my.oschina.net/u/256646/blog/174222 注意: 里面有一个设计到支付宝公钥的地方: 注 意这个是2048位的公钥应该是9行或者10行 ...

  8. 【原创】牛顿法和拟牛顿法 -- BFGS, L-BFGS, OWL-QN

    数据.特征和数值优化算法是机器学习的核心,而牛顿法及其改良(拟牛顿法)是机器最常用的一类数字优化算法,今天就从牛顿法开始,介绍几个拟牛顿法算法.本博文只介绍算法的思想,具体的数学推导过程不做介绍. 1 ...

  9. Export Farm Solution wsp Files SharePoint 2007 and 2010

    [System.Reflection.Assembly]::LoadWithPartialName("Microsoft.SharePoint")$farm = [Microsof ...

  10. 2016 Multi-University Training Contest 1 Necklace 环排+二分匹配

    链接:http://acm.hdu.edu.cn/showproblem.php?pid=5727 题意:由2*N颗宝石构成的环(阴阳宝石均为N颗且标号均从1~N) 之后给定M组 a,b;表示阳宝石a ...