mirror of
				https://github.com/InsanusMokrassar/MicroUtils.git
				synced 2025-10-26 17:50:41 +00:00 
			
		
		
		
	Compare commits
	
		
			7 Commits
		
	
	
		
			renovate/k
			...
			0.9.25
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
| 216c03205c | |||
| ab112aa7a4 | |||
| d85b3d0da9 | |||
| 67b9a03366 | |||
| 3ac56dcfd3 | |||
| d1021d283a | |||
| 97ed973cb5 | 
							
								
								
									
										7
									
								
								.github/workflows/dokka_push.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										7
									
								
								.github/workflows/dokka_push.yml
									
									
									
									
										vendored
									
									
								
							| @@ -10,9 +10,12 @@ jobs: | |||||||
|       - uses: actions/checkout@v2 |       - uses: actions/checkout@v2 | ||||||
|       - uses: actions/setup-java@v1 |       - uses: actions/setup-java@v1 | ||||||
|         with: |         with: | ||||||
|           java-version: 17 |           java-version: 11 | ||||||
|  |       - name: Fix android 32.0.0 dx | ||||||
|  |         continue-on-error: true | ||||||
|  |         run: cd /usr/local/lib/android/sdk/build-tools/32.0.0/ && mv d8 dx && cd lib  && mv d8.jar dx.jar | ||||||
|       - name: Build |       - name: Build | ||||||
|         run: ./gradlew build && ./gradlew dokkaHtml |         run: ./gradlew dokkaHtml | ||||||
|       - name: Publish KDocs |       - name: Publish KDocs | ||||||
|         uses: peaceiris/actions-gh-pages@v3 |         uses: peaceiris/actions-gh-pages@v3 | ||||||
|         with: |         with: | ||||||
|   | |||||||
| @@ -1,14 +1,17 @@ | |||||||
| 
 | 
 | ||||||
