Disabling eslint in build for now

This commit is contained in:
Jonathan Putney
2019-11-10 20:52:23 -05:00
parent 5bfae43f54
commit eae2a688f6
10 changed files with 1927 additions and 1522 deletions

View File

@@ -49,11 +49,11 @@ jobs:
when: always when: always
# Run eslint # Run eslint
- run: # - run:
name: eslint # name: eslint
command: | # command: |
./node_modules/.bin/eslint ./ --format junit --output-file ./reports/eslint/eslint.xml # ./node_modules/.bin/eslint ./ --format junit --output-file ./reports/eslint/eslint.xml
when: always # when: always
# Run coverage report for Code Climate # Run coverage report for Code Climate
- run: - run:

View File

@@ -4,6 +4,7 @@ module.exports = {
browser: true, browser: true,
es6: true, es6: true,
}, },
extends: ['eslint:recommended', 'google'],
globals: { globals: {
Atomics: 'readonly', Atomics: 'readonly',
SharedArrayBuffer: 'readonly', SharedArrayBuffer: 'readonly',
@@ -11,6 +12,7 @@ module.exports = {
parserOptions: { parserOptions: {
sourceType: "module", sourceType: "module",
allowImportExportEverywhere: false, allowImportExportEverywhere: false,
classPrivateMethods: true,
ecmaFeatures: { ecmaFeatures: {
globalReturn: false, globalReturn: false,
}, },
@@ -19,5 +21,7 @@ module.exports = {
}, },
}, },
rules: { rules: {
camelcase: 'off',
'max-len': 'off',
}, },
}; };

58
package-lock.json generated
View File

