This library's goal is to abstract all the different headset implementations behind a single interface.
This project has a React test app bootstrapped with create-react-app.
At this moment (12/10/2021) there are three supported vendors already handled with this library:
- Plantronics/Poly
- Sennheiser/EPOS
- Jabra
As of 11/3/2022 we now support:
- Yealink
# npm
npm install --save softphone-headset-vendors
# yarn
yarn softphone-headset-vendors
This will create a new instance of the headset service if one doesn't already exist. It's important to note the original instance will always be returned, even if you pass in different config options so make sure you get it right the first time. There's only one relevant option for the headset config, see below.
static getInstance(config: ImplementationConfig);
interface ImplementationConfig {
logger: any;
}
This is a computed value that returns a list of selectable headset vendors based on browser/environment compatibility. It does not ensure the required 3rd party software is installed. That gets checked when a vendor is selected or changed.
The selected headset vendor is determined by the active mic. This method notifies headset service the active mic has changed and it should determine if the "new" mic has an associated vendor implementation.
activeMicChange (newMicLabel: string): void;
Params:
newMicLabel: string
Required: This label should match the device label returned from navigator.getUserMedia()
Allows you to manually change the selected headset. This is an alternative to activeMicChange(...)
.
changeImplementation (implementation: VendorImplementation | null, deviceLabel: string): Promise<void>
params:
implementation: VendorImplementation | null
Required: The desired vendor implementationdeviceLabel: string
Required: This can be null if theimplementation
is null, otherwise this will be the device associated with the vendor implementation to which you'd like to connect.
Notifies the headset you have an incoming call. In most cases this will result in the headset ringing.
incomingCall (callInfo: CallInfo, hasOtherActiveCalls?: boolean): Promise<any>
params:
callInfo: CallInfo
Required: The desired vendor implementation- Basic interface
interface CallInfo { conversationId: string, contactName?: string }
conversationId: string
: Required: Most of the vendors use conversation or call ids in order to help maintain proper state of the headset. This is the conversationId associated with the incoming call.contactName: string
: Optional: Some vendors will announce the caller through the headset if thecontactName
is provided.
- Basic interface
hasOtherActiveCalls?: boolean
Required: This can be null if theimplementation
is null, otherwise this will be the device associated with the vendor implementation to which you'd like to connect.
Notifies the headset you are placing an outgoing call.
outgoingCall (callInfo: CallInfo): Promise<any>
params:
callInfo: CallInfo
Required: The desired vendor implementation- Basic interface
interface CallInfo { conversationId: string, contactName?: string }
conversationId: string
: Required: Most of the vendors use conversation or call ids in order to help maintain proper state of the headset. This is the conversationId associated with the incoming call.contactName: string
: Optional: Some vendors will announce the caller through the headset if thecontactName
is provided.
- Basic interface
Notifies the headset you are answering an incoming call. This will end the ringing and enable headset controls.
answerCall (conversationId: string): Promise<any>
params:
conversationId: string
: Required: Most of the vendors use conversation or call ids in order to help maintain proper state of the headset. This is the conversationId associated with the incoming call.
Notifies the headset you are rejecting an incoming call. This will end the ringing.
rejectCall (conversationId: string): Promise<any>
params:
conversationId: string
: Required: Most of the vendors use conversation or call ids in order to help maintain proper state of the headset. This is the conversationId associated with the incoming call.
Tells the headset to mute or unmute.
setMute (value: boolean): Promise<any>
params:
value: boolean
: Required: Iftrue
, the headset will be muted. Iffalse
, the headset will be unmuted.
Tells the headset you are holding or resuming a call.
setHold (conversationId: string, value: boolean): Promise<any>
params:
conversationId: string
: Required: The id associated with the conversation you'd are holding or resuming.value: boolean
: Required: Iftrue
, the headset will be held. Iffalse
, the headset will be resumed.
Tells the headset a call is ending.
endCall (conversationId: string, hasOtherActiveCalls?: boolean): Promise<any>
params:
conversationId: string
: Required: The id associated with the conversation you'd are holding or resuming.hasOtherActiveCalls?: boolean
: Optional: Some vendors differ in how much state they manage across multiple calls and it has to be shimmed. This allows us to make better decisions in those cases.
Ends all calls and returns the headset to a vanilla state.
endAllCalls (): Promise<void>
There are cases where 3rd party software needs to be started in order for the headset to connect. The method allows you to retry the connection after spinning up 3rd party software.
retryConnection (micLabel: string): Promise<void>
Params:
micLabel: string
Required: This label should match the device label returned from navigator.getUserMedia()
Returns the current connection state of the headset.
connectionStatus (): DeviceConnectionStatus
Returns:
type DeviceConnectionStatus = 'checking' | 'running' | 'notRunning' | 'noVendor';
The Headset service does not explicitly emit events itself. It uses RxJS observables to emit the events which are then subscribed to within the consuming app.
Event emitted when a user presses the answer call button during an incoming call on their selected device. The event includes the event name
as it is interpretted by the headset and a collection of items that may help with logging (event
). It can also potentially have a code
that corresponds to the event.
Declaration:
headset.headsetEvents.subscribe(event: {
event: 'deviceAnsweredCall',
payload: {
name: string,
code?: string,
event: { `containing various items mostly for logging purposes` }
}
} => {
if (event.event === 'deviceAnsweredCall') {
sdk.acceptPendingSession();
}
})
Value of event:
event: HeadsetEvents
- string value emitted by the headset library to determine what event had just occurredpayload:
- object containingname: string
- Name of the recent event as interpretted by the headset deviceevent
: { containing various items mostly for logging purposes}code?: string
- Optional: A string value of a number that represents the action that was just taken. Not all vendors supply a code which is why it is only optional
Event emitted when a user presses the answer call button while in an active call on their selected device. The event includes the event name
as it is interpretted by the headset and a collection of items that may help with logging (event
). It can also potentially have a code
that corresponds to the event.
Declaration:
headset.headsetEvents.subscribe(event: {
event: 'deviceEndedCall',
payload: {
name: string,
event: { `containing various items mostly for logging purposes` },
code?: string
} => {
if (event.event === 'deviceEndedCall') {
sdk.endSession({ conversationId });
}
}
})
Value of event:
event: HeadsetEvents
- string value emitted by the headset library to determine what event had just occurredpayload:
- object containingname: string
- Name of the recent event as interpretted by the headset deviceevent
: { containing various items mostly for logging purposes}code?: string
- Optional: A string value of a number that represents the action that was just taken. Not all vendors supply a code which is why it is only optional
Event emitted when a user presses the mute call button on their selected device. It doesn't matter if the device state is currently muted or unmuted,
this event will be emitted with the OPPOSITE value. For example, if the headset is currently muted, it will emit the event with the corresponding
value to unmute the device. The event includes the event name
as it is interpretted by the headset and a collection of items that may help with
logging (event
). It also comes with a value known as isMuted
which determines the event is trying to mute or unmute the call. It can also
potentially have a code
that corresponds to the event.
Declaration:
headset.headsetEvents.subscribe(event: {
event: 'deviceMuteStatusChanged',
payload: {
name: string,
event: { `containing various items mostly for logging purposes` },
isMuted: boolean,
code?: string
} => {
if (event.event === 'deviceMuteStatusChanged') {
sdk.setAudioMute(event.payload.isMuted);
}
}
})
Value of event:
event: HeadsetEvents
- string value emitted by the headset library to determine what event had just occurredpayload:
- object containingname: string
- Name of the recent event as interpretted by the headset deviceevent
: { containing various items mostly for logging purposes}isMuted: boolean
- the value determining if the event is to mute (true
) or unmute (false
) the devicecode?: string
- Optional: A string value of a number that represents the action that was just taken. Not all vendors supply a code which is why it is only optional
Event emitted when a user presses the hold call button on their selected device. It doesn't matter if the device state is currently on hold or not,
this event will be emitted with the OPPOSITE value. For example, if the headset is currently on hold, it will emit the event with the corresponding value to
resume the call. The event includes the event name
as it is interpretted by the headset and a collection of items that may help with logging (event
).
It also comes with a value known as holdRequested
which determines the event is trying to hold or resume the call. It will also have an optional value for toggle
.
It can also potentially have a code
that corresponds to the event.
Declaration:
headset.headsetEvents.subscribe(event: {
event: 'deviceHoldStatusChanged',
payload: {
name: string,
event: { `containing various items mostly for logging purposes` },
holdRequested: boolean,
code?: string
} => {
if (event.event === 'deviceHoldStatusChanged') {
sdk.setConversationHold(event.payload.holdRequested, event.payload.toggle);
}
}
})
Value of event:
event: HeadsetEvents
- string value emitted by the headset library to determine what event had just occurredpayload:
- object containingname: string
- Name of the recent event as interpretted by the headset deviceevent
: { containing various items mostly for logging purposes}holdRequested: boolean
- the value determining if the event is to hold (true
) or resume (false
) the callcode?: string
- Optional: A string value of a number that represents the action that was just taken. Not all vendors supply a code which is why it is only optional
This is a special event that is only necessary for specific devices. Certain devices (such as Jabra) support a technology known as
WebHID that requires additional permissions in order to use the call controls.
This event is emitted when a WebHID enabled device is selected. The event includes a callback
function that is required in order to
achieve additional permissions for WebHID
Declaration:
headset.headsetEvents.subscribe(event: {
event: 'webHidPermissionRequested',
payload: {
callback: Function
} => {
if (event.event === 'webHidPermissionRequested') {
event.payload.body.callback();
/* Please note: The above example will not work as is. You can't trigger the WebHID callback by simply calling, it must be triggered through user interaction such as clicking a button */
}
}
})
Value of event:
event: HeadsetEvents
- string value emitted by the headset library to determine what event had just occurredpayload:
- object containingcallback: Function
- the passed in function that will help achieve additional permissions for WebHID devices
Event emitted when a device implementation's connection status changes in some way. This can be the flags of isConnected
or isConnecting
changing in any way.
These flags are also included with the events payload.
Declaration:
headset.headsetEvents.subscribe(event: {
event: 'deviceConnectionStatusChanged',
payload: {
isConnected: boolean,
isConnecting: boolean
} => {
if (event.event === 'deviceConnectionStatusChanged') {
correspondingFunctionToHandleConnectionChange({event.payload.isConnected, event.payload.isConnecting});
}
}
})
Value of event:
event: HeadsetEvents
- string value emitted by the headset library to determine what event had just occurredpayload:
- object containingisConnected: boolean
- if the vendor implementation is fully connectedisConnecting: boolean
- if the vendor implementation is in the process of connecting
- The consuming app will first above anything else hit the HeadsetService (
headsets.ts
). - From there, the service will determine what vendor is currently selected out of the supported vendors above. This will also be a hub to call the proper functions that correspond with app to headset events (More on that later)
- Once the desired vendor has been determined, an instance of that vendor's adapter/service will be created. This adapter will interact with the service or sdk the vendor requires to communicate information to and from the headset.
- If an event is received from the headset itself, the vendor adapters will emit an event that
headset.ts
is listening for. This event will then be passed to the consuming app to properly reflect the state on screen to match that of the headset
Example 1 - User clicks mute in the consuming app:
- From the consuming app, the user clicks on an on-screen mute button
- The consuming app calls headsetService.mute(...)
- Which is passed to the corresponding function of the vendor adapter that aligns with the selected device (for example, plantronics.ts -> setMute(true))
- This function will then send a message to the headset itself
- The user will then see the light on their device that represents the "muted" state light up.
Example 2 - User presses the mute button from the headset:
- From the headset, the user presses the button which corresponds to mute
- This is then received by the vendor instance (for example sennheiser.ts)
- This event is then sent to
headset.ts
which in turn lets the consuming app know so that the screen can properly reflect the state of the headset
One of our supported vendors has began working with a technology known as WebHID. This is a relatively newer technology with a lot of promise but with its own caveats as well - https://wicg.github.io/webhid/
- At this moment, WebHID only works with Chromium browsers (Google Chrome/Microsoft Edge). Keep this in mind when developing and using the vendors we currently support
- In order to use WebHID, you must grant permissions for the site you are currently on. There is a function that must be called that causes a popup to show on screen where the user is then required to select their device and approve its use for WebHID purposes. This function MUST be called with user action (i.e. clicking a button). The solution we currently have in place is after the user changes and selects a new microphone, we check if it is the specific vendor that supports WebHID, then we emit an event that a consuming app should listen for. Once the consuming app receives that event, we will render an initial popup informing the user that additional permissions are required and prompting them to click either "Yes" or "No". Clicking "Yes" acts as the necessary
user action
and we render the necessary WebHID permission popup with the help of the passed in function.
This repo uses Jest for tests and code coverage
To get started in development:
npm install
cd react-app
yarn start
Then navigate to https://localhost:8443 to see the test app. This way you can see the effects of the events from the headset on the app and vice versa.
Run the tests using npm run test:watch
or npm run test:coverage
. Both commands should be run in the folder.
test:watch
will rerun the tests after changes to the code or the test itselftest:coverage
will run the test suites and produce a report on coverage of the code
All linting and tests must pass 100% and coverage should remain at 100%
Important Note: Out of the box, the test scripts will not work on Windows machines. A developer will more than likely need to make modifications to the scripts in the package.json as well as the shell scripts found in the scripts
folder. If you do not want to modify the scripts out of the box, using a Linux instance seemed to help. The author of the library used an Ubuntu instance