0

Xamarin を使用して Android Q (API 29) を対象とする Android プロジェクトで、sdcard のルートにある "Download" フォルダーに (apk) ファイルをダウンロードしようとしています。

まず、sdcard/Android/data/com.my.app/files にあるアプリのプライベート ストレージ スペースにファイルをダウンロードします。

次に、MediaStore API を使用して、ファイルをパブリックの "Download" フォルダーにコピーします。これは私のコードです:

private void SaveTest(string appPath, string fileName)
    {
        var contentValues = new ContentValues();
        contentValues.Put(MediaStore.MediaColumns.DisplayName, fileName);
        contentValues.Put(MediaStore.MediaColumns.MimeType, "application/vnd.android.package-archive");
        contentValues.Put(MediaStore.MediaColumns.RelativePath, Android.OS.Environment.DirectoryDownloads);

        var context = (Context)this;
        var resolver = context.ContentResolver;

        Stream stream = null;
        Android.Net.Uri uri = null;

        try
        {
            var contentUri = MediaStore.Downloads.ExternalContentUri;

            var cursor = resolver.Query(
                contentUri,
                new[] { MediaStore.MediaColumns.Id, MediaStore.MediaColumns.DisplayName, MediaStore.MediaColumns.MimeType, MediaStore.MediaColumns.RelativePath },
                $"{MediaStore.MediaColumns.DisplayName} = ? AND {MediaStore.MediaColumns.MimeType} = ? AND {MediaStore.MediaColumns.RelativePath} = ?",
                new[] { fileName, "application/vnd.android.package-archive", Android.OS.Environment.DirectoryDownloads },
                null
                );
            if(cursor != null && cursor.Count >= 1)
            {
                cursor.MoveToFirst();

                var id = cursor.GetLong(cursor.GetColumnIndexOrThrow(MediaStore.MediaColumns.Id));
                var displayName = cursor.GetString(cursor.GetColumnIndexOrThrow(MediaStore.MediaColumns.DisplayName));
                var relativePath = cursor.GetString(cursor.GetColumnIndexOrThrow(MediaStore.MediaColumns.MimeType));
                var lastModifiedDate = cursor.GetString(cursor.GetColumnIndexOrThrow(MediaStore.MediaColumns.RelativePath));

                uri = ContentUris.WithAppendedId(contentUri, id);
            }
            else
            {
                uri = resolver.Insert(contentUri, contentValues);
            }
            cursor?.Close();

            if (uri == null)
            {
                throw new Exception("Retrieving of creating MediaStore record failed.");
            }

            byte[] buffer = new byte[1024];
            stream = resolver.OpenOutputStream(uri);

            var fullPathAndFileName = Path.Combine(appPath, fileName);
            using (var inputstream = new FileStream(fullPathAndFileName, FileMode.Open))
            {
                CopyFile(inputstream, stream);
            }

        }
        catch
        {
            if (uri != null)
            {
                // Don't leave an orphan entry in the MediaStore
                resolver.Delete(uri, null, null);
            }

            throw;
        }
        finally
        {
            stream?.Close();
        }
    }

MediaStore への挿入は初めて機能しますが、2 回目に挿入しようとすると、返された Uri が null になります。したがって、既存のレコードを検出できるように ContentResolver クエリを追加しましたが、このクエリは結果を返しません。別のファイル名を使用した場合にのみ、再び機能します。

デバイス ログを見ると、次のエラーが表示されます。

時間 デバイス名 タイプ PID タグ メッセージ 05-14 14:14:27.442 エラー 14571 SQLiteDatabase android.database.sqlite.SQLiteConstraintException: UNIQUE 制約が失敗しました: files._data (コード 2067 SQLITE_CONSTRAINT_UNIQUE) at android.database.sqlite.SQLiteConnection.nativeExecuteForLastInsertedRowId(Nativeメソッド) で android.database.sqlite.SQLiteConnection.executeForLastInsertedRowId(SQLiteConnection.java:879) で android.database.sqlite.SQLiteSession.executeForLastInsertedRowId(SQLiteSession.java:790) で android.database.sqlite.SQLiteStatement.executeInsert(SQLiteStatement.java :88) com.android.providers.media.MediaProvider.com.android.providers.media.MediaProvider.insertInternal(MediaProvider.java:3317) の insertFile(MediaProvider.java:2776) com.android.providers.media.MediaProvider.insert(MediaProvider.java:2966) の android.content .ContentProvider$Transport.insert(ContentProvider.java:309) で android.content.ContentProviderNative.onTransact(ContentProviderNative.java:154) で android.os.Binder.execTransactInternal(Binder.java:1021) で android.os.Binder. execTransact(Binder.java:994) 05-14 14:14:27.442 エラー 14571 SQLiteDatabase バケットの挿入中にエラーが発生しました_display_name=ダウンロード owner_package_name=com.my.app 親=8 ボリューム名=external_primary date_modified=1589458467 _display_name=XXX.apk mime_type=application/vnd .android.package-archive _data=/storage/emulated/0/Download/XXX.apk title=XXX.apk group_id=102105 is_download=true date_added=1589458467 primary_directory=ダウンロード bucket_id=540528482 media_type=0 relative_path=Download/

フィルターなしでメディアストア クエリを実行しても、結果カウントは 0 です。正しい Uri を取得するにはどうすればよいですか?

4

0 に答える 0