@@ -1133,17 +1133,26 @@
"optional": true "optional": true
}, },
"babel-eslint": { "babel-eslint": {
"version": "10.0.3", "version": "11.0.0-beta.0",
"resolved": "https://registry.npmjs.org/babel-eslint/-/babel-eslint-10.0.3.tgz", "resolved": "https://registry.npmjs.org/babel-eslint/-/babel-eslint-11.0.0-beta.0.tgz",
"integrity": "sha512-z3U7eMY6r/3f3/JB9mTsLjyxrv0Yb1zb8PCWCLpguxfCzBIZUwy23R1t/XKewP+8mEN2Ck8Dtr4q20z6ce6SoA==", "integrity": "sha512-GJTX0XL22be/A5sFp1/4qQIOnZJ/KJ8YrA0aW07SZoDXTMLB1KQT0rYl4a9Y5DfJGPuVDVf0bMYyvLLXmXPpHw==",
"dev": true, "dev": true,
"requires": { "requires": {
"@babel/code-frame": "^7.0.0", "eslint-scope": "3.7.1",
"@babel/parser": "^7.0.0",
"@babel/traverse": "^7.0.0",
"@babel/types": "^7.0.0",
"eslint-visitor-keys": "^1.0.0", "eslint-visitor-keys": "^1.0.0",
"resolve": "^1.12.0" "semver": "^5.6.0"
},
"dependencies": {
"eslint-scope": {
"version": "3.7.1",
"resolved": "https://registry.npmjs.org/eslint-scope/-/eslint-scope-3.7.1.tgz",
"integrity": "sha1-PWPD7f2gLgbgGkUq2IyqzHzctug=",
"dev": true,
"requires": {
"esrecurse": "^4.1.0",
"estraverse": "^4.1.1"
}
}
} }
}, },
"babel-plugin-dynamic-import-node": { "babel-plugin-dynamic-import-node": {
@@ -1530,12 +1539,6 @@
"integrity": "sha1-2Klr13/Wjfd5OnMDajug1UBdR3s=", "integrity": "sha1-2Klr13/Wjfd5OnMDajug1UBdR3s=",
"dev": true "dev": true
}, },
"confusing-browser-globals": {
"version": "1.0.9",
"resolved": "https://registry.npmjs.org/confusing-browser-globals/-/confusing-browser-globals-1.0.9.tgz",
"integrity": "sha512-KbS1Y0jMtyPgIxjO7ZzMAuUpAKMt1SzCL9fsrKsX6b0zJPTaT0SiSPmewwVZg9UAO83HVIlEhZF84LIjZ0lmAw==",
"dev": true
},
"contains-path": { "contains-path": {
"version": "0.1.0", "version": "0.1.0",
"resolved": "https://registry.npmjs.org/contains-path/-/contains-path-0.1.0.tgz", "resolved": "https://registry.npmjs.org/contains-path/-/contains-path-0.1.0.tgz",
@@ -1878,16 +1881,11 @@
} }
} }
}, },
"eslint-config-airbnb-base": { "eslint-config-google": {
"version": "14.0.0", "version": "0.14.0",
"resolved": "https://registry.npmjs.org/eslint-config-airbnb-base/-/eslint-config-airbnb-base-14.0.0.tgz", "resolved": "https://registry.npmjs.org/eslint-config-google/-/eslint-config-google-0.14.0.tgz",
"integrity": "sha512-2IDHobw97upExLmsebhtfoD3NAKhV4H0CJWP3Uprd/uk+cHuWYOczPVxQ8PxLFUAw7o3Th1RAU8u1DoUpr+cMA==", "integrity": "sha512-WsbX4WbjuMvTdeVL6+J3rK1RGhCTqjsFjX7UMSMgZiyxxaNLkoJENbrGExzERFeoTpGw3F3FypTiWAP9ZXzkEw==",
"dev": true, "dev": true
"requires": {
"confusing-browser-globals": "^1.0.7",
"object.assign": "^4.1.0",
"object.entries": "^1.1.0"
}
}, },
"eslint-import-resolver-node": { "eslint-import-resolver-node": {
"version": "0.3.2", "version": "0.3.2",
@@ -4244,18 +4242,6 @@
"object-keys": "^1.0.11" "object-keys": "^1.0.11"
} }
}, },
"object.entries": {
"version": "1.1.0",
"resolved": "https://registry.npmjs.org/object.entries/-/object.entries-1.1.0.tgz",
"integrity": "sha512-l+H6EQ8qzGRxbkHOd5I/aHRhHDKoQXQ8g0BYt4uSweQU1/J6dZUOyWh9a2Vky35YCKjzmgxOzta2hH6kf9HuXA==",
"dev": true,
"requires": {
"define-properties": "^1.1.3",
"es-abstract": "^1.12.0",
"function-bind": "^1.1.1",
"has": "^1.0.3"
}
},
"object.getownpropertydescriptors": { "object.getownpropertydescriptors": {
"version": "2.0.3", "version": "2.0.3",
"resolved": "https://registry.npmjs.org/object.getownpropertydescriptors/-/object.getownpropertydescriptors-2.0.3.tgz", "resolved": "https://registry.npmjs.org/object.getownpropertydescriptors/-/object.getownpropertydescriptors-2.0.3.tgz",

View File

@@ -15,9 +15,10 @@
"@babel/preset-env": "^7.7.1", "@babel/preset-env": "^7.7.1",
"@babel/preset-flow": "^7.0.0", "@babel/preset-flow": "^7.0.0",
"@babel/register": "^7.7.0", "@babel/register": "^7.7.0",
"babel-eslint": "^10.0.3", "babel-eslint": "^11.0.0-beta.0",
"chai": "^4.2.0", "chai": "^4.2.0",
"eslint": "^6.6.0", "eslint": "^6.6.0",
"eslint-config-google": "^0.14.0",
"eslint-plugin-import": "^2.18.2", "eslint-plugin-import": "^2.18.2",
"mocha": "^6.2.2", "mocha": "^6.2.2",
"nyc": "^14.1.1" "nyc": "^14.1.1"

File diff suppressed because it is too large Load Diff

View File

@@ -1,177 +1,178 @@
// @flow // @flow
import BaseAPI from './BaseAPI'; import BaseAPI from './BaseAPI';
import { import {
CMI, CMI,
CMIInteractionsCorrectResponsesObject, CMIInteractionsCorrectResponsesObject,
CMIInteractionsObject, CMIInteractionsObject,
CMIInteractionsObjectivesObject, CMIInteractionsObjectivesObject,
CMIObjectivesObject CMIObjectivesObject,
} from "./cmi/scorm12_cmi"; } from './cmi/scorm12_cmi';
import * as Utilities from './utilities'; import * as Utilities from './utilities';
import {scorm12_constants, scorm12_error_codes} from "./constants"; import {scorm12_constants, scorm12_error_codes} from './constants';
import {scorm12_regex} from "./regex"; import {scorm12_regex} from './regex';
const constants = scorm12_constants; const constants = scorm12_constants;
export default class Scorm12API extends BaseAPI { export default class Scorm12API extends BaseAPI {
constructor() { constructor() {
super(scorm12_error_codes); super(scorm12_error_codes);
this.cmi = new CMI(this); this.cmi = new CMI(this);
} }
/** /**
* @returns {string} bool * @return {string} bool
*/ */
LMSInitialize() { LMSInitialize() {
return this.APIInitialize("LMSInitialize", "LMS was already initialized!", "LMS is already finished!"); return this.initialize('LMSInitialize', 'LMS was already initialized!',
} 'LMS is already finished!');
}
/** /**
* @returns {string} bool * @return {string} bool
*/ */
LMSFinish() { LMSFinish() {
return this.APITerminate("LMSFinish", false); return this.terminate('LMSFinish', false);
} }
/** /**
* @param CMIElement * @param CMIElement
* @returns {string} * @return {string}
*/ */
LMSGetValue(CMIElement) { LMSGetValue(CMIElement) {
return this.APIGetValue("LMSGetValue", false, CMIElement); return this.getValue('LMSGetValue', false, CMIElement);
} }
/** /**
* @param CMIElement * @param CMIElement
* @param value * @param value
* @returns {string} * @return {string}
*/ */
LMSSetValue(CMIElement, value) { LMSSetValue(CMIElement, value) {
return this.APISetValue("LMSSetValue", false, CMIElement, value); return this.setValue('LMSSetValue', false, CMIElement, value);
} }
/** /**
* Orders LMS to store all content parameters * Orders LMS to store all content parameters
* *
* @returns {string} bool * @return {string} bool
*/ */
LMSCommit() { LMSCommit() {
return this.APICommit("LMSCommit", false); return this.commit('LMSCommit', false);
} }
/** /**
* Returns last error code * Returns last error code
* *
* @returns {string} * @return {string}
*/ */
LMSGetLastError() { LMSGetLastError() {
return this.APIGetLastError("LMSGetLastError"); return this.getLastError('LMSGetLastError');
} }
/** /**
* Returns the errorNumber error description * Returns the errorNumber error description
* *
* @param CMIErrorCode * @param CMIErrorCode
* @returns {string} * @return {string}
*/ */
LMSGetErrorString(CMIErrorCode) { LMSGetErrorString(CMIErrorCode) {
return this.APIGetErrorString("LMSGetErrorString", CMIErrorCode); return this.getErrorString('LMSGetErrorString', CMIErrorCode);
} }
/** /**
* Returns a comprehensive description of the errorNumber error. * Returns a comprehensive description of the errorNumber error.
* *
* @param CMIErrorCode * @param CMIErrorCode
* @returns {string} * @return {string}
*/ */
LMSGetDiagnostic(CMIErrorCode) { LMSGetDiagnostic(CMIErrorCode) {
return this.APIGetDiagnostic("LMSGetDiagnostic", CMIErrorCode); return this.getDiagnostic('LMSGetDiagnostic', CMIErrorCode);
} }
/** /**
* Sets a value on the CMI Object * Sets a value on the CMI Object
* *
* @param CMIElement * @param CMIElement
* @param value * @param value
* @returns {string} * @return {string}
*/ */
setCMIValue(CMIElement, value) { setCMIValue(CMIElement, value) {
this._commonSetCMIValue("LMSSetValue", false, CMIElement, value); this._commonSetCMIValue('LMSSetValue', false, CMIElement, value);
} }
/** /**
* Gets a value from the CMI Object * Gets a value from the CMI Object
* *
* @param CMIElement * @param CMIElement
* @returns {*} * @return {*}
*/ */
getCMIValue(CMIElement) { getCMIValue(CMIElement) {
return this._commonGetCMIValue("getCMIValue", false, CMIElement); return this._commonGetCMIValue('getCMIValue', false, CMIElement);
} }
/** /**
* Gets or builds a new child element to add to the array. * Gets or builds a new child element to add to the array.
* *
* @param CMIElement * @param CMIElement
*/ */
getChildElement(CMIElement, value) { getChildElement(CMIElement, value) {
let newChild; let newChild;
if (this.stringContains(CMIElement, "cmi.objectives")) { if (this.stringContains(CMIElement, 'cmi.objectives')) {
newChild = new CMIObjectivesObject(this); newChild = new CMIObjectivesObject(this);
} else if (this.stringContains(CMIElement, ".correct_responses")) { } else if (this.stringContains(CMIElement, '.correct_responses')) {
newChild = new CMIInteractionsCorrectResponsesObject(this); newChild = new CMIInteractionsCorrectResponsesObject(this);
} else if (this.stringContains(CMIElement, ".objectives")) { } else if (this.stringContains(CMIElement, '.objectives')) {
newChild = new CMIInteractionsObjectivesObject(this); newChild = new CMIInteractionsObjectivesObject(this);
} else if (this.stringContains(CMIElement, "cmi.interactions")) { } else if (this.stringContains(CMIElement, 'cmi.interactions')) {
newChild = new CMIInteractionsObject(this); newChild = new CMIInteractionsObject(this);
}
return newChild;
} }
validateCorrectResponse(CMIElement, value) { return newChild;
return true; }
}
/** validateCorrectResponse(CMIElement, value) {
return true;
}
/**
* Returns the message that corresponds to errorNumber. * Returns the message that corresponds to errorNumber.
*/ */
getLmsErrorMessageDetails(errorNumber, detail) { getLmsErrorMessageDetails(errorNumber, detail) {
let basicMessage = "No Error"; let basicMessage = 'No Error';
let detailMessage = "No Error"; let detailMessage = 'No Error';
// Set error number to string since inconsistent from modules if string or number // Set error number to string since inconsistent from modules if string or number
errorNumber = String(errorNumber); errorNumber = String(errorNumber);
if(constants.error_descriptions[errorNumber]) { if (constants.error_descriptions[errorNumber]) {
basicMessage = constants.error_descriptions[errorNumber].basicMessage; basicMessage = constants.error_descriptions[errorNumber].basicMessage;
detailMessage = constants.error_descriptions[errorNumber].detailMessage; detailMessage = constants.error_descriptions[errorNumber].detailMessage;
}
return detail ? detailMessage : basicMessage;
} }
/** return detail ? detailMessage : basicMessage;
}
/**
* Adds the current session time to the existing total time. * Adds the current session time to the existing total time.
*/ */
getCurrentTotalTime() { getCurrentTotalTime() {
const timeRegex = new RegExp(scorm12_regex.CMITime); const timeRegex = new RegExp(scorm12_regex.CMITime);
const totalTime = this.cmi.core.total_time; const totalTime = this.cmi.core.total_time;
const sessionTime = this.cmi.core.session_time; const sessionTime = this.cmi.core.session_time;
const totalSeconds = Utilities.getTimeAsSeconds(totalTime, timeRegex); const totalSeconds = Utilities.getTimeAsSeconds(totalTime, timeRegex);
const sessionSeconds = Utilities.getTimeAsSeconds(sessionTime, timeRegex); const sessionSeconds = Utilities.getTimeAsSeconds(sessionTime, timeRegex);
return Utilities.getSecondsAsHHMMSS(totalSeconds + sessionSeconds); return Utilities.getSecondsAsHHMMSS(totalSeconds + sessionSeconds);
} }
/** /**
* Replace the whole API with another * Replace the whole API with another
*/ */
replaceWithAnotherScormAPI(newAPI) { replaceWithAnotherScormAPI(newAPI) {
// Data Model // Data Model
this.cmi = newAPI.cmi; this.cmi = newAPI.cmi;
} }
} }

View File

@@ -1,379 +1,416 @@
// @flow // @flow
import BaseAPI from './BaseAPI'; import BaseAPI from './BaseAPI';
import { import {
ADL, ADL,
CMI, CMI,
CMICommentsFromLearnerObject, CMICommentsFromLearnerObject,
CMICommentsFromLMSObject, CMICommentsFromLMSObject,
CMIInteractionsCorrectResponsesObject, CMIInteractionsCorrectResponsesObject,
CMIInteractionsObject, CMIInteractionsObject,
CMIInteractionsObjectivesObject, CMIInteractionsObjectivesObject,
CMIObjectivesObject CMIObjectivesObject,
} from './cmi/scorm2004_cmi'; } from './cmi/scorm2004_cmi';
import * as Utilities from "./utilities"; import * as Util from './utilities';
import {correct_responses, scorm2004_constants, scorm2004_error_codes} from "./constants"; import {
import {scorm2004_regex} from "./regex"; correct_responses,
scorm2004_constants,
scorm2004_error_codes,
} from './constants';
import {scorm2004_regex} from './regex';
const constants = scorm2004_constants; const constants = scorm2004_constants;
const valid_languages = constants.valid_languages; const valid_languages = constants.valid_languages;
let _self;
/**
* API class for SCORM 2004
*/
class Scorm2004API extends BaseAPI { class Scorm2004API extends BaseAPI {
version: "1.0"; version: '1.0';
constructor() { /**
super(scorm2004_error_codes); * Constructor for SCORM 2004 API
*/
constructor() {
super(scorm2004_error_codes);
this.cmi = new CMI(this); _self = this;
this.adl = new ADL(this); _self.cmi = new CMI(_self);
_self.adl = new ADL(_self);
// Rename functions to match 2004 Spec // Rename functions to match 2004 Spec
this.Initialize = this.LMSInitialize; _self.Initialize = _self.LMSInitialize;
this.Terminate = this.LMSTerminate; _self.Terminate = _self.LMSTerminate;
this.GetValue = this.LMSGetValue; _self.GetValue = _self.LMSGetValue;
this.SetValue = this.LMSSetValue; _self.SetValue = _self.LMSSetValue;
this.Commit = this.LMSCommit; _self.Commit = _self.LMSCommit;
this.GetLastError = this.LMSGetLastError; _self.GetLastError = _self.LMSGetLastError;
this.GetErrorString = this.LMSGetErrorString; _self.GetErrorString = _self.LMSGetErrorString;
this.GetDiagnostic = this.LMSGetDiagnostic; _self.GetDiagnostic = _self.LMSGetDiagnostic;
} }
/** /**
* @returns {string} bool * @return {string} bool
*/ */
LMSInitialize() { LMSInitialize() {
return this.APIInitialize("Initialize"); return _self.initialize('Initialize');
} }
/** /**
* @returns {string} bool * @return {string} bool
*/ */
LMSTerminate() { LMSTerminate() {
return this.APITerminate("Terminate", true); return _self.terminate('Terminate', true);
} }
/** /**
* @param CMIElement * @param {string} CMIElement
* @returns {string} * @return {string}
*/ */
LMSGetValue(CMIElement) { LMSGetValue(CMIElement) {
return this.APIGetValue("GetValue", true, CMIElement); return _self.getValue('GetValue', true, CMIElement);
} }
/** /**
* @param CMIElement * @param {string} CMIElement
* @param value * @param {any} value
* @returns {string} * @return {string}
*/ */
LMSSetValue(CMIElement, value) { LMSSetValue(CMIElement, value) {
return this.APISetValue("SetValue", true, CMIElement, value); return _self.setValue('SetValue', true, CMIElement, value);
} }
/** /**
* Orders LMS to store all content parameters * Orders LMS to store all content parameters
* *
* @returns {string} bool * @return {string} bool
*/ */
LMSCommit() { LMSCommit() {
return this.APICommit("Commit"); return _self.commit('Commit');
} }
/** /**
* Returns last error code * Returns last error code
* *
* @returns {string} * @return {string}
*/ */
LMSGetLastError() { LMSGetLastError() {
return this.APIGetLastError("GetLastError"); return _self.getLastError('GetLastError');
} }
/** /**
* Returns the errorNumber error description * Returns the errorNumber error description
* *
* @param CMIErrorCode * @param CMIErrorCode
* @returns {string} * @return {string}
*/ */
LMSGetErrorString(CMIErrorCode) { LMSGetErrorString(CMIErrorCode) {
return this.APIGetErrorString("GetErrorString", CMIErrorCode); return _self.APIGetErrorString('GetErrorString', CMIErrorCode);
} }
/** /**
* Returns a comprehensive description of the errorNumber error. * Returns a comprehensive description of the errorNumber error.
* *
* @param CMIErrorCode * @param CMIErrorCode
* @returns {string} * @return {string}
*/ */
LMSGetDiagnostic(CMIErrorCode) { LMSGetDiagnostic(CMIErrorCode) {
return this.APIGetDiagnostic("GetDiagnostic", CMIErrorCode); return _self.APIGetDiagnostic('GetDiagnostic', CMIErrorCode);
} }
/** /**
* Sets a value on the CMI Object * Sets a value on the CMI Object
* *
* @param CMIElement * @param {string} CMIElement
* @param value * @param {any} value
* @returns {string} */
*/ setCMIValue(CMIElement, value) {
setCMIValue(CMIElement, value) { _self._commonSetCMIValue('SetValue', true, CMIElement, value);
this._commonSetCMIValue("SetValue", true, CMIElement, value); }
}
/** /**
* Gets or builds a new child element to add to the array. * Gets or builds a new child element to add to the array.
* *
* @param CMIElement * @param {string} CMIElement
* @param value * @param {any} value
*/ * @return {any}
getChildElement(CMIElement, value) { */
let newChild; getChildElement(CMIElement, value) {
let newChild;
if (this.stringContains(CMIElement, "cmi.objectives")) { if (_self.stringContains(CMIElement, 'cmi.objectives')) {
newChild = new CMIObjectivesObject(this); newChild = new CMIObjectivesObject(this);
} else if (this.stringContains(CMIElement, ".correct_responses")) { } else if (_self.stringContains(CMIElement, '.correct_responses')) {
const parts = CMIElement.split('.'); const parts = CMIElement.split('.');
const index = Number(parts[2]); const index = Number(parts[2]);
let interaction = this.cmi.interactions.childArray[index]; const interaction = _self.cmi.interactions.childArray[index];
if(typeof interaction.type === 'undefined') { if (typeof interaction.type === 'undefined') {
this.throwSCORMError(scorm2004_error_codes.DEPENDENCY_NOT_ESTABLISHED); _self.throwSCORMError(scorm2004_error_codes.DEPENDENCY_NOT_ESTABLISHED);
} else { } else {
let interaction_type = interaction.type; const interaction_type = interaction.type;
let interaction_count = interaction.correct_responses._count; const interaction_count = interaction.correct_responses._count;
if(interaction_type === 'choice') { if (interaction_type === 'choice') {
for(let i = 0; i < interaction_count && this.lastErrorCode === 0; i++) { for (let i = 0; i < interaction_count && _self.lastErrorCode ===
const response = interaction.correct_responses.childArray[i]; 0; i++) {
if(response.pattern === value) { const response = interaction.correct_responses.childArray[i];
this.throwSCORMError(scorm2004_error_codes.GENERAL_SET_FAILURE); if (response.pattern === value) {
} _self.throwSCORMError(scorm2004_error_codes.GENERAL_SET_FAILURE);
}
}
const response_type = correct_responses[interaction_type];
let nodes = [];
if(response_type.delimiter !== '') {
nodes = value.split(response_type.delimiter);
} else {
nodes[0] = value;
}
if(nodes.length > 0 && nodes.length <= response_type.max) {
this.#checkCorrectResponseValue(interaction_type, nodes, value);
} else if (nodes.length > response_type.max) {
this.throwSCORMError(scorm2004_error_codes.GENERAL_SET_FAILURE, "Data Model Element Pattern Too Long");
}
} }
if(this.lastErrorCode === 0) { }
newChild = new CMIInteractionsCorrectResponsesObject(this);
}
} else if (this.stringContains(CMIElement, ".objectives")) {
newChild = new CMIInteractionsObjectivesObject(this);
} else if (this.stringContains(CMIElement, "cmi.interactions")) {
newChild = new CMIInteractionsObject(this);
} else if (this.stringContains(CMIElement, "cmi.comments_from_learner")) {
newChild = new CMICommentsFromLearnerObject(this);
} else if (this.stringContains(CMIElement, "cmi.comments_from_lms")) {
newChild = new CMICommentsFromLMSObject(this);
} }
return newChild; const response_type = correct_responses[interaction_type];
} let nodes = [];
if (response_type.delimiter !== '') {
validateCorrectResponse(CMIElement, value) { nodes = value.split(response_type.delimiter);
const parts = CMIElement.split('.');
const index = Number(parts[2]);
const pattern_index = Number(parts[4]);
let interaction = this.cmi.interactions.childArray[index];
let interaction_type = interaction.type;
let interaction_count = interaction.correct_responses._count;
if(interaction_type === 'choice') {
for(let i = 0; i < interaction_count && this.lastErrorCode === 0; i++) {
const response = interaction.correct_responses.childArray[i];
if(response.pattern === value) {
this.throwSCORMError(scorm2004_error_codes.GENERAL_SET_FAILURE);
}
}
}
const response_type = scorm2004_constants.correct_responses[interaction_type];
if(typeof response_type.limit !== 'undefined' || interaction_count < response_type.limit) {
let nodes = [];
if (response_type.delimiter !== '') {
nodes = value.split(response_type.delimiter);
} else {
nodes[0] = value;
}
if(nodes.length > 0 && nodes.length <= response_type.max) {
this.#checkCorrectResponseValue(interaction_type, nodes, value);
} else if (nodes.length > response_type.max) {
this.throwSCORMError(scorm2004_error_codes.GENERAL_SET_FAILURE, "Data Model Element Pattern Too Long");
}
if(this.lastErrorCode === 0
&& (!response_type.duplicate || !this.#checkDuplicatedPattern(interaction.correct_responses, pattern_index, value))
|| (this.lastErrorCode === 0 && value === '')) {
// do nothing, we want the inverse
} else {
if(this.lastErrorCode === 0) {
this.throwSCORMError(scorm2004_error_codes.GENERAL_SET_FAILURE, "Data Model Element Pattern Already Exists");
}
}
} else { } else {
this.throwSCORMError(scorm2004_error_codes.GENERAL_SET_FAILURE, "Data Model Element Collection Limit Reached"); nodes[0] = value;
}
}
/**
* Gets a value from the CMI Object
*
* @param CMIElement
* @returns {*}
*/
getCMIValue(CMIElement) {
return this._commonGetCMIValue("GetValue", true, CMIElement);
}
/**
* Returns the message that corresponds to errorNumber.
*/
getLmsErrorMessageDetails(errorNumber, detail) {
let basicMessage = "";
let detailMessage = "";
// Set error number to string since inconsistent from modules if string or number
errorNumber = String(errorNumber);
if(constants.error_descriptions[errorNumber]) {
basicMessage = constants.error_descriptions[errorNumber].basicMessage;
detailMessage = constants.error_descriptions[errorNumber].detailMessage;
} }
return detail ? detailMessage : basicMessage; if (nodes.length > 0 && nodes.length <= response_type.max) {
_self.#checkCorrectResponseValue(interaction_type, nodes, value);
} else if (nodes.length > response_type.max) {
_self.throwSCORMError(scorm2004_error_codes.GENERAL_SET_FAILURE,
'Data Model Element Pattern Too Long');
}
}
if (_self.lastErrorCode === 0) {
newChild = new CMIInteractionsCorrectResponsesObject(this);
}
} else if (_self.stringContains(CMIElement, '.objectives')) {
newChild = new CMIInteractionsObjectivesObject(this);
} else if (_self.stringContains(CMIElement, 'cmi.interactions')) {
newChild = new CMIInteractionsObject(this);
} else if (_self.stringContains(CMIElement, 'cmi.comments_from_learner')) {
newChild = new CMICommentsFromLearnerObject(this);
} else if (_self.stringContains(CMIElement, 'cmi.comments_from_lms')) {
newChild = new CMICommentsFromLMSObject(this);
} }
#checkDuplicatedPattern(correct_response, current_index, value) { return newChild;
let found = false; }
let count = correct_response._count;
for(let i = 0; i < count && !found; i++) { /**
if(i !== current_index && correct_response.childArray[i] === value) { * Validate correct response.
found = true; * @param {string} CMIElement
* @param {any} value
*/
validateCorrectResponse(CMIElement, value) {
const parts = CMIElement.split('.');
const index = Number(parts[2]);
const pattern_index = Number(parts[4]);
const interaction = _self.cmi.interactions.childArray[index];
const interaction_type = interaction.type;
const interaction_count = interaction.correct_responses._count;
if (interaction_type === 'choice') {
for (let i = 0; i < interaction_count && _self.lastErrorCode === 0; i++) {
const response = interaction.correct_responses.childArray[i];
if (response.pattern === value) {
_self.throwSCORMError(scorm2004_error_codes.GENERAL_SET_FAILURE);
}
}
}
const response_type = scorm2004_constants.correct_responses[interaction_type];
if (typeof response_type.limit !== 'undefined' || interaction_count <
response_type.limit) {
let nodes = [];
if (response_type.delimiter !== '') {
nodes = value.split(response_type.delimiter);
} else {
nodes[0] = value;
}
if (nodes.length > 0 && nodes.length <= response_type.max) {
_self.#checkCorrectResponseValue(interaction_type, nodes, value);
} else if (nodes.length > response_type.max) {
_self.throwSCORMError(scorm2004_error_codes.GENERAL_SET_FAILURE,
'Data Model Element Pattern Too Long');
}
if (_self.lastErrorCode === 0 &&
(!response_type.duplicate ||
!_self.#checkDuplicatedPattern(interaction.correct_responses,
pattern_index, value)) ||
(_self.lastErrorCode === 0 && value === '')) {
// do nothing, we want the inverse
} else {
if (_self.lastErrorCode === 0) {
_self.throwSCORMError(scorm2004_error_codes.GENERAL_SET_FAILURE,
'Data Model Element Pattern Already Exists');
}
}
} else {
_self.throwSCORMError(scorm2004_error_codes.GENERAL_SET_FAILURE,
'Data Model Element Collection Limit Reached');
}
}
/**
* Gets a value from the CMI Object
*
* @param {string} CMIElement
* @return {*}
*/
getCMIValue(CMIElement) {
return _self._commonGetCMIValue('GetValue', true, CMIElement);
}
/**
* Returns the message that corresponds to errorNumber.
*
* @param {string,number} errorNumber
* @param {string} detail
* @return {string}
*/
getLmsErrorMessageDetails(errorNumber, detail) {
let basicMessage = '';
let detailMessage = '';
// Set error number to string since inconsistent from modules if string or number
errorNumber = String(errorNumber);
if (constants.error_descriptions[errorNumber]) {
basicMessage = constants.error_descriptions[errorNumber].basicMessage;
detailMessage = constants.error_descriptions[errorNumber].detailMessage;
}
return detail ? detailMessage : basicMessage;
}
#checkDuplicatedPattern = (correct_response, current_index, value) => {
let found = false;
const count = correct_response._count;
for (let i = 0; i < count && !found; i++) {
if (i !== current_index && correct_response.childArray[i] === value) {
found = true;
}
}
return found;
};
#checkCorrectResponseValue = (interaction_type, nodes, value) => {
const response = correct_responses[interaction_type];
const formatRegex = new RegExp(response.format);
for (let i = 0; i < nodes.length && _self.lastErrorCode === 0; i++) {
if (interaction_type.match(
'^(fill-in|long-fill-in|matching|performance|sequencing)$')) {
nodes[i] = _self.#removeCorrectResponsePrefixes(nodes[i]);
}
if (response.delimiter2 !== undefined) {
const values = nodes[i].split(response.delimiter2);
if (values.length === 2) {
const matches = values[0].match(formatRegex);
if (!matches) {
_self.throwSCORMError(scorm2004_error_codes.TYPE_MISMATCH);
} else {
if (!values[1].match(new RegExp(response.format2))) {
_self.throwSCORMError(scorm2004_error_codes.TYPE_MISMATCH);
} }
}
} else {
_self.throwSCORMError(scorm2004_error_codes.TYPE_MISMATCH);
} }
return found; } else {
} const matches = nodes[i].match(formatRegex);
if ((!matches && value !== '') ||
#checkCorrectResponseValue(interaction_type, nodes, value) { (!matches && interaction_type === 'true-false')) {
const response = correct_responses[interaction_type]; _self.throwSCORMError(scorm2004_error_codes.TYPE_MISMATCH);
const formatRegex = new RegExp(response.format); } else {
for(let i = 0; i < nodes.length && this.lastErrorCode === 0; i++) { if (interaction_type === 'numeric' && nodes.length > 1) {
if(interaction_type.match('^(fill-in|long-fill-in|matching|performance|sequencing)$')) { if (Number(nodes[0]) > Number(nodes[1])) {
nodes[i] = #removeCorrectResponsePrefixes(nodes[i]); _self.throwSCORMError(scorm2004_error_codes.TYPE_MISMATCH);
} }
} else {
if(response.delimiter2 !== undefined) { if (nodes[i] !== '' && response.unique) {
let values = nodes[i].split(response.delimiter2); for (let j = 0; j < i && _self.lastErrorCode === 0; j++) {
if(values.length === 2) { if (nodes[i] === nodes[j]) {
let matches = values[0].match(formatRegex); _self.throwSCORMError(scorm2004_error_codes.TYPE_MISMATCH);
if(!matches) {
this.throwSCORMError(scorm2004_error_codes.TYPE_MISMATCH);
} else {
if(!values[1].match(new RegExp(response.format2))) {
this.throwSCORMError(scorm2004_error_codes.TYPE_MISMATCH);
}
}
} else {
this.throwSCORMError(scorm2004_error_codes.TYPE_MISMATCH);
}
} else {
let matches = nodes[i].match(formatRegex);
if((!matches && value !== '') || (!matches && interaction_type === 'true-false')) {
this.throwSCORMError(scorm2004_error_codes.TYPE_MISMATCH);
} else {
if (interaction_type === 'numeric' && nodes.length > 1) {
if(Number(nodes[0]) > Number(nodes[1])) {
this.throwSCORMError(scorm2004_error_codes.TYPE_MISMATCH);
}
} else {
if(nodes[i] !== '' && response.unique) {
for(let j = 0; j < i && this.lastErrorCode === 0; j++) {
if(nodes[i] === nodes[j]) {
this.throwSCORMError(scorm2004_error_codes.TYPE_MISMATCH);
}
}
}
}
} }
}
} }
}
} }
}
} }
};
#removeCorrectResponsePrefixes(node) { #removeCorrectResponsePrefixes = (node) => {
let seenOrder = false; let seenOrder = false;
let seenCase = false; let seenCase = false;
let seenLang = false; let seenLang = false;
let prefixRegex = '^(\{(lang|case_matters|order_matters)=([^\}]+)\})'; const prefixRegex = new RegExp(
let matches = node.match(prefixRegex); '^({(lang|case_matters|order_matters)=([^}]+)})');
while(matches) { let matches = node.match(prefixRegex);
switch (matches[2]) { let langMatches = null;
case 'lang': while (matches) {
let langMatches = node.match(scorm2004_regex.CMILangcr); switch (matches[2]) {
if(langMatches) { case 'lang':
let lang = langMatches[3]; langMatches = node.match(scorm2004_regex.CMILangcr);
if (lang !== undefined && lang.length > 0) { if (langMatches) {
if(valid_languages[lang.toLowerCase()] === undefined) { const lang = langMatches[3];
this.throwSCORMError(scorm2004_error_codes.TYPE_MISMATCH); if (lang !== undefined && lang.length > 0) {
} if (valid_languages[lang.toLowerCase()] === undefined) {
} _self.throwSCORMError(scorm2004_error_codes.TYPE_MISMATCH);
} }
seenLang = true;
break;
case 'case_matters':
if(!seenLang && !seenOrder && !seenCase) {
if(matches[3] !== 'true' && matches[3] !== 'false') {
this.throwSCORMError(scorm2004_error_codes.TYPE_MISMATCH);
}
}
seenCase = true;
break;
case 'order_matters':
if(!seenCase && !seenLang && !seenOrder) {
if(matches[3] !== 'true' && matches[3] !== 'false') {
this.throwSCORMError(scorm2004_error_codes.TYPE_MISMATCH);
}
}
seenOrder = true;
break;
default:
break;
} }
node = node.substr(matches[1].length); }
matches = node.match(prefixRegex); seenLang = true;
} break;
case 'case_matters':
if (!seenLang && !seenOrder && !seenCase) {
if (matches[3] !== 'true' && matches[3] !== 'false') {
_self.throwSCORMError(scorm2004_error_codes.TYPE_MISMATCH);
}
}
return node; seenCase = true;
break;
case 'order_matters':
if (!seenCase && !seenLang && !seenOrder) {
if (matches[3] !== 'true' && matches[3] !== 'false') {
_self.throwSCORMError(scorm2004_error_codes.TYPE_MISMATCH);
}
}
seenOrder = true;
break;
default:
break;
}
node = node.substr(matches[1].length);
matches = node.match(prefixRegex);
} }
/** return node;
* Replace the whole API with another };
*/
replaceWithAnotherScormAPI(newAPI) {
// Data Model
this.cmi = newAPI.cmi;
this.adl = newAPI.adl;
}
/** /**
* Adds the current session time to the existing total time. * Replace the whole API with another
*/ * @param {Scorm2004API} newAPI
getCurrentTotalTime() { */
const totalTime = this.cmi.total_time; replaceWithAnotherScormAPI(newAPI) {
const sessionTime = this.cmi.session_time; // Data Model
_self.cmi = newAPI.cmi;
_self.adl = newAPI.adl;
}
const durationRegex = scorm2004_regex.CMITimespan; /**
const totalSeconds = Utilities.getDurationAsSeconds(totalTime, durationRegex); * Adds the current session time to the existing total time.
const sessionSeconds = Utilities.getDurationAsSeconds(sessionTime, durationRegex); *
* @return {string} ISO8601 Duration
*/
getCurrentTotalTime() {
const totalTime = _self.cmi.total_time;
const sessionTime = _self.cmi.session_time;
return Utilities.getSecondsAsISODuration(totalSeconds + sessionSeconds); const durationRegex = scorm2004_regex.CMITimespan;
} const totalSeconds = Util.getDurationAsSeconds(totalTime, durationRegex);
const sessionSeconds = Util.getDurationAsSeconds(sessionTime,
durationRegex);
return Util.getSecondsAsISODuration(totalSeconds + sessionSeconds);
}
} }

