I cannot access files in protected directory flutter even after permission - android

I have used ACTION_OPEN_DOCUMENT_TREE to let the user choose a folder [So to avoid using MANAGE_EXTERNAL_STORAGE] but couldn't view i.e. Read the files inside the hidden
Already tried all the file piker libraries
I cannot access files in this directory even after giving permission from Document Tree to a produced directory in android studio which is below:
'/storage/emulated/0/Android/media/com.whatsapp/WhatsApp/Media/.Statuses'
below is the result I get when I list the files in the above directory but am unable to read the files using Image.file() in flutter URI permission also granted which are as follows
intent.addFlags(Intent.FLAG_GRANT_WRITE_URI_PERMISSION)
intent.addFlags(Intent.FLAG_GRANT_READ_URI_PERMISSION)
intent.addFlags(Intent.FLAG_GRANT_PREFIX_URI_PERMISSION)
intent.addFlags(Intent.FLAG_GRANT_PERSISTABLE_URI_PERMISSION)
below is one of the results I get when I list the files, its accessible in native android code but not in flutter
'content://com.android.externalstorage.documents/tree/primary%3AAndroid%2Fmedia%2Fcom.whatsapp%2FWhatsApp%2FMedia%2F.Statuses/document/primary%3AAndroid%2Fmedia%2Fcom.whatsapp%2FWhatsApp%2FMedia%2F.Statuses%2Fa9e8797092cd4bfeb6e426dfa1be7a84.jpg'
following code is used to get permission from files using native android code in mainactivity file
private fun grantAndroid11StorageAccessPermission() {
println("mydhfhsdkfsd 00")
if (isMyPackedgeInstalled("com.whatsapp")) {
val intent: Intent
val storageManager: StorageManager =
getSystemService(Context.STORAGE_SERVICE) as StorageManager
val whatsappfolderdir: String = if (File(
Environment.getExternalStorageDirectory()
.toString() + "/Android/media/com.whatsapp/WhatsApp/Media/.Statuses"
).isDirectory()
) {
"Android%2Fmedia%2Fcom.whatsapp%2FWhatsApp%2FMedia%2F.Statuses"
} else {
"WhatsApp%2FMedia%2F.Statuses"
}
if (Build.VERSION.SDK_INT >= 29) {
intent = storageManager.primaryStorageVolume.createOpenDocumentTreeIntent()
val scheme: String =
(intent.getParcelableExtra<android.os.Parcelable>("android.provider.extra.INITIAL_URI") as Uri)
.toString()
.replace("/root/", "/document/")
val stringBuilder = "$scheme%3A$whatsappfolderdir"
intent.putExtra("android.provider.extra.INITIAL_URI", Uri.parse(stringBuilder))
println("mydhfhsdkfsd $stringBuilder")
} else {
intent = Intent("android.intent.action.OPEN_DOCUMENT_TREE")
intent.putExtra("android.provider.extra.INITIAL_URI", Uri.parse(whatsappfolderdir))
}
intent.addFlags(Intent.FLAG_GRANT_WRITE_URI_PERMISSION)
intent.addFlags(Intent.FLAG_GRANT_READ_URI_PERMISSION)
intent.addFlags(Intent.FLAG_GRANT_PREFIX_URI_PERMISSION)
intent.addFlags(Intent.FLAG_GRANT_PERSISTABLE_URI_PERMISSION)
startActivityForResult(
intent,
OPEN_DOCOMENT_TREE_REQUEST_CODE
)
return
} else {
Toast.makeText(
this#MainActivity,
"Error getting ",
Toast.LENGTH_SHORT
).show()
}
}
following function is used to retrieve the files list
private fun getDataFromWhatsAppFolder(): Array<DocumentFile>? {
try {
val fromTreeUri: DocumentFile? = DocumentFile.fromTreeUri(
applicationContext,
Uri.parse(namedataprefs)
)
if (fromTreeUri!!.exists() && fromTreeUri.isDirectory() && fromTreeUri.canRead() && fromTreeUri.canWrite()) {
return fromTreeUri.listFiles()
}
} catch (e: Exception) {
e.printStackTrace()
return null
}
return null
}
if complete Mainactivity file is required I will add it below also
MainActivity
package com.example.flutterapp
import android.R.attr.path
import android.annotation.SuppressLint
import android.content.Context
import android.content.Intent
import android.content.SharedPreferences
import android.content.pm.PackageManager
import android.net.Uri
import android.os.Build
import android.os.Bundle
import android.os.Environment
import android.os.storage.StorageManager
import android.util.Log
import android.widget.Toast
import androidx.annotation.Nullable
import androidx.documentfile.provider.DocumentFile
import io.flutter.embedding.android.FlutterActivity
import io.flutter.plugin.common.MethodChannel
import io.flutter.plugins.GeneratedPluginRegistrant
import java.io.File
class MainActivity : FlutterActivity() {
private val CHANNEL = "apilevel"
private val MY_PERMISSIONS_REQUEST_WRITE_STORAGE = 1235
private val OPEN_DOCOMENT_TREE_REQUEST_CODE = 1014
var filesList: ArrayList<String> = ArrayList<String>()
private var namedataprefs: String? = null
private var callResult: MethodChannel.Result? = null
override fun onCreate(savedInstanceState: Bundle?) {
super.onCreate(savedInstanceState)
GeneratedPluginRegistrant.registerWith(flutterEngine!!)
val prefs: SharedPreferences =
getSharedPreferences("whatsapp_pref", Context.MODE_PRIVATE)
namedataprefs = prefs.getString("whatsapp", "")
MethodChannel(
flutterEngine!!.dartExecutor.binaryMessenger,
CHANNEL
).setMethodCallHandler { call, result ->
callResult = result
if (call.method == "getapilevel") {
Log.d("fffffffffff", (Build.VERSION.SDK_INT).toString())
result.success((Build.VERSION.SDK_INT).toString())
} else if (call.method == "whatsapplist") {
var mylis: List<String> = ArrayList<String>()
if (getDataFromWhatsAppFolder() != null) {
Log.d("fffffffffff", "folder done")
mylis = getData()
Log.d("fffffffffff", "folder datalengthc " + File("/storage/emulated/0/Android/media/com.whatsapp/WhatsApp/Media/.Statuses").canRead())
val directory: File = File("/storage/emulated/0/Android/media/com.whatsapp/WhatsApp/Media/.Statuses")
val files = directory.listFiles()
Log.d("Files", "Size: " + files.size)
for (i in files.indices) {
Log.d("Files", "FileName:" + files[i].name)
}
}
result.success(mylis)
} else if (call.method == "whatsapp") {
Log.d("fffffffffff", "grantwhatsapp")
if (getDataFromWhatsAppFolder() == null) {
grantAndroid11StorageAccessPermission()
}
} else {
result.notImplemented()
}
}
}
private fun isMyPackedgeInstalled(packageName: String?): Boolean {
return try {
packageManager.getPackageInfo(packageName!!, 0)
true
} catch (e: PackageManager.NameNotFoundException) {
false
}
}
private fun grantAndroid11StorageAccessPermission() {
println("mydhfhsdkfsd 00")
if (isMyPackedgeInstalled("com.whatsapp")) {
val intent: Intent
val storageManager: StorageManager =
getSystemService(Context.STORAGE_SERVICE) as StorageManager
val whatsappfolderdir: String = if (File(
Environment.getExternalStorageDirectory()
.toString() + "/Android/media/com.whatsapp/WhatsApp/Media/.Statuses"
).isDirectory()
) {
"Android%2Fmedia%2Fcom.whatsapp%2FWhatsApp%2FMedia%2F.Statuses"
} else {
"WhatsApp%2FMedia%2F.Statuses"
}
if (Build.VERSION.SDK_INT >= 29) {
intent = storageManager.primaryStorageVolume.createOpenDocumentTreeIntent()
val scheme: String =
(intent.getParcelableExtra<android.os.Parcelable>("android.provider.extra.INITIAL_URI") as Uri)
.toString()
.replace("/root/", "/document/")
val stringBuilder = "$scheme%3A$whatsappfolderdir"
intent.putExtra("android.provider.extra.INITIAL_URI", Uri.parse(stringBuilder))
println("mydhfhsdkfsd $stringBuilder")
} else {
intent = Intent("android.intent.action.OPEN_DOCUMENT_TREE")
intent.putExtra("android.provider.extra.INITIAL_URI", Uri.parse(whatsappfolderdir))
}
intent.addFlags(Intent.FLAG_GRANT_WRITE_URI_PERMISSION)
intent.addFlags(Intent.FLAG_GRANT_READ_URI_PERMISSION)
intent.addFlags(Intent.FLAG_GRANT_PREFIX_URI_PERMISSION)
intent.addFlags(Intent.FLAG_GRANT_PERSISTABLE_URI_PERMISSION)
startActivityForResult(
intent,
OPEN_DOCOMENT_TREE_REQUEST_CODE
)
return
} else {
Toast.makeText(
this#MainActivity,
"Error getting ",
Toast.LENGTH_SHORT
).show()
}
}
private fun getDataFromWhatsAppFolder(): Array<DocumentFile>? {
try {
val fromTreeUri: DocumentFile? = DocumentFile.fromTreeUri(
applicationContext,
Uri.parse(namedataprefs)
)
if (fromTreeUri!!.exists() && fromTreeUri.isDirectory() && fromTreeUri.canRead() && fromTreeUri.canWrite()) {
return fromTreeUri.listFiles()
}
} catch (e: Exception) {
e.printStackTrace()
return null
}
return null
}
private fun getData(): List<String> {
if (Build.VERSION.SDK_INT >= 30) {
if (filesList != null) {
filesList = ArrayList<String>()
}
try {
val allFileswhatsapp: Array<DocumentFile>? = getDataFromWhatsAppFolder()
///val allFileswhat: MutableList<String> = mutableListOf(Array<DocumentFile>)
val allFileswhat: Array<DocumentFile> = allFileswhatsapp!!
// allFileswhat.toMutableList().addAll(allFileswhatsapp)
// allFileswhat.addAll(allFileswhatsapp)
for (allFile in allFileswhat) {
// val file: Uri = allFile.getUri()
val file: Uri = allFile.getUri()
if (!allFile.getUri().toString().contains(".nomedia") && !allFile.getUri()
.toString().equals("")
) {
val fromTreeUri1 = DocumentFile.fromSingleUri(
this#MainActivity,
Uri.parse(allFile.getUri().toString())
)
if (fromTreeUri1 != null) {
// moveFile(fromTreeUri1.uri.toString());
filesList.add(fromTreeUri1.uri.toString())
Log.d("fffffffffff added list ", fromTreeUri1.uri.toString())
Log.d("fffffffffff added segm ", fromTreeUri1.uri.lastPathSegment.toString())
}
}
}
filesList.add("/storage/emulated/0/Download/wester.jpg")
} catch (e: Exception) {
e.printStackTrace()
}
}
return filesList
}
#SuppressLint("WrongConstant")
override fun onActivityResult(requestCode: Int, resultCode: Int, #Nullable data: Intent?) {
super.onActivityResult(requestCode, resultCode, data)
if (requestCode == OPEN_DOCOMENT_TREE_REQUEST_CODE && resultCode == -1) {
val uri: Uri? = data!!.data
try {
contentResolver.takePersistableUriPermission(
uri!!,
Intent.FLAG_GRANT_READ_URI_PERMISSION
)
contentResolver.takePersistableUriPermission(
uri!!,
Intent.FLAG_GRANT_PERSISTABLE_URI_PERMISSION
)
contentResolver.takePersistableUriPermission(
uri,
Intent.FLAG_GRANT_WRITE_URI_PERMISSION
)
} catch (e: Exception) {
e.printStackTrace()
}
namedataprefs = uri!!.toString()
val editor: SharedPreferences.Editor =
getSharedPreferences("whatsapp_pref", Context.MODE_PRIVATE)
.edit()
editor.putString("whatsapp", uri.toString())
editor.apply()
Log.d("fffffffffff", "my string location is " + uri.toString())
var mylis: List<String> = ArrayList<String>()
if (getDataFromWhatsAppFolder() != null) {
Log.d("fffffffffff", "folder done")
mylis = getData()
Log.d("fffffffffff", "folder datalength " + mylis.size)
}
callResult!!.success(mylis)
}
}
fun moveFile(sourceFile: String?): Boolean {
val despath =
"/storage/emulated/0/Download/tempwhatsapp/" + File(
sourceFile
).name
val destUri = Uri.fromFile(File(despath))
try {
val `is` = contentResolver.openInputStream(Uri.parse(sourceFile))
val os = contentResolver.openOutputStream(destUri, "w")
val buffer = ByteArray(1024)
while (true) {
val read = `is`!!.read(buffer)
if (read > 0) {
os!!.write(buffer, 0, read)
} else {
`is`.close()
os!!.flush()
os.close()
val intent = Intent("android.intent.action.MEDIA_SCANNER_SCAN_FILE")
intent.data = destUri
sendBroadcast(intent)
return true
}
}
} catch (e: java.lang.Exception) {
e.printStackTrace()
return false
}
}
}

