适配安卓11存储目录部分记录、安卓换头像

为适配Android11 对存储目录的权限修改,我们做出了部分更改,如有不对,请多指教!!
结论:
  1.需要动态请求关于flag 的两个权限
  2.相机由于传入uri之后就不会在OnActivityResult的data中返回数据,所以我们需要提前拿到相机的输出uri
  3.getContentResolver().insert 插入的位置如果已有数据,就会自增 ' (1)',需要注意
  4.我们通过 getContentResolver().delete 删掉原有位置,并且重新insert一条数据,保证uri指向的文件路径永远是同一个


头像修改记录
jar-10
  修改内容
    1.打开所有的debug ,包括 LogUnityError 和 logDebug
    2.选择照片之后的操作从裁剪改成直接读取 关键字 onActivityResult PHOTO_REQUEST_PICK
    3.把选择的照片复制到 缓存目录下 命名为 role_photo_temp.png
    4.加了在 30以上、24-29、其他 情况下的uri获取到的值输出
  预期结果以及下一步操作
    1.显示图片  证明newPath 绝对路径没有问题
    2.文件拷贝成功  对我们本应用程序的目录读写权限没有问题
  实际结果
    文件拷贝成功
    输出 role_photo.png 失败,把输出文件的 new File 相关的干掉,看是否依然会创建 role_photo.png 文件
  部分log记录
    预计的文件输出路径
    ====SDK V24 uri======content://com.special.warship.provider/external_files/Android/data/com.special.warship/files/role_photo.png
    ====SDK VOther uri======file:///storage/emulated/0/Android/data/com.special.warship/files/role_photo.png
    ====SDK V30 uri======content://media/external/images/media/2844
jar-11
  修改内容
    1.把输出文件的 new File 相关的干掉,看是否依然会创建 role_photo.png 文件
  预期结果以及下一步操作
    1.不会创建 role_photo.png 文件,证明 mediaStore 没有把选择的图片输出到指定目录
  实际结果
    file目录没有 role_photo.png 文件,实际证明 mediaStore 没有把选择的图片输出到指定目录
  部分log记录
    无
jar-12
  修改内容
    1.把点击方法更改成 直接调起裁剪 方法,不再经过选图裁剪。分别尝试两种方式 1、绝对路径 file 2、content路径
  预期结果以及下一步操作
    1.
  实际结果
    GetUri30  方法  获取的路径是 content://media/external/images/media/2883 对应的绝对路径是 file:///storage/emulated/0/Pictures/role_photo.png%20(1).jpg
    证明: GetUri30 拿到的不是可读写路径
  部分log记录
    无    
    
jar-14
  修改内容
    1.还原成原有代码,增加部分日志
    2.选择完成后,增加分支判断:使用 Uri.fromFile() 和  getContentResolver().insert(data.getData(), values); 方式获取 Uri
    3.修改 AndroidManifest.xml android.support4.的 FileProvider 改成 androidx.core 的
  预期结果以及下一步操作
    1.
  实际结果
    无效
  部分log记录
    GetHeadPhotoStorage file:/storage/emulated/0/Android/data/com.special.warship/files

jar-16
    修改内容
    1.复制一份选择的照片在原位置 命名+2
    2.动态请求了两个flag权限 Intent.FLAG_GRANT_READ_URI_PERMISSION | Intent.FLAG_GRANT_WRITE_URI_PERMISSION
  预期结果以及下一步操作
    1.
  实际结果
    选择相册中的图进行裁剪,在Picture路径下生成了裁剪后的图片   至此选择图片上传完成,需进一步标注原因
    照相上传图片,没有在Picture路径下生成裁剪后的图片,提示 发生错误,无法加载媒体
  部分log记录

 
jar-24
  修改内容
    1.如果调起相机时传入uri,在OnActivityResult 中的data 会为空,这个情况下会调取在调起相机之前的 takePhotoOutFile 文件进行裁剪。
      而 takePhotoOutFile 读取的是应用程序缓存目录,传入 Intent 的uri 为Picture路径。这个情况下就存在相机输出和裁剪输入的文件不匹配的问题
      在30的情况下,获取到输出uri之后,再次对 takePhotoOutFile 赋值,使 相机输出和裁剪输入的文件 匹配上。就可以解决
  预期结果以及下一步操作
    1.
  实际结果
    解决了相机无法完成裁剪的问题
  部分log记录
    
    
    
