Some checks failed
Close stale issues and PRs / stale (push) Has been cancelled
277 lines
9.4 KiB
TypeScript
277 lines
9.4 KiB
TypeScript
import { Participant } from '../helpers/Participant';
|
|
|
|
import BaseDialog from './BaseDialog';
|
|
import BasePageObject from './BasePageObject';
|
|
|
|
const LOCAL_VIDEO_XPATH = '//span[@id="localVideoContainer"]';
|
|
const LOCAL_VIDEO_MENU_TRIGGER = '#local-video-menu-trigger';
|
|
const LOCAL_USER_CONTROLS = 'aria/Local user controls';
|
|
const HIDE_SELF_VIEW_BUTTON_XPATH = '//div[contains(@class, "popover")]//div[@id="hideselfviewButton"]';
|
|
|
|
/**
|
|
* Filmstrip elements.
|
|
*/
|
|
export default class Filmstrip extends BasePageObject {
|
|
/**
|
|
* Asserts that {@code participant} shows or doesn't show the audio
|
|
* mute icon for the conference participant identified by
|
|
* {@code testee}.
|
|
*
|
|
* @param {Participant} testee - The {@code Participant} for whom we're checking the status of audio muted icon.
|
|
* @param {boolean} reverse - If {@code true}, the method will assert the absence of the "mute" icon;
|
|
* otherwise, it will assert its presence.
|
|
* @returns {Promise<void>}
|
|
*/
|
|
async assertAudioMuteIconIsDisplayed(testee: Participant, reverse = false): Promise<void> {
|
|
let id;
|
|
|
|
if (testee === this.participant) {
|
|
id = 'localVideoContainer';
|
|
} else {
|
|
id = `participant_${await testee.getEndpointId()}`;
|
|
}
|
|
|
|
const mutedIconXPath
|
|
= `//span[@id='${id}']//span[contains(@id, 'audioMuted')]//*[local-name()='svg' and @id='mic-disabled']`;
|
|
|
|
await this.participant.driver.$(mutedIconXPath).waitForDisplayed({
|
|
reverse,
|
|
timeout: 5_000,
|
|
timeoutMsg: `Audio mute icon is${reverse ? '' : ' not'} displayed for ${testee.name}`
|
|
});
|
|
}
|
|
|
|
/**
|
|
* Returns the remote display name for an endpoint.
|
|
* @param endpointId The endpoint id.
|
|
*/
|
|
async getRemoteDisplayName(endpointId: string) {
|
|
const remoteDisplayName = this.participant.driver.$(`span[id="participant_${endpointId}_name"]`);
|
|
|
|
await remoteDisplayName.moveTo();
|
|
|
|
return await remoteDisplayName.getText();
|
|
}
|
|
|
|
/**
|
|
* Returns the remote video id of a participant with endpointID.
|
|
* @param endpointId
|
|
*/
|
|
async getRemoteVideoId(endpointId: string) {
|
|
const remoteDisplayName = this.participant.driver.$(`span[id="participant_${endpointId}"]`);
|
|
|
|
await remoteDisplayName.moveTo();
|
|
|
|
return await this.participant.execute(eId =>
|
|
document.evaluate(`//span[@id="participant_${eId}"]//video`,
|
|
document, null, XPathResult.FIRST_ORDERED_NODE_TYPE, null).singleNodeValue?.srcObject?.id, endpointId);
|
|
}
|
|
|
|
/**
|
|
* Returns the local video id.
|
|
*/
|
|
getLocalVideoId() {
|
|
return this.participant.execute(
|
|
'return document.getElementById("localVideo_container").srcObject.id');
|
|
}
|
|
|
|
/**
|
|
* Pins a participant by clicking on their thumbnail.
|
|
* @param participant The participant.
|
|
*/
|
|
async pinParticipant(participant: Participant) {
|
|
let videoIdToSwitchTo;
|
|
|
|
if (participant === this.participant) {
|
|
videoIdToSwitchTo = await this.getLocalVideoId();
|
|
|
|
// when looking up the element and clicking it, it doesn't work if we do it twice in a row (oneOnOne.spec)
|
|
await this.participant.execute(() => document?.getElementById('localVideoContainer')?.click());
|
|
} else {
|
|
const epId = await participant.getEndpointId();
|
|
|
|
videoIdToSwitchTo = await this.getRemoteVideoId(epId);
|
|
|
|
await this.participant.driver.$(`//span[@id="participant_${epId}"]`).click();
|
|
}
|
|
|
|
await this.participant.driver.waitUntil(
|
|
async () => await this.participant.getLargeVideo().getId() === videoIdToSwitchTo,
|
|
{
|
|
timeout: 3_000,
|
|
timeoutMsg: `${this.participant.name} did not switch the large video to ${
|
|
participant.name}`
|
|
}
|
|
);
|
|
}
|
|
|
|
/**
|
|
* Unpins a participant by clicking on their thumbnail.
|
|
* @param participant
|
|
*/
|
|
async unpinParticipant(participant: Participant) {
|
|
const epId = await participant.getEndpointId();
|
|
|
|
if (participant === this.participant) {
|
|
await this.participant.execute(() => document?.getElementById('localVideoContainer')?.click());
|
|
} else {
|
|
await this.participant.driver.$(`//span[@id="participant_${epId}"]`).click();
|
|
}
|
|
|
|
await this.participant.driver.$(`//div[ @id="pin-indicator-${epId}" ]`).waitForDisplayed({
|
|
timeout: 2_000,
|
|
timeoutMsg: `${this.participant.name} did not unpin ${participant.name}`,
|
|
reverse: true
|
|
});
|
|
}
|
|
|
|
/**
|
|
* Gets avatar SRC attribute for the one displayed on small video thumbnail.
|
|
* @param endpointId
|
|
*/
|
|
async getAvatar(endpointId: string) {
|
|
const elem = this.participant.driver.$(
|
|
`//span[@id='participant_${endpointId}']//img[contains(@class,'userAvatar')]`);
|
|
|
|
return await elem.isExisting() ? await elem.getAttribute('src') : null;
|
|
}
|
|
|
|
/**
|
|
* Grants moderator rights to a participant.
|
|
* @param participant
|
|
*/
|
|
async grantModerator(participant: Participant) {
|
|
await this.clickOnRemoteMenuLink(await participant.getEndpointId(), 'grantmoderatorlink', true);
|
|
}
|
|
|
|
/**
|
|
* Clicks on the link in the remote participant actions menu.
|
|
* @param participantId
|
|
* @param linkClassname
|
|
* @param dialogConfirm
|
|
* @private
|
|
*/
|
|
private async clickOnRemoteMenuLink(participantId: string, linkClassname: string, dialogConfirm: boolean) {
|
|
await this.participant.driver.$(`//span[@id='participant_${participantId}']`).moveTo();
|
|
|
|
await this.participant.driver.$(
|
|
`//span[@id='participant_${participantId
|
|
}']//span[@id='remotevideomenu']//div[@id='remote-video-menu-trigger']`).moveTo();
|
|
|
|
const popoverElement = this.participant.driver.$(
|
|
`//div[contains(@class, 'popover')]//div[contains(@class, '${linkClassname}')]`);
|
|
|
|
await popoverElement.waitForExist();
|
|
await popoverElement.waitForDisplayed();
|
|
await popoverElement.click();
|
|
|
|
if (dialogConfirm) {
|
|
await new BaseDialog(this.participant).clickOkButton();
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Mutes the audio of a participant.
|
|
* @param participant
|
|
*/
|
|
async muteAudio(participant: Participant) {
|
|
await this.clickOnRemoteMenuLink(await participant.getEndpointId(), 'mutelink', false);
|
|
}
|
|
|
|
/**
|
|
* Mutes the video of a participant.
|
|
* @param participant
|
|
*/
|
|
async muteVideo(participant: Participant) {
|
|
await this.clickOnRemoteMenuLink(await participant.getEndpointId(), 'mutevideolink', true);
|
|
}
|
|
|
|
/**
|
|
* Kicks a participant.
|
|
* @param participantId
|
|
*/
|
|
kickParticipant(participantId: string) {
|
|
return this.clickOnRemoteMenuLink(participantId, 'kicklink', true);
|
|
}
|
|
|
|
/**
|
|
* Hover over local video.
|
|
*/
|
|
hoverOverLocalVideo() {
|
|
return this.participant.driver.$(LOCAL_VIDEO_MENU_TRIGGER).moveTo();
|
|
}
|
|
|
|
/**
|
|
* Clicks on the hide self view button from local video.
|
|
*/
|
|
async hideSelfView() {
|
|
// open local video menu
|
|
await this.hoverOverLocalVideo();
|
|
await this.participant.driver.$(LOCAL_USER_CONTROLS).moveTo();
|
|
|
|
// click Hide self view button
|
|
const hideSelfViewButton = this.participant.driver.$(HIDE_SELF_VIEW_BUTTON_XPATH);
|
|
|
|
await hideSelfViewButton.waitForExist();
|
|
await hideSelfViewButton.waitForClickable();
|
|
await hideSelfViewButton.click();
|
|
}
|
|
|
|
/**
|
|
* Checks whether the local self view is displayed or not.
|
|
*/
|
|
assertSelfViewIsHidden(hidden: boolean) {
|
|
return this.participant.driver.$(LOCAL_VIDEO_XPATH).waitForDisplayed({
|
|
reverse: hidden,
|
|
timeout: 5000,
|
|
timeoutMsg: `Local video thumbnail is${hidden ? '' : ' not'} displayed for ${this.participant.name}`
|
|
});
|
|
}
|
|
|
|
/**
|
|
* Toggles the filmstrip.
|
|
*/
|
|
async toggle() {
|
|
const toggleButton = this.participant.driver.$('#toggleFilmstripButton');
|
|
|
|
await toggleButton.moveTo();
|
|
await toggleButton.waitForDisplayed();
|
|
await toggleButton.click();
|
|
}
|
|
|
|
/**
|
|
* Asserts that the remote videos are hidden or not.
|
|
* @param reverse
|
|
*/
|
|
assertRemoteVideosHidden(reverse = false) {
|
|
return this.participant.driver.waitUntil(
|
|
async () =>
|
|
await this.participant.driver.$$('//div[@id="remoteVideos" and contains(@class, "hidden")]').length > 0,
|
|
{
|
|
timeout: 10_000, // 10 seconds
|
|
timeoutMsg: `Timeout waiting fore remote videos to be hidden: ${!reverse}.`
|
|
}
|
|
);
|
|
}
|
|
|
|
/**
|
|
* Counts the displayed remote video thumbnails.
|
|
*/
|
|
async countVisibleThumbnails() {
|
|
return (await this.participant.driver.$$('//div[@id="remoteVideos"]//span[contains(@class,"videocontainer")]')
|
|
.filter(thumbnail => thumbnail.isDisplayed())).length;
|
|
}
|
|
|
|
/**
|
|
* Check if remote videos in filmstrip are visible.
|
|
*
|
|
* @param isDisplayed whether or not filmstrip remote videos should be visible
|
|
*/
|
|
verifyRemoteVideosDisplay(isDisplayed: boolean) {
|
|
return this.participant.driver.$('//div[contains(@class, "remote-videos")]/div').waitForDisplayed({
|
|
timeout: 5_000,
|
|
reverse: !isDisplayed,
|
|
});
|
|
}
|
|
}
|