Added graph view

This commit is contained in:
Suzanne Soy 2021-06-23 02:06:06 +01:00
parent db21471d74
commit 98948d6dd5
3 changed files with 282 additions and 75 deletions

View File

@ -7,13 +7,14 @@ article#git-tutorial { position: absolute; top:0; left:0.5em; transition: right,
#git-tutorial textarea, #git-tutorial .CodeMirror { width: 100%; font-size: 1.2rem; border: thin solid black; } #git-tutorial textarea, #git-tutorial .CodeMirror { width: 100%; font-size: 1.2rem; border: thin solid black; }
#git-tutorial table { table-layout: fixed; width: 100%; font-size: 100%; font-family: monospace; min-width: 41em; } #git-tutorial table { table-layout: fixed; width: 100%; font-size: 100%; font-family: monospace; min-width: 41em; }
#git-tutorial pre.log { border: thin solid gray; padding: 0.3em; font-size: 100%; font-family: monospace; box-sizing: border-box; } #git-tutorial pre.log { border: thin solid gray; padding: 0.3em; font-size: 100%; font-family: monospace; box-sizing: border-box; }
#git-tutorial .graph-view { font-size: 100%; font-family: monospace; }
#git-tutorial td.cell-contents, #git-tutorial th.cell-contents { font-family: monospace; } #git-tutorial td.cell-contents, #git-tutorial th.cell-contents { font-family: monospace; }
article#git-tutorial p, article#git-tutorial h1 { max-width: 63rem; } article#git-tutorial p, article#git-tutorial h1 { max-width: 63rem; }
#git-tutorial td, #git-tutorial th { padding-left: 0.3em; padding-right: 0.3em; } #git-tutorial td, #git-tutorial th { padding-left: 0.3em; padding-right: 0.3em; }
#git-tutorial td.cell-contents, #git-tutorial th.cell-contents { width: 36em; } #git-tutorial td.cell-contents, #git-tutorial th.cell-contents { width: 36em; }
article#git-tutorial { left: calc(50% - ( 17.4em / 2 ) - ( 63rem / 2 ) ); right:18.4em; max-width: 63rem; } article#git-tutorial { left: calc(50% - ( 17.4em / 2 ) - ( 63rem / 2 ) ); right:18.4em; max-width: 63rem; }
article#git-tutorial table, #git-tutorial pre.log { width: 77rem; margin-left: calc( ( ( 63rem - 77rem ) / 2 ) ); } article#git-tutorial table, #git-tutorial pre.log, #git-tutorial .graph-view { width: 77rem; margin-left: calc( ( ( 63rem - 77rem ) / 2 ) ); }
#git-tutorial #toc { right:0; } #git-tutorial #toc { right:0; }
#git-tutorial #toc:hover { border-left: 1px solid gray; } #git-tutorial #toc:hover { border-left: 1px solid gray; }
@ -21,7 +22,7 @@ article#git-tutorial table, #git-tutorial pre.log { width: 77rem; margin-left: c
#git-tutorial td, #git-tutorial th { padding-left: 0.3em; padding-right: 0.3em; } #git-tutorial td, #git-tutorial th { padding-left: 0.3em; padding-right: 0.3em; }
#git-tutorial td.cell-contents, #git-tutorial th.cell-contents { width: 36em; } #git-tutorial td.cell-contents, #git-tutorial th.cell-contents { width: 36em; }
article#git-tutorial { left:0.5em; right:18.4em; max-width: 63rem; } article#git-tutorial { left:0.5em; right:18.4em; max-width: 63rem; }
article#git-tutorial table, #git-tutorial pre.log { width: 100%; margin-left: auto; } article#git-tutorial table, #git-tutorial pre.log, #git-tutorial .graph-view { width: 100%; margin-left: auto; }
#git-tutorial #toc { right:0; } #git-tutorial #toc { right:0; }
#git-tutorial #toc:hover { border-left: 1px solid gray; } #git-tutorial #toc:hover { border-left: 1px solid gray; }
} }
@ -30,7 +31,7 @@ article#git-tutorial table, #git-tutorial pre.log { width: 77rem; margin-left: c
#git-tutorial td, #git-tutorial th { padding-left: 0; padding-right: 0; } #git-tutorial td, #git-tutorial th { padding-left: 0; padding-right: 0; }
#git-tutorial td.cell-contents, #git-tutorial th.cell-contents { width: 34em; } #git-tutorial td.cell-contents, #git-tutorial th.cell-contents { width: 34em; }
article#git-tutorial { left:0.5em; right: 7em; max-width: 63rem; } article#git-tutorial { left:0.5em; right: 7em; max-width: 63rem; }
article#git-tutorial table, #git-tutorial pre.log { width: 100%; margin-left: auto; } article#git-tutorial table, #git-tutorial pre.log, #git-tutorial .graph-view { width: 100%; margin-left: auto; }
#git-tutorial #toc { right: -11em; } #git-tutorial #toc { right: -11em; }
#git-tutorial #toc:hover { border-left: 5px solid gray; } #git-tutorial #toc:hover { border-left: 5px solid gray; }
} }
@ -39,7 +40,7 @@ article#git-tutorial table, #git-tutorial pre.log { width: 77rem; margin-left: c
#git-tutorial td, #git-tutorial th { padding-left: 0; padding-right: 0; } #git-tutorial td, #git-tutorial th { padding-left: 0; padding-right: 0; }
#git-tutorial td.cell-contents, #git-tutorial th.cell-contents { width: 30em; } #git-tutorial td.cell-contents, #git-tutorial th.cell-contents { width: 30em; }
article#git-tutorial { left:0.5em; right:6em; max-width: 63rem; } article#git-tutorial { left:0.5em; right:6em; max-width: 63rem; }
article#git-tutorial table, #git-tutorial pre.log { width: 100%; margin-left: auto; } article#git-tutorial table, #git-tutorial pre.log, #git-tutorial .graph-view { width: 100%; margin-left: auto; }
#git-tutorial #toc { right: -12em; } #git-tutorial #toc { right: -12em; }
#git-tutorial #toc:hover { border-left: 5px solid gray; } #git-tutorial #toc:hover { border-left: 5px solid gray; }
} }
@ -96,3 +97,16 @@ article#git-tutorial .onlytoc { display: none; }
#git-tutorial .trivia:before { content: "Trivia"; border-bottom: thin solid slategrey; display: block; text-align: center; } #git-tutorial .trivia:before { content: "Trivia"; border-bottom: thin solid slategrey; display: block; text-align: center; }
#git-tutorial .trivia table { min-width: 90%; max-width: 90%; width: 90%; margin-left: auto; margin-right: auto; } #git-tutorial .trivia table { min-width: 90%; max-width: 90%; width: 90%; margin-left: auto; margin-right: auto; }
/*#git-tutorial .trivia table td.cell-contents, #git-tutorial .trivia table th.cell-contents { width: 30%; }*/ /*#git-tutorial .trivia table td.cell-contents, #git-tutorial .trivia table th.cell-contents { width: 30%; }*/
/* Graph view */
#git-tutorial .dimmed_previous:hover [stroke] { stroke: black; }
#git-tutorial .dimmed_previous:hover [fill=#c0c0c0] { fill: black; }
#git-tutorial .graph-view { text-align: center; }
#git-tutorial .graph-view-tooltips { white-space: nowrap; margin: 1em 0.3em; background: #f8f8f8; border: thin solid #444; text-align: left; }
#git-tutorial .graph-view-tooltips > * { display: inline-block; white-space: normal; width: 100%; margin-right: -100%; vertical-align: middle; }
#git-tutorial .graph-view-tooltips table { margin-left: auto; width: 100%; border: none }
#git-tutorial .graph-view-tooltips tr { border: none; }
#git-tutorial .graph-view-tooltips tr th { opacity: 1; border: none; border-bottom:thin solid #444; text-align: left; }
#git-tutorial .graph-view-tooltips tr td { opacity: 1; border: none; border-top:thin solid: #444; }
#git-tutorial .graph-view-tooltips > .graph-view-tooltips-default { color: #444; text-align: center; }