Related

Installing an APK Programmatically (Android 12)

I'm attempting to install an APK programmatically on Android 12 but seem to be running into unknown issues at this point. All advice I've found regarding installing an APK programmatically seem to be deprecated.
Currently, I'm able to save my file but whenever I attempt to install it using PackageManager.PackageInstaller, it fails silently and I'm unable to find anything in the logs suggesting what the failure might've been.
Here's my package installer object.
object PackageInstaller {
#SuppressLint("WrongConstant")
#Throws(IOException::class)
fun installPackage(
context: Context,
installSessionId: String?,
packageName: String?,
apkStream: InputStream?
) {
val packageManger = context.packageManager
val packageInstaller = packageManger.packageInstaller
val params = android.content.pm.PackageInstaller.SessionParams(
android.content.pm.PackageInstaller.SessionParams.MODE_FULL_INSTALL
)
params.setAppPackageName(packageName)
var session: android.content.pm.PackageInstaller.Session? = null
try {
val sessionId = packageInstaller.createSession(params)
session = packageInstaller.openSession(sessionId)
val out = session.openWrite(installSessionId!!, 0, -1)
val buffer = ByteArray(1024)
var length: Int
var count = 0
if (apkStream != null) {
while (apkStream.read(buffer).also { length = it } != -1) {
out.write(buffer, 0, length)
count += length
}
}
session.fsync(out)
out.close()
val intent = Intent
intent.addFlags(Intent.ACTION_PACKAGE_ADDED)
Log.v("installer", "Installing..?")
session.commit(
PendingIntent.getBroadcast(
context, sessionId,
intent, if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.M) {
PendingIntent.FLAG_IMMUTABLE or PendingIntent.FLAG_UPDATE_CURRENT
} else {
PendingIntent.FLAG_UPDATE_CURRENT
}
).intentSender
)
}finally {
session?.close()
}
}
}
At this point I'm pretty lost as to where to look next. Does anyone even know if this is still possible? Or a solution to this issue?
You can try with this it is working with android 12
class DownloadApk(private var context: WeakReference<Context>) {
#JvmOverloads
fun startDownloadingApk(url: String, fileName: String = "App Update") {
if (URLUtil.isValidUrl(url)) {
DownloadNewVersion(context, url, fileName).execute()
}
}
#Suppress("DEPRECATION")
private class DownloadNewVersion(
private val context: WeakReference<Context>,
val downloadUrl: String,
val fileName: String
) : AsyncTask<String, Int, Boolean>() {
private lateinit var bar: ProgressDialog
override fun onPreExecute() {
super.onPreExecute()
bar = ProgressDialog(context.get()).apply {
setCancelable(false)
setMessage("Downloading...")
isIndeterminate = true
setCanceledOnTouchOutside(false)
show()
}
}
override fun onProgressUpdate(vararg values: Int?) {
super.onProgressUpdate(*values)
var msg = ""
val progress = values[0]
if (progress != null) {
bar.progress = progress
msg = if (progress > 99) "Finishing... " else "Downloading... $progress%"
}
bar.apply {
isIndeterminate = false
max = 100
setMessage(msg)
}
}
override fun onPostExecute(result: Boolean?) {
super.onPostExecute(result)
bar.dismiss()
if (result != null && result) {
context.get()?.let {
Toast.makeText(it, "Update Done", Toast.LENGTH_SHORT).show()
}
} else {
context.get()?.let {
Toast.makeText(it, "Error: Try Again", Toast.LENGTH_SHORT).show()
}
}
}
override fun doInBackground(vararg p0: String?): Boolean {
var flag = false
try {
val path =
Environment.getExternalStoragePublicDirectory(Environment.DIRECTORY_DOWNLOADS)
.toString() + "/"
var outputFile = File("$path$fileName.apk")
var repetition = 1
while (outputFile.exists()) {
outputFile = File("$path$fileName ($repetition).apk")
repetition++
}
val directory = File(path)
if (!directory.exists()) {
directory.mkdirs()
}
val url = URL(downloadUrl)
val c = url.openConnection() as HttpURLConnection
c.requestMethod = "GET"
c.connect()
val fos = FileOutputStream(outputFile)
val inputStream = c.inputStream
val totalSize = c.contentLength.toFloat() //size of apk
val buffer = ByteArray(1024)
var len1: Int
var per: Float
var downloaded = 0f
while (inputStream.read(buffer).also { len1 = it } != -1) {
fos.write(buffer, 0, len1)
downloaded += len1
per = (downloaded * 100 / totalSize)
publishProgress(per.toInt())
}
fos.close()
inputStream.close()
openNewVersion(outputFile.path)
flag = true
} catch (e: MalformedURLException) {
Log.e("DownloadApk", "Update Error: " + e.message)
flag = false
} catch (e: IOException) {
e.printStackTrace()
}
return flag
}
private fun openNewVersion(location: String) {
val intent = Intent(Intent.ACTION_VIEW)
intent.setDataAndType(
getUriFromFile(location),
"application/vnd.android.package-archive"
)
intent.addFlags(Intent.FLAG_ACTIVITY_NEW_TASK)
intent.addFlags(Intent.FLAG_GRANT_READ_URI_PERMISSION)
context.get()?.startActivity(intent)
}
private fun getUriFromFile(filePath: String): Uri? {
return if (Build.VERSION.SDK_INT < Build.VERSION_CODES.N) {
Uri.fromFile(File(filePath))
} else {
context.get()?.let {
FileProvider.getUriForFile(
it,
it.packageName + ".provider",
File(filePath)
)
}
}
}
}
}

Type mismatch: inferred type is Boolean? but Boolean was expected

