首页 > 系统 > Android > 正文

Android实战教程第十篇仿腾讯手机助手小火箭发射效果

2019-12-12 04:42:57
字体:
来源:转载
供稿:网友

之前对系统自带的土司的源码做了简要分析,见博客:点击打开链接

这一篇给一个小案例,自定义土司,模拟腾讯卫士的小火箭发射。如果想要迅速看懂代码,建议先去看一下上篇介绍点击打开链接

首先,定义一个服务,在这个服务里面,完成土司的创建(小火箭布局创建),烟的效果属于动画播放,而且要依托一个activity。(这个activity要定义为透明状态)

定义烟的activity的布局文件

<?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" >   <ImageView   android:id="@+id/smoke_m"   android:layout_width="fill_parent"   android:layout_height="wrap_content"   android:layout_alignParentBottom="true"   android:src="@drawable/desktop_smoke_m" />   <ImageView   android:layout_above="@id/smoke_m"   android:id="@+id/smoke_t"   android:layout_width="wrap_content"   android:layout_height="wrap_content"   android:src="@drawable/desktop_smoke_t"/>  </RelativeLayout> 

在对应的Smokeactivity里面加入“烟”的动画

package com.itydl.rockets;  import android.app.Activity; import android.os.Bundle; import android.os.SystemClock; import android.view.animation.AlphaAnimation; import android.view.animation.Animation; import android.view.animation.AnimationSet; import android.view.animation.ScaleAnimation; import android.widget.ImageView;  public class SmokeActivity extends Activity {  @Override  protected void onCreate(Bundle savedInstanceState) {   // TODO Auto-generated method stub   super.onCreate(savedInstanceState);   setContentView(R.layout.smoke);   //底部烟图片   ImageView iv_m = (ImageView) findViewById(R.id.smoke_m);   //烟柱子   ImageView iv_t = (ImageView) findViewById(R.id.smoke_t);         //渐变动画   AlphaAnimation aa = new AlphaAnimation(0.0f,1.0f);   aa.setDuration(1000);   //比例动画                 设置锚点。x轴一半,y轴图片最低端y值最大处   ScaleAnimation sa = new ScaleAnimation(1.0f, 1.0f, 0.0f, 1.0f, Animation.RELATIVE_TO_SELF, 0.5f, Animation.RELATIVE_TO_SELF, 1f);   sa.setDuration(1000);      //动画集添加动画     iv_m.startAnimation(aa);//给下面这张图片实现渐变动画      AnimationSet as = new AnimationSet(true);   as.addAnimation(aa);   as.addAnimation(sa);   //给上边图片(烟柱子)设置渐变动画和比例动画   iv_t.startAnimation(as);      //1秒后关闭Activity,正好动画播完,关闭这个activity。这里也是那样,主线程动画的同时,子线程也在执行耗时操作   new Thread(){    public void run() {     //1秒后关闭当前Activity     SystemClock.sleep(1000);     runOnUiThread(new Runnable() {//activity类中的方法            @Override      public void run() {       // TODO Auto-generated method stub       finish();//关闭自己也属于更新界面操作,因此要在主线程执行。      }     });         };   }.start();     } } 

定义Service,用于自定义土司布局,加入火箭图片的动画、参数初始化、触摸事件等

package com.itydl.rockets;   import android.app.Service; import android.content.Intent; import android.graphics.PixelFormat; import android.graphics.drawable.AnimationDrawable; import android.os.Handler; import android.os.IBinder; import android.os.SystemClock; import android.view.Gravity; import android.view.MotionEvent; import android.view.View; import android.view.View.OnTouchListener; import android.view.WindowManager; import android.widget.ImageView;  public class RocketService extends Service {   private WindowManager.LayoutParams params;  private View view;  private WindowManager wm;   @Override  public IBinder onBind(Intent intent) {   // TODO Auto-generated method stub   return null;  }    @Override  public void onCreate() {      wm = (WindowManager) getSystemService(WINDOW_SERVICE);      //初始化params(土司参数)   initToastParams();      showRocket();//打开小火箭     super.onCreate();  }  /**   * 初始化土司的参数   */  private void initToastParams() {   // TODO Auto-generated method stub   // XXX This should be changed to use a Dialog, with a Theme.Toast   // defined that sets up the layout params appropriately.    params = new WindowManager.LayoutParams();   params.height = WindowManager.LayoutParams.WRAP_CONTENT;   params.width = WindowManager.LayoutParams.WRAP_CONTENT;      //对齐方式左上角   params.gravity = Gravity.LEFT | Gravity.TOP;   params.flags = WindowManager.LayoutParams.FLAG_NOT_FOCUSABLE   /* | WindowManager.LayoutParams.FLAG_NOT_TOUCHABLE */   | WindowManager.LayoutParams.FLAG_KEEP_SCREEN_ON;   params.format = PixelFormat.TRANSLUCENT;    params.type = WindowManager.LayoutParams.TYPE_SYSTEM_ALERT;// 土司天生不响应事件,改变类型。TYPE_SYSTEM_ALERT系统弹窗   params.setTitle("Toast");  }    private void closeRocket(){   if (view != null) {    wm.removeView(view);//移除小火箭   }  }    private Handler handler = new Handler(){   public void handleMessage(android.os.Message msg) {    wm.updateViewLayout(view, params);//更新小火箭在屏幕中的位置,刷新位置。属于更新ui。在主线程执行(更新土司的位置)   };  };    private void showRocket(){   //小火箭的布局      view = View.inflate(getApplicationContext(), R.layout.rocket, null);   ImageView iv_rocket = (ImageView) view.findViewById(R.id.iv_rocket);   //获取小火箭的动画背景   AnimationDrawable ad = (AnimationDrawable) iv_rocket.getBackground();   //开始小火箭动画(小火箭动画,两张图片切换)   ad.start();      //给小火箭加触摸事件(给自定义土司加触摸事件),按住拖动小火箭到屏幕正下方,松开发射火箭   view.setOnTouchListener(new OnTouchListener() {     private float startX;    private float startY;     @Override    public boolean onTouch(View v, MotionEvent event) {     System.out.println(event.getX() + ":" + event.getRawX());     // 拖动土司     switch (event.getAction()) {     case MotionEvent.ACTION_DOWN:// 按下      startX = event.getRawX();      startY = event.getRawY();      break;     case MotionEvent.ACTION_MOVE:// 按下移动,拖动      //新的 x y坐标      float moveX = event.getRawX();//移动后的x坐标      float moveY = event.getRawY();//移动后的y坐标            //dx x方向的位置变化值 dy y方向的位置变化值      float dx = moveX - startX;      float dy = moveY - startY;      //改变土司的坐标      params.x += dx;      params.y += dy;      //重新获取新的x y坐标      startX = moveX;      startY = moveY;            //更新土司的位置      wm.updateViewLayout(view, params);      break;     case MotionEvent.ACTION_UP:// 松开,接下来要发射小火箭      //判断位置 发射      //x轴方向 离两边框超过100,y轴方向大于200 就可以发射火箭      if (params.x > 100 && params.x + view.getWidth()< wm.getDefaultDisplay().getWidth() - 100 &&        params.y > 200){       //发射火箭       //1,火箭往上跑       //火箭在中心线上发射(自定义土司左上角为基准)       params.x = (wm.getDefaultDisplay().getWidth() - view.getWidth()) / 2;              new Thread(){//发射火箭改变y轴属于耗时操作,更新火箭位置是更新UI操作        public void run() {         for (int j = 0; j < view.getHeight(); ) {          SystemClock.sleep(50);//休眠50毫秒          params.y -= j;          j += 5;          handler.obtainMessage().sendToTarget();//参数y的值改变一次,发消息通知更新一次ui,更新一次土司的位置         }                  //,发射完毕,关闭小火箭         stopSelf();//关闭服务,关闭当前自己服务。这个方法用在关闭自己服务里。触发onDestroy方法,从而触发这个方法里面的关闭小火箭        };       }.start();              //2,烟的效果。因为更新火箭往上跑是在子线程执行的,因此在小火箭往上跑的同时,烟的效果也同时开始播放(子线程不影响主线程执行。两个线程可以同时进行)       //烟的效果,是一个动画,在activity完成,这个activity需要定义为透明       Intent intent = new Intent(RocketService.this,SmokeActivity.class);       //在服务中打开activity,需要设置任务栈:       intent.addFlags(Intent.FLAG_ACTIVITY_NEW_TASK);//任务栈       startActivity(intent);//启动烟的Activity      }      //冒烟的Activity     default:      break;     }     return false;//默认返回值。    }   });      wm.addView(view, params);//把小火箭加到窗体管理器  }      @Override  public void onDestroy() {   // TODO Auto-generated method stub   closeRocket();//关闭小火箭   super.onDestroy();  }  } 

对于主动人任务只是加入个按钮,打开这个服务就行了。

package com.itydl.rockets;  import android.app.Activity; import android.content.Intent; import android.os.Bundle; import android.view.View;  public class MainActivity extends Activity {   @Override  protected void onCreate(Bundle savedInstanceState) {   super.onCreate(savedInstanceState);   setContentView(R.layout.activity_main);     }   /**   * 通过点击按钮打开小火箭   * @param v   */  public void openRocket(View v){   //RocketService service = new RocketService();   Intent service = new Intent(this,RocketService.class);   startService(service);//启动小火箭服务   finish();//关闭当前界面。因为要显示火箭发射,不能在这个activity里面演示  }  } 

最后清单文件配置上两个活动和一个服务,还有一个弹出窗体的权限

<activity    android:name="com.itheima62.rockets.MainActivity"    android:label="@string/app_name" >    <intent-filter>     <action android:name="android.intent.action.MAIN" />      <category android:name="android.intent.category.LAUNCHER" />    </intent-filter>   </activity>   <!-- 配置该活动的主题,为透明、无标题、全屏 -->   <activity android:name="com.itheima62.rockets.SmokeActivity" android:theme="@android:style/Theme.Translucent.NoTitleBar.Fullscreen"></activity>   <service android:name="com.itheima62.rockets.RocketService"></service> 

复制代码 代码如下:
<uses-permission android:name="android.permission.SYSTEM_ALERT_WINDOW"/> 

好了主要代码和功能都介绍完了,看一下运行效果截图:

完整代码请查看文末的原文链接。

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持武林网。

发表评论 共有条评论
用户名: 密码:
验证码: 匿名发表