View File

@ -13,6 +13,7 @@ function ___uint8ArrayToString(a) {
} }
return s.join(''); return s.join('');
} }
// Convert bytes to hex
function ___to_hex(s) { function ___to_hex(s) {
var s = String(s); var s = String(s);
var hex = "" var hex = ""
@ -99,9 +100,10 @@ function ___hilite_off() {
} }
function ___scroll_to_dest(srcid, destclass) { function ___scroll_to_dest(srcid, destclass) {
var src = document.getElementById(srcid); var src = document.getElementById(srcid);
var wrapper = src;
while (wrapper && !wrapper.classList.contains('hilite-wrapper')) { wrapper = wrapper.parentElement; } var wrapper_and_dests = get_wrapper_and_dests(src, destclass);
var dests = (wrapper || document).getElementsByClassName(destclass); var dests = wrapper_and_dests.dests;
if (dests.length > 0) { if (dests.length > 0) {
dest = dests[dests.length - 1]; dest = dests[dests.length - 1];
while (dest && dest.tagName.toLowerCase() != 'tr') { dest = dest.parentElement; } while (dest && dest.tagName.toLowerCase() != 'tr') { dest = dest.parentElement; }
@ -119,12 +121,28 @@ function ___scroll_to_dest(srcid, destclass) {
} }
return false; return false;
} }
function get_wrapper_and_dests(src, destclass) {
var wrapper = src;
while (wrapper && !wrapper.classList.contains('hilite-wrapper')) { wrapper = wrapper.parentElement; }
var maybe_dests = (wrapper || document).getElementsByClassName(destclass);
var dests = [];
for (var i = 0; i < maybe_dests.length; i++) {
var nodest = maybe_dests[i];
while (nodest && !nodest.classList.contains('hilite-nodest')) { nodest = nodest.parentElement; }
if (nodest) {
// skip this as a destination for an arrow or for scrolling
} else {
dests.push(maybe_dests[i]);
}
}
return { wrapper: wrapper, dests: dests };
}
function ___hilite(srcid, destclass) { function ___hilite(srcid, destclass) {
___hilite_off(); ___hilite_off();
var src = document.getElementById(srcid); var src = document.getElementById(srcid);
var wrapper = src; var wrapper_and_dests = get_wrapper_and_dests(src, destclass);
while (wrapper && !wrapper.classList.contains('hilite-wrapper')) { wrapper = wrapper.parentElement; } var wrapper = wrapper_and_dests.wrapper;
var dests = (wrapper || document).getElementsByClassName(destclass); var dests = wrapper_and_dests.dests;
// circumvent glitch where the codemirror areas seem to resize themselves // circumvent glitch where the codemirror areas seem to resize themselves
// which causes the arrow to be misaligned. Instead of using a global container for lines: // which causes the arrow to be misaligned. Instead of using a global container for lines:
@ -247,30 +265,39 @@ function ___hex_hash(s) {
+ ___specialchars_and_colour(s.substr(20) /* should be empty unless there's a bug */); + ___specialchars_and_colour(s.substr(20) /* should be empty unless there's a bug */);
} }
function ___specialchars_and_colour_and_hex(s) { function ___specialchars_and_colour_and_hex(s) {
var target_hashes = [];
if (s.substr(0,5) == "tree ") { if (s.substr(0,5) == "tree ") {
var sp = s.split('\0'); var sp = s.split('\0');
sp[0] = ___specialchars_and_colour(sp[0]); sp[0] = ___specialchars_and_colour(sp[0]);
sp[1] = ___specialchars_and_colour(sp[1]); sp[1] = ___specialchars_and_colour(sp[1]);
for (i = 2; i < sp.length; i++) { for (i = 2; i < sp.length; i++) {
target_hashes.push(___to_hex(sp[i].substr(0,20)));
sp[i] = ___hex_hash(sp[i].substr(0,20)) sp[i] = ___hex_hash(sp[i].substr(0,20))
+ ___specialchars_and_colour(sp[i].substr(20)); + ___specialchars_and_colour(sp[i].substr(20));
} }
return sp.join('<span class="specialchar newline">\\000</span>'); var html = sp.join('<span class="specialchar newline">\\000</span>');
return { target_hashes: target_hashes, html: html };
} else if (/^[0-9a-f]{40}\n$/.test(s)) { } else if (/^[0-9a-f]{40}\n$/.test(s)) {
var id=___global_unique_id++; var id = ___global_unique_id++;
var hash = "object-hash-"+s.substr(0,40); var h = s.substr(0,40);
return '<span id="'+id+'" class="plain-hash-or-ref" onmouseover="___hilite('+id+',\''+hash+'\')" onmouseout="___lolite('+id+',\''+hash+'\')">' target_hashes.push(h);
+ s.substr(0,40) var hash = "object-hash-"+h;
+ '</span>' var html = '<span id="'+id+'" class="plain-hash-or-ref" onmouseover="___hilite('+id+',\''+hash+'\')" onmouseout="___lolite('+id+',\''+hash+'\')">'
+ ___specialchars_and_colour(s.substr(40)); + s.substr(0,40)
+ '</span>'
+ ___specialchars_and_colour(s.substr(40));
return { target_hashes: target_hashes, html: html };
} else if (/^ref: refs\/[^\n]*\n$/.test(s)) { } else if (/^ref: refs\/[^\n]*\n$/.test(s)) {
var id=___global_unique_id++; var id = ___global_unique_id++;
var hash = "object-hash-"+s.substr(5, s.length-6); var h = s.substr(5, s.length-6)
return s.substr(0,5) target_hashes.push(h);
+ '<span id="'+id+'" class="plain-hash-or-ref" onmouseover="___hilite('+id+',\''+hash+'\')" onmouseout="___lolite('+id+',\''+hash+'\')">' var hash = "object-hash-"+h;
+ ___specialchars_and_colour(s.substr(5, s.length-6)) var html = s.substr(0,5)
+ '</span>' + '<span id="'+id+'" class="plain-hash-or-ref" onmouseover="___hilite('+id+',\''+hash+'\')" onmouseout="___lolite('+id+',\''+hash+'\')">'
+ ___specialchars_and_colour(s.substr(s.length-1)); + ___specialchars_and_colour(s.substr(5, s.length-6))
+ '</span>'
+ ___specialchars_and_colour(s.substr(s.length-1));
return { target_hashes: target_hashes, html: html };
} else if(s.substr(0,4) == "DIRC") { } else if(s.substr(0,4) == "DIRC") {
var html = 'DIRC'; // magic var html = 'DIRC'; // magic
var i = 4; var i = 4;
@ -292,7 +319,9 @@ function ___specialchars_and_colour_and_hex(s) {
binary_span(32); // uid binary_span(32); // uid
binary_span(32); // gid binary_span(32); // gid
binary_span(32); // size binary_span(32); // size
html += ___hex_hash(s.substr(i, 20)); // hash var h = s.substr(i, 20);
target_hashes.push(___to_hex(h));
html += ___hex_hash(h); // hash
i += 20; i += 20;
var length = s.substr(i, 2); var length = s.substr(i, 2);
length = length.charCodeAt(0) * 256 + length.charCodeAt(1); length = length.charCodeAt(0) * 256 + length.charCodeAt(1);
@ -315,12 +344,14 @@ function ___specialchars_and_colour_and_hex(s) {
entry_start = i; entry_start = i;
} }
html += ___hex_hash(s.substr(i, 20)); // hash var h = s.substr(i, 20);
target_hashes.push(___to_hex(h));
html += ___hex_hash(h); // hash
i += 20; i += 20;
html += ___specialchars_and_colour(s.substr(i)); // should be empty html += ___specialchars_and_colour(s.substr(i)); // should be empty
return html; return { target_hashes: target_hashes, html: html };
} else if(s.substr(0,7) == "commit ") { } else if(s.substr(0,7) == "commit ") {
var sz = s.split('\0'); var sz = s.split('\0');
var sp = sz[1].split('\n'); var sp = sz[1].split('\n');
@ -330,7 +361,9 @@ function ___specialchars_and_colour_and_hex(s) {
if (/(tree|parent) [0-9a-f]{40}/.test(sp[i])) { if (/(tree|parent) [0-9a-f]{40}/.test(sp[i])) {
var prefix_len = sp[i].startsWith('tree ') ? 5 : 7; var prefix_len = sp[i].startsWith('tree ') ? 5 : 7;
var id=___global_unique_id++; var id=___global_unique_id++;
var hash = "object-hash-"+sp[i].substr(prefix_len); var h = sp[i].substr(prefix_len);
target_hashes.push(h);
var hash = "object-hash-"+h;
sp[i] = ___specialchars_and_colour(sp[i].substr(0,prefix_len)) sp[i] = ___specialchars_and_colour(sp[i].substr(0,prefix_len))
+ '<span id="'+id+'" class="plain-hash-or-ref" onmouseover="___hilite('+id+',\''+hash+'\')" onmouseout="___lolite('+id+',\''+hash+'\')">' + '<span id="'+id+'" class="plain-hash-or-ref" onmouseover="___hilite('+id+',\''+hash+'\')" onmouseout="___lolite('+id+',\''+hash+'\')">'
+ sp[i].substr(prefix_len) + sp[i].substr(prefix_len)
@ -343,9 +376,10 @@ function ___specialchars_and_colour_and_hex(s) {
sp[i] = ___specialchars_and_colour(sp[i]); sp[i] = ___specialchars_and_colour(sp[i]);
} }
var sp_joined = sp.join('<span class="specialchar newline">\\n</span>'); var sp_joined = sp.join('<span class="specialchar newline">\\n</span>');
return [sz[0], sp_joined].join('<span class="specialchar newline">\\000</span>'); var html = [sz[0], sp_joined].join('<span class="specialchar newline">\\000</span>');
return { target_hashes: target_hashes, html: html };
} else { } else {
return ___specialchars_and_colour(s); return { target_hashes: target_hashes, html: ___specialchars_and_colour(s) };
} }
} }
function ___specialchars_and_colour_and_hex_and_zlib(s) { function ___specialchars_and_colour_and_hex_and_zlib(s) {
@ -360,15 +394,15 @@ function ___specialchars_and_colour_and_hex_and_zlib(s) {
html: html:
'<span id="deflated'+id+'-pretty">' '<span id="deflated'+id+'-pretty">'
+ '<span class="deflated">deflated:</span>' + '<span class="deflated">deflated:</span>'
+ ___specialchars_and_colour_and_hex(___uint8ArrayToString(inflated)) + ___specialchars_and_colour_and_hex(___uint8ArrayToString(inflated)).html
+ '</span>' + '</span>'
+ '<span id="deflated'+id+'-raw" style="display:none">' + '<span id="deflated'+id+'-raw" style="display:none">'
+ ___specialchars_and_colour_and_hex(s) + ___specialchars_and_colour_and_hex(s).html
+ '</span>', + '</span>',
td: function(td) { td.classList.add('deflate-toggle'); td.setAttribute('onclick', '___deflated_click('+id+')'); } td: function(td) { td.classList.add('deflate-toggle'); td.setAttribute('onclick', '___deflated_click('+id+')'); }
}; };
} else { } else {
return { html: ___specialchars_and_colour_and_hex(s), td: function() {} }; return { html: ___specialchars_and_colour_and_hex(s).html, td: function() {} };
} }
} }
function ___bytestring_to_printf(bs, trailing_x) { function ___bytestring_to_printf(bs, trailing_x) {
@ -399,23 +433,54 @@ function ___deflated_click(id) {
document.getElementById('deflated'+id+'-raw').style.display = "none"; document.getElementById('deflated'+id+'-raw').style.display = "none";
} }
} }
function ___is_hashed_object_path(x) {
var sp = x.split('/');
return sp.length > 3 && sp[sp.length-3] == 'objects' && /^[0-9a-f]{2}$/.test(sp[sp.length-2]) && /^[0-9a-f]{38}$/.test(sp[sp.length-1]);
}
function ___get_hashed_object_path(x) {
var sp = x.split('/');
return sp.slice(sp.length-2).join('');
}
function ___is_ref_path(x) {
var sp = x.split('/');
return sp.length > 1 && sp.indexOf('refs') >= 0 && sp.length > sp.indexOf('refs') + 1;
}
function ___get_ref_path(x) {
var sp = x.split('/');
var refs_idx = sp.indexOf('refs');
return sp.slice(refs_idx).join('/');
}
function ___format_filepath(x) { function ___format_filepath(x) {
var sp = x.split('/'); var sp = x.split('/');
if (sp.length > 3 && sp[sp.length-3] == 'objects' && /^[0-9a-f]{2}$/.test(sp[sp.length-2]) && /^[0-9a-f]{38}$/.test(sp[sp.length-1])) { if (___is_hashed_object_path(x)) {
return sp.slice(0, sp.length-2).map(___specialchars_and_colour).join('/')+(sp.length > 2 ? '/' : '') return sp.slice(0, sp.length-2).map(___specialchars_and_colour).join('/')+(sp.length > 2 ? '/' : '')
+ '<span class="object-hash object-hash-'+sp.slice(sp.length-2).join('')+'">' + '<span class="object-hash object-hash-'+___get_hashed_object_path(x)+'">'
+ sp.slice(sp.length-2).map(___specialchars_and_colour).join('/') + sp.slice(sp.length-2).map(___specialchars_and_colour).join('/')
+ "</span>"; + "</span>";
} else if (sp.length > 1 && sp.indexOf('refs') >= 0 && sp.length > sp.indexOf('refs') + 1) { } else if (___is_ref_path(x)) {
var refs_idx = sp.indexOf('refs'); var refs_idx = sp.indexOf('refs');
return sp.slice(0, refs_idx).map(___specialchars_and_colour).join('/')+'/' return sp.slice(0, refs_idx).map(___specialchars_and_colour).join('/')+'/'
+ '<span class="object-hash object-hash-'+sp.slice(refs_idx).join('/')+'">'//TODO + '<span class="object-hash object-hash-'+___get_ref_path(x)+'">'//TODO
+ sp.slice(refs_idx).map(___specialchars_and_colour).join('/') + sp.slice(refs_idx).map(___specialchars_and_colour).join('/')
+ "</span>"; + "</span>";
} else { } else {
return ___specialchars_and_colour(x); return ___specialchars_and_colour(x);
} }
} }
function ___format_contents(contents) {
if (contents === null) {
return { html: '<span class="directory">Directory</span>', td: function() {} };
} else {
var specials = ___specialchars_and_colour_and_hex_and_zlib(contents);
return { html: '<code>' + specials.html + '</code>', td: specials.td };
}
}
function ___format_entry(previous_filesystem, x) { function ___format_entry(previous_filesystem, x) {
var previous_filesystem = previous_filesystem || {}; var previous_filesystem = previous_filesystem || {};
var tr = document.createElement('tr'); var tr = document.createElement('tr');
@ -434,13 +499,9 @@ function ___format_entry(previous_filesystem, x) {
var td_contents = document.createElement('td'); var td_contents = document.createElement('td');
tr.appendChild(td_contents); tr.appendChild(td_contents);
td_contents.classList.add('cell-contents'); td_contents.classList.add('cell-contents');
if (x[1] === null) { var html_and_function = ___format_contents(x[1]);
td_contents.innerHTML = '<span class="directory">Directory</span>'; td_contents.innerHTML = html_and_function.html;
} else { html_and_function.td(td_contents);
var specials = ___specialchars_and_colour_and_hex_and_zlib(x[1]);
td_contents.innerHTML = '<code>' + specials.html + '</code>';
specials.td(td_contents);
}
return tr; return tr;
} }
@ -487,7 +548,7 @@ function ___filesystem_to_table(fs, previous_filesystem) {
function ___filesystem_to_string(fs, just_table, previous_filesystem) { function ___filesystem_to_string(fs, just_table, previous_filesystem) {
var entries = ___sort_filesystem_entries(fs); var entries = ___sort_filesystem_entries(fs);
var id = ___global_unique_id++; var id = ___global_unique_id++;
var html = '<div class="hilite-wrapper">'; var html = '';
if (! just_table) { if (! just_table) {
html += 'Filesystem contents: ' + entries.length + " files and directories. " html += 'Filesystem contents: ' + entries.length + " files and directories. "
+ '<a href="javascript: ___copyprintf_click(\'elem-'+id+'\');">' + '<a href="javascript: ___copyprintf_click(\'elem-'+id+'\');">'
@ -498,8 +559,7 @@ function ___filesystem_to_string(fs, just_table, previous_filesystem) {
+ ___specialchars(___filesystem_to_printf(fs) || 'echo "Empty filesystem."') + ___specialchars(___filesystem_to_printf(fs) || 'echo "Empty filesystem."')
+ '</textarea>'; + '</textarea>';
} }
html += ___filesystem_to_table(fs, previous_filesystem).outerHTML // TODO: use DOM primitives instead. html += ___filesystem_to_table(fs, previous_filesystem).outerHTML; // TODO: use DOM primitives instead.
+ '</div>';
return html; return html;
} }
function ___textarea_value(elem) { function ___textarea_value(elem) {
@ -537,41 +597,175 @@ var ___script_log_header = ''
+ '})(window.console);\n' + '})(window.console);\n'
+ '\n'; + '\n';
function ___file_contents_to_graphviz(s) { function ___file_contents_to_graphview(filesystem, path_of_this_file, s) {
var gv = '';
try { try {
var inflated = pako.inflate(___stringToUint8Array(s)); var inflated = pako.inflate(___stringToUint8Array(s));
if (inflated) {
var s2 = ___uint8ArrayToString(inflated);
} else {
var s2 = s;
}
} catch(e) { } catch(e) {
var inflated = false; var s2 = s;
} }
if (inflated) { var target_hashes = ___specialchars_and_colour_and_hex(s2).target_hashes;
var id=___global_unique_id++; var paths = Object.keys(filesystem);
return { for (var i = 0; i < paths.length; i++) {
html: if (___is_hashed_object_path(paths[i])) {
'<span id="deflated'+id+'-pretty">' if (target_hashes.indexOf(___get_hashed_object_path(paths[i])) != -1) {
+ '<span class="deflated">deflated:</span>' gv += ___quote_gv(path_of_this_file) + ' -> ' + ___quote_gv(paths[i]);
+ ___specialchars_and_colour_and_hex(___uint8ArrayToString(inflated)) }
+ '</span>' }
+ '<span id="deflated'+id+'-raw" style="display:none">' if (___is_ref_path(paths[i])) {
+ ___specialchars_and_colour_and_hex(s) if (target_hashes.indexOf(___get_ref_path(paths[i])) != -1) {
+ '</span>', gv += ___quote_gv(path_of_this_file) + ' -> ' + ___quote_gv(paths[i]);
td: function(td) { td.classList.add('deflate-toggle'); td.setAttribute('onclick', '___deflated_click('+id+')'); } }
}; }
} else {
return { html: ___specialchars_and_colour_and_hex(s), td: function() {} };
} }
return gv;
} }
function ___filesystem_to_graphviz(filesystem, previous_filesystem) { function ___quote_gv(name) {
return "digraph graph_view {" console.log('TODO: escape GV')
+ 'a -> b' return '"' + name + '"';
+ "}";
} }
function ___eval_result_to_string(filesystem, previous_filesystem, log) { function ___entry_to_graphview(previous_filesystem, filesystem, x) {
var gv = '';
gv += ___quote_gv(x[0]) + '\n';
var components = x[0].split('/');
if (___is_hashed_object_path(x[0])) {
// var hash = components.slice(components.length-2).join('');
var shortname = components[components.length - 1].substr(0, 3) + '…';
} else {
var shortname = components[components.length - 1];
}
var parent = components.slice(0, components.length - 1).join('/');
if (parent != '') {
if (filesystem.hasOwnProperty(parent)) {
gv += ___quote_gv(parent) + ' -> ' + ___quote_gv(x[0]) + '[color="#c0c0ff"];\n';
} else {
shortname = parent + '/' + shortname;
}
}
// shortname as a label
gv += ___quote_gv(x[0]) + ' [ label = ' + ___quote_gv(shortname) + ' ]';
// Put a transparent background to make the nodes clickable.
gv += ___quote_gv(x[0]) + ' [ style="filled", fillcolor="transparent" ]';
// dim nodes that existed in the previous_filesystem
if (previous_filesystem.hasOwnProperty(x[0])) {
gv += ___quote_gv(x[0]) + ' [ color = gray, fontcolor = gray, class = dimmed_previous ]';
}
// contents of the file as a tooltip:
gv += ___quote_gv(x[0]) + ' [ tooltip = ' + '"CONTENTS x[1]"' + ' ]';
var id = ___global_unique_id++;
gv += ___quote_gv(x[0]) + ' [ id=' + id + ' ]';
if (x[1] === null) {
// This is a directory, nothing else to do.
} else {
gv += ___file_contents_to_graphview(filesystem, x[0], x[1]);
}
return { id:id, gv:gv };
}
var ___current_hover_graphview = null;
var ___sticky_hover_graphview = false;
function ___click_graphview_hover(id, default_id) {
if (___sticky_hover_graphview && ___current_hover_graphview === id) {
___sticky_hover_graphview = false;
___hide_graphview_hover(id, default_id);
} else {
___sticky_hover_graphview = true;
___show_graphview_hover(id, default_id);
}
}
function ___mouseout_graphview_hover(id, default_id) {
if (!___sticky_hover_graphview) {
___hide_graphview_hover(id, default_id);
}
}
function ___mouseover_graphview_hover(id, default_id) {
if (!___sticky_hover_graphview) {
___show_graphview_hover(id, default_id);
}
}
function ___show_graphview_hover(id, default_id) {
if (___current_hover_graphview !== null) {
___hide_graphview_hover(___current_hover_graphview, default_id);
}
___current_hover_graphview = id;
document.getElementById(default_id).style.visibility = 'hidden';
document.getElementById(id).style.visibility = 'visible';
}
function ___hide_graphview_hover(id, default_id) {
___current_hover_graphview = default_id;
document.getElementById(default_id).style.visibility = 'visible';
document.getElementById(id).style.visibility = 'hidden';
}
function ___filesystem_to_graphview(filesystem, previous_filesystem) {
var html = '';
html += '<div class="graph-view-tooltips hilite-nodest">';
var entry_hover_default_id = ___global_unique_id++;
html += '<div class="graph-view-tooltips-default" id="'+entry_hover_default_id+'">Hover a node to view its contents, click or tap to pin it.</div>'
var gv = "digraph graph_view {";
var ids = [];
var entries = ___sort_filesystem_entries(filesystem);
for (var i = 0; i < entries.length; i++) {
var entry = ___entry_to_graphview(previous_filesystem, filesystem, entries[i]);
gv += entry.gv;
var entry_hover_id = ___global_unique_id++;
var entry_hover = '';
entry_hover += '<div id="' + entry_hover_id + '" style="visibility: hidden">';
//entry_hover += 'hover for ' + entries[i][0];
entry_hover += '<table><thead><tr><th class="cell-path">' + ___format_filepath(entries[i][0]) + '</th></tr></thead>';
// TODO: use the .td function here too.
entry_hover += '<tbody><tr><td>' + ___format_contents(entries[i][1]).html + '</td></tr></tbody></table>';
entry_hover += '</div>';
html += entry_hover;
ids.push({ id: entry.id, entry_hover_id: entry_hover_id });
}
gv += '}';
var js = function () {
for (var i = 0; i < ids.length; i++) {
var el = document.getElementById(ids[i].id);
el.setAttribute('onclick', '___click_graphview_hover(' + ids[i].entry_hover_id + ', '+entry_hover_default_id+')');
el.setAttribute('onmouseover', '___mouseover_graphview_hover(' + ids[i].entry_hover_id + ', '+entry_hover_default_id+')');
el.setAttribute('onmouseout', '___mouseout_graphview_hover(' + ids[i].entry_hover_id + ', '+entry_hover_default_id+')');
}
};
html += "</div>";
html += Viz(gv, "svg");
if (entries.length == 0) {
return { js: js, html: '<br/>' };
} else {
return { js: js, html: '<div class="graph-view">' + html + '</div>' };
}
}
function ___eval_result_to_html(id, filesystem, previous_filesystem, log, quiet) {
var loghtml = '<pre class="log">' + log.map(function(l) { return l.map(function (x) { return x.toString(); }).join(', '); }).join('\n') + '</pre>' var loghtml = '<pre class="log">' + log.map(function(l) { return l.map(function (x) { return x.toString(); }).join(', '); }).join('\n') + '</pre>'
return (log.length > 0 ? '<p>Console output:</p>' + loghtml : '') var table = ___filesystem_to_string(filesystem, quiet, previous_filesystem);
+ Viz(___filesystem_to_graphviz(filesystem, previous_filesystem), "svg") var gv = ___filesystem_to_graphview(filesystem, previous_filesystem);
+ ___filesystem_to_string(filesystem, false, previous_filesystem); var html = (log.length > 0 ? '<p>Console output:</p>' + loghtml : '')
+ gv.html
+ table;
document.getElementById(id).innerHTML = '<div class="hilite-wrapper">' + html + '</div>';
gv.js();
} }
function ___git_eval(current) { function ___git_eval(current) {
document.getElementById('hide-eval-' + current).style.display = ''; document.getElementById('hide-eval-' + current).style.display = '';
@ -588,7 +782,7 @@ function ___git_eval(current) {
+ '"End of the script";\n' + '"End of the script";\n'
+ '\n' + '\n'
+ '\n' + '\n'
+ 'document.getElementById("out" + current).innerHTML = ___eval_result_to_string(filesystem, ___previous_filesystem, ___log);\n' + '___eval_result_to_html("out" + current, filesystem, ___previous_filesystem, ___log, false);\n'
+ 'filesystem;\n'; + 'filesystem;\n';
try { try {
eval(script); eval(script);

View File

@ -10,7 +10,6 @@
<script src="sha1.js"></script> <script src="sha1.js"></script>
<script src="pako.min.js"></script> <script src="pako.min.js"></script>
<script src="viz.js"></script> <script src="viz.js"></script>
<script src="lite.render.js"></script>
<link rel="stylesheet" href="codemirror-5.60.0/lib/codemirror.css"> <link rel="stylesheet" href="codemirror-5.60.0/lib/codemirror.css">
<!-- Implementation of the tutorial's helper tools (code editor, graph view, table of contents, table output and arrows): --> <!-- Implementation of the tutorial's helper tools (code editor, graph view, table of contents, table output and arrows): -->
@ -30,7 +29,7 @@ function ___example(id, f) {
for (var i = 0; i < result.previous_names.length; i++) { for (var i = 0; i < result.previous_names.length; i++) {
previous_fs[result.previous_names[i]] = filesystem[result.previous_names[i]]; previous_fs[result.previous_names[i]] = filesystem[result.previous_names[i]];
} }
document.getElementById(id).innerHTML = ___filesystem_to_string(fs, true, previous_fs); ___eval_result_to_html(id, fs, previous_fs, [], true);
}); });
} }
</script> </script>