AutorÃa | Ultima modificación | Ver Log |
// 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 <http://www.gnu.org/licenses/>.
/*
* @package atto_h5p
* @copyright 2019 Bas Brands <bas@moodle.com>
* @license http://www.gnu.org/copyleft/gpl.html GNU GPL v3 or later
*/
/**
* @module moodle-atto_h5p-button
*/
/**
* Atto h5p content tool.
*
* @namespace M.atto_h5p
* @class Button
* @extends M.editor_atto.EditorPlugin
*/
var CSS = {
CONTENTWARNING: 'att_h5p_contentwarning',
H5PBROWSER: 'openh5pbrowser',
INPUTALT: 'atto_h5p_altentry',
INPUTH5PFILE: 'atto_h5p_file',
INPUTSUBMIT: 'atto_h5p_urlentrysubmit',
OPTION_DOWNLOAD_BUTTON: 'atto_h5p_option_download_button',
OPTION_COPYRIGHT_BUTTON: 'atto_h5p_option_copyright_button',
OPTION_EMBED_BUTTON: 'atto_h5p_option_embed_button',
URLWARNING: 'atto_h5p_warning'
},
SELECTORS = {
CONTENTWARNING: '.' + CSS.CONTENTWARNING,
H5PBROWSER: '.' + CSS.H5PBROWSER,
INPUTH5PFILE: '.' + CSS.INPUTH5PFILE,
INPUTSUBMIT: '.' + CSS.INPUTSUBMIT,
OPTION_DOWNLOAD_BUTTON: '.' + CSS.OPTION_DOWNLOAD_BUTTON,
OPTION_COPYRIGHT_BUTTON: '.' + CSS.OPTION_COPYRIGHT_BUTTON,
OPTION_EMBED_BUTTON: '.' + CSS.OPTION_EMBED_BUTTON,
URLWARNING: '.' + CSS.URLWARNING
},
COMPONENTNAME = 'atto_h5p',
TEMPLATE = '' +
'<form class="atto_form mform" id="{{elementid}}_atto_h5p_form">' +
'<div style="display:none" role="alert" class="alert alert-warning mb-1 {{CSS.CONTENTWARNING}}">' +
'{{get_string "noh5pcontent" component}}' +
'</div>' +
'<div style="display:none" role="alert" class="alert alert-warning mb-1 {{CSS.URLWARNING}}">' +
'{{get_string "invalidh5purl" component}}' +
'</div>' +
'{{#if canUploadAndEmbed}}' +
'<div class="mt-2 mb-4 attoh5pinstructions">{{{get_string "instructions" component}}}</div>' +
'{{/if}}' +
'<div class="mb-4">' +
'<label for="{{elementid}}_{{CSS.H5PBROWSER}}">' +
'{{#if canUploadAndEmbed}}' +
'{{get_string "h5pfileorurl" component}}' +
'{{/if}}' +
'{{^if canUploadAndEmbed}}' +
'{{#if canUpload}}' +
'{{get_string "h5pfile" component}}' +
'{{/if}}' +
'{{#if canEmbed}}' +
'{{get_string "h5purl" component}}' +
'{{/if}}' +
'{{/if}}' +
'</label>' +
'<div class="input-group input-append w-100">' +
'<input class="form-control {{CSS.INPUTH5PFILE}}" type="url" value="{{fileURL}}" ' +
'id="{{elementid}}_{{CSS.INPUTH5PFILE}}" data-region="h5pfile" size="32"/>' +
'{{#if canUpload}}' +
'<span class="input-group-append">' +
'<button class="btn btn-secondary {{CSS.H5PBROWSER}}" type="button">' +
'{{get_string "browserepositories" component}}</button>' +
'</span>' +
'{{/if}}' +
'</div>' +
'{{#if canUpload}}' +
'<fieldset class="mt-2 collapsible" id="{{elementid}}_h5poptions">' +
'<legend class="d-flex align-items-center px-1">' +
'<div class="position-relative d-flex ftoggler align-items-center position-relative mr-1">' +
'<a role="button" data-toggle="collapse" href="#h5poptions"' +
'aria-expanded="{{#if showOptions}}true{{/if}}{{^if showOptions}}false{{/if}}"' +
'aria-controls="h5poptions"' +
'class="btn btn-icon mr-1 icons-collapse-expand stretched-link fheader collapsed">' +
'<span class="expanded-icon icon-no-margin p-2"' +
'title="{{get_string "collapse" "moodle"}}">' +
'<i class="icon fa fa-chevron-down fa-fw " aria-hidden="true"></i>' +
'</span>' +
'<span class="collapsed-icon icon-no-margin p-2"' +
'title="{{get_string "expand" "moodle"}}">' +
'<span class="dir-rtl-hide">' +
'<i class="icon fa fa-chevron-right fa-fw " aria-hidden="true"></i>' +
'</span>' +
'<span class="dir-ltr-hide">' +
'<i class="icon fa fa-chevron-left fa-fw " aria-hidden="true"></i>' +
'</span>' +
'</span>' +
'<span class="sr-only">{{get_string "h5poptions" component}}</span>' +
'</a>' +
'<h3 class="d-flex align-self-stretch align-items-center mb-0" aria-hidden="true">' +
'{{get_string "h5poptions" component}}' +
'</h3>' +
'</div>' +
'</legend>' +
'<div id="h5poptions" class="fcontainer collapseable collapse px-1 {{#if showOptions}}show{{/if}}">' +
'<div class="form-check">' +
'<input type="checkbox" {{optionDownloadButton}} ' +
'class="form-check-input {{CSS.OPTION_DOWNLOAD_BUTTON}}"' +
'aria-label="{{get_string "downloadbutton" component}}" ' +
'id="{{elementid}}_h5p-option-allow-download"/>' +
'<label class="form-check-label" for="{{elementid}}_h5p-option-allow-download">' +
'{{get_string "downloadbutton" component}}' +
'</label>' +
'</div>' +
'<div class="form-check">' +
'<input type="checkbox" {{optionEmbedButton}} ' +
'class="form-check-input {{CSS.OPTION_EMBED_BUTTON}}" ' +
'aria-label="{{get_string "embedbutton" component}}" ' +
'id="{{elementid}}_h5p-option-embed-button"/>' +
'<label class="form-check-label" for="{{elementid}}_h5p-option-embed-button">' +
'{{get_string "embedbutton" component}}' +
'</label>' +
'</div>' +
'<div class="form-check mb-2">' +
'<input type="checkbox" {{optionCopyrightButton}} ' +
'class="form-check-input {{CSS.OPTION_COPYRIGHT_BUTTON}}" ' +
'aria-label="{{get_string "copyrightbutton" component}}" ' +
'id="{{elementid}}_h5p-option-copyright-button"/>' +
'<label class="form-check-label" for="{{elementid}}_h5p-option-copyright-button">' +
'{{get_string "copyrightbutton" component}}' +
'</label>' +
'</div>' +
'</div>' +
'</fieldset>' +
'{{/if}}' +
'</div>' +
'<div class="text-center">' +
'<button class="btn btn-secondary {{CSS.INPUTSUBMIT}}" type="submit">' + '' +
'{{get_string "pluginname" component}}</button>' +
'</div>' +
'</form>',
H5PTEMPLATE = '' +
'{{#if addParagraphs}}<p><br></p>{{/if}}' +
'<div class="h5p-placeholder" contenteditable="false">' +
'{{{url}}}' +
'</div>' +
'{{#if addParagraphs}}<p><br></p>{{/if}}';
Y.namespace('M.atto_h5p').Button = Y.Base.create('button', Y.M.editor_atto.EditorPlugin, [], {
/**
* A reference to the current selection at the time that the dialogue
* was opened.
*
* @property _currentSelection
* @type Range
* @private
*/
_currentSelection: null,
/**
* A reference to the currently open form.
*
* @param _form
* @type Node
* @private
*/
_form: null,
/**
* A reference to the currently selected H5P div.
*
* @param _form
* @type Node
* @private
*/
_H5PDiv: null,
/**
* Allowed methods of adding H5P.
*
* @param _allowedmethods
* @type String
* @private
*/
_allowedmethods: 'none',
initializer: function() {
this._allowedmethods = this.get('allowedmethods');
if (this._allowedmethods === 'none') {
// Plugin not available here.
return;
}
this.addButton({
icon: 'icon',
iconComponent: 'atto_h5p',
callback: this._displayDialogue,
tags: '.h5p-placeholder',
tagMatchRequiresAll: false
});
this.editor.all('.h5p-placeholder').setAttribute('contenteditable', 'false');
this.editor.delegate('dblclick', this._handleDblClick, '.h5p-placeholder', this);
this.editor.delegate('click', this._handleClick, '.h5p-placeholder', this);
},
/**
* Handle a double click on a H5P Placeholder.
*
* @method _handleDblClick
* @private
*/
_handleDblClick: function() {
this._displayDialogue();
},
/**
* Handle a click on a H5P Placeholder.
*
* @method _handleClick
* @param {EventFacade} e
* @private
*/
_handleClick: function(e) {
var selection = this.get('host').getSelectionFromNode(e.target);
if (this.get('host').getSelection() !== selection) {
this.get('host').setSelection(selection);
}
},
/**
* Display the h5p editing tool.
*
* @method _displayDialogue
* @private
*/
_displayDialogue: function() {
// Store the current selection.
this._currentSelection = this.get('host').getSelection();
if (this._currentSelection === false) {
return;
}
this._getH5PDiv();
var dialogue = this.getDialogue({
headerContent: M.util.get_string('pluginname', COMPONENTNAME),
width: 'auto',
focusAfterHide: true
});
// Set the dialogue content, and then show the dialogue.
dialogue.set('bodyContent', this._getDialogueContent())
.show();
M.form.shortforms({formid: this.get('host').get('elementid') + '_atto_h5p_form'});
},
/**
* Get the H5P iframe
*
* @method _resolveH5P
* @return {Node} The H5P iframe selected.
* @private
*/
_getH5PDiv: function() {
var selectednodes = this.get('host').getSelectedNodes();
var H5PDiv = null;
selectednodes.each(function(selNode) {
if (selNode.hasClass('h5p-placeholder')) {
H5PDiv = selNode;
}
});
this._H5PDiv = H5PDiv;
},
/**
* Get the H5P button permissions.
*
* @return {Object} H5P button permissions.
* @private
*/
_getPermissions: function() {
var permissions = {
'canEmbed': false,
'canUpload': false,
'canUploadAndEmbed': false
};
if (this.get('host').canShowFilepicker('h5p')) {
if (this._allowedmethods === 'both') {
permissions.canUploadAndEmbed = true;
permissions.canUpload = true;
} else if (this._allowedmethods === 'upload') {
permissions.canUpload = true;
}
}
if (this._allowedmethods === 'both' || this._allowedmethods === 'embed') {
permissions.canEmbed = true;
}
return permissions;
},
/**
* Return the dialogue content for the tool, attaching any required
* events.
*
* @method _getDialogueContent
* @return {Node} The content to place in the dialogue.
* @private
*/
_getDialogueContent: function() {
var permissions = this._getPermissions();
var fileURL,
optionDownloadButton,
optionEmbedButton,
optionCopyrightButton,
showOptions = false;
if (this._H5PDiv) {
var H5PURL = this._H5PDiv.get('innerHTML');
var fileBaseUrl = M.cfg.wwwroot + '/draftfile.php';
if (fileBaseUrl == H5PURL.substring(0, fileBaseUrl.length)) {
fileURL = H5PURL.split("?")[0];
var parameters = H5PURL.split("?")[1];
if (parameters) {
if (parameters.match(/export=1/)) {
optionDownloadButton = 'checked';
showOptions = true;
}
if (parameters.match(/embed=1/)) {
optionEmbedButton = 'checked';
showOptions = true;
}
if (parameters.match(/copyright=1/)) {
optionCopyrightButton = 'checked';
showOptions = true;
}
}
} else {
fileURL = H5PURL;
}
}
var template = Y.Handlebars.compile(TEMPLATE),
content = Y.Node.create(template({
elementid: this.get('host').get('elementid'),
CSS: CSS,
component: COMPONENTNAME,
canUpload: permissions.canUpload,
canEmbed: permissions.canEmbed,
canUploadAndEmbed: permissions.canUploadAndEmbed,
showOptions: showOptions,
fileURL: fileURL,
optionDownloadButton: optionDownloadButton,
optionEmbedButton: optionEmbedButton,
optionCopyrightButton: optionCopyrightButton
}));
this._form = content;
// Listen to and act on Dialogue content events.
this._setEventListeners();
return content;
},
/**
* Update the dialogue after an h5p was selected in the File Picker.
*
* @method _filepickerCallback
* @param {object} params The parameters provided by the filepicker
* containing information about the h5p.
* @private
*/
_filepickerCallback: function(params) {
if (params.url !== '') {
var input = this._form.one(SELECTORS.INPUTH5PFILE);
input.set('value', params.url);
this._removeWarnings();
}
},
/**
* Set event Listeners for Dialogue content actions.
*
* @method _setEventListeners
* @private
*/
_setEventListeners: function() {
var form = this._form;
var permissions = this._getPermissions();
form.one(SELECTORS.INPUTSUBMIT).on('click', this._setH5P, this);
if (permissions.canUpload) {
form.one(SELECTORS.H5PBROWSER).on('click', function() {
this.get('host').showFilepicker('h5p', this._filepickerCallback, this);
}, this);
}
if (permissions.canUploadAndEmbed) {
form.one(SELECTORS.INPUTH5PFILE).on('change', function() {
this._removeWarnings();
}, this);
}
},
/**
* Remove warnings shown in the dialogue.
*
* @method _removeWarnings
* @private
*/
_removeWarnings: function() {
var form = this._form;
form.one(SELECTORS.URLWARNING).setStyle('display', 'none');
form.one(SELECTORS.CONTENTWARNING).setStyle('display', 'none');
},
/**
* Update the h5p in the contenteditable.
*
* @method _setH5P
* @param {EventFacade} e
* @private
*/
_setH5P: function(e) {
var form = this._form,
h5phtml,
host = this.get('host'),
h5pfile = form.one(SELECTORS.INPUTH5PFILE).get('value'),
permissions = this._getPermissions();
e.preventDefault();
// Check if there are any issues.
if (this._updateWarning()) {
return;
}
// Focus on the editor in preparation for inserting the H5P.
host.focus();
// Add an empty paragraph after new H5P container that can catch the cursor.
var addParagraphs = true;
// If a H5P placeholder was selected we can destroy it now.
if (this._H5PDiv) {
this._H5PDiv.remove();
addParagraphs = false;
}
if (h5pfile !== '') {
host.setSelection(this._currentSelection);
if (h5pfile.startsWith(M.cfg.wwwroot)) {
// It's a local file.
var params = '';
if (permissions.canUpload) {
var options = {};
if (form.one(SELECTORS.OPTION_DOWNLOAD_BUTTON).get('checked')) {
options['export'] = '1';
}
if (form.one(SELECTORS.OPTION_EMBED_BUTTON).get('checked')) {
options.embed = '1';
}
if (form.one(SELECTORS.OPTION_COPYRIGHT_BUTTON).get('checked')) {
options.copyright = '1';
}
for (var opt in options) {
if (params === "" && (h5pfile.indexOf("?") === -1)) {
params += "?";
} else {
params += "&";
}
params += opt + "=" + options[opt];
}
}
var h5ptemplate = Y.Handlebars.compile(H5PTEMPLATE);
h5phtml = h5ptemplate({
url: h5pfile + params,
addParagraphs: addParagraphs
});
} else {
// It's a URL.
var urltemplate = Y.Handlebars.compile(H5PTEMPLATE);
h5phtml = urltemplate({
url: h5pfile
});
}
host.insertContentAtFocusPoint(h5phtml);
this.markUpdated();
}
this.getDialogue({
focusAfterHide: null
}).hide();
},
/**
* Check if this could be a h5p embed.
*
* @method _validEmbed
* @param {String} str
* @return {boolean} whether this is a iframe tag.
* @private
*/
_validEmbed: function(str) {
var pattern = new RegExp('^(<iframe).*(<\\/iframe>)'); // Port and path.
return !!pattern.test(str);
},
/**
* Check if this could be a h5p URL.
*
* @method _validURL
* @param {String} str
* @return {boolean} whether this is a valid URL.
* @private
*/
_validURL: function(str) {
var pattern = new RegExp('^(https?:\\/\\/)?' + // Protocol.
'((([a-z\\d]([a-z\\d-]*[a-z\\d])*)\\.)+[a-z]{2,}|' + // Domain name.
'((\\d{1,3}\\.){3}\\d{1,3}))' + // OR ip (v4) address.
'(\\:\\d+)?(\\/[-a-z\\d%_.~+]*)*'); // Port and path.
return !!pattern.test(str);
},
/**
* Update the url warning.
*
* @method _updateWarning
* @return {boolean} whether a warning should be displayed.
* @private
*/
_updateWarning: function() {
var form = this._form,
state = true,
h5pfile,
permissions = this._getPermissions();
if (permissions.canUpload || permissions.canEmbed) {
h5pfile = form.one(SELECTORS.INPUTH5PFILE).get('value');
if (h5pfile !== '') {
form.one(SELECTORS.CONTENTWARNING).setStyle('display', 'none');
if (h5pfile.startsWith(M.cfg.wwwroot) || this._validURL(h5pfile)) {
// Only external URLs have to be validated.
form.one(SELECTORS.URLWARNING).setStyle('display', 'none');
state = false;
} else {
form.one(SELECTORS.URLWARNING).setStyle('display', 'block');
state = true;
}
} else {
form.one(SELECTORS.CONTENTWARNING).setStyle('display', 'block');
state = true;
}
}
return state;
}
}, {
ATTRS: {
/**
* The allowedmethods of adding h5p content.
*
* @attribute allowedmethods
* @type String
*/
allowedmethods: {
value: null
}
}
});