Android 仿火螢視頻桌面 神奇的 LiveWallPaper
一、概述
上周我的微信公眾號推送了一篇 Android 實現"透明屏幕 ,當時我看到之后就覺得特別感興趣,也立即聯系作者要了授權~~
感興趣的原因是,我是內涵段子的資深用戶,前段時間基本被一款叫火螢視頻桌面的軟件(就是將視頻作為桌面)給刷屏了,所以看了下作者的代碼,看到了SurfaceHolder,立刻想到了,肯定可以用來播放視頻實現視頻桌面的效果,于是周末嘗試了下,果然很簡單。
所以本篇文章無限感謝 Android 實現"透明屏幕 一文,代碼也部分參考自其提供的透明相機。
https://github.com/songixan/Wallpaper
效果圖是這樣的:
注:本文的測試機為小米5s ,可能不同手機會有一些兼容性問題,嘗試解決下。
二、實現
(1) 配置相關
首先編寫一個xml文件,用于描述wallpaper的 thumbnail 、 description 、 settingsActivity 等,這里為了簡單,僅設置了thumbnail。
<?xml version="1.0" encoding="utf-8"?>
<wallpaper xmlns:android="http://schemas.android.com/apk/res/android"
android:thumbnail="@mipmap/ic_launcher" />
(2)編寫代碼
Wallpaper需要在屏幕上一直顯示,其背后其實是一個Service,所以實現一個Wallpaper需要繼承自 WallpaperService ,實現其抽象方法 onCreateEngine ,如下:
public class VideoLiveWallpaper extends WallpaperService {
public Engine onCreateEngine() {
return new VideoEngine();
}
//...
}
可以看到返回值是一個Engine,Engine為WallpaperService的內部類,其內部包含 onSurfaceCreated 、 onSurfaceChanged 、 onSurfaceDestroyed 、 onTouchEvent 等方法,看到這些方法,立刻想到了SurfaceView,關于SurfaceView相關知識可以參考:
此外,大家還記得在Android播放視頻嗎?
常規的做法有通過VideoView,除此以外還有通過MediaPlayer配合SurfaceView配合來實現,今天這個例子類似后者。
我們只需要通過MediaPlayer將解碼的數據不斷的輸送到傳入的Surface中即可。
class VideoEngine extends Engine {
private MediaPlayer mMediaPlayer;
@Override
public void onSurfaceCreated(SurfaceHolder holder) {
L.d("VideoEngine#onSurfaceCreated ");
super.onSurfaceCreated(holder);
mMediaPlayer = new MediaPlayer();
mMediaPlayer.setSurface(holder.getSurface());
try {
AssetManager assetMg = getApplicationContext().getAssets();
AssetFileDescriptor fileDescriptor = assetMg.openFd("test1.mp4");
mMediaPlayer.setDataSource(fileDescriptor.getFileDescriptor(),
fileDescriptor.getStartOffset(), fileDescriptor.getLength());
mMediaPlayer.setLooping(true);
mMediaPlayer.setVolume(0, 0);
mMediaPlayer.prepare();
mMediaPlayer.start();
} catch (IOException e) {
e.printStackTrace();
}
}
@Override
public void onVisibilityChanged(boolean visible) {
L.d("VideoEngine#onVisibilityChanged visible = " + visible);
if (visible) {
mMediaPlayer.start();
} else {
mMediaPlayer.pause();
}
}
@Override
public void onSurfaceDestroyed(SurfaceHolder holder) {
L.d("VideoEngine#onSurfaceDestroyed ");
super.onSurfaceDestroyed(holder);
mMediaPlayer.release();
mMediaPlayer = null;
}</code></pre>
代碼非常簡單,在onSurfaceCreated中去初始化mMediaPlayer,核心代碼即為設置setSurface方法,這里我默認設置了靜音。
onVisibilityChanged,即當桌面不可見時,我們要暫停播放,等回到桌面繼續。
當onSurfaceDestroyed時釋放資源~~
這樣我們的VideoLiveWallpaper就寫好了,別忘了他是個Service,需要我們去注冊。
<service
android:name=".VideoLiveWallpaper"
android:label="@string/app_name"
android:permission="android.permission.BIND_WALLPAPER"
android:process=":wallpaper">
<!-- 配置intent-filter -->
<intent-filter>
<action android:name="android.service.wallpaper.WallpaperService" />
</intent-filter>
<!-- 配置meta-data -->
<meta-data
android:name="android.service.wallpaper"
android:resource="@xml/livewallpaper" />
</service>
(3)設置為壁紙
注冊完成后,我們在MainActivity中添加一個按鈕點擊設置為桌面背景,調用代碼如下
public static void setToWallPaper(Context context) {
final Intent intent = new Intent(WallpaperManager.ACTION_CHANGE_LIVE_WALLPAPER);
intent.putExtra(WallpaperManager.EXTRA_LIVE_WALLPAPER_COMPONENT,
new ComponentName(context, VideoLiveWallpaper.class));
context.startActivity(intent);
}
這樣就完成了代碼的初步編寫啦~~
(4)增加一些參數的支持
剛才我們設置了默認是靜音,可能有時候我們會希望能夠動態去控制視頻桌面的參數,正常應該嘗試去使用 settingsActivity ,不過我覺得其實廣播也挺合適的,無非就是Service(可能在獨立的進程)和Activity等通信嘛~~
這里我們增加一個復選框,支持設置開啟聲音or關閉聲音。
public static final String VIDEO_PARAMS_CONTROL_ACTION = "com.zhy.livewallpaper";
public static final String KEY_ACTION = "action";
public static final int ACTION_VOICE_SILENCE = 110;
public static final int ACTION_VOICE_NORMAL = 111;
class VideoEngine extends Engine {
// 省略其他代碼
private BroadcastReceiver mVideoParamsControlReceiver;
@Override
public void onCreate(SurfaceHolder surfaceHolder) {
super.onCreate(surfaceHolder);
IntentFilter intentFilter = new IntentFilter(VIDEO_PARAMS_CONTROL_ACTION);
registerReceiver(mVideoParamsControlReceiver = new BroadcastReceiver() {
@Override
public void onReceive(Context context, Intent intent) {
L.d("onReceive");
int action = intent.getIntExtra(KEY_ACTION, -1);
switch (action) {
case ACTION_VOICE_NORMAL:
mMediaPlayer.setVolume(1.0f, 1.0f);
break;
case ACTION_VOICE_SILENCE:
mMediaPlayer.setVolume(0, 0);
break;
}
}
}, intentFilter);
}
@Override
public void onDestroy() {
unregisterReceiver(mVideoParamsControlReceiver);
super.onDestroy();
}
}
Engine還有onCreate和onDestroy聲明周期方法,可以在onCreate中注冊動態廣播,當接受到發送的action為 ACTION_VOICE_NORMAL 則開啟聲音;接收到發送的 ACTION_VOICE_SILENCE 則為靜音狀態。
最后直接在VideoLiveWallpaper中添加兩個靜態方法用于發送廣播即可:
public static void voiceSilence(Context context) {
Intent intent = new Intent(VideoLiveWallpaper.VIDEO_PARAMS_CONTROL_ACTION);
intent.putExtra(VideoLiveWallpaper.KEY_ACTION, VideoLiveWallpaper.ACTION_VOICE_SILENCE);
context.sendBroadcast(intent);
}
public static void voiceNormal(Context context) {
Intent intent = new Intent(VideoLiveWallpaper.VIDEO_PARAMS_CONTROL_ACTION);
intent.putExtra(VideoLiveWallpaper.KEY_ACTION, VideoLiveWallpaper.ACTION_VOICE_NORMAL);
context.sendBroadcast(intent);
}
在Actiivty中:
public class MainActivity extends AppCompatActivity {
private CheckBox mCbVoice;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
mCbVoice = (CheckBox) findViewById(R.id.id_cb_voice);
mCbVoice.setOnCheckedChangeListener(
new CompoundButton.OnCheckedChangeListener() {
@Override
public void onCheckedChanged(
CompoundButton buttonView, boolean isChecked) {
if (isChecked) {
// 靜音
VideoLiveWallpaper.voiceSilence(getApplicationContext());
} else {
VideoLiveWallpaper.voiceNormal(getApplicationContext());
}
}
});
}
}
監聽一下CheckBox狀態,發送廣播即可。
ok,這樣一個簡單的視頻桌面就完成啦~~
參考
- http://www.vogella.com/tutorials/AndroidLiveWallpaper/article.html
- http://www.jianshu.com/u/befb61deec9c
項目主頁:http://www.baiduhome.net/lib/view/home/1494985986236