Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Added the new web history API with nested navigation support #821

Merged
merged 1 commit into from
Dec 7, 2024

Conversation

arkivanov
Copy link
Owner

@arkivanov arkivanov commented Dec 7, 2024

Supports Child Stack, Child Pages and Child Panels navigation models. Supporting Child Slot is a bit tricky and most likely will require additional API changes, maybe later.

As part of #111.

Summary by CodeRabbit

  • New Features

    • Introduced web navigation capabilities through new interfaces and classes, enhancing navigation and state management.
    • Added support for deep linking in various components, allowing for dynamic initialization based on URL parameters.
    • Implemented new methods for serialization and deserialization of navigation states.
    • Enhanced lifecycle management and state restoration for web applications.
  • Bug Fixes

    • Addressed issues related to navigation history and component state management.
  • Documentation

    • Updated comments and documentation to reflect new functionalities and changes in component behavior.
  • Tests

    • Added comprehensive unit tests for the new web navigation functionalities and deep linking scenarios.

Copy link

coderabbitai bot commented Dec 7, 2024

Walkthrough

The pull request introduces significant enhancements to the com.arkivanov.decompose package, primarily focusing on web navigation capabilities. New classes, interfaces, and methods have been added to support serialization for Pages and Panels, and to implement web navigation functionalities via the WebNavigation interface. Modifications to existing classes include the addition of companion objects and synthetic methods, as well as updates to constructors and methods across various components. The changes aim to improve state management and navigation handling in the API.

Changes

File Path Change Summary
decompose/api/android/decompose.api Added new classes/interfaces for serialization and web navigation, including Pages$$serializer, WebNavigation, etc.
decompose/api/decompose.klib.api Introduced WebNavigation, WebNavigationOwner, and DefaultWebHistoryController for enhanced navigation.
decompose/api/jvm/decompose.api Similar additions as in Android API, focusing on serialization and web navigation functionalities.
decompose/src/commonMain/kotlin/com/arkivanov/decompose/router/pages/Pages.kt Added @Serializable annotation to Pages class for serialization support.
decompose/src/commonMain/kotlin/com/arkivanov/decompose/router/pages/PagesWebNavigation.kt Introduced PagesWebNavigation class for managing web navigation specific to pages.
decompose/src/commonMain/kotlin/com/arkivanov/decompose/router/panels/PanelsWebNavigation.kt Added PanelsWebNavigation for managing web navigation in panel-based UIs.
decompose/src/commonMain/kotlin/com/arkivanov/decompose/router/stack/StackWebNavigation.kt Implemented StackWebNavigation for managing browser navigation in component stacks.
decompose/src/commonMain/kotlin/com/arkivanov/decompose/router/webhistory/NoOpWebNavigation.kt Created a no-op implementation of WebNavigation for placeholder functionality.
decompose/src/commonMain/kotlin/com/arkivanov/decompose/router/webhistory/WebNavigation.kt Defined WebNavigation interface for two-way navigation control linking models with browser history.
decompose/src/commonMain/kotlin/com/arkivanov/decompose/router/webhistory/WebNavigationOwner.kt Added WebNavigationOwner interface to represent holders of WebNavigation.
decompose/src/jsMain/kotlin/com/arkivanov/decompose/router/webhistory/WebHistoryNavigation.kt Introduced withWebHistory function to manage web browser history navigation.
sample/app-js-compose/src/jsMain/kotlin/com/arkivanov/decompose/sample/app/Main.kt Refactored main function to utilize withWebHistory for state management.
sample/shared/compose/build.gradle.kts Added nonWeb and js bundles to organize dependencies.
sample/shared/shared/src/commonMain/kotlin/com/arkivanov/sample/shared/Url.kt Introduced Url data class for URL handling with serialization support.
sample/shared/shared/src/commonMain/kotlin/com/arkivanov/sample/shared/utils/Utils.kt Added snakeCase extension function for string formatting.
sample/shared/shared/src/commonMain/kotlin/com/arkivanov/sample/shared/multipane/DefaultMultiPaneComponent.kt Enhanced component initialization with deep link support and web navigation.
sample/shared/shared/src/commonMain/kotlin/com/arkivanov/sample/shared/tabs/DefaultTabsComponent.kt Updated to handle deep links and introduced web navigation capabilities.

Possibly related PRs

🐇 In the meadow where we play,
New features hop in every way!
With pages and panels, we navigate,
Web histories now integrate.
Serialization's here, oh what a sight,
Hopping through code, everything feels right! 🌼


Thank you for using CodeRabbit. We offer it for free to the OSS community and would appreciate your support in helping us grow. If you find it useful, would you consider giving us a shout-out on your favorite social media?

❤️ Share
🪧 Tips

Chat

There are 3 ways to chat with CodeRabbit:

  • Review comments: Directly reply to a review comment made by CodeRabbit. Example:
    • I pushed a fix in commit <commit_id>, please review it.
    • Generate unit testing code for this file.
    • Open a follow-up GitHub issue for this discussion.
  • Files and specific lines of code (under the "Files changed" tab): Tag @coderabbitai in a new review comment at the desired location with your query. Examples:
    • @coderabbitai generate unit testing code for this file.
    • @coderabbitai modularize this function.
  • PR comments: Tag @coderabbitai in a new PR comment to ask questions about the PR branch. For the best results, please provide a very specific query, as very limited context is provided in this mode. Examples:
    • @coderabbitai gather interesting stats about this repository and render them as a table. Additionally, render a pie chart showing the language distribution in the codebase.
    • @coderabbitai read src/utils.ts and generate unit testing code.
    • @coderabbitai read the files in the src/scheduler package and generate a class diagram using mermaid and a README in the markdown format.
    • @coderabbitai help me debug CodeRabbit configuration file.

Note: Be mindful of the bot's finite context window. It's strongly recommended to break down tasks such as reading entire modules into smaller chunks. For a focused discussion, use review comments to chat about specific files and their changes, instead of using the PR comments.

CodeRabbit Commands (Invoked using PR comments)

  • @coderabbitai pause to pause the reviews on a PR.
  • @coderabbitai resume to resume the paused reviews.
  • @coderabbitai review to trigger an incremental review. This is useful when automatic reviews are disabled for the repository.
  • @coderabbitai full review to do a full review from scratch and review all the files again.
  • @coderabbitai summary to regenerate the summary of the PR.
  • @coderabbitai resolve resolve all the CodeRabbit review comments.
  • @coderabbitai configuration to show the current CodeRabbit configuration for the repository.
  • @coderabbitai help to get help.

Other keywords and placeholders

  • Add @coderabbitai ignore anywhere in the PR description to prevent this PR from being reviewed.
  • Add @coderabbitai summary to generate the high-level summary at a specific location in the PR description.
  • Add @coderabbitai anywhere in the PR title to generate the title automatically.

CodeRabbit Configuration File (.coderabbit.yaml)

  • You can programmatically configure CodeRabbit by adding a .coderabbit.yaml file to the root of your repository.
  • Please see the configuration documentation for more information.
  • If your editor has YAML language server enabled, you can add the path at the top of this file to enable auto-completion and validation: # yaml-language-server: $schema=https://coderabbit.ai/integrations/schema.v2.json

Documentation and Community

  • Visit our Documentation for detailed information on how to use CodeRabbit.
  • Join our Discord Community to get help, request features, and share feedback.
  • Follow us on X/Twitter for updates and announcements.

Copy link

@coderabbitai coderabbitai bot left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Actionable comments posted: 13

🧹 Outside diff range and nitpick comments (50)
sample/shared/shared/src/commonTest/kotlin/com/arkivanov/sample/shared/tabs/TabsComponentIntegrationTest.kt (1)

Line range hint 112-117: Consider adding test cases for deep linking functionality.

The new web navigation feature lacks test coverage. Consider adding test cases that:

  1. Verify component initialization with various deep link URL formats
  2. Test navigation state when deep links are provided
  3. Validate the interaction between deep links and tab navigation

Here's a suggested test case structure:

@Test
fun WHEN_created_with_deepLink_THEN_correct_tab_active() {
    val component = createComponent(
        deepLinkUrl = "/tabs/counters" // Add appropriate URL format
    )
    
    component.stack.assertActiveInstance<CountersChild>()
}

@Test
fun WHEN_created_with_invalid_deepLink_THEN_defaults_to_menu() {
    val component = createComponent(
        deepLinkUrl = "/invalid/path"
    )
    
    component.stack.assertActiveInstance<MenuChild>()
}
sample/shared/shared/src/commonMain/kotlin/com/arkivanov/sample/shared/root/PreviewRootComponent.kt (1)

14-18: Consider documenting experimental API usage.

The implementation looks correct, but since this component uses experimental features:

  1. Consider adding KDoc comments explaining that this component uses experimental web navigation features
  2. Document any potential limitations or behavioral differences of the NoOp implementation

Here's a suggested documentation addition:

