一、说明
在这篇自定义View之指南针(反编译别人的代码实现)文章中,我们通过反编译魅族工具箱的应用,实现了其指南针的效果,魅族工具箱里面有好几个比较实用工具,比如测量尺、水平仪、随机事件等。这篇文章分析一下其随机事件的实现。
二、界面初步分析
首先看一下魅族翻硬币的截图效果:
点击硬币后,会有一段翻转硬币的动画,动画结束后会显示硬币的翻转结果。由于不知道csdn怎么上传视频,这里就上传图片了。
用hierarchy view查看界面布局:
在界面上看到了一个VideoView控件,这个控件一般是用来播放视频的,本来我以为翻转硬币也是自定义的View,看了想多了,其实就是一个ViewView在播放视频。
三、反编译Apk查看代码实现
具体反编译过程可以查看自定义View之指南针(反编译别人的代码实现)
这里我们用jadx打开工具箱apk,搜索CoinVideoActivity,结果如下:
代码基本没有混淆,将对应的代码和使用到的资源拷贝到自己的工程中,稍作修改即可运行。
CoinVideoActivity
import android.content.Context;
import android.content.ContextWrapper;
import android.media.MediaPlayer;
import android.media.MediaPlayer.OnCompletionListener;
import android.net.Uri;
import android.os.Bundle;
import android.support.v7.app.AppCompatActivity;
import android.view.View;
import android.view.View.OnClickListener;
import android.view.WindowManager;
import android.widget.TextView;
import android.widget.VideoView;
import com.liunian.androidbasic.R;
import java.util.Random;
public class CoinVideoActivity extends AppCompatActivity {
private View mEmptyView;
private boolean mIsPause = true;
private TextView mTextView;
private VideoView mVideoView;
private class OnCoinClickListener implements OnClickListener {
private OnCoinClickListener() {
}
public void onClick(View view) {
CoinVideoActivity.this.startAnimation(); // 播放翻转硬币视频
}
}
protected void onCreate(Bundle bundle) {
super.onCreate(bundle);
setContentView(R.layout.activity_coin_vedio);
// 设置默认的窗口背景为空,减少绘制
getWindow().setBackgroundDrawable(null);
// 隐藏标题栏和状态栏
getSupportActionBar().hide();
getWindow().setFlags(WindowManager.LayoutParams.FLAG_FULLSCREEN, WindowManager.LayoutParams.FLAG_FULLSCREEN);
this.mIsPause = true;
init();
}
private void init() {
this.mTextView = (TextView) findViewById(R.id.coin_desc); // 提示文字
this.mVideoView = (VideoView) findViewById(R.id.video_view); // 播放视频控件
this.mVideoView.setOnCompletionListener(new OnCompletionListener() { // 视频播放完成后的监听
public void onCompletion(MediaPlayer mediaPlayer) {
CoinVideoActivity.this.mEmptyView.setOnClickListener(new OnCoinClickListener()); // 视频播放完成后再次给mEmptyView绑定监听事件
}
});
this.mEmptyView = findViewById(R.id.empty_view); // mEmptyView是一个透明的View,通过点击它开始播放视频
this.mEmptyView.setOnClickListener(new OnCoinClickListener());
}
private void startInitAnimation() { // 设置mVideoView最开始的视频(很短,相当于一张图片)
this.mVideoView.setVideoURI(Uri.parse("android.resource://" + getPackageName() + "/" + R.raw.coininit));
this.mVideoView.start();
}
// 播放翻转硬币视频
private void startAnimation() {
String str;
if (new Random().nextInt(2) == 0) { // 生成0或1的随机数,如果为0则播放脸朝上的视频
str = "android.resource://" + getPackageName() + "/" + R.raw.coinvideo1;
} else { // 如果为1则表示字朝上,播放字朝上的视频
str = "android.resource://" + getPackageName() + "/" + R.raw.coinvideo2;
}
this.mVideoView.setVideoURI(Uri.parse(str));
this.mVideoView.start();
this.mEmptyView.setOnClickListener(null); // 开始播放视频或将mEmptyView的点击事件设置为空,避免重复播放视频
}
protected void onResume() {
super.onResume();
if (this.mIsPause) {
this.mTextView.setText(getResources().getString(R.string.coin_click_text));
startInitAnimation();
this.mIsPause = false;
}
}
protected void onPause() {
super.onPause();
this.mIsPause = true;
}
protected void attachBaseContext(Context context) {
super.attachBaseContext(new ContextWrapper(context) {
public Object getSystemService(String str) {
if ("audio".equals(str)) {
return getApplicationContext().getSystemService(str); // 使用ApplicationContext来获得服务,避免内存泄露
}
return super.getSystemService(str);
}
});
}
}
activity_coin_vedio
<?xml version="1.0" encoding="utf-8"?>
<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
android:layout_width="match_parent"
android:layout_height="match_parent">
<VideoView
android:id="@+id/video_view"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:layout_centerInParent="true"
android:clickable="true" />
<View
android:id="@+id/empty_view"
android:layout_width="150dp"
android:layout_height="150dp"
android:layout_centerInParent="true"
android:background="@android:color/transparent" />
<TextView
android:id="@+id/coin_desc"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_centerInParent="true"
android:fontFamily="sans-serif-medium"
android:paddingTop="400dp"
android:textColor="#51ffffff" />
</RelativeLayout>
具体的翻转硬币的视频资源可以在魅族工具箱apk中找。