/** * Copyright (c) 2024 Vitor Pamplona * * Permission is hereby granted, free of charge, to any person obtaining a copy of * this software and associated documentation files (the "Software"), to deal in * the Software without restriction, including without limitation the rights to use, * copy, modify, merge, publish, distribute, sublicense, and/or sell copies of the * Software, and to permit persons to whom the Software is furnished to do so, * subject to the following conditions: * * The above copyright notice and this permission notice shall be included in all * copies or substantial portions of the Software. * * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS * FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR * COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN * AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION * WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. */ package com.vitorpamplona.amethyst.commons.compose import android.util.LruCache import androidx.compose.runtime.Composable import androidx.compose.runtime.State import androidx.compose.runtime.produceState @Composable fun produceCachedStateAsync( cache: AsyncCachedState, key: K, ): State { return produceState(initialValue = cache.cached(key), key1 = key) { cache.update(key) { value = it } } } @Composable fun produceCachedStateAsync( cache: AsyncCachedState, key: String, updateValue: K, ): State { return produceState(initialValue = cache.cached(updateValue), key1 = key) { cache.update(updateValue) { value = it } } } interface AsyncCachedState { fun cached(k: K): V? suspend fun update( k: K, onReady: (V?) -> Unit, ) } abstract class GenericBaseCacheAsync(capacity: Int) : AsyncCachedState { private val cache = LruCache(capacity) override fun cached(k: K): V? { return cache[k] } override suspend fun update( k: K, onReady: (V?) -> Unit, ) { cache[k]?.let { onReady(it) } compute(k) { if (it != null) { cache.put(k, it) } onReady(it) } } abstract suspend fun compute( key: K, onReady: (V?) -> Unit, ) }