406 lines
12 KiB
TypeScript
406 lines
12 KiB
TypeScript
// (C) Copyright 2015 Moodle Pty Ltd.
|
|
//
|
|
// Licensed under the Apache License, Version 2.0 (the "License");
|
|
// you may not use this file except in compliance with the License.
|
|
// You may obtain a copy of the License at
|
|
//
|
|
// http://www.apache.org/licenses/LICENSE-2.0
|
|
//
|
|
// Unless required by applicable law or agreed to in writing, software
|
|
// distributed under the License is distributed on an "AS IS" BASIS,
|
|
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
// See the License for the specific language governing permissions and
|
|
// limitations under the License.
|
|
|
|
import { Component, OnInit, Optional } from '@angular/core';
|
|
import { CoreError } from '@classes/errors/error';
|
|
import { CoreCourseModuleMainActivityComponent } from '@features/course/classes/main-activity-component';
|
|
import { CoreCourseContentsPage } from '@features/course/pages/contents/contents';
|
|
import { IonContent } from '@ionic/angular';
|
|
import { CoreApp } from '@services/app';
|
|
import { CoreGroupInfo, CoreGroups } from '@services/groups';
|
|
import { CoreSites } from '@services/sites';
|
|
import { CoreDomUtils } from '@services/utils/dom';
|
|
import { CoreText } from '@singletons/text';
|
|
import { CoreTimeUtils } from '@services/utils/time';
|
|
import { CoreUtils } from '@services/utils/utils';
|
|
import { Translate } from '@singletons';
|
|
import {
|
|
AddonModBBB,
|
|
AddonModBBBData,
|
|
AddonModBBBMeetingInfo,
|
|
AddonModBBBRecordingPlaybackTypes,
|
|
} from '../../services/bigbluebuttonbn';
|
|
import { ADDON_MOD_BBB_COMPONENT } from '../../constants';
|
|
import { CoreLoadings } from '@services/loadings';
|
|
import { convertTextToHTMLElement } from '@/core/utils/create-html-element';
|
|
|
|
/**
|
|
* Component that displays a Big Blue Button activity.
|
|
*/
|
|
@Component({
|
|
selector: 'addon-mod-bbb-index',
|
|
templateUrl: 'index.html',
|
|
styleUrls: ['index.scss'],
|
|
})
|
|
export class AddonModBBBIndexComponent extends CoreCourseModuleMainActivityComponent implements OnInit {
|
|
|
|
component = ADDON_MOD_BBB_COMPONENT;
|
|
pluginName = 'bigbluebuttonbn';
|
|
bbb?: AddonModBBBData;
|
|
groupInfo?: CoreGroupInfo;
|
|
groupId = 0;
|
|
meetingInfo?: AddonModBBBMeetingInfo;
|
|
recordings?: Recording[];
|
|
|
|
constructor(
|
|
protected content?: IonContent,
|
|
@Optional() courseContentsPage?: CoreCourseContentsPage,
|
|
) {
|
|
super('AddonModBBBIndexComponent', content, courseContentsPage);
|
|
}
|
|
|
|
/**
|
|
* @inheritdoc
|
|
*/
|
|
async ngOnInit(): Promise<void> {
|
|
super.ngOnInit();
|
|
|
|
await this.loadContent();
|
|
}
|
|
|
|
get showRoom(): boolean {
|
|
return !!this.meetingInfo && (!this.meetingInfo.features || this.meetingInfo.features.showroom);
|
|
}
|
|
|
|
get showRecordings(): boolean {
|
|
return !!this.meetingInfo && (!this.meetingInfo.features || this.meetingInfo.features.showrecordings);
|
|
}
|
|
|
|
/**
|
|
* @inheritdoc
|
|
*/
|
|
protected async fetchContent(): Promise<void> {
|
|
this.bbb = await AddonModBBB.getBBB(this.courseId, this.module.id);
|
|
|
|
this.description = this.bbb.intro;
|
|
this.dataRetrieved.emit(this.bbb);
|
|
|
|
this.groupInfo = await CoreGroups.getActivityGroupInfo(this.module.id, false);
|
|
|
|
this.groupId = CoreGroups.validateGroupId(this.groupId, this.groupInfo);
|
|
|
|
if (this.groupInfo.separateGroups && !this.groupInfo.groups.length) {
|
|
throw new CoreError(Translate.instant('addon.mod_bigbluebuttonbn.view_nojoin'));
|
|
}
|
|
|
|
await this.fetchMeetingInfo();
|
|
|
|
await this.fetchRecordings();
|
|
}
|
|
|
|
/**
|
|
* Get meeting info.
|
|
*
|
|
* @param updateCache Whether to update info cached data (in server).
|
|
* @returns Promise resolved when done.
|
|
*/
|
|
async fetchMeetingInfo(updateCache?: boolean): Promise<void> {
|
|
if (!this.bbb) {
|
|
return;
|
|
}
|
|
|
|
try {
|
|
this.meetingInfo = await AddonModBBB.getMeetingInfo(this.bbb.id, this.groupId, {
|
|
cmId: this.module.id,
|
|
updateCache,
|
|
});
|
|
|
|
if (this.meetingInfo.statusrunning && this.meetingInfo.userlimit > 0) {
|
|
const count = (this.meetingInfo.participantcount || 0) + (this.meetingInfo.moderatorcount || 0);
|
|
if (count === this.meetingInfo.userlimit) {
|
|
this.meetingInfo.statusmessage = Translate.instant('addon.mod_bigbluebuttonbn.userlimitreached');
|
|
}
|
|
}
|
|
} catch (error) {
|
|
if (error && error.errorcode === 'restrictedcontextexception') {
|
|
error.message = Translate.instant('addon.mod_bigbluebuttonbn.view_nojoin');
|
|
}
|
|
|
|
throw error;
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Get recordings.
|
|
*
|
|
* @returns Promise resolved when done.
|
|
*/
|
|
async fetchRecordings(): Promise<void> {
|
|
if (!this.bbb || !this.showRecordings) {
|
|
return;
|
|
}
|
|
|
|
const recordingsTable = await AddonModBBB.getRecordings(this.bbb.id, this.groupId, {
|
|
cmId: this.module.id,
|
|
});
|
|
const columns = CoreUtils.arrayToObject(recordingsTable.columns, 'key');
|
|
|
|
this.recordings = recordingsTable.parsedData.map(recordingData => {
|
|
const details: RecordingDetail[] = [];
|
|
const playbacksEl = convertTextToHTMLElement(String(recordingData.playback));
|
|
const playbacks: RecordingPlayback[] = Array.from(playbacksEl.querySelectorAll('a')).map(playbackAnchor => ({
|
|
name: playbackAnchor.textContent ?? '',
|
|
url: playbackAnchor.href,
|
|
icon: this.getPlaybackIcon(playbackAnchor),
|
|
}));
|
|
|
|
Object.entries(recordingData).forEach(([key, value]) => {
|
|
const columnData = columns[key];
|
|
if (!columnData || value === '' || key === 'actionbar' || key === 'playback') {
|
|
return;
|
|
}
|
|
|
|
if (columnData.formatter === 'customDate' && !isNaN(Number(value))) {
|
|
value = CoreTimeUtils.userDate(Number(value), 'core.strftimedaydate');
|
|
} else if (columnData.allowHTML && typeof value === 'string') {
|
|
// If the HTML is empty, don't display it.
|
|
const valueElement = convertTextToHTMLElement(value);
|
|
if (!valueElement.querySelector('img') && (valueElement.textContent ?? '').trim() === '') {
|
|
return;
|
|
}
|
|
|
|
// Treat "quick edit" buttons, they aren't supported in the app.
|
|
const quickEditLink = valueElement.querySelector('.quickeditlink');
|
|
if (quickEditLink) {
|
|
// The first span in quick edit link contains the actual HTML, use it.
|
|
value = (quickEditLink.querySelector('span')?.innerHTML ?? '').trim();
|
|
}
|
|
}
|
|
|
|
details.push({
|
|
label: columnData.label,
|
|
value: String(value),
|
|
allowHTML: !!columnData.allowHTML,
|
|
});
|
|
});
|
|
|
|
return {
|
|
name: CoreText.cleanTags(String(recordingData.recording), { singleLine: true }),
|
|
playbackLabel: columns.playback.label,
|
|
playbacks,
|
|
details,
|
|
expanded: false,
|
|
};
|
|
});
|
|
}
|
|
|
|
/**
|
|
* Get the playback icon.
|
|
*
|
|
* @param playbackAnchor Anchor element.
|
|
* @returns Icon name.
|
|
*/
|
|
protected getPlaybackIcon(playbackAnchor: HTMLAnchorElement): string {
|
|
const type = playbackAnchor.dataset.target;
|
|
switch (type) {
|
|
case AddonModBBBRecordingPlaybackTypes.NOTES:
|
|
return 'far-file-lines';
|
|
case AddonModBBBRecordingPlaybackTypes.PODCAST:
|
|
return 'fas-microphone-lines';
|
|
case AddonModBBBRecordingPlaybackTypes.SCREENSHARE:
|
|
return 'fas-display';
|
|
case AddonModBBBRecordingPlaybackTypes.STATISTICS:
|
|
return 'fas-chart-line';
|
|
case AddonModBBBRecordingPlaybackTypes.VIDEO:
|
|
return 'fas-video';
|
|
case AddonModBBBRecordingPlaybackTypes.PRESENTATION:
|
|
default:
|
|
return 'fas-circle-play';
|
|
}
|
|
}
|
|
|
|
/**
|
|
* @inheritdoc
|
|
*/
|
|
protected async logActivity(): Promise<void> {
|
|
if (!this.bbb) {
|
|
return; // Shouldn't happen.
|
|
}
|
|
|
|
await CoreUtils.ignoreErrors(AddonModBBB.logView(this.bbb.id));
|
|
|
|
this.analyticsLogEvent('mod_bigbluebuttonbn_view_bigbluebuttonbn');
|
|
}
|
|
|
|
/**
|
|
* Update meeting info.
|
|
*
|
|
* @param updateCache Whether to update info cached data (in server).
|
|
* @returns Promise resolved when done.
|
|
*/
|
|
async updateMeetingInfo(updateCache?: boolean): Promise<void> {
|
|
if (!this.bbb) {
|
|
return;
|
|
}
|
|
|
|
this.showLoading = true;
|
|
|
|
try {
|
|
await AddonModBBB.invalidateAllGroupsMeetingInfo(this.bbb.id);
|
|
|
|
await this.fetchMeetingInfo(updateCache);
|
|
} finally {
|
|
this.showLoading = false;
|
|
}
|
|
}
|
|
|
|
/**
|
|
* @inheritdoc
|
|
*/
|
|
protected async invalidateContent(): Promise<void> {
|
|
const promises: Promise<void>[] = [];
|
|
|
|
promises.push(AddonModBBB.invalidateBBBs(this.courseId));
|
|
promises.push(CoreGroups.invalidateActivityGroupInfo(this.module.id));
|
|
|
|
if (this.bbb) {
|
|
promises.push(AddonModBBB.invalidateAllGroupsMeetingInfo(this.bbb.id));
|
|
promises.push(AddonModBBB.invalidateAllGroupsRecordings(this.bbb.id));
|
|
}
|
|
|
|
await Promise.all(promises);
|
|
}
|
|
|
|
/**
|
|
* Group changed, reload some data.
|
|
*
|
|
* @returns Promise resolved when done.
|
|
*/
|
|
async groupChanged(): Promise<void> {
|
|
this.showLoading = true;
|
|
|
|
try {
|
|
await this.fetchMeetingInfo();
|
|
|
|
await this.fetchRecordings();
|
|
} catch (error) {
|
|
CoreDomUtils.showErrorModal(error);
|
|
} finally {
|
|
this.showLoading = false;
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Join the room.
|
|
*
|
|
* @returns Promise resolved when done.
|
|
*/
|
|
async joinRoom(): Promise<void> {
|
|
const modal = await CoreLoadings.show();
|
|
|
|
try {
|
|
const joinUrl = await AddonModBBB.getJoinUrl(this.module.id, this.groupId);
|
|
|
|
await CoreUtils.openInBrowser(joinUrl, {
|
|
showBrowserWarning: false,
|
|
});
|
|
|
|
// Leave some time for the room to load.
|
|
await CoreApp.waitForResume(10000);
|
|
|
|
this.updateMeetingInfo(true);
|
|
} catch (error) {
|
|
CoreDomUtils.showErrorModal(error);
|
|
} finally {
|
|
modal.dismiss();
|
|
}
|
|
}
|
|
|
|
/**
|
|
* End the meeting.
|
|
*
|
|
* @returns Promise resolved when done.
|
|
*/
|
|
async endMeeting(): Promise<void> {
|
|
if (!this.bbb) {
|
|
return;
|
|
}
|
|
|
|
try {
|
|
await CoreDomUtils.showConfirm(
|
|
Translate.instant('addon.mod_bigbluebuttonbn.end_session_confirm'),
|
|
Translate.instant('addon.mod_bigbluebuttonbn.end_session_confirm_title'),
|
|
Translate.instant('core.yes'),
|
|
);
|
|
} catch {
|
|
// User canceled.
|
|
return;
|
|
}
|
|
|
|
const modal = await CoreLoadings.show();
|
|
|
|
try {
|
|
await AddonModBBB.endMeeting(this.bbb.id, this.groupId);
|
|
|
|
this.updateMeetingInfo();
|
|
} catch (error) {
|
|
CoreDomUtils.showErrorModal(error);
|
|
} finally {
|
|
modal.dismiss();
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Toogle the visibility of a recording (expand/collapse).
|
|
*
|
|
* @param recording Recording.
|
|
*/
|
|
toggle(recording: Recording): void {
|
|
recording.expanded = !recording.expanded;
|
|
}
|
|
|
|
/**
|
|
* Open a recording playback.
|
|
*
|
|
* @param event Click event.
|
|
* @param playback Playback.
|
|
*/
|
|
openPlayback(event: MouseEvent, playback: RecordingPlayback): void {
|
|
event.preventDefault();
|
|
event.stopPropagation();
|
|
|
|
CoreSites.getCurrentSite()?.openInBrowserWithAutoLogin(playback.url);
|
|
}
|
|
|
|
}
|
|
|
|
/**
|
|
* Recording data.
|
|
*/
|
|
type Recording = {
|
|
name: string;
|
|
expanded: boolean;
|
|
playbackLabel: string;
|
|
playbacks: RecordingPlayback[];
|
|
details: RecordingDetail[];
|
|
};
|
|
|
|
/**
|
|
* Recording detail data.
|
|
*/
|
|
type RecordingDetail = {
|
|
label: string;
|
|
value: string;
|
|
allowHTML: boolean;
|
|
};
|
|
|
|
/**
|
|
* Recording playback data.
|
|
*/
|
|
type RecordingPlayback = {
|
|
name: string;
|
|
url: string;
|
|
icon: string;
|
|
};
|