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

feat: refactor & added animations to cf #2841

Open
wants to merge 12 commits into
base: dev
Choose a base branch
from

Conversation

chesterkmr
Copy link
Collaborator

@chesterkmr chesterkmr commented Nov 15, 2024

  • Cleaned code
  • Added smooth animated transition from signup to main collection flow
  • Added error pages for Network Error,Invalid Token, Missing Token, Unexpected error.

Summary by CodeRabbit

  • New Features

    • Introduced animated components using framer-motion for improved user experience in various sections, including backgrounds, headers, footers, and forms.
    • Added ServerNotAvailableError handling to enhance error management.
    • Implemented a new SignUpFormProvider for better state management during user signup.
    • Added a new context, DependenciesContext, for managing dependencies more effectively.
    • Introduced AnimatedUIRenderer for enhanced UI rendering with animations.
  • Bug Fixes

    • Enhanced error handling in the ErrorScreen to manage server availability issues.
  • Refactor

    • Updated hooks to streamline data fetching and state management, including useCustomerQuery, useUISchemasQuery, and useDependencies.
    • Removed deprecated components and hooks to simplify the codebase.
    • Modified the routing configuration to remove unnecessary higher-order components.
  • Chores

    • Updated dependencies in package.json for improved functionality.

Copy link
Contributor

coderabbitai bot commented Nov 15, 2024

Walkthrough

The pull request introduces several modifications to the @ballerine/kyb-app project, primarily focusing on dependency updates and enhancements to various components. The package.json file reinstates the @ballerine/ui dependency and adds framer-motion. Multiple components have been updated to utilize framer-motion for animations, including Background, Footer, FormContainer, Header, and Logo. Additionally, the DependenciesProvider has been enhanced with new hooks and improved error handling. Several components have been deleted, including LoadingScreen and CustomerProvider, while new contexts and hooks have been introduced.

Changes

File Change Summary
apps/kyb-app/package.json Restored dependency @ballerine/ui to version 0.5.44 and added framer-motion with version ^8.3.4.
apps/kyb-app/src/common/components/layouts/Signup/Background.tsx Updated to use motion.div and motion.img with animations based on isLoaded state.
apps/kyb-app/src/common/components/layouts/Signup/Footer.tsx Replaced <div> with <motion.div> for animated transitions.
apps/kyb-app/src/common/components/layouts/Signup/FormContainer.tsx Changed <div> to <motion.div> with animation properties.
apps/kyb-app/src/common/components/layouts/Signup/Header.tsx Updated to use motion.h1 and motion.p for animated rendering.
apps/kyb-app/src/common/components/layouts/Signup/Logo.tsx Modified to use motion.img with loading state animations.
apps/kyb-app/src/common/components/molecules/LoadingScreen/LoadingScreen.tsx Component removed.
apps/kyb-app/src/common/components/molecules/LoadingScreen/index.ts Export module deleted.
apps/kyb-app/src/common/providers/DependenciesProvider/DependenciesProvider.tsx Enhanced with new hooks and improved error handling.
apps/kyb-app/src/components/layouts/AppShell/AppShell.tsx Updated import order and wrapped children in a div with a new class.
apps/kyb-app/src/components/layouts/AppShell/Content.tsx Changed <div> to <motion.div> for animation.
apps/kyb-app/src/components/layouts/AppShell/Navigation.tsx Updated import from useCustomer to useCustomerQuery.
apps/kyb-app/src/components/molecules/CustomerProviderFallback/CustomerProviderFallback.tsx Component removed.
apps/kyb-app/src/components/providers/CustomerProvider/CustomerProvider.tsx Component removed.
apps/kyb-app/src/hooks/useSessionQuery/hocs/withSessionProtected.tsx HOC removed.
apps/kyb-app/src/hooks/useSessionQuery/index.ts Export module deleted.
apps/kyb-app/src/hooks/useSessionQuery/useSessionQuery.ts Hook removed.
apps/kyb-app/src/pages/CollectionFlow/CollectionFlow.tsx Removed withSessionProtected, replaced useCustomer with useCustomerQuery, added useDependencies.
apps/kyb-app/src/pages/CollectionFlow/components/atoms/LoadingScreen/LoadingScreen.tsx Updated to include props for animations and conditional rendering.
apps/kyb-app/src/pages/CollectionFlow/components/pages/Approved/Approved.tsx Replaced useCustomer with useCustomerQuery and removed withSessionProtected.
apps/kyb-app/src/pages/CollectionFlow/components/pages/CompletedScreen/CompletedScreen.tsx Replaced useCustomer with useCustomerQuery and removed withSessionProtected.
apps/kyb-app/src/pages/CollectionFlow/components/pages/FailedScreen/FailedScreen.tsx Updated to use useCustomerQuery.
apps/kyb-app/src/pages/CollectionFlow/components/pages/Rejected/Rejected.tsx Removed withSessionProtected and updated to useCustomerQuery.
apps/kyb-app/src/pages/Root/Root.tsx Updated navigation logic and loading state handling.
apps/kyb-app/src/pages/Root/hooks/useIsSignupRequired/useIsSignupRequired.ts Hook removed.
apps/kyb-app/src/pages/SignUpPage/SignUpPage.tsx Removed loading state management and updated theme handling.
apps/kyb-app/src/router.tsx Removed withCustomer from routing for CollectionFlow.
apps/kyb-app/src/pages/SignUpPage/components/SignUpForm/components/Submit/Submit.tsx Introduced useSignupForm for loading state management.
apps/kyb-app/src/common/providers/DependenciesProvider/dependencies-context.ts New context DependenciesContext added.
apps/kyb-app/src/common/providers/DependenciesProvider/hooks/useDependencies/useDependencies.ts New hook useDependencies added.
apps/kyb-app/src/common/providers/DependenciesProvider/types.ts New interface IDependenciesContext added.
apps/kyb-app/src/hooks/useCustomerQuery/useCustomerQuery.ts Updated to include a refetch function in the return value.
apps/kyb-app/src/hooks/useFlowContextQuery/useFlowContextQuery.ts Updated loading state logic and added isLoaded property.
apps/kyb-app/src/pages/CollectionFlow/components/organisms/AnimatedUIRenderer.tsx New component added for animated rendering.
apps/kyb-app/src/pages/SignUpPage/components/SignUpForm/SignUpForm.tsx Enhanced state management and introduced new hooks for handling signup.
apps/kyb-app/src/pages/SignUpPage/components/SignUpForm/components/SignUpFormProvider/SignUpFormProvider.tsx New provider component added for managing signup form state.
apps/kyb-app/src/pages/SignUpPage/components/SignUpForm/components/SignUpFormProvider/hooks/useSignupForm/useSignupForm.ts New hook added for consuming signup form context.
apps/kyb-app/src/pages/SignUpPage/components/SignUpForm/components/SignUpFormProvider/signup-form-context.ts New context for managing signup state added.
apps/kyb-app/src/pages/SignUpPage/components/SignUpForm/hooks/useCreateEndUserMutation/useCreateEndUserMutation.ts Simplified by removing navigation logic.
services/workflows-service/prisma/data-migrations Updated subproject commit identifier.
apps/kyb-app/src/common/components/organisms/ErrorScreen/ErrorScreen.tsx Enhanced error handling with new ServerNotAvailableError.
apps/kyb-app/src/common/components/organisms/ErrorScreen/ServerNotAvailable.tsx New component added for handling server not available errors.
apps/kyb-app/src/common/errors/server-not-available.ts New error class ServerNotAvailableError added.
apps/kyb-app/src/common/utils/request.ts Enhanced request function with improved error handling for server availability.
apps/kyb-app/src/common/providers/ThemeProvider/ThemeProvider.tsx Updated useUISchemasQuery call to accept an object parameter.
apps/kyb-app/src/components/layouts/AppShell/LanguagePicker.tsx Updated useUISchemasQuery call to accept an object parameter.
apps/kyb-app/src/domains/collection-flow/collection-flow.api.ts Modified fetchUISchema to make endUserId optional.
apps/kyb-app/src/domains/collection-flow/query-keys.ts Updated method signatures to simplify parameters.
apps/kyb-app/src/hooks/useAppExit/useAppExit.ts Updated useUISchemasQuery call to accept an object parameter.
apps/kyb-app/src/hooks/useEndUserQuery/useEndUserQuery.ts Added isFetched to return value and modified loading logic.
apps/kyb-app/src/hooks/useUIOptionsRedirect/useUIOptionsRedirect.ts Updated useUISchemasQuery call to accept an object parameter.
apps/kyb-app/src/pages/Root/hooks/useIsSignupRequired/index.ts Export module deleted.

Sequence Diagram(s)

sequenceDiagram
    participant User
    participant App
    participant DependenciesProvider
    participant LoadingScreen
    participant ErrorScreen

    User->>App: Initiates action
    App->>DependenciesProvider: Checks dependencies
    DependenciesProvider-->>App: Returns loading state
    App->>LoadingScreen: Show loading
    LoadingScreen-->>App: Loading complete
    App->>ErrorScreen: Check for errors
    ErrorScreen-->>User: Display error if any
Loading

Possibly related PRs

Suggested reviewers

  • alonp99
  • tomer-shvadron

Poem

🐇 In the land of code where rabbits play,
New dependencies hop in, brightening the day.
With motion and flair, our components now dance,
Error screens ready, giving users a chance.
So let’s celebrate changes, both big and small,
For in this code garden, we’re having a ball! 🎉


📜 Recent review details

Configuration used: CodeRabbit UI
Review profile: CHILL

📥 Commits

Reviewing files that changed from the base of the PR and between cd07107 and 8d0bbfd.

⛔ Files ignored due to path filters (1)
  • pnpm-lock.yaml is excluded by !**/pnpm-lock.yaml
📒 Files selected for processing (6)
  • apps/kyb-app/package.json (1 hunks)
  • apps/kyb-app/src/common/utils/request.ts (3 hunks)
  • apps/kyb-app/src/domains/collection-flow/collection-flow.api.ts (1 hunks)
  • apps/kyb-app/src/hooks/useEndUserQuery/useEndUserQuery.ts (2 hunks)
  • apps/kyb-app/src/pages/CollectionFlow/CollectionFlow.tsx (6 hunks)
  • services/workflows-service/prisma/data-migrations (1 hunks)
🚧 Files skipped from review as they are similar to previous changes (6)
  • apps/kyb-app/package.json
  • apps/kyb-app/src/common/utils/request.ts
  • apps/kyb-app/src/domains/collection-flow/collection-flow.api.ts
  • apps/kyb-app/src/hooks/useEndUserQuery/useEndUserQuery.ts
  • apps/kyb-app/src/pages/CollectionFlow/CollectionFlow.tsx
  • services/workflows-service/prisma/data-migrations

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
Contributor

@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: 21

🧹 Outside diff range and nitpick comments (15)
apps/kyb-app/src/hooks/useUISchemasQuery/useUISchemasQuery.ts (1)

18-18: Consider renaming isLoaded to isFetched for consistency.

The property name isLoaded deviates from React Query's conventions and could be misleading since isFetched is true even when the query fails. Consider keeping the original name for better alignment with React Query's terminology.

- isLoaded: isFetched,
+ isFetched,
apps/kyb-app/src/common/components/layouts/Signup/Footer.tsx (1)

18-23: Consider optimizing animation parameters

The animation implementation looks good, but consider these improvements:

  1. The 0.6s delay might feel too long for a footer element
  2. Animation values could be extracted into constants for reusability across components

Consider this refactor:

+const FOOTER_ANIMATION = {
+  initial: { opacity: 0, y: 20 },
+  animate: { opacity: 1, y: 0 },
+  transition: { duration: 0.5, delay: 0.3 }
+} as const;

 export const Footer: FunctionComponent<IFooterProps> = props => {
   // ...
   return (
     <motion.div
       className="font-inter text-base text-[#94A3B8]"
       style={styles}
-      initial={{ opacity: 0, y: 20 }}
-      animate={{ opacity: 1, y: 0 }}
-      transition={{ duration: 0.5, delay: 0.6 }}
+      {...FOOTER_ANIMATION}
       dangerouslySetInnerHTML={{ __html: DOMPurify(window).sanitize(rawHtml) }}
     />
   );
 };
apps/kyb-app/src/common/components/layouts/Signup/FormContainer.tsx (1)

21-27: Consider animation accessibility and performance improvements

While the animation implementation is good, consider these enhancements:

  1. Add support for users who prefer reduced motion
  2. Consider reducing the delay (0.5s might feel too long)
  3. Add layout prop for smoother layout transitions

Apply this diff to implement the suggestions:

 <motion.div
   className="my-6 flex flex-col gap-4 pr-10"
   style={containerStyles}
-  initial={{ opacity: 0, y: 20 }}
-  animate={{ opacity: 1, y: 0 }}
-  transition={{ duration: 0.5, delay: 0.5 }}
+  initial={{ opacity: 0, y: 20 }}
+  animate={{ opacity: 1, y: 0 }}
+  transition={{ duration: 0.5, delay: 0.3 }}
+  layout
+  whileHover={{ scale: 1.01 }}
+  {...(window.matchMedia('(prefers-reduced-motion: reduce)').matches && {
+    initial: { opacity: 1, y: 0 },
+    animate: { opacity: 1, y: 0 }
+  })}
apps/kyb-app/src/pages/Root/Root.tsx (1)

30-34: Consider adding animation variants for child routes

The AnimatePresence wrapper is correctly implemented with mode="wait". To complete the animation implementation, ensure that child routes rendered through the Outlet include motion variants (e.g., initial, animate, exit).

Example for child routes:

// In child route components
import { motion } from 'framer-motion';

export const ChildRoute = () => (
  <motion.div
    initial={{ opacity: 0 }}
    animate={{ opacity: 1 }}
    exit={{ opacity: 0 }}
  >
    {/* content */}
  </motion.div>
);
apps/kyb-app/src/common/components/layouts/Signup/Background.tsx (1)

19-30: Consider adding accessibility and error handling

While the animation implementation is good, consider these improvements:

  1. Add alt text for accessibility
  2. Handle image load errors
  3. Consider reducing motion based on user preferences

Apply this diff to implement the suggestions:

 <motion.img
   src={imageSrc}
+  alt="Background image"
   className="h-full w-full object-cover"
   style={styles}
   initial={{ opacity: 0, scale: 1.2 }}
   animate={isLoaded ? { opacity: 1, scale: 1 } : { opacity: 0, scale: 1.2 }}
   transition={{
     duration: 0.8,
     ease: 'easeOut',
   }}
   onLoad={() => setIsLoaded(true)}
+  onError={() => console.error('Failed to load background image')}
 />

Also, consider adding a media query for reduced motion:

const prefersReducedMotion = window.matchMedia('(prefers-reduced-motion: reduce)').matches;

// Use this in your animation config
animate={isLoaded ? 
  { 
    opacity: 1, 
    scale: prefersReducedMotion ? 1 : 1 
  } : { 
    opacity: 0, 
    scale: prefersReducedMotion ? 1 : 1.2 
  }
}
apps/kyb-app/src/common/components/layouts/Signup/Header.tsx (1)

20-35: Animation implementation looks good! Consider extracting animation variants

The animation implementation is clean and creates a nice staggered effect. However, to improve reusability and maintainability, consider extracting the animation properties into variants.

Here's how you could refactor this:

const fadeInUpVariants = {
  hidden: { opacity: 0, y: 20 },
  visible: { opacity: 1, y: 0 },
};

// In your component:
<motion.h1
  className="text-2xl font-bold"
  variants={fadeInUpVariants}
  initial="hidden"
  animate="visible"
  transition={{ duration: 0.5 }}
>
  {headingText}
</motion.h1>
<motion.p
  className="text-base"
  variants={fadeInUpVariants}
  initial="hidden"
  animate="visible"
  transition={{ duration: 0.5, delay: 0.2 }}
>
  {subheadingText}
</motion.p>

This approach makes it easier to:

  • Reuse animations across components
  • Maintain consistency in animations
  • Modify animations in one place
apps/kyb-app/src/pages/CollectionFlow/components/atoms/LoadingScreen/LoadingScreen.tsx (2)

4-10: Enhance interface documentation for better maintainability

The interface could benefit from more detailed JSDoc documentation.

+/**
+ * Props for the LoadingScreen component
+ */
 interface LoadingScreenProps {
+  /** Callback fired when the entry animation completes */
   onAnimationComplete?: () => void;
+  /** Callback fired when the exit animation completes */
   onExitComplete?: () => void;
 
-  // When providing exactly false onExit callback will be fired
+  /**
+   * Controls the visibility of the loading screen
+   * @default undefined
+   * When explicitly set to false, triggers the exit animation
+   */
   isLoading?: boolean;
 }

18-34: Consider UX improvements for the loading animation