备注:30 拿到的路径和 24 的路径不一样,把 30 的路径转换成绝对路径  比对24 的路径,或者 parsePicturePath 的路径
2021-07-30 结论: 选择照片之后调用了 intent.putExtra(MediaStore.EXTRA_OUTPUT, uri); ,但是没有输出文件到指定目录
                  也有可能是选择后不输出,裁剪完成后才输出
                  
                  
Uri.fromFile(file)
    file:///storage/emulated/0/DCIM/download/aaa.jpg
FileProvider.getUriForFile(context, BuildConfig.APPLICATION_ID + ".fileProvider", file)
    content://com.taikang.simtest.fileProvider/camera_photos/DCIM/download/aaa.jpg
Uri.parse(file.getAbsolutePath())
    /storage/emulated/0/DCIM/download/aaa.jpg
    
ContentValues values = new ContentValues();
    values.put(MediaStore.Images.Media.DISPLAY_NAME, photoFileName);
    values.put(MediaStore.Images.Media.MIME_TYPE, "image/jpeg");
    uri = this.getContentResolver().insert(MediaStore.Images.Media.EXTERNAL_CONTENT_URI, values);
得到的uri 是 content://media/external/images/media/3201 是图片库的一个虚拟地址
    对应的是 /storage/emulated/0/Pictures/role_photo.png (1).jpg
    
这两条的意思应该是指手机媒体库,不是本应用程序的目录
    MediaStore.Images.Media.EXTERNAL_CONTENT_URI  外部存储,支持写入
    MediaStore.Images.Media.INTERNAL_CONTENT_URI  内部存储,不支持写入

    