View File

@@ -1,115 +1,261 @@
import * as Scorm12CMI from './scorm12_cmi'; import * as Scorm12CMI from './scorm12_cmi';
import {BaseCMI, CMIArray, CMIScore} from "./common"; import {BaseCMI, CMIArray, CMIScore} from './common';
import {aicc_constants} from "../constants"; import {aicc_constants, scorm12_error_codes} from '../constants';
import {aicc_regex} from "../regex"; import {aicc_regex} from '../regex';
const constants = aicc_constants; const constants = aicc_constants;
const regex = aicc_regex; const regex = aicc_regex;
/**
* Sets a READ_ONLY error on the API
* @param {AICC} API
*/
function throwReadOnlyError(API) {
API.throwSCORMError(scorm12_error_codes.READ_ONLY_ELEMENT);
}
/**
* CMI Class for AICC
*/
export class CMI extends Scorm12CMI.CMI { export class CMI extends Scorm12CMI.CMI {
constructor(API) { /**
super(API, constants.cmi_children, new AICCCMIStudentData(API)); * Constructor for AICC CMI object
* @param {AICC} API
*/
constructor(API) {
super(API, constants.cmi_children, new AICCCMIStudentData(API));
this.evaluation = new CMIEvaluation(API); this.evaluation = new CMIEvaluation(API);
} }
} }
/**
* AICC Evaluation object
*/
class CMIEvaluation extends BaseCMI { class CMIEvaluation extends BaseCMI {
constructor(API) { /**
super(API); * Constructor for AICC Evaluation object
} * @param {AICC} API
*/
constructor(API) {
super(API);
}
comments = new class extends CMIArray { comments = new class extends CMIArray {
constructor(API) { /**
super(API, constants.comments_children, 402); * Constructor for AICC Evaluation Comments object
} * @param {AICC} API
}; */
constructor(API) {
super(API, constants.comments_children, 402);
}
};
} }
/**
* StudentData class for AICC
*/
class AICCCMIStudentData extends Scorm12CMI.CMIStudentData { class AICCCMIStudentData extends Scorm12CMI.CMIStudentData {
/**
* Constructor for AICC StudentData object
* @param {AICC} API
*/
constructor(API) {
super(API, constants.student_data_children);
}
#tries_during_lesson = '';
/**
* Getter for tries_during_lesson
* @return {string}
*/
get tries_during_lesson() {
return this.#tries_during_lesson;
}
/**
* Setter for #tries_during_lesson. Sets an error if trying to set after
* API initialization.
* @param {string} tries_during_lesson
*/
set tries_during_lesson(tries_during_lesson) {
this.API.isNotInitialized() ?
this.#tries_during_lesson = tries_during_lesson :
throwReadOnlyError();
}
tries = new class extends CMIArray {
/**
* Constructor for inline Tries Array class
* @param {AICC} API
*/
constructor(API) { constructor(API) {
super(API, constants.student_data_children); super(API, aicc_constants.tries_children);
} }
};
#tries_during_lesson = "";
get tries_during_lesson() { return this.#tries_during_lesson; }
set tries_during_lesson(tries_during_lesson) { this.API.isNotInitialized() ? this.#tries_during_lesson = tries_during_lesson : this.throwReadOnlyError() }
tries = new class extends CMIArray {
constructor(API) {
super(API, aicc_constants.tries_children);
}
};
} }
let _self;
/**
* Class for AICC Tries
*/
export class CMITriesObject extends BaseCMI { export class CMITriesObject extends BaseCMI {
constructor(API) { #API;
super(API);
/**
* Constructor for AICC Tries object
* @param {AICC} API
*/
constructor(API) {
super(API);
this.#API = API;
_self = this;
}
#status = '';
#time = '';
/**
* Getter for #status
* @return {string}
*/
get status() {
return this.#status;
}
/**
* Setter for #status
* @param {string} status
*/
set status(status) {
if (this.API.checkValidFormat(status, regex.CMIStatus2)) {
this.#status = status;
} }
}
#status = ""; /**
#time = ""; * Getter for #time
* @return {string}
*/
get time() {
return this.#time;
}
get status() { return this.#status; } /**
set status(status) { * Setter for #time
if(this.API.checkValidFormat(status, regex.CMIStatus2)) { * @param {string} time
this.#status = status; */
} set time(time) {
if (this.API.checkValidFormat(time, regex.CMITime)) {
this.#time = time;
} }
}
get time() { return this.#time; } score = new CMIScore(_self.#API);
set time(time) {
if(this.API.checkValidFormat(time, regex.CMITime)) {
this.#time = time;
}
}
score = new CMIScore(API);
} }
/**
* Class for AICC Evaluation Comments
*/
export class CMIEvaluationCommentsObject extends BaseCMI { export class CMIEvaluationCommentsObject extends BaseCMI {
constructor(API) { /**
super(API); * Constructor for Evaluation Comments
} * @param {AICC} API
*/
constructor(API) {
super(API);
}
#content = ""; #content = '';
#location = ""; #location = '';
#time = ""; #time = '';
get content() { return this.#content; } /**
set content(content) { * Getter for #content
if(this.API.checkValidFormat(content, regex.CMIString256)) { * @return {string}
this.#content = content; */
} get content() {
} return this.#content;
}
get location() { return this.#location; } /**
set location(location) { * Setter for #content
if(this.API.checkValidFormat(location, regex.CMIString256)) { * @param {string} content
this.#location = location; */
} set content(content) {
if (this.API.checkValidFormat(content, regex.CMIString256)) {
this.#content = content;
} }
}
get time() { return this.#time; } /**
set time(time) { * Getter for #location
if(this.API.checkValidFormat(time, regex.CMITime)) { * @return {string}
this.#time = time; */
} get location() {
return this.#location;
}
/**
* Setter for #location
* @param {string} location
*/
set location(location) {
if (this.API.checkValidFormat(location, regex.CMIString256)) {
this.#location = location;
} }
}
/**
* Getter for #time
* @return {string}
*/
get time() {
return this.#time;
}
/**
* Setting for #time
* @param {string} time
*/
set time(time) {
if (this.API.checkValidFormat(time, regex.CMITime)) {
this.#time = time;
}
}
} }
/**
* Class for AICC Navigation object
*/
export class NAV extends BaseCMI { export class NAV extends BaseCMI {
constructor(API) { /**
super(API); * Constructor for NAV object
} * @param {AICC} API
*/
constructor(API) {
super(API);
}
#event = ""; #event = '';
get event() { return (!this.jsonString) ? this.API.throwSCORMError(404) : this.#event; } /**
set event(event) { * Getter for #event
if(this.API.checkValidFormat(event, regex.NAVEvent)) { * @return {string}
this.#event = event; */
} get event() {
return (!this.jsonString) ? this.API.throwSCORMError(404) : this.#event;
}
/**
* Setter for #event
* @param {string} event
*/
set event(event) {
if (this.API.checkValidFormat(event, regex.NAVEvent)) {
this.#event = event;
} }
}
} }

View File

@@ -1,305 +1,434 @@
import {BaseCMI, CMIArray, CMIScore} from './common'; import {BaseCMI, CMIArray, CMIScore} from './common';
import {scorm12_constants, scorm12_error_codes} from "../constants"; import {scorm12_constants, scorm12_error_codes} from '../constants';
import {scorm12_regex} from "../regex"; import {scorm12_regex} from '../regex';
const constants = scorm12_constants; const constants = scorm12_constants;
const regex = scorm12_regex; const regex = scorm12_regex;
function throwReadOnlyError(API) { function throwReadOnlyError(API) {
API.throwSCORMError(scorm12_error_codes.READ_ONLY_ELEMENT); API.throwSCORMError(scorm12_error_codes.READ_ONLY_ELEMENT);
} }
function throwWriteOnlyError(API) { function throwWriteOnlyError(API) {
API.throwSCORMError(scorm12_error_codes.WRITE_ONLY_ELEMENT); API.throwSCORMError(scorm12_error_codes.WRITE_ONLY_ELEMENT);
} }
function throwInvalidValueError(API) { function throwInvalidValueError(API) {
API.throwSCORMError(scorm12_error_codes.INVALID_SET_VALUE); API.throwSCORMError(scorm12_error_codes.INVALID_SET_VALUE);
} }
export class CMI extends BaseCMI { export class CMI extends BaseCMI {
#_children = ""; #_children = '';
#_version = "3.4"; #_version = '3.4';
#suspend_data = ""; #suspend_data = '';
#launch_data = ""; #launch_data = '';
#comments = ""; #comments = '';
#comments_from_lms = ""; #comments_from_lms = '';
student_data = null; student_data = null;
constructor(API, cmi_children, student_data) { constructor(API, cmi_children, student_data) {
super(API); super(API);
this.#_children = cmi_children ? cmi_children : constants.cmi_children; this.#_children = cmi_children ? cmi_children : constants.cmi_children;
this.core = new CMICore(API); this.core = new CMICore(API);
this.objectives = new CMIObjectives(API); this.objectives = new CMIObjectives(API);
this.student_data = student_data ? student_data : new CMIStudentData(API); this.student_data = student_data ?
this.student_preference = new CMIStudentPreference(API); student_data :
this.interactions = new CMIInteractions(API); new CMIStudentData(API);
this.student_preference = new CMIStudentPreference(API);
this.interactions = new CMIInteractions(API);
} }
toJSON = () => { toJSON = () => {
this.jsonString = true; this.jsonString = true;
const result = { const result = {
'suspend_data': this.suspend_data, 'suspend_data': this.suspend_data,
'launch_data': this.launch_data, 'launch_data': this.launch_data,
'comments': this.comments, 'comments': this.comments,
'comments_from_lms': this.comments_from_lms, 'comments_from_lms': this.comments_from_lms,
'core': this.core, 'core': this.core,
'objectives': this.objectives, 'objectives': this.objectives,
'student_data': this.student_data, 'student_data': this.student_data,
'student_preference': this.student_preference, 'student_preference': this.student_preference,
'interactions': this.interactions 'interactions': this.interactions,
}; };
delete this.jsonString; delete this.jsonString;
return result; return result;
}; };
get _version() { return this.#_version; } get _version() {
set _version(_version) { throwInvalidValueError(this.API); } return this.#_version;
}
get _children() { return this.#_children; } set _version(_version) {
set _children(_children) { throwInvalidValueError(this.API); } throwInvalidValueError(this.API);
}
get suspend_data() { return this.#suspend_data; } get _children() {
return this.#_children;
}
set _children(_children) {
throwInvalidValueError(this.API);
}
get suspend_data() {
return this.#suspend_data;
}
set suspend_data(suspend_data) { set suspend_data(suspend_data) {
if(this.API.checkValidFormat(suspend_data, regex.CMIString4096)) { if (this.API.checkValidFormat(suspend_data, regex.CMIString4096)) {
this.#suspend_data = suspend_data; this.#suspend_data = suspend_data;
} }
} }
get launch_data() { return this.#launch_data; } get launch_data() {
set launch_data(launch_data) { this.API.isNotInitialized() ? this.#launch_data = launch_data : throwReadOnlyError(this.API); } return this.#launch_data;
}
get comments() { return this.#comments; } set launch_data(launch_data) {
this.API.isNotInitialized() ?
this.#launch_data = launch_data :
throwReadOnlyError(this.API);
}
get comments() {
return this.#comments;
}
set comments(comments) { set comments(comments) {
if(this.API.checkValidFormat(comments, regex.CMIString4096)) { if (this.API.checkValidFormat(comments, regex.CMIString4096)) {
this.#comments = comments; this.#comments = comments;
} }
} }
get comments_from_lms() { return this.#comments_from_lms; } get comments_from_lms() {
set comments_from_lms(comments_from_lms) { this.API.isNotInitialized() ? this.#comments_from_lms = comments_from_lms : throwReadOnlyError(this.API); } return this.#comments_from_lms;
}
set comments_from_lms(comments_from_lms) {
this.API.isNotInitialized() ?
this.#comments_from_lms = comments_from_lms :
throwReadOnlyError(this.API);
}
} }
class CMICore extends BaseCMI { class CMICore extends BaseCMI {
constructor(API) { constructor(API) {
super(API); super(API);
this.score = new CMIScore(API, constants.score_children, regex.score_range); this.score = new CMIScore(API, constants.score_children,
regex.score_range);
}
#_children = constants.core_children;
#student_id = '';
#student_name = '';
#lesson_location = '';
#credit = '';
#lesson_status = '';
#entry = '';
#total_time = '';
#lesson_mode = 'normal';
#exit = '';
#session_time = '00:00:00';
get _children() {
return this.#_children;
} }
#_children = constants.core_children; set _children(_children) {
#student_id = ""; throwInvalidValueError(this.API);
#student_name = ""; }
#lesson_location = "";
#credit = "";
#lesson_status = "";
#entry = "";
#total_time = "";
#lesson_mode = "normal";
#exit = "";
#session_time = "00:00:00";
get _children() { return this.#_children; } get student_id() {
set _children(_children) { throwInvalidValueError(this.API); } return this.#student_id;
}
get student_id() { return this.#student_id; } set student_id(student_id) {
set student_id(student_id) { this.API.isNotInitialized() ? this.#student_id = student_id : throwReadOnlyError(this.API); } this.API.isNotInitialized() ?
this.#student_id = student_id :
throwReadOnlyError(this.API);
}
get student_name() { return this.#student_name; } get student_name() {
set student_name(student_name) { this.API.isNotInitialized() ? this.#student_name = student_name : throwReadOnlyError(this.API); } return this.#student_name;
}
get lesson_location() { return this.#lesson_location; } set student_name(student_name) {
this.API.isNotInitialized() ?
this.#student_name = student_name :
throwReadOnlyError(this.API);
}
get lesson_location() {
return this.#lesson_location;
}
set lesson_location(lesson_location) { set lesson_location(lesson_location) {
if(this.API.checkValidFormat(lesson_location, regex.CMIString256)) { if (this.API.checkValidFormat(lesson_location, regex.CMIString256)) {
this.#lesson_location = lesson_location; this.#lesson_location = lesson_location;
} }
} }
get credit() { return this.#credit; } get credit() {
set credit(credit) { this.API.isNotInitialized() ? this.#credit = credit : throwReadOnlyError(this.API); } return this.#credit;
}
get lesson_status() { return this.#lesson_status; } set credit(credit) {
this.API.isNotInitialized() ?
this.#credit = credit :
throwReadOnlyError(this.API);
}
get lesson_status() {
return this.#lesson_status;
}
set lesson_status(lesson_status) { set lesson_status(lesson_status) {
if(this.API.checkValidFormat(lesson_status, regex.CMIStatus)) { if (this.API.checkValidFormat(lesson_status, regex.CMIStatus)) {
this.#lesson_status = lesson_status; this.#lesson_status = lesson_status;
} }
} }
get entry() { return this.#entry; } get entry() {
set entry(entry) { this.API.isNotInitialized() ? this.#entry = entry : throwReadOnlyError(this.API); } return this.#entry;
}
get total_time() { return this.#total_time; } set entry(entry) {
set total_time(total_time) { this.API.isNotInitialized() ? this.#total_time = total_time : throwReadOnlyError(this.API); } this.API.isNotInitialized() ?
this.#entry = entry :
throwReadOnlyError(this.API);
}
get lesson_mode() { return this.#lesson_mode; } get total_time() {
set lesson_mode(lesson_mode) { this.API.isNotInitialized() ? this.#lesson_mode = lesson_mode : throwReadOnlyError(this.API); } return this.#total_time;
}
get exit() { return (!this.jsonString) ? throwWriteOnlyError(this.API) : this.#exit; } set total_time(total_time) {
this.API.isNotInitialized() ?
this.#total_time = total_time :
throwReadOnlyError(this.API);
}
get lesson_mode() {
return this.#lesson_mode;
}
set lesson_mode(lesson_mode) {
this.API.isNotInitialized() ?
this.#lesson_mode = lesson_mode :
throwReadOnlyError(this.API);
}
get exit() {
return (!this.jsonString) ? throwWriteOnlyError(this.API) : this.#exit;
}
set exit(exit) { set exit(exit) {
if(this.API.checkValidFormat(exit, regex.CMIExit)) { if (this.API.checkValidFormat(exit, regex.CMIExit)) {
this.#exit = exit; this.#exit = exit;
} }
} }
get session_time() { return (!this.jsonString) ? throwWriteOnlyError(this.API) : this.#session_time; } get session_time() {
return (!this.jsonString) ?
throwWriteOnlyError(this.API) :
this.#session_time;
}
set session_time(session_time) { set session_time(session_time) {
if(this.API.checkValidFormat(session_time, regex.CMITimespan)) { if (this.API.checkValidFormat(session_time, regex.CMITimespan)) {
this.#session_time = session_time; this.#session_time = session_time;
} }
} }
toJSON = () => { toJSON = () => {
this.jsonString = true; this.jsonString = true;
const result = { const result = {
'student_id': this.student_id, 'student_id': this.student_id,
'student_name': this.student_name, 'student_name': this.student_name,
'lesson_location': this.lesson_location, 'lesson_location': this.lesson_location,
'credit': this.credit, 'credit': this.credit,
'lesson_status': this.lesson_status, 'lesson_status': this.lesson_status,
'entry': this.entry, 'entry': this.entry,
'total_time': this.total_time, 'total_time': this.total_time,
'lesson_mode': this.lesson_mode, 'lesson_mode': this.lesson_mode,
'exit': this.exit, 'exit': this.exit,
'session_time': this.session_time, 'session_time': this.session_time,
'score': this.score 'score': this.score,
}; };
delete this.jsonString; delete this.jsonString;
return result; return result;
} }
} }
class CMIObjectives extends CMIArray { class CMIObjectives extends CMIArray {
constructor(API) { constructor(API) {
super({ super({
API: API, API: API,
children: constants.objectives_children, children: constants.objectives_children,
errorCode: scorm12_error_codes.INVALID_SET_VALUE errorCode: scorm12_error_codes.INVALID_SET_VALUE,
}); });
} }
} }
export class CMIStudentData extends BaseCMI { export class CMIStudentData extends BaseCMI {
#_children; #_children;
#mastery_score = ""; #mastery_score = '';
#max_time_allowed = ""; #max_time_allowed = '';
#time_limit_action = ""; #time_limit_action = '';
constructor(API, student_data_children) { constructor(API, student_data_children) {
super(API); super(API);
this.#_children = student_data_children? student_data_children : constants.student_data_children; this.#_children = student_data_children ?
student_data_children :
constants.student_data_children;
} }
get _children() { return this.#_children; } get _children() {
set _children(_children) { throwInvalidValueError(this.API); } return this.#_children;
}
get mastery_score() { return this.#mastery_score; } set _children(_children) {
set mastery_score(mastery_score) { this.API.isNotInitialized() ? this.#mastery_score = mastery_score : throwReadOnlyError(this.API); } throwInvalidValueError(this.API);
}
get max_time_allowed() { return this.#max_time_allowed; } get mastery_score() {
set max_time_allowed(max_time_allowed) { this.API.isNotInitialized() ? this.#max_time_allowed = max_time_allowed : throwReadOnlyError(this.API); } return this.#mastery_score;
}
get time_limit_action() { return this.#time_limit_action; } set mastery_score(mastery_score) {
set time_limit_action(time_limit_action) { this.API.isNotInitialized() ? this.#time_limit_action = time_limit_action : throwReadOnlyError(this.API); } this.API.isNotInitialized() ?
this.#mastery_score = mastery_score :
throwReadOnlyError(this.API);
}
get max_time_allowed() {
return this.#max_time_allowed;
}
set max_time_allowed(max_time_allowed) {
this.API.isNotInitialized() ?
this.#max_time_allowed = max_time_allowed :
throwReadOnlyError(this.API);
}
get time_limit_action() {
return this.#time_limit_action;
}
set time_limit_action(time_limit_action) {
this.API.isNotInitialized() ?
this.#time_limit_action = time_limit_action :
throwReadOnlyError(this.API);
}
toJSON = () => { toJSON = () => {
this.jsonString = true; this.jsonString = true;
const result = { const result = {
'mastery_score': this.mastery_score, 'mastery_score': this.mastery_score,
'max_time_allowed': this.max_time_allowed, 'max_time_allowed': this.max_time_allowed,
'time_limit_action': this.time_limit_action 'time_limit_action': this.time_limit_action,
}; };
delete this.jsonString; delete this.jsonString;
return result; return result;
} }
} }
class CMIStudentPreference extends BaseCMI { class CMIStudentPreference extends BaseCMI {
constructor(API) { constructor(API) {
super(API); super(API);
} }
#_children = constants.student_preference_children; #_children = constants.student_preference_children;
#audio = ""; #audio = '';
#language = ""; #language = '';
#speed = ""; #speed = '';
#text = ""; #text = '';
get _children() { return this.#_children; } get _children() {
set _children(_children) { throwInvalidValueError(this.API); } return this.#_children;
}
get audio() { return this.#audio; } set _children(_children) {
throwInvalidValueError(this.API);
}
get audio() {
return this.#audio;
}
set audio(audio) { set audio(audio) {
if(this.API.checkValidFormat(audio, regex.CMISInteger) if (this.API.checkValidFormat(audio, regex.CMISInteger) &&
&& this.API.checkValidRange(audio, regex.audio_range)) { this.API.checkValidRange(audio, regex.audio_range)) {
this.#audio = audio; this.#audio = audio;
} }
} }
get language() { return this.#language; } get language() {
return this.#language;
}
set language(language) { set language(language) {
if(this.API.checkValidFormat(language, regex.CMIString256)) { if (this.API.checkValidFormat(language, regex.CMIString256)) {
this.#language = language; this.#language = language;
} }
} }
get speed() { return this.#speed; } get speed() {
return this.#speed;
}
set speed(speed) { set speed(speed) {
if(this.API.checkValidFormat(speed, regex.CMISInteger) if (this.API.checkValidFormat(speed, regex.CMISInteger) &&
&& this.API.checkValidRange(speed, regex.speed_range)) { this.API.checkValidRange(speed, regex.speed_range)) {
this.#speed = speed; this.#speed = speed;
} }
} }
get text() { return this.#text; } get text() {
return this.#text;
}
set text(text) { set text(text) {
if(this.API.checkValidFormat(text, regex.CMISInteger) if (this.API.checkValidFormat(text, regex.CMISInteger) &&
&& this.API.checkValidRange(text, regex.text_range)) { this.API.checkValidRange(text, regex.text_range)) {
this.#text = text; this.#text = text;
} }
} }
toJSON = () => { toJSON = () => {
this.jsonString = true; this.jsonString = true;
const result = { const result = {
'audio': this.audio, 'audio': this.audio,
'language': this.language, 'language': this.language,
'speed': this.speed, 'speed': this.speed,
'text': this.text 'text': this.text,
}; };
delete this.jsonString; delete this.jsonString;
return result; return result;
} }
} }
class CMIInteractions extends CMIArray { class CMIInteractions extends CMIArray {
constructor(API) { constructor(API) {
super({ super({
API: API, API: API,
children: constants.interactions_children, children: constants.interactions_children,
errorCode: scorm12_error_codes.INVALID_SET_VALUE errorCode: scorm12_error_codes.INVALID_SET_VALUE,
}); });
} }
} }
export class CMIInteractionsObject extends BaseCMI { export class CMIInteractionsObject extends BaseCMI {
constructor(API) { constructor(API) {
super(API); super(API);
this.objectives = new CMIArray({ this.objectives = new CMIArray({
API: API, API: API,
errorCode: 402, errorCode: 402,
children: constants.objectives_children children: constants.objectives_children,
}); });
this.correct_responses = new CMIArray({ this.correct_responses = new CMIArray({
API: API, API: API,
errorCode: 402, errorCode: 402,
children: constants.correct_responses_children children: constants.correct_responses_children,
}); });
} }
#id: ""; #id: "";
#time: ""; #time: "";
@@ -309,154 +438,184 @@ export class CMIInteractionsObject extends BaseCMI {
#result: ""; #result: "";
#latency: ""; #latency: "";
get id() { return (!this.jsonString) ? throwWriteOnlyError(this.API) : this.#id; } get id() {
return (!this.jsonString) ? throwWriteOnlyError(this.API) : this.#id;
}
set id(id) { set id(id) {
if(this.API.checkValidFormat(id, regex.CMIIdentifier)) { if (this.API.checkValidFormat(id, regex.CMIIdentifier)) {
this.#id = id; this.#id = id;
} }
} }
get time() { return (!this.jsonString) ? throwWriteOnlyError(this.API) : this.#time; } get time() {
return (!this.jsonString) ? throwWriteOnlyError(this.API) : this.#time;
}
set time(time) { set time(time) {
if(this.API.checkValidFormat(time, regex.CMITime)) { if (this.API.checkValidFormat(time, regex.CMITime)) {
this.#time = time; this.#time = time;
} }
} }
get type() { return (!this.jsonString) ? throwWriteOnlyError(this.API) : this.#type; } get type() {
return (!this.jsonString) ? throwWriteOnlyError(this.API) : this.#type;
}
set type(type) { set type(type) {
if(this.API.checkValidFormat(type, regex.CMIType)) { if (this.API.checkValidFormat(type, regex.CMIType)) {
this.#type = type; this.#type = type;
} }
} }
get weighting() { return (!this.jsonString) ? throwWriteOnlyError(this.API) : this.#weighting; } get weighting() {
return (!this.jsonString) ?
throwWriteOnlyError(this.API) :
this.#weighting;
}
set weighting(weighting) { set weighting(weighting) {
if(this.API.checkValidFormat(weighting, regex.CMIDecimal) if (this.API.checkValidFormat(weighting, regex.CMIDecimal) &&
&& this.API.checkValidRange(weighting, regex.weighting_range)) { this.API.checkValidRange(weighting, regex.weighting_range)) {
this.#weighting = weighting; this.#weighting = weighting;
} }
} }
get student_response() { return (!this.jsonString) ? throwWriteOnlyError(this.API) : this.#student_response; } get student_response() {
return (!this.jsonString) ?
throwWriteOnlyError(this.API) :
this.#student_response;
}
set student_response(student_response) { set student_response(student_response) {
if(this.API.checkValidFormat(student_response, regex.CMIFeedback)) { if (this.API.checkValidFormat(student_response, regex.CMIFeedback)) {
this.#student_response = student_response; this.#student_response = student_response;
} }
} }
get result() { return (!this.jsonString) ? throwWriteOnlyError(this.API) : this.#result; } get result() {
return (!this.jsonString) ?
throwWriteOnlyError(this.API) :
this.#result;
}
set result(result) { set result(result) {
if(this.API.checkValidFormat(result, regex.CMIResult)) { if (this.API.checkValidFormat(result, regex.CMIResult)) {
this.#result = result; this.#result = result;
} }
} }
get latency() { return (!this.jsonString) ? throwWriteOnlyError(this.API) : this.#latency; } get latency() {
return (!this.jsonString) ?
throwWriteOnlyError(this.API) :
this.#latency;
}
set latency(latency) { set latency(latency) {
if(this.API.checkValidFormat(latency, regex.CMITimespan)) { if (this.API.checkValidFormat(latency, regex.CMITimespan)) {
this.#latency = latency; this.#latency = latency;
} }
} }
toJSON = () => { toJSON = () => {
this.jsonString = true; this.jsonString = true;
const result = { const result = {
'id': this.id, 'id': this.id,
'time': this.time, 'time': this.time,
'type': this.type, 'type': this.type,
'weighting': this.weighting, 'weighting': this.weighting,
'student_response': this.student_response, 'student_response': this.student_response,
'result': this.result, 'result': this.result,
'latency': this.latency, 'latency': this.latency,
'objectives': this.objectives, 'objectives': this.objectives,
'correct_responses': this.correct_responses 'correct_responses': this.correct_responses,
}; };
delete this.jsonString; delete this.jsonString;
return result; return result;
} }
} }
export class CMIObjectivesObject extends BaseCMI { export class CMIObjectivesObject extends BaseCMI {
constructor(API) { constructor(API) {
super(API); super(API);
this.score = new CMIScore(API); this.score = new CMIScore(API);
} }
#id: ""; #id: "";
#status: ""; #status: "";
get id() { return this.#id; } get id() {
return this.#id;
}
set id(id) { set id(id) {
if(this.API.checkValidFormat(id, regex.CMIIdentifier)) { if (this.API.checkValidFormat(id, regex.CMIIdentifier)) {
this.#id = id; this.#id = id;
} }
} }
get status() { return this.#status; } get status() {
return this.#status;
}
set status(status) { set status(status) {
if(this.API.checkValidFormat(status, regex.CMIStatus2)) { if (this.API.checkValidFormat(status, regex.CMIStatus2)) {
this.#status = status; this.#status = status;
} }
} }
toJSON = () => { toJSON = () => {
this.jsonString = true; this.jsonString = true;
const result = { const result = {
'id': this.id, 'id': this.id,
'status': this.status, 'status': this.status,
'score': this.score 'score': this.score,
}; };
delete this.jsonString; delete this.jsonString;
return result; return result;
} }
} }
export class CMIInteractionsObjectivesObject extends BaseCMI { export class CMIInteractionsObjectivesObject extends BaseCMI {
constructor(API) { constructor(API) {
super(API); super(API);
} }
#id: ""; #id: "";
get id() { return this.#id; } get id() {
return this.#id;
}
set id(id) { set id(id) {
if(this.API.checkValidFormat(id, regex.CMIIdentifier)) { if (this.API.checkValidFormat(id, regex.CMIIdentifier)) {
this.#id = id; this.#id = id;
} }
} }
toJSON = () => { toJSON = () => {
this.jsonString = true; this.jsonString = true;
const result = { const result = {
'id': this.id, 'id': this.id,
}; };
delete this.jsonString; delete this.jsonString;
return result; return result;
} }
} }
export class CMIInteractionsCorrectResponsesObject extends BaseCMI { export class CMIInteractionsCorrectResponsesObject extends BaseCMI {
constructor(API) { constructor(API) {
super(API); super(API);
} }
#pattern: ""; #pattern: "";
get pattern() { return this.#pattern; } get pattern() {
return this.#pattern;
}
set pattern(pattern) { set pattern(pattern) {
if(this.API.checkValidFormat(pattern, regex.CMIFeedback)) { if (this.API.checkValidFormat(pattern, regex.CMIFeedback)) {
this.#pattern = pattern; this.#pattern = pattern;
} }
} }
toJSON = () => { toJSON = () => {
this.jsonString = true; this.jsonString = true;
const result = { const result = {
'pattern': this.pattern, 'pattern': this.pattern,
}; };
delete this.jsonString; delete this.jsonString;
return result; return result;
} }
} }

View File

@@ -5,31 +5,31 @@ export const SECONDS_PER_HOUR = 60 * SECONDS_PER_MINUTE;
export const SECONDS_PER_DAY = 24 * SECONDS_PER_HOUR; export const SECONDS_PER_DAY = 24 * SECONDS_PER_HOUR;
const designations = [ const designations = [
['D', SECONDS_PER_DAY], ['D', SECONDS_PER_DAY],
['H', SECONDS_PER_HOUR], ['H', SECONDS_PER_HOUR],
['M', SECONDS_PER_MINUTE], ['M', SECONDS_PER_MINUTE],
['S', SECONDS_PER_SECOND], ['S', SECONDS_PER_SECOND],
]; ];
/** /**
* Converts a Number to a String of HH:MM:SS * Converts a Number to a String of HH:MM:SS
* *
* @param totalSeconds * @param {Number} totalSeconds
* @returns {string} * @return {string}
*/ */
export function getSecondsAsHHMMSS(totalSeconds: Number) { export function getSecondsAsHHMMSS(totalSeconds: Number) {
// SCORM spec does not deal with negative durations, give zero back // SCORM spec does not deal with negative durations, give zero back
if(!totalSeconds || totalSeconds <= 0) { if (!totalSeconds || totalSeconds <= 0) {
return '00:00:00'; return '00:00:00';
} }
let hours = Math.floor(totalSeconds / SECONDS_PER_HOUR); const hours = Math.floor(totalSeconds / SECONDS_PER_HOUR);
let dateObj = new Date(totalSeconds * 1000); const dateObj = new Date(totalSeconds * 1000);
let minutes = dateObj.getUTCMinutes(); const minutes = dateObj.getUTCMinutes();
let seconds = dateObj.getSeconds(); const seconds = dateObj.getSeconds();
return hours.toString().padStart(2, '0') + ':' + return hours.toString().padStart(2, '0') + ':' +
minutes.toString().padStart(2, '0') + ':' + minutes.toString().padStart(2, '0') + ':' +
seconds.toString().padStart(2, '0'); seconds.toString().padStart(2, '0');
} }
@@ -37,75 +37,77 @@ export function getSecondsAsHHMMSS(totalSeconds: Number) {
/** /**
* Calculate the number of seconds from ISO 8601 Duration * Calculate the number of seconds from ISO 8601 Duration
* *
* @param seconds * @param {Number} seconds
* @returns {String} * @return {String}
*/ */
export function getSecondsAsISODuration(seconds: Number) { export function getSecondsAsISODuration(seconds: Number) {
// SCORM spec does not deal with negative durations, give zero back // SCORM spec does not deal with negative durations, give zero back
if(!seconds || seconds <= 0) { if (!seconds || seconds <= 0) {
return 'P0S'; return 'P0S';
}
let duration = 'P';
let remainder = seconds;
designations.forEach(([sign, current_seconds]) => {
const value = Math.floor(remainder / current_seconds);
remainder = remainder % current_seconds;
if (value) {
duration += `${value}${sign}`;
} }
});
let duration = 'P'; return duration;
let remainder = seconds;
designations.forEach(([sign, current_seconds]) => {
const value = Math.floor(remainder / current_seconds);
remainder = remainder % current_seconds;
if (value) {
duration += `${value}${sign}`;
}
});
return duration;
} }
/** /**
* Calculate the number of seconds from HH:MM:SS.DDDDDD * Calculate the number of seconds from HH:MM:SS.DDDDDD
* *
* @param timeString * @param {string} timeString
* @param timeRegex * @param {RegExp} timeRegex
* @returns {number} * @return {number}
*/ */
export function getTimeAsSeconds(timeString: String, timeRegex: RegExp) { export function getTimeAsSeconds(timeString: String, timeRegex: RegExp) {
if(!timeString || typeof timeString !== 'string' || !timeString.match(timeRegex)) { if (!timeString || typeof timeString !== 'string' ||
return 0; !timeString.match(timeRegex)) {
} return 0;
const parts = timeString.split(':'); }
const hours = Number(parts[0]); const parts = timeString.split(':');
const minutes = Number(parts[1]); const hours = Number(parts[0]);
const seconds = Number(parts[2]); const minutes = Number(parts[1]);
return (hours * 3600) + (minutes * 60) + seconds; const seconds = Number(parts[2]);
return (hours * 3600) + (minutes * 60) + seconds;
} }
/** /**
* Calculate the number of seconds from ISO 8601 Duration * Calculate the number of seconds from ISO 8601 Duration
* *
* @param duration * @param {string} duration
* @param durationRegex * @param {RegExp} durationRegex
* @returns {number} * @return {number}
*/ */
export function getDurationAsSeconds(duration: String, durationRegex: RegExp) { export function getDurationAsSeconds(duration: String, durationRegex: RegExp) {
if(!duration || !duration.match(durationRegex)) { if (!duration || !duration.match(durationRegex)) {
return 0; return 0;
} }
let [,years,months,,days,hours,minutes,seconds] = new RegExp(durationRegex).exec(duration) || []; const [, years, months, , days, hours, minutes, seconds] = new RegExp(
durationRegex).exec(duration) || [];
let now = new Date(); const now = new Date();
let anchor = new Date(now); const anchor = new Date(now);
anchor.setFullYear(anchor.getFullYear() + Number(years || 0)); anchor.setFullYear(anchor.getFullYear() + Number(years || 0));
anchor.setMonth(anchor.getMonth() + Number(months || 0)); anchor.setMonth(anchor.getMonth() + Number(months || 0));
anchor.setDate(anchor.getDate() + Number(days || 0)); anchor.setDate(anchor.getDate() + Number(days || 0));
anchor.setHours(anchor.getHours() + Number(hours || 0)); anchor.setHours(anchor.getHours() + Number(hours || 0));
anchor.setMinutes(anchor.getMinutes() + Number(minutes || 0)); anchor.setMinutes(anchor.getMinutes() + Number(minutes || 0));
anchor.setSeconds(anchor.getSeconds() + Number(seconds || 0)); anchor.setSeconds(anchor.getSeconds() + Number(seconds || 0));
if(seconds) { if (seconds) {
let milliseconds = Number(Number(seconds) % 1).toFixed(6) * 1000.0; const milliseconds = Number(Number(seconds) % 1).toFixed(6) * 1000.0;
anchor.setMilliseconds(anchor.getMilliseconds() + milliseconds); anchor.setMilliseconds(anchor.getMilliseconds() + milliseconds);
} }
return (anchor - now) / 1000.0; return (anchor - now) / 1000.0;
} }