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: Update the design and shortcut key for SearchByMultipleOptions #9851

Open
wants to merge 38 commits into
base: develop
Choose a base branch
from

Conversation

AdityaJ2305
Copy link
Contributor

@AdityaJ2305 AdityaJ2305 commented Jan 8, 2025

Proposed Changes

@ohcnetwork/care-fe-code-reviewers

Merge Checklist

  • Add specs that demonstrate bug / test a new feature.
  • Update product documentation.
  • Ensure that UI text is kept in I18n files.
  • Prep screenshot or demo video for changelog entry, and attach it to issue.
  • Request for Peer Reviews
  • Completion of QA

Summary by CodeRabbit

Release Notes

  • New Features

    • Enhanced search functionality across multiple components.
    • Introduced a new button for clearing search values.
  • Improvements

    • Refined keyboard navigation in search components.
    • More intuitive search option layouts.
    • Better handling of search field changes.
    • Updated organization selection components for improved user experience.
    • Added localization entries for improved user guidance in search functionality.
    • Improved control flow in the password reset process.
  • Bug Fixes

    • Resolved search parameter reset and validation issues.
    • Improved focus management in search interfaces.
  • Changes

    • Replaced OrganizationSelector with GovtOrganizationSelector in various forms and components.
    • Adjusted properties in search components to enhance initialization behavior.
    • Removed shortcutKey properties from search components, impacting keyboard shortcut functionality.

Copy link
Contributor

coderabbitai bot commented Jan 8, 2025

Walkthrough

This pull request introduces changes to several components to enhance search functionality and user experience. Key modifications include the removal of the shortcutKey property, the addition of a required initialOptionIndex, and improvements to keyboard navigation and search handling in the SearchByMultipleFields component. Additional updates involve the integration of new handler functions in the EncounterList and OrganizationPatients components, as well as the replacement of the OrganizationSelector with GovtOrganizationSelector across various forms. These changes collectively aim to streamline search interactions and improve input management.

Changes

File Change Summary
src/components/Common/SearchByMultipleFields.tsx - Added isAppleDevice import
- Changed initialOptionIndex to required
- Removed shortcutKey from SearchOption
- Updated state handling for selectedOptionIndex
- Enhanced keydown event handling for search shortcuts
- Added clear search button functionality
src/pages/Encounters/EncounterList.tsx - Added handleFieldChange method
- Integrated field change handler with SearchByMultipleFields
src/pages/Organization/OrganizationPatients.tsx - Updated useFilters hook configuration
- Added handleSearch and handleFieldChange methods
- Modified search parameter handling
src/components/Facility/CreateFacilityForm.tsx - Replaced OrganizationSelector with GovtOrganizationSelector
src/components/Facility/FacilityCreate.tsx - Replaced OrganizationSelector with GovtOrganizationSelector
src/components/Patient/PatientRegistration.tsx - Replaced OrganizationSelector with GovtOrganizationSelector
src/components/Users/CreateUserForm.tsx - Replaced OrganizationSelector with GovtOrganizationSelector
src/pages/Facility/FacilitiesPage.tsx - Removed shortcutKey from SearchByMultipleFields
- Added initialOptionIndex={0}
src/pages/Organization/components/GovtOrganizationSelector.tsx - Renamed interface and component from OrganizationSelector to GovtOrganizationSelector
src/pages/PublicAppointments/PatientRegistration.tsx - Replaced OrganizationSelector with GovtOrganizationSelector
public/locale/en.json - Added localization keys for search functionality

Assessment against linked issues

