Не совсем идиоматический способ, который я нашел, но, возможно, он удовлетворяет вашим потребностям:
fun String.chunkedWords(limitChars: Int,
delimiter: Char = ' ',
joinCharacter: Char = '\n') =
splitToSequence(delimiter)
.reduce { cumulatedString, word ->
val exceedsSize = cumulatedString.length - cumulatedString.indexOfLast { it == joinCharacter } + "$delimiter$word".length > limitChars
cumulatedString + if (exceedsSize) {
joinCharacter
} else {
delimiter
} + word
}
Затем вы можете использовать его следующим образом:
longText.chunkedWords(40).run(::println)
, что для данной строкизатем напечатает:
Lorem ipsum dolor sit amet, consectetur
adipisicing elit, sed do eiusmod tempor
incididunt ut labore et dolore magna
aliqua. Ut enim ad minim veniam, quis
nostrud exercitation ullamco laboris
nisi ut aliquip ex ea commodo consequat.
Duis aute irure dolor in reprehenderit
in voluptate velit esse cillum dolore eu
fugiat nulla pariatur. Excepteur sint
occaecat cupidatat non proident, sunt in
culpa qui officia deserunt mollit anim
id est laborum.
Вы также можете разбить его на строки оттуда, например, longText.chunkedWords(40).splitAsSequence("\n")
.Обратите внимание, что он также хорошо разделяется, если в строке уже есть символы новой строки, т. Е. Если у вас есть String
как "Testing shorter lines.\nAnd now there comes a very long line"
, вызов .chunkedWords(17)
даст следующий вывод:
Testing shorter
lines.
And now there // this tries to use the whole 17 characters again
comes a very
long line