在android2.3上实现截屏功能费了很大的劲,一直很好奇4.0是怎么实现截屏的,所以拿到ICS代码后就迫不及待研究了下ICS的截屏功能。
一、WindowManager端
1. 快捷键
ICS的截屏快捷键是power键+volume down键,这两个键同时按下时就启动了截屏服务。这个判定是在interceptKeyBeforeQueueing中实现的。
当power键和volume down按下时记录了各自按下的时间和并设置了按键按下标志。Power键的标志是mPowerKeyTriggered、mPowerKeyTime;volume down的标志变量是mVolumeDownKeyTriggered、mVolumeDownKeyTime;这几个标志位会在收到up事件或相应longpress后被重置,然后调用interceptScreenshotChord()判断是否需要截屏
private void interceptScreenshotChord() {
if (mVolumeDownKeyTriggered &&
mPowerKeyTriggered && !mVolumeUpKeyTriggered) {
final long now = SystemClock.uptimeMillis();
if (now <= mVolumeDownKeyTime
+ SCREENSHOT_CHORD_DEBOUNCE_DELAY_MILLIS
&& now <= mPowerKeyTime
+ SCREENSHOT_CHORD_DEBOUNCE_DELAY_MILLIS) {
mVolumeDownKeyConsumedByScreenshotChord = true;
cancelPendingPowerKeyAction();
mHandler.postDelayed(mScreenshotChordLongPress,
ViewConfiguration.getGlobalActionKeyTimeout());
}
}
}
启动截屏的条件是power和volume down都被按下,并且各自按下的时间差在一定的范围之内,这个时间差是SCREENSHOT_CHORD_DEBOUNCE_DELAY_MILLIS = 150。如果条件满足,则会先取消掉power键的longpress。Post一个Runnable。这个runnable只做了一件事就是调用takeScreenshot()方法。
2. takeScreenshot()
在这个函数里创建了一个ServiceConnection,然后bindService到TakeScreenshotService。ServiceConnection连接到TakeScreenshotService后,返回TakeScreenshotService的binder,利用这个binder创建一个Messenger向TakeScreenshotService发送一条消息,启动截屏处理。
3. 截屏结束后的处理
windowManager向TakeScreenshotService发送的消息带有一个 replyTo参数,这个参数是一个Messger类型,Messger的hander是在windowManager创建的。所以在截屏结束时,TakeScreenshotService调用replyTo通知windowManager截屏已经结束。
Handler h = new Handler(mHandler.getLooper()) { @Override public void handleMessage(Message msg) { synchronized (mScreenshotLock) { if (mScreenshotConnection == myConn) { mContext.unbindService(mScreenshotConnection); mScreenshotConnection = null; mHandler.removeCallbacks(mScreenshotTimeout); } } } }; msg.replyTo = new Messenger(h); |
二、 TakeScreenshotService
这是运行在系统app层的一个服务,在takeScreenshot调用binderService时启动它,然后对它发送了一条消息。收到消息后,调用GlobalScreenshot的takeScreenshot方法后开始截屏。
takeScreenshot中调用Surface.screenshot()截屏
原型是
public static native Bitmap screenshot(int width, int height);
然后调用startAnimation将截取的图片显示出来。