mirror of
				https://github.com/InsanusMokrassar/MicroUtils.git
				synced 2025-10-25 09:10:30 +00:00 
			
		
		
		
	Compare commits
	
		
			5 Commits
		
	
	
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
| f6d5035c1a | |||
| 43e782ab6f | |||
| f3f9920bfb | |||
| 2bfd615812 | |||
| ebfacb3659 | 
| @@ -1,5 +1,12 @@ | |||||||
| # Changelog | # Changelog | ||||||
|  |  | ||||||
|  | ## 0.20.18 | ||||||
|  |  | ||||||
|  | * `Coroutines`: | ||||||
|  |     * `SpecialMutableStateFlow` now extends `MutableStateFlow` | ||||||
|  |     * `Compose`: | ||||||
|  |         * Deprecate `FlowState` due to its complexity in fixes | ||||||
|  |  | ||||||
| ## 0.20.17 | ## 0.20.17 | ||||||
|  |  | ||||||
| * `Versions`: | * `Versions`: | ||||||
|   | |||||||
| @@ -1,10 +1,7 @@ | |||||||
| package dev.inmo.micro_utils.coroutines.compose | package dev.inmo.micro_utils.coroutines.compose | ||||||
|  |  | ||||||
| import androidx.compose.runtime.MutableState | import androidx.compose.runtime.MutableState | ||||||
| import androidx.compose.runtime.State |  | ||||||
| import androidx.compose.runtime.derivedStateOf |  | ||||||
| import dev.inmo.micro_utils.coroutines.SpecialMutableStateFlow | import dev.inmo.micro_utils.coroutines.SpecialMutableStateFlow | ||||||
| import dev.inmo.micro_utils.coroutines.doInUI |  | ||||||
| import kotlinx.coroutines.CoroutineScope | import kotlinx.coroutines.CoroutineScope | ||||||
| import kotlinx.coroutines.Dispatchers | import kotlinx.coroutines.Dispatchers | ||||||
|  |  | ||||||
| @@ -12,6 +9,7 @@ import kotlinx.coroutines.Dispatchers | |||||||
|  * This type works like [MutableState], [kotlinx.coroutines.flow.StateFlow] and [kotlinx.coroutines.flow.MutableSharedFlow]. |  * This type works like [MutableState], [kotlinx.coroutines.flow.StateFlow] and [kotlinx.coroutines.flow.MutableSharedFlow]. | ||||||
|  * Based on [SpecialMutableStateFlow] |  * Based on [SpecialMutableStateFlow] | ||||||
|  */ |  */ | ||||||
|  | @Deprecated("Will be removed soon") | ||||||
| class FlowState<T>( | class FlowState<T>( | ||||||
|     initial: T, |     initial: T, | ||||||
|     internalScope: CoroutineScope = CoroutineScope(Dispatchers.Default) |     internalScope: CoroutineScope = CoroutineScope(Dispatchers.Default) | ||||||
| @@ -25,9 +23,9 @@ class FlowState<T>( | |||||||
|             tryEmit(value) |             tryEmit(value) | ||||||
|         } |         } | ||||||
|  |  | ||||||
|     override suspend fun onChange(value: T) { |     override fun onChangeWithoutSync(value: T) { | ||||||
|         internalValue = value |         internalValue = value | ||||||
|         super.onChange(value) |         super.onChangeWithoutSync(value) | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     override fun component1(): T = value |     override fun component1(): T = value | ||||||
|   | |||||||
| @@ -3,10 +3,14 @@ package dev.inmo.micro_utils.coroutines | |||||||
| import kotlinx.coroutines.CoroutineScope | import kotlinx.coroutines.CoroutineScope | ||||||
| import kotlinx.coroutines.Dispatchers | import kotlinx.coroutines.Dispatchers | ||||||
| import kotlinx.coroutines.ExperimentalCoroutinesApi | import kotlinx.coroutines.ExperimentalCoroutinesApi | ||||||
|  | import kotlinx.coroutines.InternalCoroutinesApi | ||||||
| import kotlinx.coroutines.channels.BufferOverflow | import kotlinx.coroutines.channels.BufferOverflow | ||||||
| import kotlinx.coroutines.flow.FlowCollector | import kotlinx.coroutines.flow.FlowCollector | ||||||
| import kotlinx.coroutines.flow.MutableSharedFlow | import kotlinx.coroutines.flow.MutableSharedFlow | ||||||
|  | import kotlinx.coroutines.flow.MutableStateFlow | ||||||
| import kotlinx.coroutines.flow.StateFlow | import kotlinx.coroutines.flow.StateFlow | ||||||
|  | import kotlinx.coroutines.internal.SynchronizedObject | ||||||
|  | import kotlinx.coroutines.internal.synchronized | ||||||
|  |  | ||||||
| /** | /** | ||||||
|  * Works like [StateFlow], but guarantee that latest value update will always be delivered to |  * Works like [StateFlow], but guarantee that latest value update will always be delivered to | ||||||
| @@ -15,7 +19,9 @@ import kotlinx.coroutines.flow.StateFlow | |||||||
| open class SpecialMutableStateFlow<T>( | open class SpecialMutableStateFlow<T>( | ||||||
|     initialValue: T, |     initialValue: T, | ||||||
|     internalScope: CoroutineScope = CoroutineScope(Dispatchers.Default) |     internalScope: CoroutineScope = CoroutineScope(Dispatchers.Default) | ||||||
| ) : StateFlow<T>, FlowCollector<T>, MutableSharedFlow<T> { | ) : MutableStateFlow<T>, FlowCollector<T>, MutableSharedFlow<T> { | ||||||
|  |     @OptIn(InternalCoroutinesApi::class) | ||||||
|  |     private val syncObject = SynchronizedObject() | ||||||
|     protected val internalSharedFlow: MutableSharedFlow<T> = MutableSharedFlow( |     protected val internalSharedFlow: MutableSharedFlow<T> = MutableSharedFlow( | ||||||
|         replay = 0, |         replay = 0, | ||||||
|         extraBufferCapacity = 2, |         extraBufferCapacity = 2, | ||||||
| @@ -28,16 +34,13 @@ open class SpecialMutableStateFlow<T>( | |||||||
|     ) |     ) | ||||||
|  |  | ||||||
|     protected var _value: T = initialValue |     protected var _value: T = initialValue | ||||||
|     override val value: T |     override var value: T | ||||||
|         get() = _value |         get() = _value | ||||||
|     protected open suspend fun onChange(value: T) { |         set(value) { | ||||||
|         _value = value |             doOnChangeAction(value) | ||||||
|         publicSharedFlow.emit(value) |  | ||||||
|     } |  | ||||||
|     protected val job = internalSharedFlow.subscribe(internalScope) { |  | ||||||
|         if (_value != it) { |  | ||||||
|             onChange(it) |  | ||||||
|         } |         } | ||||||
|  |     protected val job = internalSharedFlow.subscribe(internalScope) { | ||||||
|  |         doOnChangeAction(it) | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     override val replayCache: List<T> |     override val replayCache: List<T> | ||||||
| @@ -45,6 +48,29 @@ open class SpecialMutableStateFlow<T>( | |||||||
|     override val subscriptionCount: StateFlow<Int> |     override val subscriptionCount: StateFlow<Int> | ||||||
|         get() = publicSharedFlow.subscriptionCount |         get() = publicSharedFlow.subscriptionCount | ||||||
|  |  | ||||||
|  |     @OptIn(InternalCoroutinesApi::class) | ||||||
|  |     override fun compareAndSet(expect: T, update: T): Boolean { | ||||||
|  |         return synchronized(syncObject) { | ||||||
|  |             if (expect == _value && update != _value) { | ||||||
|  |                 doOnChangeAction(update) | ||||||
|  |             } | ||||||
|  |             expect == _value | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     protected open fun onChangeWithoutSync(value: T) { | ||||||
|  |         _value = value | ||||||
|  |         publicSharedFlow.tryEmit(value) | ||||||
|  |     } | ||||||
|  |     @OptIn(InternalCoroutinesApi::class) | ||||||
|  |     protected open fun doOnChangeAction(value: T) { | ||||||
|  |         synchronized(syncObject) { | ||||||
|  |             if (_value != value) { | ||||||
|  |                 onChangeWithoutSync(value) | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |  | ||||||
|     @ExperimentalCoroutinesApi |     @ExperimentalCoroutinesApi | ||||||
|     override fun resetReplayCache() = publicSharedFlow.resetReplayCache() |     override fun resetReplayCache() = publicSharedFlow.resetReplayCache() | ||||||
|  |  | ||||||
|   | |||||||
| @@ -15,5 +15,5 @@ crypto_js_version=4.1.1 | |||||||
| # Project data | # Project data | ||||||
|  |  | ||||||
| group=dev.inmo | group=dev.inmo | ||||||
| version=0.20.17 | version=0.20.18 | ||||||
| android_code_version=223 | android_code_version=224 | ||||||
|   | |||||||
		Reference in New Issue
	
	Block a user