Fixes in "TextSourcesList" creating in from "RawMessageEntities"

This commit is contained in:
InsanusMokrassar 2022-03-27 13:37:18 +06:00
parent a5ed2a82bb
commit 5ec5c3bdfd
4 changed files with 65 additions and 26 deletions

View File

@ -2,6 +2,9 @@
## 0.38.11 ## 0.38.11
* `Core`:
* Fixes in `TextSourcesList` creating in from `RawMessageEntities`
## 0.38.10 ## 0.38.10
* `API`: * `API`:

View File

@ -20,11 +20,11 @@ internal data class RawMessageEntity(
internal fun RawMessageEntity.asTextSource( internal fun RawMessageEntity.asTextSource(
source: String, source: String,
subParts: TextSourcesList subParts: List<Pair<Int, TextSource>>
): TextSource { ): TextSource {
val sourceSubstring: String = source.substring(range) val sourceSubstring: String = source.substring(range)
val subPartsWithRegulars by lazy { val subPartsWithRegulars by lazy {
subParts.fillWithRegulars(sourceSubstring) subParts.map { (it.first - offset) to it.second }.fillWithRegulars(sourceSubstring)
} }
return when (type) { return when (type) {
"mention" -> MentionTextSource(sourceSubstring, subPartsWithRegulars) "mention" -> MentionTextSource(sourceSubstring, subPartsWithRegulars)
@ -58,16 +58,14 @@ private inline operator fun <T : Comparable<T>> ClosedRange<T>.contains(other: C
return start <= other.start && endInclusive >= other.endInclusive return start <= other.start && endInclusive >= other.endInclusive
} }
internal fun TextSourcesList.fillWithRegulars(source: String): TextSourcesList { internal fun List<Pair<Int, TextSource>>.fillWithRegulars(source: String): TextSourcesList {
var index = 0 var index = 0
val result = mutableListOf<TextSource>() val result = mutableListOf<TextSource>()
for (i in 0 until size) { for (i in indices) {
val textSource = get(i) val (offset, textSource) = get(i)
val thisSourceInStart = source.startsWith(textSource.source, index) if (offset - index > 0) {
if (!thisSourceInStart) { result.add(regular(source.substring(index, offset)))
val regularEndIndex = source.indexOf(textSource.source, index) index = offset
result.add(regular(source.substring(index, regularEndIndex)))
index = regularEndIndex
} }
result.add(textSource) result.add(textSource)
index += textSource.source.length index += textSource.source.length
@ -83,9 +81,9 @@ internal fun TextSourcesList.fillWithRegulars(source: String): TextSourcesList {
private fun createTextSources( private fun createTextSources(
originalFullString: String, originalFullString: String,
entities: RawMessageEntities entities: RawMessageEntities
): TextSourcesList { ): List<Pair<Int, TextSource>> {
val mutableEntities = entities.toMutableList().apply { sortBy { it.offset } } val mutableEntities = entities.toMutableList().apply { sortBy { it.offset } }
val resultList = mutableListOf<TextSource>() val resultList = mutableListOf<Pair<Int, TextSource>>()
while (mutableEntities.isNotEmpty()) { while (mutableEntities.isNotEmpty()) {
var parent = mutableEntities.removeFirst() var parent = mutableEntities.removeFirst()
@ -129,7 +127,7 @@ private fun createTextSources(
emptyList() emptyList()
} }
resultList.add( resultList.add(
parent.asTextSource( parent.offset to parent.asTextSource(
originalFullString, originalFullString,
subtextSources subtextSources
) )

View File

@ -3,43 +3,43 @@ package dev.inmo.tgbotapi.types.MessageEntity
import dev.inmo.tgbotapi.types.MessageEntity.textsources.* import dev.inmo.tgbotapi.types.MessageEntity.textsources.*
import kotlin.test.assertTrue import kotlin.test.assertTrue
const val testText = "It is simple hello world with #tag and @mention" const val testText = "It (is?) is simple hello world with #tag and @mention"
const val formattedV2Text = "It *_is_ ~__simple__~* ||hello world|| with \\#tag and @mention" const val formattedV2Text = "It \\(is?\\) *_is_ ~__simple__~* ||hello world|| with \\#tag and @mention"
const val formattedHtmlText = "It <b><i>is</i> <s><u>simple</u></s></b> <span class=\"tg-spoiler\">hello world</span> with #tag and @mention" const val formattedHtmlText = "It (is?) <b><i>is</i> <s><u>simple</u></s></b> <span class=\"tg-spoiler\">hello world</span> with #tag and @mention"
internal val testTextEntities = listOf( internal val testTextEntities = listOf(
RawMessageEntity( RawMessageEntity(
"bold", "bold",
3, 9,
9 9
), ),
RawMessageEntity( RawMessageEntity(
"italic", "italic",
3, 9,
2 2
), ),
RawMessageEntity( RawMessageEntity(
"strikethrough", "strikethrough",
6, 12,
6 6
), ),
RawMessageEntity( RawMessageEntity(
"underline", "underline",
6, 12,
6 6
), ),
RawMessageEntity( RawMessageEntity(
"spoiler", "spoiler",
13, 19,
11 11
), ),
RawMessageEntity( RawMessageEntity(
"hashtag", "hashtag",
30, 36,
4 4
), ),
RawMessageEntity( RawMessageEntity(
"mention", "mention",
39, 45,
8 8
) )
) )

View File

@ -2,8 +2,7 @@ package dev.inmo.tgbotapi.types.MessageEntity
import dev.inmo.tgbotapi.extensions.utils.formatting.* import dev.inmo.tgbotapi.extensions.utils.formatting.*
import dev.inmo.tgbotapi.types.MessageEntity.textsources.* import dev.inmo.tgbotapi.types.MessageEntity.textsources.*
import kotlin.test.Test import kotlin.test.*
import kotlin.test.assertEquals
class StringFormattingTests { class StringFormattingTests {
@Test @Test
@ -38,7 +37,7 @@ class StringFormattingTests {
@Test @Test
fun testThatCreatingOfStringWithSimpleDSLWorksCorrectly() { fun testThatCreatingOfStringWithSimpleDSLWorksCorrectly() {
val sources: TextSourcesList = regular("It ") + val sources: TextSourcesList = regular("It (is?) ") +
bold(italic("is") + bold(italic("is") +
" " + " " +
strikethrough(underline("simple"))) + strikethrough(underline("simple"))) +
@ -53,4 +52,43 @@ class StringFormattingTests {
assertEquals(formattedV2Text, sources.toMarkdownV2Texts().first()) assertEquals(formattedV2Text, sources.toMarkdownV2Texts().first())
assertEquals(formattedHtmlText, sources.toHtmlTexts().first()) assertEquals(formattedHtmlText, sources.toHtmlTexts().first())
} }
@Test
fun testForRepeatingWordsInOneSentenceWithTheSecondOneFormatted() {
val sourceText = "link link"
val messageEntities = listOf(
RawMessageEntity("bold", 5, 4),
RawMessageEntity("text_link", 6, 2, "google.com")
)
val textSources = messageEntities.asTextSources(sourceText)
val (regular, bold) = textSources
assertTrue(regular is RegularTextSource)
assertTrue(bold is BoldTextSource)
assertTrue(regular.source == "link ")
assertTrue(bold.source == "link")
assertTrue((bold.subsources[0] as? RegularTextSource) ?.source == "l")
assertTrue((bold.subsources[1] as? TextLinkTextSource) ?.source == "in")
assertTrue((bold.subsources[1] as? TextLinkTextSource) ?.url == "google.com")
assertTrue((bold.subsources[2] as? RegularTextSource) ?.source == "k")
assertTrue(bold.subsources.size == 3)
assertTrue(textSources.size == 2)
}
@Test
fun testForRepeatingWordsInOneSentenceWithTheSecondOneFormattedInsideOfFormatting() {
val sourceText = "text"
val messageEntities = listOf(
RawMessageEntity("bold", 0, 4),
RawMessageEntity("text_link", 3, 1, "google.com")
)
val textSources = messageEntities.asTextSources(sourceText)
val (bold) = textSources
assertTrue(bold is BoldTextSource)
assertTrue(bold.source == "text")
assertTrue((bold.subsources[0] as? RegularTextSource) ?.source == "tex")
assertTrue((bold.subsources[1] as? TextLinkTextSource) ?.source == "t")
assertTrue((bold.subsources[1] as? TextLinkTextSource) ?.url == "google.com")
assertTrue(bold.subsources.size == 2)
assertTrue(textSources.size == 1)
}
} }