+/**
+ * Preview implementation of RootComponent that provides a no-op web navigation.
+ * Note: Uses experimental Decompose API for web navigation support.
+ */
@OptIn(ExperimentalDecomposeApi::class)
class PreviewRootComponent :
    RootComponent,
    ComponentContext by PreviewComponentContext,
    WebNavigationOwner.NoOp {
sample/shared/shared/src/commonMain/kotlin/com/arkivanov/sample/shared/sharedtransitions/gallery/DefaultGalleryComponent.kt (1)

6-7: LGTM! Good architectural improvements

The changes to use ID-based selection instead of indices and making images an override property are good architectural improvements. This approach:

  • Makes the component more suitable for web navigation (IDs are more stable than indices)
  • Improves interface compliance with explicit override
  • Provides better separation of concerns by injecting images instead of generating them internally
sample/shared/shared/src/commonTest/kotlin/com/arkivanov/sample/shared/root/RootComponentIntegrationTest.kt (2)

17-47: Consider extracting the base URL constant and improving test names.

The test methods could benefit from some minor improvements:

  1. Extract the duplicated base URL into a constant
  2. Consider making test names more URL-centric since we're testing URL-based navigation
class RootComponentIntegrationTest {
+    private companion object {
+        private const val BASE_URL = "https://example.com"
+    }

-    fun WHEN_created_with_deeplink_empty_THEN_TabsChild_active() {
+    fun WHEN_created_with_root_url_THEN_TabsChild_active() {
-        val component = createComponent(deepLink = Url(url = "https://example.com"))
+        val component = createComponent(deepLink = Url(url = BASE_URL))

Line range hint 1-61: Consider documenting the DeepLink to Url migration path.

This change represents a significant API transition from DeepLink to Url. Consider adding migration documentation or deprecation warnings to help users transition their existing code.

Suggestions:

  1. Add a migration guide in the documentation
  2. Consider providing a compatibility layer or utility functions to help users migrate from DeepLink to Url
  3. Document any behavioral differences between the old and new APIs
sample/shared/shared/src/commonMain/kotlin/com/arkivanov/sample/shared/Utils.kt (1)

20-29: Consider handling additional edge cases in snakeCase()

The implementation works for basic camelCase to snake_case conversion, but could be enhanced to handle more edge cases.

Consider this improved implementation:

 internal fun String.snakeCase(): String =
     buildString {
+        if (isEmpty()) return@buildString
+        var previousIsUpperCase = false
         for (c in this@snakeCase) {
-            if (c.isUpperCase() && isNotEmpty()) {
+            val isUpperCase = c.isUpperCase()
+            if (isUpperCase && isNotEmpty() && (!previousIsUpperCase || 
+                (previousIsUpperCase && [email protected]([email protected](c) + 1)?.isLowerCase() == true))) {
                 append('_')
             }
-
             append(c.lowercaseChar())
+            previousIsUpperCase = isUpperCase
         }
     }

This handles:

  • Consecutive uppercase letters (e.g., "XMLParser" -> "xml_parser")
  • Acronyms followed by words (e.g., "myXMLParser" -> "my_xml_parser")
sample/shared/shared/src/commonMain/kotlin/com/arkivanov/sample/shared/tabs/PreviewTabsComponent.kt (1)

Line range hint 14-24: Consider documenting the NoOp behavior

While the implementation is correct, it would be helpful to add a KDoc comment explaining why this preview component implements WebNavigationOwner.NoOp and what that means for testing/preview purposes.

Here's a suggested addition:

@OptIn(ExperimentalDecomposeApi::class)
+/**
+ * A preview implementation of TabsComponent that implements WebNavigationOwner.NoOp
+ * to indicate it doesn't handle web navigation events, as it's intended for testing
+ * and preview purposes only.
+ */
class PreviewTabsComponent : TabsComponent, WebNavigationOwner.NoOp {
sample/shared/shared/src/commonMain/kotlin/com/arkivanov/sample/shared/pages/PagesComponent.kt (1)

Line range hint 8-16: Consider documenting web navigation behavior

The interface maintains good separation of concerns while adding web navigation support. Consider adding KDoc comments to document:

  • How web navigation interacts with page selection
  • Expected behavior when navigating through browser history
  • Implementation requirements for WebNavigationOwner
sample/shared/shared/src/commonMain/kotlin/com/arkivanov/sample/shared/root/RootComponent.kt (1)

21-25: Consider using annotations or constants for route paths

While the routing comments provide valuable information, consider:

  1. Using annotations or constants to define routes instead of comments
  2. Adding route documentation for all Child variants for consistency
  3. Adding validation to ensure route paths are unique

This would make the routing configuration more maintainable and type-safe.

Example implementation:

sealed class Child {
    @Route("/tabs")
    class TabsChild(val component: TabsComponent) : Child()
    
    @Route("/dynamic-features")
    class DynamicFeaturesChild(val component: DynamicFeaturesComponent) : Child()
    
    // ... similar for other variants
}
sample/shared/compose/src/commonMain/kotlin/com/arkivanov/sample/shared/utils/Utils.kt (1)

11-12: Add KDoc documentation for the WebDocumentTitle function.

The function signature looks good, but adding documentation would help explain its purpose and usage pattern in the web navigation context.

Add documentation like:

+/**
+ * Sets the document title in web environments. Has no effect on other platforms.
+ * 
+ * @param title The title to be displayed in the browser's title bar or tab
+ */
@Composable
internal expect fun WebDocumentTitle(title: String)
sample/shared/compose/src/commonMain/kotlin/com/arkivanov/sample/shared/counters/CountersContent.kt (2)

27-27: Consider parameterizing the title.

The hardcoded title "Counters" could be made more flexible:

Consider this approach:

-internal fun CountersContent(component: CountersComponent, modifier: Modifier = Modifier) {
-    WebDocumentTitle(title = "Counters")
+internal fun CountersContent(
+    component: CountersComponent,
+    modifier: Modifier = Modifier,
+    title: String = "Counters"
+) {
+    WebDocumentTitle(title = title)

This would allow for localization and dynamic titles while maintaining the default behavior.


Line range hint 27-33: Ensure title changes are announced to screen readers.

When the document title changes, screen reader users should be notified.

Consider implementing an accessibility service that announces title changes to screen readers. This could be done by:

  1. Creating a common interface for accessibility announcements
  2. Implementing platform-specific announcement mechanisms
  3. Integrating with the WebDocumentTitle component

Would you like me to provide a detailed implementation for this accessibility enhancement?

sample/shared/compose/src/commonMain/kotlin/com/arkivanov/sample/shared/menu/MenuContent.kt (1)

20-20: LGTM! Consider standardizing the title format.

The implementation of WebDocumentTitle is correct and well-placed. However, for consistency with other components (e.g., "Shared Transitions Gallery"), consider using a more descriptive title.

-    WebDocumentTitle(title = "Menu")
+    WebDocumentTitle(title = "Decompose Sample Menu")

Also applies to: 27-28

sample/shared/compose/src/commonMain/kotlin/com/arkivanov/sample/shared/sharedtransitions/photo/PhotoContent.kt (1)

20-20: Consider formatting the photo title for better user experience.

While using a dynamic title is good, using resourceId.name directly might expose implementation details or show unfriendly names to users.

Consider formatting the title more user-friendly:

-    WebDocumentTitle(title = "Shared Transitions ${component.image.resourceId.name}")
+    WebDocumentTitle(title = "Shared Transitions - Photo ${component.image.id}")

Also applies to: 29-30

sample/shared/compose/src/commonMain/kotlin/com/arkivanov/sample/shared/sharedtransitions/gallery/GalleryContent.kt (1)

Line range hint 20-34: Consider centralizing web document title management.

The current implementation requires each component to manage its own document title. Consider introducing a centralized title management system that could:

  1. Enforce consistent title formatting across components
  2. Handle title updates through the navigation system
  3. Support hierarchical titles for nested navigation
  4. Provide type-safe title templates

This would align better with the PR's objective of supporting nested navigation.

Example approach:

object WebTitleTemplate {
    const val MAIN = "Decompose Sample - {0}"
    const val PHOTO = "Shared Transitions - Photo {0}"
    // ... other templates
}

@Composable
fun rememberWebTitle(template: String, vararg args: Any) {
    WebDocumentTitle(title = template.format(*args))
}
sample/shared/compose/src/commonMain/kotlin/com/arkivanov/sample/shared/multipane/MultiPaneContent.kt (1)

37-37: Consider making the document title dynamic based on navigation state

Since this is a multi-pane layout with nested navigation, consider making the title dynamic to reflect the current navigation state (e.g., "Multi-Pane Layout - Article Title - Author Name"). This would improve the web navigation experience by providing more context in browser history.

Example implementation:

- WebDocumentTitle(title = "Multi-Pane Layout")
+ val panels by component.panels.subscribeAsState()
+ val title = buildString {
+     append("Multi-Pane Layout")
+     panels.activeChild?.instance?.let { details ->
+         append(" - ${details.article.title}")
+         details.author?.let { author ->
+             append(" - ${author.name}")
+         }
+     }
+ }
+ WebDocumentTitle(title = title)
sample/shared/compose/src/commonMain/kotlin/com/arkivanov/sample/shared/cards/CardsContent.kt (1)

59-59: Consider making the document title dynamic based on stack state

Since this is a card stack with navigation, consider making the title dynamic to reflect the current card state (e.g., "Cards (3/5)"). This would improve the web navigation experience by providing more context in browser history.

Example implementation:

- WebDocumentTitle(title = "Cards")
+ val stack by component.stack.subscribeAsState()
+ val title = "Cards (${stack.items.size})"
+ WebDocumentTitle(title = title)
sample/shared/shared/src/commonMain/kotlin/com/arkivanov/sample/shared/multipane/DefaultMultiPaneComponent.kt (2)

38-38: Consider adding documentation for the deepLinkUrl parameter.

The constructor now includes an optional deepLinkUrl: Url? parameter. Adding KDoc comments to explain its purpose and usage will enhance code readability and assist other developers in understanding how to initialize the component with deep links.


123-131: Add unit tests for the getInitialPanels function.

The getInitialPanels function initializes panels based on the deepLinkUrl. To ensure correct behavior across different scenarios (e.g., null URLs, valid parameters, invalid parameters), consider adding unit tests that cover these cases.

Would you like assistance in creating unit tests for this function?

sample/shared/shared/src/commonMain/kotlin/com/arkivanov/sample/shared/pages/DefaultPagesComponent.kt (1)

29-29: Consider renaming _pages to align with Kotlin naming conventions

In Kotlin, it's conventional to avoid leading underscores in property names. Consider renaming _pages to pagesValue or pagesState for clarity and consistency with idiomatic Kotlin code.

sample/shared/shared/src/commonMain/kotlin/com/arkivanov/sample/shared/root/DefaultRootComponent.kt (3)

77-78: Consider passing deepLinkUrl when pushing new Config instances.

In the navigation actions for onPagesItemSelected and onSharedTransitionsItemSelected, you're pushing new configurations without passing the deepLinkUrl. If deep linking is relevant, consider passing deepLinkUrl to maintain consistency and ensure proper navigation.

Apply this diff to pass the deepLinkUrl:

 onPagesItemSelected = {
-    nav.pushNew(Config.Pages())
+    nav.pushNew(Config.Pages(deepLinkUrl = config.deepLinkUrl))
 },
 onSharedTransitionsItemSelected = {
-    nav.pushNew(Config.SharedTransitions())
+    nav.pushNew(Config.SharedTransitions(deepLinkUrl = config.deepLinkUrl))
 },

126-135: Handle unrecognized paths explicitly in getInitialStack.

In the else clause of your when expression, an unrecognized path leads to Config.Tabs(deepLinkUrl = childUrl). Consider logging a warning or handling this case explicitly to aid in debugging and to improve the user experience when an unknown path is encountered.


Line range hint 141-153: Maintain consistency in Config implementations.

While Tabs, Pages, and SharedTransitions are now data classes with a deepLinkUrl parameter, DynamicFeatures and CustomNavigation remain data objects. For consistency, consider converting them to data classes, even if they don't currently have parameters. This can simplify future expansions and maintain a uniform code structure.

Apply this diff to update the implementations:

 @Serializable
-data object DynamicFeatures : Config
+data class DynamicFeatures(val deepLinkUrl: Url? = null) : Config

 @Serializable
-data object CustomNavigation : Config
+data class CustomNavigation(val deepLinkUrl: Url? = null) : Config
decompose/src/commonMain/kotlin/com/arkivanov/decompose/router/stack/StackWebNavigation.kt (6)

38-38: Typographical Error in Parameter Description

In the KDoc for the enableHistory parameter, there's a typographical error. The word "enabled" should be "enables" to match the present tense.

Apply this diff to correct the typo:

- * @param enableHistory a flag that enabled or disables the back stack history.
+ * @param enableHistory a flag that enables or disables the back stack history.

39-41: Clarify Sentence Structure for Better Readability

The sentence explaining the enableHistory parameter can be broken into two sentences for better clarity and readability.

Apply this diff to improve the documentation:

- * When `false`, the browser navigation history follows only the currently
- * active child (i.e. the back stack will not be taken into account), can be useful
- * for bottom tab navigation. Default value is `true`.
+ * When `false`, the browser navigation history follows only the currently
+ * active child (i.e., the back stack will not be taken into account). This can be useful
+ * for bottom tab navigation.
+ * Default value is `true`.

53-53: Improve Readability in Documentation

In the KDoc, replace "e.g." with "e.g.," to follow standard punctuation conventions.

Apply this diff:

- * Can be used e.g. to display an alert dialog before closing a screen. Returns `true`
+ * Can be used, e.g., to display an alert dialog before closing a screen. Returns `true`

63-68: Specify a More Precise Return Type

The function childStackWebNavigation currently returns WebNavigation<*>. For better type safety and consistency, consider specifying the generic type C in the return type.

Apply this diff to enhance type safety:

- ): WebNavigation<*> =
+ ): WebNavigation<C> =

94-101: Optimize Mapping Logic in toHistory Function

Consider simplifying the toHistory function by directly mapping over the desired list without the conditional expression.

Apply this diff to streamline the code:

-    private fun ChildStack<C, T>.toHistory(): List<HistoryItem<C>> =
-        (if (enableHistory) items else items.takeLast(1)).map { child ->
+    private fun ChildStack<C, T>.toHistory(): List<HistoryItem<C>> = 
+        val relevantItems = if (enableHistory) items else items.takeLast(1)
+        relevantItems.map { child ->
             HistoryItem(
                 path = pathMapper(child) ?: "",
                 parameters = parametersMapper(child) ?: emptyMap(),
                 key = child.configuration,
                 child = childSelector(child),
             )
         }

106-108: Add Error Handling for Navigation

Consider adding error handling or validation when navigating with the provided history list to ensure that the navigator handles unexpected input gracefully.

Apply this diff to include validation:

 override fun navigate(history: List<C>) {
+    if (history.isEmpty()) {
+        // Handle empty history case if necessary
+        return
+    }
     navigator.navigate { history }
 }
sample/app-js/src/main/kotlin/com/arkivanov/sample/app/Main.kt (1)

32-32: Consider removing console.log statement from production code

The console.log(root.stack.value) statement may be intended for debugging purposes. Consider removing it to avoid unnecessary console output in production.

sample/shared/shared/src/commonMain/kotlin/com/arkivanov/sample/shared/tabs/DefaultTabsComponent.kt (2)

48-48: Be cautious with experimental APIs

The use of @OptIn(ExperimentalDecomposeApi::class) indicates reliance on experimental features that may change in future releases. Ensure that this dependency is acceptable and consider the potential need for future adjustments.


100-100: Reminder to address the TODO comment

There is a TODO comment indicating incomplete handling of the childUrl in the path processing within getInitialConfig. Ensure this logic is implemented to correctly support deep linking functionality.

Would you like assistance in implementing the path handling logic?

decompose/src/commonMain/kotlin/com/arkivanov/decompose/router/panels/PanelsWebNavigation.kt (1)

47-70: Simplify generic type parameters for better readability

The function childPanelsWebNavigation uses multiple generic type parameters (MC, MT, DC, DT, EC, ET), which increases complexity. Consider simplifying or constraining these generics if possible to enhance readability and maintainability.

decompose/src/webMain/kotlin/com/arkivanov/decompose/router/webhistory/WebHistoryNavigation.kt (1)

133-176: Refactor for improved readability and maintainability

The subscribe method is quite complex with deep nesting and multiple responsibilities. Consider refactoring it into smaller, well-named functions or classes to enhance readability and facilitate easier future modifications.

decompose/src/webTest/kotlin/com/arkivanov/decompose/router/webhistory/WebHistoryNavigationTest.kt (2)

6-7: Use standard naming conventions for test methods

The test method names contain underscores, e.g., WHEN_created_with_no_items_in_root_THEN_ISE_thrown. Kotlin supports backticks for method names with spaces, which can improve readability. Alternatively, consider using camelCase to align with Kotlin conventions.


9-766: Consider grouping related tests and adding comments

There are numerous test cases covering various scenarios. Grouping related tests and adding brief comments or documentation can improve the understandability of the test suite.

decompose/api/jvm/decompose.api (2)

184-198: Confirm visibility of generated serializer classes

The generated serializer classes like Pages$$serializer and Panels$$serializer are marked as public synthetic. If they are not intended for public API consumption, consider marking them as internal to restrict their visibility.


501-527: Verify platform-specific placement of WebNavigation interface

The WebNavigation interface is added in the JVM-specific API. Since web navigation can be relevant for multiple platforms, consider moving it to a common module to ensure cross-platform availability.

decompose/src/commonMain/kotlin/com/arkivanov/decompose/router/pages/Pages.kt (1)

3-3: LGTM! Well-structured serialization support for web navigation.

The addition of @Serializable annotation is a clean and non-breaking change that enables state persistence for web navigation history. The existing validation logic in the init block ensures data integrity is maintained during serialization/deserialization.

This change forms a crucial part of the web history API, allowing navigation state to be preserved in the browser's history stack. The serialization capability will be particularly valuable for handling browser refresh and back/forward navigation.

Also applies to: 12-12

sample/shared/shared/src/commonMain/kotlin/com/arkivanov/sample/shared/multipane/MultiPaneComponent.kt (1)

6-6: Consider documenting web navigation behavior in interface KDoc.

The addition of WebNavigationOwner cleanly extends the component's capabilities. However, since this is using experimental API and implements complex navigation patterns, it would be beneficial to document:

  • How web navigation interacts with back handling
  • Expected URL structure for the multi-pane layout
  • State preservation behavior during navigation

Would you like me to provide a template for the interface documentation?

Also applies to: 14-14

sample/shared/shared/src/commonMain/kotlin/com/arkivanov/sample/shared/tabs/TabsComponent.kt (1)

4-4: LGTM! Consider adding documentation for web navigation.

The extension of WebNavigationOwner aligns well with the PR objectives for web history API support. The change is clean and maintains the interface's existing functionality.

Consider adding KDoc comments to document the web navigation capabilities and any specific requirements or behaviors that implementers should be aware of.

Also applies to: 11-11

sample/shared/shared/src/commonMain/kotlin/com/arkivanov/sample/shared/Url.kt (2)

28-29: Consider performance optimization for path segment consumption.

The current implementation creates a new list on every consumePathSegment call.

Consider using a more efficient data structure or caching mechanism if this operation is frequently used:

-fun Url.consumePathSegment(): Pair<String?, Url> =
-    pathSegments.firstOrNull() to copy(pathSegments = pathSegments.drop(1))
+fun Url.consumePathSegment(): Pair<String?, Url> {
+    val first = pathSegments.firstOrNull() ?: return null to this
+    return first to Url(
+        pathSegments = pathSegments.subList(1, pathSegments.size),
+        parameters = parameters
+    )
+}

32-36: Document internal utility functions.

The internal path utility functions lack documentation explaining their purpose and usage.

Add KDoc comments:

+/**
+ * Converts the class name of the receiver to snake_case format.
+ * Internal utility for generating URL path segments.
+ */
 internal fun Any.path(): String =
     this::class.simpleName?.snakeCase() ?: ""

+/**
+ * Converts the class name of type T to snake_case format.
+ * Internal utility for generating URL path segments.
+ * @return The snake_case representation of T's class name
+ */
 internal inline fun <reified T : Any> pathSegmentOf(): String =
     T::class.simpleName?.snakeCase() ?: ""
sample/shared/shared/src/commonMain/kotlin/com/arkivanov/sample/shared/multipane/PreviewMultiPaneComponent.kt (1)

21-23: LGTM! Consider documenting the NoOp implementation choice.

The implementation of WebNavigationOwner.NoOp is appropriate for a preview component.

Consider adding a comment explaining why WebNavigationOwner.NoOp is used here, to help other developers understand the design decision:

/**
 * Preview implementation of MultiPaneComponent that implements WebNavigationOwner.NoOp
 * since navigation is not needed in preview/testing scenarios.
 */
decompose/src/jsMain/kotlin/com/arkivanov/decompose/router/webhistory/WebHistoryNavigation.kt (3)

14-20: Enhance documentation with usage examples and deep linking format

While the documentation explains the basic purpose, it would be beneficial to include:

  • Example usage code
  • Expected deep link format
  • State persistence behavior

25-33: Consider potential race condition in state restoration

The state restoration logic might have a race condition if multiple tabs are opened simultaneously. Consider adding a tab-specific identifier to the storage keys.


35-40: Consider using a more robust state persistence mechanism

The current implementation uses sessionStorage which is cleared when the browser session ends. For better user experience, consider:

  1. Adding an option to use localStorage for persistent state
  2. Implementing a cleanup mechanism for old states
decompose/src/webTest/kotlin/com/arkivanov/decompose/router/webhistory/TestWebNavigation.kt (3)

12-16: LGTM! Consider adding KDoc documentation.

The class constructor is well-designed with sensible defaults for testing. Consider adding KDoc documentation to describe the purpose of this test implementation and its parameters.

+/**
+ * Test implementation of WebNavigation interface that uses Integer configurations.
+ * 
+ * @param initialHistory The initial navigation history to set up
+ * @param onBeforeNavigate Callback to control navigation permission, defaults to always allow
+ * @param childFactory Factory function to create child navigations, defaults to no children
+ */
 class TestWebNavigation(
     initialHistory: List<Int>,
     private val onBeforeNavigate: () -> Boolean = { true },
     private val childFactory: (Int) -> TestWebNavigation? = { null },
 ) : WebNavigation<Int> {

35-56: Consider handling edge cases in navigation.

The navigation logic could be more robust by:

  1. Validating input history list (null check, empty check)
  2. Handling potential exceptions during config.toString()
  3. Adding bounds checking for large history lists
 override fun navigate(history: List<Int>) {
+    requireNotNull(history) { "History list cannot be null" }
+    
     children.keys.removeAll { it !in history }
 
     history.forEach { config ->
         if (config !in children) {
             val child = childFactory(config)
             if (child != null) {
                 children[config] = child
             }
         }
     }
 
     _history.value =
         history.map { config ->
             HistoryItem(
                 key = config,
                 path = config.toString(),
                 parameters = emptyMap(),
                 child = children[config]?.let(TestWebNavigation::Owner),
             )
         }
 }

58-82: Enhance assertion error messages for better debugging.

The assertions would benefit from more descriptive error messages to help identify test failures quickly.

 fun assertHistory(configs: Iterable<Int>) {
-    assertContentEquals(configs, _history.value.map { it.key })
+    assertContentEquals(
+        configs,
+        _history.value.map { it.key },
+        "Navigation history mismatch: expected=$configs, actual=${_history.value.map { it.key }}"
+    )
 }
📜 Review details

Configuration used: CodeRabbit UI
Review profile: CHILL

📥 Commits

Reviewing files that changed from the base of the PR and between a3f0869 and fc78ab9.

📒 Files selected for processing (51)
  • decompose/api/android/decompose.api (6 hunks)
  • decompose/api/decompose.klib.api (6 hunks)
  • decompose/api/jvm/decompose.api (6 hunks)
  • decompose/src/commonMain/kotlin/com/arkivanov/decompose/router/pages/Pages.kt (1 hunks)
  • decompose/src/commonMain/kotlin/com/arkivanov/decompose/router/pages/PagesWebNavigation.kt (1 hunks)
  • decompose/src/commonMain/kotlin/com/arkivanov/decompose/router/panels/PanelsWebNavigation.kt (1 hunks)
  • decompose/src/commonMain/kotlin/com/arkivanov/decompose/router/stack/ChildStackFactory.kt (1 hunks)
  • decompose/src/commonMain/kotlin/com/arkivanov/decompose/router/stack/StackWebNavigation.kt (1 hunks)
  • decompose/src/commonMain/kotlin/com/arkivanov/decompose/router/webhistory/NoOpWebNavigation.kt (1 hunks)
  • decompose/src/commonMain/kotlin/com/arkivanov/decompose/router/webhistory/WebNavigation.kt (1 hunks)
  • decompose/src/commonMain/kotlin/com/arkivanov/decompose/router/webhistory/WebNavigationOwner.kt (1 hunks)
  • decompose/src/jsMain/kotlin/com/arkivanov/decompose/router/webhistory/WebHistoryNavigation.kt (1 hunks)
  • decompose/src/wasmJsMain/kotlin/com/arkivanov/decompose/router/webhistory/WebHistoryNavigation.kt (1 hunks)
  • decompose/src/webMain/kotlin/com/arkivanov/decompose/Utils.kt (1 hunks)
  • decompose/src/webMain/kotlin/com/arkivanov/decompose/router/stack/Utils.kt (2 hunks)
  • decompose/src/webMain/kotlin/com/arkivanov/decompose/router/webhistory/WebHistoryNavigation.kt (1 hunks)
  • decompose/src/webTest/kotlin/com/arkivanov/decompose/router/webhistory/TestWebNavigation.kt (1 hunks)
  • decompose/src/webTest/kotlin/com/arkivanov/decompose/router/webhistory/WebHistoryNavigationTest.kt (1 hunks)
  • sample/app-js-compose/src/jsMain/kotlin/com/arkivanov/decompose/sample/app/Main.kt (1 hunks)
  • sample/app-js-compose/src/jsMain/resources/index.html (1 hunks)
  • sample/app-js/src/main/kotlin/com/arkivanov/sample/app/Main.kt (2 hunks)
  • sample/app-js/src/main/resources/index.html (1 hunks)
  • sample/shared/compose/build.gradle.kts (1 hunks)
  • sample/shared/compose/src/commonMain/kotlin/com/arkivanov/sample/shared/cards/CardsContent.kt (1 hunks)
  • sample/shared/compose/src/commonMain/kotlin/com/arkivanov/sample/shared/counters/CountersContent.kt (1 hunks)
  • sample/shared/compose/src/commonMain/kotlin/com/arkivanov/sample/shared/menu/MenuContent.kt (1 hunks)
  • sample/shared/compose/src/commonMain/kotlin/com/arkivanov/sample/shared/multipane/MultiPaneContent.kt (1 hunks)
  • sample/shared/compose/src/commonMain/kotlin/com/arkivanov/sample/shared/sharedtransitions/gallery/GalleryContent.kt (2 hunks)
  • sample/shared/compose/src/commonMain/kotlin/com/arkivanov/sample/shared/sharedtransitions/photo/PhotoContent.kt (2 hunks)
  • sample/shared/compose/src/commonMain/kotlin/com/arkivanov/sample/shared/utils/Utils.kt (1 hunks)
  • sample/shared/compose/src/jsMain/kotlin/com/arkivanov/sample/shared/utils/Utils.js.kt (1 hunks)
  • sample/shared/compose/src/nonWebMain/kotlin/com/arkivanov/sample/shared/utils/Utils.nonWeb.kt (1 hunks)
  • sample/shared/shared/src/commonMain/kotlin/com/arkivanov/sample/shared/Url.kt (1 hunks)
  • sample/shared/shared/src/commonMain/kotlin/com/arkivanov/sample/shared/Utils.kt (1 hunks)
  • sample/shared/shared/src/commonMain/kotlin/com/arkivanov/sample/shared/multipane/DefaultMultiPaneComponent.kt (4 hunks)
  • sample/shared/shared/src/commonMain/kotlin/com/arkivanov/sample/shared/multipane/MultiPaneComponent.kt (1 hunks)
  • sample/shared/shared/src/commonMain/kotlin/com/arkivanov/sample/shared/multipane/PreviewMultiPaneComponent.kt (2 hunks)
  • sample/shared/shared/src/commonMain/kotlin/com/arkivanov/sample/shared/pages/DefaultPagesComponent.kt (2 hunks)
  • sample/shared/shared/src/commonMain/kotlin/com/arkivanov/sample/shared/pages/PagesComponent.kt (1 hunks)
  • sample/shared/shared/src/commonMain/kotlin/com/arkivanov/sample/shared/root/DefaultRootComponent.kt (5 hunks)
  • sample/shared/shared/src/commonMain/kotlin/com/arkivanov/sample/shared/root/PreviewRootComponent.kt (1 hunks)
  • sample/shared/shared/src/commonMain/kotlin/com/arkivanov/sample/shared/root/RootComponent.kt (2 hunks)
  • sample/shared/shared/src/commonMain/kotlin/com/arkivanov/sample/shared/sharedtransitions/DefaultSharedTransitionsComponent.kt (3 hunks)
  • sample/shared/shared/src/commonMain/kotlin/com/arkivanov/sample/shared/sharedtransitions/SharedTransitionsComponent.kt (1 hunks)
  • sample/shared/shared/src/commonMain/kotlin/com/arkivanov/sample/shared/sharedtransitions/gallery/DefaultGalleryComponent.kt (1 hunks)
  • sample/shared/shared/src/commonMain/kotlin/com/arkivanov/sample/shared/sharedtransitions/photo/Image.kt (0 hunks)
  • sample/shared/shared/src/commonMain/kotlin/com/arkivanov/sample/shared/tabs/DefaultTabsComponent.kt (6 hunks)
  • sample/shared/shared/src/commonMain/kotlin/com/arkivanov/sample/shared/tabs/PreviewTabsComponent.kt (1 hunks)
  • sample/shared/shared/src/commonMain/kotlin/com/arkivanov/sample/shared/tabs/TabsComponent.kt (1 hunks)
  • sample/shared/shared/src/commonTest/kotlin/com/arkivanov/sample/shared/root/RootComponentIntegrationTest.kt (2 hunks)
  • sample/shared/shared/src/commonTest/kotlin/com/arkivanov/sample/shared/tabs/TabsComponentIntegrationTest.kt (1 hunks)
💤 Files with no reviewable changes (1)
  • sample/shared/shared/src/commonMain/kotlin/com/arkivanov/sample/shared/sharedtransitions/photo/Image.kt
✅ Files skipped from review due to trivial changes (5)
  • sample/shared/compose/src/nonWebMain/kotlin/com/arkivanov/sample/shared/utils/Utils.nonWeb.kt
  • sample/app-js/src/main/resources/index.html
  • sample/app-js-compose/src/jsMain/resources/index.html
  • decompose/src/commonMain/kotlin/com/arkivanov/decompose/router/stack/ChildStackFactory.kt
  • decompose/src/commonMain/kotlin/com/arkivanov/decompose/router/webhistory/NoOpWebNavigation.kt
🧰 Additional context used
🪛 detekt (1.23.7)
decompose/src/webMain/kotlin/com/arkivanov/decompose/Utils.kt

[warning] 17-17: The caught exception is swallowed. The original exception could be lost.

(detekt.exceptions.SwallowedException)

🔇 Additional comments (43)
sample/shared/shared/src/commonTest/kotlin/com/arkivanov/sample/shared/tabs/TabsComponentIntegrationTest.kt (1)

112-112: LGTM! The change maintains backward compatibility.

The addition of deepLinkUrl = null aligns with the updated DefaultTabsComponent constructor while preserving existing test behavior.

sample/shared/shared/src/commonMain/kotlin/com/arkivanov/sample/shared/root/PreviewRootComponent.kt (2)

4-4: LGTM: Import additions align with web navigation implementation.

The new imports support the web navigation functionality and experimental API usage.

Also applies to: 6-6


15-18: Verify consistent web navigation implementation across preview components.

Let's ensure this implementation pattern is consistent across other preview components.

✅ Verification successful

Implementation pattern is consistent across preview components

The verification shows consistent implementation of WebNavigationOwner.NoOp across all preview components:

  • PreviewRootComponent
  • PreviewTabsComponent
  • PreviewMultiPaneComponent

All follow the same pattern of implementing WebNavigationOwner.NoOp interface along with their respective component interfaces.

🏁 Scripts executed

The following scripts were executed for the analysis:

Script:

#!/bin/bash
# Description: Check implementation patterns across preview components
# Expected: Other preview components should follow similar pattern for web navigation

# Search for other preview components implementing WebNavigationOwner
echo "Checking WebNavigationOwner implementation in preview components:"
rg "class Preview.*Component.*WebNavigationOwner" -A 3

# Search for NoOp usage pattern
echo "Checking NoOp implementation pattern:"
rg "WebNavigationOwner\.NoOp" -B 2 -A 2

Length of output: 4017

sample/shared/shared/src/commonMain/kotlin/com/arkivanov/sample/shared/sharedtransitions/gallery/DefaultGalleryComponent.kt (1)

11-12: LGTM! Verify ID propagation

The implementation correctly propagates the image ID to the callback. The change from index-based to ID-based selection is consistently applied.

Let's verify that all image IDs are properly handled throughout the navigation chain:

✅ Verification successful

Let me gather more information about the image ID handling across the codebase.


ID propagation is correctly implemented

The verification confirms proper ID handling throughout the navigation chain:

  • Image data class correctly defines id: Int as a property
  • DefaultGalleryComponent properly propagates the ID from onImageClicked to onImageSelected
  • DefaultSharedTransitionsComponent correctly uses the ID to create a new Photo configuration (Config.Photo(id = it))
  • The navigation flow maintains ID consistency from gallery selection through to photo display
🏁 Scripts executed

The following scripts were executed for the analysis:

Script:

#!/bin/bash
# Description: Check how image IDs are used in the navigation flow

# Look for Image class definition and ID usage
ast-grep --pattern 'class Image {
  $$$
}'

# Check for other components handling image IDs
rg -A 3 "onImageSelected.*id.*Int" 

Length of output: 693


Script:

#!/bin/bash
# Search for Image data class and its usages
ast-grep --pattern 'data class Image'

# Look for image ID references in function parameters and properties
rg "id: Int" -A 2

# Check for image selection handling in interfaces and implementations
rg "onImageClicked|onImageSelected" -A 2

Length of output: 7009

sample/shared/shared/src/commonTest/kotlin/com/arkivanov/sample/shared/root/RootComponentIntegrationTest.kt (1)

51-59: Verify all DefaultRootComponent usages are updated.

The parameter type change from DeepLink to Url? looks good, but we should verify that all usages of DefaultRootComponent across the codebase have been updated to use the new deepLinkUrl parameter.

✅ Verification successful

All DefaultRootComponent usages have been updated correctly

The search results show that all instantiations of DefaultRootComponent across the codebase are consistent with the new parameter type. The key findings:

  • The main implementation in DefaultRootComponent.kt correctly defines deepLinkUrl: Url? = null
  • All actual usages in application code (Android, iOS, JS, Desktop) either:
    • Pass the deepLinkUrl parameter correctly (e.g., in JS apps with deepLinkUrl = deepLink?.let(::Url))
    • Or rely on the default null value by not specifying the parameter
  • The test implementation in RootComponentIntegrationTest.kt correctly uses the new parameter name and type
🏁 Scripts executed

The following scripts were executed for the analysis:

Script:

#!/bin/bash
# Search for any remaining DeepLink usage in DefaultRootComponent instantiations
ast-grep --pattern 'DefaultRootComponent($$$deepLink$$$)'

# Search for all DefaultRootComponent instantiations to verify parameter usage
rg "DefaultRootComponent\(" -A 5

Length of output: 16785

sample/shared/shared/src/commonMain/kotlin/com/arkivanov/sample/shared/Utils.kt (1)

20-29: Verify usage patterns of snakeCase() in web navigation

Since this utility is likely used for URL path segments or state serialization, let's verify its usage patterns.

✅ Verification successful

Let me gather more context about the usage patterns by examining the files where snakeCase() is used.


Based on the gathered information, I can now provide the final response:

The snakeCase() utility is correctly implemented for URL path generation

The utility is used internally for converting class names to URL path segments in a consistent manner. The implementation correctly handles:

  • Converting class names to lowercase URL-friendly segments
  • Adding underscores between words for readability
  • Being used only in internal path generation utilities (path() and pathSegmentOf())

The usage is well-scoped and follows web URL conventions for path segments.

🏁 Scripts executed

The following scripts were executed for the analysis:

Script:

#!/bin/bash
# Find all usages of snakeCase() in the codebase
echo "Searching for snakeCase() usage patterns..."
rg -l "\.snakeCase\(\)" | while read -r file; do
    echo "=== $file ==="
    rg -B 2 -A 2 "\.snakeCase\(\)" "$file"
done

# Look for potential web navigation related files that might use this
echo -e "\nSearching for potential web navigation files..."
rg -l "WebNavigation|WebPath|WebSerializer" 

Length of output: 4017


Script:

#!/bin/bash
# Check the content of Url.kt to understand how snakeCase is used in path generation
echo "=== Content of Url.kt ==="
cat sample/shared/shared/src/commonMain/kotlin/com/arkivanov/sample/shared/Url.kt

# Check for any path-related usages in web navigation files
echo -e "\n=== Checking path generation in web navigation files ==="
rg -B 3 -A 3 "path|Path" sample/shared/shared/src/commonMain/kotlin/com/arkivanov/sample/shared/Url.kt decompose/src/commonMain/kotlin/com/arkivanov/decompose/router/webhistory/WebNavigation.kt

# Look for any URL path segment handling in web navigation
echo -e "\n=== Checking URL path handling in WebNavigation ==="
rg -B 3 -A 3 "url|URL|path|Path" decompose/src/commonMain/kotlin/com/arkivanov/decompose/router/webhistory/WebNavigation.kt

Length of output: 8068

sample/shared/shared/src/commonMain/kotlin/com/arkivanov/sample/shared/tabs/PreviewTabsComponent.kt (2)

3-3: LGTM: Required imports added for web navigation support

The new imports support the web navigation functionality and experimental API usage.

Also applies to: 5-5


11-12: Verify the implications of using experimental API

The component now implements WebNavigationOwner.NoOp and uses experimental features. A few points to consider:

  1. The @OptIn annotation correctly acknowledges the use of experimental API
  2. The NoOp implementation suggests this is a preview/test component that intentionally doesn't handle web navigation

Let's verify the consistent usage of this pattern across other preview components:

✅ Verification successful

Let me gather more information about the implementation of other preview components to verify if they consistently implement WebNavigationOwner.NoOp.


Let me check specifically for the WebNavigationOwner interface and its usage to understand the complete context.


The experimental API usage and WebNavigationOwner.NoOp implementation are correctly applied

The verification shows consistent patterns across preview components:

  1. All preview components that require web navigation (PreviewRootComponent, PreviewTabsComponent, PreviewMultiPaneComponent) correctly implement WebNavigationOwner.NoOp
  2. The @OptIn(ExperimentalDecomposeApi::class) annotation is consistently used in all preview components
  3. The NoOp implementation is documented in the source as being specifically designed for "fake or preview component implementations"
🏁 Scripts executed

The following scripts were executed for the analysis:

Script:

#!/bin/bash
# Description: Check if other preview components follow the same pattern
# Expected: Other preview components should also implement WebNavigationOwner.NoOp

# Search for other preview components and their implementation
ast-grep --pattern 'class Preview$_Component : $_, WebNavigationOwner.NoOp'

# Check for consistent @OptIn usage in preview components
rg -l "class Preview.*Component" | xargs rg "@OptIn.*ExperimentalDecomposeApi"

Length of output: 588


Script:

#!/bin/bash
# Check the implementation details of all preview components
rg "class Preview.*Component.*:.*" -A 2 sample/shared/shared/src/commonMain/kotlin/com/arkivanov/sample/shared/

Length of output: 4249


Script:

#!/bin/bash
# Search for WebNavigationOwner interface definition and its usages
rg "WebNavigationOwner" -A 3

# Also check for any other components that might implement NoOp
rg "WebNavigationOwner\.NoOp" -A 2

Length of output: 39462

sample/shared/shared/src/commonMain/kotlin/com/arkivanov/sample/shared/pages/PagesComponent.kt (2)

4-4: LGTM: Import added correctly

The WebNavigationOwner import is properly placed and necessary for the interface extension.


8-8: Implementation aligns with web history API objectives

The addition of WebNavigationOwner interface extends PagesComponent with web navigation capabilities while maintaining its existing contract. This change supports the PR's goal of implementing nested navigation support.

Let's verify the implementation impact:

sample/shared/shared/src/commonMain/kotlin/com/arkivanov/sample/shared/root/RootComponent.kt (3)

4-4: LGTM!

The WebNavigationOwner import is correctly added to support the new web navigation capabilities.


13-13: Verify implementations of RootComponent

The addition of WebNavigationOwner interface is aligned with the PR objectives. Let's verify that all implementing classes properly handle this new interface requirement.

✅ Verification successful

Both implementations properly handle WebNavigationOwner interface

The implementations are correctly handling the new WebNavigationOwner interface:

  • DefaultRootComponent implements it by providing the required webNavigation property
  • PreviewRootComponent explicitly implements WebNavigationOwner.NoOp, which is a valid implementation for preview purposes
🏁 Scripts executed

The following scripts were executed for the analysis:

Script:

#!/bin/bash
# Description: Find all implementations of RootComponent to verify they implement WebNavigationOwner

# Search for RootComponent implementations
ast-grep --pattern 'class $_ : $$$RootComponent$$$'

Length of output: 53


Script:

#!/bin/bash
# Let's try a different approach to find RootComponent implementations
# First, let's find the file containing RootComponent definition
fd "RootComponent.kt"

# Then search for class definitions implementing RootComponent using ripgrep
# Using -A and -B for context to better understand the implementation
rg "class.*:.*RootComponent" -A 5 -B 2

Length of output: 985


Script:

#!/bin/bash
# Let's examine the actual implementations found in the shared directory
cat "sample/shared/shared/src/commonMain/kotlin/com/arkivanov/sample/shared/root/DefaultRootComponent.kt"
cat "sample/shared/shared/src/commonMain/kotlin/com/arkivanov/sample/shared/root/PreviewRootComponent.kt"

Length of output: 7685


13-26: Verify integration with WebNavigation system

Since RootComponent is a key component in the navigation hierarchy, let's verify its integration with the web history API.

✅ Verification successful

Based on the results, I can now provide a final response as the verification is complete. The implementation shows proper integration with the web navigation system:

Web navigation integration is correctly implemented

The RootComponent properly integrates with the web navigation system through:

  • Correct implementation of WebNavigationOwner interface
  • Proper web navigation setup in DefaultRootComponent with childStackWebNavigation
  • Appropriate child component selection for navigation handling
  • Clear path mapping for web navigation states

Key implementation details:

  • WebNavigation is configured with proper serialization and path mapping
  • Child components (Tabs, Pages, SharedTransitions) are correctly wired for web navigation
  • Navigation state handling is properly implemented for the component stack
🏁 Scripts executed

The following scripts were executed for the analysis:

Script:

#!/bin/bash
# Description: Analyze the integration between RootComponent and WebNavigation

# Search for WebNavigation usage in relation to RootComponent
ast-grep --pattern 'class $_ {
  $$$
  WebNavigation$$$
  $$$
  RootComponent$$$
}'

# Search for navigation state handling
rg -A 5 "WebNavigation.*RootComponent"

Length of output: 126


Script:

#!/bin/bash
# Let's try a different approach to analyze the web navigation integration

# 1. First, let's find all files that implement or use WebNavigationOwner
rg "WebNavigationOwner"

# 2. Look for WebNavigation related implementations
rg -l "class.*implements.*WebNavigation"

# 3. Search for navigation state management in RootComponent implementation
fd RootComponent.kt --exec cat {}

# 4. Look for web navigation related methods and their usage
rg "fun.*navigate|handleNavigation|onNavigation" --type kotlin

Length of output: 33872

sample/shared/compose/src/commonMain/kotlin/com/arkivanov/sample/shared/sharedtransitions/gallery/GalleryContent.kt (1)

24-24: LGTM! Clean and consistent implementation.

The WebDocumentTitle implementation is well-placed and uses a clear, descriptive title that aligns with the component's purpose.

Also applies to: 33-34

sample/shared/compose/src/commonMain/kotlin/com/arkivanov/sample/shared/multipane/MultiPaneContent.kt (1)

32-32: LGTM: Web document title integration

The integration of WebDocumentTitle is correctly placed and matches the TopAppBar title, which is essential for web navigation history.

Also applies to: 37-37

sample/shared/compose/src/commonMain/kotlin/com/arkivanov/sample/shared/cards/CardsContent.kt (1)

54-54: LGTM: Web document title integration

The integration of WebDocumentTitle is correctly placed and matches the TopAppBar title, which is essential for web navigation history.

Also applies to: 59-59

sample/shared/shared/src/commonMain/kotlin/com/arkivanov/sample/shared/pages/DefaultPagesComponent.kt (1)

39-46: Implementation of webNavigation is correct

The use of childPagesWebNavigation to implement webNavigation enhances deep linking and web navigation support. The integration appears correct and follows the intended design.

sample/shared/shared/src/commonMain/kotlin/com/arkivanov/sample/shared/root/DefaultRootComponent.kt (1)

50-65: Verify the childSelector mapping in webNavigation.

Ensure that the childSelector function correctly handles all child types. Returning null for certain children like CustomNavigationChild and DynamicFeaturesChild should be intentional and properly managed within the navigation logic to prevent unexpected behavior.

decompose/src/commonMain/kotlin/com/arkivanov/decompose/router/stack/StackWebNavigation.kt (1)

80-89: Ensure All Parameters Are Utilized

Verify that all constructor parameters in StackWebNavigation are utilized appropriately within the class. Specifically, ensure that enableHistory, pathMapper, parametersMapper, onBeforeNavigate, and childSelector are actively used in the class methods.

sample/app-js/src/main/kotlin/com/arkivanov/sample/app/Main.kt (1)

24-30: Proper use of withWebHistory for web navigation

The withWebHistory function is correctly utilized to integrate web history support with the DefaultRootComponent. The stateKeeper and deepLink parameters are appropriately handled.

sample/app-js-compose/src/jsMain/kotlin/com/arkivanov/decompose/sample/app/Main.kt (2)

22-22: Appropriate use of @OptIn for experimental APIs

The code correctly opts into ExperimentalComposeUiApi and ExperimentalDecomposeApi to utilize experimental features.


27-33: Correct integration of web history using withWebHistory

The withWebHistory function is properly used to embed web history support with the DefaultRootComponent. The handling of stateKeeper and deepLink parameters is appropriate.

decompose/src/commonMain/kotlin/com/arkivanov/decompose/router/webhistory/WebNavigation.kt (1)

1-63: Well-defined WebNavigation interface for browser navigation

The WebNavigation interface is properly defined, providing essential properties and methods for managing web navigation and browser history integration. The documentation is comprehensive and enhances understandability.

sample/shared/shared/src/commonMain/kotlin/com/arkivanov/sample/shared/sharedtransitions/DefaultSharedTransitionsComponent.kt (1)

Line range hint 28-109: Enhanced deep link support improves user navigation

The addition of deepLinkUrl and the implementation of the getInitialStack function enhance user experience by allowing direct navigation to specific images via deep links.

decompose/src/commonMain/kotlin/com/arkivanov/decompose/router/pages/PagesWebNavigation.kt (1)

1-96: Implementation of PagesWebNavigation for enhanced browser history integration

The introduction of PagesWebNavigation and the childPagesWebNavigation function provides a robust mechanism for integrating page navigation with the browser's history. The code is well-structured, and the detailed documentation aids in understanding the functionality.

sample/shared/shared/src/commonMain/kotlin/com/arkivanov/sample/shared/tabs/DefaultTabsComponent.kt (1)

79-79: ⚠️ Potential issue

Verify implications of changing Config.MultiPane from an object to a data class

Converting Config.MultiPane from an object to a data class changes it from a singleton to multiple instances. This may impact any reference equality checks or assumptions about its uniqueness. Review usage throughout the codebase to ensure this change doesn't introduce bugs.

Run the following script to find all usages of Config.MultiPane:

✅ Verification successful

Let me search for the definition and usage patterns of Config.MultiPane to better understand the context.


Let me search specifically for the Config class definition in the sample project where we see the MultiPane usage.


Let me try one more search with a different pattern to find the Config class and its usage.


Let me try one final search to see the specific file content.


No issues with Config.MultiPane being a data class

Based on the code analysis, the change from object to data class for Config.MultiPane is safe because:

  1. The code only uses Config.MultiPane for type checking (is Config.MultiPane) and instantiation
  2. There are no reference equality checks (===) or value equality checks (==)
  3. The data class is used appropriately to carry a nullable deepLinkUrl parameter
  4. All usages are contained within this single file and follow the sealed interface pattern correctly
🏁 Scripts executed

The following scripts were executed for the analysis:

Script:

#!/bin/bash
# Description: Find all usages of Config.MultiPane in the codebase.

# Search for Config.MultiPane references
rg 'Config\.MultiPane\b'

Length of output: 576


Script:

#!/bin/bash
# Find the definition of Config class and MultiPane
ast-grep --pattern 'sealed class Config {
  $$$
}'

# Also search for any equality checks on Config.MultiPane
rg -B2 -A2 'Config\.MultiPane.*==' || rg -B2 -A2 'Config\.MultiPane.*==='

Length of output: 1291


Script:

#!/bin/bash
# Find Config class definition in the sample project
rg -B2 -A5 'sealed class Config' sample/shared/

# Also search for any pattern matching or when expressions with Config.MultiPane
rg -B2 -A2 'when.*Config' sample/shared/

Length of output: 92


Script:

#!/bin/bash
# Search for files containing both "Config" and "MultiPane" to locate the definition
rg -l 'Config.*MultiPane|MultiPane.*Config' sample/shared/

# Then cat the contents of any found files to see the full context
fd -t f . sample/shared/ -x cat {} \; | grep -A 10 -B 10 'sealed.*Config'

Length of output: 283


Script:

#!/bin/bash
# Get the content of the specific file where Config.MultiPane is used
cat sample/shared/shared/src/commonMain/kotlin/com/arkivanov/sample/shared/tabs/DefaultTabsComponent.kt

Length of output: 4919

decompose/api/android/decompose.api (5)

Line range hint 190-218: Serialization support added for Pages class looks good

The addition of the Companion object and the Pages$$serializer class correctly implements serialization for the Pages class, following Kotlin serialization conventions.


253-257: Addition of childPagesWebNavigation function enhances web navigation capabilities

The new childPagesWebNavigation function in PagesWebNavigationKt provides web navigation support for pages, effectively integrating web navigation into the pages component.


378-382: Addition of childPanelsWebNavigation function extends web navigation support

The childPanelsWebNavigation function introduced in PanelsWebNavigationKt enhances web navigation capabilities for panels, aligning with the overall navigation enhancements.


501-505: Addition of childStackWebNavigation function complements navigation features

The childStackWebNavigation function in StackWebNavigationKt provides web navigation capabilities for child stacks, complementing the other navigation enhancements.


521-547: Introduction of WebNavigation interfaces and classes enhances web navigation framework

The addition of the WebNavigation interface, HistoryItem class, and WebNavigationOwner interfaces significantly expands the web navigation API. These additions enhance the navigation capabilities and allow for better management of navigation states and history.

decompose/api/decompose.klib.api (6)

117-138: Introduction of WebNavigation interface and HistoryItem class

The addition of the WebNavigation interface and its nested HistoryItem class provides essential functionality for web navigation, including management of navigation history and serialization support.


183-192: Addition of WebNavigationOwner interface and NoOp subinterface

The WebNavigationOwner interface and its NoOp subinterface define a standard for components that provide WebNavigation instances, with NoOp offering a default no-operation implementation. This structure enhances consistency and flexibility in components' navigation capabilities.


370-390: Serialization support added for Pages class

The inclusion of the $serializer nested class and the Companion object with a serializer method adds serialization capabilities to the Pages class, adhering to Kotlin serialization patterns.


Line range hint 498-512: Addition of web navigation functions for panels, pages, and stacks

The new functions childPanelsWebNavigation, childPagesWebNavigation, and childStackWebNavigation extend web navigation support to panels, pages, and stacks, enhancing the library's navigation capabilities and providing developers with more tools for web navigation management.


556-558: Introduction of DefaultWebHistoryController for web history management

The implementation of DefaultWebHistoryController provides concrete functionality for managing web history on supported platforms, aligning with the newly introduced web navigation features and improving state management.


558-558: Addition of withWebHistory function enhances root component navigation

The withWebHistory function facilitates web browser history navigation for root WebNavigationOwner components, improving integration with web platforms and simplifying navigation management.

decompose/src/commonMain/kotlin/com/arkivanov/decompose/router/webhistory/WebNavigationOwner.kt (1)

1-21: Addition of WebNavigationOwner interface and NoOp subinterface

The new WebNavigationOwner interface and its NoOp subinterface define a standard for components that own web navigation, facilitating consistency and allowing for simplified implementations in cases where full navigation functionality is not required.

sample/shared/shared/src/commonMain/kotlin/com/arkivanov/sample/shared/sharedtransitions/SharedTransitionsComponent.kt (1)

4-4: Update SharedTransitionsComponent to implement WebNavigationOwner

By extending WebNavigationOwner, SharedTransitionsComponent gains web navigation capabilities, enhancing its functionality and ensuring consistency with the new navigation framework.

Also applies to: 10-10

decompose/src/webMain/kotlin/com/arkivanov/decompose/router/stack/Utils.kt (1)

Line range hint 36-43: Consider thread-safety implications of value tracking.

The subscription handling is improved with proper cancellation support, but storing the old value in a var could be problematic in concurrent scenarios. Consider using atomic references or synchronization if this code might run in a multi-threaded context.

Let's check if this code might be accessed from multiple threads:

Consider this alternative implementation using AtomicReference if concurrent access is possible:

internal fun <T : Any> Value<T>.subscribe(observer: (new: T, old: T) -> Unit): Cancellation {
    val oldRef = AtomicReference(value)
    
    return subscribe { new ->
        val old = oldRef.getAndSet(new)
        observer(new, old)
    }
}
sample/shared/compose/build.gradle.kts (1)

52-58: LGTM! Clean separation of web and non-web targets

The bundle structure properly separates web-specific code from other targets, which is essential for the new web history API.

Let's verify no web-specific dependencies leak into non-web targets:

✅ Verification successful

The search results show that web-specific imports and dependencies are present in the common source set (commonMain), but they are properly isolated through interfaces and default implementations. The pattern observed is:

  1. Interfaces in common code define web navigation contracts (WebNavigationOwner)
  2. Default implementations handle web-specific logic
  3. Preview/NoOp implementations provide non-web alternatives

This is a standard way to handle web-specific features in Kotlin Multiplatform projects. The bundle structure in the build file correctly reflects this architecture by having nonWeb dependencies separate from js.

LGTM! Web dependencies are properly isolated through interfaces

The codebase follows a clean architecture pattern where web-specific functionality is abstracted behind interfaces in common code, with concrete implementations provided for each target platform. The bundle structure in build.gradle.kts correctly enforces this separation.

🏁 Scripts executed

The following scripts were executed for the analysis:

Script:

#!/bin/bash
# Search for web-specific imports in non-web source directories
rg -g '!**/js*/**' -g '!**/wasm*/**' \
   -e 'kotlinx\.browser' \
   -e 'org\.w3c\.dom' \
   -e 'WebNavigation' \
   -e 'WebHistoryNavigation'

Length of output: 50331

decompose/src/webTest/kotlin/com/arkivanov/decompose/router/webhistory/TestWebNavigation.kt (2)

84-89: LGTM! Clean and focused implementation.

The Owner class provides a clean implementation of WebNavigationOwner with helpful toString() override for debugging.


18-27: Consider thread safety for shared state.

While the initialization is thread-safe, the children map might need synchronization if accessed from multiple threads during testing.

Consider using a concurrent map implementation if concurrent testing is needed:

-    private var children = HashMap<Int, TestWebNavigation>()
+    private var children = Collections.synchronizedMap(HashMap<Int, TestWebNavigation>())

Copy link

@coderabbitai coderabbitai bot left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Actionable comments posted: 2

🧹 Outside diff range and nitpick comments (18)
sample/shared/shared/src/commonMain/kotlin/com/arkivanov/sample/shared/multipane/PreviewMultiPaneComponent.kt (1)

21-23: Consider adding documentation for WebNavigationOwner.NoOp implementation

While using NoOp is appropriate for a preview component, it would be helpful to add a KDoc comment explaining why this implementation was chosen and what it means for preview/testing scenarios.

Here's a suggested addition:

@OptIn(ExperimentalDecomposeApi::class)
+/**
+ * Preview implementation of MultiPaneComponent that implements WebNavigationOwner.NoOp
+ * to disable web navigation functionality in preview/testing scenarios.
+ */
class PreviewMultiPaneComponent(
    isMultiPane: Boolean = false,
) : MultiPaneComponent,
    ComponentContext by PreviewComponentContext,
    WebNavigationOwner.NoOp {
sample/shared/shared/src/commonMain/kotlin/com/arkivanov/sample/shared/sharedtransitions/gallery/DefaultGalleryComponent.kt (1)

6-7: LGTM! Good move towards identifier-based navigation.

The switch from passing Image objects to using integer IDs in the callback is a positive architectural change that aligns well with web navigation patterns. This makes it easier to serialize navigation state and construct deep links.

Consider documenting the ID format/constraints in the interface to ensure consistent usage across the codebase.

sample/shared/shared/src/commonMain/kotlin/com/arkivanov/sample/shared/sharedtransitions/DefaultSharedTransitionsComponent.kt (2)

32-38: Consider adding const values for magic numbers

The image list initialization uses hardcoded values that could be better expressed as named constants.

Consider this improvement:

+    private companion object {
+        private const val MAX_IMAGES = 100
+    }
-    private val images =
-        List(100) { index ->
+    private val images =
+        List(MAX_IMAGES) { index ->

53-67: Clarify the purpose of onBeforeNavigate callback

The onBeforeNavigate callback always returns false. Consider adding a comment explaining why this behavior is desired, or if this is intentional.

decompose/src/webTest/kotlin/com/arkivanov/decompose/router/webhistory/WebHistoryNavigationTest.kt (4)

6-6: Consider using backticks for test function names instead of suppressing 'TestFunctionName' warning

By using backticks in the test function names, you can write more readable and expressive names without suppressing the TestFunctionName warning. This aligns with Kotlin best practices and enhances code clarity.

You can remove the suppression and update the test function names as follows:

-@Suppress("TestFunctionName")
 class WebHistoryNavigationTest {

     @Test
-    fun WHEN_created_with_no_items_in_root_THEN_ISE_thrown() {
+    fun `WHEN created with no items in root THEN ISE thrown`() {
         // Test code
     }
     // ... other test functions ...
 }

7-768: Simplify test method names for better readability

The test method names are quite long and use underscores for separation, which can impact readability and make test reports harder to navigate. Consider simplifying the method names using descriptive phrases within backticks, which is a common practice in Kotlin for test functions.

For example:

@Test
fun `created with no items in root throws ISE`() {
    // Test code
}

This approach enhances readability and makes the test suite more maintainable.


165-176: Refactor repeated initialization code into helper functions

The initialization of TestWebNavigation with similar configurations is repeated across multiple tests. To adhere to the DRY (Don't Repeat Yourself) principle, consider extracting this repeated code into helper functions. This will reduce duplication and improve maintainability.

For example, you can create a helper function:

private fun createTestNavigation(
    rootConfigs: List<Int>,
    childConfigs: Map<Int, List<Int>> = emptyMap()
): TestWebNavigation {
    return TestWebNavigation(
        initialHistory = rootConfigs,
        childFactory = { config ->
            childConfigs[config]?.let { TestWebNavigation(initialHistory = it) }
        }
    )
}

Then, in your tests, you can use:

@Test
fun `test scenario`() {
    val nav = createTestNavigation(
        rootConfigs = listOf(1),
        childConfigs = mapOf(1 to listOf(11))
    )
    // Test code...
}

Also applies to: 184-195, 205-216, 225-236, 245-256


764-767: Enhance assertion messages for better test diagnostics

In the assertHistory function, consider adding custom messages to your assertions. This will provide more context when a test fails, making debugging easier.

For example:

private fun assertHistory(nav: TestWebNavigation, urls: List<String>, index: Int = urls.lastIndex) {
    history.assertStack(urls = urls, index = index)
    nav.assertHistory(
        expectedConfigs = urls.slice(0..index),
        message = "Expected navigation history to be ${urls.slice(0..index)}, but was ${nav.currentHistory()}"
    )
}
decompose/src/commonMain/kotlin/com/arkivanov/decompose/router/webhistory/WebNavigation.kt (1)

57-62: Consider making HistoryItem a data class

Converting HistoryItem to a data class provides automatically generated equals, hashCode, and toString methods, which can be helpful for debugging and state comparisons.

Apply this diff to make the change:

- class HistoryItem<out T>(
+ data class HistoryItem<out T>(
    val key: T,
    val path: String,
    val parameters: Map<String, String>,
    val child: WebNavigationOwner?,
)
sample/app-js/src/main/kotlin/com/arkivanov/sample/app/Main.kt (1)

32-32: Consider removing debug logging

The console log statement appears to be for debugging purposes and should be removed before merging.

-    console.log(root.stack.value)
sample/shared/shared/src/commonMain/kotlin/com/arkivanov/sample/shared/tabs/DefaultTabsComponent.kt (2)

100-101: Address TODO comment

The TODO comment suggests incomplete implementation or documentation. Please clarify the purpose of path and childUrl variables.

Would you like me to help document these variables or implement any missing functionality?


56-63: Consider documenting childSelector behavior

The childSelector implementation returns null for most child types except MultiPaneChild. Consider adding documentation explaining this design decision.

decompose/src/commonMain/kotlin/com/arkivanov/decompose/router/stack/StackWebNavigation.kt (2)

93-101: Consider adding error handling for invalid configurations

The toHistory method assumes all configurations can be properly converted to history items. Consider adding error handling for invalid configurations.

 private fun ChildStack<C, T>.toHistory(): List<HistoryItem<C>> =
     (if (enableHistory) items else items.takeLast(1)).map { child ->
+        try {
             HistoryItem(
                 path = pathMapper(child) ?: "",
                 parameters = parametersMapper(child) ?: emptyMap(),
                 key = child.configuration,
                 child = childSelector(child),
             )
+        } catch (e: Exception) {
+            throw IllegalStateException("Failed to convert configuration to history item", e)
+        }
     }

106-108: Consider adding validation for history list

The navigate method accepts any list of configurations without validation. Consider adding checks for empty lists or invalid configurations.

 override fun navigate(history: List<C>) {
+    require(history.isNotEmpty()) { "History list cannot be empty" }
+    history.forEach { config ->
+        require(config != null) { "Configuration cannot be null" }
+    }
     navigator.navigate { history }
 }
decompose/src/webMain/kotlin/com/arkivanov/decompose/router/webhistory/WebHistoryNavigation.kt (2)

85-92: Consider adding input validation for navigation nodes

While the implementation is functionally correct, it would benefit from additional validation to ensure data consistency.

Consider adding these validations:

 private fun <T : Any> WebNavigation<T>.navigate(nodes: List<SerializableNode>) {
+    require(nodes.isNotEmpty()) { "Navigation nodes list cannot be empty" }
     val items: List<T> = nodes.map { it.data.consumeRequired(serializer) }
     navigate(items)

     history.value.forEachIndexed { index, item ->
+        require(index < nodes.size) { "Navigation history size mismatch" }
         item.child?.webNavigation?.navigate(nodes[index].children)
     }
 }

317-360: Consider adding versioning support for serialized state

To support future schema changes and backward compatibility, consider adding version information to the serialized state.

Apply this diff to add versioning:

 @Serializable
 private data class SerializableNode(
+    val version: Int = 1,
     val data: SerializableContainer,
     val children: List<SerializableNode>,
 )

 @Serializable
 private class HistoryState(
+    val version: Int = 1,
     val index: Int,
     val nodesContainer: SerializableContainer,
 )
decompose/api/decompose.klib.api (1)

Line range hint 498-513: Consider documenting platform support for web navigation functions

The web navigation functions are platform-specific but this isn't immediately obvious from the API.

Consider adding KDoc comments to clearly indicate platform support:

+/**
+ * Enables web navigation support for panels.
+ * Available only on JS and Wasm platforms.
+ */
 final fun <A: kotlin/Any, B: kotlin/Any> childPanelsWebNavigation(...)
decompose/src/commonMain/kotlin/com/arkivanov/decompose/router/pages/PagesWebNavigation.kt (1)

76-88: Clarify variable naming in lambda expressions for better readability

In the history property, the lambda parameter is named pages, which is the same as the outer variable. This can be confusing. Consider renaming the lambda parameter to improve clarity.

Apply this diff to rename the lambda parameter:

override val history: Value<List<WebNavigation.HistoryItem<Pages<C>>>> =
-    pages.map { pages ->
+    pages.map { childPages ->
        listOf(
            WebNavigation.HistoryItem(
-                path = pathMapper(pages) ?: "",
-                parameters = parametersMapper(pages) ?: emptyMap(),
+                path = pathMapper(childPages) ?: "",
+                parameters = parametersMapper(childPages) ?: emptyMap(),
                key = Pages(
-                    items = pages.items.map { it.configuration },
-                    selectedIndex = pages.selectedIndex,
+                    items = childPages.items.map { it.configuration },
+                    selectedIndex = childPages.selectedIndex,
                ),
-               child = (pages.items.getOrNull(pages.selectedIndex) as? Child.Created)?.let(childSelector),
+               child = (childPages.items.getOrNull(childPages.selectedIndex) as? Child.Created)?.let(childSelector),
            )
        )
    }
📜 Review details

Configuration used: CodeRabbit UI
Review profile: CHILL

📥 Commits

Reviewing files that changed from the base of the PR and between fc78ab9 and 8cd8d9c.

📒 Files selected for processing (53)
  • decompose/api/android/decompose.api (6 hunks)
  • decompose/api/decompose.klib.api (6 hunks)
  • decompose/api/jvm/decompose.api (6 hunks)
  • decompose/src/commonMain/kotlin/com/arkivanov/decompose/router/pages/Pages.kt (1 hunks)
  • decompose/src/commonMain/kotlin/com/arkivanov/decompose/router/pages/PagesWebNavigation.kt (1 hunks)
  • decompose/src/commonMain/kotlin/com/arkivanov/decompose/router/panels/PanelsWebNavigation.kt (1 hunks)
  • decompose/src/commonMain/kotlin/com/arkivanov/decompose/router/stack/ChildStackFactory.kt (1 hunks)
  • decompose/src/commonMain/kotlin/com/arkivanov/decompose/router/stack/StackWebNavigation.kt (1 hunks)
  • decompose/src/commonMain/kotlin/com/arkivanov/decompose/router/webhistory/NoOpWebNavigation.kt (1 hunks)
  • decompose/src/commonMain/kotlin/com/arkivanov/decompose/router/webhistory/WebNavigation.kt (1 hunks)
  • decompose/src/commonMain/kotlin/com/arkivanov/decompose/router/webhistory/WebNavigationOwner.kt (1 hunks)
  • decompose/src/jsMain/kotlin/com/arkivanov/decompose/router/webhistory/WebHistoryNavigation.kt (1 hunks)
  • decompose/src/wasmJsMain/kotlin/com/arkivanov/decompose/router/webhistory/WebHistoryNavigation.kt (1 hunks)
  • decompose/src/webMain/kotlin/com/arkivanov/decompose/Utils.kt (1 hunks)
  • decompose/src/webMain/kotlin/com/arkivanov/decompose/router/stack/Utils.kt (2 hunks)
  • decompose/src/webMain/kotlin/com/arkivanov/decompose/router/webhistory/WebHistoryNavigation.kt (1 hunks)
  • decompose/src/webTest/kotlin/com/arkivanov/decompose/router/webhistory/TestWebNavigation.kt (1 hunks)
  • decompose/src/webTest/kotlin/com/arkivanov/decompose/router/webhistory/WebHistoryNavigationTest.kt (1 hunks)
  • sample/app-ios-compose/app-ios-compose/iOSApp.swift (1 hunks)
  • sample/app-ios/app-ios/app_iosApp.swift (1 hunks)
  • sample/app-js-compose/src/jsMain/kotlin/com/arkivanov/decompose/sample/app/Main.kt (1 hunks)
  • sample/app-js-compose/src/jsMain/resources/index.html (1 hunks)
  • sample/app-js/src/main/kotlin/com/arkivanov/sample/app/Main.kt (2 hunks)
  • sample/app-js/src/main/resources/index.html (1 hunks)
  • sample/shared/compose/build.gradle.kts (1 hunks)
  • sample/shared/compose/src/commonMain/kotlin/com/arkivanov/sample/shared/cards/CardsContent.kt (1 hunks)
  • sample/shared/compose/src/commonMain/kotlin/com/arkivanov/sample/shared/counters/CountersContent.kt (1 hunks)
  • sample/shared/compose/src/commonMain/kotlin/com/arkivanov/sample/shared/menu/MenuContent.kt (1 hunks)
  • sample/shared/compose/src/commonMain/kotlin/com/arkivanov/sample/shared/multipane/MultiPaneContent.kt (1 hunks)
  • sample/shared/compose/src/commonMain/kotlin/com/arkivanov/sample/shared/sharedtransitions/gallery/GalleryContent.kt (2 hunks)
  • sample/shared/compose/src/commonMain/kotlin/com/arkivanov/sample/shared/sharedtransitions/photo/PhotoContent.kt (2 hunks)
  • sample/shared/compose/src/commonMain/kotlin/com/arkivanov/sample/shared/utils/Utils.kt (1 hunks)
  • sample/shared/compose/src/jsMain/kotlin/com/arkivanov/sample/shared/utils/Utils.js.kt (1 hunks)
  • sample/shared/compose/src/nonWebMain/kotlin/com/arkivanov/sample/shared/utils/Utils.nonWeb.kt (1 hunks)
  • sample/shared/shared/src/commonMain/kotlin/com/arkivanov/sample/shared/Url.kt (1 hunks)
  • sample/shared/shared/src/commonMain/kotlin/com/arkivanov/sample/shared/Utils.kt (1 hunks)
  • sample/shared/shared/src/commonMain/kotlin/com/arkivanov/sample/shared/multipane/DefaultMultiPaneComponent.kt (4 hunks)
  • sample/shared/shared/src/commonMain/kotlin/com/arkivanov/sample/shared/multipane/MultiPaneComponent.kt (1 hunks)
  • sample/shared/shared/src/commonMain/kotlin/com/arkivanov/sample/shared/multipane/PreviewMultiPaneComponent.kt (2 hunks)
  • sample/shared/shared/src/commonMain/kotlin/com/arkivanov/sample/shared/pages/DefaultPagesComponent.kt (2 hunks)
  • sample/shared/shared/src/commonMain/kotlin/com/arkivanov/sample/shared/pages/PagesComponent.kt (1 hunks)
  • sample/shared/shared/src/commonMain/kotlin/com/arkivanov/sample/shared/root/DefaultRootComponent.kt (5 hunks)
  • sample/shared/shared/src/commonMain/kotlin/com/arkivanov/sample/shared/root/PreviewRootComponent.kt (1 hunks)
  • sample/shared/shared/src/commonMain/kotlin/com/arkivanov/sample/shared/root/RootComponent.kt (2 hunks)
  • sample/shared/shared/src/commonMain/kotlin/com/arkivanov/sample/shared/sharedtransitions/DefaultSharedTransitionsComponent.kt (3 hunks)
  • sample/shared/shared/src/commonMain/kotlin/com/arkivanov/sample/shared/sharedtransitions/SharedTransitionsComponent.kt (1 hunks)
  • sample/shared/shared/src/commonMain/kotlin/com/arkivanov/sample/shared/sharedtransitions/gallery/DefaultGalleryComponent.kt (1 hunks)
  • sample/shared/shared/src/commonMain/kotlin/com/arkivanov/sample/shared/sharedtransitions/photo/Image.kt (0 hunks)
  • sample/shared/shared/src/commonMain/kotlin/com/arkivanov/sample/shared/tabs/DefaultTabsComponent.kt (6 hunks)
  • sample/shared/shared/src/commonMain/kotlin/com/arkivanov/sample/shared/tabs/PreviewTabsComponent.kt (1 hunks)
  • sample/shared/shared/src/commonMain/kotlin/com/arkivanov/sample/shared/tabs/TabsComponent.kt (1 hunks)
  • sample/shared/shared/src/commonTest/kotlin/com/arkivanov/sample/shared/root/RootComponentIntegrationTest.kt (2 hunks)
  • sample/shared/shared/src/commonTest/kotlin/com/arkivanov/sample/shared/tabs/TabsComponentIntegrationTest.kt (1 hunks)
💤 Files with no reviewable changes (1)
  • sample/shared/shared/src/commonMain/kotlin/com/arkivanov/sample/shared/sharedtransitions/photo/Image.kt
🚧 Files skipped from review as they are similar to previous changes (32)
  • sample/app-js/src/main/resources/index.html
  • sample/app-js-compose/src/jsMain/resources/index.html
  • sample/shared/shared/src/commonMain/kotlin/com/arkivanov/sample/shared/tabs/TabsComponent.kt
  • sample/shared/shared/src/commonMain/kotlin/com/arkivanov/sample/shared/sharedtransitions/SharedTransitionsComponent.kt
  • sample/shared/compose/src/commonMain/kotlin/com/arkivanov/sample/shared/utils/Utils.kt
  • sample/shared/shared/src/commonMain/kotlin/com/arkivanov/sample/shared/Utils.kt
  • sample/shared/shared/src/commonMain/kotlin/com/arkivanov/sample/shared/pages/PagesComponent.kt
  • sample/shared/compose/src/commonMain/kotlin/com/arkivanov/sample/shared/cards/CardsContent.kt
  • decompose/src/commonMain/kotlin/com/arkivanov/decompose/router/pages/Pages.kt
  • sample/shared/compose/src/commonMain/kotlin/com/arkivanov/sample/shared/sharedtransitions/gallery/GalleryContent.kt
  • sample/shared/shared/src/commonMain/kotlin/com/arkivanov/sample/shared/multipane/MultiPaneComponent.kt
  • sample/shared/compose/src/jsMain/kotlin/com/arkivanov/sample/shared/utils/Utils.js.kt
  • sample/shared/compose/src/commonMain/kotlin/com/arkivanov/sample/shared/sharedtransitions/photo/PhotoContent.kt
  • sample/shared/compose/src/commonMain/kotlin/com/arkivanov/sample/shared/menu/MenuContent.kt
  • sample/shared/shared/src/commonMain/kotlin/com/arkivanov/sample/shared/root/PreviewRootComponent.kt
  • sample/shared/compose/src/commonMain/kotlin/com/arkivanov/sample/shared/multipane/MultiPaneContent.kt
  • sample/shared/shared/src/commonTest/kotlin/com/arkivanov/sample/shared/tabs/TabsComponentIntegrationTest.kt
  • decompose/src/commonMain/kotlin/com/arkivanov/decompose/router/stack/ChildStackFactory.kt
  • sample/shared/shared/src/commonMain/kotlin/com/arkivanov/sample/shared/root/RootComponent.kt
  • decompose/src/webMain/kotlin/com/arkivanov/decompose/router/stack/Utils.kt
  • sample/shared/compose/build.gradle.kts
  • decompose/src/jsMain/kotlin/com/arkivanov/decompose/router/webhistory/WebHistoryNavigation.kt
  • decompose/src/wasmJsMain/kotlin/com/arkivanov/decompose/router/webhistory/WebHistoryNavigation.kt
  • sample/shared/compose/src/nonWebMain/kotlin/com/arkivanov/sample/shared/utils/Utils.nonWeb.kt
  • sample/shared/shared/src/commonMain/kotlin/com/arkivanov/sample/shared/Url.kt
  • sample/shared/shared/src/commonTest/kotlin/com/arkivanov/sample/shared/root/RootComponentIntegrationTest.kt
  • sample/shared/compose/src/commonMain/kotlin/com/arkivanov/sample/shared/counters/CountersContent.kt
  • sample/shared/shared/src/commonMain/kotlin/com/arkivanov/sample/shared/pages/DefaultPagesComponent.kt
  • decompose/src/webTest/kotlin/com/arkivanov/decompose/router/webhistory/TestWebNavigation.kt
  • decompose/src/commonMain/kotlin/com/arkivanov/decompose/router/webhistory/WebNavigationOwner.kt
  • sample/shared/shared/src/commonMain/kotlin/com/arkivanov/sample/shared/tabs/PreviewTabsComponent.kt
  • decompose/src/commonMain/kotlin/com/arkivanov/decompose/router/webhistory/NoOpWebNavigation.kt
🧰 Additional context used
🪛 detekt (1.23.7)
decompose/src/webMain/kotlin/com/arkivanov/decompose/Utils.kt

[warning] 17-17: The caught exception is swallowed. The original exception could be lost.

(detekt.exceptions.SwallowedException)

🔇 Additional comments (36)
sample/shared/shared/src/commonMain/kotlin/com/arkivanov/sample/shared/multipane/PreviewMultiPaneComponent.kt (1)

Line range hint 1-35: Implementation looks good!

The integration of WebNavigationOwner.NoOp into this preview component is clean and appropriate. The implementation maintains its testing/preview purpose while adapting to the new web navigation architecture.

sample/shared/shared/src/commonMain/kotlin/com/arkivanov/sample/shared/sharedtransitions/gallery/DefaultGalleryComponent.kt (1)

11-12: LGTM! Verify ID usage consistency.

The implementation correctly forwards the ID to the callback.

Let's verify that other components are consistently using IDs for navigation:

✅ Verification successful

Let me gather more information to verify the ID usage consistency across the codebase.


ID usage is consistent across the codebase

The verification shows consistent ID handling in the image navigation flow:

  • GalleryComponent interface defines onImageClicked(index: Int)
  • DefaultGalleryComponent correctly implements this by forwarding to onImageSelected(id: Int)
  • The navigation handler in DefaultSharedTransitionsComponent properly uses the ID to push new Photo config: onImageSelected = { nav.pushNew(Config.Photo(id = it)) }
  • UI layer in GalleryContent correctly passes the index parameter: clickable { component.onImageClicked(index = index) }
🏁 Scripts executed

The following scripts were executed for the analysis:

Script:

#!/bin/bash
# Description: Check for consistent ID-based navigation pattern across components
# Look for other image selection handlers to ensure consistent ID usage

# Search for image selection related methods
ast-grep --pattern 'fun $FUNC_NAME($ARG: Int) {
  $$$
}'

# Search for image selection related callbacks
rg -A 2 'onImageSelected.*Int.*->' 

Length of output: 1060


Script:

#!/bin/bash
# Search for image-related method calls and handlers
rg -A 3 "onImageClicked|onImageSelected" 

# Search for any Int parameters in image-related functions
ast-grep --pattern 'fun $FUNC_NAME($ARG_NAME: Int) {
  onImage$_($ARG_NAME)
}'

# Search for Gallery component interface definition
rg -A 5 "interface.*Gallery.*Component"

Length of output: 4895

sample/shared/shared/src/commonMain/kotlin/com/arkivanov/sample/shared/sharedtransitions/DefaultSharedTransitionsComponent.kt (5)

27-28: LGTM: Constructor updated for deep link support

The addition of the deepLinkUrl parameter aligns with the PR's objective of supporting web navigation.


42-51: LGTM: Stack navigation properly configured

The stack navigation setup correctly integrates deep linking support while maintaining existing functionality.


102-104: Validate deep link parameters to prevent navigation issues

The image ID from the deep link URL should be validated against existing images.


117-117: LGTM: Config properly updated for serialization

The change from storing Image to id in Photo config supports proper serialization for web navigation.


83-83: ⚠️ Potential issue

Handle potential NoSuchElementException

Using first() to find an image by ID could throw an exception if the ID doesn't exist.

Consider using this safer approach:

-                        image = images.first { it.id == config.id },
+                        image = images.find { it.id == config.id }
+                            ?: throw IllegalArgumentException("Invalid image ID: ${config.id}"),

Likely invalid or redundant comment.

sample/shared/shared/src/commonMain/kotlin/com/arkivanov/sample/shared/multipane/DefaultMultiPaneComponent.kt (6)

12-12: LGTM: Required imports added for web navigation support

The new imports align with the PR objectives and are necessary for implementing the web history API functionality.

Also applies to: 17-17, 20-20


38-38: LGTM: Deep linking support added via constructor parameter

The nullable deepLinkUrl parameter is correctly added to support deep linking functionality.


Line range hint 46-58: LGTM: Panels initialization properly configured

The panels initialization is correctly implemented with the new deep linking support.


67-71: Ensure parametersMapper returns a non-null Map

When panels.details is null, the lambda returns null. Consider returning an empty map to prevent potential NullPointerException.


132-134: LGTM: Well-structured constant definition

The article ID key constant is properly defined and scoped within the companion object.


123-130: Verify article ID validation

While the code safely handles null and invalid article IDs, consider verifying that the article exists in the database before creating the Details instance.

decompose/src/webTest/kotlin/com/arkivanov/decompose/router/webhistory/WebHistoryNavigationTest.kt (1)

630-637: Verify the handling of onBeforeNavigate returning false

In tests where onBeforeNavigate returns false, ensure that the navigation prevention behaves as expected. Confirm that the browser history remains unchanged and that the navigation stack in TestWebNavigation reflects the correct state.

To automate this verification, you can run the following script:

Please ensure that the assertions in your tests accurately reflect the expected behavior when navigation is canceled.

Also applies to: 640-648, 651-661, 664-674

✅ Verification successful

Let me gather more information about the test assertions to verify the behavior when navigation is prevented.


Based on the test assertions and implementation, I can now provide the final response:

The handling of onBeforeNavigate=false is correctly verified

The test cases demonstrate proper verification of navigation prevention:

  • The navigation stack maintains the correct state as shown by assertHistory checks
  • Browser history operations are properly tested with various scenarios (forward/backward navigation)
  • The test assertions verify both the navigation stack and URL states through assertHistory(nav = nav, urls = ...) calls
🏁 Scripts executed

The following scripts were executed for the analysis:

Script:

#!/bin/bash
# Description: Verify that when onBeforeNavigate returns false, navigation is prevented.

# Search for instances where onBeforeNavigate is set to return false
rg 'onBeforeNavigate\s*=\s*\{\s*false\s*\}' -A 5

# Ensure that the navigation stack does not change after navigation is attempted
# This would require checking the state before and after, which can be done via test assertions

Length of output: 7581


Script:

#!/bin/bash
# Search for assertHistory function and its usage in the test file
rg -A 5 'fun assertHistory' decompose/src/webTest/kotlin/com/arkivanov/decompose/router/webhistory/WebHistoryNavigationTest.kt

# Search for test cases with onBeforeNavigate=false and their assertions
rg -B 5 'assertHistory.*nav.*urls' decompose/src/webTest/kotlin/com/arkivanov/decompose/router/webhistory/WebHistoryNavigationTest.kt

Length of output: 9734

sample/app-js-compose/src/jsMain/kotlin/com/arkivanov/decompose/sample/app/Main.kt (1)

27-33: Good implementation of withWebHistory

The integration of withWebHistory in the main function is correctly done. Passing stateKeeper and converting deepLink to Url enhances deep link handling effectively.

decompose/src/commonMain/kotlin/com/arkivanov/decompose/router/panels/PanelsWebNavigation.kt (1)

71-116: Ensure correct serializers are provided

In PanelsWebNavigation, the serializers mainSerializer, detailsSerializer, and extraSerializer are crucial for proper serialization and deserialization of configurations. Verify that the correct serializers are supplied to prevent runtime exceptions or data inconsistencies.

sample/shared/shared/src/commonMain/kotlin/com/arkivanov/sample/shared/root/DefaultRootComponent.kt (2)

74-78: Deep link handling in Tabs component is correctly implemented

Passing deepLinkUrl to DefaultTabsComponent ensures proper handling of deep links within the Tabs component.


126-135: Efficient initial stack determination with getInitialStack

The getInitialStack function effectively determines the initial navigation stack based on the deepLinkUrl, enhancing deep link support throughout the application.

decompose/api/jvm/decompose.api (5)

Line range hint 170-197: Approval: Enhancements to Pages Class

The addition of the Companion object and the Pages$$serializer class to the Pages class improves serialization capabilities, aligning with the introduced web navigation support.


233-237: Approval: Introduction of PagesWebNavigationKt

The new PagesWebNavigationKt class provides necessary functionalities for web navigation of pages, enhancing the API's navigation features.


358-362: Approval: Addition of PanelsWebNavigationKt

The inclusion of PanelsWebNavigationKt adds web navigation support for panels, which is consistent with the PR objectives to enhance nested navigation.


481-485: Approval: Enhancement with StackWebNavigationKt

The introduction of StackWebNavigationKt extends stack navigation capabilities for web interfaces, contributing to the comprehensive navigation enhancements.


501-507: Approval: Introduction of WebNavigation Interface

The addition of the WebNavigation interface is a significant enhancement for managing web navigation history and actions, aligning with the goals of supporting nested navigation models.

decompose/api/android/decompose.api (5)

Line range hint 190-218: Approval: Serialization Enhancements in Pages Class

The addition of the Companion object and the Pages$$serializer class to the Pages class enhances serialization support, which is essential for the new web navigation functionalities.


253-257: Approval: Addition of PagesWebNavigationKt

The inclusion of PagesWebNavigationKt provides web navigation capabilities for pages, aligning with the PR's objective to add nested navigation support.


378-382: Approval: Introduction of PanelsWebNavigationKt

The new PanelsWebNavigationKt class offers web navigation functionalities for panels, contributing to the expanded navigation features.


501-505: Approval: Enhancement with StackWebNavigationKt

The addition of StackWebNavigationKt enhances stack navigation for web interfaces, supporting the comprehensive navigation enhancements in the API.


521-527: Approval: Introduction of WebNavigation Interface

The new WebNavigation interface and related classes expand the library's web navigation capabilities, which is crucial for the nested navigation support added in this PR.

decompose/src/webMain/kotlin/com/arkivanov/decompose/Utils.kt (1)

14-20: ⚠️ Potential issue

Catch Specific Exceptions Instead of General Exception

Catching all exceptions can unintentionally suppress critical issues. It's advisable to catch more specific exceptions, such as SerializationException, to avoid masking underlying problems and to handle different exception types appropriately.

Apply this diff to catch only SerializationException:

internal fun JsonString.decodeContainer(): SerializableContainer? =
    try {
        Json.decodeFromString(SerializableContainer.serializer(), value)
-   } catch (e: Exception) {
+   } catch (e: SerializationException) {
        null
+   } catch (e: Exception) {
+       // Optionally log or handle other exceptions
+       throw e
    }

This change ensures that serialization errors are handled gracefully while allowing other exceptions to propagate, preserving critical error information.

🧰 Tools
🪛 detekt (1.23.7)

[warning] 17-17: The caught exception is swallowed. The original exception could be lost.

(detekt.exceptions.SwallowedException)

sample/app-ios/app-ios/app_iosApp.swift (1)

34-34: Approval: Update to deepLinkUrl Initialization

Replacing deepLink with deepLinkUrl: nil aligns with the updated deep link handling approach. The change is appropriate and maintains the intended functionality.

sample/app-ios-compose/app-ios-compose/iOSApp.swift (1)

29-29: Approval: Change to deepLinkUrl Parameter

Updating the initialization from deepLink to deepLinkUrl: nil reflects the new method of handling deep links within the application and aligns with the changes made in the associated components.

sample/app-js/src/main/kotlin/com/arkivanov/sample/app/Main.kt (1)

24-30: LGTM: Proper initialization with web history support

The implementation correctly uses the new withWebHistory function to initialize the root component with state keeper and deep link support.

sample/shared/shared/src/commonMain/kotlin/com/arkivanov/sample/shared/tabs/DefaultTabsComponent.kt (1)

48-64: Verify history disabled flag

The webNavigation implementation disables history tracking (enableHistory = false). This might affect the navigation experience if history tracking is required for tabs.

✅ Verification successful

Let me gather more context about the navigation implementation in the codebase.


Let me check one more thing about the tabs implementation to understand the navigation pattern better.


Based on the gathered information, I can now provide the final response:

Disabling history in tabs navigation is intentional and correct

The enableHistory = false setting in DefaultTabsComponent is appropriate because:

  • This is the only component in the codebase that explicitly disables history
  • The component implements tab-based navigation where each tab maintains its own state
  • Other navigation components (root, pages, multipane) handle history differently based on their specific navigation patterns
  • The tabs component acts as a container where the actual navigation history is managed by the active tab's child component
🏁 Scripts executed

The following scripts were executed for the analysis:

Script:

#!/bin/bash
# Check if other tab-based components also disable history
rg -l "enableHistory\s*=\s*false" --type kotlin

Length of output: 149


Script:

#!/bin/bash
# Check for other web navigation implementations and their history settings
rg "webNavigation.*=" --type kotlin -A 5

# Also check for any documentation or comments about web navigation history
rg -i "web.*navigation.*history" --type kotlin

Length of output: 17238


Script:

#!/bin/bash
# Check the TabsComponent interface and implementation for navigation-related code
rg -A 10 "interface TabsComponent" --type kotlin
rg -A 10 "class DefaultTabsComponent" --type kotlin

Length of output: 2964

decompose/src/webMain/kotlin/com/arkivanov/decompose/router/webhistory/WebHistoryNavigation.kt (2)

79-83: LGTM: Recursive navigation check is well implemented

The recursive navigation check is implemented correctly, handling both parent and child navigation permissions.


14-77: ⚠️ Potential issue

Review state management and error handling in enableWebHistory function

Several concerns need to be addressed:

  1. The state deserialization lacks error handling which could lead to runtime exceptions
  2. The function uses a boolean flag for synchronization which might be insufficient for complex navigation scenarios
  3. The recursive navigation handling could potentially cause stack overflow for deeply nested structures

Apply this diff to add error handling and improve state management:

 fun onPopState(state: String?) {
-    val deserializedState = state?.deserializeState() ?: return
+    val deserializedState = try {
+        state?.deserializeState()
+    } catch (e: Exception) {
+        // Log the error for debugging
+        console.error("Failed to deserialize navigation state", e)
+        return
+    } ?: return

     if (navigation.onBeforeNavigateRecursive()) {
         isEnabled = false
         navigation.navigate(deserializedState.nodesContainer.consumeNodes())
         isEnabled = true
     } else {
         val delta = navigation.history().lastIndex - deserializedState.index
         if (delta != 0) {
             browserHistory.setOnPopStateListener { browserHistory.setOnPopStateListener(::onPopState) }
             browserHistory.go(delta)
         }
     }
 }

Consider implementing a more robust synchronization mechanism using a state machine or mutex to handle complex navigation scenarios.

Let's verify the navigation depth handling:

decompose/api/decompose.klib.api (1)

117-138: LGTM: Well-designed API for web navigation

The WebNavigation and WebNavigationOwner interfaces are well-designed with clear separation of concerns and extensibility support.

Also applies to: 183-191

Copy link

@coderabbitai coderabbitai bot left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Actionable comments posted: 0

🧹 Outside diff range and nitpick comments (18)
sample/shared/shared/src/commonMain/kotlin/com/arkivanov/sample/shared/sharedtransitions/DefaultSharedTransitionsComponent.kt (2)

32-38: Consider lazy initialization for the images list

The images list is eagerly initialized with 100 items during component construction. Consider using lazy initialization to defer creation until first use, especially since not all images may be needed immediately.

-    private val images =
+    private val images by lazy {
         List(100) { index ->
             Image(
                 id = index,
                 resourceId = ImageResourceId.entries[index % ImageResourceId.entries.size],
             )
-        }
+        }

53-67: Consider implementing meaningful navigation control

The onBeforeNavigate callback always returns false, which means navigation is never prevented. Consider implementing actual navigation validation logic if there are cases where navigation should be restricted (e.g., unsaved changes).

-            onBeforeNavigate = { false },
+            onBeforeNavigate = { navigationEvent ->
+                // Add logic to determine if navigation should be prevented
+                // Example: check for unsaved changes
+                false
+            },
sample/shared/shared/src/commonMain/kotlin/com/arkivanov/sample/shared/pages/DefaultPagesComponent.kt (2)

29-35: Consider extracting initialization logic

The pages initialization logic could be moved to a separate private function for better maintainability and testing.

Consider refactoring to:

-    private val _pages: Value<ChildPages<ImageResourceId, KittenComponent>> =
-        childPages(
-            source = nav,
-            serializer = ImageResourceId.serializer(),
-            initialPages = { Pages(items = ImageResourceId.entries, selectedIndex = getInitialPageIndex(deepLinkUrl)) },
-            childFactory = { imageType, ctx -> DefaultKittenComponent(ctx, imageType) },
-        )
+    private val _pages: Value<ChildPages<ImageResourceId, KittenComponent>> = createPages(deepLinkUrl)
+
+    private fun createPages(deepLinkUrl: Url?): Value<ChildPages<ImageResourceId, KittenComponent>> =
+        childPages(
+            source = nav,
+            serializer = ImageResourceId.serializer(),
+            initialPages = { createInitialPages(deepLinkUrl) },
+            childFactory = { imageType, ctx -> DefaultKittenComponent(ctx, imageType) },
+        )
+
+    private fun createInitialPages(deepLinkUrl: Url?) = 
+        Pages(
+            items = ImageResourceId.entries,
+            selectedIndex = getInitialPageIndex(deepLinkUrl)
+        )

39-46: Document experimental API usage

The code uses @ExperimentalDecomposeApi which indicates potential API instability. Consider adding documentation comments about this experimental status and any migration plans.

Add documentation:

+    /**
+     * Web navigation implementation using experimental Decompose API.
+     * Note: This API is experimental and may change in future releases.
+     * Track issue #111 for API stabilization updates.
+     */
     @OptIn(ExperimentalDecomposeApi::class)
     override val webNavigation: WebNavigation<*> =
         childPagesWebNavigation(
decompose/src/commonMain/kotlin/com/arkivanov/decompose/router/webhistory/WebNavigation.kt (1)

57-62: Consider making HistoryItem a data class

Since HistoryItem is primarily used for holding data, converting it into a data class will automatically provide equals, hashCode, toString, and copy methods. This aligns with Kotlin best practices for classes that are intended to be simple data carriers.

sample/shared/shared/src/commonMain/kotlin/com/arkivanov/sample/shared/tabs/DefaultTabsComponent.kt (1)

100-100: Address the TODO comment regarding childUrl handling

The TODO comment indicates that handling of childUrl in the path is pending. Implementing this will ensure proper deep link processing and navigation.

Would you like assistance in implementing the handling of childUrl within the getInitialConfig method?

sample/shared/shared/src/commonMain/kotlin/com/arkivanov/sample/shared/sharedtransitions/gallery/DefaultGalleryComponent.kt (1)

7-7: Consider passing the Image object instead of only the id to onImageSelected

Changing the onImageSelected callback to accept id: Int instead of image: Image reduces the information available in the callback. If additional properties of the Image are needed for processing, it might be beneficial to pass the entire Image object to retain flexibility.

decompose/src/webMain/kotlin/com/arkivanov/decompose/Utils.kt (1)

6-9: Consider documenting the Json configuration

The Json configuration allows structured map keys, but the reason for this setting isn't immediately clear. Consider adding a comment explaining why this configuration is necessary.

sample/app-js/src/main/kotlin/com/arkivanov/sample/app/Main.kt (1)

32-32: Consider using more descriptive logging

The console log message could be more informative about what's being logged.

-    console.log(root.stack.value)
+    console.log("Root stack state:", root.stack.value)
decompose/src/commonMain/kotlin/com/arkivanov/decompose/router/stack/StackWebNavigation.kt (2)

12-57: Consider adding example usage in the documentation.

The documentation is comprehensive but would benefit from a code example demonstrating typical usage patterns, especially for the pathMapper and parametersMapper functions.


93-101: Consider optimizing the history mapping logic.

The toHistory function could be more efficient by:

  1. Combining the map operation with takeLast
  2. Using a shared empty map instance
+private val EMPTY_MAP = emptyMap<String, String>()

 private fun ChildStack<C, T>.toHistory(): List<HistoryItem<C>> =
-    (if (enableHistory) items else items.takeLast(1)).map { child ->
+    items.let { if (enableHistory) it else it.takeLast(1) }.map { child ->
         HistoryItem(
             path = pathMapper(child) ?: "",
-            parameters = parametersMapper(child) ?: emptyMap(),
+            parameters = parametersMapper(child) ?: EMPTY_MAP,
             key = child.configuration,
             child = childSelector(child),
         )
     }
sample/shared/shared/src/commonMain/kotlin/com/arkivanov/sample/shared/multipane/DefaultMultiPaneComponent.kt (1)

Line range hint 46-57: Consider extracting panel configuration to a separate function.

The panel configuration is complex and would benefit from being extracted into a separate private function for better readability and maintainability.

+    private fun createPanelsConfiguration() =
+        childPanels(
+            source = navigation,
+            initialPanels = { getInitialPanels(deepLinkUrl) },
+            serializers = Triple(Unit.serializer(), Details.serializer(), Extra.serializer()),
+            onStateChanged = { newState, _ -> _navState.onNext(newState) },
+            handleBackButton = true,
+            mainFactory = { _, ctx -> listComponent(ctx) },
+            detailsFactory = ::detailsComponent,
+            extraFactory = ::authorComponent,
+        )

-    private val _panels =
-        childPanels(
-            source = navigation,
-            initialPanels = { getInitialPanels(deepLinkUrl) },
-            serializers = Triple(Unit.serializer(), Details.serializer(), Extra.serializer()),
-            onStateChanged = { newState, _ -> _navState.onNext(newState) },
-            handleBackButton = true,
-            mainFactory = { _, ctx -> listComponent(ctx) },
-            detailsFactory = ::detailsComponent,
-            extraFactory = ::authorComponent,
-        )
+    private val _panels = createPanelsConfiguration()
sample/shared/shared/src/commonMain/kotlin/com/arkivanov/sample/shared/root/DefaultRootComponent.kt (2)

126-135: Consider making the path handling more type-safe.

The getInitialStack function uses string-based path handling which could be error-prone. Consider using a sealed class or enum for path segments to make it more type-safe.

+private sealed class PathSegment(val value: String) {
+    object DynamicFeatures : PathSegment("dynamic-features")
+    object CustomNavigation : PathSegment("custom-navigation")
+    object Pages : PathSegment("pages")
+    object SharedTransitions : PathSegment("shared-transitions")
+
+    companion object {
+        fun fromString(value: String): PathSegment? =
+            when (value) {
+                DynamicFeatures.value -> DynamicFeatures
+                CustomNavigation.value -> CustomNavigation
+                Pages.value -> Pages
+                SharedTransitions.value -> SharedTransitions
+                else -> null
+            }
+    }
+}

 private fun getInitialStack(deepLinkUrl: Url?): List<Config> {
     val (path, childUrl) = deepLinkUrl?.consumePathSegment() ?: return listOf(Config.Tabs())
 
-    return when (path) {
-        pathSegmentOf<Config.DynamicFeatures>() -> listOf(Config.Tabs(), Config.DynamicFeatures)
-        pathSegmentOf<Config.CustomNavigation>() -> listOf(Config.Tabs(), Config.CustomNavigation)
-        pathSegmentOf<Config.Pages>() -> listOf(Config.Tabs(), Config.Pages(deepLinkUrl = childUrl))
-        pathSegmentOf<Config.SharedTransitions>() -> listOf(Config.Tabs(), Config.SharedTransitions(deepLinkUrl = childUrl))
-        else -> listOf(Config.Tabs(deepLinkUrl = childUrl))
+    return when (PathSegment.fromString(path)) {
+        is PathSegment.DynamicFeatures -> listOf(Config.Tabs(), Config.DynamicFeatures)
+        is PathSegment.CustomNavigation -> listOf(Config.Tabs(), Config.CustomNavigation)
+        is PathSegment.Pages -> listOf(Config.Tabs(), Config.Pages(deepLinkUrl = childUrl))
+        is PathSegment.SharedTransitions -> listOf(Config.Tabs(), Config.SharedTransitions(deepLinkUrl = childUrl))
+        null -> listOf(Config.Tabs(deepLinkUrl = childUrl))
     }
 }

57-65: Consider adding error handling for invalid child types.

The childSelector function assumes all child types are handled in the when expression. Consider adding an else branch or making it more explicit that all cases are covered.

 childSelector = {
     when (val child = it.instance) {
         is CustomNavigationChild -> null
         is TabsChild -> child.component
         is DynamicFeaturesChild -> null
         is PagesChild -> child.component
         is SharedTransitionsChild -> child.component
+        // All cases are covered as Child is a sealed interface
     }
 },
decompose/src/webMain/kotlin/com/arkivanov/decompose/router/webhistory/WebHistoryNavigation.kt (2)

15-21: Add validation for browser history state initialization

The browser history state initialization should validate that the navigation has a valid history before replacing the state.

 internal fun <T : Any> enableWebHistory(navigation: WebNavigation<T>, browserHistory: BrowserHistory) {
+    val initialHistory = navigation.nodeHistory()
+    require(initialHistory.isNotEmpty()) { "Navigation history cannot be empty" }
+
     if (browserHistory.state == null) {
-        browserHistory.replaceState(navigation.nodeHistory())
+        browserHistory.replaceState(initialHistory)
     }

40-77: Document the complex navigation logic

The navigation subscription logic is complex and would benefit from documentation explaining the different navigation scenarios.

+    // Subscribe to navigation changes and handle different scenarios:
+    // 1. Push: Add new states to history
+    // 2. Pop: Remove states from history
+    // 3. Rewrite: Replace existing history
+    // 4. URL Update: Update URL without changing history
     navigation.subscribe(
         isEnabled = { isEnabled },
         onPush = { history ->
decompose/api/decompose.klib.api (2)

117-138: Well-designed WebNavigation interface with comprehensive functionality!

The interface provides a robust foundation for web navigation with:

  • History tracking via Value<List>
  • Type-safe serialization support
  • Navigation control methods
  • Pre-navigation validation
  • Nested navigation support through WebNavigationOwner in HistoryItem

The design allows for flexible implementation of different navigation patterns while maintaining type safety.

Consider documenting the following aspects for implementers:

  1. Thread-safety requirements for history updates
  2. Lifecycle management best practices
  3. Guidelines for handling deep linking

183-191: Clean interface design with proper separation of concerns!

The WebNavigationOwner interface effectively:

  • Separates navigation ownership from navigation implementation
  • Provides a NoOp implementation for graceful degradation
  • Uses wildcard type parameter for flexibility

This design allows for easy composition and testing.

Consider adding:

  1. Factory methods for common WebNavigationOwner implementations
  2. Extension functions for common navigation patterns
📜 Review details

Configuration used: CodeRabbit UI
Review profile: CHILL

📥 Commits

Reviewing files that changed from the base of the PR and between 8cd8d9c and 0137389.

📒 Files selected for processing (54)
  • decompose/api/android/decompose.api (6 hunks)
  • decompose/api/decompose.klib.api (6 hunks)
  • decompose/api/jvm/decompose.api (6 hunks)
  • decompose/src/commonMain/kotlin/com/arkivanov/decompose/router/pages/Pages.kt (1 hunks)
  • decompose/src/commonMain/kotlin/com/arkivanov/decompose/router/pages/PagesWebNavigation.kt (1 hunks)
  • decompose/src/commonMain/kotlin/com/arkivanov/decompose/router/panels/PanelsWebNavigation.kt (1 hunks)
  • decompose/src/commonMain/kotlin/com/arkivanov/decompose/router/stack/ChildStackFactory.kt (1 hunks)
  • decompose/src/commonMain/kotlin/com/arkivanov/decompose/router/stack/StackWebNavigation.kt (1 hunks)
  • decompose/src/commonMain/kotlin/com/arkivanov/decompose/router/webhistory/NoOpWebNavigation.kt (1 hunks)
  • decompose/src/commonMain/kotlin/com/arkivanov/decompose/router/webhistory/WebNavigation.kt (1 hunks)
  • decompose/src/commonMain/kotlin/com/arkivanov/decompose/router/webhistory/WebNavigationOwner.kt (1 hunks)
  • decompose/src/jsMain/kotlin/com/arkivanov/decompose/router/webhistory/WebHistoryNavigation.kt (1 hunks)
  • decompose/src/wasmJsMain/kotlin/com/arkivanov/decompose/router/webhistory/WebHistoryNavigation.kt (1 hunks)
  • decompose/src/webMain/kotlin/com/arkivanov/decompose/Utils.kt (1 hunks)
  • decompose/src/webMain/kotlin/com/arkivanov/decompose/router/stack/Utils.kt (2 hunks)
  • decompose/src/webMain/kotlin/com/arkivanov/decompose/router/webhistory/WebHistoryNavigation.kt (1 hunks)
  • decompose/src/webTest/kotlin/com/arkivanov/decompose/UrlEncodeTest.kt (1 hunks)
  • decompose/src/webTest/kotlin/com/arkivanov/decompose/router/webhistory/TestWebNavigation.kt (1 hunks)
  • decompose/src/webTest/kotlin/com/arkivanov/decompose/router/webhistory/WebHistoryNavigationTest.kt (1 hunks)
  • sample/app-ios-compose/app-ios-compose/iOSApp.swift (1 hunks)
  • sample/app-ios/app-ios/app_iosApp.swift (1 hunks)
  • sample/app-js-compose/src/jsMain/kotlin/com/arkivanov/decompose/sample/app/Main.kt (1 hunks)
  • sample/app-js-compose/src/jsMain/resources/index.html (1 hunks)
  • sample/app-js/src/main/kotlin/com/arkivanov/sample/app/Main.kt (2 hunks)
  • sample/app-js/src/main/resources/index.html (1 hunks)
  • sample/shared/compose/build.gradle.kts (1 hunks)
  • sample/shared/compose/src/commonMain/kotlin/com/arkivanov/sample/shared/cards/CardsContent.kt (1 hunks)
  • sample/shared/compose/src/commonMain/kotlin/com/arkivanov/sample/shared/counters/CountersContent.kt (1 hunks)
  • sample/shared/compose/src/commonMain/kotlin/com/arkivanov/sample/shared/menu/MenuContent.kt (1 hunks)
  • sample/shared/compose/src/commonMain/kotlin/com/arkivanov/sample/shared/multipane/MultiPaneContent.kt (1 hunks)
  • sample/shared/compose/src/commonMain/kotlin/com/arkivanov/sample/shared/sharedtransitions/gallery/GalleryContent.kt (2 hunks)
  • sample/shared/compose/src/commonMain/kotlin/com/arkivanov/sample/shared/sharedtransitions/photo/PhotoContent.kt (2 hunks)
  • sample/shared/compose/src/commonMain/kotlin/com/arkivanov/sample/shared/utils/Utils.kt (1 hunks)
  • sample/shared/compose/src/jsMain/kotlin/com/arkivanov/sample/shared/utils/Utils.js.kt (1 hunks)
  • sample/shared/compose/src/nonWebMain/kotlin/com/arkivanov/sample/shared/utils/Utils.nonWeb.kt (1 hunks)
  • sample/shared/shared/src/commonMain/kotlin/com/arkivanov/sample/shared/Url.kt (1 hunks)
  • sample/shared/shared/src/commonMain/kotlin/com/arkivanov/sample/shared/Utils.kt (1 hunks)
  • sample/shared/shared/src/commonMain/kotlin/com/arkivanov/sample/shared/multipane/DefaultMultiPaneComponent.kt (4 hunks)
  • sample/shared/shared/src/commonMain/kotlin/com/arkivanov/sample/shared/multipane/MultiPaneComponent.kt (1 hunks)
  • sample/shared/shared/src/commonMain/kotlin/com/arkivanov/sample/shared/multipane/PreviewMultiPaneComponent.kt (2 hunks)
  • sample/shared/shared/src/commonMain/kotlin/com/arkivanov/sample/shared/pages/DefaultPagesComponent.kt (2 hunks)
  • sample/shared/shared/src/commonMain/kotlin/com/arkivanov/sample/shared/pages/PagesComponent.kt (1 hunks)
  • sample/shared/shared/src/commonMain/kotlin/com/arkivanov/sample/shared/root/DefaultRootComponent.kt (5 hunks)
  • sample/shared/shared/src/commonMain/kotlin/com/arkivanov/sample/shared/root/PreviewRootComponent.kt (1 hunks)
  • sample/shared/shared/src/commonMain/kotlin/com/arkivanov/sample/shared/root/RootComponent.kt (2 hunks)
  • sample/shared/shared/src/commonMain/kotlin/com/arkivanov/sample/shared/sharedtransitions/DefaultSharedTransitionsComponent.kt (3 hunks)
  • sample/shared/shared/src/commonMain/kotlin/com/arkivanov/sample/shared/sharedtransitions/SharedTransitionsComponent.kt (1 hunks)
  • sample/shared/shared/src/commonMain/kotlin/com/arkivanov/sample/shared/sharedtransitions/gallery/DefaultGalleryComponent.kt (1 hunks)
  • sample/shared/shared/src/commonMain/kotlin/com/arkivanov/sample/shared/sharedtransitions/photo/Image.kt (0 hunks)
  • sample/shared/shared/src/commonMain/kotlin/com/arkivanov/sample/shared/tabs/DefaultTabsComponent.kt (6 hunks)
  • sample/shared/shared/src/commonMain/kotlin/com/arkivanov/sample/shared/tabs/PreviewTabsComponent.kt (1 hunks)
  • sample/shared/shared/src/commonMain/kotlin/com/arkivanov/sample/shared/tabs/TabsComponent.kt (1 hunks)
  • sample/shared/shared/src/commonTest/kotlin/com/arkivanov/sample/shared/root/RootComponentIntegrationTest.kt (2 hunks)
  • sample/shared/shared/src/commonTest/kotlin/com/arkivanov/sample/shared/tabs/TabsComponentIntegrationTest.kt (1 hunks)
💤 Files with no reviewable changes (1)
  • sample/shared/shared/src/commonMain/kotlin/com/arkivanov/sample/shared/sharedtransitions/photo/Image.kt
🚧 Files skipped from review as they are similar to previous changes (36)
  • sample/app-js/src/main/resources/index.html
  • sample/shared/compose/src/commonMain/kotlin/com/arkivanov/sample/shared/utils/Utils.kt
  • decompose/src/commonMain/kotlin/com/arkivanov/decompose/router/pages/Pages.kt
  • sample/shared/shared/src/commonMain/kotlin/com/arkivanov/sample/shared/tabs/TabsComponent.kt
  • sample/shared/shared/src/commonMain/kotlin/com/arkivanov/sample/shared/multipane/MultiPaneComponent.kt
  • sample/shared/compose/src/commonMain/kotlin/com/arkivanov/sample/shared/multipane/MultiPaneContent.kt
  • sample/app-js-compose/src/jsMain/resources/index.html
  • sample/shared/compose/src/commonMain/kotlin/com/arkivanov/sample/shared/sharedtransitions/gallery/GalleryContent.kt
  • sample/app-ios/app-ios/app_iosApp.swift
  • sample/shared/shared/src/commonMain/kotlin/com/arkivanov/sample/shared/Utils.kt
  • sample/shared/compose/src/commonMain/kotlin/com/arkivanov/sample/shared/cards/CardsContent.kt
  • sample/shared/shared/src/commonMain/kotlin/com/arkivanov/sample/shared/pages/PagesComponent.kt
  • sample/shared/compose/src/commonMain/kotlin/com/arkivanov/sample/shared/sharedtransitions/photo/PhotoContent.kt
  • sample/shared/shared/src/commonMain/kotlin/com/arkivanov/sample/shared/sharedtransitions/SharedTransitionsComponent.kt
  • sample/shared/shared/src/commonMain/kotlin/com/arkivanov/sample/shared/root/RootComponent.kt
  • sample/shared/compose/src/jsMain/kotlin/com/arkivanov/sample/shared/utils/Utils.js.kt
  • sample/shared/compose/src/nonWebMain/kotlin/com/arkivanov/sample/shared/utils/Utils.nonWeb.kt
  • sample/shared/compose/build.gradle.kts
  • sample/shared/compose/src/commonMain/kotlin/com/arkivanov/sample/shared/menu/MenuContent.kt
  • sample/shared/shared/src/commonTest/kotlin/com/arkivanov/sample/shared/tabs/TabsComponentIntegrationTest.kt
  • decompose/src/webMain/kotlin/com/arkivanov/decompose/router/stack/Utils.kt
  • sample/shared/compose/src/commonMain/kotlin/com/arkivanov/sample/shared/counters/CountersContent.kt
  • sample/shared/shared/src/commonMain/kotlin/com/arkivanov/sample/shared/tabs/PreviewTabsComponent.kt
  • sample/shared/shared/src/commonMain/kotlin/com/arkivanov/sample/shared/multipane/PreviewMultiPaneComponent.kt
  • decompose/src/jsMain/kotlin/com/arkivanov/decompose/router/webhistory/WebHistoryNavigation.kt
  • sample/shared/shared/src/commonTest/kotlin/com/arkivanov/sample/shared/root/RootComponentIntegrationTest.kt
  • decompose/src/wasmJsMain/kotlin/com/arkivanov/decompose/router/webhistory/WebHistoryNavigation.kt
  • decompose/src/commonMain/kotlin/com/arkivanov/decompose/router/webhistory/WebNavigationOwner.kt
  • sample/shared/shared/src/commonMain/kotlin/com/arkivanov/sample/shared/root/PreviewRootComponent.kt
  • decompose/src/commonMain/kotlin/com/arkivanov/decompose/router/stack/ChildStackFactory.kt
  • sample/shared/shared/src/commonMain/kotlin/com/arkivanov/sample/shared/Url.kt
  • decompose/src/commonMain/kotlin/com/arkivanov/decompose/router/panels/PanelsWebNavigation.kt
  • decompose/src/commonMain/kotlin/com/arkivanov/decompose/router/webhistory/NoOpWebNavigation.kt
  • sample/app-ios-compose/app-ios-compose/iOSApp.swift
  • decompose/src/webTest/kotlin/com/arkivanov/decompose/router/webhistory/TestWebNavigation.kt
  • decompose/src/webTest/kotlin/com/arkivanov/decompose/router/webhistory/WebHistoryNavigationTest.kt
🧰 Additional context used
🪛 detekt (1.23.7)
decompose/src/webMain/kotlin/com/arkivanov/decompose/Utils.kt

[warning] 17-17: The caught exception is swallowed. The original exception could be lost.

(detekt.exceptions.SwallowedException)

🔇 Additional comments (26)
sample/shared/shared/src/commonMain/kotlin/com/arkivanov/sample/shared/sharedtransitions/DefaultSharedTransitionsComponent.kt (4)

42-51: LGTM! Stack initialization is well implemented

The stack setup properly handles serialization, initial state from deep links, and back button navigation.


111-118: LGTM! Config serialization is well implemented

The change from storing Image reference to ID in Photo config improves serialization and follows best practices for state management.


83-83: ⚠️ Potential issue

Handle potential NoSuchElementException when retrieving images

Using first() to find an image can throw if the ID doesn't exist. Consider using a safer approach.


93-109: ⚠️ Potential issue

Validate deep link parameters to prevent navigation issues

The image ID from deep link parameters should be validated before creating the Config.Photo.

sample/shared/shared/src/commonMain/kotlin/com/arkivanov/sample/shared/pages/DefaultPagesComponent.kt (2)

22-23: LGTM: Constructor parameter addition is well-designed

The nullable deepLinkUrl parameter appropriately handles cases where deep linking may or may not be present.


64-68: ⚠️ Potential issue

Ensure safe destructuring when consuming the path segment

The destructuring operation val (path) = deepLinkUrl?.consumePathSegment() could cause a NPE if consumePathSegment() returns null.

Consider safer implementation:

     private fun getInitialPageIndex(deepLinkUrl: Url?): Int {
-        val (path) = deepLinkUrl?.consumePathSegment() ?: return 0
+        val pathSegment = deepLinkUrl?.consumePathSegment() ?: return 0
+        val path = pathSegment.first

         return ImageResourceId.entries.indexOfFirst { it.name == path }.takeIf { it >= 0 } ?: 0
     }
decompose/src/commonMain/kotlin/com/arkivanov/decompose/router/pages/PagesWebNavigation.kt (1)

94-95: Ensure history list is not empty before accessing history.first()

In the navigate method, history.first() is called without checking if history is not empty. Although the current implementation may guarantee a non-empty history, adding a safety check can prevent potential NoSuchElementException in case of future changes.

sample/shared/shared/src/commonMain/kotlin/com/arkivanov/sample/shared/tabs/DefaultTabsComponent.kt (2)

29-29: Ensure backward compatibility with constructor changes

Adding deepLinkUrl: Url? to the constructor changes its signature. This may affect existing code that instantiates DefaultTabsComponent without this parameter. Consider providing a default value or an overloaded constructor to maintain backward compatibility.


123-123: Evaluate the impact of changing MultiPane from an object to a data class

Converting Config.MultiPane from a data object to a data class changes it from a singleton to multiple instances. This may have implications if any code relies on MultiPane being a singleton. Please ensure that this change does not adversely affect the application logic.

decompose/src/webTest/kotlin/com/arkivanov/decompose/UrlEncodeTest.kt (1)

12-12: Ensure encodeURIComponent and decodeURIComponent are properly referenced

The functions encodeURIComponent and decodeURIComponent are JavaScript global functions. Since this is a Kotlin/JS test, verify that these functions are correctly accessible or consider importing them explicitly if necessary to prevent any reference errors.

decompose/src/webMain/kotlin/com/arkivanov/decompose/Utils.kt (3)

11-12: LGTM: Serialization implementation looks correct

The encodeToJson function correctly uses the SerializableContainer serializer.


14-19: ⚠️ Potential issue

Improve exception handling in decodeContainer

The current implementation swallows all exceptions, which could mask critical issues.

Apply this diff to catch specific exceptions:

 internal fun JsonString.decodeContainer(): SerializableContainer? =
     try {
         Json.decodeFromString(SerializableContainer.serializer(), value)
-    } catch (e: Exception) {
+    } catch (e: kotlinx.serialization.SerializationException) {
         null
+    } catch (e: IllegalArgumentException) {
+        null
+    } catch (e: Exception) {
+        // Unexpected exception during deserialization
+        throw e
     }
🧰 Tools
🪛 detekt (1.23.7)

[warning] 17-17: The caught exception is swallowed. The original exception could be lost.

(detekt.exceptions.SwallowedException)


23-25: Verify browser compatibility for external functions

The external functions encodeURIComponent and decodeURIComponent rely on browser APIs.

✅ Verification successful

Let me analyze the usage of these functions further by checking the web platform target configuration and any potential polyfills or fallbacks.


Let me check the web-specific implementation and usage of these functions.


Let me check the actual usage context and implementation details of these functions.


Based on the analysis of the code, I can now provide the final response:

Browser compatibility is properly handled for URI component functions

The encodeURIComponent and decodeURIComponent functions are:

  • Properly declared as external functions targeting the web platform
  • Used in a web-specific module (webMain) with appropriate platform checks
  • Tested in web-specific tests that verify encoding/decoding functionality
  • Used safely in URL parameter handling within the web history navigation component
  • Standard JavaScript functions with universal browser support (IE5.5+)
🏁 Scripts executed

The following scripts were executed for the analysis:

Script:

#!/bin/bash
# Description: Check if these functions are used with proper null/undefined checks in the codebase

# Search for direct usage of these functions
rg "encodeURIComponent|decodeURIComponent" --type kotlin

# Search for potential error handling around these calls
rg -B 2 -A 2 "encodeURIComponent|decodeURIComponent" --type kotlin

Length of output: 3159


Script:

#!/bin/bash
# Check build configuration and web-specific setup
fd "build.gradle" --exec cat {}

# Check for any web-specific initialization or platform checks
rg -B 3 -A 3 "expect|actual.*web" --type kotlin

# Look for any browser-specific checks or initialization code
rg -B 3 -A 3 "browser|window|document" --type kotlin

Length of output: 87388


Script:

#!/bin/bash
# Check web-specific implementation and usage
rg -B 3 -A 3 "actual.*encodeURIComponent|actual.*decodeURIComponent" --type kotlin

# Look for any error handling or web platform checks around these functions
fd -e kt -e js -x cat {} | rg -B 3 -A 3 "try.*encodeURIComponent|catch.*encodeURIComponent|try.*decodeURIComponent|catch.*decodeURIComponent"

# Check web platform target configuration
fd "build.gradle.kts|build.gradle" --exec grep -A 5 "js.*browser" {}

Length of output: 1313


Script:

#!/bin/bash
# Check the implementation and usage in web-specific code
rg -B 5 -A 5 "encodeURIComponent|decodeURIComponent" --type kotlin --type js

# Look for any web-specific platform checks or initialization
fd -e kt -x cat {} | rg -B 3 -A 3 "external.*fun.*Component|@JsName|@JsModule"

# Check for any web platform specific tests
fd -e kt | rg "web.*test|test.*web" | xargs cat

Length of output: 4963

sample/app-js/src/main/kotlin/com/arkivanov/sample/app/Main.kt (1)

24-30: 🛠️ Refactor suggestion

Consider handling deep link parsing errors

The deep link URL parsing could fail if the URL format is invalid.

Consider adding error handling:

 withWebHistory { stateKeeper, deepLink ->
     DefaultRootComponent(
         componentContext = DefaultComponentContext(lifecycle = lifecycle, stateKeeper = stateKeeper),
         featureInstaller = DefaultFeatureInstaller,
-        deepLinkUrl = deepLink?.let(::Url),
+        deepLinkUrl = try {
+            deepLink?.let(::Url)
+        } catch (e: Exception) {
+            console.error("Invalid deep link URL: $deepLink")
+            null
+        },
     )
 }
sample/app-js-compose/src/jsMain/kotlin/com/arkivanov/decompose/sample/app/Main.kt (1)

27-32: 🛠️ Refactor suggestion

Consider handling deep link parsing errors

Similar to the non-Compose version, the deep link URL parsing needs error handling.

Apply the same error handling pattern:

 withWebHistory { stateKeeper, deepLink ->
     DefaultRootComponent(
         componentContext = DefaultComponentContext(lifecycle = lifecycle, stateKeeper = stateKeeper),
         featureInstaller = DefaultFeatureInstaller,
-        deepLinkUrl = deepLink?.let(::Url),
+        deepLinkUrl = try {
+            deepLink?.let(::Url)
+        } catch (e: Exception) {
+            console.error("Invalid deep link URL: $deepLink")
+            null
+        },
     )
 }
decompose/src/commonMain/kotlin/com/arkivanov/decompose/router/stack/StackWebNavigation.kt (1)

58-78: LGTM!

The factory function is well-designed with appropriate default values and clear parameter naming.

sample/shared/shared/src/commonMain/kotlin/com/arkivanov/sample/shared/multipane/DefaultMultiPaneComponent.kt (1)

67-71: Ensure parametersMapper returns a non-null Map.

The parametersMapper lambda returns null when panels.details is null, which could lead to issues.

decompose/src/webMain/kotlin/com/arkivanov/decompose/router/webhistory/WebHistoryNavigation.kt (5)

80-84: LGTM: Clean recursive implementation

The recursive navigation check is well-implemented and handles the child navigation hierarchy correctly.


235-269: LGTM: Proper URL encoding implementation

The URL handling functions correctly implement URL encoding and parameter handling.


320-328: LGTM: Well-structured serialization implementation

The SerializableNode class follows Kotlin serialization best practices with proper companion object setup.


86-93: 🛠️ Refactor suggestion

Add error handling for serialization

The serialization process should handle potential exceptions to prevent app crashes.

 private fun <T : Any> WebNavigation<T>.navigate(nodes: List<SerializableNode>) {
-    val items: List<T> = nodes.map { it.data.consumeRequired(serializer) }
+    val items: List<T> = try {
+        nodes.map { it.data.consumeRequired(serializer) }
+    } catch (e: Exception) {
+        // Log the error and return to maintain app stability
+        return
+    }

Likely invalid or redundant comment.


22-36: ⚠️ Potential issue

Add error handling for state deserialization

The state deserialization in onPopState should handle potential exceptions to prevent app crashes.

 fun onPopState(state: String?) {
-    val deserializedState = state?.deserializeState() ?: return
+    val deserializedState = try {
+        state?.deserializeState()
+    } catch (e: Exception) {
+        // Log the error and return to maintain app stability
+        return
+    } ?: return

Likely invalid or redundant comment.

decompose/api/jvm/decompose.api (1)

501-527: LGTM: Well-designed web navigation API

The WebNavigation interface and related components are well-designed with clear responsibilities and proper separation of concerns.

decompose/api/android/decompose.api (1)

521-547: LGTM: Consistent Android API implementation

The Android API declarations maintain consistency with the JVM version while properly integrating with Android components.

decompose/api/decompose.klib.api (2)

557-558: Platform-specific implementation properly isolated!

The withWebHistory function is correctly restricted to JS and Wasm-JS platforms where web history APIs are available.

Let's verify the platform support:

✅ Verification successful

Platform-specific implementation verified and correctly isolated

The verification confirms that:

  • The common interface WebHistoryController is platform-agnostic and properly defined in commonMain
  • Both JS and WasmJS have identical implementations of withWebHistory using browser-specific APIs (window, sessionStorage)
  • The DefaultBrowserHistory is properly implemented with expect/actual declarations for both JS and WasmJS targets
  • No implementation exists for non-web platforms, as expected
🏁 Scripts executed

The following scripts were executed for the analysis:

Script:

#!/bin/bash
# Description: Check for proper platform-specific implementations
# Expected: Implementation files for supported platforms only

# Search for platform-specific implementations
fd -e kt ".*WebHistory.*" --type f

Length of output: 746


Script:

#!/bin/bash
# Let's verify the actual implementations and platform restrictions

# Check the content of the common interface
cat decompose/src/commonMain/kotlin/com/arkivanov/decompose/router/stack/webhistory/WebHistoryController.kt

# Check JS implementation
cat decompose/src/jsMain/kotlin/com/arkivanov/decompose/router/webhistory/WebHistoryNavigation.kt

# Check WasmJS implementation
cat decompose/src/wasmJsMain/kotlin/com/arkivanov/decompose/router/webhistory/WebHistoryNavigation.kt

# Check if there are any expect/actual declarations
rg "expect|actual" decompose/src/*/kotlin/com/arkivanov/decompose/router/webhistory/

Length of output: 8441


498-498: Consistent implementation of web navigation across different models!

The functions provide a unified approach to web navigation while respecting the unique characteristics of each navigation model (panels, pages, stack).

Let's verify the consistency of navigation patterns:

Also applies to: 512-513

✅ Verification successful

Let me gather more specific information about the implementation patterns.


Let me try a different approach to examine the implementations.


Let me try one final search pattern to verify the implementations.


Consistent web navigation implementation verified across all navigation models!

Based on the code analysis, all three web navigation implementations (childPanelsWebNavigation, childPagesWebNavigation, and childStackWebNavigation) follow a consistent pattern with:

  • Common core parameters: navigator, serializer, and model-specific state (panels/pages/stack)
  • Consistent path and parameter mapping functionality
  • Uniform child selection mechanism via childSelector: (Child) -> WebNavigationOwner?
  • Proper type safety through generics
  • Consistent return type of WebNavigation<*>
🏁 Scripts executed

The following scripts were executed for the analysis:

Script:

#!/bin/bash
# Description: Check for consistent parameter patterns across navigation functions
# Expected: Similar parameter patterns for all navigation functions

# Search for navigation function declarations
rg "Web[nN]avigation.*\(" --type kotlin

Length of output: 18098


Script:

#!/bin/bash
# Get implementation details of the web navigation functions
ast-grep --pattern 'fun childPanelsWebNavigation($$$)'
ast-grep --pattern 'fun childPagesWebNavigation($$$)'
ast-grep --pattern 'fun childStackWebNavigation($$$)'

Length of output: 168


Script:

#!/bin/bash
# Search for the web navigation function implementations with context
rg "fun child.*WebNavigation" -A 10 --type kotlin

Length of output: 51


Script:

#!/bin/bash
# Search for web navigation implementations in specific files
rg "WebNavigation" --type kotlin -A 5 decompose/src/commonMain/kotlin/com/arkivanov/decompose/router/panels/
rg "WebNavigation" --type kotlin -A 5 decompose/src/commonMain/kotlin/com/arkivanov/decompose/router/pages/
rg "WebNavigation" --type kotlin -A 5 decompose/src/commonMain/kotlin/com/arkivanov/decompose/router/stack/

Length of output: 20813

@arkivanov arkivanov merged commit 828d5b5 into master Dec 7, 2024
3 checks passed
@arkivanov arkivanov deleted the web-history-2 branch December 7, 2024 17:20
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
None yet
Projects
None yet
Development

Successfully merging this pull request may close these issues.

1 participant