mirror of
https://github.com/vitorpamplona/amethyst.git
synced 2025-06-06 13:39:14 +02:00
Refactoring RichTextViewer file
This commit is contained in:
parent
f2badce3b8
commit
00981ef15c
@ -0,0 +1,70 @@
|
|||||||
|
package com.vitorpamplona.amethyst.ui.components
|
||||||
|
|
||||||
|
import androidx.compose.foundation.background
|
||||||
|
import androidx.compose.foundation.layout.Arrangement
|
||||||
|
import androidx.compose.foundation.layout.Box
|
||||||
|
import androidx.compose.foundation.layout.PaddingValues
|
||||||
|
import androidx.compose.foundation.layout.Row
|
||||||
|
import androidx.compose.foundation.layout.fillMaxWidth
|
||||||
|
import androidx.compose.foundation.layout.padding
|
||||||
|
import androidx.compose.foundation.shape.RoundedCornerShape
|
||||||
|
import androidx.compose.material.Button
|
||||||
|
import androidx.compose.material.ButtonDefaults
|
||||||
|
import androidx.compose.material.MaterialTheme
|
||||||
|
import androidx.compose.material.Text
|
||||||
|
import androidx.compose.runtime.Composable
|
||||||
|
import androidx.compose.runtime.getValue
|
||||||
|
import androidx.compose.runtime.mutableStateOf
|
||||||
|
import androidx.compose.runtime.remember
|
||||||
|
import androidx.compose.runtime.setValue
|
||||||
|
import androidx.compose.ui.Alignment
|
||||||
|
import androidx.compose.ui.Modifier
|
||||||
|
import androidx.compose.ui.graphics.Brush
|
||||||
|
import androidx.compose.ui.graphics.Color
|
||||||
|
import androidx.compose.ui.unit.dp
|
||||||
|
import androidx.navigation.NavController
|
||||||
|
|
||||||
|
@Composable
|
||||||
|
fun ExpandableRichTextViewer(
|
||||||
|
content: String,
|
||||||
|
canPreview: Boolean,
|
||||||
|
tags: List<List<String>>?,
|
||||||
|
navController: NavController
|
||||||
|
) {
|
||||||
|
var showFullText by remember { mutableStateOf(false) }
|
||||||
|
|
||||||
|
val text = if (showFullText) content else content.take(350)
|
||||||
|
|
||||||
|
Box(contentAlignment = Alignment.BottomCenter) {
|
||||||
|
RichTextViewer(text, canPreview, tags, navController)
|
||||||
|
|
||||||
|
if (content.length > 350 && !showFullText) {
|
||||||
|
Row(
|
||||||
|
verticalAlignment = Alignment.CenterVertically,
|
||||||
|
horizontalArrangement = Arrangement.Center,
|
||||||
|
modifier = Modifier
|
||||||
|
.fillMaxWidth()
|
||||||
|
.background(
|
||||||
|
brush = Brush.verticalGradient(
|
||||||
|
colors = listOf(
|
||||||
|
MaterialTheme.colors.background.copy(alpha = 0f),
|
||||||
|
MaterialTheme.colors.background
|
||||||
|
)
|
||||||
|
)
|
||||||
|
)
|
||||||
|
) {
|
||||||
|
Button(
|
||||||
|
modifier = Modifier.padding(top = 10.dp),
|
||||||
|
onClick = { showFullText = !showFullText },
|
||||||
|
shape = RoundedCornerShape(20.dp),
|
||||||
|
colors = ButtonDefaults.buttonColors(
|
||||||
|
backgroundColor = MaterialTheme.colors.primary
|
||||||
|
),
|
||||||
|
contentPadding = PaddingValues(vertical = 6.dp, horizontal = 16.dp)
|
||||||
|
) {
|
||||||
|
Text(text = "Show More", color = Color.White)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
@ -1,45 +1,24 @@
|
|||||||
package com.vitorpamplona.amethyst.ui.components
|
package com.vitorpamplona.amethyst.ui.components
|
||||||
|
|
||||||
import android.content.res.Resources
|
|
||||||
import android.util.Patterns
|
import android.util.Patterns
|
||||||
import androidx.compose.animation.animateContentSize
|
import androidx.compose.animation.animateContentSize
|
||||||
import androidx.compose.foundation.background
|
|
||||||
import androidx.compose.foundation.layout.*
|
import androidx.compose.foundation.layout.*
|
||||||
import androidx.compose.foundation.shape.RoundedCornerShape
|
|
||||||
import androidx.compose.foundation.text.ClickableText
|
|
||||||
import androidx.compose.material.*
|
import androidx.compose.material.*
|
||||||
import androidx.compose.runtime.*
|
import androidx.compose.runtime.*
|
||||||
import androidx.compose.runtime.livedata.observeAsState
|
|
||||||
import androidx.compose.ui.Alignment
|
|
||||||
import androidx.compose.ui.Modifier
|
import androidx.compose.ui.Modifier
|
||||||
import androidx.compose.ui.graphics.Brush
|
|
||||||
import androidx.compose.ui.graphics.Color
|
|
||||||
import androidx.compose.ui.platform.LocalContext
|
|
||||||
import androidx.compose.ui.text.SpanStyle
|
|
||||||
import androidx.compose.ui.text.buildAnnotatedString
|
|
||||||
import androidx.compose.ui.text.style.TextDirection
|
import androidx.compose.ui.text.style.TextDirection
|
||||||
import androidx.compose.ui.text.style.TextOverflow
|
|
||||||
import androidx.compose.ui.text.withStyle
|
|
||||||
import androidx.compose.ui.unit.dp
|
|
||||||
import androidx.core.os.ConfigurationCompat
|
|
||||||
import androidx.navigation.NavController
|
import androidx.navigation.NavController
|
||||||
import com.google.accompanist.flowlayout.FlowRow
|
import com.google.accompanist.flowlayout.FlowRow
|
||||||
import com.vitorpamplona.amethyst.LocalPreferences
|
|
||||||
import com.vitorpamplona.amethyst.lnurl.LnInvoiceUtil
|
import com.vitorpamplona.amethyst.lnurl.LnInvoiceUtil
|
||||||
import com.vitorpamplona.amethyst.model.Account
|
|
||||||
import com.vitorpamplona.amethyst.model.LocalCache
|
import com.vitorpamplona.amethyst.model.LocalCache
|
||||||
import com.vitorpamplona.amethyst.model.toByteArray
|
import com.vitorpamplona.amethyst.model.toByteArray
|
||||||
import com.vitorpamplona.amethyst.model.toNote
|
import com.vitorpamplona.amethyst.model.toNote
|
||||||
import com.vitorpamplona.amethyst.service.Nip19
|
import com.vitorpamplona.amethyst.service.Nip19
|
||||||
import com.vitorpamplona.amethyst.service.lang.LanguageTranslatorService
|
|
||||||
import com.vitorpamplona.amethyst.service.lang.ResultOrError
|
|
||||||
import com.vitorpamplona.amethyst.ui.note.toShortenHex
|
import com.vitorpamplona.amethyst.ui.note.toShortenHex
|
||||||
import com.vitorpamplona.amethyst.ui.screen.loggedIn.AccountViewModel
|
|
||||||
import nostr.postr.toNpub
|
import nostr.postr.toNpub
|
||||||
import java.net.MalformedURLException
|
import java.net.MalformedURLException
|
||||||
import java.net.URISyntaxException
|
import java.net.URISyntaxException
|
||||||
import java.net.URL
|
import java.net.URL
|
||||||
import java.util.*
|
|
||||||
import java.util.regex.Pattern
|
import java.util.regex.Pattern
|
||||||
|
|
||||||
val imageExtension = Pattern.compile("(.*/)*.+\\.(png|jpg|gif|bmp|jpeg|webp|svg)$")
|
val imageExtension = Pattern.compile("(.*/)*.+\\.(png|jpg|gif|bmp|jpeg|webp|svg)$")
|
||||||
@ -62,165 +41,6 @@ fun isValidURL(url: String?): Boolean {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@Composable
|
|
||||||
fun TranslateableRichTextViewer(
|
|
||||||
content: String,
|
|
||||||
canPreview: Boolean,
|
|
||||||
tags: List<List<String>>?,
|
|
||||||
accountViewModel: AccountViewModel,
|
|
||||||
navController: NavController
|
|
||||||
) {
|
|
||||||
val translatedTextState = remember {
|
|
||||||
mutableStateOf(ResultOrError(content, null, null, null))
|
|
||||||
}
|
|
||||||
|
|
||||||
var showOriginal by remember { mutableStateOf(false) }
|
|
||||||
var langSettingsPopupExpanded by remember { mutableStateOf(false) }
|
|
||||||
|
|
||||||
val context = LocalContext.current
|
|
||||||
|
|
||||||
val accountState by accountViewModel.accountLanguagesLiveData.observeAsState()
|
|
||||||
val account = accountState?.account ?: return
|
|
||||||
|
|
||||||
LaunchedEffect(accountState) {
|
|
||||||
LanguageTranslatorService.autoTranslate(content, account.dontTranslateFrom, account.translateTo).addOnCompleteListener { task ->
|
|
||||||
if (task.isSuccessful) {
|
|
||||||
translatedTextState.value = task.result
|
|
||||||
} else {
|
|
||||||
translatedTextState.value = ResultOrError(content, null, null, null)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
val toBeViewed = if (showOriginal) content else translatedTextState.value.result ?: content
|
|
||||||
|
|
||||||
Column(modifier = Modifier.padding(top = 5.dp)) {
|
|
||||||
ExpandableRichTextViewer(
|
|
||||||
toBeViewed,
|
|
||||||
canPreview,
|
|
||||||
tags,
|
|
||||||
navController
|
|
||||||
)
|
|
||||||
|
|
||||||
val target = translatedTextState.value.targetLang
|
|
||||||
val source = translatedTextState.value.sourceLang
|
|
||||||
|
|
||||||
if (source != null && target != null) {
|
|
||||||
if (source != target) {
|
|
||||||
Row(modifier = Modifier.fillMaxWidth().padding(top = 5.dp)) {
|
|
||||||
val clickableTextStyle = SpanStyle(color = MaterialTheme.colors.primary.copy(alpha = 0.52f))
|
|
||||||
|
|
||||||
val annotatedTranslationString= buildAnnotatedString {
|
|
||||||
withStyle(clickableTextStyle) {
|
|
||||||
pushStringAnnotation("langSettings", true.toString())
|
|
||||||
append("Auto")
|
|
||||||
}
|
|
||||||
|
|
||||||
append("-translated from ")
|
|
||||||
|
|
||||||
withStyle(clickableTextStyle) {
|
|
||||||
pushStringAnnotation("showOriginal", true.toString())
|
|
||||||
append(Locale(source).displayName)
|
|
||||||
}
|
|
||||||
|
|
||||||
append(" to ")
|
|
||||||
|
|
||||||
withStyle(clickableTextStyle) {
|
|
||||||
pushStringAnnotation("showOriginal", false.toString())
|
|
||||||
append(Locale(target).displayName)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
ClickableText(
|
|
||||||
text = annotatedTranslationString,
|
|
||||||
style = LocalTextStyle.current.copy(color = MaterialTheme.colors.onSurface.copy(alpha = 0.32f)),
|
|
||||||
overflow = TextOverflow.Visible,
|
|
||||||
maxLines = 3
|
|
||||||
) { spanOffset -> annotatedTranslationString.getStringAnnotations(spanOffset, spanOffset)
|
|
||||||
.firstOrNull()
|
|
||||||
?.also { span ->
|
|
||||||
if (span.tag == "showOriginal")
|
|
||||||
showOriginal = span.item.toBoolean()
|
|
||||||
else
|
|
||||||
langSettingsPopupExpanded = !langSettingsPopupExpanded
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
DropdownMenu(
|
|
||||||
expanded = langSettingsPopupExpanded,
|
|
||||||
onDismissRequest = { langSettingsPopupExpanded = false }
|
|
||||||
) {
|
|
||||||
DropdownMenuItem(onClick = {
|
|
||||||
accountViewModel.dontTranslateFrom(source, context)
|
|
||||||
langSettingsPopupExpanded = false
|
|
||||||
}) {
|
|
||||||
Text("Never translate from ${Locale(source).displayName}")
|
|
||||||
}
|
|
||||||
Divider()
|
|
||||||
val languageList = ConfigurationCompat.getLocales(Resources.getSystem().getConfiguration())
|
|
||||||
for (i in 0 until languageList.size()) {
|
|
||||||
languageList.get(i)?.let { lang ->
|
|
||||||
DropdownMenuItem(onClick = {
|
|
||||||
accountViewModel.translateTo(lang, context)
|
|
||||||
langSettingsPopupExpanded = false
|
|
||||||
}) {
|
|
||||||
Text("Always translate to ${lang.displayName}")
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
@Composable
|
|
||||||
fun ExpandableRichTextViewer(
|
|
||||||
content: String,
|
|
||||||
canPreview: Boolean,
|
|
||||||
tags: List<List<String>>?,
|
|
||||||
navController: NavController
|
|
||||||
) {
|
|
||||||
var showFullText by remember { mutableStateOf(false) }
|
|
||||||
|
|
||||||
val text = if (showFullText) content else content.take(350)
|
|
||||||
|
|
||||||
Box(contentAlignment = Alignment.BottomCenter) {
|
|
||||||
RichTextViewer(text, canPreview, tags, navController)
|
|
||||||
|
|
||||||
if (content.length > 350 && !showFullText) {
|
|
||||||
Row(
|
|
||||||
verticalAlignment = Alignment.CenterVertically,
|
|
||||||
horizontalArrangement = Arrangement.Center,
|
|
||||||
modifier = Modifier
|
|
||||||
.fillMaxWidth()
|
|
||||||
.background(
|
|
||||||
brush = Brush.verticalGradient(
|
|
||||||
colors = listOf(
|
|
||||||
MaterialTheme.colors.background.copy(alpha = 0f),
|
|
||||||
MaterialTheme.colors.background
|
|
||||||
)
|
|
||||||
)
|
|
||||||
)
|
|
||||||
) {
|
|
||||||
Button(
|
|
||||||
modifier = Modifier.padding(top = 10.dp),
|
|
||||||
onClick = { showFullText = !showFullText },
|
|
||||||
shape = RoundedCornerShape(20.dp),
|
|
||||||
colors = ButtonDefaults
|
|
||||||
.buttonColors(
|
|
||||||
backgroundColor = MaterialTheme.colors.primary
|
|
||||||
),
|
|
||||||
contentPadding = PaddingValues(vertical = 6.dp, horizontal = 16.dp)
|
|
||||||
) {
|
|
||||||
Text(text = "Show More", color = Color.White)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
@Composable
|
@Composable
|
||||||
fun RichTextViewer(
|
fun RichTextViewer(
|
||||||
content: String,
|
content: String,
|
||||||
|
@ -0,0 +1,151 @@
|
|||||||
|
package com.vitorpamplona.amethyst.ui.components
|
||||||
|
|
||||||
|
import android.content.res.Resources
|
||||||
|
import androidx.compose.foundation.layout.Column
|
||||||
|
import androidx.compose.foundation.layout.Row
|
||||||
|
import androidx.compose.foundation.layout.fillMaxWidth
|
||||||
|
import androidx.compose.foundation.layout.padding
|
||||||
|
import androidx.compose.foundation.text.ClickableText
|
||||||
|
import androidx.compose.material.Divider
|
||||||
|
import androidx.compose.material.DropdownMenu
|
||||||
|
import androidx.compose.material.DropdownMenuItem
|
||||||
|
import androidx.compose.material.LocalTextStyle
|
||||||
|
import androidx.compose.material.MaterialTheme
|
||||||
|
import androidx.compose.material.Text
|
||||||
|
import androidx.compose.runtime.Composable
|
||||||
|
import androidx.compose.runtime.LaunchedEffect
|
||||||
|
import androidx.compose.runtime.getValue
|
||||||
|
import androidx.compose.runtime.livedata.observeAsState
|
||||||
|
import androidx.compose.runtime.mutableStateOf
|
||||||
|
import androidx.compose.runtime.remember
|
||||||
|
import androidx.compose.runtime.setValue
|
||||||
|
import androidx.compose.ui.Modifier
|
||||||
|
import androidx.compose.ui.platform.LocalContext
|
||||||
|
import androidx.compose.ui.text.SpanStyle
|
||||||
|
import androidx.compose.ui.text.buildAnnotatedString
|
||||||
|
import androidx.compose.ui.text.style.TextOverflow
|
||||||
|
import androidx.compose.ui.text.withStyle
|
||||||
|
import androidx.compose.ui.unit.dp
|
||||||
|
import androidx.core.os.ConfigurationCompat
|
||||||
|
import androidx.navigation.NavController
|
||||||
|
import com.vitorpamplona.amethyst.service.lang.LanguageTranslatorService
|
||||||
|
import com.vitorpamplona.amethyst.service.lang.ResultOrError
|
||||||
|
import com.vitorpamplona.amethyst.ui.screen.loggedIn.AccountViewModel
|
||||||
|
import java.util.Locale
|
||||||
|
|
||||||
|
@Composable
|
||||||
|
fun TranslateableRichTextViewer(
|
||||||
|
content: String,
|
||||||
|
canPreview: Boolean,
|
||||||
|
tags: List<List<String>>?,
|
||||||
|
accountViewModel: AccountViewModel,
|
||||||
|
navController: NavController
|
||||||
|
) {
|
||||||
|
val translatedTextState = remember {
|
||||||
|
mutableStateOf(ResultOrError(content, null, null, null))
|
||||||
|
}
|
||||||
|
|
||||||
|
var showOriginal by remember { mutableStateOf(false) }
|
||||||
|
var langSettingsPopupExpanded by remember { mutableStateOf(false) }
|
||||||
|
|
||||||
|
val context = LocalContext.current
|
||||||
|
|
||||||
|
val accountState by accountViewModel.accountLanguagesLiveData.observeAsState()
|
||||||
|
val account = accountState?.account ?: return
|
||||||
|
|
||||||
|
LaunchedEffect(accountState) {
|
||||||
|
LanguageTranslatorService.autoTranslate(content, account.dontTranslateFrom, account.translateTo)
|
||||||
|
.addOnCompleteListener { task ->
|
||||||
|
if (task.isSuccessful) {
|
||||||
|
translatedTextState.value = task.result
|
||||||
|
} else {
|
||||||
|
translatedTextState.value = ResultOrError(content, null, null, null)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
val toBeViewed = if (showOriginal) content else translatedTextState.value.result ?: content
|
||||||
|
|
||||||
|
Column(modifier = Modifier.padding(top = 5.dp)) {
|
||||||
|
ExpandableRichTextViewer(
|
||||||
|
toBeViewed,
|
||||||
|
canPreview,
|
||||||
|
tags,
|
||||||
|
navController
|
||||||
|
)
|
||||||
|
|
||||||
|
val target = translatedTextState.value.targetLang
|
||||||
|
val source = translatedTextState.value.sourceLang
|
||||||
|
|
||||||
|
if (source != null && target != null) {
|
||||||
|
if (source != target) {
|
||||||
|
Row(modifier = Modifier.fillMaxWidth().padding(top = 5.dp)) {
|
||||||
|
val clickableTextStyle =
|
||||||
|
SpanStyle(color = MaterialTheme.colors.primary.copy(alpha = 0.52f))
|
||||||
|
|
||||||
|
val annotatedTranslationString = buildAnnotatedString {
|
||||||
|
withStyle(clickableTextStyle) {
|
||||||
|
pushStringAnnotation("langSettings", true.toString())
|
||||||
|
append("Auto")
|
||||||
|
}
|
||||||
|
|
||||||
|
append("-translated from ")
|
||||||
|
|
||||||
|
withStyle(clickableTextStyle) {
|
||||||
|
pushStringAnnotation("showOriginal", true.toString())
|
||||||
|
append(Locale(source).displayName)
|
||||||
|
}
|
||||||
|
|
||||||
|
append(" to ")
|
||||||
|
|
||||||
|
withStyle(clickableTextStyle) {
|
||||||
|
pushStringAnnotation("showOriginal", false.toString())
|
||||||
|
append(Locale(target).displayName)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
ClickableText(
|
||||||
|
text = annotatedTranslationString,
|
||||||
|
style = LocalTextStyle.current.copy(color = MaterialTheme.colors.onSurface.copy(alpha = 0.32f)),
|
||||||
|
overflow = TextOverflow.Visible,
|
||||||
|
maxLines = 3
|
||||||
|
) { spanOffset ->
|
||||||
|
annotatedTranslationString.getStringAnnotations(spanOffset, spanOffset)
|
||||||
|
.firstOrNull()
|
||||||
|
?.also { span ->
|
||||||
|
if (span.tag == "showOriginal")
|
||||||
|
showOriginal = span.item.toBoolean()
|
||||||
|
else
|
||||||
|
langSettingsPopupExpanded = !langSettingsPopupExpanded
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
DropdownMenu(
|
||||||
|
expanded = langSettingsPopupExpanded,
|
||||||
|
onDismissRequest = { langSettingsPopupExpanded = false }
|
||||||
|
) {
|
||||||
|
DropdownMenuItem(onClick = {
|
||||||
|
accountViewModel.dontTranslateFrom(source, context)
|
||||||
|
langSettingsPopupExpanded = false
|
||||||
|
}) {
|
||||||
|
Text("Never translate from ${Locale(source).displayName}")
|
||||||
|
}
|
||||||
|
Divider()
|
||||||
|
val languageList =
|
||||||
|
ConfigurationCompat.getLocales(Resources.getSystem().getConfiguration())
|
||||||
|
for (i in 0 until languageList.size()) {
|
||||||
|
languageList.get(i)?.let { lang ->
|
||||||
|
DropdownMenuItem(onClick = {
|
||||||
|
accountViewModel.translateTo(lang, context)
|
||||||
|
langSettingsPopupExpanded = false
|
||||||
|
}) {
|
||||||
|
Text("Always translate to ${lang.displayName}")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
Loading…
x
Reference in New Issue
Block a user