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(widgets): added-blind-transfer #422

Closed
wants to merge 8 commits into from

Conversation

adhmenon
Copy link
Contributor

COMPLETES SPARK-568988

Description

  • Added code for consult & transfer user interface
  • Code has also been added for blind transfer working
  • Unit tests for the above and some missed out cases

Vidcast - https://app.vidcast.io/share/24962b05-3a16-4b60-8c83-9333efa121f3?playerMode=vidcast

Considerations

  • Right now, this PR will get merge conflicts once the components PR is in, will resolve and update.
  • I have broken up the ticket into 3 - one for transfer, consult & queues
  • Test coverage is low in some areas as I was not able to get them working in time. WIP
  • There is a small UI glitch in the popover, will fix it with the next PR.
  • Error screens are a WIP for now until UX finalises - will be done in the next PR.

Tests

  • Tested with EXTENSION and BROWSER mode
  • Tested old flows such as decline/cancel call to see that nothing was broken
  • Tested it with multi login to see that state persists.

Copy link

coderabbitai bot commented Mar 25, 2025

Important

Review skipped

Auto reviews are disabled on base/target branches other than the default branch.

Please check the settings in the CodeRabbit UI or the .coderabbit.yaml file in this repository. To trigger a single review, invoke the @coderabbitai review command.

You can disable this status message by setting the reviews.review_status to false in the CodeRabbit configuration file.

📝 Walkthrough

Walkthrough

This PR introduces a new boolean property isAgentTransferred into the core state management, with corresponding methods and event handlers to track and update agent transfer status. The Store and associated type declarations are extended with this property along with a setter method. The changes extend into the event handling layer by adding a new event (AGENT_BLIND_TRANSFERRED) and its handling logic in the StoreWrapper, which adjusts task wrap-up behavior based on the transfer state. In parallel, new React components and presentational elements are implemented to support interactive call control features, including lists and popovers for selecting buddy agents. Enhancements also include updates to the helper hook for managing buddy agents and call transfers, as well as complementary changes in SCSS and unit tests to verify and maintain the expanded functionality.

Possibly related PRs


🪧 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.
  • @coderabbitai resolve resolve all the CodeRabbit review comments.
  • @coderabbitai plan to trigger planning for file edits and PR creation.
  • @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.

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.

@adhmenon
Copy link
Contributor Author

@coderabbitai review

Copy link

coderabbitai bot commented Mar 25, 2025

✅ Actions performed

Review triggered.

Note: CodeRabbit is an incremental review system and does not re-review already reviewed commits. This command is applicable only when automatic reviews are paused.

Copy link

This pull request is automatically being deployed by Amplify Hosting (learn more).

Access this pull request here: https://pr-422.d1b38q61t1z947.amplifyapp.com

Copy link

@coderabbitai coderabbitai bot left a comment

Choose a reason for hiding this comment

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

Actionable comments posted: 2

🧹 Nitpick comments (16)
packages/contact-center/task/src/CallControl/CallControlCustomComponents/call-control-popover.presentational.tsx (3)

19-19: Consider renaming the variable or implementing actual filtering

The variable name filteredAgents suggests that some filtering is being applied, but it's directly assigned from buddyAgents without any filtering logic. Either rename it to something like displayedAgents or implement actual filtering if that's the intended functionality.

-  const filteredAgents = buddyAgents;
+  const displayedAgents = buddyAgents;

37-52: Consider displaying a loading state when fetching agents

There's currently no handling for the loading state when fetching buddy agents. Users might benefit from seeing a loading indicator while agents are being loaded.

Consider adding a loading state to improve user experience:

