Fix potential build race condition with country code select fragments.

This commit is contained in:
Cody Henthorne
2025-02-11 13:17:28 -05:00
committed by Greyson Parrelli
parent 88cf4c3399
commit 3237072c40
8 changed files with 396 additions and 625 deletions

View File

@@ -0,0 +1,68 @@
/*
* Copyright 2025 Signal Messenger, LLC
* SPDX-License-Identifier: AGPL-3.0-only
*/
package org.thoughtcrime.securesms.components.settings.app.changenumber
import android.os.Bundle
import android.view.View
import androidx.compose.runtime.Composable
import androidx.compose.runtime.getValue
import androidx.compose.ui.res.stringResource
import androidx.core.os.bundleOf
import androidx.fragment.app.setFragmentResult
import androidx.fragment.app.viewModels
import androidx.lifecycle.compose.collectAsStateWithLifecycle
import androidx.navigation.fragment.findNavController
import org.signal.core.util.getParcelableCompat
import org.thoughtcrime.securesms.R
import org.thoughtcrime.securesms.compose.ComposeFragment
import org.thoughtcrime.securesms.registration.ui.countrycode.Country
import org.thoughtcrime.securesms.registration.ui.countrycode.CountryCodeSelectScreen
import org.thoughtcrime.securesms.registration.ui.countrycode.CountryCodeViewModel
/**
* Country code picker specific to change number flow.
*/
class ChangeNumberCountryCodeFragment : ComposeFragment() {
companion object {
const val RESULT_KEY = "result_key"
const val REQUEST_KEY_COUNTRY = "request_key_country"
const val REQUEST_COUNTRY = "country"
const val RESULT_COUNTRY = "country"
}
private val viewModel: CountryCodeViewModel by viewModels()
@Composable
override fun FragmentContent() {
val state by viewModel.state.collectAsStateWithLifecycle()
val resultKey = arguments?.getString(RESULT_KEY) ?: REQUEST_KEY_COUNTRY
CountryCodeSelectScreen(
state = state,
title = stringResource(R.string.CountryCodeFragment__your_country),
onSearch = { search -> viewModel.filterCountries(search) },
onDismissed = { findNavController().popBackStack() },
onClick = { country ->
setFragmentResult(
resultKey,
bundleOf(
RESULT_COUNTRY to country
)
)
findNavController().popBackStack()
}
)
}
override fun onViewCreated(view: View, savedInstanceState: Bundle?) {
super.onViewCreated(view, savedInstanceState)
val initialCountry = arguments?.getParcelableCompat(REQUEST_COUNTRY, Country::class.java)
viewModel.loadCountries(initialCountry)
}
}

View File

