所谓实时壁纸,就是指手机桌面不再是简单的图片,而是运行中的动画,这个动画是由程序实时绘制的,因此被称为实时壁纸。

为了开发实时壁纸,Android提供了WallpaperService基类,实时壁纸的实现类需要继承该基类,在Android应用中开发实时壁纸的步骤如下:

1、开发一个子类继承WallpaperService基类。

2、继承WallpaperService基类时必须重写onCreateEngine()方法,该方法返回WallpaperService.Engine子类对象。

3、开发者需要实现WallpaperService.Engine子类,并重写其中的public  void  onVisibilityChanged(boolean  visible)、public  void  onOffsetsChanged()方法。不仅如此,由于WallpaperService.Engine子类采用了与SurfaceView相同的绘图机制,因此还可选择性地重写在SurfaceHolder.Callback中的三个方法。重写这些方法时可通过SurfaceHolder动态地绘制图形。

下面的Service类就代表了实时壁纸服务,程序代码如下:

import android.graphics.Canvas;
import android.graphics.Paint;
import android.os.Handler;
import android.service.wallpaper.WallpaperService;
import android.view.MotionEvent;
import android.view.SurfaceHolder;

public class LiveWallPaper extends WallpaperService{
  //实现WallpaperService必须实现的抽象方法
  @Override
  public Engine onCreateEngine() {
    // 返回自定义的Engine
    return new MyEngine();
  }
  class MyEngine extends Engine{
    //记录程序界面是否可见
    private boolean mVisible;
    //记录当前用户动作事件的发生位置
    private float mTouchX = -1;
    private float mTouchY = -1;
    //记录当前圆圈的绘制位置
    private float cx = 15;
    private float cy = 20;
    //定义画笔
    private Paint mPaint = new Paint();
    //定义一个Handler
    Handler mHandler = new Handler();
    //定义一个周期性执行的任务
    private final Runnable drawTarget = new Runnable() {

      @Override
      public void run() {
        drawFrame();
      }
    };
    public void onCreate(android.view.SurfaceHolder surfaceHolder) {
      super.onCreate(surfaceHolder);
      //初始化画笔
      mPaint.setColor(0xffffffff);
      mPaint.setAntiAlias(true);
      mPaint.setStrokeWidth(2);
      mPaint.setStrokeCap(Paint.Cap.ROUND);
      mPaint.setStyle(Paint.Style.STROKE);
      //设置处理触摸事件
      setTouchEventsEnabled(true);
    };
    @Override
    public void onDestroy() {
      super.onDestroy();
      //删除回调
      mHandler.removeCallbacks(drawTarget);
    }
    @Override
    public void onVisibilityChanged(boolean visible) {
      mVisible = visible;
      //当界面可见时,执行drawFrame()方法
      if(visible){
        //动态地绘制图形
        drawFrame();
      }else{
        //如果界面不可见,删除回调
        mHandler.removeCallbacks(drawTarget);
      }
    }
    @Override
    public void onOffsetsChanged(float xOffset, float yOffset,
            float xOffsetStep, float yOffsetStep, int xPixelOffset,
            int yPixelOffset) {
      drawFrame();
    }
    // //当屏幕大小改变时调用该方法
    // @Override
    // public void onSurfaceChanged(SurfaceHolder holder, int format,
    //              int width, int height) {
    //   super.onSurfaceChanged(holder, format, width, height);
    //    drawFrame();
    // }
    // @Override
    // public void onSurfaceCreated(SurfaceHolder holder) {
    //    super.onSurfaceCreated(holder);
    // }
    // @Override
    // public void onSurfaceDestroyed(SurfaceHolder holder) {
    //    super.onSurfaceDestroyed(holder);
    //     mVisible = false;
    //    mHandler.removeCallbacks(drawTarget);
    // }
    @Override
    public void onTouchEvent(MotionEvent event) {
      // 如果检测到滑动操作
      if(event.getAction() == MotionEvent.ACTION_MOVE){
        mTouchX = event.getX();
        mTouchY = event.getY();
      }else{
        mTouchX = -1;
        mTouchY = -1;
      }
      super.onTouchEvent(event);
    }
    //定义绘制图形的工具方法
    private void drawFrame(){
      //获取该壁纸的SurfaceHolder
      final SurfaceHolder holder = getSurfaceHolder();
      Canvas c = null;
      try {
        //获取该壁纸的SurfaceHolder
        c = holder.lockCanvas();
        if(c != null){
          c.save();
          //绘制背景色
          c.drawColor(0xff000000);
          //在触碰点绘制圆圈
          drawTouchPoint(c);
          //绘制圆圈
          c.drawCircle(cx, cy, 80, mPaint);
          c.restore();
        }
      } catch (Exception e) {
        e.printStackTrace();
      }finally{
        if(c != null){
          holder.unlockCanvasAndPost(c);
        }
      }
      mHandler.removeCallbacks(drawTarget);
      //调度下一次重绘
      if(mVisible){
        cx += 15;
        cy += 20;
        //如果cx、cy移除屏幕,从左上角重新开始
        if(cx > 320){
          cx = 15;
        }
        if(cy > 400){
          cy = 20;
        }
        //指定0.1秒后重新执行mDrawCube一次
        mHandler.postDelayed(drawTarget, 100);
      }

    }
    //在屏幕触碰点绘制圆圈
    private void drawTouchPoint(Canvas c){
      if(mTouchX >= 0 && mTouchY >= 0){
        c.drawCircle(mTouchX, mTouchY, 40, mPaint);
      }
    }
  }

}

