ÿØÿà 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 . /** * AJAX helper for the inline editing a value. * * This script is automatically included from template core/inplace_editable * It registers a click-listener on [data-inplaceeditablelink] link (the "inplace edit" icon), * then replaces the displayed value with an input field. On "Enter" it sends a request * to web service core_update_inplace_editable, which invokes the specified callback. * Any exception thrown by the web service (or callback) is displayed as an error popup. * * @module core/inplace_editable * @copyright 2016 Marina Glancy * @license http://www.gnu.org/copyleft/gpl.html GNU GPL v3 or later * @since 3.1 */ define( ['jquery', 'core/ajax', 'core/templates', 'core/notification', 'core/str', 'core/config', 'core/url', 'core/form-autocomplete', 'core/pending', 'core/local/inplace_editable/events', ], function($, ajax, templates, notification, str, cfg, url, autocomplete, Pending, Events) { const removeSpinner = function(element) { element.removeClass('updating'); element.find('img.spinner').hide(); }; /** * Update an inplace editable value. * * @param {Jquery} mainelement the element to update * @param {string} value the new value * @param {bool} silent if true the change won't alter the current page focus * @fires event:core/inplace_editable:updated * @fires event:core/inplace_editable:updateFailed */ const updateValue = function(mainelement, value, silent) { var pendingId = [ mainelement.attr('data-itemid'), mainelement.attr('data-component'), mainelement.attr('data-itemtype'), ].join('-'); var pendingPromise = new Pending(pendingId); addSpinner(mainelement); ajax.call([{ methodname: 'core_update_inplace_editable', args: { itemid: mainelement.attr('data-itemid'), component: mainelement.attr('data-component'), itemtype: mainelement.attr('data-itemtype'), value: value, }, }])[0] .then(function(data) { return templates.render('core/inplace_editable', data) .then(function(html, js) { var oldvalue = mainelement.attr('data-value'); var newelement = $(html); templates.replaceNode(mainelement, newelement, js); if (!silent) { newelement.find('[data-inplaceeditablelink]').focus(); } // Trigger updated event on the DOM element. Events.notifyElementUpdated(newelement.get(0), data, oldvalue); return; }); }) .then(function() { return pendingPromise.resolve(); }) .fail(function(ex) { removeSpinner(mainelement); M.util.js_complete(pendingId); // Trigger update failed event on the DOM element. let updateFailedEvent = Events.notifyElementUpdateFailed(mainelement.get(0), ex, value); if (!updateFailedEvent.defaultPrevented) { notification.exception(ex); } }); }; const addSpinner = function(element) { element.addClass('updating'); var spinner = element.find('img.spinner'); if (spinner.length) { spinner.show(); } else { spinner = $('') .attr('src', url.imageUrl('i/loading_small')) .addClass('spinner').addClass('smallicon') ; element.append(spinner); } }; $('body').on('click keypress', '[data-inplaceeditable] [data-inplaceeditablelink]', function(e) { if (e.type === 'keypress' && e.keyCode !== 13) { return; } var editingEnabledPromise = new Pending('autocomplete-start-editing'); e.stopImmediatePropagation(); e.preventDefault(); var target = $(this), mainelement = target.closest('[data-inplaceeditable]'); var turnEditingOff = function(el) { el.find('input').off(); el.find('select').off(); el.html(el.attr('data-oldcontent')); el.removeAttr('data-oldcontent'); el.removeClass('inplaceeditingon'); el.find('[data-inplaceeditablelink]').focus(); // Re-enable any parent draggable attribute. el.parents(`[data-inplace-in-draggable="true"]`) .attr('draggable', true) .attr('data-inplace-in-draggable', false); }; var turnEditingOffEverywhere = function() { // Re-enable any disabled draggable attribute. $(`[data-inplace-in-draggable="true"]`) .attr('draggable', true) .attr('data-inplace-in-draggable', false); $('span.inplaceeditable.inplaceeditingon').each(function() { turnEditingOff($(this)); }); }; var uniqueId = function(prefix, idlength) { var uniqid = prefix, i; for (i = 0; i < idlength; i++) { uniqid += String(Math.floor(Math.random() * 10)); } // Make sure this ID is not already taken by an existing element. if ($("#" + uniqid).length === 0) { return uniqid; } return uniqueId(prefix, idlength); }; var turnEditingOnText = function(el) { str.get_string('edittitleinstructions').done(function(s) { var instr = $('' + s + ''). attr('id', uniqueId('id_editinstructions_', 20)), inputelement = $(''). attr('id', uniqueId('id_inplacevalue_', 20)). attr('value', el.attr('data-value')). attr('aria-describedby', instr.attr('id')). addClass('ignoredirty'). addClass('form-control'), lbl = $(''). attr('for', inputelement.attr('id')); el.html('').append(instr).append(lbl).append(inputelement); inputelement.focus(); inputelement.select(); inputelement.on('keyup keypress focusout', function(e) { if (cfg.behatsiterunning && e.type === 'focusout') { // Behat triggers focusout too often. return; } if (e.type === 'keypress' && e.keyCode === 13) { // We need 'keypress' event for Enter because keyup/keydown would catch Enter that was // pressed in other fields. var val = inputelement.val(); turnEditingOff(el); updateValue(el, val); } if ((e.type === 'keyup' && e.keyCode === 27) || e.type === 'focusout') { // We need 'keyup' event for Escape because keypress does not work with Escape. turnEditingOff(el); } }); }); }; var turnEditingOnToggle = function(el, newvalue) { turnEditingOff(el); updateValue(el, newvalue); }; var turnEditingOnSelect = function(el, options) { var i, inputelement = $(''). attr('id', uniqueId('id_inplacevalue_', 20)). addClass('custom-select'), lbl = $('') .attr('for', inputelement.attr('id')); for (i in options) { inputelement .append($('