commit
d23160df19
|
@ -980,10 +980,7 @@
|
||||||
"addon.mod_scorm.errorcreateofflineattempt": "local_moodlemobileapp",
|
"addon.mod_scorm.errorcreateofflineattempt": "local_moodlemobileapp",
|
||||||
"addon.mod_scorm.errordownloadscorm": "local_moodlemobileapp",
|
"addon.mod_scorm.errordownloadscorm": "local_moodlemobileapp",
|
||||||
"addon.mod_scorm.errorgetscorm": "local_moodlemobileapp",
|
"addon.mod_scorm.errorgetscorm": "local_moodlemobileapp",
|
||||||
"addon.mod_scorm.errorinvalidversion": "local_moodlemobileapp",
|
|
||||||
"addon.mod_scorm.errornotdownloadable": "local_moodlemobileapp",
|
|
||||||
"addon.mod_scorm.errornovalidsco": "local_moodlemobileapp",
|
"addon.mod_scorm.errornovalidsco": "local_moodlemobileapp",
|
||||||
"addon.mod_scorm.errorpackagefile": "local_moodlemobileapp",
|
|
||||||
"addon.mod_scorm.errorsyncscorm": "local_moodlemobileapp",
|
"addon.mod_scorm.errorsyncscorm": "local_moodlemobileapp",
|
||||||
"addon.mod_scorm.exceededmaxattempts": "scorm",
|
"addon.mod_scorm.exceededmaxattempts": "scorm",
|
||||||
"addon.mod_scorm.failed": "scorm",
|
"addon.mod_scorm.failed": "scorm",
|
||||||
|
|
|
@ -17,7 +17,7 @@
|
||||||
|
|
||||||
<!-- Activity info. -->
|
<!-- Activity info. -->
|
||||||
<core-course-module-info [module]="module" [description]="description" [component]="component" [componentId]="componentId"
|
<core-course-module-info [module]="module" [description]="description" [component]="component" [componentId]="componentId"
|
||||||
[courseId]="courseId" [hasDataToSync]="!errorMessage && hasOffline" (completionChanged)="onCompletionChange()" />
|
[courseId]="courseId" [hasDataToSync]="hasOffline" (completionChanged)="onCompletionChange()" />
|
||||||
|
|
||||||
<!-- Warning message. -->
|
<!-- Warning message. -->
|
||||||
<ion-card class="core-info-card" *ngIf="scorm && scorm.warningMessage">
|
<ion-card class="core-info-card" *ngIf="scorm && scorm.warningMessage">
|
||||||
|
@ -162,21 +162,8 @@
|
||||||
|
|
||||||
<div collapsible-footer *ngIf="!showLoading" slot="fixed">
|
<div collapsible-footer *ngIf="!showLoading" slot="fixed">
|
||||||
<div class="list-item-limited-width" *ngIf="scorm && !scorm.warningMessage">
|
<div class="list-item-limited-width" *ngIf="scorm && !scorm.warningMessage">
|
||||||
<!-- Open in browser button. -->
|
|
||||||
<ng-container *ngIf="errorMessage">
|
|
||||||
<ion-item class="ion-text-wrap">
|
|
||||||
<ion-label>
|
|
||||||
<p class="text-danger">{{ errorMessage | translate }}</p>
|
|
||||||
</ion-label>
|
|
||||||
</ion-item>
|
|
||||||
<ion-button class="ion-margin ion-text-wrap" expand="block" [href]="module.url" core-link [showBrowserWarning]="false">
|
|
||||||
{{ 'core.openinbrowser' | translate }}
|
|
||||||
<ion-icon name="fas-up-right-from-square" slot="end" aria-hidden="true" />
|
|
||||||
</ion-button>
|
|
||||||
</ng-container>
|
|
||||||
|
|
||||||
<!-- Warning that user doesn't have any more attempts. -->
|
<!-- Warning that user doesn't have any more attempts. -->
|
||||||
<ion-card *ngIf="!errorMessage && attemptsLeft === 0" class="core-danger-card">
|
<ion-card *ngIf="attemptsLeft === 0" class="core-danger-card">
|
||||||
<ion-item class="ion-text-wrap">
|
<ion-item class="ion-text-wrap">
|
||||||
<ion-label>
|
<ion-label>
|
||||||
<p>{{ 'addon.mod_scorm.exceededmaxattempts' | translate }}</p>
|
<p>{{ 'addon.mod_scorm.exceededmaxattempts' | translate }}</p>
|
||||||
|
@ -184,7 +171,16 @@
|
||||||
</ion-item>
|
</ion-item>
|
||||||
</ion-card>
|
</ion-card>
|
||||||
|
|
||||||
<ng-container *ngIf="!errorMessage && (!scorm.lastattemptlock || attemptsLeft > 0)">
|
<ion-card *ngIf="useOnlinePlayer && !isOnline" class="core-warning-card">
|
||||||
|
<ion-item class="ion-text-wrap">
|
||||||
|
<ion-icon name="fas-triangle-exclamation" slot="start" aria-hidden="true" />
|
||||||
|
<ion-label>
|
||||||
|
{{ 'core.course.activitynotavailableoffline' | translate }} {{ 'core.needinternettoaccessit' | translate }}
|
||||||
|
</ion-label>
|
||||||
|
</ion-item>
|
||||||
|
</ion-card>
|
||||||
|
|
||||||
|
<ng-container *ngIf="(!scorm.lastattemptlock || attemptsLeft > 0) && (!useOnlinePlayer || isOnline)">
|
||||||
<!-- Open SCORM in app form -->
|
<!-- Open SCORM in app form -->
|
||||||
<ng-container *ngIf="!downloading && !skip">
|
<ng-container *ngIf="!downloading && !skip">
|
||||||
<!-- Create new attempt -->
|
<!-- Create new attempt -->
|
||||||
|
|
|
@ -13,7 +13,7 @@
|
||||||
// limitations under the License.
|
// limitations under the License.
|
||||||
|
|
||||||
import { DownloadStatus } from '@/core/constants';
|
import { DownloadStatus } from '@/core/constants';
|
||||||
import { Component, Input, OnInit, Optional } from '@angular/core';
|
import { Component, Input, OnDestroy, OnInit, Optional } from '@angular/core';
|
||||||
import { CoreError } from '@classes/errors/error';
|
import { CoreError } from '@classes/errors/error';
|
||||||
import { CoreCourseModuleMainActivityComponent } from '@features/course/classes/main-activity-component';
|
import { CoreCourseModuleMainActivityComponent } from '@features/course/classes/main-activity-component';
|
||||||
import { CoreCourseContentsPage } from '@features/course/pages/contents/contents';
|
import { CoreCourseContentsPage } from '@features/course/pages/contents/contents';
|
||||||
|
@ -23,7 +23,7 @@ import { CoreNavigator } from '@services/navigator';
|
||||||
import { CoreSync } from '@services/sync';
|
import { CoreSync } from '@services/sync';
|
||||||
import { CoreDomUtils } from '@services/utils/dom';
|
import { CoreDomUtils } from '@services/utils/dom';
|
||||||
import { CoreObject } from '@singletons/object';
|
import { CoreObject } from '@singletons/object';
|
||||||
import { Translate } from '@singletons';
|
import { NgZone, Translate } from '@singletons';
|
||||||
import { CoreEventObserver, CoreEvents } from '@singletons/events';
|
import { CoreEventObserver, CoreEvents } from '@singletons/events';
|
||||||
import { AddonModScormPrefetchHandler } from '../../services/handlers/prefetch';
|
import { AddonModScormPrefetchHandler } from '../../services/handlers/prefetch';
|
||||||
import {
|
import {
|
||||||
|
@ -51,6 +51,8 @@ import {
|
||||||
} from '../../constants';
|
} from '../../constants';
|
||||||
import { CoreWait } from '@singletons/wait';
|
import { CoreWait } from '@singletons/wait';
|
||||||
import { CorePromiseUtils } from '@singletons/promise-utils';
|
import { CorePromiseUtils } from '@singletons/promise-utils';
|
||||||
|
import { CoreNetwork } from '@services/network';
|
||||||
|
import { Subscription } from 'rxjs';
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Component that displays a SCORM entry page.
|
* Component that displays a SCORM entry page.
|
||||||
|
@ -60,7 +62,7 @@ import { CorePromiseUtils } from '@singletons/promise-utils';
|
||||||
templateUrl: 'addon-mod-scorm-index.html',
|
templateUrl: 'addon-mod-scorm-index.html',
|
||||||
styleUrl: 'index.scss',
|
styleUrl: 'index.scss',
|
||||||
})
|
})
|
||||||
export class AddonModScormIndexComponent extends CoreCourseModuleMainActivityComponent implements OnInit {
|
export class AddonModScormIndexComponent extends CoreCourseModuleMainActivityComponent implements OnInit, OnDestroy {
|
||||||
|
|
||||||
@Input() autoPlayData?: AddonModScormAutoPlayData; // Data to use to play the SCORM automatically.
|
@Input() autoPlayData?: AddonModScormAutoPlayData; // Data to use to play the SCORM automatically.
|
||||||
|
|
||||||
|
@ -73,7 +75,6 @@ export class AddonModScormIndexComponent extends CoreCourseModuleMainActivityCom
|
||||||
}; // Selected organization.
|
}; // Selected organization.
|
||||||
|
|
||||||
startNewAttempt = false;
|
startNewAttempt = false;
|
||||||
errorMessage?: string; // Error message.
|
|
||||||
syncTime?: string; // Last sync time.
|
syncTime?: string; // Last sync time.
|
||||||
hasOffline = false; // Whether the SCORM has offline data.
|
hasOffline = false; // Whether the SCORM has offline data.
|
||||||
attemptToContinue?: number; // The attempt to continue or review.
|
attemptToContinue?: number; // The attempt to continue or review.
|
||||||
|
@ -96,6 +97,7 @@ export class AddonModScormIndexComponent extends CoreCourseModuleMainActivityCom
|
||||||
onlineAttempts: AttemptGrade[] = []; // Grades for online attempts.
|
onlineAttempts: AttemptGrade[] = []; // Grades for online attempts.
|
||||||
offlineAttempts: AttemptGrade[] = []; // Grades for offline attempts.
|
offlineAttempts: AttemptGrade[] = []; // Grades for offline attempts.
|
||||||
gradesExpanded = false;
|
gradesExpanded = false;
|
||||||
|
isOnline: boolean;
|
||||||
|
|
||||||
protected fetchContentDefaultError = 'addon.mod_scorm.errorgetscorm'; // Default error to show when loading contents.
|
protected fetchContentDefaultError = 'addon.mod_scorm.errorgetscorm'; // Default error to show when loading contents.
|
||||||
protected syncEventName = ADDON_MOD_SCORM_DATA_AUTO_SYNCED;
|
protected syncEventName = ADDON_MOD_SCORM_DATA_AUTO_SYNCED;
|
||||||
|
@ -105,12 +107,22 @@ export class AddonModScormIndexComponent extends CoreCourseModuleMainActivityCom
|
||||||
protected hasPlayed = false; // Whether the user has opened the player page.
|
protected hasPlayed = false; // Whether the user has opened the player page.
|
||||||
protected dataSentObserver?: CoreEventObserver; // To detect data sent to server.
|
protected dataSentObserver?: CoreEventObserver; // To detect data sent to server.
|
||||||
protected dataSent = false; // Whether some data was sent to server while playing the SCORM.
|
protected dataSent = false; // Whether some data was sent to server while playing the SCORM.
|
||||||
|
protected useOnlinePlayer = false; // Whether the SCORM needs to be played using an online player.
|
||||||
|
protected onlineObserver: Subscription;
|
||||||
|
|
||||||
constructor(
|
constructor(
|
||||||
protected content?: IonContent,
|
protected content?: IonContent,
|
||||||
@Optional() courseContentsPage?: CoreCourseContentsPage,
|
@Optional() courseContentsPage?: CoreCourseContentsPage,
|
||||||
) {
|
) {
|
||||||
super('AddonModScormIndexComponent', content, courseContentsPage);
|
super('AddonModScormIndexComponent', content, courseContentsPage);
|
||||||
|
|
||||||
|
this.isOnline = CoreNetwork.isOnline();
|
||||||
|
this.onlineObserver = CoreNetwork.onChange().subscribe(() => {
|
||||||
|
// Execute the callback in the Angular zone, so change detection doesn't stop working.
|
||||||
|
NgZone.run(() => {
|
||||||
|
this.isOnline = CoreNetwork.isOnline();
|
||||||
|
});
|
||||||
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -181,19 +193,19 @@ export class AddonModScormIndexComponent extends CoreCourseModuleMainActivityCom
|
||||||
|
|
||||||
this.dataRetrieved.emit(this.scorm);
|
this.dataRetrieved.emit(this.scorm);
|
||||||
this.description = this.scorm.intro || this.description;
|
this.description = this.scorm.intro || this.description;
|
||||||
this.errorMessage = AddonModScorm.isScormUnsupported(this.scorm);
|
this.useOnlinePlayer = AddonModScorm.useOnlinePlayer(this.scorm);
|
||||||
|
|
||||||
if (this.scorm.warningMessage) {
|
if (this.scorm.warningMessage) {
|
||||||
return; // SCORM is closed or not open yet, we can't get more data.
|
return; // SCORM is closed or not open yet, we can't get more data.
|
||||||
}
|
}
|
||||||
|
|
||||||
if (sync) {
|
if (sync && !this.useOnlinePlayer) {
|
||||||
// Try to synchronize the SCORM.
|
// Try to synchronize the SCORM.
|
||||||
await CorePromiseUtils.ignoreErrors(this.syncActivity(showErrors));
|
await CorePromiseUtils.ignoreErrors(this.syncActivity(showErrors));
|
||||||
}
|
}
|
||||||
|
|
||||||
const [syncTime, accessInfo] = await Promise.all([
|
const [syncTime, accessInfo] = await Promise.all([
|
||||||
AddonModScormSync.getReadableSyncTime(this.scorm.id),
|
this.useOnlinePlayer ? undefined : AddonModScormSync.getReadableSyncTime(this.scorm.id),
|
||||||
AddonModScorm.getAccessInformation(this.scorm.id, { cmId: this.module.id }),
|
AddonModScorm.getAccessInformation(this.scorm.id, { cmId: this.module.id }),
|
||||||
this.fetchAttemptData(this.scorm),
|
this.fetchAttemptData(this.scorm),
|
||||||
]);
|
]);
|
||||||
|
@ -203,7 +215,7 @@ export class AddonModScormIndexComponent extends CoreCourseModuleMainActivityCom
|
||||||
|
|
||||||
// Check whether to launch the SCORM immediately.
|
// Check whether to launch the SCORM immediately.
|
||||||
if (this.skip === undefined) {
|
if (this.skip === undefined) {
|
||||||
this.skip = !this.hasOffline && !this.errorMessage && (!this.scorm.lastattemptlock || this.attemptsLeft > 0) &&
|
this.skip = !this.hasOffline && (!this.scorm.lastattemptlock || this.attemptsLeft > 0) &&
|
||||||
(
|
(
|
||||||
!!this.autoPlayData
|
!!this.autoPlayData
|
||||||
||
|
||
|
||||||
|
@ -268,7 +280,7 @@ export class AddonModScormIndexComponent extends CoreCourseModuleMainActivityCom
|
||||||
* @returns Promise resolved when done.
|
* @returns Promise resolved when done.
|
||||||
*/
|
*/
|
||||||
protected async loadPackageSize(scorm: AddonModScormScorm): Promise<void> {
|
protected async loadPackageSize(scorm: AddonModScormScorm): Promise<void> {
|
||||||
if (scorm.packagesize || this.errorMessage) {
|
if (scorm.packagesize || this.useOnlinePlayer) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -498,6 +510,15 @@ export class AddonModScormIndexComponent extends CoreCourseModuleMainActivityCom
|
||||||
event?.preventDefault();
|
event?.preventDefault();
|
||||||
event?.stopPropagation();
|
event?.stopPropagation();
|
||||||
|
|
||||||
|
if (this.useOnlinePlayer) {
|
||||||
|
// No need to download the package, just open it.
|
||||||
|
if (this.isOnline) {
|
||||||
|
this.openScorm(scoId, preview);
|
||||||
|
}
|
||||||
|
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
if (this.downloading || !this.scorm) {
|
if (this.downloading || !this.scorm) {
|
||||||
// Scope is being downloaded, abort.
|
// Scope is being downloaded, abort.
|
||||||
return;
|
return;
|
||||||
|
@ -569,8 +590,10 @@ export class AddonModScormIndexComponent extends CoreCourseModuleMainActivityCom
|
||||||
}
|
}
|
||||||
}, this.siteId);
|
}, this.siteId);
|
||||||
|
|
||||||
|
const pageRoute = this.useOnlinePlayer ? 'online-player' : 'player';
|
||||||
|
|
||||||
CoreNavigator.navigateToSitePath(
|
CoreNavigator.navigateToSitePath(
|
||||||
`${ADDON_MOD_SCORM_PAGE_NAME}/${this.courseId}/${this.module.id}/player`,
|
`${ADDON_MOD_SCORM_PAGE_NAME}/${this.courseId}/${this.module.id}/${pageRoute}`,
|
||||||
{
|
{
|
||||||
params: {
|
params: {
|
||||||
mode: autoPlayData?.mode ?? (preview ? AddonModScormMode.BROWSE : AddonModScormMode.NORMAL),
|
mode: autoPlayData?.mode ?? (preview ? AddonModScormMode.BROWSE : AddonModScormMode.NORMAL),
|
||||||
|
@ -587,6 +610,11 @@ export class AddonModScormIndexComponent extends CoreCourseModuleMainActivityCom
|
||||||
* @inheritdoc
|
* @inheritdoc
|
||||||
*/
|
*/
|
||||||
protected async showStatus(status: DownloadStatus): Promise<void> {
|
protected async showStatus(status: DownloadStatus): Promise<void> {
|
||||||
|
if (this.useOnlinePlayer) {
|
||||||
|
this.statusMessage = '';
|
||||||
|
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
if (status === DownloadStatus.OUTDATED && this.scorm) {
|
if (status === DownloadStatus.OUTDATED && this.scorm) {
|
||||||
// Only show the outdated message if the file should be downloaded.
|
// Only show the outdated message if the file should be downloaded.
|
||||||
|
@ -635,6 +663,13 @@ export class AddonModScormIndexComponent extends CoreCourseModuleMainActivityCom
|
||||||
return result;
|
return result;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @inheritdoc
|
||||||
|
*/
|
||||||
|
ngOnDestroy(): void {
|
||||||
|
this.onlineObserver.unsubscribe();
|
||||||
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
|
|
@ -14,10 +14,7 @@
|
||||||
"errorcreateofflineattempt": "An error occurred while creating a new offline attempt. Please try again.",
|
"errorcreateofflineattempt": "An error occurred while creating a new offline attempt. Please try again.",
|
||||||
"errordownloadscorm": "Error downloading SCORM: \"{{name}}\".",
|
"errordownloadscorm": "Error downloading SCORM: \"{{name}}\".",
|
||||||
"errorgetscorm": "Error getting SCORM data.",
|
"errorgetscorm": "Error getting SCORM data.",
|
||||||
"errorinvalidversion": "Sorry, the application only supports SCORM 1.2.",
|
|
||||||
"errornotdownloadable": "Your school or learning provider has disabled the download of SCORM packages.",
|
|
||||||
"errornovalidsco": "This SCORM package doesn't have a visible SCO to load.",
|
"errornovalidsco": "This SCORM package doesn't have a visible SCO to load.",
|
||||||
"errorpackagefile": "Sorry, the application only supports ZIP packages.",
|
|
||||||
"errorsyncscorm": "An error occurred while synchronising. Please try again.",
|
"errorsyncscorm": "An error occurred while synchronising. Please try again.",
|
||||||
"exceededmaxattempts": "You have reached the maximum number of attempts.",
|
"exceededmaxattempts": "You have reached the maximum number of attempts.",
|
||||||
"failed": "Failed",
|
"failed": "Failed",
|
||||||
|
|
|
@ -0,0 +1,23 @@
|
||||||
|
<ion-header>
|
||||||
|
<ion-toolbar>
|
||||||
|
<ion-buttons slot="start">
|
||||||
|
<ion-back-button [text]="'core.back' | translate" />
|
||||||
|
</ion-buttons>
|
||||||
|
<ion-title>
|
||||||
|
<h1>
|
||||||
|
<core-format-text *ngIf="scorm" [text]="scorm.name" contextLevel="module" [contextInstanceId]="cmId"
|
||||||
|
[courseId]="courseId" />
|
||||||
|
</h1>
|
||||||
|
</ion-title>
|
||||||
|
<!-- Add empty ion-buttons to let iframe add the full screen button -->
|
||||||
|
<ion-buttons slot="end" />
|
||||||
|
</ion-toolbar>
|
||||||
|
</ion-header>
|
||||||
|
<ion-content>
|
||||||
|
<core-loading [hideUntil]="loaded">
|
||||||
|
<core-iframe *ngIf="loaded" id="scorm_object" [src]="src" [iframeWidth]="scormWidth" [iframeHeight]="scormHeight"
|
||||||
|
[showFullscreenOnToolbar]="true" [autoFullscreenOnRotate]="enableFullScreenOnRotate" (loaded)="iframeLoaded()" />
|
||||||
|
|
||||||
|
<p *ngIf="!src && errorMessage">{{ errorMessage | translate }}</p>
|
||||||
|
</core-loading>
|
||||||
|
</ion-content>
|
|
@ -0,0 +1,299 @@
|
||||||
|
// (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, OnDestroy, ViewChild } from '@angular/core';
|
||||||
|
import { CoreNavigator } from '@services/navigator';
|
||||||
|
import { CoreSitesReadingStrategy } from '@services/sites';
|
||||||
|
import { CoreDomUtils } from '@services/utils/dom';
|
||||||
|
import { CoreEvents } from '@singletons/events';
|
||||||
|
import {
|
||||||
|
AddonModScorm,
|
||||||
|
AddonModScormAttemptCountResult,
|
||||||
|
AddonModScormGetScormAccessInformationWSResponse,
|
||||||
|
AddonModScormScorm,
|
||||||
|
AddonModScormScoWithData,
|
||||||
|
} from '../../services/scorm';
|
||||||
|
import { AddonModScormHelper } from '../../services/scorm-helper';
|
||||||
|
import { AddonModScormMode } from '../../constants';
|
||||||
|
import { CoreSharedModule } from '@/core/shared.module';
|
||||||
|
import { Subscription } from 'rxjs';
|
||||||
|
import { CoreNetwork } from '@services/network';
|
||||||
|
import { NgZone, Translate } from '@singletons';
|
||||||
|
import { CoreError } from '@classes/errors/error';
|
||||||
|
import { CoreWait } from '@singletons/wait';
|
||||||
|
import { CoreIframeComponent } from '@components/iframe/iframe';
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Page that allows playing a SCORM in online, served from the server.
|
||||||
|
*/
|
||||||
|
@Component({
|
||||||
|
selector: 'page-addon-mod-scorm-online-player',
|
||||||
|
templateUrl: 'online-player.html',
|
||||||
|
standalone: true,
|
||||||
|
imports: [
|
||||||
|
CoreSharedModule,
|
||||||
|
],
|
||||||
|
})
|
||||||
|
export default class AddonModScormOnlinePlayerPage implements OnInit, OnDestroy {
|
||||||
|
|
||||||
|
@ViewChild(CoreIframeComponent) iframe?: CoreIframeComponent;
|
||||||
|
|
||||||
|
scorm!: AddonModScormScorm; // The SCORM object.
|
||||||
|
loaded = false; // Whether the data has been loaded.
|
||||||
|
src?: string; // Iframe src.
|
||||||
|
errorMessage?: string; // Error message.
|
||||||
|
scormWidth?: number; // Width applied to scorm iframe.
|
||||||
|
scormHeight?: number; // Height applied to scorm iframe.
|
||||||
|
cmId!: number; // Course module ID.
|
||||||
|
courseId!: number; // Course ID.
|
||||||
|
enableFullScreenOnRotate = false;
|
||||||
|
|
||||||
|
protected mode!: AddonModScormMode; // Mode to play the SCORM.
|
||||||
|
protected moduleUrl!: string; // Module URL.
|
||||||
|
protected newAttempt = false; // Whether to start a new attempt.
|
||||||
|
protected organizationId?: string; // Organization ID to load.
|
||||||
|
protected attempt = 0; // The attempt number.
|
||||||
|
protected initialScoId?: number; // Initial SCO ID to load.
|
||||||
|
protected onlineObserver: Subscription;
|
||||||
|
protected isDestroyed = false;
|
||||||
|
|
||||||
|
constructor() {
|
||||||
|
let isOnline = CoreNetwork.isOnline();
|
||||||
|
this.onlineObserver = CoreNetwork.onChange().subscribe(() => {
|
||||||
|
// Execute the callback in the Angular zone, so change detection doesn't stop working.
|
||||||
|
NgZone.run(() => {
|
||||||
|
const wasOnline = isOnline;
|
||||||
|
isOnline = CoreNetwork.isOnline();
|
||||||
|
|
||||||
|
if (!isOnline && wasOnline) {
|
||||||
|
// User lost connection while playing an online package. Show an error.
|
||||||
|
CoreDomUtils.showErrorModal(new CoreError(Translate.instant('core.course.changesofflinemaybelost'), {
|
||||||
|
title: Translate.instant('core.youreoffline'),
|
||||||
|
}));
|
||||||
|
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
});
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @inheritdoc
|
||||||
|
*/
|
||||||
|
async ngOnInit(): Promise<void> {
|
||||||
|
try {
|
||||||
|
this.cmId = CoreNavigator.getRequiredRouteNumberParam('cmId');
|
||||||
|
this.courseId = CoreNavigator.getRequiredRouteNumberParam('courseId');
|
||||||
|
this.mode = CoreNavigator.getRouteParam('mode') || AddonModScormMode.NORMAL;
|
||||||
|
this.moduleUrl = CoreNavigator.getRouteParam('moduleUrl') || '';
|
||||||
|
this.newAttempt = !!CoreNavigator.getRouteBooleanParam('newAttempt');
|
||||||
|
this.organizationId = CoreNavigator.getRouteParam('organizationId');
|
||||||
|
this.initialScoId = CoreNavigator.getRouteNumberParam('scoId');
|
||||||
|
} catch (error) {
|
||||||
|
CoreDomUtils.showErrorModal(error);
|
||||||
|
CoreNavigator.back();
|
||||||
|
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
try {
|
||||||
|
// Fetch the SCORM data.
|
||||||
|
await this.fetchData();
|
||||||
|
|
||||||
|
if (!this.src) {
|
||||||
|
CoreNavigator.back();
|
||||||
|
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
} finally {
|
||||||
|
this.loaded = true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Initialize.
|
||||||
|
*/
|
||||||
|
protected async initialize(): Promise<void> {
|
||||||
|
// Get the SCORM instance.
|
||||||
|
this.scorm = await AddonModScorm.getScorm(this.courseId, this.cmId, {
|
||||||
|
moduleUrl: this.moduleUrl,
|
||||||
|
readingStrategy: CoreSitesReadingStrategy.PREFER_CACHE,
|
||||||
|
});
|
||||||
|
|
||||||
|
if (!this.scorm.popup || !this.scorm.width || this.scorm.width <= 100) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
// If we receive a value > 100 we assume it's a fixed pixel size.
|
||||||
|
this.scormWidth = this.scorm.width;
|
||||||
|
|
||||||
|
// Only get fixed size on height if width is also fixed.
|
||||||
|
if (this.scorm.height && this.scorm.height > 100) {
|
||||||
|
this.scormHeight = this.scorm.height;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Determine the attempt to use, the mode (normal/preview) and if it's offline or online.
|
||||||
|
*
|
||||||
|
* @param attemptsData Attempts count.
|
||||||
|
* @param accessInfo Access info.
|
||||||
|
*/
|
||||||
|
protected async determineAttemptAndMode(
|
||||||
|
attemptsData: AddonModScormAttemptCountResult,
|
||||||
|
accessInfo: AddonModScormGetScormAccessInformationWSResponse,
|
||||||
|
): Promise<void> {
|
||||||
|
const data = await AddonModScormHelper.determineAttemptToContinue(this.scorm, attemptsData);
|
||||||
|
|
||||||
|
let incomplete = false;
|
||||||
|
this.attempt = data.num;
|
||||||
|
|
||||||
|
// Check if current attempt is incomplete.
|
||||||
|
if (this.attempt > 0) {
|
||||||
|
incomplete = await AddonModScorm.isAttemptIncomplete(this.scorm.id, this.attempt, {
|
||||||
|
cmId: this.cmId,
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
// Determine mode and attempt to use.
|
||||||
|
const result = AddonModScorm.determineAttemptAndMode(
|
||||||
|
this.scorm,
|
||||||
|
this.mode,
|
||||||
|
this.attempt,
|
||||||
|
this.newAttempt,
|
||||||
|
incomplete,
|
||||||
|
accessInfo.cansavetrack,
|
||||||
|
);
|
||||||
|
|
||||||
|
if (result.attempt > this.attempt) {
|
||||||
|
// We're creating a new attempt, verify that we can create a new online attempt. We ignore cache.
|
||||||
|
await AddonModScorm.getScormUserData(this.scorm.id, result.attempt, {
|
||||||
|
cmId: this.cmId,
|
||||||
|
readingStrategy: CoreSitesReadingStrategy.ONLY_NETWORK,
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
this.mode = result.mode;
|
||||||
|
this.newAttempt = result.newAttempt;
|
||||||
|
this.attempt = result.attempt;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Fetch data needed to play the SCORM.
|
||||||
|
*/
|
||||||
|
protected async fetchData(): Promise<void> {
|
||||||
|
await this.initialize();
|
||||||
|
|
||||||
|
try {
|
||||||
|
// Get attempts data.
|
||||||
|
const [attemptsData, accessInfo] = await Promise.all([
|
||||||
|
AddonModScorm.getAttemptCount(this.scorm.id, { cmId: this.cmId }),
|
||||||
|
AddonModScorm.getAccessInformation(this.scorm.id, {
|
||||||
|
cmId: this.cmId,
|
||||||
|
}),
|
||||||
|
]);
|
||||||
|
|
||||||
|
await this.determineAttemptAndMode(attemptsData, accessInfo);
|
||||||
|
|
||||||
|
const sco = await this.getScoToLoad();
|
||||||
|
if (!sco) {
|
||||||
|
// We couldn't find a SCO to load: they're all inactive or without launch URL.
|
||||||
|
this.errorMessage = 'addon.mod_scorm.errornovalidsco';
|
||||||
|
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Load SCO.
|
||||||
|
this.src = await AddonModScorm.getScoSrcForOnlinePlayer(this.scorm, sco, {
|
||||||
|
mode: this.mode,
|
||||||
|
organization: this.organizationId,
|
||||||
|
newAttempt: this.newAttempt,
|
||||||
|
});
|
||||||
|
} catch (error) {
|
||||||
|
CoreDomUtils.showErrorModalDefault(error, 'addon.mod_scorm.errorgetscorm', true);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Fetch the TOC.
|
||||||
|
*
|
||||||
|
* @returns SCO to load.
|
||||||
|
*/
|
||||||
|
protected async getScoToLoad(): Promise<AddonModScormScoWithData | undefined> {
|
||||||
|
// We need to check incomplete again: attempt number or status might have changed.
|
||||||
|
const incomplete = await AddonModScorm.isAttemptIncomplete(this.scorm.id, this.attempt, {
|
||||||
|
cmId: this.cmId,
|
||||||
|
});
|
||||||
|
|
||||||
|
// Get TOC.
|
||||||
|
const toc = await AddonModScormHelper.getToc(this.scorm.id, this.attempt, incomplete, {
|
||||||
|
organization: this.organizationId,
|
||||||
|
cmId: this.cmId,
|
||||||
|
});
|
||||||
|
|
||||||
|
if (this.newAttempt) {
|
||||||
|
// Creating a new attempt, use the first SCO defined by the SCORM.
|
||||||
|
this.initialScoId = this.scorm.launch;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Determine current SCO if we received an ID.
|
||||||
|
let currentSco: AddonModScormScoWithData | undefined;
|
||||||
|
if (this.initialScoId && this.initialScoId > 0) {
|
||||||
|
// SCO set by parameter, get it from TOC.
|
||||||
|
currentSco = AddonModScormHelper.getScoFromToc(toc, this.initialScoId);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (currentSco) {
|
||||||
|
return currentSco;
|
||||||
|
}
|
||||||
|
|
||||||
|
// No SCO defined. Get the first valid one.
|
||||||
|
return await AddonModScormHelper.getFirstSco(this.scorm.id, this.attempt, {
|
||||||
|
toc,
|
||||||
|
organization: this.organizationId,
|
||||||
|
mode: this.mode,
|
||||||
|
cmId: this.cmId,
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @inheritdoc
|
||||||
|
*/
|
||||||
|
ngOnDestroy(): void {
|
||||||
|
this.isDestroyed = true;
|
||||||
|
this.onlineObserver.unsubscribe();
|
||||||
|
|
||||||
|
// Empty src when leaving the state so unload event is triggered in the iframe.
|
||||||
|
this.src = '';
|
||||||
|
CoreEvents.trigger(CoreEvents.ACTIVITY_DATA_SENT, { module: 'scorm' });
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* SCORM iframe has been loaded.
|
||||||
|
*/
|
||||||
|
async iframeLoaded(): Promise<void> {
|
||||||
|
// When using online player, some packages don't calculate the right height. Sending a 'resize' event doesn't fix it, but
|
||||||
|
// changing the iframe size makes the SCORM recalculate the size.
|
||||||
|
// Wait 1 second (to let inner iframes load) and then force full screen to make the SCORM recalculate the size.
|
||||||
|
await CoreWait.wait(1000);
|
||||||
|
|
||||||
|
if (this.isDestroyed) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
this.iframe?.toggleFullscreen(true);
|
||||||
|
this.enableFullScreenOnRotate = true;
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
|
@ -14,7 +14,6 @@
|
||||||
|
|
||||||
import { Component, OnInit, OnDestroy } from '@angular/core';
|
import { Component, OnInit, OnDestroy } from '@angular/core';
|
||||||
import { CoreNavigationBarItem } from '@components/navigation-bar/navigation-bar';
|
import { CoreNavigationBarItem } from '@components/navigation-bar/navigation-bar';
|
||||||
import { CoreMainMenuPage } from '@features/mainmenu/pages/menu/menu';
|
|
||||||
import { CoreNavigator } from '@services/navigator';
|
import { CoreNavigator } from '@services/navigator';
|
||||||
import { CoreSites, CoreSitesReadingStrategy } from '@services/sites';
|
import { CoreSites, CoreSitesReadingStrategy } from '@services/sites';
|
||||||
import { CoreSync } from '@services/sync';
|
import { CoreSync } from '@services/sync';
|
||||||
|
@ -89,10 +88,6 @@ export class AddonModScormPlayerPage implements OnInit, OnDestroy {
|
||||||
protected launchPrevObserver?: CoreEventObserver;
|
protected launchPrevObserver?: CoreEventObserver;
|
||||||
protected goOfflineObserver?: CoreEventObserver;
|
protected goOfflineObserver?: CoreEventObserver;
|
||||||
|
|
||||||
constructor(
|
|
||||||
protected mainMenuPage: CoreMainMenuPage,
|
|
||||||
) {}
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @inheritdoc
|
* @inheritdoc
|
||||||
*/
|
*/
|
||||||
|
@ -311,8 +306,6 @@ export class AddonModScormPlayerPage implements OnInit, OnDestroy {
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Fetch data needed to play the SCORM.
|
* Fetch data needed to play the SCORM.
|
||||||
*
|
|
||||||
* @returns Promise resolved when done.
|
|
||||||
*/
|
*/
|
||||||
protected async fetchData(): Promise<void> {
|
protected async fetchData(): Promise<void> {
|
||||||
if (!this.scorm) {
|
if (!this.scorm) {
|
||||||
|
|
|
@ -29,6 +29,10 @@ const routes: Routes = [
|
||||||
path: ':courseId/:cmId/player',
|
path: ':courseId/:cmId/player',
|
||||||
component: AddonModScormPlayerPage,
|
component: AddonModScormPlayerPage,
|
||||||
},
|
},
|
||||||
|
{
|
||||||
|
path: ':courseId/:cmId/online-player',
|
||||||
|
loadComponent: () => import('./pages/online-player/online-player'),
|
||||||
|
},
|
||||||
];
|
];
|
||||||
|
|
||||||
@NgModule({
|
@NgModule({
|
||||||
|
|
|
@ -23,7 +23,7 @@ import { CoreFileSizeSum } from '@services/plugin-file-delegate';
|
||||||
import { CoreSites, CoreSitesReadingStrategy } from '@services/sites';
|
import { CoreSites, CoreSitesReadingStrategy } from '@services/sites';
|
||||||
import { CorePromiseUtils } from '@singletons/promise-utils';
|
import { CorePromiseUtils } from '@singletons/promise-utils';
|
||||||
import { CoreWSFile } from '@services/ws';
|
import { CoreWSFile } from '@services/ws';
|
||||||
import { makeSingleton, Translate } from '@singletons';
|
import { makeSingleton } from '@singletons';
|
||||||
import { AddonModScorm, AddonModScormScorm } from '../scorm';
|
import { AddonModScorm, AddonModScormScorm } from '../scorm';
|
||||||
import { AddonModScormSync } from '../scorm-sync';
|
import { AddonModScormSync } from '../scorm-sync';
|
||||||
import { ADDON_MOD_SCORM_COMPONENT } from '../../constants';
|
import { ADDON_MOD_SCORM_COMPONENT } from '../../constants';
|
||||||
|
@ -177,10 +177,9 @@ export class AddonModScormPrefetchHandlerService extends CoreCourseActivityPrefe
|
||||||
|
|
||||||
siteId = siteId || CoreSites.getCurrentSiteId();
|
siteId = siteId || CoreSites.getCurrentSiteId();
|
||||||
|
|
||||||
const result = AddonModScorm.isScormUnsupported(scorm);
|
if (AddonModScorm.useOnlinePlayer(scorm)) {
|
||||||
|
// Shouldn't happen, if scorm uses online player it shouldn't be downloaded.
|
||||||
if (result) {
|
throw new CoreError('This SCORM cannot be downloaded.');
|
||||||
throw new CoreError(Translate.instant(result));
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// First verify that the file needs to be downloaded.
|
// First verify that the file needs to be downloaded.
|
||||||
|
@ -270,7 +269,7 @@ export class AddonModScormPrefetchHandlerService extends CoreCourseActivityPrefe
|
||||||
async getDownloadSize(module: CoreCourseAnyModuleData, courseId: number): Promise<CoreFileSizeSum> {
|
async getDownloadSize(module: CoreCourseAnyModuleData, courseId: number): Promise<CoreFileSizeSum> {
|
||||||
const scorm = await this.getScorm(module, courseId);
|
const scorm = await this.getScorm(module, courseId);
|
||||||
|
|
||||||
if (AddonModScorm.isScormUnsupported(scorm)) {
|
if (AddonModScorm.useOnlinePlayer(scorm)) {
|
||||||
return { size: -1, total: false };
|
return { size: -1, total: false };
|
||||||
} else if (!scorm.packagesize) {
|
} else if (!scorm.packagesize) {
|
||||||
// We don't have package size, try to calculate it.
|
// We don't have package size, try to calculate it.
|
||||||
|
@ -353,7 +352,7 @@ export class AddonModScormPrefetchHandlerService extends CoreCourseActivityPrefe
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (AddonModScorm.isScormUnsupported(scorm)) {
|
if (AddonModScorm.useOnlinePlayer(scorm)) {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -962,6 +962,32 @@ export class AddonModScormProvider {
|
||||||
return CorePath.concatenatePaths(dirPath, launchUrl);
|
return CorePath.concatenatePaths(dirPath, launchUrl);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Given a SCORM and a SCO, returns the full launch URL for the SCO to be used in an online player.
|
||||||
|
*
|
||||||
|
* @param scorm SCORM.
|
||||||
|
* @param sco SCO.
|
||||||
|
* @param options Other options.
|
||||||
|
* @returns The URL.
|
||||||
|
*/
|
||||||
|
async getScoSrcForOnlinePlayer(
|
||||||
|
scorm: AddonModScormScorm,
|
||||||
|
sco: AddonModScormWSSco,
|
||||||
|
options: AddonModScormGetScoSrcForOnlinePlayerOptions = {},
|
||||||
|
): Promise<string> {
|
||||||
|
const site = await CoreSites.getSite(options.siteId);
|
||||||
|
|
||||||
|
// Use online player.
|
||||||
|
return CoreUrl.addParamsToUrl(CorePath.concatenatePaths(site.getURL(), '/mod/scorm/player.php'), {
|
||||||
|
a: scorm.id,
|
||||||
|
scoid: sco.id,
|
||||||
|
display: 'popup',
|
||||||
|
mode: options.mode,
|
||||||
|
currentorg: options.organization,
|
||||||
|
newattempt: options.newAttempt ? 'on' : 'off',
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Get the path to the folder where a SCORM is downloaded.
|
* Get the path to the folder where a SCORM is downloaded.
|
||||||
*
|
*
|
||||||
|
@ -985,7 +1011,7 @@ export class AddonModScormProvider {
|
||||||
getScormFileList(scorm: AddonModScormScorm): CoreWSFile[] {
|
getScormFileList(scorm: AddonModScormScorm): CoreWSFile[] {
|
||||||
const files: CoreWSFile[] = [];
|
const files: CoreWSFile[] = [];
|
||||||
|
|
||||||
if (!this.isScormUnsupported(scorm) && !scorm.warningMessage) {
|
if (!this.useOnlinePlayer(scorm) && !scorm.warningMessage) {
|
||||||
files.push({
|
files.push({
|
||||||
fileurl: this.getPackageUrl(scorm),
|
fileurl: this.getPackageUrl(scorm),
|
||||||
filepath: '/',
|
filepath: '/',
|
||||||
|
@ -1359,22 +1385,6 @@ export class AddonModScormProvider {
|
||||||
return !!(scorm.timeopen && scorm.timeopen > CoreTimeUtils.timestamp());
|
return !!(scorm.timeopen && scorm.timeopen > CoreTimeUtils.timestamp());
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
|
||||||
* Check if a SCORM is unsupported in the app. If it's not, returns the error code to show.
|
|
||||||
*
|
|
||||||
* @param scorm SCORM to check.
|
|
||||||
* @returns String with error code if unsupported, undefined if supported.
|
|
||||||
*/
|
|
||||||
isScormUnsupported(scorm: AddonModScormScorm): string | undefined {
|
|
||||||
if (!this.isScormValidVersion(scorm)) {
|
|
||||||
return 'addon.mod_scorm.errorinvalidversion';
|
|
||||||
} else if (!this.isScormDownloadable(scorm)) {
|
|
||||||
return 'addon.mod_scorm.errornotdownloadable';
|
|
||||||
} else if (!this.isValidPackageUrl(this.getPackageUrl(scorm))) {
|
|
||||||
return 'addon.mod_scorm.errorpackagefile';
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Check if it's a valid SCORM 1.2.
|
* Check if it's a valid SCORM 1.2.
|
||||||
*
|
*
|
||||||
|
@ -1698,6 +1708,17 @@ export class AddonModScormProvider {
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Check if a SCORM should use an online player.
|
||||||
|
*
|
||||||
|
* @param scorm SCORM to check.
|
||||||
|
* @returns True if it should use an online player.
|
||||||
|
*/
|
||||||
|
useOnlinePlayer(scorm: AddonModScormScorm): boolean {
|
||||||
|
return !this.isScormValidVersion(scorm) || !this.isScormDownloadable(scorm) ||
|
||||||
|
!this.isValidPackageUrl(this.getPackageUrl(scorm));
|
||||||
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
export const AddonModScorm = makeSingleton(AddonModScormProvider);
|
export const AddonModScorm = makeSingleton(AddonModScormProvider);
|
||||||
|
@ -2066,6 +2087,16 @@ export type AddonModScormScoIcon = {
|
||||||
description: string;
|
description: string;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Options to pass to getScoSrcForOnlinePlayer.
|
||||||
|
*/
|
||||||
|
export type AddonModScormGetScoSrcForOnlinePlayerOptions = {
|
||||||
|
siteId?: string;
|
||||||
|
mode?: string; // Navigation mode.
|
||||||
|
organization?: string; // Organization ID.
|
||||||
|
newAttempt?: boolean; // Whether to start a new attempt.
|
||||||
|
};
|
||||||
|
|
||||||
declare module '@singletons/events' {
|
declare module '@singletons/events' {
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
|
|
@ -159,13 +159,27 @@ Feature: Test basic usage of SCORM activity in app
|
||||||
# And I go back in the app
|
# And I go back in the app
|
||||||
# Then I should find "1" within "Number of attempts you have made" "ion-item" in the app
|
# Then I should find "1" within "Number of attempts you have made" "ion-item" in the app
|
||||||
|
|
||||||
Scenario: Unsupported SCORM
|
Scenario: SCORM 2004 works online
|
||||||
Given the following "activities" exist:
|
Given the following "activities" exist:
|
||||||
| activity | name | course | idnumber | packagefilepath |
|
| activity | name | course | idnumber | packagefilepath |
|
||||||
| scorm | SCORM 1.2 | C1 | scorm | mod/scorm/tests/packages/RuntimeBasicCalls_SCORM20043rdEdition.zip |
|
| scorm | SCORM 2004 | C1 | scorm | mod/scorm/tests/packages/RuntimeBasicCalls_SCORM20043rdEdition.zip |
|
||||||
And I entered the course "Course 1" as "student1" in the app
|
And I entered the course "Course 1" as "student1" in the app
|
||||||
When I press "SCORM 1.2" in the app
|
When I press "SCORM 2004" in the app
|
||||||
Then I should find "Sorry, the application only supports SCORM 1.2." in the app
|
Then I should be able to press "Enter" in the app
|
||||||
|
|
||||||
|
When I switch network connection to offline
|
||||||
|
Then I should find "This activity is not available offline" in the app
|
||||||
|
And I should not be able to press "Enter" in the app
|
||||||
|
|
||||||
|
When I switch network connection to wifi
|
||||||
|
And I press "Enter" in the app
|
||||||
|
And I press "Disable fullscreen" in the app
|
||||||
|
Then I should not be able to press "TOC" in the app
|
||||||
|
|
||||||
|
When I switch network connection to offline
|
||||||
|
Then I should find "Any changes you make to this activity while offline may not be saved" in the app
|
||||||
|
|
||||||
|
# TODO: When iframes are fixed, test that the iframe actually works. However, the Golf 2004 SCORM has some issues.
|
||||||
|
|
||||||
Scenario: Hidden SCOs not displayed in TOC
|
Scenario: Hidden SCOs not displayed in TOC
|
||||||
Given the following "activities" exist:
|
Given the following "activities" exist:
|
||||||
|
|
|
@ -64,6 +64,7 @@ export class CoreIframeComponent implements OnChanges, OnDestroy {
|
||||||
|
|
||||||
initialized = false;
|
initialized = false;
|
||||||
|
|
||||||
|
protected fullScreenInitialized = false;
|
||||||
protected iframe?: HTMLIFrameElement;
|
protected iframe?: HTMLIFrameElement;
|
||||||
protected style?: HTMLStyleElement;
|
protected style?: HTMLStyleElement;
|
||||||
protected orientationObs?: CoreEventObserver;
|
protected orientationObs?: CoreEventObserver;
|
||||||
|
@ -87,36 +88,6 @@ export class CoreIframeComponent implements OnChanges, OnDestroy {
|
||||||
|
|
||||||
this.initialized = true;
|
this.initialized = true;
|
||||||
|
|
||||||
if (this.showFullscreenOnToolbar || this.autoFullscreenOnRotate) {
|
|
||||||
// Leave fullscreen when navigating.
|
|
||||||
this.navSubscription = Router.events
|
|
||||||
.pipe(filter(event => event instanceof NavigationStart))
|
|
||||||
.subscribe(async () => {
|
|
||||||
if (this.fullscreen) {
|
|
||||||
this.toggleFullscreen(false);
|
|
||||||
}
|
|
||||||
});
|
|
||||||
|
|
||||||
const shadow =
|
|
||||||
this.elementRef.nativeElement.closest('.ion-page')?.querySelector('ion-header ion-toolbar')?.shadowRoot;
|
|
||||||
if (shadow) {
|
|
||||||
this.style = document.createElement('style');
|
|
||||||
shadow.appendChild(this.style);
|
|
||||||
}
|
|
||||||
|
|
||||||
if (this.autoFullscreenOnRotate) {
|
|
||||||
this.toggleFullscreen(CoreScreen.isLandscape);
|
|
||||||
|
|
||||||
this.orientationObs = CoreEvents.on(CoreEvents.ORIENTATION_CHANGE, (data) => {
|
|
||||||
if (this.isInHiddenPage()) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
this.toggleFullscreen(data.orientation == CoreScreenOrientation.LANDSCAPE);
|
|
||||||
});
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// Show loading only with external URLs.
|
// Show loading only with external URLs.
|
||||||
this.loading = !this.src || !CoreUrl.isLocalFileUrl(this.src);
|
this.loading = !this.src || !CoreUrl.isLocalFileUrl(this.src);
|
||||||
|
|
||||||
|
@ -127,6 +98,72 @@ export class CoreIframeComponent implements OnChanges, OnDestroy {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Configure fullscreen based on the inputs.
|
||||||
|
*/
|
||||||
|
protected configureFullScreen(): void {
|
||||||
|
if (!this.showFullscreenOnToolbar && !this.autoFullscreenOnRotate) {
|
||||||
|
// Full screen disabled, stop watchers if enabled.
|
||||||
|
this.navSubscription?.unsubscribe();
|
||||||
|
this.orientationObs?.off();
|
||||||
|
this.style?.remove();
|
||||||
|
this.navSubscription = undefined;
|
||||||
|
this.orientationObs = undefined;
|
||||||
|
this.style = undefined;
|
||||||
|
this.fullScreenInitialized = true;
|
||||||
|
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!this.navSubscription) {
|
||||||
|
// Leave fullscreen when navigating.
|
||||||
|
this.navSubscription = Router.events
|
||||||
|
.pipe(filter(event => event instanceof NavigationStart))
|
||||||
|
.subscribe(async () => {
|
||||||
|
if (this.fullscreen) {
|
||||||
|
this.toggleFullscreen(false);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!this.style) {
|
||||||
|
const shadow = this.elementRef.nativeElement.closest('.ion-page')?.querySelector('ion-header ion-toolbar')?.shadowRoot;
|
||||||
|
if (shadow) {
|
||||||
|
this.style = document.createElement('style');
|
||||||
|
shadow.appendChild(this.style);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!this.autoFullscreenOnRotate) {
|
||||||
|
this.orientationObs?.off();
|
||||||
|
this.orientationObs = undefined;
|
||||||
|
this.fullScreenInitialized = true;
|
||||||
|
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (this.orientationObs) {
|
||||||
|
this.fullScreenInitialized = true;
|
||||||
|
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!this.fullScreenInitialized) {
|
||||||
|
// Only change full screen value if it's being initialized.
|
||||||
|
this.toggleFullscreen(CoreScreen.isLandscape);
|
||||||
|
}
|
||||||
|
|
||||||
|
this.orientationObs = CoreEvents.on(CoreEvents.ORIENTATION_CHANGE, (data) => {
|
||||||
|
if (this.isInHiddenPage()) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
this.toggleFullscreen(data.orientation == CoreScreenOrientation.LANDSCAPE);
|
||||||
|
});
|
||||||
|
|
||||||
|
this.fullScreenInitialized = true;
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Initialize things related to the iframe element.
|
* Initialize things related to the iframe element.
|
||||||
*/
|
*/
|
||||||
|
@ -170,6 +207,10 @@ export class CoreIframeComponent implements OnChanges, OnDestroy {
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!changes.src) {
|
if (!changes.src) {
|
||||||
|
if (changes.showFullscreenOnToolbar || changes.autoFullscreenOnRotate) {
|
||||||
|
this.configureFullScreen();
|
||||||
|
}
|
||||||
|
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -212,6 +253,9 @@ export class CoreIframeComponent implements OnChanges, OnDestroy {
|
||||||
// Now that the URL has been set, initialize the iframe. Wait for the iframe to the added to the DOM.
|
// Now that the URL has been set, initialize the iframe. Wait for the iframe to the added to the DOM.
|
||||||
setTimeout(() => {
|
setTimeout(() => {
|
||||||
this.init();
|
this.init();
|
||||||
|
if (changes.showFullscreenOnToolbar || changes.autoFullscreenOnRotate) {
|
||||||
|
this.configureFullScreen();
|
||||||
|
}
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -229,6 +273,12 @@ export class CoreIframeComponent implements OnChanges, OnDestroy {
|
||||||
this.orientationObs?.off();
|
this.orientationObs?.off();
|
||||||
this.navSubscription?.unsubscribe();
|
this.navSubscription?.unsubscribe();
|
||||||
window.removeEventListener('message', this.messageListenerFunction);
|
window.removeEventListener('message', this.messageListenerFunction);
|
||||||
|
|
||||||
|
if (this.fullscreen) {
|
||||||
|
// Make sure to leave fullscreen mode when the iframe is destroyed. This can happen if there's a race condition
|
||||||
|
// between clicking back button and some code toggling the fullscreen on.
|
||||||
|
this.toggleFullscreen(false);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
|
|
@ -2,6 +2,7 @@
|
||||||
@import "utilities/borders";
|
@import "utilities/borders";
|
||||||
@import "utilities/flex";
|
@import "utilities/flex";
|
||||||
@import "utilities/float";
|
@import "utilities/float";
|
||||||
|
@import "utilities/screenreaders";
|
||||||
@import "utilities/spacing";
|
@import "utilities/spacing";
|
||||||
@import "utilities/text";
|
@import "utilities/text";
|
||||||
@import "utilities/visibility";
|
@import "utilities/visibility";
|
||||||
|
|
|
@ -3,12 +3,24 @@
|
||||||
* This is partially done as part of https://tracker.moodle.org/browse/MDL-71979.
|
* This is partially done as part of https://tracker.moodle.org/browse/MDL-71979.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
/* Bootstrap 5 bridge classes */
|
/* Bootstrap 5 bridge classes */
|
||||||
|
|
||||||
|
/*
|
||||||
|
* These variables used to bridge the gap between Bootstrap 4 and Bootstrap 5 for
|
||||||
|
* alert and list-group-item.
|
||||||
|
*/
|
||||||
|
|
||||||
|
// Reduces the background color intensity by 80%. This is the definition in BS5.
|
||||||
|
$alert-bg-scale: -80% !default;
|
||||||
|
// Reduces the border color intensity by 70%. This is the definition in BS5.
|
||||||
|
$alert-border-scale: -70% !default;
|
||||||
|
// Increases the text color intensity by 50%. This is the definition in BS5.
|
||||||
|
$alert-color-scale: 50% !default;
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* These function used to bridge the gap between Bootstrap 4 and Bootstrap 5 and
|
* These function used to bridge the gap between Bootstrap 4 and Bootstrap 5 and
|
||||||
* and will be located in __functions.scss in Bootstrap 5
|
* and will be located in __functions.scss in Bootstrap 5
|
||||||
|
* This file should be removed as part of MDL-75669.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
// Tint a color: mix a color with white based on the provided weight.
|
// Tint a color: mix a color with white based on the provided weight.
|
||||||
|
@ -27,6 +39,15 @@
|
||||||
@return if($weight > 0, shade-color($color, $weight), tint-color($color, -$weight));
|
@return if($weight > 0, shade-color($color, $weight), tint-color($color, -$weight));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Visually hidden mixins
|
||||||
|
@mixin visually-hidden() {
|
||||||
|
@include sr-only();
|
||||||
|
}
|
||||||
|
|
||||||
|
@mixin visually-hidden-focusable() {
|
||||||
|
@include sr-only-focusable();
|
||||||
|
}
|
||||||
|
|
||||||
/* These classes are used to bridge the gap between Bootstrap 4 and Bootstrap 5. */
|
/* These classes are used to bridge the gap between Bootstrap 4 and Bootstrap 5. */
|
||||||
/* This file should be removed as part of MDL-75669. */
|
/* This file should be removed as part of MDL-75669. */
|
||||||
.g-0 {
|
.g-0 {
|
||||||
|
@ -107,3 +128,39 @@
|
||||||
.rounded-end {
|
.rounded-end {
|
||||||
@extend .rounded-right;
|
@extend .rounded-right;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Generate sized rounded classes.
|
||||||
|
.rounded-1 {
|
||||||
|
@extend .rounded-sm;
|
||||||
|
}
|
||||||
|
.rounded-3 {
|
||||||
|
@extend .rounded-lg;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Generate all font-weight and font-style classes.
|
||||||
|
.fw-light {
|
||||||
|
@extend .font-weight-light;
|
||||||
|
}
|
||||||
|
.fw-lighter {
|
||||||
|
@extend .font-weight-lighter;
|
||||||
|
}
|
||||||
|
.fw-normal {
|
||||||
|
@extend .font-weight-normal;
|
||||||
|
}
|
||||||
|
.fw-bold {
|
||||||
|
@extend .font-weight-bold;
|
||||||
|
}
|
||||||
|
.fw-bolder {
|
||||||
|
@extend .font-weight-bolder;
|
||||||
|
}
|
||||||
|
.fst-italic {
|
||||||
|
@extend .font-italic;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Generate visually-hidden classes.
|
||||||
|
.visually-hidden {
|
||||||
|
@extend .sr-only;
|
||||||
|
}
|
||||||
|
.visually-hidden-focusable {
|
||||||
|
@extend .sr-only-focusable;
|
||||||
|
}
|
||||||
|
|
Loading…
Reference in New Issue