上面的程序中粗体字代码就是实现实时壁纸Service的关键代码,这两段粗体字代码重写了WallpaperService.Engine的onVisibilityChanged、onOffsetsChanged方法,并指定当桌面显示时调用drawFrame()方法进行绘制,drawFrame()方法绘制完成后通过Handler对象指定0.1秒后重绘。

上面的程序中被注释的代码是可有可无地,如果程序需要桌面的Surface改变事件提供响应,则可以考虑重写被注释代码。

定义了该Service类之后,接下来还需要在AndroidManifest.xml文件中配置该Service,配置实时壁纸Service与配置普通Service存在小小的区别。它需要指定如下两项:

1、指定运行实时壁纸需要android.permission.BIND_WALLPAPER权限。

2、为实时壁纸指定meta-data配置。

在AndroidManifest.xml文件中配置实时壁纸,也就是需要增加如下配置片段:

<!-- 配置实时壁纸Service -->
<service android:name=".LiveWallPaper"
  android:label="@string/app_name"
  android:permission="android.permission.BIND_WALLPAPER"
  >
  <!-- 为实时壁纸配置intent-filter -->

  <intent-filter >
    <action android:name="android.service.wallpaper.WallPapreService"/>
  </intent-filter>
  <!-- 为实时壁纸配置meta-filter -->

  <meta-data
    android:name="android.service.wallpaper"
    android:resource="@xml/livewallpaper"
  />
</service>

上面的配置文件中粗体字配置部分就是配置实时壁纸的关键代码,上面的配置文件中指定了实时壁纸的meta-data放在@xml/livewallpaper中定义,因此程序还需要zaires\xml\目录下增加一个livewallpaper.xml文件,该代码如下:

<?xml version="1.0" encoding="utf-8"?>
<wallpaper xmls:android="http://schemas.android.com/apk/res/android"/>

