Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Migrated Custom Image Selector to Jetpack Compose #5964

Draft
wants to merge 24 commits into
base: main
Choose a base branch
from
Draft
Show file tree
Hide file tree
Changes from all commits
Commits
Show all changes
24 commits
Select commit Hold shift + click to select a range
b0bf55b
UI: add material theme for compose
rohit9625 Sep 20, 2024
3f3df7d
add ui components for custom selector
rohit9625 Sep 20, 2024
8558594
add adaptive layout, image loading, and compose navigation dependencies
rohit9625 Oct 3, 2024
74cf035
add logic to fetch images from storage and manage in viewmodel
rohit9625 Oct 3, 2024
9f7ae75
create custom selector main screen with UI events and folder data class
rohit9625 Oct 3, 2024
0ff2880
create image grid screen/pane to display images from any folder
rohit9625 Oct 3, 2024
5da9097
refactor: add new cs screen into custom selector activity
rohit9625 Oct 3, 2024
1a86883
add actions lambda to bottom bar and replace hard-coded strings
rohit9625 Oct 13, 2024
ca30bf1
Add selection count indicator in top bar and refactor
rohit9625 Oct 13, 2024
55c0939
Add drag and tap gestures to select images
rohit9625 Oct 13, 2024
31c012f
refactor holder screen for both folder and image panes
rohit9625 Oct 13, 2024
071bffb
refactor preview for folder item
rohit9625 Oct 13, 2024
a930d8e
add view image screen and enable edge to edge for custom selector
rohit9625 Nov 25, 2024
dcd31f9
update dependencies
rohit9625 Nov 25, 2024
130158f
remove state-changing argument causing unnecessary recompositions
rohit9625 Nov 25, 2024
03713dd
add functionality for unselecting all pictures at once
rohit9625 Nov 26, 2024
178154c
fix overlapping navigation bar adding navigation bar padding
rohit9625 Nov 26, 2024
4ebb945
Merge branch 'main' into jetpack-custom-selector
rohit9625 Nov 27, 2024
c688059
remove onBackPressed override
rohit9625 Nov 27, 2024
c033003
move models package inside domain package
rohit9625 Nov 29, 2024
1d11ab7
fix imports and refactor code
rohit9625 Nov 29, 2024
443e713
refactor Ui components for custom selector
rohit9625 Jan 3, 2025
e611cbc
update UI states and refactor code
rohit9625 Jan 3, 2025
d1fdab4
add logic to mark or unmark images as not for upload
rohit9625 Jan 3, 2025
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
15 changes: 11 additions & 4 deletions app/build.gradle
Original file line number Diff line number Diff line change
Expand Up @@ -47,15 +47,15 @@ dependencies {

implementation 'com.jakewharton.timber:timber:4.7.1'
implementation 'com.github.deano2390:MaterialShowcaseView:1.2.0'
implementation "com.google.android.material:material:1.9.0"
implementation "com.google.android.material:material:1.12.0"
implementation 'com.karumi:dexter:5.0.0'
implementation 'androidx.lifecycle:lifecycle-extensions:2.2.0'

// Jetpack Compose
def composeBom = platform('androidx.compose:compose-bom:2024.08.00')
def composeBom = platform('androidx.compose:compose-bom:2024.10.00')

implementation "androidx.activity:activity-compose:1.9.1"
implementation "androidx.lifecycle:lifecycle-runtime-ktx:2.8.4"
implementation "androidx.activity:activity-compose:1.9.3"
implementation "androidx.lifecycle:lifecycle-runtime-ktx:2.8.6"
implementation (composeBom)
implementation "androidx.compose.runtime:runtime"
implementation "androidx.compose.ui:ui"
Expand All @@ -65,6 +65,13 @@ dependencies {
implementation "androidx.compose.foundation:foundation-layout"
implementation "androidx.compose.material3:material3"
androidTestImplementation(composeBom)
// Adaptive Layout APIs
implementation "androidx.compose.material3.adaptive:adaptive:1.0.0"
implementation "androidx.compose.material3.adaptive:adaptive-layout-android:1.0.0"
implementation "androidx.compose.material3.adaptive:adaptive-navigation-android:1.0.0"

implementation "io.coil-kt:coil-compose:2.6.0"
implementation "androidx.navigation:navigation-compose:2.8.3"

implementation "com.hannesdorfmann:adapterdelegates4-kotlin-dsl-viewbinding:$ADAPTER_DELEGATES_VERSION"
implementation "com.hannesdorfmann:adapterdelegates4-pagination:$ADAPTER_DELEGATES_VERSION"
Expand Down
1 change: 1 addition & 0 deletions app/src/main/AndroidManifest.xml
Original file line number Diff line number Diff line change
Expand Up @@ -144,6 +144,7 @@
android:label="@string/result" />
<activity
android:name=".customselector.ui.selector.CustomSelectorActivity"
android:windowSoftInputMode="adjustResize"
android:configChanges="screenSize|keyboard|orientation"
android:label="@string/title_activity_custom_selector"
android:parentActivityName=".contributions.MainActivity" />
Expand Down
Original file line number Diff line number Diff line change
@@ -0,0 +1,29 @@
package fr.free.nrw.commons.customselector.data

import fr.free.nrw.commons.customselector.database.NotForUploadStatus
import fr.free.nrw.commons.customselector.database.NotForUploadStatusDao
import fr.free.nrw.commons.customselector.domain.ImageRepository
import fr.free.nrw.commons.customselector.domain.model.Image
import kotlinx.coroutines.flow.Flow
import javax.inject.Inject

class ImageRepositoryImpl @Inject constructor(
private val mediaReader: MediaReader,
private val notForUploadStatusDao: NotForUploadStatusDao
): ImageRepository {
override suspend fun getImagesFromDevice(): Flow<Image> {
return mediaReader.getImages()
}

override suspend fun markAsNotForUpload(imageSHA: String) {
notForUploadStatusDao.insert(NotForUploadStatus(imageSHA))
}

override suspend fun unmarkAsNotForUpload(imageSHA: String) {
notForUploadStatusDao.deleteWithImageSHA1(imageSHA)
}

override suspend fun isNotForUpload(imageSHA: String): Boolean {
return notForUploadStatusDao.find(imageSHA) > 0
}
}
Original file line number Diff line number Diff line change
@@ -0,0 +1,71 @@
package fr.free.nrw.commons.customselector.data

import android.content.ContentUris
import android.content.Context
import android.provider.MediaStore
import android.text.format.DateFormat
import fr.free.nrw.commons.customselector.domain.model.Image
import kotlinx.coroutines.Dispatchers
import kotlinx.coroutines.flow.flow
import kotlinx.coroutines.flow.flowOn
import java.util.Calendar
import java.util.Date
import javax.inject.Inject

class MediaReader @Inject constructor(private val context: Context) {
fun getImages() = flow {
val projection = arrayOf(
MediaStore.Images.Media._ID,
MediaStore.Images.Media.DISPLAY_NAME,
MediaStore.Images.Media.DATA,
MediaStore.Images.Media.BUCKET_ID,
MediaStore.Images.Media.BUCKET_DISPLAY_NAME,
MediaStore.Images.Media.DATE_ADDED,
MediaStore.Images.Media.MIME_TYPE
)
val cursor = context.contentResolver.query(
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Can you add a // TODO: comment - we also use Room in the app, and it supports returning a Flow from a query. Eventually migrating this query to Room and returning the Flow directly would simplify this code, and for now, a TODO comment will serve as a nice breadcrumb trail toward a better overall architecture.

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Thanks for reviewing @psh. I doubt how Room can help fetch images from phone storage.

MediaStore.Images.Media.EXTERNAL_CONTENT_URI, projection,
null, null, MediaStore.Images.Media.DATE_ADDED + " DESC"
)

cursor?.use {
val idColumn = cursor.getColumnIndex(MediaStore.Images.Media._ID)
val nameColumn = cursor.getColumnIndex(MediaStore.Images.Media.DISPLAY_NAME)
val dataColumn = cursor.getColumnIndex(MediaStore.Images.Media.DATA)
val bucketIdColumn = cursor.getColumnIndex(MediaStore.Images.Media.BUCKET_ID)
val bucketNameColumn = cursor.getColumnIndex(MediaStore.Images.Media.BUCKET_DISPLAY_NAME)
val dateColumn = cursor.getColumnIndex(MediaStore.Images.Media.DATE_ADDED)
val mimeTypeColumn = cursor.getColumnIndex(MediaStore.Images.Media.MIME_TYPE)

while(cursor.moveToNext()) {
val id = cursor.getLong(idColumn)
val name = cursor.getString(nameColumn)
val path = cursor.getString(dataColumn)
val bucketId = cursor.getLong(bucketIdColumn)
val bucketName = cursor.getString(bucketNameColumn)
val date = cursor.getLong(dateColumn)
val mimeType = cursor.getString(mimeTypeColumn)

val validMimeTypes = arrayOf(
"image/jpeg", "image/png", "image/svg+xml", "image/gif",
"image/tiff", "image/webp", "image/x-xcf"
)
// Skip the media items with unsupported MIME types
if(mimeType.lowercase() !in validMimeTypes) continue

// URI to access the image
val uri = ContentUris.withAppendedId(
MediaStore.Images.Media.EXTERNAL_CONTENT_URI, id
)

val calendar = Calendar.getInstance()
calendar.timeInMillis = date * 1000L
val calendarDate: Date = calendar.time
val dateFormat = DateFormat.getMediumDateFormat(context)
val formattedDate = dateFormat.format(calendarDate)

emit(Image(id, name, uri, path, bucketId, bucketName, date = formattedDate))
}
}
}.flowOn(Dispatchers.IO)
}
Original file line number Diff line number Diff line change
@@ -0,0 +1,15 @@
package fr.free.nrw.commons.customselector.domain

import fr.free.nrw.commons.customselector.domain.model.Image
import kotlinx.coroutines.flow.Flow

interface ImageRepository {

suspend fun getImagesFromDevice(): Flow<Image>

suspend fun markAsNotForUpload(imageSHA: String)

suspend fun unmarkAsNotForUpload(imageSHA: String)

suspend fun isNotForUpload(imageSHA: String): Boolean
}
Original file line number Diff line number Diff line change
@@ -1,4 +1,4 @@
package fr.free.nrw.commons.customselector.model
package fr.free.nrw.commons.customselector.domain.model

/**
* sealed class Callback Status.
Expand Down
Original file line number Diff line number Diff line change
@@ -1,4 +1,4 @@
package fr.free.nrw.commons.customselector.model
package fr.free.nrw.commons.customselector.domain.model

/**
* Custom selector data class Folder.
Expand Down
Original file line number Diff line number Diff line change
@@ -1,4 +1,4 @@
package fr.free.nrw.commons.customselector.model
package fr.free.nrw.commons.customselector.domain.model

import android.net.Uri
import android.os.Parcel
Expand Down
Original file line number Diff line number Diff line change
@@ -1,4 +1,4 @@
package fr.free.nrw.commons.customselector.model
package fr.free.nrw.commons.customselector.domain.model

/**
* Custom selector data class Result.
Expand Down
Original file line number Diff line number Diff line change
@@ -0,0 +1,89 @@
package fr.free.nrw.commons.customselector.domain.use_case

import android.content.Context
import android.net.Uri
import androidx.exifinterface.media.ExifInterface
import fr.free.nrw.commons.customselector.ui.selector.ImageLoader
import fr.free.nrw.commons.filepicker.PickedFiles
import fr.free.nrw.commons.media.MediaClient
import fr.free.nrw.commons.upload.FileProcessor
import fr.free.nrw.commons.upload.FileUtilsWrapper
import kotlinx.coroutines.CoroutineDispatcher
import kotlinx.coroutines.Dispatchers
import kotlinx.coroutines.withContext
import okio.FileNotFoundException
import timber.log.Timber
import java.io.IOException
import java.net.UnknownHostException
import javax.inject.Inject

class ImageUseCase @Inject constructor(
private val fileUtilsWrapper: FileUtilsWrapper,
private val fileProcessor: FileProcessor,
private val mediaClient: MediaClient,
private val context: Context
) {
private val ioDispatcher: CoroutineDispatcher = Dispatchers.IO

/**
* Retrieves the SHA1 hash of an image from its URI.
*
* @param uri The URI of the image.
* @return The SHA1 hash of the image, or an empty string if the image is not found.
*/
suspend fun getImageSHA1(uri: Uri): String = withContext(ioDispatcher) {
try {
val inputStream = context.contentResolver.openInputStream(uri)
fileUtilsWrapper.getSHA1(inputStream)
} catch (e: FileNotFoundException) {
Timber.e(e)
""
}
}

/**
* Generates a modified SHA1 hash of an image after redacting sensitive EXIF tags.
*
* @param imageUri The URI of the image to process.
* @return The modified SHA1 hash of the image.
*/
suspend fun generateModifiedSHA1(imageUri: Uri): String = withContext(ioDispatcher) {
val uploadableFile = PickedFiles.pickedExistingPicture(context, imageUri)
val exifInterface: ExifInterface? = try {
ExifInterface(uploadableFile.file!!)
} catch (e: IOException) {
Timber.e(e)
null
}
fileProcessor.redactExifTags(exifInterface, fileProcessor.getExifTagsToRedact())

val sha1 = fileUtilsWrapper.getSHA1(
fileUtilsWrapper.getFileInputStream(uploadableFile.filePath))
uploadableFile.file.delete()
sha1
}

/**
* Checks whether a file with the given SHA1 hash exists on Wikimedia Commons.
*
* @param sha1 The SHA1 hash of the file to check.
* @return An ImageLoader.Result indicating the existence of the file on Commons.
*/
suspend fun checkWhetherFileExistsOnCommonsUsingSHA1(
sha1: String
): ImageLoader.Result = withContext(ioDispatcher) {
return@withContext try {
if (mediaClient.checkFileExistsUsingSha(sha1).blockingGet()) {
ImageLoader.Result.TRUE
} else {
ImageLoader.Result.FALSE
}
} catch (e: UnknownHostException) {
Timber.e(e, "Network Connection Error")
ImageLoader.Result.ERROR
} catch (e: Exception) {
e.printStackTrace()
ImageLoader.Result.ERROR
}
}
}
Original file line number Diff line number Diff line change
@@ -1,7 +1,7 @@
package fr.free.nrw.commons.customselector.helper

import fr.free.nrw.commons.customselector.model.Folder
import fr.free.nrw.commons.customselector.model.Image
import fr.free.nrw.commons.customselector.domain.model.Folder
import fr.free.nrw.commons.customselector.domain.model.Image

/**
* Image Helper object, includes all the static functions and variables required by custom selector.
Expand Down
Original file line number Diff line number Diff line change
@@ -1,6 +1,6 @@
package fr.free.nrw.commons.customselector.listeners

import fr.free.nrw.commons.customselector.model.Image
import fr.free.nrw.commons.customselector.domain.model.Image

/**
* Custom Selector Image Loader Listener
Expand Down
Original file line number Diff line number Diff line change
@@ -1,6 +1,6 @@
package fr.free.nrw.commons.customselector.listeners

import fr.free.nrw.commons.customselector.model.Image
import fr.free.nrw.commons.customselector.domain.model.Image

/**
* Custom selector Image select listener
Expand Down
Original file line number Diff line number Diff line change
@@ -1,6 +1,6 @@
package fr.free.nrw.commons.customselector.listeners

import fr.free.nrw.commons.customselector.model.Image
import fr.free.nrw.commons.customselector.domain.model.Image

/**
* Interface to pass data between fragment and activity
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -10,8 +10,8 @@ import androidx.recyclerview.widget.RecyclerView
import com.bumptech.glide.Glide
import fr.free.nrw.commons.R
import fr.free.nrw.commons.customselector.listeners.FolderClickListener
import fr.free.nrw.commons.customselector.model.Folder
import fr.free.nrw.commons.customselector.model.Image
import fr.free.nrw.commons.customselector.domain.model.Folder
import fr.free.nrw.commons.customselector.domain.model.Image

/**
* Custom selector FolderAdapter.
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -17,7 +17,7 @@ import fr.free.nrw.commons.customselector.helper.ImageHelper
import fr.free.nrw.commons.customselector.helper.ImageHelper.CUSTOM_SELECTOR_PREFERENCE_KEY
import fr.free.nrw.commons.customselector.helper.ImageHelper.SHOW_ALREADY_ACTIONED_IMAGES_PREFERENCE_KEY
import fr.free.nrw.commons.customselector.listeners.ImageSelectListener
import fr.free.nrw.commons.customselector.model.Image
import fr.free.nrw.commons.customselector.domain.model.Image
import fr.free.nrw.commons.customselector.ui.selector.ImageLoader
import kotlinx.coroutines.CoroutineDispatcher
import kotlinx.coroutines.CoroutineScope
Expand Down
Loading