Please assist,
I have upgraded my flutter to 3.0.1 • channel stable ,
dart sdk version to Dart SDK version: 2.17.1 (stable) (Tue May 17 17:58:21 2022 +0000) on "windows_x64".
kotlin version ext.kotlin_version = '1.6.20' in /android/build.gradle file.
When I try to start my app I get the following error
C:\Users\[MY_USER]\AppData\Local\Pub\Cache\hosted\pub.dartlang.org\photo_manager-
1.3.10\android\src\main\kotlin\top\kikt\imagescanner\core\PhotoManagerPlugin.kt: (128, 36): Type
mismatch: inferred type is Boolean? but Boolean was expected
e: C:\Users\[MY_USER]\AppData\Local\Pub\Cache\hosted\pub.dartlang.org\photo_manager-
1.3.10\android\src\main\kotlin\top\kikt\imagescanner\core\PhotoManagerPlugin.kt: (128, 41): Type
mismatch: inferred type is Boolean? but Boolean was expected
2
FAILURE: Build failed with an exception.
* What went wrong:
Execution failed for task ':photo_manager:compileDebugKotlin'.
> A failure occurred while executing org.jetbrains.kotlin.compilerRunner.GradleCompilerRunnerWithWorkers$GradleKotlinCompilerWorkAction
> Compilation error. See log for more details
* Try:
> Run with --stacktrace option to get the stack trace.
> Run with --info or --debug option to get more log output.
> Run with --scan to get full insights.
* Get more help at https://help.gradle.org
BUILD FAILED in 6m 26s
Exception: Gradle task assembleDebug failed with exit code 1
Exited (sigterm)
I'm not sure what else is missing. I did not edit any kotlin code.
Kotlin MainActivity.kt
package com.company.app
import io.flutter.embedding.android.FlutterActivity
class MainActivity: FlutterActivity() {
}
PhotoManagerPlugin.kt code
package top.kikt.imagescanner.core
import android.Manifest
import android.app.Activity
import android.content.Context
import android.content.pm.PackageManager
import android.os.Build
import android.os.Handler
import androidx.annotation.RequiresApi
import com.bumptech.glide.Glide
import io.flutter.plugin.common.BinaryMessenger
import io.flutter.plugin.common.MethodCall
import io.flutter.plugin.common.MethodChannel
import top.kikt.imagescanner.core.entity.AssetEntity
import top.kikt.imagescanner.core.entity.FilterOption
import top.kikt.imagescanner.core.entity.PermissionResult
import top.kikt.imagescanner.core.entity.ThumbLoadOption
import top.kikt.imagescanner.core.utils.ConvertUtils
import top.kikt.imagescanner.core.utils.IDBUtils
import top.kikt.imagescanner.core.utils.belowSdk
import top.kikt.imagescanner.permission.PermissionsListener
import top.kikt.imagescanner.permission.PermissionsUtils
import top.kikt.imagescanner.util.LogUtils
import top.kikt.imagescanner.util.ResultHandler
import java.util.concurrent.ArrayBlockingQueue
import java.util.concurrent.ThreadPoolExecutor
import java.util.concurrent.TimeUnit
/// create 2019-09-05 by cai
class PhotoManagerPlugin(
private val applicationContext: Context,
messenger: BinaryMessenger,
private var activity: Activity?,
private val permissionsUtils: PermissionsUtils
) : MethodChannel.MethodCallHandler {
val deleteManager = PhotoManagerDeleteManager(applicationContext, activity)
fun bindActivity(activity: Activity?) {
this.activity = activity
deleteManager.bindActivity(activity)
}
companion object {
private const val poolSize = 8
private val threadPool: ThreadPoolExecutor = ThreadPoolExecutor(
poolSize + 3,
1000,
200,
TimeUnit.MINUTES,
ArrayBlockingQueue(poolSize + 3)
)
fun runOnBackground(runnable: () -> Unit) {
threadPool.execute(runnable)
}
var cacheOriginBytes = true
}
private val notifyChannel = PhotoManagerNotifyChannel(applicationContext, messenger, Handler())
init {
permissionsUtils.permissionsListener = object : PermissionsListener {
override fun onDenied(
deniedPermissions: MutableList<String>,
grantedPermissions: MutableList<String>
) {
}
override fun onGranted() {
}
}
}
private val photoManager = PhotoManager(applicationContext)
private var ignorePermissionCheck = false
override fun onMethodCall(call: MethodCall, result: MethodChannel.Result) {
val resultHandler = ResultHandler(result, call)
if (call.method == "ignorePermissionCheck") {
val ignore = call.argument<Boolean>("ignore")!!
ignorePermissionCheck = ignore
resultHandler.reply(ignore)
return
}
var needLocationPermissions = false
val handleResult = when (call.method) {
"releaseMemCache" -> {
photoManager.clearCache()
resultHandler.reply(1)
true
}
"log" -> {
LogUtils.isLog = call.arguments()
resultHandler.reply(1)
true
}
"openSetting" -> {
permissionsUtils.getAppDetailSettingIntent(activity)
resultHandler.reply(1)
true
}
"clearFileCache" -> {
Glide.get(applicationContext).clearMemory()
runOnBackground {
photoManager.clearFileCache()
resultHandler.reply(1)
}
true
}
"forceOldApi" -> {
photoManager.useOldApi = true
resultHandler.reply(1)
true
}
"systemVersion" -> {
resultHandler.reply(Build.VERSION.SDK_INT.toString())
true
}
"cacheOriginBytes" -> {
cacheOriginBytes = call.arguments()
resultHandler.reply(cacheOriginBytes)
true
}
"getLatLngAndroidQ" -> {
/// 这里不拦截, 然后额外添加gps权限
needLocationPermissions = true
false
}
"copyAsset" -> {
needLocationPermissions = true
false
}
"getFullFile" -> {
val isOrigin = call.argument<Boolean>("isOrigin")!!
if (isOrigin && Build.VERSION.SDK_INT >= Build.VERSION_CODES.Q) {
needLocationPermissions = true
}
false
}
"getOriginBytes" -> {
needLocationPermissions = true
false
}
"getMediaUrl" -> {
false
}
else -> false
}
if (handleResult) {
return
}
if (ignorePermissionCheck) {
onHandlePermissionResult(call, resultHandler, true)
return
}
if (permissionsUtils.isRequesting) {
resultHandler.replyError(
"PERMISSION_REQUESTING",
"Another permission request is still ongoing. Please request after the existing one is done.",
null
)
return
}
val utils = permissionsUtils.apply {
withActivity(activity)
permissionsListener = object : PermissionsListener {
override fun onDenied(
deniedPermissions: MutableList<String>,
grantedPermissions: MutableList<String>
) {
LogUtils.info("onDenied call.method = ${call.method}")
if (call.method == "requestPermissionExtend") {
resultHandler.reply(PermissionResult.Denied.value)
return
}
if (grantedPermissions.containsAll(
arrayListOf(
Manifest.permission.READ_EXTERNAL_STORAGE,
Manifest.permission.WRITE_EXTERNAL_STORAGE
)
)
) {
LogUtils.info("onGranted call.method = ${call.method}")
onHandlePermissionResult(call, resultHandler, false)
} else {
replyPermissionError(resultHandler)
}
}
override fun onGranted() {
LogUtils.info("onGranted call.method = ${call.method}")
onHandlePermissionResult(call, resultHandler, true)
}
}
}
val permissions = arrayListOf(
Manifest.permission.READ_EXTERNAL_STORAGE,
Manifest.permission.WRITE_EXTERNAL_STORAGE
)
if (needLocationPermissions && Build.VERSION.SDK_INT >= Build.VERSION_CODES.Q && haveManifestMediaLocation(
applicationContext
)
) {
permissions.add(Manifest.permission.ACCESS_MEDIA_LOCATION)
}
utils.getPermissions(3001, permissions)
}
#RequiresApi(Build.VERSION_CODES.Q)
private fun haveManifestMediaLocation(context: Context): Boolean {
// Debug.waitForDebugger()
val applicationInfo = context.applicationInfo
val packageInfo = context.packageManager.getPackageInfo(
applicationInfo.packageName,
PackageManager.GET_PERMISSIONS
)
return packageInfo.requestedPermissions.contains(Manifest.permission.ACCESS_MEDIA_LOCATION)
}
private fun replyPermissionError(resultHandler: ResultHandler) {
resultHandler.replyError("Request for permission failed.", "User denied permission.", null)
}
private fun onHandlePermissionResult(
call: MethodCall,
resultHandler: ResultHandler,
haveLocationPermission: Boolean
) {
when (call.method) {
"requestPermissionExtend" -> resultHandler.reply(PermissionResult.Authorized.value)
"getGalleryList" -> {
if (Build.VERSION.SDK_INT >= 29) {
notifyChannel.setAndroidQExperimental(true)
}
runOnBackground {
val type = call.argument<Int>("type")!!
val hasAll = call.argument<Boolean>("hasAll")!!
val option = call.getOption()
val onlyAll = call.argument<Boolean>("onlyAll")!!
val list = photoManager.getGalleryList(type, hasAll, onlyAll, option)
resultHandler.reply(ConvertUtils.convertToGalleryResult(list))
}
}
"getAssetWithGalleryId" -> {
runOnBackground {
val id = call.argument<String>("id")!!
val page = call.argument<Int>("page")!!
val pageCount = call.argument<Int>("pageCount")!!
val type = call.argument<Int>("type")!!
val option = call.getOption()
val list = photoManager.getAssetList(id, page, pageCount, type, option)
resultHandler.reply(ConvertUtils.convertToAssetResult(list))
}
}
"getAssetListWithRange" -> {
runOnBackground {
val galleryId = call.getString("galleryId")
val type = call.getInt("type")
val start = call.getInt("start")
val end = call.getInt("end")
val option = call.getOption()
val list: List<AssetEntity> =
photoManager.getAssetListWithRange(galleryId, type, start, end, option)
resultHandler.reply(ConvertUtils.convertToAssetResult(list))
}
}
"getThumb" -> {
runOnBackground {
val id = call.argument<String>("id")!!
val optionMap = call.argument<Map<*, *>>("option")!!
val option = ThumbLoadOption.fromMap(optionMap)
photoManager.getThumb(id, option, resultHandler)
}
}
"requestCacheAssetsThumb" -> {
runOnBackground {
val ids = call.argument<List<String>>("ids")!!
val optionMap = call.argument<Map<*, *>>("option")!!
val option = ThumbLoadOption.fromMap(optionMap)
photoManager.requestCache(ids, option, resultHandler)
}
}
"cancelCacheRequests" -> {
runOnBackground {
photoManager.cancelCacheRequests()
}
}
"assetExists" -> {
runOnBackground {
val id = call.argument<String>("id")!!
photoManager.assetExists(id, resultHandler)
}
}
"getFullFile" -> {
runOnBackground {
val id = call.argument<String>("id")!!
val isOrigin =
if (!haveLocationPermission) false else call.argument<Boolean>("isOrigin")!!
photoManager.getFile(id, isOrigin, resultHandler)
}
}
"getOriginBytes" -> {
runOnBackground {
val id = call.argument<String>("id")!!
photoManager.getOriginBytes(
id,
cacheOriginBytes,
haveLocationPermission,
resultHandler
)
}
}
"getMediaUrl" -> {
runOnBackground {
val id = call.argument<String>("id")!!
val type = call.argument<Int>("type")!!
val mediaUri = photoManager.getMediaUri(id, type)
resultHandler.reply(mediaUri)
}
}
"getPropertiesFromAssetEntity" -> {
runOnBackground {
val id = call.argument<String>("id")!!
val asset = photoManager.getAssetProperties(id)
val assetResult = if (asset != null) {
ConvertUtils.convertToAssetResult(asset)
} else {
null
}
resultHandler.reply(assetResult)
}
}
"fetchPathProperties" -> {
runOnBackground {
val id = call.argument<String>("id")!!
val type = call.argument<Int>("type")!!
val option = call.getOption()
val pathEntity = photoManager.getPathEntity(id, type, option)
if (pathEntity != null) {
val mapResult = ConvertUtils.convertToGalleryResult(listOf(pathEntity))
resultHandler.reply(mapResult)
} else {
resultHandler.reply(null)
}
}
}
"getLatLngAndroidQ" -> {
runOnBackground {
val id = call.argument<String>("id")!!
// 读取id
val location = photoManager.getLocation(id)
resultHandler.reply(location)
}
}
"notify" -> {
runOnBackground {
val notify = call.argument<Boolean>("notify")
if (notify == true) {
notifyChannel.startNotify()
} else {
notifyChannel.stopNotify()
}
}
}
"deleteWithIds" -> {
runOnBackground {
try {
val ids = call.argument<List<String>>("ids")!!
if (belowSdk(29)) {
deleteManager.deleteInApi28(ids)
resultHandler.reply(ids)
} else if (IDBUtils.isAndroidR) {
val uris = ids.map {
photoManager.getUri(it)
}.toList()
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.R) {
deleteManager.deleteInApi30(uris, resultHandler)
}
} else {
val uris = ids.mapNotNull { photoManager.getUri(it) }
// for (id in ids) {
// val uri = photoManager.getUri(id)
// if (uri != null) {
// deleteManager.deleteWithUriInApi29(uri, false)
// }
// }
deleteManager.deleteWithUriInApi29(ids, uris, resultHandler, false)
}
} catch (e: Exception) {
LogUtils.error("deleteWithIds failed", e)
resultHandler.replyError("deleteWithIds failed")
}
}
}
"saveImage" -> {
runOnBackground {
try {
val image = call.argument<ByteArray>("image")!!
val title = call.argument<String>("title") ?: ""
val desc = call.argument<String>("desc") ?: ""
val relativePath = call.argument<String>("relativePath") ?: ""
val entity = photoManager.saveImage(image, title, desc, relativePath)
if (entity == null) {
resultHandler.reply(null)
return#runOnBackground
}
val map = ConvertUtils.convertToAssetResult(entity)
resultHandler.reply(map)
} catch (e: Exception) {
LogUtils.error("save image error", e)
resultHandler.reply(null)
}
}
}
"saveImageWithPath" -> {
runOnBackground {
try {
val imagePath = call.argument<String>("path")!!
val title = call.argument<String>("title") ?: ""
val desc = call.argument<String>("desc") ?: ""
val relativePath = call.argument<String>("relativePath") ?: ""
val entity = photoManager.saveImage(imagePath, title, desc, relativePath)
if (entity == null) {
resultHandler.reply(null)
return#runOnBackground
}
val map = ConvertUtils.convertToAssetResult(entity)
resultHandler.reply(map)
} catch (e: Exception) {
LogUtils.error("save image error", e)
resultHandler.reply(null)
}
}
}
"saveVideo" -> {
runOnBackground {
try {
val videoPath = call.argument<String>("path")!!
val title = call.argument<String>("title")!!
val desc = call.argument<String>("desc") ?: ""
val relativePath = call.argument<String>("relativePath") ?: ""
val entity = photoManager.saveVideo(videoPath, title, desc, relativePath)
if (entity == null) {
resultHandler.reply(null)
return#runOnBackground
}
val map = ConvertUtils.convertToAssetResult(entity)
resultHandler.reply(map)
} catch (e: Exception) {
LogUtils.error("save video error", e)
resultHandler.reply(null)
}
}
}
"copyAsset" -> {
runOnBackground {
val assetId = call.argument<String>("assetId")!!
val galleryId = call.argument<String>("galleryId")!!
photoManager.copyToGallery(assetId, galleryId, resultHandler)
}
}
"moveAssetToPath" -> {
runOnBackground {
val assetId = call.argument<String>("assetId")!!
val albumId = call.argument<String>("albumId")!!
photoManager.moveToGallery(assetId, albumId, resultHandler)
}
}
"removeNoExistsAssets" -> {
runOnBackground {
photoManager.removeAllExistsAssets(resultHandler)
}
}
else -> resultHandler.notImplemented()
}
}
private fun MethodCall.getString(key: String): String {
return this.argument<String>(key)!!
}
private fun MethodCall.getInt(key: String): Int {
return this.argument<Int>(key)!!
}
private fun MethodCall.getOption(): FilterOption {
val arguments = argument<Map<*, *>>("option")!!
return ConvertUtils.convertFilterOptionsFromMap(arguments)
}
}
The issue was due to the use of multi_crop_picker: ^0.0.5, commented it out and app compiled successfully.
I had the same exact problem.
Just check your dependencies versions and update the necessary ones.
In my case, my flutter SDK version was 3.3.6 , and I discovered that two of my dependencies versions were NOT compatible with the flutter version.
The dependencies were :
1- wechat_assets_picker 6.3.0
2-wechat_camera_picker 2.6

