From 4c190a22b91130c05c319c8c0ef3e2ddf6db04ea Mon Sep 17 00:00:00 2001 From: Matt Ramotar Date: Sat, 16 Mar 2024 11:48:30 -0400 Subject: [PATCH] Implement RealMutablePagingBuffer (#623) Signed-off-by: mramotar --- .../core/impl/RealMutablePagingBuffer.kt | 93 +++++++++++++++++++ .../paging/core/mutablePagingBufferOf.kt | 7 ++ 2 files changed, 100 insertions(+) create mode 100644 paging/core/src/commonMain/kotlin/org/mobilenativefoundation/paging/core/impl/RealMutablePagingBuffer.kt create mode 100644 paging/core/src/commonMain/kotlin/org/mobilenativefoundation/paging/core/mutablePagingBufferOf.kt diff --git a/paging/core/src/commonMain/kotlin/org/mobilenativefoundation/paging/core/impl/RealMutablePagingBuffer.kt b/paging/core/src/commonMain/kotlin/org/mobilenativefoundation/paging/core/impl/RealMutablePagingBuffer.kt new file mode 100644 index 000000000..d73194e9e --- /dev/null +++ b/paging/core/src/commonMain/kotlin/org/mobilenativefoundation/paging/core/impl/RealMutablePagingBuffer.kt @@ -0,0 +1,93 @@ +package org.mobilenativefoundation.paging.core.impl + +import org.mobilenativefoundation.paging.core.MutablePagingBuffer +import org.mobilenativefoundation.paging.core.PagingKey +import org.mobilenativefoundation.paging.core.PagingSource + +/** + * A concrete implementation of [MutablePagingBuffer], a custom data structure for efficiently storing and retrieving paging data. + * + * @param Id The type of the unique identifier for each item in the paged data. + * @param K The type of the key used for paging. + * @param P The type of the parameters associated with each page of data. + * @param D The type of the data items. + * @param E The type of errors that can occur during the paging process. + * @param A The type of custom actions that can be dispatched to modify the paging state. + * @property maxSize The maximum size of the buffer. + */ +class RealMutablePagingBuffer, K : Any, P : Any, D : Any, E : Any, A : Any>( + private val maxSize: Int +) : MutablePagingBuffer { + + private val buffer: Array?> = arrayOfNulls(maxSize) + private val paramsToIndex: MutableMap, Int> = mutableMapOf() + private val keyToIndex: MutableMap, Int> = mutableMapOf() + private var head = 0 + private var tail = 0 + private var size = 0 + + override fun get(params: PagingSource.LoadParams): PagingSource.LoadResult.Data? { + val index = paramsToIndex[params] + return if (index != null) buffer[index] else null + } + + + override fun get(key: PagingKey): PagingSource.LoadResult.Data? { + val index = keyToIndex[key] + return if (index != null) buffer[index] else null + } + + + override fun put(params: PagingSource.LoadParams, page: PagingSource.LoadResult.Data) { + // Check if the buffer is full + if (size == maxSize) { + // Get the index of the oldest entry in the buffer + val oldestIndex = head + // Find the params associated with the oldest entry + val oldestParams = paramsToIndex.entries.first { it.value == oldestIndex }.key + // Remove the oldest entry from the maps + paramsToIndex.remove(oldestParams) + keyToIndex.remove(oldestParams.key) + // Remove the oldest entry from the buffer + buffer[oldestIndex] = null + // Update the head index to point to the next entry + head = (head + 1) % maxSize + } + // Get the index to insert the new page + val index = tail + // Insert the new page at the tail index + buffer[index] = page + // Update the maps with the new entry + paramsToIndex[params] = index + keyToIndex[params.key] = index + // Update the tail index to point to the next empty slot + tail = (tail + 1) % maxSize + // Update the size of the buffer + size = minOf(size + 1, maxSize) + } + + override fun head(): PagingSource.LoadResult.Data? { + return buffer[head] + } + + override fun getAll(): List> { + val pages = mutableListOf>() + var index = head + var count = 0 + while (count < size) { + val page = buffer[index] + if (page != null) { + pages.add(page) + } + index = (index + 1) % maxSize + count++ + } + return pages + } + + override fun isEmpty(): Boolean = size == 0 + + override fun indexOf(key: PagingKey): Int { + return keyToIndex[key] ?: -1 + } +} diff --git a/paging/core/src/commonMain/kotlin/org/mobilenativefoundation/paging/core/mutablePagingBufferOf.kt b/paging/core/src/commonMain/kotlin/org/mobilenativefoundation/paging/core/mutablePagingBufferOf.kt new file mode 100644 index 000000000..05d88fd64 --- /dev/null +++ b/paging/core/src/commonMain/kotlin/org/mobilenativefoundation/paging/core/mutablePagingBufferOf.kt @@ -0,0 +1,7 @@ +package org.mobilenativefoundation.paging.core + +import org.mobilenativefoundation.paging.core.impl.RealMutablePagingBuffer + +inline fun , K : Any, P : Any, D : Any, E : Any, A : Any> mutablePagingBufferOf(maxSize: Int): MutablePagingBuffer { + return RealMutablePagingBuffer(maxSize) +} \ No newline at end of file