While the implementation is solid, consider the following improvements:

  1. The animation duration of 0.7s might feel slow for frequent loading states
  2. The scale animation from 0.5 might be too dramatic and distracting
  3. Add accessibility attributes for screen readers
     <AnimatePresence mode="wait" onExitComplete={onExitComplete}>
       {isLoading === false ? null : (
         <motion.div
           key="loading"
-          initial={{ opacity: 0, scale: 0.5 }}
-          animate={{ opacity: 1, scale: 1 }}
-          exit={{ opacity: 0, scale: 0.5 }}
+          initial={{ opacity: 0, scale: 0.8 }}
+          animate={{ opacity: 1, scale: 1 }}
+          exit={{ opacity: 0, scale: 0.8 }}
           transition={{
-            duration: 0.7,
+            duration: 0.4,
             ease: [0, 0.71, 0.2, 1.01],
           }}
-          className="fixed inset-0 flex h-screen w-screen items-center justify-center bg-white/80 backdrop-blur-sm"
+          className="fixed inset-0 flex h-screen w-screen items-center justify-center bg-white/90 backdrop-blur-sm"
           onAnimationComplete={onAnimationComplete}
+          role="alert"
+          aria-live="polite"
+          aria-label="Loading, please wait"
         >
           <Loader2 className="text-black animate-spin" size={72} />
         </motion.div>
apps/kyb-app/src/router.tsx (1)

Line range hint 1-54: Clean router structure, consider implementing the TODO for 404 page

The router structure is clean and well-organized. The changes align with modern React practices by moving away from HOCs.

Would you like me to help implement a 404 page component and its route configuration to address the TODO comment?

apps/kyb-app/src/pages/CollectionFlow/components/pages/CompletedScreen/CompletedScreen.tsx (1)

Line range hint 12-52: Consider adding loading and error states

The component could benefit from explicit handling of loading and error states from the query.

Consider implementing loading and error states:

 export const CompletedScreen = () => {
   const { t } = useTranslation();
-  const { customer } = useCustomerQuery();
+  const { customer, isLoading, error } = useCustomerQuery();
   const { trackEvent } = useFlowTracking();
 
   const { exit, isExitAvailable } = useAppExit();
 
   useEffect(() => {
     trackEvent(CollectionFlowEvents.FLOW_COMPLETED);
   }, [trackEvent]);
 
   useUIOptionsRedirect('success');
 
+  if (isLoading) {
+    return (
+      <div className="flex h-full items-center justify-center">
+        <Card className="w-full max-w-[646px] p-12">
+          <div className="text-center">Loading...</div>
+        </Card>
+      </div>
+    );
+  }
+
+  if (error) {
+    return (
+      <div className="flex h-full items-center justify-center">
+        <Card className="w-full max-w-[646px] p-12">
+          <div className="text-center text-red-500">
+            {t('error.generic')}
+          </div>
+        </Card>
+      </div>
+    );
+  }
apps/kyb-app/src/components/layouts/AppShell/Navigation.tsx (1)

17-17: Consider handling loading and error states from useCustomerQuery

The hook likely provides loading and error states that should be handled explicitly. Consider destructuring these states to provide better user feedback.

-  const { customer } = useCustomerQuery();
+  const { customer, isLoading, error } = useCustomerQuery();
+  
+  if (error) {
+    // Handle error state appropriately
+    return null;
+  }
apps/kyb-app/src/common/providers/DependenciesProvider/DependenciesProvider.tsx (1)

Line range hint 1-83: Consider splitting the provider into smaller, focused components

The DependenciesProvider is handling multiple concerns (data fetching, error handling, loading states, and animations). Consider splitting it into:

  1. DataProvider - Handle data fetching and error handling
  2. LoadingProvider - Manage loading states and animations

This would improve maintainability and make the code easier to test.

Would you like me to provide an example of this split architecture?

apps/kyb-app/package.json (1)

51-51: Review motion library configuration

The motion library has been added for animations. Consider the following recommendations:

  1. The current React version (^18.2.0) is compatible with [email protected]
  2. For optimal performance, consider adding motion-specific configuration to reduce bundle size

Consider adding a motion configuration file (e.g., motion.config.ts):

import { MotionConfig } from 'motion'

export const motionConfig = {
  reducedMotion: 'user', // Respect user's reduced motion preferences
  transition: {
    // Default animation settings
    type: 'spring',
    duration: 0.3,
  },
}
apps/kyb-app/src/hooks/useUIOptionsRedirect/useUIOptionsRedirect.unit.test.ts (1)

Line range hint 124-144: Add missing useUISchemasQuery mocks in the last two test cases.

The last two test cases don't mock useUISchemasQuery, which could lead to inconsistent test behavior if the hook implementation changes. For consistency, these test cases should also include the mock setup.

Add the mock before each test case:

it('should redirect to failure URL when state is failure', () => {
  vi.mocked(useStateManagerContext).mockReturnValue({
    config: mockConfig,
  } as StateManagerContext);
  vi.mocked(useUISchemasQuery).mockReturnValue({
    data: null,
    isLoading: false,
    error: null,
    isLoaded: true,
  });

  renderHook(() => useUIOptionsRedirect('failure'));
  // ... rest of the test
});

it('should log info message when redirecting', () => {
  vi.mocked(useStateManagerContext).mockReturnValue({
    config: mockConfig,
  } as StateManagerContext);
  vi.mocked(useUISchemasQuery).mockReturnValue({
    data: null,
    isLoading: false,
    error: null,
    isLoaded: true,
  });

  renderHook(() => useUIOptionsRedirect('success'));
  // ... rest of the test
});
apps/kyb-app/src/pages/CollectionFlow/CollectionFlow.tsx (1)

Line range hint 205-273: Optimize animation components to improve performance

While adding animations enhances the user experience, nesting multiple motion.div components with similar animation properties may impact performance. Consider simplifying the animation structure by reducing unnecessary nesting and reusing common animation configurations.

You might refactor by extracting common animation properties into variants or shared variables to manage animations more efficiently.

📜 Review details

Configuration used: CodeRabbit UI
Review profile: CHILL

📥 Commits

Reviewing files that changed from the base of the PR and between 37a8a40 and 3c3b21d.

⛔ Files ignored due to path filters (1)
  • pnpm-lock.yaml is excluded by !**/pnpm-lock.yaml
📒 Files selected for processing (47)
  • apps/kyb-app/package.json (2 hunks)
  • apps/kyb-app/src/common/components/layouts/Signup/Background.tsx (2 hunks)
  • apps/kyb-app/src/common/components/layouts/Signup/Footer.tsx (2 hunks)
  • apps/kyb-app/src/common/components/layouts/Signup/FormContainer.tsx (2 hunks)
  • apps/kyb-app/src/common/components/layouts/Signup/Header.tsx (2 hunks)
  • apps/kyb-app/src/common/components/layouts/Signup/Logo.tsx (2 hunks)
  • apps/kyb-app/src/common/components/layouts/Signup/Signup.tsx (2 hunks)
  • apps/kyb-app/src/common/components/molecules/LoadingScreen/LoadingScreen.tsx (0 hunks)
  • apps/kyb-app/src/common/components/molecules/LoadingScreen/index.ts (0 hunks)
  • apps/kyb-app/src/common/providers/DependenciesProvider/DependenciesProvider.tsx (3 hunks)
  • apps/kyb-app/src/components/layouts/AppShell/AppShell.tsx (1 hunks)
  • apps/kyb-app/src/components/layouts/AppShell/Content.tsx (1 hunks)
  • apps/kyb-app/src/components/layouts/AppShell/Navigation.tsx (1 hunks)
  • apps/kyb-app/src/components/molecules/CustomerProviderFallback/CustomerProviderFallback.tsx (0 hunks)
  • apps/kyb-app/src/components/molecules/CustomerProviderFallback/index.ts (0 hunks)
  • apps/kyb-app/src/components/providers/CustomerProvider/CustomerProvider.tsx (0 hunks)
  • apps/kyb-app/src/components/providers/CustomerProvider/customer.context.ts (0 hunks)
  • apps/kyb-app/src/components/providers/CustomerProvider/hooks/useCustomer/index.ts (0 hunks)
  • apps/kyb-app/src/components/providers/CustomerProvider/hooks/useCustomer/useCustomer.ts (0 hunks)
  • apps/kyb-app/src/components/providers/CustomerProvider/index.ts (0 hunks)
  • apps/kyb-app/src/components/providers/CustomerProvider/types.ts (0 hunks)
  • apps/kyb-app/src/hocs/withCustomer/index.ts (0 hunks)
  • apps/kyb-app/src/hocs/withCustomer/withCustomer.tsx (0 hunks)
  • apps/kyb-app/src/hooks/useSessionQuery/hocs/withSessionProtected.tsx (0 hunks)
  • apps/kyb-app/src/hooks/useSessionQuery/index.ts (0 hunks)
  • apps/kyb-app/src/hooks/useSessionQuery/useSessionQuery.ts (0 hunks)
  • apps/kyb-app/src/hooks/useUIOptionsRedirect/useUIOptionsRedirect.unit.test.ts (4 hunks)
  • apps/kyb-app/src/hooks/useUISchemasQuery/useUISchemasQuery.ts (2 hunks)
  • apps/kyb-app/src/pages/CollectionFlow/CollectionFlow.tsx (10 hunks)
  • apps/kyb-app/src/pages/CollectionFlow/components/atoms/LoadingScreen/LoadingScreen.tsx (1 hunks)
  • apps/kyb-app/src/pages/CollectionFlow/components/pages/Approved/Approved.tsx (2 hunks)
  • apps/kyb-app/src/pages/CollectionFlow/components/pages/CompletedScreen/CompletedScreen.tsx (2 hunks)
  • apps/kyb-app/src/pages/CollectionFlow/components/pages/FailedScreen/FailedScreen.tsx (2 hunks)
  • apps/kyb-app/src/pages/CollectionFlow/components/pages/Rejected/Rejected.tsx (2 hunks)
  • apps/kyb-app/src/pages/Root/Root.tsx (2 hunks)
  • apps/kyb-app/src/pages/Root/hooks/useIsSignupRequired/useIsSignupRequired.ts (1 hunks)
  • apps/kyb-app/src/pages/SignIn/SignIn.tsx (0 hunks)
  • apps/kyb-app/src/pages/SignIn/components/layouts/SigninLayout/SigninLayout.tsx (0 hunks)
  • apps/kyb-app/src/pages/SignIn/components/layouts/SigninLayout/index.ts (0 hunks)
  • apps/kyb-app/src/pages/SignIn/components/organisms/SigninForm/SigninForm.tsx (0 hunks)
  • apps/kyb-app/src/pages/SignIn/components/organisms/SigninForm/index.ts (0 hunks)
  • apps/kyb-app/src/pages/SignIn/components/organisms/SigninForm/signin.schema.ts (0 hunks)
  • apps/kyb-app/src/pages/SignIn/components/organisms/SigninForm/signin.ui-schema.ts (0 hunks)
  • apps/kyb-app/src/pages/SignIn/components/organisms/SigninForm/types.ts (0 hunks)
  • apps/kyb-app/src/pages/SignIn/index.ts (0 hunks)
  • apps/kyb-app/src/pages/SignUpPage/SignUpPage.tsx (1 hunks)
  • apps/kyb-app/src/router.tsx (1 hunks)
💤 Files with no reviewable changes (24)
  • apps/kyb-app/src/common/components/molecules/LoadingScreen/LoadingScreen.tsx
  • apps/kyb-app/src/common/components/molecules/LoadingScreen/index.ts
  • apps/kyb-app/src/components/molecules/CustomerProviderFallback/CustomerProviderFallback.tsx
  • apps/kyb-app/src/components/molecules/CustomerProviderFallback/index.ts
  • apps/kyb-app/src/components/providers/CustomerProvider/CustomerProvider.tsx
  • apps/kyb-app/src/components/providers/CustomerProvider/customer.context.ts
  • apps/kyb-app/src/components/providers/CustomerProvider/hooks/useCustomer/index.ts
  • apps/kyb-app/src/components/providers/CustomerProvider/hooks/useCustomer/useCustomer.ts
  • apps/kyb-app/src/components/providers/CustomerProvider/index.ts
  • apps/kyb-app/src/components/providers/CustomerProvider/types.ts
  • apps/kyb-app/src/hocs/withCustomer/index.ts
  • apps/kyb-app/src/hocs/withCustomer/withCustomer.tsx
  • apps/kyb-app/src/hooks/useSessionQuery/hocs/withSessionProtected.tsx
  • apps/kyb-app/src/hooks/useSessionQuery/index.ts
  • apps/kyb-app/src/hooks/useSessionQuery/useSessionQuery.ts
  • apps/kyb-app/src/pages/SignIn/SignIn.tsx
  • apps/kyb-app/src/pages/SignIn/components/layouts/SigninLayout/SigninLayout.tsx
  • apps/kyb-app/src/pages/SignIn/components/layouts/SigninLayout/index.ts
  • apps/kyb-app/src/pages/SignIn/components/organisms/SigninForm/SigninForm.tsx
  • apps/kyb-app/src/pages/SignIn/components/organisms/SigninForm/index.ts
  • apps/kyb-app/src/pages/SignIn/components/organisms/SigninForm/signin.schema.ts
  • apps/kyb-app/src/pages/SignIn/components/organisms/SigninForm/signin.ui-schema.ts
  • apps/kyb-app/src/pages/SignIn/components/organisms/SigninForm/types.ts
  • apps/kyb-app/src/pages/SignIn/index.ts
🔇 Additional comments (32)
apps/kyb-app/src/pages/Root/hooks/useIsSignupRequired/useIsSignupRequired.ts (2)

Line range hint 1-17: Clean and well-structured implementation

The hook follows React best practices with proper memoization, clear separation of concerns, and a logical flow for determining signup requirements.


10-10: Consider if Boolean() wrapper is necessary

The Boolean() wrapper appears redundant as error || isLoading would already evaluate to a boolean value. If this was added for TypeScript strict mode compliance, consider explicitly typing the variables instead.

Let's verify the types of error and isLoading from useEndUserQuery:

✅ Verification successful

Let me gather more information about the error type since I see it's being cast as HTTPError.


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

Boolean() wrapper is necessary for type safety

The Boolean() wrapper is actually necessary here because:

  • The error from useEndUserQuery is of type HTTPError | null
  • When combining error || isLoading where one operand could be null, TypeScript's type inference benefits from the explicit Boolean() coercion
  • This ensures consistent boolean output regardless of whether we have an HTTPError object or null
🏁 Scripts executed

The following scripts were executed for the analysis:

Script:

#!/bin/bash
# Description: Check the implementation and return types of useEndUserQuery

# Search for useEndUserQuery implementation
ast-grep --pattern 'export const useEndUserQuery = () => {
  $$$
}'

# Search for any type definitions
rg -A 5 "type.*EndUserQuery"

Length of output: 1788


Script:

#!/bin/bash
# Check HTTPError type and usage
rg "HTTPError" -A 3

# Check the actual implementation of useIsSignupRequired to understand the full context
ast-grep --pattern 'export const useIsSignupRequired = () => {
  $$$
}'

Length of output: 18114

apps/kyb-app/src/components/layouts/AppShell/Content.tsx (1)

2-2: Verify the motion library import path

The import path motion/react seems unusual. The common motion libraries are either @motionone/react or framer-motion. Please verify if this is the correct import path.

#!/bin/bash
# Check package.json for motion-related dependencies
echo "Checking package.json for motion dependencies:"
fd package.json | xargs cat | jq '.dependencies, .devDependencies' | grep -i motion

# Check for any other motion imports in the codebase
echo "Checking other motion imports:"
rg "from ['\"](.*motion.*)['\"]"
apps/kyb-app/src/hooks/useUISchemasQuery/useUISchemasQuery.ts (2)

9-9: LGTM! Good addition of fetch completion tracking.

Adding isFetched helps distinguish between "never fetched" and "currently loading" states, which is important for proper UI state management.


12-12: Investigate and fix the @ts-ignore comment.

The presence of @ts-ignore suggests a potential type issue with the staleTime configuration. Let's understand why this type assertion is needed.

Would you like help investigating and fixing the type issue to remove the @ts-ignore?

apps/kyb-app/src/pages/SignUpPage/SignUpPage.tsx (1)

14-16: Verify handling of loading states

The removal of LoadingScreen and loading-related hooks might affect user experience during data fetching.

Let's verify if loading states are handled elsewhere:

apps/kyb-app/src/common/components/layouts/Signup/Footer.tsx (1)

24-24: Verify HTML sanitization across the application

While DOMPurify is correctly used here, let's ensure rawHtml is properly sanitized throughout its lifecycle.

✅ Verification successful

HTML sanitization is properly implemented across the application

After reviewing the codebase, all instances of dangerouslySetInnerHTML are properly sanitized using DOMPurify:

  • In Footer.tsx: DOMPurify(window).sanitize(rawHtml)
  • In CollectionFlow.tsx: DOMPurify.sanitize(props.options.descriptionRaw)
  • In CompletedScreen.tsx: DOMPurify.sanitize(t('success.header'))
  • In Approved.tsx: DOMPurify.sanitize(t('approved.header'))
  • In Note.tsx: DOMPurify.sanitize(content, { ADD_ATTR: ['target'] })

The rawHtml value is properly typed as an optional string in the settings types and is consistently sanitized before being rendered.

🏁 Scripts executed

The following scripts were executed for the analysis:

Script:

#!/bin/bash
# Search for other instances where rawHtml is used or modified
rg -A 3 "rawHtml" --type typescript

# Search for potential XSS vulnerabilities in HTML handling
ast-grep --pattern 'dangerouslySetInnerHTML = { $$ }'

Length of output: 126


Script:

#!/bin/bash
# Search for rawHtml usage with ripgrep
rg -A 3 "rawHtml"

# Search for dangerouslySetInnerHTML usage
rg -A 3 "dangerouslySetInnerHTML"

# Search for DOMPurify usage
rg -A 3 "DOMPurify"

# Search for HTML sanitization patterns using ast-grep
ast-grep --pattern 'dangerouslySetInnerHTML={{ __html: $$ }}'

Length of output: 8570

🧰 Tools
🪛 Biome

[error] 24-24: Avoid passing content using the dangerouslySetInnerHTML prop.

Setting content using code can expose users to cross-site scripting (XSS) attacks

(lint/security/noDangerouslySetInnerHtml)

apps/kyb-app/src/common/components/layouts/Signup/Logo.tsx (1)

13-13: Well-implemented animation with loading state!

The animation implementation is clean and follows best practices:

  • Uses state to manage loading
  • Smooth scale and fade-in transition
  • Only animates after the image is loaded
  • Good use of initial and animate states

Also applies to: 17-26

apps/kyb-app/src/components/layouts/AppShell/AppShell.tsx (2)

4-6: LGTM: Import organization

The reordering of imports improves code organization by grouping related components together.


14-18: Consider the implications of overflow-y-hidden

While the addition of motion.div aligns with the animation objectives, the overflow-y-hidden class might prevent scrolling in child components. Ensure this doesn't affect the usability of longer forms or content areas.

Consider these alternatives:

  1. Move the overflow control to specific child components that need it
  2. Use overflow-y-auto instead if scrolling might be needed
  3. Add a CSS custom property to make it configurable by consumers
✅ Verification successful

The previous search didn't yield useful results. Let's try a more targeted approach to understand the component structure and potential scroll areas.


Based on the search results, I can now provide a final response:

The overflow-y-hidden is necessary and correctly implemented

The AppShell component serves as the main layout wrapper with h-screen height, and the search results show that child components (like Sidebar) are also designed with h-screen to fill the viewport. The overflow-y-hidden on AppShell is intentional to prevent double scrollbars, as scrolling should be managed by the content areas within the layout rather than the shell itself.

  • The Sidebar component already has its own height management with h-screen
  • No scrolling issues were found in the direct children of AppShell
  • The pattern follows a common layout structure where the main shell controls the viewport and inner components manage their own overflow
🏁 Scripts executed

The following scripts were executed for the analysis:

Script:

#!/bin/bash
# Look for potential scroll containers within AppShell
ast-grep --pattern 'className="$_scroll$_"'
# Check for overflow-y-auto or overflow-y-scroll in related components
rg "overflow-y-(auto|scroll)" "apps/kyb-app/src/components/layouts"

Length of output: 16190


Script:

#!/bin/bash
# Let's check the content of AppShell and its direct children components
rg -A 5 -B 5 "overflow-y" apps/kyb-app/src
# Check if there are any scroll containers or long content areas within the app
rg -A 5 -B 5 "(overflow-y|h-\[|min-h-\[|h-screen)" apps/kyb-app/src/components

Length of output: 10209

apps/kyb-app/src/common/components/layouts/Signup/Signup.tsx (1)

2-2: Let me verify if there's a custom motion package in the repository.

#!/bin/bash
# Check for any motion package in the repository
fd -t f "package.json" -x cat {} \; | jq -c 'select(.name | contains("motion"))'

# Look for any local motion package
fd -t d "motion"

# Check for motion configuration or setup files
fd -t f "motion.config" -o -t f "motion.setup"
apps/kyb-app/src/pages/CollectionFlow/components/pages/Rejected/Rejected.tsx (2)

6-6: ⚠️ Potential issue

Security: Verify authentication protection after HOC removal.

The removal of withSessionProtected HOC might expose sensitive business information to unauthorized users. Ensure that authentication is handled at a higher level (e.g., route protection).

Let's verify the authentication protection:

✅ Verification successful

Based on the shell script results, I can see that authentication is handled at a higher level through multiple mechanisms:

  1. Route-level protection via AuthenticatedLayout component that wraps protected routes and handles auth redirects
  2. Global auth state management through AuthProvider context
  3. Auth checks in data fetching hooks with isAuthenticated guards
  4. Session management with useSession hook

The removal of withSessionProtected HOC from the Rejected component is safe since the authentication is already handled at the router level through the AuthenticatedLayout wrapper component.

Authentication is properly handled at the router level - no security concerns

The component is already protected by being rendered within AuthenticatedLayout which handles authentication checks and redirects unauthorized users. The removal of the redundant HOC wrapper is a valid refactoring.

🏁 Scripts executed

The following scripts were executed for the analysis:

Script:

#!/bin/bash
# Description: Check for route-level or parent-level authentication protection
# Expected: Find router configuration or parent components with auth checks

# Check router configuration for auth protection
rg -A 10 "Rejected.*path|route"

# Look for authentication wrapper components
ast-grep --pattern 'const $_ = ({ children }) => {
  $$$
  auth
  $$$
}'