-      <ListNext listSize={filteredAgents.length} className="agent-agent-list">
-        {filteredAgents.map((agent) => (
+      {isLoading ? (
+        <div className="agent-loading">
+          <Text tagName="small" type="body-secondary">
+            Loading agents...
+          </Text>
+        </div>
+      ) : (
+        <ListNext listSize={displayedAgents.length} className="agent-agent-list">
+          {displayedAgents.map((agent) => (

33-35: The Tab system appears to be over-engineered for a single tab

The component is using a TabList component with only one tab option. If there are no plans to add more tabs in the future, consider simplifying this to a plain header.

If no additional tabs are planned, consider simplifying:

-      <TabListNext
-        aria-label="Tabs"
-        className="agent-tablist"
-        hasBackground={false}
-        style={{marginTop: '0'}}
-        onTabSelection={(key) => setSelectedTab(key as string)}
-      >
-        <TabNext key="Agents" active={selectedTab === 'Agents'}>
-          Agents
-        </TabNext>
-      </TabListNext>
+      <div className="agent-tablist" style={{marginTop: '0'}}>
+        <Text tagName="h4" type="body-medium-bold">
+          Agents
+        </Text>
+      </div>
packages/contact-center/store/src/store.types.ts (3)

44-44: Add documentation for the new isAgentTransferred property

The new property lacks JSDoc comments to explain its purpose and usage. Adding documentation will help other developers understand when and how to use this property.

Add appropriate documentation:

+  /**
+   * Flag indicating whether the agent has been transferred.
+   */
   isAgentTransferred: boolean;

70-70: Add documentation for the new setIsAgentTransferred method

This method lacks JSDoc comments to explain its purpose. Adding documentation will improve code maintainability.

Add appropriate documentation:

+  /**
+   * Sets the agent transfer status.
+   * @param value - Boolean indicating whether the agent has been transferred.
+   */
   setIsAgentTransferred(value: boolean): void;

97-97: Add documentation for the new AGENT_BLIND_TRANSFERRED enum value

The new enum value lacks an explanatory comment. Adding a brief comment will clarify its purpose.

Add a comment for the new enum value:

+  // Event triggered when an agent is blind transferred
   AGENT_BLIND_TRANSFERRED = 'AgentBlindTransferred',
packages/contact-center/task/tests/CallControl/call-control-list-item.presentational.tsx (1)

31-37: Be more specific about which console errors to suppress

Currently, all console errors are being suppressed, which could hide actual issues during testing. It would be better to specifically mock only the expected errors.

Consider a more targeted approach to error suppression:

beforeAll(() => {
-  jest.spyOn(console, 'error').mockImplementation(() => {});
+  // Only mock specific React-related warnings that are expected
+  const originalError = console.error;
+  jest.spyOn(console, 'error').mockImplementation((message, ...args) => {
+    // Filter out specific expected warnings
+    if (message.includes('Warning: ReactDOM.render is no longer supported') || 
+        message.includes('Warning: Failed prop type')) {
+      return;
+    }
+    // Log other errors normally
+    originalError(message, ...args);
+  });
});
packages/contact-center/task/src/CallControl/call-control.presentational.tsx (2)

153-156: Consider adding a loading state when fetching buddy agents.

The code calls loadBuddyAgents() when the popover opens, but doesn't provide any visual feedback to the user while the agents are being loaded.

Add a loading state:

 const [showAgentMenu, setShowAgentMenu] = useState(false);
 const [agentMenuType, setAgentMenuType] = useState<'Consult' | 'Transfer' | null>(null);
+const [isLoadingAgents, setIsLoadingAgents] = useState(false);

 // ...

 onPress={() => {
   // If popover is already visible, we close it
   if (showAgentMenu && agentMenuType === button.menuType) {
     setShowAgentMenu(false);
     setAgentMenuType(null);
   } else {
     setAgentMenuType(button.menuType as 'Consult' | 'Transfer');
     setShowAgentMenu(true);
+    setIsLoadingAgents(true);
     loadBuddyAgents()
+      .finally(() => setIsLoadingAgents(false));
   }
 }}

Then in the CallControlPopoverPresentational component, pass the loading state:

 <CallControlPopoverPresentational
   heading={button.menuType}
   buttonIcon={button.icon}
   buddyAgents={buddyAgents}
+  isLoading={isLoadingAgents}
   onAgentSelect={(agentId) => {
     // ...
   }}
 />

173-188: Add empty state handling for buddy agents list.

The code doesn't handle the case when buddyAgents is empty or failed to load. Providing feedback to the user in these scenarios would improve the user experience.

Consider adding an empty state message:

 {showAgentMenu && agentMenuType === button.menuType ? (
   <CallControlPopoverPresentational
     heading={button.menuType}
     buttonIcon={button.icon}
     buddyAgents={buddyAgents}
+    emptyStateMessage="No available agents found"
     onAgentSelect={(agentId) => {
       setShowAgentMenu(false);
       setAgentMenuType(null);
       if (agentMenuType === 'Consult') {
         consultCall();
       } else {
         transferCall(agentId, 'agent');
       }
     }}
   />
 ) : null}
packages/contact-center/task/src/CallControl/call-control.styles.scss (2)

151-160: Add keyboard focus styles for accessibility.

The hover button styles only address mouse interactions. For better accessibility, add equivalent focus styles for keyboard navigation.

 .call-control-list-item .hover-button {
   opacity: 0;
   transition: opacity 0.3s;
   pointer-events: none;
 }

 .call-control-list-item:hover .hover-button {
   opacity: 1;
   pointer-events: auto;
 }
+
+.call-control-list-item .hover-button:focus-visible {
+  opacity: 1;
+  pointer-events: auto;
+  outline: 2px solid var(--mds-color-theme-outline-focus-normal);
+}

162-165: Consider more descriptive class name.

The class name .custom-text is very generic and might clash with other styles. Consider using a more specific name related to its usage context.

-.custom-text {
+.agent-popover-description-text {
   margin: 0 !important;
   line-height: 1.2 !important;
 }
packages/contact-center/store/src/storeEventsWrapper.ts (2)

246-249: Improve handling of unused parameter.

Instead of using an eslint-disable comment for the unused parameter, it's better to follow the JavaScript convention of prefixing unused parameters with an underscore.

-// eslint-disable-next-line @typescript-eslint/no-unused-vars
-handleAgentBlindTransferred = (event) => {
+handleAgentBlindTransferred = (_event) => {
   this.setIsAgentTransferred(true);
 };

441-455: Well-implemented getBuddyAgents method with error handling.

The method correctly fetches buddy agents with proper error handling and logging. The implementation follows the pattern of other methods in the file.

Consider enhancing the method to support more filtering options:

-getBuddyAgents = async (): Promise<Array<BuddyDetails>> => {
+getBuddyAgents = async (
+  options: { mediaType?: string; state?: string } = { mediaType: 'telephony', state: 'Available' }
+): Promise<Array<BuddyDetails>> => {
   try {
     const response = await this.store.cc.getBuddyAgents({
-      mediaType: 'telephony',
-      state: 'Available',
+      mediaType: options.mediaType || 'telephony',
+      state: options.state || 'Available',
     });
     return response?.data?.agentList || [];
   } catch (error) {
     this.store.logger.error(`Error fetching buddy agents: ${error}`, {
       module: 'cc-store#storeEventsWrapper.ts',
       method: 'getBuddyAgents',
     });
     return [];
   }
 };
packages/contact-center/task/tests/CallControl/call-control.presentational.tsx (3)

43-53: Well structured mock for CallControlPopoverPresentational.

The mock implementation includes a test button that triggers the onAgentSelect callback, which allows for testing agent selection functionality.

Consider using a dynamic agent ID instead of hardcoding 'agent1' to make the tests more flexible and maintainable:

- <button data-testid="AgentSelectButton" onClick={() => props.onAgentSelect('agent1')}>
+ <button data-testid="AgentSelectButton" onClick={() => props.onAgentSelect(props.buddyAgents?.[0]?.id || 'agent1')}>

105-110: Targeted test for hold button icon.

Good test that verifies the correct icon is displayed when the call is not on hold.

Consider adding a test for when the call is on hold as well:

it('renders the correct hold button icon based on currentTask (on hold)', () => {
  const onHoldProps = {
    ...baseProps,
    currentTask: {
      ...baseProps.currentTask,
      data: {
        ...baseProps.currentTask.data,
        interaction: {
          ...baseProps.currentTask.data.interaction,
          media: {
            '1': { isHold: true },
          },
        },
      },
    },
  };
  render(<CallControlPresentational {...onHoldProps} />);
  const holdButton = screen.getAllByTestId('ButtonCircle')[0];
  expect(holdButton).toHaveTextContent('play-bold');
});

55-144: Consider adding tests for recording functionality.

I notice you have mock functions for toggleRecording and setIsRecording, but no tests that specifically verify this functionality. Consider adding a test to ensure recording can be toggled correctly:

it('calls toggleRecording when recording button is clicked', () => {
  render(<CallControlPresentational {...baseProps} />);
  // Assuming the recording button is the 1st or 5th button in the array
  // Adjust index based on actual button order
  const buttons = screen.getAllByTestId('ButtonCircle');
  fireEvent.click(buttons[1]); // Adjust index as needed
  expect(mockToggleRecording).toHaveBeenCalled();
});
📜 Review details

Configuration used: .coderabbit.yaml
Review profile: CHILL
Plan: Pro

📥 Commits

Reviewing files that changed from the base of the PR and between 6f62524 and 3eb97f7.

📒 Files selected for processing (13)
  • packages/contact-center/store/src/store.ts (1 hunks)
  • packages/contact-center/store/src/store.types.ts (5 hunks)
  • packages/contact-center/store/src/storeEventsWrapper.ts (8 hunks)
  • packages/contact-center/task/src/CallControl/CallControlCustomComponents/call-control-list-item.presentational.tsx (1 hunks)
  • packages/contact-center/task/src/CallControl/CallControlCustomComponents/call-control-popover.presentational.tsx (1 hunks)
  • packages/contact-center/task/src/CallControl/call-control.presentational.tsx (4 hunks)
  • packages/contact-center/task/src/CallControl/call-control.styles.scss (2 hunks)
  • packages/contact-center/task/src/helper.ts (4 hunks)
  • packages/contact-center/task/src/task.types.ts (3 hunks)
  • packages/contact-center/task/tests/CallControl/call-control-list-item.presentational.tsx (1 hunks)
  • packages/contact-center/task/tests/CallControl/call-control-popover.presentational.tsx (1 hunks)
  • packages/contact-center/task/tests/CallControl/call-control.presentational.tsx (3 hunks)
  • packages/contact-center/task/tests/helper.ts (1 hunks)
🧰 Additional context used
🧠 Learnings (2)
packages/contact-center/task/tests/CallControl/call-control-popover.presentational.tsx (1)
Learnt from: Shreyas281299
PR: webex/widgets#362
File: packages/contact-center/task/tests/CallControl/call-control.presentational.tsx:0-0
Timestamp: 2025-03-20T09:47:38.389Z
Learning: UI testing is not mandatory for React components in the webex/widgets repository.
packages/contact-center/task/tests/CallControl/call-control-list-item.presentational.tsx (1)
Learnt from: Shreyas281299
PR: webex/widgets#362
File: packages/contact-center/task/tests/CallControl/call-control.presentational.tsx:0-0
Timestamp: 2025-03-20T09:47:38.389Z
Learning: UI testing is not mandatory for React components in the webex/widgets repository.
🧬 Code Definitions (2)
packages/contact-center/task/src/task.types.ts (1)
packages/contact-center/store/src/store.types.ts (1)
  • BuddyDetails (135-135)
packages/contact-center/task/src/CallControl/CallControlCustomComponents/call-control-popover.presentational.tsx (1)
packages/contact-center/store/src/storeEventsWrapper.ts (1)
  • agentId (44-46)
⏰ Context from checks skipped due to timeout of 90000ms (1)
  • GitHub Check: AWS Amplify Console Web Preview
🔇 Additional comments (32)
packages/contact-center/store/src/store.ts (1)

38-38: Good implementation of the isAgentTransferred property!

This property addition supports the blind transfer functionality as specified in the PR objectives. The naming is clear and follows the same conventions as other boolean state flags in the Store class.

packages/contact-center/task/tests/CallControl/call-control-popover.presentational.tsx (1)

1-74: Well-structured test suite for the CallControlPopoverPresentational component!

The tests thoroughly verify rendering and interaction behaviors, including heading display, agent list rendering, empty state handling, and agent selection functionality. The mocks are set up appropriately to isolate component testing.

Note from learnings: I acknowledge that UI testing is not mandatory for React components in this repository, but these tests add good coverage to the new functionality.

packages/contact-center/task/src/CallControl/CallControlCustomComponents/call-control-list-item.presentational.tsx (1)

1-59: Clean implementation of the call control list item component!

This component is well-designed with:

  • Clear props interface
  • Good use of Momentum UI components
  • Smart handling of avatar initials
  • Consistent styling approach
  • Proper separation of concerns

The implementation supports the blind transfer functionality by providing a reusable list item that can display agent information with action buttons.

packages/contact-center/task/tests/helper.ts (2)

1087-1108: Good test coverage for buddy agent loading functionality!

The test properly verifies that the loadBuddyAgents method correctly fetches and stores buddy agents. The use of mocks and assertions is appropriate for validating this functionality.


1110-1130: Well-implemented test for the transferCall method!

This test properly verifies that the transferCall method correctly calls the task's transfer function with the expected parameters. The test confirms essential functionality for the blind transfer feature.

packages/contact-center/task/tests/CallControl/call-control-list-item.presentational.tsx (2)

53-59: LGTM! Good test coverage for the component rendering

The test effectively verifies that the component renders with the correct content and properly displays user initials.


69-74: LGTM! Good interaction testing

The test properly verifies that clicking the button triggers the callback function.

packages/contact-center/task/src/task.types.ts (2)

201-221: Well-documented new properties and methods!

The new properties and methods for buddy agents and call transfers have clear and comprehensive JSDoc comments, which is excellent for code maintainability.


240-243: LGTM! Properly updated type exports

The CallControlPresentationalProps type has been correctly updated to include the new properties.

packages/contact-center/task/src/CallControl/call-control.presentational.tsx (4)

8-8: Good import of the new component.

The addition of CallControlPopoverPresentational enhances the component architecture by separating the popover UI logic into its own component.


13-14: State variables appropriately manage popover visibility and type.

These state variables cleanly control the display and behavior of the agent menu, supporting both Consult and Transfer operations.


29-32: New props enhance component functionality.

The addition of buddyAgents, loadBuddyAgents, transferCall, and consultCall props properly enables the new agent selection and transfer features.


89-102: Button configurations properly define new call features.

The new Consult and Transfer buttons are well-configured with appropriate icons, tooltips, and menuType properties.

packages/contact-center/task/src/helper.ts (3)

133-142: Well-implemented buddy agents loading with error handling.

The implementation for loading buddy agents is clean, with proper error handling and state management. The use of useCallback is appropriate to prevent unnecessary function recreations.


272-283: Good implementation of transferCall with error handling.

The transfer call implementation is solid with proper error handling. The function is well-structured and follows the pattern of other methods in this file.


301-304: Properly exposed new functionality in hook return.

The new functions are correctly added to the hook's return object, making them available to components.

packages/contact-center/task/src/CallControl/call-control.styles.scss (1)

118-149: Well-structured styles for the agent popover.

The styles for the agent popover are well-organized, following the existing style patterns in the file. The CSS nesting and naming conventions are consistent.

packages/contact-center/store/src/storeEventsWrapper.ts (5)

88-90: Good implementation of isAgentTransferred getter.

The getter correctly accesses the underlying store state.


139-141: Well-implemented setter for isAgentTransferred.

The setter follows the same pattern as other setters in the file.


256-256: Good condition update to consider transferred state.

Adding the check for this.store.isAgentTransferred ensures proper handling of transferred calls.


279-279: Appropriately reset transfer state on wrap-up.

Resetting isAgentTransferred to false on task wrap-up ensures clean state management.


302-303: Good implementation of event listeners for blind transfers.

The event listeners are correctly set up for the new blind transfer functionality, following the same pattern as other event listeners in the file.

Also applies to: 338-339

packages/contact-center/task/tests/CallControl/call-control.presentational.tsx (10)

3-3: Good addition of fireEvent for interaction testing.

Adding the fireEvent import enables you to test user interactions like button clicks, which is essential for properly testing the component's behavior.


7-32: Well implemented mock components with interactive functionality.

The enhanced mock implementations for ButtonPill, ButtonCircle, PopoverNext, SelectNext, and TooltipNext now properly handle props like onPress, disabled, and className. This allows for more realistic testing of component interactions.


35-41: Improved Icon mock with className support.

The updated Icon mock now handles the className prop, which provides better testing for styling conditions.


64-67: Good addition of mock functions for new features.

The new mock functions for buddy agents and call operations are essential for testing the new transfer and consult functionality.


69-94: Well structured baseProps with all required component properties.

Organizing the props in a baseProps object makes the tests more maintainable and reduces duplication. All necessary props for the component are included.


100-104: Simple test for component rendering.

This test ensures the basic component renders correctly.


111-117: Good interaction test for the hold button.

This test verifies that the toggleHold function is called with the correct value when the hold button is clicked.


118-124: Good interaction test for the end call button.

This test verifies that the endCall function is called when the end call button is clicked.


125-134: Well structured test for agent consultation.

This test properly verifies the consultation flow by clicking the consult button and then selecting an agent.


135-143: Comprehensive test for agent transfer.

This test correctly verifies that transferCall is called with the right parameters when an agent is selected for transfer.

@@ -35,6 +35,7 @@ class Store implements IStore {
lastStateChangeTimestamp?: number;
lastIdleCodeChangeTimestamp?: number;
showMultipleLoginAlert: boolean = false;
isAgentTransferred: boolean = false;
Copy link
Contributor Author

Choose a reason for hiding this comment

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

State so we can check when the agent has been transferred.

@@ -93,6 +94,7 @@ enum TASK_EVENTS {
CONTACT_RECORDING_PAUSED = 'ContactRecordingPaused',
CONTACT_RECORDING_RESUMED = 'ContactRecordingResumed',
AGENT_WRAPPEDUP = 'AgentWrappedUp',
AGENT_BLIND_TRANSFERRED = 'AgentBlindTransferred',
Copy link
Contributor Author

Choose a reason for hiding this comment

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

As discussed, we will update above state on this event.

@@ -10,6 +10,7 @@ import {
IdleCode,
IContactCenter,
ITask,
BuddyDetails,
Copy link
Contributor Author

Choose a reason for hiding this comment

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

We need this for the budyd agents details.

@@ -418,6 +437,22 @@ class StoreWrapper implements IStoreWrapper {
});
});
};

getBuddyAgents = async (): Promise<Array<BuddyDetails>> => {
Copy link
Contributor Author

Choose a reason for hiding this comment

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

Method so we can extract buddy agents - queues will also be similar to this.

import {Icon} from '@momentum-design/components/dist/react';
import classnames from 'classnames';

export interface CallControlListItemPresentationalProps {
Copy link
Contributor Author

Choose a reason for hiding this comment

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

This entire class is for the custom list item, right now all it consists of is an avatar, title, subtitle and a button which are all highly customisable.
Open to any suggestions to make it better.

@@ -130,6 +130,17 @@ export const useCallControl = (props: useCallControlProps) => {
const [isHeld, setIsHeld] = useState<boolean | undefined>(undefined);
const [isRecording, setIsRecording] = useState(true);

const [buddyAgents, setBuddyAgents] = useState<BuddyDetails[]>([]);
Copy link
Contributor Author

Choose a reason for hiding this comment

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

Code to extract buddy agents, used logger to log any errors here.

};

// New consult callback method (empty for now)
const consultCall = async () => {
Copy link
Contributor Author

Choose a reason for hiding this comment

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

Temporary, will add in the next PR.

import '@testing-library/jest-dom';
import CallControlListItemPresentational from '../../src/CallControl/CallControlCustomComponents/call-control-list-item.presentational';

jest.mock('@momentum-ui/react-collaboration', () => ({
Copy link
Contributor Author

Choose a reason for hiding this comment

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

Added tests here to get 100% coverage.

import '@testing-library/jest-dom';
import CallControlPopoverPresentational from '../../src/CallControl/CallControlCustomComponents/call-control-popover.presentational';

jest.mock('../../src/CallControl/CallControlCustomComponents/call-control-list-item.presentational', () => {
Copy link
Contributor Author

Choose a reason for hiding this comment

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

Same for this.

import React from 'react';
import {render, screen} from '@testing-library/react';
import {render, screen, fireEvent} from '@testing-library/react';
Copy link
Contributor Author

Choose a reason for hiding this comment

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

Here coverage is still an issue, added tests for cases like hold/end etc. which were missed before, however testing the popover is an issue - some rendering problems for the test. WIP, but because of that coverage is a bit low.

@adhmenon adhmenon added the validated Indicates that the PR is ready for actions label Mar 25, 2025
@bhabalan bhabalan changed the title feat(widgets): added-blind-trasnfer feat(widgets): added-blind-transfer Mar 26, 2025
@adhmenon adhmenon changed the title feat(widgets): added-blind-transfer feat(widgets): added-blind-trasnfer Mar 26, 2025
@adhmenon adhmenon changed the title feat(widgets): added-blind-trasnfer feat(widgets): added-blind-transfer Mar 26, 2025
if (agentMenuType === 'Consult') {
consultCall();
} else {
transferCall(agentId, 'agent');
Copy link
Contributor

Choose a reason for hiding this comment

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

We should have second parameter being picked up based on selected type right. Also these buttons should appear as radio buttons and option to select them. Queue is not implemented yet but dial number can be added right as one of the menu options

Copy link
Contributor Author

@adhmenon adhmenon Mar 26, 2025

Choose a reason for hiding this comment

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

Figma doesn't seem to have dial number... we only have "Agents" and "Queues" as of now.
Also, yes this method will change once I have consult implemented and also once I add queues.

.agent-popover-content {
display: flex;
flex-direction: column;
padding: 8px;
Copy link
Contributor

Choose a reason for hiding this comment

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

Can these be not converted to rem? 1 rem is 16 px so we can convert the values

@@ -286,6 +299,9 @@ class StoreWrapper implements IStoreWrapper {

task.on(TASK_EVENTS.AGENT_WRAPPEDUP, this.handleTaskWrapUp);

// Listen for AGENT_BLIND_TRANSFERRED event
task.on(TASK_EVENTS.AGENT_BLIND_TRANSFERRED, () => this.handleAgentBlindTransferred(task));
Copy link
Contributor

Choose a reason for hiding this comment

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

Why are we listening for this event? I don't see SDK emitting any event for trasnfer. Complete dependency is there on promise resolve for transfer

Copy link
Contributor Author

@adhmenon adhmenon Mar 26, 2025

Choose a reason for hiding this comment

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

Edit:
I did some testing... and it was working with just promise based... I was in the wrong here.. thought it might cause some async issues and that end call would arrive before we set the state in store...

mediaType: 'telephony',
state: 'Available',
});
return response?.data?.agentList || [];
Copy link
Contributor

Choose a reason for hiding this comment

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

Why do we have ? after response? In case of success, response will always be there and have data right. And even agentList is mandatory field but it can be empty array sometime.
And failure should go to catch block

};

try {
await currentTask.transfer(transferPayload);
Copy link
Contributor

Choose a reason for hiding this comment

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

Why are we not using the response here that we receive as part of promise resolve?

Copy link
Contributor Author

Choose a reason for hiding this comment

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

I think we don't need to use the response here, not sure how helpful it would be also.

* @param destination - The destination to transfer the call to.
* @param destinationType - The type of destination.
*/
transferCall: (destination: string, destinationType: string) => void;
Copy link
Contributor

Choose a reason for hiding this comment

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

We have a type in SDK for destinationType so instead of keeping it as string why don't we use that enum type

<TooltipNext
key={index}
triggerComponent={
<ButtonCircle className={button.className} onPress={button.onClick} disabled={button.disabled}>
Copy link
Contributor

Choose a reason for hiding this comment

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

We should add aria-label to all the Button or icons we have for accessibility purpose

@adhmenon adhmenon requested a review from Kesari3008 March 26, 2025 10:34
@adhmenon
Copy link
Contributor Author

Closing this PR, opened a new one #423

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
validated Indicates that the PR is ready for actions
Projects
None yet
Development

Successfully merging this pull request may close these issues.

2 participants