简体   繁体   中英

How to save Document file in external storage after android API LEVEL28 With Android SAF(Storage Access Framework))

This Code Works Fine With Media Files I want a solution For Document Files

I Don't Know how to put contentValues For Document Files

fun getFile(fileName: String): File? {
    with(sharePrefHelper.app){
        if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.Q) {
            val values = ContentValues()
            // Here is My Question That what should i Do Here Because this is for document not for image
            values.put(MediaStore.Images.Media.DISPLAY_NAME, fileName)
            // for MIME_TYPE "image/jpg" this is working
            values.put(MediaStore.Images.Media.MIME_TYPE, "text/csv")
            values.put(MediaStore.MediaColumns.RELATIVE_PATH, "DCIM/Donny")
            contentResolver.insert(MediaStore.Images.Media.EXTERNAL_CONTENT_URI, values)?.let {
                it.path?.let { finalPath ->
                    return File(finalPath)
                }
            }
        } else {
            val directory: File = Environment.getExternalStoragePublicDirectory(Environment.DIRECTORY_DCIM + "/Donny")
            if (!directory.exists()){
                directory.mkdirs()
            }
            return File(directory, fileName)
        }
        return null
    }
}

This Code Works Fine with media Files

My Question Here is How to save documents like CSV File in outer folder of android device

EDIT :

Well well well, I'm still trying to add anytype of file in the "download" directory. Personnaly, I'm trying to copy a file from my assetFolder and paste it to the "Download" folder. I haven't succeeded yet. However, I can currently CREATE anytype of file in that folder, it's working with the method below. I hope this can help you.

Here is my code :

    public void saveFileToPhone(InputStream inputStream, String filename) {
    OutputStream outputStream;
    Context myContext = requireContext();
    try {
        if(Build.VERSION.SDK_INT >=Build.VERSION_CODES.Q){
            ContentResolver contentResolver = requireContext().getContentResolver();
            ContentValues contentValues = new ContentValues();
            contentValues.put(MediaStore.Downloads.DISPLAY_NAME,filename);
            contentValues.put(MediaStore.MediaColumns.RELATIVE_PATH, Environment.DIRECTORY_DOWNLOADS);
            Uri collection = MediaStore.Downloads.getContentUri(MediaStore.VOLUME_EXTERNAL_PRIMARY);
            Uri fileUri = contentResolver.insert(collection, contentValues);
            outputStream = contentResolver.openOutputStream(Objects.requireNonNull(fileUri));
            Objects.requireNonNull(outputStream);

        }
    }catch (FileNotFoundException e) {

        e.printStackTrace();
    }
}

Here it is what i have done with clean way

This is file provider activity

class FileProviderActivity : AppCompatActivity() {

    var commonIntentLauncher: ActivityResultLauncher<Intent?> = registerForActivityResult(
        ActivityResultContracts.StartActivityForResult()) { result ->
        if (result.resultCode == Activity.RESULT_OK) {
            result.data?.let {
                intent.getParcelableExtra<ResultReceiver>("FileReceiver")?.send(0, bundleOf(
                    "FileUri" to result?.data?.data.toString()
                ))
                finish()
            }
        }
    }


    override fun onCreate(savedInstanceState: Bundle?) {
        super.onCreate(savedInstanceState)
        if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.Q){
            val activityIntent = Intent(Intent.ACTION_CREATE_DOCUMENT).apply {
                addCategory(Intent.CATEGORY_OPENABLE)
                type = intent.getStringExtra("fileType")
                putExtra(Intent.EXTRA_TITLE, intent.getStringExtra("fileName"))
                putExtra(DocumentsContract.EXTRA_INITIAL_URI, MediaStore.Downloads.EXTERNAL_CONTENT_URI)
            }
            commonIntentLauncher.launch(activityIntent)
        }else{
            val directory: File = Environment.getExternalStoragePublicDirectory(Environment.DIRECTORY_DOWNLOADS + "/Dunny/CSVFiles")
            if (!directory.exists()){
                directory.mkdirs()
            }

            intent.getParcelableExtra<ResultReceiver>("FileReceiver")?.send(0, bundleOf(
                "FileUri" to File(directory, intent.getStringExtra("fileName")!!).toURI().toString()
            ))
            finish()
        }
    }
}

This FileProviderHelper

class MyFileProvider {

    companion object {
        fun with(context: Context) = FileRequest(context)
    }

    class FileRequest(private val context: Context) {

        fun request(fileName: String, fileType: String = "application/*", file: (Uri?) -> Unit ) {
            val intent = Intent(context, FileProviderActivity::class.java)
                .addFlags(Intent.FLAG_ACTIVITY_SINGLE_TOP)
                .addFlags(Intent.FLAG_ACTIVITY_NEW_TASK)
                .putExtra("fileName",fileName)
                .putExtra("fileType", fileType)
                .putExtras(bundleOf("FileReceiver" to FileReceiver(file)))
            context.startActivity(intent)
        }
    }

    internal class FileReceiver(private val file: (Uri?) -> Unit) : ResultReceiver(Handler(Looper.getMainLooper())) {
        override fun onReceiveResult(resultCode: Int, resultData: Bundle?) {
            super.onReceiveResult(resultCode, resultData)
            resultData?.let {
                file(it.getString("FileUri")?.toUri())
            }
        }
    }
}

Here Is Use Of this Function

MyFileProvider.with(this).request("TestFile.csv","application/*") { fileUri ->
    toast(fileUri.toString())
}

The technical post webpages of this site follow the CC BY-SA 4.0 protocol. If you need to reprint, please indicate the site URL or the original address.Any question please contact:yoyou2525@163.com.

 
粤ICP备18138465号  © 2020-2024 STACKOOM.COM