diff --git a/app/src/main/java/foundation/e/apps/FdroidFetchViewModel.kt b/app/src/main/java/foundation/e/apps/AppInfoFetchViewModel.kt similarity index 63% rename from app/src/main/java/foundation/e/apps/FdroidFetchViewModel.kt rename to app/src/main/java/foundation/e/apps/AppInfoFetchViewModel.kt index 8b1f24cae936b884d0c4cc629e331817391e7723..b8a9c1e8dc4907a728b6cf2a560c1e6533f16551 100644 --- a/app/src/main/java/foundation/e/apps/FdroidFetchViewModel.kt +++ b/app/src/main/java/foundation/e/apps/AppInfoFetchViewModel.kt @@ -1,13 +1,19 @@ package foundation.e.apps import android.widget.TextView +import androidx.lifecycle.LiveData import androidx.lifecycle.ViewModel +import androidx.lifecycle.liveData import androidx.lifecycle.viewModelScope +import com.aurora.gplayapi.data.models.AuthData +import com.google.gson.Gson import dagger.hilt.android.lifecycle.HiltViewModel import foundation.e.apps.api.fdroid.FdroidRepository import foundation.e.apps.api.fdroid.models.FdroidEntity import foundation.e.apps.api.fused.data.FusedApp +import foundation.e.apps.api.gplay.GPlayAPIRepository import foundation.e.apps.utils.enums.Origin +import foundation.e.apps.utils.modules.DataStoreModule import kotlinx.coroutines.Dispatchers import kotlinx.coroutines.launch import kotlinx.coroutines.withContext @@ -17,8 +23,11 @@ import javax.inject.Inject * */ @HiltViewModel -class FdroidFetchViewModel @Inject constructor( - private val fdroidRepository: FdroidRepository +class AppInfoFetchViewModel @Inject constructor( + private val fdroidRepository: FdroidRepository, + private val gPlayAPIRepository: GPlayAPIRepository, + private val dataStoreModule: DataStoreModule, + private val gson: Gson ) : ViewModel() { private val fdroidEntries = mutableMapOf() @@ -55,4 +64,23 @@ class FdroidFetchViewModel @Inject constructor( } } } + + fun isAppPurchased(app: FusedApp): LiveData { + return liveData { + val authData = gson.fromJson(dataStoreModule.getAuthDataSync(), AuthData::class.java) + try { + gPlayAPIRepository.getDownloadInfo( + app.package_name, + app.latest_version_code, + app.offer_type, + authData + ) + app.isPurchased = true + emit(true) + } catch (e: Exception) { + app.isPurchased = false + emit(false) + } + } + } } diff --git a/app/src/main/java/foundation/e/apps/api/fused/data/FusedApp.kt b/app/src/main/java/foundation/e/apps/api/fused/data/FusedApp.kt index b4f349b6305de180baa2a139fbcc32a762554924..d48829311e7cd8a9517fabd08d35d53fe9919c3c 100644 --- a/app/src/main/java/foundation/e/apps/api/fused/data/FusedApp.kt +++ b/app/src/main/java/foundation/e/apps/api/fused/data/FusedApp.kt @@ -53,6 +53,7 @@ data class FusedApp( val url: String = String(), var type: Type = Type.NATIVE, var privacyScore: Int = -1, + var isPurchased: Boolean = false, /* * List of permissions from Exodus API. diff --git a/app/src/main/java/foundation/e/apps/application/ApplicationFragment.kt b/app/src/main/java/foundation/e/apps/application/ApplicationFragment.kt index 6f8d2aab1b743dfba56d18db47a57cff60a30cea..1c896b79ade6969c5a7dc577c19f21d224a66dc9 100644 --- a/app/src/main/java/foundation/e/apps/application/ApplicationFragment.kt +++ b/app/src/main/java/foundation/e/apps/application/ApplicationFragment.kt @@ -44,7 +44,7 @@ import com.google.android.material.button.MaterialButton import com.google.android.material.snackbar.Snackbar import com.google.android.material.textview.MaterialTextView import dagger.hilt.android.AndroidEntryPoint -import foundation.e.apps.FdroidFetchViewModel +import foundation.e.apps.AppInfoFetchViewModel import foundation.e.apps.MainActivityViewModel import foundation.e.apps.PrivacyInfoViewModel import foundation.e.apps.R @@ -81,7 +81,7 @@ class ApplicationFragment : Fragment(R.layout.fragment_application) { private val applicationViewModel: ApplicationViewModel by viewModels() private val privacyInfoViewModel: PrivacyInfoViewModel by viewModels() - private val fdroidFetchViewModel: FdroidFetchViewModel by viewModels() + private val appInfoFetchViewModel: AppInfoFetchViewModel by viewModels() private val mainActivityViewModel: MainActivityViewModel by activityViewModels() private var applicationIcon: ImageView? = null @@ -147,7 +147,7 @@ class ApplicationFragment : Fragment(R.layout.fragment_application) { applicationIcon = appIcon appName.text = it.name appAuthor.text = it.author - fdroidFetchViewModel.setAuthorNameIfNeeded(appAuthor, it) + appInfoFetchViewModel.setAuthorNameIfNeeded(appAuthor, it) categoryTitle.text = it.category if (args.origin == Origin.CLEANAPK) { appIcon.load(CleanAPKInterface.ASSET_URL + it.icon_image_path) diff --git a/app/src/main/java/foundation/e/apps/applicationlist/ApplicationListFragment.kt b/app/src/main/java/foundation/e/apps/applicationlist/ApplicationListFragment.kt index 57c0eb48baa06fb99efe45c7c95e38520d44ae4e..e0b206aa5e3605f427b93123abd4c4139adc1e77 100644 --- a/app/src/main/java/foundation/e/apps/applicationlist/ApplicationListFragment.kt +++ b/app/src/main/java/foundation/e/apps/applicationlist/ApplicationListFragment.kt @@ -31,8 +31,8 @@ import androidx.navigation.fragment.navArgs import androidx.recyclerview.widget.LinearLayoutManager import androidx.recyclerview.widget.RecyclerView import dagger.hilt.android.AndroidEntryPoint +import foundation.e.apps.AppInfoFetchViewModel import foundation.e.apps.AppProgressViewModel -import foundation.e.apps.FdroidFetchViewModel import foundation.e.apps.MainActivityViewModel import foundation.e.apps.PrivacyInfoViewModel import foundation.e.apps.R @@ -62,7 +62,7 @@ class ApplicationListFragment : Fragment(R.layout.fragment_application_list), Fu private val viewModel: ApplicationListViewModel by viewModels() private val privacyInfoViewModel: PrivacyInfoViewModel by viewModels() - private val fdroidFetchViewModel: FdroidFetchViewModel by viewModels() + private val appInfoFetchViewModel: AppInfoFetchViewModel by viewModels() private val mainActivityViewModel: MainActivityViewModel by activityViewModels() private val appProgressViewModel: AppProgressViewModel by viewModels() @@ -112,7 +112,7 @@ class ApplicationListFragment : Fragment(R.layout.fragment_application_list), Fu ApplicationListRVAdapter( this, privacyInfoViewModel, - fdroidFetchViewModel, + appInfoFetchViewModel, mainActivityViewModel, it, pkgManagerModule, diff --git a/app/src/main/java/foundation/e/apps/applicationlist/model/ApplicationListRVAdapter.kt b/app/src/main/java/foundation/e/apps/applicationlist/model/ApplicationListRVAdapter.kt index 1d6245d746bd489bbd6bb436db25a0e6459fe529..b94d70af43631d0471fe4907f2075b0d05d82311 100644 --- a/app/src/main/java/foundation/e/apps/applicationlist/model/ApplicationListRVAdapter.kt +++ b/app/src/main/java/foundation/e/apps/applicationlist/model/ApplicationListRVAdapter.kt @@ -33,8 +33,9 @@ import coil.load import com.facebook.shimmer.Shimmer import com.facebook.shimmer.Shimmer.Direction.LEFT_TO_RIGHT import com.facebook.shimmer.ShimmerDrawable +import com.google.android.material.button.MaterialButton import com.google.android.material.snackbar.Snackbar -import foundation.e.apps.FdroidFetchViewModel +import foundation.e.apps.AppInfoFetchViewModel import foundation.e.apps.MainActivityViewModel import foundation.e.apps.PrivacyInfoViewModel import foundation.e.apps.R @@ -56,7 +57,7 @@ import javax.inject.Singleton class ApplicationListRVAdapter( private val fusedAPIInterface: FusedAPIInterface, private val privacyInfoViewModel: PrivacyInfoViewModel, - private val fdroidFetchViewModel: FdroidFetchViewModel, + private val appInfoFetchViewModel: AppInfoFetchViewModel, private val mainActivityViewModel: MainActivityViewModel, private val currentDestinationId: Int, private val pkgManagerModule: PkgManagerModule, @@ -127,7 +128,7 @@ class ApplicationListRVAdapter( } appTitle.text = searchApp.name appAuthor.text = searchApp.author - fdroidFetchViewModel.setAuthorNameIfNeeded(appAuthor, searchApp) + appInfoFetchViewModel.setAuthorNameIfNeeded(appAuthor, searchApp) if (searchApp.ratings.usageQualityScore != -1.0) { appRating.text = searchApp.ratings.usageQualityScore.toString() appRatingBar.rating = searchApp.ratings.usageQualityScore.toFloat() @@ -202,6 +203,7 @@ class ApplicationListRVAdapter( installApplication(searchApp, appIcon) } } + progressBarInstall.visibility = View.GONE } private fun ApplicationListItemBinding.handleBlocked(view: View) { @@ -218,6 +220,7 @@ class ApplicationListRVAdapter( } } } + progressBarInstall.visibility = View.GONE } private fun ApplicationListItemBinding.showCalculatedPrivacyScoreData( @@ -274,11 +277,13 @@ class ApplicationListRVAdapter( private fun ApplicationListItemBinding.handleInstalling(view: View, holder: ViewHolder) { installButton.apply { isEnabled = false + text = context.getString(R.string.installing) setTextColor(context.getColor(R.color.light_grey)) backgroundTintList = ContextCompat.getColorStateList(view.context, android.R.color.transparent) strokeColor = ContextCompat.getColorStateList(view.context, R.color.light_grey) } + progressBarInstall.visibility = View.GONE } private fun ApplicationListItemBinding.handleDownloading( @@ -295,7 +300,9 @@ class ApplicationListRVAdapter( setOnClickListener { cancelDownload(searchApp) } + progressBarInstall.visibility = View.GONE } + progressBarInstall.visibility = View.GONE } private fun ApplicationListItemBinding.handleUnavailable( @@ -303,13 +310,7 @@ class ApplicationListRVAdapter( searchApp: FusedApp, ) { installButton.apply { - isEnabled = true - text = when { - mainActivityViewModel.checkUnsupportedApplication(searchApp) -> - context.getString(R.string.not_available) - searchApp.isFree -> context.getString(R.string.install) - else -> searchApp.price - } + updateUIByPaymentType(searchApp, this, this@handleUnavailable) setTextColor(context.getColor(R.color.colorAccent)) backgroundTintList = ContextCompat.getColorStateList(view.context, android.R.color.transparent) @@ -318,7 +319,7 @@ class ApplicationListRVAdapter( if (mainActivityViewModel.checkUnsupportedApplication(searchApp, context)) { return@setOnClickListener } - if (searchApp.isFree) { + if (searchApp.isFree || searchApp.isPurchased) { installApplication(searchApp, appIcon) } else { paidAppHandler?.invoke(searchApp) @@ -327,6 +328,35 @@ class ApplicationListRVAdapter( } } + private fun updateUIByPaymentType( + searchApp: FusedApp, + materialButton: MaterialButton, + applicationListItemBinding: ApplicationListItemBinding + ) { + when { + mainActivityViewModel.checkUnsupportedApplication(searchApp) -> { + materialButton.isEnabled = true + materialButton.text = materialButton.context.getString(R.string.not_available) + } + searchApp.isFree -> { + materialButton.isEnabled = true + materialButton.text = materialButton.context.getString(R.string.install) + applicationListItemBinding.progressBarInstall.visibility = View.GONE + } + else -> { + materialButton.isEnabled = false + materialButton.text = "" + applicationListItemBinding.progressBarInstall.visibility = View.VISIBLE + appInfoFetchViewModel.isAppPurchased(searchApp).observe(lifecycleOwner) { + materialButton.isEnabled = true + applicationListItemBinding.progressBarInstall.visibility = View.GONE + materialButton.text = + if (it) materialButton.context.getString(R.string.install) else searchApp.price + } + } + } + } + private fun ApplicationListItemBinding.handleUpdatable( view: View, searchApp: FusedApp @@ -346,6 +376,7 @@ class ApplicationListRVAdapter( installApplication(searchApp, appIcon) } } + progressBarInstall.visibility = View.GONE } private fun ApplicationListItemBinding.handleInstalled( @@ -366,6 +397,7 @@ class ApplicationListRVAdapter( } } } + progressBarInstall.visibility = View.GONE } fun setData(newList: List) { diff --git a/app/src/main/java/foundation/e/apps/home/HomeFragment.kt b/app/src/main/java/foundation/e/apps/home/HomeFragment.kt index 8fc6189389d3c317e535502cdcb0210c29539e22..d4f0dfb882dcef487501775e7cef8f402cbbe659 100644 --- a/app/src/main/java/foundation/e/apps/home/HomeFragment.kt +++ b/app/src/main/java/foundation/e/apps/home/HomeFragment.kt @@ -29,6 +29,7 @@ import androidx.navigation.findNavController import androidx.recyclerview.widget.LinearLayoutManager import androidx.recyclerview.widget.RecyclerView import dagger.hilt.android.AndroidEntryPoint +import foundation.e.apps.AppInfoFetchViewModel import foundation.e.apps.AppProgressViewModel import foundation.e.apps.MainActivityViewModel import foundation.e.apps.R @@ -56,6 +57,7 @@ class HomeFragment : Fragment(R.layout.fragment_home), FusedAPIInterface { private val homeViewModel: HomeViewModel by viewModels() private val mainActivityViewModel: MainActivityViewModel by activityViewModels() private val appProgressViewModel: AppProgressViewModel by viewModels() + private val appInfoFetchViewModel: AppInfoFetchViewModel by viewModels() @Inject lateinit var pkgManagerModule: PkgManagerModule @@ -90,7 +92,7 @@ class HomeFragment : Fragment(R.layout.fragment_home), FusedAPIInterface { pkgManagerModule, pwaManagerModule, User.valueOf(mainActivityViewModel.userType.value ?: User.UNAVAILABLE.name), - mainActivityViewModel, viewLifecycleOwner + mainActivityViewModel, appInfoFetchViewModel, viewLifecycleOwner ) { fusedApp -> if (!mainActivityViewModel.shouldShowPaidAppsSnackBar(fusedApp)) { ApplicationDialogFragment( diff --git a/app/src/main/java/foundation/e/apps/home/model/HomeChildRVAdapter.kt b/app/src/main/java/foundation/e/apps/home/model/HomeChildRVAdapter.kt index 6d3a3d35bb0255e2a1ec34f030d32ac8bc85b7dc..59fe1b90827e7abf4862a40e305e437e149d378b 100644 --- a/app/src/main/java/foundation/e/apps/home/model/HomeChildRVAdapter.kt +++ b/app/src/main/java/foundation/e/apps/home/model/HomeChildRVAdapter.kt @@ -20,16 +20,20 @@ package foundation.e.apps.home.model import android.graphics.Color import android.view.LayoutInflater +import android.view.View import android.view.ViewGroup import android.widget.ImageView import androidx.core.content.ContextCompat +import androidx.lifecycle.LifecycleOwner import androidx.navigation.findNavController import androidx.recyclerview.widget.ListAdapter import androidx.recyclerview.widget.RecyclerView import coil.load import com.facebook.shimmer.Shimmer import com.facebook.shimmer.ShimmerDrawable +import com.google.android.material.button.MaterialButton import com.google.android.material.snackbar.Snackbar +import foundation.e.apps.AppInfoFetchViewModel import foundation.e.apps.MainActivityViewModel import foundation.e.apps.R import foundation.e.apps.api.cleanapk.CleanAPKInterface @@ -47,8 +51,10 @@ class HomeChildRVAdapter( private val fusedAPIInterface: FusedAPIInterface, private val pkgManagerModule: PkgManagerModule, private val pwaManagerModule: PWAManagerModule, + private val appInfoFetchViewModel: AppInfoFetchViewModel, private val mainActivityViewModel: MainActivityViewModel, private val user: User, + private val lifecycleOwner: LifecycleOwner, private val paidAppHandler: ((FusedApp) -> Unit)? = null ) : ListAdapter(HomeChildFusedAppDiffUtil()) { @@ -101,122 +107,197 @@ class HomeChildRVAdapter( when (homeApp.status) { Status.INSTALLED -> { - installButton.apply { - isEnabled = true - text = context.getString(R.string.open) - setTextColor(Color.WHITE) - backgroundTintList = - ContextCompat.getColorStateList(view.context, R.color.colorAccent) - strokeColor = - ContextCompat.getColorStateList(view.context, R.color.colorAccent) - setOnClickListener { - if (homeApp.is_pwa) { - pwaManagerModule.launchPwa(homeApp) - } else { - context.startActivity(pkgManagerModule.getLaunchIntent(homeApp.package_name)) - } - } - } + handleInstalled(view, homeApp) } Status.UPDATABLE -> { - installButton.apply { - text = if (mainActivityViewModel.checkUnsupportedApplication(homeApp)) - context.getString(R.string.not_available) - else context.getString(R.string.update) - setTextColor(Color.WHITE) - backgroundTintList = - ContextCompat.getColorStateList(view.context, R.color.colorAccent) - strokeColor = - ContextCompat.getColorStateList(view.context, R.color.colorAccent) - setOnClickListener { - if (mainActivityViewModel.checkUnsupportedApplication(homeApp, context)) { - return@setOnClickListener - } - installApplication(homeApp, appIcon) - } - } + handleUpdatable(view, homeApp) } Status.UNAVAILABLE -> { - installButton.apply { - text = when { - mainActivityViewModel.checkUnsupportedApplication(homeApp) -> - context.getString(R.string.not_available) - homeApp.isFree -> context.getString(R.string.install) - else -> homeApp.price - } - setTextColor(context.getColor(R.color.colorAccent)) - backgroundTintList = ContextCompat.getColorStateList( - view.context, - android.R.color.transparent - ) - strokeColor = - ContextCompat.getColorStateList(view.context, R.color.colorAccent) - setOnClickListener { - if (mainActivityViewModel.checkUnsupportedApplication(homeApp, context)) { - return@setOnClickListener - } - if (homeApp.isFree) { - installApplication(homeApp, appIcon) - } else { - paidAppHandler?.invoke(homeApp) - } - } - } + handleUnavailable(homeApp, holder, view) } Status.QUEUED, Status.AWAITING, Status.DOWNLOADING -> { - installButton.apply { - text = context.getString(R.string.cancel) - setTextColor(context.getColor(R.color.colorAccent)) - backgroundTintList = ContextCompat.getColorStateList( - view.context, - android.R.color.transparent - ) - strokeColor = - ContextCompat.getColorStateList(view.context, R.color.colorAccent) - - setOnClickListener { - cancelDownload(homeApp) - } - } + handleQueued(view, homeApp) } Status.INSTALLING, Status.UNINSTALLING -> { - installButton.apply { - isEnabled = false - setTextColor(context.getColor(R.color.light_grey)) - backgroundTintList = ContextCompat.getColorStateList( - view.context, - android.R.color.transparent - ) - strokeColor = - ContextCompat.getColorStateList(view.context, R.color.light_grey) - } + handleInstalling(view) } Status.BLOCKED -> { - installButton.setOnClickListener { - val errorMsg = when (user) { - User.ANONYMOUS, - User.UNAVAILABLE -> view.context.getString(R.string.install_blocked_anonymous) - User.GOOGLE -> view.context.getString(R.string.install_blocked_google) - } - if (errorMsg.isNotBlank()) { - Snackbar.make(view, errorMsg, Snackbar.LENGTH_SHORT).show() - } - } + handleBlocked(view) } Status.INSTALLATION_ISSUE -> { - installButton.apply { - text = view.context.getString(R.string.retry) - setTextColor(context.getColor(R.color.colorAccent)) - backgroundTintList = ContextCompat.getColorStateList( - view.context, - android.R.color.transparent - ) - strokeColor = - ContextCompat.getColorStateList(view.context, R.color.colorAccent) - setOnClickListener { - installApplication(homeApp, appIcon) - } - } + handleInstallationIssue(view, homeApp) + } + } + } + } + + private fun HomeChildListItemBinding.handleInstallationIssue( + view: View, + homeApp: FusedApp + ) { + installButton.apply { + text = view.context.getString(R.string.retry) + setTextColor(context.getColor(R.color.colorAccent)) + backgroundTintList = ContextCompat.getColorStateList( + view.context, + android.R.color.transparent + ) + strokeColor = + ContextCompat.getColorStateList(view.context, R.color.colorAccent) + setOnClickListener { + installApplication(homeApp, appIcon) + } + } + progressBarInstall.visibility = View.GONE + } + + private fun HomeChildListItemBinding.handleBlocked(view: View) { + installButton.setOnClickListener { + val errorMsg = when (user) { + User.ANONYMOUS, + User.UNAVAILABLE -> view.context.getString(R.string.install_blocked_anonymous) + User.GOOGLE -> view.context.getString(R.string.install_blocked_google) + } + if (errorMsg.isNotBlank()) { + Snackbar.make(view, errorMsg, Snackbar.LENGTH_SHORT).show() + } + } + progressBarInstall.visibility = View.GONE + } + + private fun HomeChildListItemBinding.handleInstalling(view: View) { + installButton.apply { + isEnabled = false + setTextColor(context.getColor(R.color.light_grey)) + text = context.getString(R.string.installing) + backgroundTintList = ContextCompat.getColorStateList( + view.context, + android.R.color.transparent + ) + strokeColor = + ContextCompat.getColorStateList(view.context, R.color.light_grey) + } + progressBarInstall.visibility = View.GONE + } + + private fun HomeChildListItemBinding.handleQueued( + view: View, + homeApp: FusedApp + ) { + installButton.apply { + text = context.getString(R.string.cancel) + setTextColor(context.getColor(R.color.colorAccent)) + backgroundTintList = ContextCompat.getColorStateList( + view.context, + android.R.color.transparent + ) + strokeColor = + ContextCompat.getColorStateList(view.context, R.color.colorAccent) + + setOnClickListener { + cancelDownload(homeApp) + } + } + progressBarInstall.visibility = View.GONE + } + + private fun HomeChildListItemBinding.handleUnavailable( + homeApp: FusedApp, + holder: ViewHolder, + view: View + ) { + installButton.apply { + updateUIByPaymentType(homeApp, this, holder.binding) + setTextColor(context.getColor(R.color.colorAccent)) + backgroundTintList = ContextCompat.getColorStateList( + view.context, + android.R.color.transparent + ) + strokeColor = + ContextCompat.getColorStateList(view.context, R.color.colorAccent) + setOnClickListener { + if (mainActivityViewModel.checkUnsupportedApplication(homeApp, context)) { + return@setOnClickListener + } + if (homeApp.isFree) { + installApplication(homeApp, appIcon) + } else { + paidAppHandler?.invoke(homeApp) + } + } + } + } + + private fun HomeChildListItemBinding.handleUpdatable( + view: View, + homeApp: FusedApp + ) { + installButton.apply { + text = if (mainActivityViewModel.checkUnsupportedApplication(homeApp)) + context.getString(R.string.not_available) + else context.getString(R.string.update) + setTextColor(Color.WHITE) + backgroundTintList = + ContextCompat.getColorStateList(view.context, R.color.colorAccent) + strokeColor = + ContextCompat.getColorStateList(view.context, R.color.colorAccent) + setOnClickListener { + if (mainActivityViewModel.checkUnsupportedApplication(homeApp, context)) { + return@setOnClickListener + } + installApplication(homeApp, appIcon) + } + } + progressBarInstall.visibility = View.GONE + } + + private fun HomeChildListItemBinding.handleInstalled( + view: View, + homeApp: FusedApp + ) { + installButton.apply { + isEnabled = true + text = context.getString(R.string.open) + setTextColor(Color.WHITE) + backgroundTintList = + ContextCompat.getColorStateList(view.context, R.color.colorAccent) + strokeColor = + ContextCompat.getColorStateList(view.context, R.color.colorAccent) + setOnClickListener { + if (homeApp.is_pwa) { + pwaManagerModule.launchPwa(homeApp) + } else { + context.startActivity(pkgManagerModule.getLaunchIntent(homeApp.package_name)) + } + } + } + progressBarInstall.visibility = View.GONE + } + + private fun updateUIByPaymentType( + homeApp: FusedApp, + materialButton: MaterialButton, + homeChildListItemBinding: HomeChildListItemBinding + ) { + when { + mainActivityViewModel.checkUnsupportedApplication(homeApp) -> { + materialButton.isEnabled = true + materialButton.text = materialButton.context.getString(R.string.not_available) + } + homeApp.isFree -> { + materialButton.isEnabled = true + materialButton.text = materialButton.context.getString(R.string.install) + homeChildListItemBinding.progressBarInstall.visibility = View.GONE + } + else -> { + materialButton.isEnabled = false + materialButton.text = "" + homeChildListItemBinding.progressBarInstall.visibility = View.VISIBLE + appInfoFetchViewModel.isAppPurchased(homeApp).observe(lifecycleOwner) { + materialButton.isEnabled = true + homeChildListItemBinding.progressBarInstall.visibility = View.GONE + materialButton.text = + if (it) materialButton.context.getString(R.string.install) else homeApp.price } } } diff --git a/app/src/main/java/foundation/e/apps/home/model/HomeParentRVAdapter.kt b/app/src/main/java/foundation/e/apps/home/model/HomeParentRVAdapter.kt index e400a7cb6062fe6f93c84be7d6f91ebaa3c8cd62..7e81364fa73ae51263a5e17ba92fb6ae95a6c7b5 100644 --- a/app/src/main/java/foundation/e/apps/home/model/HomeParentRVAdapter.kt +++ b/app/src/main/java/foundation/e/apps/home/model/HomeParentRVAdapter.kt @@ -24,6 +24,7 @@ import androidx.lifecycle.LifecycleOwner import androidx.recyclerview.widget.LinearLayoutManager import androidx.recyclerview.widget.ListAdapter import androidx.recyclerview.widget.RecyclerView +import foundation.e.apps.AppInfoFetchViewModel import foundation.e.apps.MainActivityViewModel import foundation.e.apps.api.fused.FusedAPIInterface import foundation.e.apps.api.fused.data.FusedApp @@ -39,6 +40,7 @@ class HomeParentRVAdapter( private val pwaManagerModule: PWAManagerModule, private val user: User, private val mainActivityViewModel: MainActivityViewModel, + private val appInfoFetchViewModel: AppInfoFetchViewModel, private val lifecycleOwner: LifecycleOwner, private val paidAppHandler: ((FusedApp) -> Unit)? = null ) : ListAdapter(FusedHomeDiffUtil()) { @@ -62,8 +64,10 @@ class HomeParentRVAdapter( fusedAPIInterface, pkgManagerModule, pwaManagerModule, + appInfoFetchViewModel, mainActivityViewModel, user, + lifecycleOwner, paidAppHandler ) homeChildRVAdapter.setData(fusedHome.list) diff --git a/app/src/main/java/foundation/e/apps/search/SearchFragment.kt b/app/src/main/java/foundation/e/apps/search/SearchFragment.kt index 74af37fbf253b0ad419e4851e887bfdf11256620..3791b0db0444e3b50ba796ed558c428060c697eb 100644 --- a/app/src/main/java/foundation/e/apps/search/SearchFragment.kt +++ b/app/src/main/java/foundation/e/apps/search/SearchFragment.kt @@ -39,8 +39,8 @@ import androidx.recyclerview.widget.RecyclerView import com.aurora.gplayapi.SearchSuggestEntry import com.facebook.shimmer.ShimmerFrameLayout import dagger.hilt.android.AndroidEntryPoint +import foundation.e.apps.AppInfoFetchViewModel import foundation.e.apps.AppProgressViewModel -import foundation.e.apps.FdroidFetchViewModel import foundation.e.apps.MainActivityViewModel import foundation.e.apps.PrivacyInfoViewModel import foundation.e.apps.R @@ -74,7 +74,7 @@ class SearchFragment : private val searchViewModel: SearchViewModel by viewModels() private val privacyInfoViewModel: PrivacyInfoViewModel by viewModels() - private val fdroidFetchViewModel: FdroidFetchViewModel by viewModels() + private val appInfoFetchViewModel: AppInfoFetchViewModel by viewModels() private val mainActivityViewModel: MainActivityViewModel by activityViewModels() private val appProgressViewModel: AppProgressViewModel by viewModels() @@ -120,7 +120,7 @@ class SearchFragment : ApplicationListRVAdapter( this, privacyInfoViewModel, - fdroidFetchViewModel, + appInfoFetchViewModel, mainActivityViewModel, it, pkgManagerModule, diff --git a/app/src/main/java/foundation/e/apps/updates/UpdatesFragment.kt b/app/src/main/java/foundation/e/apps/updates/UpdatesFragment.kt index 32165f0b4d62d19581afc862452089e74dfee7b7..cd6116e1eb5801805e3aa02bb120045af2172960 100644 --- a/app/src/main/java/foundation/e/apps/updates/UpdatesFragment.kt +++ b/app/src/main/java/foundation/e/apps/updates/UpdatesFragment.kt @@ -29,8 +29,8 @@ import androidx.navigation.fragment.findNavController import androidx.recyclerview.widget.LinearLayoutManager import androidx.recyclerview.widget.RecyclerView import dagger.hilt.android.AndroidEntryPoint +import foundation.e.apps.AppInfoFetchViewModel import foundation.e.apps.AppProgressViewModel -import foundation.e.apps.FdroidFetchViewModel import foundation.e.apps.MainActivityViewModel import foundation.e.apps.PrivacyInfoViewModel import foundation.e.apps.R @@ -61,7 +61,7 @@ class UpdatesFragment : Fragment(R.layout.fragment_updates), FusedAPIInterface { private val updatesViewModel: UpdatesViewModel by viewModels() private val privacyInfoViewModel: PrivacyInfoViewModel by viewModels() - private val fdroidFetchViewModel: FdroidFetchViewModel by viewModels() + private val appInfoFetchViewModel: AppInfoFetchViewModel by viewModels() private val mainActivityViewModel: MainActivityViewModel by activityViewModels() private val appProgressViewModel: AppProgressViewModel by viewModels() @@ -91,7 +91,7 @@ class UpdatesFragment : Fragment(R.layout.fragment_updates), FusedAPIInterface { ApplicationListRVAdapter( this, privacyInfoViewModel, - fdroidFetchViewModel, + appInfoFetchViewModel, mainActivityViewModel, it, pkgManagerModule, diff --git a/app/src/main/res/layout/application_list_item.xml b/app/src/main/res/layout/application_list_item.xml index 874abbe4db4ecb602d45c97e94ae0c50a0e3dcc1..eb8ad1af61459a009d972aed157817dac9e8ef3f 100644 --- a/app/src/main/res/layout/application_list_item.xml +++ b/app/src/main/res/layout/application_list_item.xml @@ -103,82 +103,91 @@ app:layout_constraintBottom_toBottomOf="@+id/app_rating_bar" app:layout_constraintLeft_toRightOf="@+id/app_rating_bar" /> - - - - - - + + android:layout_height="16dp" + app:layout_constraintTop_toTopOf="@+id/installButton" + app:layout_constraintBottom_toBottomOf="@+id/installButton" + app:layout_constraintLeft_toLeftOf="@+id/installButton" + app:layout_constraintRight_toRightOf="@+id/installButton"/> - + + + + + + + + + - - - + + diff --git a/app/src/main/res/layout/home_child_list_item.xml b/app/src/main/res/layout/home_child_list_item.xml index 4472644222a080a0f5bf68a9a4e1376a7122c0d3..548f48ac46720bfc2d2331bc783c0a99cf0a0ecb 100644 --- a/app/src/main/res/layout/home_child_list_item.xml +++ b/app/src/main/res/layout/home_child_list_item.xml @@ -57,16 +57,27 @@ android:textColor="?android:textColorPrimary" android:textSize="12sp" /> - + android:layout_height="wrap_content"> + + + \ No newline at end of file diff --git a/app/src/main/res/values/strings.xml b/app/src/main/res/values/strings.xml index 695f6cf5034e7a28d257c0e9234ffac8b4dbb3b6..fb2863aae72c4a2467ce86aee75621882ecbf62b 100644 --- a/app/src/main/res/values/strings.xml +++ b/app/src/main/res/values/strings.xml @@ -74,6 +74,7 @@ Check out \"%1$s\"\n%2$s OK Install + Installing Cancel App icon N/A