Skip to content

Commit

Permalink
Merge pull request #3 from maifeeulasad/develop
Browse files Browse the repository at this point in the history
  • Loading branch information
maifeeulasad authored Mar 19, 2021
2 parents 598d24a + cfca5d0 commit f3cb485
Show file tree
Hide file tree
Showing 15 changed files with 405 additions and 109 deletions.
15 changes: 3 additions & 12 deletions .idea/codeStyles/Project.xml

Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.

3 changes: 3 additions & 0 deletions app/build.gradle
Original file line number Diff line number Diff line change
Expand Up @@ -96,4 +96,7 @@ dependencies {
//implementation "com.google.dagger:hilt-android-testing:$hilt_ver"
implementation "androidx.hilt:hilt-common:1.0.0-alpha03"
implementation "androidx.hilt:hilt-lifecycle-viewmodel:1.0.0-alpha03"

//Bio-metric
implementation 'androidx.biometric:biometric:1.2.0-alpha03'
}
3 changes: 2 additions & 1 deletion app/src/main/AndroidManifest.xml
Original file line number Diff line number Diff line change
Expand Up @@ -3,7 +3,8 @@
package="dev.spikeysanju.expensetracker">

<uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE" />

<uses-permission android:name="android.permission.USE_BIOMETRIC"
android:requiredFeature="true"/>

<application
android:name=".app.ExpenseTracker"
Expand Down
Original file line number Diff line number Diff line change
@@ -0,0 +1,42 @@
package dev.spikeysanju.expensetracker.data.local.datastore

import android.content.Context
import androidx.datastore.preferences.core.booleanPreferencesKey
import androidx.datastore.preferences.core.edit
import kotlinx.coroutines.flow.Flow
import kotlinx.coroutines.flow.map
import javax.inject.Singleton


class SettingsDataStore(context: Context) :
PrefsDataStore(
context,
PREF_FILE_SETTINGS
),
SettingsImpl {

// used to get the data from datastore
override val biometric: Flow<Boolean>
get() = dataStore.data.map { preferences ->
val biometric = preferences[BIOMETRIC_KEY] ?: false
biometric
}

// used to save the bio-metric preference to datastore
override suspend fun saveToDataStore(isBiometricEnabled: Boolean) {
dataStore.edit { preferences ->
preferences[BIOMETRIC_KEY] = isBiometricEnabled
}
}

companion object {
private const val PREF_FILE_SETTINGS = "settings_preference"
private val BIOMETRIC_KEY = booleanPreferencesKey("biometric_mode")
}
}

@Singleton
interface SettingsImpl {
val biometric: Flow<Boolean>
suspend fun saveToDataStore(isBiometricEnabled: Boolean)
}
Original file line number Diff line number Diff line change
Expand Up @@ -7,6 +7,7 @@ import android.widget.ArrayAdapter
import android.widget.Spinner
import android.widget.TextView
import androidx.appcompat.app.AppCompatDelegate
import androidx.core.widget.NestedScrollView
import androidx.fragment.app.activityViewModels
import androidx.lifecycle.lifecycleScope
import androidx.navigation.fragment.findNavController
Expand All @@ -27,6 +28,7 @@ import indianRupee
import kotlinx.coroutines.flow.collect
import kotlinx.coroutines.flow.first
import show
import kotlin.math.abs

@AndroidEntryPoint
class DashboardFragment :
Expand Down Expand Up @@ -183,6 +185,17 @@ class DashboardFragment :
findNavController().navigate(R.id.action_dashboardFragment_to_addTransactionFragment)
}

mainDashboardScrollView.setOnScrollChangeListener(
NestedScrollView.OnScrollChangeListener { _, sX, sY, oX, oY ->
if (abs(sY - oY) > 10) {
when {
sY > oY -> btnAddTransaction.hide()
oY > sY -> btnAddTransaction.show()
}
}
}
)

transactionAdapter.setOnItemClickListener {
val bundle = Bundle().apply {
putSerializable("transaction", it)
Expand Down Expand Up @@ -267,6 +280,10 @@ class DashboardFragment :
findNavController().navigate(R.id.action_dashboardFragment_to_aboutFragment)
true
}
R.id.action_settings -> {
findNavController().navigate(R.id.action_dashboardFragment_to_settingsFragment)
true
}
else -> super.onOptionsItemSelected(item)
}
}
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -23,8 +23,10 @@ import dev.spikeysanju.expensetracker.utils.saveBitmap
import dev.spikeysanju.expensetracker.utils.viewState.DetailState
import dev.spikeysanju.expensetracker.view.base.BaseFragment
import dev.spikeysanju.expensetracker.view.main.viewmodel.TransactionViewModel
import hide
import indianRupee
import kotlinx.coroutines.flow.collect
import show

