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

为了开发实时壁纸,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. Android 5.1 open data flow 数据开启流程

    首先我们来看看下面的关系图: 底层Settings.apk 在Settings -> Data Usage Summary中的某个SIM tab下开启数据开关 android/packages/ ...

  2. HAL驱动库学习-SPI

    如何使用SPI库1 声明SPI hanlde, 例如: SPI_HandleTypeDef hspi2 通过实现HAL_SPI_MspInit()函数初始化底层资源 以下两个必须进行初始化 a 使能s ...

  3. Servlet-cookies机制

    通过cookies,可以保存用户的使用习惯,优化用户体验,同时能减轻服务端压力.下面说下在Servlet中cookies机制的使用 就用保存用户登录数据来举例子: 打开网页的处理Servlet: pa ...

  4. Android 网络请求库volley的封装,让请求更方便

    首先封装一下volley 请求 public class CustomRequest extends StringRequest { private static final String TAG = ...

  5. win10查看连接过的wifi密码

    cmd窗口 运行   “netsh wlan show profiles name="linasd" key=clear”

  6. Linux 打通ssh无密码登录

    像hadoop和spark这类的集群,因为master节点要控制slave节点,以及各节点之间要交互信息,所以需要各节点之间能够互相无密码登录. 通过RSA保存密码, 基本操作如下: Step 1: ...

  7. Python3基础 把一个列表中内容给另外一个列表,形成两个独立的列表

    镇场诗:---大梦谁觉,水月中建博客.百千磨难,才知世事无常.---今持佛语,技术无量愿学.愿尽所学,铸一良心博客.------------------------------------------ ...

  8. javascript guid(uuid)

    http://stackoverflow.com/questions/105034/how-to-create-a-guid-uuid-in-javascript function generateU ...

  9. iOS开发数据库篇—SQLite的应用

    iOS开发数据库篇—SQLite的应用 一.简单说明 在iOS中使用SQLite3,首先要添加库文件libsqlite3.dylib和导入主头文件. 导入头文件,可以使用库中的函数(是纯C语言的) 二 ...

  10. python走起之第四话

    本节大纲: 一:双层装饰器:一个函数可以被多层装饰器进行装饰,函数渲染(编译)从下到上,函数执行从上到下.如下程序: 1 #!/usr/bin/env python 2 #-*-coding:utf- ...