├── .editorconfig ├── .github └── workflows │ ├── check.yml │ └── documentation.yml ├── .gitignore ├── LICENSE ├── README.md ├── build.gradle.kts ├── gradle.properties ├── gradle ├── android.versions.toml ├── common.versions.toml └── wrapper │ ├── gradle-wrapper.jar │ └── gradle-wrapper.properties ├── gradlew ├── gradlew.bat ├── kotlin-js-store └── yarn.lock ├── navigation-screens-viewmodel ├── build.gradle.kts └── src │ ├── androidInstrumentedTest │ └── kotlin │ │ └── io │ │ └── github │ │ └── lukwol │ │ └── navigation │ │ └── screens │ │ └── viewmodel │ │ └── components │ │ ├── ViewModelScreenNavigationTest.kt │ │ ├── components │ │ ├── FirstScreen.kt │ │ ├── FirstScreenViewModel.kt │ │ ├── SecondScreen.kt │ │ ├── SecondScreenViewModel.kt │ │ └── TestScreenNavigation.kt │ │ └── data │ │ └── TestRoutes.kt │ ├── androidMain │ └── kotlin │ │ └── io │ │ └── github │ │ └── lukwol │ │ └── navigation │ │ └── screens │ │ └── viewmodel │ │ ├── NavArgs.kt │ │ ├── ScreensNavigation.kt │ │ ├── VMScreensMapBuilder.kt │ │ └── ViewModel.kt │ ├── commonMain │ └── kotlin │ │ └── io │ │ └── github │ │ └── lukwol │ │ └── navigation │ │ └── screens │ │ └── viewmodel │ │ ├── ScreensNavigation.kt │ │ ├── VMScreensMapBuilder.kt │ │ └── ViewModel.kt │ ├── desktopTest │ └── kotlin │ │ └── io │ │ └── github │ │ └── lukwol │ │ └── navigation │ │ └── screens │ │ └── viewmodel │ │ └── components │ │ ├── ViewModelScreenNavigationTest.kt │ │ ├── components │ │ ├── FirstScreen.kt │ │ ├── FirstScreenViewModel.kt │ │ ├── SecondScreen.kt │ │ ├── SecondScreenViewModel.kt │ │ └── TestScreenNavigation.kt │ │ └── data │ │ └── TestRoutes.kt │ └── nonAndroidMain │ └── kotlin │ └── io │ └── github │ └── lukwol │ └── navigation │ └── screens │ └── viewmodel │ ├── ScreensNavigation.kt │ ├── VMScreensMapBuilder.kt │ ├── ViewModel.kt │ └── ViewModelStore.kt ├── navigation-screens ├── build.gradle.kts └── src │ ├── androidInstrumentedTest │ └── kotlin │ │ └── io │ │ └── github │ │ └── lukwol │ │ └── navigation │ │ └── screens │ │ └── android │ │ ├── ScreensNavigationTest.kt │ │ ├── components │ │ ├── FirstScreen.kt │ │ ├── SecondScreen.kt │ │ ├── TestScreenNavigation.kt │ │ └── ThirdScreen.kt │ │ └── data │ │ ├── TestRoutes.kt │ │ └── ThirdScreenArgs.kt │ ├── androidMain │ └── kotlin │ │ └── io │ │ └── github │ │ └── lukwol │ │ └── navigation │ │ └── screens │ │ ├── NavArgs.kt │ │ ├── ScreensController.kt │ │ ├── ScreensMapBuilder.kt │ │ └── ScreensNavigation.kt │ ├── commonMain │ └── kotlin │ │ └── io │ │ └── github │ │ └── lukwol │ │ └── navigation │ │ └── screens │ │ ├── LocalScreensController.kt │ │ ├── RouteWithArgs.kt │ │ ├── ScreensController.kt │ │ ├── ScreensMapBuilder.kt │ │ └── ScreensNavigation.kt │ ├── desktopTest │ └── kotlin │ │ └── io │ │ └── github │ │ └── lukwol │ │ └── navigation │ │ └── screens │ │ └── desktop │ │ ├── ScreensControllerTest.kt │ │ ├── ScreensMapBuilderTest.kt │ │ ├── ScreensNavigationTest.kt │ │ ├── components │ │ ├── FirstScreen.kt │ │ ├── SecondScreen.kt │ │ ├── TestScreenNavigation.kt │ │ └── ThirdScreen.kt │ │ └── data │ │ ├── TestRoutes.kt │ │ └── ThirdScreenArgs.kt │ └── nonAndroidMain │ └── kotlin │ └── io │ └── github │ └── lukwol │ └── navigation │ └── screens │ ├── NavigationType.kt │ ├── ScreensController.kt │ ├── ScreensMapBuilder.kt │ └── ScreensNavigation.kt ├── navigation-windows ├── build.gradle.kts └── src │ ├── desktopMain │ └── kotlin │ │ └── io │ │ └── github │ │ └── lukwol │ │ └── navigation │ │ └── windows │ │ ├── LocalWindowController.kt │ │ ├── RouteWithArgs.kt │ │ ├── WindowsController.kt │ │ ├── WindowsMapBuilder.kt │ │ └── WindowsNavigation.kt │ └── desktopTest │ └── kotlin │ └── io │ └── github │ └── lukwol │ └── navigation │ └── windows │ ├── WindowsControllerTest.kt │ ├── WindowsMapBuilderTest.kt │ └── data │ └── TestRoutes.kt └── settings.gradle.kts /.editorconfig: -------------------------------------------------------------------------------- 1 | root = true 2 | 3 | # noinspection EditorConfigKeyCorrectness 4 | [*.{kt,kts}] 5 | ktlint_standard_property-naming = disabled 6 | ktlint_function_naming_ignore_when_annotated_with = Composable 7 | -------------------------------------------------------------------------------- /.github/workflows/check.yml: -------------------------------------------------------------------------------- 1 | name: check 2 | 3 | on: 4 | push: 5 | branches: main 6 | pull_request: 7 | branches: main 8 | workflow_dispatch: 9 | 10 | jobs: 11 | check: 12 | runs-on: macos-latest 13 | steps: 14 | - name: Checkout 15 | uses: actions/checkout@v3 16 | - name: Setup JDK 17 | uses: actions/setup-java@v3 18 | with: 19 | distribution: adopt 20 | java-version: 21 21 | - name: Validate Gradle wrapper 22 | uses: gradle/wrapper-validation-action@v1 23 | - name: Check all modules 24 | run: ./gradlew check 25 | - name: Run instrumented unit tests 26 | uses: reactivecircus/android-emulator-runner@v2 27 | with: 28 | api-level: 33 29 | target: google_apis 30 | arch: x86_64 31 | channel: canary 32 | script: ./gradlew connectedCheck 33 | -------------------------------------------------------------------------------- /.github/workflows/documentation.yml: -------------------------------------------------------------------------------- 1 | name: documentation 2 | 3 | on: 4 | push: 5 | tags: 6 | - '[0-9]+.[0-9]+.[0-9]+' 7 | 8 | permissions: 9 | contents: read 10 | pages: write 11 | id-token: write 12 | 13 | concurrency: 14 | group: pages 15 | cancel-in-progress: true 16 | 17 | jobs: 18 | deploy: 19 | environment: 20 | name: github-pages 21 | url: ${{ steps.deployment.outputs.page_url }} 22 | runs-on: ubuntu-latest 23 | steps: 24 | - name: Checkout 25 | uses: actions/checkout@v3 26 | - name: Setup JDK 27 | uses: actions/setup-java@v3 28 | with: 29 | distribution: adopt 30 | java-version: 21 31 | - name: Validate Gradle wrapper 32 | uses: gradle/wrapper-validation-action@v1 33 | - name: Generate documentation 34 | run: ./gradlew dokkaHtmlMultiModule 35 | - name: Setup Pages 36 | uses: actions/configure-pages@v2 37 | - name: Upload artifact 38 | uses: actions/upload-pages-artifact@v1 39 | with: 40 | path: './build/dokka/htmlMultiModule' 41 | - name: Deploy to GitHub Pages 42 | id: deployment 43 | uses: actions/deploy-pages@v1 44 | -------------------------------------------------------------------------------- /.gitignore: -------------------------------------------------------------------------------- 1 | *.iml 2 | .gradle 3 | .idea 4 | .DS_Store 5 | build 6 | captures 7 | .externalNativeBuild 8 | .cxx 9 | local.properties 10 | xcuserdata 11 | -------------------------------------------------------------------------------- /LICENSE: -------------------------------------------------------------------------------- 1 | MIT License 2 | 3 | Copyright (c) 2023 Łukasz Wolańczyk 4 | 5 | Permission is hereby granted, free of charge, to any person obtaining a copy 6 | of this software and associated documentation files (the "Software"), to deal 7 | in the Software without restriction, including without limitation the rights 8 | to use, copy, modify, merge, publish, distribute, sublicense, and/or sell 9 | copies of the Software, and to permit persons to whom the Software is 10 | furnished to do so, subject to the following conditions: 11 | 12 | The above copyright notice and this permission notice shall be included in all 13 | copies or substantial portions of the Software. 14 | 15 | THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR 16 | IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, 17 | FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE 18 | AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER 19 | LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, 20 | OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE 21 | SOFTWARE. 22 | -------------------------------------------------------------------------------- /README.md: -------------------------------------------------------------------------------- 1 | # navigation 2 | 3 | [![maven-central](https://img.shields.io/badge/Maven-Central-download.svg?style=flat-square)](https://central.sonatype.com/namespace/io.github.lukwol) 4 | [![check](https://github.com/lukwol/navigation/actions/workflows/check.yml/badge.svg)](https://github.com/lukwol/navigation/actions/workflows/check.yml) 5 | 6 | Tiny library for easy navigation 7 | in [Compose Multiplatform](https://github.com/JetBrains/compose-jb/) 8 | applications. 9 | 10 | Provides: 11 | 12 | * Screens navigation - multiplatform 13 | * Optional `ViewModel` with work cancellation support - multiplatform 14 | * Windows navigation - desktop only 15 | 16 | ## Versions 17 | 18 | For each version of `navigation` specific version of Compose Multiplatform is required 19 | 20 | | navigation | compose-multiplatform | 21 | |:----------:|:---------------------:| 22 | | 1.4.0 | 1.6.1 | 23 | | 1.3.2 | 1.5.12 | 24 | | 1.3.1 | 1.5.11 | 25 | | 1.3.0 | 1.5.10 | 26 | | 1.2.0 | 1.5.3 | 27 | | 1.1.0 | 1.5.2 | 28 | | 1.0.0 | 1.5.1 | 29 | 30 | ## Installation 31 | 32 | Add `mavenCentral` and `google` repositories: 33 | 34 | ```kotlin 35 | repositories { 36 | mavenCentral() 37 | google() 38 | } 39 | ``` 40 | 41 | Declare dependencies in `build.gradle.kts`: 42 | 43 | ```kotlin 44 | dependencies { 45 | // Screens navigation - multiplatform 46 | implementation("io.github.lukwol:navigation-screens:1.4.0") 47 | // Screens navigation with ViewModel support - multiplatform 48 | implementation("io.github.lukwol:navigation-screens-viewmodel:1.4.0") 49 | // Windows navigation - desktop application only 50 | implementation("io.github.lukwol:navigation-windows:1.4.0") 51 | } 52 | ``` 53 | 54 | ## Usage 55 | 56 | Bootstrap new project with handy [`app-template`](https://github.com/lukwol/app-template/). 57 | 58 | ### Build screens navigation 59 | 60 | Basic screens navigation: 61 | 62 | ```kotlin 63 | ScreensNavigation( 64 | startRoute = AppRoutes.FirstScreenRoute, 65 | ) { 66 | screen(AppRoutes.FirstScreenRoute) { 67 | FirstScreen() 68 | } 69 | 70 | screen(AppRoutes.SecondScreenRoute) { args: String? -> 71 | SecondScreen(args) 72 | } 73 | } 74 | ``` 75 | 76 | Screens navigation with `ViewModel` and custom animations: 77 | 78 | ```kotlin 79 | ScreensNavigation( 80 | startRoute = AppRoutes.FirstScreenRoute, 81 | enterTransition = { 82 | slideIntoContainer(AnimatedContentTransitionScope.SlideDirection.Left) 83 | }, 84 | exitTransition = { 85 | slideOutOfContainer(AnimatedContentTransitionScope.SlideDirection.Left) 86 | }, 87 | popEnterTransition = { 88 | slideIntoContainer(AnimatedContentTransitionScope.SlideDirection.Right) 89 | }, 90 | popExitTransition = { 91 | slideOutOfContainer(AnimatedContentTransitionScope.SlideDirection.Right) 92 | }, 93 | ) { 94 | screen( 95 | route = AppRoutes.FirstScreenRoute, 96 | viewModelFactory = { 97 | FirstScreenViewModel() 98 | } 99 | ) { viewModel -> 100 | FirstScreen(viewModel) 101 | } 102 | 103 | screen( 104 | route = AppRoutes.SecondScreenRoute, 105 | viewModelWithArgs = { args: SomeArgs? -> 106 | SecondScreenViewModel(args) 107 | } 108 | ) { viewModel -> 109 | SecondScreen(viewModel) 110 | } 111 | } 112 | ``` 113 | 114 | ### Build windows navigation if needed (desktop) 115 | 116 | ```kotlin 117 | WindowsNavigation( 118 | startRoute = AppRoutes.FirstWindowRoute 119 | ) { 120 | window( 121 | route = AppRoutes.FirstWindowRoute, 122 | title = "First Window" 123 | ) { 124 | ScreensNavigation( 125 | startRoute = AppRoutes.FirstScreenRoute 126 | ) { 127 | // ... 128 | } 129 | } 130 | 131 | window( 132 | route = AppRoutes.SecondWindowRoute, 133 | title = "Second Window" 134 | ) { 135 | ScreensNavigation( 136 | startRoute = AppRoutes.SecondScreenRoute 137 | ) { 138 | // ... 139 | } 140 | } 141 | } 142 | ``` 143 | 144 | ### Navigate to screen 145 | 146 | ```kotlin 147 | // Obtain LocalScreensController in your view 148 | val screensController = LocalScreensController.current 149 | 150 | // Push screen 151 | screensController.push(AppRoutes.SecondScreenRoute) 152 | 153 | // Optionally pass arguments if needed 154 | screensController.push(AppRoutes.SecondScreenRoute, SomeArguments) 155 | 156 | // Pop screen to navigate back 157 | screensController.pop() 158 | 159 | // Optionally pass route, up to which screens should be dismissed 160 | screensController.pop(AppRoutes.FirstScreenRoute) 161 | ``` 162 | 163 | ### Navigate to window (desktop) 164 | 165 | ```kotlin 166 | // Obtain LocalWindowController in your view 167 | val windowsController = LocalWindowController.current 168 | 169 | // Open window 170 | windowsController.open(AppRoutes.SecondWindowRoute) 171 | 172 | // Optionally pass arguments if needed 173 | windowsController.open(AppRoutes.SecondWindowRoute, SomeArguments) 174 | 175 | // Close window 176 | windowsController.close(AppRoutes.SecondWindowRoute) 177 | ``` 178 | 179 | ## Documentation 180 | 181 | API Reference is available at https://lukwol.github.io/navigation/ 182 | 183 | ## Licensing 184 | 185 | Project is available under [MIT](https://github.com/lukwol/navigation/blob/main/LICENSE) License. 186 | -------------------------------------------------------------------------------- /build.gradle.kts: -------------------------------------------------------------------------------- 1 | import com.github.benmanes.gradle.versions.updates.DependencyUpdatesTask 2 | import org.jetbrains.kotlin.gradle.tasks.KotlinCompile 3 | import org.jetbrains.kotlin.util.capitalizeDecapitalize.toLowerCaseAsciiOnly 4 | 5 | plugins { 6 | alias(androidLibs.plugins.library) apply false 7 | alias(commonLibs.plugins.kotlin.multiplatform) apply false 8 | alias(commonLibs.plugins.compose.multiplatform) apply false 9 | alias(commonLibs.plugins.kotlin.serialization) apply false 10 | alias(commonLibs.plugins.kotlinter) apply false 11 | alias(commonLibs.plugins.dokka) 12 | alias(commonLibs.plugins.dependency.updates) 13 | } 14 | 15 | subprojects { 16 | apply { 17 | plugin("org.jetbrains.dokka") 18 | plugin("org.jmailen.kotlinter") 19 | plugin("maven-publish") 20 | plugin("signing") 21 | } 22 | 23 | group = "io.github.lukwol" 24 | version = "1.4.0" 25 | 26 | extensions.configure { 27 | repositories { 28 | maven { 29 | name = "Staging" 30 | setUrl("https://s01.oss.sonatype.org/service/local/staging/deploy/maven2/") 31 | credentials { 32 | username = findProperty("ossrhUsername").toString() 33 | password = findProperty("ossrhPassword").toString() 34 | } 35 | } 36 | maven { 37 | name = "Snapshots" 38 | setUrl("https://s01.oss.sonatype.org/content/repositories/snapshots/") 39 | credentials { 40 | username = findProperty("ossrhUsername").toString() 41 | password = findProperty("ossrhPassword").toString() 42 | } 43 | } 44 | } 45 | 46 | extensions.configure { 47 | useInMemoryPgpKeys( 48 | /* defaultKeyId = */ findProperty("keyId").toString(), 49 | /* defaultSecretKey = */ findProperty("keyBase64").toString(), 50 | /* defaultPassword = */ findProperty("keyPassword").toString() 51 | ) 52 | } 53 | 54 | val javadocJar by tasks.registering(Jar::class) { 55 | archiveClassifier.set("javadoc") 56 | } 57 | 58 | publications { 59 | withType { 60 | // artifact(javadocJar) 61 | 62 | pom { 63 | name.set(project.name) 64 | description.set("Simple navigation in Compose Multiplatform apps") 65 | url.set("https://github.com/lukwol/navigation") 66 | 67 | licenses { 68 | license { 69 | name.set("MIT License") 70 | url.set("https://github.com/lukwol/navigation/blob/main/LICENSE") 71 | } 72 | } 73 | scm { 74 | url.set("https://github.com/lukwol/navigation") 75 | connection.set("scm:git:git://github.com/lukwol/navigation.git") 76 | } 77 | developers { 78 | developer { 79 | name.set("Lukasz Wolanczyk") 80 | url.set("https://github.com/lukwol") 81 | } 82 | } 83 | } 84 | 85 | the().sign(this) 86 | } 87 | } 88 | } 89 | 90 | tasks.withType { 91 | compilerOptions { 92 | if (project.findProperty("enableComposeCompilerReports") == "true") { 93 | val destinationDir = project.layout.buildDirectory.asFile.get().absolutePath + "/compose_metrics" 94 | freeCompilerArgs.addAll( 95 | listOf( 96 | "-P", 97 | "plugin:androidx.compose.compiler.plugins.kotlin:reportsDestination=$destinationDir", 98 | "-P", 99 | "plugin:androidx.compose.compiler.plugins.kotlin:metricsDestination=$destinationDir" 100 | ) 101 | ) 102 | } 103 | } 104 | } 105 | } 106 | 107 | tasks.withType { 108 | rejectVersionIf { 109 | val version = candidate.version.toLowerCaseAsciiOnly() 110 | 111 | listOf("-dev", "-alpha", "-beta", "-rc") 112 | .any { it in version } 113 | } 114 | } 115 | -------------------------------------------------------------------------------- /gradle.properties: -------------------------------------------------------------------------------- 1 | # Gradle 2 | org.gradle.jvmargs=-Xmx2048M -Dfile.encoding=UTF-8 -Dkotlin.daemon.jvm.options\="-Xmx2048M" 3 | 4 | # Kotlin 5 | kotlin.code.style=official 6 | 7 | # Android 8 | android.useAndroidX=true 9 | android.nonTransitiveRClass=true 10 | 11 | # MPP 12 | kotlin.mpp.enableCInteropCommonization=true 13 | kotlin.mpp.androidSourceSetLayoutVersion=2 14 | 15 | # Compose 16 | org.jetbrains.compose.experimental.uikit.enabled=true 17 | org.jetbrains.compose.experimental.wasm.enabled=true 18 | 19 | # Dokka 20 | org.jetbrains.dokka.classpath.useNativeDistributionAccessor=true 21 | -------------------------------------------------------------------------------- /gradle/android.versions.toml: -------------------------------------------------------------------------------- 1 | [versions] 2 | min-sdk = "29" 3 | compile-sdk = "34" 4 | agp = "8.2.2" 5 | compose = "1.6.1" 6 | compose-compiler = "1.5.10" 7 | compose-navigation = "2.7.7" 8 | lifecycle = "2.7.0" 9 | test-runner = "1.5.2" 10 | 11 | [plugins] 12 | library = { id = "com.android.library", version.ref = "agp" } 13 | 14 | [libraries] 15 | compose-navigation = { module = "androidx.navigation:navigation-compose", version.ref = "compose-navigation" } 16 | compose-ui-test-junit4 = { module = "androidx.compose.ui:ui-test-junit4", version.ref = "compose" } 17 | compose-ui-test-manifest = { module = "androidx.compose.ui:ui-test-manifest", version.ref = "compose" } 18 | lifecycle-viewmodel = { module = "androidx.lifecycle:lifecycle-viewmodel-ktx", version.ref = "lifecycle" } 19 | test-runner = { module = "androidx.test:runner", version.ref = "test-runner" } 20 | -------------------------------------------------------------------------------- /gradle/common.versions.toml: -------------------------------------------------------------------------------- 1 | [versions] 2 | kotlin = "1.9.22" 3 | compose = "1.6.1" 4 | coroutines = "1.8.0" 5 | kotlin-serialization-json = "1.6.3" 6 | dokka = "1.9.20" 7 | kotlinter = "4.2.0" 8 | dependency-updates = "0.51.0" 9 | 10 | [plugins] 11 | kotlin-multiplatform = { id = "org.jetbrains.kotlin.multiplatform", version.ref = "kotlin" } 12 | dokka = { id = "org.jetbrains.dokka", version.ref = "dokka" } 13 | kotlin-serialization = { id = "org.jetbrains.kotlin.plugin.serialization", version.ref = "kotlin" } 14 | compose-multiplatform = { id = "org.jetbrains.compose", version.ref = "compose" } 15 | kotlinter = { id = "org.jmailen.kotlinter", version.ref = "kotlinter" } 16 | dependency-updates = { id = "com.github.ben-manes.versions", version.ref = "dependency-updates" } 17 | 18 | [libraries] 19 | kotlin-serialization-json = { module = "org.jetbrains.kotlinx:kotlinx-serialization-json", version.ref = "kotlin-serialization-json" } 20 | coroutines-core = { module = "org.jetbrains.kotlinx:kotlinx-coroutines-core", version.ref = "coroutines" } 21 | -------------------------------------------------------------------------------- /gradle/wrapper/gradle-wrapper.jar: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/lukwol/navigation/07d666afd54d8becc9ca4e895fc87dd0bed6610d/gradle/wrapper/gradle-wrapper.jar -------------------------------------------------------------------------------- /gradle/wrapper/gradle-wrapper.properties: -------------------------------------------------------------------------------- 1 | distributionBase=GRADLE_USER_HOME 2 | distributionPath=wrapper/dists 3 | distributionUrl=https\://services.gradle.org/distributions/gradle-8.6-bin.zip 4 | networkTimeout=10000 5 | validateDistributionUrl=true 6 | zipStoreBase=GRADLE_USER_HOME 7 | zipStorePath=wrapper/dists 8 | -------------------------------------------------------------------------------- /gradlew: -------------------------------------------------------------------------------- 1 | #!/bin/sh 2 | 3 | # 4 | # Copyright © 2015-2021 the original authors. 5 | # 6 | # Licensed under the Apache License, Version 2.0 (the "License"); 7 | # you may not use this file except in compliance with the License. 8 | # You may obtain a copy of the License at 9 | # 10 | # https://www.apache.org/licenses/LICENSE-2.0 11 | # 12 | # Unless required by applicable law or agreed to in writing, software 13 | # distributed under the License is distributed on an "AS IS" BASIS, 14 | # WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 15 | # See the License for the specific language governing permissions and 16 | # limitations under the License. 17 | # 18 | 19 | ############################################################################## 20 | # 21 | # Gradle start up script for POSIX generated by Gradle. 22 | # 23 | # Important for running: 24 | # 25 | # (1) You need a POSIX-compliant shell to run this script. If your /bin/sh is 26 | # noncompliant, but you have some other compliant shell such as ksh or 27 | # bash, then to run this script, type that shell name before the whole 28 | # command line, like: 29 | # 30 | # ksh Gradle 31 | # 32 | # Busybox and similar reduced shells will NOT work, because this script 33 | # requires all of these POSIX shell features: 34 | # * functions; 35 | # * expansions «$var», «${var}», «${var:-default}», «${var+SET}», 36 | # «${var#prefix}», «${var%suffix}», and «$( cmd )»; 37 | # * compound commands having a testable exit status, especially «case»; 38 | # * various built-in commands including «command», «set», and «ulimit». 39 | # 40 | # Important for patching: 41 | # 42 | # (2) This script targets any POSIX shell, so it avoids extensions provided 43 | # by Bash, Ksh, etc; in particular arrays are avoided. 44 | # 45 | # The "traditional" practice of packing multiple parameters into a 46 | # space-separated string is a well documented source of bugs and security 47 | # problems, so this is (mostly) avoided, by progressively accumulating 48 | # options in "$@", and eventually passing that to Java. 49 | # 50 | # Where the inherited environment variables (DEFAULT_JVM_OPTS, JAVA_OPTS, 51 | # and GRADLE_OPTS) rely on word-splitting, this is performed explicitly; 52 | # see the in-line comments for details. 53 | # 54 | # There are tweaks for specific operating systems such as AIX, CygWin, 55 | # Darwin, MinGW, and NonStop. 56 | # 57 | # (3) This script is generated from the Groovy template 58 | # https://github.com/gradle/gradle/blob/HEAD/subprojects/plugins/src/main/resources/org/gradle/api/internal/plugins/unixStartScript.txt 59 | # within the Gradle project. 60 | # 61 | # You can find Gradle at https://github.com/gradle/gradle/. 62 | # 63 | ############################################################################## 64 | 65 | # Attempt to set APP_HOME 66 | 67 | # Resolve links: $0 may be a link 68 | app_path=$0 69 | 70 | # Need this for daisy-chained symlinks. 71 | while 72 | APP_HOME=${app_path%"${app_path##*/}"} # leaves a trailing /; empty if no leading path 73 | [ -h "$app_path" ] 74 | do 75 | ls=$( ls -ld "$app_path" ) 76 | link=${ls#*' -> '} 77 | case $link in #( 78 | /*) app_path=$link ;; #( 79 | *) app_path=$APP_HOME$link ;; 80 | esac 81 | done 82 | 83 | # This is normally unused 84 | # shellcheck disable=SC2034 85 | APP_BASE_NAME=${0##*/} 86 | # Discard cd standard output in case $CDPATH is set (https://github.com/gradle/gradle/issues/25036) 87 | APP_HOME=$( cd "${APP_HOME:-./}" > /dev/null && pwd -P ) || exit 88 | 89 | # Use the maximum available, or set MAX_FD != -1 to use that value. 90 | MAX_FD=maximum 91 | 92 | warn () { 93 | echo "$*" 94 | } >&2 95 | 96 | die () { 97 | echo 98 | echo "$*" 99 | echo 100 | exit 1 101 | } >&2 102 | 103 | # OS specific support (must be 'true' or 'false'). 104 | cygwin=false 105 | msys=false 106 | darwin=false 107 | nonstop=false 108 | case "$( uname )" in #( 109 | CYGWIN* ) cygwin=true ;; #( 110 | Darwin* ) darwin=true ;; #( 111 | MSYS* | MINGW* ) msys=true ;; #( 112 | NONSTOP* ) nonstop=true ;; 113 | esac 114 | 115 | CLASSPATH=$APP_HOME/gradle/wrapper/gradle-wrapper.jar 116 | 117 | 118 | # Determine the Java command to use to start the JVM. 119 | if [ -n "$JAVA_HOME" ] ; then 120 | if [ -x "$JAVA_HOME/jre/sh/java" ] ; then 121 | # IBM's JDK on AIX uses strange locations for the executables 122 | JAVACMD=$JAVA_HOME/jre/sh/java 123 | else 124 | JAVACMD=$JAVA_HOME/bin/java 125 | fi 126 | if [ ! -x "$JAVACMD" ] ; then 127 | die "ERROR: JAVA_HOME is set to an invalid directory: $JAVA_HOME 128 | 129 | Please set the JAVA_HOME variable in your environment to match the 130 | location of your Java installation." 131 | fi 132 | else 133 | JAVACMD=java 134 | if ! command -v java >/dev/null 2>&1 135 | then 136 | die "ERROR: JAVA_HOME is not set and no 'java' command could be found in your PATH. 137 | 138 | Please set the JAVA_HOME variable in your environment to match the 139 | location of your Java installation." 140 | fi 141 | fi 142 | 143 | # Increase the maximum file descriptors if we can. 144 | if ! "$cygwin" && ! "$darwin" && ! "$nonstop" ; then 145 | case $MAX_FD in #( 146 | max*) 147 | # In POSIX sh, ulimit -H is undefined. That's why the result is checked to see if it worked. 148 | # shellcheck disable=SC2039,SC3045 149 | MAX_FD=$( ulimit -H -n ) || 150 | warn "Could not query maximum file descriptor limit" 151 | esac 152 | case $MAX_FD in #( 153 | '' | soft) :;; #( 154 | *) 155 | # In POSIX sh, ulimit -n is undefined. That's why the result is checked to see if it worked. 156 | # shellcheck disable=SC2039,SC3045 157 | ulimit -n "$MAX_FD" || 158 | warn "Could not set maximum file descriptor limit to $MAX_FD" 159 | esac 160 | fi 161 | 162 | # Collect all arguments for the java command, stacking in reverse order: 163 | # * args from the command line 164 | # * the main class name 165 | # * -classpath 166 | # * -D...appname settings 167 | # * --module-path (only if needed) 168 | # * DEFAULT_JVM_OPTS, JAVA_OPTS, and GRADLE_OPTS environment variables. 169 | 170 | # For Cygwin or MSYS, switch paths to Windows format before running java 171 | if "$cygwin" || "$msys" ; then 172 | APP_HOME=$( cygpath --path --mixed "$APP_HOME" ) 173 | CLASSPATH=$( cygpath --path --mixed "$CLASSPATH" ) 174 | 175 | JAVACMD=$( cygpath --unix "$JAVACMD" ) 176 | 177 | # Now convert the arguments - kludge to limit ourselves to /bin/sh 178 | for arg do 179 | if 180 | case $arg in #( 181 | -*) false ;; # don't mess with options #( 182 | /?*) t=${arg#/} t=/${t%%/*} # looks like a POSIX filepath 183 | [ -e "$t" ] ;; #( 184 | *) false ;; 185 | esac 186 | then 187 | arg=$( cygpath --path --ignore --mixed "$arg" ) 188 | fi 189 | # Roll the args list around exactly as many times as the number of 190 | # args, so each arg winds up back in the position where it started, but 191 | # possibly modified. 192 | # 193 | # NB: a `for` loop captures its iteration list before it begins, so 194 | # changing the positional parameters here affects neither the number of 195 | # iterations, nor the values presented in `arg`. 196 | shift # remove old arg 197 | set -- "$@" "$arg" # push replacement arg 198 | done 199 | fi 200 | 201 | 202 | # Add default JVM options here. You can also use JAVA_OPTS and GRADLE_OPTS to pass JVM options to this script. 203 | DEFAULT_JVM_OPTS='"-Xmx64m" "-Xms64m"' 204 | 205 | # Collect all arguments for the java command: 206 | # * DEFAULT_JVM_OPTS, JAVA_OPTS, JAVA_OPTS, and optsEnvironmentVar are not allowed to contain shell fragments, 207 | # and any embedded shellness will be escaped. 208 | # * For example: A user cannot expect ${Hostname} to be expanded, as it is an environment variable and will be 209 | # treated as '${Hostname}' itself on the command line. 210 | 211 | set -- \ 212 | "-Dorg.gradle.appname=$APP_BASE_NAME" \ 213 | -classpath "$CLASSPATH" \ 214 | org.gradle.wrapper.GradleWrapperMain \ 215 | "$@" 216 | 217 | # Stop when "xargs" is not available. 218 | if ! command -v xargs >/dev/null 2>&1 219 | then 220 | die "xargs is not available" 221 | fi 222 | 223 | # Use "xargs" to parse quoted args. 224 | # 225 | # With -n1 it outputs one arg per line, with the quotes and backslashes removed. 226 | # 227 | # In Bash we could simply go: 228 | # 229 | # readarray ARGS < <( xargs -n1 <<<"$var" ) && 230 | # set -- "${ARGS[@]}" "$@" 231 | # 232 | # but POSIX shell has neither arrays nor command substitution, so instead we 233 | # post-process each arg (as a line of input to sed) to backslash-escape any 234 | # character that might be a shell metacharacter, then use eval to reverse 235 | # that process (while maintaining the separation between arguments), and wrap 236 | # the whole thing up as a single "set" statement. 237 | # 238 | # This will of course break if any of these variables contains a newline or 239 | # an unmatched quote. 240 | # 241 | 242 | eval "set -- $( 243 | printf '%s\n' "$DEFAULT_JVM_OPTS $JAVA_OPTS $GRADLE_OPTS" | 244 | xargs -n1 | 245 | sed ' s~[^-[:alnum:]+,./:=@_]~\\&~g; ' | 246 | tr '\n' ' ' 247 | )" '"$@"' 248 | 249 | exec "$JAVACMD" "$@" 250 | -------------------------------------------------------------------------------- /gradlew.bat: -------------------------------------------------------------------------------- 1 | @rem 2 | @rem Copyright 2015 the original author or authors. 3 | @rem 4 | @rem Licensed under the Apache License, Version 2.0 (the "License"); 5 | @rem you may not use this file except in compliance with the License. 6 | @rem You may obtain a copy of the License at 7 | @rem 8 | @rem https://www.apache.org/licenses/LICENSE-2.0 9 | @rem 10 | @rem Unless required by applicable law or agreed to in writing, software 11 | @rem distributed under the License is distributed on an "AS IS" BASIS, 12 | @rem WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 13 | @rem See the License for the specific language governing permissions and 14 | @rem limitations under the License. 15 | @rem 16 | 17 | @if "%DEBUG%"=="" @echo off 18 | @rem ########################################################################## 19 | @rem 20 | @rem Gradle startup script for Windows 21 | @rem 22 | @rem ########################################################################## 23 | 24 | @rem Set local scope for the variables with windows NT shell 25 | if "%OS%"=="Windows_NT" setlocal 26 | 27 | set DIRNAME=%~dp0 28 | if "%DIRNAME%"=="" set DIRNAME=. 29 | @rem This is normally unused 30 | set APP_BASE_NAME=%~n0 31 | set APP_HOME=%DIRNAME% 32 | 33 | @rem Resolve any "." and ".." in APP_HOME to make it shorter. 34 | for %%i in ("%APP_HOME%") do set APP_HOME=%%~fi 35 | 36 | @rem Add default JVM options here. You can also use JAVA_OPTS and GRADLE_OPTS to pass JVM options to this script. 37 | set DEFAULT_JVM_OPTS="-Xmx64m" "-Xms64m" 38 | 39 | @rem Find java.exe 40 | if defined JAVA_HOME goto findJavaFromJavaHome 41 | 42 | set JAVA_EXE=java.exe 43 | %JAVA_EXE% -version >NUL 2>&1 44 | if %ERRORLEVEL% equ 0 goto execute 45 | 46 | echo. 47 | echo ERROR: JAVA_HOME is not set and no 'java' command could be found in your PATH. 48 | echo. 49 | echo Please set the JAVA_HOME variable in your environment to match the 50 | echo location of your Java installation. 51 | 52 | goto fail 53 | 54 | :findJavaFromJavaHome 55 | set JAVA_HOME=%JAVA_HOME:"=% 56 | set JAVA_EXE=%JAVA_HOME%/bin/java.exe 57 | 58 | if exist "%JAVA_EXE%" goto execute 59 | 60 | echo. 61 | echo ERROR: JAVA_HOME is set to an invalid directory: %JAVA_HOME% 62 | echo. 63 | echo Please set the JAVA_HOME variable in your environment to match the 64 | echo location of your Java installation. 65 | 66 | goto fail 67 | 68 | :execute 69 | @rem Setup the command line 70 | 71 | set CLASSPATH=%APP_HOME%\gradle\wrapper\gradle-wrapper.jar 72 | 73 | 74 | @rem Execute Gradle 75 | "%JAVA_EXE%" %DEFAULT_JVM_OPTS% %JAVA_OPTS% %GRADLE_OPTS% "-Dorg.gradle.appname=%APP_BASE_NAME%" -classpath "%CLASSPATH%" org.gradle.wrapper.GradleWrapperMain %* 76 | 77 | :end 78 | @rem End local scope for the variables with windows NT shell 79 | if %ERRORLEVEL% equ 0 goto mainEnd 80 | 81 | :fail 82 | rem Set variable GRADLE_EXIT_CONSOLE if you need the _script_ return code instead of 83 | rem the _cmd.exe /c_ return code! 84 | set EXIT_CODE=%ERRORLEVEL% 85 | if %EXIT_CODE% equ 0 set EXIT_CODE=1 86 | if not ""=="%GRADLE_EXIT_CONSOLE%" exit %EXIT_CODE% 87 | exit /b %EXIT_CODE% 88 | 89 | :mainEnd 90 | if "%OS%"=="Windows_NT" endlocal 91 | 92 | :omega 93 | -------------------------------------------------------------------------------- /kotlin-js-store/yarn.lock: -------------------------------------------------------------------------------- 1 | # THIS IS AN AUTOGENERATED FILE. DO NOT EDIT THIS FILE DIRECTLY. 2 | # yarn lockfile v1 3 | 4 | 5 | "@colors/colors@1.5.0": 6 | version "1.5.0" 7 | resolved "https://registry.yarnpkg.com/@colors/colors/-/colors-1.5.0.tgz#bb504579c1cae923e6576a4f5da43d25f97bdbd9" 8 | integrity sha512-ooWCrlZP11i8GImSjTHYHLkvFDP48nS4+204nGb1RiX/WXYHmJA2III9/e2DWVabCESdW7hBAEzHRqUn9OUVvQ== 9 | 10 | "@discoveryjs/json-ext@^0.5.0": 11 | version "0.5.7" 12 | resolved "https://registry.yarnpkg.com/@discoveryjs/json-ext/-/json-ext-0.5.7.tgz#1d572bfbbe14b7704e0ba0f39b74815b84870d70" 13 | integrity sha512-dBVuXR082gk3jsFp7Rd/JI4kytwGHecnCoTtXFb7DB6CNHp4rg5k1bhg0nWdLGLnOV71lmDzGQaLMy8iPLY0pw== 14 | 15 | "@jridgewell/gen-mapping@^0.3.0": 16 | version "0.3.3" 17 | resolved "https://registry.yarnpkg.com/@jridgewell/gen-mapping/-/gen-mapping-0.3.3.tgz#7e02e6eb5df901aaedb08514203b096614024098" 18 | integrity sha512-HLhSWOLRi875zjjMG/r+Nv0oCW8umGb0BgEhyX3dDX3egwZtB8PqLnjz3yedt8R5StBrzcg4aBpnh8UA9D1BoQ== 19 | dependencies: 20 | "@jridgewell/set-array" "^1.0.1" 21 | "@jridgewell/sourcemap-codec" "^1.4.10" 22 | "@jridgewell/trace-mapping" "^0.3.9" 23 | 24 | "@jridgewell/resolve-uri@^3.1.0": 25 | version "3.1.1" 26 | resolved "https://registry.yarnpkg.com/@jridgewell/resolve-uri/-/resolve-uri-3.1.1.tgz#c08679063f279615a3326583ba3a90d1d82cc721" 27 | integrity sha512-dSYZh7HhCDtCKm4QakX0xFpsRDqjjtZf/kjI/v3T3Nwt5r8/qz/M19F9ySyOqU94SXBmeG9ttTul+YnR4LOxFA== 28 | 29 | "@jridgewell/set-array@^1.0.1": 30 | version "1.1.2" 31 | resolved "https://registry.yarnpkg.com/@jridgewell/set-array/-/set-array-1.1.2.tgz#7c6cf998d6d20b914c0a55a91ae928ff25965e72" 32 | integrity sha512-xnkseuNADM0gt2bs+BvhO0p78Mk762YnZdsuzFV018NoG1Sj1SCQvpSqa7XUaTam5vAGasABV9qXASMKnFMwMw== 33 | 34 | "@jridgewell/source-map@^0.3.3": 35 | version "0.3.5" 36 | resolved "https://registry.yarnpkg.com/@jridgewell/source-map/-/source-map-0.3.5.tgz#a3bb4d5c6825aab0d281268f47f6ad5853431e91" 37 | integrity sha512-UTYAUj/wviwdsMfzoSJspJxbkH5o1snzwX0//0ENX1u/55kkZZkcTZP6u9bwKGkv+dkk9at4m1Cpt0uY80kcpQ== 38 | dependencies: 39 | "@jridgewell/gen-mapping" "^0.3.0" 40 | "@jridgewell/trace-mapping" "^0.3.9" 41 | 42 | "@jridgewell/sourcemap-codec@^1.4.10", "@jridgewell/sourcemap-codec@^1.4.14": 43 | version "1.4.15" 44 | resolved "https://registry.yarnpkg.com/@jridgewell/sourcemap-codec/-/sourcemap-codec-1.4.15.tgz#d7c6e6755c78567a951e04ab52ef0fd26de59f32" 45 | integrity sha512-eF2rxCRulEKXHTRiDrDy6erMYWqNw4LPdQ8UQA4huuxaQsVeRPFl2oM8oDGxMFhJUWZf9McpLtJasDDZb/Bpeg== 46 | 47 | "@jridgewell/trace-mapping@^0.3.20", "@jridgewell/trace-mapping@^0.3.9": 48 | version "0.3.22" 49 | resolved "https://registry.yarnpkg.com/@jridgewell/trace-mapping/-/trace-mapping-0.3.22.tgz#72a621e5de59f5f1ef792d0793a82ee20f645e4c" 50 | integrity sha512-Wf963MzWtA2sjrNt+g18IAln9lKnlRp+K2eH4jjIoF1wYeq3aMREpG09xhlhdzS0EjwU7qmUJYangWa+151vZw== 51 | dependencies: 52 | "@jridgewell/resolve-uri" "^3.1.0" 53 | "@jridgewell/sourcemap-codec" "^1.4.14" 54 | 55 | "@socket.io/component-emitter@~3.1.0": 56 | version "3.1.0" 57 | resolved "https://registry.yarnpkg.com/@socket.io/component-emitter/-/component-emitter-3.1.0.tgz#96116f2a912e0c02817345b3c10751069920d553" 58 | integrity sha512-+9jVqKhRSpsc591z5vX+X5Yyw+he/HCB4iQ/RYxw35CEPaY1gnsNE43nf9n9AaYjAQrTiI/mOwKUKdUs9vf7Xg== 59 | 60 | "@types/cookie@^0.4.1": 61 | version "0.4.1" 62 | resolved "https://registry.yarnpkg.com/@types/cookie/-/cookie-0.4.1.tgz#bfd02c1f2224567676c1545199f87c3a861d878d" 63 | integrity sha512-XW/Aa8APYr6jSVVA1y/DEIZX0/GMKLEVekNG727R8cs56ahETkRAy/3DR7+fJyh7oUgGwNQaRfXCun0+KbWY7Q== 64 | 65 | "@types/cors@^2.8.12": 66 | version "2.8.17" 67 | resolved "https://registry.yarnpkg.com/@types/cors/-/cors-2.8.17.tgz#5d718a5e494a8166f569d986794e49c48b216b2b" 68 | integrity sha512-8CGDvrBj1zgo2qE+oS3pOCyYNqCPryMWY2bGfwA0dcfopWGgxs+78df0Rs3rc9THP4JkOhLsAa+15VdpAqkcUA== 69 | dependencies: 70 | "@types/node" "*" 71 | 72 | "@types/eslint-scope@^3.7.3": 73 | version "3.7.7" 74 | resolved "https://registry.yarnpkg.com/@types/eslint-scope/-/eslint-scope-3.7.7.tgz#3108bd5f18b0cdb277c867b3dd449c9ed7079ac5" 75 | integrity sha512-MzMFlSLBqNF2gcHWO0G1vP/YQyfvrxZ0bF+u7mzUdZ1/xK4A4sru+nraZz5i3iEIk1l1uyicaDVTB4QbbEkAYg== 76 | dependencies: 77 | "@types/eslint" "*" 78 | "@types/estree" "*" 79 | 80 | "@types/eslint@*": 81 | version "8.56.2" 82 | resolved "https://registry.yarnpkg.com/@types/eslint/-/eslint-8.56.2.tgz#1c72a9b794aa26a8b94ad26d5b9aa51c8a6384bb" 83 | integrity sha512-uQDwm1wFHmbBbCZCqAlq6Do9LYwByNZHWzXppSnay9SuwJ+VRbjkbLABer54kcPnMSlG6Fdiy2yaFXm/z9Z5gw== 84 | dependencies: 85 | "@types/estree" "*" 86 | "@types/json-schema" "*" 87 | 88 | "@types/estree@*", "@types/estree@^1.0.0": 89 | version "1.0.5" 90 | resolved "https://registry.yarnpkg.com/@types/estree/-/estree-1.0.5.tgz#a6ce3e556e00fd9895dd872dd172ad0d4bd687f4" 91 | integrity sha512-/kYRxGDLWzHOB7q+wtSUQlFrtcdUccpfy+X+9iMBpHK8QLLhx2wIPYuS5DYtR9Wa/YlZAbIovy7qVdB1Aq6Lyw== 92 | 93 | "@types/json-schema@*", "@types/json-schema@^7.0.8": 94 | version "7.0.15" 95 | resolved "https://registry.yarnpkg.com/@types/json-schema/-/json-schema-7.0.15.tgz#596a1747233694d50f6ad8a7869fcb6f56cf5841" 96 | integrity sha512-5+fP8P8MFNC+AyZCDxrB2pkZFPGzqQWUzpSeuuVLvm8VMcorNYavBqoFcxK8bQz4Qsbn4oUEEem4wDLfcysGHA== 97 | 98 | "@types/node@*", "@types/node@>=10.0.0": 99 | version "20.11.10" 100 | resolved "https://registry.yarnpkg.com/@types/node/-/node-20.11.10.tgz#6c3de8974d65c362f82ee29db6b5adf4205462f9" 101 | integrity sha512-rZEfe/hJSGYmdfX9tvcPMYeYPW2sNl50nsw4jZmRcaG0HIAb0WYEpsB05GOb53vjqpyE9GUhlDQ4jLSoB5q9kg== 102 | dependencies: 103 | undici-types "~5.26.4" 104 | 105 | "@webassemblyjs/ast@1.11.6", "@webassemblyjs/ast@^1.11.5": 106 | version "1.11.6" 107 | resolved "https://registry.yarnpkg.com/@webassemblyjs/ast/-/ast-1.11.6.tgz#db046555d3c413f8966ca50a95176a0e2c642e24" 108 | integrity sha512-IN1xI7PwOvLPgjcf180gC1bqn3q/QaOCwYUahIOhbYUu8KA/3tw2RT/T0Gidi1l7Hhj5D/INhJxiICObqpMu4Q== 109 | dependencies: 110 | "@webassemblyjs/helper-numbers" "1.11.6" 111 | "@webassemblyjs/helper-wasm-bytecode" "1.11.6" 112 | 113 | "@webassemblyjs/floating-point-hex-parser@1.11.6": 114 | version "1.11.6" 115 | resolved "https://registry.yarnpkg.com/@webassemblyjs/floating-point-hex-parser/-/floating-point-hex-parser-1.11.6.tgz#dacbcb95aff135c8260f77fa3b4c5fea600a6431" 116 | integrity sha512-ejAj9hfRJ2XMsNHk/v6Fu2dGS+i4UaXBXGemOfQ/JfQ6mdQg/WXtwleQRLLS4OvfDhv8rYnVwH27YJLMyYsxhw== 117 | 118 | "@webassemblyjs/helper-api-error@1.11.6": 119 | version "1.11.6" 120 | resolved "https://registry.yarnpkg.com/@webassemblyjs/helper-api-error/-/helper-api-error-1.11.6.tgz#6132f68c4acd59dcd141c44b18cbebbd9f2fa768" 121 | integrity sha512-o0YkoP4pVu4rN8aTJgAyj9hC2Sv5UlkzCHhxqWj8butaLvnpdc2jOwh4ewE6CX0txSfLn/UYaV/pheS2Txg//Q== 122 | 123 | "@webassemblyjs/helper-buffer@1.11.6": 124 | version "1.11.6" 125 | resolved "https://registry.yarnpkg.com/@webassemblyjs/helper-buffer/-/helper-buffer-1.11.6.tgz#b66d73c43e296fd5e88006f18524feb0f2c7c093" 126 | integrity sha512-z3nFzdcp1mb8nEOFFk8DrYLpHvhKC3grJD2ardfKOzmbmJvEf/tPIqCY+sNcwZIY8ZD7IkB2l7/pqhUhqm7hLA== 127 | 128 | "@webassemblyjs/helper-numbers@1.11.6": 129 | version "1.11.6" 130 | resolved "https://registry.yarnpkg.com/@webassemblyjs/helper-numbers/-/helper-numbers-1.11.6.tgz#cbce5e7e0c1bd32cf4905ae444ef64cea919f1b5" 131 | integrity sha512-vUIhZ8LZoIWHBohiEObxVm6hwP034jwmc9kuq5GdHZH0wiLVLIPcMCdpJzG4C11cHoQ25TFIQj9kaVADVX7N3g== 132 | dependencies: 133 | "@webassemblyjs/floating-point-hex-parser" "1.11.6" 134 | "@webassemblyjs/helper-api-error" "1.11.6" 135 | "@xtuc/long" "4.2.2" 136 | 137 | "@webassemblyjs/helper-wasm-bytecode@1.11.6": 138 | version "1.11.6" 139 | resolved "https://registry.yarnpkg.com/@webassemblyjs/helper-wasm-bytecode/-/helper-wasm-bytecode-1.11.6.tgz#bb2ebdb3b83aa26d9baad4c46d4315283acd51e9" 140 | integrity sha512-sFFHKwcmBprO9e7Icf0+gddyWYDViL8bpPjJJl0WHxCdETktXdmtWLGVzoHbqUcY4Be1LkNfwTmXOJUFZYSJdA== 141 | 142 | "@webassemblyjs/helper-wasm-section@1.11.6": 143 | version "1.11.6" 144 | resolved "https://registry.yarnpkg.com/@webassemblyjs/helper-wasm-section/-/helper-wasm-section-1.11.6.tgz#ff97f3863c55ee7f580fd5c41a381e9def4aa577" 145 | integrity sha512-LPpZbSOwTpEC2cgn4hTydySy1Ke+XEu+ETXuoyvuyezHO3Kjdu90KK95Sh9xTbmjrCsUwvWwCOQQNta37VrS9g== 146 | dependencies: 147 | "@webassemblyjs/ast" "1.11.6" 148 | "@webassemblyjs/helper-buffer" "1.11.6" 149 | "@webassemblyjs/helper-wasm-bytecode" "1.11.6" 150 | "@webassemblyjs/wasm-gen" "1.11.6" 151 | 152 | "@webassemblyjs/ieee754@1.11.6": 153 | version "1.11.6" 154 | resolved "https://registry.yarnpkg.com/@webassemblyjs/ieee754/-/ieee754-1.11.6.tgz#bb665c91d0b14fffceb0e38298c329af043c6e3a" 155 | integrity sha512-LM4p2csPNvbij6U1f19v6WR56QZ8JcHg3QIJTlSwzFcmx6WSORicYj6I63f9yU1kEUtrpG+kjkiIAkevHpDXrg== 156 | dependencies: 157 | "@xtuc/ieee754" "^1.2.0" 158 | 159 | "@webassemblyjs/leb128@1.11.6": 160 | version "1.11.6" 161 | resolved "https://registry.yarnpkg.com/@webassemblyjs/leb128/-/leb128-1.11.6.tgz#70e60e5e82f9ac81118bc25381a0b283893240d7" 162 | integrity sha512-m7a0FhE67DQXgouf1tbN5XQcdWoNgaAuoULHIfGFIEVKA6tu/edls6XnIlkmS6FrXAquJRPni3ZZKjw6FSPjPQ== 163 | dependencies: 164 | "@xtuc/long" "4.2.2" 165 | 166 | "@webassemblyjs/utf8@1.11.6": 167 | version "1.11.6" 168 | resolved "https://registry.yarnpkg.com/@webassemblyjs/utf8/-/utf8-1.11.6.tgz#90f8bc34c561595fe156603be7253cdbcd0fab5a" 169 | integrity sha512-vtXf2wTQ3+up9Zsg8sa2yWiQpzSsMyXj0qViVP6xKGCUT8p8YJ6HqI7l5eCnWx1T/FYdsv07HQs2wTFbbof/RA== 170 | 171 | "@webassemblyjs/wasm-edit@^1.11.5": 172 | version "1.11.6" 173 | resolved "https://registry.yarnpkg.com/@webassemblyjs/wasm-edit/-/wasm-edit-1.11.6.tgz#c72fa8220524c9b416249f3d94c2958dfe70ceab" 174 | integrity sha512-Ybn2I6fnfIGuCR+Faaz7YcvtBKxvoLV3Lebn1tM4o/IAJzmi9AWYIPWpyBfU8cC+JxAO57bk4+zdsTjJR+VTOw== 175 | dependencies: 176 | "@webassemblyjs/ast" "1.11.6" 177 | "@webassemblyjs/helper-buffer" "1.11.6" 178 | "@webassemblyjs/helper-wasm-bytecode" "1.11.6" 179 | "@webassemblyjs/helper-wasm-section" "1.11.6" 180 | "@webassemblyjs/wasm-gen" "1.11.6" 181 | "@webassemblyjs/wasm-opt" "1.11.6" 182 | "@webassemblyjs/wasm-parser" "1.11.6" 183 | "@webassemblyjs/wast-printer" "1.11.6" 184 | 185 | "@webassemblyjs/wasm-gen@1.11.6": 186 | version "1.11.6" 187 | resolved "https://registry.yarnpkg.com/@webassemblyjs/wasm-gen/-/wasm-gen-1.11.6.tgz#fb5283e0e8b4551cc4e9c3c0d7184a65faf7c268" 188 | integrity sha512-3XOqkZP/y6B4F0PBAXvI1/bky7GryoogUtfwExeP/v7Nzwo1QLcq5oQmpKlftZLbT+ERUOAZVQjuNVak6UXjPA== 189 | dependencies: 190 | "@webassemblyjs/ast" "1.11.6" 191 | "@webassemblyjs/helper-wasm-bytecode" "1.11.6" 192 | "@webassemblyjs/ieee754" "1.11.6" 193 | "@webassemblyjs/leb128" "1.11.6" 194 | "@webassemblyjs/utf8" "1.11.6" 195 | 196 | "@webassemblyjs/wasm-opt@1.11.6": 197 | version "1.11.6" 198 | resolved "https://registry.yarnpkg.com/@webassemblyjs/wasm-opt/-/wasm-opt-1.11.6.tgz#d9a22d651248422ca498b09aa3232a81041487c2" 199 | integrity sha512-cOrKuLRE7PCe6AsOVl7WasYf3wbSo4CeOk6PkrjS7g57MFfVUF9u6ysQBBODX0LdgSvQqRiGz3CXvIDKcPNy4g== 200 | dependencies: 201 | "@webassemblyjs/ast" "1.11.6" 202 | "@webassemblyjs/helper-buffer" "1.11.6" 203 | "@webassemblyjs/wasm-gen" "1.11.6" 204 | "@webassemblyjs/wasm-parser" "1.11.6" 205 | 206 | "@webassemblyjs/wasm-parser@1.11.6", "@webassemblyjs/wasm-parser@^1.11.5": 207 | version "1.11.6" 208 | resolved "https://registry.yarnpkg.com/@webassemblyjs/wasm-parser/-/wasm-parser-1.11.6.tgz#bb85378c527df824004812bbdb784eea539174a1" 209 | integrity sha512-6ZwPeGzMJM3Dqp3hCsLgESxBGtT/OeCvCZ4TA1JUPYgmhAx38tTPR9JaKy0S5H3evQpO/h2uWs2j6Yc/fjkpTQ== 210 | dependencies: 211 | "@webassemblyjs/ast" "1.11.6" 212 | "@webassemblyjs/helper-api-error" "1.11.6" 213 | "@webassemblyjs/helper-wasm-bytecode" "1.11.6" 214 | "@webassemblyjs/ieee754" "1.11.6" 215 | "@webassemblyjs/leb128" "1.11.6" 216 | "@webassemblyjs/utf8" "1.11.6" 217 | 218 | "@webassemblyjs/wast-printer@1.11.6": 219 | version "1.11.6" 220 | resolved "https://registry.yarnpkg.com/@webassemblyjs/wast-printer/-/wast-printer-1.11.6.tgz#a7bf8dd7e362aeb1668ff43f35cb849f188eff20" 221 | integrity sha512-JM7AhRcE+yW2GWYaKeHL5vt4xqee5N2WcezptmgyhNS+ScggqcT1OtXykhAb13Sn5Yas0j2uv9tHgrjwvzAP4A== 222 | dependencies: 223 | "@webassemblyjs/ast" "1.11.6" 224 | "@xtuc/long" "4.2.2" 225 | 226 | "@webpack-cli/configtest@^2.1.0": 227 | version "2.1.1" 228 | resolved "https://registry.yarnpkg.com/@webpack-cli/configtest/-/configtest-2.1.1.tgz#3b2f852e91dac6e3b85fb2a314fb8bef46d94646" 229 | integrity sha512-wy0mglZpDSiSS0XHrVR+BAdId2+yxPSoJW8fsna3ZpYSlufjvxnP4YbKTCBZnNIcGN4r6ZPXV55X4mYExOfLmw== 230 | 231 | "@webpack-cli/info@^2.0.1": 232 | version "2.0.2" 233 | resolved "https://registry.yarnpkg.com/@webpack-cli/info/-/info-2.0.2.tgz#cc3fbf22efeb88ff62310cf885c5b09f44ae0fdd" 234 | integrity sha512-zLHQdI/Qs1UyT5UBdWNqsARasIA+AaF8t+4u2aS2nEpBQh2mWIVb8qAklq0eUENnC5mOItrIB4LiS9xMtph18A== 235 | 236 | "@webpack-cli/serve@^2.0.3": 237 | version "2.0.5" 238 | resolved "https://registry.yarnpkg.com/@webpack-cli/serve/-/serve-2.0.5.tgz#325db42395cd49fe6c14057f9a900e427df8810e" 239 | integrity sha512-lqaoKnRYBdo1UgDX8uF24AfGMifWK19TxPmM5FHc2vAGxrJ/qtyUyFBWoY1tISZdelsQ5fBcOusifo5o5wSJxQ== 240 | 241 | "@xtuc/ieee754@^1.2.0": 242 | version "1.2.0" 243 | resolved "https://registry.yarnpkg.com/@xtuc/ieee754/-/ieee754-1.2.0.tgz#eef014a3145ae477a1cbc00cd1e552336dceb790" 244 | integrity sha512-DX8nKgqcGwsc0eJSqYt5lwP4DH5FlHnmuWWBRy7X0NcaGR0ZtuyeESgMwTYVEtxmsNGY+qit4QYT/MIYTOTPeA== 245 | 246 | "@xtuc/long@4.2.2": 247 | version "4.2.2" 248 | resolved "https://registry.yarnpkg.com/@xtuc/long/-/long-4.2.2.tgz#d291c6a4e97989b5c61d9acf396ae4fe133a718d" 249 | integrity sha512-NuHqBY1PB/D8xU6s/thBgOAiAP7HOYDQ32+BFZILJ8ivkUkAHQnWfn6WhL79Owj1qmUnoN/YPhktdIoucipkAQ== 250 | 251 | abab@^2.0.6: 252 | version "2.0.6" 253 | resolved "https://registry.yarnpkg.com/abab/-/abab-2.0.6.tgz#41b80f2c871d19686216b82309231cfd3cb3d291" 254 | integrity sha512-j2afSsaIENvHZN2B8GOpF566vZ5WVk5opAiMTvWgaQT8DkbOqsTfvNAvHoRGU2zzP8cPoqys+xHTRDWW8L+/BA== 255 | 256 | accepts@~1.3.4: 257 | version "1.3.8" 258 | resolved "https://registry.yarnpkg.com/accepts/-/accepts-1.3.8.tgz#0bf0be125b67014adcb0b0921e62db7bffe16b2e" 259 | integrity sha512-PYAthTa2m2VKxuvSD3DPC/Gy+U+sOA1LAuT8mkmRuvw+NACSaeXEQ+NHcVF7rONl6qcaxV3Uuemwawk+7+SJLw== 260 | dependencies: 261 | mime-types "~2.1.34" 262 | negotiator "0.6.3" 263 | 264 | acorn-import-assertions@^1.7.6: 265 | version "1.9.0" 266 | resolved "https://registry.yarnpkg.com/acorn-import-assertions/-/acorn-import-assertions-1.9.0.tgz#507276249d684797c84e0734ef84860334cfb1ac" 267 | integrity sha512-cmMwop9x+8KFhxvKrKfPYmN6/pKTYYHBqLa0DfvVZcKMJWNyWLnaqND7dx/qn66R7ewM1UX5XMaDVP5wlVTaVA== 268 | 269 | acorn@^8.7.1, acorn@^8.8.2: 270 | version "8.11.3" 271 | resolved "https://registry.yarnpkg.com/acorn/-/acorn-8.11.3.tgz#71e0b14e13a4ec160724b38fb7b0f233b1b81d7a" 272 | integrity sha512-Y9rRfJG5jcKOE0CLisYbojUjIrIEE7AGMzA/Sm4BslANhbS+cDMpgBdcPT91oJ7OuJ9hYJBx59RjbhxVnrF8Xg== 273 | 274 | ajv-keywords@^3.5.2: 275 | version "3.5.2" 276 | resolved "https://registry.yarnpkg.com/ajv-keywords/-/ajv-keywords-3.5.2.tgz#31f29da5ab6e00d1c2d329acf7b5929614d5014d" 277 | integrity sha512-5p6WTN0DdTGVQk6VjcEju19IgaHudalcfabD7yhDGeA6bcQnmL+CpveLJq/3hvfwd1aof6L386Ougkx6RfyMIQ== 278 | 279 | ajv@^6.12.5: 280 | version "6.12.6" 281 | resolved "https://registry.yarnpkg.com/ajv/-/ajv-6.12.6.tgz#baf5a62e802b07d977034586f8c3baf5adf26df4" 282 | integrity sha512-j3fVLgvTo527anyYyJOGTYJbG+vnnQYvE0m5mmkc1TK+nxAppkCLMIL0aZ4dblVCNoGShhm+kzE4ZUykBoMg4g== 283 | dependencies: 284 | fast-deep-equal "^3.1.1" 285 | fast-json-stable-stringify "^2.0.0" 286 | json-schema-traverse "^0.4.1" 287 | uri-js "^4.2.2" 288 | 289 | ansi-colors@4.1.1: 290 | version "4.1.1" 291 | resolved "https://registry.yarnpkg.com/ansi-colors/-/ansi-colors-4.1.1.tgz#cbb9ae256bf750af1eab344f229aa27fe94ba348" 292 | integrity sha512-JoX0apGbHaUJBNl6yF+p6JAFYZ666/hhCGKN5t9QFjbJQKUU/g8MNbFDbvfrgKXvI1QpZplPOnwIo99lX/AAmA== 293 | 294 | ansi-regex@^5.0.1: 295 | version "5.0.1" 296 | resolved "https://registry.yarnpkg.com/ansi-regex/-/ansi-regex-5.0.1.tgz#082cb2c89c9fe8659a311a53bd6a4dc5301db304" 297 | integrity sha512-quJQXlTSUGL2LH9SUXo8VwsY4soanhgo6LNSm84E1LBcE8s3O0wpdiRzyR9z/ZZJMlMWv37qOOb9pdJlMUEKFQ== 298 | 299 | ansi-styles@^4.0.0, ansi-styles@^4.1.0: 300 | version "4.3.0" 301 | resolved "https://registry.yarnpkg.com/ansi-styles/-/ansi-styles-4.3.0.tgz#edd803628ae71c04c85ae7a0906edad34b648937" 302 | integrity sha512-zbB9rCJAT1rbjiVDb2hqKFHNYLxgtk8NURxZ3IZwD3F6NtxbXZQCnnSi1Lkx+IDohdPlFp222wVALIheZJQSEg== 303 | dependencies: 304 | color-convert "^2.0.1" 305 | 306 | anymatch@~3.1.2: 307 | version "3.1.3" 308 | resolved "https://registry.yarnpkg.com/anymatch/-/anymatch-3.1.3.tgz#790c58b19ba1720a84205b57c618d5ad8524973e" 309 | integrity sha512-KMReFUr0B4t+D+OBkjR3KYqvocp2XaSzO55UcB6mgQMd3KbcE+mWTyvVV7D/zsdEbNnV6acZUutkiHQXvTr1Rw== 310 | dependencies: 311 | normalize-path "^3.0.0" 312 | picomatch "^2.0.4" 313 | 314 | argparse@^2.0.1: 315 | version "2.0.1" 316 | resolved "https://registry.yarnpkg.com/argparse/-/argparse-2.0.1.tgz#246f50f3ca78a3240f6c997e8a9bd1eac49e4b38" 317 | integrity sha512-8+9WqebbFzpX9OR+Wa6O29asIogeRMzcGtAINdpMHHyAg10f05aSFVBbcEqGf/PXw1EjAZ+q2/bEBg3DvurK3Q== 318 | 319 | balanced-match@^1.0.0: 320 | version "1.0.2" 321 | resolved "https://registry.yarnpkg.com/balanced-match/-/balanced-match-1.0.2.tgz#e83e3a7e3f300b34cb9d87f615fa0cbf357690ee" 322 | integrity sha512-3oSeUO0TMV67hN1AmbXsK4yaqU7tjiHlbxRDZOpH0KW9+CeX4bRAaX0Anxt0tx2MrpRpWwQaPwIlISEJhYU5Pw== 323 | 324 | base64id@2.0.0, base64id@~2.0.0: 325 | version "2.0.0" 326 | resolved "https://registry.yarnpkg.com/base64id/-/base64id-2.0.0.tgz#2770ac6bc47d312af97a8bf9a634342e0cd25cb6" 327 | integrity sha512-lGe34o6EHj9y3Kts9R4ZYs/Gr+6N7MCaMlIFA3F1R2O5/m7K06AxfSeO5530PEERE6/WyEg3lsuyw4GHlPZHog== 328 | 329 | binary-extensions@^2.0.0: 330 | version "2.2.0" 331 | resolved "https://registry.yarnpkg.com/binary-extensions/-/binary-extensions-2.2.0.tgz#75f502eeaf9ffde42fc98829645be4ea76bd9e2d" 332 | integrity sha512-jDctJ/IVQbZoJykoeHbhXpOlNBqGNcwXJKJog42E5HDPUwQTSdjCHdihjj0DlnheQ7blbT6dHOafNAiS8ooQKA== 333 | 334 | body-parser@^1.19.0: 335 | version "1.20.2" 336 | resolved "https://registry.yarnpkg.com/body-parser/-/body-parser-1.20.2.tgz#6feb0e21c4724d06de7ff38da36dad4f57a747fd" 337 | integrity sha512-ml9pReCu3M61kGlqoTm2umSXTlRTuGTx0bfYj+uIUKKYycG5NtSbeetV3faSU6R7ajOPw0g/J1PvK4qNy7s5bA== 338 | dependencies: 339 | bytes "3.1.2" 340 | content-type "~1.0.5" 341 | debug "2.6.9" 342 | depd "2.0.0" 343 | destroy "1.2.0" 344 | http-errors "2.0.0" 345 | iconv-lite "0.4.24" 346 | on-finished "2.4.1" 347 | qs "6.11.0" 348 | raw-body "2.5.2" 349 | type-is "~1.6.18" 350 | unpipe "1.0.0" 351 | 352 | brace-expansion@^1.1.7: 353 | version "1.1.11" 354 | resolved "https://registry.yarnpkg.com/brace-expansion/-/brace-expansion-1.1.11.tgz#3c7fcbf529d87226f3d2f52b966ff5271eb441dd" 355 | integrity sha512-iCuPHDFgrHX7H2vEI/5xpz07zSHB00TpugqhmYtVmMO6518mCuRMoOYFldEBl0g187ufozdaHgWKcYFb61qGiA== 356 | dependencies: 357 | balanced-match "^1.0.0" 358 | concat-map "0.0.1" 359 | 360 | brace-expansion@^2.0.1: 361 | version "2.0.1" 362 | resolved "https://registry.yarnpkg.com/brace-expansion/-/brace-expansion-2.0.1.tgz#1edc459e0f0c548486ecf9fc99f2221364b9a0ae" 363 | integrity sha512-XnAIvQ8eM+kC6aULx6wuQiwVsnzsi9d3WxzV3FpWTGA19F621kwdbsAcFKXgKUHZWsy+mY6iL1sHTxWEFCytDA== 364 | dependencies: 365 | balanced-match "^1.0.0" 366 | 367 | braces@^3.0.2, braces@~3.0.2: 368 | version "3.0.2" 369 | resolved "https://registry.yarnpkg.com/braces/-/braces-3.0.2.tgz#3454e1a462ee8d599e236df336cd9ea4f8afe107" 370 | integrity sha512-b8um+L1RzM3WDSzvhm6gIz1yfTbBt6YTlcEKAvsmqCZZFw46z626lVj9j1yEPW33H5H+lBQpZMP1k8l+78Ha0A== 371 | dependencies: 372 | fill-range "^7.0.1" 373 | 374 | browser-stdout@1.3.1: 375 | version "1.3.1" 376 | resolved "https://registry.yarnpkg.com/browser-stdout/-/browser-stdout-1.3.1.tgz#baa559ee14ced73452229bad7326467c61fabd60" 377 | integrity sha512-qhAVI1+Av2X7qelOfAIYwXONood6XlZE/fXaBSmW/T5SzLAmCgzi+eiWE7fUvbHaeNBQH13UftjpXxsfLkMpgw== 378 | 379 | browserslist@^4.14.5: 380 | version "4.22.3" 381 | resolved "https://registry.yarnpkg.com/browserslist/-/browserslist-4.22.3.tgz#299d11b7e947a6b843981392721169e27d60c5a6" 382 | integrity sha512-UAp55yfwNv0klWNapjs/ktHoguxuQNGnOzxYmfnXIS+8AsRDZkSDxg7R1AX3GKzn078SBI5dzwzj/Yx0Or0e3A== 383 | dependencies: 384 | caniuse-lite "^1.0.30001580" 385 | electron-to-chromium "^1.4.648" 386 | node-releases "^2.0.14" 387 | update-browserslist-db "^1.0.13" 388 | 389 | buffer-from@^1.0.0: 390 | version "1.1.2" 391 | resolved "https://registry.yarnpkg.com/buffer-from/-/buffer-from-1.1.2.tgz#2b146a6fd72e80b4f55d255f35ed59a3a9a41bd5" 392 | integrity sha512-E+XQCRwSbaaiChtv6k6Dwgc+bx+Bs6vuKJHHl5kox/BaKbhiXzqQOwK4cO22yElGp2OCmjwVhT3HmxgyPGnJfQ== 393 | 394 | bytes@3.1.2: 395 | version "3.1.2" 396 | resolved "https://registry.yarnpkg.com/bytes/-/bytes-3.1.2.tgz#8b0beeb98605adf1b128fa4386403c009e0221a5" 397 | integrity sha512-/Nf7TyzTx6S3yRJObOAV7956r8cr2+Oj8AC5dt8wSP3BQAoeX58NoHyCU8P8zGkNXStjTSi6fzO6F0pBdcYbEg== 398 | 399 | call-bind@^1.0.0: 400 | version "1.0.5" 401 | resolved "https://registry.yarnpkg.com/call-bind/-/call-bind-1.0.5.tgz#6fa2b7845ce0ea49bf4d8b9ef64727a2c2e2e513" 402 | integrity sha512-C3nQxfFZxFRVoJoGKKI8y3MOEo129NQ+FgQ08iye+Mk4zNZZGdjfs06bVTr+DBSlA66Q2VEcMki/cUCP4SercQ== 403 | dependencies: 404 | function-bind "^1.1.2" 405 | get-intrinsic "^1.2.1" 406 | set-function-length "^1.1.1" 407 | 408 | camelcase@^6.0.0: 409 | version "6.3.0" 410 | resolved "https://registry.yarnpkg.com/camelcase/-/camelcase-6.3.0.tgz#5685b95eb209ac9c0c177467778c9c84df58ba9a" 411 | integrity sha512-Gmy6FhYlCY7uOElZUSbxo2UCDH8owEk996gkbrpsgGtrJLM3J7jGxl9Ic7Qwwj4ivOE5AWZWRMecDdF7hqGjFA== 412 | 413 | caniuse-lite@^1.0.30001580: 414 | version "1.0.30001581" 415 | resolved "https://registry.yarnpkg.com/caniuse-lite/-/caniuse-lite-1.0.30001581.tgz#0dfd4db9e94edbdca67d57348ebc070dece279f4" 416 | integrity sha512-whlTkwhqV2tUmP3oYhtNfaWGYHDdS3JYFQBKXxcUR9qqPWsRhFHhoISO2Xnl/g0xyKzht9mI1LZpiNWfMzHixQ== 417 | 418 | chalk@^4.1.0: 419 | version "4.1.2" 420 | resolved "https://registry.yarnpkg.com/chalk/-/chalk-4.1.2.tgz#aac4e2b7734a740867aeb16bf02aad556a1e7a01" 421 | integrity sha512-oKnbhFyRIXpUuez8iBMmyEa4nbj4IOQyuhc/wy9kY7/WVPcwIO9VA668Pu8RkO7+0G76SLROeyw9CpQ061i4mA== 422 | dependencies: 423 | ansi-styles "^4.1.0" 424 | supports-color "^7.1.0" 425 | 426 | chokidar@3.5.3, chokidar@^3.5.1: 427 | version "3.5.3" 428 | resolved "https://registry.yarnpkg.com/chokidar/-/chokidar-3.5.3.tgz#1cf37c8707b932bd1af1ae22c0432e2acd1903bd" 429 | integrity sha512-Dr3sfKRP6oTcjf2JmUmFJfeVMvXBdegxB0iVQ5eb2V10uFJUCAS8OByZdVAyVb8xXNz3GjjTgj9kLWsZTqE6kw== 430 | dependencies: 431 | anymatch "~3.1.2" 432 | braces "~3.0.2" 433 | glob-parent "~5.1.2" 434 | is-binary-path "~2.1.0" 435 | is-glob "~4.0.1" 436 | normalize-path "~3.0.0" 437 | readdirp "~3.6.0" 438 | optionalDependencies: 439 | fsevents "~2.3.2" 440 | 441 | chrome-trace-event@^1.0.2: 442 | version "1.0.3" 443 | resolved "https://registry.yarnpkg.com/chrome-trace-event/-/chrome-trace-event-1.0.3.tgz#1015eced4741e15d06664a957dbbf50d041e26ac" 444 | integrity sha512-p3KULyQg4S7NIHixdwbGX+nFHkoBiA4YQmyWtjb8XngSKV124nJmRysgAeujbUVb15vh+RvFUfCPqU7rXk+hZg== 445 | 446 | cliui@^7.0.2: 447 | version "7.0.4" 448 | resolved "https://registry.yarnpkg.com/cliui/-/cliui-7.0.4.tgz#a0265ee655476fc807aea9df3df8df7783808b4f" 449 | integrity sha512-OcRE68cOsVMXp1Yvonl/fzkQOyjLSu/8bhPDfQt0e0/Eb283TKP20Fs2MqoPsr9SwA595rRCA+QMzYc9nBP+JQ== 450 | dependencies: 451 | string-width "^4.2.0" 452 | strip-ansi "^6.0.0" 453 | wrap-ansi "^7.0.0" 454 | 455 | clone-deep@^4.0.1: 456 | version "4.0.1" 457 | resolved "https://registry.yarnpkg.com/clone-deep/-/clone-deep-4.0.1.tgz#c19fd9bdbbf85942b4fd979c84dcf7d5f07c2387" 458 | integrity sha512-neHB9xuzh/wk0dIHweyAXv2aPGZIVk3pLMe+/RNzINf17fe0OG96QroktYAUm7SM1PBnzTabaLboqqxDyMU+SQ== 459 | dependencies: 460 | is-plain-object "^2.0.4" 461 | kind-of "^6.0.2" 462 | shallow-clone "^3.0.0" 463 | 464 | color-convert@^2.0.1: 465 | version "2.0.1" 466 | resolved "https://registry.yarnpkg.com/color-convert/-/color-convert-2.0.1.tgz#72d3a68d598c9bdb3af2ad1e84f21d896abd4de3" 467 | integrity sha512-RRECPsj7iu/xb5oKYcsFHSppFNnsj/52OVTRKb4zP5onXwVF3zVmmToNcOfGC+CRDpfK/U584fMg38ZHCaElKQ== 468 | dependencies: 469 | color-name "~1.1.4" 470 | 471 | color-name@~1.1.4: 472 | version "1.1.4" 473 | resolved "https://registry.yarnpkg.com/color-name/-/color-name-1.1.4.tgz#c2a09a87acbde69543de6f63fa3995c826c536a2" 474 | integrity sha512-dOy+3AuW3a2wNbZHIuMZpTcgjGuLU/uBL/ubcZF9OXbDo8ff4O8yVp5Bf0efS8uEoYo5q4Fx7dY9OgQGXgAsQA== 475 | 476 | colorette@^2.0.14: 477 | version "2.0.20" 478 | resolved "https://registry.yarnpkg.com/colorette/-/colorette-2.0.20.tgz#9eb793e6833067f7235902fcd3b09917a000a95a" 479 | integrity sha512-IfEDxwoWIjkeXL1eXcDiow4UbKjhLdq6/EuSVR9GMN7KVH3r9gQ83e73hsz1Nd1T3ijd5xv1wcWRYO+D6kCI2w== 480 | 481 | commander@^10.0.1: 482 | version "10.0.1" 483 | resolved "https://registry.yarnpkg.com/commander/-/commander-10.0.1.tgz#881ee46b4f77d1c1dccc5823433aa39b022cbe06" 484 | integrity sha512-y4Mg2tXshplEbSGzx7amzPwKKOCGuoSRP/CjEdwwk0FOGlUbq6lKuoyDZTNZkmxHdJtp54hdfY/JUrdL7Xfdug== 485 | 486 | commander@^2.20.0: 487 | version "2.20.3" 488 | resolved "https://registry.yarnpkg.com/commander/-/commander-2.20.3.tgz#fd485e84c03eb4881c20722ba48035e8531aeb33" 489 | integrity sha512-GpVkmM8vF2vQUkj2LvZmD35JxeJOLCwJ9cUkugyk2nuhbv3+mJvpLYYt+0+USMxE+oj+ey/lJEnhZw75x/OMcQ== 490 | 491 | concat-map@0.0.1: 492 | version "0.0.1" 493 | resolved "https://registry.yarnpkg.com/concat-map/-/concat-map-0.0.1.tgz#d8a96bd77fd68df7793a73036a3ba0d5405d477b" 494 | integrity sha512-/Srv4dswyQNBfohGpz9o6Yb3Gz3SrUDqBH5rTuhGR7ahtlbYKnVxw2bCFMRljaA7EXHaXZ8wsHdodFvbkhKmqg== 495 | 496 | connect@^3.7.0: 497 | version "3.7.0" 498 | resolved "https://registry.yarnpkg.com/connect/-/connect-3.7.0.tgz#5d49348910caa5e07a01800b030d0c35f20484f8" 499 | integrity sha512-ZqRXc+tZukToSNmh5C2iWMSoV3X1YUcPbqEM4DkEG5tNQXrQUZCNVGGv3IuicnkMtPfGf3Xtp8WCXs295iQ1pQ== 500 | dependencies: 501 | debug "2.6.9" 502 | finalhandler "1.1.2" 503 | parseurl "~1.3.3" 504 | utils-merge "1.0.1" 505 | 506 | content-type@~1.0.5: 507 | version "1.0.5" 508 | resolved "https://registry.yarnpkg.com/content-type/-/content-type-1.0.5.tgz#8b773162656d1d1086784c8f23a54ce6d73d7918" 509 | integrity sha512-nTjqfcBFEipKdXCv4YDQWCfmcLZKm81ldF0pAopTvyrFGVbcR6P/VAAd5G7N+0tTr8QqiU0tFadD6FK4NtJwOA== 510 | 511 | cookie@~0.4.1: 512 | version "0.4.2" 513 | resolved "https://registry.yarnpkg.com/cookie/-/cookie-0.4.2.tgz#0e41f24de5ecf317947c82fc789e06a884824432" 514 | integrity sha512-aSWTXFzaKWkvHO1Ny/s+ePFpvKsPnjc551iI41v3ny/ow6tBG5Vd+FuqGNhh1LxOmVzOlGUriIlOaokOvhaStA== 515 | 516 | cors@~2.8.5: 517 | version "2.8.5" 518 | resolved "https://registry.yarnpkg.com/cors/-/cors-2.8.5.tgz#eac11da51592dd86b9f06f6e7ac293b3df875d29" 519 | integrity sha512-KIHbLJqu73RGr/hnbrO9uBeixNGuvSQjul/jdFvS/KFSIH1hWVd1ng7zOHx+YrEfInLG7q4n6GHQ9cDtxv/P6g== 520 | dependencies: 521 | object-assign "^4" 522 | vary "^1" 523 | 524 | cross-spawn@^7.0.3: 525 | version "7.0.3" 526 | resolved "https://registry.yarnpkg.com/cross-spawn/-/cross-spawn-7.0.3.tgz#f73a85b9d5d41d045551c177e2882d4ac85728a6" 527 | integrity sha512-iRDPJKUPVEND7dHPO8rkbOnPpyDygcDFtWjpeWNCgy8WP2rXcxXL8TskReQl6OrB2G7+UJrags1q15Fudc7G6w== 528 | dependencies: 529 | path-key "^3.1.0" 530 | shebang-command "^2.0.0" 531 | which "^2.0.1" 532 | 533 | custom-event@~1.0.0: 534 | version "1.0.1" 535 | resolved "https://registry.yarnpkg.com/custom-event/-/custom-event-1.0.1.tgz#5d02a46850adf1b4a317946a3928fccb5bfd0425" 536 | integrity sha512-GAj5FOq0Hd+RsCGVJxZuKaIDXDf3h6GQoNEjFgbLLI/trgtavwUbSnZ5pVfg27DVCaWjIohryS0JFwIJyT2cMg== 537 | 538 | date-format@^4.0.14: 539 | version "4.0.14" 540 | resolved "https://registry.yarnpkg.com/date-format/-/date-format-4.0.14.tgz#7a8e584434fb169a521c8b7aa481f355810d9400" 541 | integrity sha512-39BOQLs9ZjKh0/patS9nrT8wc3ioX3/eA/zgbKNopnF2wCqJEoxywwwElATYvRsXdnOxA/OQeQoFZ3rFjVajhg== 542 | 543 | debug@2.6.9: 544 | version "2.6.9" 545 | resolved "https://registry.yarnpkg.com/debug/-/debug-2.6.9.tgz#5d128515df134ff327e90a4c93f4e077a536341f" 546 | integrity sha512-bC7ElrdJaJnPbAP+1EotYvqZsb3ecl5wi6Bfi6BJTUcNowp6cvspg0jXznRTKDjm/E7AdgFBVeAPVMNcKGsHMA== 547 | dependencies: 548 | ms "2.0.0" 549 | 550 | debug@4.3.4, debug@^4.3.4, debug@~4.3.1, debug@~4.3.2: 551 | version "4.3.4" 552 | resolved "https://registry.yarnpkg.com/debug/-/debug-4.3.4.tgz#1319f6579357f2338d3337d2cdd4914bb5dcc865" 553 | integrity sha512-PRWFHuSU3eDtQJPvnNY7Jcket1j0t5OuOsFzPPzsekD52Zl8qUfFIPEiswXqIvHWGVHOgX+7G/vCNNhehwxfkQ== 554 | dependencies: 555 | ms "2.1.2" 556 | 557 | decamelize@^4.0.0: 558 | version "4.0.0" 559 | resolved "https://registry.yarnpkg.com/decamelize/-/decamelize-4.0.0.tgz#aa472d7bf660eb15f3494efd531cab7f2a709837" 560 | integrity sha512-9iE1PgSik9HeIIw2JO94IidnE3eBoQrFJ3w7sFuzSX4DpmZ3v5sZpUiV5Swcf6mQEF+Y0ru8Neo+p+nyh2J+hQ== 561 | 562 | define-data-property@^1.1.1: 563 | version "1.1.1" 564 | resolved "https://registry.yarnpkg.com/define-data-property/-/define-data-property-1.1.1.tgz#c35f7cd0ab09883480d12ac5cb213715587800b3" 565 | integrity sha512-E7uGkTzkk1d0ByLeSc6ZsFS79Axg+m1P/VsgYsxHgiuc3tFSj+MjMIwe90FC4lOAZzNBdY7kkO2P2wKdsQ1vgQ== 566 | dependencies: 567 | get-intrinsic "^1.2.1" 568 | gopd "^1.0.1" 569 | has-property-descriptors "^1.0.0" 570 | 571 | depd@2.0.0: 572 | version "2.0.0" 573 | resolved "https://registry.yarnpkg.com/depd/-/depd-2.0.0.tgz#b696163cc757560d09cf22cc8fad1571b79e76df" 574 | integrity sha512-g7nH6P6dyDioJogAAGprGpCtVImJhpPk/roCzdb3fIh61/s/nPsfR6onyMwkCAR/OlC3yBC0lESvUoQEAssIrw== 575 | 576 | destroy@1.2.0: 577 | version "1.2.0" 578 | resolved "https://registry.yarnpkg.com/destroy/-/destroy-1.2.0.tgz#4803735509ad8be552934c67df614f94e66fa015" 579 | integrity sha512-2sJGJTaXIIaR1w4iJSNoN0hnMY7Gpc/n8D4qSCJw8QqFWXf7cuAgnEHxBpweaVcPevC2l3KpjYCx3NypQQgaJg== 580 | 581 | di@^0.0.1: 582 | version "0.0.1" 583 | resolved "https://registry.yarnpkg.com/di/-/di-0.0.1.tgz#806649326ceaa7caa3306d75d985ea2748ba913c" 584 | integrity sha512-uJaamHkagcZtHPqCIHZxnFrXlunQXgBOsZSUOWwFw31QJCAbyTBoHMW75YOTur5ZNx8pIeAKgf6GWIgaqqiLhA== 585 | 586 | diff@5.0.0: 587 | version "5.0.0" 588 | resolved "https://registry.yarnpkg.com/diff/-/diff-5.0.0.tgz#7ed6ad76d859d030787ec35855f5b1daf31d852b" 589 | integrity sha512-/VTCrvm5Z0JGty/BWHljh+BAiw3IK+2j87NGMu8Nwc/f48WoDAC395uomO9ZD117ZOBaHmkX1oyLvkVM/aIT3w== 590 | 591 | dom-serialize@^2.2.1: 592 | version "2.2.1" 593 | resolved "https://registry.yarnpkg.com/dom-serialize/-/dom-serialize-2.2.1.tgz#562ae8999f44be5ea3076f5419dcd59eb43ac95b" 594 | integrity sha512-Yra4DbvoW7/Z6LBN560ZwXMjoNOSAN2wRsKFGc4iBeso+mpIA6qj1vfdf9HpMaKAqG6wXTy+1SYEzmNpKXOSsQ== 595 | dependencies: 596 | custom-event "~1.0.0" 597 | ent "~2.2.0" 598 | extend "^3.0.0" 599 | void-elements "^2.0.0" 600 | 601 | ee-first@1.1.1: 602 | version "1.1.1" 603 | resolved "https://registry.yarnpkg.com/ee-first/-/ee-first-1.1.1.tgz#590c61156b0ae2f4f0255732a158b266bc56b21d" 604 | integrity sha512-WMwm9LhRUo+WUaRN+vRuETqG89IgZphVSNkdFgeb6sS/E4OrDIN7t48CAewSHXc6C8lefD8KKfr5vY61brQlow== 605 | 606 | electron-to-chromium@^1.4.648: 607 | version "1.4.648" 608 | resolved "https://registry.yarnpkg.com/electron-to-chromium/-/electron-to-chromium-1.4.648.tgz#c7b46c9010752c37bb4322739d6d2dd82354fbe4" 609 | integrity sha512-EmFMarXeqJp9cUKu/QEciEApn0S/xRcpZWuAm32U7NgoZCimjsilKXHRO9saeEW55eHZagIDg6XTUOv32w9pjg== 610 | 611 | emoji-regex@^8.0.0: 612 | version "8.0.0" 613 | resolved "https://registry.yarnpkg.com/emoji-regex/-/emoji-regex-8.0.0.tgz#e818fd69ce5ccfcb404594f842963bf53164cc37" 614 | integrity sha512-MSjYzcWNOA0ewAHpz0MxpYFvwg6yjy1NG3xteoqz644VCo/RPgnr1/GGt+ic3iJTzQ8Eu3TdM14SawnVUmGE6A== 615 | 616 | encodeurl@~1.0.2: 617 | version "1.0.2" 618 | resolved "https://registry.yarnpkg.com/encodeurl/-/encodeurl-1.0.2.tgz#ad3ff4c86ec2d029322f5a02c3a9a606c95b3f59" 619 | integrity sha512-TPJXq8JqFaVYm2CWmPvnP2Iyo4ZSM7/QKcSmuMLDObfpH5fi7RUGmd/rTDf+rut/saiDiQEeVTNgAmJEdAOx0w== 620 | 621 | engine.io-parser@~5.2.1: 622 | version "5.2.1" 623 | resolved "https://registry.yarnpkg.com/engine.io-parser/-/engine.io-parser-5.2.1.tgz#9f213c77512ff1a6cc0c7a86108a7ffceb16fcfb" 624 | integrity sha512-9JktcM3u18nU9N2Lz3bWeBgxVgOKpw7yhRaoxQA3FUDZzzw+9WlA6p4G4u0RixNkg14fH7EfEc/RhpurtiROTQ== 625 | 626 | engine.io@~6.5.2: 627 | version "6.5.4" 628 | resolved "https://registry.yarnpkg.com/engine.io/-/engine.io-6.5.4.tgz#6822debf324e781add2254e912f8568508850cdc" 629 | integrity sha512-KdVSDKhVKyOi+r5uEabrDLZw2qXStVvCsEB/LN3mw4WFi6Gx50jTyuxYVCwAAC0U46FdnzP/ScKRBTXb/NiEOg== 630 | dependencies: 631 | "@types/cookie" "^0.4.1" 632 | "@types/cors" "^2.8.12" 633 | "@types/node" ">=10.0.0" 634 | accepts "~1.3.4" 635 | base64id "2.0.0" 636 | cookie "~0.4.1" 637 | cors "~2.8.5" 638 | debug "~4.3.1" 639 | engine.io-parser "~5.2.1" 640 | ws "~8.11.0" 641 | 642 | enhanced-resolve@^5.13.0: 643 | version "5.15.0" 644 | resolved "https://registry.yarnpkg.com/enhanced-resolve/-/enhanced-resolve-5.15.0.tgz#1af946c7d93603eb88e9896cee4904dc012e9c35" 645 | integrity sha512-LXYT42KJ7lpIKECr2mAXIaMldcNCh/7E0KBKOu4KSfkHmP+mZmSs+8V5gBAqisWBy0OO4W5Oyys0GO1Y8KtdKg== 646 | dependencies: 647 | graceful-fs "^4.2.4" 648 | tapable "^2.2.0" 649 | 650 | ent@~2.2.0: 651 | version "2.2.0" 652 | resolved "https://registry.yarnpkg.com/ent/-/ent-2.2.0.tgz#e964219325a21d05f44466a2f686ed6ce5f5dd1d" 653 | integrity sha512-GHrMyVZQWvTIdDtpiEXdHZnFQKzeO09apj8Cbl4pKWy4i0Oprcq17usfDt5aO63swf0JOeMWjWQE/LzgSRuWpA== 654 | 655 | envinfo@^7.7.3: 656 | version "7.11.0" 657 | resolved "https://registry.yarnpkg.com/envinfo/-/envinfo-7.11.0.tgz#c3793f44284a55ff8c82faf1ffd91bc6478ea01f" 658 | integrity sha512-G9/6xF1FPbIw0TtalAMaVPpiq2aDEuKLXM314jPVAO9r2fo2a4BLqMNkmRS7O/xPPZ+COAhGIz3ETvHEV3eUcg== 659 | 660 | es-module-lexer@^1.2.1: 661 | version "1.4.1" 662 | resolved "https://registry.yarnpkg.com/es-module-lexer/-/es-module-lexer-1.4.1.tgz#41ea21b43908fe6a287ffcbe4300f790555331f5" 663 | integrity sha512-cXLGjP0c4T3flZJKQSuziYoq7MlT+rnvfZjfp7h+I7K9BNX54kP9nyWvdbwjQ4u1iWbOL4u96fgeZLToQlZC7w== 664 | 665 | escalade@^3.1.1: 666 | version "3.1.1" 667 | resolved "https://registry.yarnpkg.com/escalade/-/escalade-3.1.1.tgz#d8cfdc7000965c5a0174b4a82eaa5c0552742e40" 668 | integrity sha512-k0er2gUkLf8O0zKJiAhmkTnJlTvINGv7ygDNPbeIsX/TJjGJZHuh9B2UxbsaEkmlEo9MfhrSzmhIlhRlI2GXnw== 669 | 670 | escape-html@~1.0.3: 671 | version "1.0.3" 672 | resolved "https://registry.yarnpkg.com/escape-html/-/escape-html-1.0.3.tgz#0258eae4d3d0c0974de1c169188ef0051d1d1988" 673 | integrity sha512-NiSupZ4OeuGwr68lGIeym/ksIZMJodUGOSCZ/FSnTxcrekbvqrgdUxlJOMpijaKZVjAJrWrGs/6Jy8OMuyj9ow== 674 | 675 | escape-string-regexp@4.0.0: 676 | version "4.0.0" 677 | resolved "https://registry.yarnpkg.com/escape-string-regexp/-/escape-string-regexp-4.0.0.tgz#14ba83a5d373e3d311e5afca29cf5bfad965bf34" 678 | integrity sha512-TtpcNJ3XAzx3Gq8sWRzJaVajRs0uVxA2YAkdb1jm2YkPz4G6egUFAyA3n5vtEIZefPk5Wa4UXbKuS5fKkJWdgA== 679 | 680 | eslint-scope@5.1.1: 681 | version "5.1.1" 682 | resolved "https://registry.yarnpkg.com/eslint-scope/-/eslint-scope-5.1.1.tgz#e786e59a66cb92b3f6c1fb0d508aab174848f48c" 683 | integrity sha512-2NxwbF/hZ0KpepYN0cNbo+FN6XoK7GaHlQhgx/hIZl6Va0bF45RQOOwhLIy8lQDbuCiadSLCBnH2CFYquit5bw== 684 | dependencies: 685 | esrecurse "^4.3.0" 686 | estraverse "^4.1.1" 687 | 688 | esrecurse@^4.3.0: 689 | version "4.3.0" 690 | resolved "https://registry.yarnpkg.com/esrecurse/-/esrecurse-4.3.0.tgz#7ad7964d679abb28bee72cec63758b1c5d2c9921" 691 | integrity sha512-KmfKL3b6G+RXvP8N1vr3Tq1kL/oCFgn2NYXEtqP8/L3pKapUA4G8cFVaoF3SU323CD4XypR/ffioHmkti6/Tag== 692 | dependencies: 693 | estraverse "^5.2.0" 694 | 695 | estraverse@^4.1.1: 696 | version "4.3.0" 697 | resolved "https://registry.yarnpkg.com/estraverse/-/estraverse-4.3.0.tgz#398ad3f3c5a24948be7725e83d11a7de28cdbd1d" 698 | integrity sha512-39nnKffWz8xN1BU/2c79n9nB9HDzo0niYUqx6xyqUnyoAnQyyWpOTdZEeiCch8BBu515t4wp9ZmgVfVhn9EBpw== 699 | 700 | estraverse@^5.2.0: 701 | version "5.3.0" 702 | resolved "https://registry.yarnpkg.com/estraverse/-/estraverse-5.3.0.tgz#2eea5290702f26ab8fe5370370ff86c965d21123" 703 | integrity sha512-MMdARuVEQziNTeJD8DgMqmhwR11BRQ/cBP+pLtYdSTnf3MIO8fFeiINEbX36ZdNlfU/7A9f3gUw49B3oQsvwBA== 704 | 705 | eventemitter3@^4.0.0: 706 | version "4.0.7" 707 | resolved "https://registry.yarnpkg.com/eventemitter3/-/eventemitter3-4.0.7.tgz#2de9b68f6528d5644ef5c59526a1b4a07306169f" 708 | integrity sha512-8guHBZCwKnFhYdHr2ysuRWErTwhoN2X8XELRlrRwpmfeY2jjuUN4taQMsULKUVo1K4DvZl+0pgfyoysHxvmvEw== 709 | 710 | events@^3.2.0: 711 | version "3.3.0" 712 | resolved "https://registry.yarnpkg.com/events/-/events-3.3.0.tgz#31a95ad0a924e2d2c419a813aeb2c4e878ea7400" 713 | integrity sha512-mQw+2fkQbALzQ7V0MY0IqdnXNOeTtP4r0lN9z7AAawCXgqea7bDii20AYrIBrFd/Hx0M2Ocz6S111CaFkUcb0Q== 714 | 715 | extend@^3.0.0: 716 | version "3.0.2" 717 | resolved "https://registry.yarnpkg.com/extend/-/extend-3.0.2.tgz#f8b1136b4071fbd8eb140aff858b1019ec2915fa" 718 | integrity sha512-fjquC59cD7CyW6urNXK0FBufkZcoiGG80wTuPujX590cB5Ttln20E2UB4S/WARVqhXffZl2LNgS+gQdPIIim/g== 719 | 720 | fast-deep-equal@^3.1.1: 721 | version "3.1.3" 722 | resolved "https://registry.yarnpkg.com/fast-deep-equal/-/fast-deep-equal-3.1.3.tgz#3a7d56b559d6cbc3eb512325244e619a65c6c525" 723 | integrity sha512-f3qQ9oQy9j2AhBe/H9VC91wLmKBCCU/gDOnKNAYG5hswO7BLKj09Hc5HYNz9cGI++xlpDCIgDaitVs03ATR84Q== 724 | 725 | fast-json-stable-stringify@^2.0.0: 726 | version "2.1.0" 727 | resolved "https://registry.yarnpkg.com/fast-json-stable-stringify/-/fast-json-stable-stringify-2.1.0.tgz#874bf69c6f404c2b5d99c481341399fd55892633" 728 | integrity sha512-lhd/wF+Lk98HZoTCtlVraHtfh5XYijIjalXck7saUtuanSDyLMxnHhSXEDJqHxD7msR8D0uCmqlkwjCV8xvwHw== 729 | 730 | fastest-levenshtein@^1.0.12: 731 | version "1.0.16" 732 | resolved "https://registry.yarnpkg.com/fastest-levenshtein/-/fastest-levenshtein-1.0.16.tgz#210e61b6ff181de91ea9b3d1b84fdedd47e034e5" 733 | integrity sha512-eRnCtTTtGZFpQCwhJiUOuxPQWRXVKYDn0b2PeHfXL6/Zi53SLAzAHfVhVWK2AryC/WH05kGfxhFIPvTF0SXQzg== 734 | 735 | fill-range@^7.0.1: 736 | version "7.0.1" 737 | resolved "https://registry.yarnpkg.com/fill-range/-/fill-range-7.0.1.tgz#1919a6a7c75fe38b2c7c77e5198535da9acdda40" 738 | integrity sha512-qOo9F+dMUmC2Lcb4BbVvnKJxTPjCm+RRpe4gDuGrzkL7mEVl/djYSu2OdQ2Pa302N4oqkSg9ir6jaLWJ2USVpQ== 739 | dependencies: 740 | to-regex-range "^5.0.1" 741 | 742 | finalhandler@1.1.2: 743 | version "1.1.2" 744 | resolved "https://registry.yarnpkg.com/finalhandler/-/finalhandler-1.1.2.tgz#b7e7d000ffd11938d0fdb053506f6ebabe9f587d" 745 | integrity sha512-aAWcW57uxVNrQZqFXjITpW3sIUQmHGG3qSb9mUah9MgMC4NeWhNOlNjXEYq3HjRAvL6arUviZGGJsBg6z0zsWA== 746 | dependencies: 747 | debug "2.6.9" 748 | encodeurl "~1.0.2" 749 | escape-html "~1.0.3" 750 | on-finished "~2.3.0" 751 | parseurl "~1.3.3" 752 | statuses "~1.5.0" 753 | unpipe "~1.0.0" 754 | 755 | find-up@5.0.0: 756 | version "5.0.0" 757 | resolved "https://registry.yarnpkg.com/find-up/-/find-up-5.0.0.tgz#4c92819ecb7083561e4f4a240a86be5198f536fc" 758 | integrity sha512-78/PXT1wlLLDgTzDs7sjq9hzz0vXD+zn+7wypEe4fXQxCmdmqfGsEPQxmiCSQI3ajFV91bVSsvNtrJRiW6nGng== 759 | dependencies: 760 | locate-path "^6.0.0" 761 | path-exists "^4.0.0" 762 | 763 | find-up@^4.0.0: 764 | version "4.1.0" 765 | resolved "https://registry.yarnpkg.com/find-up/-/find-up-4.1.0.tgz#97afe7d6cdc0bc5928584b7c8d7b16e8a9aa5d19" 766 | integrity sha512-PpOwAdQ/YlXQ2vj8a3h8IipDuYRi3wceVQQGYWxNINccq40Anw7BlsEXCMbt1Zt+OLA6Fq9suIpIWD0OsnISlw== 767 | dependencies: 768 | locate-path "^5.0.0" 769 | path-exists "^4.0.0" 770 | 771 | flat@^5.0.2: 772 | version "5.0.2" 773 | resolved "https://registry.yarnpkg.com/flat/-/flat-5.0.2.tgz#8ca6fe332069ffa9d324c327198c598259ceb241" 774 | integrity sha512-b6suED+5/3rTpUBdG1gupIl8MPFCAMA0QXwmljLhvCUKcUvdE4gWky9zpuGCcXHOsz4J9wPGNWq6OKpmIzz3hQ== 775 | 776 | flatted@^3.2.7: 777 | version "3.2.9" 778 | resolved "https://registry.yarnpkg.com/flatted/-/flatted-3.2.9.tgz#7eb4c67ca1ba34232ca9d2d93e9886e611ad7daf" 779 | integrity sha512-36yxDn5H7OFZQla0/jFJmbIKTdZAQHngCedGxiMmpNfEZM0sdEeT+WczLQrjK6D7o2aiyLYDnkw0R3JK0Qv1RQ== 780 | 781 | follow-redirects@^1.0.0: 782 | version "1.15.5" 783 | resolved "https://registry.yarnpkg.com/follow-redirects/-/follow-redirects-1.15.5.tgz#54d4d6d062c0fa7d9d17feb008461550e3ba8020" 784 | integrity sha512-vSFWUON1B+yAw1VN4xMfxgn5fTUiaOzAJCKBwIIgT/+7CuGy9+r+5gITvP62j3RmaD5Ph65UaERdOSRGUzZtgw== 785 | 786 | format-util@^1.0.5: 787 | version "1.0.5" 788 | resolved "https://registry.yarnpkg.com/format-util/-/format-util-1.0.5.tgz#1ffb450c8a03e7bccffe40643180918cc297d271" 789 | integrity sha512-varLbTj0e0yVyRpqQhuWV+8hlePAgaoFRhNFj50BNjEIrw1/DphHSObtqwskVCPWNgzwPoQrZAbfa/SBiicNeg== 790 | 791 | fs-extra@^8.1.0: 792 | version "8.1.0" 793 | resolved "https://registry.yarnpkg.com/fs-extra/-/fs-extra-8.1.0.tgz#49d43c45a88cd9677668cb7be1b46efdb8d2e1c0" 794 | integrity sha512-yhlQgA6mnOJUKOsRUFsgJdQCvkKhcz8tlZG5HBQfReYZy46OwLcY+Zia0mtdHsOo9y/hP+CxMN0TU9QxoOtG4g== 795 | dependencies: 796 | graceful-fs "^4.2.0" 797 | jsonfile "^4.0.0" 798 | universalify "^0.1.0" 799 | 800 | fs.realpath@^1.0.0: 801 | version "1.0.0" 802 | resolved "https://registry.yarnpkg.com/fs.realpath/-/fs.realpath-1.0.0.tgz#1504ad2523158caa40db4a2787cb01411994ea4f" 803 | integrity sha512-OO0pH2lK6a0hZnAdau5ItzHPI6pUlvI7jMVnxUQRtw4owF2wk8lOSabtGDCTP4Ggrg2MbGnWO9X8K1t4+fGMDw== 804 | 805 | fsevents@~2.3.2: 806 | version "2.3.3" 807 | resolved "https://registry.yarnpkg.com/fsevents/-/fsevents-2.3.3.tgz#cac6407785d03675a2a5e1a5305c697b347d90d6" 808 | integrity sha512-5xoDfX+fL7faATnagmWPpbFtwh/R77WmMMqqHGS65C3vvB0YHrgF+B1YmZ3441tMj5n63k0212XNoJwzlhffQw== 809 | 810 | function-bind@^1.1.2: 811 | version "1.1.2" 812 | resolved "https://registry.yarnpkg.com/function-bind/-/function-bind-1.1.2.tgz#2c02d864d97f3ea6c8830c464cbd11ab6eab7a1c" 813 | integrity sha512-7XHNxH7qX9xG5mIwxkhumTox/MIRNcOgDrxWsMt2pAr23WHp6MrRlN7FBSFpCpr+oVO0F744iUgR82nJMfG2SA== 814 | 815 | get-caller-file@^2.0.5: 816 | version "2.0.5" 817 | resolved "https://registry.yarnpkg.com/get-caller-file/-/get-caller-file-2.0.5.tgz#4f94412a82db32f36e3b0b9741f8a97feb031f7e" 818 | integrity sha512-DyFP3BM/3YHTQOCUL/w0OZHR0lpKeGrxotcHWcqNEdnltqFwXVfhEBQ94eIo34AfQpo0rGki4cyIiftY06h2Fg== 819 | 820 | get-intrinsic@^1.0.2, get-intrinsic@^1.1.3, get-intrinsic@^1.2.1, get-intrinsic@^1.2.2: 821 | version "1.2.2" 822 | resolved "https://registry.yarnpkg.com/get-intrinsic/-/get-intrinsic-1.2.2.tgz#281b7622971123e1ef4b3c90fd7539306da93f3b" 823 | integrity sha512-0gSo4ml/0j98Y3lngkFEot/zhiCeWsbYIlZ+uZOVgzLyLaUw7wxUL+nCTP0XJvJg1AXulJRI3UJi8GsbDuxdGA== 824 | dependencies: 825 | function-bind "^1.1.2" 826 | has-proto "^1.0.1" 827 | has-symbols "^1.0.3" 828 | hasown "^2.0.0" 829 | 830 | glob-parent@~5.1.2: 831 | version "5.1.2" 832 | resolved "https://registry.yarnpkg.com/glob-parent/-/glob-parent-5.1.2.tgz#869832c58034fe68a4093c17dc15e8340d8401c4" 833 | integrity sha512-AOIgSQCepiJYwP3ARnGx+5VnTu2HBYdzbGP45eLw1vr3zB3vZLeyed1sC9hnbcOc9/SrMyM5RPQrkGz4aS9Zow== 834 | dependencies: 835 | is-glob "^4.0.1" 836 | 837 | glob-to-regexp@^0.4.1: 838 | version "0.4.1" 839 | resolved "https://registry.yarnpkg.com/glob-to-regexp/-/glob-to-regexp-0.4.1.tgz#c75297087c851b9a578bd217dd59a92f59fe546e" 840 | integrity sha512-lkX1HJXwyMcprw/5YUZc2s7DrpAiHB21/V+E1rHUrVNokkvB6bqMzT0VfV6/86ZNabt1k14YOIaT7nDvOX3Iiw== 841 | 842 | glob@7.2.0: 843 | version "7.2.0" 844 | resolved "https://registry.yarnpkg.com/glob/-/glob-7.2.0.tgz#d15535af7732e02e948f4c41628bd910293f6023" 845 | integrity sha512-lmLf6gtyrPq8tTjSmrO94wBeQbFR3HbLHbuyD69wuyQkImp2hWqMGB47OX65FBkPffO641IP9jWa1z4ivqG26Q== 846 | dependencies: 847 | fs.realpath "^1.0.0" 848 | inflight "^1.0.4" 849 | inherits "2" 850 | minimatch "^3.0.4" 851 | once "^1.3.0" 852 | path-is-absolute "^1.0.0" 853 | 854 | glob@^7.1.3, glob@^7.1.7: 855 | version "7.2.3" 856 | resolved "https://registry.yarnpkg.com/glob/-/glob-7.2.3.tgz#b8df0fb802bbfa8e89bd1d938b4e16578ed44f2b" 857 | integrity sha512-nFR0zLpU2YCaRxwoCJvL6UvCH2JFyFVIvwTLsIf21AuHlMskA1hhTdk+LlYJtOlYt9v6dvszD2BGRqBL+iQK9Q== 858 | dependencies: 859 | fs.realpath "^1.0.0" 860 | inflight "^1.0.4" 861 | inherits "2" 862 | minimatch "^3.1.1" 863 | once "^1.3.0" 864 | path-is-absolute "^1.0.0" 865 | 866 | gopd@^1.0.1: 867 | version "1.0.1" 868 | resolved "https://registry.yarnpkg.com/gopd/-/gopd-1.0.1.tgz#29ff76de69dac7489b7c0918a5788e56477c332c" 869 | integrity sha512-d65bNlIadxvpb/A2abVdlqKqV563juRnZ1Wtk6s1sIR8uNsXR70xqIzVqxVf1eTqDunwT2MkczEeaezCKTZhwA== 870 | dependencies: 871 | get-intrinsic "^1.1.3" 872 | 873 | graceful-fs@^4.1.2, graceful-fs@^4.1.6, graceful-fs@^4.2.0, graceful-fs@^4.2.10, graceful-fs@^4.2.4, graceful-fs@^4.2.6, graceful-fs@^4.2.9: 874 | version "4.2.11" 875 | resolved "https://registry.yarnpkg.com/graceful-fs/-/graceful-fs-4.2.11.tgz#4183e4e8bf08bb6e05bbb2f7d2e0c8f712ca40e3" 876 | integrity sha512-RbJ5/jmFcNNCcDV5o9eTnBLJ/HszWV0P73bc+Ff4nS/rJj+YaS6IGyiOL0VoBYX+l1Wrl3k63h/KrH+nhJ0XvQ== 877 | 878 | has-flag@^4.0.0: 879 | version "4.0.0" 880 | resolved "https://registry.yarnpkg.com/has-flag/-/has-flag-4.0.0.tgz#944771fd9c81c81265c4d6941860da06bb59479b" 881 | integrity sha512-EykJT/Q1KjTWctppgIAgfSO0tKVuZUjhgMr17kqTumMl6Afv3EISleU7qZUzoXDFTAHTDC4NOoG/ZxU3EvlMPQ== 882 | 883 | has-property-descriptors@^1.0.0, has-property-descriptors@^1.0.1: 884 | version "1.0.1" 885 | resolved "https://registry.yarnpkg.com/has-property-descriptors/-/has-property-descriptors-1.0.1.tgz#52ba30b6c5ec87fd89fa574bc1c39125c6f65340" 886 | integrity sha512-VsX8eaIewvas0xnvinAe9bw4WfIeODpGYikiWYLH+dma0Jw6KHYqWiWfhQlgOVK8D6PvjubK5Uc4P0iIhIcNVg== 887 | dependencies: 888 | get-intrinsic "^1.2.2" 889 | 890 | has-proto@^1.0.1: 891 | version "1.0.1" 892 | resolved "https://registry.yarnpkg.com/has-proto/-/has-proto-1.0.1.tgz#1885c1305538958aff469fef37937c22795408e0" 893 | integrity sha512-7qE+iP+O+bgF9clE5+UoBFzE65mlBiVj3tKCrlNQ0Ogwm0BjpT/gK4SlLYDMybDh5I3TCTKnPPa0oMG7JDYrhg== 894 | 895 | has-symbols@^1.0.3: 896 | version "1.0.3" 897 | resolved "https://registry.yarnpkg.com/has-symbols/-/has-symbols-1.0.3.tgz#bb7b2c4349251dce87b125f7bdf874aa7c8b39f8" 898 | integrity sha512-l3LCuF6MgDNwTDKkdYGEihYjt5pRPbEg46rtlmnSPlUbgmB8LOIrKJbYYFBSbnPaJexMKtiPO8hmeRjRz2Td+A== 899 | 900 | hasown@^2.0.0: 901 | version "2.0.0" 902 | resolved "https://registry.yarnpkg.com/hasown/-/hasown-2.0.0.tgz#f4c513d454a57b7c7e1650778de226b11700546c" 903 | integrity sha512-vUptKVTpIJhcczKBbgnS+RtcuYMB8+oNzPK2/Hp3hanz8JmpATdmmgLgSaadVREkDm+e2giHwY3ZRkyjSIDDFA== 904 | dependencies: 905 | function-bind "^1.1.2" 906 | 907 | he@1.2.0: 908 | version "1.2.0" 909 | resolved "https://registry.yarnpkg.com/he/-/he-1.2.0.tgz#84ae65fa7eafb165fddb61566ae14baf05664f0f" 910 | integrity sha512-F/1DnUGPopORZi0ni+CvrCgHQ5FyEAHRLSApuYWMmrbSwoN2Mn/7k+Gl38gJnR7yyDZk6WLXwiGod1JOWNDKGw== 911 | 912 | http-errors@2.0.0: 913 | version "2.0.0" 914 | resolved "https://registry.yarnpkg.com/http-errors/-/http-errors-2.0.0.tgz#b7774a1486ef73cf7667ac9ae0858c012c57b9d3" 915 | integrity sha512-FtwrG/euBzaEjYeRqOgly7G0qviiXoJWnvEH2Z1plBdXgbyjv34pHTSb9zoeHMyDy33+DWy5Wt9Wo+TURtOYSQ== 916 | dependencies: 917 | depd "2.0.0" 918 | inherits "2.0.4" 919 | setprototypeof "1.2.0" 920 | statuses "2.0.1" 921 | toidentifier "1.0.1" 922 | 923 | http-proxy@^1.18.1: 924 | version "1.18.1" 925 | resolved "https://registry.yarnpkg.com/http-proxy/-/http-proxy-1.18.1.tgz#401541f0534884bbf95260334e72f88ee3976549" 926 | integrity sha512-7mz/721AbnJwIVbnaSv1Cz3Am0ZLT/UBwkC92VlxhXv/k/BBQfM2fXElQNC27BVGr0uwUpplYPQM9LnaBMR5NQ== 927 | dependencies: 928 | eventemitter3 "^4.0.0" 929 | follow-redirects "^1.0.0" 930 | requires-port "^1.0.0" 931 | 932 | iconv-lite@0.4.24: 933 | version "0.4.24" 934 | resolved "https://registry.yarnpkg.com/iconv-lite/-/iconv-lite-0.4.24.tgz#2022b4b25fbddc21d2f524974a474aafe733908b" 935 | integrity sha512-v3MXnZAcvnywkTUEZomIActle7RXXeedOR31wwl7VlyoXO4Qi9arvSenNQWne1TcRwhCL1HwLI21bEqdpj8/rA== 936 | dependencies: 937 | safer-buffer ">= 2.1.2 < 3" 938 | 939 | iconv-lite@^0.6.3: 940 | version "0.6.3" 941 | resolved "https://registry.yarnpkg.com/iconv-lite/-/iconv-lite-0.6.3.tgz#a52f80bf38da1952eb5c681790719871a1a72501" 942 | integrity sha512-4fCk79wshMdzMp2rH06qWrJE4iolqLhCUH+OiuIgU++RB0+94NlDL81atO7GX55uUKueo0txHNtvEyI6D7WdMw== 943 | dependencies: 944 | safer-buffer ">= 2.1.2 < 3.0.0" 945 | 946 | import-local@^3.0.2: 947 | version "3.1.0" 948 | resolved "https://registry.yarnpkg.com/import-local/-/import-local-3.1.0.tgz#b4479df8a5fd44f6cdce24070675676063c95cb4" 949 | integrity sha512-ASB07uLtnDs1o6EHjKpX34BKYDSqnFerfTOJL2HvMqF70LnxpjkzDB8J44oT9pu4AMPkQwf8jl6szgvNd2tRIg== 950 | dependencies: 951 | pkg-dir "^4.2.0" 952 | resolve-cwd "^3.0.0" 953 | 954 | inflight@^1.0.4: 955 | version "1.0.6" 956 | resolved "https://registry.yarnpkg.com/inflight/-/inflight-1.0.6.tgz#49bd6331d7d02d0c09bc910a1075ba8165b56df9" 957 | integrity sha512-k92I/b08q4wvFscXCLvqfsHCrjrF7yiXsQuIVvVE7N82W3+aqpzuUdBbfhWcy/FZR3/4IgflMgKLOsvPDrGCJA== 958 | dependencies: 959 | once "^1.3.0" 960 | wrappy "1" 961 | 962 | inherits@2, inherits@2.0.4: 963 | version "2.0.4" 964 | resolved "https://registry.yarnpkg.com/inherits/-/inherits-2.0.4.tgz#0fa2c64f932917c3433a0ded55363aae37416b7c" 965 | integrity sha512-k/vGaX4/Yla3WzyMCvTQOXYeIHvqOKtnqBduzTHpzpQZzAskKMhZ2K+EnBiSM9zGSoIFeMpXKxa4dYeZIQqewQ== 966 | 967 | interpret@^3.1.1: 968 | version "3.1.1" 969 | resolved "https://registry.yarnpkg.com/interpret/-/interpret-3.1.1.tgz#5be0ceed67ca79c6c4bc5cf0d7ee843dcea110c4" 970 | integrity sha512-6xwYfHbajpoF0xLW+iwLkhwgvLoZDfjYfoFNu8ftMoXINzwuymNLd9u/KmwtdT2GbR+/Cz66otEGEVVUHX9QLQ== 971 | 972 | is-binary-path@~2.1.0: 973 | version "2.1.0" 974 | resolved "https://registry.yarnpkg.com/is-binary-path/-/is-binary-path-2.1.0.tgz#ea1f7f3b80f064236e83470f86c09c254fb45b09" 975 | integrity sha512-ZMERYes6pDydyuGidse7OsHxtbI7WVeUEozgR/g7rd0xUimYNlvZRE/K2MgZTjWy725IfelLeVcEM97mmtRGXw== 976 | dependencies: 977 | binary-extensions "^2.0.0" 978 | 979 | is-core-module@^2.13.0: 980 | version "2.13.1" 981 | resolved "https://registry.yarnpkg.com/is-core-module/-/is-core-module-2.13.1.tgz#ad0d7532c6fea9da1ebdc82742d74525c6273384" 982 | integrity sha512-hHrIjvZsftOsvKSn2TRYl63zvxsgE0K+0mYMoH6gD4omR5IWB2KynivBQczo3+wF1cCkjzvptnI9Q0sPU66ilw== 983 | dependencies: 984 | hasown "^2.0.0" 985 | 986 | is-extglob@^2.1.1: 987 | version "2.1.1" 988 | resolved "https://registry.yarnpkg.com/is-extglob/-/is-extglob-2.1.1.tgz#a88c02535791f02ed37c76a1b9ea9773c833f8c2" 989 | integrity sha512-SbKbANkN603Vi4jEZv49LeVJMn4yGwsbzZworEoyEiutsN3nJYdbO36zfhGJ6QEDpOZIFkDtnq5JRxmvl3jsoQ== 990 | 991 | is-fullwidth-code-point@^3.0.0: 992 | version "3.0.0" 993 | resolved "https://registry.yarnpkg.com/is-fullwidth-code-point/-/is-fullwidth-code-point-3.0.0.tgz#f116f8064fe90b3f7844a38997c0b75051269f1d" 994 | integrity sha512-zymm5+u+sCsSWyD9qNaejV3DFvhCKclKdizYaJUuHA83RLjb7nSuGnddCHGv0hk+KY7BMAlsWeK4Ueg6EV6XQg== 995 | 996 | is-glob@^4.0.1, is-glob@~4.0.1: 997 | version "4.0.3" 998 | resolved "https://registry.yarnpkg.com/is-glob/-/is-glob-4.0.3.tgz#64f61e42cbbb2eec2071a9dac0b28ba1e65d5084" 999 | integrity sha512-xelSayHH36ZgE7ZWhli7pW34hNbNl8Ojv5KVmkJD4hBdD3th8Tfk9vYasLM+mXWOZhFkgZfxhLSnrwRr4elSSg== 1000 | dependencies: 1001 | is-extglob "^2.1.1" 1002 | 1003 | is-number@^7.0.0: 1004 | version "7.0.0" 1005 | resolved "https://registry.yarnpkg.com/is-number/-/is-number-7.0.0.tgz#7535345b896734d5f80c4d06c50955527a14f12b" 1006 | integrity sha512-41Cifkg6e8TylSpdtTpeLVMqvSBEVzTttHvERD741+pnZ8ANv0004MRL43QKPDlK9cGvNp6NZWZUBlbGXYxxng== 1007 | 1008 | is-plain-obj@^2.1.0: 1009 | version "2.1.0" 1010 | resolved "https://registry.yarnpkg.com/is-plain-obj/-/is-plain-obj-2.1.0.tgz#45e42e37fccf1f40da8e5f76ee21515840c09287" 1011 | integrity sha512-YWnfyRwxL/+SsrWYfOpUtz5b3YD+nyfkHvjbcanzk8zgyO4ASD67uVMRt8k5bM4lLMDnXfriRhOpemw+NfT1eA== 1012 | 1013 | is-plain-object@^2.0.4: 1014 | version "2.0.4" 1015 | resolved "https://registry.yarnpkg.com/is-plain-object/-/is-plain-object-2.0.4.tgz#2c163b3fafb1b606d9d17928f05c2a1c38e07677" 1016 | integrity sha512-h5PpgXkWitc38BBMYawTYMWJHFZJVnBquFE57xFpjB8pJFiF6gZ+bU+WyI/yqXiFR5mdLsgYNaPe8uao6Uv9Og== 1017 | dependencies: 1018 | isobject "^3.0.1" 1019 | 1020 | is-unicode-supported@^0.1.0: 1021 | version "0.1.0" 1022 | resolved "https://registry.yarnpkg.com/is-unicode-supported/-/is-unicode-supported-0.1.0.tgz#3f26c76a809593b52bfa2ecb5710ed2779b522a7" 1023 | integrity sha512-knxG2q4UC3u8stRGyAVJCOdxFmv5DZiRcdlIaAQXAbSfJya+OhopNotLQrstBhququ4ZpuKbDc/8S6mgXgPFPw== 1024 | 1025 | isbinaryfile@^4.0.8: 1026 | version "4.0.10" 1027 | resolved "https://registry.yarnpkg.com/isbinaryfile/-/isbinaryfile-4.0.10.tgz#0c5b5e30c2557a2f06febd37b7322946aaee42b3" 1028 | integrity sha512-iHrqe5shvBUcFbmZq9zOQHBoeOhZJu6RQGrDpBgenUm/Am+F3JM2MgQj+rK3Z601fzrL5gLZWtAPH2OBaSVcyw== 1029 | 1030 | isexe@^2.0.0: 1031 | version "2.0.0" 1032 | resolved "https://registry.yarnpkg.com/isexe/-/isexe-2.0.0.tgz#e8fbf374dc556ff8947a10dcb0572d633f2cfa10" 1033 | integrity sha512-RHxMLp9lnKHGHRng9QFhRCMbYAcVpn69smSGcq3f36xjgVVWThj4qqLbTLlq7Ssj8B+fIQ1EuCEGI2lKsyQeIw== 1034 | 1035 | isobject@^3.0.1: 1036 | version "3.0.1" 1037 | resolved "https://registry.yarnpkg.com/isobject/-/isobject-3.0.1.tgz#4e431e92b11a9731636aa1f9c8d1ccbcfdab78df" 1038 | integrity sha512-WhB9zCku7EGTj/HQQRz5aUQEUeoQZH2bWcltRErOpymJ4boYE6wL9Tbr23krRPSZ+C5zqNSrSw+Cc7sZZ4b7vg== 1039 | 1040 | jest-worker@^27.4.5: 1041 | version "27.5.1" 1042 | resolved "https://registry.yarnpkg.com/jest-worker/-/jest-worker-27.5.1.tgz#8d146f0900e8973b106b6f73cc1e9a8cb86f8db0" 1043 | integrity sha512-7vuh85V5cdDofPyxn58nrPjBktZo0u9x1g8WtjQol+jZDaE+fhN+cIvTj11GndBnMnyfrUOG1sZQxCdjKh+DKg== 1044 | dependencies: 1045 | "@types/node" "*" 1046 | merge-stream "^2.0.0" 1047 | supports-color "^8.0.0" 1048 | 1049 | js-yaml@4.1.0: 1050 | version "4.1.0" 1051 | resolved "https://registry.yarnpkg.com/js-yaml/-/js-yaml-4.1.0.tgz#c1fb65f8f5017901cdd2c951864ba18458a10602" 1052 | integrity sha512-wpxZs9NoxZaJESJGIZTyDEaYpl0FKSA+FB9aJiyemKhMwkxQg63h4T1KJgUGHpTqPDNRcmmYLugrRjJlBtWvRA== 1053 | dependencies: 1054 | argparse "^2.0.1" 1055 | 1056 | json-parse-even-better-errors@^2.3.1: 1057 | version "2.3.1" 1058 | resolved "https://registry.yarnpkg.com/json-parse-even-better-errors/-/json-parse-even-better-errors-2.3.1.tgz#7c47805a94319928e05777405dc12e1f7a4ee02d" 1059 | integrity sha512-xyFwyhro/JEof6Ghe2iz2NcXoj2sloNsWr/XsERDK/oiPCfaNhl5ONfp+jQdAZRQQ0IJWNzH9zIZF7li91kh2w== 1060 | 1061 | json-schema-traverse@^0.4.1: 1062 | version "0.4.1" 1063 | resolved "https://registry.yarnpkg.com/json-schema-traverse/-/json-schema-traverse-0.4.1.tgz#69f6a87d9513ab8bb8fe63bdb0979c448e684660" 1064 | integrity sha512-xbbCH5dCYU5T8LcEhhuh7HJ88HXuW3qsI3Y0zOZFKfZEHcpWiHU/Jxzk629Brsab/mMiHQti9wMP+845RPe3Vg== 1065 | 1066 | jsonfile@^4.0.0: 1067 | version "4.0.0" 1068 | resolved "https://registry.yarnpkg.com/jsonfile/-/jsonfile-4.0.0.tgz#8771aae0799b64076b76640fca058f9c10e33ecb" 1069 | integrity sha512-m6F1R3z8jjlf2imQHS2Qez5sjKWQzbuuhuJ/FKYFRZvPE3PuHcSMVZzfsLhGVOkfd20obL5SWEBew5ShlquNxg== 1070 | optionalDependencies: 1071 | graceful-fs "^4.1.6" 1072 | 1073 | karma-chrome-launcher@3.2.0: 1074 | version "3.2.0" 1075 | resolved "https://registry.yarnpkg.com/karma-chrome-launcher/-/karma-chrome-launcher-3.2.0.tgz#eb9c95024f2d6dfbb3748d3415ac9b381906b9a9" 1076 | integrity sha512-rE9RkUPI7I9mAxByQWkGJFXfFD6lE4gC5nPuZdobf/QdTEJI6EU4yIay/cfU/xV4ZxlM5JiTv7zWYgA64NpS5Q== 1077 | dependencies: 1078 | which "^1.2.1" 1079 | 1080 | karma-mocha@2.0.1: 1081 | version "2.0.1" 1082 | resolved "https://registry.yarnpkg.com/karma-mocha/-/karma-mocha-2.0.1.tgz#4b0254a18dfee71bdbe6188d9a6861bf86b0cd7d" 1083 | integrity sha512-Tzd5HBjm8his2OA4bouAsATYEpZrp9vC7z5E5j4C5Of5Rrs1jY67RAwXNcVmd/Bnk1wgvQRou0zGVLey44G4tQ== 1084 | dependencies: 1085 | minimist "^1.2.3" 1086 | 1087 | karma-sourcemap-loader@0.4.0: 1088 | version "0.4.0" 1089 | resolved "https://registry.yarnpkg.com/karma-sourcemap-loader/-/karma-sourcemap-loader-0.4.0.tgz#b01d73f8f688f533bcc8f5d273d43458e13b5488" 1090 | integrity sha512-xCRL3/pmhAYF3I6qOrcn0uhbQevitc2DERMPH82FMnG+4WReoGcGFZb1pURf2a5apyrOHRdvD+O6K7NljqKHyA== 1091 | dependencies: 1092 | graceful-fs "^4.2.10" 1093 | 1094 | karma-webpack@5.0.0: 1095 | version "5.0.0" 1096 | resolved "https://registry.yarnpkg.com/karma-webpack/-/karma-webpack-5.0.0.tgz#2a2c7b80163fe7ffd1010f83f5507f95ef39f840" 1097 | integrity sha512-+54i/cd3/piZuP3dr54+NcFeKOPnys5QeM1IY+0SPASwrtHsliXUiCL50iW+K9WWA7RvamC4macvvQ86l3KtaA== 1098 | dependencies: 1099 | glob "^7.1.3" 1100 | minimatch "^3.0.4" 1101 | webpack-merge "^4.1.5" 1102 | 1103 | karma@6.4.2: 1104 | version "6.4.2" 1105 | resolved "https://registry.yarnpkg.com/karma/-/karma-6.4.2.tgz#a983f874cee6f35990c4b2dcc3d274653714de8e" 1106 | integrity sha512-C6SU/53LB31BEgRg+omznBEMY4SjHU3ricV6zBcAe1EeILKkeScr+fZXtaI5WyDbkVowJxxAI6h73NcFPmXolQ== 1107 | dependencies: 1108 | "@colors/colors" "1.5.0" 1109 | body-parser "^1.19.0" 1110 | braces "^3.0.2" 1111 | chokidar "^3.5.1" 1112 | connect "^3.7.0" 1113 | di "^0.0.1" 1114 | dom-serialize "^2.2.1" 1115 | glob "^7.1.7" 1116 | graceful-fs "^4.2.6" 1117 | http-proxy "^1.18.1" 1118 | isbinaryfile "^4.0.8" 1119 | lodash "^4.17.21" 1120 | log4js "^6.4.1" 1121 | mime "^2.5.2" 1122 | minimatch "^3.0.4" 1123 | mkdirp "^0.5.5" 1124 | qjobs "^1.2.0" 1125 | range-parser "^1.2.1" 1126 | rimraf "^3.0.2" 1127 | socket.io "^4.4.1" 1128 | source-map "^0.6.1" 1129 | tmp "^0.2.1" 1130 | ua-parser-js "^0.7.30" 1131 | yargs "^16.1.1" 1132 | 1133 | kind-of@^6.0.2: 1134 | version "6.0.3" 1135 | resolved "https://registry.yarnpkg.com/kind-of/-/kind-of-6.0.3.tgz#07c05034a6c349fa06e24fa35aa76db4580ce4dd" 1136 | integrity sha512-dcS1ul+9tmeD95T+x28/ehLgd9mENa3LsvDTtzm3vyBEO7RPptvAD+t44WVXaUjTBRcrpFeFlC8WCruUR456hw== 1137 | 1138 | loader-runner@^4.2.0: 1139 | version "4.3.0" 1140 | resolved "https://registry.yarnpkg.com/loader-runner/-/loader-runner-4.3.0.tgz#c1b4a163b99f614830353b16755e7149ac2314e1" 1141 | integrity sha512-3R/1M+yS3j5ou80Me59j7F9IMs4PXs3VqRrm0TU3AbKPxlmpoY1TNscJV/oGJXo8qCatFGTfDbY6W6ipGOYXfg== 1142 | 1143 | locate-path@^5.0.0: 1144 | version "5.0.0" 1145 | resolved "https://registry.yarnpkg.com/locate-path/-/locate-path-5.0.0.tgz#1afba396afd676a6d42504d0a67a3a7eb9f62aa0" 1146 | integrity sha512-t7hw9pI+WvuwNJXwk5zVHpyhIqzg2qTlklJOf0mVxGSbe3Fp2VieZcduNYjaLDoy6p9uGpQEGWG87WpMKlNq8g== 1147 | dependencies: 1148 | p-locate "^4.1.0" 1149 | 1150 | locate-path@^6.0.0: 1151 | version "6.0.0" 1152 | resolved "https://registry.yarnpkg.com/locate-path/-/locate-path-6.0.0.tgz#55321eb309febbc59c4801d931a72452a681d286" 1153 | integrity sha512-iPZK6eYjbxRu3uB4/WZ3EsEIMJFMqAoopl3R+zuq0UjcAm/MO6KCweDgPfP3elTztoKP3KtnVHxTn2NHBSDVUw== 1154 | dependencies: 1155 | p-locate "^5.0.0" 1156 | 1157 | lodash@^4.17.15, lodash@^4.17.21: 1158 | version "4.17.21" 1159 | resolved "https://registry.yarnpkg.com/lodash/-/lodash-4.17.21.tgz#679591c564c3bffaae8454cf0b3df370c3d6911c" 1160 | integrity sha512-v2kDEe57lecTulaDIuNTPy3Ry4gLGJ6Z1O3vE1krgXZNrsQ+LFTGHVxVjcXPs17LhbZVGedAJv8XZ1tvj5FvSg== 1161 | 1162 | log-symbols@4.1.0: 1163 | version "4.1.0" 1164 | resolved "https://registry.yarnpkg.com/log-symbols/-/log-symbols-4.1.0.tgz#3fbdbb95b4683ac9fc785111e792e558d4abd503" 1165 | integrity sha512-8XPvpAA8uyhfteu8pIvQxpJZ7SYYdpUivZpGy6sFsBuKRY/7rQGavedeB8aK+Zkyq6upMFVL/9AW6vOYzfRyLg== 1166 | dependencies: 1167 | chalk "^4.1.0" 1168 | is-unicode-supported "^0.1.0" 1169 | 1170 | log4js@^6.4.1: 1171 | version "6.9.1" 1172 | resolved "https://registry.yarnpkg.com/log4js/-/log4js-6.9.1.tgz#aba5a3ff4e7872ae34f8b4c533706753709e38b6" 1173 | integrity sha512-1somDdy9sChrr9/f4UlzhdaGfDR2c/SaD2a4T7qEkG4jTS57/B3qmnjLYePwQ8cqWnUHZI0iAKxMBpCZICiZ2g== 1174 | dependencies: 1175 | date-format "^4.0.14" 1176 | debug "^4.3.4" 1177 | flatted "^3.2.7" 1178 | rfdc "^1.3.0" 1179 | streamroller "^3.1.5" 1180 | 1181 | media-typer@0.3.0: 1182 | version "0.3.0" 1183 | resolved "https://registry.yarnpkg.com/media-typer/-/media-typer-0.3.0.tgz#8710d7af0aa626f8fffa1ce00168545263255748" 1184 | integrity sha512-dq+qelQ9akHpcOl/gUVRTxVIOkAJ1wR3QAvb4RsVjS8oVoFjDGTc679wJYmUmknUF5HwMLOgb5O+a3KxfWapPQ== 1185 | 1186 | merge-stream@^2.0.0: 1187 | version "2.0.0" 1188 | resolved "https://registry.yarnpkg.com/merge-stream/-/merge-stream-2.0.0.tgz#52823629a14dd00c9770fb6ad47dc6310f2c1f60" 1189 | integrity sha512-abv/qOcuPfk3URPfDzmZU1LKmuw8kT+0nIHvKrKgFrwifol/doWcdA4ZqsWQ8ENrFKkd67Mfpo/LovbIUsbt3w== 1190 | 1191 | mime-db@1.52.0: 1192 | version "1.52.0" 1193 | resolved "https://registry.yarnpkg.com/mime-db/-/mime-db-1.52.0.tgz#bbabcdc02859f4987301c856e3387ce5ec43bf70" 1194 | integrity sha512-sPU4uV7dYlvtWJxwwxHD0PuihVNiE7TyAbQ5SWxDCB9mUYvOgroQOwYQQOKPJ8CIbE+1ETVlOoK1UC2nU3gYvg== 1195 | 1196 | mime-types@^2.1.27, mime-types@~2.1.24, mime-types@~2.1.34: 1197 | version "2.1.35" 1198 | resolved "https://registry.yarnpkg.com/mime-types/-/mime-types-2.1.35.tgz#381a871b62a734450660ae3deee44813f70d959a" 1199 | integrity sha512-ZDY+bPm5zTTF+YpCrAU9nK0UgICYPT0QtT1NZWFv4s++TNkcgVaT0g6+4R2uI4MjQjzysHB1zxuWL50hzaeXiw== 1200 | dependencies: 1201 | mime-db "1.52.0" 1202 | 1203 | mime@^2.5.2: 1204 | version "2.6.0" 1205 | resolved "https://registry.yarnpkg.com/mime/-/mime-2.6.0.tgz#a2a682a95cd4d0cb1d6257e28f83da7e35800367" 1206 | integrity sha512-USPkMeET31rOMiarsBNIHZKLGgvKc/LrjofAnBlOttf5ajRvqiRA8QsenbcooctK6d6Ts6aqZXBA+XbkKthiQg== 1207 | 1208 | minimatch@5.0.1: 1209 | version "5.0.1" 1210 | resolved "https://registry.yarnpkg.com/minimatch/-/minimatch-5.0.1.tgz#fb9022f7528125187c92bd9e9b6366be1cf3415b" 1211 | integrity sha512-nLDxIFRyhDblz3qMuq+SoRZED4+miJ/G+tdDrjkkkRnjAsBexeGpgjLEQ0blJy7rHhR2b93rhQY4SvyWu9v03g== 1212 | dependencies: 1213 | brace-expansion "^2.0.1" 1214 | 1215 | minimatch@^3.0.4, minimatch@^3.1.1: 1216 | version "3.1.2" 1217 | resolved "https://registry.yarnpkg.com/minimatch/-/minimatch-3.1.2.tgz#19cd194bfd3e428f049a70817c038d89ab4be35b" 1218 | integrity sha512-J7p63hRiAjw1NDEww1W7i37+ByIrOWO5XQQAzZ3VOcL0PNybwpfmV/N05zFAzwQ9USyEcX6t3UO+K5aqBQOIHw== 1219 | dependencies: 1220 | brace-expansion "^1.1.7" 1221 | 1222 | minimist@^1.2.3, minimist@^1.2.6: 1223 | version "1.2.8" 1224 | resolved "https://registry.yarnpkg.com/minimist/-/minimist-1.2.8.tgz#c1a464e7693302e082a075cee0c057741ac4772c" 1225 | integrity sha512-2yyAR8qBkN3YuheJanUpWC5U3bb5osDywNB8RzDVlDwDHbocAJveqqj1u8+SVD7jkWT4yvsHCpWqqWqAxb0zCA== 1226 | 1227 | mkdirp@^0.5.5: 1228 | version "0.5.6" 1229 | resolved "https://registry.yarnpkg.com/mkdirp/-/mkdirp-0.5.6.tgz#7def03d2432dcae4ba1d611445c48396062255f6" 1230 | integrity sha512-FP+p8RB8OWpF3YZBCrP5gtADmtXApB5AMLn+vdyA+PyxCjrCs00mjyUozssO33cwDeT3wNGdLxJ5M//YqtHAJw== 1231 | dependencies: 1232 | minimist "^1.2.6" 1233 | 1234 | mocha@10.2.0: 1235 | version "10.2.0" 1236 | resolved "https://registry.yarnpkg.com/mocha/-/mocha-10.2.0.tgz#1fd4a7c32ba5ac372e03a17eef435bd00e5c68b8" 1237 | integrity sha512-IDY7fl/BecMwFHzoqF2sg/SHHANeBoMMXFlS9r0OXKDssYE1M5O43wUY/9BVPeIvfH2zmEbBfseqN9gBQZzXkg== 1238 | dependencies: 1239 | ansi-colors "4.1.1" 1240 | browser-stdout "1.3.1" 1241 | chokidar "3.5.3" 1242 | debug "4.3.4" 1243 | diff "5.0.0" 1244 | escape-string-regexp "4.0.0" 1245 | find-up "5.0.0" 1246 | glob "7.2.0" 1247 | he "1.2.0" 1248 | js-yaml "4.1.0" 1249 | log-symbols "4.1.0" 1250 | minimatch "5.0.1" 1251 | ms "2.1.3" 1252 | nanoid "3.3.3" 1253 | serialize-javascript "6.0.0" 1254 | strip-json-comments "3.1.1" 1255 | supports-color "8.1.1" 1256 | workerpool "6.2.1" 1257 | yargs "16.2.0" 1258 | yargs-parser "20.2.4" 1259 | yargs-unparser "2.0.0" 1260 | 1261 | ms@2.0.0: 1262 | version "2.0.0" 1263 | resolved "https://registry.yarnpkg.com/ms/-/ms-2.0.0.tgz#5608aeadfc00be6c2901df5f9861788de0d597c8" 1264 | integrity sha512-Tpp60P6IUJDTuOq/5Z8cdskzJujfwqfOTkrwIwj7IRISpnkJnT6SyJ4PCPnGMoFjC9ddhal5KVIYtAt97ix05A== 1265 | 1266 | ms@2.1.2: 1267 | version "2.1.2" 1268 | resolved "https://registry.yarnpkg.com/ms/-/ms-2.1.2.tgz#d09d1f357b443f493382a8eb3ccd183872ae6009" 1269 | integrity sha512-sGkPx+VjMtmA6MX27oA4FBFELFCZZ4S4XqeGOXCv68tT+jb3vk/RyaKWP0PTKyWtmLSM0b+adUTEvbs1PEaH2w== 1270 | 1271 | ms@2.1.3: 1272 | version "2.1.3" 1273 | resolved "https://registry.yarnpkg.com/ms/-/ms-2.1.3.tgz#574c8138ce1d2b5861f0b44579dbadd60c6615b2" 1274 | integrity sha512-6FlzubTLZG3J2a/NVCAleEhjzq5oxgHyaCU9yYXvcLsvoVaHJq/s5xXI6/XXP6tz7R9xAOtHnSO/tXtF3WRTlA== 1275 | 1276 | nanoid@3.3.3: 1277 | version "3.3.3" 1278 | resolved "https://registry.yarnpkg.com/nanoid/-/nanoid-3.3.3.tgz#fd8e8b7aa761fe807dba2d1b98fb7241bb724a25" 1279 | integrity sha512-p1sjXuopFs0xg+fPASzQ28agW1oHD7xDsd9Xkf3T15H3c/cifrFHVwrh74PdoklAPi+i7MdRsE47vm2r6JoB+w== 1280 | 1281 | negotiator@0.6.3: 1282 | version "0.6.3" 1283 | resolved "https://registry.yarnpkg.com/negotiator/-/negotiator-0.6.3.tgz#58e323a72fedc0d6f9cd4d31fe49f51479590ccd" 1284 | integrity sha512-+EUsqGPLsM+j/zdChZjsnX51g4XrHFOIXwfnCVPGlQk/k5giakcKsuxCObBRu6DSm9opw/O6slWbJdghQM4bBg== 1285 | 1286 | neo-async@^2.6.2: 1287 | version "2.6.2" 1288 | resolved "https://registry.yarnpkg.com/neo-async/-/neo-async-2.6.2.tgz#b4aafb93e3aeb2d8174ca53cf163ab7d7308305f" 1289 | integrity sha512-Yd3UES5mWCSqR+qNT93S3UoYUkqAZ9lLg8a7g9rimsWmYGK8cVToA4/sF3RrshdyV3sAGMXVUmpMYOw+dLpOuw== 1290 | 1291 | node-releases@^2.0.14: 1292 | version "2.0.14" 1293 | resolved "https://registry.yarnpkg.com/node-releases/-/node-releases-2.0.14.tgz#2ffb053bceb8b2be8495ece1ab6ce600c4461b0b" 1294 | integrity sha512-y10wOWt8yZpqXmOgRo77WaHEmhYQYGNA6y421PKsKYWEK8aW+cqAphborZDhqfyKrbZEN92CN1X2KbafY2s7Yw== 1295 | 1296 | normalize-path@^3.0.0, normalize-path@~3.0.0: 1297 | version "3.0.0" 1298 | resolved "https://registry.yarnpkg.com/normalize-path/-/normalize-path-3.0.0.tgz#0dcd69ff23a1c9b11fd0978316644a0388216a65" 1299 | integrity sha512-6eZs5Ls3WtCisHWp9S2GUy8dqkpGi4BVSz3GaqiE6ezub0512ESztXUwUB6C6IKbQkY2Pnb/mD4WYojCRwcwLA== 1300 | 1301 | object-assign@^4: 1302 | version "4.1.1" 1303 | resolved "https://registry.yarnpkg.com/object-assign/-/object-assign-4.1.1.tgz#2109adc7965887cfc05cbbd442cac8bfbb360863" 1304 | integrity sha512-rJgTQnkUnH1sFw8yT6VSU3zD3sWmu6sZhIseY8VX+GRu3P6F7Fu+JNDoXfklElbLJSnc3FUQHVe4cU5hj+BcUg== 1305 | 1306 | object-inspect@^1.9.0: 1307 | version "1.13.1" 1308 | resolved "https://registry.yarnpkg.com/object-inspect/-/object-inspect-1.13.1.tgz#b96c6109324ccfef6b12216a956ca4dc2ff94bc2" 1309 | integrity sha512-5qoj1RUiKOMsCCNLV1CBiPYE10sziTsnmNxkAI/rZhiD63CF7IqdFGC/XzjWjpSgLf0LxXX3bDFIh0E18f6UhQ== 1310 | 1311 | on-finished@2.4.1: 1312 | version "2.4.1" 1313 | resolved "https://registry.yarnpkg.com/on-finished/-/on-finished-2.4.1.tgz#58c8c44116e54845ad57f14ab10b03533184ac3f" 1314 | integrity sha512-oVlzkg3ENAhCk2zdv7IJwd/QUD4z2RxRwpkcGY8psCVcCYZNq4wYnVWALHM+brtuJjePWiYF/ClmuDr8Ch5+kg== 1315 | dependencies: 1316 | ee-first "1.1.1" 1317 | 1318 | on-finished@~2.3.0: 1319 | version "2.3.0" 1320 | resolved "https://registry.yarnpkg.com/on-finished/-/on-finished-2.3.0.tgz#20f1336481b083cd75337992a16971aa2d906947" 1321 | integrity sha512-ikqdkGAAyf/X/gPhXGvfgAytDZtDbr+bkNUJ0N9h5MI/dmdgCs3l6hoHrcUv41sRKew3jIwrp4qQDXiK99Utww== 1322 | dependencies: 1323 | ee-first "1.1.1" 1324 | 1325 | once@^1.3.0: 1326 | version "1.4.0" 1327 | resolved "https://registry.yarnpkg.com/once/-/once-1.4.0.tgz#583b1aa775961d4b113ac17d9c50baef9dd76bd1" 1328 | integrity sha512-lNaJgI+2Q5URQBkccEKHTQOPaXdUxnZZElQTZY0MFUAuaEqe1E+Nyvgdz/aIyNi6Z9MzO5dv1H8n58/GELp3+w== 1329 | dependencies: 1330 | wrappy "1" 1331 | 1332 | p-limit@^2.2.0: 1333 | version "2.3.0" 1334 | resolved "https://registry.yarnpkg.com/p-limit/-/p-limit-2.3.0.tgz#3dd33c647a214fdfffd835933eb086da0dc21db1" 1335 | integrity sha512-//88mFWSJx8lxCzwdAABTJL2MyWB12+eIY7MDL2SqLmAkeKU9qxRvWuSyTjm3FUmpBEMuFfckAIqEaVGUDxb6w== 1336 | dependencies: 1337 | p-try "^2.0.0" 1338 | 1339 | p-limit@^3.0.2: 1340 | version "3.1.0" 1341 | resolved "https://registry.yarnpkg.com/p-limit/-/p-limit-3.1.0.tgz#e1daccbe78d0d1388ca18c64fea38e3e57e3706b" 1342 | integrity sha512-TYOanM3wGwNGsZN2cVTYPArw454xnXj5qmWF1bEoAc4+cU/ol7GVh7odevjp1FNHduHc3KZMcFduxU5Xc6uJRQ== 1343 | dependencies: 1344 | yocto-queue "^0.1.0" 1345 | 1346 | p-locate@^4.1.0: 1347 | version "4.1.0" 1348 | resolved "https://registry.yarnpkg.com/p-locate/-/p-locate-4.1.0.tgz#a3428bb7088b3a60292f66919278b7c297ad4f07" 1349 | integrity sha512-R79ZZ/0wAxKGu3oYMlz8jy/kbhsNrS7SKZ7PxEHBgJ5+F2mtFW2fK2cOtBh1cHYkQsbzFV7I+EoRKe6Yt0oK7A== 1350 | dependencies: 1351 | p-limit "^2.2.0" 1352 | 1353 | p-locate@^5.0.0: 1354 | version "5.0.0" 1355 | resolved "https://registry.yarnpkg.com/p-locate/-/p-locate-5.0.0.tgz#83c8315c6785005e3bd021839411c9e110e6d834" 1356 | integrity sha512-LaNjtRWUBY++zB5nE/NwcaoMylSPk+S+ZHNB1TzdbMJMny6dynpAGt7X/tl/QYq3TIeE6nxHppbo2LGymrG5Pw== 1357 | dependencies: 1358 | p-limit "^3.0.2" 1359 | 1360 | p-try@^2.0.0: 1361 | version "2.2.0" 1362 | resolved "https://registry.yarnpkg.com/p-try/-/p-try-2.2.0.tgz#cb2868540e313d61de58fafbe35ce9004d5540e6" 1363 | integrity sha512-R4nPAVTAU0B9D35/Gk3uJf/7XYbQcyohSKdvAxIRSNghFl4e71hVoGnBNQz9cWaXxO2I10KTC+3jMdvvoKw6dQ== 1364 | 1365 | parseurl@~1.3.3: 1366 | version "1.3.3" 1367 | resolved "https://registry.yarnpkg.com/parseurl/-/parseurl-1.3.3.tgz#9da19e7bee8d12dff0513ed5b76957793bc2e8d4" 1368 | integrity sha512-CiyeOxFT/JZyN5m0z9PfXw4SCBJ6Sygz1Dpl0wqjlhDEGGBP1GnsUVEL0p63hoG1fcj3fHynXi9NYO4nWOL+qQ== 1369 | 1370 | path-exists@^4.0.0: 1371 | version "4.0.0" 1372 | resolved "https://registry.yarnpkg.com/path-exists/-/path-exists-4.0.0.tgz#513bdbe2d3b95d7762e8c1137efa195c6c61b5b3" 1373 | integrity sha512-ak9Qy5Q7jYb2Wwcey5Fpvg2KoAc/ZIhLSLOSBmRmygPsGwkVVt0fZa0qrtMz+m6tJTAHfZQ8FnmB4MG4LWy7/w== 1374 | 1375 | path-is-absolute@^1.0.0: 1376 | version "1.0.1" 1377 | resolved "https://registry.yarnpkg.com/path-is-absolute/-/path-is-absolute-1.0.1.tgz#174b9268735534ffbc7ace6bf53a5a9e1b5c5f5f" 1378 | integrity sha512-AVbw3UJ2e9bq64vSaS9Am0fje1Pa8pbGqTTsmXfaIiMpnr5DlDhfJOuLj9Sf95ZPVDAUerDfEk88MPmPe7UCQg== 1379 | 1380 | path-key@^3.1.0: 1381 | version "3.1.1" 1382 | resolved "https://registry.yarnpkg.com/path-key/-/path-key-3.1.1.tgz#581f6ade658cbba65a0d3380de7753295054f375" 1383 | integrity sha512-ojmeN0qd+y0jszEtoY48r0Peq5dwMEkIlCOu6Q5f41lfkswXuKtYrhgoTpLnyIcHm24Uhqx+5Tqm2InSwLhE6Q== 1384 | 1385 | path-parse@^1.0.7: 1386 | version "1.0.7" 1387 | resolved "https://registry.yarnpkg.com/path-parse/-/path-parse-1.0.7.tgz#fbc114b60ca42b30d9daf5858e4bd68bbedb6735" 1388 | integrity sha512-LDJzPVEEEPR+y48z93A0Ed0yXb8pAByGWo/k5YYdYgpY2/2EsOsksJrq7lOHxryrVOn1ejG6oAp8ahvOIQD8sw== 1389 | 1390 | picocolors@^1.0.0: 1391 | version "1.0.0" 1392 | resolved "https://registry.yarnpkg.com/picocolors/-/picocolors-1.0.0.tgz#cb5bdc74ff3f51892236eaf79d68bc44564ab81c" 1393 | integrity sha512-1fygroTLlHu66zi26VoTDv8yRgm0Fccecssto+MhsZ0D/DGW2sm8E8AjW7NU5VVTRt5GxbeZ5qBuJr+HyLYkjQ== 1394 | 1395 | picomatch@^2.0.4, picomatch@^2.2.1: 1396 | version "2.3.1" 1397 | resolved "https://registry.yarnpkg.com/picomatch/-/picomatch-2.3.1.tgz#3ba3833733646d9d3e4995946c1365a67fb07a42" 1398 | integrity sha512-JU3teHTNjmE2VCGFzuY8EXzCDVwEqB2a8fsIvwaStHhAWJEeVd1o1QD80CU6+ZdEXXSLbSsuLwJjkCBWqRQUVA== 1399 | 1400 | pkg-dir@^4.2.0: 1401 | version "4.2.0" 1402 | resolved "https://registry.yarnpkg.com/pkg-dir/-/pkg-dir-4.2.0.tgz#f099133df7ede422e81d1d8448270eeb3e4261f3" 1403 | integrity sha512-HRDzbaKjC+AOWVXxAU/x54COGeIv9eb+6CkDSQoNTt4XyWoIJvuPsXizxu/Fr23EiekbtZwmh1IcIG/l/a10GQ== 1404 | dependencies: 1405 | find-up "^4.0.0" 1406 | 1407 | punycode@^2.1.0: 1408 | version "2.3.1" 1409 | resolved "https://registry.yarnpkg.com/punycode/-/punycode-2.3.1.tgz#027422e2faec0b25e1549c3e1bd8309b9133b6e5" 1410 | integrity sha512-vYt7UD1U9Wg6138shLtLOvdAu+8DsC/ilFtEVHcH+wydcSpNE20AfSOduf6MkRFahL5FY7X1oU7nKVZFtfq8Fg== 1411 | 1412 | qjobs@^1.2.0: 1413 | version "1.2.0" 1414 | resolved "https://registry.yarnpkg.com/qjobs/-/qjobs-1.2.0.tgz#c45e9c61800bd087ef88d7e256423bdd49e5d071" 1415 | integrity sha512-8YOJEHtxpySA3fFDyCRxA+UUV+fA+rTWnuWvylOK/NCjhY+b4ocCtmu8TtsWb+mYeU+GCHf/S66KZF/AsteKHg== 1416 | 1417 | qs@6.11.0: 1418 | version "6.11.0" 1419 | resolved "https://registry.yarnpkg.com/qs/-/qs-6.11.0.tgz#fd0d963446f7a65e1367e01abd85429453f0c37a" 1420 | integrity sha512-MvjoMCJwEarSbUYk5O+nmoSzSutSsTwF85zcHPQ9OrlFoZOYIjaqBAJIqIXjptyD5vThxGq52Xu/MaJzRkIk4Q== 1421 | dependencies: 1422 | side-channel "^1.0.4" 1423 | 1424 | randombytes@^2.1.0: 1425 | version "2.1.0" 1426 | resolved "https://registry.yarnpkg.com/randombytes/-/randombytes-2.1.0.tgz#df6f84372f0270dc65cdf6291349ab7a473d4f2a" 1427 | integrity sha512-vYl3iOX+4CKUWuxGi9Ukhie6fsqXqS9FE2Zaic4tNFD2N2QQaXOMFbuKK4QmDHC0JO6B1Zp41J0LpT0oR68amQ== 1428 | dependencies: 1429 | safe-buffer "^5.1.0" 1430 | 1431 | range-parser@^1.2.1: 1432 | version "1.2.1" 1433 | resolved "https://registry.yarnpkg.com/range-parser/-/range-parser-1.2.1.tgz#3cf37023d199e1c24d1a55b84800c2f3e6468031" 1434 | integrity sha512-Hrgsx+orqoygnmhFbKaHE6c296J+HTAQXoxEF6gNupROmmGJRoyzfG3ccAveqCBrwr/2yxQ5BVd/GTl5agOwSg== 1435 | 1436 | raw-body@2.5.2: 1437 | version "2.5.2" 1438 | resolved "https://registry.yarnpkg.com/raw-body/-/raw-body-2.5.2.tgz#99febd83b90e08975087e8f1f9419a149366b68a" 1439 | integrity sha512-8zGqypfENjCIqGhgXToC8aB2r7YrBX+AQAfIPs/Mlk+BtPTztOvTS01NRW/3Eh60J+a48lt8qsCzirQ6loCVfA== 1440 | dependencies: 1441 | bytes "3.1.2" 1442 | http-errors "2.0.0" 1443 | iconv-lite "0.4.24" 1444 | unpipe "1.0.0" 1445 | 1446 | readdirp@~3.6.0: 1447 | version "3.6.0" 1448 | resolved "https://registry.yarnpkg.com/readdirp/-/readdirp-3.6.0.tgz#74a370bd857116e245b29cc97340cd431a02a6c7" 1449 | integrity sha512-hOS089on8RduqdbhvQ5Z37A0ESjsqz6qnRcffsMU3495FuTdqSm+7bhJ29JvIOsBDEEnan5DPu9t3To9VRlMzA== 1450 | dependencies: 1451 | picomatch "^2.2.1" 1452 | 1453 | rechoir@^0.8.0: 1454 | version "0.8.0" 1455 | resolved "https://registry.yarnpkg.com/rechoir/-/rechoir-0.8.0.tgz#49f866e0d32146142da3ad8f0eff352b3215ff22" 1456 | integrity sha512-/vxpCXddiX8NGfGO/mTafwjq4aFa/71pvamip0++IQk3zG8cbCj0fifNPrjjF1XMXUne91jL9OoxmdykoEtifQ== 1457 | dependencies: 1458 | resolve "^1.20.0" 1459 | 1460 | require-directory@^2.1.1: 1461 | version "2.1.1" 1462 | resolved "https://registry.yarnpkg.com/require-directory/-/require-directory-2.1.1.tgz#8c64ad5fd30dab1c976e2344ffe7f792a6a6df42" 1463 | integrity sha512-fGxEI7+wsG9xrvdjsrlmL22OMTTiHRwAMroiEeMgq8gzoLC/PQr7RsRDSTLUg/bZAZtF+TVIkHc6/4RIKrui+Q== 1464 | 1465 | requires-port@^1.0.0: 1466 | version "1.0.0" 1467 | resolved "https://registry.yarnpkg.com/requires-port/-/requires-port-1.0.0.tgz#925d2601d39ac485e091cf0da5c6e694dc3dcaff" 1468 | integrity sha512-KigOCHcocU3XODJxsu8i/j8T9tzT4adHiecwORRQ0ZZFcp7ahwXuRU1m+yuO90C5ZUyGeGfocHDI14M3L3yDAQ== 1469 | 1470 | resolve-cwd@^3.0.0: 1471 | version "3.0.0" 1472 | resolved "https://registry.yarnpkg.com/resolve-cwd/-/resolve-cwd-3.0.0.tgz#0f0075f1bb2544766cf73ba6a6e2adfebcb13f2d" 1473 | integrity sha512-OrZaX2Mb+rJCpH/6CpSqt9xFVpN++x01XnN2ie9g6P5/3xelLAkXWVADpdz1IHD/KFfEXyE6V0U01OQ3UO2rEg== 1474 | dependencies: 1475 | resolve-from "^5.0.0" 1476 | 1477 | resolve-from@^5.0.0: 1478 | version "5.0.0" 1479 | resolved "https://registry.yarnpkg.com/resolve-from/-/resolve-from-5.0.0.tgz#c35225843df8f776df21c57557bc087e9dfdfc69" 1480 | integrity sha512-qYg9KP24dD5qka9J47d0aVky0N+b4fTU89LN9iDnjB5waksiC49rvMB0PrUJQGoTmH50XPiqOvAjDfaijGxYZw== 1481 | 1482 | resolve@^1.20.0: 1483 | version "1.22.8" 1484 | resolved "https://registry.yarnpkg.com/resolve/-/resolve-1.22.8.tgz#b6c87a9f2aa06dfab52e3d70ac8cde321fa5a48d" 1485 | integrity sha512-oKWePCxqpd6FlLvGV1VU0x7bkPmmCNolxzjMf4NczoDnQcIWrAF+cPtZn5i6n+RfD2d9i0tzpKnG6Yk168yIyw== 1486 | dependencies: 1487 | is-core-module "^2.13.0" 1488 | path-parse "^1.0.7" 1489 | supports-preserve-symlinks-flag "^1.0.0" 1490 | 1491 | rfdc@^1.3.0: 1492 | version "1.3.1" 1493 | resolved "https://registry.yarnpkg.com/rfdc/-/rfdc-1.3.1.tgz#2b6d4df52dffe8bb346992a10ea9451f24373a8f" 1494 | integrity sha512-r5a3l5HzYlIC68TpmYKlxWjmOP6wiPJ1vWv2HeLhNsRZMrCkxeqxiHlQ21oXmQ4F3SiryXBHhAD7JZqvOJjFmg== 1495 | 1496 | rimraf@^3.0.0, rimraf@^3.0.2: 1497 | version "3.0.2" 1498 | resolved "https://registry.yarnpkg.com/rimraf/-/rimraf-3.0.2.tgz#f1a5402ba6220ad52cc1282bac1ae3aa49fd061a" 1499 | integrity sha512-JZkJMZkAGFFPP2YqXZXPbMlMBgsxzE8ILs4lMIX/2o0L9UBw9O/Y3o6wFw/i9YLapcUJWwqbi3kdxIPdC62TIA== 1500 | dependencies: 1501 | glob "^7.1.3" 1502 | 1503 | safe-buffer@^5.1.0: 1504 | version "5.2.1" 1505 | resolved "https://registry.yarnpkg.com/safe-buffer/-/safe-buffer-5.2.1.tgz#1eaf9fa9bdb1fdd4ec75f58f9cdb4e6b7827eec6" 1506 | integrity sha512-rp3So07KcdmmKbGvgaNxQSJr7bGVSVk5S9Eq1F+ppbRo70+YeaDxkw5Dd8NPN+GD6bjnYm2VuPuCXmpuYvmCXQ== 1507 | 1508 | "safer-buffer@>= 2.1.2 < 3", "safer-buffer@>= 2.1.2 < 3.0.0": 1509 | version "2.1.2" 1510 | resolved "https://registry.yarnpkg.com/safer-buffer/-/safer-buffer-2.1.2.tgz#44fa161b0187b9549dd84bb91802f9bd8385cd6a" 1511 | integrity sha512-YZo3K82SD7Riyi0E1EQPojLz7kpepnSQI9IyPbHHg1XXXevb5dJI7tpyN2ADxGcQbHG7vcyRHk0cbwqcQriUtg== 1512 | 1513 | schema-utils@^3.1.1, schema-utils@^3.1.2: 1514 | version "3.3.0" 1515 | resolved "https://registry.yarnpkg.com/schema-utils/-/schema-utils-3.3.0.tgz#f50a88877c3c01652a15b622ae9e9795df7a60fe" 1516 | integrity sha512-pN/yOAvcC+5rQ5nERGuwrjLlYvLTbCibnZ1I7B1LaiAz9BRBlE9GMgE/eqV30P7aJQUf7Ddimy/RsbYO/GrVGg== 1517 | dependencies: 1518 | "@types/json-schema" "^7.0.8" 1519 | ajv "^6.12.5" 1520 | ajv-keywords "^3.5.2" 1521 | 1522 | serialize-javascript@6.0.0: 1523 | version "6.0.0" 1524 | resolved "https://registry.yarnpkg.com/serialize-javascript/-/serialize-javascript-6.0.0.tgz#efae5d88f45d7924141da8b5c3a7a7e663fefeb8" 1525 | integrity sha512-Qr3TosvguFt8ePWqsvRfrKyQXIiW+nGbYpy8XK24NQHE83caxWt+mIymTT19DGFbNWNLfEwsrkSmN64lVWB9ag== 1526 | dependencies: 1527 | randombytes "^2.1.0" 1528 | 1529 | serialize-javascript@^6.0.1: 1530 | version "6.0.2" 1531 | resolved "https://registry.yarnpkg.com/serialize-javascript/-/serialize-javascript-6.0.2.tgz#defa1e055c83bf6d59ea805d8da862254eb6a6c2" 1532 | integrity sha512-Saa1xPByTTq2gdeFZYLLo+RFE35NHZkAbqZeWNd3BpzppeVisAqpDjcp8dyf6uIvEqJRd46jemmyA4iFIeVk8g== 1533 | dependencies: 1534 | randombytes "^2.1.0" 1535 | 1536 | set-function-length@^1.1.1: 1537 | version "1.2.0" 1538 | resolved "https://registry.yarnpkg.com/set-function-length/-/set-function-length-1.2.0.tgz#2f81dc6c16c7059bda5ab7c82c11f03a515ed8e1" 1539 | integrity sha512-4DBHDoyHlM1IRPGYcoxexgh67y4ueR53FKV1yyxwFMY7aCqcN/38M1+SwZ/qJQ8iLv7+ck385ot4CcisOAPT9w== 1540 | dependencies: 1541 | define-data-property "^1.1.1" 1542 | function-bind "^1.1.2" 1543 | get-intrinsic "^1.2.2" 1544 | gopd "^1.0.1" 1545 | has-property-descriptors "^1.0.1" 1546 | 1547 | setprototypeof@1.2.0: 1548 | version "1.2.0" 1549 | resolved "https://registry.yarnpkg.com/setprototypeof/-/setprototypeof-1.2.0.tgz#66c9a24a73f9fc28cbe66b09fed3d33dcaf1b424" 1550 | integrity sha512-E5LDX7Wrp85Kil5bhZv46j8jOeboKq5JMmYM3gVGdGH8xFpPWXUMsNrlODCrkoxMEeNi/XZIwuRvY4XNwYMJpw== 1551 | 1552 | shallow-clone@^3.0.0: 1553 | version "3.0.1" 1554 | resolved "https://registry.yarnpkg.com/shallow-clone/-/shallow-clone-3.0.1.tgz#8f2981ad92531f55035b01fb230769a40e02efa3" 1555 | integrity sha512-/6KqX+GVUdqPuPPd2LxDDxzX6CAbjJehAAOKlNpqqUpAqPM6HeL8f+o3a+JsyGjn2lv0WY8UsTgUJjU9Ok55NA== 1556 | dependencies: 1557 | kind-of "^6.0.2" 1558 | 1559 | shebang-command@^2.0.0: 1560 | version "2.0.0" 1561 | resolved "https://registry.yarnpkg.com/shebang-command/-/shebang-command-2.0.0.tgz#ccd0af4f8835fbdc265b82461aaf0c36663f34ea" 1562 | integrity sha512-kHxr2zZpYtdmrN1qDjrrX/Z1rR1kG8Dx+gkpK1G4eXmvXswmcE1hTWBWYUzlraYw1/yZp6YuDY77YtvbN0dmDA== 1563 | dependencies: 1564 | shebang-regex "^3.0.0" 1565 | 1566 | shebang-regex@^3.0.0: 1567 | version "3.0.0" 1568 | resolved "https://registry.yarnpkg.com/shebang-regex/-/shebang-regex-3.0.0.tgz#ae16f1644d873ecad843b0307b143362d4c42172" 1569 | integrity sha512-7++dFhtcx3353uBaq8DDR4NuxBetBzC7ZQOhmTQInHEd6bSrXdiEyzCvG07Z44UYdLShWUyXt5M/yhz8ekcb1A== 1570 | 1571 | side-channel@^1.0.4: 1572 | version "1.0.4" 1573 | resolved "https://registry.yarnpkg.com/side-channel/-/side-channel-1.0.4.tgz#efce5c8fdc104ee751b25c58d4290011fa5ea2cf" 1574 | integrity sha512-q5XPytqFEIKHkGdiMIrY10mvLRvnQh42/+GoBlFW3b2LXLE2xxJpZFdm94we0BaoV3RwJyGqg5wS7epxTv0Zvw== 1575 | dependencies: 1576 | call-bind "^1.0.0" 1577 | get-intrinsic "^1.0.2" 1578 | object-inspect "^1.9.0" 1579 | 1580 | socket.io-adapter@~2.5.2: 1581 | version "2.5.2" 1582 | resolved "https://registry.yarnpkg.com/socket.io-adapter/-/socket.io-adapter-2.5.2.tgz#5de9477c9182fdc171cd8c8364b9a8894ec75d12" 1583 | integrity sha512-87C3LO/NOMc+eMcpcxUBebGjkpMDkNBS9tf7KJqcDsmL936EChtVva71Dw2q4tQcuVC+hAUy4an2NO/sYXmwRA== 1584 | dependencies: 1585 | ws "~8.11.0" 1586 | 1587 | socket.io-parser@~4.2.4: 1588 | version "4.2.4" 1589 | resolved "https://registry.yarnpkg.com/socket.io-parser/-/socket.io-parser-4.2.4.tgz#c806966cf7270601e47469ddeec30fbdfda44c83" 1590 | integrity sha512-/GbIKmo8ioc+NIWIhwdecY0ge+qVBSMdgxGygevmdHj24bsfgtCmcUUcQ5ZzcylGFHsN3k4HB4Cgkl96KVnuew== 1591 | dependencies: 1592 | "@socket.io/component-emitter" "~3.1.0" 1593 | debug "~4.3.1" 1594 | 1595 | socket.io@^4.4.1: 1596 | version "4.7.4" 1597 | resolved "https://registry.yarnpkg.com/socket.io/-/socket.io-4.7.4.tgz#2401a2d7101e4bdc64da80b140d5d8b6a8c7738b" 1598 | integrity sha512-DcotgfP1Zg9iP/dH9zvAQcWrE0TtbMVwXmlV4T4mqsvY+gw+LqUGPfx2AoVyRk0FLME+GQhufDMyacFmw7ksqw== 1599 | dependencies: 1600 | accepts "~1.3.4" 1601 | base64id "~2.0.0" 1602 | cors "~2.8.5" 1603 | debug "~4.3.2" 1604 | engine.io "~6.5.2" 1605 | socket.io-adapter "~2.5.2" 1606 | socket.io-parser "~4.2.4" 1607 | 1608 | source-map-js@^1.0.2: 1609 | version "1.0.2" 1610 | resolved "https://registry.yarnpkg.com/source-map-js/-/source-map-js-1.0.2.tgz#adbc361d9c62df380125e7f161f71c826f1e490c" 1611 | integrity sha512-R0XvVJ9WusLiqTCEiGCmICCMplcCkIwwR11mOSD9CR5u+IXYdiseeEuXCVAjS54zqwkLcPNnmU4OeJ6tUrWhDw== 1612 | 1613 | source-map-loader@4.0.1: 1614 | version "4.0.1" 1615 | resolved "https://registry.yarnpkg.com/source-map-loader/-/source-map-loader-4.0.1.tgz#72f00d05f5d1f90f80974eda781cbd7107c125f2" 1616 | integrity sha512-oqXpzDIByKONVY8g1NUPOTQhe0UTU5bWUl32GSkqK2LjJj0HmwTMVKxcUip0RgAYhY1mqgOxjbQM48a0mmeNfA== 1617 | dependencies: 1618 | abab "^2.0.6" 1619 | iconv-lite "^0.6.3" 1620 | source-map-js "^1.0.2" 1621 | 1622 | source-map-support@~0.5.20: 1623 | version "0.5.21" 1624 | resolved "https://registry.yarnpkg.com/source-map-support/-/source-map-support-0.5.21.tgz#04fe7c7f9e1ed2d662233c28cb2b35b9f63f6e4f" 1625 | integrity sha512-uBHU3L3czsIyYXKX88fdrGovxdSCoTGDRZ6SYXtSRxLZUzHg5P/66Ht6uoUlHu9EZod+inXhKo3qQgwXUT/y1w== 1626 | dependencies: 1627 | buffer-from "^1.0.0" 1628 | source-map "^0.6.0" 1629 | 1630 | source-map@^0.6.0, source-map@^0.6.1: 1631 | version "0.6.1" 1632 | resolved "https://registry.yarnpkg.com/source-map/-/source-map-0.6.1.tgz#74722af32e9614e9c287a8d0bbde48b5e2f1a263" 1633 | integrity sha512-UjgapumWlbMhkBgzT7Ykc5YXUT46F0iKu8SGXq0bcwP5dz/h0Plj6enJqjz1Zbq2l5WaqYnrVbwWOWMyF3F47g== 1634 | 1635 | statuses@2.0.1: 1636 | version "2.0.1" 1637 | resolved "https://registry.yarnpkg.com/statuses/-/statuses-2.0.1.tgz#55cb000ccf1d48728bd23c685a063998cf1a1b63" 1638 | integrity sha512-RwNA9Z/7PrK06rYLIzFMlaF+l73iwpzsqRIFgbMLbTcLD6cOao82TaWefPXQvB2fOC4AjuYSEndS7N/mTCbkdQ== 1639 | 1640 | statuses@~1.5.0: 1641 | version "1.5.0" 1642 | resolved "https://registry.yarnpkg.com/statuses/-/statuses-1.5.0.tgz#161c7dac177659fd9811f43771fa99381478628c" 1643 | integrity sha512-OpZ3zP+jT1PI7I8nemJX4AKmAX070ZkYPVWV/AaKTJl+tXCTGyVdC1a4SL8RUQYEwk/f34ZX8UTykN68FwrqAA== 1644 | 1645 | streamroller@^3.1.5: 1646 | version "3.1.5" 1647 | resolved "https://registry.yarnpkg.com/streamroller/-/streamroller-3.1.5.tgz#1263182329a45def1ffaef58d31b15d13d2ee7ff" 1648 | integrity sha512-KFxaM7XT+irxvdqSP1LGLgNWbYN7ay5owZ3r/8t77p+EtSUAfUgtl7be3xtqtOmGUl9K9YPO2ca8133RlTjvKw== 1649 | dependencies: 1650 | date-format "^4.0.14" 1651 | debug "^4.3.4" 1652 | fs-extra "^8.1.0" 1653 | 1654 | string-width@^4.1.0, string-width@^4.2.0: 1655 | version "4.2.3" 1656 | resolved "https://registry.yarnpkg.com/string-width/-/string-width-4.2.3.tgz#269c7117d27b05ad2e536830a8ec895ef9c6d010" 1657 | integrity sha512-wKyQRQpjJ0sIp62ErSZdGsjMJWsap5oRNihHhu6G7JVO/9jIB6UyevL+tXuOqrng8j/cxKTWyWUwvSTriiZz/g== 1658 | dependencies: 1659 | emoji-regex "^8.0.0" 1660 | is-fullwidth-code-point "^3.0.0" 1661 | strip-ansi "^6.0.1" 1662 | 1663 | strip-ansi@^6.0.0, strip-ansi@^6.0.1: 1664 | version "6.0.1" 1665 | resolved "https://registry.yarnpkg.com/strip-ansi/-/strip-ansi-6.0.1.tgz#9e26c63d30f53443e9489495b2105d37b67a85d9" 1666 | integrity sha512-Y38VPSHcqkFrCpFnQ9vuSXmquuv5oXOKpGeT6aGrr3o3Gc9AlVa6JBfUSOCnbxGGZF+/0ooI7KrPuUSztUdU5A== 1667 | dependencies: 1668 | ansi-regex "^5.0.1" 1669 | 1670 | strip-json-comments@3.1.1: 1671 | version "3.1.1" 1672 | resolved "https://registry.yarnpkg.com/strip-json-comments/-/strip-json-comments-3.1.1.tgz#31f1281b3832630434831c310c01cccda8cbe006" 1673 | integrity sha512-6fPc+R4ihwqP6N/aIv2f1gMH8lOVtWQHoqC4yK6oSDVVocumAsfCqjkXnqiYMhmMwS/mEHLp7Vehlt3ql6lEig== 1674 | 1675 | supports-color@8.1.1, supports-color@^8.0.0: 1676 | version "8.1.1" 1677 | resolved "https://registry.yarnpkg.com/supports-color/-/supports-color-8.1.1.tgz#cd6fc17e28500cff56c1b86c0a7fd4a54a73005c" 1678 | integrity sha512-MpUEN2OodtUzxvKQl72cUF7RQ5EiHsGvSsVG0ia9c5RbWGL2CI4C7EpPS8UTBIplnlzZiNuV56w+FuNxy3ty2Q== 1679 | dependencies: 1680 | has-flag "^4.0.0" 1681 | 1682 | supports-color@^7.1.0: 1683 | version "7.2.0" 1684 | resolved "https://registry.yarnpkg.com/supports-color/-/supports-color-7.2.0.tgz#1b7dcdcb32b8138801b3e478ba6a51caa89648da" 1685 | integrity sha512-qpCAvRl9stuOHveKsn7HncJRvv501qIacKzQlO/+Lwxc9+0q2wLyv4Dfvt80/DPn2pqOBsJdDiogXGR9+OvwRw== 1686 | dependencies: 1687 | has-flag "^4.0.0" 1688 | 1689 | supports-preserve-symlinks-flag@^1.0.0: 1690 | version "1.0.0" 1691 | resolved "https://registry.yarnpkg.com/supports-preserve-symlinks-flag/-/supports-preserve-symlinks-flag-1.0.0.tgz#6eda4bd344a3c94aea376d4cc31bc77311039e09" 1692 | integrity sha512-ot0WnXS9fgdkgIcePe6RHNk1WA8+muPa6cSjeR3V8K27q9BB1rTE3R1p7Hv0z1ZyAc8s6Vvv8DIyWf681MAt0w== 1693 | 1694 | tapable@^2.1.1, tapable@^2.2.0: 1695 | version "2.2.1" 1696 | resolved "https://registry.yarnpkg.com/tapable/-/tapable-2.2.1.tgz#1967a73ef4060a82f12ab96af86d52fdb76eeca0" 1697 | integrity sha512-GNzQvQTOIP6RyTfE2Qxb8ZVlNmw0n88vp1szwWRimP02mnTsx3Wtn5qRdqY9w2XduFNUgvOwhNnQsjwCp+kqaQ== 1698 | 1699 | terser-webpack-plugin@^5.3.7: 1700 | version "5.3.10" 1701 | resolved "https://registry.yarnpkg.com/terser-webpack-plugin/-/terser-webpack-plugin-5.3.10.tgz#904f4c9193c6fd2a03f693a2150c62a92f40d199" 1702 | integrity sha512-BKFPWlPDndPs+NGGCr1U59t0XScL5317Y0UReNrHaw9/FwhPENlq6bfgs+4yPfyP51vqC1bQ4rp1EfXW5ZSH9w== 1703 | dependencies: 1704 | "@jridgewell/trace-mapping" "^0.3.20" 1705 | jest-worker "^27.4.5" 1706 | schema-utils "^3.1.1" 1707 | serialize-javascript "^6.0.1" 1708 | terser "^5.26.0" 1709 | 1710 | terser@^5.26.0: 1711 | version "5.27.0" 1712 | resolved "https://registry.yarnpkg.com/terser/-/terser-5.27.0.tgz#70108689d9ab25fef61c4e93e808e9fd092bf20c" 1713 | integrity sha512-bi1HRwVRskAjheeYl291n3JC4GgO/Ty4z1nVs5AAsmonJulGxpSektecnNedrwK9C7vpvVtcX3cw00VSLt7U2A== 1714 | dependencies: 1715 | "@jridgewell/source-map" "^0.3.3" 1716 | acorn "^8.8.2" 1717 | commander "^2.20.0" 1718 | source-map-support "~0.5.20" 1719 | 1720 | tmp@^0.2.1: 1721 | version "0.2.1" 1722 | resolved "https://registry.yarnpkg.com/tmp/-/tmp-0.2.1.tgz#8457fc3037dcf4719c251367a1af6500ee1ccf14" 1723 | integrity sha512-76SUhtfqR2Ijn+xllcI5P1oyannHNHByD80W1q447gU3mp9G9PSpGdWmjUOHRDPiHYacIk66W7ubDTuPF3BEtQ== 1724 | dependencies: 1725 | rimraf "^3.0.0" 1726 | 1727 | to-regex-range@^5.0.1: 1728 | version "5.0.1" 1729 | resolved "https://registry.yarnpkg.com/to-regex-range/-/to-regex-range-5.0.1.tgz#1648c44aae7c8d988a326018ed72f5b4dd0392e4" 1730 | integrity sha512-65P7iz6X5yEr1cwcgvQxbbIw7Uk3gOy5dIdtZ4rDveLqhrdJP+Li/Hx6tyK0NEb+2GCyneCMJiGqrADCSNk8sQ== 1731 | dependencies: 1732 | is-number "^7.0.0" 1733 | 1734 | toidentifier@1.0.1: 1735 | version "1.0.1" 1736 | resolved "https://registry.yarnpkg.com/toidentifier/-/toidentifier-1.0.1.tgz#3be34321a88a820ed1bd80dfaa33e479fbb8dd35" 1737 | integrity sha512-o5sSPKEkg/DIQNmH43V0/uerLrpzVedkUh8tGNvaeXpfpuwjKenlSox/2O/BTlZUtEe+JG7s5YhEz608PlAHRA== 1738 | 1739 | type-is@~1.6.18: 1740 | version "1.6.18" 1741 | resolved "https://registry.yarnpkg.com/type-is/-/type-is-1.6.18.tgz#4e552cd05df09467dcbc4ef739de89f2cf37c131" 1742 | integrity sha512-TkRKr9sUTxEH8MdfuCSP7VizJyzRNMjj2J2do2Jr3Kym598JVdEksuzPQCnlFPW4ky9Q+iA+ma9BGm06XQBy8g== 1743 | dependencies: 1744 | media-typer "0.3.0" 1745 | mime-types "~2.1.24" 1746 | 1747 | typescript@5.0.4: 1748 | version "5.0.4" 1749 | resolved "https://registry.yarnpkg.com/typescript/-/typescript-5.0.4.tgz#b217fd20119bd61a94d4011274e0ab369058da3b" 1750 | integrity sha512-cW9T5W9xY37cc+jfEnaUvX91foxtHkza3Nw3wkoF4sSlKn0MONdkdEndig/qPBWXNkmplh3NzayQzCiHM4/hqw== 1751 | 1752 | ua-parser-js@^0.7.30: 1753 | version "0.7.37" 1754 | resolved "https://registry.yarnpkg.com/ua-parser-js/-/ua-parser-js-0.7.37.tgz#e464e66dac2d33a7a1251d7d7a99d6157ec27832" 1755 | integrity sha512-xV8kqRKM+jhMvcHWUKthV9fNebIzrNy//2O9ZwWcfiBFR5f25XVZPLlEajk/sf3Ra15V92isyQqnIEXRDaZWEA== 1756 | 1757 | undici-types@~5.26.4: 1758 | version "5.26.5" 1759 | resolved "https://registry.yarnpkg.com/undici-types/-/undici-types-5.26.5.tgz#bcd539893d00b56e964fd2657a4866b221a65617" 1760 | integrity sha512-JlCMO+ehdEIKqlFxk6IfVoAUVmgz7cU7zD/h9XZ0qzeosSHmUJVOzSQvvYSYWXkFXC+IfLKSIffhv0sVZup6pA== 1761 | 1762 | universalify@^0.1.0: 1763 | version "0.1.2" 1764 | resolved "https://registry.yarnpkg.com/universalify/-/universalify-0.1.2.tgz#b646f69be3942dabcecc9d6639c80dc105efaa66" 1765 | integrity sha512-rBJeI5CXAlmy1pV+617WB9J63U6XcazHHF2f2dbJix4XzpUF0RS3Zbj0FGIOCAva5P/d/GBOYaACQ1w+0azUkg== 1766 | 1767 | unpipe@1.0.0, unpipe@~1.0.0: 1768 | version "1.0.0" 1769 | resolved "https://registry.yarnpkg.com/unpipe/-/unpipe-1.0.0.tgz#b2bf4ee8514aae6165b4817829d21b2ef49904ec" 1770 | integrity sha512-pjy2bYhSsufwWlKwPc+l3cN7+wuJlK6uz0YdJEOlQDbl6jo/YlPi4mb8agUkVC8BF7V8NuzeyPNqRksA3hztKQ== 1771 | 1772 | update-browserslist-db@^1.0.13: 1773 | version "1.0.13" 1774 | resolved "https://registry.yarnpkg.com/update-browserslist-db/-/update-browserslist-db-1.0.13.tgz#3c5e4f5c083661bd38ef64b6328c26ed6c8248c4" 1775 | integrity sha512-xebP81SNcPuNpPP3uzeW1NYXxI3rxyJzF3pD6sH4jE7o/IX+WtSpwnVU+qIsDPyk0d3hmFQ7mjqc6AtV604hbg== 1776 | dependencies: 1777 | escalade "^3.1.1" 1778 | picocolors "^1.0.0" 1779 | 1780 | uri-js@^4.2.2: 1781 | version "4.4.1" 1782 | resolved "https://registry.yarnpkg.com/uri-js/-/uri-js-4.4.1.tgz#9b1a52595225859e55f669d928f88c6c57f2a77e" 1783 | integrity sha512-7rKUyy33Q1yc98pQ1DAmLtwX109F7TIfWlW1Ydo8Wl1ii1SeHieeh0HHfPeL2fMXK6z0s8ecKs9frCuLJvndBg== 1784 | dependencies: 1785 | punycode "^2.1.0" 1786 | 1787 | utils-merge@1.0.1: 1788 | version "1.0.1" 1789 | resolved "https://registry.yarnpkg.com/utils-merge/-/utils-merge-1.0.1.tgz#9f95710f50a267947b2ccc124741c1028427e713" 1790 | integrity sha512-pMZTvIkT1d+TFGvDOqodOclx0QWkkgi6Tdoa8gC8ffGAAqz9pzPTZWAybbsHHoED/ztMtkv/VoYTYyShUn81hA== 1791 | 1792 | vary@^1: 1793 | version "1.1.2" 1794 | resolved "https://registry.yarnpkg.com/vary/-/vary-1.1.2.tgz#2299f02c6ded30d4a5961b0b9f74524a18f634fc" 1795 | integrity sha512-BNGbWLfd0eUPabhkXUVm0j8uuvREyTh5ovRa/dyow/BqAbZJyC+5fU+IzQOzmAKzYqYRAISoRhdQr3eIZ/PXqg== 1796 | 1797 | void-elements@^2.0.0: 1798 | version "2.0.1" 1799 | resolved "https://registry.yarnpkg.com/void-elements/-/void-elements-2.0.1.tgz#c066afb582bb1cb4128d60ea92392e94d5e9dbec" 1800 | integrity sha512-qZKX4RnBzH2ugr8Lxa7x+0V6XD9Sb/ouARtiasEQCHB1EVU4NXtmHsDDrx1dO4ne5fc3J6EW05BP1Dl0z0iung== 1801 | 1802 | watchpack@^2.4.0: 1803 | version "2.4.0" 1804 | resolved "https://registry.yarnpkg.com/watchpack/-/watchpack-2.4.0.tgz#fa33032374962c78113f93c7f2fb4c54c9862a5d" 1805 | integrity sha512-Lcvm7MGST/4fup+ifyKi2hjyIAwcdI4HRgtvTpIUxBRhB+RFtUh8XtDOxUfctVCnhVi+QQj49i91OyvzkJl6cg== 1806 | dependencies: 1807 | glob-to-regexp "^0.4.1" 1808 | graceful-fs "^4.1.2" 1809 | 1810 | webpack-cli@5.1.0: 1811 | version "5.1.0" 1812 | resolved "https://registry.yarnpkg.com/webpack-cli/-/webpack-cli-5.1.0.tgz#abc4b1f44b50250f2632d8b8b536cfe2f6257891" 1813 | integrity sha512-a7KRJnCxejFoDpYTOwzm5o21ZXMaNqtRlvS183XzGDUPRdVEzJNImcQokqYZ8BNTnk9DkKiuWxw75+DCCoZ26w== 1814 | dependencies: 1815 | "@discoveryjs/json-ext" "^0.5.0" 1816 | "@webpack-cli/configtest" "^2.1.0" 1817 | "@webpack-cli/info" "^2.0.1" 1818 | "@webpack-cli/serve" "^2.0.3" 1819 | colorette "^2.0.14" 1820 | commander "^10.0.1" 1821 | cross-spawn "^7.0.3" 1822 | envinfo "^7.7.3" 1823 | fastest-levenshtein "^1.0.12" 1824 | import-local "^3.0.2" 1825 | interpret "^3.1.1" 1826 | rechoir "^0.8.0" 1827 | webpack-merge "^5.7.3" 1828 | 1829 | webpack-merge@^4.1.5: 1830 | version "4.2.2" 1831 | resolved "https://registry.yarnpkg.com/webpack-merge/-/webpack-merge-4.2.2.tgz#a27c52ea783d1398afd2087f547d7b9d2f43634d" 1832 | integrity sha512-TUE1UGoTX2Cd42j3krGYqObZbOD+xF7u28WB7tfUordytSjbWTIjK/8V0amkBfTYN4/pB/GIDlJZZ657BGG19g== 1833 | dependencies: 1834 | lodash "^4.17.15" 1835 | 1836 | webpack-merge@^5.7.3: 1837 | version "5.10.0" 1838 | resolved "https://registry.yarnpkg.com/webpack-merge/-/webpack-merge-5.10.0.tgz#a3ad5d773241e9c682803abf628d4cd62b8a4177" 1839 | integrity sha512-+4zXKdx7UnO+1jaN4l2lHVD+mFvnlZQP/6ljaJVb4SZiwIKeUnrT5l0gkT8z+n4hKpC+jpOv6O9R+gLtag7pSA== 1840 | dependencies: 1841 | clone-deep "^4.0.1" 1842 | flat "^5.0.2" 1843 | wildcard "^2.0.0" 1844 | 1845 | webpack-sources@^3.2.3: 1846 | version "3.2.3" 1847 | resolved "https://registry.yarnpkg.com/webpack-sources/-/webpack-sources-3.2.3.tgz#2d4daab8451fd4b240cc27055ff6a0c2ccea0cde" 1848 | integrity sha512-/DyMEOrDgLKKIG0fmvtz+4dUX/3Ghozwgm6iPp8KRhvn+eQf9+Q7GWxVNMk3+uCPWfdXYC4ExGBckIXdFEfH1w== 1849 | 1850 | webpack@5.82.0: 1851 | version "5.82.0" 1852 | resolved "https://registry.yarnpkg.com/webpack/-/webpack-5.82.0.tgz#3c0d074dec79401db026b4ba0fb23d6333f88e7d" 1853 | integrity sha512-iGNA2fHhnDcV1bONdUu554eZx+XeldsaeQ8T67H6KKHl2nUSwX8Zm7cmzOA46ox/X1ARxf7Bjv8wQ/HsB5fxBg== 1854 | dependencies: 1855 | "@types/eslint-scope" "^3.7.3" 1856 | "@types/estree" "^1.0.0" 1857 | "@webassemblyjs/ast" "^1.11.5" 1858 | "@webassemblyjs/wasm-edit" "^1.11.5" 1859 | "@webassemblyjs/wasm-parser" "^1.11.5" 1860 | acorn "^8.7.1" 1861 | acorn-import-assertions "^1.7.6" 1862 | browserslist "^4.14.5" 1863 | chrome-trace-event "^1.0.2" 1864 | enhanced-resolve "^5.13.0" 1865 | es-module-lexer "^1.2.1" 1866 | eslint-scope "5.1.1" 1867 | events "^3.2.0" 1868 | glob-to-regexp "^0.4.1" 1869 | graceful-fs "^4.2.9" 1870 | json-parse-even-better-errors "^2.3.1" 1871 | loader-runner "^4.2.0" 1872 | mime-types "^2.1.27" 1873 | neo-async "^2.6.2" 1874 | schema-utils "^3.1.2" 1875 | tapable "^2.1.1" 1876 | terser-webpack-plugin "^5.3.7" 1877 | watchpack "^2.4.0" 1878 | webpack-sources "^3.2.3" 1879 | 1880 | which@^1.2.1: 1881 | version "1.3.1" 1882 | resolved "https://registry.yarnpkg.com/which/-/which-1.3.1.tgz#a45043d54f5805316da8d62f9f50918d3da70b0a" 1883 | integrity sha512-HxJdYWq1MTIQbJ3nw0cqssHoTNU267KlrDuGZ1WYlxDStUtKUhOaJmh112/TZmHxxUfuJqPXSOm7tDyas0OSIQ== 1884 | dependencies: 1885 | isexe "^2.0.0" 1886 | 1887 | which@^2.0.1: 1888 | version "2.0.2" 1889 | resolved "https://registry.yarnpkg.com/which/-/which-2.0.2.tgz#7c6a8dd0a636a0327e10b59c9286eee93f3f51b1" 1890 | integrity sha512-BLI3Tl1TW3Pvl70l3yq3Y64i+awpwXqsGBYWkkqMtnbXgrMD+yj7rhW0kuEDxzJaYXGjEW5ogapKNMEKNMjibA== 1891 | dependencies: 1892 | isexe "^2.0.0" 1893 | 1894 | wildcard@^2.0.0: 1895 | version "2.0.1" 1896 | resolved "https://registry.yarnpkg.com/wildcard/-/wildcard-2.0.1.tgz#5ab10d02487198954836b6349f74fff961e10f67" 1897 | integrity sha512-CC1bOL87PIWSBhDcTrdeLo6eGT7mCFtrg0uIJtqJUFyK+eJnzl8A1niH56uu7KMa5XFrtiV+AQuHO3n7DsHnLQ== 1898 | 1899 | workerpool@6.2.1: 1900 | version "6.2.1" 1901 | resolved "https://registry.yarnpkg.com/workerpool/-/workerpool-6.2.1.tgz#46fc150c17d826b86a008e5a4508656777e9c343" 1902 | integrity sha512-ILEIE97kDZvF9Wb9f6h5aXK4swSlKGUcOEGiIYb2OOu/IrDU9iwj0fD//SsA6E5ibwJxpEvhullJY4Sl4GcpAw== 1903 | 1904 | wrap-ansi@^7.0.0: 1905 | version "7.0.0" 1906 | resolved "https://registry.yarnpkg.com/wrap-ansi/-/wrap-ansi-7.0.0.tgz#67e145cff510a6a6984bdf1152911d69d2eb9e43" 1907 | integrity sha512-YVGIj2kamLSTxw6NsZjoBxfSwsn0ycdesmc4p+Q21c5zPuZ1pl+NfxVdxPtdHvmNVOQ6XSYG4AUtyt/Fi7D16Q== 1908 | dependencies: 1909 | ansi-styles "^4.0.0" 1910 | string-width "^4.1.0" 1911 | strip-ansi "^6.0.0" 1912 | 1913 | wrappy@1: 1914 | version "1.0.2" 1915 | resolved "https://registry.yarnpkg.com/wrappy/-/wrappy-1.0.2.tgz#b5243d8f3ec1aa35f1364605bc0d1036e30ab69f" 1916 | integrity sha512-l4Sp/DRseor9wL6EvV2+TuQn63dMkPjZ/sp9XkghTEbV9KlPS1xUsZ3u7/IQO4wxtcFB4bgpQPRcR3QCvezPcQ== 1917 | 1918 | ws@~8.11.0: 1919 | version "8.11.0" 1920 | resolved "https://registry.yarnpkg.com/ws/-/ws-8.11.0.tgz#6a0d36b8edfd9f96d8b25683db2f8d7de6e8e143" 1921 | integrity sha512-HPG3wQd9sNQoT9xHyNCXoDUa+Xw/VevmY9FoHyQ+g+rrMn4j6FB4np7Z0OhdTgjx6MgQLK7jwSy1YecU1+4Asg== 1922 | 1923 | y18n@^5.0.5: 1924 | version "5.0.8" 1925 | resolved "https://registry.yarnpkg.com/y18n/-/y18n-5.0.8.tgz#7f4934d0f7ca8c56f95314939ddcd2dd91ce1d55" 1926 | integrity sha512-0pfFzegeDWJHJIAmTLRP2DwHjdF5s7jo9tuztdQxAhINCdvS+3nGINqPd00AphqJR/0LhANUS6/+7SCb98YOfA== 1927 | 1928 | yargs-parser@20.2.4: 1929 | version "20.2.4" 1930 | resolved "https://registry.yarnpkg.com/yargs-parser/-/yargs-parser-20.2.4.tgz#b42890f14566796f85ae8e3a25290d205f154a54" 1931 | integrity sha512-WOkpgNhPTlE73h4VFAFsOnomJVaovO8VqLDzy5saChRBFQFBoMYirowyW+Q9HB4HFF4Z7VZTiG3iSzJJA29yRA== 1932 | 1933 | yargs-parser@^20.2.2: 1934 | version "20.2.9" 1935 | resolved "https://registry.yarnpkg.com/yargs-parser/-/yargs-parser-20.2.9.tgz#2eb7dc3b0289718fc295f362753845c41a0c94ee" 1936 | integrity sha512-y11nGElTIV+CT3Zv9t7VKl+Q3hTQoT9a1Qzezhhl6Rp21gJ/IVTW7Z3y9EWXhuUBC2Shnf+DX0antecpAwSP8w== 1937 | 1938 | yargs-unparser@2.0.0: 1939 | version "2.0.0" 1940 | resolved "https://registry.yarnpkg.com/yargs-unparser/-/yargs-unparser-2.0.0.tgz#f131f9226911ae5d9ad38c432fe809366c2325eb" 1941 | integrity sha512-7pRTIA9Qc1caZ0bZ6RYRGbHJthJWuakf+WmHK0rVeLkNrrGhfoabBNdue6kdINI6r4if7ocq9aD/n7xwKOdzOA== 1942 | dependencies: 1943 | camelcase "^6.0.0" 1944 | decamelize "^4.0.0" 1945 | flat "^5.0.2" 1946 | is-plain-obj "^2.1.0" 1947 | 1948 | yargs@16.2.0, yargs@^16.1.1: 1949 | version "16.2.0" 1950 | resolved "https://registry.yarnpkg.com/yargs/-/yargs-16.2.0.tgz#1c82bf0f6b6a66eafce7ef30e376f49a12477f66" 1951 | integrity sha512-D1mvvtDG0L5ft/jGWkLpG1+m0eQxOfaBvTNELraWj22wSVUMWxZUvYgJYcKh6jGGIkJFhH4IZPQhR4TKpc8mBw== 1952 | dependencies: 1953 | cliui "^7.0.2" 1954 | escalade "^3.1.1" 1955 | get-caller-file "^2.0.5" 1956 | require-directory "^2.1.1" 1957 | string-width "^4.2.0" 1958 | y18n "^5.0.5" 1959 | yargs-parser "^20.2.2" 1960 | 1961 | yocto-queue@^0.1.0: 1962 | version "0.1.0" 1963 | resolved "https://registry.yarnpkg.com/yocto-queue/-/yocto-queue-0.1.0.tgz#0294eb3dee05028d31ee1a5fa2c556a6aaf10a1b" 1964 | integrity sha512-rVksvsnNCdJ/ohGc6xgPwyN8eheCxsiLM8mxuE/t/mOVqJewPuO1miLpTHQiRgTKCLexL4MeAFVagts7HmNZ2Q== 1965 | -------------------------------------------------------------------------------- /navigation-screens-viewmodel/build.gradle.kts: -------------------------------------------------------------------------------- 1 | import org.jetbrains.kotlin.gradle.targets.js.dsl.ExperimentalWasmDsl 2 | 3 | plugins { 4 | alias(androidLibs.plugins.library) 5 | alias(commonLibs.plugins.kotlin.multiplatform) 6 | alias(commonLibs.plugins.compose.multiplatform) 7 | } 8 | 9 | kotlin { 10 | applyDefaultHierarchyTemplate() 11 | 12 | @OptIn(ExperimentalWasmDsl::class) 13 | wasmJs { 14 | browser() 15 | } 16 | 17 | jvm("desktop") 18 | 19 | androidTarget { 20 | publishLibraryVariants("release") 21 | } 22 | 23 | iosX64() 24 | iosArm64() 25 | iosSimulatorArm64() 26 | 27 | sourceSets { 28 | commonMain.dependencies { 29 | api(projects.navigationScreens) 30 | 31 | implementation(compose.animation) 32 | implementation(commonLibs.coroutines.core) 33 | } 34 | androidMain.dependencies { 35 | implementation(androidLibs.lifecycle.viewmodel) 36 | implementation(androidLibs.compose.navigation) 37 | implementation(commonLibs.kotlin.serialization.json) 38 | } 39 | create("nonAndroidMain") 40 | .dependsOn(getByName("commonMain")) 41 | getByName("desktopMain") 42 | .dependsOn(getByName("nonAndroidMain")) 43 | iosMain { 44 | dependsOn(getByName("nonAndroidMain")) 45 | } 46 | getByName("wasmJsMain") { 47 | dependsOn(getByName("nonAndroidMain")) 48 | } 49 | getByName("androidInstrumentedTest").dependencies { 50 | implementation(kotlin("test")) 51 | implementation(compose.material3) 52 | implementation(commonLibs.kotlin.serialization.json) 53 | implementation(androidLibs.test.runner) 54 | implementation(androidLibs.compose.ui.test.junit4) 55 | } 56 | getByName("desktopTest").dependencies { 57 | implementation(kotlin("test")) 58 | implementation(commonLibs.kotlin.serialization.json) 59 | implementation(compose.material3) 60 | implementation(compose.desktop.currentOs) 61 | implementation(compose.desktop.uiTestJUnit4) 62 | } 63 | } 64 | } 65 | 66 | android { 67 | compileSdk = androidLibs.versions.compile.sdk.get().toInt() 68 | namespace = "io.github.lukwol.navigation.screens.viewmodel" 69 | 70 | defaultConfig { 71 | minSdk = androidLibs.versions.min.sdk.get().toInt() 72 | testInstrumentationRunner = "androidx.test.runner.AndroidJUnitRunner" 73 | } 74 | 75 | compileOptions { 76 | sourceCompatibility = JavaVersion.VERSION_21 77 | targetCompatibility = JavaVersion.VERSION_21 78 | } 79 | 80 | buildFeatures { 81 | compose = true 82 | } 83 | composeOptions { 84 | kotlinCompilerExtensionVersion = androidLibs.versions.compose.compiler.get() 85 | } 86 | 87 | dependencies { 88 | debugImplementation(androidLibs.compose.ui.test.manifest) 89 | } 90 | } 91 | -------------------------------------------------------------------------------- /navigation-screens-viewmodel/src/androidInstrumentedTest/kotlin/io/github/lukwol/navigation/screens/viewmodel/components/ViewModelScreenNavigationTest.kt: -------------------------------------------------------------------------------- 1 | package io.github.lukwol.navigation.screens.viewmodel.components 2 | 3 | import androidx.compose.ui.test.assert 4 | import androidx.compose.ui.test.hasSetTextAction 5 | import androidx.compose.ui.test.hasText 6 | import androidx.compose.ui.test.junit4.createComposeRule 7 | import androidx.compose.ui.test.onNodeWithText 8 | import androidx.compose.ui.test.performClick 9 | import androidx.compose.ui.test.performTextClearance 10 | import androidx.compose.ui.test.performTextInput 11 | import io.github.lukwol.navigation.screens.viewmodel.components.components.TestScreenNavigation 12 | import org.junit.Rule 13 | import org.junit.Test 14 | 15 | class ViewModelScreenNavigationTest { 16 | @get:Rule 17 | val compose = createComposeRule() 18 | 19 | @Test 20 | fun firstScreen(): Unit = 21 | with(compose) { 22 | setContent { TestScreenNavigation() } 23 | 24 | onNode(hasSetTextAction()).run { 25 | assert(hasText("")) 26 | performTextInput("Foo") 27 | assert(hasText("Foo")) 28 | } 29 | } 30 | 31 | @Test 32 | fun navigateToSecondScreenAndUpdateItsState() { 33 | with(compose) { 34 | setContent { TestScreenNavigation() } 35 | 36 | onNodeWithText("Push Second Screen").performClick() 37 | waitForIdle() 38 | 39 | onNode(hasSetTextAction()).run { 40 | assert(hasText("")) 41 | performTextInput("Bar") 42 | assert(hasText("Bar")) 43 | } 44 | } 45 | } 46 | 47 | @Test 48 | fun navigateBackAndForth() { 49 | with(compose) { 50 | setContent { TestScreenNavigation() } 51 | 52 | onNode(hasSetTextAction()).run { 53 | assert(hasText("")) 54 | performTextInput("Foo") 55 | assert(hasText("Foo")) 56 | } 57 | 58 | onNodeWithText("Push Second Screen").performClick() 59 | waitForIdle() 60 | 61 | onNode(hasSetTextAction()).run { 62 | assert(hasText("Foo")) 63 | performTextClearance() 64 | performTextInput("The quick brown fox") 65 | assert(hasText("The quick brown fox")) 66 | } 67 | 68 | onNodeWithText("Pop Screen").performClick() 69 | waitForIdle() 70 | 71 | onNode(hasSetTextAction()).run { 72 | assert(hasText("Foo")) 73 | performTextClearance() 74 | performTextInput("FooBar") 75 | assert(hasText("FooBar")) 76 | } 77 | 78 | onNodeWithText("Push Second Screen").performClick() 79 | waitForIdle() 80 | 81 | onNode(hasSetTextAction()).assert(hasText("FooBar")) 82 | } 83 | } 84 | } 85 | -------------------------------------------------------------------------------- /navigation-screens-viewmodel/src/androidInstrumentedTest/kotlin/io/github/lukwol/navigation/screens/viewmodel/components/components/FirstScreen.kt: -------------------------------------------------------------------------------- 1 | package io.github.lukwol.navigation.screens.viewmodel.components.components 2 | 3 | import androidx.compose.foundation.layout.Column 4 | import androidx.compose.material3.Button 5 | import androidx.compose.material3.Text 6 | import androidx.compose.material3.TextField 7 | import androidx.compose.runtime.Composable 8 | import io.github.lukwol.navigation.screens.LocalScreensController 9 | import io.github.lukwol.navigation.screens.viewmodel.components.data.TestRoutes 10 | 11 | @Suppress("TestFunctionName") 12 | @Composable 13 | fun FirstScreen(viewModel: FirstScreenViewModel) { 14 | val screensController = LocalScreensController.current 15 | 16 | Column { 17 | TextField( 18 | value = viewModel.text, 19 | onValueChange = { viewModel.text = it }, 20 | ) 21 | 22 | Button( 23 | onClick = { screensController.push(TestRoutes.SecondScreen, viewModel.text) }, 24 | ) { 25 | Text("Push Second Screen") 26 | } 27 | } 28 | } 29 | -------------------------------------------------------------------------------- /navigation-screens-viewmodel/src/androidInstrumentedTest/kotlin/io/github/lukwol/navigation/screens/viewmodel/components/components/FirstScreenViewModel.kt: -------------------------------------------------------------------------------- 1 | package io.github.lukwol.navigation.screens.viewmodel.components.components 2 | 3 | import androidx.compose.runtime.getValue 4 | import androidx.compose.runtime.mutableStateOf 5 | import androidx.compose.runtime.setValue 6 | import io.github.lukwol.navigation.screens.viewmodel.ViewModel 7 | 8 | class FirstScreenViewModel : ViewModel() { 9 | var text by mutableStateOf("") 10 | } 11 | -------------------------------------------------------------------------------- /navigation-screens-viewmodel/src/androidInstrumentedTest/kotlin/io/github/lukwol/navigation/screens/viewmodel/components/components/SecondScreen.kt: -------------------------------------------------------------------------------- 1 | package io.github.lukwol.navigation.screens.viewmodel.components.components 2 | 3 | import androidx.compose.foundation.layout.Column 4 | import androidx.compose.material3.Button 5 | import androidx.compose.material3.Text 6 | import androidx.compose.material3.TextField 7 | import androidx.compose.runtime.Composable 8 | import io.github.lukwol.navigation.screens.LocalScreensController 9 | 10 | @Suppress("TestFunctionName") 11 | @Composable 12 | fun SecondScreen(viewModel: SecondScreenViewModel) { 13 | val screensController = LocalScreensController.current 14 | 15 | Column { 16 | TextField( 17 | value = viewModel.text.orEmpty(), 18 | onValueChange = { viewModel.text = it }, 19 | ) 20 | 21 | Button( 22 | onClick = { screensController.pop() }, 23 | ) { 24 | Text("Pop Screen") 25 | } 26 | } 27 | } 28 | -------------------------------------------------------------------------------- /navigation-screens-viewmodel/src/androidInstrumentedTest/kotlin/io/github/lukwol/navigation/screens/viewmodel/components/components/SecondScreenViewModel.kt: -------------------------------------------------------------------------------- 1 | package io.github.lukwol.navigation.screens.viewmodel.components.components 2 | 3 | import androidx.compose.runtime.getValue 4 | import androidx.compose.runtime.mutableStateOf 5 | import androidx.compose.runtime.setValue 6 | import io.github.lukwol.navigation.screens.viewmodel.ViewModel 7 | 8 | class SecondScreenViewModel : ViewModel() { 9 | var text: String? by mutableStateOf(null) 10 | } 11 | -------------------------------------------------------------------------------- /navigation-screens-viewmodel/src/androidInstrumentedTest/kotlin/io/github/lukwol/navigation/screens/viewmodel/components/components/TestScreenNavigation.kt: -------------------------------------------------------------------------------- 1 | package io.github.lukwol.navigation.screens.viewmodel.components.components 2 | 3 | import androidx.compose.runtime.Composable 4 | import androidx.lifecycle.viewmodel.compose.viewModel 5 | import io.github.lukwol.navigation.screens.viewmodel.ScreensNavigation 6 | import io.github.lukwol.navigation.screens.viewmodel.components.data.TestRoutes 7 | 8 | @Suppress("TestFunctionName") 9 | @Composable 10 | fun TestScreenNavigation() { 11 | ScreensNavigation( 12 | startRoute = TestRoutes.FirstScreen, 13 | ) { 14 | screen( 15 | route = TestRoutes.FirstScreen, 16 | viewModelFactory = { viewModel() }, 17 | ) { viewModel -> 18 | FirstScreen(viewModel) 19 | } 20 | 21 | screen( 22 | route = TestRoutes.SecondScreen, 23 | viewModelWithArgs = { args: String? -> 24 | viewModel().apply { text = args } 25 | }, 26 | ) { viewModel -> 27 | SecondScreen(viewModel) 28 | } 29 | } 30 | } 31 | -------------------------------------------------------------------------------- /navigation-screens-viewmodel/src/androidInstrumentedTest/kotlin/io/github/lukwol/navigation/screens/viewmodel/components/data/TestRoutes.kt: -------------------------------------------------------------------------------- 1 | package io.github.lukwol.navigation.screens.viewmodel.components.data 2 | 3 | object TestRoutes { 4 | const val FirstScreen = "first-screen" 5 | const val SecondScreen = "second-screen" 6 | } 7 | -------------------------------------------------------------------------------- /navigation-screens-viewmodel/src/androidMain/kotlin/io/github/lukwol/navigation/screens/viewmodel/NavArgs.kt: -------------------------------------------------------------------------------- 1 | package io.github.lukwol.navigation.screens.viewmodel 2 | 3 | import androidx.lifecycle.SavedStateHandle 4 | import io.github.lukwol.navigation.screens.NavArgsKey 5 | import kotlinx.serialization.json.Json 6 | 7 | /** 8 | * Accesses and decode navigation arguments from [SavedStateHandle]. 9 | * 10 | * @param Args type of arguments that will be decoded from JSON 11 | * @return Decoded navigation arguments 12 | */ 13 | @Suppress("unused") 14 | inline fun SavedStateHandle.navArgs(): Args? = 15 | get(NavArgsKey) 16 | ?.let(Json.Default::decodeFromString) 17 | -------------------------------------------------------------------------------- /navigation-screens-viewmodel/src/androidMain/kotlin/io/github/lukwol/navigation/screens/viewmodel/ScreensNavigation.kt: -------------------------------------------------------------------------------- 1 | package io.github.lukwol.navigation.screens.viewmodel 2 | 3 | import androidx.compose.animation.AnimatedContentTransitionScope 4 | import androidx.compose.animation.EnterTransition 5 | import androidx.compose.animation.ExitTransition 6 | import androidx.compose.runtime.Composable 7 | import io.github.lukwol.navigation.screens.ScreensNavigation as BasicScreensNavigation 8 | 9 | @Composable 10 | actual fun ScreensNavigation( 11 | startRoute: String, 12 | enterTransition: (AnimatedContentTransitionScope<*>.() -> EnterTransition), 13 | exitTransition: (AnimatedContentTransitionScope<*>.() -> ExitTransition), 14 | popEnterTransition: (AnimatedContentTransitionScope<*>.() -> EnterTransition), 15 | popExitTransition: (AnimatedContentTransitionScope<*>.() -> ExitTransition), 16 | builder: VMScreensMapBuilder.() -> Unit, 17 | ) { 18 | BasicScreensNavigation( 19 | startRoute = startRoute, 20 | enterTransition = enterTransition, 21 | exitTransition = exitTransition, 22 | popEnterTransition = popEnterTransition, 23 | popExitTransition = popExitTransition, 24 | ) { 25 | builder(VMScreensMapBuilder(this)) 26 | } 27 | } 28 | -------------------------------------------------------------------------------- /navigation-screens-viewmodel/src/androidMain/kotlin/io/github/lukwol/navigation/screens/viewmodel/VMScreensMapBuilder.kt: -------------------------------------------------------------------------------- 1 | package io.github.lukwol.navigation.screens.viewmodel 2 | 3 | import androidx.compose.runtime.Composable 4 | import io.github.lukwol.navigation.screens.ScreensMapBuilder 5 | 6 | actual class VMScreensMapBuilder( 7 | actual val screensMapBuilder: ScreensMapBuilder, 8 | ) { 9 | actual inline fun screen( 10 | route: String, 11 | noinline viewModelWithArgs: @Composable (args: Args?) -> VM, 12 | noinline content: @Composable (viewModel: VM) -> Unit, 13 | ) = screensMapBuilder.screen(route) { args: Args? -> 14 | val viewModel = viewModelWithArgs(args) 15 | content(viewModel) 16 | } 17 | 18 | actual fun screen( 19 | route: String, 20 | viewModelFactory: @Composable () -> VM, 21 | content: @Composable (viewModel: VM) -> Unit, 22 | ): Unit = 23 | screen( 24 | route = route, 25 | viewModelWithArgs = { _: Any? -> 26 | viewModelFactory() 27 | }, 28 | content = content, 29 | ) 30 | } 31 | -------------------------------------------------------------------------------- /navigation-screens-viewmodel/src/androidMain/kotlin/io/github/lukwol/navigation/screens/viewmodel/ViewModel.kt: -------------------------------------------------------------------------------- 1 | package io.github.lukwol.navigation.screens.viewmodel 2 | 3 | import androidx.lifecycle.viewModelScope 4 | import androidx.lifecycle.ViewModel as AndroidViewModel 5 | 6 | actual abstract class ViewModel : AndroidViewModel() { 7 | actual val coroutineScope get() = viewModelScope 8 | } 9 | -------------------------------------------------------------------------------- /navigation-screens-viewmodel/src/commonMain/kotlin/io/github/lukwol/navigation/screens/viewmodel/ScreensNavigation.kt: -------------------------------------------------------------------------------- 1 | package io.github.lukwol.navigation.screens.viewmodel 2 | 3 | import androidx.compose.animation.AnimatedContentTransitionScope 4 | import androidx.compose.animation.EnterTransition 5 | import androidx.compose.animation.ExitTransition 6 | import androidx.compose.animation.core.tween 7 | import androidx.compose.animation.fadeIn 8 | import androidx.compose.animation.fadeOut 9 | import androidx.compose.runtime.Composable 10 | import io.github.lukwol.navigation.screens.ScreensNavigation as BasicScreensNavigation 11 | 12 | /** 13 | * Declare screens navigation same as [ScreensNavigation][BasicScreensNavigation] 14 | * but additionally allows declaration of screens with [ViewModel]. 15 | * 16 | * Has separate implementations for Android and non Android targets. 17 | * 18 | * Both implementation uses [ScreensNavigation] underneath and adds support for lifecycle aware [ViewModel]. 19 | * 20 | * Android target utilizes official ViewModel class from **Lifecycle** framework to leverage 21 | * view models persistence, management and make them lifecycle aware. 22 | * 23 | * Other targets share same [ViewModel] implementation and takes care 24 | * of cancelling [ViewModel.coroutineScope] when screens or windows are disposed. 25 | */ 26 | @Composable 27 | expect fun ScreensNavigation( 28 | startRoute: String, 29 | enterTransition: (AnimatedContentTransitionScope<*>.() -> EnterTransition) = 30 | { fadeIn(animationSpec = tween(700)) }, 31 | exitTransition: (AnimatedContentTransitionScope<*>.() -> ExitTransition) = 32 | { fadeOut(animationSpec = tween(700)) }, 33 | popEnterTransition: (AnimatedContentTransitionScope<*>.() -> EnterTransition) = 34 | enterTransition, 35 | popExitTransition: (AnimatedContentTransitionScope<*>.() -> ExitTransition) = 36 | exitTransition, 37 | builder: VMScreensMapBuilder.() -> Unit, 38 | ) 39 | -------------------------------------------------------------------------------- /navigation-screens-viewmodel/src/commonMain/kotlin/io/github/lukwol/navigation/screens/viewmodel/VMScreensMapBuilder.kt: -------------------------------------------------------------------------------- 1 | package io.github.lukwol.navigation.screens.viewmodel 2 | 3 | import androidx.compose.runtime.Composable 4 | import io.github.lukwol.navigation.screens.ScreensMapBuilder 5 | 6 | /** 7 | * Builds screens navigation map same as [ScreensMapBuilder], 8 | * additionally allows passing [ViewModel] via lambda. 9 | * 10 | * Has separate implementations for Android and non Android targets. 11 | * 12 | * Both implementation uses [ScreensMapBuilder] underneath and adds support for lifecycle aware [ViewModel]. 13 | * 14 | * Non Android implementation manually manages [ViewModel] persistence 15 | * using `ViewModelStore` and cancels [ViewModel.coroutineScope] whenever screen or window is disposed. 16 | * 17 | * Android implementation relies on built in solutions from Google **Lifecycle* framework. 18 | */ 19 | expect class VMScreensMapBuilder { 20 | @PublishedApi 21 | internal val screensMapBuilder: ScreensMapBuilder 22 | 23 | /** 24 | * Declare screen [content] for [route] and add it to underlying screens map. 25 | * The screens map will be used to navigate between the screens. 26 | * 27 | * @param VM generic type of [ViewModel] 28 | * @param route used to identify the screen when navigating, in order to present it's content 29 | * @param viewModelWithArgs lambda that takes screen arguments as a parameter 30 | * and creates [ViewModel] that is later passed to screen [content] lambda 31 | * @param content [Composable] lambda with content of the screen that takes 32 | * [view model][VM] as a parameter 33 | * 34 | * @throws IllegalArgumentException when attempting to add new screen for already registered [route] 35 | */ 36 | inline fun screen( 37 | route: String, 38 | noinline viewModelWithArgs: @Composable (args: Args?) -> VM, 39 | noinline content: @Composable (viewModel: VM) -> Unit, 40 | ) 41 | 42 | /** 43 | * Alternative implementation of [screen] that does not pass any arguments. 44 | * 45 | * @see screen 46 | */ 47 | fun screen( 48 | route: String, 49 | viewModelFactory: @Composable () -> VM, 50 | content: @Composable (viewModel: VM) -> Unit, 51 | ) 52 | } 53 | 54 | /** 55 | * Implementation forwarded from ScreensNavigation 56 | * which does not involve [ViewModel] initialization. 57 | * 58 | * @see ScreensMapBuilder.screen 59 | */ 60 | inline fun VMScreensMapBuilder.screen( 61 | route: String, 62 | noinline content: @Composable (args: Args?) -> Unit, 63 | ) = screensMapBuilder.screen(route, content) 64 | 65 | /** 66 | * Implementation forwarded from ScreensNavigation that does not pass any arguments 67 | * and does not involve [ViewModel] initialization. 68 | * 69 | * @see ScreensMapBuilder.screen 70 | */ 71 | fun VMScreensMapBuilder.screen( 72 | route: String, 73 | content: @Composable () -> Unit, 74 | ) = screensMapBuilder.screen(route, content) 75 | -------------------------------------------------------------------------------- /navigation-screens-viewmodel/src/commonMain/kotlin/io/github/lukwol/navigation/screens/viewmodel/ViewModel.kt: -------------------------------------------------------------------------------- 1 | package io.github.lukwol.navigation.screens.viewmodel 2 | 3 | import kotlinx.coroutines.CoroutineScope 4 | 5 | /** 6 | * [ViewModel] is tied to specific view, it exposes the view state and encapsulates related business logic. 7 | * 8 | * When used in a combination with screens and/or windows navigation, [ViewModel] is tightly integrated 9 | * into screens and windows navigation, by exposing lifecycle aware [coroutineScope], 10 | * that is canceled whenever screen or window is dismissed. 11 | * 12 | * [ViewModel] allows to persists the view state when screen is not currently active until it's dismissed. 13 | * 14 | * [ViewModel] separate implementations for Android and non Android targets. 15 | * 16 | * On Android [ViewModel] inherits from official Android `ViewModel` class and [coroutineScope] 17 | * is just another way to reference `viewModelScope`. 18 | * 19 | * Other targets have common implementation where `ViewModel` does not inherit from any class 20 | * and creates it's own new `coroutineScope` when instantiating this class. 21 | */ 22 | expect abstract class ViewModel() { 23 | /** 24 | * This scope will be canceled when view connected to this CoroutineScopeAware implementation is cleared. 25 | */ 26 | val coroutineScope: CoroutineScope 27 | } 28 | -------------------------------------------------------------------------------- /navigation-screens-viewmodel/src/desktopTest/kotlin/io/github/lukwol/navigation/screens/viewmodel/components/ViewModelScreenNavigationTest.kt: -------------------------------------------------------------------------------- 1 | package io.github.lukwol.navigation.screens.viewmodel.components 2 | 3 | import androidx.compose.ui.test.assert 4 | import androidx.compose.ui.test.hasSetTextAction 5 | import androidx.compose.ui.test.hasText 6 | import androidx.compose.ui.test.junit4.createComposeRule 7 | import androidx.compose.ui.test.onNodeWithText 8 | import androidx.compose.ui.test.performClick 9 | import androidx.compose.ui.test.performTextClearance 10 | import androidx.compose.ui.test.performTextInput 11 | import io.github.lukwol.navigation.screens.viewmodel.components.components.TestScreenNavigation 12 | import org.junit.Rule 13 | import org.junit.Test 14 | 15 | class ViewModelScreenNavigationTest { 16 | @get:Rule 17 | val compose = createComposeRule() 18 | 19 | @Test 20 | fun firstScreen(): Unit = 21 | with(compose) { 22 | setContent { TestScreenNavigation() } 23 | 24 | onNode(hasSetTextAction()).run { 25 | assert(hasText("")) 26 | performTextInput("Foo") 27 | assert(hasText("Foo")) 28 | } 29 | } 30 | 31 | @Test 32 | fun navigateToSecondScreenAndUpdateItsState() { 33 | with(compose) { 34 | setContent { TestScreenNavigation() } 35 | 36 | onNodeWithText("Push Second Screen").performClick() 37 | waitForIdle() 38 | 39 | onNode(hasSetTextAction()).run { 40 | assert(hasText("")) 41 | performTextInput("Bar") 42 | assert(hasText("Bar")) 43 | } 44 | } 45 | } 46 | 47 | @Test 48 | fun navigateBackAndForth() { 49 | with(compose) { 50 | setContent { TestScreenNavigation() } 51 | 52 | onNode(hasSetTextAction()).run { 53 | assert(hasText("")) 54 | performTextInput("Foo") 55 | assert(hasText("Foo")) 56 | } 57 | 58 | onNodeWithText("Push Second Screen").performClick() 59 | waitForIdle() 60 | 61 | onNode(hasSetTextAction()).run { 62 | assert(hasText("Foo")) 63 | performTextClearance() 64 | performTextInput("The quick brown fox") 65 | assert(hasText("The quick brown fox")) 66 | } 67 | 68 | onNodeWithText("Pop Screen").performClick() 69 | waitForIdle() 70 | 71 | onNode(hasSetTextAction()).run { 72 | assert(hasText("Foo")) 73 | performTextClearance() 74 | performTextInput("FooBar") 75 | assert(hasText("FooBar")) 76 | } 77 | 78 | onNodeWithText("Push Second Screen").performClick() 79 | waitForIdle() 80 | 81 | onNode(hasSetTextAction()).assert(hasText("FooBar")) 82 | } 83 | } 84 | } 85 | -------------------------------------------------------------------------------- /navigation-screens-viewmodel/src/desktopTest/kotlin/io/github/lukwol/navigation/screens/viewmodel/components/components/FirstScreen.kt: -------------------------------------------------------------------------------- 1 | package io.github.lukwol.navigation.screens.viewmodel.components.components 2 | 3 | import androidx.compose.foundation.layout.Column 4 | import androidx.compose.material3.Button 5 | import androidx.compose.material3.Text 6 | import androidx.compose.material3.TextField 7 | import androidx.compose.runtime.Composable 8 | import io.github.lukwol.navigation.screens.LocalScreensController 9 | import io.github.lukwol.navigation.screens.viewmodel.components.data.TestRoutes 10 | 11 | @Suppress("TestFunctionName") 12 | @Composable 13 | fun FirstScreen(viewModel: FirstScreenViewModel) { 14 | val screensController = LocalScreensController.current 15 | 16 | Column { 17 | TextField( 18 | value = viewModel.text, 19 | onValueChange = { viewModel.text = it }, 20 | ) 21 | 22 | Button( 23 | onClick = { screensController.push(TestRoutes.SecondScreen, viewModel.text) }, 24 | ) { 25 | Text("Push Second Screen") 26 | } 27 | } 28 | } 29 | -------------------------------------------------------------------------------- /navigation-screens-viewmodel/src/desktopTest/kotlin/io/github/lukwol/navigation/screens/viewmodel/components/components/FirstScreenViewModel.kt: -------------------------------------------------------------------------------- 1 | package io.github.lukwol.navigation.screens.viewmodel.components.components 2 | 3 | import androidx.compose.runtime.getValue 4 | import androidx.compose.runtime.mutableStateOf 5 | import androidx.compose.runtime.setValue 6 | import io.github.lukwol.navigation.screens.viewmodel.ViewModel 7 | 8 | class FirstScreenViewModel : ViewModel() { 9 | var text by mutableStateOf("") 10 | } 11 | -------------------------------------------------------------------------------- /navigation-screens-viewmodel/src/desktopTest/kotlin/io/github/lukwol/navigation/screens/viewmodel/components/components/SecondScreen.kt: -------------------------------------------------------------------------------- 1 | package io.github.lukwol.navigation.screens.viewmodel.components.components 2 | 3 | import androidx.compose.foundation.layout.Column 4 | import androidx.compose.material3.Button 5 | import androidx.compose.material3.Text 6 | import androidx.compose.material3.TextField 7 | import androidx.compose.runtime.Composable 8 | import io.github.lukwol.navigation.screens.LocalScreensController 9 | 10 | @Suppress("TestFunctionName") 11 | @Composable 12 | fun SecondScreen(viewModel: SecondScreenViewModel) { 13 | val screensController = LocalScreensController.current 14 | 15 | Column { 16 | TextField( 17 | value = viewModel.text.orEmpty(), 18 | onValueChange = { viewModel.text = it }, 19 | ) 20 | 21 | Button( 22 | onClick = { screensController.pop() }, 23 | ) { 24 | Text("Pop Screen") 25 | } 26 | } 27 | } 28 | -------------------------------------------------------------------------------- /navigation-screens-viewmodel/src/desktopTest/kotlin/io/github/lukwol/navigation/screens/viewmodel/components/components/SecondScreenViewModel.kt: -------------------------------------------------------------------------------- 1 | package io.github.lukwol.navigation.screens.viewmodel.components.components 2 | 3 | import androidx.compose.runtime.getValue 4 | import androidx.compose.runtime.mutableStateOf 5 | import androidx.compose.runtime.setValue 6 | import io.github.lukwol.navigation.screens.viewmodel.ViewModel 7 | 8 | class SecondScreenViewModel : ViewModel() { 9 | var text: String? by mutableStateOf(null) 10 | } 11 | -------------------------------------------------------------------------------- /navigation-screens-viewmodel/src/desktopTest/kotlin/io/github/lukwol/navigation/screens/viewmodel/components/components/TestScreenNavigation.kt: -------------------------------------------------------------------------------- 1 | package io.github.lukwol.navigation.screens.viewmodel.components.components 2 | 3 | import androidx.compose.runtime.Composable 4 | import io.github.lukwol.navigation.screens.viewmodel.ScreensNavigation 5 | import io.github.lukwol.navigation.screens.viewmodel.components.data.TestRoutes 6 | 7 | @Suppress("TestFunctionName") 8 | @Composable 9 | fun TestScreenNavigation() { 10 | ScreensNavigation( 11 | startRoute = TestRoutes.FirstScreen, 12 | ) { 13 | screen( 14 | route = TestRoutes.FirstScreen, 15 | viewModelFactory = { FirstScreenViewModel() }, 16 | ) { viewModel -> 17 | FirstScreen(viewModel) 18 | } 19 | 20 | screen( 21 | route = TestRoutes.SecondScreen, 22 | viewModelWithArgs = { args: String? -> 23 | SecondScreenViewModel().apply { text = args } 24 | }, 25 | ) { viewModel -> 26 | SecondScreen(viewModel) 27 | } 28 | } 29 | } 30 | -------------------------------------------------------------------------------- /navigation-screens-viewmodel/src/desktopTest/kotlin/io/github/lukwol/navigation/screens/viewmodel/components/data/TestRoutes.kt: -------------------------------------------------------------------------------- 1 | package io.github.lukwol.navigation.screens.viewmodel.components.data 2 | 3 | object TestRoutes { 4 | const val FirstScreen = "first-screen" 5 | const val SecondScreen = "second-screen" 6 | } 7 | -------------------------------------------------------------------------------- /navigation-screens-viewmodel/src/nonAndroidMain/kotlin/io/github/lukwol/navigation/screens/viewmodel/ScreensNavigation.kt: -------------------------------------------------------------------------------- 1 | package io.github.lukwol.navigation.screens.viewmodel 2 | 3 | import androidx.compose.animation.AnimatedContentTransitionScope 4 | import androidx.compose.animation.EnterTransition 5 | import androidx.compose.animation.ExitTransition 6 | import androidx.compose.runtime.Composable 7 | import androidx.compose.runtime.DisposableEffect 8 | import androidx.compose.runtime.remember 9 | import kotlinx.coroutines.CoroutineScope 10 | import kotlinx.coroutines.cancel 11 | import io.github.lukwol.navigation.screens.ScreensNavigation as BasicScreensNavigation 12 | 13 | @Composable 14 | actual fun ScreensNavigation( 15 | startRoute: String, 16 | enterTransition: (AnimatedContentTransitionScope<*>.() -> EnterTransition), 17 | exitTransition: (AnimatedContentTransitionScope<*>.() -> ExitTransition), 18 | popEnterTransition: (AnimatedContentTransitionScope<*>.() -> EnterTransition), 19 | popExitTransition: (AnimatedContentTransitionScope<*>.() -> ExitTransition), 20 | builder: VMScreensMapBuilder.() -> Unit, 21 | ) { 22 | val viewModelStore = remember { ViewModelStore() } 23 | 24 | BasicScreensNavigation( 25 | startRoute = startRoute, 26 | enterTransition = enterTransition, 27 | exitTransition = exitTransition, 28 | popEnterTransition = popEnterTransition, 29 | popExitTransition = popExitTransition, 30 | ) { 31 | builder(VMScreensMapBuilder(viewModelStore, this)) 32 | } 33 | 34 | DisposableEffect(Unit) { 35 | onDispose { 36 | viewModelStore.values 37 | .map(ViewModel::coroutineScope) 38 | .forEach(CoroutineScope::cancel) 39 | } 40 | } 41 | } 42 | -------------------------------------------------------------------------------- /navigation-screens-viewmodel/src/nonAndroidMain/kotlin/io/github/lukwol/navigation/screens/viewmodel/VMScreensMapBuilder.kt: -------------------------------------------------------------------------------- 1 | package io.github.lukwol.navigation.screens.viewmodel 2 | 3 | import androidx.compose.runtime.Composable 4 | import androidx.compose.runtime.LaunchedEffect 5 | import io.github.lukwol.navigation.screens.LocalScreensController 6 | import io.github.lukwol.navigation.screens.ScreensMapBuilder 7 | import kotlinx.coroutines.cancel 8 | 9 | actual class VMScreensMapBuilder internal constructor( 10 | @PublishedApi 11 | internal val viewModelStore: ViewModelStore, 12 | @PublishedApi 13 | internal actual val screensMapBuilder: ScreensMapBuilder = ScreensMapBuilder(), 14 | ) { 15 | @Suppress("UNCHECKED_CAST") 16 | actual inline fun screen( 17 | route: String, 18 | noinline viewModelWithArgs: @Composable (args: Args?) -> VM, 19 | noinline content: @Composable (viewModel: VM) -> Unit, 20 | ) = screensMapBuilder.screen(route) { args: Args? -> 21 | val screensController = LocalScreensController.current 22 | 23 | val viewModel = viewModelStore.getOrPut(route) { viewModelWithArgs(args) } as VM 24 | content(viewModel) 25 | 26 | LaunchedEffect(route) { 27 | viewModelStore 28 | .keys 29 | .filterNot { it in screensController.routes } 30 | .forEach { disposedRoute -> 31 | viewModelStore[disposedRoute]?.coroutineScope?.cancel() 32 | viewModelStore.remove(disposedRoute) 33 | } 34 | } 35 | } 36 | 37 | actual fun screen( 38 | route: String, 39 | viewModelFactory: @Composable () -> VM, 40 | content: @Composable (viewModel: VM) -> Unit, 41 | ): Unit = 42 | screen( 43 | route = route, 44 | viewModelWithArgs = { _: Any? -> 45 | viewModelFactory() 46 | }, 47 | content = content, 48 | ) 49 | } 50 | -------------------------------------------------------------------------------- /navigation-screens-viewmodel/src/nonAndroidMain/kotlin/io/github/lukwol/navigation/screens/viewmodel/ViewModel.kt: -------------------------------------------------------------------------------- 1 | package io.github.lukwol.navigation.screens.viewmodel 2 | 3 | import kotlinx.coroutines.CoroutineScope 4 | import kotlinx.coroutines.Dispatchers 5 | import kotlinx.coroutines.SupervisorJob 6 | 7 | actual abstract class ViewModel { 8 | actual val coroutineScope = 9 | CoroutineScope( 10 | context = SupervisorJob() + Dispatchers.Main.immediate, 11 | ) 12 | } 13 | 14 | val ViewModel.viewModelScope get() = coroutineScope 15 | -------------------------------------------------------------------------------- /navigation-screens-viewmodel/src/nonAndroidMain/kotlin/io/github/lukwol/navigation/screens/viewmodel/ViewModelStore.kt: -------------------------------------------------------------------------------- 1 | package io.github.lukwol.navigation.screens.viewmodel 2 | 3 | /** 4 | * View model store is responsible for persisting [view models][ViewModel] state, 5 | * for screens that are not currently displayed but are on the navigation stack. 6 | * 7 | * @property viewModels underlying mutable map 8 | * that registers [view models][ViewModel] for screen routes 9 | */ 10 | @PublishedApi 11 | internal class ViewModelStore private constructor( 12 | private val viewModels: MutableMap, 13 | ) : MutableMap by viewModels { 14 | constructor() : this(mutableMapOf()) 15 | } 16 | -------------------------------------------------------------------------------- /navigation-screens/build.gradle.kts: -------------------------------------------------------------------------------- 1 | import org.jetbrains.kotlin.gradle.targets.js.dsl.ExperimentalWasmDsl 2 | 3 | plugins { 4 | alias(commonLibs.plugins.kotlin.multiplatform) 5 | alias(commonLibs.plugins.compose.multiplatform) 6 | alias(androidLibs.plugins.library) 7 | alias(commonLibs.plugins.kotlin.serialization) 8 | } 9 | 10 | kotlin { 11 | applyDefaultHierarchyTemplate() 12 | 13 | @OptIn(ExperimentalWasmDsl::class) 14 | wasmJs { 15 | browser() 16 | } 17 | 18 | jvm("desktop") 19 | 20 | androidTarget { 21 | publishLibraryVariants("release") 22 | } 23 | 24 | iosX64() 25 | iosArm64() 26 | iosSimulatorArm64() 27 | 28 | sourceSets { 29 | commonMain { 30 | dependencies { 31 | implementation(compose.animation) 32 | } 33 | } 34 | androidMain { 35 | dependencies { 36 | implementation(androidLibs.compose.navigation) 37 | implementation(commonLibs.kotlin.serialization.json) 38 | } 39 | } 40 | create("nonAndroidMain") { 41 | dependsOn(getByName("commonMain")) 42 | } 43 | getByName("desktopMain") { 44 | dependsOn(getByName("nonAndroidMain")) 45 | } 46 | iosMain { 47 | dependsOn(getByName("nonAndroidMain")) 48 | } 49 | getByName("wasmJsMain") { 50 | dependsOn(getByName("nonAndroidMain")) 51 | } 52 | getByName("androidInstrumentedTest").dependencies { 53 | implementation(kotlin("test")) 54 | implementation(compose.material3) 55 | implementation(commonLibs.kotlin.serialization.json) 56 | implementation(androidLibs.test.runner) 57 | implementation(androidLibs.compose.ui.test.junit4) 58 | } 59 | getByName("desktopTest").dependencies { 60 | implementation(kotlin("test")) 61 | implementation(compose.material3) 62 | implementation(compose.desktop.currentOs) 63 | implementation(compose.desktop.uiTestJUnit4) 64 | } 65 | } 66 | } 67 | 68 | android { 69 | compileSdk = androidLibs.versions.compile.sdk.get().toInt() 70 | namespace = "io.github.lukwol.navigation.screens" 71 | 72 | defaultConfig { 73 | minSdk = androidLibs.versions.min.sdk.get().toInt() 74 | testInstrumentationRunner = "androidx.test.runner.AndroidJUnitRunner" 75 | } 76 | 77 | compileOptions { 78 | sourceCompatibility = JavaVersion.VERSION_21 79 | targetCompatibility = JavaVersion.VERSION_21 80 | } 81 | 82 | buildFeatures { 83 | compose = true 84 | } 85 | composeOptions { 86 | kotlinCompilerExtensionVersion = androidLibs.versions.compose.compiler.get() 87 | } 88 | 89 | dependencies { 90 | debugImplementation(androidLibs.compose.ui.test.manifest) 91 | } 92 | } 93 | -------------------------------------------------------------------------------- /navigation-screens/src/androidInstrumentedTest/kotlin/io/github/lukwol/navigation/screens/android/ScreensNavigationTest.kt: -------------------------------------------------------------------------------- 1 | package io.github.lukwol.navigation.screens.android 2 | 3 | import androidx.compose.ui.test.junit4.createComposeRule 4 | import androidx.compose.ui.test.onNodeWithText 5 | import androidx.compose.ui.test.performClick 6 | import io.github.lukwol.navigation.screens.ScreensNavigation 7 | import io.github.lukwol.navigation.screens.android.components.TestScreenNavigation 8 | import io.github.lukwol.navigation.screens.android.data.TestRoutes 9 | import org.junit.Rule 10 | import org.junit.Test 11 | import kotlin.test.assertFailsWith 12 | 13 | class ScreensNavigationTest { 14 | @get:Rule 15 | val compose = createComposeRule() 16 | 17 | @Test 18 | fun firstScreen(): Unit = 19 | with(compose) { 20 | setContent { TestScreenNavigation() } 21 | 22 | onNodeWithText("Push Second Screen").assertExists() 23 | } 24 | 25 | @Test 26 | fun navigateToSecondScreen() { 27 | with(compose) { 28 | setContent { TestScreenNavigation() } 29 | 30 | onNodeWithText("Push Second Screen").assertExists() 31 | onNodeWithText("Foo").assertDoesNotExist() 32 | 33 | onNodeWithText("Push Second Screen").performClick() 34 | waitForIdle() 35 | 36 | onNodeWithText("Push Second Screen").assertDoesNotExist() 37 | onNodeWithText("Foo").assertExists() 38 | } 39 | } 40 | 41 | @Test 42 | fun navigateToThirdScreen() { 43 | with(compose) { 44 | setContent { TestScreenNavigation() } 45 | 46 | onNodeWithText("Push Second Screen").performClick() 47 | waitForIdle() 48 | 49 | onNodeWithText("Foo").assertExists() 50 | 51 | onNodeWithText("Push Third Screen").performClick() 52 | waitForIdle() 53 | 54 | onNodeWithText("Foo").assertDoesNotExist() 55 | onNodeWithText("text = Bar, number = 42").assertExists() 56 | } 57 | } 58 | 59 | @Test 60 | fun navigateToThirdScreenThenPopToSecondScreen() { 61 | with(compose) { 62 | setContent { TestScreenNavigation() } 63 | 64 | onNodeWithText("Push Second Screen").performClick() 65 | waitForIdle() 66 | 67 | onNodeWithText("Push Third Screen").performClick() 68 | waitForIdle() 69 | 70 | onNodeWithText("text = Bar, number = 42").assertExists() 71 | onNodeWithText("Pop Screen").performClick() 72 | waitForIdle() 73 | 74 | onNodeWithText("text = Bar, number = 42").assertDoesNotExist() 75 | onNodeWithText("Foo").assertExists() 76 | } 77 | } 78 | 79 | @Test 80 | fun navigateToThirdScreenThenPopToFirstScreen() { 81 | with(compose) { 82 | setContent { TestScreenNavigation() } 83 | 84 | onNodeWithText("Push Second Screen").performClick() 85 | waitForIdle() 86 | 87 | onNodeWithText("Push Third Screen").performClick() 88 | waitForIdle() 89 | 90 | onNodeWithText("text = Bar, number = 42").assertExists() 91 | onNodeWithText("Pop To First Screen").performClick() 92 | waitForIdle() 93 | 94 | onNodeWithText("text = Bar, number = 42").assertDoesNotExist() 95 | onNodeWithText("Push Second Screen").assertExists() 96 | } 97 | } 98 | 99 | @Test 100 | fun missingStartRouteScreen() { 101 | assertFailsWith { 102 | compose.setContent { 103 | ScreensNavigation( 104 | startRoute = TestRoutes.FirstScreen, 105 | ) { 106 | screen(TestRoutes.SecondScreen) {} 107 | } 108 | } 109 | } 110 | } 111 | 112 | @Test 113 | fun emptyNavigationGraph() { 114 | assertFailsWith { 115 | compose.setContent { 116 | ScreensNavigation( 117 | startRoute = TestRoutes.FirstScreen, 118 | ) {} 119 | } 120 | } 121 | } 122 | } 123 | -------------------------------------------------------------------------------- /navigation-screens/src/androidInstrumentedTest/kotlin/io/github/lukwol/navigation/screens/android/components/FirstScreen.kt: -------------------------------------------------------------------------------- 1 | package io.github.lukwol.navigation.screens.android.components 2 | 3 | import androidx.compose.material3.Button 4 | import androidx.compose.material3.Text 5 | import androidx.compose.runtime.Composable 6 | import io.github.lukwol.navigation.screens.LocalScreensController 7 | import io.github.lukwol.navigation.screens.android.data.TestRoutes 8 | 9 | @Suppress("TestFunctionName") 10 | @Composable 11 | fun FirstScreen() { 12 | val screensController = LocalScreensController.current 13 | 14 | Button( 15 | onClick = { screensController.push(TestRoutes.SecondScreen, "Foo") }, 16 | ) { 17 | Text("Push Second Screen") 18 | } 19 | } 20 | -------------------------------------------------------------------------------- /navigation-screens/src/androidInstrumentedTest/kotlin/io/github/lukwol/navigation/screens/android/components/SecondScreen.kt: -------------------------------------------------------------------------------- 1 | package io.github.lukwol.navigation.screens.android.components 2 | 3 | import androidx.compose.foundation.layout.Column 4 | import androidx.compose.material3.Button 5 | import androidx.compose.material3.Text 6 | import androidx.compose.runtime.Composable 7 | import io.github.lukwol.navigation.screens.LocalScreensController 8 | import io.github.lukwol.navigation.screens.android.data.TestRoutes 9 | import io.github.lukwol.navigation.screens.android.data.ThirdScreenArgs 10 | 11 | @Suppress("TestFunctionName") 12 | @Composable 13 | fun SecondScreen(args: String) { 14 | val screensController = LocalScreensController.current 15 | 16 | Column { 17 | Text(args) 18 | Button( 19 | onClick = { 20 | screensController.push( 21 | TestRoutes.ThirdScreen, 22 | ThirdScreenArgs(text = "Bar", number = 42), 23 | ) 24 | }, 25 | ) { 26 | Text("Push Third Screen") 27 | } 28 | } 29 | } 30 | -------------------------------------------------------------------------------- /navigation-screens/src/androidInstrumentedTest/kotlin/io/github/lukwol/navigation/screens/android/components/TestScreenNavigation.kt: -------------------------------------------------------------------------------- 1 | package io.github.lukwol.navigation.screens.android.components 2 | 3 | import androidx.compose.runtime.Composable 4 | import io.github.lukwol.navigation.screens.ScreensNavigation 5 | import io.github.lukwol.navigation.screens.android.data.TestRoutes 6 | import io.github.lukwol.navigation.screens.android.data.ThirdScreenArgs 7 | 8 | @Suppress("TestFunctionName") 9 | @Composable 10 | fun TestScreenNavigation() { 11 | ScreensNavigation( 12 | startRoute = TestRoutes.FirstScreen, 13 | ) { 14 | screen(TestRoutes.FirstScreen) { 15 | FirstScreen() 16 | } 17 | 18 | screen(TestRoutes.SecondScreen) { args: String? -> 19 | SecondScreen(args!!) 20 | } 21 | 22 | screen(TestRoutes.ThirdScreen) { args: ThirdScreenArgs? -> 23 | ThirdScreen(args!!) 24 | } 25 | } 26 | } 27 | -------------------------------------------------------------------------------- /navigation-screens/src/androidInstrumentedTest/kotlin/io/github/lukwol/navigation/screens/android/components/ThirdScreen.kt: -------------------------------------------------------------------------------- 1 | package io.github.lukwol.navigation.screens.android.components 2 | 3 | import androidx.compose.foundation.layout.Column 4 | import androidx.compose.material3.Button 5 | import androidx.compose.material3.Text 6 | import androidx.compose.runtime.Composable 7 | import io.github.lukwol.navigation.screens.LocalScreensController 8 | import io.github.lukwol.navigation.screens.android.data.TestRoutes 9 | import io.github.lukwol.navigation.screens.android.data.ThirdScreenArgs 10 | 11 | @Suppress("TestFunctionName") 12 | @Composable 13 | fun ThirdScreen(args: ThirdScreenArgs) { 14 | val screensController = LocalScreensController.current 15 | 16 | Column { 17 | Text(args.toString()) 18 | Button( 19 | onClick = { screensController.pop() }, 20 | ) { 21 | Text("Pop Screen") 22 | } 23 | Button( 24 | onClick = { screensController.pop(upToRoute = TestRoutes.FirstScreen) }, 25 | ) { 26 | Text("Pop To First Screen") 27 | } 28 | } 29 | } 30 | -------------------------------------------------------------------------------- /navigation-screens/src/androidInstrumentedTest/kotlin/io/github/lukwol/navigation/screens/android/data/TestRoutes.kt: -------------------------------------------------------------------------------- 1 | package io.github.lukwol.navigation.screens.android.data 2 | 3 | object TestRoutes { 4 | const val FirstScreen = "first-screen" 5 | const val SecondScreen = "second-screen" 6 | const val ThirdScreen = "third-screen" 7 | } 8 | -------------------------------------------------------------------------------- /navigation-screens/src/androidInstrumentedTest/kotlin/io/github/lukwol/navigation/screens/android/data/ThirdScreenArgs.kt: -------------------------------------------------------------------------------- 1 | package io.github.lukwol.navigation.screens.android.data 2 | 3 | import kotlinx.serialization.Serializable 4 | 5 | @Suppress("PLUGIN_IS_NOT_ENABLED") 6 | @Serializable 7 | class ThirdScreenArgs( 8 | val text: String, 9 | val number: Int, 10 | ) { 11 | override fun toString() = "text = $text, number = $number" 12 | } 13 | -------------------------------------------------------------------------------- /navigation-screens/src/androidMain/kotlin/io/github/lukwol/navigation/screens/NavArgs.kt: -------------------------------------------------------------------------------- 1 | package io.github.lukwol.navigation.screens 2 | 3 | import androidx.lifecycle.SavedStateHandle 4 | 5 | /** 6 | * Key for which JSON encoded navigation arguments are stored in [SavedStateHandle] 7 | * and are appended to screen route. 8 | */ 9 | const val NavArgsKey = "navArgs" 10 | 11 | /** 12 | * Screen route to which optional navigation arguments can be appended. 13 | */ 14 | @PublishedApi 15 | internal fun appendNavArgs(screenRoute: String): String = "$screenRoute?$NavArgsKey={$NavArgsKey}" 16 | -------------------------------------------------------------------------------- /navigation-screens/src/androidMain/kotlin/io/github/lukwol/navigation/screens/ScreensController.kt: -------------------------------------------------------------------------------- 1 | package io.github.lukwol.navigation.screens 2 | 3 | import androidx.navigation.NavHostController 4 | import kotlinx.serialization.encodeToString 5 | import kotlinx.serialization.json.Json 6 | 7 | actual class ScreensController( 8 | val navController: NavHostController?, 9 | ) { 10 | actual constructor( 11 | startRoute: String, 12 | ) : this(null) 13 | 14 | actual inline fun push( 15 | route: String, 16 | args: Args, 17 | ) { 18 | navController?.navigate( 19 | route = "$route?$NavArgsKey=${Json.Default.encodeToString(args)}", 20 | ) 21 | } 22 | 23 | actual fun push(route: String) { 24 | navController?.navigate(route) 25 | } 26 | 27 | actual fun pop(upToRoute: String?) = 28 | if (upToRoute == null) { 29 | navController?.popBackStack() 30 | } else { 31 | navController?.popBackStack( 32 | route = appendNavArgs(upToRoute), 33 | inclusive = false, 34 | ) 35 | }.let { 36 | when (it) { 37 | true -> Result.success(Unit) 38 | else -> 39 | Result.failure( 40 | RuntimeException("Could not pop the back stack"), 41 | ) 42 | } 43 | } 44 | } 45 | -------------------------------------------------------------------------------- /navigation-screens/src/androidMain/kotlin/io/github/lukwol/navigation/screens/ScreensMapBuilder.kt: -------------------------------------------------------------------------------- 1 | package io.github.lukwol.navigation.screens 2 | 3 | import androidx.compose.runtime.Composable 4 | import androidx.navigation.NavGraphBuilder 5 | import androidx.navigation.compose.composable 6 | import kotlinx.serialization.json.Json 7 | 8 | actual open class ScreensMapBuilder( 9 | val navGraphBuilder: NavGraphBuilder, 10 | ) { 11 | actual inline fun screen( 12 | route: String, 13 | noinline content: @Composable (args: Args?) -> Unit, 14 | ) { 15 | navGraphBuilder.composable( 16 | route = appendNavArgs(route), 17 | ) { navBackStackEntry -> 18 | val args: Args? = 19 | navBackStackEntry 20 | .arguments 21 | ?.getString(NavArgsKey) 22 | ?.let(Json.Default::decodeFromString) 23 | content(args) 24 | } 25 | } 26 | 27 | actual fun screen( 28 | route: String, 29 | content: @Composable () -> Unit, 30 | ) = screen( 31 | route = route, 32 | content = { content() }, 33 | ) 34 | } 35 | -------------------------------------------------------------------------------- /navigation-screens/src/androidMain/kotlin/io/github/lukwol/navigation/screens/ScreensNavigation.kt: -------------------------------------------------------------------------------- 1 | package io.github.lukwol.navigation.screens 2 | 3 | import androidx.compose.animation.AnimatedContentTransitionScope 4 | import androidx.compose.animation.EnterTransition 5 | import androidx.compose.animation.ExitTransition 6 | import androidx.compose.runtime.Composable 7 | import androidx.compose.runtime.CompositionLocalProvider 8 | import androidx.navigation.compose.NavHost 9 | import androidx.navigation.compose.rememberNavController 10 | 11 | @Composable 12 | actual fun ScreensNavigation( 13 | startRoute: String, 14 | enterTransition: (AnimatedContentTransitionScope<*>.() -> EnterTransition), 15 | exitTransition: (AnimatedContentTransitionScope<*>.() -> ExitTransition), 16 | popEnterTransition: (AnimatedContentTransitionScope<*>.() -> EnterTransition), 17 | popExitTransition: (AnimatedContentTransitionScope<*>.() -> ExitTransition), 18 | builder: ScreensMapBuilder.() -> Unit, 19 | ) { 20 | val navController = rememberNavController() 21 | 22 | CompositionLocalProvider( 23 | LocalScreensController provides ScreensController(navController), 24 | ) { 25 | NavHost( 26 | navController = navController, 27 | enterTransition = enterTransition, 28 | exitTransition = exitTransition, 29 | popEnterTransition = popEnterTransition, 30 | popExitTransition = popExitTransition, 31 | startDestination = appendNavArgs(startRoute), 32 | ) { 33 | val mapBuilder = ScreensMapBuilder(this) 34 | builder(mapBuilder) 35 | } 36 | } 37 | } 38 | -------------------------------------------------------------------------------- /navigation-screens/src/commonMain/kotlin/io/github/lukwol/navigation/screens/LocalScreensController.kt: -------------------------------------------------------------------------------- 1 | package io.github.lukwol.navigation.screens 2 | 3 | import androidx.compose.runtime.compositionLocalOf 4 | 5 | /** 6 | * Provides local [ScreensController] that can be used 7 | * for pushing or popping screens from within currently displayed screen. 8 | * 9 | * Defaults value is set to prevent problems with screens previews. 10 | */ 11 | val LocalScreensController = compositionLocalOf { ScreensController("") } 12 | -------------------------------------------------------------------------------- /navigation-screens/src/commonMain/kotlin/io/github/lukwol/navigation/screens/RouteWithArgs.kt: -------------------------------------------------------------------------------- 1 | package io.github.lukwol.navigation.screens 2 | 3 | import androidx.compose.runtime.Immutable 4 | 5 | /** 6 | * A pair of screen route and screen [args]. 7 | */ 8 | @Immutable 9 | @PublishedApi 10 | internal data class RouteWithArgs( 11 | val route: String, 12 | val args: Any? = null, 13 | ) 14 | -------------------------------------------------------------------------------- /navigation-screens/src/commonMain/kotlin/io/github/lukwol/navigation/screens/ScreensController.kt: -------------------------------------------------------------------------------- 1 | package io.github.lukwol.navigation.screens 2 | 3 | /** 4 | * Manages navigation between the screens declared when building ScreensNavigation. 5 | * 6 | * Available via [LocalScreensController]. 7 | * 8 | * Has separate implementations for Android and non Android targets. 9 | */ 10 | expect class ScreensController(startRoute: String) { 11 | /** 12 | * Pushes new screen route on the routes stack. 13 | * Effectively it changes current screen to the new screen for given [route] 14 | * which are declared when building ScreensNavigation. 15 | * 16 | * @param route route for which new screen will be navigated to 17 | * @param args optional screen arguments passed when navigating to [route] 18 | */ 19 | inline fun push( 20 | route: String, 21 | args: Args, 22 | ) 23 | 24 | /** 25 | * Pushes new screen route on the routes stack without passing arguments. 26 | * Effectively it changes current screen to the new screen for given [route] 27 | * which are declared when building ScreensNavigation. 28 | * 29 | * @param route route for which new screen will be navigated to 30 | */ 31 | fun push(route: String) 32 | 33 | /** 34 | * Pops back to previous screen route on the routes stack. 35 | * 36 | * @param upToRoute if provided optionally pops multiple routes up to given [upToRoute]. 37 | * 38 | * @return [Result.success] if screen was popped successfully or [Result.failure] with an error: 39 | * 40 | * [IllegalArgumentException] if there is no [upToRoute] on the stack 41 | * or [upToRoute] is the last route on the stack. 42 | * 43 | * [IllegalStateException] if it's about to pop last route on the routes stack. 44 | * 45 | * When there was an error, error screens do not change on the stack. 46 | */ 47 | fun pop(upToRoute: String? = null): Result 48 | } 49 | -------------------------------------------------------------------------------- /navigation-screens/src/commonMain/kotlin/io/github/lukwol/navigation/screens/ScreensMapBuilder.kt: -------------------------------------------------------------------------------- 1 | package io.github.lukwol.navigation.screens 2 | 3 | import androidx.compose.runtime.Composable 4 | 5 | /** 6 | * Builds screens navigation map. 7 | * 8 | * Has separate implementations for Android and non Android targets. 9 | */ 10 | expect open class ScreensMapBuilder { 11 | /** 12 | * Declare screen [content] for [route] and add it to underlying screens map. 13 | * The screens map will be used to navigate between the screens. 14 | * 15 | * @param route used to identify the screen when navigating, in order to present it's content 16 | * @param content [Composable] lambda with content of the screen that takes 17 | * screen arguments as a parameter 18 | * 19 | * @throws IllegalArgumentException when attempting to add new screen for already registered [route] 20 | */ 21 | inline fun screen( 22 | route: String, 23 | noinline content: @Composable (args: Args?) -> Unit, 24 | ) 25 | 26 | /** 27 | * Alternative implementation of [screen] that does not pass any arguments. 28 | * 29 | * @see screen 30 | */ 31 | fun screen( 32 | route: String, 33 | content: @Composable () -> Unit, 34 | ) 35 | } 36 | -------------------------------------------------------------------------------- /navigation-screens/src/commonMain/kotlin/io/github/lukwol/navigation/screens/ScreensNavigation.kt: -------------------------------------------------------------------------------- 1 | package io.github.lukwol.navigation.screens 2 | 3 | import androidx.compose.animation.AnimatedContentTransitionScope 4 | import androidx.compose.animation.EnterTransition 5 | import androidx.compose.animation.ExitTransition 6 | import androidx.compose.animation.core.tween 7 | import androidx.compose.animation.fadeIn 8 | import androidx.compose.animation.fadeOut 9 | import androidx.compose.runtime.Composable 10 | import androidx.compose.runtime.compositionLocalOf 11 | 12 | /** 13 | * Defines screens navigation map by declaring all possible 14 | * [screens][ScreensMapBuilder.screen] and their contents identified by their routes. 15 | * 16 | * It's later provided via [compositionLocalOf] to allow navigation 17 | * to another screen within currently presented screen. 18 | * 19 | * Has separate implementations for Android and non Android targets. 20 | * 21 | * Android target underneath uses official **Compose Navigation** framework made by Google. 22 | * Android implementation utilizes optional route arguments 23 | * and JSON serialization via Kotlin Serialization. 24 | * 25 | * Other targets have common custom implementation that does not involve any serialization. 26 | * 27 | * @param startRoute first screen route for which the initial screen will be displayed 28 | * @param enterTransition defines enter transitions when pushing new screen on the stack, 29 | * by default uses fade in animation 30 | * @param exitTransition defines exit transitions when pushing new screen on the stack 31 | * by default uses fade out animation 32 | * @param popEnterTransition defines enter transitions when popping screen from the stack 33 | * by default uses animation specified in [enterTransition] 34 | * @param popExitTransition defines exit transitions when popping screen from the stack 35 | * by default uses animation specified in [exitTransition] 36 | * @param builder the builder used to construct the underlying screens navigation map 37 | * 38 | * @see LocalScreensController 39 | */ 40 | @Composable 41 | expect fun ScreensNavigation( 42 | startRoute: String, 43 | enterTransition: (AnimatedContentTransitionScope<*>.() -> EnterTransition) = 44 | { fadeIn(animationSpec = tween(700)) }, 45 | exitTransition: (AnimatedContentTransitionScope<*>.() -> ExitTransition) = 46 | { fadeOut(animationSpec = tween(700)) }, 47 | popEnterTransition: (AnimatedContentTransitionScope<*>.() -> EnterTransition) = 48 | enterTransition, 49 | popExitTransition: (AnimatedContentTransitionScope<*>.() -> ExitTransition) = 50 | exitTransition, 51 | builder: ScreensMapBuilder.() -> Unit, 52 | ) 53 | -------------------------------------------------------------------------------- /navigation-screens/src/desktopTest/kotlin/io/github/lukwol/navigation/screens/desktop/ScreensControllerTest.kt: -------------------------------------------------------------------------------- 1 | package io.github.lukwol.navigation.screens.desktop 2 | 3 | import io.github.lukwol.navigation.screens.ScreensController 4 | import io.github.lukwol.navigation.screens.desktop.data.TestRoutes 5 | import kotlin.test.BeforeTest 6 | import kotlin.test.Test 7 | import kotlin.test.assertEquals 8 | import kotlin.test.assertFailsWith 9 | 10 | class ScreensControllerTest { 11 | private lateinit var screensController: ScreensController 12 | 13 | @BeforeTest 14 | fun setUp() { 15 | screensController = ScreensController(TestRoutes.FirstScreen) 16 | } 17 | 18 | @Test 19 | fun initialRoutes() { 20 | assertEquals(screensController.routes, listOf(TestRoutes.FirstScreen)) 21 | } 22 | 23 | @Test 24 | fun pushingSameRouteMultipleTimesThenPoppingUpToIt() { 25 | screensController.push(TestRoutes.SecondScreen) 26 | screensController.push(TestRoutes.ThirdScreen) 27 | screensController.push(TestRoutes.FourthScreen) 28 | screensController.push(TestRoutes.SecondScreen) 29 | screensController.push(TestRoutes.FifthScreen) 30 | assertEquals( 31 | listOf( 32 | TestRoutes.FirstScreen, 33 | TestRoutes.SecondScreen, 34 | TestRoutes.ThirdScreen, 35 | TestRoutes.FourthScreen, 36 | TestRoutes.SecondScreen, 37 | TestRoutes.FifthScreen, 38 | ), 39 | screensController.routes, 40 | ) 41 | screensController.pop(TestRoutes.SecondScreen) 42 | assertEquals( 43 | listOf( 44 | TestRoutes.FirstScreen, 45 | TestRoutes.SecondScreen, 46 | TestRoutes.ThirdScreen, 47 | TestRoutes.FourthScreen, 48 | TestRoutes.SecondScreen, 49 | ), 50 | screensController.routes, 51 | ) 52 | screensController.pop() 53 | screensController.pop(TestRoutes.SecondScreen) 54 | assertEquals( 55 | listOf( 56 | TestRoutes.FirstScreen, 57 | TestRoutes.SecondScreen, 58 | ), 59 | screensController.routes, 60 | ) 61 | screensController.pop() 62 | assertEquals( 63 | listOf(TestRoutes.FirstScreen), 64 | screensController.routes, 65 | ) 66 | } 67 | 68 | @Test 69 | fun poppingStartRoute() { 70 | assertFailsWith { 71 | screensController.pop().getOrThrow() 72 | } 73 | } 74 | 75 | @Test 76 | fun poppingRouteThatIsNotOnTheStack() { 77 | assertFailsWith { 78 | screensController.pop(TestRoutes.SecondScreen).getOrThrow() 79 | } 80 | } 81 | 82 | @Test 83 | fun poppingToCurrentRoute() { 84 | screensController.push(TestRoutes.SecondScreen) 85 | assertFailsWith { 86 | screensController.pop(TestRoutes.SecondScreen).getOrThrow() 87 | } 88 | } 89 | } 90 | -------------------------------------------------------------------------------- /navigation-screens/src/desktopTest/kotlin/io/github/lukwol/navigation/screens/desktop/ScreensMapBuilderTest.kt: -------------------------------------------------------------------------------- 1 | package io.github.lukwol.navigation.screens.desktop 2 | 3 | import io.github.lukwol.navigation.screens.ScreensMapBuilder 4 | import io.github.lukwol.navigation.screens.desktop.data.TestRoutes 5 | import kotlin.test.BeforeTest 6 | import kotlin.test.Test 7 | import kotlin.test.assertContentEquals 8 | import kotlin.test.assertFailsWith 9 | 10 | class ScreensMapBuilderTest { 11 | private lateinit var screensMapBuilder: ScreensMapBuilder 12 | 13 | @BeforeTest 14 | fun setUp() { 15 | screensMapBuilder = ScreensMapBuilder() 16 | } 17 | 18 | @Test 19 | fun screensWithUniqueRoutes() { 20 | with(screensMapBuilder) { 21 | screen(route = TestRoutes.SecondScreen, content = {}) 22 | screen(route = TestRoutes.ThirdScreen, content = {}) 23 | assertContentEquals(listOf(TestRoutes.SecondScreen, TestRoutes.ThirdScreen), build().keys) 24 | } 25 | } 26 | 27 | @Test 28 | fun screensWithNotUniqueRoutes() { 29 | with(screensMapBuilder) { 30 | screen(route = TestRoutes.SecondScreen, content = {}) 31 | screen(route = TestRoutes.ThirdScreen, content = {}) 32 | assertFailsWith { 33 | screen(route = TestRoutes.SecondScreen, content = {}) 34 | } 35 | } 36 | } 37 | } 38 | -------------------------------------------------------------------------------- /navigation-screens/src/desktopTest/kotlin/io/github/lukwol/navigation/screens/desktop/ScreensNavigationTest.kt: -------------------------------------------------------------------------------- 1 | package io.github.lukwol.navigation.screens.desktop 2 | 3 | import androidx.compose.ui.test.junit4.createComposeRule 4 | import androidx.compose.ui.test.onNodeWithText 5 | import androidx.compose.ui.test.performClick 6 | import io.github.lukwol.navigation.screens.ScreensNavigation 7 | import io.github.lukwol.navigation.screens.desktop.components.TestScreenNavigation 8 | import io.github.lukwol.navigation.screens.desktop.data.TestRoutes 9 | import org.junit.Rule 10 | import org.junit.Test 11 | import kotlin.test.assertFailsWith 12 | 13 | class ScreensNavigationTest { 14 | @get:Rule 15 | val compose = createComposeRule() 16 | 17 | @Test 18 | fun firstScreen(): Unit = 19 | with(compose) { 20 | setContent { TestScreenNavigation() } 21 | 22 | onNodeWithText("Push Second Screen").assertExists() 23 | } 24 | 25 | @Test 26 | fun navigateToSecondScreen() { 27 | with(compose) { 28 | setContent { TestScreenNavigation() } 29 | 30 | onNodeWithText("Push Second Screen").assertExists() 31 | onNodeWithText("Foo").assertDoesNotExist() 32 | 33 | onNodeWithText("Push Second Screen").performClick() 34 | waitForIdle() 35 | 36 | onNodeWithText("Push Second Screen").assertDoesNotExist() 37 | onNodeWithText("Foo").assertExists() 38 | } 39 | } 40 | 41 | @Test 42 | fun navigateToThirdScreen() { 43 | with(compose) { 44 | setContent { TestScreenNavigation() } 45 | 46 | onNodeWithText("Push Second Screen").performClick() 47 | waitForIdle() 48 | 49 | onNodeWithText("Foo").assertExists() 50 | 51 | onNodeWithText("Push Third Screen").performClick() 52 | waitForIdle() 53 | 54 | onNodeWithText("Foo").assertDoesNotExist() 55 | onNodeWithText("text = Bar, number = 42").assertExists() 56 | } 57 | } 58 | 59 | @Test 60 | fun navigateToThirdScreenThenPopToSecondScreen() { 61 | with(compose) { 62 | setContent { TestScreenNavigation() } 63 | 64 | onNodeWithText("Push Second Screen").performClick() 65 | waitForIdle() 66 | 67 | onNodeWithText("Push Third Screen").performClick() 68 | waitForIdle() 69 | 70 | onNodeWithText("text = Bar, number = 42").assertExists() 71 | onNodeWithText("Pop Screen").performClick() 72 | waitForIdle() 73 | 74 | onNodeWithText("text = Bar, number = 42").assertDoesNotExist() 75 | onNodeWithText("Foo").assertExists() 76 | } 77 | } 78 | 79 | @Test 80 | fun navigateToThirdScreenThenPopToFirstScreen() { 81 | with(compose) { 82 | setContent { TestScreenNavigation() } 83 | 84 | onNodeWithText("Push Second Screen").performClick() 85 | waitForIdle() 86 | 87 | onNodeWithText("Push Third Screen").performClick() 88 | waitForIdle() 89 | 90 | onNodeWithText("text = Bar, number = 42").assertExists() 91 | onNodeWithText("Pop To First Screen").performClick() 92 | waitForIdle() 93 | 94 | onNodeWithText("text = Bar, number = 42").assertDoesNotExist() 95 | onNodeWithText("Push Second Screen").assertExists() 96 | } 97 | } 98 | 99 | @Test 100 | fun `missing start route screen`() { 101 | assertFailsWith { 102 | compose.setContent { 103 | ScreensNavigation( 104 | startRoute = TestRoutes.FirstScreen, 105 | ) { 106 | screen(TestRoutes.SecondScreen) {} 107 | } 108 | } 109 | } 110 | } 111 | 112 | @Test 113 | fun `empty navigation graph`() { 114 | assertFailsWith { 115 | compose.setContent { 116 | ScreensNavigation( 117 | startRoute = TestRoutes.FirstScreen, 118 | ) {} 119 | } 120 | } 121 | } 122 | } 123 | -------------------------------------------------------------------------------- /navigation-screens/src/desktopTest/kotlin/io/github/lukwol/navigation/screens/desktop/components/FirstScreen.kt: -------------------------------------------------------------------------------- 1 | package io.github.lukwol.navigation.screens.desktop.components 2 | 3 | import androidx.compose.material3.Button 4 | import androidx.compose.material3.Text 5 | import androidx.compose.runtime.Composable 6 | import io.github.lukwol.navigation.screens.LocalScreensController 7 | import io.github.lukwol.navigation.screens.desktop.data.TestRoutes 8 | 9 | @Suppress("TestFunctionName") 10 | @Composable 11 | fun FirstScreen() { 12 | val screensController = LocalScreensController.current 13 | 14 | Button( 15 | onClick = { screensController.push(TestRoutes.SecondScreen, "Foo") }, 16 | ) { 17 | Text("Push Second Screen") 18 | } 19 | } 20 | -------------------------------------------------------------------------------- /navigation-screens/src/desktopTest/kotlin/io/github/lukwol/navigation/screens/desktop/components/SecondScreen.kt: -------------------------------------------------------------------------------- 1 | package io.github.lukwol.navigation.screens.desktop.components 2 | 3 | import androidx.compose.foundation.layout.Column 4 | import androidx.compose.material3.Button 5 | import androidx.compose.material3.Text 6 | import androidx.compose.runtime.Composable 7 | import io.github.lukwol.navigation.screens.LocalScreensController 8 | import io.github.lukwol.navigation.screens.desktop.data.TestRoutes 9 | import io.github.lukwol.navigation.screens.desktop.data.ThirdScreenArgs 10 | 11 | @Suppress("TestFunctionName") 12 | @Composable 13 | fun SecondScreen(args: String) { 14 | val screensController = LocalScreensController.current 15 | 16 | Column { 17 | Text(args) 18 | Button( 19 | onClick = { 20 | screensController.push( 21 | TestRoutes.ThirdScreen, 22 | ThirdScreenArgs(text = "Bar", number = 42), 23 | ) 24 | }, 25 | ) { 26 | Text("Push Third Screen") 27 | } 28 | } 29 | } 30 | -------------------------------------------------------------------------------- /navigation-screens/src/desktopTest/kotlin/io/github/lukwol/navigation/screens/desktop/components/TestScreenNavigation.kt: -------------------------------------------------------------------------------- 1 | package io.github.lukwol.navigation.screens.desktop.components 2 | 3 | import androidx.compose.runtime.Composable 4 | import io.github.lukwol.navigation.screens.ScreensNavigation 5 | import io.github.lukwol.navigation.screens.desktop.data.TestRoutes 6 | import io.github.lukwol.navigation.screens.desktop.data.ThirdScreenArgs 7 | 8 | @Suppress("TestFunctionName") 9 | @Composable 10 | fun TestScreenNavigation() { 11 | ScreensNavigation( 12 | startRoute = TestRoutes.FirstScreen, 13 | ) { 14 | screen(TestRoutes.FirstScreen) { 15 | FirstScreen() 16 | } 17 | 18 | screen(TestRoutes.SecondScreen) { args: String? -> 19 | SecondScreen(args!!) 20 | } 21 | 22 | screen(TestRoutes.ThirdScreen) { args: ThirdScreenArgs? -> 23 | ThirdScreen(args!!) 24 | } 25 | } 26 | } 27 | -------------------------------------------------------------------------------- /navigation-screens/src/desktopTest/kotlin/io/github/lukwol/navigation/screens/desktop/components/ThirdScreen.kt: -------------------------------------------------------------------------------- 1 | package io.github.lukwol.navigation.screens.desktop.components 2 | 3 | import androidx.compose.foundation.layout.Column 4 | import androidx.compose.material3.Button 5 | import androidx.compose.material3.Text 6 | import androidx.compose.runtime.Composable 7 | import io.github.lukwol.navigation.screens.LocalScreensController 8 | import io.github.lukwol.navigation.screens.desktop.data.TestRoutes 9 | import io.github.lukwol.navigation.screens.desktop.data.ThirdScreenArgs 10 | 11 | @Suppress("TestFunctionName") 12 | @Composable 13 | fun ThirdScreen(args: ThirdScreenArgs) { 14 | val screensController = LocalScreensController.current 15 | 16 | Column { 17 | Text(args.toString()) 18 | Button( 19 | onClick = { screensController.pop() }, 20 | ) { 21 | Text("Pop Screen") 22 | } 23 | Button( 24 | onClick = { screensController.pop(upToRoute = TestRoutes.FirstScreen) }, 25 | ) { 26 | Text("Pop To First Screen") 27 | } 28 | } 29 | } 30 | -------------------------------------------------------------------------------- /navigation-screens/src/desktopTest/kotlin/io/github/lukwol/navigation/screens/desktop/data/TestRoutes.kt: -------------------------------------------------------------------------------- 1 | package io.github.lukwol.navigation.screens.desktop.data 2 | 3 | object TestRoutes { 4 | const val FirstScreen = "first-screen" 5 | const val SecondScreen = "second-screen" 6 | const val ThirdScreen = "third-screen" 7 | const val FourthScreen = "fourth-screen" 8 | const val FifthScreen = "fifth-screen" 9 | } 10 | -------------------------------------------------------------------------------- /navigation-screens/src/desktopTest/kotlin/io/github/lukwol/navigation/screens/desktop/data/ThirdScreenArgs.kt: -------------------------------------------------------------------------------- 1 | package io.github.lukwol.navigation.screens.desktop.data 2 | 3 | class ThirdScreenArgs( 4 | val text: String, 5 | val number: Int, 6 | ) { 7 | override fun toString() = "text = $text, number = $number" 8 | } 9 | -------------------------------------------------------------------------------- /navigation-screens/src/nonAndroidMain/kotlin/io/github/lukwol/navigation/screens/NavigationType.kt: -------------------------------------------------------------------------------- 1 | package io.github.lukwol.navigation.screens 2 | 3 | /** 4 | * Describes whether screen is being pushed or popped from the stack. 5 | */ 6 | internal enum class NavigationType { 7 | Push, 8 | Pop, 9 | } 10 | -------------------------------------------------------------------------------- /navigation-screens/src/nonAndroidMain/kotlin/io/github/lukwol/navigation/screens/ScreensController.kt: -------------------------------------------------------------------------------- 1 | package io.github.lukwol.navigation.screens 2 | 3 | import androidx.compose.runtime.mutableStateListOf 4 | 5 | actual class ScreensController actual constructor(startRoute: String) { 6 | /** 7 | * Mutable list that serves as a stack of presented screens and their arguments. 8 | * 9 | * Routes with args can be pushed on and popped from the stack. 10 | * 11 | * Whenever last route is changed, currently presented screen is changed. 12 | */ 13 | @PublishedApi 14 | internal val routesStack = mutableStateListOf(RouteWithArgs(startRoute)) 15 | 16 | /** 17 | * Read only list of current screen routes along with their navigation arguments, 18 | * the stack is based on `routesStack` 19 | * 20 | * @see routesStack 21 | */ 22 | val routes get() = routesStack.map(RouteWithArgs::route) 23 | 24 | @Suppress("NOTHING_TO_INLINE", "KotlinRedundantDiagnosticSuppress") 25 | actual inline fun push( 26 | route: String, 27 | args: Args, 28 | ) { 29 | routesStack += RouteWithArgs(route, args) 30 | } 31 | 32 | actual fun push(route: String) { 33 | routesStack += RouteWithArgs(route, null) 34 | } 35 | 36 | actual fun pop(upToRoute: String?): Result = 37 | runCatching { 38 | when { 39 | upToRoute != null && upToRoute !in routes -> throw IllegalArgumentException("There is no $upToRoute on the stack") 40 | upToRoute == routes.last() -> throw IllegalArgumentException("Cannot pop up to current route $upToRoute") 41 | routes.size == 1 -> throw IllegalStateException("Cannot pop start route ${routes.first()}") 42 | upToRoute == null -> routesStack.removeLast() 43 | else -> while (routes.last() != upToRoute) routesStack.removeLast() 44 | } 45 | } 46 | } 47 | -------------------------------------------------------------------------------- /navigation-screens/src/nonAndroidMain/kotlin/io/github/lukwol/navigation/screens/ScreensMapBuilder.kt: -------------------------------------------------------------------------------- 1 | package io.github.lukwol.navigation.screens 2 | 3 | import androidx.compose.runtime.Composable 4 | 5 | actual open class ScreensMapBuilder { 6 | /** 7 | * Map of all registered screen routes and their corresponding contents 8 | * represented as a [Composable] lambda. 9 | */ 10 | val screensMap = mutableMapOf Unit>() 11 | 12 | @Suppress("UNCHECKED_CAST", "NOTHING_TO_INLINE", "KotlinRedundantDiagnosticSuppress") 13 | actual inline fun screen( 14 | route: String, 15 | noinline content: @Composable (args: Args?) -> Unit, 16 | ) { 17 | if (screensMap.containsKey(route)) { 18 | throw IllegalArgumentException("$route is already registered") 19 | } else { 20 | screensMap[route] = { 21 | content(it as Args?) 22 | } 23 | } 24 | } 25 | 26 | actual fun screen( 27 | route: String, 28 | content: @Composable () -> Unit, 29 | ) = screen( 30 | route = route, 31 | content = { content() }, 32 | ) 33 | 34 | /** 35 | * Build immutable map of with screen route as keys and contents as values. 36 | */ 37 | fun build() = screensMap.toMap() 38 | } 39 | -------------------------------------------------------------------------------- /navigation-screens/src/nonAndroidMain/kotlin/io/github/lukwol/navigation/screens/ScreensNavigation.kt: -------------------------------------------------------------------------------- 1 | package io.github.lukwol.navigation.screens 2 | 3 | import androidx.compose.animation.AnimatedContent 4 | import androidx.compose.animation.AnimatedContentTransitionScope 5 | import androidx.compose.animation.EnterTransition 6 | import androidx.compose.animation.ExitTransition 7 | import androidx.compose.animation.togetherWith 8 | import androidx.compose.runtime.Composable 9 | import androidx.compose.runtime.CompositionLocalProvider 10 | import androidx.compose.runtime.SideEffect 11 | import androidx.compose.runtime.remember 12 | 13 | @Composable 14 | actual fun ScreensNavigation( 15 | startRoute: String, 16 | enterTransition: (AnimatedContentTransitionScope<*>.() -> EnterTransition), 17 | exitTransition: (AnimatedContentTransitionScope<*>.() -> ExitTransition), 18 | popEnterTransition: (AnimatedContentTransitionScope<*>.() -> EnterTransition), 19 | popExitTransition: (AnimatedContentTransitionScope<*>.() -> ExitTransition), 20 | builder: ScreensMapBuilder.() -> Unit, 21 | ) { 22 | val mapBuilder = ScreensMapBuilder() 23 | builder(mapBuilder) 24 | 25 | val screensMap = remember { mapBuilder.build() } 26 | val screensController = remember { ScreensController(startRoute) } 27 | var previousRoute: String? = remember { null } 28 | 29 | CompositionLocalProvider( 30 | LocalScreensController provides screensController, 31 | ) { 32 | val routeWithArgs = screensController.routesStack.last() 33 | val navigationType = 34 | when (previousRoute) { 35 | in screensController.routes -> NavigationType.Push 36 | else -> NavigationType.Pop 37 | } 38 | AnimatedContent( 39 | targetState = routeWithArgs, 40 | transitionSpec = { 41 | when (navigationType) { 42 | NavigationType.Push -> enterTransition() togetherWith exitTransition() 43 | NavigationType.Pop -> popEnterTransition() togetherWith popExitTransition() 44 | } 45 | }, 46 | content = { (route, args) -> 47 | screensMap.getValue(route)(args) 48 | }, 49 | ) 50 | 51 | SideEffect { 52 | previousRoute = routeWithArgs.route 53 | } 54 | } 55 | } 56 | -------------------------------------------------------------------------------- /navigation-windows/build.gradle.kts: -------------------------------------------------------------------------------- 1 | plugins { 2 | alias(commonLibs.plugins.kotlin.multiplatform) 3 | alias(commonLibs.plugins.compose.multiplatform) 4 | } 5 | 6 | kotlin { 7 | applyDefaultHierarchyTemplate() 8 | 9 | jvm("desktop") 10 | 11 | sourceSets { 12 | getByName("desktopMain").dependencies { 13 | implementation(compose.desktop.currentOs) 14 | } 15 | getByName("desktopTest").dependencies { 16 | implementation(kotlin("test")) 17 | } 18 | } 19 | } 20 | -------------------------------------------------------------------------------- /navigation-windows/src/desktopMain/kotlin/io/github/lukwol/navigation/windows/LocalWindowController.kt: -------------------------------------------------------------------------------- 1 | package io.github.lukwol.navigation.windows 2 | 3 | import androidx.compose.runtime.compositionLocalOf 4 | 5 | /** 6 | * Provides local [WindowsController] that can be used 7 | * for opening or closing screens from within currently displayed screens 8 | * in any window. 9 | * 10 | * Defaults value is set to prevent problems with screens previews. 11 | */ 12 | val LocalWindowController = compositionLocalOf { WindowsController("") } 13 | -------------------------------------------------------------------------------- /navigation-windows/src/desktopMain/kotlin/io/github/lukwol/navigation/windows/RouteWithArgs.kt: -------------------------------------------------------------------------------- 1 | package io.github.lukwol.navigation.windows 2 | 3 | import androidx.compose.runtime.Immutable 4 | 5 | /** 6 | * A pair of window route and window arguments. 7 | */ 8 | @Immutable 9 | internal data class RouteWithArgs( 10 | val route: String, 11 | val args: Any? = null, 12 | ) 13 | -------------------------------------------------------------------------------- /navigation-windows/src/desktopMain/kotlin/io/github/lukwol/navigation/windows/WindowsController.kt: -------------------------------------------------------------------------------- 1 | package io.github.lukwol.navigation.windows 2 | 3 | import androidx.compose.runtime.mutableStateListOf 4 | 5 | /** 6 | * Manages navigation between the windows declared when building [WindowsNavigation]. 7 | * 8 | * Available via [LocalWindowController]. 9 | */ 10 | class WindowsController(startRoute: String) { 11 | /** 12 | * Mutable list of presented windows and their arguments. 13 | * 14 | * Routes can be added to the list by opening new windows, 15 | * and removed from the list by closing windows. 16 | * 17 | * Whenever content of the list changes, opened windows are changed. 18 | */ 19 | internal val routesList = mutableStateListOf(RouteWithArgs(startRoute)) 20 | 21 | /** 22 | * Read only list of current window routes along with their navigation arguments, 23 | * the list is based on the [routesList]. 24 | * 25 | * @see routesList 26 | */ 27 | val routes get() = routesList.map(RouteWithArgs::route).toSet() 28 | 29 | /** 30 | * Opens new window for the [route] and adds it to the [routes] stack. 31 | * Content of the windows is based on what was declared in [WindowsNavigation]. 32 | * 33 | * @param route for which new window will be opened 34 | * 35 | * @return [Result.success] if window was opened successfully or [Result.failure] with an error: 36 | * 37 | * [IllegalArgumentException] if window for the [route] is already opened. 38 | * 39 | * When there was an error, error opened windows do not change. 40 | */ 41 | fun open( 42 | route: String, 43 | args: Any? = null, 44 | ): Result = 45 | runCatching { 46 | if (route in routes) { 47 | throw IllegalArgumentException("Window for $route is already opened") 48 | } else { 49 | routesList += RouteWithArgs(route, args) 50 | } 51 | } 52 | 53 | /** 54 | * Closes the window for given [route] and removes it from the [routes] list. 55 | * When there was an error, error opened windows do not change. 56 | * 57 | * @param route for which currently opened window will be closed 58 | * 59 | * @return [Result.success] if window was closed successfully or [Result.failure] with an error: 60 | * 61 | * [IllegalArgumentException] if window for the [route] was not opened. 62 | * 63 | * When there was an error, error opened windows do not change. 64 | * 65 | * @see [WindowsController.close] 66 | */ 67 | fun close(route: String): Result = 68 | runCatching { 69 | if (route !in routes) { 70 | throw IllegalArgumentException("Window for $route is not opened") 71 | } else { 72 | routesList.removeAll { it.route == route } 73 | } 74 | } 75 | } 76 | -------------------------------------------------------------------------------- /navigation-windows/src/desktopMain/kotlin/io/github/lukwol/navigation/windows/WindowsMapBuilder.kt: -------------------------------------------------------------------------------- 1 | package io.github.lukwol.navigation.windows 2 | 3 | import androidx.compose.runtime.Composable 4 | import androidx.compose.ui.window.FrameWindowScope 5 | import androidx.compose.ui.window.Window 6 | 7 | /** 8 | * Builds windows navigation map. 9 | * 10 | * @see WindowsNavigation 11 | */ 12 | class WindowsMapBuilder { 13 | /** 14 | * Map of all registered window routes and their corresponding contents 15 | * represented as a [Composable] lambda. 16 | */ 17 | private val windowsMap = mutableMapOf Unit>() 18 | 19 | /** 20 | * Declare window [content] for [route] and add it to underlying [windowsMap]. 21 | * The [windowsMap] will be used to navigate between the windows. 22 | * 23 | * @param route used to identify the window when navigating, in order to present it's content 24 | * @param windowFactory lambda that takes [content] argument 25 | * and allows to wrap it in custom [Window] implementation. 26 | * @param content view with [Composable] content of the window 27 | * 28 | * @throws IllegalArgumentException when attempting to add new window for already registered [route] 29 | */ 30 | @Suppress("UNCHECKED_CAST") 31 | fun window( 32 | route: String, 33 | windowFactory: @Composable (content: @Composable FrameWindowScope.() -> Unit) -> Unit, 34 | content: @Composable FrameWindowScope.(args: Args?) -> Unit, 35 | ) { 36 | if (windowsMap.containsKey(route)) { 37 | throw IllegalArgumentException("$route is already registered") 38 | } else { 39 | windowsMap[route] = { 40 | windowFactory { content(it as Args?) } 41 | } 42 | } 43 | } 44 | 45 | /** 46 | * Alternative implementation of [window] that does not pass any arguments. 47 | * 48 | * @see window 49 | */ 50 | fun window( 51 | route: String, 52 | windowFactory: @Composable (content: @Composable FrameWindowScope.() -> Unit) -> Unit, 53 | content: @Composable FrameWindowScope.() -> Unit, 54 | ) = window( 55 | route = route, 56 | windowFactory = windowFactory, 57 | content = { _: Any? -> 58 | content() 59 | }, 60 | ) 61 | 62 | /** 63 | * Alternative implementation of [window] that provides [Window] with basic setup. 64 | * Window title is set to [title] parameter and `onCloseRequest` is set to close this window. 65 | * 66 | * @see window 67 | */ 68 | fun window( 69 | route: String, 70 | title: String = route, 71 | content: @Composable FrameWindowScope.(args: Args?) -> Unit, 72 | ) { 73 | window( 74 | route = route, 75 | windowFactory = { 76 | val windowsController = LocalWindowController.current 77 | 78 | Window( 79 | title = title, 80 | onCloseRequest = { 81 | windowsController.close(route) 82 | }, 83 | content = it, 84 | ) 85 | }, 86 | content = content, 87 | ) 88 | } 89 | 90 | /** 91 | * Alternative implementation of [window] that does not pass any arguments 92 | * and that provides [Window] with basic setup. 93 | * Window title is set to [title] parameter and `onCloseRequest` is set to close this window. 94 | * 95 | * @see window 96 | */ 97 | fun window( 98 | route: String, 99 | title: String = route, 100 | content: @Composable FrameWindowScope.() -> Unit, 101 | ) = window( 102 | route = route, 103 | title = title, 104 | content = { _: Any? -> 105 | content() 106 | }, 107 | ) 108 | 109 | /** 110 | * Build immutable [windowsMap]. 111 | */ 112 | fun build() = windowsMap.toMap() 113 | } 114 | -------------------------------------------------------------------------------- /navigation-windows/src/desktopMain/kotlin/io/github/lukwol/navigation/windows/WindowsNavigation.kt: -------------------------------------------------------------------------------- 1 | package io.github.lukwol.navigation.windows 2 | 3 | import androidx.compose.runtime.Composable 4 | import androidx.compose.runtime.CompositionLocalProvider 5 | import androidx.compose.runtime.remember 6 | 7 | /** 8 | * Declare windows navigation by defining all [windows][WindowsMapBuilder.window] for window routes. 9 | * 10 | * Provides [LocalWindowController] that can be used for navigation inside each window. 11 | * 12 | * @param startRoute first window route for which window will be displayed 13 | * @param builder the builder used to construct the windows navigation map 14 | */ 15 | 16 | @Composable 17 | fun WindowsNavigation( 18 | startRoute: String, 19 | builder: WindowsMapBuilder.() -> Unit, 20 | ) { 21 | val mapBuilder = WindowsMapBuilder() 22 | builder(mapBuilder) 23 | 24 | val windowsMap = remember { mapBuilder.build() } 25 | val windowsController = remember { WindowsController(startRoute) } 26 | 27 | val routes = windowsController.routesList 28 | 29 | CompositionLocalProvider( 30 | LocalWindowController provides windowsController, 31 | ) { 32 | windowsMap.forEach { (route, content) -> 33 | @Composable 34 | fun window() = 35 | routes 36 | .find { it.route == route } 37 | ?.let { content(it.args) } 38 | window() 39 | } 40 | } 41 | } 42 | -------------------------------------------------------------------------------- /navigation-windows/src/desktopTest/kotlin/io/github/lukwol/navigation/windows/WindowsControllerTest.kt: -------------------------------------------------------------------------------- 1 | package io.github.lukwol.navigation.windows 2 | 3 | import io.github.lukwol.navigation.windows.data.TestRoutes 4 | import junit.framework.TestCase.assertEquals 5 | import org.junit.Before 6 | import org.junit.Test 7 | import kotlin.test.assertFailsWith 8 | 9 | class WindowsControllerTest { 10 | private lateinit var windowsController: WindowsController 11 | 12 | @Before 13 | fun setUp() { 14 | windowsController = WindowsController(TestRoutes.FirstWindow) 15 | } 16 | 17 | @Test 18 | fun `initial routes`() { 19 | assertEquals(windowsController.routes, setOf(TestRoutes.FirstWindow)) 20 | } 21 | 22 | @Test 23 | fun `opening and closing multiple windows`() { 24 | windowsController.open(TestRoutes.SecondWindow) 25 | assertEquals( 26 | windowsController.routes, 27 | setOf( 28 | TestRoutes.FirstWindow, 29 | TestRoutes.SecondWindow, 30 | ), 31 | ) 32 | 33 | windowsController.open(TestRoutes.ThirdWindow) 34 | assertEquals( 35 | windowsController.routes, 36 | setOf( 37 | TestRoutes.FirstWindow, 38 | TestRoutes.SecondWindow, 39 | TestRoutes.ThirdWindow, 40 | ), 41 | ) 42 | 43 | windowsController.close(TestRoutes.FirstWindow) 44 | assertEquals( 45 | windowsController.routes, 46 | setOf( 47 | TestRoutes.SecondWindow, 48 | TestRoutes.ThirdWindow, 49 | ), 50 | ) 51 | 52 | windowsController.close(TestRoutes.SecondWindow) 53 | assertEquals(windowsController.routes, setOf(TestRoutes.ThirdWindow)) 54 | } 55 | 56 | @Test 57 | fun `opening window that is already opened`() { 58 | windowsController.open(TestRoutes.SecondWindow) 59 | assertFailsWith { 60 | windowsController.open(TestRoutes.SecondWindow).getOrThrow() 61 | } 62 | } 63 | 64 | @Test 65 | fun `closing window that is not opened`() { 66 | assertFailsWith { 67 | windowsController.close(TestRoutes.SecondWindow).getOrThrow() 68 | } 69 | } 70 | } 71 | -------------------------------------------------------------------------------- /navigation-windows/src/desktopTest/kotlin/io/github/lukwol/navigation/windows/WindowsMapBuilderTest.kt: -------------------------------------------------------------------------------- 1 | package io.github.lukwol.navigation.windows 2 | 3 | import androidx.compose.ui.window.Window 4 | import io.github.lukwol.navigation.windows.data.TestRoutes 5 | import org.junit.Before 6 | import org.junit.Test 7 | import kotlin.test.assertContentEquals 8 | import kotlin.test.assertFailsWith 9 | 10 | class WindowsMapBuilderTest { 11 | private lateinit var windowsMapBuilder: WindowsMapBuilder 12 | 13 | @Before 14 | fun setUp() { 15 | windowsMapBuilder = WindowsMapBuilder() 16 | } 17 | 18 | @Test 19 | fun `windows with unique routes`() { 20 | with(windowsMapBuilder) { 21 | window(TestRoutes.FirstWindow) {} 22 | window(TestRoutes.SecondWindow) {} 23 | assertContentEquals(build().keys, listOf(TestRoutes.FirstWindow, TestRoutes.SecondWindow)) 24 | } 25 | } 26 | 27 | @Test 28 | fun `windows with not unique routes`() { 29 | with(windowsMapBuilder) { 30 | window(TestRoutes.FirstWindow) {} 31 | window(TestRoutes.SecondWindow) {} 32 | assertFailsWith { 33 | window(TestRoutes.FirstWindow) {} 34 | } 35 | } 36 | } 37 | 38 | @Test 39 | fun `custom window`() { 40 | with(windowsMapBuilder) { 41 | window( 42 | route = TestRoutes.FirstWindow, 43 | windowFactory = { 44 | Window( 45 | onCloseRequest = { }, 46 | ) {} 47 | }, 48 | ) {} 49 | assertContentEquals(build().keys, listOf(TestRoutes.FirstWindow)) 50 | } 51 | } 52 | } 53 | -------------------------------------------------------------------------------- /navigation-windows/src/desktopTest/kotlin/io/github/lukwol/navigation/windows/data/TestRoutes.kt: -------------------------------------------------------------------------------- 1 | package io.github.lukwol.navigation.windows.data 2 | 3 | object TestRoutes { 4 | const val FirstWindow = "first-window" 5 | const val SecondWindow = "second-window" 6 | const val ThirdWindow = "third-window" 7 | } 8 | -------------------------------------------------------------------------------- /settings.gradle.kts: -------------------------------------------------------------------------------- 1 | @file:Suppress("UnstableApiUsage") 2 | 3 | rootProject.name = "navigation" 4 | 5 | pluginManagement { 6 | repositories { 7 | gradlePluginPortal() 8 | mavenCentral() 9 | google() 10 | } 11 | } 12 | 13 | dependencyResolutionManagement { 14 | repositories { 15 | mavenCentral() 16 | google() 17 | } 18 | 19 | versionCatalogs { 20 | create("commonLibs") { 21 | from(files("gradle/common.versions.toml")) 22 | } 23 | create("androidLibs") { 24 | from(files("gradle/android.versions.toml")) 25 | } 26 | } 27 | } 28 | 29 | include(":navigation-screens") 30 | include(":navigation-screens-viewmodel") 31 | include(":navigation-windows") 32 | 33 | enableFeaturePreview("TYPESAFE_PROJECT_ACCESSORS") 34 | --------------------------------------------------------------------------------