Objective Addressed Explanation
⌘K shortcut to focus search input
Navigation through dropdown with arrow keys
ENTER key for option selection
ESC key to close dropdown and clear input
Phone number search validation [#9868]

Possibly related PRs

  • New Cypress Test for HCX Workflow in the platform #9007: The main PR modifies the SearchByMultipleFields component, which is also referenced in the EncounterList and OrganizationPatients components that handle search functionalities, indicating a direct relationship in terms of search logic and UI updates.
  • Fix: Duplicate network requests in Patient Details page #9297: This PR also involves changes to the searchOptions array, specifically removing the shortcutKey property, which aligns with the changes made in the main PR regarding the SearchByMultipleFields component.
  • Search UI for discharge patients #9320: Similar to the main PR, this PR updates the searchOptions and integrates the SearchByMultipleFields component, indicating a connection in terms of search functionality.

Suggested reviewers

  • rithviknishad
  • Jacobjeevan

Poem

🐰 A rabbit's search, swift and bright,

Keys dancing with digital might,

⌘K opens the gate,

Options navigate,

Searching becomes pure delight! 🔍


📜 Recent review details

Configuration used: CodeRabbit UI
Review profile: CHILL
Plan: Pro

📥 Commits

Reviewing files that changed from the base of the PR and between 7f98d95 and e2cbb84.

📒 Files selected for processing (3)
  • public/locale/en.json (2 hunks)
  • src/pages/Facility/FacilitiesPage.tsx (1 hunks)
  • src/pages/Organization/components/GovtOrganizationSelector.tsx (2 hunks)
🚧 Files skipped from review as they are similar to previous changes (3)
  • src/pages/Facility/FacilitiesPage.tsx
  • public/locale/en.json
  • src/pages/Organization/components/GovtOrganizationSelector.tsx
⏰ Context from checks skipped due to timeout of 90000ms (1)
  • GitHub Check: cypress-run (1)

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 generate docstrings to generate docstrings for this PR. (Beta)
  • @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

netlify bot commented Jan 8, 2025

Deploy Preview for care-ohc ready!

Name Link
🔨 Latest commit e2cbb84
🔍 Latest deploy log https://app.netlify.com/sites/care-ohc/deploys/67850640eb43fd00086499a9
😎 Deploy Preview https://deploy-preview-9851--care-ohc.netlify.app
📱 Preview on mobile
Toggle QR Code...

QR Code

Use your smartphone camera to open QR code link.

To edit notification comments on pull requests, go to your Netlify site configuration.

@JavidSumra
Copy link
Contributor

@AdityaJ2305 I think you can make use of useKeyBoardShortcut hook instead of manual handling keyboard events

@Jacobjeevan Jacobjeevan mentioned this pull request Jan 10, 2025
6 tasks
@AdityaJ2305
Copy link
Contributor Author

Fixed: #9867 (comment)

@AdityaJ2305 AdityaJ2305 marked this pull request as ready for review January 10, 2025 18:14
@AdityaJ2305 AdityaJ2305 requested a review from a team as a code owner January 10, 2025 18:14
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

🧹 Nitpick comments (3)
src/pages/Organization/OrganizationPatients.tsx (1)

41-52: Consider improving phone number validation.

The phone number validation logic could be more robust:

  1. The magic number 13 should be extracted as a constant for better maintainability.
  2. The validation should consider minimum length requirements for different phone number formats.

Apply this diff to improve the validation:

+const MIN_PHONE_LENGTH = 13;
+
 const handleSearch = useCallback((key: string, value: string) => {
   const searchParams = {
     name: key === "name" ? value : "",
     phone_number:
       key === "phone_number"
-        ? value.length >= 13 || value === ""
+        ? value.length >= MIN_PHONE_LENGTH || value === ""
           ? value
           : undefined
         : undefined,
   };
   updateQuery(searchParams);
 }, []);
src/components/Common/SearchByMultipleFields.tsx (2)

191-214: Reduce code duplication in keyboard shortcut hints.

The keyboard shortcut hint UI is duplicated between the phone number input and text input sections. Consider extracting it into a reusable component.

Create a new component for the keyboard hints:

const KeyboardShortcutHint: React.FC<{ isOpen: boolean }> = ({ isOpen }) => {
  if (isOpen) {
    return (
      <span className="border border-gray-300 rounded px-1 py-0.5 bg-white text-gray-500">
        ESC
      </span>
    );
  }
  
  return (
    <span>
      {isAppleDevice ? (
        <span className="border border-gray-300 rounded px-1 py-0.5 bg-white text-gray-500">
          ⌘K
        </span>
      ) : (
        <div className="flex gap-1 font-medium">
          <span className="border border-gray-300 rounded px-1 py-0.5 bg-white text-gray-500">
            Ctrl
          </span>
          <span className="border border-gray-300 rounded px-1 py-0.5 bg-white text-gray-500">
            K
          </span>
        </div>
      )}
    </span>
  );
};

Then use it in both places:

 <div className="absolute top-1/2 right-2 transform -translate-y-1/2 flex items-center space-x-2 text-xs text-gray-500">
-  {open ? (
-    <span className="border border-gray-300 rounded px-1 py-0.5 bg-white text-gray-500">
-      ESC
-    </span>
-  ) : (
-    <span>
-      {isAppleDevice ? (
-        <span className="border border-gray-300 rounded px-1 py-0.5 bg-white text-gray-500">
-          ⌘K
-        </span>
-      ) : (
-        <div className="flex gap-1 font-medium">
-          <span className="border border-gray-300 rounded px-1 py-0.5 bg-white text-gray-500">
-            Ctrl
-          </span>
-          <span className="border border-gray-300 rounded px-1 py-0.5 bg-white text-gray-500">
-            K
-          </span>
-        </div>
-      )}
-    </span>
-  )}
+  <KeyboardShortcutHint isOpen={open} />
 </div>

Also applies to: 226-249


130-140: Enhance keyboard navigation robustness.

The keyboard navigation logic could be improved in two areas:

  1. Add visual feedback when no item is focused (focusedIndex === -1)
  2. Make the Enter key handler more defensive

Apply these improvements:

 } else if (e.key === "Enter" && focusedIndex !== -1) {
+  if (focusedIndex >= unselectedOptions.length) {
+    return;
+  }
   const selectedOptionIndex = options.findIndex(
     (option) => option.key === unselectedOptions[focusedIndex].key,
   );
+  if (selectedOptionIndex === -1) {
+    return;
+  }
   handleOptionChange(selectedOptionIndex);
 }

