android5.0截图,Android5.0以上截图方法,适配targetSdkVersion 29以上

本文详细介绍了如何在Android 5.0及以上版本实现后台连续截图的方法,包括通过MediaProjectionManager获取权限,启动服务进行截图,并处理ImageReader的回调以实现连续截图功能。同时,代码示例展示了如何创建并使用虚拟显示以及处理图片数据。

适配了Android5.0以上的截图方法,实现了后台连续截图;使用该方法请求授权同意后开始截图:

private MediaProjectionManager mMediaProjectionManager;

public void getMediaProjectionManger() {

mMediaProjectionManager = (MediaProjectionManager) getSystemService(Context.MEDIA_PROJECTION_SERVICE);

if (mMediaProjectionManager != null) {

startActivityForResult(mMediaProjectionManager.createScreenCaptureIntent(), RECORD_REQUEST_CODE);

}

}

由于targetSdkVersion 29及以上要求getMediaProjection需要再前台服务执行,所以在onActivityResult里面启动服务:

@Override

protected void onActivityResult(int requestCode, int resultCode, @Nullable Intent data) {

super.onActivityResult(requestCode, resultCode, data);

if (requestCode == RECORD_REQUEST_CODE && resultCode == RESULT_OK) {

Intent intent = new Intent(ZkScreenshotActivity.this, ZkA5AboveScreenshotService.class);

intent.putExtra("code",resultCode);

intent.putExtra("data",data);

ZkUtlis.startService(ZkScreenshotActivity.this,intent);

}

}

Service方法如下:

public class ZkA5AboveScreenshotService extends Service {

private String mTag = "ZkA5AboveScreenshotService";

private int mResultCode;

private Intent mResultData;

private MediaProjection mMediaProjection;

private MediaProjectionManager mMediaProjectionManager;

private int mWidth;

private int mHeight;

private int mDensity;

private ImageReader mImageReader;

private int mImagesProduced;

private int mCount = 0;

@Override

public void onCreate() {

super.onCreate();

mMediaProjectionManager = (MediaProjectionManager) getSystemService(Context.MEDIA_PROJECTION_SERVICE);

}

@Nullable

@Override

public IBinder onBind(Intent intent) {

return null;

}

@Override

public int onStartCommand(Intent intent, int flags, int startId) {

createNotificationChannel();

mResultCode = intent.getIntExtra("code", -1);

mResultData = intent.getParcelableExtra("data");

mMediaProjection = mMediaProjectionManager.getMediaProjection(mResultCode, Objects.requireNonNull(mResultData));

Log.d(mTag, "mMediaProjection created: " + mMediaProjection);

startCaptureReader();

return super.onStartCommand(intent, flags, startId);

}

/**

* startCaptureReader

*/

@SuppressLint("WrongConstant")

private void startCaptureReader() {

WindowManager wm = (WindowManager) getBaseContext().getSystemService(Context.WINDOW_SERVICE);

Display display = wm.getDefaultDisplay();

Point size = new Point();

display.getSize(size);

mWidth = size.x;

mHeight = size.y;

Log.d(mTag,"width:" + mWidth + " height:" + mHeight);

DisplayMetrics metrics = getResources().getDisplayMetrics();

mDensity = metrics.densityDpi;

// start capture reader

mImageReader = ImageReader.newInstance(mWidth, mHeight, 0x01, 2);

mMediaProjection.createVirtualDisplay("screencap", mWidth, mHeight, mDensity, DisplayManager.VIRTUAL_DISPLAY_FLAG_AUTO_MIRROR, mImageReader.getSurface(), null, null);

mImageReader.setOnImageAvailableListener(new ImageAvailableListener(), null);

}

/**

* ImageAvailableListener

*/

private class ImageAvailableListener implements ImageReader.OnImageAvailableListener {

@Override

public void onImageAvailable(ImageReader reader) {

try (Image image = reader.acquireLatestImage()) {

if (image != null) {

/**

* 截图结果,会在内容更新自动回调,该方法可以实现连续截图,无需重复启用截图方法

*/

}

image.close();

}

} catch (Exception e) {

e.printStackTrace();

}

}

}

/**

* Create Notification Channel

*/