| name: Build | name: Publish package to GitHub Packages | ||||||
| on: [push] | on: [push] | ||||||
| jobs: | jobs: | ||||||
|   build: |   publishing: | ||||||
|     runs-on: ubuntu-latest |     runs-on: ubuntu-latest | ||||||
|     steps: |     steps: | ||||||
|       - uses: actions/checkout@v2 |       - uses: actions/checkout@v2 | ||||||
|       - uses: actions/setup-java@v1 |       - uses: actions/setup-java@v1 | ||||||
|         with: |         with: | ||||||
|           java-version: 17 |           java-version: 11 | ||||||
|  |       - name: Fix android 32.0.0 dx | ||||||
|  |         continue-on-error: true | ||||||
|  |         run: cd /usr/local/lib/android/sdk/build-tools/32.0.0/ && mv d8 dx && cd lib  && mv d8.jar dx.jar | ||||||
|       - name: Rewrite version |       - name: Rewrite version | ||||||
|         run: | |         run: | | ||||||
|           branch="`echo "${{ github.ref }}" | grep -o "[^/]*$"`" |           branch="`echo "${{ github.ref }}" | grep -o "[^/]*$"`" | ||||||
| @@ -17,9 +20,9 @@ jobs: | |||||||
|           mv gradle.properties.tmp gradle.properties |           mv gradle.properties.tmp gradle.properties | ||||||
|       - name: Build |       - name: Build | ||||||
|         run: ./gradlew build |         run: ./gradlew build | ||||||
|       - name: Publish to InmoNexus |       - name: Publish | ||||||
|         continue-on-error: true |         continue-on-error: true | ||||||
|         run: ./gradlew publishAllPublicationsToInmoNexusRepository |         run: ./gradlew --no-parallel publishAllPublicationsToGithubPackagesRepository | ||||||
|         env: |         env: | ||||||
|           INMONEXUS_USER: ${{ secrets.INMONEXUS_USER }} |           GITHUBPACKAGES_USER: ${{ github.actor }} | ||||||
|           INMONEXUS_PASSWORD: ${{ secrets.INMONEXUS_PASSWORD }} |           GITHUBPACKAGES_PASSWORD: ${{ secrets.GITHUB_TOKEN }} | ||||||
							
								
								
									
										9
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										9
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							| @@ -1,6 +1,4 @@ | |||||||
| .idea | .idea | ||||||
| .vscode |  | ||||||
| .kotlin |  | ||||||
| out/* | out/* | ||||||
| *.iml | *.iml | ||||||
| target | target | ||||||
| @@ -10,16 +8,9 @@ settings.xml | |||||||
| .gradle/ | .gradle/ | ||||||
| build/ | build/ | ||||||
| out/ | out/ | ||||||
| bin/ |  | ||||||
|  |  | ||||||
| secret.gradle | secret.gradle | ||||||
| local.properties | local.properties | ||||||
| kotlin-js-store | kotlin-js-store | ||||||
|  |  | ||||||
| publishing.sh | publishing.sh | ||||||
|  |  | ||||||
| local.* |  | ||||||
| local/ |  | ||||||
| **/*.local.* |  | ||||||
|  |  | ||||||
| .kotlin/ |  | ||||||
|   | |||||||
							
								
								
									
										8
									
								
								.space.kts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										8
									
								
								.space.kts
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,8 @@ | |||||||
|  | job("Build and run tests") { | ||||||
|  |     container(displayName = "Run gradle build", image = "openjdk:11") { | ||||||
|  |         kotlinScript { api -> | ||||||
|  |             // here can be your complex logic | ||||||
|  |             api.gradlew("build") | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | } | ||||||
							
								
								
									
										1542
									
								
								CHANGELOG.md
									
									
									
									
									
								
							
							
						
						
									
										1542
									
								
								CHANGELOG.md
									
									
									
									
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							
							
								
								
									
										48
									
								
								README.md
									
									
									
									
									
								
							
							
						
						
									
										48
									
								
								README.md
									
									
									
									
									
								
							| @@ -35,51 +35,3 @@ Most of complex modules are built with next hierarchy: | |||||||
|     * `common` part contains routes which are common for clients and servers |     * `common` part contains routes which are common for clients and servers | ||||||
|     * `client` submodule contains clients which are usually using `UnifiedRequester` to make requests using routes from `ktor/common` module and some internal logic of requests |     * `client` submodule contains clients which are usually using `UnifiedRequester` to make requests using routes from `ktor/common` module and some internal logic of requests | ||||||
|     * `server` submodule (in most cases `JVM`-only) contains some extensions for `Route` instances which usually will give opportunity to proxy internet requests from `ktor/client` realization to some proxy object |     * `server` submodule (in most cases `JVM`-only) contains some extensions for `Route` instances which usually will give opportunity to proxy internet requests from `ktor/client` realization to some proxy object | ||||||
|  |  | ||||||
| ## Gradle Templates |  | ||||||
|  |  | ||||||
| All templates can be used by applying them in your project's build.gradle files using the `apply from` directive. For example: |  | ||||||
|  |  | ||||||
| ```gradle |  | ||||||
| apply from: "$defaultProject" |  | ||||||
| ``` |  | ||||||
|  |  | ||||||
| In the sample has been used `defaultProject.gradle` as a basic template. |  | ||||||
|  |  | ||||||
| The project includes a collection of Gradle templates to simplify project setup and configuration. These templates are located in the `gradle/templates` directory and can be used to quickly set up different types of projects: |  | ||||||
|  |  | ||||||
| ### Project Setup Templates |  | ||||||
|  |  | ||||||
| * `defaultProject.gradle` (usage `apply from: "$defaultProject"`) - Basic project configuration |  | ||||||
| * `defaultProjectWithSerialization.gradle` (usage `apply from: "$defaultProjectWithSerialization"`) - Project configuration with Kotlin Serialization support |  | ||||||
| * `mppJavaProject.gradle` (usage `apply from: "$mppJavaProject"`) - Multiplatform project with Java support |  | ||||||
| * `mppAndroidProject.gradle` (usage `apply from: "$mppAndroidProject"`) - Multiplatform project with Android support |  | ||||||
|  |  | ||||||
| ### Multiplatform Configuration Templates |  | ||||||
|  |  | ||||||
| * `enableMPPAndroid.gradle` (usage `apply from: "$enableMPPAndroid"`) - Enable Android target in multiplatform project |  | ||||||
| * `enableMPPJs.gradle` (usage `apply from: "$enableMPPJs"`) - Enable JavaScript target in multiplatform project |  | ||||||
| * `enableMPPJvm.gradle` (usage `apply from: "$enableMPPJvm"`) - Enable JVM target in multiplatform project |  | ||||||
| * `enableMPPNativeArm64.gradle` (usage `apply from: "$enableMPPNativeArm64"`) - Enable ARM64 native target |  | ||||||
| * `enableMPPNativeX64.gradle` (usage `apply from: "$enableMPPNativeX64"`) - Enable x64 native target |  | ||||||
| * `enableMPPWasmJs.gradle` (usage `apply from: "$enableMPPWasmJs"`) - Enable WebAssembly JavaScript target |  | ||||||
|  |  | ||||||
| ### Compose Integration Templates |  | ||||||
|  |  | ||||||
| * `addCompose.gradle` (usage `apply from: "$addCompose"`) - Basic Compose configuration |  | ||||||
| * `addComposeForAndroid.gradle` (usage `apply from: "$addComposeForAndroid"`) - Compose configuration for Android |  | ||||||
| * `addComposeForDesktop.gradle` (usage `apply from: "$addComposeForDesktop"`) - Compose configuration for Desktop |  | ||||||
| * `addComposeForJs.gradle` (usage `apply from: "$addComposeForJs"`) - Compose configuration for JavaScript |  | ||||||
|  |  | ||||||
| ### Publishing Templates |  | ||||||
|  |  | ||||||
| * `publish.gradle` (usage `apply from: "$publish"`) - General publishing configuration |  | ||||||
| * `publish_jvm.gradle` (usage `apply from: "$publish_jvm"`) - JVM-specific publishing configuration |  | ||||||
| * `publish.kpsb` and `publish_jvm.kpsb` (usage `apply from: "$publish_kpsb"` and `apply from: "$publish_jvm_kpsb"`) - Publishing configuration for Kotlin Multiplatform and JVM |  | ||||||
|  |  | ||||||
| ### Combined Project Templates |  | ||||||
|  |  | ||||||
| * `mppJvmJsWasmJsLinuxMingwProject.gradle` (usage `apply from: "$mppJvmJsWasmJsLinuxMingwProject"`) - Multiplatform project for JVM, JS, Wasm, Linux, and MinGW |  | ||||||
| * `mppJvmJsWasmJsAndroidLinuxMingwLinuxArm64Project.gradle` (usage `apply from: "$mppJvmJsWasmJsAndroidLinuxMingwLinuxArm64Project"`) - Multiplatform project with additional Android and ARM64 support |  | ||||||
| * `mppComposeJvmJsWasmJsAndroidLinuxMingwLinuxArm64Project.gradle` (usage `apply from: "$mppComposeJvmJsWasmJsAndroidLinuxMingwLinuxArm64Project"`) - Multiplatform project with Compose support |  | ||||||
| * `mppProjectWithSerializationAndCompose.gradle` (usage `apply from: "$mppProjectWithSerializationAndCompose"`) - Multiplatform project with both Serialization and Compose support |  | ||||||
|   | |||||||
| @@ -4,7 +4,7 @@ plugins { | |||||||
|     id "com.android.library" |     id "com.android.library" | ||||||
| } | } | ||||||
|  |  | ||||||
| apply from: "$mppAndroidProject" | apply from: "$mppAndroidProjectPresetPath" | ||||||
|  |  | ||||||
| kotlin { | kotlin { | ||||||
|     sourceSets { |     sourceSets { | ||||||
|   | |||||||
| @@ -1 +0,0 @@ | |||||||
| <manifest/> |  | ||||||
							
								
								
									
										1
									
								
								android/alerts/common/src/main/AndroidManifest.xml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										1
									
								
								android/alerts/common/src/main/AndroidManifest.xml
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1 @@ | |||||||
|  | <manifest package="dev.inmo.micro_utils.android.alerts.common"/> | ||||||
| @@ -4,7 +4,7 @@ plugins { | |||||||
|     id "com.android.library" |     id "com.android.library" | ||||||
| } | } | ||||||
|  |  | ||||||
| apply from: "$mppAndroidProject" | apply from: "$mppAndroidProjectPresetPath" | ||||||
|  |  | ||||||
| kotlin { | kotlin { | ||||||
|     sourceSets { |     sourceSets { | ||||||
|   | |||||||
| @@ -1 +0,0 @@ | |||||||
| <manifest/> |  | ||||||
							
								
								
									
										1
									
								
								android/alerts/recyclerview/src/main/AndroidManifest.xml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										1
									
								
								android/alerts/recyclerview/src/main/AndroidManifest.xml
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1 @@ | |||||||
|  | <manifest package="dev.inmo.micro_utils.android.alerts.recyclerview"/> | ||||||
| @@ -1,19 +0,0 @@ | |||||||
| plugins { |  | ||||||
|     id "org.jetbrains.kotlin.multiplatform" |  | ||||||
|     id "org.jetbrains.kotlin.plugin.serialization" |  | ||||||
|     id "com.android.library" |  | ||||||
|     alias(libs.plugins.jb.compose) |  | ||||||
|     alias(libs.plugins.kt.jb.compose) |  | ||||||
| } |  | ||||||
|  |  | ||||||
| apply from: "$mppProjectWithSerializationAndCompose" |  | ||||||
|  |  | ||||||
| kotlin { |  | ||||||
|     sourceSets { |  | ||||||
|         androidMain { |  | ||||||
|             dependencies { |  | ||||||
|                 api project(":micro_utils.android.smalltextfield") |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| } |  | ||||||
| @@ -1 +0,0 @@ | |||||||
| <manifest/> |  | ||||||
| @@ -1,27 +0,0 @@ | |||||||
| package dev.inmo.micro_utils.android.pickers |  | ||||||
|  |  | ||||||
| import androidx.compose.animation.core.* |  | ||||||
|  |  | ||||||
| internal suspend fun Animatable<Float, AnimationVector1D>.fling( |  | ||||||
|     initialVelocity: Float, |  | ||||||
|     animationSpec: DecayAnimationSpec<Float>, |  | ||||||
|     adjustTarget: ((Float) -> Float)?, |  | ||||||
|     block: (Animatable<Float, AnimationVector1D>.() -> Unit)? = null, |  | ||||||
| ): AnimationResult<Float, AnimationVector1D> { |  | ||||||
|     val targetValue = animationSpec.calculateTargetValue(value, initialVelocity) |  | ||||||
|     val adjustedTarget = adjustTarget?.invoke(targetValue) |  | ||||||
|  |  | ||||||
|     return if (adjustedTarget != null) { |  | ||||||
|         animateTo( |  | ||||||
|             targetValue = adjustedTarget, |  | ||||||
|             initialVelocity = initialVelocity, |  | ||||||
|             block = block |  | ||||||
|         ) |  | ||||||
|     } else { |  | ||||||
|         animateDecay( |  | ||||||
|             initialVelocity = initialVelocity, |  | ||||||
|             animationSpec = animationSpec, |  | ||||||
|             block = block, |  | ||||||
|         ) |  | ||||||
|     } |  | ||||||
| } |  | ||||||
| @@ -1,219 +0,0 @@ | |||||||
| package dev.inmo.micro_utils.android.pickers |  | ||||||
|  |  | ||||||
| import androidx.compose.animation.core.Animatable |  | ||||||
| import androidx.compose.animation.core.exponentialDecay |  | ||||||
| import androidx.compose.foundation.clickable |  | ||||||
| import androidx.compose.foundation.gestures.* |  | ||||||
| import androidx.compose.foundation.layout.* |  | ||||||
| import androidx.compose.foundation.text.KeyboardActions |  | ||||||
| import androidx.compose.foundation.text.KeyboardOptions |  | ||||||
| import androidx.compose.material.icons.Icons |  | ||||||
| import androidx.compose.material.icons.filled.KeyboardArrowDown |  | ||||||
| import androidx.compose.material.icons.filled.KeyboardArrowUp |  | ||||||
| import androidx.compose.material3.* |  | ||||||
| import androidx.compose.runtime.* |  | ||||||
| import androidx.compose.ui.Alignment |  | ||||||
| import androidx.compose.ui.ExperimentalComposeUiApi |  | ||||||
| import androidx.compose.ui.Modifier |  | ||||||
| import androidx.compose.ui.draw.alpha |  | ||||||
| import androidx.compose.ui.focus.FocusRequester |  | ||||||
| import androidx.compose.ui.focus.focusRequester |  | ||||||
| import androidx.compose.ui.geometry.Offset |  | ||||||
| import androidx.compose.ui.graphics.Color |  | ||||||
| import androidx.compose.ui.input.pointer.PointerInputScope |  | ||||||
| import androidx.compose.ui.input.pointer.pointerInput |  | ||||||
| import androidx.compose.ui.platform.LocalDensity |  | ||||||
| import androidx.compose.ui.text.ExperimentalTextApi |  | ||||||
| import androidx.compose.ui.text.TextStyle |  | ||||||
| import androidx.compose.ui.text.input.KeyboardType |  | ||||||
| import androidx.compose.ui.unit.IntOffset |  | ||||||
| import androidx.compose.ui.unit.center |  | ||||||
| import androidx.compose.ui.unit.dp |  | ||||||
| import dev.inmo.micro_utils.android.smalltextfield.SmallTextField |  | ||||||
| import kotlinx.coroutines.launch |  | ||||||
| import kotlin.math.abs |  | ||||||
| import kotlin.math.absoluteValue |  | ||||||
| import kotlin.math.roundToInt |  | ||||||
|  |  | ||||||
| private inline fun PointerInputScope.checkContains(offset: Offset): Boolean { |  | ||||||
|     return ((size.center.x - offset.x).absoluteValue < size.width / 2) && ((size.center.y - offset.y).absoluteValue < size.height / 2) |  | ||||||
| } |  | ||||||
|  |  | ||||||
| // src: https://gist.github.com/vganin/a9a84653a9f48a2d669910fbd48e32d5 |  | ||||||
|  |  | ||||||
| @OptIn(ExperimentalTextApi::class, ExperimentalComposeUiApi::class) |  | ||||||
| @Composable |  | ||||||
| fun NumberPicker( |  | ||||||
|     number: Int, |  | ||||||
|     modifier: Modifier = Modifier, |  | ||||||
|     range: IntRange? = null, |  | ||||||
|     textStyle: TextStyle = LocalTextStyle.current, |  | ||||||
|     arrowsColor: Color = MaterialTheme.colorScheme.primary, |  | ||||||
|     allowUseManualInput: Boolean = true, |  | ||||||
|     onStateChanged: (Int) -> Unit = {}, |  | ||||||
| ) { |  | ||||||
|     val coroutineScope = rememberCoroutineScope() |  | ||||||
|     val numbersColumnHeight = 36.dp |  | ||||||
|     val halvedNumbersColumnHeight = numbersColumnHeight / 2 |  | ||||||
|     val halvedNumbersColumnHeightPx = with(LocalDensity.current) { halvedNumbersColumnHeight.toPx() } |  | ||||||
|  |  | ||||||
|     fun animatedStateValue(offset: Float): Int = number - (offset / halvedNumbersColumnHeightPx).toInt() |  | ||||||
|  |  | ||||||
|     val animatedOffset = remember { Animatable(0f) }.apply { |  | ||||||
|         if (range != null) { |  | ||||||
|             val offsetRange = remember(number, range) { |  | ||||||
|                 val value = number |  | ||||||
|                 val first = -(range.last - value) * halvedNumbersColumnHeightPx |  | ||||||
|                 val last = -(range.first - value) * halvedNumbersColumnHeightPx |  | ||||||
|                 first..last |  | ||||||
|             } |  | ||||||
|             updateBounds(offsetRange.start, offsetRange.endInclusive) |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
|     val coercedAnimatedOffset = animatedOffset.value % halvedNumbersColumnHeightPx |  | ||||||
|     val animatedStateValue = animatedStateValue(animatedOffset.value) |  | ||||||
|     val disabledArrowsColor = arrowsColor.copy(alpha = 0f) |  | ||||||
|  |  | ||||||
|     val inputFieldShown = if (allowUseManualInput) { |  | ||||||
|         remember { mutableStateOf(false) } |  | ||||||
|     } else { |  | ||||||
|         null |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     Column( |  | ||||||
|         modifier = modifier |  | ||||||
|             .wrapContentSize() |  | ||||||
|             .draggable( |  | ||||||
|                 orientation = Orientation.Vertical, |  | ||||||
|                 state = rememberDraggableState { deltaY -> |  | ||||||
|                     if (inputFieldShown ?.value != true) { |  | ||||||
|                         coroutineScope.launch { |  | ||||||
|                             animatedOffset.snapTo(animatedOffset.value + deltaY) |  | ||||||
|                         } |  | ||||||
|                     } |  | ||||||
|                 }, |  | ||||||
|                 onDragStopped = { velocity -> |  | ||||||
|                     if (inputFieldShown ?.value != true) { |  | ||||||
|                         coroutineScope.launch { |  | ||||||
|                             val endValue = animatedOffset.fling( |  | ||||||
|                                 initialVelocity = velocity, |  | ||||||
|                                 animationSpec = exponentialDecay(frictionMultiplier = 20f), |  | ||||||
|                                 adjustTarget = { target -> |  | ||||||
|                                     val coercedTarget = target % halvedNumbersColumnHeightPx |  | ||||||
|                                     val coercedAnchors = |  | ||||||
|                                         listOf(-halvedNumbersColumnHeightPx, 0f, halvedNumbersColumnHeightPx) |  | ||||||
|                                     val coercedPoint = coercedAnchors.minByOrNull { abs(it - coercedTarget) }!! |  | ||||||
|                                     val base = |  | ||||||
|                                         halvedNumbersColumnHeightPx * (target / halvedNumbersColumnHeightPx).toInt() |  | ||||||
|                                     coercedPoint + base |  | ||||||
|                                 } |  | ||||||
|                             ).endState.value |  | ||||||
|  |  | ||||||
|                             onStateChanged(animatedStateValue(endValue)) |  | ||||||
|                             animatedOffset.snapTo(0f) |  | ||||||
|                         } |  | ||||||
|                     } |  | ||||||
|                 } |  | ||||||
|             ), |  | ||||||
|         horizontalAlignment = Alignment.CenterHorizontally |  | ||||||
|     ) { |  | ||||||
|         val spacing = 4.dp |  | ||||||
|  |  | ||||||
|         val upEnabled = range == null || range.first < number |  | ||||||
|         IconButton( |  | ||||||
|             { |  | ||||||
|                 onStateChanged(number - 1) |  | ||||||
|                 inputFieldShown ?.value = false |  | ||||||
|             }, |  | ||||||
|             enabled = upEnabled |  | ||||||
|         ) { |  | ||||||
|             Icon(Icons.Default.KeyboardArrowUp, "", tint = if (upEnabled) arrowsColor else disabledArrowsColor) |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|         Spacer(modifier = Modifier.height(spacing)) |  | ||||||
|         Box( |  | ||||||
|             modifier = Modifier |  | ||||||
|                 .offset { IntOffset(x = 0, y = coercedAnimatedOffset.roundToInt()) }, |  | ||||||
|             contentAlignment = Alignment.Center |  | ||||||
|         ) { |  | ||||||
|             val baseLabelModifier = Modifier.align(Alignment.Center) |  | ||||||
|             ProvideTextStyle(textStyle) { |  | ||||||
|                 Text( |  | ||||||
|                     text = (animatedStateValue - 1).toString(), |  | ||||||
|                     modifier = baseLabelModifier |  | ||||||
|                         .offset(y = -halvedNumbersColumnHeight) |  | ||||||
|                         .alpha(coercedAnimatedOffset / halvedNumbersColumnHeightPx) |  | ||||||
|                 ) |  | ||||||
|  |  | ||||||
|                 if (inputFieldShown ?.value == true) { |  | ||||||
|                     val currentValue = remember { mutableStateOf(number.toString()) } |  | ||||||
|  |  | ||||||
|                     val focusRequester = remember { FocusRequester() } |  | ||||||
|                     SmallTextField( |  | ||||||
|                         currentValue.value, |  | ||||||
|                         { |  | ||||||
|                             val asDigit = it.toIntOrNull() |  | ||||||
|                             when { |  | ||||||
|                                 (asDigit == null && it.isEmpty()) -> currentValue.value = (range ?.first ?: 0).toString() |  | ||||||
|                                 (asDigit != null && (range == null || asDigit in range)) -> currentValue.value = it |  | ||||||
|                                 else -> { /* do nothing */ } |  | ||||||
|                             } |  | ||||||
|                         }, |  | ||||||
|                         baseLabelModifier.focusRequester(focusRequester).width(IntrinsicSize.Min).pointerInput(number) { |  | ||||||
|                             detectTapGestures { |  | ||||||
|                                 if (!checkContains(it)) { |  | ||||||
|                                     currentValue.value.toIntOrNull() ?.let(onStateChanged) |  | ||||||
|                                     inputFieldShown.value = false |  | ||||||
|                                 } |  | ||||||
|                             } |  | ||||||
|                         }, |  | ||||||
|                         keyboardOptions = KeyboardOptions( |  | ||||||
|                             keyboardType = KeyboardType.Number |  | ||||||
|                         ), |  | ||||||
|                         keyboardActions = KeyboardActions { |  | ||||||
|                             currentValue.value.toIntOrNull() ?.let(onStateChanged) |  | ||||||
|                             inputFieldShown.value = false |  | ||||||
|                         }, |  | ||||||
|                         singleLine = true, |  | ||||||
|                         textStyle = textStyle |  | ||||||
|                     ) |  | ||||||
|                     LaunchedEffect(Unit) { |  | ||||||
|                         focusRequester.requestFocus() |  | ||||||
|                     } |  | ||||||
|                 } else { |  | ||||||
|                     Text( |  | ||||||
|                         text = animatedStateValue.toString(), |  | ||||||
|                         modifier = baseLabelModifier |  | ||||||
|                             .alpha(1 - abs(coercedAnimatedOffset) / halvedNumbersColumnHeightPx) |  | ||||||
|                             .clickable { |  | ||||||
|                                 if (inputFieldShown ?.value == false) { |  | ||||||
|                                     inputFieldShown.value = true |  | ||||||
|                                 } |  | ||||||
|                             } |  | ||||||
|                     ) |  | ||||||
|                 } |  | ||||||
|                 Text( |  | ||||||
|                     text = (animatedStateValue + 1).toString(), |  | ||||||
|                     modifier = baseLabelModifier |  | ||||||
|                         .offset(y = halvedNumbersColumnHeight) |  | ||||||
|                         .alpha(-coercedAnimatedOffset / halvedNumbersColumnHeightPx) |  | ||||||
|                 ) |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|  |  | ||||||
|         Spacer(modifier = Modifier.height(spacing)) |  | ||||||
|  |  | ||||||
|         val downEnabled = range == null || range.last > number |  | ||||||
|         IconButton( |  | ||||||
|             { |  | ||||||
|                 onStateChanged(number + 1) |  | ||||||
|                 inputFieldShown ?.value = false |  | ||||||
|             }, |  | ||||||
|             enabled = downEnabled |  | ||||||
|         ) { |  | ||||||
|             Icon(Icons.Default.KeyboardArrowDown, "", tint = if (downEnabled) arrowsColor else disabledArrowsColor) |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| } |  | ||||||
|  |  | ||||||
| @@ -1,153 +0,0 @@ | |||||||
| package dev.inmo.micro_utils.android.pickers |  | ||||||
|  |  | ||||||
| import androidx.compose.animation.core.Animatable |  | ||||||
| import androidx.compose.animation.core.exponentialDecay |  | ||||||
| import androidx.compose.foundation.gestures.* |  | ||||||
| import androidx.compose.foundation.layout.* |  | ||||||
| import androidx.compose.foundation.rememberScrollState |  | ||||||
| import androidx.compose.material.icons.Icons |  | ||||||
| import androidx.compose.material.icons.filled.KeyboardArrowDown |  | ||||||
| import androidx.compose.material.icons.filled.KeyboardArrowUp |  | ||||||
| import androidx.compose.material3.* |  | ||||||
| import androidx.compose.runtime.* |  | ||||||
| import androidx.compose.ui.Alignment |  | ||||||
| import androidx.compose.ui.ExperimentalComposeUiApi |  | ||||||
| import androidx.compose.ui.Modifier |  | ||||||
| import androidx.compose.ui.draw.alpha |  | ||||||
| import androidx.compose.ui.graphics.Color |  | ||||||
| import androidx.compose.ui.platform.LocalDensity |  | ||||||
| import androidx.compose.ui.text.ExperimentalTextApi |  | ||||||
| import androidx.compose.ui.text.TextStyle |  | ||||||
| import androidx.compose.ui.unit.dp |  | ||||||
| import kotlinx.coroutines.launch |  | ||||||
| import kotlin.math.* |  | ||||||
|  |  | ||||||
| @OptIn(ExperimentalTextApi::class, ExperimentalComposeUiApi::class) |  | ||||||
| @Composable |  | ||||||
| fun <T> SetPicker( |  | ||||||
|     current: T, |  | ||||||
|     dataList: List<T>, |  | ||||||
|     modifier: Modifier = Modifier, |  | ||||||
|     textStyle: TextStyle = LocalTextStyle.current, |  | ||||||
|     arrowsColor: Color = MaterialTheme.colorScheme.primary, |  | ||||||
|     dataToString: @Composable (T) -> String = { it.toString() }, |  | ||||||
|     onStateChanged: (T) -> Unit = {}, |  | ||||||
| ) { |  | ||||||
|     val coroutineScope = rememberCoroutineScope() |  | ||||||
|     val numbersColumnHeight = 8.dp + with(LocalDensity.current) { |  | ||||||
|         textStyle.lineHeight.toDp() |  | ||||||
|     } |  | ||||||
|     val numbersColumnHeightPx = with(LocalDensity.current) { numbersColumnHeight.toPx() } |  | ||||||
|     val halvedNumbersColumnHeight = numbersColumnHeight / 2 |  | ||||||
|     val halvedNumbersColumnHeightPx = with(LocalDensity.current) { halvedNumbersColumnHeight.toPx() } |  | ||||||
|  |  | ||||||
|     val index = dataList.indexOfFirst { it === current }.takeIf { it > -1 } ?: dataList.indexOf(current) |  | ||||||
|     val lastIndex = dataList.size - 1 |  | ||||||
|  |  | ||||||
|     fun animatedStateValue(offset: Float): Int = index - (offset / halvedNumbersColumnHeightPx).toInt() |  | ||||||
|  |  | ||||||
|     val animatedOffset = remember { Animatable(0f) }.apply { |  | ||||||
|         val offsetRange = remember(index, lastIndex) { |  | ||||||
|             val value = index |  | ||||||
|             val first = -(lastIndex - value) * halvedNumbersColumnHeightPx |  | ||||||
|             val last = value * halvedNumbersColumnHeightPx |  | ||||||
|             first..last |  | ||||||
|         } |  | ||||||
|         updateBounds(offsetRange.start, offsetRange.endInclusive) |  | ||||||
|     } |  | ||||||
|     val indexAnimatedOffset = if (animatedOffset.value > 0) { |  | ||||||
|         (index - floor(animatedOffset.value / halvedNumbersColumnHeightPx).toInt()) |  | ||||||
|     } else { |  | ||||||
|         (index - ceil(animatedOffset.value / halvedNumbersColumnHeightPx).toInt()) |  | ||||||
|     } |  | ||||||
|     val coercedAnimatedOffset = animatedOffset.value % halvedNumbersColumnHeightPx |  | ||||||
|     val disabledArrowsColor = arrowsColor.copy(alpha = 0f) |  | ||||||
|  |  | ||||||
|     Column( |  | ||||||
|         modifier = modifier |  | ||||||
|             .wrapContentSize() |  | ||||||
|             .draggable( |  | ||||||
|                 orientation = Orientation.Vertical, |  | ||||||
|                 state = rememberDraggableState { deltaY -> |  | ||||||
|                     coroutineScope.launch { |  | ||||||
|                         animatedOffset.snapTo(animatedOffset.value + deltaY) |  | ||||||
|                     } |  | ||||||
|                 }, |  | ||||||
|                 onDragStopped = { velocity -> |  | ||||||
|                     coroutineScope.launch { |  | ||||||
|                         val endValue = animatedOffset.fling( |  | ||||||
|                             initialVelocity = velocity, |  | ||||||
|                             animationSpec = exponentialDecay(frictionMultiplier = 20f), |  | ||||||
|                             adjustTarget = { target -> |  | ||||||
|                                 val coercedTarget = target % halvedNumbersColumnHeightPx |  | ||||||
|                                 val coercedAnchors = |  | ||||||
|                                     listOf(-halvedNumbersColumnHeightPx, 0f, halvedNumbersColumnHeightPx) |  | ||||||
|                                 val coercedPoint = coercedAnchors.minByOrNull { abs(it - coercedTarget) }!! |  | ||||||
|                                 val base = |  | ||||||
|                                     halvedNumbersColumnHeightPx * (target / halvedNumbersColumnHeightPx).toInt() |  | ||||||
|                                 coercedPoint + base |  | ||||||
|                             } |  | ||||||
|                         ).endState.value |  | ||||||
|  |  | ||||||
|                         onStateChanged(dataList.elementAt(animatedStateValue(endValue))) |  | ||||||
|                         animatedOffset.snapTo(0f) |  | ||||||
|                     } |  | ||||||
|                 } |  | ||||||
|             ), |  | ||||||
|         horizontalAlignment = Alignment.CenterHorizontally |  | ||||||
|     ) { |  | ||||||
|         val spacing = 4.dp |  | ||||||
|  |  | ||||||
|         val upEnabled = index > 0 |  | ||||||
|         IconButton( |  | ||||||
|             { |  | ||||||
|                 onStateChanged(dataList.elementAt(index - 1)) |  | ||||||
|             }, |  | ||||||
|             enabled = upEnabled |  | ||||||
|         ) { |  | ||||||
|             Icon(Icons.Default.KeyboardArrowUp, "", tint = if (upEnabled) arrowsColor else disabledArrowsColor) |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|         Spacer(modifier = Modifier.height(spacing)) |  | ||||||
|         Box( |  | ||||||
|             modifier = Modifier, |  | ||||||
|             contentAlignment = Alignment.Center |  | ||||||
|         ) { |  | ||||||
|             ProvideTextStyle(textStyle) { |  | ||||||
|                 dataList.forEachIndexed { i, t -> |  | ||||||
|                     val alpha = when { |  | ||||||
|                         i == indexAnimatedOffset - 1 -> coercedAnimatedOffset / halvedNumbersColumnHeightPx |  | ||||||
|                         i == indexAnimatedOffset -> 1 - (abs(coercedAnimatedOffset) / halvedNumbersColumnHeightPx) |  | ||||||
|                         i == indexAnimatedOffset + 1 -> -coercedAnimatedOffset / halvedNumbersColumnHeightPx |  | ||||||
|                         else -> return@forEachIndexed |  | ||||||
|                     } |  | ||||||
|                     val offset = when { |  | ||||||
|                         i == indexAnimatedOffset - 1 && coercedAnimatedOffset > 0 -> coercedAnimatedOffset - halvedNumbersColumnHeightPx |  | ||||||
|                         i == indexAnimatedOffset -> coercedAnimatedOffset |  | ||||||
|                         i == indexAnimatedOffset + 1 && coercedAnimatedOffset < 0 -> coercedAnimatedOffset + halvedNumbersColumnHeightPx |  | ||||||
|                         else -> return@forEachIndexed |  | ||||||
|                     } |  | ||||||
|                     Text( |  | ||||||
|                         text = dataToString(t), |  | ||||||
|                         modifier = Modifier |  | ||||||
|                             .alpha(alpha) |  | ||||||
|                             .offset(y = with(LocalDensity.current) { offset.toDp() }) |  | ||||||
|                     ) |  | ||||||
|                 } |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|  |  | ||||||
|         Spacer(modifier = Modifier.height(spacing)) |  | ||||||
|  |  | ||||||
|         val downEnabled = index < lastIndex |  | ||||||
|         IconButton( |  | ||||||
|             { |  | ||||||
|                 onStateChanged(dataList.elementAt(index + 1)) |  | ||||||
|             }, |  | ||||||
|             enabled = downEnabled |  | ||||||
|         ) { |  | ||||||
|             Icon(Icons.Default.KeyboardArrowDown, "", tint = if (downEnabled) arrowsColor else disabledArrowsColor) |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| } |  | ||||||
| @@ -4,7 +4,7 @@ plugins { | |||||||
|     id "com.android.library" |     id "com.android.library" | ||||||
| } | } | ||||||
|  |  | ||||||
| apply from: "$mppAndroidProject" | apply from: "$mppAndroidProjectPresetPath" | ||||||
|  |  | ||||||
| kotlin { | kotlin { | ||||||
|     sourceSets { |     sourceSets { | ||||||
|   | |||||||
| @@ -1 +0,0 @@ | |||||||
| <manifest/> |  | ||||||
							
								
								
									
										1
									
								
								android/recyclerview/src/main/AndroidManifest.xml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										1
									
								
								android/recyclerview/src/main/AndroidManifest.xml
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1 @@ | |||||||
|  | <manifest package="dev.inmo.micro_utils.android.recyclerview"/> | ||||||
| @@ -1,19 +0,0 @@ | |||||||
| plugins { |  | ||||||
|     id "org.jetbrains.kotlin.multiplatform" |  | ||||||
|     id "org.jetbrains.kotlin.plugin.serialization" |  | ||||||
|     id "com.android.library" |  | ||||||
|     alias(libs.plugins.jb.compose) |  | ||||||
|     alias(libs.plugins.kt.jb.compose) |  | ||||||
| } |  | ||||||
|  |  | ||||||
| apply from: "$mppProjectWithSerializationAndCompose" |  | ||||||
|  |  | ||||||
| kotlin { |  | ||||||
|     sourceSets { |  | ||||||
|         androidMain { |  | ||||||
|             dependencies { |  | ||||||
|                 api libs.jb.compose.material3 |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| } |  | ||||||
| @@ -1 +0,0 @@ | |||||||
| <manifest/> |  | ||||||
| @@ -1,66 +0,0 @@ | |||||||
| package dev.inmo.micro_utils.android.smalltextfield |  | ||||||
|  |  | ||||||
| import androidx.compose.foundation.interaction.MutableInteractionSource |  | ||||||
| import androidx.compose.foundation.layout.Box |  | ||||||
| import androidx.compose.foundation.layout.Row |  | ||||||
| import androidx.compose.foundation.layout.defaultMinSize |  | ||||||
| import androidx.compose.foundation.text.BasicTextField |  | ||||||
| import androidx.compose.foundation.text.KeyboardActions |  | ||||||
| import androidx.compose.foundation.text.KeyboardOptions |  | ||||||
| import androidx.compose.foundation.text.selection.LocalTextSelectionColors |  | ||||||
| import androidx.compose.material3.* |  | ||||||
| import androidx.compose.runtime.Composable |  | ||||||
| import androidx.compose.runtime.CompositionLocalProvider |  | ||||||
| import androidx.compose.runtime.remember |  | ||||||
| import androidx.compose.ui.Alignment |  | ||||||
| import androidx.compose.ui.Modifier |  | ||||||
| import androidx.compose.ui.graphics.Color |  | ||||||
| import androidx.compose.ui.graphics.Shape |  | ||||||
| import androidx.compose.ui.graphics.SolidColor |  | ||||||
| import androidx.compose.ui.graphics.takeOrElse |  | ||||||
| import androidx.compose.ui.text.TextStyle |  | ||||||
| import androidx.compose.ui.text.input.VisualTransformation |  | ||||||
|  |  | ||||||
| @OptIn(ExperimentalMaterial3Api::class) |  | ||||||
| @Composable |  | ||||||
| fun SmallTextField( |  | ||||||
|     value: String, |  | ||||||
|     onValueChange: (String) -> Unit, |  | ||||||
|     modifier: Modifier = Modifier, |  | ||||||
|     enabled: Boolean = true, |  | ||||||
|     readOnly: Boolean = false, |  | ||||||
|     textStyle: TextStyle = LocalTextStyle.current, |  | ||||||
|     textColor: Color = textStyle.color.takeOrElse { |  | ||||||
|         LocalContentColor.current |  | ||||||
|     }, |  | ||||||
|     visualTransformation: VisualTransformation = VisualTransformation.None, |  | ||||||
|     keyboardOptions: KeyboardOptions = KeyboardOptions.Default, |  | ||||||
|     keyboardActions: KeyboardActions = KeyboardActions.Default, |  | ||||||
|     singleLine: Boolean = false, |  | ||||||
|     maxLines: Int = if (singleLine) 1 else Int.MAX_VALUE, |  | ||||||
|     minLines: Int = 1, |  | ||||||
|     interactionSource: MutableInteractionSource = remember { MutableInteractionSource() }, |  | ||||||
| ) { |  | ||||||
|     BasicTextField( |  | ||||||
|         value = value, |  | ||||||
|         modifier = modifier, |  | ||||||
|         onValueChange = onValueChange, |  | ||||||
|         enabled = enabled, |  | ||||||
|         readOnly = readOnly, |  | ||||||
|         textStyle = textStyle.copy( |  | ||||||
|             color = textColor |  | ||||||
|         ), |  | ||||||
|         visualTransformation = visualTransformation, |  | ||||||
|         keyboardOptions = keyboardOptions, |  | ||||||
|         keyboardActions = keyboardActions, |  | ||||||
|         interactionSource = interactionSource, |  | ||||||
|         singleLine = singleLine, |  | ||||||
|         maxLines = maxLines, |  | ||||||
|         minLines = minLines, |  | ||||||
|         cursorBrush = SolidColor( |  | ||||||
|             textStyle.color.takeOrElse { |  | ||||||
|                 LocalContentColor.current |  | ||||||
|             } |  | ||||||
|         ) |  | ||||||
|     ) |  | ||||||
| } |  | ||||||
							
								
								
									
										34
									
								
								build.gradle
									
									
									
									
									
								
							
							
						
						
									
										34
									
								
								build.gradle
									
									
									
									
									
								
							| @@ -9,7 +9,6 @@ buildscript { | |||||||
|     dependencies { |     dependencies { | ||||||
|         classpath libs.buildscript.kt.gradle |         classpath libs.buildscript.kt.gradle | ||||||
|         classpath libs.buildscript.kt.serialization |         classpath libs.buildscript.kt.serialization | ||||||
|         classpath libs.buildscript.kt.ksp |  | ||||||
|         classpath libs.buildscript.jb.dokka |         classpath libs.buildscript.jb.dokka | ||||||
|         classpath libs.buildscript.gh.release |         classpath libs.buildscript.gh.release | ||||||
|         classpath libs.buildscript.android.gradle |         classpath libs.buildscript.android.gradle | ||||||
| @@ -17,35 +16,22 @@ buildscript { | |||||||
|     } |     } | ||||||
| } | } | ||||||
|  |  | ||||||
| plugins { |  | ||||||
|     alias(libs.plugins.versions) |  | ||||||
|     alias(libs.plugins.nmcp.aggregation) |  | ||||||
| } |  | ||||||
|  |  | ||||||
|  |  | ||||||
| if ((project.hasProperty('SONATYPE_USER') || System.getenv('SONATYPE_USER') != null) && (project.hasProperty('SONATYPE_PASSWORD') || System.getenv('SONATYPE_PASSWORD') != null)) { |  | ||||||
|     nmcpAggregation { |  | ||||||
|         centralPortal { |  | ||||||
|             username = project.hasProperty('SONATYPE_USER') ? project.property('SONATYPE_USER') : System.getenv('SONATYPE_USER') |  | ||||||
|             password = project.hasProperty('SONATYPE_PASSWORD') ? project.property('SONATYPE_PASSWORD') : System.getenv('SONATYPE_PASSWORD') |  | ||||||
|             validationTimeout = Duration.ofHours(4) |  | ||||||
|             publishingType = System.getenv('PUBLISHING_TYPE') != "" ? System.getenv('PUBLISHING_TYPE') : "USER_MANAGED" |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|         publishAllProjectsProbablyBreakingProjectIsolation() |  | ||||||
|     } |  | ||||||
| } |  | ||||||
|  |  | ||||||
| allprojects { | allprojects { | ||||||
|     repositories { |     repositories { | ||||||
|  |         mavenLocal() | ||||||
|         mavenCentral() |         mavenCentral() | ||||||
|         google() |         google() | ||||||
|         maven { url "https://maven.pkg.jetbrains.space/public/p/compose/dev" } |     } | ||||||
|         maven { url "https://nexus.inmo.dev/repository/maven-releases/" } |  | ||||||
|         mavenLocal() |     // temporal crutch until legacy tests will be stabled or legacy target will be removed | ||||||
|  |     if (it != rootProject.findProject("docs")) { | ||||||
|  |         tasks.whenTaskAdded { task -> | ||||||
|  |             if(task.name == "jsLegacyBrowserTest" || task.name == "jsLegacyNodeTest") { | ||||||
|  |                 task.enabled = false | ||||||
|  |             } | ||||||
|  |         } | ||||||
|     } |     } | ||||||
| } | } | ||||||
|  |  | ||||||
| apply from: "./extensions.gradle" | apply from: "./extensions.gradle" | ||||||
| apply from: "./github_release.gradle" | apply from: "./github_release.gradle" | ||||||
| apply from: "./versions_plugin_setup.gradle" |  | ||||||
|   | |||||||
| @@ -1,17 +0,0 @@ | |||||||
| plugins { |  | ||||||
|     id "org.jetbrains.kotlin.multiplatform" |  | ||||||
|     id "org.jetbrains.kotlin.plugin.serialization" |  | ||||||
|     id "com.android.library" |  | ||||||
| } |  | ||||||
|  |  | ||||||
| apply from: "$mppJvmJsWasmJsAndroidLinuxMingwLinuxArm64Project" |  | ||||||
|  |  | ||||||
| kotlin { |  | ||||||
|     sourceSets { |  | ||||||
|         commonMain { |  | ||||||
|             dependencies { |  | ||||||
|                 api project(":micro_utils.colors.common") |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| } |  | ||||||
| @@ -1,7 +0,0 @@ | |||||||
| plugins { |  | ||||||
|     id "org.jetbrains.kotlin.multiplatform" |  | ||||||
|     id "org.jetbrains.kotlin.plugin.serialization" |  | ||||||
|     id "com.android.library" |  | ||||||
| } |  | ||||||
|  |  | ||||||
| apply from: "$mppJvmJsWasmJsAndroidLinuxMingwLinuxArm64Project" |  | ||||||
| @@ -1,174 +0,0 @@ | |||||||
| package dev.inmo.micro_utils.colors.common |  | ||||||
|  |  | ||||||
| import kotlinx.serialization.Serializable |  | ||||||
| import kotlin.jvm.JvmInline |  | ||||||
| import kotlin.math.floor |  | ||||||
|  |  | ||||||
| /** |  | ||||||
|  * Wrapper for RGBA colors. Receiving [UInt] in main constructor. Each part in main constructor |  | ||||||
|  * configured with `0x00 - 0xff` range. Examples: |  | ||||||
|  * |  | ||||||
|  * * Red: `0xff0000ffu` |  | ||||||
|  * * Red (0.5 capacity): `0xff000088u` |  | ||||||
|  * |  | ||||||
|  * Anyway it is recommended to use |  | ||||||
|  * |  | ||||||
|  * @param hexaUInt rgba [UInt] in format `0xRRGGBBAA` where RR - red, GG - green, BB - blue` and AA - alpha |  | ||||||
|  */ |  | ||||||
| @Serializable |  | ||||||
| @JvmInline |  | ||||||
| value class HEXAColor ( |  | ||||||
|     val hexaUInt: UInt |  | ||||||
| ) : Comparable<HEXAColor> { |  | ||||||
|     /** |  | ||||||
|      * @returns [hexaUInt] as a string with format `#RRGGBBAA` where RR - red, GG - green, BB - blue and AA - alpha |  | ||||||
|      */ |  | ||||||
|     val hexa: String |  | ||||||
|         get() = "#${hexaUInt.toString(16).padStart(8, '0')}" |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * @returns [hexaUInt] as a string with format `#RRGGBB` where RR - red, GG - green and BB - blue |  | ||||||
|      */ |  | ||||||
|     val hex: String |  | ||||||
|         get() = hexa.take(7) |  | ||||||
|     /** |  | ||||||
|      * @returns [hexaUInt] as a string with format `#AARRGGBB` where AA - alpha, RR - red, GG - green and BB - blue |  | ||||||
|      */ |  | ||||||
|     val ahex: String |  | ||||||
|         get() = "#${a.toString(16).padStart(2, '2')}${hex.drop(1)}" |  | ||||||
|     val rgba: String |  | ||||||
|         get() = "rgba($r,$g,$b,${aOfOne.toString().take(5)})" |  | ||||||
|     val rgb: String |  | ||||||
|         get() = "rgb($r,$g,$b)" |  | ||||||
|     val shortHex: String |  | ||||||
|         get() = "#${r.shortPart()}${g.shortPart()}${b.shortPart()}" |  | ||||||
|     val shortHexa: String |  | ||||||
|         get() = "$shortHex${a.shortPart()}" |  | ||||||
|     val rgbUInt: UInt |  | ||||||
|         get() = (hexaUInt / 256u) |  | ||||||
|     val rgbInt: Int |  | ||||||
|         get() = rgbUInt.toInt() |  | ||||||
|     val ahexUInt |  | ||||||
|         get() = (a * 0x1000000).toUInt() + rgbUInt |  | ||||||
|  |  | ||||||
|     val r: Int |  | ||||||
|         get() = ((hexaUInt and 0xff000000u) / 0x1000000u).toInt() |  | ||||||
|     val g: Int |  | ||||||
|         get() = ((hexaUInt and 0x00ff0000u) / 0x10000u).toInt() |  | ||||||
|     val b: Int |  | ||||||
|         get() = ((hexaUInt and 0x0000ff00u) / 0x100u).toInt() |  | ||||||
|     val a: Int |  | ||||||
|         get() = ((hexaUInt and 0x000000ffu)).toInt() |  | ||||||
|     val aOfOne: Float |  | ||||||
|         get() = a.toFloat() / (0xff) |  | ||||||
|     init { |  | ||||||
|         require(hexaUInt in 0u ..0xffffffffu) |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     constructor(r: Int, g: Int, b: Int, a: Int) : this( |  | ||||||
|         ((r * 0x1000000).toLong() + g * 0x10000 + b * 0x100 + a).toUInt() |  | ||||||
|     ) { |  | ||||||
|         require(r in 0 ..0xff) |  | ||||||
|         require(g in 0 ..0xff) |  | ||||||
|         require(b in 0 ..0xff) |  | ||||||
|         require(a in 0 ..0xff) |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     constructor(r: Int, g: Int, b: Int, aOfOne: Float = 1f) : this( |  | ||||||
|         r = r, g = g, b = b, a = (aOfOne * 0xff).toInt() |  | ||||||
|     ) |  | ||||||
|  |  | ||||||
|     override fun toString(): String { |  | ||||||
|         return hexa |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     override fun compareTo(other: HEXAColor): Int = (hexaUInt - other.hexaUInt).coerceIn(Int.MIN_VALUE.toUInt(), Int.MAX_VALUE.toLong().toUInt()).toInt() |  | ||||||
|  |  | ||||||
|     fun copy( |  | ||||||
|         r: Int = this.r, |  | ||||||
|         g: Int = this.g, |  | ||||||
|         b: Int = this.b, |  | ||||||
|         aOfOne: Float = this.aOfOne |  | ||||||
|     ) = HEXAColor(r = r, g = g, b = b, aOfOne = aOfOne) |  | ||||||
|     fun copy( |  | ||||||
|         r: Int = this.r, |  | ||||||
|         g: Int = this.g, |  | ||||||
|         b: Int = this.b, |  | ||||||
|         a: Int |  | ||||||
|     ) = HEXAColor(r = r, g = g, b = b, a = a) |  | ||||||
|  |  | ||||||
|     companion object { |  | ||||||
|         /** |  | ||||||
|          * Parsing color from [color] |  | ||||||
|          * |  | ||||||
|          * Supported formats samples (on Red color based): |  | ||||||
|          * |  | ||||||
|          * * `#f00` |  | ||||||
|          * * `#f00f` |  | ||||||
|          * * `#ff0000` |  | ||||||
|          * * `#ff0000ff` |  | ||||||
|          * * `rgb(255, 0, 0)` |  | ||||||
|          * * `rgba(255, 0, 0, 1)` |  | ||||||
|          */ |  | ||||||
|         fun parseStringColor(color: String): HEXAColor = when { |  | ||||||
|             color.startsWith("#") -> color.removePrefix("#").let { color -> |  | ||||||
|                 when (color.length) { |  | ||||||
|                     3 -> color.map { "$it$it" }.joinToString(separator = "", postfix = "ff") |  | ||||||
|                     4 -> color.take(3).map { "$it$it" }.joinToString(separator = "", postfix = color.takeLast(1).let { "${it}0" }) |  | ||||||
|                     6 -> "${color}ff" |  | ||||||
|                     8 -> color |  | ||||||
|                     else -> error("Malfurmed color string: $color. It is expected that color started with # will contains 3, 6 or 8 valuable parts") |  | ||||||
|                 } |  | ||||||
|             } |  | ||||||
|             color.startsWith("rgb(") -> color |  | ||||||
|                 .removePrefix("rgb(") |  | ||||||
|                 .removeSuffix(")") |  | ||||||
|                 .replace(Regex("\\s"), "") |  | ||||||
|                 .split(",") |  | ||||||
|                 .joinToString("", postfix = "ff") { |  | ||||||
|                     it.toInt().toString(16).padStart(2, '0') |  | ||||||
|                 } |  | ||||||
|             color.startsWith("rgba(") -> color |  | ||||||
|                 .removePrefix("rgba(") |  | ||||||
|                 .removeSuffix(")") |  | ||||||
|                 .replace(Regex("\\s"), "") |  | ||||||
|                 .split(",").let { |  | ||||||
|                     it.take(3).map { it.toInt().toString(16).padStart(2, '0') } + (it.last().toFloat() * 0xff).toInt().toString(16).padStart(2, '0') |  | ||||||
|                 } |  | ||||||
|                 .joinToString("") |  | ||||||
|             else -> color |  | ||||||
|         }.lowercase().toUInt(16).let(::HEXAColor) |  | ||||||
|  |  | ||||||
|         /** |  | ||||||
|          * Creates [HEXAColor] from [uint] presume it is in format `0xRRGGBBAA` where RR - red, GG - green, BB - blue` and AA - alpha |  | ||||||
|          */ |  | ||||||
|         fun fromHexa(uint: UInt) = HEXAColor(uint) |  | ||||||
|  |  | ||||||
|         /** |  | ||||||
|          * Creates [HEXAColor] from [uint] presume it is in format `0xAARRGGBB` where AA - alpha, RR - red, GG - green and BB - blue` |  | ||||||
|          */ |  | ||||||
|         fun fromAhex(uint: UInt) = HEXAColor( |  | ||||||
|             a = ((uint and 0xff000000u) / 0x1000000u).toInt(), |  | ||||||
|             r = ((uint and 0x00ff0000u) / 0x10000u).toInt(), |  | ||||||
|             g = ((uint and 0x0000ff00u) / 0x100u).toInt(), |  | ||||||
|             b = ((uint and 0x000000ffu)).toInt() |  | ||||||
|         ) |  | ||||||
|  |  | ||||||
|         /** |  | ||||||
|          * Parsing color from [color] |  | ||||||
|          * |  | ||||||
|          * Supported formats samples (on Red color based): |  | ||||||
|          * |  | ||||||
|          * * `#f00` |  | ||||||
|          * * `#ff0000` |  | ||||||
|          * * `#ff0000ff` |  | ||||||
|          * * `rgb(255, 0, 0)` |  | ||||||
|          * * `rgba(255, 0, 0, 1)` |  | ||||||
|          */ |  | ||||||
|         operator fun invoke(color: String) = parseStringColor(color) |  | ||||||
|  |  | ||||||
|         private fun Int.shortPart(): String { |  | ||||||
|             return (floor(toFloat() / 16)).toInt().toString(16) |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| } |  | ||||||
| @@ -1,209 +0,0 @@ | |||||||
| package dev.inmo.micro_utils.colors.common |  | ||||||
|  |  | ||||||
| import kotlin.math.floor |  | ||||||
| import kotlin.test.Test |  | ||||||
| import kotlin.test.assertEquals |  | ||||||
| import kotlin.test.assertTrue |  | ||||||
|  |  | ||||||
| class HexColorTests { |  | ||||||
|     val alphaRgbaPrecision = 5 |  | ||||||
|     class TestColor( |  | ||||||
|         val color: HEXAColor, |  | ||||||
|         val shortHex: String, |  | ||||||
|         val shortHexa: String, |  | ||||||
|         val hex: String, |  | ||||||
|         val hexa: String, |  | ||||||
|         val ahex: String, |  | ||||||
|         val ahexUInt: UInt, |  | ||||||
|         val rgbUInt: UInt, |  | ||||||
|         val rgb: String, |  | ||||||
|         val rgba: String, |  | ||||||
|         val r: Int, |  | ||||||
|         val g: Int, |  | ||||||
|         val b: Int, |  | ||||||
|         val a: Int, |  | ||||||
|         vararg val additionalRGBAVariants: String |  | ||||||
|     ) |  | ||||||
|     val testColors: List<TestColor> |  | ||||||
|         get() = listOf( |  | ||||||
|             TestColor( |  | ||||||
|                 color = HEXAColor(hexaUInt = 0xff0000ffu), |  | ||||||
|                 shortHex = "#f00", |  | ||||||
|                 shortHexa = "#f00f", |  | ||||||
|                 hex = "#ff0000", |  | ||||||
|                 hexa = "#ff0000ff", |  | ||||||
|                 ahex = "#ffff0000", |  | ||||||
|                 ahexUInt = 0xffff0000u, |  | ||||||
|                 rgbUInt = 0xff0000u, |  | ||||||
|                 rgb = "rgb(255,0,0)", |  | ||||||
|                 rgba = "rgba(255,0,0,1.0)", |  | ||||||
|                 r = 0xff, |  | ||||||
|                 g = 0x00, |  | ||||||
|                 b = 0x00, |  | ||||||
|                 a = 0xff, |  | ||||||
|                 "rgba(255,0,0,1)", |  | ||||||
|             ), |  | ||||||
|             TestColor( |  | ||||||
|                 color = HEXAColor(hexaUInt = 0x00ff00ffu), |  | ||||||
|                 shortHex = "#0f0", |  | ||||||
|                 shortHexa = "#0f0f", |  | ||||||
|                 hex = "#00ff00", |  | ||||||
|                 hexa = "#00ff00ff", |  | ||||||
|                 ahex = "#ff00ff00", |  | ||||||
|                 ahexUInt = 0xff00ff00u, |  | ||||||
|                 rgbUInt = 0x00ff00u, |  | ||||||
|                 rgb = "rgb(0,255,0)", |  | ||||||
|                 rgba = "rgba(0,255,0,1.0)", |  | ||||||
|                 r = 0x00, |  | ||||||
|                 g = 0xff, |  | ||||||
|                 b = 0x00, |  | ||||||
|                 a = 0xff, |  | ||||||
|                 "rgba(0,255,0,1)" |  | ||||||
|             ), |  | ||||||
|             TestColor( |  | ||||||
|                 color = HEXAColor(0x0000ffffu), |  | ||||||
|                 shortHex = "#00f", |  | ||||||
|                 shortHexa = "#00ff", |  | ||||||
|                 hex = "#0000ff", |  | ||||||
|                 hexa = "#0000ffff", |  | ||||||
|                 ahex = "#ff0000ff", |  | ||||||
|                 ahexUInt = 0xff0000ffu, |  | ||||||
|                 rgbUInt = 0x0000ffu, |  | ||||||
|                 rgb = "rgb(0,0,255)", |  | ||||||
|                 rgba = "rgba(0,0,255,1.0)", |  | ||||||
|                 r = 0x00, |  | ||||||
|                 g = 0x00, |  | ||||||
|                 b = 0xff, |  | ||||||
|                 a = 0xff, |  | ||||||
|                 "rgba(0,0,255,1)" |  | ||||||
|             ), |  | ||||||
|             TestColor( |  | ||||||
|                 color = HEXAColor(0xff000088u), |  | ||||||
|                 shortHex = "#f00", |  | ||||||
|                 shortHexa = "#f008", |  | ||||||
|                 hex = "#ff0000", |  | ||||||
|                 hexa = "#ff000088", |  | ||||||
|                 ahex = "#88ff0000", |  | ||||||
|                 ahexUInt = 0x88ff0000u, |  | ||||||
|                 rgbUInt = 0xff0000u, |  | ||||||
|                 rgb = "rgb(255,0,0)", |  | ||||||
|                 rgba = "rgba(255,0,0,0.533)", |  | ||||||
|                 r = 0xff, |  | ||||||
|                 g = 0x00, |  | ||||||
|                 b = 0x00, |  | ||||||
|                 a = 0x88, |  | ||||||
|             ), |  | ||||||
|             TestColor( |  | ||||||
|                 color = HEXAColor(0x00ff0088u), |  | ||||||
|                 shortHex = "#0f0", |  | ||||||
|                 shortHexa = "#0f08", |  | ||||||
|                 hex = "#00ff00", |  | ||||||
|                 hexa = "#00ff0088", |  | ||||||
|                 ahex = "#8800ff00", |  | ||||||
|                 ahexUInt = 0x8800ff00u, |  | ||||||
|                 rgbUInt = 0x00ff00u, |  | ||||||
|                 rgb = "rgb(0,255,0)", |  | ||||||
|                 rgba = "rgba(0,255,0,0.533)", |  | ||||||
|                 r = 0x00, |  | ||||||
|                 g = 0xff, |  | ||||||
|                 b = 0x00, |  | ||||||
|                 a = 0x88, |  | ||||||
|             ), |  | ||||||
|             TestColor( |  | ||||||
|                 color = HEXAColor(0x0000ff88u), |  | ||||||
|                 shortHex = "#00f", |  | ||||||
|                 shortHexa = "#00f8", |  | ||||||
|                 hex = "#0000ff", |  | ||||||
|                 hexa = "#0000ff88", |  | ||||||
|                 ahex = "#880000ff", |  | ||||||
|                 ahexUInt = 0x880000ffu, |  | ||||||
|                 rgbUInt = 0x0000ffu, |  | ||||||
|                 rgb = "rgb(0,0,255)", |  | ||||||
|                 rgba = "rgba(0,0,255,0.533)", |  | ||||||
|                 r = 0x00, |  | ||||||
|                 g = 0x00, |  | ||||||
|                 b = 0xff, |  | ||||||
|                 a = 0x88, |  | ||||||
|             ), |  | ||||||
|             TestColor( |  | ||||||
|                 color = HEXAColor(0xff000022u), |  | ||||||
|                 shortHex = "#f00", |  | ||||||
|                 shortHexa = "#f002", |  | ||||||
|                 hex = "#ff0000", |  | ||||||
|                 hexa = "#ff000022", |  | ||||||
|                 ahex = "#22ff0000", |  | ||||||
|                 ahexUInt = 0x22ff0000u, |  | ||||||
|                 rgbUInt = 0xff0000u, |  | ||||||
|                 rgb = "rgb(255,0,0)", |  | ||||||
|                 rgba = "rgba(255,0,0,0.133)", |  | ||||||
|                 r = 0xff, |  | ||||||
|                 g = 0x00, |  | ||||||
|                 b = 0x00, |  | ||||||
|                 a = 0x22, |  | ||||||
|             ), |  | ||||||
|             TestColor( |  | ||||||
|                 color = HEXAColor(0x00ff0022u), |  | ||||||
|                 shortHex = "#0f0", |  | ||||||
|                 shortHexa = "#0f02", |  | ||||||
|                 hex = "#00ff00", |  | ||||||
|                 hexa = "#00ff0022", |  | ||||||
|                 ahex = "#2200ff00", |  | ||||||
|                 ahexUInt = 0x2200ff00u, |  | ||||||
|                 rgbUInt = 0x00ff00u, |  | ||||||
|                 rgb = "rgb(0,255,0)", |  | ||||||
|                 rgba = "rgba(0,255,0,0.133)", |  | ||||||
|                 r = 0x00, |  | ||||||
|                 g = 0xff, |  | ||||||
|                 b = 0x00, |  | ||||||
|                 a = 0x22, |  | ||||||
|             ), |  | ||||||
|             TestColor( |  | ||||||
|                 color = HEXAColor(0x0000ff22u), |  | ||||||
|                 shortHex = "#00f", |  | ||||||
|                 shortHexa = "#00f2", |  | ||||||
|                 hex = "#0000ff", |  | ||||||
|                 hexa = "#0000ff22", |  | ||||||
|                 ahex = "#220000ff", |  | ||||||
|                 ahexUInt = 0x220000ffu, |  | ||||||
|                 rgbUInt = 0x0000ffu, |  | ||||||
|                 rgb = "rgb(0,0,255)", |  | ||||||
|                 rgba = "rgba(0,0,255,0.133)", |  | ||||||
|                 r = 0x00, |  | ||||||
|                 g = 0x00, |  | ||||||
|                 b = 0xff, |  | ||||||
|                 a = 0x22, |  | ||||||
|             ), |  | ||||||
|         ) |  | ||||||
|  |  | ||||||
|     @Test |  | ||||||
|     fun baseTest() { |  | ||||||
|         testColors.forEach { |  | ||||||
|             assertEquals(it.hex, it.color.hex) |  | ||||||
|             assertEquals(it.hexa, it.color.hexa) |  | ||||||
|             assertEquals(it.ahex, it.color.ahex) |  | ||||||
|             assertEquals(it.rgbUInt, it.color.rgbUInt) |  | ||||||
|             assertEquals(it.ahexUInt, it.color.ahexUInt) |  | ||||||
|             assertEquals(it.shortHex, it.color.shortHex) |  | ||||||
|             assertEquals(it.shortHexa, it.color.shortHexa) |  | ||||||
|             assertEquals(it.rgb, it.color.rgb) |  | ||||||
|             assertTrue(it.rgba == it.color.rgba || it.color.rgba in it.additionalRGBAVariants) |  | ||||||
|             assertEquals(it.r, it.color.r) |  | ||||||
|             assertEquals(it.g, it.color.g) |  | ||||||
|             assertEquals(it.b, it.color.b) |  | ||||||
|             assertEquals(it.a, it.color.a) |  | ||||||
|             assertEquals(it.color, HEXAColor.fromAhex(it.ahexUInt)) |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     @Test |  | ||||||
|     fun testHexParseColor() { |  | ||||||
|         testColors.forEach { |  | ||||||
|             assertEquals(it.color.copy(aOfOne = 1f), HEXAColor.parseStringColor(it.hex)) |  | ||||||
|             assertEquals(it.color, HEXAColor.parseStringColor(it.hexa)) |  | ||||||
|             assertEquals(it.color.copy(aOfOne = 1f), HEXAColor.parseStringColor(it.rgb)) |  | ||||||
|             assertTrue(it.color.hexaUInt.toInt() - HEXAColor.parseStringColor(it.rgba).hexaUInt.toInt() in -0x1 .. 0x1, ) |  | ||||||
|             assertEquals(it.color.copy(aOfOne = 1f), HEXAColor.parseStringColor(it.shortHex)) |  | ||||||
|             assertEquals(it.color.copy(a = floor(it.color.a.toFloat() / 16).toInt() * 0x10), HEXAColor.parseStringColor(it.shortHexa)) |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| } |  | ||||||
| @@ -1,298 +0,0 @@ | |||||||
| package dev.inmo.micro_utils.colors |  | ||||||
|  |  | ||||||
| import dev.inmo.micro_utils.colors.common.HEXAColor |  | ||||||
|  |  | ||||||
| val HEXAColor.Companion.aliceblue |  | ||||||
|     get() = HEXAColor(0xF0F8FFFFu) |  | ||||||
| val HEXAColor.Companion.antiquewhite |  | ||||||
|     get() = HEXAColor(0xFAEBD7FFu) |  | ||||||
| val HEXAColor.Companion.aqua |  | ||||||
|     get() = HEXAColor(0x00FFFFFFu) |  | ||||||
| val HEXAColor.Companion.aquamarine |  | ||||||
|     get() = HEXAColor(0x7FFFD4FFu) |  | ||||||
| val HEXAColor.Companion.azure |  | ||||||
|     get() = HEXAColor(0xF0FFFFFFu) |  | ||||||
| val HEXAColor.Companion.beige |  | ||||||
|     get() = HEXAColor(0xF5F5DCFFu) |  | ||||||
| val HEXAColor.Companion.bisque |  | ||||||
|     get() = HEXAColor(0xFFE4C4FFu) |  | ||||||
| val HEXAColor.Companion.black |  | ||||||
|     get() = HEXAColor(0x000000FFu) |  | ||||||
| val HEXAColor.Companion.blanchedalmond |  | ||||||
|     get() = HEXAColor(0xFFEBCDFFu) |  | ||||||
| val HEXAColor.Companion.blue |  | ||||||
|     get() = HEXAColor(0x0000FFFFu) |  | ||||||
| val HEXAColor.Companion.blueviolet |  | ||||||
|     get() = HEXAColor(0x8A2BE2FFu) |  | ||||||
| val HEXAColor.Companion.brown |  | ||||||
|     get() = HEXAColor(0xA52A2AFFu) |  | ||||||
| val HEXAColor.Companion.burlywood |  | ||||||
|     get() = HEXAColor(0xDEB887FFu) |  | ||||||
| val HEXAColor.Companion.cadetblue |  | ||||||
|     get() = HEXAColor(0x5F9EA0FFu) |  | ||||||
| val HEXAColor.Companion.chartreuse |  | ||||||
|     get() = HEXAColor(0x7FFF00FFu) |  | ||||||
| val HEXAColor.Companion.chocolate |  | ||||||
|     get() = HEXAColor(0xD2691EFFu) |  | ||||||
| val HEXAColor.Companion.coral |  | ||||||
|     get() = HEXAColor(0xFF7F50FFu) |  | ||||||
| val HEXAColor.Companion.cornflowerblue |  | ||||||
|     get() = HEXAColor(0x6495EDFFu) |  | ||||||
| val HEXAColor.Companion.cornsilk |  | ||||||
|     get() = HEXAColor(0xFFF8DCFFu) |  | ||||||
| val HEXAColor.Companion.crimson |  | ||||||
|     get() = HEXAColor(0xDC143CFFu) |  | ||||||
| val HEXAColor.Companion.cyan |  | ||||||
|     get() = HEXAColor(0x00FFFFFFu) |  | ||||||
| val HEXAColor.Companion.darkblue |  | ||||||
|     get() = HEXAColor(0x00008BFFu) |  | ||||||
| val HEXAColor.Companion.darkcyan |  | ||||||
|     get() = HEXAColor(0x008B8BFFu) |  | ||||||
| val HEXAColor.Companion.darkgoldenrod |  | ||||||
|     get() = HEXAColor(0xB8860BFFu) |  | ||||||
| val HEXAColor.Companion.darkgray |  | ||||||
|     get() = HEXAColor(0xA9A9A9FFu) |  | ||||||
| val HEXAColor.Companion.darkgreen |  | ||||||
|     get() = HEXAColor(0x006400FFu) |  | ||||||
| val HEXAColor.Companion.darkgrey |  | ||||||
|     get() = HEXAColor(0xA9A9A9FFu) |  | ||||||
| val HEXAColor.Companion.darkkhaki |  | ||||||
|     get() = HEXAColor(0xBDB76BFFu) |  | ||||||
| val HEXAColor.Companion.darkmagenta |  | ||||||
|     get() = HEXAColor(0x8B008BFFu) |  | ||||||
| val HEXAColor.Companion.darkolivegreen |  | ||||||
|     get() = HEXAColor(0x556B2FFFu) |  | ||||||
| val HEXAColor.Companion.darkorange |  | ||||||
|     get() = HEXAColor(0xFF8C00FFu) |  | ||||||
| val HEXAColor.Companion.darkorchid |  | ||||||
|     get() = HEXAColor(0x9932CCFFu) |  | ||||||
| val HEXAColor.Companion.darkred |  | ||||||
|     get() = HEXAColor(0x8B0000FFu) |  | ||||||
| val HEXAColor.Companion.darksalmon |  | ||||||
|     get() = HEXAColor(0xE9967AFFu) |  | ||||||
| val HEXAColor.Companion.darkseagreen |  | ||||||
|     get() = HEXAColor(0x8FBC8FFFu) |  | ||||||
| val HEXAColor.Companion.darkslateblue |  | ||||||
|     get() = HEXAColor(0x483D8BFFu) |  | ||||||
| val HEXAColor.Companion.darkslategray |  | ||||||
|     get() = HEXAColor(0x2F4F4FFFu) |  | ||||||
| val HEXAColor.Companion.darkslategrey |  | ||||||
|     get() = HEXAColor(0x2F4F4FFFu) |  | ||||||
| val HEXAColor.Companion.darkturquoise |  | ||||||
|     get() = HEXAColor(0x00CED1FFu) |  | ||||||
| val HEXAColor.Companion.darkviolet |  | ||||||
|     get() = HEXAColor(0x9400D3FFu) |  | ||||||
| val HEXAColor.Companion.deeppink |  | ||||||
|     get() = HEXAColor(0xFF1493FFu) |  | ||||||
| val HEXAColor.Companion.deepskyblue |  | ||||||
|     get() = HEXAColor(0x00BFFFFFu) |  | ||||||
| val HEXAColor.Companion.dimgray |  | ||||||
|     get() = HEXAColor(0x696969FFu) |  | ||||||
| val HEXAColor.Companion.dimgrey |  | ||||||
|     get() = HEXAColor(0x696969FFu) |  | ||||||
| val HEXAColor.Companion.dodgerblue |  | ||||||
|     get() = HEXAColor(0x1E90FFFFu) |  | ||||||
| val HEXAColor.Companion.firebrick |  | ||||||
|     get() = HEXAColor(0xB22222FFu) |  | ||||||
| val HEXAColor.Companion.floralwhite |  | ||||||
|     get() = HEXAColor(0xFFFAF0FFu) |  | ||||||
| val HEXAColor.Companion.forestgreen |  | ||||||
|     get() = HEXAColor(0x228B22FFu) |  | ||||||
| val HEXAColor.Companion.fuchsia |  | ||||||
|     get() = HEXAColor(0xFF00FFFFu) |  | ||||||
| val HEXAColor.Companion.gainsboro |  | ||||||
|     get() = HEXAColor(0xDCDCDCFFu) |  | ||||||
| val HEXAColor.Companion.ghostwhite |  | ||||||
|     get() = HEXAColor(0xF8F8FFFFu) |  | ||||||
| val HEXAColor.Companion.gold |  | ||||||
|     get() = HEXAColor(0xFFD700FFu) |  | ||||||
| val HEXAColor.Companion.goldenrod |  | ||||||
|     get() = HEXAColor(0xDAA520FFu) |  | ||||||
| val HEXAColor.Companion.gray |  | ||||||
|     get() = HEXAColor(0x808080FFu) |  | ||||||
| val HEXAColor.Companion.green |  | ||||||
|     get() = HEXAColor(0x008000FFu) |  | ||||||
| val HEXAColor.Companion.greenyellow |  | ||||||
|     get() = HEXAColor(0xADFF2FFFu) |  | ||||||
| val HEXAColor.Companion.grey |  | ||||||
|     get() = HEXAColor(0x808080FFu) |  | ||||||
| val HEXAColor.Companion.honeydew |  | ||||||
|     get() = HEXAColor(0xF0FFF0FFu) |  | ||||||
| val HEXAColor.Companion.hotpink |  | ||||||
|     get() = HEXAColor(0xFF69B4FFu) |  | ||||||
| val HEXAColor.Companion.indianred |  | ||||||
|     get() = HEXAColor(0xCD5C5CFFu) |  | ||||||
| val HEXAColor.Companion.indigo |  | ||||||
|     get() = HEXAColor(0x4B0082FFu) |  | ||||||
| val HEXAColor.Companion.ivory |  | ||||||
|     get() = HEXAColor(0xFFFFF0FFu) |  | ||||||
| val HEXAColor.Companion.khaki |  | ||||||
|     get() = HEXAColor(0xF0E68CFFu) |  | ||||||
| val HEXAColor.Companion.lavender |  | ||||||
|     get() = HEXAColor(0xE6E6FAFFu) |  | ||||||
| val HEXAColor.Companion.lavenderblush |  | ||||||
|     get() = HEXAColor(0xFFF0F5FFu) |  | ||||||
| val HEXAColor.Companion.lawngreen |  | ||||||
|     get() = HEXAColor(0x7CFC00FFu) |  | ||||||
| val HEXAColor.Companion.lemonchiffon |  | ||||||
|     get() = HEXAColor(0xFFFACDFFu) |  | ||||||
| val HEXAColor.Companion.lightblue |  | ||||||
|     get() = HEXAColor(0xADD8E6FFu) |  | ||||||
| val HEXAColor.Companion.lightcoral |  | ||||||
|     get() = HEXAColor(0xF08080FFu) |  | ||||||
| val HEXAColor.Companion.lightcyan |  | ||||||
|     get() = HEXAColor(0xE0FFFFFFu) |  | ||||||
| val HEXAColor.Companion.lightgoldenrodyellow |  | ||||||
|     get() = HEXAColor(0xFAFAD2FFu) |  | ||||||
| val HEXAColor.Companion.lightgray |  | ||||||
|     get() = HEXAColor(0xD3D3D3FFu) |  | ||||||
| val HEXAColor.Companion.lightgreen |  | ||||||
|     get() = HEXAColor(0x90EE90FFu) |  | ||||||
| val HEXAColor.Companion.lightgrey |  | ||||||
|     get() = HEXAColor(0xD3D3D3FFu) |  | ||||||
| val HEXAColor.Companion.lightpink |  | ||||||
|     get() = HEXAColor(0xFFB6C1FFu) |  | ||||||
| val HEXAColor.Companion.lightsalmon |  | ||||||
|     get() = HEXAColor(0xFFA07AFFu) |  | ||||||
| val HEXAColor.Companion.lightseagreen |  | ||||||
|     get() = HEXAColor(0x20B2AAFFu) |  | ||||||
| val HEXAColor.Companion.lightskyblue |  | ||||||
|     get() = HEXAColor(0x87CEFAFFu) |  | ||||||
| val HEXAColor.Companion.lightslategray |  | ||||||
|     get() = HEXAColor(0x778899FFu) |  | ||||||
| val HEXAColor.Companion.lightslategrey |  | ||||||
|     get() = HEXAColor(0x778899FFu) |  | ||||||
| val HEXAColor.Companion.lightsteelblue |  | ||||||
|     get() = HEXAColor(0xB0C4DEFFu) |  | ||||||
| val HEXAColor.Companion.lightyellow |  | ||||||
|     get() = HEXAColor(0xFFFFE0FFu) |  | ||||||
| val HEXAColor.Companion.lime |  | ||||||
|     get() = HEXAColor(0x00FF00FFu) |  | ||||||
| val HEXAColor.Companion.limegreen |  | ||||||
|     get() = HEXAColor(0x32CD32FFu) |  | ||||||
| val HEXAColor.Companion.linen |  | ||||||
|     get() = HEXAColor(0xFAF0E6FFu) |  | ||||||
| val HEXAColor.Companion.magenta |  | ||||||
|     get() = HEXAColor(0xFF00FFFFu) |  | ||||||
| val HEXAColor.Companion.maroon |  | ||||||
|     get() = HEXAColor(0x800000FFu) |  | ||||||
| val HEXAColor.Companion.mediumaquamarine |  | ||||||
|     get() = HEXAColor(0x66CDAAFFu) |  | ||||||
| val HEXAColor.Companion.mediumblue |  | ||||||
|     get() = HEXAColor(0x0000CDFFu) |  | ||||||
| val HEXAColor.Companion.mediumorchid |  | ||||||
|     get() = HEXAColor(0xBA55D3FFu) |  | ||||||
| val HEXAColor.Companion.mediumpurple |  | ||||||
|     get() = HEXAColor(0x9370DBFFu) |  | ||||||
| val HEXAColor.Companion.mediumseagreen |  | ||||||
|     get() = HEXAColor(0x3CB371FFu) |  | ||||||
| val HEXAColor.Companion.mediumslateblue |  | ||||||
|     get() = HEXAColor(0x7B68EEFFu) |  | ||||||
| val HEXAColor.Companion.mediumspringgreen |  | ||||||
|     get() = HEXAColor(0x00FA9AFFu) |  | ||||||
| val HEXAColor.Companion.mediumturquoise |  | ||||||
|     get() = HEXAColor(0x48D1CCFFu) |  | ||||||
| val HEXAColor.Companion.mediumvioletred |  | ||||||
|     get() = HEXAColor(0xC71585FFu) |  | ||||||
| val HEXAColor.Companion.midnightblue |  | ||||||
|     get() = HEXAColor(0x191970FFu) |  | ||||||
| val HEXAColor.Companion.mintcream |  | ||||||
|     get() = HEXAColor(0xF5FFFAFFu) |  | ||||||
| val HEXAColor.Companion.mistyrose |  | ||||||
|     get() = HEXAColor(0xFFE4E1FFu) |  | ||||||
| val HEXAColor.Companion.moccasin |  | ||||||
|     get() = HEXAColor(0xFFE4B5FFu) |  | ||||||
| val HEXAColor.Companion.navajowhite |  | ||||||
|     get() = HEXAColor(0xFFDEADFFu) |  | ||||||
| val HEXAColor.Companion.navy |  | ||||||
|     get() = HEXAColor(0x000080FFu) |  | ||||||
| val HEXAColor.Companion.oldlace |  | ||||||
|     get() = HEXAColor(0xFDF5E6FFu) |  | ||||||
| val HEXAColor.Companion.olive |  | ||||||
|     get() = HEXAColor(0x808000FFu) |  | ||||||
| val HEXAColor.Companion.olivedrab |  | ||||||
|     get() = HEXAColor(0x6B8E23FFu) |  | ||||||
| val HEXAColor.Companion.orange |  | ||||||
|     get() = HEXAColor(0xFFA500FFu) |  | ||||||
| val HEXAColor.Companion.orangered |  | ||||||
|     get() = HEXAColor(0xFF4500FFu) |  | ||||||
| val HEXAColor.Companion.orchid |  | ||||||
|     get() = HEXAColor(0xDA70D6FFu) |  | ||||||
| val HEXAColor.Companion.palegoldenrod |  | ||||||
|     get() = HEXAColor(0xEEE8AAFFu) |  | ||||||
| val HEXAColor.Companion.palegreen |  | ||||||
|     get() = HEXAColor(0x98FB98FFu) |  | ||||||
| val HEXAColor.Companion.paleturquoise |  | ||||||
|     get() = HEXAColor(0xAFEEEEFFu) |  | ||||||
| val HEXAColor.Companion.palevioletred |  | ||||||
|     get() = HEXAColor(0xDB7093FFu) |  | ||||||
| val HEXAColor.Companion.papayawhip |  | ||||||
|     get() = HEXAColor(0xFFEFD5FFu) |  | ||||||
| val HEXAColor.Companion.peachpuff |  | ||||||
|     get() = HEXAColor(0xFFDAB9FFu) |  | ||||||
| val HEXAColor.Companion.peru |  | ||||||
|     get() = HEXAColor(0xCD853FFFu) |  | ||||||
| val HEXAColor.Companion.pink |  | ||||||
|     get() = HEXAColor(0xFFC0CBFFu) |  | ||||||
| val HEXAColor.Companion.plum |  | ||||||
|     get() = HEXAColor(0xDDA0DDFFu) |  | ||||||
| val HEXAColor.Companion.powderblue |  | ||||||
|     get() = HEXAColor(0xB0E0E6FFu) |  | ||||||
| val HEXAColor.Companion.purple |  | ||||||
|     get() = HEXAColor(0x800080FFu) |  | ||||||
| val HEXAColor.Companion.red |  | ||||||
|     get() = HEXAColor(0xFF0000FFu) |  | ||||||
| val HEXAColor.Companion.rosybrown |  | ||||||
|     get() = HEXAColor(0xBC8F8FFFu) |  | ||||||
| val HEXAColor.Companion.royalblue |  | ||||||
|     get() = HEXAColor(0x4169E1FFu) |  | ||||||
| val HEXAColor.Companion.saddlebrown |  | ||||||
|     get() = HEXAColor(0x8B4513FFu) |  | ||||||
| val HEXAColor.Companion.salmon |  | ||||||
|     get() = HEXAColor(0xFA8072FFu) |  | ||||||
| val HEXAColor.Companion.sandybrown |  | ||||||
|     get() = HEXAColor(0xF4A460FFu) |  | ||||||
| val HEXAColor.Companion.seagreen |  | ||||||
|     get() = HEXAColor(0x2E8B57FFu) |  | ||||||
| val HEXAColor.Companion.seashell |  | ||||||
|     get() = HEXAColor(0xFFF5EEFFu) |  | ||||||
| val HEXAColor.Companion.sienna |  | ||||||
|     get() = HEXAColor(0xA0522DFFu) |  | ||||||
| val HEXAColor.Companion.silver |  | ||||||
|     get() = HEXAColor(0xC0C0C0FFu) |  | ||||||
| val HEXAColor.Companion.skyblue |  | ||||||
|     get() = HEXAColor(0x87CEEBFFu) |  | ||||||
| val HEXAColor.Companion.slateblue |  | ||||||
|     get() = HEXAColor(0x6A5ACDFFu) |  | ||||||
| val HEXAColor.Companion.slategray |  | ||||||
|     get() = HEXAColor(0x708090FFu) |  | ||||||
| val HEXAColor.Companion.slategrey |  | ||||||
|     get() = HEXAColor(0x708090FFu) |  | ||||||
| val HEXAColor.Companion.snow |  | ||||||
|     get() = HEXAColor(0xFFFAFAFFu) |  | ||||||
| val HEXAColor.Companion.springgreen |  | ||||||
|     get() = HEXAColor(0x00FF7FFFu) |  | ||||||
| val HEXAColor.Companion.steelblue |  | ||||||
|     get() = HEXAColor(0x4682B4FFu) |  | ||||||
| val HEXAColor.Companion.tan |  | ||||||
|     get() = HEXAColor(0xD2B48CFFu) |  | ||||||
| val HEXAColor.Companion.teal |  | ||||||
|     get() = HEXAColor(0x008080FFu) |  | ||||||
| val HEXAColor.Companion.thistle |  | ||||||
|     get() = HEXAColor(0xD8BFD8FFu) |  | ||||||
| val HEXAColor.Companion.tomato |  | ||||||
|     get() = HEXAColor(0xFF6347FFu) |  | ||||||
| val HEXAColor.Companion.turquoise |  | ||||||
|     get() = HEXAColor(0x40E0D0FFu) |  | ||||||
| val HEXAColor.Companion.violet |  | ||||||
|     get() = HEXAColor(0xEE82EEFFu) |  | ||||||
| val HEXAColor.Companion.wheat |  | ||||||
|     get() = HEXAColor(0xF5DEB3FFu) |  | ||||||
| val HEXAColor.Companion.white |  | ||||||
|     get() = HEXAColor(0xFFFFFFFFu) |  | ||||||
| val HEXAColor.Companion.whitesmoke |  | ||||||
|     get() = HEXAColor(0xF5F5F5FFu) |  | ||||||
| val HEXAColor.Companion.yellow |  | ||||||
|     get() = HEXAColor(0xFFFF00FFu) |  | ||||||
| val HEXAColor.Companion.yellowgreen |  | ||||||
|     get() = HEXAColor(0x9ACD32FFu) |  | ||||||
| @@ -4,15 +4,10 @@ plugins { | |||||||
|     id "com.android.library" |     id "com.android.library" | ||||||
| } | } | ||||||
|  |  | ||||||
| apply from: "$mppJvmJsWasmJsAndroidLinuxMingwLinuxArm64Project" | apply from: "$mppProjectWithSerializationPresetPath" | ||||||
|  |  | ||||||
| kotlin { | kotlin { | ||||||
|     sourceSets { |     sourceSets { | ||||||
|         commonMain { |  | ||||||
|             dependencies { |  | ||||||
|                 api libs.klock |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
|         jvmMain { |         jvmMain { | ||||||
|             dependencies { |             dependencies { | ||||||
|                 api project(":micro_utils.coroutines") |                 api project(":micro_utils.coroutines") | ||||||
| @@ -20,20 +15,7 @@ kotlin { | |||||||
|         } |         } | ||||||
|         androidMain { |         androidMain { | ||||||
|             dependencies { |             dependencies { | ||||||
|                 api libs.android.fragment |                 api project(":micro_utils.coroutines") | ||||||
|             } |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|         nativeMain { |  | ||||||
|             dependencies { |  | ||||||
|                 api libs.okio |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|         wasmJsMain { |  | ||||||
|             dependencies { |  | ||||||
|                 api libs.kotlinx.browser |  | ||||||
|                 api libs.kt.coroutines |  | ||||||
|             } |             } | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
|   | |||||||
| @@ -3,17 +3,15 @@ plugins { | |||||||
|     id "org.jetbrains.kotlin.plugin.serialization" |     id "org.jetbrains.kotlin.plugin.serialization" | ||||||
|     id "com.android.library" |     id "com.android.library" | ||||||
|     alias(libs.plugins.jb.compose) |     alias(libs.plugins.jb.compose) | ||||||
|     alias(libs.plugins.kt.jb.compose) |  | ||||||
| } | } | ||||||
|  |  | ||||||
| apply from: "$mppComposeJvmJsWasmJsAndroidLinuxMingwLinuxArm64Project" | apply from: "$mppProjectWithSerializationAndComposePresetPath" | ||||||
|  |  | ||||||
| kotlin { | kotlin { | ||||||
|     sourceSets { |     sourceSets { | ||||||
|         commonMain { |         commonMain { | ||||||
|             dependencies { |             dependencies { | ||||||
|                 api project(":micro_utils.common") |                 api project(":micro_utils.common") | ||||||
|                 api libs.kt.coroutines |  | ||||||
|             } |             } | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
|   | |||||||
| @@ -1 +0,0 @@ | |||||||
| <manifest/> |  | ||||||
| @@ -1,74 +0,0 @@ | |||||||
| package dev.inmo.micro_utils.common.compose |  | ||||||
|  |  | ||||||
| import androidx.compose.runtime.* |  | ||||||
| import dev.inmo.micro_utils.common.Optional |  | ||||||
| import dev.inmo.micro_utils.common.dataOrThrow |  | ||||||
| import dev.inmo.micro_utils.common.optional |  | ||||||
|  |  | ||||||
| class LoadableComponentContext<T> internal constructor( |  | ||||||
|     presetOptional: Optional<T>, |  | ||||||
| ) { |  | ||||||
|     internal val iterationState: MutableState<Int> = mutableStateOf(0) |  | ||||||
|  |  | ||||||
|     internal var dataOptional: Optional<T> = if (presetOptional.dataPresented) presetOptional else Optional.absent() |  | ||||||
|         private set |  | ||||||
|     internal val dataState: MutableState<Optional<T>> = mutableStateOf(dataOptional) |  | ||||||
|  |  | ||||||
|     fun reload() { |  | ||||||
|         iterationState.value++ |  | ||||||
|     } |  | ||||||
| } |  | ||||||
|  |  | ||||||
| /** |  | ||||||
|  * Showing data with ability to reload data |  | ||||||
|  * |  | ||||||
|  * [block] will be shown when [loader] will complete loading. If you want to reload data, just call |  | ||||||
|  * [LoadableComponentContext.reload] |  | ||||||
|  */ |  | ||||||
| @Composable |  | ||||||
| fun <T> LoadableComponent( |  | ||||||
|     preload: Optional<T>, |  | ||||||
|     loader: suspend LoadableComponentContext<T>.() -> T, |  | ||||||
|     block: @Composable LoadableComponentContext<T>.(T) -> Unit |  | ||||||
| ) { |  | ||||||
|     val context = remember { LoadableComponentContext(preload) } |  | ||||||
|  |  | ||||||
|     LaunchedEffect(context.iterationState.value) { |  | ||||||
|         context.dataState.value = loader(context).optional |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     context.dataState.let { |  | ||||||
|         if (it.value.dataPresented) { |  | ||||||
|             context.block(it.value.dataOrThrow(IllegalStateException("Data must be presented, but optional has been changed by some way"))) |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| } |  | ||||||
|  |  | ||||||
| /** |  | ||||||
|  * Showing data with ability to reload data |  | ||||||
|  * |  | ||||||
|  * [block] will be shown when [loader] will complete loading. If you want to reload data, just call |  | ||||||
|  * [LoadableComponentContext.reload] |  | ||||||
|  */ |  | ||||||
| @Composable |  | ||||||
| fun <T> LoadableComponent( |  | ||||||
|     preload: T, |  | ||||||
|     loader: suspend LoadableComponentContext<T>.() -> T, |  | ||||||
|     block: @Composable LoadableComponentContext<T>.(T) -> Unit |  | ||||||
| ) { |  | ||||||
|     LoadableComponent(preload.optional, loader, block) |  | ||||||
| } |  | ||||||
|  |  | ||||||
| /** |  | ||||||
|  * Showing data with ability to reload data |  | ||||||
|  * |  | ||||||
|  * [block] will be shown when [loader] will complete loading. If you want to reload data, just call |  | ||||||
|  * [LoadableComponentContext.reload] |  | ||||||
|  */ |  | ||||||
| @Composable |  | ||||||
| fun <T> LoadableComponent( |  | ||||||
|     loader: suspend LoadableComponentContext<T>.() -> T, |  | ||||||
|     block: @Composable LoadableComponentContext<T>.(T) -> Unit |  | ||||||
| ) { |  | ||||||
|     LoadableComponent(Optional.absent(), loader, block) |  | ||||||
| } |  | ||||||
| @@ -1,10 +0,0 @@ | |||||||
| package dev.inmo.micro_utils.common.compose |  | ||||||
|  |  | ||||||
| import androidx.compose.runtime.MutableState |  | ||||||
| import androidx.compose.runtime.State |  | ||||||
| import androidx.compose.runtime.derivedStateOf |  | ||||||
|  |  | ||||||
| /** |  | ||||||
|  * Converts current [MutableState] to immutable [State] using [derivedStateOf] |  | ||||||
|  */ |  | ||||||
| fun <T> MutableState<T>.asState(): State<T> = derivedStateOf { this.value } |  | ||||||
| @@ -1,19 +0,0 @@ | |||||||
| package dev.inmo.micro_utils.common.compose |  | ||||||
|  |  | ||||||
| import org.jetbrains.compose.web.dom.AttrBuilderContext |  | ||||||
| import org.w3c.dom.Element |  | ||||||
|  |  | ||||||
| operator fun <T : Element> AttrBuilderContext<T>?.plus( |  | ||||||
|     other: AttrBuilderContext<T>? |  | ||||||
| ) = when (this) { |  | ||||||
|     null -> other ?: {} |  | ||||||
|     else -> when (other) { |  | ||||||
|         null -> this ?: {} |  | ||||||
|         else -> { |  | ||||||
|             { |  | ||||||
|                 invoke(this) |  | ||||||
|                 other(this) |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| } |  | ||||||
| @@ -1,22 +0,0 @@ | |||||||
| package dev.inmo.micro_utils.common.compose |  | ||||||
|  |  | ||||||
| import androidx.compose.runtime.Composable |  | ||||||
| import androidx.compose.runtime.DisposableEffect |  | ||||||
| import androidx.compose.runtime.DisposableEffectResult |  | ||||||
| import androidx.compose.runtime.DisposableEffectScope |  | ||||||
| import org.jetbrains.compose.web.attributes.AttrsScope |  | ||||||
| import org.jetbrains.compose.web.dom.ElementScope |  | ||||||
| import org.w3c.dom.Element |  | ||||||
|  |  | ||||||
| /** |  | ||||||
|  * This function must be called in the context of your tag content. It works like default [AttrsScope.ref], |  | ||||||
|  * but able to be used several times. Uses [DisposableEffect] under the hood |  | ||||||
|  */ |  | ||||||
| @Composable |  | ||||||
| fun <T : Element> ElementScope<T>.ref( |  | ||||||
|     block: DisposableEffectScope.(T) -> DisposableEffectResult |  | ||||||
| ) { |  | ||||||
|     DisposableEffect(0) { |  | ||||||
|         block(scopeElement) |  | ||||||
|     } |  | ||||||
| } |  | ||||||
| @@ -1,43 +0,0 @@ | |||||||
| package dev.inmo.micro_utils.common.compose |  | ||||||
|  |  | ||||||
| import org.jetbrains.compose.web.css.* |  | ||||||
|  |  | ||||||
| object SkeletonAnimation : StyleSheet() { |  | ||||||
|     val skeletonKeyFrames: CSSNamedKeyframes by keyframes { |  | ||||||
|         to { backgroundPosition("-20% 0") } |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     fun CSSBuilder.includeSkeletonStyle( |  | ||||||
|         duration: CSSSizeValue<out CSSUnitTime> = 2.s, |  | ||||||
|         timingFunction: AnimationTimingFunction = AnimationTimingFunction.EaseInOut, |  | ||||||
|         iterationCount: Int? = null, |  | ||||||
|         direction: AnimationDirection = AnimationDirection.Normal, |  | ||||||
|         keyFrames: CSSNamedKeyframes = skeletonKeyFrames, |  | ||||||
|         hideChildren: Boolean = true, |  | ||||||
|         hideText: Boolean = hideChildren |  | ||||||
|     ) { |  | ||||||
|         backgroundImage("linear-gradient(110deg, rgb(236, 236, 236) 40%, rgb(245, 245, 245) 50%, rgb(236, 236, 236) 65%)") |  | ||||||
|         backgroundSize("200% 100%") |  | ||||||
|         backgroundPosition("180% 0") |  | ||||||
|         animation(keyFrames) { |  | ||||||
|             duration(duration) |  | ||||||
|             timingFunction(timingFunction) |  | ||||||
|             iterationCount(iterationCount) |  | ||||||
|             direction(direction) |  | ||||||
|         } |  | ||||||
|         if (hideText) { |  | ||||||
|             property("color", "${Color.transparent} !important") |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|         if (hideChildren) { |  | ||||||
|             child(self, universal) style { |  | ||||||
|                 property("visibility", "hidden") |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     val skeleton by style { |  | ||||||
|         includeSkeletonStyle() |  | ||||||
|     } |  | ||||||
| } |  | ||||||
|  |  | ||||||
| @@ -1,11 +0,0 @@ | |||||||
| package dev.inmo.micro_utils.common.compose |  | ||||||
|  |  | ||||||
| import org.jetbrains.compose.web.dom.AttrBuilderContext |  | ||||||
|  |  | ||||||
| fun tagClasses(vararg classnames: String): AttrBuilderContext<*> = { |  | ||||||
|     classes(*classnames) |  | ||||||
| } |  | ||||||
|  |  | ||||||
| fun tagId(id: String): AttrBuilderContext<*> = { |  | ||||||
|     id(id) |  | ||||||
| } |  | ||||||
| @@ -1,40 +0,0 @@ | |||||||
| import androidx.compose.runtime.remember |  | ||||||
| import androidx.compose.ui.test.ExperimentalTestApi |  | ||||||
| import androidx.compose.ui.test.runComposeUiTest |  | ||||||
| import dev.inmo.micro_utils.common.compose.LoadableComponent |  | ||||||
| import dev.inmo.micro_utils.coroutines.MutableRedeliverStateFlow |  | ||||||
| import kotlinx.coroutines.flow.filter |  | ||||||
| import kotlinx.coroutines.flow.first |  | ||||||
| import org.jetbrains.annotations.TestOnly |  | ||||||
| import kotlin.test.Test |  | ||||||
| import kotlin.test.assertTrue |  | ||||||
|  |  | ||||||
| class LoadableComponentTests { |  | ||||||
|     @OptIn(ExperimentalTestApi::class) |  | ||||||
|     @Test |  | ||||||
|     @TestOnly |  | ||||||
|     fun testSimpleLoad() = runComposeUiTest { |  | ||||||
|         val loadingFlow = MutableRedeliverStateFlow<Int>(0) |  | ||||||
|         val loadedFlow = MutableRedeliverStateFlow<Int>(0) |  | ||||||
|         setContent { |  | ||||||
|             LoadableComponent<Int>({ |  | ||||||
|                 loadingFlow.filter { it == 1 }.first() |  | ||||||
|             }) { |  | ||||||
|                 assert(dataState.value.data == 1) |  | ||||||
|                 remember { |  | ||||||
|                     loadedFlow.value = 2 |  | ||||||
|                 } |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|         waitForIdle() |  | ||||||
|  |  | ||||||
|         assertTrue(loadedFlow.value == 0) |  | ||||||
|  |  | ||||||
|         loadingFlow.value = 1 |  | ||||||
|  |  | ||||||
|         waitForIdle() |  | ||||||
|  |  | ||||||
|         assertTrue(loadedFlow.value == 2) |  | ||||||
|     } |  | ||||||
| } |  | ||||||
							
								
								
									
										1
									
								
								common/compose/src/main/AndroidManifest.xml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										1
									
								
								common/compose/src/main/AndroidManifest.xml
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1 @@ | |||||||
|  | <manifest package="dev.inmo.micro_utils.common.compose"/> | ||||||
| @@ -1,76 +0,0 @@ | |||||||
| package dev.inmo.micro_utils.common |  | ||||||
|  |  | ||||||
| import android.os.Bundle |  | ||||||
| import android.os.Parcelable |  | ||||||
| import androidx.fragment.app.Fragment |  | ||||||
| import java.io.Serializable |  | ||||||
| import kotlin.reflect.KProperty |  | ||||||
|  |  | ||||||
| object ArgumentPropertyNullableDelegate { |  | ||||||
|     operator fun <T: Any> getValue(thisRef: Fragment, property: KProperty<*>): T? { |  | ||||||
|         val arguments = thisRef.arguments ?: return null |  | ||||||
|         val key = property.name |  | ||||||
|         return when (property.getter.returnType.classifier) { |  | ||||||
|             // Scalars |  | ||||||
|             String::class -> arguments.getString(key) |  | ||||||
|             Boolean::class -> arguments.getBoolean(key) |  | ||||||
|             Byte::class -> arguments.getByte(key) |  | ||||||
|             Char::class -> arguments.getChar(key) |  | ||||||
|             Double::class -> arguments.getDouble(key) |  | ||||||
|             Float::class -> arguments.getFloat(key) |  | ||||||
|             Int::class -> arguments.getInt(key) |  | ||||||
|             Long::class -> arguments.getLong(key) |  | ||||||
|             Short::class -> arguments.getShort(key) |  | ||||||
|  |  | ||||||
|             // References |  | ||||||
|             Bundle::class -> arguments.getBundle(key) |  | ||||||
|             CharSequence::class -> arguments.getCharSequence(key) |  | ||||||
|             Parcelable::class -> arguments.getParcelable(key) |  | ||||||
|  |  | ||||||
|             // Scalar arrays |  | ||||||
|             BooleanArray::class -> arguments.getBooleanArray(key) |  | ||||||
|             ByteArray::class -> arguments.getByteArray(key) |  | ||||||
|             CharArray::class -> arguments.getCharArray(key) |  | ||||||
|             DoubleArray::class -> arguments.getDoubleArray(key) |  | ||||||
|             FloatArray::class -> arguments.getFloatArray(key) |  | ||||||
|             IntArray::class -> arguments.getIntArray(key) |  | ||||||
|             LongArray::class -> arguments.getLongArray(key) |  | ||||||
|             ShortArray::class -> arguments.getShortArray(key) |  | ||||||
|             Array::class -> { |  | ||||||
|                 val componentType = property.returnType.classifier ?.javaClass ?.componentType!! |  | ||||||
|                 @Suppress("UNCHECKED_CAST") // Checked by reflection. |  | ||||||
|                 when { |  | ||||||
|                     Parcelable::class.java.isAssignableFrom(componentType) -> { |  | ||||||
|                         arguments.getParcelableArray(key) |  | ||||||
|                     } |  | ||||||
|                     String::class.java.isAssignableFrom(componentType) -> { |  | ||||||
|                         arguments.getStringArray(key) |  | ||||||
|                     } |  | ||||||
|                     CharSequence::class.java.isAssignableFrom(componentType) -> { |  | ||||||
|                         arguments.getCharSequenceArray(key) |  | ||||||
|                     } |  | ||||||
|                     Serializable::class.java.isAssignableFrom(componentType) -> { |  | ||||||
|                         arguments.getSerializable(key) |  | ||||||
|                     } |  | ||||||
|                     else -> { |  | ||||||
|                         val valueType = componentType.canonicalName |  | ||||||
|                         throw IllegalArgumentException( |  | ||||||
|                             "Illegal value array type $valueType for key \"$key\"" |  | ||||||
|                         ) |  | ||||||
|                     } |  | ||||||
|                 } |  | ||||||
|             } |  | ||||||
|             Serializable::class -> arguments.getSerializable(key) |  | ||||||
|             else -> null |  | ||||||
|         } as? T |  | ||||||
|     } |  | ||||||
| } |  | ||||||
|  |  | ||||||
| object ArgumentPropertyNonNullableDelegate { |  | ||||||
|     operator fun <T: Any> getValue(thisRef: Fragment, property: KProperty<*>): T { |  | ||||||
|         return ArgumentPropertyNullableDelegate.getValue<T>(thisRef, property)!! |  | ||||||
|     } |  | ||||||
| } |  | ||||||
|  |  | ||||||
| fun argumentOrNull() = ArgumentPropertyNullableDelegate |  | ||||||
| fun argumentOrThrow() = ArgumentPropertyNonNullableDelegate |  | ||||||
| @@ -1,61 +0,0 @@ | |||||||
| package dev.inmo.micro_utils.common |  | ||||||
|  |  | ||||||
| import android.app.Activity |  | ||||||
| import android.view.View |  | ||||||
| import android.view.ViewGroup |  | ||||||
| import androidx.core.view.children |  | ||||||
| import androidx.fragment.app.Fragment |  | ||||||
|  |  | ||||||
| fun findViewsByTag(viewGroup: ViewGroup, tag: Any?): List<View> { |  | ||||||
|     return viewGroup.children.flatMap { |  | ||||||
|         findViewsByTag(it, tag) |  | ||||||
|     }.toList() |  | ||||||
| } |  | ||||||
|  |  | ||||||
| fun findViewsByTag(viewGroup: ViewGroup, key: Int, tag: Any?): List<View> { |  | ||||||
|     return viewGroup.children.flatMap { |  | ||||||
|         findViewsByTag(it, key, tag) |  | ||||||
|     }.toList() |  | ||||||
| } |  | ||||||
|  |  | ||||||
| fun findViewsByTag(view: View, tag: Any?): List<View> { |  | ||||||
|     val result = mutableListOf<View>() |  | ||||||
|     if (view.tag == tag) { |  | ||||||
|         result.add(view) |  | ||||||
|     } |  | ||||||
|     if (view is ViewGroup) { |  | ||||||
|         result.addAll(findViewsByTag(view, tag)) |  | ||||||
|     } |  | ||||||
|     return result.toList() |  | ||||||
| } |  | ||||||
|  |  | ||||||
| fun findViewsByTag(view: View, key: Int, tag: Any?): List<View> { |  | ||||||
|     val result = mutableListOf<View>() |  | ||||||
|     if (view.getTag(key) == tag) { |  | ||||||
|         result.add(view) |  | ||||||
|     } |  | ||||||
|     if (view is ViewGroup) { |  | ||||||
|         result.addAll(findViewsByTag(view, key, tag)) |  | ||||||
|     } |  | ||||||
|     return result.toList() |  | ||||||
| } |  | ||||||
|  |  | ||||||
| fun Activity.findViewsByTag(tag: Any?) = rootView ?.let { |  | ||||||
|     findViewsByTag(it, tag) |  | ||||||
| } |  | ||||||
|  |  | ||||||
| fun Activity.findViewsByTag(key: Int, tag: Any?) = rootView ?.let { |  | ||||||
|     findViewsByTag(it, key, tag) |  | ||||||
| } |  | ||||||
|  |  | ||||||
| fun Fragment.findViewsByTag(tag: Any?) = view ?.let { |  | ||||||
|     findViewsByTag(it, tag) |  | ||||||
| } |  | ||||||
|  |  | ||||||
| fun Fragment.findViewsByTag(key: Int, tag: Any?) = view ?.let { |  | ||||||
|     findViewsByTag(it, key, tag) |  | ||||||
| } |  | ||||||
|  |  | ||||||
| fun Fragment.findViewsByTagInActivity(tag: Any?) = activity ?.findViewsByTag(tag) |  | ||||||
|  |  | ||||||
| fun Fragment.findViewsByTagInActivity(key: Int, tag: Any?) = activity ?.findViewsByTag(key, tag) |  | ||||||
| @@ -1,7 +0,0 @@ | |||||||
| package dev.inmo.micro_utils.common |  | ||||||
|  |  | ||||||
| import android.app.Activity |  | ||||||
| import android.view.View |  | ||||||
|  |  | ||||||
| val Activity.rootView: View? |  | ||||||
|     get() = findViewById<View?>(android.R.id.content) ?.rootView ?: window.decorView.findViewById<View?>(android.R.id.content) ?.rootView |  | ||||||
| @@ -1,5 +1,3 @@ | |||||||
| @file:Suppress("OPT_IN_IS_NOT_ENABLED") |  | ||||||
|  |  | ||||||
| package dev.inmo.micro_utils.common | package dev.inmo.micro_utils.common | ||||||
|  |  | ||||||
| @RequiresOptIn( | @RequiresOptIn( | ||||||
|   | |||||||
| @@ -1,13 +0,0 @@ | |||||||
| package dev.inmo.micro_utils.common |  | ||||||
|  |  | ||||||
| fun <T> List<T>.breakAsPairs(): List<Pair<T, T>> { |  | ||||||
|     val result = mutableListOf<Pair<T, T>>() |  | ||||||
|  |  | ||||||
|     for (i in 0 until size - 1) { |  | ||||||
|         val first = get(i) |  | ||||||
|         val second = get(i + 1) |  | ||||||
|         result.add(first to second) |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     return result |  | ||||||
| } |  | ||||||
| @@ -1,6 +0,0 @@ | |||||||
| package dev.inmo.micro_utils.common |  | ||||||
|  |  | ||||||
| /** |  | ||||||
|  * Creates simple [Comparator] which will use [compareTo] of [T] for both objects |  | ||||||
|  */ |  | ||||||
| fun <T : Comparable<C>, C : T> T.createComparator() = Comparator<C> { o1, o2 -> o1.compareTo(o2) } |  | ||||||
| @@ -1,25 +0,0 @@ | |||||||
| package dev.inmo.micro_utils.common |  | ||||||
|  |  | ||||||
| import korlibs.time.DateTime |  | ||||||
| import kotlinx.serialization.KSerializer |  | ||||||
| import kotlinx.serialization.Serializer |  | ||||||
| import kotlinx.serialization.builtins.serializer |  | ||||||
| import kotlinx.serialization.descriptors.SerialDescriptor |  | ||||||
| import kotlinx.serialization.encoding.Decoder |  | ||||||
| import kotlinx.serialization.encoding.Encoder |  | ||||||
|  |  | ||||||
| /** |  | ||||||
|  * Serializes [DateTime] as its raw [DateTime.unixMillis] and deserializes in the same way |  | ||||||
|  */ |  | ||||||
| object DateTimeSerializer : KSerializer<DateTime> { |  | ||||||
|     override val descriptor: SerialDescriptor |  | ||||||
|         get() = Double.serializer().descriptor |  | ||||||
|  |  | ||||||
|     override fun deserialize(decoder: Decoder): DateTime { |  | ||||||
|         return DateTime(decoder.decodeDouble()) |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     override fun serialize(encoder: Encoder, value: DateTime) { |  | ||||||
|         encoder.encodeDouble(value.unixMillis) |  | ||||||
|     } |  | ||||||
| } |  | ||||||
| @@ -2,13 +2,11 @@ | |||||||
|  |  | ||||||
| package dev.inmo.micro_utils.common | package dev.inmo.micro_utils.common | ||||||
|  |  | ||||||
| import kotlinx.serialization.Serializable |  | ||||||
|  |  | ||||||
| private inline fun <T> getObject( | private inline fun <T> getObject( | ||||||
|     additional: MutableList<T>, |     additional: MutableList<T>, | ||||||
|     iterator: Iterator<T> |     iterator: Iterator<T> | ||||||
| ): T? = when { | ): T? = when { | ||||||
|     additional.isNotEmpty() -> additional.removeAt(0) |     additional.isNotEmpty() -> additional.removeFirst() | ||||||
|     iterator.hasNext() -> iterator.next() |     iterator.hasNext() -> iterator.next() | ||||||
|     else -> null |     else -> null | ||||||
| } | } | ||||||
| @@ -16,33 +14,16 @@ private inline fun <T> getObject( | |||||||
| /** | /** | ||||||
|  * Diff object which contains information about differences between two [Iterable]s |  * Diff object which contains information about differences between two [Iterable]s | ||||||
|  * |  * | ||||||
|  * See tests for more info |  | ||||||
|  * |  | ||||||
|  * @param removed The objects which has been presented in the old collection but absent in new one. Index here is the index in the old collection |  | ||||||
|  * @param added The object which appear in new collection only. Indexes here show the index in the new collection |  | ||||||
|  * @param replaced Pair of old-new changes. First object has been presented in the old collection on its |  | ||||||
|  * [IndexedValue.index] place, the second one is the object in new collection. Both have indexes due to the fact that in |  | ||||||
|  * case when some value has been replaced after adds or removes in original collection the object index will be changed |  | ||||||
|  * |  | ||||||
|  * @see calculateDiff |  * @see calculateDiff | ||||||
|  */ |  */ | ||||||
| @Serializable | data class Diff<T> internal constructor( | ||||||
| data class Diff<T> @Warning(warning) constructor( |     val removed: List<IndexedValue<T>>, | ||||||
|     val removed: List<@Serializable(IndexedValueSerializer::class) IndexedValue<T>>, |  | ||||||
|     /** |     /** | ||||||
|      * Old-New values pairs |      * Old-New values pairs | ||||||
|      */ |      */ | ||||||
|     val replaced: List<Pair<@Serializable(IndexedValueSerializer::class) IndexedValue<T>, @Serializable(IndexedValueSerializer::class) IndexedValue<T>>>, |     val replaced: List<Pair<IndexedValue<T>, IndexedValue<T>>>, | ||||||
|     val added: List<@Serializable(IndexedValueSerializer::class) IndexedValue<T>> |     val added: List<IndexedValue<T>> | ||||||
| ) { | ) | ||||||
|     fun isEmpty(): Boolean = removed.isEmpty() && replaced.isEmpty() && added.isEmpty() |  | ||||||
|  |  | ||||||
|     companion object { |  | ||||||
|         private const val warning = "This feature can be changed without any warranties. Use with caution and only in case you know what you are doing" |  | ||||||
|     } |  | ||||||
| } |  | ||||||
|  |  | ||||||
| fun <T> emptyDiff(): Diff<T> = Diff(emptyList(), emptyList(), emptyList()) |  | ||||||
|  |  | ||||||
| private inline fun <T> performChanges( | private inline fun <T> performChanges( | ||||||
|     potentialChanges: MutableList<Pair<IndexedValue<T>?, IndexedValue<T>?>>, |     potentialChanges: MutableList<Pair<IndexedValue<T>?, IndexedValue<T>?>>, | ||||||
| @@ -51,18 +32,17 @@ private inline fun <T> performChanges( | |||||||
|     changedList: MutableList<Pair<IndexedValue<T>, IndexedValue<T>>>, |     changedList: MutableList<Pair<IndexedValue<T>, IndexedValue<T>>>, | ||||||
|     removedList: MutableList<IndexedValue<T>>, |     removedList: MutableList<IndexedValue<T>>, | ||||||
|     addedList: MutableList<IndexedValue<T>>, |     addedList: MutableList<IndexedValue<T>>, | ||||||
|     comparisonFun: (T?, T?) -> Boolean |     strictComparison: Boolean | ||||||
| ) { | ) { | ||||||
|     var i = -1 |     var i = -1 | ||||||
|     val (oldObject, newObject) = potentialChanges.lastOrNull() ?: return |     val (oldObject, newObject) = potentialChanges.lastOrNull() ?: return | ||||||
|     for ((old, new) in potentialChanges.take(potentialChanges.size - 1)) { |     for ((old, new) in potentialChanges.take(potentialChanges.size - 1)) { | ||||||
|         i++ |         i++ | ||||||
|         val oldOneEqualToNewObject = comparisonFun(old ?.value, newObject ?.value) |         val oldOneEqualToNewObject = old ?.value === newObject ?.value || (old ?.value == newObject ?.value && !strictComparison) | ||||||
|         val newOneEqualToOldObject = comparisonFun(new ?.value, oldObject ?.value) |         val newOneEqualToOldObject = new ?.value === oldObject ?.value || (new ?.value == oldObject ?.value && !strictComparison) | ||||||
|         if (oldOneEqualToNewObject || newOneEqualToOldObject) { |         if (oldOneEqualToNewObject || newOneEqualToOldObject) { | ||||||
|             changedList.addAll( |             changedList.addAll( | ||||||
|                 potentialChanges.take(i).mapNotNull { |                 potentialChanges.take(i).mapNotNull { | ||||||
|                     @Suppress("UNCHECKED_CAST") |  | ||||||
|                     if (it.first != null && it.second != null) it as Pair<IndexedValue<T>, IndexedValue<T>> else null |                     if (it.first != null && it.second != null) it as Pair<IndexedValue<T>, IndexedValue<T>> else null | ||||||
|                 } |                 } | ||||||
|             ) |             ) | ||||||
| @@ -112,7 +92,7 @@ private inline fun <T> performChanges( | |||||||
|  */ |  */ | ||||||
| fun <T> Iterable<T>.calculateDiff( | fun <T> Iterable<T>.calculateDiff( | ||||||
|     other: Iterable<T>, |     other: Iterable<T>, | ||||||
|     comparisonFun: (T?, T?) -> Boolean |     strictComparison: Boolean = false | ||||||
| ): Diff<T> { | ): Diff<T> { | ||||||
|     var i = -1 |     var i = -1 | ||||||
|     var j = -1 |     var j = -1 | ||||||
| @@ -140,60 +120,31 @@ fun <T> Iterable<T>.calculateDiff( | |||||||
|         } |         } | ||||||
|  |  | ||||||
|         when { |         when { | ||||||
|             comparisonFun(oldObject, newObject) -> { |             oldObject === newObject || (oldObject == newObject && !strictComparison) -> { | ||||||
|                 changedObjects.addAll(potentiallyChangedObjects.map { |                 changedObjects.addAll(potentiallyChangedObjects.map { it as Pair<IndexedValue<T>, IndexedValue<T>> }) | ||||||
|                     @Suppress("UNCHECKED_CAST") |  | ||||||
|                     it as Pair<IndexedValue<T>, IndexedValue<T>> |  | ||||||
|                 }) |  | ||||||
|                 potentiallyChangedObjects.clear() |                 potentiallyChangedObjects.clear() | ||||||
|             } |             } | ||||||
|             else -> { |             else -> { | ||||||
|                 potentiallyChangedObjects.add(oldObject ?.let { IndexedValue(i, oldObject) } to newObject ?.let { IndexedValue(j, newObject) }) |                 potentiallyChangedObjects.add(oldObject ?.let { IndexedValue(i, oldObject) } to newObject ?.let { IndexedValue(j, newObject) }) | ||||||
|                 val previousOldsAdditionsSize = additionalInOld.size |                 val previousOldsAdditionsSize = additionalInOld.size | ||||||
|                 val previousNewsAdditionsSize = additionalInNew.size |                 val previousNewsAdditionsSize = additionalInNew.size | ||||||
|                 performChanges(potentiallyChangedObjects, additionalInOld, additionalInNew, changedObjects, removedObjects, addedObjects, comparisonFun) |                 performChanges(potentiallyChangedObjects, additionalInOld, additionalInNew, changedObjects, removedObjects, addedObjects, strictComparison) | ||||||
|                 i -= (additionalInOld.size - previousOldsAdditionsSize) |                 i -= (additionalInOld.size - previousOldsAdditionsSize) | ||||||
|                 j -= (additionalInNew.size - previousNewsAdditionsSize) |                 j -= (additionalInNew.size - previousNewsAdditionsSize) | ||||||
|             } |             } | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
|     potentiallyChangedObjects.add(null to null) |     potentiallyChangedObjects.add(null to null) | ||||||
|     performChanges(potentiallyChangedObjects, additionalInOld, additionalInNew, changedObjects, removedObjects, addedObjects, comparisonFun) |     performChanges(potentiallyChangedObjects, additionalInOld, additionalInNew, changedObjects, removedObjects, addedObjects, strictComparison) | ||||||
|  |  | ||||||
|     return Diff(removedObjects.toList(), changedObjects.toList(), addedObjects.toList()) |     return Diff(removedObjects.toList(), changedObjects.toList(), addedObjects.toList()) | ||||||
| } | } | ||||||
|  |  | ||||||
| /** |  | ||||||
|  * Calculating [Diff] object |  | ||||||
|  * |  | ||||||
|  * @param strictComparison If this parameter set to true, objects which are not equal by links will be used as different |  | ||||||
|  * objects. For example, in case of two "Example" string they will be equal by value, but CAN be different by links |  | ||||||
|  */ |  | ||||||
| fun <T> Iterable<T>.calculateDiff( |  | ||||||
|     other: Iterable<T>, |  | ||||||
|     strictComparison: Boolean = false |  | ||||||
| ): Diff<T> = calculateDiff( |  | ||||||
|     other, |  | ||||||
|     comparisonFun = if (strictComparison) { |  | ||||||
|         { t1, t2 -> |  | ||||||
|             t1 === t2 |  | ||||||
|         } |  | ||||||
|     } else { |  | ||||||
|         { t1, t2 -> |  | ||||||
|             t1 === t2 || t1 == t2 // small optimization for cases when t1 and t2 are the same - comparison will be faster potentially |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| ) |  | ||||||
| inline fun <T> Iterable<T>.diff( | inline fun <T> Iterable<T>.diff( | ||||||
|     other: Iterable<T>, |     other: Iterable<T>, | ||||||
|     strictComparison: Boolean = false |     strictComparison: Boolean = false | ||||||
| ): Diff<T> = calculateDiff(other, strictComparison) | ): Diff<T> = calculateDiff(other, strictComparison) | ||||||
| inline fun <T> Iterable<T>.diff( |  | ||||||
|     other: Iterable<T>, |  | ||||||
|     noinline comparisonFun: (T?, T?) -> Boolean |  | ||||||
| ): Diff<T> = calculateDiff(other, comparisonFun) |  | ||||||
|  |  | ||||||
| inline fun <T> Diff(old: Iterable<T>, new: Iterable<T>) = old.calculateDiff(new, strictComparison = false) | inline fun <T> Diff(old: Iterable<T>, new: Iterable<T>) = old.calculateDiff(new) | ||||||
| inline fun <T> StrictDiff(old: Iterable<T>, new: Iterable<T>) = old.calculateDiff(new, true) | inline fun <T> StrictDiff(old: Iterable<T>, new: Iterable<T>) = old.calculateDiff(new, true) | ||||||
|  |  | ||||||
| /** | /** | ||||||
| @@ -203,23 +154,6 @@ inline fun <T> Iterable<T>.calculateStrictDiff( | |||||||
|     other: Iterable<T> |     other: Iterable<T> | ||||||
| ) = calculateDiff(other, strictComparison = true) | ) = calculateDiff(other, strictComparison = true) | ||||||
|  |  | ||||||
| /** |  | ||||||
|  * Applies [diff] to [this] [MutableList] |  | ||||||
|  */ |  | ||||||
| fun <T> MutableList<T>.applyDiff( |  | ||||||
|     diff: Diff<T> |  | ||||||
| ) { |  | ||||||
|     for (i in diff.removed.indices.sortedDescending()) { |  | ||||||
|         removeAt(diff.removed[i].index) |  | ||||||
|     } |  | ||||||
|     diff.added.forEach { (i, t) -> |  | ||||||
|         add(i, t) |  | ||||||
|     } |  | ||||||
|     diff.replaced.forEach { (_, new) -> |  | ||||||
|         set(new.index, new.value) |  | ||||||
|     } |  | ||||||
| } |  | ||||||
|  |  | ||||||
| /** | /** | ||||||
|  * This method call [calculateDiff] with strict mode [strictComparison] and then apply differences to [this] |  * This method call [calculateDiff] with strict mode [strictComparison] and then apply differences to [this] | ||||||
|  * mutable list |  * mutable list | ||||||
| @@ -227,27 +161,14 @@ fun <T> MutableList<T>.applyDiff( | |||||||
| fun <T> MutableList<T>.applyDiff( | fun <T> MutableList<T>.applyDiff( | ||||||
|     source: Iterable<T>, |     source: Iterable<T>, | ||||||
|     strictComparison: Boolean = false |     strictComparison: Boolean = false | ||||||
| ): Diff<T> = calculateDiff(source, strictComparison).also { | ) = calculateDiff(source, strictComparison).let { | ||||||
|     applyDiff(it) |     for (i in it.removed.indices.sortedDescending()) { | ||||||
|  |         removeAt(it.removed[i].index) | ||||||
|  |     } | ||||||
|  |     it.added.forEach { (i, t) -> | ||||||
|  |         add(i, t) | ||||||
|  |     } | ||||||
|  |     it.replaced.forEach { (_, new) -> | ||||||
|  |         set(new.index, new.value) | ||||||
|     } |     } | ||||||
|  |  | ||||||
| /** |  | ||||||
|  * This method call [calculateDiff] and then apply differences to [this] |  | ||||||
|  * mutable list |  | ||||||
|  */ |  | ||||||
| fun <T> MutableList<T>.applyDiff( |  | ||||||
|     source: Iterable<T>, |  | ||||||
|     comparisonFun: (T?, T?) -> Boolean |  | ||||||
| ): Diff<T> = calculateDiff(source, comparisonFun).also { |  | ||||||
|     applyDiff(it) |  | ||||||
| } | } | ||||||
|  |  | ||||||
| /** |  | ||||||
|  * Reverse [this] [Diff]. Result will contain [Diff.added] on [Diff.removed] (and vice-verse), all the |  | ||||||
|  * [Diff.replaced] values will be reversed too |  | ||||||
|  */ |  | ||||||
| fun <T> Diff<T>.reversed() = Diff( |  | ||||||
|     removed = added, |  | ||||||
|     replaced = replaced.map { it.second to it.first }, |  | ||||||
|     added = removed |  | ||||||
| ) |  | ||||||
|   | |||||||
| @@ -1,5 +1,3 @@ | |||||||
| @file:Suppress("unused", "NOTHING_TO_INLINE") |  | ||||||
|  |  | ||||||
| package dev.inmo.micro_utils.common | package dev.inmo.micro_utils.common | ||||||
|  |  | ||||||
| import kotlinx.serialization.* | import kotlinx.serialization.* | ||||||
| @@ -23,18 +21,26 @@ import kotlinx.serialization.encoding.* | |||||||
| sealed interface Either<T1, T2> { | sealed interface Either<T1, T2> { | ||||||
|     val optionalT1: Optional<T1> |     val optionalT1: Optional<T1> | ||||||
|     val optionalT2: Optional<T2> |     val optionalT2: Optional<T2> | ||||||
|  |     @Deprecated("Use optionalT1 instead", ReplaceWith("optionalT1")) | ||||||
|     val t1OrNull: T1? |     val t1: T1? | ||||||
|         get() = optionalT1.dataOrNull() |         get() = optionalT1.dataOrNull() | ||||||
|     val t2OrNull: T2? |     @Deprecated("Use optionalT2 instead", ReplaceWith("optionalT2")) | ||||||
|  |     val t2: T2? | ||||||
|         get() = optionalT2.dataOrNull() |         get() = optionalT2.dataOrNull() | ||||||
|  |  | ||||||
|  |     companion object { | ||||||
|  |         fun <T1, T2> serializer( | ||||||
|  |             t1Serializer: KSerializer<T1>, | ||||||
|  |             t2Serializer: KSerializer<T2>, | ||||||
|  |         ): KSerializer<Either<T1, T2>> = EitherSerializer(t1Serializer, t2Serializer) | ||||||
|  |     } | ||||||
| } | } | ||||||
|  |  | ||||||
| class EitherSerializer<T1, T2>( | class EitherSerializer<T1, T2>( | ||||||
|     t1Serializer: KSerializer<T1>, |     t1Serializer: KSerializer<T1>, | ||||||
|     t2Serializer: KSerializer<T2>, |     t2Serializer: KSerializer<T2>, | ||||||
| ) : KSerializer<Either<T1, T2>> { | ) : KSerializer<Either<T1, T2>> { | ||||||
|     @OptIn(InternalSerializationApi::class, ExperimentalSerializationApi::class) |     @OptIn(ExperimentalSerializationApi::class, InternalSerializationApi::class) | ||||||
|     override val descriptor: SerialDescriptor = buildSerialDescriptor( |     override val descriptor: SerialDescriptor = buildSerialDescriptor( | ||||||
|         "TypedSerializer", |         "TypedSerializer", | ||||||
|         SerialKind.CONTEXTUAL |         SerialKind.CONTEXTUAL | ||||||
| @@ -97,7 +103,7 @@ class EitherSerializer<T1, T2>( | |||||||
|  */ |  */ | ||||||
| @Serializable | @Serializable | ||||||
| data class EitherFirst<T1, T2>( | data class EitherFirst<T1, T2>( | ||||||
|     val t1: T1 |     override val t1: T1 | ||||||
| ) : Either<T1, T2> { | ) : Either<T1, T2> { | ||||||
|     override val optionalT1: Optional<T1> = t1.optional |     override val optionalT1: Optional<T1> = t1.optional | ||||||
|     override val optionalT2: Optional<T2> = Optional.absent() |     override val optionalT2: Optional<T2> = Optional.absent() | ||||||
| @@ -108,7 +114,7 @@ data class EitherFirst<T1, T2>( | |||||||
|  */ |  */ | ||||||
| @Serializable | @Serializable | ||||||
| data class EitherSecond<T1, T2>( | data class EitherSecond<T1, T2>( | ||||||
|     val t2: T2 |     override val t2: T2 | ||||||
| ) : Either<T1, T2> { | ) : Either<T1, T2> { | ||||||
|     override val optionalT1: Optional<T1> = Optional.absent() |     override val optionalT1: Optional<T1> = Optional.absent() | ||||||
|     override val optionalT2: Optional<T2> = t2.optional |     override val optionalT2: Optional<T2> = t2.optional | ||||||
|   | |||||||
| @@ -1,43 +0,0 @@ | |||||||
| package dev.inmo.micro_utils.common |  | ||||||
|  |  | ||||||
| inline fun <T> Boolean.letIfTrue(block: () -> T): T? { |  | ||||||
|     return if (this) { |  | ||||||
|         block() |  | ||||||
|     } else { |  | ||||||
|         null |  | ||||||
|     } |  | ||||||
| } |  | ||||||
|  |  | ||||||
| inline fun <T> Boolean.letIfFalse(block: () -> T): T? { |  | ||||||
|     return if (this) { |  | ||||||
|         null |  | ||||||
|     } else { |  | ||||||
|         block() |  | ||||||
|     } |  | ||||||
| } |  | ||||||
|  |  | ||||||
| inline fun Boolean.alsoIfTrue(block: () -> Unit): Boolean { |  | ||||||
|     letIfTrue(block) |  | ||||||
|     return this |  | ||||||
| } |  | ||||||
|  |  | ||||||
| inline fun Boolean.alsoIfFalse(block: () -> Unit): Boolean { |  | ||||||
|     letIfFalse(block) |  | ||||||
|     return this |  | ||||||
| } |  | ||||||
|  |  | ||||||
| inline fun <T> Boolean.ifTrue(block: () -> T): T? { |  | ||||||
|     return if (this) { |  | ||||||
|         block() |  | ||||||
|     } else { |  | ||||||
|         null |  | ||||||
|     } |  | ||||||
| } |  | ||||||
|  |  | ||||||
| inline fun <T> Boolean.ifFalse(block: () -> T): T? { |  | ||||||
|     return if (this) { |  | ||||||
|         null |  | ||||||
|     } else { |  | ||||||
|         block() |  | ||||||
|     } |  | ||||||
| } |  | ||||||
| @@ -1,30 +0,0 @@ | |||||||
| package dev.inmo.micro_utils.common |  | ||||||
|  |  | ||||||
| import kotlinx.serialization.KSerializer |  | ||||||
| import kotlinx.serialization.Serializer |  | ||||||
| import kotlinx.serialization.builtins.PairSerializer |  | ||||||
| import kotlinx.serialization.builtins.serializer |  | ||||||
| import kotlinx.serialization.descriptors.SerialDescriptor |  | ||||||
| import kotlinx.serialization.encoding.Decoder |  | ||||||
| import kotlinx.serialization.encoding.Encoder |  | ||||||
|  |  | ||||||
| class IndexedValueSerializer<T>(private val subSerializer: KSerializer<T>) : KSerializer<IndexedValue<T>> { |  | ||||||
|     private val originalSerializer = PairSerializer(Int.serializer(), subSerializer) |  | ||||||
|     override val descriptor: SerialDescriptor |  | ||||||
|         get() = originalSerializer.descriptor |  | ||||||
|  |  | ||||||
|     override fun deserialize(decoder: Decoder): IndexedValue<T> { |  | ||||||
|         val pair = originalSerializer.deserialize(decoder) |  | ||||||
|         return IndexedValue( |  | ||||||
|             pair.first, |  | ||||||
|             pair.second |  | ||||||
|         ) |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     override fun serialize(encoder: Encoder, value: IndexedValue<T>) { |  | ||||||
|         originalSerializer.serialize( |  | ||||||
|             encoder, |  | ||||||
|             Pair(value.index, value.value) |  | ||||||
|         ) |  | ||||||
|     } |  | ||||||
| } |  | ||||||
| @@ -7,7 +7,7 @@ import kotlin.jvm.JvmInline | |||||||
| @JvmInline | @JvmInline | ||||||
| value class FileName(val string: String) { | value class FileName(val string: String) { | ||||||
|     val name: String |     val name: String | ||||||
|         get() = withoutSlashAtTheEnd.takeLastWhile { it != MPPFilePathSeparator } |         get() = withoutSlashAtTheEnd.takeLastWhile { it != '/' } | ||||||
|     val extension: String |     val extension: String | ||||||
|         get() = name.takeLastWhile { it != '.' } |         get() = name.takeLastWhile { it != '.' } | ||||||
|     val nameWithoutExtension: String |     val nameWithoutExtension: String | ||||||
| @@ -18,7 +18,7 @@ value class FileName(val string: String) { | |||||||
|             } ?: filename |             } ?: filename | ||||||
|         } |         } | ||||||
|     val withoutSlashAtTheEnd: String |     val withoutSlashAtTheEnd: String | ||||||
|         get() = string.dropLastWhile { it == MPPFilePathSeparator } |         get() = string.dropLastWhile { it == '/' } | ||||||
|     override fun toString(): String = string |     override fun toString(): String = string | ||||||
| } | } | ||||||
|  |  | ||||||
| @@ -26,7 +26,6 @@ value class FileName(val string: String) { | |||||||
| expect class MPPFile | expect class MPPFile | ||||||
|  |  | ||||||
| expect val MPPFile.filename: FileName | expect val MPPFile.filename: FileName | ||||||
| expect val MPPFilePathSeparator: Char |  | ||||||
| expect val MPPFile.filesize: Long | expect val MPPFile.filesize: Long | ||||||
| expect val MPPFile.bytesAllocatorSync: ByteArrayAllocator | expect val MPPFile.bytesAllocatorSync: ByteArrayAllocator | ||||||
| expect val MPPFile.bytesAllocator: SuspendByteArrayAllocator | expect val MPPFile.bytesAllocator: SuspendByteArrayAllocator | ||||||
|   | |||||||
| @@ -1,133 +0,0 @@ | |||||||
| package dev.inmo.micro_utils.common |  | ||||||
|  |  | ||||||
| /** |  | ||||||
|  * Contains diff based on the comparison of objects with the same [K]. |  | ||||||
|  * |  | ||||||
|  * @param removed Contains map with keys removed from parent map |  | ||||||
|  * @param changed Contains map with keys values changed new map in comparison with old one |  | ||||||
|  * @param added Contains map with new keys and values |  | ||||||
|  */ |  | ||||||
| data class MapDiff<K, V> @Warning(warning) constructor( |  | ||||||
|     val removed: Map<K, V>, |  | ||||||
|     val changed: Map<K, Pair<V, V>>, |  | ||||||
|     val added: Map<K, V> |  | ||||||
| ) { |  | ||||||
|     fun isEmpty() = removed.isEmpty() && changed.isEmpty() && added.isEmpty() |  | ||||||
|     inline fun isNotEmpty() = !isEmpty() |  | ||||||
|  |  | ||||||
|     companion object { |  | ||||||
|         private const val warning = "This feature can be changed without any warranties. Use with caution and only in case you know what you are doing" |  | ||||||
|         fun <K, V> empty() = MapDiff<K, V>(emptyMap(), emptyMap(), emptyMap()) |  | ||||||
|     } |  | ||||||
| } |  | ||||||
|  |  | ||||||
| private inline fun <K, V> createCompareFun( |  | ||||||
|     strictComparison: Boolean |  | ||||||
| ): (K, V, V) -> Boolean = if (strictComparison) { |  | ||||||
|     { _, first, second -> first === second } |  | ||||||
| } else { |  | ||||||
|     { _, first, second -> first == second } |  | ||||||
| } |  | ||||||
|  |  | ||||||
| /** |  | ||||||
|  * Compare [this] [Map] with the [other] one in principle when [other] is newer than [this] |  | ||||||
|  * |  | ||||||
|  * @param compareFun Will be used to determine changed values |  | ||||||
|  */ |  | ||||||
| fun <K, V> Map<K, V>.diff( |  | ||||||
|     other: Map<K, V>, |  | ||||||
|     compareFun: (K, V, V) -> Boolean |  | ||||||
| ): MapDiff<K, V> { |  | ||||||
|     val removed: Map<K, V> = (keys - other.keys).associateWith { |  | ||||||
|         getValue(it) |  | ||||||
|     } |  | ||||||
|     val added: Map<K, V> = (other.keys - keys).associateWith { |  | ||||||
|         other.getValue(it) |  | ||||||
|     } |  | ||||||
|     val changed = keys.intersect(other.keys).mapNotNull { |  | ||||||
|         val old = getValue(it) |  | ||||||
|         val new = other.getValue(it) |  | ||||||
|         if (compareFun(it, old, new)) { |  | ||||||
|             return@mapNotNull null |  | ||||||
|         } else { |  | ||||||
|             it to (old to new) |  | ||||||
|         } |  | ||||||
|     }.toMap() |  | ||||||
|  |  | ||||||
|     return MapDiff( |  | ||||||
|         removed, |  | ||||||
|         changed, |  | ||||||
|         added |  | ||||||
|     ) |  | ||||||
| } |  | ||||||
|  |  | ||||||
| /** |  | ||||||
|  * Compare [this] [Map] with the [other] one in principle when [other] is newer than [this] |  | ||||||
|  * |  | ||||||
|  * @param strictComparison If true, will use strict (===) comparison for the values' comparison. Otherwise, standard |  | ||||||
|  * `equals` will be used |  | ||||||
|  */ |  | ||||||
| fun <K, V> Map<K, V>.diff( |  | ||||||
|     other: Map<K, V>, |  | ||||||
|     strictComparison: Boolean = false |  | ||||||
| ): MapDiff<K, V> = diff( |  | ||||||
|     other, |  | ||||||
|     compareFun = createCompareFun(strictComparison) |  | ||||||
| ) |  | ||||||
|  |  | ||||||
| /** |  | ||||||
|  * Will apply [mapDiff] to [this] [MutableMap] |  | ||||||
|  */ |  | ||||||
| fun <K, V> MutableMap<K, V>.applyDiff( |  | ||||||
|     mapDiff: MapDiff<K, V> |  | ||||||
| ) { |  | ||||||
|     mapDiff.apply { |  | ||||||
|         keys.removeAll(removed.keys) |  | ||||||
|         changed.forEach { (k, oldNew) -> |  | ||||||
|             put(k, oldNew.second) |  | ||||||
|         } |  | ||||||
|         putAll(added) |  | ||||||
|     } |  | ||||||
| } |  | ||||||
|  |  | ||||||
| /** |  | ||||||
|  * Will apply changes with [from] map into [this] one |  | ||||||
|  * |  | ||||||
|  * @param compareFun Will be used to determine changed values |  | ||||||
|  * |  | ||||||
|  * @return [MapDiff] applied to [this] [MutableMap] |  | ||||||
|  */ |  | ||||||
| fun <K, V> MutableMap<K, V>.applyDiff( |  | ||||||
|     from: Map<K, V>, |  | ||||||
|     compareFun: (K, V, V) -> Boolean |  | ||||||
| ): MapDiff<K, V> { |  | ||||||
|     return diff(from, compareFun).also { |  | ||||||
|         applyDiff(it) |  | ||||||
|     } |  | ||||||
| } |  | ||||||
|  |  | ||||||
| /** |  | ||||||
|  * Will apply changes with [from] map into [this] one |  | ||||||
|  * |  | ||||||
|  * @param strictComparison If true, will use strict (===) comparison for the values' comparison. Otherwise, standard |  | ||||||
|  * `equals` will be used |  | ||||||
|  * |  | ||||||
|  * @return [MapDiff] applied to [this] [MutableMap] |  | ||||||
|  */ |  | ||||||
| fun <K, V> MutableMap<K, V>.applyDiff( |  | ||||||
|     from: Map<K, V>, |  | ||||||
|     strictComparison: Boolean = false |  | ||||||
| ): MapDiff<K, V> = applyDiff( |  | ||||||
|     from, |  | ||||||
|     compareFun = createCompareFun(strictComparison) |  | ||||||
| ) |  | ||||||
|  |  | ||||||
| /** |  | ||||||
|  * Reverse [this] [MapDiff]. Result will contain [MapDiff.added] on [MapDiff.removed] (and vice-verse), all the |  | ||||||
|  * [MapDiff.changed] values will be reversed too |  | ||||||
|  */ |  | ||||||
| fun <K, V> MapDiff<K, V>.reversed(): MapDiff<K, V> = MapDiff( |  | ||||||
|     removed = added, |  | ||||||
|     changed = changed.mapValues { (_, oldNew) -> oldNew.second to oldNew.first }, |  | ||||||
|     added = removed |  | ||||||
| ) |  | ||||||
| @@ -1,53 +0,0 @@ | |||||||
| package dev.inmo.micro_utils.common |  | ||||||
|  |  | ||||||
| import kotlin.jvm.JvmName |  | ||||||
|  |  | ||||||
| interface SimpleMapper<T1, T2> { |  | ||||||
|     fun convertToT1(from: T2): T1 |  | ||||||
|     fun convertToT2(from: T1): T2 |  | ||||||
| } |  | ||||||
|  |  | ||||||
| @JvmName("convertFromT2") |  | ||||||
| fun <T1, T2> SimpleMapper<T1, T2>.convert(from: T2) = convertToT1(from) |  | ||||||
| @JvmName("convertFromT1") |  | ||||||
| fun <T1, T2> SimpleMapper<T1, T2>.convert(from: T1) = convertToT2(from) |  | ||||||
|  |  | ||||||
| class SimpleMapperImpl<T1, T2>( |  | ||||||
|     private val t1: (T2) -> T1, |  | ||||||
|     private val t2: (T1) -> T2, |  | ||||||
| ) : SimpleMapper<T1, T2> { |  | ||||||
|     override fun convertToT1(from: T2): T1 = t1.invoke(from) |  | ||||||
|  |  | ||||||
|     override fun convertToT2(from: T1): T2 = t2.invoke(from) |  | ||||||
| } |  | ||||||
|  |  | ||||||
| @Suppress("NOTHING_TO_INLINE") |  | ||||||
| inline fun <T1, T2> simpleMapper( |  | ||||||
|     noinline t1: (T2) -> T1, |  | ||||||
|     noinline t2: (T1) -> T2, |  | ||||||
| ) = SimpleMapperImpl(t1, t2) |  | ||||||
|  |  | ||||||
| interface SimpleSuspendableMapper<T1, T2> { |  | ||||||
|     suspend fun convertToT1(from: T2): T1 |  | ||||||
|     suspend fun convertToT2(from: T1): T2 |  | ||||||
| } |  | ||||||
|  |  | ||||||
| @JvmName("convertFromT2") |  | ||||||
| suspend fun <T1, T2> SimpleSuspendableMapper<T1, T2>.convert(from: T2) = convertToT1(from) |  | ||||||
| @JvmName("convertFromT1") |  | ||||||
| suspend fun <T1, T2> SimpleSuspendableMapper<T1, T2>.convert(from: T1) = convertToT2(from) |  | ||||||
|  |  | ||||||
| class SimpleSuspendableMapperImpl<T1, T2>( |  | ||||||
|     private val t1: suspend (T2) -> T1, |  | ||||||
|     private val t2: suspend (T1) -> T2, |  | ||||||
| ) : SimpleSuspendableMapper<T1, T2> { |  | ||||||
|     override suspend fun convertToT1(from: T2): T1 = t1.invoke(from) |  | ||||||
|  |  | ||||||
|     override suspend fun convertToT2(from: T1): T2 = t2.invoke(from) |  | ||||||
| } |  | ||||||
|  |  | ||||||
| @Suppress("NOTHING_TO_INLINE") |  | ||||||
| inline fun <T1, T2> simpleSuspendableMapper( |  | ||||||
|     noinline t1: suspend (T2) -> T1, |  | ||||||
|     noinline t2: suspend (T1) -> T2, |  | ||||||
| ) = SimpleSuspendableMapperImpl(t1, t2) |  | ||||||
| @@ -41,18 +41,10 @@ data class Optional<T> internal constructor( | |||||||
| inline val <T> T.optional | inline val <T> T.optional | ||||||
|     get() = Optional.presented(this) |     get() = Optional.presented(this) | ||||||
|  |  | ||||||
| inline val <T : Any> T?.optionalOrAbsentIfNull |  | ||||||
|     get() = if (this == null) { |  | ||||||
|         Optional.absent<T>() |  | ||||||
|     } else { |  | ||||||
|         Optional.presented(this) |  | ||||||
|     } |  | ||||||
|  |  | ||||||
| /** | /** | ||||||
|  * Will call [block] when data presented ([Optional.dataPresented] == true) |  * Will call [block] when data presented ([Optional.dataPresented] == true) | ||||||
|  */ |  */ | ||||||
| inline fun <T> Optional<T>.onPresented(block: (T) -> Unit): Optional<T> = apply { | inline fun <T> Optional<T>.onPresented(block: (T) -> Unit): Optional<T> = apply { | ||||||
|     @OptIn(Warning::class) |  | ||||||
|     if (dataPresented) { @Suppress("UNCHECKED_CAST") block(data as T) } |     if (dataPresented) { @Suppress("UNCHECKED_CAST") block(data as T) } | ||||||
| } | } | ||||||
|  |  | ||||||
| @@ -60,7 +52,6 @@ inline fun <T> Optional<T>.onPresented(block: (T) -> Unit): Optional<T> = apply | |||||||
|  * Will call [block] when data presented ([Optional.dataPresented] == true) |  * Will call [block] when data presented ([Optional.dataPresented] == true) | ||||||
|  */ |  */ | ||||||
| inline fun <T, R> Optional<T>.mapOnPresented(block: (T) -> R): R? = run { | inline fun <T, R> Optional<T>.mapOnPresented(block: (T) -> R): R? = run { | ||||||
|     @OptIn(Warning::class) |  | ||||||
|     if (dataPresented) { @Suppress("UNCHECKED_CAST") block(data as T) } else null |     if (dataPresented) { @Suppress("UNCHECKED_CAST") block(data as T) } else null | ||||||
| } | } | ||||||
|  |  | ||||||
| @@ -68,7 +59,6 @@ inline fun <T, R> Optional<T>.mapOnPresented(block: (T) -> R): R? = run { | |||||||
|  * Will call [block] when data absent ([Optional.dataPresented] == false) |  * Will call [block] when data absent ([Optional.dataPresented] == false) | ||||||
|  */ |  */ | ||||||
| inline fun <T> Optional<T>.onAbsent(block: () -> Unit): Optional<T> = apply { | inline fun <T> Optional<T>.onAbsent(block: () -> Unit): Optional<T> = apply { | ||||||
|     @OptIn(Warning::class) |  | ||||||
|     if (!dataPresented) { block() } |     if (!dataPresented) { block() } | ||||||
| } | } | ||||||
|  |  | ||||||
| @@ -76,22 +66,27 @@ inline fun <T> Optional<T>.onAbsent(block: () -> Unit): Optional<T> = apply { | |||||||
|  * Will call [block] when data presented ([Optional.dataPresented] == true) |  * Will call [block] when data presented ([Optional.dataPresented] == true) | ||||||
|  */ |  */ | ||||||
| inline fun <T, R> Optional<T>.mapOnAbsent(block: () -> R): R? = run { | inline fun <T, R> Optional<T>.mapOnAbsent(block: () -> R): R? = run { | ||||||
|     @OptIn(Warning::class) |     if (!dataPresented) { @Suppress("UNCHECKED_CAST") block() } else null | ||||||
|     if (!dataPresented) { block() } else null |  | ||||||
| } | } | ||||||
|  |  | ||||||
| /** | /** | ||||||
|  * Returns [Optional.data] if [Optional.dataPresented] of [this] is true, or null otherwise |  * Returns [Optional.data] if [Optional.dataPresented] of [this] is true, or null otherwise | ||||||
|  */ |  */ | ||||||
| fun <T> Optional<T>.dataOrNull() = @OptIn(Warning::class) if (dataPresented) @Suppress("UNCHECKED_CAST") (data as T) else null | fun <T> Optional<T>.dataOrNull() = if (dataPresented) @Suppress("UNCHECKED_CAST") (data as T) else null | ||||||
|  |  | ||||||
| /** | /** | ||||||
|  * Returns [Optional.data] if [Optional.dataPresented] of [this] is true, or throw [throwable] otherwise |  * Returns [Optional.data] if [Optional.dataPresented] of [this] is true, or throw [throwable] otherwise | ||||||
|  */ |  */ | ||||||
| fun <T> Optional<T>.dataOrThrow(throwable: Throwable) = @OptIn(Warning::class) if (dataPresented) @Suppress("UNCHECKED_CAST") (data as T) else throw throwable | fun <T> Optional<T>.dataOrThrow(throwable: Throwable) = if (dataPresented) @Suppress("UNCHECKED_CAST") (data as T) else throw throwable | ||||||
|  |  | ||||||
|  |  | ||||||
| /** | /** | ||||||
|  * Returns [Optional.data] if [Optional.dataPresented] of [this] is true, or call [block] and returns the result of it |  * Returns [Optional.data] if [Optional.dataPresented] of [this] is true, or call [block] and returns the result of it | ||||||
|  */ |  */ | ||||||
| inline fun <T> Optional<T>.dataOrElse(block: () -> T) = @OptIn(Warning::class) if (dataPresented) @Suppress("UNCHECKED_CAST") (data as T) else block() | inline fun <T> Optional<T>.dataOrElse(block: () -> T) = if (dataPresented) @Suppress("UNCHECKED_CAST") (data as T) else block() | ||||||
|  |  | ||||||
|  | /** | ||||||
|  |  * Returns [Optional.data] if [Optional.dataPresented] of [this] is true, or call [block] and returns the result of it | ||||||
|  |  */ | ||||||
|  | @Deprecated("dataOrElse now is inline", ReplaceWith("dataOrElse", "dev.inmo.micro_utils.common.dataOrElse")) | ||||||
|  | suspend fun <T> Optional<T>.dataOrElseSuspendable(block: suspend () -> T) = if (dataPresented) @Suppress("UNCHECKED_CAST") (data as T) else block() | ||||||
|   | |||||||
| @@ -1,32 +0,0 @@ | |||||||
| package dev.inmo.micro_utils.common |  | ||||||
|  |  | ||||||
| inline fun <T> Sequence<T>.padWith(size: Int, inserter: (Sequence<T>) -> Sequence<T>): Sequence<T> { |  | ||||||
|     var result = this |  | ||||||
|     while (result.count() < size) { |  | ||||||
|         result = inserter(result) |  | ||||||
|     } |  | ||||||
|     return result |  | ||||||
| } |  | ||||||
|  |  | ||||||
| inline fun <T> Sequence<T>.padEnd(size: Int, padBlock: (Int) -> T): Sequence<T> = padWith(size) { it + padBlock(it.count()) } |  | ||||||
|  |  | ||||||
| inline fun <T> Sequence<T>.padEnd(size: Int, o: T) = padEnd(size) { o } |  | ||||||
|  |  | ||||||
| inline fun <T> List<T>.padWith(size: Int, inserter: (List<T>) -> List<T>): List<T> { |  | ||||||
|     var result = this |  | ||||||
|     while (result.size < size) { |  | ||||||
|         result = inserter(result) |  | ||||||
|     } |  | ||||||
|     return result |  | ||||||
| } |  | ||||||
| inline fun <T> List<T>.padEnd(size: Int, padBlock: (Int) -> T): List<T> = asSequence().padEnd(size, padBlock).toList() |  | ||||||
|  |  | ||||||
| inline fun <T> List<T>.padEnd(size: Int, o: T): List<T> = asSequence().padEnd(size, o).toList() |  | ||||||
|  |  | ||||||
| inline fun <T> Sequence<T>.padStart(size: Int, padBlock: (Int) -> T): Sequence<T> = padWith(size) { sequenceOf(padBlock(it.count())) + it } |  | ||||||
|  |  | ||||||
| inline fun <T> Sequence<T>.padStart(size: Int, o: T) = padStart(size) { o } |  | ||||||
|  |  | ||||||
| inline fun <T> List<T>.padStart(size: Int, padBlock: (Int) -> T): List<T> = asSequence().padStart(size, padBlock).toList() |  | ||||||
|  |  | ||||||
| inline fun <T> List<T>.padStart(size: Int, o: T): List<T> = asSequence().padStart(size, o).toList() |  | ||||||
| @@ -1,76 +0,0 @@ | |||||||
| package dev.inmo.micro_utils.common |  | ||||||
|  |  | ||||||
| import kotlinx.serialization.Serializable |  | ||||||
| import kotlin.jvm.JvmInline |  | ||||||
|  |  | ||||||
| /** |  | ||||||
|  * Contains [of1] as main value, where 100% of percentage is when of1 == 1 |  | ||||||
|  * |  | ||||||
|  * @see invoke |  | ||||||
|  * @see partOfTotal |  | ||||||
|  * @see of100 |  | ||||||
|  */ |  | ||||||
| @Serializable |  | ||||||
| @JvmInline |  | ||||||
| value class Percentage private constructor( |  | ||||||
|     /** |  | ||||||
|      * Value of percentage. When it equals to 1, means 100% |  | ||||||
|      */ |  | ||||||
|     val of1: Double |  | ||||||
| ) { |  | ||||||
|     /** |  | ||||||
|      * Same as [of1], but float (using [Double.toFloat]) |  | ||||||
|      */ |  | ||||||
|     val of1Float |  | ||||||
|         get() = of1.toFloat() |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Represent this percentage as common percentage where 100% is 100% |  | ||||||
|      */ |  | ||||||
|     val of100 |  | ||||||
|         get() = of1 * 100 |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Same as [of100], but float (using [Double.toFloat]) |  | ||||||
|      */ |  | ||||||
|     val of100Float |  | ||||||
|         get() = of100.toFloat() |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Same as [of100], but int (using [Double.toInt]) |  | ||||||
|      */ |  | ||||||
|     val of100Int |  | ||||||
|         get() = of100.toInt() |  | ||||||
|  |  | ||||||
|     companion object { |  | ||||||
|         val rangeOfValues = 0.0 .. 1.0 |  | ||||||
|  |  | ||||||
|         val START = Percentage(rangeOfValues.start) |  | ||||||
|         val COMPLETED = Percentage(rangeOfValues.endInclusive) |  | ||||||
|  |  | ||||||
|         operator fun invoke(of1: Double) = Percentage(of1.coerceIn(rangeOfValues)) |  | ||||||
|         operator fun invoke(part: Number, total: Number) = Percentage( |  | ||||||
|             part.toDouble() / total.toDouble() |  | ||||||
|         ) |  | ||||||
|         fun of1(of1: Double) = Percentage(of1 = of1) |  | ||||||
|         fun of100(of100: Double) = Percentage(of1 = of100 / 100) |  | ||||||
|         fun partOfTotal(part: Number, total: Number) = Percentage(part = part, total = total) |  | ||||||
|     } |  | ||||||
| } |  | ||||||
|  |  | ||||||
| typealias Progress = Percentage |  | ||||||
|  |  | ||||||
| /** |  | ||||||
|  * Will return [this] [Progress] if [Percentage.of1] in `0 .. 1` range |  | ||||||
|  */ |  | ||||||
| fun Progress.ensureStrictOrNull(): Progress? = if (of1 in Percentage.rangeOfValues) this else null |  | ||||||
| /** |  | ||||||
|  * Will return [this] [Progress] if [Percentage.of1] in `0 .. 1` range. Otherwise, will throw error |  | ||||||
|  * [IllegalArgumentException] due to [require] failure |  | ||||||
|  */ |  | ||||||
| fun Progress.ensureStrictOrThrow(): Progress { |  | ||||||
|     require(of1 in Percentage.rangeOfValues) { |  | ||||||
|         "For strict checks value of percentage must be in ${Percentage.rangeOfValues}, but actual value is $of1" |  | ||||||
|     } |  | ||||||
|     return this |  | ||||||
| } |  | ||||||
| @@ -1,80 +0,0 @@ | |||||||
| @file:Suppress( |  | ||||||
|   "RemoveRedundantCallsOfConversionMethods", |  | ||||||
|   "RedundantVisibilityModifier", |  | ||||||
| ) |  | ||||||
|  |  | ||||||
| package dev.inmo.micro_utils.common |  | ||||||
|  |  | ||||||
| import kotlin.Byte |  | ||||||
| import kotlin.Double |  | ||||||
| import kotlin.Float |  | ||||||
| import kotlin.Int |  | ||||||
| import kotlin.Long |  | ||||||
| import kotlin.Short |  | ||||||
| import kotlin.Suppress |  | ||||||
|  |  | ||||||
| public operator fun Percentage.plus(other: Percentage): Percentage = Percentage(of1 + other.of1) |  | ||||||
|  |  | ||||||
| public operator fun Percentage.minus(other: Percentage): Percentage = Percentage(of1 - other.of1) |  | ||||||
|  |  | ||||||
| public operator fun Percentage.plus(i: Byte): Percentage = Percentage((of1 + i).toDouble()) |  | ||||||
|  |  | ||||||
| public operator fun Percentage.minus(i: Byte): Percentage = Percentage((of1 - i).toDouble()) |  | ||||||
|  |  | ||||||
| public operator fun Percentage.times(i: Byte): Percentage = Percentage((of1 * i).toDouble()) |  | ||||||
|  |  | ||||||
| public operator fun Percentage.div(i: Byte): Percentage = Percentage((of1 / i).toDouble()) |  | ||||||
|  |  | ||||||
| public operator fun Percentage.rem(i: Byte): Percentage = Percentage((of1 % i).toDouble()) |  | ||||||
|  |  | ||||||
| public operator fun Percentage.plus(i: Short): Percentage = Percentage((of1 + i).toDouble()) |  | ||||||
|  |  | ||||||
| public operator fun Percentage.minus(i: Short): Percentage = Percentage((of1 - i).toDouble()) |  | ||||||
|  |  | ||||||
| public operator fun Percentage.times(i: Short): Percentage = Percentage((of1 * i).toDouble()) |  | ||||||
|  |  | ||||||
| public operator fun Percentage.div(i: Short): Percentage = Percentage((of1 / i).toDouble()) |  | ||||||
|  |  | ||||||
| public operator fun Percentage.rem(i: Short): Percentage = Percentage((of1 % i).toDouble()) |  | ||||||
|  |  | ||||||
| public operator fun Percentage.plus(i: Int): Percentage = Percentage((of1 + i).toDouble()) |  | ||||||
|  |  | ||||||
| public operator fun Percentage.minus(i: Int): Percentage = Percentage((of1 - i).toDouble()) |  | ||||||
|  |  | ||||||
| public operator fun Percentage.times(i: Int): Percentage = Percentage((of1 * i).toDouble()) |  | ||||||
|  |  | ||||||
| public operator fun Percentage.div(i: Int): Percentage = Percentage((of1 / i).toDouble()) |  | ||||||
|  |  | ||||||
| public operator fun Percentage.rem(i: Int): Percentage = Percentage((of1 % i).toDouble()) |  | ||||||
|  |  | ||||||
| public operator fun Percentage.plus(i: Long): Percentage = Percentage((of1 + i).toDouble()) |  | ||||||
|  |  | ||||||
| public operator fun Percentage.minus(i: Long): Percentage = Percentage((of1 - i).toDouble()) |  | ||||||
|  |  | ||||||
| public operator fun Percentage.times(i: Long): Percentage = Percentage((of1 * i).toDouble()) |  | ||||||
|  |  | ||||||
| public operator fun Percentage.div(i: Long): Percentage = Percentage((of1 / i).toDouble()) |  | ||||||
|  |  | ||||||
| public operator fun Percentage.rem(i: Long): Percentage = Percentage((of1 % i).toDouble()) |  | ||||||
|  |  | ||||||
| public operator fun Percentage.plus(i: Float): Percentage = Percentage((of1 + i).toDouble()) |  | ||||||
|  |  | ||||||
| public operator fun Percentage.minus(i: Float): Percentage = Percentage((of1 - i).toDouble()) |  | ||||||
|  |  | ||||||
| public operator fun Percentage.times(i: Float): Percentage = Percentage((of1 * i).toDouble()) |  | ||||||
|  |  | ||||||
| public operator fun Percentage.div(i: Float): Percentage = Percentage((of1 / i).toDouble()) |  | ||||||
|  |  | ||||||
| public operator fun Percentage.rem(i: Float): Percentage = Percentage((of1 % i).toDouble()) |  | ||||||
|  |  | ||||||
| public operator fun Percentage.plus(i: Double): Percentage = Percentage((of1 + i).toDouble()) |  | ||||||
|  |  | ||||||
| public operator fun Percentage.minus(i: Double): Percentage = Percentage((of1 - i).toDouble()) |  | ||||||
|  |  | ||||||
| public operator fun Percentage.times(i: Double): Percentage = Percentage((of1 * i).toDouble()) |  | ||||||
|  |  | ||||||
| public operator fun Percentage.div(i: Double): Percentage = Percentage((of1 / i).toDouble()) |  | ||||||
|  |  | ||||||
| public operator fun Percentage.rem(i: Double): Percentage = Percentage((of1 % i).toDouble()) |  | ||||||
|  |  | ||||||
| public operator fun Percentage.compareTo(other: Percentage): Int = (of1.compareTo(other.of1)) |  | ||||||
| @@ -1,28 +0,0 @@ | |||||||
| package dev.inmo.micro_utils.common |  | ||||||
|  |  | ||||||
| /** |  | ||||||
|  * Convert [this] [Long] to [Int] with bounds of [Int.MIN_VALUE] and [Int.MAX_VALUE] |  | ||||||
|  */ |  | ||||||
| fun Long.toCoercedInt(): Int = coerceIn(Int.MIN_VALUE.toLong(), Int.MAX_VALUE.toLong()).toInt() |  | ||||||
| /** |  | ||||||
|  * Convert [this] [Long] to [Short] with bounds of [Short.MIN_VALUE] and [Short.MAX_VALUE] |  | ||||||
|  */ |  | ||||||
| fun Long.toCoercedShort(): Short = coerceIn(Short.MIN_VALUE.toLong(), Short.MAX_VALUE.toLong()).toShort() |  | ||||||
| /** |  | ||||||
|  * Convert [this] [Long] to [Byte] with bounds of [Byte.MIN_VALUE] and [Byte.MAX_VALUE] |  | ||||||
|  */ |  | ||||||
| fun Long.toCoercedByte(): Byte = coerceIn(Byte.MIN_VALUE.toLong(), Byte.MAX_VALUE.toLong()).toByte() |  | ||||||
|  |  | ||||||
| /** |  | ||||||
|  * Convert [this] [Int] to [Short] with bounds of [Short.MIN_VALUE] and [Short.MAX_VALUE] |  | ||||||
|  */ |  | ||||||
| fun Int.toCoercedShort(): Short = coerceIn(Short.MIN_VALUE.toInt(), Short.MAX_VALUE.toInt()).toShort() |  | ||||||
| /** |  | ||||||
|  * Convert [this] [Int] to [Byte] with bounds of [Byte.MIN_VALUE] and [Byte.MAX_VALUE] |  | ||||||
|  */ |  | ||||||
| fun Int.toCoercedByte(): Byte = coerceIn(Byte.MIN_VALUE.toInt(), Byte.MAX_VALUE.toInt()).toByte() |  | ||||||
|  |  | ||||||
| /** |  | ||||||
|  * Convert [this] [Short] to [Byte] with bounds of [Byte.MIN_VALUE] and [Byte.MAX_VALUE] |  | ||||||
|  */ |  | ||||||
| fun Short.toCoercedByte(): Byte = coerceIn(Byte.MIN_VALUE.toShort(), Byte.MAX_VALUE.toShort()).toByte() |  | ||||||
| @@ -1,27 +1,5 @@ | |||||||
| package dev.inmo.micro_utils.common | package dev.inmo.micro_utils.common | ||||||
|  |  | ||||||
| /** |  | ||||||
|  * Executes the given [action] until getting of successful result specified number of [times]. |  | ||||||
|  * |  | ||||||
|  * A zero-based index of current iteration is passed as a parameter to [action]. |  | ||||||
|  */ |  | ||||||
| inline fun <R> repeatOnFailure( |  | ||||||
|     onFailure: (Throwable) -> Boolean, |  | ||||||
|     action: () -> R |  | ||||||
| ): Result<R> { |  | ||||||
|     do { |  | ||||||
|         runCatching { |  | ||||||
|             action() |  | ||||||
|         }.onFailure { |  | ||||||
|             if (!onFailure(it)) { |  | ||||||
|                 return Result.failure(it) |  | ||||||
|             } |  | ||||||
|         }.onSuccess { |  | ||||||
|             return Result.success(it) |  | ||||||
|         } |  | ||||||
|     } while (true) |  | ||||||
| } |  | ||||||
|  |  | ||||||
| /** | /** | ||||||
|  * Executes the given [action] until getting of successful result specified number of [times]. |  * Executes the given [action] until getting of successful result specified number of [times]. | ||||||
|  * |  * | ||||||
| @@ -32,23 +10,12 @@ inline fun <R> repeatOnFailure( | |||||||
|     onEachFailure: (Throwable) -> Unit = {}, |     onEachFailure: (Throwable) -> Unit = {}, | ||||||
|     action: (Int) -> R |     action: (Int) -> R | ||||||
| ): Optional<R> { | ): Optional<R> { | ||||||
|     var i = 0 |     repeat(times) { | ||||||
|     val result = repeatOnFailure( |         runCatching { | ||||||
|         { |             action(it) | ||||||
|             onEachFailure(it) |         }.onFailure(onEachFailure).onSuccess { | ||||||
|             if (i < times) { |             return Optional.presented(it) | ||||||
|                 i++ |  | ||||||
|                 true |  | ||||||
|             } else { |  | ||||||
|                 false |  | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
|     ) { |     return Optional.absent() | ||||||
|         action(i) |  | ||||||
|     } |  | ||||||
|     return if (result.isSuccess) { |  | ||||||
|         Optional.presented(result.getOrThrow()) |  | ||||||
|     } else { |  | ||||||
|         Optional.absent() |  | ||||||
|     } |  | ||||||
| } | } | ||||||
|   | |||||||
| @@ -1,26 +0,0 @@ | |||||||
| package dev.inmo.micro_utils.common |  | ||||||
|  |  | ||||||
| /** |  | ||||||
|  * Will try to execute [action] and, if any exception will happen, execution will be retried. |  | ||||||
|  * This process will happen at most [count] times. There is no any limits on [count] value, but [action] will run at |  | ||||||
|  * least once and [retryOnFailure] will return its result if it is successful |  | ||||||
|  */ |  | ||||||
| inline fun <T> retryOnFailure(count: Int, action: () -> T): T { |  | ||||||
|     var triesCount = 0 |  | ||||||
|     while (true) { |  | ||||||
|         val result = runCatching { |  | ||||||
|             action() |  | ||||||
|         }.onFailure { |  | ||||||
|             triesCount++ |  | ||||||
|  |  | ||||||
|             if (triesCount >= count) { |  | ||||||
|                 throw it |  | ||||||
|             } else { |  | ||||||
|                 null |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|         if (result.isSuccess) return result.getOrThrow() |  | ||||||
|     } |  | ||||||
|     error("Unreachable code: retry must throw latest exception if error happen or success value if not") |  | ||||||
| } |  | ||||||
| @@ -1,6 +0,0 @@ | |||||||
| package dev.inmo.micro_utils.common |  | ||||||
|  |  | ||||||
| val FixedSignsRange = 0 .. 100 |  | ||||||
|  |  | ||||||
| expect fun Float.fixed(signs: Int): Float |  | ||||||
| expect fun Double.fixed(signs: Int): Double |  | ||||||
| @@ -1,5 +0,0 @@ | |||||||
| package dev.inmo.micro_utils.common |  | ||||||
|  |  | ||||||
| fun <T> Iterable<T>.withReplacedAt(i: Int, block: (T) -> T): List<T> = take(i) + block(elementAt(i)) + drop(i + 1) |  | ||||||
| fun <T> Iterable<T>.withReplaced(t: T, block: (T) -> T): List<T> = withReplacedAt(indexOf(t), block) |  | ||||||
|  |  | ||||||
| @@ -32,7 +32,7 @@ class DiffUtilsTests { | |||||||
|         val withIndex = oldList.withIndex() |         val withIndex = oldList.withIndex() | ||||||
|  |  | ||||||
|         for (count in 1 .. (floor(oldList.size.toFloat() / 2).toInt())) { |         for (count in 1 .. (floor(oldList.size.toFloat() / 2).toInt())) { | ||||||
|             for ((i, _) in withIndex) { |             for ((i, v) in withIndex) { | ||||||
|                 if (i + count > oldList.lastIndex) { |                 if (i + count > oldList.lastIndex) { | ||||||
|                     continue |                     continue | ||||||
|                 } |                 } | ||||||
| @@ -55,7 +55,7 @@ class DiffUtilsTests { | |||||||
|         val withIndex = oldList.withIndex() |         val withIndex = oldList.withIndex() | ||||||
|  |  | ||||||
|         for (step in oldList.indices) { |         for (step in oldList.indices) { | ||||||
|             for ((i, _) in withIndex) { |             for ((i, v) in withIndex) { | ||||||
|                 val mutable = oldList.toMutableList() |                 val mutable = oldList.toMutableList() | ||||||
|                 val changes = ( |                 val changes = ( | ||||||
|                     if (step == 0) i until oldList.size else (i until oldList.size step step) |                     if (step == 0) i until oldList.size else (i until oldList.size step step) | ||||||
| @@ -104,7 +104,7 @@ class DiffUtilsTests { | |||||||
|         val withIndex = oldList.withIndex() |         val withIndex = oldList.withIndex() | ||||||
|  |  | ||||||
|         for (count in 1 .. (floor(oldList.size.toFloat() / 2).toInt())) { |         for (count in 1 .. (floor(oldList.size.toFloat() / 2).toInt())) { | ||||||
|             for ((i, _) in withIndex) { |             for ((i, v) in withIndex) { | ||||||
|                 if (i + count > oldList.lastIndex) { |                 if (i + count > oldList.lastIndex) { | ||||||
|                     continue |                     continue | ||||||
|                 } |                 } | ||||||
| @@ -129,20 +129,15 @@ class DiffUtilsTests { | |||||||
|         val withIndex = oldList.withIndex() |         val withIndex = oldList.withIndex() | ||||||
|  |  | ||||||
|         for (step in oldList.indices) { |         for (step in oldList.indices) { | ||||||
|             for ((i, _) in withIndex) { |             for ((i, v) in withIndex) { | ||||||
|                 val mutable = oldList.toMutableList() |                 val mutable = oldList.toMutableList() | ||||||
|  |                 val changes = ( | ||||||
|                 val newList = if (step == 0) { |                     if (step == 0) i until oldList.size else (i until oldList.size step step) | ||||||
|                     i until oldList.size |                 ).map { index -> | ||||||
|                 } else { |  | ||||||
|                     i until oldList.size step step |  | ||||||
|                 } |  | ||||||
|                 newList.forEach { index -> |  | ||||||
|                     IndexedValue(index, mutable[index]) to IndexedValue(index, "changed$index").also { |                     IndexedValue(index, mutable[index]) to IndexedValue(index, "changed$index").also { | ||||||
|                         mutable[index] = it.value |                         mutable[index] = it.value | ||||||
|                     } |                     } | ||||||
|                 } |                 } | ||||||
|  |  | ||||||
|                 val mutableOldList = oldList.toMutableList() |                 val mutableOldList = oldList.toMutableList() | ||||||
|                 mutableOldList.applyDiff(mutable) |                 mutableOldList.applyDiff(mutable) | ||||||
|                 assertEquals( |                 assertEquals( | ||||||
|   | |||||||
| @@ -1,29 +0,0 @@ | |||||||
| package dev.inmo.micro_utils.common |  | ||||||
|  |  | ||||||
| import kotlin.test.Test |  | ||||||
| import kotlin.test.assertEquals |  | ||||||
|  |  | ||||||
| class PercentageTests { |  | ||||||
|     @Test |  | ||||||
|     fun testCompareTo() { |  | ||||||
|         val step = 0.01 |  | ||||||
|  |  | ||||||
|         var i = Percentage.START.of1 |  | ||||||
|         while (i <= Percentage.COMPLETED.of1) { |  | ||||||
|             val percentageI = Percentage(i) |  | ||||||
|  |  | ||||||
|             var j = Percentage.START.of1 |  | ||||||
|             while (j <= Percentage.COMPLETED.of1) { |  | ||||||
|                 val percentageJ = Percentage(j) |  | ||||||
|  |  | ||||||
|                 assertEquals(percentageI.of1.compareTo(percentageJ.of1), percentageI.compareTo(percentageJ)) |  | ||||||
|                 assertEquals(percentageI.of1 > percentageJ.of1, percentageI > percentageJ) |  | ||||||
|                 assertEquals(percentageI.of1 < percentageJ.of1, percentageI < percentageJ) |  | ||||||
|  |  | ||||||
|                 j += step |  | ||||||
|             } |  | ||||||
|  |  | ||||||
|             i += step |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| } |  | ||||||
| @@ -1,21 +0,0 @@ | |||||||
| package dev.inmo.micro_utils.common |  | ||||||
|  |  | ||||||
| import kotlin.test.Test |  | ||||||
| import kotlin.test.assertEquals |  | ||||||
|  |  | ||||||
| class WithReplacedTest { |  | ||||||
|     @Test |  | ||||||
|     fun testReplaced() { |  | ||||||
|         val data = 0 until 10 |  | ||||||
|         val testData = Int.MAX_VALUE |  | ||||||
|  |  | ||||||
|         for (i in 0 until data.last) { |  | ||||||
|             val withReplaced = data.withReplacedAt(i) { |  | ||||||
|                 testData |  | ||||||
|             } |  | ||||||
|             val dataAsMutableList = data.toMutableList() |  | ||||||
|             dataAsMutableList[i] = testData |  | ||||||
|             assertEquals(withReplaced, dataAsMutableList.toList()) |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| } |  | ||||||
| @@ -1,13 +0,0 @@ | |||||||
| package dev.inmo.micro_utils.common |  | ||||||
|  |  | ||||||
| import kotlinx.browser.window |  | ||||||
|  |  | ||||||
| fun copyToClipboard(text: String): Boolean { |  | ||||||
|     return runCatching { |  | ||||||
|         window.navigator.clipboard.writeText( |  | ||||||
|             text |  | ||||||
|         ) |  | ||||||
|     }.onFailure { |  | ||||||
|         it.printStackTrace() |  | ||||||
|     }.isSuccess |  | ||||||
| } |  | ||||||
| @@ -1,29 +0,0 @@ | |||||||
| package dev.inmo.micro_utils.common |  | ||||||
|  |  | ||||||
| import kotlinx.browser.window |  | ||||||
| import org.w3c.files.Blob |  | ||||||
| import org.w3c.files.BlobPropertyBag |  | ||||||
| import kotlin.js.json |  | ||||||
|  |  | ||||||
| external class ClipboardItem(data: dynamic) |  | ||||||
|  |  | ||||||
| inline fun Blob.convertToClipboardItem(): ClipboardItem { |  | ||||||
|     val itemData: dynamic = json(this.type to this) |  | ||||||
|     return ClipboardItem(itemData) |  | ||||||
| } |  | ||||||
|  |  | ||||||
| suspend fun copyImageURLToClipboard(imageUrl: String): Boolean { |  | ||||||
|     return runCatching { |  | ||||||
|         val response = window.fetch(imageUrl).await() |  | ||||||
|         val blob = response.blob().await() |  | ||||||
|         val data = arrayOf( |  | ||||||
|             Blob( |  | ||||||
|                 arrayOf(blob), |  | ||||||
|                 BlobPropertyBag("image/png") |  | ||||||
|             ).convertToClipboardItem() |  | ||||||
|         ).asDynamic() |  | ||||||
|         window.navigator.clipboard.write(data) |  | ||||||
|     }.onFailure { |  | ||||||
|         it.printStackTrace() |  | ||||||
|     }.isSuccess |  | ||||||
| } |  | ||||||
| @@ -1,12 +0,0 @@ | |||||||
| package dev.inmo.micro_utils.common |  | ||||||
|  |  | ||||||
| import org.w3c.dom.Element |  | ||||||
|  |  | ||||||
| inline val Element.isOverflowWidth |  | ||||||
|     get() = scrollWidth > clientWidth |  | ||||||
|  |  | ||||||
| inline val Element.isOverflowHeight |  | ||||||
|     get() = scrollHeight > clientHeight |  | ||||||
|  |  | ||||||
| inline val Element.isOverflow |  | ||||||
|     get() = isOverflowHeight || isOverflowWidth |  | ||||||
| @@ -35,10 +35,6 @@ private suspend fun MPPFile.dirtyReadBytes(): ByteArray = readBytesPromise().awa | |||||||
|  */ |  */ | ||||||
| actual val MPPFile.filename: FileName | actual val MPPFile.filename: FileName | ||||||
|     get() = FileName(name) |     get() = FileName(name) | ||||||
|  |  | ||||||
| actual val MPPFilePathSeparator: Char |  | ||||||
|     get() = '/' |  | ||||||
|  |  | ||||||
| /** | /** | ||||||
|  * @suppress |  * @suppress | ||||||
|  */ |  */ | ||||||
|   | |||||||
| @@ -1,58 +0,0 @@ | |||||||
| package dev.inmo.micro_utils.common |  | ||||||
|  |  | ||||||
| import org.w3c.dom.* |  | ||||||
| import kotlin.js.Json |  | ||||||
| import kotlin.js.json |  | ||||||
|  |  | ||||||
| external class ResizeObserver( |  | ||||||
|     callback: (Array<ResizeObserverEntry>, ResizeObserver) -> Unit |  | ||||||
| ) { |  | ||||||
|     fun observe(target: Element, options: Json = definedExternally) |  | ||||||
|  |  | ||||||
|     fun unobserve(target: Element) |  | ||||||
|  |  | ||||||
|     fun disconnect() |  | ||||||
| } |  | ||||||
|  |  | ||||||
| external interface ResizeObserverSize { |  | ||||||
|     val blockSize: Float |  | ||||||
|     val inlineSize: Float |  | ||||||
| } |  | ||||||
|  |  | ||||||
| external interface ResizeObserverEntry { |  | ||||||
|     val borderBoxSize: Array<ResizeObserverSize> |  | ||||||
|     val contentBoxSize: Array<ResizeObserverSize> |  | ||||||
|     val devicePixelContentBoxSize: Array<ResizeObserverSize> |  | ||||||
|     val contentRect: DOMRectReadOnly |  | ||||||
|     val target: Element |  | ||||||
| } |  | ||||||
|  |  | ||||||
| fun ResizeObserver.observe(target: Element, options: ResizeObserverObserveOptions) = observe( |  | ||||||
|     target, |  | ||||||
|     json( |  | ||||||
|         "box" to options.box ?.name |  | ||||||
|     ) |  | ||||||
| ) |  | ||||||
|  |  | ||||||
| class ResizeObserverObserveOptions( |  | ||||||
|     val box: Box? = null |  | ||||||
| ) { |  | ||||||
|     sealed interface Box { |  | ||||||
|         val name: String |  | ||||||
|  |  | ||||||
|         object Content : Box { |  | ||||||
|             override val name: String |  | ||||||
|                 get() = "content-box" |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|         object Border : Box { |  | ||||||
|             override val name: String |  | ||||||
|                 get() = "border-box" |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|         object DevicePixelContent : Box { |  | ||||||
|             override val name: String |  | ||||||
|                 get() = "device-pixel-content-box" |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| } |  | ||||||
| @@ -1,4 +0,0 @@ | |||||||
| package dev.inmo.micro_utils.common |  | ||||||
|  |  | ||||||
| actual fun Float.fixed(signs: Int): Float = this.asDynamic().toFixed(signs.coerceIn(FixedSignsRange)).unsafeCast<String>().toFloat() |  | ||||||
| actual fun Double.fixed(signs: Int): Double = this.asDynamic().toFixed(signs.coerceIn(FixedSignsRange)).unsafeCast<String>().toDouble() |  | ||||||
| @@ -1,20 +0,0 @@ | |||||||
| package dev.inmo.micro_utils.common |  | ||||||
|  |  | ||||||
| import java.io.File |  | ||||||
| import java.io.InputStream |  | ||||||
| import java.util.UUID |  | ||||||
|  |  | ||||||
| fun InputStream.downloadToTempFile( |  | ||||||
|     fileName: String = UUID.randomUUID().toString(), |  | ||||||
|     fileExtension: String? = ".temp", |  | ||||||
|     folder: File? = null |  | ||||||
| ) = File.createTempFile( |  | ||||||
|     fileName, |  | ||||||
|     fileExtension, |  | ||||||
|     folder |  | ||||||
| ).apply { |  | ||||||
|     outputStream().use { |  | ||||||
|         copyTo(it) |  | ||||||
|     } |  | ||||||
|     deleteOnExit() |  | ||||||
| } |  | ||||||
| @@ -14,10 +14,6 @@ actual typealias MPPFile = File | |||||||
|  */ |  */ | ||||||
| actual val MPPFile.filename: FileName | actual val MPPFile.filename: FileName | ||||||
|     get() = FileName(name) |     get() = FileName(name) | ||||||
|  |  | ||||||
| actual val MPPFilePathSeparator: Char |  | ||||||
|     get() = File.separatorChar |  | ||||||
|  |  | ||||||
| /** | /** | ||||||
|  * @suppress |  * @suppress | ||||||
|  */ |  */ | ||||||
|   | |||||||
| @@ -1,12 +0,0 @@ | |||||||
| package dev.inmo.micro_utils.common |  | ||||||
|  |  | ||||||
| import java.math.BigDecimal |  | ||||||
| import java.math.RoundingMode |  | ||||||
|  |  | ||||||
| actual fun Float.fixed(signs: Int): Float = BigDecimal.valueOf(this.toDouble()) |  | ||||||
|     .setScale(signs.coerceIn(FixedSignsRange), RoundingMode.HALF_UP) |  | ||||||
|     .toFloat(); |  | ||||||
|  |  | ||||||
| actual fun Double.fixed(signs: Int): Double = BigDecimal.valueOf(this) |  | ||||||
|     .setScale(signs.coerceIn(FixedSignsRange), RoundingMode.HALF_UP) |  | ||||||
|     .toDouble(); |  | ||||||
							
								
								
									
										1
									
								
								common/src/main/AndroidManifest.xml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										1
									
								
								common/src/main/AndroidManifest.xml
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1 @@ | |||||||
|  | <manifest package="dev.inmo.micro_utils.common"/> | ||||||
| @@ -1,40 +0,0 @@ | |||||||
| package dev.inmo.micro_utils.common |  | ||||||
|  |  | ||||||
| import okio.FileSystem |  | ||||||
| import okio.Path |  | ||||||
| import okio.use |  | ||||||
|  |  | ||||||
| actual typealias MPPFile = Path |  | ||||||
|  |  | ||||||
| /** |  | ||||||
|  * @suppress |  | ||||||
|  */ |  | ||||||
| actual val MPPFile.filename: FileName |  | ||||||
|     get() = FileName(toString()) |  | ||||||
|  |  | ||||||
|  |  | ||||||
| actual val MPPFilePathSeparator: Char = Path.DIRECTORY_SEPARATOR.first() |  | ||||||
|  |  | ||||||
| /** |  | ||||||
|  * @suppress |  | ||||||
|  */ |  | ||||||
| actual val MPPFile.filesize: Long |  | ||||||
|     get() = FileSystem.SYSTEM.openReadOnly(this).use { |  | ||||||
|         it.size() |  | ||||||
|     } |  | ||||||
| /** |  | ||||||
|  * @suppress |  | ||||||
|  */ |  | ||||||
| actual val MPPFile.bytesAllocatorSync: ByteArrayAllocator |  | ||||||
|     get() = { |  | ||||||
|         FileSystem.SYSTEM.read(this) { |  | ||||||
|             readByteArray() |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| /** |  | ||||||
|  * @suppress |  | ||||||
|  */ |  | ||||||
| actual val MPPFile.bytesAllocator: SuspendByteArrayAllocator |  | ||||||
|     get() = { |  | ||||||
|         bytesAllocatorSync() |  | ||||||
|     } |  | ||||||
| @@ -1,25 +0,0 @@ | |||||||
| package dev.inmo.micro_utils.common |  | ||||||
|  |  | ||||||
| import kotlinx.cinterop.* |  | ||||||
| import platform.posix.snprintf |  | ||||||
| import platform.posix.sprintf |  | ||||||
|  |  | ||||||
| @OptIn(ExperimentalForeignApi::class) |  | ||||||
| actual fun Float.fixed(signs: Int): Float { |  | ||||||
|     return memScoped { |  | ||||||
|         val buff = allocArray<ByteVar>(Float.SIZE_BYTES * 2) |  | ||||||
|  |  | ||||||
|         sprintf(buff, "%.${signs}f", this@fixed) |  | ||||||
|         buff.toKString().toFloat() |  | ||||||
|     } |  | ||||||
| } |  | ||||||
|  |  | ||||||
| @OptIn(ExperimentalForeignApi::class) |  | ||||||
| actual fun Double.fixed(signs: Int): Double { |  | ||||||
|     return memScoped { |  | ||||||
|         val buff = allocArray<ByteVar>(Double.SIZE_BYTES * 2) |  | ||||||
|  |  | ||||||
|         sprintf(buff, "%.${signs}f", this@fixed) |  | ||||||
|         buff.toKString().toDouble() |  | ||||||
|     } |  | ||||||
| } |  | ||||||
| @@ -1,15 +0,0 @@ | |||||||
| package dev.inmo.micro_utils.common |  | ||||||
|  |  | ||||||
| import org.khronos.webgl.* |  | ||||||
|  |  | ||||||
| fun DataView.toByteArray() = ByteArray(this.byteLength) { |  | ||||||
|     getInt8(it) |  | ||||||
| } |  | ||||||
|  |  | ||||||
| fun ArrayBuffer.toByteArray() = Int8Array(this).toByteArray() |  | ||||||
|  |  | ||||||
| fun ByteArray.toDataView() = DataView(ArrayBuffer(size)).also { |  | ||||||
|     forEachIndexed { i, byte -> it.setInt8(i, byte) } |  | ||||||
| } |  | ||||||
|  |  | ||||||
| fun ByteArray.toArrayBuffer() = toDataView().buffer |  | ||||||
| @@ -1,13 +0,0 @@ | |||||||
| package dev.inmo.micro_utils.common |  | ||||||
|  |  | ||||||
| import kotlinx.browser.window |  | ||||||
|  |  | ||||||
| fun copyToClipboard(text: String): Boolean { |  | ||||||
|     return runCatching { |  | ||||||
|         window.navigator.clipboard.writeText( |  | ||||||
|             text |  | ||||||
|         ) |  | ||||||
|     }.onFailure { |  | ||||||
|         it.printStackTrace() |  | ||||||
|     }.isSuccess |  | ||||||
| } |  | ||||||
| @@ -1,31 +0,0 @@ | |||||||
| package dev.inmo.micro_utils.common |  | ||||||
|  |  | ||||||
| import kotlinx.browser.window |  | ||||||
| import kotlinx.coroutines.await |  | ||||||
| import org.w3c.fetch.Response |  | ||||||
| import org.w3c.files.Blob |  | ||||||
| import org.w3c.files.BlobPropertyBag |  | ||||||
|  |  | ||||||
| external class ClipboardItem(data: JsAny?) : JsAny |  | ||||||
|  |  | ||||||
| fun createBlobData(blob: Blob): JsAny = js("""({[blob.type]: blob})""") |  | ||||||
|  |  | ||||||
| inline fun Blob.convertToClipboardItem(): ClipboardItem { |  | ||||||
|     return ClipboardItem(createBlobData(this)) |  | ||||||
| } |  | ||||||
|  |  | ||||||
| suspend fun copyImageURLToClipboard(imageUrl: String): Boolean { |  | ||||||
|     return runCatching { |  | ||||||
|         val response = window.fetch(imageUrl).await<Response>() |  | ||||||
|         val blob = response.blob().await<Blob>() |  | ||||||
|         val data = arrayOf( |  | ||||||
|             Blob( |  | ||||||
|                 arrayOf(blob).toJsArray().unsafeCast(), |  | ||||||
|                 BlobPropertyBag("image/png") |  | ||||||
|             ).convertToClipboardItem() |  | ||||||
|         ).toJsArray() |  | ||||||
|         window.navigator.clipboard.write(data.unsafeCast()) |  | ||||||
|     }.onFailure { |  | ||||||
|         it.printStackTrace() |  | ||||||
|     }.isSuccess |  | ||||||
| } |  | ||||||
| @@ -1,63 +0,0 @@ | |||||||
| package dev.inmo.micro_utils.common |  | ||||||
|  |  | ||||||
| import kotlinx.browser.document |  | ||||||
| import org.w3c.dom.* |  | ||||||
|  |  | ||||||
| private fun createMutationObserverInit(childList: Boolean, subtree: Boolean): JsAny = js("({childList, subtree})") |  | ||||||
|  |  | ||||||
| fun Node.onRemoved(block: () -> Unit): MutationObserver { |  | ||||||
|     lateinit var observer: MutationObserver |  | ||||||
|  |  | ||||||
|     observer = MutationObserver { _, _ -> |  | ||||||
|         fun checkIfRemoved(node: Node): Boolean { |  | ||||||
|             return node.parentNode != document && (node.parentNode ?.let { checkIfRemoved(it) } ?: true) |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|         if (checkIfRemoved(this)) { |  | ||||||
|             observer.disconnect() |  | ||||||
|             block() |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     observer.observe(document, createMutationObserverInit(childList = true, subtree = true).unsafeCast()) |  | ||||||
|     return observer |  | ||||||
| } |  | ||||||
|  |  | ||||||
| fun Element.onVisibilityChanged(block: IntersectionObserverEntry.(Float, IntersectionObserver) -> Unit): IntersectionObserver { |  | ||||||
|     var previousIntersectionRatio = -1f |  | ||||||
|     val observer = IntersectionObserver { entries, observer -> |  | ||||||
|         entries.toArray().forEach { |  | ||||||
|             if (previousIntersectionRatio.toDouble() != it.intersectionRatio.toDouble()) { |  | ||||||
|                 previousIntersectionRatio = it.intersectionRatio.toDouble().toFloat() |  | ||||||
|                 it.block(previousIntersectionRatio, observer) |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     observer.observe(this) |  | ||||||
|     return observer |  | ||||||
| } |  | ||||||
|  |  | ||||||
| fun Element.onVisible(block: Element.(IntersectionObserver) -> Unit) { |  | ||||||
|     var previous = -1f |  | ||||||
|     onVisibilityChanged { intersectionRatio, observer -> |  | ||||||
|         if (previous != intersectionRatio) { |  | ||||||
|             if (intersectionRatio > 0 && previous == 0f) { |  | ||||||
|                 block(observer) |  | ||||||
|             } |  | ||||||
|             previous = intersectionRatio |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| } |  | ||||||
|  |  | ||||||
| fun Element.onInvisible(block: Element.(IntersectionObserver) -> Unit): IntersectionObserver { |  | ||||||
|     var previous = -1f |  | ||||||
|     return onVisibilityChanged { intersectionRatio, observer -> |  | ||||||
|         if (previous != intersectionRatio) { |  | ||||||
|             if (intersectionRatio == 0f && previous != 0f) { |  | ||||||
|                 block(observer) |  | ||||||
|             } |  | ||||||
|             previous = intersectionRatio |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| } |  | ||||||
| @@ -1,43 +0,0 @@ | |||||||
| package dev.inmo.micro_utils.common |  | ||||||
|  |  | ||||||
| import kotlinx.browser.window |  | ||||||
| import org.w3c.dom.DOMRect |  | ||||||
| import org.w3c.dom.Element |  | ||||||
|  |  | ||||||
| val DOMRect.isOnScreenByLeftEdge: Boolean |  | ||||||
|     get() = left >= 0 && left <= window.innerWidth |  | ||||||
| inline val Element.isOnScreenByLeftEdge |  | ||||||
|     get() = getBoundingClientRect().isOnScreenByLeftEdge |  | ||||||
|  |  | ||||||
| val DOMRect.isOnScreenByRightEdge: Boolean |  | ||||||
|     get() = right >= 0 && right <= window.innerWidth |  | ||||||
| inline val Element.isOnScreenByRightEdge |  | ||||||
|     get() = getBoundingClientRect().isOnScreenByRightEdge |  | ||||||
|  |  | ||||||
| internal val DOMRect.isOnScreenHorizontally: Boolean |  | ||||||
|     get() = isOnScreenByLeftEdge || isOnScreenByRightEdge |  | ||||||
|  |  | ||||||
|  |  | ||||||
| val DOMRect.isOnScreenByTopEdge: Boolean |  | ||||||
|     get() = top >= 0 && top <= window.innerHeight |  | ||||||
| inline val Element.isOnScreenByTopEdge |  | ||||||
|     get() = getBoundingClientRect().isOnScreenByTopEdge |  | ||||||
|  |  | ||||||
| val DOMRect.isOnScreenByBottomEdge: Boolean |  | ||||||
|     get() = bottom >= 0 && bottom <= window.innerHeight |  | ||||||
| inline val Element.isOnScreenByBottomEdge |  | ||||||
|     get() = getBoundingClientRect().isOnScreenByBottomEdge |  | ||||||
|  |  | ||||||
| internal val DOMRect.isOnScreenVertically: Boolean |  | ||||||
|     get() = isOnScreenByLeftEdge || isOnScreenByRightEdge |  | ||||||
|  |  | ||||||
|  |  | ||||||
| val DOMRect.isOnScreenFully: Boolean |  | ||||||
|     get() = isOnScreenByLeftEdge && isOnScreenByTopEdge && isOnScreenByRightEdge && isOnScreenByBottomEdge |  | ||||||
| val Element.isOnScreenFully: Boolean |  | ||||||
|     get() = getBoundingClientRect().isOnScreenFully |  | ||||||
|  |  | ||||||
| val DOMRect.isOnScreen: Boolean |  | ||||||
|     get() = isOnScreenFully || (isOnScreenHorizontally && isOnScreenVertically) |  | ||||||
| inline val Element.isOnScreen: Boolean |  | ||||||
|     get() = getBoundingClientRect().isOnScreen |  | ||||||
| @@ -1,127 +0,0 @@ | |||||||
| package dev.inmo.micro_utils.common |  | ||||||
|  |  | ||||||
| import org.w3c.dom.DOMRectReadOnly |  | ||||||
| import org.w3c.dom.Element |  | ||||||
|  |  | ||||||
| external interface IntersectionObserverOptions: JsAny { |  | ||||||
|     /** |  | ||||||
|      * An Element or Document object which is an ancestor of the intended target, whose bounding rectangle will be |  | ||||||
|      * considered the viewport. Any part of the target not visible in the visible area of the root is not considered |  | ||||||
|      * visible. |  | ||||||
|      */ |  | ||||||
|     var root: Element? |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * A string which specifies a set of offsets to add to the root's bounding_box when calculating intersections, |  | ||||||
|      * effectively shrinking or growing the root for calculation purposes. The syntax is approximately the same as that |  | ||||||
|      * for the CSS margin property; see The root element and root margin in Intersection Observer API for more |  | ||||||
|      * information on how the margin works and the syntax. The default is "0px 0px 0px 0px". |  | ||||||
|      */ |  | ||||||
|     var rootMargin: String? |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Either a single number or an array of numbers between 0.0 and 1.0, specifying a ratio of intersection area to |  | ||||||
|      * total bounding box area for the observed target. A value of 0.0 means that even a single visible pixel counts as |  | ||||||
|      * the target being visible. 1.0 means that the entire target element is visible. See Thresholds in Intersection |  | ||||||
|      * Observer API for a more in-depth description of how thresholds are used. The default is a threshold of 0.0. |  | ||||||
|      */ |  | ||||||
|     var threshold: JsArray<JsNumber>? |  | ||||||
| } |  | ||||||
|  |  | ||||||
| private fun createEmptyJsObject(): JsAny = js("{}") |  | ||||||
|  |  | ||||||
| fun IntersectionObserverOptions( |  | ||||||
|     block: IntersectionObserverOptions.() -> Unit = {} |  | ||||||
| ): IntersectionObserverOptions = createEmptyJsObject().unsafeCast<IntersectionObserverOptions>().apply(block) |  | ||||||
|  |  | ||||||
| external interface IntersectionObserverEntry: JsAny { |  | ||||||
|     /** |  | ||||||
|      * Returns the bounds rectangle of the target element as a DOMRectReadOnly. The bounds are computed as described in |  | ||||||
|      * the documentation for Element.getBoundingClientRect(). |  | ||||||
|      */ |  | ||||||
|     val boundingClientRect: DOMRectReadOnly |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Returns the ratio of the intersectionRect to the boundingClientRect. |  | ||||||
|      */ |  | ||||||
|     val intersectionRatio: JsNumber |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Returns a DOMRectReadOnly representing the target's visible area. |  | ||||||
|      */ |  | ||||||
|     val intersectionRect: DOMRectReadOnly |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * A Boolean value which is true if the target element intersects with the intersection observer's root. If this is |  | ||||||
|      * true, then, the IntersectionObserverEntry describes a transition into a state of intersection; if it's false, |  | ||||||
|      * then you know the transition is from intersecting to not-intersecting. |  | ||||||
|      */ |  | ||||||
|     val isIntersecting: Boolean |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Returns a DOMRectReadOnly for the intersection observer's root. |  | ||||||
|      */ |  | ||||||
|     val rootBounds: DOMRectReadOnly |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * The Element whose intersection with the root changed. |  | ||||||
|      */ |  | ||||||
|     val target: Element |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * A DOMHighResTimeStamp indicating the time at which the intersection was recorded, relative to the |  | ||||||
|      * IntersectionObserver's time origin. |  | ||||||
|      */ |  | ||||||
|     val time: Double |  | ||||||
| } |  | ||||||
|  |  | ||||||
| typealias IntersectionObserverCallback = (entries: JsArray<IntersectionObserverEntry>, observer: IntersectionObserver) -> Unit |  | ||||||
|  |  | ||||||
| /** |  | ||||||
|  * This is just an implementation from [this commentary](https://youtrack.jetbrains.com/issue/KT-43157#focus=Comments-27-4498582.0-0) |  | ||||||
|  * of Kotlin JS issue related to the absence of [Intersection Observer API](https://developer.mozilla.org/en-US/docs/Web/API/IntersectionObserver) |  | ||||||
|  */ |  | ||||||
| external class IntersectionObserver(callback: IntersectionObserverCallback): JsAny { |  | ||||||
|     constructor(callback: IntersectionObserverCallback, options: IntersectionObserverOptions) |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * The Element or Document whose bounds are used as the bounding box when testing for intersection. If no root value |  | ||||||
|      * was passed to the constructor or its value is null, the top-level document's viewport is used. |  | ||||||
|      */ |  | ||||||
|     val root: Element |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * An offset rectangle applied to the root's bounding box when calculating intersections, effectively shrinking or |  | ||||||
|      * growing the root for calculation purposes. The value returned by this property may not be the same as the one |  | ||||||
|      * specified when calling the constructor as it may be changed to match internal requirements. Each offset can be |  | ||||||
|      * expressed in pixels (px) or as a percentage (%). The default is "0px 0px 0px 0px". |  | ||||||
|      */ |  | ||||||
|     val rootMargin: String |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * A list of thresholds, sorted in increasing numeric order, where each threshold is a ratio of intersection area to |  | ||||||
|      * bounding box area of an observed target. Notifications for a target are generated when any of the thresholds are |  | ||||||
|      * crossed for that target. If no value was passed to the constructor, 0 is used. |  | ||||||
|      */ |  | ||||||
|     val thresholds: JsArray<JsNumber> |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Stops the IntersectionObserver object from observing any target. |  | ||||||
|      */ |  | ||||||
|     fun disconnect() |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Tells the IntersectionObserver a target element to observe. |  | ||||||
|      */ |  | ||||||
|     fun observe(targetElement: Element) |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Returns an array of IntersectionObserverEntry objects for all observed targets. |  | ||||||
|      */ |  | ||||||
|     fun takeRecords(): JsArray<IntersectionObserverEntry> |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Tells the IntersectionObserver to stop observing a particular target element. |  | ||||||
|      */ |  | ||||||
|     fun unobserve(targetElement: Element) |  | ||||||
| } |  | ||||||
Some files were not shown because too many files have changed in this diff Show More
		Reference in New Issue
	
	Block a user