hamiltonianFromSpanningTree = (st) => {
const { nrows, ncols, row, col, edges, connect } = st;
const nrows2 = 2 * nrows;
const ncols2 = 2 * ncols;
const edges2 = [];
const grid2 = new Array(nrows2 * ncols2);
const index2 = (i, dcol, drow) =>
(row(i) * 2 + drow) * ncols2 + col(i) * 2 + dcol;
const col2 = (i) => i % ncols2;
const row2 = (i) => ~~(i / ncols2);
connect.forEach((cell, i) => {
let { left, right, up, down } = cell;
if (right) {
edges2.push([index2(i, 1, 0), index2(i, 2, 0)]);
edges2.push([index2(i, 1, 1), index2(i, 2, 1)]);
} else {
edges2.push([index2(i, 1, 0), index2(i, 1, 1)]);
}
if (!left) {
edges2.push([index2(i, 0, 0), index2(i, 0, 1)]);
}
if (down) {
edges2.push([index2(i, 0, 1), index2(i, 0, 2)]);
edges2.push([index2(i, 1, 1), index2(i, 1, 2)]);
} else {
edges2.push([index2(i, 0, 1), index2(i, 1, 1)]);
}
if (!up) {
edges2.push([index2(i, 0, 0), index2(i, 1, 0)]);
}
});
const link = d3.range(ncols2 * nrows2).map((x) => []);
for (let [i, j] of edges2) {
link[i].push(j);
link[j].push(i);
}
let j = 0;
const path = [];
const visited = [];
for (let k = edges2.length; k > 0; k--) {
path.push(j);
visited[j] = true;
j = visited[link[j][0]] ? link[j][1] : link[j][0];
}
return {
nrows: nrows2,
ncols: ncols2,
col: col2,
row: row2,
edges: edges2,
path
};
}