Android10填坑適配指南,包含實際經驗代碼,毫不照搬翻譯文檔php
當 targetSdkVersion >= Build.VERSION_CODES.P 時調用 canvas.clipPath(path, Region.Op.XXX); 引發的異常,參考源碼以下:java
@Deprecated
public boolean clipPath(@NonNull Path path, @NonNull Region.Op op) {
checkValidClipOp(op);
return nClipPath(mNativeCanvasWrapper, path.readOnlyNI(), op.nativeInt);
}
private static void checkValidClipOp(@NonNull Region.Op op) {
if (sCompatiblityVersion >= Build.VERSION_CODES.P
&& op != Region.Op.INTERSECT && op != Region.Op.DIFFERENCE) {
throw new IllegalArgumentException(
"Invalid Region.Op - only INTERSECT and DIFFERENCE are allowed");
}
}
複製代碼
咱們能夠看到當目標版本從Android P開始,Canvas.clipPath(@NonNull Path path, @NonNull Region.Op op) ; 已經被廢棄,並且是包含異常風險的廢棄API,只有 Region.Op.INTERSECT 和 Region.Op.DIFFERENCE 獲得兼容,幾乎全部的博客解決方案都是以下簡單粗暴:android
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.P) {
canvas.clipPath(path);
} else {
canvas.clipPath(path, Region.Op.XOR);// REPLACE、UNION 等
}
複製代碼
但咱們必定須要一些高級邏輯運算效果怎麼辦?如小說的仿真翻頁閱讀效果,解決方案以下,用Path.op代替,先運算Path,再給canvas.clipPath:數據庫
if(Build.VERSION.SDK_INT >= Build.VERSION_CODES.P){
Path mPathXOR = new Path();
mPathXOR.moveTo(0,0);
mPathXOR.lineTo(getWidth(),0);
mPathXOR.lineTo(getWidth(),getHeight());
mPathXOR.lineTo(0,getHeight());
mPathXOR.close();
//以上根據實際的Canvas或View的大小,畫出相同大小的Path便可
mPathXOR.op(mPath0, Path.Op.XOR);
canvas.clipPath(mPathXOR);
}else {
canvas.clipPath(mPath0, Region.Op.XOR);
}
複製代碼
當 targetSdkVersion >= Build.VERSION_CODES.P 時,默認限制了HTTP請求,並出現相關日誌:canvas
java.net.UnknownServiceException: CLEARTEXT communication to xxx not permitted by network security policy
app
第一種解決方案:在AndroidManifest.xml中Application添加以下節點代碼ide
<application android:usesCleartextTraffic="true">
佈局
第二種解決方案:在res目錄新建xml目錄,已建的跳過 在xml目錄新建一個xml文件network_security_config.xml,而後在AndroidManifest.xml中Application添加以下節點代碼post
android:networkSecurityConfig="@xml/network_config"
ui
名字隨機,內容以下:
<?xml version="1.0" encoding="utf-8"?>
<network-security-config>
<base-config cleartextTrafficPermitted="true" />
</network-security-config>
複製代碼
一、掃描系統相冊、視頻等,圖片、視頻選擇器都是經過ContentResolver來提供,主要代碼以下:
private static final String[] IMAGE_PROJECTION = {
MediaStore.Images.Media.DATA,
MediaStore.Images.Media.DISPLAY_NAME,
MediaStore.Images.Media._ID,
MediaStore.Images.Media.BUCKET_ID,
MediaStore.Images.Media.BUCKET_DISPLAY_NAME};
Cursor imageCursor = mContext.getContentResolver().query(
MediaStore.Images.Media.EXTERNAL_CONTENT_URI,
IMAGE_PROJECTION, null, null, IMAGE_PROJECTION[0] + " DESC");
String path = imageCursor.getString(imageCursor.getColumnIndexOrThrow(IMAGE_PROJECTION[0]));
String name = imageCursor.getString(imageCursor.getColumnIndexOrThrow(IMAGE_PROJECTION[1]));
int id = imageCursor.getInt(imageCursor.getColumnIndexOrThrow(IMAGE_PROJECTION[2]));
String folderPath = imageCursor.getString(imageCursor.getColumnIndexOrThrow(IMAGE_PROJECTION[3]));
String folderName = imageCursor.getString(imageCursor.getColumnIndexOrThrow(IMAGE_PROJECTION[4]));
//Android Q 公有目錄只能經過Content Uri + id的方式訪問,之前的File路徑所有無效,若是是Video,記得換成MediaStore.Videos
if(Build.VERSION.SDK_INT >= Build.VERSION_CODES.Q){
path = MediaStore.Images.Media
.EXTERNAL_CONTENT_URI
.buildUpon()
.appendPath(String.valueOf(id)).build().toString();
}
複製代碼
二、判斷公有目錄文件是否存在,自Android Q開始,公有目錄File API都失效,不能直接經過new File(path).exists();判斷公有目錄文件是否存在,正確方式以下:
public static boolean isAndroidQFileExists(Context context, String path){
AssetFileDescriptor afd = null;
ContentResolver cr = context.getContentResolver();
try {
Uri uri = Uri.parse(path);
afd = cr.openAssetFileDescriptor(uri, "r");
if (afd == null) {
return false;
} else {
close(afd);
}
} catch (FileNotFoundException e) {
return false;
}finally {
close(afd);
}
return true;
}
複製代碼
三、copy或者下載文件到公有目錄,保存Bitmap同理,如Download,MIME_TYPE類型能夠自行參考對應的文件類型,這裏只對APK做出說明,從私有目錄copy到公有目錄demo以下(遠程下載同理,只要拿到OutputStream便可,亦可下載到私有目錄再copy到公有目錄):
public static void copyToDownloadAndroidQ(Context context, String sourcePath, String fileName, String saveDirName){
ContentValues values = new ContentValues();
values.put(MediaStore.Downloads.DISPLAY_NAME, fileName);
values.put(MediaStore.Downloads.MIME_TYPE, "application/vnd.android.package-archive");
values.put(MediaStore.Downloads.RELATIVE_PATH, "Download/" + saveDirName.replaceAll("/","") + "/");
Uri external = MediaStore.Downloads.EXTERNAL_CONTENT_URI;
ContentResolver resolver = context.getContentResolver();
Uri insertUri = resolver.insert(external, values);
if(insertUri == null) {
return;
}
String mFilePath = insertUri.toString();
InputStream is = null;
OutputStream os = null;
try {
os = resolver.openOutputStream(insertUri);
if(os == null){
return;
}
int read;
File sourceFile = new File(sourcePath);
if (sourceFile.exists()) { // 文件存在時
is = new FileInputStream(sourceFile); // 讀入原文件
byte[] buffer = new byte[1444];
while ((read = is.read(buffer)) != -1) {
os.write(buffer, 0, read);
}
}
} catch (Exception e) {
e.printStackTrace();
}finally {
close(is,os);
}
}
複製代碼
四、保存圖片相關
/** * 經過MediaStore保存,兼容AndroidQ,保存成功自動添加到相冊數據庫,無需再發送廣播告訴系統插入相冊 * * @param context context * @param sourceFile 源文件 * @param saveFileName 保存的文件名 * @param saveDirName picture子目錄 * @return 成功或者失敗 */
public static boolean saveImageWithAndroidQ(Context context, File sourceFile, String saveFileName, String saveDirName) {
String extension = BitmapUtil.getExtension(sourceFile.getAbsolutePath());
ContentValues values = new ContentValues();
values.put(MediaStore.Images.Media.DESCRIPTION, "This is an image");
values.put(MediaStore.Images.Media.DISPLAY_NAME, saveFileName);
values.put(MediaStore.Images.Media.MIME_TYPE, "image/png");
values.put(MediaStore.Images.Media.TITLE, "Image.png");
values.put(MediaStore.Images.Media.RELATIVE_PATH, "Pictures/" + saveDirName);
Uri external = MediaStore.Images.Media.EXTERNAL_CONTENT_URI;
ContentResolver resolver = context.getContentResolver();
Uri insertUri = resolver.insert(external, values);
BufferedInputStream inputStream = null;
OutputStream os = null;
boolean result = false;
try {
inputStream = new BufferedInputStream(new FileInputStream(sourceFile));
if (insertUri != null) {
os = resolver.openOutputStream(insertUri);
}
if (os != null) {
byte[] buffer = new byte[1024 * 4];
int len;
while ((len = inputStream.read(buffer)) != -1) {
os.write(buffer, 0, len);
}
os.flush();
}
result = true;
} catch (IOException e) {
result = false;
} finally {
close(os, inputStream);
}
return result;
}
複製代碼
該問題出如今 targetSdkVersion >= Build.VERSION_CODES.P 狀況下,且設備版本爲Android P以上版本,解決方法在onCreate中加入以下代碼,可得到焦點,如須要彈出鍵盤可延遲一下:
mEditText.post(() -> {
mEditText.requestFocus();
mEditText.setFocusable(true);
mEditText.setFocusableInTouchMode(true);
});
複製代碼
/* * 自Android N開始,是經過FileProvider共享相關文件,可是Android Q對公有目錄 File API進行了限制,只能經過Uri來操做, * 從代碼上看,又變得和之前低版本同樣了,只是必須加上權限代碼Intent.FLAG_GRANT_READ_URI_PERMISSION */
private void installApk() {
if(Build.VERSION.SDK_INT >= Build.VERSION_CODES.Q){
//適配Android Q,注意mFilePath是經過ContentResolver獲得的,上述有相關代碼
Intent intent = new Intent(Intent.ACTION_VIEW);
intent.setDataAndType(Uri.parse(mFilePath) ,"application/vnd.android.package-archive");
intent.setFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
intent.addFlags(Intent.FLAG_GRANT_READ_URI_PERMISSION);
startActivity(intent);
return ;
}
File file = new File(saveFileName + "demo.apk");
if (!file.exists())
return;
Intent intent = new Intent(Intent.ACTION_VIEW);
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.N) {
intent.setFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
intent.addFlags(Intent.FLAG_GRANT_READ_URI_PERMISSION);
Uri contentUri = FileProvider.getUriForFile(getApplicationContext(), "net.oschina.app.provider", file);
intent.setDataAndType(contentUri, "application/vnd.android.package-archive");
} else {
intent.setDataAndType(Uri.fromFile(file), "application/vnd.android.package-archive");
intent.setFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
}
startActivity(intent);
}
複製代碼
Android Q 又一個天坑,若是你要顯示一個半透明的Activity,這在android10以前普通樣式Activity只須要設置windowIsTranslucent=true便可,可是到了AndroidQ,它沒有效果了,並且若是動態設置View.setVisibility(),界面還會出現殘影...
解決辦法:使用Dialog樣式Activity,且設置windowIsFloating=true,此時問題又來了,若是Activity根佈局沒有設置fitsSystemWindow=true,默認是沒有侵入狀態欄的,使界面看上去正常。
Android Q中只有當應用處於可交互狀況(默認輸入法自己就可交互)才能訪問剪切板和監聽剪切板變化,在onResume回調也沒法直接訪問剪切板,這麼作的好處是避免了一些應用後臺瘋狂監聽響應剪切板的內容,瘋狂彈窗。
所以若是還須要監聽剪切板,可使用應用生命週期回調,監聽APP後臺返回,延遲幾毫秒訪問剪切板,再保存最後一次訪問獲得的剪切板內容,每次都比較一下是否有變化,再進行下一步操做。