# Check for authentication hooks usage in parent components
rg -A 5 "useAuth|isAuthenticated|requireAuth"

Length of output: 251051


3-3: LGTM! Verify query implementation details.

The change to useCustomerQuery aligns with modern data fetching patterns.

Let's verify the query implementation:

✅ Verification successful

Based on the shell script results, I can see that the useCustomerQuery hook is properly implemented with appropriate error handling and loading states. The hook returns an object containing:

  • customer: The customer data or null
  • isLoading: Loading state boolean
  • isLoaded: Boolean indicating if the fetch completed
  • error: Error object or null

The implementation follows React Query patterns and includes proper TypeScript typing, error handling, and loading states. The hook is also used consistently across the codebase.

LGTM! The query implementation is well structured and handles all necessary states.

🏁 Scripts executed

The following scripts were executed for the analysis:

Script:

#!/bin/bash
# Description: Check the implementation of useCustomerQuery hook
# Expected: Find the hook implementation and verify error handling, loading states

# Search for the useCustomerQuery implementation
ast-grep --pattern 'export const useCustomerQuery = () => {
  $$$
}'

# Check for error handling patterns in the hook
rg -A 5 'useCustomerQuery.*error'

# Check for loading state handling
rg -A 5 'useCustomerQuery.*loading|isLoading'

Length of output: 321982

apps/kyb-app/src/common/components/layouts/Signup/Background.tsx (1)

18-31: LGTM! Well-implemented animations

The animation implementation using Framer Motion is clean and well-structured. The transition configuration with proper duration and easing looks good.

apps/kyb-app/src/pages/CollectionFlow/components/atoms/LoadingScreen/LoadingScreen.tsx (2)

2-2: Verify the motion import path

The import path motion/react seems unusual. The standard import path for Framer Motion is 'framer-motion'.

✅ Verification successful

Import path motion/react is correct and consistently used

The import path is valid as the codebase:

  • Uses the motion package (version 11.11.17) as shown in package.json
  • Consistently uses motion/react imports across multiple files in the kyb-app
  • Has a separate usage pattern where framer-motion is used exclusively in the backoffice-v2 app
🏁 Scripts executed

The following scripts were executed for the analysis:

Script:

#!/bin/bash
# Check if there's a local alias or if this is a typo
rg -l "from 'motion/react'" 
# Check package.json for the correct dependency
fd -t f "package.json" -x cat {} | jq '.dependencies | select(has("framer-motion") or has("motion"))'

Length of output: 321


Script:

#!/bin/bash
# Check for motion/react imports across the codebase
rg -l "from ['|\"]motion/react['|\"]"

# Check package.json files for dependencies
fd "package.json" -x cat {} | grep -A 5 -B 5 "\"motion\""

# Check for framer-motion imports to see if they coexist
rg -l "from ['|\"]framer-motion['|\"]"

Length of output: 1851


20-31: Verify animation consistency across components

Since this PR involves adding animations to multiple components, let's ensure the animation style (duration, easing, scale) is consistent with other components.

apps/kyb-app/src/router.tsx (1)

38-38: Verify customer data handling in CollectionFlow component

The removal of withCustomer HOC suggests that customer data fetching has been moved into the CollectionFlow component using useCustomerQuery. Please ensure that:

  1. Customer data is properly fetched and handled
  2. Loading and error states are managed appropriately
  3. Authentication checks are still enforced

Let's verify the implementation:

Also applies to: 42-42

✅ Verification successful

Customer data handling has been properly implemented in CollectionFlow

The verification confirms that the CollectionFlow component correctly implements customer data handling:

  • Uses useCustomerQuery hook to fetch customer data
  • Properly handles customer information (logo, display name)
  • Includes loading states (via isLogoLoaded state)
  • Contains appropriate error handling and status checks
  • Maintains authentication flow through the collection flow states

The removal of withCustomer HOC in favor of the useCustomerQuery hook represents a valid architectural improvement.

🏁 Scripts executed

The following scripts were executed for the analysis:

Script:

#!/bin/bash
# Description: Check CollectionFlow implementation for customer data handling

# Check CollectionFlow component implementation
ast-grep --pattern 'export function CollectionFlow() {
  $$$
  useCustomerQuery($$$)
  $$$
}'

# Check for any remaining withCustomer HOC usage
rg "withCustomer" -A 3

Length of output: 118


Script:

#!/bin/bash
# Let's try a broader search for CollectionFlow and customer-related hooks

# Search for CollectionFlow component definition
ast-grep --pattern 'export $_ function CollectionFlow$_() {
  $$$
}'

# Search for CollectionFlow class or const definition
ast-grep --pattern 'export $_ CollectionFlow = $_'

# Search for useCustomer or useCustomerQuery usage
rg "useCustomer" -A 3

# Search for any file containing CollectionFlow
rg "CollectionFlow" -A 5

Length of output: 555


Script:

#!/bin/bash
# Let's try a more focused search within the kyb-app directory

# Find and examine CollectionFlow file
fd CollectionFlow.tsx -t f apps/kyb-app/src

# Search for CollectionFlow in the kyb-app directory
rg -t tsx "CollectionFlow" apps/kyb-app/src

# Look for customer-related hooks in kyb-app
rg -t tsx "useCustomer" apps/kyb-app/src

# Check the imports in router.tsx
cat apps/kyb-app/src/router.tsx

Length of output: 1604


Script:

#!/bin/bash
# Let's examine the CollectionFlow component implementation
cat apps/kyb-app/src/pages/CollectionFlow/CollectionFlow.tsx

# Also check for any HOCs or customer-related hooks in the same directory
cat apps/kyb-app/src/pages/CollectionFlow/index.ts

Length of output: 15197

apps/kyb-app/src/pages/CollectionFlow/components/pages/FailedScreen/FailedScreen.tsx (2)

Line range hint 1-38: LGTM! Clean and well-structured component

The component follows React best practices with:

  • Proper separation of concerns
  • Clean UI organization
  • Responsive design
  • Internationalization support
  • Event tracking

1-1: Verify error handling in the new useCustomerQuery hook

The switch from useCustomer to useCustomerQuery looks good, but we should ensure proper error handling is in place since this is a failure screen.

Let's check the implementation of the new hook:

Also applies to: 11-11

✅ Verification successful

Based on the shell script results, I can now generate the final response as the implementation of useCustomerQuery is clear and shows proper error handling:

Error handling in useCustomerQuery is properly implemented

The hook correctly handles and exposes error states:

  • Returns typed error via error: error ? (error as HTTPError) : null
  • Exposes loading state via isLoading
  • Provides fetch completion status via isLoaded: isFetched
🏁 Scripts executed

The following scripts were executed for the analysis:

Script:

#!/bin/bash
# Description: Verify error handling in useCustomerQuery hook
# Look for error handling patterns in the hook implementation

# Search for the hook implementation
ast-grep --pattern 'export const useCustomerQuery = () => {
  $$$
}'

# Search for error handling patterns in the hook
rg -A 5 "useCustomerQuery" | rg -A 5 "(error|catch|try)"

Length of output: 4332

apps/kyb-app/src/pages/CollectionFlow/components/pages/Approved/Approved.tsx (2)

Line range hint 5-41: Verify customer data retrieval behavior

The switch from useCustomer to useCustomerQuery might affect how customer data is retrieved and cached. Ensure the new implementation maintains the same functionality.

#!/bin/bash
# Description: Check the implementation of both hooks to verify functionality remains the same

# Look for the implementation of both hooks
rg -A 10 "export.*(useCustomer|useCustomerQuery)"

# Check for any references to customer data caching
rg -l "queryClient.*customer" --type ts

8-8: Verify session protection after HOC removal

The removal of withSessionProtected HOC might impact security. Ensure that session protection is handled at a higher level in the component hierarchy or through route guards.

✅ Verification successful

Let me gather more information about the authentication and protection mechanisms in the codebase.


Let me check one more aspect to understand the authentication setup in the app's root and routing.


Let me check one final aspect about the GlobalProviders component since it wraps all routes.


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

Removal of withSessionProtected HOC is safe

The application already implements authentication protection at a higher level through:

  • AccessTokenProvider wrapping all routes via GlobalProviders
  • useAccessToken hook controlling access in the Root component
  • Automatic redirection logic in Root that ensures users are directed to appropriate flows based on their auth state

The component-level HOC was redundant since authentication is properly handled at the application level.

🏁 Scripts executed

The following scripts were executed for the analysis:

Script:

#!/bin/bash
# Description: Check if session protection is handled at a higher level

# Check for session protection in parent components or route configuration
ast-grep --pattern 'withSessionProtected($$$)'

# Look for route protection patterns
rg -l "protect|guard|auth" --type ts

Length of output: 2422


Script:

#!/bin/bash
# Check for route protection patterns in the kyb-app
rg -A 5 "Route.*protect|auth.*route|private.*route" --type ts apps/kyb-app/

# Check for authentication hooks usage
ast-grep --pattern 'useAccessToken($$$)'

# Check for any authentication providers or wrappers
rg -A 5 "AuthProvider|ProtectedRoute" --type ts apps/kyb-app/

Length of output: 648


Script:

#!/bin/bash
# Check the app's root component and main routing setup
rg -A 10 "createBrowserRouter|RouterProvider" apps/kyb-app/
rg -A 10 "const.*routes" apps/kyb-app/

# Check for any auth-related wrappers in the app structure
fd -e tsx -e ts -x cat {} \; apps/kyb-app/src/App.tsx apps/kyb-app/src/main.tsx 2>/dev/null

Length of output: 2321


Script:

#!/bin/bash
# Check GlobalProviders implementation for auth protection
rg -A 10 "GlobalProviders" apps/kyb-app/src/pages/
cat apps/kyb-app/src/pages/GlobalProviders.tsx 2>/dev/null

# Check Root component for auth protection
rg -A 10 "function Root|const Root" apps/kyb-app/src/pages/Root/
cat apps/kyb-app/src/pages/Root/Root.tsx 2>/dev/null

Length of output: 2746

apps/kyb-app/src/pages/CollectionFlow/components/pages/CompletedScreen/CompletedScreen.tsx (2)

5-5: LGTM: Hook replacement looks good

The replacement of the customer hook aligns with the broader refactoring pattern mentioned in the summary.


12-12: Verify session protection after HOC removal

The removal of withSessionProtected HOC might expose this route to unauthorized access. Ensure that session protection is handled at a higher level in the routing hierarchy.

apps/kyb-app/src/components/layouts/AppShell/Navigation.tsx (2)

Line range hint 13-65: Implementation looks good!

The component is well-structured and follows React best practices:

  • Proper use of useCallback for event handlers
  • Clear separation of concerns
  • Proper TypeScript typing
  • Appropriate null checks for customer data
  • Good handling of disabled states

9-9: Verify the implementation of useCustomerQuery hook

The import statement suggests a local hook implementation. Let's verify its existence and implementation.

✅ Verification successful

Let me verify the implementation by checking the actual hook code.


Hook implementation is valid and consistently used across the codebase

The useCustomerQuery hook exists at the correct import path and is properly implemented using React Query. The hook is widely used across multiple components in both kyb-app and backoffice-v2, showing consistent usage patterns. The implementation follows React Query best practices for data fetching.

🏁 Scripts executed

The following scripts were executed for the analysis:

Script:

#!/bin/bash
# Description: Verify the existence and implementation of useCustomerQuery hook

# Check if the hook file exists and show its implementation
rg -p "export.*useCustomerQuery" --type ts

# Look for other usages of this hook to ensure consistent implementation
rg -p "useCustomerQuery" --type ts

Length of output: 6020


Script:

#!/bin/bash
# Show the implementation of useCustomerQuery in kyb-app
cat apps/kyb-app/src/hooks/useCustomerQuery/useCustomerQuery.ts

Length of output: 793

apps/kyb-app/src/common/providers/DependenciesProvider/DependenciesProvider.tsx (2)

4-5: LGTM: Import additions align with new language-aware functionality

The new imports support the internationalization enhancement mentioned in the PR summary.


26-26: Verify consistent usage of language-aware schema fetching

The hook now requires a language parameter. Let's ensure all usages are updated accordingly.

Also applies to: 30-30

✅ Verification successful

All usages of useUISchemasQuery are properly updated with language parameter

The verification shows that all 6 occurrences of useUISchemasQuery across the codebase are correctly using the language parameter:

  • Using language prop directly in most components
  • Using useLanguage() hook in useUIOptionsRedirect
  • Using appLanguage in useAppExit
🏁 Scripts executed

The following scripts were executed for the analysis:

Script:

#!/bin/bash
# Search for all useUISchemasQuery usages to verify they include the language parameter
ast-grep --pattern 'useUISchemasQuery($$$)'

Length of output: 783

apps/kyb-app/package.json (1)

20-20: Verify @ballerine/ui version

The package has been restored with a pinned version. Let's verify if this is the latest compatible version.

✅ Verification successful

@ballerine/ui version is consistent with other packages

The version 0.5.44 is currently used across multiple applications in the repository, with some using the exact version and others using the caret range (^0.5.44). The pinned version in kyb-app aligns with the ecosystem.

🏁 Scripts executed

The following scripts were executed for the analysis:

Script:

#!/bin/bash
# Description: Check the latest version of @ballerine/ui package
# Note: Using GitHub API since this appears to be an internal package

gh api \
  -H "Accept: application/vnd.github+json" \
  repos/ballerine-io/ballerine/contents/packages/ui/package.json | \
  jq -r '.content' | base64 -d | jq '.version'

Length of output: 229


Script:

#!/bin/bash
# Let's try a different approach using ripgrep to find references to @ballerine/ui
# in package.json files across the repository

rg -U "@ballerine/ui.*version|@ballerine/ui.*\":" --type json

Length of output: 324

apps/kyb-app/src/hooks/useUIOptionsRedirect/useUIOptionsRedirect.unit.test.ts (1)

53-58: Consider adding test cases for isLoaded: false state.

The mock setup includes isLoaded: true, but there are no test cases verifying the behavior when isLoaded is false. This could miss potential edge cases in the redirection logic.

Let's verify if the useUIOptionsRedirect hook depends on the isLoaded state:

Consider adding a test case like:

it('should not redirect when data is not loaded', () => {
  vi.mocked(useUISchemasQuery).mockReturnValue({
    data: mockData as UISchema,
    isLoading: false,
    error: null,
    isLoaded: false,
  });

  const originalHref = window.location.href;
  renderHook(() => useUIOptionsRedirect('success'));

  expect(window.location.href).toBe(originalHref);
});
apps/kyb-app/src/pages/CollectionFlow/CollectionFlow.tsx (3)

69-69: Component and hook usage appear correct

The CollectionFlow component definition and the usage of the useCustomerQuery hook are implemented correctly.

Also applies to: 73-73


86-87: Logic for determining revision state is appropriate

The check for CollectionFlowStatusesEnum.revision using getCollectionFlowState is correctly implemented.


24-24: Ensure consistent use of useCustomerQuery hook

Please verify that all instances of the old useCustomer hook have been replaced with the new useCustomerQuery hook throughout the codebase to prevent inconsistencies.

Run the following script to find any remaining usages of useCustomer:

✅ Verification successful

No instances of old useCustomer hook found

The search results show that all instances in the codebase are using the new useCustomerQuery hook. The old useCustomer hook is not present anywhere. The only other "useCustomer" references found are for a different decorator UseCustomerAuthGuard in the workflows service, which is unrelated to this hook migration.

🏁 Scripts executed

The following scripts were executed for the analysis:

Script:

#!/bin/bash
# Description: Find all usages of the 'useCustomer' hook.

rg 'useCustomer\('

Length of output: 20


Script:

#!/bin/bash
# Let's try a more comprehensive search for any references to useCustomer
rg -i "usecustomer"

# Also search for the new hook to understand its usage
rg -i "usecustomerquery"

# Let's also check the imports specifically
rg "import.*useCustomer"

Length of output: 24558

apps/kyb-app/package.json Outdated Show resolved Hide resolved
Copy link

changeset-bot bot commented Nov 18, 2024

⚠️ No Changeset found

Latest commit: cd07107

Merging this PR will not cause a version bump for any packages. If these changes should not result in a new version, you're good to go. If these changes should result in a version bump, you need to add a changeset.

This PR includes no changesets

When changesets are added to this PR, you'll see the packages that this PR includes changesets for and the associated semver types

Click here to learn what changesets are, and how to add one.

Click here if you're a maintainer who wants to add a changeset to this PR

Copy link
Contributor

@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 (2)
apps/kyb-app/src/pages/SignUpPage/components/SignUpForm/components/Submit/Submit.tsx (2)

34-103: Optimize nested AnimatePresence components for better performance