Also, consider adding a CSS class to indicate when no item is focused:

 className={cn(
   "flex items-center p-2 rounded-md cursor-pointer",
   {
     "bg-gray-100": focusedIndex === index,
+    "bg-gray-50": focusedIndex === -1,
     "hover:bg-secondary-100": true,
   },
 )}
📜 Review details

Configuration used: CodeRabbit UI
Review profile: CHILL
Plan: Pro

📥 Commits

Reviewing files that changed from the base of the PR and between b998f5d and 7f2735f.

📒 Files selected for processing (3)
  • src/components/Common/SearchByMultipleFields.tsx (5 hunks)
  • src/pages/Encounters/EncounterList.tsx (2 hunks)
  • src/pages/Organization/OrganizationPatients.tsx (3 hunks)
🧰 Additional context used
🪛 Biome (1.9.4)
src/pages/Encounters/EncounterList.tsx

[error] 165-175: This block statement doesn't serve any purpose and can be safely removed.

Standalone block statements without any block-level declarations are redundant in JavaScript and can be removed to simplify the code.
Safe fix: Remove redundant block.

(lint/complexity/noUselessLoneBlockStatements)

src/pages/Organization/OrganizationPatients.tsx

[error] 55-61: This block statement doesn't serve any purpose and can be safely removed.

Standalone block statements without any block-level declarations are redundant in JavaScript and can be removed to simplify the code.
Safe fix: Remove redundant block.

(lint/complexity/noUselessLoneBlockStatements)

🔇 Additional comments (1)
src/components/Common/SearchByMultipleFields.tsx (1)

118-126: LGTM! Keyboard shortcut implementation follows best practices.

The change from "/" to "Cmd/Ctrl + K" aligns with modern web application patterns and provides a more intuitive search experience. The implementation correctly handles both Mac and Windows/Linux modifier keys.

src/pages/Encounters/EncounterList.tsx Show resolved Hide resolved
@AdityaJ2305 AdityaJ2305 marked this pull request as draft January 11, 2025 06:23
@github-actions github-actions bot added the merge conflict pull requests with merge conflict label Jan 12, 2025
Copy link

👋 Hi, @AdityaJ2305,
Conflicts have been detected against the base branch. Please rebase your branch against the base branch.


This message is automatically generated by prince-chrismc/label-merge-conflicts-action so don't hesitate to report issues/improvements there.

