荣添创意网站建设,wordpress 删除文章,网站 空间,上海公共招聘网官网下载Android监听用户的截屏、投屏、录屏行为
一.截屏
方案一#xff1a;使用系统广播监听截屏操作
从Android Q#xff08;10.0#xff09;开始#xff0c;Intent.ACTION_SCREEN_CAPTURED_CHANGED字段不再被支持。这是因为Google在安卓10 中引入了一个新的隐私限制#…Android监听用户的截屏、投屏、录屏行为
一.截屏
方案一使用系统广播监听截屏操作
从Android Q10.0开始Intent.ACTION_SCREEN_CAPTURED_CHANGED字段不再被支持。这是因为Google在安卓10 中引入了一个新的隐私限制即限制应用在用户开启了屏幕录制功能或截屏功能时获取相应的广播。
创建一个BroadcastReceiver类来接收截屏广播
public class ScreenCaptureReceiver extends BroadcastReceiver {Overridepublic void onReceive(Context context, Intent intent) {String action intent.getAction();if (Intent.ACTION_SCREEN_CAPTURES_CHANGED.equals(action)) {// 截屏操作// 在这里执行你的逻辑操作}}
}在AndroidManifest.xml文件中声明截屏广播接收器
receiverandroid:name.ScreenCaptureReceiverandroid:enabledtrueintent-filteraction android:nameandroid.intent.action.SCREEN_CAPTURE_CHANGED //intent-filter
/receiver注册截屏广播接收器开始监听截屏操作
ScreenCaptureReceiver receiver new ScreenCaptureReceiver();
IntentFilter filter new IntentFilter(Intent.ACTION_SCREEN_CAPTURE_CHANGED);
registerReceiver(receiver, filter);在不需要监听时记得取消注册截屏广播接收器
unregisterReceiver(receiver);在BroadcastReceiver中的onReceive方法中可以执行一些逻辑操作例如显示一个提示消息
public class ScreenCaptureReceiver extends BroadcastReceiver {Overridepublic void onReceive(Context context, Intent intent) {String action intent.getAction();if (Intent.ACTION_SCREEN_CAPTURE_CHANGED.equals(action)) {// 截屏操作Toast.makeText(context, 用户进行了截屏操作, Toast.LENGTH_SHORT).show();}}
}方案二使用ContentObserver监听截屏操作
另一种监听截屏操作的方法是使用ContentObserver来监听系统截屏文件的变化。以下是一个示例代码
public class ScreenCaptureObserver extends ContentObserver {private static final String TAG ScreenCaptureObserver;private static final String SCREENSHOTS_DIR Environment.getExternalStorageDirectory().toString() /Pictures/Screenshots;private Context mContext;public ScreenCaptureObserver(Context context) {super(null);mContext context;}Overridepublic void onChange(boolean selfChange) {super.onChange(selfChange);Log.d(TAG, Screen capture detected);if (uri.toString().matches(MediaStore.Images.Media.EXTERNAL_CONTENT_URI.toString() /[0-9])) {Cursor cursor null;try {cursor mContext.getContentResolver().query(uri, new String[]{MediaStore.Images.Media.DATA}, null, null, null);if (cursor ! null cursor.moveToFirst()) {String path cursor.getString(cursor.getColumnIndex(MediaStore.Images.Media.DATA));if (path ! null path.startsWith(SCREENSHOTS_PATH)) {// 此处为用户截屏行为的响应逻辑}}} finally {if (cursor ! null) {cursor.close();}}}}public void start() {ContentResolver contentResolver mContext.getContentResolver();contentResolver.registerContentObserver(MediaStore.Images.Media.EXTERNAL_CONTENT_URI, true, this);}public void stop() {ContentResolver contentResolver mContext.getContentResolver();contentResolver.unregisterContentObserver(this);}
}使用示例
ScreenCaptureObserver observer new ScreenCaptureObserver(this);
observer.start();对比与选型建议
使用 BroadcastReceiver
方案一的优点
直接监听 Intent.ACTION_SCREENSHOT无需处理 Uri 的变化。适用于监听应用内的截屏操作。
方案一的缺点
只能监听到整个屏幕的截屏操作无法获取到具体的截屏内容。有版本限制。从Android Q10.0开始Intent.ACTION_SCREEN_CAPTURED_CHANGED字段不再被支持。
使用 ContentObserver
方案二的优点
可以监听到截屏操作发生的具体 Uri可以进一步获取截屏的文件路径和信息。适用于监听系统级别的截屏操作。
方案二的缺点
需要指定监听的 Uri可能需要考虑兼容性问题。需要在代码中处理 Uri 的变化并解析截屏的文件路径。
选型建议
根据需求如果只关心截屏的发生与否并不需要获取截屏的具体内容方案一可以考虑。如果需要获取截屏内容的具体信息方案二比较适合。如果只需要监听应用内的截屏操作方案一比较方便。如果需要监听系统级别的截屏操作需要使用方案二。
二.录屏
在 Android 开发中要监听用户的录屏操作可以使用以下两种方案
方案一使用 MediaProjection API
Android 录屏 - 简书 (jianshu.com)
Android Q之后又前台服务限制
首先在你的应用中创建一个 MediaProjectionManager 对象来获取用户录屏的权限。
private MediaProjectionManager mediaProjectionManager;
private MediaProjection mediaProjection;
private MediaProjection.Callback projectionCallback;mediaProjectionManager (MediaProjectionManager) getSystemService(Context.MEDIA_PROJECTION_SERVICE);请求用户授权录屏权限在回调中获取 MediaProjection 对象。
private static final int REQUEST_CODE_SCREEN_CAPTURE 1;Intent intent mediaProjectionManager.createScreenCaptureIntent();
startActivityForResult(intent, REQUEST_CODE_SCREEN_CAPTURE);Override
protected void onActivityResult(int requestCode, int resultCode, Intent data) {if (requestCode REQUEST_CODE_SCREEN_CAPTURE resultCode RESULT_OK) {mediaProjection mediaProjectionManager.getMediaProjection(resultCode, data);startScreenCapture();}
}开始屏幕录制并通过 MediaProjection.Callback 监听录屏状态。
private VirtualDisplay virtualDisplay;
private MediaRecorder mediaRecorder;private void startScreenCapture() {DisplayMetrics metrics getResources().getDisplayMetrics();int screenWidth metrics.widthPixels;int screenHeight metrics.heightPixels;int screenDensity metrics.densityDpi;mediaRecorder new MediaRecorder();mediaRecorder.setVideoSource(MediaRecorder.VideoSource.SURFACE);mediaRecorder.setOutputFormat(MediaRecorder.OutputFormat.THREE_GPP);mediaRecorder.setVideoEncoder(MediaRecorder.VideoEncoder.H264);mediaRecorder.setVideoSize(screenWidth, screenHeight);mediaRecorder.setVideoFrameRate(30);mediaRecorder.setOutputFile(getOutputFilePath());Surface surface mediaRecorder.getSurface();virtualDisplay mediaProjection.createVirtualDisplay(ScreenCapture,screenWidth,screenHeight,screenDensity,DisplayManager.VIRTUAL_DISPLAY_FLAG_AUTO_MIRROR,surface,null,null);mediaRecorder.prepare();mediaRecorder.start();projectionCallback new MediaProjection.Callback() {Overridepublic void onStop() {stopScreenCapture();}};mediaProjection.registerCallback(projectionCallback, null);
}private void stopScreenCapture() {if (virtualDisplay ! null) {virtualDisplay.release();virtualDisplay null;}if (mediaRecorder ! null) {mediaRecorder.stop();mediaRecorder.reset();mediaRecorder.release();mediaRecorder null;}if (mediaProjection ! null) {mediaProjection.unregisterCallback(projectionCallback);mediaProjection.stop();mediaProjection null;}
}方案二使用 ContentObserver 监听屏幕录制状态变化
创建一个继承自 ContentObserver 的观察者类并重写 onChange() 方法。
public class ScreenRecordObserver extends ContentObserver {private static final String SCREEN_RECORD_STATE_PATH /sys/class/graphics/fb0/screen_state;private OnScreenRecordStateChangedListener listener;public ScreenRecordObserver(Handler handler) {super(handler);}public void setOnScreenRecordStateChangedListener(OnScreenRecordStateChangedListener listener) {this.listener listener;}Overridepublic void onChange(boolean selfChange, Uri uri) {if (listener ! null) {boolean isRecording isScreenRecording();listener.onScreenRecordStateChanged(isRecording);}}private boolean isScreenRecording() {try {FileInputStream fis new FileInputStream(SCREEN_RECORD_STATE_PATH);BufferedReader bufferedReader new BufferedReader(new InputStreamReader(fis));String state bufferedReader.readLine();bufferedReader.close();return !0.equals(state);} catch (Exception e) {e.printStackTrace();}return false;}
}public interface OnScreenRecordStateChangedListener {void onScreenRecordStateChanged(boolean isRecording);
}注册观察者监听屏幕录制状态变化。
private ScreenRecordObserver screenRecordObserver;screenRecordObserver new ScreenRecordObserver(new Handler());
screenRecordObserver.setOnScreenRecordStateChangedListener(new OnScreenRecordStateChangedListener() {Overridepublic void onScreenRecordStateChanged(boolean isRecording) {if (isRecording) {// 屏幕开始录制} else {// 屏幕停止录制}}
});getContentResolver().registerContentObserver(Uri.parse(content:// SCREEN_RECORD_STATE_PATH), true, screenRecordObserver);对比与选型建议
使用 MediaProjection API
方案一的优点
监听到的录屏操作更加准确可靠可以监听到具体的录屏开始和结束时间同时可以获取到录屏内容。可以对录屏过程进行更加灵活的控制如录制分辨率、帧率等。
方案一的缺点
需要申请录屏权限用户需要给予应用访问屏幕内容的权限。需要编写更多的代码进行屏幕录制的控制。
使用 ContentObserver 监听屏幕录制状态变化
方案二的优点
不需要申请额外的权限。监听录屏状态变化的实现相对简单。
方案二的缺点 不能精确确定录屏的开始和结束时间。 无法获取到录屏的具体内容。
选型建议
根据需求如果需要准确监听录屏的开始和结束时间并且需要获取到录屏的具体内容推荐使用方案一。
但如果只是需要知道录屏状态变化而不关心具体的时间和内容方案二比较简单方便。
三.投屏
在 Android 开发中要监听用户的投屏操作可以使用以下两种方案
方案一使用 MediaRouter API
首先在你的 AndroidManifest.xml 文件中添加以下权限
uses-permission android:nameandroid.permission.ACCESS_NETWORK_STATE /
uses-permission android:nameandroid.permission.CHANGE_NETWORK_STATE /
uses-permission android:nameandroid.permission.CHANGE_WIFI_STATE /
uses-permission android:nameandroid.permission.ACCESS_WIFI_STATE /创建一个 MediaRouter.Callback 对象并重写 onRouteSelected() 和 onRouteUnselected() 方法。
private MediaRouter mediaRouter;
private MediaRouter.Callback mediaRouterCallback;mediaRouter (MediaRouter) getSystemService(Context.MEDIA_ROUTER_SERVICE);mediaRouterCallback new MediaRouter.Callback() {Overridepublic void onRouteSelected(MediaRouter router, MediaRouter.RouteInfo route) {// 投屏开始}Overridepublic void onRouteUnselected(MediaRouter router, MediaRouter.RouteInfo route) {// 投屏结束}
};注册监听器并指定监听的路由类型。
mediaRouter.addCallback(MediaRouter.ROUTE_TYPE_LIVE_VIDEO, mediaRouterCallback);方案二使用 DisplayManager API
首先在 AndroidManifest.xml 文件中添加以下权限
uses-permission android:nameandroid.permission.ACCESS_NETWORK_STATE /
uses-permission android:nameandroid.permission.CHANGE_NETWORK_STATE /
uses-permission android:nameandroid.permission.CHANGE_WIFI_STATE /
uses-permission android:nameandroid.permission.ACCESS_WIFI_STATE /创建一个 DisplayManager.DisplayListener 对象并重写 onDisplayAdded()、onDisplayRemoved() 和 onDisplayChanged() 方法。
private DisplayManager displayManager;
private DisplayManager.DisplayListener displayListener;displayManager (DisplayManager) getSystemService(Context.DISPLAY_SERVICE);displayListener new DisplayManager.DisplayListener() {Overridepublic void onDisplayAdded(int displayId) {// 投屏开始}Overridepublic void onDisplayRemoved(int displayId) {// 投屏结束}Overridepublic void onDisplayChanged(int displayId) {// 投屏状态变化}
};注册监听器。
displayManager.registerDisplayListener(displayListener, null);对比与选型建议
使用 MediaRouter API
方案一的优点
使用方便只需要注册一个回调即可监听到投屏操作。可以监听到投屏的开始和结束时间。
方案一的缺点
只能监听到路由类型为 ROUTE_TYPE_LIVE_VIDEO 的投屏操作其他类型的投屏操作无法监听到。
使用 DisplayManager API
方案二的优点
可以监听到所有的投屏操作不限制特定的路由类型。可以监听到投屏的开始和结束时间。
方案二的缺点
需要注册更多的监听器和处理投屏状态变化的逻辑。无法获取到具体的投屏内容。
选型建议
根据需求如果只需要监听特定类型的投屏操作并且需要获取投屏的具体内容推荐使用方案一。
如果需要监听所有类型的投屏操作方案二比较适合。
同时如果只关心投屏的开始和结束时间并不需要具体的投屏内容两种方案都可以考虑。