There are multiple nested AnimatePresence components within the render method. While nesting is sometimes necessary, excessive nesting can impact performance and make the component harder to maintain. Review the structure to see if some AnimatePresence components can be combined or removed.


20-28: Add dependencies to the useCallback hook

The onClick function uses setIsSubmitted, setIsLoading, and setIsSaved, which are stable since they're from useState. However, if any props or external variables are used inside onClick in the future, forgetting to add them to the dependency array may cause bugs.

Consider adding dependencies to the useCallback hook for future-proofing:

- }, []);
+ }, [setIsSubmitted, setIsLoading, setIsSaved]);
📜 Review details

Configuration used: CodeRabbit UI
Review profile: CHILL

📥 Commits

Reviewing files that changed from the base of the PR and between 3c3b21d and ac4bbb0.

📒 Files selected for processing (1)
  • apps/kyb-app/src/pages/SignUpPage/components/SignUpForm/components/Submit/Submit.tsx (1 hunks)
🔇 Additional comments (1)
apps/kyb-app/src/pages/SignUpPage/components/SignUpForm/components/Submit/Submit.tsx (1)

65-93: Ensure all possible states are accounted for in conditional rendering

The icon rendering inside the Chip component depends on isLoading and isSaved states. Ensure that all possible combinations of these states are handled to prevent the icon from not rendering or causing UI glitches.

Please verify that the conditions cover all expected states and the UI behaves as intended during transitions.

Copy link
Contributor

@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: 9

🧹 Outside diff range and nitpick comments (19)
apps/kyb-app/src/common/providers/DependenciesProvider/types.ts (1)

1-4: Consider enhancing error handling and cancellation support.

The interface could benefit from additional features:

  1. Error handling for failed refetch operations
  2. Cancellation mechanism for long-running refetch operations
  3. Type information for the refetched dependencies

Consider extending the interface like this:

 export interface IDependenciesContext {
-  refetchAll: () => Promise<void>;
+  refetchAll: () => Promise<void> & { cancel: () => void };
   isLoading: boolean;
+  error: Error | null;
+  dependencies: {
+    // Add specific dependency types here
+    [key: string]: unknown;
+  };
 }
apps/kyb-app/src/pages/SignUpPage/components/SignUpForm/components/SignUpFormProvider/SignUpFormProvider.tsx (1)

4-10: Consider adding readonly modifiers for better type safety.

While the interface is well-structured, adding readonly modifiers would prevent accidental mutations and improve type safety.

Here's the suggested improvement:

 interface ISignUpFormProviderProps {
-  children: React.ReactNode;
-  context: {
-    isLoading: boolean;
-    isSuccess: boolean;
-  };
+  readonly children: React.ReactNode;
+  readonly context: {
+    readonly isLoading: boolean;
+    readonly isSuccess: boolean;
+  };
 }
apps/kyb-app/src/pages/CollectionFlow/components/organisms/AnimatedUIRenderer.tsx (2)

6-10: Consider adding prop validation

While the typing is good, consider adding runtime prop validation using PropTypes or implementing validation checks for the currentPage prop, especially since this component is part of a critical flow.