开发实时壁纸(Live Wallpapers)的更多相关文章

  1. Andoird开发手机壁纸

    Android 开发手机壁纸3种方法 首先使用WallpaperManager wpm = (WallpaperManager) getActivity().getSystemService(Cont ...

  2. 使用Node.js的socket.io模块开发实时web程序

    首发:个人博客,更新&纠错&回复 今天的思维漫游如下:从.net的windows程序开发,摸到nodejs的桌面程序开发,又熟悉了一下nodejs,对“异步”的理解有了上上周对操作系统 ...

  3. c# 利用AForge和百度AI开发实时人脸识别

    baiduAIFaceIdentify项目是C#语言,集成百度AI的SDK利用AForge开发的实时人脸识别的小demo,里边包含了人脸检测识别,人脸注册,人脸登录等功能 人脸实时检测识别功能 思路是 ...

  4. 基于flink快速开发实时TopN程序

    TopN 是统计报表和大屏非常常见的功能,主要用来实时计算排行榜.流式的TopN可以使业务方在内存中按照某个统计指标(如出现次数)计算排名并快速出发出更新后的排行榜. 我们以统计词频为例展示一下如何快 ...

  5. React开发实时聊天招聘工具 -第一章

    第一章 课程道学 6个页面 弱化css Antd-mobile作为组件库 Redux 状态管理 React-Router 路由 Axios异步请求 后端Express框架 Socket.io 数据库: ...

  6. 大众点评Java开发实时应用监控平台-CAT

    CAT介绍 CAT是基于Java开发的实时应用监控平台,包括实时应用监控,业务监控. CAT作为大众点评网基础监控组件,它已经在中间件框架(MVC框架,RPC框架,数据库框架,缓存框架等)中得到广泛应 ...

  7. flex4+fms3.5+cs4开发实时音视频直播及点播详解

    开发工具及环境: 1)flash builder4 2)flash cs4 3)flash media server3.5 fms部分 fms是adobe的流媒体服务器,不过是收费的,价格大概是ora ...

  8. iOS开发——实时监控网速(仅作参考,发现一点问题)

    开发中用到获取网速的地方,应该就两种: 1.下载速度,这种可以直接在接受数据的地方统计计算.这个就不讲了. 2.获取手机网卡的数据,可以监控网卡的进出流量,下面就是. #import "Vi ...

  9. React开发实时聊天招聘工具 -第二章

    2-1 介绍React开发环境 npm install -g create-react-app xxx npm run eject   来配置webpack 2-2 ES6常用语法 其他 还有一些特性 ...

随机推荐

  1. JQuery MultiSelect(左右选择框)

    <!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" "http://www.w3.org/ ...

  2. js 格式化数字保留2位小数

    function toDecimal2(x) { var f = parseFloat(x); if (isNaN(f)) { return false; } var f = Math.round(x ...

  3. 随机采样方法整理与讲解(MCMC、Gibbs Sampling等)

    本文是对参考资料中多篇关于sampling的内容进行总结+搬运,方便以后自己翻阅.其实参考资料中的资料写的比我好,大家可以看一下!好东西多分享!PRML的第11章也是sampling,有时间后面写到P ...

  4. IE7 float:left失效的解决方法

    <div id="a" style="width:500px"> <div id="b" style="widt ...

  5. python核心编程学习记录之错误与异常

  6. shell & dialog

    最近使用dialog写图形自动化shell脚本,  功能很强大,功能不是非常多但是足够用.想写一篇linux下dialog的使用方法,虽然命令不多,但是写起来也需要下很大功夫,而且不一定写得更好,在网 ...

  7. [问题2014S06] 解答

    [问题2014S06]  解答  (本解答由巴闻嘉同学给出) 设特征多项式 \[f(x)=\det(xI_V-\varphi)=x^n+a_{n-1}x^{n-1}+\cdots+a_1x+a_0,\ ...

  8. GZFramwork数据库层《二》单据表增删改查(自动生成单据号码)

    运行效果: 使用代码生成器(GZCodeGenerate)生成tb_EmpLeave的Model 生成器源代码下载地址: https://github.com/GarsonZhang/GZCodeGe ...

  9. Creating Custom Login Screen In Oracle Forms 10g

    Below is the example plsql unit to validate login credentials and after successful validation open a ...

  10. 利用backtrace和objdump进行分析挂掉的程序

    转自:http://blog.csdn.net/hanchaoman/article/details/5583457 汇编不懂,先把方法记下来. glibc为我们提供了此类能够dump栈内容的函数簇, ...