@AdityaJ2305 AdityaJ2305 force-pushed the serachinput_design_update branch from 91e9334 to 3cfb24a Compare January 12, 2025 09:37
@AdityaJ2305
Copy link
Contributor Author

AdityaJ2305 commented Jan 12, 2025

  • Rename to GovtOrganizationSelector

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

🧹 Nitpick comments (4)
src/components/Common/SearchByMultipleFields.tsx (4)

117-125: Document the new keyboard shortcut

The change from "/" to "Cmd/Ctrl + K" is a good improvement, but should be documented for users.

Consider adding a comment explaining the shortcut change:

+// Using Cmd/Ctrl + K as it's a more standard shortcut for search functionality
 if (e.key === "k" && (e.metaKey || e.ctrlKey)) {

155-159: Improve phone number validation logic

The phone number validation logic could be more robust. Consider extracting the validation to a separate function.

+const isValidPhoneNumber = (value: string) => value.length === 13 || value.length === 3;
+
 if (
-  selectedOption.key === "phone_number"
-    ? searchValue.length == 13 || searchValue.length == 3
-    : selectedOption.value !== searchValue
+  selectedOption.key === "phone_number"
+    ? isValidPhoneNumber(searchValue)
+    : selectedOption.value !== searchValue
 ) {

397-409: Add data-testid for clear search button

The clear search button is a good addition, but should include a data-testid for testing.

 <Button
   variant="ghost"
   size="sm"
+  data-testid={`${id}__clear-search`}
   className="w-full justify-start text-muted-foreground"
   onClick={() => {
     setSearchValue("");
   }}
 >

281-292: Enhance accessibility with ARIA labels

While the basic ARIA implementation is good, consider adding more descriptive labels.

 <Button
   variant="ghost"
   className="focus:ring-0 px-2 ml-1"
   size="sm"
+  aria-label={t("open_search_options")}
   onClick={() => setOpen(true)}
 >
📜 Review details

Configuration used: CodeRabbit UI
Review profile: CHILL
Plan: Pro

📥 Commits

Reviewing files that changed from the base of the PR and between ed62635 and 45d3717.

📒 Files selected for processing (1)
  • src/components/Common/SearchByMultipleFields.tsx (7 hunks)
⏰ Context from checks skipped due to timeout of 90000ms (2)
  • GitHub Check: cypress-run (1)
  • GitHub Check: OSSAR-Scan
🔇 Additional comments (1)
src/components/Common/SearchByMultipleFields.tsx (1)

70-70: LGTM! Improved state management

Good improvements:

  • Edge case handling for initialOptionIndex
  • Performance optimization with memoized unselectedOptions

Also applies to: 104-107

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

🧹 Nitpick comments (2)
src/components/Common/SearchByMultipleFields.tsx (2)

117-124: Consider adding ARIA attributes for keyboard shortcuts

While the keyboard shortcut implementation is good, consider adding ARIA attributes to improve accessibility.

+  aria-keyshortcuts="Control+K"

391-403: Consider adding keyboard shortcut for clear search

The clear search button is a good addition, but consider adding a keyboard shortcut (e.g., Esc when the input is focused) for better user experience.

 useEffect(() => {
   const handleKeyDown = (e: KeyboardEvent) => {
+    if (e.key === "Escape" && document.activeElement === inputRef.current) {
+      setSearchValue("");
+      return;
+    }
     // ... existing code
   };
 }, []);
📜 Review details

Configuration used: CodeRabbit UI
Review profile: CHILL
Plan: Pro

📥 Commits

Reviewing files that changed from the base of the PR and between a3d83b5 and 73b33ec.

📒 Files selected for processing (1)
  • src/components/Common/SearchByMultipleFields.tsx (6 hunks)
🔇 Additional comments (5)
src/components/Common/SearchByMultipleFields.tsx (5)

31-31: LGTM! Platform-specific import added

The addition of isAppleDevice utility enables platform-specific keyboard shortcut display.


45-45: Breaking Change: initialOptionIndex is now required

This change was previously flagged and remains a concern.


104-107: LGTM! Efficient memoization

The memoization of unselectedOptions is well-implemented and will prevent unnecessary recalculations.


178-215: LGTM! Well-implemented device-specific UI

The keyboard shortcut display is properly implemented with device-specific considerations and good visual feedback.

Also applies to: 219-250


70-70: Consider edge case handling for initialOptionIndex

The fallback to 0 when initialOptionIndex is -1 might not be the best approach. Consider throwing an error or providing better validation.

✅ Verification successful

Fallback to 0 for initialOptionIndex is appropriate

The current implementation is correct and safe because:

  • The component requires at least one search option to function
  • The fallback ensures a valid initial state for the selectedOption reference
  • The component already has proper error handling for invalid inputs
🏁 Scripts executed

The following scripts were executed for the analysis:

Script:

#!/bin/bash
# Search for usages of SearchByMultipleFields to verify initialOptionIndex values
rg -A 2 "initialOptionIndex={" --type ts --type tsx

Length of output: 157


Script:

#!/bin/bash
# Find usages of SearchByMultipleFields component
rg -l "SearchByMultipleFields" --type ts --type tsx

# Get the full implementation of SearchByMultipleFields
cat src/components/Common/SearchByMultipleFields.tsx

Length of output: 14004

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

🧹 Nitpick comments (2)
src/components/Users/UserResetPassword.tsx (2)

228-231: Consider adding loading spinner for better UX.

While the button text changes to "updating", consider adding a loading spinner icon for better visual feedback.

 <Button
   type="submit"
   disabled={!form.formState.isDirty || isPending}
   variant="primary"
 >
+  {isPending && <CareIcon icon="l-spinner" className="mr-2 h-4 w-4 animate-spin" />}
   {isPending ? t("updating") : t("update_password")}
 </Button>

Line range hint 41-137: Consider rate limiting password attempts.

The password reset functionality could benefit from rate limiting to prevent brute force attempts. Consider implementing a mechanism to track failed attempts and temporarily lock the account after a certain threshold.

📜 Review details

Configuration used: CodeRabbit UI
Review profile: CHILL
Plan: Pro

📥 Commits

Reviewing files that changed from the base of the PR and between 73b33ec and 5bbfb59.

📒 Files selected for processing (1)
  • src/components/Users/UserResetPassword.tsx (1 hunks)
⏰ Context from checks skipped due to timeout of 90000ms (5)
  • GitHub Check: Redirect rules - care-ohc
  • GitHub Check: Header rules - care-ohc
  • GitHub Check: Pages changed - care-ohc
  • GitHub Check: cypress-run (1)
  • GitHub Check: OSSAR-Scan
🔇 Additional comments (2)
src/components/Users/UserResetPassword.tsx (2)

228-228: LGTM! Good improvement to prevent double submissions.

The addition of isPending to the button's disabled state is a good practice that prevents duplicate form submissions while the password reset is in progress.


Line range hint 89-91: Verify password reset mutation error handling.

The mutation setup doesn't include error handling. Ensure that API errors are properly caught and displayed to the user.

@AdityaJ2305
Copy link
Contributor Author

isPending was pending for submit button 😅
#9683

@AdityaJ2305
Copy link
Contributor Author

@rithviknishad @Jacobjeevan Ready for Review

Copy link

👋 Hi, @AdityaJ2305,
Conflicts have been detected against the base branch. Please rebase your branch against the base branch.


This message is automatically generated by prince-chrismc/label-merge-conflicts-action so don't hesitate to report issues/improvements there.

@github-actions github-actions bot added the Deploy-Failed Deplyment is not showing preview label Jan 13, 2025
@github-actions github-actions bot added merge conflict pull requests with merge conflict and removed merge conflict pull requests with merge conflict labels Jan 13, 2025
Copy link

👋 Hi, @AdityaJ2305,
Conflicts have been detected against the base branch. Please rebase your branch against the base branch.


This message is automatically generated by prince-chrismc/label-merge-conflicts-action so don't hesitate to report issues/improvements there.

@github-actions github-actions bot removed the merge conflict pull requests with merge conflict label Jan 13, 2025
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
Deploy-Failed Deplyment is not showing preview needs review needs testing
Projects
None yet
3 participants