use prism-tomorrow.css
This commit is contained in:
598
docs/_style/prism-master/download.js
Normal file
598
docs/_style/prism-master/download.js
Normal file
@ -0,0 +1,598 @@
|
||||
/**
|
||||
* 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;
|
||||
});
|
||||
}
|
||||
|
||||
})();
|
Reference in New Issue
Block a user