You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
157 lines
5.7 KiB
157 lines
5.7 KiB
package org.jetbrains.compose.resources |
|
|
|
import androidx.compose.runtime.Composable |
|
import androidx.compose.runtime.getValue |
|
import kotlinx.coroutines.Dispatchers |
|
import kotlinx.coroutines.ExperimentalCoroutinesApi |
|
import kotlinx.coroutines.withContext |
|
import org.jetbrains.compose.resources.vector.xmldom.Element |
|
import org.jetbrains.compose.resources.vector.xmldom.NodeList |
|
|
|
private const val STRINGS_XML = "strings.xml" //todo |
|
private val SimpleStringFormatRegex = Regex("""%(\d)\$[ds]""") |
|
|
|
private sealed interface StringItem { |
|
data class Value(val text: String) : StringItem |
|
data class Array(val items: List<String>) : StringItem |
|
} |
|
|
|
@OptIn(ExperimentalCoroutinesApi::class) |
|
private val stringsCacheDispatcher = Dispatchers.Default.limitedParallelism(1) |
|
private val parsedStringsCache = mutableMapOf<String, Map<String, StringItem>>() |
|
|
|
//@TestOnly |
|
internal fun dropStringsCache() { |
|
parsedStringsCache.clear() |
|
} |
|
|
|
private suspend fun getParsedStrings(path: String, resourceReader: ResourceReader): Map<String, StringItem> = |
|
withContext(stringsCacheDispatcher) { |
|
parsedStringsCache.getOrPut(path) { |
|
val nodes = resourceReader.read(path).toXmlElement().childNodes |
|
val strings = nodes.getElementsWithName("string").associate { element -> |
|
element.getAttribute("name") to StringItem.Value(element.textContent.orEmpty()) |
|
} |
|
val arrays = nodes.getElementsWithName("string-array").associate { arrayElement -> |
|
val items = arrayElement.childNodes.getElementsWithName("item").map { element -> |
|
element.textContent.orEmpty() |
|
} |
|
arrayElement.getAttribute("name") to StringItem.Array(items) |
|
} |
|
strings + arrays |
|
} |
|
} |
|
|
|
/** |
|
* Retrieves a string resource using the provided ID. |
|
* |
|
* @param id The ID of the string resource to retrieve. |
|
* @return The retrieved string resource. |
|
* |
|
* @throws IllegalArgumentException If the provided ID is not found in the resource file. |
|
*/ |
|
@ExperimentalResourceApi |
|
@Composable |
|
fun getString(id: ResourceId): String { |
|
val resourceReader = LocalResourceReader.current |
|
val str by rememberState(id, { "" }) { loadString(id, resourceReader) } |
|
return str |
|
} |
|
|
|
/** |
|
* Loads a string resource using the provided ID. |
|
* |
|
* @param id The ID of the string resource to load. |
|
* @return The loaded string resource. |
|
* |
|
* @throws IllegalArgumentException If the provided ID is not found in the resource file. |
|
*/ |
|
@ExperimentalResourceApi |
|
suspend fun loadString(id: ResourceId): String = loadString(id, DefaultResourceReader) |
|
|
|
private suspend fun loadString(id: ResourceId, resourceReader: ResourceReader): String { |
|
val nameToValue = getParsedStrings(getPathById(STRINGS_XML), resourceReader) |
|
val item = nameToValue[id.stringKey] as? StringItem.Value |
|
?: error("String ID=`${id.stringKey}` is not found!") |
|
return item.text |
|
} |
|
|
|
/** |
|
* Retrieves a formatted string resource using the provided ID and arguments. |
|
* |
|
* @param id The ID of the string resource to retrieve. |
|
* @param formatArgs The arguments to be inserted into the formatted string. |
|
* @return The formatted string resource. |
|
* |
|
* @throws IllegalArgumentException If the provided ID is not found in the resource file. |
|
*/ |
|
@ExperimentalResourceApi |
|
@Composable |
|
fun getString(id: ResourceId, vararg formatArgs: Any): String { |
|
val resourceReader = LocalResourceReader.current |
|
val args = formatArgs.map { it.toString() } |
|
val str by rememberState(id, { "" }) { loadString(id, args, resourceReader) } |
|
return str |
|
} |
|
|
|
/** |
|
* Loads a formatted string resource using the provided ID and arguments. |
|
* |
|
* @param id The ID of the string resource to load. |
|
* @param formatArgs The arguments to be inserted into the formatted string. |
|
* @return The formatted string resource. |
|
* |
|
* @throws IllegalArgumentException If the provided ID is not found in the resource file. |
|
*/ |
|
@ExperimentalResourceApi |
|
suspend fun loadString(id: ResourceId, vararg formatArgs: Any): String = loadString( |
|
id, |
|
formatArgs.map { it.toString() }, |
|
DefaultResourceReader |
|
) |
|
|
|
private suspend fun loadString(id: ResourceId, args: List<String>, resourceReader: ResourceReader): String { |
|
val str = loadString(id, resourceReader) |
|
return SimpleStringFormatRegex.replace(str) { matchResult -> |
|
args[matchResult.groupValues[1].toInt() - 1] |
|
} |
|
} |
|
|
|
/** |
|
* Retrieves a list of strings from a string array resource. |
|
* |
|
* @param id The ID of the string array resource. |
|
* @return A list of strings representing the items in the string array. |
|
* |
|
* @throws IllegalStateException if the string array with the given ID is not found. |
|
*/ |
|
@ExperimentalResourceApi |
|
@Composable |
|
fun getStringArray(id: ResourceId): List<String> { |
|
val resourceReader = LocalResourceReader.current |
|
val array by rememberState(id, { emptyList() }) { loadStringArray(id, resourceReader) } |
|
return array |
|
} |
|
|
|
/** |
|
* Loads a string array from a resource file. |
|
* |
|
* @param id The ID of the string array resource. |
|
* @return A list of strings representing the items in the string array. |
|
* |
|
* @throws IllegalStateException if the string array with the given ID is not found. |
|
*/ |
|
@ExperimentalResourceApi |
|
suspend fun loadStringArray(id: ResourceId): List<String> = loadStringArray(id, DefaultResourceReader) |
|
|
|
private suspend fun loadStringArray(id: ResourceId, resourceReader: ResourceReader): List<String> { |
|
val nameToValue = getParsedStrings(getPathById(STRINGS_XML), resourceReader) |
|
val item = nameToValue[id.stringKey] as? StringItem.Array |
|
?: error("String array ID=`${id.stringKey}` is not found!") |
|
return item.items |
|
} |
|
|
|
private fun NodeList.getElementsWithName(name: String): List<Element> = |
|
List(length) { item(it) } |
|
.filterIsInstance<Element>() |
|
.filter { it.localName == name } |