How to fetch files inside hidden folder using Media Store API on Android 11

I need to fetch data inside WhatsApp folders on External Storage.
As i am targeting API Level 30 i am no longer able to access WhatsApp folders on External Storage. I have implemented Storage Access Framework and got Android/media folder Uri and Document File. And using listFiles() i am able to list files but with filter() and sortedByDescending() functions it becomes very slow.
What i have tried?
Used Cursor loader with Projection and Selection Arguments but it only
worked for non hidden folders like WhatsApp Images and WhatsApp Videos
It returns empty cursor for hidden folder .Statuses
Tried replacing MediaStore.Video.Media.EXTERNAL_CONTENT_URI with MediaStore.Video.Media.getContentUri(MediaStore.VOLUME_EXTERNAL)
What is required?
List images and videos from .Statuses folder same as i am listing WhatsApp Images using Media Store in HomeActivity.java
Below is my code
In this activity i get permision to Android/media and set all WhatsApp folders URIs for status fetching and other use, but fetched WhatsApp Images with projection and selection from WhatsApp Images folder
class HomeActivity : AppCompatActivity(), InternetListener, PurchasesUpdatedListener,
CoroutineScope {
private val exceptionHandler = CoroutineExceptionHandler { context, exception ->
Toast.makeText(this, exception.message, Toast.LENGTH_LONG).show()
}
private val dataRepository: DataRepository by inject()
val tinyDB: TinyDB by inject()
val REQUEST_CODE = 12123
init {
newNativeAdSetUp = null
}
val sharedViewModel by viewModel<SharedViewModel>()
val viewModel by viewModel<HomeViewModel>()
val handler = CoroutineExceptionHandler { _, exception ->
Log.d("CoroutineException", "$exception handled !")
}
override val coroutineContext: CoroutineContext
get() = Dispatchers.Main + job + handler
private lateinit var job: Job
val sdk30PermissionListener = object : PermissionListener {
override fun onPermissionGranted() {
openDocumentTree()
}
override fun onPermissionDenied(deniedPermissions: MutableList<String>?) {
}
}
override fun onCreate(savedInstanceState: Bundle?) {
super.onCreate(savedInstanceState)
setContentView(R.layout.activity_home)
handlePermissionsByVersion()
}
private fun handlePermissionsByVersion() {
if (SDK_INT >= Build.VERSION_CODES.R) {
if ((ContextCompat.checkSelfPermission(
this,
Manifest.permission.WRITE_EXTERNAL_STORAGE
)
== PackageManager.PERMISSION_GRANTED) && (ContextCompat.checkSelfPermission(
this,
Manifest.permission.READ_EXTERNAL_STORAGE
)
== PackageManager.PERMISSION_GRANTED)
) {
//if granted load whatsapp images and some uris setup to viewmodel
loadWhatsAppImages()
if (arePermissionsGranted()) {
if (dataRepository.mrWhatsAppImages == null || dataRepository.mrWhatsAppBusinessImages == null) {
setUpWAURIs()
}
}
} else {
TedPermission.with(this)
.setPermissionListener(sdk30PermissionListener)
.setDeniedMessage("If you reject permission,you can not use this service\n\nPlease turn on permissions at [Setting] > [Permission]")
.setPermissions(
Manifest.permission.WRITE_EXTERNAL_STORAGE,
Manifest.permission.READ_EXTERNAL_STORAGE
)
.check()
}
}
}
override fun onActivityResult(requestCode: Int, resultCode: Int, #Nullable data: Intent?) {
super.onActivityResult(requestCode, resultCode, data)
if (resultCode == RESULT_OK && requestCode == REQUEST_CODE) {
if (data != null) {
//this is the uri user has provided us
val treeUri: Uri? = data.data
if (treeUri != null) {
sharedViewModel.treeUri = treeUri
val decoded = Uri.decode(treeUri.toString())
Log.i(LOGTAG, "got uri: ${treeUri.toString()}")
// here we should do some checks on the uri, we do not want root uri
// because it will not work on Android 11, or perhaps we have some specific
// folder name that we want, etc
if (Uri.decode(treeUri.toString()).endsWith(":")) {
showWrongFolderSelection()
return
}
if (!decoded.equals(Constants.WHATSAPP_MEDIA_URI_DECODED)) {
showWrongFolderSelection()
return
}
// here we ask the content resolver to persist the permission for us
val takeFlags: Int = Intent.FLAG_GRANT_READ_URI_PERMISSION or
Intent.FLAG_GRANT_WRITE_URI_PERMISSION
contentResolver.takePersistableUriPermission(
treeUri,
takeFlags
)
val treeUriAsString = treeUri.toString()
tinyDB.putString("FOLDER_URI", treeUriAsString)
if (SDK_INT >= Build.VERSION_CODES.R) {
setupPaths()
}
}
}
}
}
private fun setupPaths() {
setUpOverlay()
fetchWhatsAppRootURIs(
this,
sharedViewModel,
dataRepository,
tinyDB
) {
fetchWhatsAppBusinessRootURIs(
this,
sharedViewModel,
dataRepository,
tinyDB
) {
tinyDB.putBoolean("WARootPathsDone", true)
removeOverlay()
}
}
}
override fun onDestroy() {
dialogHandler.removeCallbacksAndMessages(null)
super.onDestroy()
}
val loadmanagerImages = object : LoaderManager.LoaderCallbacks<Cursor> {
val whatsAppImagesArrayList = arrayListOf<File>()
override fun onCreateLoader(id: Int, args: Bundle?): Loader<Cursor> {
var location: File = File(
Environment.getExternalStorageDirectory()
.toString() + Constants.whatsapp_images_path
)
if (!location.exists()) {
location = File(
Environment.getExternalStorageDirectory()
.toString() + Constants.whatsapp_images_path11
)
}
if (location != null && location.exists()) {
whatsAppImagesArrayList.clear()
Timber.e("checkLoaded-onCreateLoader $id")
if (id == 0) {
var folder = location.absolutePath
val projection = arrayOf(
MediaStore.MediaColumns.DATA,
MediaStore.MediaColumns.DATE_MODIFIED
)
val selection = MediaStore.Images.Media.DATA + " like ? "
val selectionArgs: String = "%$folder%"
return CursorLoader(
this#HomeActivity,
MediaStore.Images.Media.EXTERNAL_CONTENT_URI,
projection,
selection,
arrayOf(selectionArgs),
"${MediaStore.Images.Media.DATE_MODIFIED} DESC"
)
}
}
return null!!
}
override fun onLoadFinished(loader: Loader<Cursor>, cursor: Cursor?) {
Timber.e("checkLoaded-onLoadFinished")
var absolutePathOfImage: String
if (loader.id == 0) {
cursor?.let {
val columnIndexData = cursor.getColumnIndexOrThrow(MediaStore.MediaColumns.DATA)
GlobalScope.launch(Dispatchers.Main + exceptionHandler) {
async(Dispatchers.IO + exceptionHandler) {
while (!cursor.isClosed && cursor.moveToNext() == true) {
absolutePathOfImage = cursor.getString(columnIndexData!!)
whatsAppImagesArrayList.add(File(absolutePathOfImage))
}
}.await()
LoaderManager.getInstance(this#HomeActivity).destroyLoader(0)
Timber.e("checkLoaded-Completion")
galleryViewModel.whatsAppImagesList.postValue(whatsAppImagesArrayList)
}
}
}
}
override fun onLoaderReset(loader: Loader<Cursor>) {
}
}
fun loadWhatsAppImages() {
try {
tinyDB.putBoolean("whatsAppMediaLoadCalled", true)
LoaderManager.getInstance(this).initLoader(
0,
null,
loadmanagerImages
)
} catch (e: RuntimeException) {
Log.e("exVideos ", "ex : ${e.localizedMessage}")
}
}
companion object {
const val ANDROID_DOCID = "primary:Android/media/"
const val EXTERNAL_STORAGE_PROVIDER_AUTHORITY = "com.android.externalstorage.documents"
private val androidUri = DocumentsContract.buildDocumentUri(
EXTERNAL_STORAGE_PROVIDER_AUTHORITY, ANDROID_DOCID
)
val androidTreeUri = DocumentsContract.buildTreeDocumentUri(
EXTERNAL_STORAGE_PROVIDER_AUTHORITY, ANDROID_DOCID
)
}
private fun openDocumentTree() {
val uriString = tinyDB.getString("FOLDER_URI", "")
when {
uriString == "" -> {
Log.w(LOGTAG, "uri not stored")
askPermission()
}
arePermissionsGranted() -> {
}
else -> {
Log.w(LOGTAG, "uri permission not stored")
askPermission()
}
}
}
// this will present the user with folder browser to select a folder for our data
private fun askPermission() {
val intent = Intent(Intent.ACTION_OPEN_DOCUMENT_TREE)
intent.putExtra(DocumentsContract.EXTRA_INITIAL_URI, androidUri)
startActivityForResult(intent, REQUEST_CODE)
}
private fun arePermissionsGranted(): Boolean {
var uriString = tinyDB.getString("FOLDER_URI", "")
val list = contentResolver.persistedUriPermissions
for (i in list.indices) {
val persistedUriString = list[i].uri.toString()
if (persistedUriString == uriString && list[i].isWritePermission && list[i].isReadPermission) {
return true
}
}
return false
}
private fun showWrongFolderSelection() {
val layoutInflaterAndroid = LayoutInflater.from(this)
val mView = layoutInflaterAndroid.inflate(R.layout.layout_dialog_wrong_folder, null)
val builder = AlertDialog.Builder(this, R.style.ThemePageSearchDialog)
builder.setView(mView)
val alertDialog = builder.show()
alertDialog.setCancelable(false)
val btnOk = mView.findViewById(R.id.tvExit) as TextView
val tvCancel = mView.findViewById(R.id.tvCancel) as TextView
btnOk.setOnClickListener {
alertDialog.dismiss()
openDocumentTree()
}
tvCancel.setOnClickListener {
alertDialog.dismiss()
}
}
private fun setUpWAURIs() {
dataRepository.mrWhatsAppImages =
getDocumentFileFromStringURI(
this,
tinyDB.getString("mrWhatsAppImages")
)
dataRepository.mrWhatsAppVN =
getDocumentFileFromStringURI(
this,
tinyDB.getString("mrWhatsAppVN")
)
dataRepository.mrWhatsAppDocs =
getDocumentFileFromStringURI(
this,
tinyDB.getString("mrWhatsAppDocs")
)
dataRepository.mrWhatsAppVideo =
getDocumentFileFromStringURI(
this,
tinyDB.getString("mrWhatsAppVideo")
)
dataRepository.mrWhatsAppAudio =
getDocumentFileFromStringURI(
this,
tinyDB.getString("mrWhatsAppAudio")
)
dataRepository.WhatsAppStatuses =
getDocumentFileFromStringURIStatuses(
this,
tinyDB.getString("WhatsAppStatuses")
)
dataRepository.mrWhatsAppBusinessImages =
getDocumentFileFromStringURI(
this,
tinyDB.getString("mrWhatsAppBusinessImages")
)
dataRepository.mrWhatsAppBusinessVN =
getDocumentFileFromStringURI(
this,
tinyDB.getString("mrWhatsAppBusinessVN")
)
dataRepository.mrWhatsAppBusinessDocs =
getDocumentFileFromStringURI(
this,
tinyDB.getString("mrWhatsAppBusinessDocs")
)
dataRepository.mrWhatsAppBusinessVideo =
getDocumentFileFromStringURI(
this,
tinyDB.getString("mrWhatsAppBusinessVideo")
)
dataRepository.mrWhatsAppBusinessAudio =
getDocumentFileFromStringURI(
this,
tinyDB.getString("mrWhatsAppBusinessAudio")
)
dataRepository.WhatsAppBusinessStatuses =
getDocumentFileFromStringURIStatuses(
this,
tinyDB.getString("WhatsAppBusinessStatuses")
)
}
fun setUpOverlay() {
val dialogfragment = FullScreenLoadingDialog()
dialogfragment.isCancelable = false
dialogfragment.setisAdmobAd(true)
val ft: FragmentTransaction =
supportFragmentManager.beginTransaction()
ft.add(dialogfragment, "DialogFragment_FLAG")
ft.commitAllowingStateLoss()
}
fun removeOverlay() {
val fragment: Fragment? = supportFragmentManager.findFragmentByTag("DialogFragment_FLAG")
if (fragment != null && fragment is DialogFragment) {
fragment.dismissAllowingStateLoss()
}
}
fun fetchWhatsAppRootURIs(
context: Context,
sharedViewModel: SharedViewModel,
dataRepository: DataRepository,
tinyDB: TinyDB, completed: () -> Unit
) {
val selectedPackageName = Constants.WHATSAPP_PKG_NAME
val selectedRootName = Constants.WHATSAPP_ROOT_NAME
var waImages: DocumentFile? = null
var waVN: DocumentFile? = null
var waDocs: DocumentFile? = null
var waVideos: DocumentFile? = null
var waAudio: DocumentFile? = null
var waStatus: DocumentFile? = null
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.R && sharedViewModel.treeUri != null) {
CoroutineScope(Dispatchers.Main).launch {
async(Dispatchers.IO) {
val dir = DocumentFile.fromTreeUri(
context,
sharedViewModel.treeUri!!
)
dir?.listFiles()?.forEach {
if (it.name.equals(selectedPackageName)) {
it.listFiles().forEach {
if (it.name.equals(selectedRootName)) {
it.listFiles().forEach {
if (it.name.equals(Constants.WHATSAPP_MEDIA_FOLDER_NAME)) {
it.listFiles().forEach {
if (it.name.equals(Constants.FOLDER_NAME_WHATSAPP_IMAGES)) {
waImages = it
} else if (it.name.equals(Constants.FOLDER_NAME_WHATSAPP_VN)) {
waVN = it
} else if (it.name.equals(Constants.FOLDER_NAME_WHATSAPP_DOCUMENTS)) {
waDocs = it
} else if (it.name.equals(Constants.FOLDER_NAME_WHATSAPP_VIDEO)) {
waVideos = it
} else if (it.name.equals(Constants.FOLDER_NAME_WHATSAPP_AUDIO)) {
waAudio = it
} else if (it.name.equals(Constants.FOLDER_NAME_STATUSES)) {
waStatus = it
}
}
}
}
}
}
}
}
}.await()
Timber.e("processStatusFetch:Done")
tinyDB.putString("mrWhatsAppImages", waImages?.uri.toString())
tinyDB.putString("mrWhatsAppVN", waImages?.uri.toString())
tinyDB.putString("mrWhatsAppDocs", waImages?.uri.toString())
tinyDB.putString("mrWhatsAppVideo", waImages?.uri.toString())
tinyDB.putString("mrWhatsAppAudio", waImages?.uri.toString())
tinyDB.putString("WhatsAppStatuses", waStatus?.uri.toString())
dataRepository.mrWhatsAppImages = waImages
dataRepository.mrWhatsAppVN = waVN
dataRepository.mrWhatsAppDocs = waDocs
dataRepository.mrWhatsAppVideo = waVideos
dataRepository.mrWhatsAppAudio = waAudio
dataRepository.WhatsAppStatuses = waStatus
completed()
}
}
}
Here i am using .Statuses folder URI to list DocumentFiles and display but this way it is slow
class StatusImageFragment : Fragment(), StatusListener, CoroutineScope {
companion object {
fun newInstance() = StatusImageFragment()
}
val handler = CoroutineExceptionHandler { _, exception ->
Log.d("CoroutineException", "$exception handled !")
}
override val coroutineContext: CoroutineContext
get() = Dispatchers.Main + job + handler
private lateinit var job: Job
private var adapterSDK30 = StatusImageAdapterSDK30()
private var no_image: ImageView? = null
private var no_image_txt: TextView? = null
val tinyDB: TinyDB by inject()
val sharedViewModel by viewModel<SharedViewModel>()
private val dataRepository: DataRepository by inject()
override fun onCreateView(
inflater: LayoutInflater, container: ViewGroup?,
savedInstanceState: Bundle?
): View? {
job = Job()
return inflater.inflate(R.layout.status_image_fragment, container, false)
}
override fun onViewCreated(view: View, savedInstanceState: Bundle?) {
super.onViewCreated(view, savedInstanceState)
swipeRefresh(false, false)
}
public fun swipeRefresh(isReloadRequired: Boolean, isFromModeChanged: Boolean) {
try {
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.R) {
if (isFromModeChanged) {
status_image_recycler.visibility = View.GONE
progressbar.visibility = View.VISIBLE
no_image?.let {
it.visibility = View.GONE
}
no_image_txt?.let {
it.visibility = View.GONE
}
go_to_app?.let {
it.visibility = View.GONE
}
} else {
if (adapterSDK30.listImages == null || adapterSDK30.listImages.size == 0) {
no_image?.let {
it.visibility = View.GONE
}
no_image_txt?.let {
it.visibility = View.GONE
}
go_to_app?.let {
it.visibility = View.GONE
}
progressbar.visibility = View.VISIBLE
}
}
if (isReloadRequired) {
processStatusFetchFromChild({
sharedViewModel.statusImages.observe(viewLifecycleOwner, Observer {
val arrayList = it
adapterSDK30.listImages = arrayList
postFetchingExecutionSDK30()
})
})
} else {
sharedViewModel.statusImages.observe(viewLifecycleOwner, Observer {
val arrayList = it
adapterSDK30.listImages = arrayList
adapterSDK30.listImages = it
postFetchingExecutionSDK30()
})
}
}
} catch (ex: Exception) {
ex.printStackTrace()
}
}
private fun postFetchingExecutionSDK30() {
progressbar.visibility = View.GONE
status_image_recycler.visibility = View.VISIBLE
if (adapterSDK30!!.listImages != null && adapterSDK30!!.listImages.size > 0) {
no_image?.let {
it.visibility = View.GONE
}
no_image_txt?.let {
it.visibility = View.GONE
}
go_to_app?.let {
it.visibility = View.GONE
}
} else {
no_image?.let {
it.visibility = View.VISIBLE
}
no_image_txt?.let {
it.visibility = View.VISIBLE
}
go_to_app?.let {
it.visibility = View.VISIBLE
}
}
adapterSDK30!!.notifyDataSetChanged()
status_img_swipe.isRefreshing = false
}
override fun onDestroyView() {
job.cancel()
super.onDestroyView()
}
fun processStatusFetchFromChild(completed: () -> Unit) {
val statusSelection = tinyDB.getInt(Constants.status_accounts)
if (statusSelection == 0 || statusSelection == 1) {
if (dataRepository.WhatsAppStatuses == null) {
(activity as StatusActivity).setUpWAURIs()
}
var documentFileStatuses: DocumentFile? = dataRepository.WhatsAppStatuses
if (statusSelection == 1) {
documentFileStatuses = dataRepository.WhatsAppBusinessStatuses
}
if (documentFileStatuses != null) {
launch(Dispatchers.Main) {
val statusImages1 = arrayListOf<DocumentFile>()
async(Dispatchers.IO) {
//this takes time ; want to fetch this same as WhatsApp Gallery
statusImages1.addAll(documentFileStatuses!!.listFiles().filter {
it.mimeType.equals(Constants.MIME_TYPE_IMG_PNG) || it.mimeType.equals(
Constants.MIME_TYPE_IMG_JPG
) || it.mimeType.equals(Constants.MIME_TYPE_IMG_JPEG)
}.sortedByDescending { it.lastModified() })
}.await()
Timber.e("processStatusFetch:Done")
sharedViewModel.statusImages.postValue(statusImages1)
completed()
}
} else {
Timber.e("processStatusFetch:Done")
sharedViewModel.statusImages.postValue(arrayListOf<DocumentFile>())
completed()
}
} else {
Timber.e("processStatusFetch:Done")
sharedViewModel.statusImages.postValue(arrayListOf<DocumentFile>())
completed()
}
}
}
Please note WhatsApp folder path which i used is
val whatsapp_images_path11 = "/Android/media/“ +"com.whatsapp" +"/WhatsApp/Media/WhatsAppImages/"
How i can use MediaStore in this case so that i don't need to use sort and filter functions of list? Its not important to get java.io File only i can work with URIs as well.
What I have finally implemented, in android 10+ you need to ask the user for your specific directory access. Then you can use this functions to fetch statuses:
#RequiresApi(Build.VERSION_CODES.LOLLIPOP)
private fun readSDKFrom30(): ArrayList<String> {
val treeUri = DocumentsContract.buildTreeDocumentUri(
EXTERNAL_STORAGE_PROVIDER_AUTHORITY,
"primary:Android/media/com.whatsapp/WhatsApp/Media/.Statuses"
)
val tree = DocumentFile.fromTreeUri(context, treeUri)!!
val pathList = ArrayList<String>()
listFolderContent(tree).forEach { uri ->
val file = createFileFromContentUri(uri)
pathList.add(file.toString())
}
return pathList
}
private fun listFolderContent(folder: DocumentFile): List<Uri> {
return if (folder.isDirectory) {
val files = folder.listFiles().toMutableList()
files.sortByDescending { it.lastModified() }
files.mapNotNull { file ->
if (file.name != null) file.uri else null
}
} else {
emptyList()
}
}
#RequiresApi(Build.VERSION_CODES.O)
private fun createFileFromContentUri(fileUri: Uri): File {
var fileName = ""
fileUri.let { returnUri ->
context.contentResolver.query(returnUri, null, null, null)
}?.use { cursor ->
val nameIndex = cursor.getColumnIndex(OpenableColumns.DISPLAY_NAME)
cursor.moveToFirst()
fileName = cursor.getString(nameIndex)
}
val iStream: InputStream =
context.contentResolver.openInputStream(fileUri)!!
val outputDir: File = context.cacheDir!!
val outputFile = File(outputDir, fileName)
copyStreamToFile(iStream, outputFile)
iStream.close()
return outputFile
}
private fun copyStreamToFile(inputStream: InputStream, outputFile: File) {
inputStream.use { input ->
val outputStream = FileOutputStream(outputFile)
outputStream.use { output ->
val buffer = ByteArray(4 * 1024) // buffer size
while (true) {
val byteCount = input.read(buffer)
if (byteCount < 0) break
output.write(buffer, 0, byteCount)
}
output.flush()
}
}
}
Using DocumentFile to handle SAF uries is slow indeed.
Better use DocumentsContract to do so.
Its about twenty times as fast as DocumentFile and about as fast as classic File class stuff.
Using MediaStore for hidden folders should be possible. You cannot create hidden folders with the mediastore. But if you managed to make them not using mediastore you should be able to list files in them using mediastore. Well if they are scanned. And if they belong to your app.