如果您需要与其他应用共享特定文件,请使用 FileProvider API。

  • 5
    点赞
  • 7
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
package com.example.tigongzhe; import android.R.integer; import android.content.ContentProvider; import android.content.ContentUris; import android.content.ContentValues; import android.content.Context; import android.content.UriMatcher; import android.database.Cursor; import android.database.sqlite.SQLiteDatabase; import android.database.sqlite.SQLiteDatabase.CursorFactory; import android.database.sqlite.SQLiteOpenHelper; import android.net.Uri; import android.provider.SyncStateContract.Helpers; import android.text.Selection; import android.util.Log; public class provider extends ContentProvider { private MyOpenHelper myOpenHelper; private SQLiteDatabase sqLiteDatabase; private static final UriMatcher URI_MATCHER=new UriMatcher(UriMatcher.NO_MATCH); private final String TAG="provider"; private static final String authority="com.example.tigongzhe.provider"; static { URI_MATCHER.addURI(authority, "contacter", 1); URI_MATCHER.addURI(authority, "contacter/#", 2); } private static final String _id="id"; private static final String name="name"; private static final String num="num"; @Override public boolean onCreate() { // TODO Auto-generated method stub myOpenHelper=new MyOpenHelper(getContext(), DB_Name, null, version_1); return true; } @Override public String getType(Uri uri) { // TODO Auto-generated method stub int flag=URI_MATCHER.match(uri); switch (flag) { case 2: return "vnd.android.cursor.item/contacter"; case 1: return "vnd.android.dir.item/contacter"; default: throw new IllegalArgumentException("异常参数"); } } @Override public Uri insert(Uri uri, ContentValues values) { // TODO Auto-generated method stub sqLiteDatabase=myOpenHelper.getWritableDatabase(); int flag=URI_MATCHER.match(uri); switch (flag) { case 1: sqLiteDatabase.insert(Table_Name, name, values); break; case 2: long id=sqLiteDatabase.insert(Table_Name, name, values); ContentUris.withAppendedId(uri, id); default: break; } return uri; } @Override public Cursor query(Uri uri, String[] arg1, String arg2, String[] arg3, String arg4) { // TODO Auto-generated method stub Cursor cursor; sqLiteDatabase=myOpenHelper.getReadableDatabase(); int flag=URI_MATCHER.match(uri); switch (flag) { case 1: cursor=sqLiteDatabase.query(Table_Name, arg1, arg2, arg3, null, null,arg4); break; case 2: long id=ContentUris.parseId(uri); arg2=(arg2==null||"".equals(arg2.trim()))? _id+"="+id:arg2+"and"+_id+"="+id; cursor=sqLiteDatabase.query(Table_Name, arg1, arg2, arg3, null, null,arg4); default: throw new IllegalArgumentException("参数错误"); } return cursor; } @Override public int update(Uri uri, ContentValues values, String selection, String[] selectionArgs) { // TODO Auto-generated method stub int num=0; sqLiteDatabase=myOpenHelper.getWritableDatabase(); int flag=URI_MATCHER.match(uri); switch (flag) { case 1: num=sqLiteDatabase.update(Table_Name, values,selection, selectionArgs); break; case 2: long id=ContentUris.parseId(uri); selection=(selection==null||"".equals(selection.trim()))? _id+"="+id:selection+"and"+_id+"="+id; num=sqLiteDatabase.update(Table_Name, values,selection, selectionArgs); default: break; } return num; } @Override public int delete(Uri uri, String selection, String[] selectionArgs) { // TODO Auto-generated method stub int num=0; sqLiteDatabase=myOpenHelper.getWritableDatabase(); int flag=URI_MATCHER.match(uri); switch (flag) { case 1: num=sqLiteDatabase.delete(Table_Name, selection, selectionArgs); break; case 2: long id=ContentUris.parseId(uri); selection=(selection==null||"".equals(selection.trim()))?_id+"="+id:selection+"and"+_id+"="+id; num=sqLiteDatabase.delete(Table_Name, selection, selectionArgs); default: throw new IllegalArgumentException("异常参数"); } return num; } private final String DB_Name = "mydb.db"; private final String Table_Name="contacter"; private final int version_1=1; private class MyOpenHelper extends SQLiteOpenHelper { public MyOpenHelper(Context context, String name, CursorFactory factory, int version) { super(context, name, factory, version); } /** * @description 当数据表无连接时创建新的表 */ @Override public void onCreate(SQLiteDatabase db) { String sql = " create table if not exists " + Table_Name + "(id INTEGER PRIMARY KEY AUTOINCREMENT," + "name varchar(64),num varchar(64))"; db.execSQL(sql); } /** * @description 当版本更新时触发的方法 */ @Override public void onUpgrade(SQLiteDatabase db, int oldVersion, int newVersion) { String sql = " drop table if exists " + Table_Name; db.execSQL(sql); onCreate(db); } } } +++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++ <?xml version="1.0" encoding="utf-8"?> <manifest xmlns:android="http://schemas.android.com/apk/res/android" package="com.example.tigongzhe" android:versionCode="1" android:versionName="1.0" > <uses-sdk android:minSdkVersion="8" android:targetSdkVersion="18" /> <application android:allowBackup="true" android:icon="@drawable/ic_launcher" android:label="@string/app_name" android:theme="@style/AppTheme" > <activity android:name="com.example.tigongzhe.MainActivity" android:label="@string/app_name" > <intent-filter> <action android:name="android.intent.action.MAIN" /> <category android:name="android.intent.category.LAUNCHER" /> </intent-filter> </activity> <provider android:name=".provider" android:authorities="com.example.tigongzhe.provider" android:multiprocess="true" android:exported="true" android:permission="com.example.tigongzhe.permission" ></provider> </application> <permission android:name="com.example.tigongzhe.permission" android:protectionLevel="normal"></permission> </manifest> ++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++package com.example.tigongzhe; import android.R.integer; import android.content.ContentProvider; import android.content.ContentUris; import android.content.ContentValues; import android.content.Context; import android.content.UriMatcher; import android.database.Cursor; import android.database.sqlite.SQLiteDatabase; import android.database.sqlite.SQLiteDatabase.CursorFactory; import android.database.sqlite.SQLiteOpenHelper; import android.net.Uri; import android.provider.SyncStateContract.Helpers; import android.text.Selection; import android.util.Log; public class provider extends ContentProvider { private MyOpenHelper myOpenHelper; private SQLiteDatabase sqLiteDatabase; private static final UriMatcher URI_MATCHER=new UriMatcher(UriMatcher.NO_MATCH); private final String TAG="provider"; private static final String authority="com.example.tigongzhe.provider"; static { URI_MATCHER.addURI(authority, "contacter", 1); URI_MATCHER.addURI(authority, "contacter/#", 2); } private static final String _id="id"; private static final String name="name"; private static final String num="num"; @Override public boolean onCreate() { // TODO Auto-generated method stub myOpenHelper=new MyOpenHelper(getContext(), DB_Name, null, version_1); return true; } @Override public String getType(Uri uri) { // TODO Auto-generated method stub int flag=URI_MATCHER.match(uri); switch (flag) { case 2: return "vnd.android.cursor.item/contacter"; case 1: return "vnd.android.dir.item/contacter"; default: throw new IllegalArgumentException("异常参数"); } } @Override public Uri insert(Uri uri, ContentValues values) { // TODO Auto-generated method stub sqLiteDatabase=myOpenHelper.getWritableDatabase(); int flag=URI_MATCHER.match(uri); switch (flag) { case 1: sqLiteDatabase.insert(Table_Name, name, values); break; case 2: long id=sqLiteDatabase.insert(Table_Name, name, values); ContentUris.withAppendedId(uri, id); default: break; } return uri; } @Override public Cursor query(Uri uri, String[] arg1, String arg2, String[] arg3, String arg4) { // TODO Auto-generated method stub Cursor cursor; sqLiteDatabase=myOpenHelper.getReadableDatabase(); int flag=URI_MATCHER.match(uri); switch (flag) { case 1: cursor=sqLiteDatabase.query(Table_Name, arg1, arg2, arg3, null, null,arg4); break; case 2: long id=ContentUris.parseId(uri); arg2=(arg2==null||"".equals(arg2.trim()))? _id+"="+id:arg2+"and"+_id+"="+id; cursor=sqLiteDatabase.query(Table_Name, arg1, arg2, arg3, null, null,arg4); default: throw new IllegalArgumentException("参数错误"); } return cursor; } @Override public int update(Uri uri, ContentValues values, String selection, String[] selectionArgs) { // TODO Auto-generated method stub int num=0; sqLiteDatabase=myOpenHelper.getWritableDatabase(); int flag=URI_MATCHER.match(uri); switch (flag) { case 1: num=sqLiteDatabase.update(Table_Name, values,selection, selectionArgs); break; case 2: long id=ContentUris.parseId(uri); selection=(selection==null||"".equals(selection.trim()))? _id+"="+id:selection+"and"+_id+"="+id; num=sqLiteDatabase.update(Table_Name, values,selection, selectionArgs); default: break; } return num; } @Override public int delete(Uri uri, String selection, String[] selectionArgs) { // TODO Auto-generated method stub int num=0; sqLiteDatabase=myOpenHelper.getWritableDatabase(); int flag=URI_MATCHER.match(uri); switch (flag) { case 1: num=sqLiteDatabase.delete(Table_Name, selection, selectionArgs); break; case 2: long id=ContentUris.parseId(uri); selection=(selection==null||"".equals(selection.trim()))?_id+"="+id:selection+"and"+_id+"="+id; num=sqLiteDatabase.delete(Table_Name, selection, selectionArgs); default: throw new IllegalArgumentException("异常参数"); } return num; } private final String DB_Name = "mydb.db"; private final String Table_Name="contacter"; private final int version_1=1; private class MyOpenHelper extends SQLiteOpenHelper { public MyOpenHelper(Context context, String name, CursorFactory factory, int version) { super(context, name, factory, version); } /** * @description 当数据表无连接时创建新的表 */ @Override public void onCreate(SQLiteDatabase db) { String sql = " create table if not exists " + Table_Name + "(id INTEGER PRIMARY KEY AUTOINCREMENT," + "name varchar(64),num varchar(64))"; db.execSQL(sql); } /** * @description 当版本更新时触发的方法 */ @Override public void onUpgrade(SQLiteDatabase db, int oldVersion, int newVersion) { String sql = " drop table if exists " + Table_Name; db.execSQL(sql); onCreate(db); } } }
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值