private void createNotificationChannel() {

//获取一个Notification构造器

Notification.Builder builder = new Notification.Builder(this.getApplicationContext());

//点击后跳转的界面,可以设置跳转数据

Intent nfIntent = new Intent(this, ZkScreenshotActivity.class);

// 设置PendingIntent

builder.setContentIntent(PendingIntent.getActivity(this, 0, nfIntent, 0))

// 设置下拉列表中的图标(大图标)

.setLargeIcon(BitmapFactory.decodeResource(this.getResources(), R.mipmap.ic_launcher))

// 设置下拉列表里的标题

//.setContentTitle("SMI InstantView")

// 设置状态栏内的小图标

.setSmallIcon(R.mipmap.ic_launcher)

// 设置上下文内容

.setContentText("is running......")

// 设置该通知发生的时间

.setWhen(System.currentTimeMillis());

/*以下是对Android 8.0的适配*/

//普通notification适配

if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.O) {

builder.setChannelId("notification_id");

}

//前台服务notification适配

if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.O) {

NotificationManager notificationManager = (NotificationManager)getSystemService(NOTIFICATION_SERVICE);

NotificationChannel channel = new NotificationChannel("notification_id", "notification_name", NotificationManager.IMPORTANCE_LOW);

notificationManager.createNotificationChannel(channel);

}

// 获取构建好的Notification

Notification notification = builder.build();

//设置为默认的声音

notification.defaults = Notification.DEFAULT_SOUND;

startForeground(110, notification);

}

@Override

public void onDestroy() {

super.onDestroy();

Log.d(mTag, "Service onDestroy");

if (mImageReader != null) {

mImageReader.setOnImageAvailableListener(null,null);

mImageReader.close();

mImageReader = null;

}

if (mMediaProjection != null) {

mMediaProjection.stop();

mMediaProjection = null;

}

mMediaProjectionManager = null;

}

下面是将Image转Bitmap

/**

* Image 转 Bitmap

* @param image

*/

private Bitmap imageToBitmap(Image image) {

Image.Plane[] planes = image.getPlanes();

ByteBuffer buffer = planes[0].getBuffer();

int pixelStride = planes[0].getPixelStride();

int rowStride = planes[0].getRowStride();

int rowPadding = rowStride - pixelStride * mWidth;

Bitmap bitmap = Bitmap.createBitmap(mWidth + rowPadding / pixelStride, mHeight, Bitmap.Config.ARGB_8888);

bitmap.copyPixelsFromBuffer(buffer);

return bitmap;

}

适配 Android 29 SDK(即 Android Q 或 Android 10),需要选择与之兼容的 Gradle 版本和 Android Gradle 插件(AGP)版本。通常,Android SDK 版本与 Gradle 及 AGP 的兼容性依赖于三者之间的版本匹配关系。 Android Gradle 插件(AGP)8.0.x 及以上版本开始支持 Android 13(SDK 33),而 Android 29(SDK 29)则在 AGP 3.5.x 到 5.0.x 范围内得到良好支持。对应的 Gradle 版本应为 5.4.1 到 6.7 之间,以确保兼容性和稳定性 [^1]。 具体推荐如下: - **Android Gradle 插件版本**:`5.0.1` 或 `3.5.4` - **Gradle 版本**:`5.6.4` 或 `6.1.1` - **SDK 版本**:`29` 在 `build.gradle` 项目的根级文件中,指定 AGP 插件版本: ```gradle buildscript { repositories { google() mavenCentral() } dependencies { classpath 'com.android.tools.build:gradle:5.0.1' } } ``` 在 `gradle-wrapper.properties` 文件中指定 Gradle 版本: ```properties distributionUrl=https\://services.gradle.org/distributions/gradle-6.1.1-all.zip ``` 此外,在模块级 `build.gradle` 文件中设置 SDK 版本: ```gradle android { compileSdkVersion 29 defaultConfig { minSdkVersion 21 targetSdkVersion 29 versionCode 1 versionName "1.0" } } ``` 如果使用的是较新版本的 Android Studio,可能需要通过设置中的 Gradle 配置选择本地已安装的低版本 Gradle 分发包,以避免版本不兼容问题 [^2]。 ### 相关问题 1. 如何配置 Gradle 以支持 Android 11(SDK 30)? 2. Android Gradle 插件版本与 Gradle 版本之间的兼容性如何确定? 3. 使用旧版 SDK 开发时为何会出现 Gradle 版本不兼容错误? 4. 如何手动下载并配置特定版本的 Gradle? 5. Android SDK 版本升级后,Gradle 和 AGP 应该如何调整?
评论
成就一亿技术人!
拼手气红包6.0元
还能输入1000个字符  | 博主筛选后可见
 
红包 添加红包
表情包 插入表情
 条评论被折叠 查看
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值