Android file chooser & upload getting file not found error

I'm able to select file, selected pdf file got uri as content://somefile but when i'm sending the file using unirest getting error file not found exception below is my code
//file chooser
File file;
Intent chooseFile = new Intent(Intent.ACTION_GET_CONTENT);
chooseFile.setType("*/*");
chooseFile.addCategory(Intent.CATEGORY_OPENABLE);
startActivityForResult(Intent.createChooser(chooseFile, "Select a File to Upload"),1);
#Override
public void onActivityResult(int requestCode, int resultCode, Intent data) {
super.onActivityResult(requestCode, resultCode, data);
switch (requestCode) {
case 1:
if (resultCode == -1) {
file = new File(data.getData().getPath());
}
break;
}
}
HttpResponse<String> response = Unirest.post("http://url/")
.field("file", file)
you have to do the typical method so please use retrofit for saving so much time
file upload example - How to Upload Image file in Retrofit 2
for networking, you have to also visit the official retrofit site
https://square.github.io/retrofit/
99% of people recommend retrofit for networking and saving so much time
import android.content.ContentUris
import android.content.Context
import android.database.Cursor
import android.net.Uri
import android.os.Build
import android.os.Environment
import android.provider.DocumentsContract
import android.provider.MediaStore
object SelectedFilePath {
fun getPath(context: Context, uri: Uri): String? {
//check here to KITKAT or new version
val isKitKat = Build.VERSION.SDK_INT >= Build.VERSION_CODES.KITKAT
// DocumentProvider
if (isKitKat && DocumentsContract.isDocumentUri(context, uri)) {
// ExternalStorageProvider
if (isExternalStorageDocument(uri)) {
val docId = DocumentsContract.getDocumentId(uri)
val split = docId.split(":").toTypedArray()
val type = split[0]
if ("primary".equals(type, ignoreCase = true)) {
return Environment.getExternalStorageDirectory().toString() + "/" + split[1]
}
} else if (isDownloadsDocument(uri)) {
val id = DocumentsContract.getDocumentId(uri)
val contentUri = ContentUris.withAppendedId(
Uri.parse("content://<span id=\"IL_AD1\" class=\"IL_AD\">downloads</span>/public_downloads"),
java.lang.Long.valueOf(id)
)
return getDataColumn(context, contentUri, null, null)
} else if (isMediaDocument(uri)) {
val docId = DocumentsContract.getDocumentId(uri)
val split = docId.split(":").toTypedArray()
val type = split[0]
var contentUri: Uri? = null
if ("image" == type) {
contentUri = MediaStore.Images.Media.EXTERNAL_CONTENT_URI
} else if ("video" == type) {
contentUri = MediaStore.Video.Media.EXTERNAL_CONTENT_URI
} else if ("audio" == type) {
contentUri = MediaStore.Audio.Media.EXTERNAL_CONTENT_URI
}
val selection = "_id=?"
val selectionArgs = arrayOf(
split[1]
)
return getDataColumn(context, contentUri, selection, selectionArgs)
}
} else if ("content".equals(uri.scheme, ignoreCase = true)) {
// Return the remote address
return if (isGooglePhotosUri(uri)) uri.lastPathSegment else getDataColumn(
context,
uri,
null,
null
)
} else if ("file".equals(uri.scheme, ignoreCase = true)) {
return uri.path
}
return null
}
fun getDataColumn(
context: Context, uri: Uri?, selection: String?,
selectionArgs: Array<String>?
): String? {
var cursor: Cursor? = null
val column = "_data"
val projection = arrayOf(
column
)
try {
cursor = context.contentResolver.query(
uri!!, projection, selection, selectionArgs,
null
)
if (cursor != null && cursor.moveToFirst()) {
val index = cursor.getColumnIndexOrThrow(column)
return cursor.getString(index)
}
} finally {
cursor?.close()
}
return null
}
fun isExternalStorageDocument(uri: Uri): Boolean {
return "com.android.externalstorage.documents" == uri.authority
}
fun isDownloadsDocument(uri: Uri): Boolean {
return "com.android.providers.downloads.documents" == uri.authority
}
fun isMediaDocument(uri: Uri): Boolean {
return "com.android.providers.media.documents" == uri.authority
}
fun isGooglePhotosUri(uri: Uri): Boolean {
return "com.google.android.apps.photos.content" == uri.authority
}
}
Now onActivityResult use this class
MY_RESULT_CODE_FILECHOOSER -> if (resultCode == Activity.RESULT_OK) {
if (data != null) {
val fileUri: Uri? = data.data
Log.d(LOG_TAG, "Uri: $fileUri")
var filePath: String? = null
try {
filePath = fileUri?.let { SelectedFilePath.getPath(this, it) }
uploadFile(filePath)
} catch (e: Exception) {
Log.e(LOG_TAG, "Error: $e")
showToast("Error: $e")
}
if (filePath != null) {
Log.e(LOG_TAG, filePath)
}
}
}
Full code is:
private fun askPermissionAndBrowseFile() {
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.M) { // Level 23
// Check if we have Call permission
val permisson: Int = ActivityCompat.checkSelfPermission(
this,
Manifest.permission.READ_EXTERNAL_STORAGE
)
if (permisson != PackageManager.PERMISSION_GRANTED) {
requestPermissions(
arrayOf(Manifest.permission.READ_EXTERNAL_STORAGE),
MY_REQUEST_CODE_PERMISSION
)
return
}
}
browseFiles()
}
private fun browseFiles() {
var chooseFileIntent = Intent(Intent.ACTION_GET_CONTENT)
chooseFileIntent.type = "*/*"
chooseFileIntent.addCategory(Intent.CATEGORY_OPENABLE)
chooseFileIntent = Intent.createChooser(chooseFileIntent, "Choose a file")
startActivityForResult(chooseFileIntent, MY_RESULT_CODE_FILECHOOSER)
}
override fun onRequestPermissionsResult(
requestCode: Int,
permissions: Array<out String>,
grantResults: IntArray
) {
super.onRequestPermissionsResult(requestCode, permissions, grantResults)
when (requestCode) {
MY_REQUEST_CODE_PERMISSION -> {
// Note: If request is cancelled, the result arrays are empty.
// Permissions granted (CALL_PHONE).
if (grantResults.size > 0
&& grantResults[0] == PackageManager.PERMISSION_GRANTED
) {
Log.d(LOG_TAG, "Permission granted!")
showToast("Permission granted!")
browseFiles()
} else {
Log.i(LOG_TAG, "Permission denied!")
showToast("Permission denied!")
}
}
}
}
override fun onActivityResult(requestCode: Int, resultCode: Int, data: Intent?) {
when (requestCode) {
MY_RESULT_CODE_FILECHOOSER -> if (resultCode == Activity.RESULT_OK) {
if (data != null) {
val fileUri: Uri? = data.data
Log.d(LOG_TAG, "Uri: $fileUri")
var filePath: String? = null
try {
filePath = fileUri?.let { SelectedFilePath.getPath(this, it) }
uploadFile(filePath)
} catch (e: Exception) {
Log.e(LOG_TAG, "Error: $e")
showToast("Error: $e")
}
if (filePath != null) {
Log.e(LOG_TAG, filePath)
}
}
}
}
super.onActivityResult(requestCode, resultCode, data)
}
private fun uploadFile(filePath: String?) {
val myData = Data.Builder().putString(AppConstants.FILE_TO_UPLOAD, filePath).build()
val oneTimeWorkRequest = OneTimeWorkRequest.Builder(UploadFileWorker::class.java)
.setInputData(myData)
.build()
WorkManager.getInstance(this).enqueue(oneTimeWorkRequest)
Toast.makeText(this, resources.getString(R.string.file_will_be_uploaded), Toast.LENGTH_SHORT).show()
}
enter code here
Upload worker class :
import android.app.NotificationChannel
import android.app.NotificationManager
import android.content.Context
import android.os.Build
import android.util.Log
import android.widget.Toast
import androidx.core.app.NotificationCompat
import androidx.work.Worker
import androidx.work.WorkerParameters
import com.thirdeyegen.mr_workspace.R
import com.thirdeyegen.mr_workspace.constants.AppConstants
import com.thirdeyegen.mr_workspace.network.ProgressRequestBody
import com.thirdeyegen.mr_workspace.network.RestClient
import okhttp3.MultipartBody
import retrofit2.Call
import retrofit2.Callback
import retrofit2.Response
import java.io.File
infix fun Int.hasReached(limit: Int): Boolean {
return this == limit
}
class UploadFileWorker(context: Context, workerParams: WorkerParameters) : Worker(context, workerParams) {
private var isSuccess = true
private val TAG = "UPLOAD_FILE_WORKER==>"
val PROGRESS_MAX = 100
var PROGRESS_CURRENT = 0
val mContext = context
private lateinit var builder: NotificationCompat.Builder
private lateinit var notificationManager: NotificationManager
override fun doWork(): Result {
Log.d(TAG, "DO_WORK")
createNotificationChannel()
return startUploading()
}
private fun startUploading(): Result {
val filePath = inputData.getString(AppConstants.FILE_TO_UPLOAD)
if (filePath != null) {
val file = File(filePath)
val fileBody = ProgressRequestBody(file, "*", object : ProgressRequestBody.UploadCallbacks {
override fun onFinish() {
if (PROGRESS_CURRENT hasReached 100) {
notificationManager.cancel(1)
}
}
override fun onProgressUpdate(percentage: Int) {
PROGRESS_CURRENT = percentage
builder.setProgress(PROGRESS_MAX, PROGRESS_CURRENT, false)
notificationManager.notify(1, builder.build());
if (percentage hasReached 100) {
notificationManager.cancel(1)
}
Log.d(TAG, "UPLOAD_PROGRESS====> $percentage")
}
override fun onError() {
notificationManager.cancel(1)
Log.d(TAG, "UPLOAD_ERROR===>")
}
})
val filePart: MultipartBody.Part = MultipartBody.Part.createFormData("file", file.name, fileBody)
RestClient.apiInterfaceForUploader.uploadVideo(filePart, "record").enqueue(object : Callback<Any?> {
override fun onResponse(call: Call<Any?>, response: Response<Any?>) {
Toast.makeText(applicationContext, R.string.msg_file_uploaded_successfully, Toast.LENGTH_LONG).show()
isSuccess = true
}
override fun onFailure(call: Call<Any?>, t: Throwable) {
Toast.makeText(applicationContext, R.string.msg_error_while_uploading_file, Toast.LENGTH_SHORT).show()
isSuccess = false
}
})
}
return if (isSuccess) {
Result.success()
} else {
Result.failure()
}
}
private fun createNotificationChannel() {
// Create the NotificationChannel, but only on API 26+ because
// the NotificationChannel class is new and not in the support library
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.O) {
val name: CharSequence = "My Channel"
val description = mContext.getString(R.string.upload_file_text)
val importance: Int = NotificationManager.IMPORTANCE_DEFAULT
val channel = NotificationChannel("upload_worker", name, importance)
channel.description = description
// Register the channel with the system; you can't change the importance
// or other notification behaviors after this
val notificationManager: NotificationManager = mContext.getSystemService(NotificationManager::class.java)
notificationManager.createNotificationChannel(channel)
}
setupNotification(mContext)
}
private fun setupNotification(context: Context) {
builder = NotificationCompat.Builder(context, "upload_worker").apply {
setContentTitle(context.getString(R.string.file_upload_text))
setOngoing(false)
setAutoCancel(true)
setContentText(context.getString(R.string.upload_in_progress))
setSmallIcon(R.drawable.ic_files)
setSound(null)
setNotificationSilent()
priority = NotificationCompat.PRIORITY_LOW
}
notificationManager = mContext.getSystemService(Context.NOTIFICATION_SERVICE) as NotificationManager
notificationManager.notify(1, builder.build())
builder.setOngoing(true)
}
}

