mirror of
https://github.com/kalmarek/SmallHyperbolic
synced 2024-11-23 15:35:27 +01:00
add SmallHyperbolic/morphisms page
This commit is contained in:
parent
f84aa07e9e
commit
0ae4f333ed
89
docs/morphisms/index.html
Normal file
89
docs/morphisms/index.html
Normal file
@ -0,0 +1,89 @@
|
||||
<!DOCTYPE html>
|
||||
<html lang="en">
|
||||
<head>
|
||||
<script src="https://cdn.jsdelivr.net/npm/d3@7"></script>
|
||||
|
||||
<!-- Bootstrap -->
|
||||
<link href="https://cdn.jsdelivr.net/npm/bootstrap@5.1.3/dist/css/bootstrap.min.css" rel="stylesheet"
|
||||
integrity="sha384-1BmE4kWBq78iYhFldvKuhfTAU6auU8tT94WrHftjDbrCEXSU1oBoqyl2QvZ6jIW3" crossorigin="anonymous">
|
||||
<link rel="stylesheet" href="https://cdn.jsdelivr.net/npm/bootstrap-icons@1.8.1/font/bootstrap-icons.css">
|
||||
|
||||
<!-- KaTeX -->
|
||||
<link rel="stylesheet" href="https://cdn.jsdelivr.net/npm/katex@0.15.2/dist/katex.min.css"
|
||||
integrity="sha384-MlJdn/WNKDGXveldHDdyRP1R4CTHr3FeuDNfhsLPYrq2t0UBkUdK2jyTnXPEK1NQ" crossorigin="anonymous">
|
||||
<!-- The loading of KaTeX is deferred to speed up page rendering -->
|
||||
<script defer src="https://cdn.jsdelivr.net/npm/katex@0.15.2/dist/katex.min.js"
|
||||
integrity="sha384-VQ8d8WVFw0yHhCk5E8I86oOhv48xLpnDZx5T9GogA/Y84DcCKWXDmSDfn13bzFZY"
|
||||
crossorigin="anonymous"></script>
|
||||
<!-- To automatically render math in text elements, include the auto-render extension: -->
|
||||
<script defer src="https://cdn.jsdelivr.net/npm/katex@0.15.2/dist/contrib/auto-render.min.js"
|
||||
integrity="sha384-+XBljXPPiv+OzfbB3cVmLHf4hdUFHlWNZN5spNQ7rmHTXpd7WvJum6fIACpNNfIR" crossorigin="anonymous"
|
||||
onload="renderMathInElement(document.body);"></script>
|
||||
|
||||
<meta name="viewport" content="width=device-width, initial-scale=1.0">
|
||||
<style type="text/css">
|
||||
body {
|
||||
background-color: #eee;
|
||||
font-family: "Helvetica", sans-serif;
|
||||
font-weight: 300;
|
||||
}
|
||||
|
||||
.height {
|
||||
height: 100vh
|
||||
}
|
||||
|
||||
.canvas {
|
||||
display: inline-block;
|
||||
position: absolute;
|
||||
width: 100%;
|
||||
height: 100vh;
|
||||
vertical-align: top;
|
||||
overflow: hidden;
|
||||
border: 1mm black;
|
||||
}
|
||||
.svg-content {
|
||||
display: inline-block;
|
||||
position: absolute;
|
||||
top: 0;
|
||||
left: 0;
|
||||
}
|
||||
|
||||
.math {
|
||||
position: absolute;
|
||||
top: 2px;
|
||||
}
|
||||
|
||||
.foreground {
|
||||
opacity: 1;
|
||||
filter: sepia(0.0);
|
||||
}
|
||||
|
||||
.search-field {
|
||||
position: relative;
|
||||
/* width: 30%; */
|
||||
z-index: 1;
|
||||
|
||||
}
|
||||
|
||||
</style>
|
||||
</head>
|
||||
<body>
|
||||
<div class="canvas">
|
||||
<!-- <div class="container">
|
||||
<div class="search-field">
|
||||
<label for="exampleDataList" class="form-label">Datalist example</label>
|
||||
<input class="form-control" list="datalistOptions" id="exampleDataList" placeholder="Type to search...">
|
||||
<datalist id="datalistOptions">
|
||||
</datalist>
|
||||
</div>
|
||||
</div> -->
|
||||
</div>
|
||||
|
||||
</body>
|
||||
|
||||
<script type="text/javascript" src="../math_render.js"></script>
|
||||
|
||||
<script type="text/javascript" src="js/d3_visualisation.js"></script>
|
||||
<script type="text/javascript" src="js/morphisms.js"></script>
|
||||
|
||||
</html>
|
221
docs/morphisms/js/d3_visualisation.js
vendored
Normal file
221
docs/morphisms/js/d3_visualisation.js
vendored
Normal file
@ -0,0 +1,221 @@
|
||||
function drag(simulation) {
|
||||
|
||||
function dragstarted(event, d) {
|
||||
if (!event.active) simulation.alphaTarget(0.3).restart();
|
||||
d.fx = d.x;
|
||||
d.fy = d.y;
|
||||
}
|
||||
|
||||
function dragged(event, d) {
|
||||
d.fx = event.x;
|
||||
d.fy = event.y;
|
||||
}
|
||||
|
||||
function dragended(event, d) {
|
||||
if (!event.active) simulation.alphaTarget(0);
|
||||
d.fx = null;
|
||||
d.fy = null;
|
||||
}
|
||||
|
||||
return d3.drag()
|
||||
.on("start", dragstarted)
|
||||
.on("drag", dragged)
|
||||
.on("end", dragended);
|
||||
}
|
||||
|
||||
function linkArc(d) {
|
||||
const r = Math.hypot(d.target.x - d.source.x, d.target.y - d.source.y);
|
||||
return `
|
||||
M${d.source.x},${d.source.y}
|
||||
A${r},${r} 0 0,1 ${d.target.x},${d.target.y}
|
||||
`;
|
||||
}
|
||||
|
||||
function highlight(node) {
|
||||
return node.transition()
|
||||
.duration('400')
|
||||
.attr('opacity', 1)
|
||||
.attr('filter', 'sepia(0.0)')
|
||||
;
|
||||
}
|
||||
|
||||
function dehighlight(node) {
|
||||
return node.transition()
|
||||
.duration('400')
|
||||
.attr('opacity', 0.3)
|
||||
.attr('filter', 'sepia(0.8)')
|
||||
;
|
||||
}
|
||||
|
||||
function _union(...arr) {
|
||||
return arr.reduce((first, second) => [...new Set(first.concat(second))]);
|
||||
}
|
||||
|
||||
async function create_svg(
|
||||
data,
|
||||
width,
|
||||
height,
|
||||
) {
|
||||
let links = data.links;
|
||||
let nodes = data.nodes;
|
||||
let types = Array.from(new Set(nodes.map(d => d.level)));
|
||||
let color = d3.scaleOrdinal(types, d3.schemeSet1);
|
||||
|
||||
d3.select("datalist")
|
||||
.selectAll("option")
|
||||
.data(nodes)
|
||||
.join("option")
|
||||
.attr("value", n=>n.id)
|
||||
.text(n=>n.id)
|
||||
|
||||
|
||||
|
||||
const simulation = d3.forceSimulation(nodes)
|
||||
.force("link", d3.forceLink(links).id(d => d.id))
|
||||
.force("charge", d3.forceManyBody().strength(-400))
|
||||
.force("center", d3.forceCenter(width/2, height/2))
|
||||
.force("x", d3.forceX())
|
||||
.force("y", d3.forceY());
|
||||
|
||||
const svg = d3.create("svg")
|
||||
.attr("preserveAspectRatio", "xMinYMin meet")
|
||||
.attr("viewBox", [0, 0, width, height])
|
||||
.classed("svg-content", true)
|
||||
.style("font", "12px sans-serif");
|
||||
|
||||
// Per-type markers, as they don't inherit styles.
|
||||
svg.append("defs").selectAll("marker")
|
||||
.data(types)
|
||||
.join("marker")
|
||||
.attr("id", d => `arrow-${d}`)
|
||||
.attr("viewBox", "0 -5 10 10")
|
||||
.attr("refX", 15)
|
||||
.attr("refY", 0)
|
||||
.attr("markerWidth", 6)
|
||||
.attr("markerHeight", 6)
|
||||
.attr("orient", "auto")
|
||||
.append("path")
|
||||
.attr("fill", d=>color(d))
|
||||
.attr("d", "M0,-5 L10,0 L0,5 z")
|
||||
;
|
||||
|
||||
const svg_content = svg.append("g")
|
||||
|
||||
const link = svg_content.append("g")
|
||||
.attr("fill", "none")
|
||||
.attr("stroke-width", 1.5)
|
||||
.selectAll("path")
|
||||
.data(links)
|
||||
.join("path")
|
||||
.attr("stroke", d => color(d.source.level))
|
||||
.attr("marker-end", d => `url(${new URL(`#arrow-${d.source.level}`, location)})`)
|
||||
.attr("opacity", 0.3)
|
||||
.attr("filter", "sepia(0.8)")
|
||||
;
|
||||
|
||||
function find_descendants(node) {
|
||||
let desc = links.filter(l => l.source.id == node.id );
|
||||
if (desc.length == 0) {
|
||||
return []
|
||||
} else {
|
||||
let all_desc = desc
|
||||
.map(d => find_descendants(d.target))
|
||||
.reduce(
|
||||
(total, item) => Array.from(new Set(total.concat(item))),
|
||||
desc.map(l=>l.target)
|
||||
)
|
||||
;
|
||||
return all_desc;
|
||||
// return _union(desc, _union(...desc.map(l=>find_descendants(l.target))));
|
||||
}
|
||||
};
|
||||
|
||||
const descendants = {};
|
||||
|
||||
nodes.forEach((node) => {
|
||||
let desc = find_descendants(node);
|
||||
desc.push(node)
|
||||
descendants[node.id] = desc;
|
||||
});
|
||||
|
||||
console.log(descendants)
|
||||
|
||||
function foreground_descendants(id) {
|
||||
node.classed("foreground", (n) => {
|
||||
return (descendants[id].find(v => v.id == n.id)) ? true : false;
|
||||
});
|
||||
|
||||
link.classed("foreground", (n) => {
|
||||
let verts = descendants[id]
|
||||
return (verts.includes(n.source) && verts.includes(n.target)) ? true : false;
|
||||
});
|
||||
}
|
||||
|
||||
d3.select("input").on("input", function () {
|
||||
let n = nodes.find(n => n.id == this.value)
|
||||
if (n) {
|
||||
foreground_descendants(n.id)
|
||||
// let transform = {k:1, x:n.x, y:n.y}
|
||||
console.log(n)
|
||||
|
||||
d3.zoom().translateTo(svg_content,n.x, n.y)
|
||||
// console.log(transform)
|
||||
// svg_content.attr("transform", `translate(${n.x/2},${n.y/2})`)
|
||||
}
|
||||
});
|
||||
|
||||
const node = svg_content.append("g")
|
||||
.attr("stroke-linecap", "round")
|
||||
.attr("stroke-linejoin", "round")
|
||||
.selectAll("g")
|
||||
.data(nodes)
|
||||
.join("g")
|
||||
.attr("class", d=>d.id)
|
||||
.attr("opacity", 0.3)
|
||||
.attr("filter", "sepia(0.8)")
|
||||
.on("mouseover", function (d, i) {
|
||||
highlight(d3.select(this));
|
||||
})
|
||||
.on("mouseout", function (d, i) {
|
||||
dehighlight(d3.select(this))
|
||||
})
|
||||
.on("click", function (d, i) {
|
||||
console.log(this)
|
||||
let id = this.classList[0];
|
||||
foreground_descendants(id);
|
||||
})
|
||||
.call(drag(simulation));
|
||||
|
||||
// circles for nodes:
|
||||
node.append("circle")
|
||||
.attr("stroke", "white")
|
||||
.attr("stroke-width", 1.5)
|
||||
.attr("r", 5)
|
||||
.attr("fill", d => color(d.level));
|
||||
|
||||
node.append("foreignObject")
|
||||
.attr("x", 10)
|
||||
.attr("y", "0.31em")
|
||||
.clone(true).lower()
|
||||
.attr("fill", "none")
|
||||
.attr("stroke", "white")
|
||||
.attr("stroke-width", 5)
|
||||
.append(d => createMathSpan(d.id));
|
||||
|
||||
const zoom = d3.zoom()
|
||||
.scaleExtent([0.2, 5])
|
||||
// .translateExtent([[0, 0], [width, height]])
|
||||
.on("zoom", (e) => {
|
||||
console.log(e.transform)
|
||||
svg_content.attr("transform", e.transform)
|
||||
});
|
||||
|
||||
svg.call(zoom)
|
||||
|
||||
simulation.on("tick", () => {
|
||||
link.attr("d", linkArc);
|
||||
node.attr("transform", d => `translate(${d.x},${d.y})`);
|
||||
});
|
||||
|
||||
return svg;
|
||||
}
|
83
docs/morphisms/js/morphisms.js
Normal file
83
docs/morphisms/js/morphisms.js
Normal file
@ -0,0 +1,83 @@
|
||||
const morphisms_url = new URL("https://raw.githubusercontent.com/kalmarek/SmallHyperbolic/mk/morphisms/data/triangle_groups_morphisms.json")
|
||||
|
||||
async function fetch_json(url) {
|
||||
try {
|
||||
let response = await fetch(url);
|
||||
let json = await response.json();
|
||||
return json;
|
||||
} catch (err) {
|
||||
console.log("Error while fetching json:" + err);
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
;
|
||||
|
||||
async function place_svg(svg) {
|
||||
d3.select("div.canvas")
|
||||
.append("div")
|
||||
.attr("class", "container-fluid")
|
||||
.attr("class", "svg-container")
|
||||
.node()
|
||||
.appendChild(svg.node());
|
||||
};
|
||||
|
||||
async function add_search() {
|
||||
let input_grp = d3.select("div.canvas")
|
||||
.append("div")
|
||||
.classed("search-field", true)
|
||||
.append("div")
|
||||
.classed("container", true)
|
||||
// .append("div")
|
||||
// .classed("input-group", true)
|
||||
;
|
||||
// let floating = input_grp.insert("div")
|
||||
// .attr("class", "form-floating")
|
||||
|
||||
let input = input_grp.insert("input")
|
||||
.attr("class", "form-control")
|
||||
.attr("list", "datalistOptions")
|
||||
.attr("id", "groupSearch")
|
||||
.attr("placeholder", "Type to search...");
|
||||
|
||||
// input_grp.insert("label")
|
||||
// .attr("for", "groupSearch")
|
||||
// .text("Type to search...")
|
||||
|
||||
input_grp.insert("datalist")
|
||||
.attr("id", "datalistOptions")
|
||||
|
||||
// input_grp.append("button")
|
||||
// .classed("btn btn-primary", true)
|
||||
// .attr("type", "button")
|
||||
// .attr("id", "searchBtn")
|
||||
// .append("i")
|
||||
// .classed("bi-search", true)
|
||||
// ;
|
||||
}
|
||||
|
||||
async function show_katex() {
|
||||
let math_objects = document.getElementsByClassName("math");
|
||||
let toggle = true;
|
||||
for (let elt of math_objects) {
|
||||
toggleKaTeX(elt, toggle);
|
||||
let fObj = elt.parentElement;
|
||||
let rect = elt.getElementsByClassName("math-tex")[0].getBoundingClientRect();
|
||||
fObj.setAttribute("width", rect.width+4);
|
||||
fObj.setAttribute("height", rect.height+4);
|
||||
}
|
||||
};
|
||||
|
||||
add_search()
|
||||
|
||||
fetch_json(morphisms_url)
|
||||
// .then(async (data) => { console.log(data); return data;})
|
||||
.then(async (data) => {
|
||||
return create_svg(data, window.innerWidth, window.innerHeight);
|
||||
})
|
||||
// .then(async (data) => { console.log(data); return data; })
|
||||
.then(place_svg)
|
||||
.then(show_katex)
|
||||
// .then(add_search)
|
||||
;
|
||||
|
Loading…
Reference in New Issue
Block a user