@@ -15,14 +15,12 @@ import androidx.navigation.fragment.findNavController
import com.google.android.material.dialog.MaterialAlertDialogBuilder
import com.google.android.material.snackbar.Snackbar
import org.signal.core.util.getParcelableCompat
import org.signal.core.util.logging.Log
import org.thoughtcrime.securesms.LoggingFragment
import org.thoughtcrime.securesms.R
import org.thoughtcrime.securesms.components.ViewBinderDelegate
import org.thoughtcrime.securesms.databinding.FragmentChangeNumberEnterPhoneNumberBinding
import org.thoughtcrime.securesms.registration.ui.countrycode.Country
import org.thoughtcrime.securesms.registration.util.ChangeNumberInputController
import org.thoughtcrime.securesms.registrationv3.ui.countrycode.CountryCodeFragment
import org.thoughtcrime.securesms.util.Dialogs
import org.thoughtcrime.securesms.util.navigation.safeNavigate
@@ -32,8 +30,6 @@ import org.thoughtcrime.securesms.util.navigation.safeNavigate
class ChangeNumberEnterPhoneNumberFragment : LoggingFragment(R.layout.fragment_change_number_enter_phone_number) {
companion object {
private val TAG: String = Log.tag(ChangeNumberEnterPhoneNumberFragment::class.java)
private const val OLD_NUMBER_COUNTRY_SELECT = "old_number_country"
private const val NEW_NUMBER_COUNTRY_SELECT = "new_number_country"
}
@@ -113,12 +109,12 @@ class ChangeNumberEnterPhoneNumberFragment : LoggingFragment(R.layout.fragment_c
)
parentFragmentManager.setFragmentResultListener(OLD_NUMBER_COUNTRY_SELECT, this) { _: String, bundle: Bundle ->
val country = bundle.getParcelableCompat(CountryCodeFragment.RESULT_COUNTRY, Country::class.java)!!
val country = bundle.getParcelableCompat(ChangeNumberCountryCodeFragment.RESULT_COUNTRY, Country::class.java)!!
viewModel.setOldCountry(country)
}
parentFragmentManager.setFragmentResultListener(NEW_NUMBER_COUNTRY_SELECT, this) { _: String, bundle: Bundle ->
val country = bundle.getParcelableCompat(CountryCodeFragment.RESULT_COUNTRY, Country::class.java)!!
val country = bundle.getParcelableCompat(ChangeNumberCountryCodeFragment.RESULT_COUNTRY, Country::class.java)!!
viewModel.setNewCountry(country)
}

View File

@@ -76,8 +76,8 @@ import org.thoughtcrime.securesms.invites.InviteActions
import org.thoughtcrime.securesms.phonenumbers.PhoneNumberVisualTransformation
import org.thoughtcrime.securesms.recipients.RecipientId
import org.thoughtcrime.securesms.registration.ui.countrycode.Country
import org.thoughtcrime.securesms.registration.ui.countrycode.CountryCodeSelectScreen
import org.thoughtcrime.securesms.registration.ui.countrycode.CountryCodeState
import org.thoughtcrime.securesms.registrationv3.ui.countrycode.Screen
import org.thoughtcrime.securesms.util.DynamicNoActionBarTheme
import org.thoughtcrime.securesms.util.viewModel
import org.whispersystems.signalservice.api.util.PhoneNumberFormatter
@@ -165,7 +165,7 @@ class FindByActivity : PassphraseRequiredActivity() {
}
composable("select-country-prefix") {
Screen(
CountryCodeSelectScreen(
state = CountryCodeState(
query = state.query,
filteredList = state.filteredCountries,

View File

@@ -4,62 +4,15 @@ package org.thoughtcrime.securesms.registration.ui.countrycode
import android.os.Bundle
import android.view.View
import androidx.compose.foundation.ExperimentalFoundationApi
import androidx.compose.foundation.background
import androidx.compose.foundation.clickable
import androidx.compose.foundation.layout.Row
import androidx.compose.foundation.layout.defaultMinSize
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.lazy.LazyColumn
import androidx.compose.foundation.lazy.items
import androidx.compose.foundation.lazy.rememberLazyListState
import androidx.compose.foundation.shape.RoundedCornerShape
import androidx.compose.foundation.text.KeyboardOptions
import androidx.compose.material3.CircularProgressIndicator
import androidx.compose.material3.ExperimentalMaterial3Api
import androidx.compose.material3.Icon
import androidx.compose.material3.MaterialTheme
import androidx.compose.material3.Scaffold
import androidx.compose.material3.Text
import androidx.compose.material3.TextField
import androidx.compose.material3.TextFieldDefaults
import androidx.compose.runtime.Composable
import androidx.compose.runtime.LaunchedEffect
import androidx.compose.runtime.getValue
import androidx.compose.runtime.mutableStateOf
import androidx.compose.runtime.remember
import androidx.compose.runtime.rememberCoroutineScope
import androidx.compose.runtime.setValue
import androidx.compose.ui.Alignment
import androidx.compose.ui.Modifier
import androidx.compose.ui.focus.FocusRequester
import androidx.compose.ui.focus.focusRequester
import androidx.compose.ui.graphics.Color
import androidx.compose.ui.graphics.vector.ImageVector
import androidx.compose.ui.graphics.vector.rememberVectorPainter
import androidx.compose.ui.res.stringResource
import androidx.compose.ui.res.vectorResource
import androidx.compose.ui.text.SpanStyle
import androidx.compose.ui.text.buildAnnotatedString
import androidx.compose.ui.text.font.FontWeight
import androidx.compose.ui.text.input.KeyboardType
import androidx.compose.ui.text.input.VisualTransformation
import androidx.compose.ui.text.withStyle
import androidx.compose.ui.unit.dp
import androidx.core.os.bundleOf
import androidx.fragment.app.setFragmentResult
import androidx.fragment.app.viewModels
import androidx.lifecycle.compose.collectAsStateWithLifecycle
import androidx.navigation.fragment.findNavController
import kotlinx.coroutines.launch
import org.signal.core.ui.Dividers
import org.signal.core.ui.IconButtons.IconButton
import org.signal.core.ui.Previews
import org.signal.core.ui.Scaffolds
import org.signal.core.ui.SignalPreview
import org.signal.core.util.getParcelableCompat
import org.signal.core.util.logging.Log
import org.thoughtcrime.securesms.R
@@ -83,7 +36,7 @@ class CountryCodeFragment : ComposeFragment() {
override fun FragmentContent() {
val state by viewModel.state.collectAsStateWithLifecycle()
Screen(
CountryCodeSelectScreen(
state = state,
title = stringResource(R.string.CountryCodeFragment__your_country),
onSearch = { search -> viewModel.filterCountries(search) },
@@ -107,259 +60,3 @@ class CountryCodeFragment : ComposeFragment() {
viewModel.loadCountries(initialCountry)
}
}
@OptIn(ExperimentalMaterial3Api::class, ExperimentalFoundationApi::class)
@Composable
private fun Screen(
state: CountryCodeState,
title: String,
onSearch: (String) -> Unit = {},
onDismissed: () -> Unit = {},
onClick: (Country) -> Unit = {}
) {
Scaffold(
topBar = {
Scaffolds.DefaultTopAppBar(
title = title,
titleContent = { _, title ->
Text(text = title, style = MaterialTheme.typography.titleLarge)
},
onNavigationClick = onDismissed,
navigationIconPainter = rememberVectorPainter(ImageVector.vectorResource(R.drawable.symbol_x_24))
)
}
) { padding ->
val listState = rememberLazyListState()
val coroutineScope = rememberCoroutineScope()
LazyColumn(
state = listState,
horizontalAlignment = Alignment.CenterHorizontally,
modifier = Modifier.padding(padding)
) {
stickyHeader {
SearchBar(
text = state.query,
onSearch = onSearch
)
}
if (state.countryList.isEmpty()) {
item {
CircularProgressIndicator(
modifier = Modifier.size(56.dp)
)
}
} else if (state.query.isEmpty()) {
items(state.commonCountryList) { country ->
CountryItem(country, onClick)
}
item {
Dividers.Default()
}
items(state.countryList) { country ->
CountryItem(country, onClick)
}
} else {
items(state.filteredList) { country ->
CountryItem(country, onClick, state.query)
}
}
}
LaunchedEffect(state.startingIndex) {
coroutineScope.launch {
listState.scrollToItem(index = state.startingIndex)
}
}
}
}
@Composable
fun CountryItem(
country: Country,
onClick: (Country) -> Unit = {},
query: String = ""
) {
val emoji = country.emoji
val name = country.name
val code = "+${country.countryCode}"
Row(
verticalAlignment = Alignment.CenterVertically,
modifier = Modifier
.padding(horizontal = 24.dp)
.fillMaxWidth()
.defaultMinSize(minHeight = 56.dp)
.clickable { onClick(country) }
) {
Text(
text = emoji,
modifier = Modifier.size(24.dp)
)
if (query.isEmpty()) {
Text(
text = name.ifEmpty { stringResource(R.string.CountryCodeFragment__unknown_country) },
modifier = Modifier
.padding(start = 24.dp)
.weight(1f),
style = MaterialTheme.typography.bodyLarge
)
Text(
text = code,
modifier = Modifier.padding(start = 24.dp),
style = MaterialTheme.typography.bodyLarge,
color = MaterialTheme.colorScheme.onSurfaceVariant
)
} else {
val annotatedName = buildAnnotatedString {
val startIndex = name.indexOf(query, ignoreCase = true)
if (startIndex >= 0) {
append(name.substring(0, startIndex))
withStyle(style = SpanStyle(fontWeight = FontWeight.Bold)) {
append(name.substring(startIndex, startIndex + query.length))
}
append(name.substring(startIndex + query.length))
} else {
append(name)
}
}
val annotatedCode = buildAnnotatedString {
val startIndex = code.indexOf(query, ignoreCase = true)
if (startIndex >= 0) {
append(code.substring(0, startIndex))
withStyle(style = SpanStyle(fontWeight = FontWeight.Bold)) {
append(code.substring(startIndex, startIndex + query.length))
}
append(code.substring(startIndex + query.length))
} else {
append(code)
}
}
Text(
text = annotatedName,
modifier = Modifier
.padding(start = 24.dp)
.weight(1f),
style = MaterialTheme.typography.bodyLarge
)
Text(
text = annotatedCode,
modifier = Modifier.padding(start = 24.dp),
style = MaterialTheme.typography.bodyLarge,
color = MaterialTheme.colorScheme.onSurfaceVariant
)
}
}
}
@Composable
fun SearchBar(
text: String,
modifier: Modifier = Modifier,
hint: String = stringResource(R.string.CountryCodeFragment__search_by),
onSearch: (String) -> Unit = {}
) {
val focusRequester = remember { FocusRequester() }
var showKeyboard by remember { mutableStateOf(false) }
TextField(
value = text,
onValueChange = { onSearch(it) },
placeholder = { Text(hint) },
trailingIcon = {
if (text.isNotEmpty()) {
IconButton(onClick = { onSearch("") }) {
Icon(
imageVector = ImageVector.vectorResource(R.drawable.symbol_x_24),
contentDescription = null
)
}
} else {
IconButton(onClick = {
showKeyboard = !showKeyboard
focusRequester.requestFocus()
}) {
if (showKeyboard) {
Icon(
imageVector = ImageVector.vectorResource(R.drawable.symbol_keyboard_24),
contentDescription = null
)
} else {
Icon(
imageVector = ImageVector.vectorResource(R.drawable.symbol_number_pad_24),
contentDescription = null
)
}
}
}
},
keyboardOptions = KeyboardOptions(
keyboardType = if (showKeyboard) {
KeyboardType.Number
} else {
KeyboardType.Text
}
),
shape = RoundedCornerShape(32.dp),
modifier = modifier
.background(MaterialTheme.colorScheme.background)
.padding(bottom = 18.dp, start = 16.dp, end = 16.dp)
.fillMaxWidth()
.height(54.dp)
.focusRequester(focusRequester),
visualTransformation = VisualTransformation.None,
colors = TextFieldDefaults.colors(
// TODO move to SignalTheme
focusedContainerColor = MaterialTheme.colorScheme.surfaceVariant,
unfocusedContainerColor = MaterialTheme.colorScheme.surfaceVariant,
disabledContainerColor = MaterialTheme.colorScheme.surfaceVariant,
focusedIndicatorColor = Color.Transparent,
unfocusedIndicatorColor = Color.Transparent
)
)
}
@SignalPreview
@Composable
private fun ScreenPreview() {
Previews.Preview {
Screen(
state = CountryCodeState(
countryList = mutableListOf(
Country("\uD83C\uDDFA\uD83C\uDDF8", "United States", 1, "US"),
Country("\uD83C\uDDE8\uD83C\uDDE6", "Canada", 2, "CA"),
Country("\uD83C\uDDF2\uD83C\uDDFD", "Mexico", 3, "MX")
),
commonCountryList = mutableListOf(
Country("\uD83C\uDDFA\uD83C\uDDF8", "United States", 4, "US"),
Country("\uD83C\uDDE8\uD83C\uDDE6", "Canada", 5, "CA")
)
),
title = "Your country"
)
}
}
@SignalPreview
@Composable
private fun LoadingScreenPreview() {
Previews.Preview {
Screen(
state = CountryCodeState(
countryList = emptyList()
),
title = "Your country"
)
}
}

View File

@@ -0,0 +1,320 @@
/*
* Copyright 2025 Signal Messenger, LLC
* SPDX-License-Identifier: AGPL-3.0-only
*/
package org.thoughtcrime.securesms.registration.ui.countrycode
import androidx.compose.foundation.ExperimentalFoundationApi
import androidx.compose.foundation.background
import androidx.compose.foundation.clickable
import androidx.compose.foundation.layout.Row
import androidx.compose.foundation.layout.defaultMinSize
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.lazy.LazyColumn
import androidx.compose.foundation.lazy.items
import androidx.compose.foundation.lazy.rememberLazyListState
import androidx.compose.foundation.shape.RoundedCornerShape
import androidx.compose.foundation.text.KeyboardOptions
import androidx.compose.material3.CircularProgressIndicator
import androidx.compose.material3.ExperimentalMaterial3Api
import androidx.compose.material3.Icon
import androidx.compose.material3.MaterialTheme
import androidx.compose.material3.Scaffold
import androidx.compose.material3.Text
import androidx.compose.material3.TextField
import androidx.compose.material3.TextFieldDefaults
import androidx.compose.runtime.Composable
import androidx.compose.runtime.LaunchedEffect
import androidx.compose.runtime.getValue
import androidx.compose.runtime.mutableStateOf
import androidx.compose.runtime.remember
import androidx.compose.runtime.rememberCoroutineScope
import androidx.compose.runtime.setValue
import androidx.compose.ui.Alignment
import androidx.compose.ui.Modifier
import androidx.compose.ui.focus.FocusRequester
import androidx.compose.ui.focus.focusRequester
import androidx.compose.ui.graphics.Color
import androidx.compose.ui.graphics.vector.ImageVector
import androidx.compose.ui.graphics.vector.rememberVectorPainter
import androidx.compose.ui.res.stringResource
import androidx.compose.ui.res.vectorResource
import androidx.compose.ui.text.SpanStyle
import androidx.compose.ui.text.buildAnnotatedString
import androidx.compose.ui.text.font.FontWeight
import androidx.compose.ui.text.input.KeyboardType
import androidx.compose.ui.text.input.VisualTransformation
import androidx.compose.ui.text.withStyle
import androidx.compose.ui.unit.dp
import kotlinx.coroutines.launch
import org.signal.core.ui.Dividers
import org.signal.core.ui.IconButtons.IconButton
import org.signal.core.ui.Previews
import org.signal.core.ui.Scaffolds
import org.signal.core.ui.SignalPreview
import org.thoughtcrime.securesms.R
/**
* Screen that allows someone to search and select a country code from a supported list of countries.
*/
@OptIn(ExperimentalMaterial3Api::class, ExperimentalFoundationApi::class)
@Composable
fun CountryCodeSelectScreen(
state: CountryCodeState,
title: String,
onSearch: (String) -> Unit = {},
onDismissed: () -> Unit = {},
onClick: (Country) -> Unit = {}
) {
Scaffold(
topBar = {
Scaffolds.DefaultTopAppBar(
title = title,
titleContent = { _, title ->
Text(text = title, style = MaterialTheme.typography.titleLarge)
},
onNavigationClick = onDismissed,
navigationIconPainter = rememberVectorPainter(ImageVector.vectorResource(R.drawable.symbol_x_24))
)
}
) { padding ->
val listState = rememberLazyListState()
val coroutineScope = rememberCoroutineScope()
LazyColumn(
state = listState,
horizontalAlignment = Alignment.CenterHorizontally,
modifier = Modifier.padding(padding)
) {
stickyHeader {
SearchBar(
text = state.query,
onSearch = onSearch
)
}
if (state.countryList.isEmpty()) {
item {
CircularProgressIndicator(
modifier = Modifier.size(56.dp)
)
}
} else if (state.query.isEmpty()) {
if (state.commonCountryList.isNotEmpty()) {
items(state.commonCountryList) { country ->
CountryItem(country, onClick)
}
item {
Dividers.Default()
}
}
items(state.countryList) { country ->
CountryItem(country, onClick)
}
} else {
items(state.filteredList) { country ->
CountryItem(country, onClick, state.query)
}
}
}
LaunchedEffect(state.startingIndex) {
coroutineScope.launch {
listState.scrollToItem(index = state.startingIndex)
}
}
}
}
@Composable
private fun CountryItem(
country: Country,
onClick: (Country) -> Unit = {},
query: String = ""
) {
val emoji = country.emoji
val name = country.name
val code = "+${country.countryCode}"
Row(
verticalAlignment = Alignment.CenterVertically,
modifier = Modifier
.padding(horizontal = 24.dp)
.fillMaxWidth()
.defaultMinSize(minHeight = 56.dp)
.clickable { onClick(country) }
) {
Text(
text = emoji,
modifier = Modifier.size(24.dp)
)
if (query.isEmpty()) {
Text(
text = name.ifEmpty { stringResource(R.string.CountryCodeFragment__unknown_country) },
modifier = Modifier
.padding(start = 24.dp)
.weight(1f),
style = MaterialTheme.typography.bodyLarge
)
Text(
text = code,
modifier = Modifier.padding(start = 24.dp),
style = MaterialTheme.typography.bodyLarge,
color = MaterialTheme.colorScheme.onSurfaceVariant
)
} else {
val annotatedName = buildAnnotatedString {
val startIndex = name.indexOf(query, ignoreCase = true)
if (startIndex >= 0) {
append(name.substring(0, startIndex))
withStyle(style = SpanStyle(fontWeight = FontWeight.Bold)) {
append(name.substring(startIndex, startIndex + query.length))
}
append(name.substring(startIndex + query.length))
} else {
append(name)
}
}
val annotatedCode = buildAnnotatedString {
val startIndex = code.indexOf(query, ignoreCase = true)
if (startIndex >= 0) {
append(code.substring(0, startIndex))
withStyle(style = SpanStyle(fontWeight = FontWeight.Bold)) {
append(code.substring(startIndex, startIndex + query.length))
}
append(code.substring(startIndex + query.length))
} else {
append(code)
}
}
Text(
text = annotatedName,
modifier = Modifier
.padding(start = 24.dp)
.weight(1f),
style = MaterialTheme.typography.bodyLarge
)
Text(
text = annotatedCode,
modifier = Modifier.padding(start = 24.dp),
style = MaterialTheme.typography.bodyLarge,
color = MaterialTheme.colorScheme.onSurfaceVariant
)
}
}
}
@Composable
private fun SearchBar(
text: String,
modifier: Modifier = Modifier,
hint: String = stringResource(R.string.CountryCodeFragment__search_by),
onSearch: (String) -> Unit = {}
) {
val focusRequester = remember { FocusRequester() }
var showKeyboard by remember { mutableStateOf(false) }
TextField(
value = text,
onValueChange = { onSearch(it) },
placeholder = { Text(hint) },
trailingIcon = {
if (text.isNotEmpty()) {
IconButton(onClick = { onSearch("") }) {
Icon(
imageVector = ImageVector.vectorResource(R.drawable.symbol_x_24),
contentDescription = null
)
}
} else {
IconButton(onClick = {
showKeyboard = !showKeyboard
focusRequester.requestFocus()
}) {
if (showKeyboard) {
Icon(
imageVector = ImageVector.vectorResource(R.drawable.symbol_keyboard_24),
contentDescription = null
)
} else {
Icon(
imageVector = ImageVector.vectorResource(R.drawable.symbol_number_pad_24),
contentDescription = null
)
}
}
}
},
keyboardOptions = KeyboardOptions(
keyboardType = if (showKeyboard) {
KeyboardType.Number
} else {
KeyboardType.Text
}
),
shape = RoundedCornerShape(32.dp),
modifier = modifier
.background(MaterialTheme.colorScheme.background)
.padding(bottom = 18.dp, start = 16.dp, end = 16.dp)
.fillMaxWidth()
.height(54.dp)
.focusRequester(focusRequester),
visualTransformation = VisualTransformation.None,
colors = TextFieldDefaults.colors(
// TODO move to SignalTheme
focusedContainerColor = MaterialTheme.colorScheme.surfaceVariant,
unfocusedContainerColor = MaterialTheme.colorScheme.surfaceVariant,
disabledContainerColor = MaterialTheme.colorScheme.surfaceVariant,
focusedIndicatorColor = Color.Transparent,
unfocusedIndicatorColor = Color.Transparent
)
)
}
@SignalPreview
@Composable
private fun ScreenPreview() {
Previews.Preview {
CountryCodeSelectScreen(
state = CountryCodeState(
countryList = mutableListOf(
Country("\uD83C\uDDFA\uD83C\uDDF8", "United States", 1, "US"),
Country("\uD83C\uDDE8\uD83C\uDDE6", "Canada", 2, "CA"),
Country("\uD83C\uDDF2\uD83C\uDDFD", "Mexico", 3, "MX")
),
commonCountryList = mutableListOf(
Country("\uD83C\uDDFA\uD83C\uDDF8", "United States", 4, "US"),
Country("\uD83C\uDDE8\uD83C\uDDE6", "Canada", 5, "CA")
)
),
title = "Your country"
)
}
}
@SignalPreview
@Composable
private fun LoadingScreenPreview() {
Previews.Preview {
CountryCodeSelectScreen(
state = CountryCodeState(
countryList = emptyList()
),
title = "Your country"
)
}
}

View File

@@ -12,17 +12,12 @@ import kotlinx.coroutines.flow.MutableStateFlow
import kotlinx.coroutines.flow.asStateFlow
import kotlinx.coroutines.flow.update
import kotlinx.coroutines.launch
import org.signal.core.util.logging.Log
/**
* View model to support [CountryCodeFragment] and track the countries
*/
class CountryCodeViewModel : ViewModel() {
companion object {
private val TAG = Log.tag(CountryCodeViewModel::class.java)
}
private val internalState = MutableStateFlow(CountryCodeState())
val state = internalState.asStateFlow()

View File

@@ -4,68 +4,21 @@ package org.thoughtcrime.securesms.registrationv3.ui.countrycode
import android.os.Bundle
import android.view.View
import androidx.compose.foundation.ExperimentalFoundationApi
import androidx.compose.foundation.background
import androidx.compose.foundation.clickable
import androidx.compose.foundation.layout.Row
import androidx.compose.foundation.layout.defaultMinSize
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.lazy.LazyColumn
import androidx.compose.foundation.lazy.items
import androidx.compose.foundation.lazy.rememberLazyListState
import androidx.compose.foundation.shape.RoundedCornerShape
import androidx.compose.foundation.text.KeyboardOptions
import androidx.compose.material3.CircularProgressIndicator
import androidx.compose.material3.ExperimentalMaterial3Api
import androidx.compose.material3.Icon
import androidx.compose.material3.MaterialTheme
import androidx.compose.material3.Scaffold
import androidx.compose.material3.Text
import androidx.compose.material3.TextField
import androidx.compose.material3.TextFieldDefaults
import androidx.compose.runtime.Composable
import androidx.compose.runtime.LaunchedEffect
import androidx.compose.runtime.getValue
import androidx.compose.runtime.mutableStateOf
import androidx.compose.runtime.remember
import androidx.compose.runtime.rememberCoroutineScope
import androidx.compose.runtime.setValue
import androidx.compose.ui.Alignment
import androidx.compose.ui.Modifier
import androidx.compose.ui.focus.FocusRequester
import androidx.compose.ui.focus.focusRequester
import androidx.compose.ui.graphics.Color
import androidx.compose.ui.graphics.vector.ImageVector
import androidx.compose.ui.graphics.vector.rememberVectorPainter
import androidx.compose.ui.res.stringResource
import androidx.compose.ui.res.vectorResource
import androidx.compose.ui.text.SpanStyle
import androidx.compose.ui.text.buildAnnotatedString
import androidx.compose.ui.text.font.FontWeight
import androidx.compose.ui.text.input.KeyboardType
import androidx.compose.ui.text.input.VisualTransformation
import androidx.compose.ui.text.withStyle
import androidx.compose.ui.unit.dp
import androidx.core.os.bundleOf
import androidx.fragment.app.setFragmentResult
import androidx.fragment.app.viewModels
import androidx.lifecycle.compose.collectAsStateWithLifecycle
import androidx.navigation.fragment.findNavController
import kotlinx.coroutines.launch
import org.signal.core.ui.Dividers
import org.signal.core.ui.IconButtons.IconButton
import org.signal.core.ui.Previews
import org.signal.core.ui.Scaffolds
import org.signal.core.ui.SignalPreview
import org.signal.core.util.getParcelableCompat
import org.signal.core.util.logging.Log
import org.thoughtcrime.securesms.R
import org.thoughtcrime.securesms.compose.ComposeFragment
import org.thoughtcrime.securesms.registration.ui.countrycode.Country
import org.thoughtcrime.securesms.registration.ui.countrycode.CountryCodeState
import org.thoughtcrime.securesms.registration.ui.countrycode.CountryCodeSelectScreen
import org.thoughtcrime.securesms.registration.ui.countrycode.CountryCodeViewModel
/**
@@ -89,7 +42,7 @@ class CountryCodeFragment : ComposeFragment() {
val resultKey = arguments?.getString(RESULT_KEY) ?: REQUEST_KEY_COUNTRY
Screen(
CountryCodeSelectScreen(
state = state,
title = stringResource(R.string.CountryCodeFragment__your_country),
onSearch = { search -> viewModel.filterCountries(search) },
@@ -113,261 +66,3 @@ class CountryCodeFragment : ComposeFragment() {
viewModel.loadCountries(initialCountry)
}
}
@OptIn(ExperimentalMaterial3Api::class, ExperimentalFoundationApi::class)
@Composable
fun Screen(
state: CountryCodeState,
title: String,
onSearch: (String) -> Unit = {},
onDismissed: () -> Unit = {},
onClick: (Country) -> Unit = {}
) {
Scaffold(
topBar = {
Scaffolds.DefaultTopAppBar(
title = title,
titleContent = { _, title ->
Text(text = title, style = MaterialTheme.typography.titleLarge)
},
onNavigationClick = onDismissed,
navigationIconPainter = rememberVectorPainter(ImageVector.vectorResource(R.drawable.symbol_x_24))
)
}
) { padding ->
val listState = rememberLazyListState()
val coroutineScope = rememberCoroutineScope()
LazyColumn(
state = listState,
horizontalAlignment = Alignment.CenterHorizontally,
modifier = Modifier.padding(padding)
) {
stickyHeader {
SearchBar(
text = state.query,
onSearch = onSearch
)
}
if (state.countryList.isEmpty()) {
item {
CircularProgressIndicator(
modifier = Modifier.size(56.dp)
)
}
} else if (state.query.isEmpty()) {
if (state.commonCountryList.isNotEmpty()) {
items(state.commonCountryList) { country ->
CountryItem(country, onClick)
}
item {
Dividers.Default()
}
}
items(state.countryList) { country ->
CountryItem(country, onClick)
}
} else {
items(state.filteredList) { country ->
CountryItem(country, onClick, state.query)
}
}
}
LaunchedEffect(state.startingIndex) {
coroutineScope.launch {
listState.scrollToItem(index = state.startingIndex)
}
}
}
}
@Composable
fun CountryItem(
country: Country,
onClick: (Country) -> Unit = {},
query: String = ""
) {
val emoji = country.emoji
val name = country.name
val code = "+${country.countryCode}"
Row(
verticalAlignment = Alignment.CenterVertically,
modifier = Modifier
.padding(horizontal = 24.dp)
.fillMaxWidth()
.defaultMinSize(minHeight = 56.dp)
.clickable { onClick(country) }
) {
Text(
text = emoji,
modifier = Modifier.size(24.dp)
)
if (query.isEmpty()) {
Text(
text = name.ifEmpty { stringResource(R.string.CountryCodeFragment__unknown_country) },
modifier = Modifier
.padding(start = 24.dp)
.weight(1f),
style = MaterialTheme.typography.bodyLarge
)
Text(
text = code,
modifier = Modifier.padding(start = 24.dp),
style = MaterialTheme.typography.bodyLarge,
color = MaterialTheme.colorScheme.onSurfaceVariant
)
} else {
val annotatedName = buildAnnotatedString {
val startIndex = name.indexOf(query, ignoreCase = true)
if (startIndex >= 0) {
append(name.substring(0, startIndex))
withStyle(style = SpanStyle(fontWeight = FontWeight.Bold)) {
append(name.substring(startIndex, startIndex + query.length))
}
append(name.substring(startIndex + query.length))
} else {
append(name)
}
}
val annotatedCode = buildAnnotatedString {
val startIndex = code.indexOf(query, ignoreCase = true)
if (startIndex >= 0) {
append(code.substring(0, startIndex))
withStyle(style = SpanStyle(fontWeight = FontWeight.Bold)) {
append(code.substring(startIndex, startIndex + query.length))
}
append(code.substring(startIndex + query.length))
} else {
append(code)
}
}
Text(
text = annotatedName,
modifier = Modifier
.padding(start = 24.dp)
.weight(1f),
style = MaterialTheme.typography.bodyLarge
)
Text(
text = annotatedCode,
modifier = Modifier.padding(start = 24.dp),
style = MaterialTheme.typography.bodyLarge,
color = MaterialTheme.colorScheme.onSurfaceVariant
)
}
}
}
@Composable
fun SearchBar(
text: String,
modifier: Modifier = Modifier,
hint: String = stringResource(R.string.CountryCodeFragment__search_by),
onSearch: (String) -> Unit = {}
) {
val focusRequester = remember { FocusRequester() }
var showKeyboard by remember { mutableStateOf(false) }
TextField(
value = text,
onValueChange = { onSearch(it) },
placeholder = { Text(hint) },
trailingIcon = {
if (text.isNotEmpty()) {
IconButton(onClick = { onSearch("") }) {
Icon(
imageVector = ImageVector.vectorResource(R.drawable.symbol_x_24),
contentDescription = null
)
}
} else {
IconButton(onClick = {
showKeyboard = !showKeyboard
focusRequester.requestFocus()
}) {
if (showKeyboard) {
Icon(
imageVector = ImageVector.vectorResource(R.drawable.symbol_keyboard_24),
contentDescription = null
)
} else {
Icon(
imageVector = ImageVector.vectorResource(R.drawable.symbol_number_pad_24),
contentDescription = null
)
}
}
}
},
keyboardOptions = KeyboardOptions(
keyboardType = if (showKeyboard) {
KeyboardType.Number
} else {
KeyboardType.Text
}
),
shape = RoundedCornerShape(32.dp),
modifier = modifier
.background(MaterialTheme.colorScheme.background)
.padding(bottom = 18.dp, start = 16.dp, end = 16.dp)
.fillMaxWidth()
.height(54.dp)
.focusRequester(focusRequester),
visualTransformation = VisualTransformation.None,
colors = TextFieldDefaults.colors(
// TODO move to SignalTheme
focusedContainerColor = MaterialTheme.colorScheme.surfaceVariant,
unfocusedContainerColor = MaterialTheme.colorScheme.surfaceVariant,
disabledContainerColor = MaterialTheme.colorScheme.surfaceVariant,
focusedIndicatorColor = Color.Transparent,
unfocusedIndicatorColor = Color.Transparent
)
)
}
@SignalPreview
@Composable
private fun ScreenPreview() {
Previews.Preview {
Screen(
state = CountryCodeState(
countryList = mutableListOf(
Country("\uD83C\uDDFA\uD83C\uDDF8", "United States", 1, "US"),
Country("\uD83C\uDDE8\uD83C\uDDE6", "Canada", 2, "CA"),
Country("\uD83C\uDDF2\uD83C\uDDFD", "Mexico", 3, "MX")
),
commonCountryList = mutableListOf(
Country("\uD83C\uDDFA\uD83C\uDDF8", "United States", 4, "US"),
Country("\uD83C\uDDE8\uD83C\uDDE6", "Canada", 5, "CA")
)
),
title = "Your country"
)
}
}
@SignalPreview
@Composable
private fun LoadingScreenPreview() {
Previews.Preview {
Screen(
state = CountryCodeState(
countryList = emptyList()
),
title = "Your country"
)
}
}

View File

@@ -58,7 +58,7 @@
<fragment
android:id="@+id/countryPickerFragment"
android:name="org.thoughtcrime.securesms.registrationv3.ui.countrycode.CountryCodeFragment"
android:name="org.thoughtcrime.securesms.components.settings.app.changenumber.ChangeNumberCountryCodeFragment"
tools:layout="@layout/fragment_registration_country_picker">
<argument