Getting a class not found exception along with cannot decode stream exception while trying to retrieve images

I am getting a class not found exception when trying to click an image from the camera or while fetching it from the directory. The code is in kotlin and has been given below.
This is the class which implements the functionality to capture an image or picks them from the gallery.
The methods _openCamera() and openFileSelector() are implemented.
The main motive was to capture the images and upload them in the server but the implemented methods doesn't give the proper results.
class MainActivity : AppCompatActivity() {
private var drawerResult: Drawer? = null
private var jobschedular: JobScheduler? = null
private var jobschedularCode: Int = 1
private var phoneNumber: String? = null
private var toolbar: Toolbar? = null
private var familyId: String? = null
val TAG: String? = "Activity_Name"
val REQUEST_IMAGE_CAPTURE = 1
val REQUEST_CODE_FOR_GALLERY_CAPTURE = 2
var photoFile: File? = null
var progressDialog: Dialog? = null
private var doubleBackToExitPressedOnce = false
#SuppressLint("PrivateResource")
override fun onCreate(savedInstanceState: Bundle?) {
super.onCreate(savedInstanceState)
setContentView(R.layout.activity_main)
overridePendingTransition(R.anim.fade_in, R.anim.fade_out)
Log.d(TAG, "Inside MainActivity")
//onclick listener for open camera
onclickListenerForOpenCamera()
//starting the services here . .
val service_checkAddedtoFamily = Intent(this, checkAddedToFamily::class.java)
startService(service_checkAddedtoFamily)
val service_checkDocsToBeVerified = Intent(this, checkDocsToBeVerified::class.java)
startService(service_checkDocsToBeVerified)
/*findViewById<Button>(R.id.scan).setOnClickListener {
val i = Intent(this, Testers::class.java)
i.addFlags(Intent.FLAG_GRANT_READ_URI_PERMISSION)
i.addFlags(Intent.FLAG_GRANT_WRITE_URI_PERMISSION)
startActivity(i)
overridePendingTransition(R.anim.fade_in, R.anim.fade_out)
}*/
//onclick listener for select image button
attach_onclick_listener_to_add_photos_from_gallery()
//onclick listener for select pdf files
onclickListenerForSelectPdfFile()
//get toolbar for drawer
toolbar = findViewById(R.id.toolbar_tabs)
//get phone number
val loginInfo = applicationContext.getSharedPreferences("loginInfo", Context.MODE_PRIVATE)
phoneNumber = loginInfo.getString("phoneNumber", "")
//onclick listener for upload button
//onclickListenerForUploadButton()
//onclick listener for retrieve button
onclickListenerForRetrieveButton()
//on click permanent diseases button
//onclickPermanentDiseasesButtton()
//navigation drawer
left_drawer(this, this#MainActivity, toolbar!!).createNavigationDrawer()
//verify auto upload
verifyAutoLoginInformation()
//create Sqlite database
DB_HELPER(this#MainActivity).writableDatabase
//get job schedular service
jobschedular = applicationContext.getSystemService(Context.JOB_SCHEDULER_SERVICE) as JobScheduler
schedulTheJobForHealthGoals()
schedulTheJobForHealthInsurance()
setPreferencesForNutrition()
schedulTheJobForNutrition()
schedulTheJobForSyncNutritionOnline()
}
/*override fun onBackPressed() {
if (doubleBackToExitPressedOnce) {
super.onBackPressed()
return
}
this.doubleBackToExitPressedOnce = true
Toast.makeText(this, "Press back again to exit", Toast.LENGTH_SHORT).show()
Handler().postDelayed(Runnable { doubleBackToExitPressedOnce = false }, 2000)
}*/
//job schedular
fun schedulTheJobForHealthGoals() {
val builder = JobInfo.Builder(jobschedularCode, ComponentName(this#MainActivity, health_goals_services::class.java))
.setPersisted(true)
.setPeriodic(5000)
.setRequiredNetworkType(JobInfo.NETWORK_TYPE_UNMETERED)
.setRequiresCharging(false)
.setRequiresDeviceIdle(false)
val bundle = PersistableBundle()
bundle.putString("key", "value")
builder.setExtras(bundle)
val s_response = jobschedular!!.schedule(builder.build())
if (s_response <= 0) {
//something goes wrong
}
}
fun schedulTheJobForHealthInsurance() {
val builder = JobInfo.Builder(jobschedularCode, ComponentName(this#MainActivity, health_insurance_service::class.java))
.setPersisted(true)
.setPeriodic(5000)
.setRequiredNetworkType(JobInfo.NETWORK_TYPE_UNMETERED)
.setRequiresCharging(false)
.setRequiresDeviceIdle(false)
val bundle = PersistableBundle()
bundle.putString("key", "value")
builder.setExtras(bundle)
val s_response = jobschedular!!.schedule(builder.build())
if (s_response <= 0) {
//something goes wrong
}
}
fun schedulTheJobForNutrition() {
val builder = JobInfo.Builder(jobschedularCode, ComponentName(this#MainActivity, nutrition_service::class.java))
.setPersisted(true)
.setPeriodic(5000) //change to 1 hour
.setRequiredNetworkType(JobInfo.NETWORK_TYPE_UNMETERED)
.setRequiresCharging(false)
.setRequiresDeviceIdle(false)
val bundle = PersistableBundle()
bundle.putString("key", "value")
builder.setExtras(bundle)
val s_response = jobschedular!!.schedule(builder.build())
if (s_response <= 0) {
//something goes wrong
}
}
fun setPreferencesForNutrition() {
val nutritionInfo = getSharedPreferences("nutrition", Context.MODE_PRIVATE)
val editor = nutritionInfo.edit()
editor.putString("breakFastTime_Hour", "7")
editor.putString("lunchTime_Hour", "14") //TODO: change to 13
editor.putString("DinnerTime_Hour", "20")
editor.apply()
}
fun schedulTheJobForSyncNutritionOnline() {
val builder = JobInfo.Builder(jobschedularCode, ComponentName(this#MainActivity, sync_nutrition_online::class.java))
.setPersisted(true)
.setPeriodic(5000)
.setRequiredNetworkType(JobInfo.NETWORK_TYPE_UNMETERED)
.setRequiresCharging(false)
.setRequiresDeviceIdle(false)
val bundle = PersistableBundle()
bundle.putString("key", "value")
builder.setExtras(bundle)
val s_response = jobschedular!!.schedule(builder.build())
if (s_response <= 0) {
//something goes wrong
}
}
//buttons on home screen
/*fun onclickListenerForUploadButton(){
findViewById<ImageView>(R.id.uploadButton).setOnClickListener{
openModeOfUploadActivity()
}
}*/
fun onclickListenerForRetrieveButton() {
findViewById<Button>(R.id.retrieveButton).setOnClickListener {
openHistoryActivity()
}
}
/*fun onclickPermanentDiseasesButtton(){
findViewById<Button>(R.id.permanentDiseasesButton).setOnClickListener{
openPermanentDiseases()
}
}*/
/*fun openModeOfUploadActivity(){
val intent = Intent(this,MainActivity::class.java)
startActivity(intent)
}*/
fun openHistoryActivity() {
val intent = Intent(this, history_pickFamilyMember::class.java)
startActivity(intent)
overridePendingTransition(R.anim.fade_in, R.anim.fade_out)
}
/*fun openPermanentDiseases(){
val intent = Intent(this,permanentDiseaese::class.java)
startActivity(intent)
}
*/
//verify auto login information
fun verifyAutoLoginInformation() {
val loginInfo = applicationContext.getSharedPreferences("loginInfo", Context.MODE_PRIVATE)
if (loginInfo.contains("familyOrIndividual") == true) {
//for family
if (loginInfo.getString("familyOrIndividual", "").toString() == "f") {
if (loginInfo.contains("phoneNumber") == true && loginInfo.contains("password") == true) {
val phoneNumber = loginInfo.getString("phoneNumber", "")
val password = loginInfo.getString("password", "")
individual_family_login(this#MainActivity).makeFamilyLoginApiRequest(phoneNumber, password)
} else {
left_drawer(this, this#MainActivity, toolbar!!).makeUserLogOut()
}
}
//for individual
if (loginInfo.getString("familyOrIndividual", "").toString() == "i") {
if (loginInfo.contains("phoneNumber") == true && loginInfo.contains("password") == true) {
val phoneNumber = loginInfo.getString("phoneNumber", "")
val password = loginInfo.getString("password", "")
individual_family_login(this#MainActivity).makeLoginApiRequest(phoneNumber, password)
} else {
left_drawer(this, this#MainActivity, toolbar!!).makeUserLogOut()
}
}
//for security
if (loginInfo.getString("familyOrIndividual", "").toString() != "i" && loginInfo.getString("familyOrIndividual", "").toString() != "f") {
left_drawer(this, this#MainActivity, toolbar!!).makeUserLogOut()
}
} else {
left_drawer(this, this#MainActivity, toolbar!!).makeUserLogOut()
}
}
//camera scan
fun onclickListenerForOpenCamera() {
findViewById<ImageView>(R.id.openCamera).setOnClickListener {
get_permissions_camera()
}
}
fun _openCamera() {
Log.d("Errors__", "inside _openCamera()")
try {
val takePictureIntent = Intent(MediaStore.ACTION_IMAGE_CAPTURE)
if (takePictureIntent.resolveActivity(getPackageManager()) != null) {
try {
photoFile = createImageFile()
} catch (ex: Exception) {
Log.d("Errors__", "inside: " + ex.toString())
}
if (photoFile != null) {
val builder: StrictMode.VmPolicy.Builder = StrictMode.VmPolicy.Builder()
StrictMode.setVmPolicy(builder.build())
val photoURI: Uri = Uri.fromFile(photoFile!!)
Log.d("Path__", "photoURI: $photoURI")
Log.d("Path__", "photoURI.path: " + photoURI.path)
takePictureIntent.putExtra(MediaStore.EXTRA_OUTPUT, photoURI)
takePictureIntent.addFlags(Intent.FLAG_GRANT_READ_URI_PERMISSION)
startActivityForResult(takePictureIntent, REQUEST_IMAGE_CAPTURE)
}
}
} catch (e: Exception) {
Log.d("Errors__", "_openCamera" + e.toString())
}
}
fun createImageFile(): File {
Log.d("Errors__", "inside createImageFile()")
val mCurrentPhotoPath: String
val imageFileName = "camera"
val storageDir: File = getExternalFilesDir(Environment.DIRECTORY_PICTURES)
val image = File.createTempFile(
imageFileName, /* prefix */
".jpg", /* suffix */
storageDir /* directory */
)
// Save a file: path for use with ACTION_VIEW intents
mCurrentPhotoPath = image.getAbsolutePath()
Log.d("Path__", "Image: $image")
return image
}
//file selector
fun onclickListenerForSelectPdfFile() {
findViewById<ImageView>(R.id.selectPdfFile).setOnClickListener {
get_permissions_fileExplorer()
}
}
#SuppressLint("SdCardPath")
fun openFileSelector() {
val properties = DialogProperties()
properties.selection_mode = DialogConfigs.MULTI_MODE;
properties.selection_type = DialogConfigs.FILE_SELECT;
properties.root = File(DialogConfigs.DEFAULT_DIR);
properties.error_dir = File(DialogConfigs.DEFAULT_DIR);
properties.offset = File(DialogConfigs.DEFAULT_DIR);
properties.extensions = null;
val dialog: FilePickerDialog = FilePickerDialog(this#MainActivity, properties)
dialog.setTitle("Select a File")
dialog.setDialogSelectionListener(object : DialogSelectionListener {
override fun onSelectedFilePaths(files: Array<out String>?) {
convertPdfToImages(files!!)
}
})
dialog.show()
}
fun convertPdfToImages(files: Array<out String>) {
showProcessProgress()
doAsync {
var uriList: MutableList<Uri>? = mutableListOf()
val no_of_files = files.size
var counter = 0
while (counter < no_of_files) {
var pdfFile = File(files[counter])
val decodeService = DecodeServiceBase(PdfContext())
decodeService.setContentResolver(applicationContext.getContentResolver())
decodeService.open(Uri.fromFile(pdfFile))
val pageCount: Int = decodeService.getPageCount()
var i = 0
while (i < pageCount) {
val page: PdfPage = decodeService.getPage(i) as PdfPage
val rectF = RectF(0.toFloat(), 0.toFloat(), 1.toFloat(), 1.toFloat())
// do a fit center to 1920x1080
val scaleBy = 1
val with: Int = (page.getWidth() * scaleBy)
val height: Int = (page.getHeight() * scaleBy)
val bitmap: Bitmap = page.renderBitmap(with, height, rectF)
try {
val outputFile = File(applicationContext.externalCacheDir,
System.currentTimeMillis().toString() + ".jpg")
val outputStream = FileOutputStream(outputFile)
// a bit long running
bitmap.compress(Bitmap.CompressFormat.JPEG, 100, outputStream)
uriList!!.add(Uri.fromFile(outputFile))
outputStream.close()
} catch (e: IOException) {
}
i++
}
counter++
}
uiThread {
progressDialog!!.hide()
openPreview(uriList!!)
Log.d("mess", "size: " + uriList.size + " " + uriList.toString())
}
}
}
//select image
fun attach_onclick_listener_to_add_photos_from_gallery() {
findViewById<ImageView>(R.id.selectImage).setOnClickListener {
get_permissions_gallery()
}
}
fun open_selector() {
Matisse.from(this)
.choose(MimeType.allOf())
.countable(true)
.maxSelectable(200)
.restrictOrientation(ActivityInfo.SCREEN_ORIENTATION_PORTRAIT)
.thumbnailScale(0.85f)
.imageEngine(PicassoEngine())
.forResult(REQUEST_CODE_FOR_GALLERY_CAPTURE)
}
//activity results
override fun onActivityResult(requestCode: Int, resultCode: Int, data: Intent?) {
if (requestCode == REQUEST_IMAGE_CAPTURE && resultCode == Activity.RESULT_OK) {
var uriOfImage = Uri.fromFile(photoFile)
Log.d("fileCapturing__", "URI Image: $uriOfImage")
//start croper
CropImage.activity(uriOfImage)
.start(this)
}
if (requestCode == REQUEST_CODE_FOR_GALLERY_CAPTURE && resultCode == Activity.RESULT_OK) {
var selected_images = Matisse.obtainResult(data)
openPreview(selected_images!!)
}
//for croper
if (requestCode == CropImage.CROP_IMAGE_ACTIVITY_REQUEST_CODE) {
val result: CropImage.ActivityResult = CropImage.getActivityResult(data)
if (resultCode == RESULT_OK) {
doAsync {
val builder: StrictMode.VmPolicy.Builder = StrictMode.VmPolicy.Builder()
StrictMode.setVmPolicy(builder.build())
var resultUri: Uri = result.getUri()
//save cropped image for persisitance
val croppedImage = createImageFile() //empty
val outputStream = FileOutputStream(croppedImage)
// a bit long running
(Picasso.with(this#MainActivity)
.load(resultUri)
.get()
).compress(Bitmap.CompressFormat.JPEG, 100, outputStream)
Log.d("fileCapturing__", "outputStream: $outputStream")
resultUri = Uri.fromFile(croppedImage)
outputStream.close()
uiThread {
//add to mu list
var mu_list = ArrayList<Uri>(1)
mu_list.add(resultUri)
Log.d("fileCapturing__", "camera uri" + resultUri.toString())
openPreview(mu_list)
}
}
} else if (resultCode == CropImage.CROP_IMAGE_ACTIVITY_RESULT_ERROR_CODE) {
val error: Exception = result.getError()
Log.d("fileCapturing__", "Error: $error")
}
}
}
//preview
fun openPreview(list: MutableList<Uri>) {
val _object = list
val i = Intent(this, typeOfDocument::class.java)
val args = Bundle()
args.putSerializable("ARRAYLIST", _object as java.io.Serializable)
i.putExtra("BUNDLE", args)
startActivity(i)
finish()
}
//get permissions
//Camera
fun get_permissions_camera() {
if (ContextCompat.checkSelfPermission(this#MainActivity, android.Manifest.permission.CAMERA) != PackageManager.PERMISSION_GRANTED) {
MaterialDialog.Builder(this#MainActivity)
.title("Camera permission")
.content("Camera permissions are required for opening Camera")
.negativeText("Cancel")
.onNegative(object : MaterialDialog.SingleButtonCallback {
override fun onClick(dialog: MaterialDialog, which: DialogAction) {
}
})
.positiveText("Give Permissions")
.onPositive(object : MaterialDialog.SingleButtonCallback {
override fun onClick(dialog: MaterialDialog, which: DialogAction) {
getPermissionsUsingDexter_camera(
android.Manifest.permission.CAMERA
)
}
})
.show()
} else {
_openCamera()
}
}
fun getPermissionsUsingDexter_camera(permissionString: String) {
Dexter.withActivity(this)
.withPermissions(
permissionString
).withListener(object : MultiplePermissionsListener {
override fun onPermissionRationaleShouldBeShown(permissions: MutableList<PermissionRequest>?, token: PermissionToken?) {
}
override fun onPermissionsChecked(report: MultiplePermissionsReport?) {
if (report!!.areAllPermissionsGranted() == true) {
_openCamera()
Log.d("mess", "permission given")
} else {
Log.d("mess", "permission not granted")
}
}
})
.check()
}
//gallery
fun get_permissions_gallery() {
if (ContextCompat.checkSelfPermission(this#MainActivity, android.Manifest.permission.READ_EXTERNAL_STORAGE) != PackageManager.PERMISSION_GRANTED) {
MaterialDialog.Builder(this#MainActivity)
.title("Storage permission")
.content("Storage permissions are required for opening the Gallery")
.negativeText("Cancel")
.onNegative(object : MaterialDialog.SingleButtonCallback {
override fun onClick(dialog: MaterialDialog, which: DialogAction) {
}
})
.positiveText("Give Permissions")
.onPositive(object : MaterialDialog.SingleButtonCallback {
override fun onClick(dialog: MaterialDialog, which: DialogAction) {
getPermissionsUsingDexter_gallery(
android.Manifest.permission.READ_EXTERNAL_STORAGE
)
}
})
.show()
} else {
open_selector()
}
}
fun getPermissionsUsingDexter_gallery(permissionString: String) {
Dexter.withActivity(this)
.withPermissions(
permissionString
).withListener(object : MultiplePermissionsListener {
override fun onPermissionRationaleShouldBeShown(permissions: MutableList<PermissionRequest>?, token: PermissionToken?) {
}
override fun onPermissionsChecked(report: MultiplePermissionsReport?) {
if (report!!.areAllPermissionsGranted() == true) {
open_selector()
Log.d("mess", "permission given")
} else {
Log.d("mess", "permission not granted")
}
}
})
.check()
}
//file exploer
fun get_permissions_fileExplorer() {
if (ContextCompat.checkSelfPermission(this#MainActivity, android.Manifest.permission.READ_EXTERNAL_STORAGE) != PackageManager.PERMISSION_GRANTED) {
MaterialDialog.Builder(this#MainActivity)
.title("Storage permission")
.content("Storage access permissions are required for opening File Explorer")
.negativeText("Cancel")
.onNegative(object : MaterialDialog.SingleButtonCallback {
override fun onClick(dialog: MaterialDialog, which: DialogAction) {
}
})
.positiveText("Give Permissions")
.onPositive(object : MaterialDialog.SingleButtonCallback {
override fun onClick(dialog: MaterialDialog, which: DialogAction) {
getPermissionsUsingDexter_fileExplores(
android.Manifest.permission.READ_EXTERNAL_STORAGE
)
}
})
.show()
} else {
openFileSelector()
}
}
fun getPermissionsUsingDexter_fileExplores(permissionString: String) {
Dexter.withActivity(this)
.withPermissions(
permissionString
).withListener(object : MultiplePermissionsListener {
override fun onPermissionRationaleShouldBeShown(permissions: MutableList<PermissionRequest>?, token: PermissionToken?) {
}
override fun onPermissionsChecked(report: MultiplePermissionsReport?) {
if (report!!.areAllPermissionsGranted() == true) {
openFileSelector()
Log.d("mess", "permission given")
} else {
Log.d("mess", "permission not granted")
}
}
})
.check()
}
//progress bar
fun showProcessProgress() {
progressDialog = MaterialDialog.Builder(this)
.title("Please Wait")
.content("Converting Pdf to Images")
.progress(true, 0)
.show()
}
}
The error shows when I try to click an image or fetch from the library.
The provider path used is given below:
<paths xmlns:android="http://schemas.android.com/apk/res/android">
<external-path
name="external_files"
path="." />
</paths>
I have tried changing the path to "/" but it didn't work. The error wasn't showing earlier but the error exists now.
Here is the snapshot of the logs.
All suggestions are accepted. Thanks in advance.

Categories

Resources