599 lines
14 KiB
JavaScript
599 lines
14 KiB
JavaScript
|
/**
|
|||
|
* Manage downloads
|
|||
|
*/
|
|||
|
|
|||
|
(function() {
|
|||
|
|
|||
|
var cache = {};
|
|||
|
var form = $('form');
|
|||
|
var minified = true;
|
|||
|
|
|||
|
var dependencies = {};
|
|||
|
|
|||
|
var treeURL = 'https://api.github.com/repos/PrismJS/prism/git/trees/gh-pages?recursive=1';
|
|||
|
var treePromise = new Promise(function(resolve) {
|
|||
|
$u.xhr({
|
|||
|
url: treeURL,
|
|||
|
callback: function(xhr) {
|
|||
|
if (xhr.status < 400) {
|
|||
|
resolve(JSON.parse(xhr.responseText).tree);
|
|||
|
}
|
|||
|
}
|
|||
|
});
|
|||
|
});
|
|||
|
|
|||
|
var hstr = window.location.hash.match(/(?:languages|plugins)=[-+\w]+|themes=[-\w]+/g);
|
|||
|
if (hstr) {
|
|||
|
hstr.forEach(function(str) {
|
|||
|
var kv = str.split('=', 2),
|
|||
|
category = kv[0],
|
|||
|
ids = kv[1].split('+');
|
|||
|
if (category !== 'meta' && category !== 'core' && components[category]) {
|
|||
|
for (var id in components[category]) {
|
|||
|
if (components[category][id].option) {
|
|||
|
delete components[category][id].option;
|
|||
|
}
|
|||
|
}
|
|||
|
if (category === 'themes' && ids.length) {
|
|||
|
var themeInput = $('#theme input[value="' + ids[0] + '"]');
|
|||
|
if (themeInput) {
|
|||
|
themeInput.checked = true;
|
|||
|
}
|
|||
|
setTheme(ids[0]);
|
|||
|
}
|
|||
|
var makeDefault = function (id) {
|
|||
|
if (id !== 'meta') {
|
|||
|
if (components[category][id]) {
|
|||
|
if (components[category][id].option !== 'default') {
|
|||
|
if (typeof components[category][id] === 'string') {
|
|||
|
components[category][id] = { title: components[category][id] }
|
|||
|
}
|
|||
|
components[category][id].option = 'default';
|
|||
|
}
|
|||
|
if (components[category][id].require) {
|
|||
|
var deps = components[category][id].require;
|
|||
|
if ($u.type(deps) !== 'array') {
|
|||
|
deps = [deps];
|
|||
|
}
|
|||
|
deps.forEach(makeDefault);
|
|||
|
}
|
|||
|
}
|
|||
|
}
|
|||
|
};
|
|||
|
ids.forEach(makeDefault);
|
|||
|
}
|
|||
|
});
|
|||
|
}
|
|||
|
|
|||
|
// Stay compatible with old querystring feature
|
|||
|
var qstr = window.location.search.match(/(?:languages|plugins)=[-+\w]+|themes=[-\w]+/g);
|
|||
|
if (qstr && !hstr) {
|
|||
|
window.location.hash = window.location.search.replace(/^\?/, '');
|
|||
|
window.location.search = '';
|
|||
|
}
|
|||
|
|
|||
|
var storedTheme = localStorage.getItem('theme');
|
|||
|
|
|||
|
for (var category in components) {
|
|||
|
var all = components[category];
|
|||
|
|
|||
|
all.meta.section = $u.element.create('section', {
|
|||
|
className: 'options',
|
|||
|
id: 'category-' + category,
|
|||
|
contents: {
|
|||
|
tag: 'h1',
|
|||
|
contents: category.charAt(0).toUpperCase() + category.slice(1)
|
|||
|
},
|
|||
|
inside: '#components'
|
|||
|
});
|
|||
|
|
|||
|
if (all.meta.addCheckAll) {
|
|||
|
$u.element.create('label', {
|
|||
|
attributes: {
|
|||
|
'data-id': 'check-all-' + category
|
|||
|
},
|
|||
|
contents: [
|
|||
|
{
|
|||
|
tag: 'input',
|
|||
|
properties: {
|
|||
|
type: 'checkbox',
|
|||
|
name: 'check-all-' + category,
|
|||
|
value: '',
|
|||
|
checked: false,
|
|||
|
onclick: (function(category, all){
|
|||
|
return function () {
|
|||
|
var checkAll = this;
|
|||
|
$$('input[name="download-' + category + '"]').forEach(function(input) {
|
|||
|
all[input.value].enabled = input.checked = checkAll.checked;
|
|||
|
});
|
|||
|
|
|||
|
update(category);
|
|||
|
};
|
|||
|
})(category, all)
|
|||
|
}
|
|||
|
},
|
|||
|
'Select/unselect all'
|
|||
|
],
|
|||
|
inside: all.meta.section
|
|||
|
});
|
|||
|
}
|
|||
|
|
|||
|
for (var id in all) {
|
|||
|
if(id === 'meta') {
|
|||
|
continue;
|
|||
|
}
|
|||
|
|
|||
|
var checked = false, disabled = false;
|
|||
|
var option = all[id].option || all.meta.option;
|
|||
|
|
|||
|
switch (option) {
|
|||
|
case 'mandatory': disabled = true; // fallthrough
|
|||
|
case 'default': checked = true;
|
|||
|
}
|
|||
|
if (category === 'themes' && storedTheme) {
|
|||
|
checked = id === storedTheme;
|
|||
|
}
|
|||
|
|
|||
|
var filepath = all.meta.path.replace(/\{id}/g, id);
|
|||
|
|
|||
|
var info = all[id] = {
|
|||
|
title: all[id].title || all[id],
|
|||
|
aliasTitles: all[id].aliasTitles,
|
|||
|
noCSS: all[id].noCSS || all.meta.noCSS,
|
|||
|
noJS: all[id].noJS || all.meta.noJS,
|
|||
|
enabled: checked,
|
|||
|
require: $u.type(all[id].require) === 'string' ? [all[id].require] : all[id].require,
|
|||
|
after: $u.type(all[id].after) === 'string' ? [all[id].after] : all[id].after,
|
|||
|
peerDependencies: $u.type(all[id].peerDependencies) === 'string' ? [all[id].peerDependencies] : all[id].peerDependencies,
|
|||
|
owner: all[id].owner,
|
|||
|
files: {
|
|||
|
minified: {
|
|||
|
paths: [],
|
|||
|
size: 0
|
|||
|
},
|
|||
|
dev: {
|
|||
|
paths: [],
|
|||
|
size: 0
|
|||
|
}
|
|||
|
}
|
|||
|
};
|
|||
|
|
|||
|
if (info.require) {
|
|||
|
info.require.forEach(function (v) {
|
|||
|
dependencies[v] = (dependencies[v] || []).concat(id);
|
|||
|
});
|
|||
|
}
|
|||
|
|
|||
|
if (!all[id].noJS && !/\.css$/.test(filepath)) {
|
|||
|
info.files.minified.paths.push(filepath.replace(/(\.js)?$/, '.min.js'));
|
|||
|
info.files.dev.paths.push(filepath.replace(/(\.js)?$/, '.js'));
|
|||
|
}
|
|||
|
|
|||
|
|
|||
|
if ((!all[id].noCSS && !/\.js$/.test(filepath)) || /\.css$/.test(filepath)) {
|
|||
|
var cssFile = filepath.replace(/(\.css)?$/, '.css');
|
|||
|
|
|||
|
info.files.minified.paths.push(cssFile);
|
|||
|
info.files.dev.paths.push(cssFile);
|
|||
|
}
|
|||
|
|
|||
|
function getLanguageTitle(lang) {
|
|||
|
if (!lang.aliasTitles)
|
|||
|
return lang.title;
|
|||
|
|
|||
|
var titles = [lang.title];
|
|||
|
for (var alias in lang.aliasTitles)
|
|||
|
if (lang.aliasTitles.hasOwnProperty(alias))
|
|||
|
titles.push(lang.aliasTitles[alias]);
|
|||
|
return titles.join(" + ");
|
|||
|
}
|
|||
|
|
|||
|
var label = $u.element.create('label', {
|
|||
|
attributes: {
|
|||
|
'data-id': id
|
|||
|
},
|
|||
|
contents: [
|
|||
|
{
|
|||
|
tag: 'input',
|
|||
|
properties: {
|
|||
|
type: all.meta.exclusive? 'radio' : 'checkbox',
|
|||
|
name: 'download-' + category,
|
|||
|
value: id,
|
|||
|
checked: checked,
|
|||
|
disabled: disabled,
|
|||
|
onclick: (function(id, category, all){
|
|||
|
return function () {
|
|||
|
$$('input[name="' + this.name + '"]').forEach(function(input) {
|
|||
|
all[input.value].enabled = input.checked;
|
|||
|
});
|
|||
|
|
|||
|
if (all[id].require && this.checked) {
|
|||
|
all[id].require.forEach(function(v) {
|
|||
|
var input = $('label[data-id="' + v + '"] > input');
|
|||
|
input.checked = true;
|
|||
|
|
|||
|
input.onclick();
|
|||
|
});
|
|||
|
}
|
|||
|
|
|||
|
if (dependencies[id] && !this.checked) { // It’s required by others
|
|||
|
dependencies[id].forEach(function(dependent) {
|
|||
|
var input = $('label[data-id="' + dependent + '"] > input');
|
|||
|
input.checked = false;
|
|||
|
|
|||
|
input.onclick();
|
|||
|
});
|
|||
|
}
|
|||
|
|
|||
|
update(category, id);
|
|||
|
};
|
|||
|
})(id, category, all)
|
|||
|
}
|
|||
|
},
|
|||
|
all.meta.link? {
|
|||
|
tag: 'a',
|
|||
|
properties: {
|
|||
|
href: all.meta.link.replace(/\{id}/g, id),
|
|||
|
className: 'name'
|
|||
|
},
|
|||
|
contents: info.title
|
|||
|
} : {
|
|||
|
tag: 'span',
|
|||
|
properties: {
|
|||
|
className: 'name'
|
|||
|
},
|
|||
|
contents: getLanguageTitle(info)
|
|||
|
},
|
|||
|
' ',
|
|||
|
all[id].owner? {
|
|||
|
tag: 'a',
|
|||
|
properties: {
|
|||
|
href: 'https://github.com/' + all[id].owner,
|
|||
|
className: 'owner',
|
|||
|
target: '_blank'
|
|||
|
},
|
|||
|
contents: all[id].owner
|
|||
|
} : ' ',
|
|||
|
{
|
|||
|
tag: 'strong',
|
|||
|
className: 'filesize'
|
|||
|
}
|
|||
|
],
|
|||
|
inside: all.meta.section
|
|||
|
});
|
|||
|
|
|||
|
// Add click events on main theme selector too.
|
|||
|
(function (label) {
|
|||
|
if (category === 'themes') {
|
|||
|
var themeInput = $('#theme input[value="' + id + '"]');
|
|||
|
var input = $('input', label);
|
|||
|
if (themeInput) {
|
|||
|
var themeInputOnclick = themeInput.onclick;
|
|||
|
themeInput.onclick = function () {
|
|||
|
input.checked = true;
|
|||
|
input.onclick();
|
|||
|
themeInputOnclick && themeInputOnclick.call(themeInput);
|
|||
|
};
|
|||
|
}
|
|||
|
}
|
|||
|
}(label));
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
form.elements.compression[0].onclick =
|
|||
|
form.elements.compression[1].onclick = function() {
|
|||
|
minified = !!+this.value;
|
|||
|
|
|||
|
getFilesSizes();
|
|||
|
};
|
|||
|
|
|||
|
function getFileSize(filepath) {
|
|||
|
return treePromise.then(function(tree) {
|
|||
|
for(var i=0, l=tree.length; i<l; i++) {
|
|||
|
if(tree[i].path === filepath) {
|
|||
|
return tree[i].size;
|
|||
|
}
|
|||
|
}
|
|||
|
});
|
|||
|
}
|
|||
|
|
|||
|
function getFilesSizes() {
|
|||
|
for (var category in components) {
|
|||
|
var all = components[category];
|
|||
|
|
|||
|
for (var id in all) {
|
|||
|
if(id === 'meta') {
|
|||
|
continue;
|
|||
|
}
|
|||
|
|
|||
|
var distro = all[id].files[minified? 'minified' : 'dev'],
|
|||
|
files = distro.paths;
|
|||
|
|
|||
|
files.forEach(function (filepath) {
|
|||
|
var file = cache[filepath] = cache[filepath] || {};
|
|||
|
|
|||
|
if(!file.size) {
|
|||
|
|
|||
|
(function(category, id) {
|
|||
|
getFileSize(filepath).then(function(size) {
|
|||
|
if(size) {
|
|||
|
file.size = size;
|
|||
|
distro.size += file.size;
|
|||
|
|
|||
|
update(category, id);
|
|||
|
}
|
|||
|
});
|
|||
|
}(category, id));
|
|||
|
}
|
|||
|
else {
|
|||
|
update(category, id);
|
|||
|
}
|
|||
|
});
|
|||
|
}
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
getFilesSizes();
|
|||
|
|
|||
|
function getFileContents(filepath) {
|
|||
|
return new Promise(function(resolve, reject) {
|
|||
|
$u.xhr({
|
|||
|
url: filepath,
|
|||
|
callback: function(xhr) {
|
|||
|
if (xhr.status < 400 && xhr.responseText) {
|
|||
|
resolve(xhr.responseText);
|
|||
|
} else {
|
|||
|
reject();
|
|||
|
}
|
|||
|
}
|
|||
|
});
|
|||
|
});
|
|||
|
}
|
|||
|
|
|||
|
function prettySize(size) {
|
|||
|
return Math.round(100 * size / 1024)/100 + 'KB';
|
|||
|
}
|
|||
|
|
|||
|
function update(updatedCategory, updatedId){
|
|||
|
// Update total size
|
|||
|
var total = {js: 0, css: 0}, updated = {js: 0, css: 0};
|
|||
|
|
|||
|
for (var category in components) {
|
|||
|
var all = components[category];
|
|||
|
var allChecked = true;
|
|||
|
|
|||
|
for (var id in all) {
|
|||
|
var info = all[id];
|
|||
|
|
|||
|
if (info.enabled || id == updatedId) {
|
|||
|
var distro = info.files[minified? 'minified' : 'dev'];
|
|||
|
|
|||
|
distro.paths.forEach(function(path) {
|
|||
|
if (cache[path]) {
|
|||
|
var file = cache[path];
|
|||
|
|
|||
|
var type = path.match(/\.(\w+)$/)[1],
|
|||
|
size = file.size || 0;
|
|||
|
|
|||
|
if (info.enabled) {
|
|||
|
|
|||
|
if (!file.contentsPromise) {
|
|||
|
file.contentsPromise = getFileContents(path);
|
|||
|
}
|
|||
|
|
|||
|
total[type] += size;
|
|||
|
}
|
|||
|
|
|||
|
if (id == updatedId) {
|
|||
|
updated[type] += size;
|
|||
|
}
|
|||
|
}
|
|||
|
});
|
|||
|
}
|
|||
|
if (id !== 'meta' && !info.enabled) {
|
|||
|
allChecked = false;
|
|||
|
}
|
|||
|
|
|||
|
// Select main theme
|
|||
|
if (category === 'themes' && id === updatedId && info.enabled) {
|
|||
|
var themeInput = $('#theme input[value="' + updatedId + '"]');
|
|||
|
if (themeInput) {
|
|||
|
themeInput.checked = true;
|
|||
|
}
|
|||
|
setTheme(updatedId);
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
if (all.meta.addCheckAll) {
|
|||
|
$('input[name="check-all-' + category + '"]').checked = allChecked;
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
total.all = total.js + total.css;
|
|||
|
|
|||
|
if (updatedId) {
|
|||
|
updated.all = updated.js + updated.css;
|
|||
|
|
|||
|
$u.element.prop($('label[data-id="' + updatedId + '"] .filesize'), {
|
|||
|
textContent: prettySize(updated.all),
|
|||
|
title: (updated.js ? Math.round(100 * updated.js / updated.all) + '% JavaScript' : '') +
|
|||
|
(updated.js && updated.css ? ' + ' : '') +
|
|||
|
(updated.css ? Math.round(100 * updated.css / updated.all) + '% CSS' : '')
|
|||
|
});
|
|||
|
}
|
|||
|
|
|||
|
$('#filesize').textContent = prettySize(total.all);
|
|||
|
|
|||
|
$u.element.prop($('#percent-js'), {
|
|||
|
textContent: Math.round(100 * total.js / total.all) + '%',
|
|||
|
title: prettySize(total.js)
|
|||
|
});
|
|||
|
|
|||
|
$u.element.prop($('#percent-css'), {
|
|||
|
textContent: Math.round(100 * total.css / total.all) + '%',
|
|||
|
title: prettySize(total.css)
|
|||
|
});
|
|||
|
|
|||
|
delayedGenerateCode();
|
|||
|
}
|
|||
|
|
|||
|
var timerId = 0;
|
|||
|
// "debounce" multiple rapid requests to generate and highlight code
|
|||
|
function delayedGenerateCode(){
|
|||
|
if ( timerId !== 0 ) {
|
|||
|
clearTimeout(timerId);
|
|||
|
}
|
|||
|
timerId = setTimeout(generateCode, 500);
|
|||
|
}
|
|||
|
|
|||
|
function getSortedComponents(components, requireName, sorted) {
|
|||
|
if (!sorted) {
|
|||
|
sorted = [];
|
|||
|
for (var component in components) {
|
|||
|
sorted.push(component);
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
var i = 0;
|
|||
|
while (i < sorted.length) {
|
|||
|
var id = sorted[i];
|
|||
|
var indexOfRequirement = i;
|
|||
|
var notNow = false;
|
|||
|
for (var requirement in components[id][requireName]) {
|
|||
|
indexOfRequirement = sorted.indexOf(components[id][requireName][requirement]);
|
|||
|
if (indexOfRequirement > i) {
|
|||
|
notNow = true;
|
|||
|
break;
|
|||
|
}
|
|||
|
}
|
|||
|
if (notNow) {
|
|||
|
var tmp = sorted[i];
|
|||
|
sorted[i] = sorted[indexOfRequirement];
|
|||
|
sorted[indexOfRequirement] = tmp;
|
|||
|
}
|
|||
|
else {
|
|||
|
i++;
|
|||
|
}
|
|||
|
}
|
|||
|
return sorted;
|
|||
|
}
|
|||
|
|
|||
|
function getSortedComponentsByRequirements(components, afterName) {
|
|||
|
var sorted = getSortedComponents(components, afterName);
|
|||
|
return getSortedComponents(components, "require", sorted);
|
|||
|
}
|
|||
|
|
|||
|
function generateCode(){
|
|||
|
var promises = [];
|
|||
|
var redownload = {};
|
|||
|
|
|||
|
for (var category in components) {
|
|||
|
var all = components[category];
|
|||
|
|
|||
|
// In case if one component requires other, required component should go first.
|
|||
|
var sorted = getSortedComponentsByRequirements(all, category === 'languages' ? 'peerDependencies' : 'after');
|
|||
|
|
|||
|
for (var i = 0; i < sorted.length; i++) {
|
|||
|
var id = sorted[i];
|
|||
|
|
|||
|
if(id === 'meta') {
|
|||
|
continue;
|
|||
|
}
|
|||
|
|
|||
|
var info = all[id];
|
|||
|
if (info.enabled) {
|
|||
|
if (category !== 'core') {
|
|||
|
redownload[category] = redownload[category] || [];
|
|||
|
redownload[category].push(id);
|
|||
|
}
|
|||
|
info.files[minified? 'minified' : 'dev'].paths.forEach(function (path) {
|
|||
|
if (cache[path]) {
|
|||
|
var type = path.match(/\.(\w+)$/)[1];
|
|||
|
|
|||
|
promises.push({
|
|||
|
contentsPromise: cache[path].contentsPromise,
|
|||
|
path: path,
|
|||
|
type: type
|
|||
|
});
|
|||
|
}
|
|||
|
});
|
|||
|
}
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
// Hide error message if visible
|
|||
|
var error = $('#download .error');
|
|||
|
error.style.display = '';
|
|||
|
|
|||
|
Promise.all([buildCode(promises), getVersion()]).then(function(arr) {
|
|||
|
var res = arr[0];
|
|||
|
var version = arr[1];
|
|||
|
var code = res.code;
|
|||
|
var errors = res.errors;
|
|||
|
|
|||
|
if(errors.length) {
|
|||
|
error.style.display = 'block';
|
|||
|
error.innerHTML = '';
|
|||
|
$u.element.contents(error, errors);
|
|||
|
}
|
|||
|
|
|||
|
var redownloadUrl = window.location.href.split("#")[0] + "#";
|
|||
|
for (var category in redownload) {
|
|||
|
redownloadUrl += category + "=" + redownload[category].join('+') + "&";
|
|||
|
}
|
|||
|
redownloadUrl = redownloadUrl.replace(/&$/,"");
|
|||
|
window.location.replace(redownloadUrl);
|
|||
|
|
|||
|
var versionComment = "/* PrismJS " + version + "\n" + redownloadUrl + " */";
|
|||
|
|
|||
|
for (var type in code) {
|
|||
|
var codeElement = $('#download-' + type + ' code');
|
|||
|
|
|||
|
codeElement.textContent = versionComment + "\n" + code[type];
|
|||
|
Prism.highlightElement(codeElement, true);
|
|||
|
|
|||
|
$('#download-' + type + ' .download-button').href = 'data:application/octet-stream;charset=utf-8,' + encodeURIComponent(versionComment + "\n" + code[type]);
|
|||
|
}
|
|||
|
});
|
|||
|
}
|
|||
|
|
|||
|
function buildCode(promises) {
|
|||
|
var i = 0,
|
|||
|
l = promises.length;
|
|||
|
var code = {js: '', css: ''};
|
|||
|
var errors = [];
|
|||
|
|
|||
|
var f = function(resolve) {
|
|||
|
if(i < l) {
|
|||
|
var p = promises[i];
|
|||
|
p.contentsPromise.then(function(contents) {
|
|||
|
code[p.type] += contents + (p.type === 'js' && !/;\s*$/.test(contents) ? ';' : '') + '\n';
|
|||
|
i++;
|
|||
|
f(resolve);
|
|||
|
});
|
|||
|
p.contentsPromise['catch'](function() {
|
|||
|
errors.push($u.element.create({
|
|||
|
tag: 'p',
|
|||
|
prop: {
|
|||
|
textContent: 'An error occurred while fetching the file "' + p.path + '".'
|
|||
|
}
|
|||
|
}));
|
|||
|
i++;
|
|||
|
f(resolve);
|
|||
|
});
|
|||
|
} else {
|
|||
|
resolve({code: code, errors: errors});
|
|||
|
}
|
|||
|
};
|
|||
|
|
|||
|
return new Promise(f);
|
|||
|
}
|
|||
|
|
|||
|
function getVersion() {
|
|||
|
return getFileContents('./package.json').then(function (jsonStr) {
|
|||
|
return JSON.parse(jsonStr).version;
|
|||
|
});
|
|||
|
}
|
|||
|
|
|||
|
})();
|