@AndroidEntryPoint
class TransactionDetailsFragment : BaseFragment<FragmentTransactionDetailsBinding, TransactionViewModel>() {
Expand Down Expand Up @@ -126,7 +128,10 @@ class TransactionDetailsFragment : BaseFragment<FragmentTransactionDetailsBindin
return
}

// unHide the app logo and name
showAppNameAndLogo()
val imageURI = binding.transactionDetails.detailView.drawToBitmap().let { bitmap ->
hideAppNameAndLogo()
saveBitmap(requireActivity(), bitmap)
} ?: run {
toast("Error occurred!")
Expand All @@ -141,6 +146,16 @@ class TransactionDetailsFragment : BaseFragment<FragmentTransactionDetailsBindin
startActivity(Intent.createChooser(intent, null))
}

private fun showAppNameAndLogo() = with(binding.transactionDetails) {
appIconForShare.show()
appNameForShare.show()
}

private fun hideAppNameAndLogo() = with(binding.transactionDetails) {
appIconForShare.hide()
appNameForShare.hide()
}

private fun isStoragePermissionGranted(): Boolean = ContextCompat.checkSelfPermission(
requireContext(),
Manifest.permission.WRITE_EXTERNAL_STORAGE
Expand Down
Original file line number Diff line number Diff line change
@@ -1,8 +1,12 @@
package dev.spikeysanju.expensetracker.view.main

import android.os.Bundle
import android.widget.Toast
import androidx.activity.viewModels
import androidx.appcompat.app.AppCompatActivity
import androidx.biometric.BiometricPrompt
import androidx.core.content.ContextCompat
import androidx.lifecycle.lifecycleScope
import androidx.navigation.NavController
import androidx.navigation.fragment.NavHostFragment
import androidx.navigation.ui.AppBarConfiguration
Expand All @@ -14,17 +18,29 @@ import dev.spikeysanju.expensetracker.databinding.ActivityMainBinding
import dev.spikeysanju.expensetracker.repo.TransactionRepo
import dev.spikeysanju.expensetracker.utils.viewModelFactory
import dev.spikeysanju.expensetracker.view.main.viewmodel.TransactionViewModel
import dev.spikeysanju.expensetracker.view.settings.SettingsViewModel
import kotlinx.coroutines.flow.first
import java.util.concurrent.Executor
import kotlin.system.exitProcess

@AndroidEntryPoint
class MainActivity : AppCompatActivity() {
private lateinit var navHostFragment: NavHostFragment
private lateinit var appBarConfiguration: AppBarConfiguration

private lateinit var executor: Executor
private lateinit var biometricPrompt: BiometricPrompt
private lateinit var promptInfo: BiometricPrompt.PromptInfo

private val repo by lazy { TransactionRepo(AppDatabase(this)) }
private val viewModel: TransactionViewModel by viewModels {
viewModelFactory { TransactionViewModel(this.application, repo) }
}

private val settingsViewModel: SettingsViewModel by viewModels{
viewModelFactory { SettingsViewModel(this.application) }
}

override fun onCreate(savedInstanceState: Bundle?) {
super.onCreate(savedInstanceState)
val binding = ActivityMainBinding.inflate(layoutInflater)
Expand All @@ -36,6 +52,11 @@ class MainActivity : AppCompatActivity() {
*/
viewModel

lifecycleScope.launchWhenStarted {
if(settingsViewModel.bioMetricPreference.first())
authenticate()
}

initViews(binding)
observeNavElements(binding, navHostFragment.navController)
}
Expand Down Expand Up @@ -79,4 +100,38 @@ class MainActivity : AppCompatActivity() {
navHostFragment.navController.navigateUp()
return super.onSupportNavigateUp()
}

private fun authenticate(){
executor = ContextCompat.getMainExecutor(this)
biometricPrompt = BiometricPrompt(this, executor,
object : BiometricPrompt.AuthenticationCallback() {
override fun onAuthenticationSucceeded(
result: BiometricPrompt.AuthenticationResult) {
super.onAuthenticationSucceeded(result)
Toast.makeText(applicationContext,
"Welcome to Expenso", Toast.LENGTH_LONG)
.show()
}

override fun onAuthenticationFailed() {
super.onAuthenticationFailed()
//todo: show message wait for a second; static err code
exitProcess(-1)
}

override fun onAuthenticationError(errorCode: Int,
errString: CharSequence) {
super.onAuthenticationError(errorCode, errString)
//todo: show message wait for a second; static err code
exitProcess(-2)
}
})

promptInfo = BiometricPrompt.PromptInfo.Builder()
.setTitle("Please complete bio-metric login to continue")
.setNegativeButtonText("Exit")
.build()
biometricPrompt.authenticate(promptInfo)
}

}
Original file line number Diff line number Diff line change
@@ -0,0 +1,38 @@
package dev.spikeysanju.expensetracker.view.settings


import android.os.Bundle
import android.view.LayoutInflater
import android.view.View
import android.view.ViewGroup
import androidx.fragment.app.viewModels
import androidx.lifecycle.lifecycleScope
import dagger.hilt.android.AndroidEntryPoint
import dev.spikeysanju.expensetracker.databinding.FragmentSettingsBinding
import dev.spikeysanju.expensetracker.view.base.BaseFragment
import kotlinx.coroutines.flow.first

@AndroidEntryPoint
class SettingsFragment : BaseFragment<FragmentSettingsBinding, SettingsViewModel>() {
override val viewModel: SettingsViewModel by viewModels()

override fun onViewCreated(view: View, savedInstanceState: Bundle?) {
super.onViewCreated(view, savedInstanceState)

// Set the item state
lifecycleScope.launchWhenStarted {
binding.biometric.isChecked = viewModel.bioMetricPreference.first()
}

initViews()
}

private fun initViews() = with(binding) {
biometric.setOnCheckedChangeListener { _, biometricEnabled ->
viewModel.setBioMetricLock(biometricEnabled)
}
}

override fun getViewBinding(inflater: LayoutInflater, container: ViewGroup?) =
FragmentSettingsBinding.inflate(inflater, container, false)
}
Original file line number Diff line number Diff line change
@@ -0,0 +1,31 @@
package dev.spikeysanju.expensetracker.view.settings


import android.app.Application
import androidx.lifecycle.AndroidViewModel
import androidx.lifecycle.viewModelScope
import dagger.hilt.android.lifecycle.HiltViewModel
import dev.spikeysanju.expensetracker.data.local.datastore.SettingsDataStore
import kotlinx.coroutines.Dispatchers
import kotlinx.coroutines.launch
import javax.inject.Inject

@HiltViewModel
class SettingsViewModel @Inject constructor(
application: Application
) :
AndroidViewModel(application) {

// init datastore
private val settingsDataStore = SettingsDataStore(application)

// get bio-metric preference
val bioMetricPreference = settingsDataStore.biometric

fun setBioMetricLock(bioMetricLock: Boolean) {
viewModelScope.launch(Dispatchers.IO) {
settingsDataStore.saveToDataStore(bioMetricLock)
}
}

}
24 changes: 24 additions & 0 deletions app/src/main/res/layout/content_transaction_details.xml
Original file line number Diff line number Diff line change
Expand Up @@ -193,6 +193,30 @@
app:layout_constraintTop_toBottomOf="@id/caption_createdAt"
tools:text="Sunday, 18 Dec 2021" />

<com.google.android.material.imageview.ShapeableImageView
android:id="@+id/app_icon_for_share"
app:srcCompat="@drawable/ic_logo"
android:visibility="invisible"
app:layout_constraintEnd_toStartOf="@+id/app_name_for_share"
app:layout_constraintBottom_toBottomOf="parent"
android:layout_width="@dimen/dimen_64"
android:layout_height="@dimen/dimen_64" />

<TextView
android:id="@+id/app_name_for_share"
android:padding="@dimen/dimen_16"
android:gravity="center"
android:visibility="invisible"
android:text="@string/app_name"
app:layout_constraintEnd_toEndOf="parent"
android:layout_marginStart="@dimen/dimen_16"
android:fontFamily="@font/open_sans_regular"
app:layout_constraintTop_toTopOf="@id/app_icon_for_share"
android:textAppearance="@style/TextAppearance.MaterialComponents.Subtitle1"
app:layout_constraintBottom_toBottomOf="parent"
android:layout_width="wrap_content"
android:layout_height="wrap_content" />

</androidx.constraintlayout.widget.ConstraintLayout>

</ScrollView>
5 changes: 3 additions & 2 deletions app/src/main/res/layout/fragment_dashboard.xml
Original file line number Diff line number Diff line change
Expand Up @@ -8,7 +8,8 @@
tools:context=".view.dashboard.DashboardFragment">


<ScrollView
<androidx.core.widget.NestedScrollView
android:id="@+id/main_dashboard_scroll_view"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:fillViewport="true"
Expand Down Expand Up @@ -88,7 +89,7 @@

</androidx.constraintlayout.widget.ConstraintLayout>

</ScrollView>
</androidx.core.widget.NestedScrollView>

<ViewStub
android:id="@+id/emptyStateLayout"
Expand Down
Loading

0 comments on commit f3cb485

Please sign in to comment.