Skip to content

Commit

Permalink
Add goal icon picker dialog & backend
Browse files Browse the repository at this point in the history
Signed-off-by: starry-shivam <[email protected]>
  • Loading branch information
starry-shivam committed Mar 24, 2024
1 parent caadc53 commit e5bd922
Show file tree
Hide file tree
Showing 16 changed files with 2,585 additions and 37 deletions.
15 changes: 1 addition & 14 deletions .idea/deploymentTargetDropDown.xml

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

2 changes: 2 additions & 0 deletions app/build.gradle
Original file line number Diff line number Diff line change
Expand Up @@ -100,6 +100,8 @@ dependencies {
implementation 'com.google.android.material:material:1.11.0'
// Android 12+ splash API.
implementation 'androidx.core:core-splashscreen:1.0.1'
// Material icons.
implementation 'androidx.compose.material:material-icons-extended:1.6.4'
// Room database
implementation "androidx.room:room-ktx:$room_version"
ksp "androidx.room:room-compiler:$room_version"
Expand Down
5 changes: 4 additions & 1 deletion app/proguard-rules.pro
Original file line number Diff line number Diff line change
Expand Up @@ -25,4 +25,7 @@
-keep class * implements com.google.gson.TypeAdapter
-keep class * implements com.google.gson.TypeAdapterFactory
-keep class * implements com.google.gson.JsonSerializer
-keep class * implements com.google.gson.JsonDeserializer
-keep class * implements com.google.gson.JsonDeserializer

# Prevent R8 from removing icons used in goal icon picker.
-keep class androidx.compose.material.icons.filled.** { *; }
Original file line number Diff line number Diff line change
Expand Up @@ -434,7 +434,7 @@ fun GoalItemCompact(
@Composable
@Preview(showBackground = true)
fun GoalItemsPV() {
Column {
Column(modifier = Modifier.padding(10.dp)) {
GoalItemClassic(
title = "Home Decorations",
primaryText = "You're off to a great start!\nCurrently saved $0.00 out of $1,000.00.",
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -54,7 +54,7 @@ fun HomeDialogs(
openDeleteDialog.value = false
onDeleteConfirmed()
}) {
Text(stringResource(id = R.string.dialog_confirm_button))
Text(stringResource(id = R.string.confirm))
}
}, dismissButton = {
TextButton(onClick = {
Expand Down
Original file line number Diff line number Diff line change
@@ -0,0 +1,288 @@
package com.starry.greenstash.ui.screens.input.composables

import androidx.compose.animation.ExperimentalAnimationApi
import androidx.compose.foundation.ExperimentalFoundationApi
import androidx.compose.foundation.background
import androidx.compose.foundation.clickable
import androidx.compose.foundation.layout.Arrangement
import androidx.compose.foundation.layout.Box
import androidx.compose.foundation.layout.Column
import androidx.compose.foundation.layout.Row
import androidx.compose.foundation.layout.Spacer
import androidx.compose.foundation.layout.fillMaxHeight
import androidx.compose.foundation.layout.fillMaxSize
import androidx.compose.foundation.layout.fillMaxWidth
import androidx.compose.foundation.layout.height
import androidx.compose.foundation.layout.padding
import androidx.compose.foundation.layout.size
import androidx.compose.foundation.layout.width
import androidx.compose.foundation.lazy.LazyColumn
import androidx.compose.foundation.lazy.items
import androidx.compose.foundation.shape.RoundedCornerShape
import androidx.compose.material.ExperimentalMaterialApi
import androidx.compose.material.icons.Icons
import androidx.compose.material.icons.filled.Image
import androidx.compose.material3.Button
import androidx.compose.material3.Card
import androidx.compose.material3.CardDefaults
import androidx.compose.material3.CircularProgressIndicator
import androidx.compose.material3.ExperimentalMaterial3Api
import androidx.compose.material3.Icon
import androidx.compose.material3.MaterialTheme
import androidx.compose.material3.OutlinedTextField
import androidx.compose.material3.OutlinedTextFieldDefaults
import androidx.compose.material3.Text
import androidx.compose.material3.TextButton
import androidx.compose.runtime.Composable
import androidx.compose.runtime.LaunchedEffect
import androidx.compose.runtime.MutableState
import androidx.compose.runtime.getValue
import androidx.compose.ui.Alignment
import androidx.compose.ui.ExperimentalComposeUiApi
import androidx.compose.ui.Modifier
import androidx.compose.ui.draw.clip
import androidx.compose.ui.platform.LocalContext
import androidx.compose.ui.res.stringResource
import androidx.compose.ui.text.font.FontWeight
import androidx.compose.ui.text.style.TextAlign
import androidx.compose.ui.text.style.TextOverflow
import androidx.compose.ui.unit.dp
import androidx.compose.ui.unit.sp
import androidx.compose.ui.window.Dialog
import com.starry.greenstash.R
import com.starry.greenstash.ui.screens.input.viewmodels.IconItem
import com.starry.greenstash.ui.screens.input.viewmodels.IconsState
import com.starry.greenstash.ui.screens.input.viewmodels.InputViewModel
import kotlinx.coroutines.ExperimentalCoroutinesApi
import kotlinx.coroutines.delay

@ExperimentalCoroutinesApi
@ExperimentalMaterial3Api
@ExperimentalAnimationApi
@ExperimentalComposeUiApi
@ExperimentalFoundationApi
@ExperimentalMaterialApi
@Composable
fun ExtendedIconsPicker(
viewModel: InputViewModel,
showDialog: MutableState<Boolean>,
) {

val context = LocalContext.current
val state by viewModel.iconState

LaunchedEffect(key1 = true) {
viewModel.updateIconSearch(context = context, search = "")
}

if (showDialog.value) {
Dialog(
onDismissRequest = {
println("Dialog dismissed 11")
}
) {
Column(
modifier = Modifier.fillMaxWidth(0.99f)
) {
Card(
modifier = Modifier.fillMaxWidth(),
colors = CardDefaults.elevatedCardColors()
) {
Column(
modifier = Modifier.fillMaxWidth(),
horizontalAlignment = Alignment.CenterHorizontally,
verticalArrangement = Arrangement.Center
) {

SearchTextField(
viewModel = viewModel,
onSearchChanged = { search ->
viewModel.updateIconSearch(
context = context,
search = search
)
}
)

Spacer(modifier = Modifier.height(10.dp))

IconsList(
iconState = state,
onIconClick = { viewModel.onIconClick(it) }
)

Row(
modifier = Modifier
.fillMaxWidth()
.padding(10.dp)
) {
Spacer(modifier = Modifier.weight(1f))

TextButton(onClick = {
if (viewModel.iconState.value.searchText.isNotEmpty()) {
viewModel.updateIconSearch(context = context, search = "")
} else {
showDialog.value = false
}
}) {
Text(text = stringResource(id = R.string.cancel))
}

Spacer(modifier = Modifier.width(10.dp))

Button(onClick = { showDialog.value = false }) {
Text(text = stringResource(id = R.string.confirm))
}
}
}
}
}
}
}

}

@ExperimentalCoroutinesApi
@ExperimentalMaterial3Api
@ExperimentalAnimationApi
@ExperimentalComposeUiApi
@ExperimentalFoundationApi
@ExperimentalMaterialApi
@Composable
private fun SearchTextField(
viewModel: InputViewModel,
onSearchChanged: (String) -> Unit
) {

Text(
modifier = Modifier.padding(12.dp),
text = "Select an Icon",
color = MaterialTheme.colorScheme.onSurface,
fontWeight = FontWeight.SemiBold,
fontSize = 20.sp
)

OutlinedTextField(
modifier = Modifier.height(60.dp),
label = {
Text(
text = "Search",
color = MaterialTheme.colorScheme.primary
)
},
value = viewModel.iconState.value.searchText,
onValueChange = { onSearchChanged(it) },
colors = OutlinedTextFieldDefaults.colors(
unfocusedBorderColor = MaterialTheme.colorScheme.primary,
unfocusedTextColor = MaterialTheme.colorScheme.primary,
focusedTextColor = MaterialTheme.colorScheme.primary
),
trailingIcon = {
Icon(
Icons.Filled.Image, "Image",
tint = MaterialTheme.colorScheme.primary
)
},
maxLines = 1,
shape = RoundedCornerShape(16.dp)
)
}

@Composable
private fun IconsList(
iconState: IconsState,
onIconClick: (IconItem) -> Unit
) {
val icons = iconState.icons
Column(
modifier = Modifier.fillMaxHeight(0.4f),
horizontalAlignment = Alignment.CenterHorizontally
) {
if (iconState.loading) {
Box(modifier = Modifier.fillMaxSize(), contentAlignment = Alignment.Center) {
CircularProgressIndicator()
}
} else {
LazyColumn {
items(
items = icons,
itemContent = { iconItems ->
IconListRow(
icons = iconItems,
selectedIcon = iconState.selectedIcon,
onClick = { onIconClick(it) }
)
}
)
}
}
}
}

@Composable
private fun IconListRow(
icons: List<IconItem>,
selectedIcon: IconItem? = null,
onClick: (IconItem) -> Unit = {}
) {
Row(
modifier = Modifier.fillMaxWidth(),
horizontalArrangement = Arrangement.SpaceAround
) {
icons.forEach {
IconItem(
icon = it,
selected = it == selectedIcon,
onClick = { onClick(it) }
)
}
}
}

@Composable
private fun IconItem(
icon: IconItem,
selected: Boolean,
onClick: () -> Unit
) {
val backgroundColor = if (selected)
MaterialTheme.colorScheme.onSurface.copy(alpha = 0.25f)
else
MaterialTheme.colorScheme.surface

Column(
modifier = Modifier
.width(100.dp)
.padding(10.dp)
.clickable { onClick() }
.clip(RoundedCornerShape(16.dp))
.background(backgroundColor),
horizontalAlignment = Alignment.CenterHorizontally,
verticalArrangement = Arrangement.Center
) {
var text = "Not found"
var vector = Icons.Filled.Image
var color = MaterialTheme.colorScheme.error


if (icon.image != null) {
text = icon.name
color = MaterialTheme.colorScheme.primary
vector = icon.image!!
}

Icon(
vector, icon.name,
tint = color,
modifier = Modifier.size(40.dp)
)
Text(
text = text,
fontSize = MaterialTheme.typography.labelSmall.fontSize,
textAlign = TextAlign.Center,
overflow = TextOverflow.Ellipsis,
maxLines = 1
)
}
}

Loading

0 comments on commit e5bd922

Please sign in to comment.