ÿØÿà JFIF    ÿÛ „  ( %"1!%)+...383,7(-.+  -+++--++++---+-+-----+---------------+---+-++7-----ÿÀ  ß â" ÿÄ     ÿÄ H    !1AQaq"‘¡2B±ÁÑð#R“Ò Tbr‚²á3csƒ’ÂñDS¢³$CÿÄ   ÿÄ %  !1AQa"23‘ÿÚ   ? ôÿ ¨pŸªáÿ —åYõõ\?àÒü©ŠÄï¨pŸªáÿ —åYõõ\?àÓü©ŠÄá 0Ÿªáÿ Ÿå[úƒ ú®ði~TÁbqÐ8OÕpÿ ƒOò¤Oè`–RÂáœá™êi€ßÉ< FtŸI“öÌ8úDf´°å}“¾œ6  öFá°y¥jñÇh†ˆ¢ã/ÃÐ:ªcÈ "Y¡ðÑl>ÿ ”ÏËte:qž\oäŠe÷󲍷˜HT4&ÿ ÓÐü6ö®¿øþßèô Ÿ•7Ñi’•j|“ñì>b…þS?*Óôÿ ÓÐü*h¥£ír¶ü UãS炟[AÐaè[ûª•õ&õj?†Éö+EzP—WeÒírJFt ‘BŒ†Ï‡%#tE Øz ¥OÛ«!1›üä±Í™%ºÍãö]°î(–:@<‹ŒÊö×òÆt¦ãº+‡¦%ÌÁ²h´OƒJŒtMÜ>ÀÜÊw3Y´•牋4ǍýʏTì>œú=Íwhyë,¾Ôò×õ¿ßÊa»«þˆѪQ|%6ž™A õ%:øj<>É—ÿ Å_ˆCbõ¥š±ý¯Ýƒï…¶|RëócÍf溪“t.СøTÿ *Ä¿-{†çàczůŽ_–^XþŒ±miB[X±d 1,é”zEù»& î9gœf™9Ð'.;—™i}!ôšåîqêÛ٤ёý£½ÆA–àôe"A$˝Úsäÿ ÷Û #°xŸëí(l »ý3—¥5m! rt`†0~'j2(]S¦¦kv,ÚÇ l¦øJA£Šƒ J3E8ÙiŽ:cÉžúeZ°€¯\®kÖ(79«Ž:¯X”¾³Š&¡* ….‰Ž(ÜíŸ2¥ª‡×Hi²TF¤ò[¨íÈRëÉ䢍mgÑ.Ÿ<öäS0í„ǹÁU´f#Vß;Õ–…P@3ío<ä-±»Ž.L|kªÀê›fÂ6@»eu‚|ÓaÞÆŸ…¨ááå>åŠ?cKü6ùTÍÆ”†sĤÚ;H2RÚ†õ\Ö·Ÿn'¾ ñ#ºI¤Å´%çÁ­‚â7›‹qT3Iï¨ÖÚ5I7Ë!ÅOóŸ¶øÝñØôת¦$Tcö‘[«Ö³šÒ';Aþ ¸èíg A2Z"i¸vdÄ÷.iõ®§)¿]¤À†–‡É&ä{V¶iŽ”.Ó×Õÿ û?h¬Mt–íª[ÿ Ñÿ ÌV(í}=ibÔ¡›¥¢±b Lô¥‡piη_Z<‡z§èŒ)iÖwiÇ 2hÙ3·=’d÷8éŽ1¦¸c¤µ€7›7Ø ð\á)} ¹fËí›pAÃL%âc2 í§æQz¿;T8sæ°qø)QFMð‰XŒÂ±N¢aF¨…8¯!U  Z©RÊ ÖPVÄÀÍin™Ì-GˆªÅËŠ›•zË}º±ŽÍFò¹}Uw×#ä5B¤{î}Ð<ÙD é©¤&‡ïDbàÁôMÁ.// This file is part of Moodle - http://moodle.org/ // // Moodle is free software: you can redistribute it and/or modify // it under the terms of the GNU General Public License as published by // the Free Software Foundation, either version 3 of the License, or // (at your option) any later version. // // Moodle is distributed in the hope that it will be useful, // but WITHOUT ANY WARRANTY; without even the implied warranty of // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the // GNU General Public License for more details. // // You should have received a copy of the GNU General Public License // along with Moodle. If not, see . /* global H5PEmbedCommunicator:true */ /** * When embedded the communicator helps talk to the parent page. * This is a copy of the H5P.communicator, which we need to communicate in this context * * @type {H5PEmbedCommunicator} * @module core_h5p * @copyright 2019 Joubel AS * @license http://www.gnu.org/copyleft/gpl.html GNU GPL v3 or later */ H5PEmbedCommunicator = (function() { /** * @class * @private */ function Communicator() { var self = this; // Maps actions to functions. var actionHandlers = {}; // Register message listener. window.addEventListener('message', function receiveMessage(event) { if (window.parent !== event.source || event.data.context !== 'h5p') { return; // Only handle messages from parent and in the correct context. } if (actionHandlers[event.data.action] !== undefined) { actionHandlers[event.data.action](event.data); } }, false); /** * Register action listener. * * @param {string} action What you are waiting for * @param {function} handler What you want done */ self.on = function(action, handler) { actionHandlers[action] = handler; }; /** * Send a message to the all mighty father. * * @param {string} action * @param {Object} [data] payload */ self.send = function(action, data) { if (data === undefined) { data = {}; } data.context = 'h5p'; data.action = action; // Parent origin can be anything. window.parent.postMessage(data, '*'); }; /* eslint-disable promise/avoid-new */ const repositoryPromise = new Promise((resolve) => { require(['core_h5p/repository'], (Repository) => { // Replace the default versions. self.post = Repository.postStatement; self.postState = Repository.postState; self.deleteState = Repository.deleteState; // Resolve the Promise with Repository to allow any queued calls to be executed. resolve(Repository); }); }); /** * Send a xAPI statement to LMS. * * @param {string} component * @param {Object} statements * @returns {Promise} */ self.post = (component, statements) => repositoryPromise.then((Repository) => Repository.postStatement( component, statements, )); /** * Send a xAPI state to LMS. * * @param {string} component * @param {string} activityId * @param {Object} agent * @param {string} stateId * @param {string} stateData * @returns {void} */ self.postState = ( component, activityId, agent, stateId, stateData, ) => repositoryPromise.then((Repository) => Repository.postState( component, activityId, agent, stateId, stateData, )); /** * Delete a xAPI state from LMS. * * @param {string} component * @param {string} activityId * @param {Object} agent * @param {string} stateId * @returns {Promise} */ self.deleteState = (component, activityId, agent, stateId) => repositoryPromise.then((Repository) => Repository.deleteState( component, activityId, agent, stateId, )); } return (window.postMessage && window.addEventListener ? new Communicator() : undefined); })(); var getH5PObject = async (iFrame) => { var H5P = iFrame.contentWindow.H5P; if (H5P?.instances?.[0]) { return H5P; } // In some cases, the H5P takes a while to be initialized (which causes some random behat failures). const sleep = (delay) => new Promise((resolve) => setTimeout(resolve, delay)); let remainingAttemps = 10; while (!H5P?.instances?.[0] && remainingAttemps > 0) { await sleep(100); H5P = iFrame.contentWindow.H5P; remainingAttemps--; } return H5P; }; document.onreadystatechange = async() => { // Wait for instances to be initialize. if (document.readyState !== 'complete') { return; } /** @var {boolean} statementPosted Whether the statement has been sent or not, to avoid sending xAPI State after it. */ var statementPosted = false; // Check for H5P iFrame. var iFrame = document.querySelector('.h5p-iframe'); if (!iFrame || !iFrame.contentWindow) { return; } var H5P = await getH5PObject(iFrame); if (!H5P?.instances?.[0]) { return; } var resizeDelay; var instance = H5P.instances[0]; var parentIsFriendly = false; // Handle that the resizer is loaded after the iframe. H5PEmbedCommunicator.on('ready', function() { H5PEmbedCommunicator.send('hello'); }); // Handle hello message from our parent window. H5PEmbedCommunicator.on('hello', function() { // Initial setup/handshake is done. parentIsFriendly = true; // Hide scrollbars for correct size. iFrame.contentDocument.body.style.overflow = 'hidden'; document.body.classList.add('h5p-resizing'); // Content need to be resized to fit the new iframe size. H5P.trigger(instance, 'resize'); }); // When resize has been prepared tell parent window to resize. H5PEmbedCommunicator.on('resizePrepared', function() { H5PEmbedCommunicator.send('resize', { scrollHeight: iFrame.contentDocument.body.scrollHeight }); }); H5PEmbedCommunicator.on('resize', function() { H5P.trigger(instance, 'resize'); }); H5P.on(instance, 'resize', function() { if (H5P.isFullscreen) { return; // Skip iframe resize. } // Use a delay to make sure iframe is resized to the correct size. clearTimeout(resizeDelay); resizeDelay = setTimeout(function() { // Only resize if the iframe can be resized. if (parentIsFriendly) { H5PEmbedCommunicator.send('prepareResize', { scrollHeight: iFrame.contentDocument.body.scrollHeight, clientHeight: iFrame.contentDocument.body.clientHeight } ); } else { H5PEmbedCommunicator.send('hello'); } }, 0); }); // Get emitted xAPI data. H5P.externalDispatcher.on('xAPI', function(event) { statementPosted = false; var moodlecomponent = H5P.getMoodleComponent(); if (moodlecomponent == undefined) { return; } // Skip malformed events. var hasStatement = event && event.data && event.data.statement; if (!hasStatement) { return; } var statement = event.data.statement; var validVerb = statement.verb && statement.verb.id; if (!validVerb) { return; } var isCompleted = statement.verb.id === 'http://adlnet.gov/expapi/verbs/answered' || statement.verb.id === 'http://adlnet.gov/expapi/verbs/completed'; var isChild = statement.context && statement.context.contextActivities && statement.context.contextActivities.parent && statement.context.contextActivities.parent[0] && statement.context.contextActivities.parent[0].id; if (isCompleted && !isChild) { var statements = H5P.getXAPIStatements(this.contentId, statement); H5PEmbedCommunicator.post(moodlecomponent, statements); // Mark the statement has been sent, to avoid sending xAPI State after it. statementPosted = true; } }); H5P.externalDispatcher.on('xAPIState', function(event) { var moodlecomponent = H5P.getMoodleComponent(); var contentId = event.data.activityId; var stateId = event.data.stateId; var state = event.data.state; if (state === undefined) { // When state is undefined, a call to the WS for getting the state could be done. However, for now, this is not // required because the content state is initialised with PHP. return; } if (state === null) { // When this method is called from the H5P API with null state, the state must be deleted using the rest of attributes. H5PEmbedCommunicator.deleteState(moodlecomponent, contentId, H5P.getxAPIActor(), stateId); } else if (!statementPosted) { // Only update the state if a statement hasn't been posted recently. // When state is defined, it needs to be updated. As not all the H5P content types are returning a JSON, we need // to simulate it because xAPI State defines statedata as a JSON. var statedata = { h5p: state }; H5PEmbedCommunicator.postState(moodlecomponent, contentId, H5P.getxAPIActor(), stateId, JSON.stringify(statedata)); } }); // Trigger initial resize for instance. H5P.trigger(instance, 'resize'); };