+import PropTypes from 'prop-types';

 export const AnimatedUIRenderer: FunctionComponent<UIRendererProps & { currentPage: UIPage }> = ({
   schema,
   elements,
   currentPage,
 }) => {
+  if (!currentPage?.stateName) {
+    throw new Error('AnimatedUIRenderer: currentPage.stateName is required');
+  }

11-21: Consider enhancing animation configuration

The animation implementation could be improved in several ways:

  1. Extract animation values as constants for reusability
  2. Add exit animations for smoother transitions
  3. Consider making duration configurable via props
+const ANIMATION_CONFIG = {
+  initial: { x: '100%', opacity: 0 },
+  animate: { x: 0, opacity: 1 },
+  exit: { x: '-100%', opacity: 0 },
+  transition: { duration: 0.4 },
+} as const;

 return (
   <motion.div
-    initial={{ x: '100%', opacity: 0 }}
-    animate={{ x: 0, opacity: 1 }}
-    transition={{ duration: 0.4 }}
+    {...ANIMATION_CONFIG}
     key={currentPage.stateName}
   >
apps/kyb-app/src/hooks/useFlowContextQuery/useFlowContextQuery.ts (1)

Line range hint 11-15: Clean up TypeScript configuration

The @ts-ignore comment and as const type assertion suggest underlying type issues that should be addressed:

  1. The staleTime: Infinity type issue can be resolved by properly typing the query options
  2. Consider removing the type assertion and fixing the underlying type issue

Consider this improvement:

  const { data, isLoading, isFetched, error, refetch } = useQuery({
    ...collectionFlowQuerykeys.getContext(endUser?.id ?? null),
-   // @ts-ignore
-   staleTime: Infinity as const,
+   staleTime: Infinity,
    enabled: !!accessToken,
  });
apps/kyb-app/src/pages/SignUpPage/SignUpPage.tsx (1)

27-27: Enhance animation implementation

The current animation configuration could be improved:

  1. Add entry animation for a smoother user experience
  2. Consider reducing the transition duration (0.7s is quite long)
  3. Consider using a less dramatic exit animation

Consider applying these changes:

- <motion.div className="w-full" exit={{ x: '-100%', opacity: 0 }} transition={{ duration: 0.7 }}>
+ <motion.div
+   className="w-full"
+   initial={{ opacity: 0, y: 20 }}
+   animate={{ opacity: 1, y: 0 }}
+   exit={{ opacity: 0, y: 20 }}
+   transition={{ duration: 0.3 }}
+ >
apps/kyb-app/src/common/components/layouts/Signup/Background.tsx (1)

23-29: Make animation configuration customizable

Consider making the animation parameters configurable through props to allow for different animation styles across the application.

Apply this diff:

 interface IBackgroundProps {
   imageSrc?: string;
   styles?: React.CSSProperties;
+  animation?: {
+    initial?: Record<string, number>;
+    animate?: Record<string, number>;
+    duration?: number;
+    ease?: string;
+  };
 }

 export const Background: FunctionComponent<IBackgroundProps> = props => {
   const { themeParams } = useSignupLayout();
-  const { imageSrc, styles } = { ...props, ...themeParams?.background };
+  const { imageSrc, styles, animation = {} } = { ...props, ...themeParams?.background };
+  const {
+    initial = { opacity: 0, scale: 1.2 },
+    animate = { opacity: 1, scale: 1 },
+    duration = 0.6,
+    ease = 'easeOut',
+  } = animation;

   // ... rest of the component

         initial={{ opacity: 0, scale: 1.2 }}
-        animate={isLoaded ? { opacity: 1, scale: 1 } : { opacity: 0, scale: 1.2 }}
+        animate={isLoaded ? animate : initial}
         exit={{ opacity: 0 }}
         transition={{
-          duration: 0.6,
-          ease: 'easeOut',
+          duration,
+          ease,
         }}
apps/kyb-app/src/router.tsx (2)

39-44: Consider consolidating duplicate routes

The root path ('') and collection-flow path both render the same CollectionFlow component. This duplication could lead to maintenance issues and confusion.

Consider:

  1. Using only one route and redirecting from the other, or
  2. Documenting the purpose of having two identical routes if there's a specific requirement

63-63: Implement 404 page handling

The TODO comment indicates missing 404 page handling. This is a basic requirement for proper routing and user experience.

Would you like me to help create a basic 404 page component and add it to the router configuration?

apps/kyb-app/src/common/providers/DependenciesProvider/DependenciesProvider.tsx (1)

42-46: Consider adding cleanup for loading state

While the loading state management is improved, consider adding a cleanup function to reset isInitialLoading when the component unmounts.

 useEffect(() => {
   if (isLoading) return;

   setIsInitialLoading(false);
+  return () => {
+    setIsInitialLoading(true);
+  };
 }, [isLoading]);
apps/kyb-app/src/pages/CollectionFlow/CollectionFlow.tsx (2)

225-301: Consider optimizing nested animations for performance

While the sequential animations create a nice visual effect, having multiple nested motion.div elements with individual animations might impact performance, especially on lower-end devices.

Consider consolidating animations or using CSS transitions for simpler animations:

  1. Use CSS transitions for opacity changes:
- <motion.div
-   className="flex h-full flex-col"
-   initial={{ opacity: 0 }}
-   animate={{ opacity: 1 }}
-   transition={{ duration: 0.5, delay: 0.2 }}
- >
+ <div 
+   className="flex h-full flex-col transition-opacity duration-500 delay-200"
+   style={{ opacity: isVisible ? 1 : 0 }}
+ >
  1. Or use Framer Motion's staggerChildren for better performance:
 <motion.div
   className="flex h-full flex-col"
+  variants={{
+    hidden: { opacity: 0 },
+    show: {
+      opacity: 1,
+      transition: {
+        staggerChildren: 0.2
+      }
+    }
+  }}
+  initial="hidden"
+  animate="show"
 >

167-167: Remove debug console.log statements

Debug console.log statements should be removed before merging to production.

Apply this diff to remove the debug statements:

- console.log({ collectionFlow });
  
  if (collectionFlow) {
    const steps = collectionFlow?.steps || [];

    const isAnyStepCompleted = steps.some(step => step.isCompleted);

-   console.log('Collection flow touched, changing state to inprogress');
    setCollectionFlowStatus(context, CollectionFlowStatusesEnum.inprogress);

Also applies to: 182-182

apps/kyb-app/src/pages/SignUpPage/components/SignUpForm/SignUpForm.tsx (3)

89-89: Avoid using as any to maintain type safety

Using as any bypasses TypeScript's type checking, which can lead to potential runtime errors.

Consider properly typing the layouts object to match the expected type of the layouts prop in DynamicForm. You can define an interface or type that reflects the structure of layouts to ensure type safety.


31-32: Simplify ref usage for current values

The usage of useRefValue and storing isLoading in a ref may not be necessary.

Since accessToken and isLoading are already reactive, you might not need to store them in refs. Directly using these values can simplify the code:

- const prevIsLoadingRef = useRef(isLoading);
- const accessTokenRef = useRefValue(accessToken);

If you need to track the previous value of isLoading, you can use a state variable or the usePrevious hook pattern.


48-51: Initialize signupState with a type-safe interface

Defining a type or interface for signupState enhances type safety and code readability.

Consider defining an interface for signupState:

interface SignupState {
  isLoading: boolean;
  isSuccess: boolean;
}

const [signupState, setSignupState] = useState<SignupState>({
  isLoading: false,
  isSuccess: false,
});
apps/kyb-app/src/pages/SignUpPage/components/SignUpForm/components/Submit/Submit.tsx (4)

14-14: Remove unused import of 'useTranslation'

The useTranslation hook is imported but not used within the component. Unused imports can lead to unnecessary code bloat and potential confusion.

Apply this diff to remove the unused import:

-import { useTranslation } from 'react-i18next';

37-37: Ensure submit text is correctly displayed

The submit button label uses {themeParams?.form?.submitText || submitText}. If both themeParams?.form?.submitText and submitText are undefined or empty, the button may not display any text. Consider providing a default fallback text to ensure the button always displays meaningful text.

Apply this diff to include a default text:

-              {themeParams?.form?.submitText || submitText}
+              {themeParams?.form?.submitText || submitText || 'Submit'}

23-90: Simplify nested conditional rendering for better readability

The component contains multiple nested AnimatePresence and motion.div components with complex conditional logic. This may affect readability and maintainability. Consider refactoring the code to simplify the conditional rendering.

For example, extract the repeated animation configurations and conditional blocks into separate components or functions. This can make the code cleaner and easier to understand.


75-76: Use theme variables for colors to maintain consistency

Hardcoding colors using hex values like color="#fff" and bg-[#00BD59] may lead to inconsistencies across the application. It's better to use theme variables or CSS classes that reference a centralized theme.

Consider updating the code to use theme colors:

-                            color="#fff"
-                            className="flex h-3 w-3 items-center justify-center rounded-full bg-[#00BD59]"
+                            color="var(--color-text-inverse)"
+                            className="flex h-3 w-3 items-center justify-center rounded-full bg-success"

Ensure that var(--color-text-inverse) and bg-success are defined in your theme.

📜 Review details

Configuration used: CodeRabbit UI
Review profile: CHILL

📥 Commits

Reviewing files that changed from the base of the PR and between ac4bbb0 and 94ad1ec.

📒 Files selected for processing (26)
  • apps/kyb-app/src/common/components/layouts/Signup/Background.tsx (2 hunks)
  • apps/kyb-app/src/common/providers/DependenciesProvider/DependenciesProvider.tsx (4 hunks)
  • apps/kyb-app/src/common/providers/DependenciesProvider/dependencies-context.ts (1 hunks)
  • apps/kyb-app/src/common/providers/DependenciesProvider/hooks/useDependencies/index.ts (1 hunks)
  • apps/kyb-app/src/common/providers/DependenciesProvider/hooks/useDependencies/useDependencies.ts (1 hunks)
  • apps/kyb-app/src/common/providers/DependenciesProvider/index.ts (1 hunks)
  • apps/kyb-app/src/common/providers/DependenciesProvider/types.ts (1 hunks)
  • apps/kyb-app/src/components/layouts/AppShell/AppShell.tsx (1 hunks)
  • apps/kyb-app/src/hooks/useCustomerQuery/useCustomerQuery.ts (1 hunks)
  • apps/kyb-app/src/hooks/useFlowContextQuery/useFlowContextQuery.ts (1 hunks)
  • apps/kyb-app/src/hooks/useUIOptionsRedirect/useUIOptionsRedirect.unit.test.ts (4 hunks)
  • apps/kyb-app/src/hooks/useUISchemasQuery/useUISchemasQuery.ts (1 hunks)
  • apps/kyb-app/src/pages/CollectionFlow/CollectionFlow.tsx (5 hunks)
  • apps/kyb-app/src/pages/CollectionFlow/components/organisms/AnimatedUIRenderer.tsx (1 hunks)
  • apps/kyb-app/src/pages/Root/Root.tsx (1 hunks)
  • apps/kyb-app/src/pages/SignUpPage/SignUpPage.tsx (1 hunks)
  • apps/kyb-app/src/pages/SignUpPage/components/SignUpForm/SignUpForm.tsx (3 hunks)
  • apps/kyb-app/src/pages/SignUpPage/components/SignUpForm/components/SignUpFormProvider/SignUpFormProvider.tsx (1 hunks)
  • apps/kyb-app/src/pages/SignUpPage/components/SignUpForm/components/SignUpFormProvider/hooks/useSignupForm/index.ts (1 hunks)
  • apps/kyb-app/src/pages/SignUpPage/components/SignUpForm/components/SignUpFormProvider/hooks/useSignupForm/useSignupForm.ts (1 hunks)
  • apps/kyb-app/src/pages/SignUpPage/components/SignUpForm/components/SignUpFormProvider/index.ts (1 hunks)
  • apps/kyb-app/src/pages/SignUpPage/components/SignUpForm/components/SignUpFormProvider/signup-form-context.ts (1 hunks)
  • apps/kyb-app/src/pages/SignUpPage/components/SignUpForm/components/Submit/Submit.tsx (1 hunks)
  • apps/kyb-app/src/pages/SignUpPage/components/SignUpForm/hooks/useCreateEndUserMutation/useCreateEndUserMutation.ts (0 hunks)
  • apps/kyb-app/src/router.tsx (2 hunks)
  • services/workflows-service/prisma/data-migrations (1 hunks)
💤 Files with no reviewable changes (1)
  • apps/kyb-app/src/pages/SignUpPage/components/SignUpForm/hooks/useCreateEndUserMutation/useCreateEndUserMutation.ts
✅ Files skipped from review due to trivial changes (9)
  • apps/kyb-app/src/common/providers/DependenciesProvider/dependencies-context.ts
  • apps/kyb-app/src/common/providers/DependenciesProvider/hooks/useDependencies/index.ts
  • apps/kyb-app/src/common/providers/DependenciesProvider/hooks/useDependencies/useDependencies.ts
  • apps/kyb-app/src/common/providers/DependenciesProvider/index.ts
  • apps/kyb-app/src/pages/SignUpPage/components/SignUpForm/components/SignUpFormProvider/hooks/useSignupForm/index.ts
  • apps/kyb-app/src/pages/SignUpPage/components/SignUpForm/components/SignUpFormProvider/hooks/useSignupForm/useSignupForm.ts
  • apps/kyb-app/src/pages/SignUpPage/components/SignUpForm/components/SignUpFormProvider/index.ts
  • apps/kyb-app/src/pages/SignUpPage/components/SignUpForm/components/SignUpFormProvider/signup-form-context.ts
  • services/workflows-service/prisma/data-migrations
🚧 Files skipped from review as they are similar to previous changes (4)
  • apps/kyb-app/src/components/layouts/AppShell/AppShell.tsx
  • apps/kyb-app/src/hooks/useUIOptionsRedirect/useUIOptionsRedirect.unit.test.ts
  • apps/kyb-app/src/hooks/useUISchemasQuery/useUISchemasQuery.ts
  • apps/kyb-app/src/pages/Root/Root.tsx
🔇 Additional comments (17)
apps/kyb-app/src/common/providers/DependenciesProvider/types.ts (1)

1-4: LGTM! Clean and focused interface definition.

The interface follows TypeScript naming conventions and provides a clear contract for dependency management with async operation support and loading state tracking.

apps/kyb-app/src/pages/SignUpPage/components/SignUpForm/components/SignUpFormProvider/SignUpFormProvider.tsx (2)

1-2: LGTM! Clean and well-organized imports.

The imports are properly structured with named imports and good separation of concerns.


12-17: Clean implementation, but verify animation requirements.

The component implementation is clean and follows best practices. However, since this PR is focused on adding animations, we should verify if this component needs any animation-related functionality.

Let's check if other form-related components have animation implementations that might be relevant here:

✅ Verification successful

No animation requirements needed for SignUpFormProvider component

The SignUpFormProvider component serves as a context provider wrapper and doesn't need animation functionality. The existing animations in the SignUp flow are appropriately placed in:

  • SignUpPage.tsx (page-level transitions)
  • Submit.tsx (form submission animations)

The context provider's role is strictly to manage state distribution, and adding animations here would violate its single responsibility principle.

🏁 Scripts executed

The following scripts were executed for the analysis:

Script:

#!/bin/bash
# Description: Check for animation-related code in nearby components
# Look for imports of 'motion' library and animation-related code in the SignUp components

# Search for motion imports in nearby files
echo "Checking for motion imports in SignUp components..."
rg -l "import.*motion" "apps/kyb-app/src/pages/SignUpPage"

# Search for animation-related properties
echo "Checking for animation properties..."
rg "animate|transition|motion" "apps/kyb-app/src/pages/SignUpPage"

Length of output: 3182

apps/kyb-app/src/pages/CollectionFlow/components/organisms/AnimatedUIRenderer.tsx (1)

23-23: LGTM!

Setting the displayName is good practice for debugging and development tools.

apps/kyb-app/src/hooks/useCustomerQuery/useCustomerQuery.ts (2)

19-19: Improved loading state logic

The loading state now correctly differentiates between initial loading and subsequent refetches by combining isLoading with !isFetched. This prevents showing loading indicators during refetches when data is already available.


11-11: Verify the refetch functionality usage

The addition of the refetch function enhances the hook's capabilities for manual data refresh. Let's verify where this new functionality is being used to ensure proper implementation.

Also applies to: 22-22

✅ Verification successful

The refetch functionality is properly implemented and used

The refetch functionality from useCustomerQuery is correctly exposed and integrated into the application's dependency management system. It's specifically used in the DependenciesProvider component as part of the refetchAll mechanism, which handles data refresh across the application. The implementation follows the standard React Query patterns and is properly typed and exported.

  • apps/kyb-app/src/common/providers/DependenciesProvider/DependenciesProvider.tsx: Uses refetch through useEndUserQuery to trigger a refresh of all dependencies
  • apps/kyb-app/src/hooks/useCustomerQuery/useCustomerQuery.ts: Correctly exposes the refetch function from React Query
🏁 Scripts executed

The following scripts were executed for the analysis:

Script:

#!/bin/bash
# Description: Find components using the refetch functionality
# Expected: Components that need manual refresh capabilities

# Search for useCustomerQuery hook usage with refetch
rg -A 5 "useCustomerQuery.*refetch" apps/kyb-app/src

# Search for direct refetch calls
rg "refetch\(\)" apps/kyb-app/src

Length of output: 90


Script:

#!/bin/bash
# Let's first find all files that import useCustomerQuery
rg -l "useCustomerQuery" apps/kyb-app/src

# Then let's look for any destructured usage of refetch from useCustomerQuery
rg -A 3 "const.*\{.*refetch.*\}.*=.*useCustomerQuery" apps/kyb-app/src

# Also search for any usage of refetch in the same files
rg "refetch" apps/kyb-app/src

# Let's also check the implementation of useCustomerQuery itself
cat apps/kyb-app/src/hooks/useCustomerQuery/useCustomerQuery.ts

Length of output: 4263

apps/kyb-app/src/hooks/useFlowContextQuery/useFlowContextQuery.ts (2)

20-21: LGTM! Improved loading state management

The changes improve the accuracy of loading states by:

  • Preventing false loading states when data is already fetched
  • Adding an explicit isLoaded state for better state management

Line range hint 1-24: Verify error handling in consuming components

The hook now has more granular loading states which might affect components using this hook.

Let's verify the usage of this hook's states in components:

✅ Verification successful

Based on the search results, I can see that the hook useFlowContextQuery is used consistently across the codebase with the loading states isLoading and isLoaded. The changes in the review comment align with the existing pattern where loading states are handled granularly with both isLoading and isLoaded flags.

Loading states are handled consistently across components

The changes to the loading states in useFlowContextQuery follow the established pattern used throughout the codebase where components check both isLoading and isLoaded flags to properly handle loading states. This is evidenced by similar implementations in hooks like:

  • useUISchemasQuery
  • useCustomerQuery
  • useEndUserQuery

The granular loading states allow components to handle different loading scenarios appropriately.

🏁 Scripts executed

The following scripts were executed for the analysis:

Script:

#!/bin/bash
# Description: Find components using useFlowContextQuery to verify they handle the new states correctly

# Search for components using the hook
rg -l "useFlowContextQuery" --type tsx

# Check specific usage of loading states
rg "isLoading|isLoaded" -C 3 $(rg -l "useFlowContextQuery" --type tsx)

Length of output: 392403

apps/kyb-app/src/pages/SignUpPage/SignUpPage.tsx (2)

11-11: Incorrect import path for motion library

This issue was previously identified in an earlier review.


28-38: Clean component structure

The component hierarchy is well-organized and follows a logical structure with clear separation of concerns.

apps/kyb-app/src/common/components/layouts/Signup/Background.tsx (1)

1-1: Fix incorrect import path for Framer Motion

The import path 'motion/react' is incorrect and needs to be updated.

apps/kyb-app/src/common/providers/DependenciesProvider/DependenciesProvider.tsx (3)

26-33: Well-structured dependency queries with proper typing

Good use of the 'satisfies' operator for type safety and proper integration of language parameter for UI schemas.


84-88: Clean rendering logic with proper loading state handling

The implementation properly addresses potential race conditions by using isInitialLoading state. The context provider is correctly set up with memoized values.


21-24: Verify refetchEndUser integration with existing dependencies

The refetchEndUser hook is used to trigger a refetch of all dependencies. Let's verify this behavior.

✅ Verification successful

Based on the search results, I can now provide a final response:

The refetchEndUser integration is correctly implemented

The code shows a proper dependency chain where:

  • useEndUserQuery is the primary query that other queries depend on
  • Other queries like useUISchemasQuery, useFlowContextQuery, and useCustomerQuery use the endUser data and will automatically refetch when endUser changes
  • The refetchAll function in DependenciesProvider correctly uses refetchEndUser which will trigger a cascade of refetches due to the dependency chain
🏁 Scripts executed

The following scripts were executed for the analysis:

Script:

#!/bin/bash
# Search for dependencies that should be refetched
ast-grep --pattern 'useEffect($$$, [$$$refetchEndUser$$$])'

Length of output: 61


Script:

#!/bin/bash
# Let's search for any usage of refetchEndUser in the file
rg "refetchEndUser" -A 5 -B 5 "apps/kyb-app/src/common/providers/DependenciesProvider/DependenciesProvider.tsx"

Length of output: 938


Script:

#!/bin/bash
# Let's check how useEndUserQuery is defined and if it triggers other dependencies
rg "useEndUserQuery" -A 10 -B 10

Length of output: 12223

apps/kyb-app/src/pages/CollectionFlow/CollectionFlow.tsx (1)

23-23: LGTM: Hook replacement aligns with data fetching standardization

The replacement of useCustomer with useCustomerQuery is consistent with the broader refactoring effort to standardize customer data retrieval across the application.

apps/kyb-app/src/pages/SignUpPage/components/SignUpForm/components/Submit/Submit.tsx (2)

41-89: Validate the state transitions for loading and success states

The conditions (isSignupLoading || isSignupSuccess) control the rendering of the Chip component. Ensure that the state transitions between loading and success are handled correctly, and there's no unintended overlap where both states might be true simultaneously.

Review the logic managing isSignupLoading and isSignupSuccess to confirm that they transition as expected. If necessary, adjust the state management to prevent conflicting states.


7-7: ⚠️ Potential issue

Verify the import path for 'motion/react'

The import statement import { AnimatePresence, motion } from 'motion/react'; may be incorrect. Typically, AnimatePresence and motion are imported from framer-motion using import { AnimatePresence, motion } from 'framer-motion';. Please verify that motion/react is the correct module in your project setup.

Run the following script to check for the correct import path and dependencies:

Copy link
Contributor

@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: 1

🧹 Outside diff range and nitpick comments (6)
apps/kyb-app/src/pages/CollectionFlow/hooks/useSignupGuard/useSignupGuard.ts (2)

5-14: Add TypeScript types and return value documentation

The hook implementation would benefit from explicit TypeScript types and return value documentation.

-export const useSignupGuard = () => {
+/**
+ * Guards routes by redirecting to signup page when signup is required
+ * @returns void
+ */
+export const useSignupGuard = (): void => {

10-12: Add error handling for navigation

Consider adding error handling for the navigation operation to gracefully handle potential failures.

 if (isSignupRequired) {
-  navigate('/signup');
+  try {
+    navigate('/signup');
+  } catch (error) {
+    console.error('Failed to navigate to signup page:', error);
+  }
 }
apps/kyb-app/src/pages/SignUpPage/components/SignUpForm/SignUpForm.tsx (1)

75-80: Enhance error handling with specific error messages

The current error handling uses a generic message. Consider extracting and displaying more specific error information to help users understand and resolve issues.

  try {
    await createEndUserRequest(values);
  } catch (error) {
    setSignupState({ isLoading: false, isSuccess: false });
-   toast.error('Failed to create user. Please try again.');
+   const errorMessage = error instanceof Error 
+     ? error.message 
+     : 'Failed to create user. Please try again.';
+   toast.error(errorMessage);
  }
apps/kyb-app/src/pages/CollectionFlow/CollectionFlow.tsx (3)

146-150: Consider animation performance and accessibility

The animation implementation could be improved in several ways:

  1. Extract animation constants for better maintainability
  2. Consider users who prefer reduced motion
  3. Long animation chains with sequential delays might impact performance on slower devices

Consider these improvements:

+ // Create animation constants
+ const ANIMATION_CONFIG = {
+   slideIn: {
+     initial: { x: '100%', opacity: 1 },
+     animate: { x: 0, opacity: 1 },
+     transition: { duration: 0.7 }
+   },
+   fadeIn: {
+     initial: { opacity: 0 },
+     animate: { opacity: 1 },
+     transition: { duration: 0.5 }
+   }
+ };

+ // Add reduced motion support
+ import { useReducedMotion } from 'framer-motion';
+ 
+ const prefersReducedMotion = useReducedMotion();
+ const animationProps = prefersReducedMotion ? {} : ANIMATION_CONFIG.slideIn;

- <motion.div
-   initial={{ x: '100%', opacity: 1 }}
-   animate={{ x: 0, opacity: 1 }}
-   transition={{ duration: 0.7 }}
+ <motion.div {...animationProps}

Also applies to: 229-234, 235-240, 241-246, 252-257, 270-275, 278-282


308-325: Improve debug mode implementation

The current debug mode implementation has several areas for improvement:

  1. Direct localStorage access
  2. No type safety for the debug flag
  3. Inline debug controls

Consider creating a proper debug utilities module:

+ // Create a debug utilities module
+ const DEBUG_FLAGS = {
+   DEV_MODE: 'devmode'
+ } as const;
+ 
+ const useDebugMode = () => {
+   const [isDebugMode, setDebugMode] = useState(() => 
+     localStorage.getItem(DEBUG_FLAGS.DEV_MODE) !== null
+   );
+   
+   return isDebugMode;
+ };
+ 
+ const DebugControls = ({ stateApi }: { stateApi: StateApi }) => {
+   if (!useDebugMode()) return null;
+   
+   return (
+     <div className="flex flex-col gap-4 p-4 bg-gray-100 rounded-md">
+       <h3 className="font-bold">Debug Controls</h3>
+       <div>Current State: {currentPage?.stateName ?? 'Unknown'}</div>
+       <div className="flex gap-4">
+         <button 
+           className="px-2 py-1 bg-blue-500 text-white rounded"
+           onClick={() => stateApi.sendEvent('PREVIOUS')}
+         >
+           Previous
+         </button>
+         <button
+           className="px-2 py-1 bg-blue-500 text-white rounded"
+           onClick={() => stateApi.sendEvent('NEXT')}
+         >
+           Next
+         </button>
+       </div>
+     </div>
+   );
+ };

- {localStorage.getItem('devmode') ? (
-   <div className="flex flex-col gap-4">
-     DEBUG
-     <div>{currentPage ? currentPage.stateName : 'Page not found and state ' + state}</div>
-     <div className="flex gap-4">
-       <button onClick={() => stateApi.sendEvent('PREVIOUS')}>prev</button>
-       <button onClick={() => stateApi.sendEvent('NEXT')}>next</button>
-     </div>
-   </div>
- ) : null}
+ <DebugControls stateApi={stateApi} />

211-215: Simplify screen resolution logic

The current implementation uses multiple separate conditions to determine which screen to show. This could be simplified using a mapping approach.

Consider using a screen resolver pattern:

+ const SCREEN_COMPONENTS = {
+   done: LoadingScreen,
+   completed: CompletedScreen,
+   failed: FailedScreen,
+ } as const;
+ 
+ const resolveScreen = (state: string) => {
+   if (isCompleted(state)) return SCREEN_COMPONENTS.completed;
+   if (isFailed(state)) return SCREEN_COMPONENTS.failed;
+   if (state === 'done') return SCREEN_COMPONENTS.done;
+   return null;
+ };

- if (state === 'done') return <LoadingScreen />;
- if (isCompleted(state)) return <CompletedScreen />;
- if (isFailed(state)) return <FailedScreen />;
+ const ScreenComponent = resolveScreen(state);
+ if (ScreenComponent) return <ScreenComponent />;
📜 Review details

Configuration used: CodeRabbit UI
Review profile: CHILL

📥 Commits

Reviewing files that changed from the base of the PR and between 94ad1ec and 3ef32d2.

📒 Files selected for processing (5)
  • apps/kyb-app/src/pages/CollectionFlow/CollectionFlow.tsx (5 hunks)
  • apps/kyb-app/src/pages/CollectionFlow/hooks/useSignupGuard/index.ts (1 hunks)
  • apps/kyb-app/src/pages/CollectionFlow/hooks/useSignupGuard/useSignupGuard.ts (1 hunks)
  • apps/kyb-app/src/pages/Root/Root.tsx (1 hunks)
  • apps/kyb-app/src/pages/SignUpPage/components/SignUpForm/SignUpForm.tsx (3 hunks)
✅ Files skipped from review due to trivial changes (1)
  • apps/kyb-app/src/pages/CollectionFlow/hooks/useSignupGuard/index.ts
🚧 Files skipped from review as they are similar to previous changes (1)
  • apps/kyb-app/src/pages/Root/Root.tsx
🔇 Additional comments (5)
apps/kyb-app/src/pages/CollectionFlow/hooks/useSignupGuard/useSignupGuard.ts (1)

1-3: LGTM: Clean and focused imports

The imports are well-organized and only include what's necessary for the hook's functionality.

apps/kyb-app/src/pages/SignUpPage/components/SignUpForm/SignUpForm.tsx (3)

26-32: LGTM! Clean hook setup and state management.

The component properly initializes necessary hooks and refs for tracking state changes.


89-98: LGTM! Clean component rendering with proper context usage.

The component properly uses context and handles form state management.


34-52: ⚠️ Potential issue

Critical: Navigation logic needs refactoring

The current implementation still uses setTimeout for navigation, which can lead to timing issues and race conditions. This was previously flagged in earlier reviews.

Consider implementing the navigation logic directly in the handleSubmit function after successful signup, as suggested in the previous review:

- useEffect(() => {
-   if (prevIsLoadingRef.current !== isLoading && !isSignupRequired) {
-     setTimeout(() => {
-       setSignupState({ isLoading: false, isSuccess: true });
-     }, 1500);
-     setTimeout(() => {
-       navigate(`/collection-flow?token=${accessTokenRef.current}`);
-     }, 3000);
-     return;
-   }
-   if (!isSignupRequired) {
-     navigate(`/collection-flow?token=${accessTokenRef.current}`);
-   }
-   prevIsLoadingRef.current = isLoading;
- }, [isLoading, isSignupRequired, accessTokenRef, navigate]);
apps/kyb-app/src/pages/CollectionFlow/CollectionFlow.tsx (1)

70-71: LGTM: Good architectural shift from HOC to hooks

The change from withSessionProtected HOC to useSignupGuard hook follows modern React best practices for better composition and reusability.

Copy link
Contributor

@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 (7)
apps/kyb-app/src/common/errors/server-not-available.ts (1)

1-5: Consider enhancing the error class with additional features.

Consider these improvements to make the error more robust and maintainable:

+/** Error thrown when the server is not available or cannot be reached */
 export class ServerNotAvailableError extends Error {
   constructor() {
     super('Server not available');
+    this.name = 'ServerNotAvailableError';
+    this.code = 'SERVER_NOT_AVAILABLE';
   }
+
+  /** Unique error code for programmatic handling */
+  readonly code: string;
 }

This enhancement:

  • Adds JSDoc documentation for better IDE support
  • Sets the error name to match the class name
  • Includes an error code for programmatic error handling
apps/kyb-app/src/common/components/organisms/ErrorScreen/ServerNotAvailable.tsx (3)

6-23: Consider internationalizing the error messages.

To support multiple languages and maintain consistency, consider moving the error messages to a translation file.

+ import { useTranslation } from 'react-i18next';
  
  export const ServerNotAvailableErrorScreen = () => {
+   const { t } = useTranslation();
    return (
      <AppErrorScreen
-       title="Server Not Available"
+       title={t('errors.server.title')}
        description={
          <div className="!text-muted-foreground flex flex-col gap-1">
-           <p>We are unable to connect to our servers at this time.</p>
+           <p>{t('errors.server.connection')}</p>
            // ... similar changes for other text

8-23: Enhance accessibility for error messages.

Add proper ARIA attributes and semantic HTML to improve screen reader support.

- <div className="!text-muted-foreground flex flex-col gap-1">
+ <div 
+   className="!text-muted-foreground flex flex-col gap-1"
+   role="alert"
+   aria-live="polite"
+ >
-   <ul>
+   <ul aria-label="Troubleshooting steps">

3-27: Consider adding animations as mentioned in PR objectives.

Since this PR includes animation updates, consider adding subtle animations to improve the user experience when the error screen appears.

+ import { motion } from 'framer-motion';

  export const ServerNotAvailableErrorScreen = () => {
    return (
+     <motion.div
+       initial={{ opacity: 0, y: 20 }}
+       animate={{ opacity: 1, y: 0 }}
+       transition={{ duration: 0.5 }}
+     >
        <AppErrorScreen
          // ... existing props
        />
+     </motion.div>
    );
  };
apps/kyb-app/src/common/components/organisms/ErrorScreen/ErrorScreen.tsx (1)

21-23: Consider adding error details to the ServerNotAvailableErrorScreen.

The error handling logic is correct, but consider passing the error details to ServerNotAvailableErrorScreen for more detailed error reporting or logging purposes.

- return <ServerNotAvailableErrorScreen />;
+ return <ServerNotAvailableErrorScreen error={error} />;
apps/kyb-app/src/common/providers/DependenciesProvider/DependenciesProvider.tsx (1)

53-55: Consider adding documentation for the refetch behavior.

Since the comment indicates this triggers a refetch of all dependencies, it would be helpful to document the cascade effect in JSDoc format.

+/**
+ * Triggers a refetch of the end user data, which cascades to refresh all dependencies.
+ * @returns Promise<void>
+ */
 const refetchAll = useCallback(async () => {
   await refetchEndUser(); // It will trigger refetch of all dependencies
 }, [refetchEndUser]);
apps/kyb-app/src/common/utils/request.ts (1)

10-10: Remove duplicate status code in 'statusCodes' array

The statusCodes array contains a duplicate entry for 404. Consider removing the duplicate for clarity.

Apply this diff to remove the duplicate:

     statusCodes: [500, 408, 404, 404, 403, 401, 0],
+    statusCodes: [500, 408, 404, 403, 401, 0],
📜 Review details

Configuration used: CodeRabbit UI
Review profile: CHILL

📥 Commits

Reviewing files that changed from the base of the PR and between 3ef32d2 and f917e6b.

📒 Files selected for processing (6)
  • apps/kyb-app/src/common/components/organisms/ErrorScreen/ErrorScreen.tsx (2 hunks)
  • apps/kyb-app/src/common/components/organisms/ErrorScreen/ServerNotAvailable.tsx (1 hunks)
  • apps/kyb-app/src/common/errors/server-not-available.ts (1 hunks)
  • apps/kyb-app/src/common/providers/DependenciesProvider/DependenciesProvider.tsx (4 hunks)
  • apps/kyb-app/src/common/utils/request.ts (4 hunks)
  • apps/kyb-app/src/router.tsx (1 hunks)
🚧 Files skipped from review as they are similar to previous changes (1)
  • apps/kyb-app/src/router.tsx
🔇 Additional comments (12)
apps/kyb-app/src/common/errors/server-not-available.ts (1)

1-5: LGTM! Clean and focused implementation.

The error class follows the standard pattern for custom errors and serves its specific purpose well.

apps/kyb-app/src/common/components/organisms/ErrorScreen/ServerNotAvailable.tsx (1)

3-27: LGTM! Clean and well-structured implementation.

The component is well-organized, follows React best practices, and provides clear error messaging with actionable steps.

apps/kyb-app/src/common/components/organisms/ErrorScreen/ErrorScreen.tsx (2)

3-3: LGTM! Imports are well-organized and necessary.

The new imports for server availability error handling are properly placed alongside other error-related imports, maintaining a consistent organization pattern.

Also applies to: 8-8


Line range hint 11-42: Verify error handling hierarchy and consider TypeScript improvements.

The error handling implementation looks good, but consider these improvements:

  1. Verify that the order of error checks is appropriate for your use case (most specific to most general).
  2. Consider adding TypeScript type guards to ensure exhaustive error handling.

Example type improvement:

type AppError = 
  | AccessTokenIsMissingError 
  | InvalidAccessTokenError 
  | ServerNotAvailableError 
  | Error;

// Add type assertion to ensure error is of known type
const error = useRouteError() as AppError;
apps/kyb-app/src/common/providers/DependenciesProvider/DependenciesProvider.tsx (5)

2-2: LGTM! New imports align with the enhanced functionality.

The added imports support the new error handling, language-based UI schemas, and end-user query functionality.

Also applies to: 4-4, 6-7


Line range hint 22-47: Well-implemented loading state management!

The changes effectively address the previous race condition concerns by:

  1. Using isInitialLoading to handle the initial load state
  2. Adding isLoaded check in the loading condition
  3. Properly managing state transitions in the useEffect

57-57: LGTM! Context value is properly memoized.

The context value is correctly constructed and memoized with appropriate dependencies.


63-69: LGTM! Robust server availability check implementation.

The error handling enhancement properly checks for server availability before proceeding with other error checks, providing better error feedback to users.


92-96: LGTM! Clean and efficient rendering logic.

The conditional rendering based on isInitialLoading provides a clear separation between initial load and normal operation states.

apps/kyb-app/src/common/utils/request.ts (3)

14-22: Verify error handling in custom fetch function

In the custom fetch function, the error handling checks for error?.response?.statusCode. Please verify that the error object from the fetch API provides a response property with a statusCode. If not, this condition may not work as intended, and network errors without a response might not be properly handled.


26-28: Appropriate offline handling before requests

The check for navigator.onLine effectively prevents requests when the user is offline by throwing a ServerNotAvailableError. This enhances user experience by providing immediate feedback without waiting for a network timeout.


38-41: Enhanced error handling for server availability

The condition !response || error.name === 'TimeoutError' correctly identifies scenarios where the server is not available or a timeout has occurred, throwing a ServerNotAvailableError. This improves the robustness of error handling in the application.

@chesterkmr chesterkmr force-pushed the illiar/feat/cf-ui-enhancements branch from f917e6b to d572e64 Compare November 20, 2024 11:20
Copy link
Contributor

@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: 1

🧹 Outside diff range and nitpick comments (6)
apps/kyb-app/src/common/utils/request.ts (3)

14-22: Consider a more concise implementation

The custom fetch implementation can be simplified while maintaining the same functionality.

-  fetch: (input, init) => {
-    return fetch(input, init).catch(error => {
-      if (!error?.response?.statusCode) {
-        throw new ServerNotAvailableError();
-      }
-
-      throw error;
-    });
-  },
+  fetch: (input, init) => 
+    fetch(input, init).catch(error => {
+      if (!error?.response?.statusCode) throw new ServerNotAvailableError();
+      throw error;
+    }),

Line range hint 51-74: Consider enhancing Sentry error reporting

While the current implementation captures basic error information, consider these improvements for better error tracking:

  1. Add more specific error grouping:
 scope.setFingerprint([
   request.method,
   request.url,
   String(error.response.status),
+  error.name,
+  error.message,
   getAccessToken() || 'anonymous',
 ]);
  1. Include more context in error capture:
 Sentry.captureException(error, {
   extra: {
-    ErrorMessage: `StatusCode: ${response?.status}, URL:${response?.url}`,
+    statusCode: response?.status,
+    url: response?.url,
+    method: request.method,
     // @ts-ignore
     reqId: response?.headers?.['X-Request-ID'],
     bodyRaw: responseBody,
+    timestamp: new Date().toISOString(),
   },
 });

13-13: Consider making timeout configurable

The hardcoded timeout value of 30 seconds might not be suitable for all environments or scenarios. Consider making this configurable through environment variables or configuration files.

-  timeout: 30_000,
+  timeout: Number(import.meta.env.VITE_API_TIMEOUT) || 30_000,
apps/kyb-app/src/pages/CollectionFlow/CollectionFlow.tsx (3)

229-305: Consider optimizing animation implementation

The current implementation has multiple nested motion components with similar animation properties. Consider:

  1. Extracting common animation configurations into constants
  2. Reducing nesting depth of motion components to optimize performance

Example refactor:

const fadeInAnimation = {
  initial: { opacity: 0 },
  animate: { opacity: 1 },
  transition: { duration: 0.5 }
};

const sidebarAnimation = {
  initial: { opacity: 0, x: -20 },
  animate: { opacity: 1, x: 0 },
  transition: { duration: 0.5 }
};

Then use these configurations:

- <motion.div
-   initial={{ opacity: 0 }}
-   animate={{ opacity: 1 }}
-   transition={{ duration: 0.5, delay: 0.2 }}
+ <motion.div
+   {...fadeInAnimation}
+   transition={{ ...fadeInAnimation.transition, delay: 0.2 }}

309-326: Enhance debug mode implementation

The current debug implementation could be improved for better developer experience:

  1. Consider using environment variables instead of localStorage
  2. Add more structured debug information
  3. Improve debug controls styling

Consider this implementation:

- {localStorage.getItem('devmode') ? (
+ {process.env.NODE_ENV === 'development' && (
-   <div className="flex flex-col gap-4">
+   <div className="flex flex-col gap-4 p-4 bg-gray-100 rounded-md mb-4">
-     DEBUG
+     <h3 className="font-bold text-lg">Debug Panel</h3>
      <div>
+       <span className="font-semibold">Current State: </span>
        {currentPage
          ? currentPage.stateName
          : 'Page not found and state ' + state}
      </div>
-     <div className="flex gap-4">
+     <div className="flex gap-4 mt-2">
        <button
+         className="px-4 py-2 bg-gray-200 rounded hover:bg-gray-300"
          onClick={() => stateApi.sendEvent('PREVIOUS')}
        >
-         prev
+         Previous
        </button>
        <button
+         className="px-4 py-2 bg-gray-200 rounded hover:bg-gray-300"
          onClick={() => stateApi.sendEvent('NEXT')}
        >
-         next
+         Next
        </button>
      </div>
    </div>
  ) : null}

171-171: Remove console.log statement

Console.log statements should be removed from production code. Consider using proper logging service if debugging information is needed in production.

- console.log({ collectionFlow });
📜 Review details

Configuration used: CodeRabbit UI
Review profile: CHILL

📥 Commits

Reviewing files that changed from the base of the PR and between f917e6b and d572e64.

📒 Files selected for processing (7)
  • apps/kyb-app/src/common/components/organisms/ErrorScreen/ErrorScreen.tsx (2 hunks)
  • apps/kyb-app/src/common/components/organisms/ErrorScreen/ServerNotAvailable.tsx (1 hunks)
  • apps/kyb-app/src/common/errors/server-not-available.ts (1 hunks)
  • apps/kyb-app/src/common/providers/DependenciesProvider/DependenciesProvider.tsx (4 hunks)
  • apps/kyb-app/src/common/utils/request.ts (4 hunks)
  • apps/kyb-app/src/pages/CollectionFlow/CollectionFlow.tsx (5 hunks)
  • apps/kyb-app/src/router.tsx (1 hunks)
🚧 Files skipped from review as they are similar to previous changes (4)
  • apps/kyb-app/src/common/components/organisms/ErrorScreen/ErrorScreen.tsx
  • apps/kyb-app/src/common/components/organisms/ErrorScreen/ServerNotAvailable.tsx
  • apps/kyb-app/src/common/errors/server-not-available.ts
  • apps/kyb-app/src/router.tsx
🔇 Additional comments (7)
apps/kyb-app/src/common/utils/request.ts (1)

26-28: LGTM! Good offline detection implementation

The offline check using navigator.onLine is a good practice for early detection of network issues.

apps/kyb-app/src/common/providers/DependenciesProvider/DependenciesProvider.tsx (5)

22-35: LGTM! Well-structured dependency management

The implementation properly handles state management with TypeScript type safety and clear separation of concerns.


43-47: LGTM! Clean loading state management

The loading state management is well-implemented with clear transitions and no risk of race conditions.


53-57: Consider enhancing refetchAll with error handling and dependency tracking

While the implementation works, it could benefit from:

  1. Error handling as suggested in past reviews
  2. Clear documentation of how it triggers all dependency refetches

63-69: LGTM! Comprehensive error handling

The error handling implementation properly distinguishes between server availability and other error types, providing a better user experience.


92-96: LGTM! Clean rendering logic

The rendering implementation properly handles loading states and context provision with good performance considerations through memoization.

apps/kyb-app/src/pages/CollectionFlow/CollectionFlow.tsx (1)

23-23: LGTM: Hook changes improve code organization

The migration to useCustomerQuery and addition of useSignupGuard hook improves the separation of concerns and maintainability.

Also applies to: 41-41

apps/kyb-app/src/common/utils/request.ts Outdated Show resolved Hide resolved
Copy link
Contributor

@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: 1

🧹 Outside diff range and nitpick comments (8)
apps/kyb-app/src/common/components/layouts/Signup/Logo.tsx (1)

13-13: Consider enhancing type safety and maintainability

The animation implementation looks good, but could benefit from some improvements:

  1. Add explicit type for the loading state
  2. Extract animation constants for better maintainability
+ const ANIMATION_CONSTANTS = {
+   duration: 0.5,
+   initial: { opacity: 0, scale: 0.8 },
+   loaded: { opacity: 1, scale: 1 },
+ } as const;
+
- const [isLoaded, setIsLoaded] = useState(false);
+ const [isLoaded, setIsLoaded] = useState<boolean>(false);

  return (
    <motion.img
      src={imageSrc}
      style={styles}
-     initial={{ opacity: 0, scale: 0.8 }}
-     animate={isLoaded ? { opacity: 1, scale: 1 } : { opacity: 0, scale: 0.8 }}
-     transition={{ duration: 0.5 }}
+     initial={ANIMATION_CONSTANTS.initial}
+     animate={isLoaded ? ANIMATION_CONSTANTS.loaded : ANIMATION_CONSTANTS.initial}
+     transition={{ duration: ANIMATION_CONSTANTS.duration }}
      onLoad={() => setIsLoaded(true)}
    />

Also applies to: 17-26

apps/kyb-app/src/pages/CollectionFlow/components/atoms/LoadingScreen/LoadingScreen.tsx (3)

4-10: Consider improving type definitions and documentation.

  1. The comment about onExit could be clearer
  2. Consider making isLoading required to prevent undefined behavior
  3. Add JSDoc documentation for better IDE support
+/**
+ * Props for the LoadingScreen component
+ * @property onAnimationComplete - Callback fired when the entry animation completes
+ * @property onExitComplete - Callback fired when the exit animation completes
+ * @property isLoading - Controls the visibility of the loading screen
+ */
 interface LoadingScreenProps {
   onAnimationComplete?: () => void;
   onExitComplete?: () => void;
 
-  // When providing exactly false onExit callback will be fired
-  isLoading?: boolean;
+  // Controls the visibility of the loading screen.
+  // When false, triggers the exit animation and onExitComplete callback
+  isLoading: boolean;
 }

18-35: Consider performance and accessibility improvements.

While the animation implementation is good, consider the following improvements:

  1. The 0.7s animation duration might feel slow for frequent loading states
  2. The backdrop blur might impact performance on lower-end devices
  3. Consider making the spinner size configurable via props
  4. The background opacity (80%) might not provide sufficient contrast for accessibility

Here's a suggested improvement:

 <AnimatePresence mode="wait" onExitComplete={onExitComplete}>
   {isLoading === false ? null : (
     <motion.div
       key="loading"
       initial={{ opacity: 0, scale: 0.5 }}
       animate={{ opacity: 1, scale: 1 }}
       exit={{ opacity: 0, scale: 0.5 }}
       transition={{
-        duration: 0.7,
+        duration: 0.4,
         ease: [0, 0.71, 0.2, 1.01],
       }}
-      className="fixed inset-0 flex h-screen w-screen items-center justify-center bg-white/80 backdrop-blur-sm"
+      className="fixed inset-0 flex h-screen w-screen items-center justify-center bg-white/90"
       onAnimationComplete={onAnimationComplete}
     >
-      <Loader2 className="animate-spin text-black" size={72} />
+      <Loader2 className="animate-spin text-black" size={props.spinnerSize ?? 72} />
     </motion.div>
   )}
 </AnimatePresence>

12-37: Consider implementing a loading state management system.

Given this component's role in providing loading feedback, consider implementing a centralized loading state management system (e.g., using React Context or a state management library) to handle multiple loading states across the application consistently.

This would allow for:

  • Coordinated loading states across components
  • Prevention of multiple loading screens appearing simultaneously
  • Consistent animation timing and styling across the application
apps/kyb-app/src/pages/CollectionFlow/CollectionFlow.tsx (4)

171-171: Remove console.log statement

Remove debug logging before production deployment.

-  console.log({ collectionFlow });

186-187: Remove console.log statement

Remove debug logging before production deployment.

-  console.log('Collection flow touched, changing state to inprogress');

309-326: Improve debug mode implementation

The debug mode implementation could benefit from:

  1. Type-safe localStorage access
  2. Proper styling for debug controls
  3. Environment-based conditional rendering
+ const isDevMode = process.env.NODE_ENV === 'development' && localStorage.getItem('devmode') === 'true';

- {localStorage.getItem('devmode') ? (
+ {isDevMode ? (
-   <div className="flex flex-col gap-4">
+   <div className="flex flex-col gap-4 p-4 bg-gray-100 rounded-md mb-4">
-     DEBUG
+     <h3 className="font-bold text-gray-700">Debug Controls</h3>
      <div>
        {currentPage
          ? currentPage.stateName
          : 'Page not found and state ' + state}
      </div>
      <div className="flex gap-4">
-       <button onClick={() => stateApi.sendEvent('PREVIOUS')}>
+       <button
+         className="px-3 py-1 bg-gray-200 rounded hover:bg-gray-300"
+         onClick={() => stateApi.sendEvent('PREVIOUS')}>
          prev
        </button>
-       <button onClick={() => stateApi.sendEvent('NEXT')}>
+       <button
+         className="px-3 py-1 bg-gray-200 rounded hover:bg-gray-300"
+         onClick={() => stateApi.sendEvent('NEXT')}>
          next
        </button>
      </div>
    </div>
  ) : null}

229-305: Consider adjusting animation timing

The sidebar animations have sequential delays (0.2s, 0.5s, 0.7s, 0.9s, 1.1s) which might feel too slow for returning users. Consider reducing delays or adding a preference to disable animations.

📜 Review details

Configuration used: CodeRabbit UI
Review profile: CHILL

📥 Commits

Reviewing files that changed from the base of the PR and between d572e64 and af1c45d.

⛔ Files ignored due to path filters (1)
  • pnpm-lock.yaml is excluded by !**/pnpm-lock.yaml
📒 Files selected for processing (14)
  • apps/kyb-app/package.json (2 hunks)
  • apps/kyb-app/src/common/components/layouts/Signup/Background.tsx (2 hunks)
  • apps/kyb-app/src/common/components/layouts/Signup/Footer.tsx (2 hunks)
  • apps/kyb-app/src/common/components/layouts/Signup/FormContainer.tsx (2 hunks)
  • apps/kyb-app/src/common/components/layouts/Signup/Header.tsx (2 hunks)
  • apps/kyb-app/src/common/components/layouts/Signup/Logo.tsx (2 hunks)
  • apps/kyb-app/src/common/components/layouts/Signup/Signup.tsx (2 hunks)
  • apps/kyb-app/src/components/layouts/AppShell/Content.tsx (1 hunks)
  • apps/kyb-app/src/pages/CollectionFlow/CollectionFlow.tsx (5 hunks)
  • apps/kyb-app/src/pages/CollectionFlow/components/atoms/LoadingScreen/LoadingScreen.tsx (1 hunks)
  • apps/kyb-app/src/pages/CollectionFlow/components/organisms/AnimatedUIRenderer.tsx (1 hunks)
  • apps/kyb-app/src/pages/Root/Root.tsx (1 hunks)
  • apps/kyb-app/src/pages/SignUpPage/SignUpPage.tsx (1 hunks)
  • apps/kyb-app/src/pages/SignUpPage/components/SignUpForm/components/Submit/Submit.tsx (1 hunks)
🚧 Files skipped from review as they are similar to previous changes (11)
  • apps/kyb-app/package.json
  • apps/kyb-app/src/common/components/layouts/Signup/Background.tsx
  • apps/kyb-app/src/common/components/layouts/Signup/Footer.tsx
  • apps/kyb-app/src/common/components/layouts/Signup/FormContainer.tsx
  • apps/kyb-app/src/common/components/layouts/Signup/Header.tsx
  • apps/kyb-app/src/common/components/layouts/Signup/Signup.tsx
  • apps/kyb-app/src/components/layouts/AppShell/Content.tsx
  • apps/kyb-app/src/pages/CollectionFlow/components/organisms/AnimatedUIRenderer.tsx
  • apps/kyb-app/src/pages/Root/Root.tsx
  • apps/kyb-app/src/pages/SignUpPage/SignUpPage.tsx
  • apps/kyb-app/src/pages/SignUpPage/components/SignUpForm/components/Submit/Submit.tsx
🔇 Additional comments (5)
apps/kyb-app/src/common/components/layouts/Signup/Logo.tsx (1)

1-2: LGTM! Import issue has been resolved.

The Framer Motion import has been correctly updated to use the proper package path.

apps/kyb-app/src/pages/CollectionFlow/components/atoms/LoadingScreen/LoadingScreen.tsx (1)

1-3: LGTM!

The imports are appropriate and necessary for the component's functionality.

apps/kyb-app/src/pages/CollectionFlow/CollectionFlow.tsx (3)

70-76: LGTM! Good refactoring of session protection

The shift from HOC-based session protection to hook-based protection with useSignupGuard is a good improvement. This approach provides better composition and flexibility.


146-151: LGTM! Clean animation implementation

The framer-motion implementation provides a smooth transition effect with good timing (0.7s) and combined transforms (translation + opacity).


139-144: 🛠️ Refactor suggestion

Consider adding error boundary integration

While the edge case handling is in place, integrating with an error boundary would provide better error recovery and maintainability. This would complement the existing error screens (NetworkError, InvalidToken, etc.) mentioned in the PR objectives.

Copy link
Contributor

@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 (2)
apps/kyb-app/src/pages/CollectionFlow/CollectionFlow.tsx (2)

177-177: Remove debugging console.log statement

This console.log statement appears to be left over from debugging.

-  console.log({ collectionFlow });

154-156: Consider extracting animation constants

The animation configuration is duplicated across multiple motion components. Consider extracting these values into constants for better maintainability and consistency.

+const ANIMATION_DEFAULTS = {
+  duration: 0.5,
+  fadeInFromLeft: {
+    initial: { opacity: 0, x: -20 },
+    animate: { opacity: 1, x: 0 },
+  },
+  fadeIn: {
+    initial: { opacity: 0 },
+    animate: { opacity: 1 },
+  },
+};

+const getDelayedTransition = (delay: number) => ({
+  duration: ANIMATION_DEFAULTS.duration,
+  delay,
+});

 // Example usage:
-<motion.div
-  initial={{ opacity: 0, x: -20 }}
-  animate={{ opacity: 1, x: 0 }}
-  transition={{ duration: 0.5 }}
+<motion.div
+  {...ANIMATION_DEFAULTS.fadeInFromLeft}
+  transition={getDelayedTransition(0)}

Also applies to: 238-240, 244-246, 250-252, 261-263, 279-281, 286-288

📜 Review details

Configuration used: CodeRabbit UI
Review profile: CHILL

📥 Commits

Reviewing files that changed from the base of the PR and between af1c45d and c40a2c9.

📒 Files selected for processing (1)
  • apps/kyb-app/src/pages/CollectionFlow/CollectionFlow.tsx (6 hunks)
🔇 Additional comments (2)
apps/kyb-app/src/pages/CollectionFlow/CollectionFlow.tsx (2)

144-149: LGTM! Improved error handling with user-friendly feedback

The error handling has been improved by replacing the previous placeholder with a proper AppErrorScreen component that provides clear feedback to users.


315-332: Consider removing or protecting debug mode

The debug interface is controlled by a simple localStorage check. Consider:

  1. Using an environment variable for better security
  2. Adding a more secure mechanism to enable debug mode
  3. Removing it if it's not needed in production

Copy link
Contributor

@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: 1

🧹 Outside diff range and nitpick comments (2)
apps/kyb-app/src/pages/CollectionFlow/CollectionFlow.tsx (2)

153-157: Consider performance optimization for animations

While the animation implementation looks good, consider:

  1. Using will-change CSS property to optimize performance
  2. Implementing layoutId prop for shared element transitions
  3. Using useReducedMotion hook to respect user preferences
 <motion.div
   initial={{ x: '100%', opacity: 1 }}
   animate={{ x: 0, opacity: 1 }}
   transition={{ duration: 0.7 }}
+  style={{ willChange: 'transform' }}
 >

313-330: Improve debug mode implementation

The current debug mode implementation could be enhanced:

  1. Use environment variables instead of localStorage
  2. Add more structured debug information
  3. Consider using a dedicated debugging component
-{localStorage.getItem('devmode') ? (
+{process.env.NODE_ENV === 'development' && (
   <DebugPanel
     currentPage={currentPage}
     state={state}
     stateApi={stateApi}
   />
 )}
📜 Review details

Configuration used: CodeRabbit UI
Review profile: CHILL

📥 Commits

Reviewing files that changed from the base of the PR and between c40a2c9 and ebe5fd1.

📒 Files selected for processing (2)
  • apps/kyb-app/src/common/utils/request.ts (4 hunks)
  • apps/kyb-app/src/pages/CollectionFlow/CollectionFlow.tsx (6 hunks)
🚧 Files skipped from review as they are similar to previous changes (1)
  • apps/kyb-app/src/common/utils/request.ts
🔇 Additional comments (1)
apps/kyb-app/src/pages/CollectionFlow/CollectionFlow.tsx (1)

71-77: Verify authentication strategy change

The shift from withSessionProtected HOC to useSignupGuard hook changes the authentication approach. Ensure that:

  1. All authentication scenarios are still properly handled
  2. Session validation occurs before any sensitive data is loaded
  3. Proper redirection happens for unauthenticated users

apps/kyb-app/src/pages/CollectionFlow/CollectionFlow.tsx Outdated Show resolved Hide resolved
Copy link
Contributor

@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: 9

🧹 Outside diff range and nitpick comments (22)
apps/kyb-app/src/common/providers/DependenciesProvider/dependencies-context.ts (3)

4-8: Add JSDoc documentation for the context.

Consider adding documentation to explain:

  • The purpose of this context
  • The difference between isLoading and isFetching states
  • The expected behavior of refetchAll
+/**
+ * Context for managing application dependencies and their loading states.
+ * @property {() => Promise<void>} refetchAll - Triggers a refresh of all dependencies
+ * @property {boolean} isLoading - Indicates initial loading state
+ * @property {boolean} isFetching - Indicates background refresh state
+ */
 export const DependenciesContext = createContext<IDependenciesContext>({
   refetchAll: async () => {},
   isLoading: false,
   isFetching: false,
 });

4-8: Consider adding error handling to the context.

The context should include error state management to handle potential failures during dependency fetching.

 export const DependenciesContext = createContext<IDependenciesContext>({
   refetchAll: async () => {},
   isLoading: false,
   isFetching: false,
+  error: null,
+  clearError: () => {},
 });

5-5: Add return type annotation to the async function.

For better type safety, consider adding a return type to the refetchAll function.

-  refetchAll: async () => {},
+  refetchAll: async (): Promise<void> => {},
apps/kyb-app/src/hooks/useEndUserQuery/useEndUserQuery.ts (1)

15-15: Consider type validation before error casting.

The current error casting assumes all errors are HTTPError instances. Consider adding type validation to handle different error types gracefully.

-    error: error ? (error as HTTPError) : null,
+    error: error instanceof HTTPError ? error : null,
apps/kyb-app/src/hooks/useUISchemasQuery/useUISchemasQuery.ts (1)

17-21: Consider removing redundant isLoaded property

The isLoaded property is redundant as it's just an alias for isFetched. Consider removing it to maintain a cleaner API, or document why both properties are necessary if there's a specific use case.

  return {
    isLoading: isLoading && !isFetched,
-    isLoaded: isFetched,
    data: data ?? null,
    error: error ? (error as HTTPError) : null,
    refetch,
  };
apps/kyb-app/src/domains/collection-flow/query-keys.ts (1)

20-31: Consider documenting the architectural change

The shift from explicit endUserId passing to what appears to be a session/context-based approach is a significant architectural change. Consider:

  1. Adding comments explaining the new approach
  2. Updating relevant documentation
  3. Adding examples of how to access user context in the new architecture
apps/kyb-app/src/pages/SignUpPage/SignUpPage.tsx (1)

29-36: Add exit animations to child components

While AnimatePresence is properly set up, the child components should have exit animations to ensure smooth transitions when unmounting.

Consider adding exit animations to the child components. For example:

- <Logo />
+ <motion.div
+   exit={{ opacity: 0 }}
+   transition={{ duration: 0.3 }}
+ >
+   <Logo />
+ </motion.div>

Apply similar patterns to Header, FormContainer, and Footer components for consistent animation behavior.

apps/kyb-app/src/hooks/useAppExit/useAppExit.ts (2)

Line range hint 1-33: Consider adding type safety and error handling improvements

While the implementation is functional, it could benefit from:

  1. TypeScript interfaces for the return type and UI schema structure
  2. Error handling for failed UI schema fetching
  3. URL validation before redirection

Consider applying these improvements:

+ interface AppExitConfig {
+   kybOnExitAction: 'send-event' | 'redirect-to-customer-portal';
+ }
+
+ interface UISchema {
+   config?: AppExitConfig;
+ }
+
+ interface AppExitHook {
+   exit: () => void;
+   isExitAvailable: boolean;
+ }
+
 export const useAppExit = () => {
   const appLanguage = useLanguage();
-  const { data: uiSchema } = useUISchemasQuery({ language: appLanguage });
+  const { data: uiSchema, error } = useUISchemasQuery({ language: appLanguage });
   const { customer } = useCustomerQuery();
   const { trackEvent } = useFlowTracking();

+  if (error) {
+    console.error('Failed to fetch UI schema:', error);
+  }
+
   const kybOnExitAction = uiSchema?.config?.kybOnExitAction || 'send-event';

   const exit = useCallback(() => {
     if (kybOnExitAction === 'send-event') {
       trackEvent(CollectionFlowEvents.USER_EXITED);
       return;
     }

     if (kybOnExitAction === 'redirect-to-customer-portal') {
-      if (customer?.websiteUrl) {
+      if (customer?.websiteUrl && isValidUrl(customer.websiteUrl)) {
         location.href = customer.websiteUrl;
+      } else {
+        console.error('Invalid or missing website URL');
       }
     }
-  }, [trackEvent, customer]);
+  }, [trackEvent, customer, kybOnExitAction]);

   return {
     exit,
     isExitAvailable: kybOnExitAction === 'send-event' ? true : !!customer?.websiteUrl,
-  };
+  } as AppExitHook;
 };
+
+ function isValidUrl(url: string): boolean {
+   try {
+     new URL(url);
+     return true;
+   } catch {
+     return false;
+   }
+ }

Line range hint 24-26: Add URL sanitization before redirect

Direct assignment to location.href with unsanitized URLs could pose a security risk. Consider adding URL validation and sanitization.

Consider:

  1. Implementing a URL allowlist for known customer domains
  2. Adding URL sanitization to prevent javascript: URLs
  3. Using a dedicated navigation service that enforces security policies
apps/kyb-app/src/hooks/useUIOptionsRedirect/useUIOptionsRedirect.ts (3)

Line range hint 26-32: Add URL validation before redirection

Direct assignment to location.href with unvalidated URLs could pose a security risk. Consider adding URL validation to prevent open redirect vulnerabilities.

   useEffect(() => {
     if (redirectUrls?.[state]) {
       const redirectUrl = redirectUrls[state] as string;
+      try {
+        const url = new URL(redirectUrl);
+        if (!['http:', 'https:'].includes(url.protocol)) {
+          throw new Error('Invalid URL protocol');
+        }
       console.info(`Collection Flow resolved to ${state}. Redirecting to ${redirectUrl}`);
-      location.href = redirectUrls[state] as string;
+      location.href = url.toString();
+      } catch (error) {
+        console.error('Invalid redirect URL:', error);
+      }
     }
   }, [redirectUrls, state]);

Line range hint 29-29: Remove or conditionally include console logging

Production builds should not include console logs. Consider removing the log or wrapping it in a development check.

-      console.info(`Collection Flow resolved to ${state}. Redirecting to ${redirectUrl}`);
+      if (process.env.NODE_ENV === 'development') {
+        console.info(`Collection Flow resolved to ${state}. Redirecting to ${redirectUrl}`);
+      }

Line range hint 20-24: Improve type safety by using type guards

Replace type assertions with proper type guards to ensure type safety at runtime.

   const redirectUrls: UIOptions['redirectUrls'] | null = useMemo(() => {
     if (!uiOptions) return null;
 
+    const isValidRedirectUrls = (urls: unknown): urls is Record<string, string> => {
+      return typeof urls === 'object' && urls !== null;
+    };
+
+    if (!isValidRedirectUrls(uiOptions.redirectUrls)) return null;
     return uiOptions.redirectUrls;
   }, [uiOptions]);
apps/kyb-app/src/common/providers/ThemeProvider/ThemeProvider.tsx (2)

Line range hint 22-27: Consider enhancing error handling with error boundaries.

While the current error handling with console.warn is functional, consider implementing React Error Boundaries for more robust error handling, especially since theme loading is critical for the UI.

Here's a suggested implementation:

class ThemeErrorBoundary extends React.Component<{children: React.ReactNode}, {hasError: boolean}> {
  constructor(props) {
    super(props);
    this.state = { hasError: false };
  }

  static getDerivedStateFromError(error) {
    return { hasError: true };
  }

  componentDidCatch(error, errorInfo) {
    console.error('Theme loading failed:', error, errorInfo);
  }

  render() {
    if (this.state.hasError) {
      return <div>Theme loading failed. Using default theme.</div>;
    }

    return this.props.children;
  }
}

Line range hint 36-41: Consider adding cleanup in useLayoutEffect.

The current useLayoutEffect implementation adds styles but doesn't clean them up when the component unmounts.

Consider adding a cleanup function:

 useLayoutEffect(() => {
   if (theme) {
     document
       .getElementsByTagName('html')[0]
       ?.setAttribute('style', transformThemeToInlineStyles(theme as ITheme));
   }
+  return () => {
+    document.getElementsByTagName('html')[0]?.removeAttribute('style');
+  };
 }, [theme]);
apps/kyb-app/src/components/layouts/AppShell/LanguagePicker.tsx (1)

Line range hint 33-61: Consider enhancing accessibility and user experience.

While the implementation is functionally correct, consider these improvements:

  1. Add aria-label to the language picker for better screen reader support
  2. Consider adding language names in their native script (e.g., "English (English)", "中文 (Chinese)")
  3. Add keyboard navigation hints for dropdown

Here's a suggested enhancement:

       <DropdownInput
         value={language}
         name="languagePicker"
+        aria-label="Select language"
         options={supportedLanguages}
         props={{
-          item: { variant: 'inverted' },
+          item: { 
+            variant: 'inverted',
+            className: 'focus:ring-2 focus:ring-primary'
+          },
           content: { className: 'w-[204px] p-1', align: 'start' },
           trigger: {
             icon: (
               <span className="text-primary-foreground">
                 <GlobeIcon />
               </span>
             ),
-            className: 'px-3 gap-x-2 bg-primary text-primary-foreground',
+            className: 'px-3 gap-x-2 bg-primary text-primary-foreground focus:ring-2 focus:ring-primary',
+            'aria-keyshortcuts': 'Alt+L'
           },
         }}
apps/kyb-app/src/domains/collection-flow/collection-flow.api.ts (1)

Line range hint 1-109: Consider implementing consistent error handling across all API functions

While fetchFlowContext has proper error handling with specific error messages and type validation, other functions like fetchUISchema, fetchCustomer, etc., could benefit from similar error handling patterns. This would improve reliability and debugging.

Example implementation for fetchUISchema:

 export const fetchUISchema = async (language: string, endUserId?: string): Promise<UISchema> => {
+  try {
+    const result = await request
+      .get(`collection-flow/${!endUserId ? 'no-user/' : ''}configuration/${language}`, {
+        searchParams: {
+          uiContext: 'collection_flow',
+        },
+      });
+    
+    const schema = await result.json<UISchema>();
+    
+    if (!schema || typeof schema !== 'object') {
+      throw new Error('Invalid UI schema response');
+    }
+    
+    return schema;
+  } catch (error) {
+    console.error('Error fetching UI schema:', error);
+    throw error;
+  }
 };
apps/kyb-app/src/pages/SignUpPage/components/SignUpForm/SignUpForm.tsx (3)

31-34: Add TypeScript interface for signup state

Consider adding a type definition for the signup state to improve type safety and maintainability.

+interface SignupState {
+  isLoading: boolean;
+  isSuccess: boolean;
+}

-const [signupState, setSignupState] = useState({
+const [signupState, setSignupState] = useState<SignupState>({
   isLoading: false,
   isSuccess: false,
});

52-63: Handle potential race conditions in success flow

The success flow might have race conditions between state updates, refetching, and navigation. Consider using async/await properly and updating the state after all operations complete.

        try {
          await createEndUserRequest(values);
          await refetchAll();
+         setSignupState({ isLoading: false, isSuccess: true });
          navigate(`/collection-flow?token=${accessTokenRef.current}`);
-         setSignupState({ isLoading: false, isSuccess: false });
        } catch (error) {
          setSignupState({ isLoading: false, isSuccess: false });
          toast.error('Failed to create user. Please try again.');
        }

78-78: Remove 'any' type casting for layouts

The layouts prop is currently cast to 'any'. Consider defining proper types for the layouts object to improve type safety.

-        layouts={layouts as any}
+        layouts={layouts as DynamicFormLayouts}

You'll need to import or define the DynamicFormLayouts type from @ballerine/ui.

apps/kyb-app/src/common/providers/DependenciesProvider/DependenciesProvider.tsx (3)

29-31: Consider initializing error state with a type-safe approach

The error state could benefit from a more type-safe initialization using a discriminated union type to handle different error scenarios more explicitly.

- const [error, setError] = useState<Error | null>(null);
+ type DependencyError =
+   | { type: 'server_unavailable'; error: ServerNotAvailableError }
+   | { type: 'invalid_token'; error: InvalidAccessTokenError }
+   | { type: 'unknown'; error: Error }
+   | null;
+ const [error, setError] = useState<DependencyError>(null);

118-118: Remove debug console.log statement

Debug logging should be removed before merging to production.

- console.log({ error });

Line range hint 92-116: Enhance error handling with structured error responses

The error handling could be more robust with better typing and structured error responses.

- const handleErrors = async (errors: HTTPError[]) => {
+ type ErrorResponse = {
+   statusCode: number;
+   message: string;
+   code?: string;
+ };

+ const handleErrors = async (errors: HTTPError[]) => {
+   try {
      if (errors.every(error => !error.response)) {
        setError(new ServerNotAvailableError());
        return;
      }

      const isShouldIgnore = await isShouldIgnoreErrors(errors);
      if (isShouldIgnore) return;

-     const errorResponses = await getJsonErrors(errors);
+     const errorResponses = await getJsonErrors(errors) as ErrorResponse[];

      if (errorResponses.every(error => error.statusCode === 401)) {
-       setError(new InvalidAccessTokenError());
+       setError({ type: 'invalid_token', error: new InvalidAccessTokenError() });
        return;
      }

-     setError(new Error('Something went wrong'));
+     setError({ 
+       type: 'unknown',
+       error: new Error(`Multiple errors occurred: ${errorResponses.map(e => e.message).join(', ')}`)
+     });
+   } catch (e) {
+     setError({
+       type: 'unknown',
+       error: new Error('Failed to process error responses')
+     });
+   }
  };
📜 Review details

Configuration used: CodeRabbit UI
Review profile: CHILL

📥 Commits

Reviewing files that changed from the base of the PR and between ebe5fd1 and 2bfbb49.

📒 Files selected for processing (24)
  • apps/kyb-app/src/common/components/layouts/Signup/FormContainer.tsx (2 hunks)
  • apps/kyb-app/src/common/components/layouts/Signup/Header.tsx (2 hunks)
  • apps/kyb-app/src/common/components/layouts/Signup/Logo.tsx (2 hunks)
  • apps/kyb-app/src/common/providers/DependenciesProvider/DependenciesProvider.tsx (3 hunks)
  • apps/kyb-app/src/common/providers/DependenciesProvider/dependencies-context.ts (1 hunks)
  • apps/kyb-app/src/common/providers/DependenciesProvider/types.ts (1 hunks)
  • apps/kyb-app/src/common/providers/ThemeProvider/ThemeProvider.tsx (1 hunks)
  • apps/kyb-app/src/components/layouts/AppShell/LanguagePicker.tsx (1 hunks)
  • apps/kyb-app/src/domains/collection-flow/collection-flow.api.ts (1 hunks)
  • apps/kyb-app/src/domains/collection-flow/query-keys.ts (1 hunks)
  • apps/kyb-app/src/hooks/useAppExit/useAppExit.ts (1 hunks)
  • apps/kyb-app/src/hooks/useCustomerQuery/useCustomerQuery.ts (1 hunks)
  • apps/kyb-app/src/hooks/useEndUserQuery/useEndUserQuery.ts (1 hunks)
  • apps/kyb-app/src/hooks/useFlowContextQuery/useFlowContextQuery.ts (1 hunks)
  • apps/kyb-app/src/hooks/useUIOptionsRedirect/useUIOptionsRedirect.ts (1 hunks)
  • apps/kyb-app/src/hooks/useUISchemasQuery/useUISchemasQuery.ts (1 hunks)
  • apps/kyb-app/src/pages/CollectionFlow/CollectionFlow.tsx (6 hunks)
  • apps/kyb-app/src/pages/CollectionFlow/components/organisms/AnimatedUIRenderer.tsx (1 hunks)
  • apps/kyb-app/src/pages/Root/Root.tsx (1 hunks)
  • apps/kyb-app/src/pages/Root/hooks/useIsSignupRequired/index.ts (0 hunks)
  • apps/kyb-app/src/pages/Root/hooks/useIsSignupRequired/useIsSignupRequired.ts (0 hunks)
  • apps/kyb-app/src/pages/SignUpPage/SignUpPage.tsx (1 hunks)
  • apps/kyb-app/src/pages/SignUpPage/components/SignUpForm/SignUpForm.tsx (3 hunks)
  • apps/kyb-app/src/pages/SignUpPage/components/SignUpForm/components/Submit/Submit.tsx (1 hunks)
💤 Files with no reviewable changes (2)
  • apps/kyb-app/src/pages/Root/hooks/useIsSignupRequired/index.ts
  • apps/kyb-app/src/pages/Root/hooks/useIsSignupRequired/useIsSignupRequired.ts
🚧 Files skipped from review as they are similar to previous changes (9)
  • apps/kyb-app/src/common/components/layouts/Signup/FormContainer.tsx
  • apps/kyb-app/src/common/components/layouts/Signup/Header.tsx
  • apps/kyb-app/src/common/components/layouts/Signup/Logo.tsx
  • apps/kyb-app/src/common/providers/DependenciesProvider/types.ts
  • apps/kyb-app/src/hooks/useCustomerQuery/useCustomerQuery.ts
  • apps/kyb-app/src/hooks/useFlowContextQuery/useFlowContextQuery.ts
  • apps/kyb-app/src/pages/CollectionFlow/components/organisms/AnimatedUIRenderer.tsx
  • apps/kyb-app/src/pages/Root/Root.tsx
  • apps/kyb-app/src/pages/SignUpPage/components/SignUpForm/components/Submit/Submit.tsx
🔇 Additional comments (14)
apps/kyb-app/src/common/providers/DependenciesProvider/dependencies-context.ts (1)

1-8: Verify the context usage across components.

Let's ensure this context is properly consumed and that components handle the loading states correctly.

✅ Verification successful

Context usage is properly implemented across components

The context is well-structured and correctly consumed:

  • Properly wrapped at the top level in GlobalProviders.tsx
  • Used via useDependencies hook in multiple components:
    • Root.tsx - handles loading state
    • SignUpForm.tsx - uses refetchAll functionality
    • CollectionFlow.tsx - handles loading state
  • Clean implementation with a dedicated hook in useDependencies.ts
🏁 Scripts executed

The following scripts were executed for the analysis:

Script:

#!/bin/bash
# Description: Find components consuming this context and verify proper usage

# Find direct context imports
echo "Components importing DependenciesContext:"
rg -l "import.*DependenciesContext" --type ts --type tsx

# Find usages of the context hook
echo "\nComponents using dependency hooks:"
rg -l "useDependencies" --type ts --type tsx

# Check for proper error boundary implementation
echo "\nVerifying error boundary implementation:"
rg -l "ErrorBoundary.*DependenciesProvider" --type ts --type tsx

Length of output: 525


Script:

#!/bin/bash
# Let's try without file type restrictions since tsx might not be recognized
echo "Components importing DependenciesContext:"
rg "import.*DependenciesContext"

echo -e "\nComponents using dependency hooks:"
rg "useDependencies"

echo -e "\nComponents wrapped with DependenciesProvider:"
rg "DependenciesProvider"

echo -e "\nContext consumer implementations:"
rg "useContext.*DependenciesContext"

Length of output: 3373

apps/kyb-app/src/hooks/useEndUserQuery/useEndUserQuery.ts (3)

14-14: LGTM! Improved loading state logic.

The modified loading state logic isLoading && !isFetched prevents false loading indicators after the initial fetch, which is particularly useful for animations and transitions.


17-17: LGTM! Added isLoaded state.

The addition of isLoaded state aligns well with the PR objectives for managing animations and transitions between states.


8-8: Verify the implications of Infinity staleTime.

Setting staleTime: Infinity means the data will never be considered stale. Ensure this aligns with the business requirements and won't cause issues with data freshness.

✅ Verification successful

Let me analyze the findings from the shell script results. I can see a clear pattern of staleTime usage across the codebase:

  1. In the KYB app, there are three instances of staleTime: Infinity:

    • useEndUserQuery
    • useUISchemasQuery
    • useFlowContextQuery
  2. These queries appear to be related to user context and UI schemas, which are typically static during a user session.

  3. Other queries in the codebase use finite staleTime values:

    • Most queries use 100,000ms (100 seconds)
    • Some use 1,000,000ms (16.6 minutes)
    • A few use 10,000ms (10 seconds)

Setting staleTime: Infinity is appropriate for this end-user query.

The end user data and related UI schemas are session-specific and shouldn't change during the user's session. This pattern is consistently used across the KYB app for session-context related queries. Other dynamic data in the system uses appropriate finite staleTime values, showing a clear distinction between static session data and dynamic application data.

🏁 Scripts executed

The following scripts were executed for the analysis:

Script:

#!/bin/bash
# Description: Check for other instances of staleTime configuration and data refresh patterns

# Search for other staleTime configurations
echo "Checking staleTime configurations:"
rg -A 2 "staleTime:" --type ts

# Search for manual refresh patterns
echo "Checking refetch usage:"
rg "refetch\(" --type ts

Length of output: 8536

apps/kyb-app/src/hooks/useUISchemasQuery/useUISchemasQuery.ts (1)

6-6: Verify all usages of useUISchemasQuery

The change to object parameters is a good practice for extensibility. However, this is a breaking change that requires updates to all consumers of this hook.

✅ Verification successful

All usages of useUISchemasQuery have been properly updated

All 6 call sites of useUISchemasQuery in the codebase are using the new object parameter syntax { language }:

  • DependenciesProvider.tsx
  • ThemeProvider.tsx
  • CollectionFlow.tsx
  • useUIOptionsRedirect.ts
  • useAppExit.ts
  • LanguagePicker.tsx

The test mocks in useUIOptionsRedirect.unit.test.ts are also correctly structured to match the hook's return type.

🏁 Scripts executed

The following scripts were executed for the analysis:

Script:

#!/bin/bash
# Description: Find all usages of useUISchemasQuery to verify they've been updated
# to use the new object parameter syntax

# Search for function calls
rg -A 2 "useUISchemasQuery\(" 

# Search for potential test mocks
rg -A 2 "useUISchemasQuery.*mock"

Length of output: 3344

apps/kyb-app/src/domains/collection-flow/query-keys.ts (3)

20-23: Verify handling of undefined endUserId in fetchUISchema

The endUserId parameter is now optional, but it's still being passed directly to fetchUISchema. We should verify that fetchUISchema properly handles the case when endUserId is undefined.

✅ Verification successful

Optional endUserId is properly handled in fetchUISchema implementation

The implementation in collection-flow.api.ts shows that fetchUISchema is designed to handle undefined endUserId:

  • It uses a ternary operator in the URL path: collection-flow/${!endUserId ? 'no-user/' : ''}configuration/${language}
  • When endUserId is undefined, it appends 'no-user/' to the path, providing a fallback behavior
  • The parameter is explicitly typed as optional with endUserId?: string
🏁 Scripts executed

The following scripts were executed for the analysis:

Script:

#!/bin/bash
# Description: Check how fetchUISchema handles undefined endUserId

# Search for fetchUISchema implementation
ast-grep --pattern 'function fetchUISchema($language, $endUserId) {
  $$$
}'

# Search for any null checks on endUserId
rg -A 5 'fetchUISchema.*\(' 

Length of output: 1864


28-31: Consistent refactoring pattern observed

The removal of endUserId from both getCustomer and getContext suggests a deliberate architectural shift away from explicit user ID passing. While this simplifies the API, we should ensure this change:

  1. Maintains proper access control
  2. Doesn't break existing flows
  3. Aligns with the PR's objective of cleaning up the code

24-27: Verify the architectural change in customer data fetching

The removal of endUserId parameter suggests that fetchCustomer now retrieves the customer ID from another source (possibly a context or session). This is a significant architectural change.

✅ Verification successful

Let me gather more information about the customer fetching implementation.


Let me gather more information about the customer fetching implementation and flow context.


Based on the gathered information, I can now generate the final response.

Architectural change in customer data fetching is verified

The removal of endUserId parameter is part of a broader architectural pattern where:

  1. fetchCustomer now retrieves customer data based on the current project ID (../external/customers/by-current-project-id)
  2. The customer context is managed through flow context and authentication tokens
  3. The customer data is shared across components through React Query's state management
🏁 Scripts executed

The following scripts were executed for the analysis:

Script:

#!/bin/bash
# Description: Verify the new customer data fetching approach

# Check if customer ID is retrieved from session or context
ast-grep --pattern 'function fetchCustomer() {
  $$$
}'

# Look for any session/context usage related to customer ID
rg -A 5 'getFlowSession|useFlowContext'

Length of output: 5849


Script:

#!/bin/bash
# Check the implementation of fetchCustomer and related customer query hooks
ast-grep --pattern 'function fetchCustomer() {
  $$$
}'

ast-grep --pattern 'export const useCustomerQuery = () => {
  $$$
}'

# Look for any customer ID usage in the flow context
rg -B 2 -A 5 'customer.*Id|endUserId' --type ts

Length of output: 147302


Script:

#!/bin/bash
# Check the implementation of fetchCustomer and related flow context
ast-grep --pattern 'export const fetchCustomer = async () => {
  $$$
}'

# Look for customer ID usage in flow context
rg -B 2 -A 5 'getFlowContext|useFlowContext' --type ts

# Check collection flow API implementation
ast-grep --pattern 'export const getCollectionFlowContext = async (token: string) => {
  $$$
}'

Length of output: 6114

apps/kyb-app/src/pages/SignUpPage/SignUpPage.tsx (2)

11-11: LGTM: Proper import for animations

The AnimatePresence import is correctly added to support the new animation features.


18-24: Reusing existing review comment

A previous review already suggested simplifying the theme handling by removing the unnecessary ref pattern.

apps/kyb-app/src/hooks/useAppExit/useAppExit.ts (1)

10-10: Verify consistent usage of useUISchemasQuery across the codebase

The parameter structure for useUISchemasQuery has been changed to use an object. Let's ensure this change is consistent across all usages.

✅ Verification successful

All usages of useUISchemasQuery are consistent with the object parameter style

The verification shows that all 6 occurrences of useUISchemasQuery across the codebase consistently use the object parameter style with the language property:

  • CollectionFlow.tsx: useUISchemasQuery({ language })
  • useUIOptionsRedirect.ts: useUISchemasQuery({ language: useLanguage() })
  • useAppExit.ts: useUISchemasQuery({ language: appLanguage })
  • DependenciesProvider.tsx: useUISchemasQuery({ language })
  • ThemeProvider.tsx: useUISchemasQuery({ language })
  • LanguagePicker.tsx: useUISchemasQuery({ language })

No instances of old-style usage were found.

🏁 Scripts executed

The following scripts were executed for the analysis:

Script:

#!/bin/bash
# Description: Find all usages of useUISchemasQuery to verify parameter consistency
# Expected: All calls should use the new object parameter style

# Search for direct useUISchemasQuery calls
echo "Checking direct hook usage patterns:"
rg -A 1 "useUISchemasQuery\(" --type ts

# Search for potential old style usage
echo -e "\nChecking for potential old style usage:"
rg "useUISchemasQuery\([^{]" --type ts

Length of output: 1611

apps/kyb-app/src/common/providers/ThemeProvider/ThemeProvider.tsx (1)

17-17: LGTM! API parameter structure update looks good.

The change to pass an object parameter to useUISchemasQuery aligns with modern API patterns and makes the code more maintainable for future parameter additions.

Let's verify that this change is consistent across all usages of useUISchemasQuery:

✅ Verification successful

All usages of useUISchemasQuery consistently use object parameter style

The verification confirms that all 6 occurrences of useUISchemasQuery across the codebase use the new object parameter style with { language }. No inconsistencies were found.

🏁 Scripts executed

The following scripts were executed for the analysis:

Script:

#!/bin/bash
# Description: Check for any inconsistent usage of useUISchemasQuery
# Expect: All calls should use object parameter style

ast-grep --pattern 'useUISchemasQuery($ARG)'

Length of output: 828

apps/kyb-app/src/components/layouts/AppShell/LanguagePicker.tsx (1)

19-19: LGTM! Parameter change aligns with codebase standardization.

The update to pass an object parameter to useUISchemasQuery is consistent with the broader changes in the codebase.

Let's verify this pattern is consistently applied across the codebase:

✅ Verification successful

All usages of useUISchemasQuery consistently use object parameter pattern

The verification shows that all 6 instances of useUISchemasQuery across the codebase consistently use the object parameter pattern with { language }. This confirms that the change aligns with the standardized approach throughout the application.

🏁 Scripts executed

The following scripts were executed for the analysis:

Script:

#!/bin/bash
# Description: Check if useUISchemasQuery is consistently called with object parameter
# Expected: All calls should use the object parameter pattern

# Search for all useUISchemasQuery calls
ast-grep --pattern 'useUISchemasQuery($ARGS)' 

Length of output: 829

apps/kyb-app/src/domains/collection-flow/collection-flow.api.ts (1)

Line range hint 52-59: Verify the impact of making endUserId optional

The change from string | null to optional string parameter looks good, but there are some considerations:

  1. The URL path construction using !endUserId will treat both undefined and empty string as falsy values, which might not be the intended behavior.
  2. The function doesn't validate if the provided endUserId (when defined) is a non-empty string.

Let's verify the usage of this function across the codebase:

✅ Verification successful

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

The optional endUserId parameter is correctly handled throughout the codebase

The verification shows that:

  1. The EndUser interface in collection-flow.api.ts properly defines id as a string.
  2. All usages of fetchUISchema correctly handle the optional parameter:
    • In DependenciesProvider.tsx: Uses optional chaining with endUser?.id
    • In query-keys.ts: Properly types and passes the optional endUserId
  3. The URL path construction using !endUserId is actually correct as it handles both undefined and empty string cases appropriately, which aligns with the API's requirements for distinguishing between user and no-user contexts.
🏁 Scripts executed

The following scripts were executed for the analysis:

Script:

#!/bin/bash
# Description: Check how fetchUISchema is called across the codebase
# Expected: Calls should handle the optional parameter correctly

# Search for function calls
rg -A 3 "fetchUISchema\(" 

# Search for imports to ensure we catch all usages
rg "import.*fetchUISchema.*from"

Length of output: 811


Script:

#!/bin/bash
# Let's check the context around the usage to understand how endUserId is handled

# Check DependenciesProvider implementation
rg -B 5 -A 10 "fetchUISchema\(language, endUser\?.id\)" apps/kyb-app/src/common/providers/DependenciesProvider/DependenciesProvider.tsx

# Check query-keys implementation
rg -B 5 -A 10 "fetchUISchema\(language, endUserId\)" apps/kyb-app/src/domains/collection-flow/query-keys.ts

# Check if endUser is properly typed
ast-grep --pattern 'interface $_ {
  $$$
  id: $_
  $$$
}'

Length of output: 30056

@chesterkmr chesterkmr changed the title feat: refactor & added animations to cf (WIP) feat: refactor & added animations to cf Nov 25, 2024
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.

2 participants