<!doctype html>
<html lang="en"> 
  <head>
    <meta charset="UTF-8" />
    <meta name="viewport" content="width=device-width, initial-scale=1.0" />
    <script
      id="p5scripttag"
      src="https://cdnjs.cloudflare.com/ajax/libs/p5.js/1.9.0/p5.min.js"
      integrity="sha512-uaz5GpnQoE6t5echKlX8P52czvsIGgLPcvlzfvRubLZ1Hp8JemUDnbUiAahbVtPb+jUVrNETuXvAhDDF/N3M4w=="
      crossorigin="anonymous"
      referrerpolicy="no-referrer"
    ></script>

    <link
      rel="stylesheet"
      href="https://cdnjs.cloudflare.com/ajax/libs/highlight.js/11.9.0/styles/atom-one-dark.min.css"
    />
    <script src="https://cdnjs.cloudflare.com/ajax/libs/highlight.js/11.9.0/highlight.min.js"></script>
    <script src="https://cdnjs.cloudflare.com/ajax/libs/highlight.js/11.9.0/languages/python.min.js"></script>

    <script>
      const bgCol = "#F2F0E7";
const accentCol = "#fd4578";

hljs.initHighlightingOnLoad();

const updateTargetDims = () => {
  // width is max-width of `.contentContainer` - its padding
  // return [min(windowWidth, 900 - 80), 700]
  return [windowWidth * (1 / 2), windowHeight];
};

const setCodeAndPlan = (code, plan) => {
  const codeElm = document.getElementById("code");
  if (codeElm) {
    // codeElm.innerText = code;
    codeElm.innerHTML = hljs.highlight(code, { language: "python" }).value;
  }

  const planElm = document.getElementById("plan");
  if (planElm) {
    // planElm.innerText = plan.trim();
    planElm.innerHTML = hljs.highlight(plan, { language: "plaintext" }).value;
  }
};

windowResized = () => {
  resizeCanvas(...updateTargetDims());
  awaitingPostResizeOps = true;
};

const animEase = (t) => 1 - (1 - Math.min(t, 1.0)) ** 5;

// ---- global constants ----

const globalAnimSpeed = 1.1;
const scaleFactor = 0.57;

// ---- global vars ----

let globalTime = 0;
let manualSelection = false;

let currentElemInd = 0;

let treeStructData = {"edges": [[1, 5]], "layout": [[0.0, 0.0], [0.25, 0.0], [0.5, 0.0], [0.75, 0.0], [1.0, 0.0], [0.25, 1.0]], "plan": ["To integrate single-cell RNA-seq multimodal data, I will use a simple approach\nbased on canonical correlation analysis (CCA) to align the RNA and ATAC data\nfrom the two modalities. After performing CCA, I will project both datasets into\na common space and store the integrated representations in\n`adata_mod1.obsm['X_integrated']` and `adata_mod2.obsm['X_integrated']`.\nFinally, I will evaluate the integration using the provided `compute_knn_auc`\nfunction to measure the overlap of nearest neighbors between the two modalities.", "To tackle the task of integrating single-cell RNA-seq multimodal data, we will\nuse a simple approach based on the alignment of the latent spaces of the two\nmodalities. We will leverage the normalized data from both `adata_mod1` and\n`adata_mod2`, applying dimensionality reduction techniques, such as PCA or UMAP,\nto obtain a shared representation. The integrated representations will be stored\nin `adata_mod1.obsm['X_integrated']` and `adata_mod2.obsm['X_integrated']`.\nFinally, we will evaluate the integration quality using the provided\n`compute_knn_auc` function, which assesses the overlap of nearest neighbors\nbetween the two datasets.", "To integrate the single-cell RNA-seq multimodal data, we will utilize a simple\napproach that leverages PCA for dimensionality reduction. We will extract the\nnormalized RNA data from `adata_mod1` and the normalized ATAC data from\n`adata_mod2`, apply PCA to both datasets to obtain a shared latent space, and\nstore the integrated representations in `adata_mod1.obsm['X_integrated']` and\n`adata_mod2.obsm['X_integrated']`. Finally, we will evaluate the integration\nquality using the provided `compute_knn_auc` function, which measures the\noverlap of nearest neighbors between the two modalities.", "To integrate the single-cell RNA-seq multimodal data, we will utilize the\nCanonical Correlation Analysis (CCA) method, which is effective for aligning\ndifferent modalities by maximizing the correlation between them. We will extract\nthe normalized RNA data from `adata_mod1` and the normalized ATAC data from\n`adata_mod2`, apply CCA to obtain a shared latent space, and store the\nintegrated representations in `adata_mod1.obsm['X_integrated']` and\n`adata_mod2.obsm['X_integrated']`. Finally, we will evaluate the integration\nquality using the provided `compute_knn_auc` function to measure the overlap of\nnearest neighbors between the two datasets.", "To integrate the single-cell RNA-seq multimodal data, I will utilize a canonical\ncorrelation analysis (CCA) approach, which is effective for aligning two\ndifferent modalities. I will extract the normalized RNA data from `adata_mod1`\nand the normalized ATAC data from `adata_mod2`, apply CCA to find a shared\nlatent space, and store the integrated representations in\n`adata_mod1.obsm['X_integrated']` and `adata_mod2.obsm['X_integrated']`.\nFinally, I will evaluate the integration quality using the provided\n`compute_knn_auc` function to measure the overlap of nearest neighbors between\nthe two datasets.", "To improve the integration of single-cell RNA-seq multimodal data, we can\nenhance the dimensionality reduction step by using UMAP instead of PCA. UMAP\n(Uniform Manifold Approximation and Projection) is known to preserve local\nstructure and can better capture complex relationships in the data. This change\nmay lead to a more accurate shared representation of the two modalities,\npotentially increasing the overlap of nearest neighbors and thus improving the\nKNN AUC score."], "code": ["import numpy as np\nimport pandas as pd\nimport anndata as ad\nfrom sklearn.neighbors import NearestNeighbors\nfrom sklearn.cross_decomposition import CCA\n\n# Load data\nadata_mod1 = ad.read_h5ad(\"./input/scicar_mouse_kidney_mod1_subset.h5ad\")\nadata_mod2 = ad.read_h5ad(\"./input/scicar_mouse_kidney_mod2_subset.h5ad\")\n\n# Extract normalized layers\nX_mod1 = adata_mod1.layers[\"normalized\"]\nX_mod2 = adata_mod2.layers[\"normalized\"]\n\n# Perform CCA\ncca = CCA(n_components=50)  # Choose the number of components\nX_mod1_cca, X_mod2_cca = cca.fit_transform(X_mod1, X_mod2)\n\n# Store integrated data\nadata_mod1.obsm[\"X_integrated\"] = X_mod1_cca\nadata_mod2.obsm[\"X_integrated\"] = X_mod2_cca\n\n\n# Evaluation function\ndef compute_knn_auc(adata_mod1, adata_mod2, k=30):\n    X_mod1 = adata_mod1.obsm[\"X_integrated\"]\n    X_mod2 = adata_mod2.obsm[\"X_integrated\"]\n\n    n_cells = X_mod1.shape[0]\n    knn_mod1 = NearestNeighbors(n_neighbors=k + 1).fit(X_mod1)\n    knn_mod2 = NearestNeighbors(n_neighbors=k + 1).fit(X_mod2)\n\n    neighbors1 = knn_mod1.kneighbors(return_distance=False)[:, 1:]\n    neighbors2 = knn_mod2.kneighbors(return_distance=False)[:, 1:]\n\n    overlaps = [\n        len(set(neighbors1[i]) & set(neighbors2[i])) / k for i in range(n_cells)\n    ]\n    return np.mean(overlaps)\n\n\n# Compute and print the evaluation metric\nauc_score = compute_knn_auc(adata_mod1, adata_mod2)\nprint(\"KNN AUC Score:\", auc_score)\n\n# Optionally, save the integrated data for further analysis or submission\nadata_mod1.write(\"./working/adata_mod1_integrated.h5ad\")\nadata_mod2.write(\"./working/adata_mod2_integrated.h5ad\")\n", "import numpy as np\nimport pandas as pd\nimport scanpy as sc\nfrom sklearn.decomposition import PCA\nfrom sklearn.neighbors import NearestNeighbors\n\n# Load the datasets\nadata_mod1 = sc.read(\"./input/scicar_mouse_kidney_mod1_subset.h5ad\")\nadata_mod2 = sc.read(\"./input/scicar_mouse_kidney_mod2_subset.h5ad\")\n\n# Normalize and prepare the data\nX_mod1 = adata_mod1.layers[\"normalized\"]\nX_mod2 = adata_mod2.layers[\"normalized\"]\n\n# Dimensionality reduction using PCA\npca_mod1 = PCA(n_components=50)\nX_integrated_mod1 = pca_mod1.fit_transform(X_mod1)\n\npca_mod2 = PCA(n_components=50)\nX_integrated_mod2 = pca_mod2.fit_transform(X_mod2)\n\n# Store the integrated representations\nadata_mod1.obsm[\"X_integrated\"] = X_integrated_mod1\nadata_mod2.obsm[\"X_integrated\"] = X_integrated_mod2\n\n\n# Define the evaluation function\ndef compute_knn_auc(adata_mod1, adata_mod2, k=30):\n    X_mod1 = adata_mod1.obsm[\"X_integrated\"]\n    X_mod2 = adata_mod2.obsm[\"X_integrated\"]\n\n    n_cells = X_mod1.shape[0]\n    knn_mod1 = NearestNeighbors(n_neighbors=k + 1).fit(X_mod1)\n    knn_mod2 = NearestNeighbors(n_neighbors=k + 1).fit(X_mod2)\n\n    neighbors1 = knn_mod1.kneighbors(return_distance=False)[:, 1:]\n    neighbors2 = knn_mod2.kneighbors(return_distance=False)[:, 1:]\n\n    overlaps = [\n        len(set(neighbors1[i]) & set(neighbors2[i])) / k for i in range(n_cells)\n    ]\n    return np.mean(overlaps)\n\n\n# Evaluate the integration\nauc_score = compute_knn_auc(adata_mod1, adata_mod2)\nprint(f\"KNN AUC Score: {auc_score}\")\n\n# Save predictions (if applicable)\nsubmission_df = pd.DataFrame({\"KNN_AUC_Score\": [auc_score]})\nsubmission_df.to_csv(\"./working/submission.csv\", index=False)\n", "import numpy as np\nimport pandas as pd\nimport scanpy as sc\nfrom sklearn.decomposition import PCA\nfrom sklearn.neighbors import NearestNeighbors\n\n# Load data\nadata_mod1 = sc.read(\"./input/scicar_mouse_kidney_mod1_subset.h5ad\")\nadata_mod2 = sc.read(\"./input/scicar_mouse_kidney_mod2_subset.h5ad\")\n\n# Extract normalized data\nX_mod1 = adata_mod1.layers[\"normalized\"]\nX_mod2 = adata_mod2.layers[\"normalized\"]\n\n# Apply PCA\npca = PCA(n_components=50)  # Choose a suitable number of components\nX_mod1_pca = pca.fit_transform(X_mod1)\nX_mod2_pca = pca.fit_transform(X_mod2)\n\n# Store integrated representations\nadata_mod1.obsm[\"X_integrated\"] = X_mod1_pca\nadata_mod2.obsm[\"X_integrated\"] = X_mod2_pca\n\n\n# Evaluate integration quality\ndef compute_knn_auc(adata_mod1, adata_mod2, k=30):\n    X_mod1 = adata_mod1.obsm[\"X_integrated\"]\n    X_mod2 = adata_mod2.obsm[\"X_integrated\"]\n\n    n_cells = X_mod1.shape[0]\n    knn_mod1 = NearestNeighbors(n_neighbors=k + 1).fit(X_mod1)\n    knn_mod2 = NearestNeighbors(n_neighbors=k + 1).fit(X_mod2)\n\n    neighbors1 = knn_mod1.kneighbors(return_distance=False)[:, 1:]\n    neighbors2 = knn_mod2.kneighbors(return_distance=False)[:, 1:]\n\n    overlaps = [\n        len(set(neighbors1[i]) & set(neighbors2[i])) / k for i in range(n_cells)\n    ]\n    return np.mean(overlaps)\n\n\n# Compute and print the KNN AUC score\nknn_auc_score = compute_knn_auc(adata_mod1, adata_mod2)\nprint(f\"KNN AUC Score: {knn_auc_score}\")\n\n# Save submission file\nsubmission = pd.DataFrame({\"KNN_AUC_Score\": [knn_auc_score]})\nsubmission.to_csv(\"./working/submission.csv\", index=False)\n", "import numpy as np\nimport pandas as pd\nimport scanpy as sc\nfrom sklearn.cross_decomposition import CCA\nfrom sklearn.neighbors import NearestNeighbors\n\n# Load the datasets\nadata_mod1 = sc.read(\"./input/scicar_mouse_kidney_mod1_subset.h5ad\")\nadata_mod2 = sc.read(\"./input/scicar_mouse_kidney_mod2_subset.h5ad\")\n\n# Extract normalized data\nrna_data = adata_mod1.layers[\"normalized\"]\natac_data = adata_mod2.layers[\"normalized\"]\n\n# Apply Canonical Correlation Analysis (CCA)\ncca = CCA(n_components=30)\nX_combined = cca.fit_transform(rna_data, atac_data)\n\n# Store the integrated data\nadata_mod1.obsm[\"X_integrated\"] = X_combined\nadata_mod2.obsm[\"X_integrated\"] = X_combined\n\n\n# Define the evaluation function\ndef compute_knn_auc(adata_mod1, adata_mod2, k=30):\n    X_mod1 = adata_mod1.obsm[\"X_integrated\"]\n    X_mod2 = adata_mod2.obsm[\"X_integrated\"]\n\n    n_cells = X_mod1.shape[0]\n    knn_mod1 = NearestNeighbors(n_neighbors=k + 1).fit(X_mod1)\n    knn_mod2 = NearestNeighbors(n_neighbors=k + 1).fit(X_mod2)\n\n    neighbors1 = knn_mod1.kneighbors(return_distance=False)[:, 1:]\n    neighbors2 = knn_mod2.kneighbors(return_distance=False)[:, 1:]\n\n    overlaps = [\n        len(set(neighbors1[i]) & set(neighbors2[i])) / k for i in range(n_cells)\n    ]\n    return np.mean(overlaps)\n\n\n# Evaluate the integration\nknn_auc_score = compute_knn_auc(adata_mod1, adata_mod2)\nprint(\"KNN AUC Score:\", knn_auc_score)\n\n# Save the predictions\nsubmission_data = pd.DataFrame({\"KNN_AUC_Score\": [knn_auc_score]})\nsubmission_data.to_csv(\"./working/submission.csv\", index=False)\n", "import numpy as np\nimport pandas as pd\nimport scanpy as sc\nfrom sklearn.cross_decomposition import CCA\nfrom sklearn.neighbors import NearestNeighbors\n\n# Load the data\nadata_mod1 = sc.read(\"./input/scicar_mouse_kidney_mod1_subset.h5ad\")\nadata_mod2 = sc.read(\"./input/scicar_mouse_kidney_mod2_subset.h5ad\")\n\n# Extract normalized data\nX_mod1 = adata_mod1.layers[\"normalized\"]\nX_mod2 = adata_mod2.layers[\"normalized\"]\n\n# Apply Canonical Correlation Analysis (CCA)\ncca = CCA(n_components=50)\nX_combined = np.concatenate([X_mod1, X_mod2], axis=0)\ncca.fit(X_combined)\n\n# Transform the data to the integrated space\nX_mod1_integrated = cca.transform(X_mod1)\nX_mod2_integrated = cca.transform(X_mod2)\n\n# Store the integrated representations\nadata_mod1.obsm[\"X_integrated\"] = X_mod1_integrated\nadata_mod2.obsm[\"X_integrated\"] = X_mod2_integrated\n\n# Evaluate integration quality\nknn_auc_score = compute_knn_auc(adata_mod1, adata_mod2)\nprint(f\"KNN AUC Score: {knn_auc_score}\")\n\n# Save predictions (if applicable)\nsubmission_df = pd.DataFrame({\"KNN_AUC_Score\": [knn_auc_score]})\nsubmission_df.to_csv(\"./working/submission.csv\", index=False)\n", "import numpy as np\nimport pandas as pd\nimport scanpy as sc\nimport umap\nfrom sklearn.neighbors import NearestNeighbors\n\n# Load the datasets\nadata_mod1 = sc.read(\"./input/scicar_mouse_kidney_mod1_subset.h5ad\")\nadata_mod2 = sc.read(\"./input/scicar_mouse_kidney_mod2_subset.h5ad\")\n\n# Normalize and prepare the data\nX_mod1 = adata_mod1.layers[\"normalized\"]\nX_mod2 = adata_mod2.layers[\"normalized\"]\n\n# Dimensionality reduction using UMAP\numap_model = umap.UMAP(n_components=50, random_state=42)\nX_integrated_mod1 = umap_model.fit_transform(X_mod1)\nX_integrated_mod2 = umap_model.fit_transform(X_mod2)\n\n# Store the integrated representations\nadata_mod1.obsm[\"X_integrated\"] = X_integrated_mod1\nadata_mod2.obsm[\"X_integrated\"] = X_integrated_mod2\n\n\n# Define the evaluation function\ndef compute_knn_auc(adata_mod1, adata_mod2, k=30):\n    X_mod1 = adata_mod1.obsm[\"X_integrated\"]\n    X_mod2 = adata_mod2.obsm[\"X_integrated\"]\n\n    n_cells = X_mod1.shape[0]\n    knn_mod1 = NearestNeighbors(n_neighbors=k + 1).fit(X_mod1)\n    knn_mod2 = NearestNeighbors(n_neighbors=k + 1).fit(X_mod2)\n\n    neighbors1 = knn_mod1.kneighbors(return_distance=False)[:, 1:]\n    neighbors2 = knn_mod2.kneighbors(return_distance=False)[:, 1:]\n\n    overlaps = [\n        len(set(neighbors1[i]) & set(neighbors2[i])) / k for i in range(n_cells)\n    ]\n    return np.mean(overlaps)\n\n\n# Evaluate the integration\nauc_score = compute_knn_auc(adata_mod1, adata_mod2)\nprint(f\"KNN AUC Score: {auc_score}\")\n\n# Save predictions\nsubmission_df = pd.DataFrame({\"KNN_AUC_Score\": [auc_score]})\nsubmission_df.to_csv(\"./working/submission.csv\", index=False)\n"], "term_out": ["Traceback (most recent call last):\n  File \"runfile.py\", line 17, in <module>\n    X_mod1_cca, X_mod2_cca = cca.fit_transform(X_mod1, X_mod2)\n  File \"/home/aturcan/miniconda3/envs/aide3/lib/python3.10/site-packages/sklearn/utils/_set_output.py\", line 316, in wrapped\n    data_to_wrap = f(self, X, *args, **kwargs)\n  File \"/home/aturcan/miniconda3/envs/aide3/lib/python3.10/site-packages/sklearn/cross_decomposition/_pls.py\", line 498, in fit_transform\n    return self.fit(X, y).transform(X, y)\n  File \"/home/aturcan/miniconda3/envs/aide3/lib/python3.10/site-packages/sklearn/base.py\", line 1365, in wrapper\n    return fit_method(estimator, *args, **kwargs)\n  File \"/home/aturcan/miniconda3/envs/aide3/lib/python3.10/site-packages/sklearn/cross_decomposition/_pls.py\", line 225, in fit\n    X = validate_data(\n  File \"/home/aturcan/miniconda3/envs/aide3/lib/python3.10/site-packages/sklearn/utils/validation.py\", line 2954, in validate_data\n    out = check_array(X, input_name=\"X\", **check_params)\n  File \"/home/aturcan/miniconda3/envs/aide3/lib/python3.10/site-packages/sklearn/utils/validation.py\", line 1012, in check_array\n    array = _ensure_sparse_format(\n  File \"/home/aturcan/miniconda3/envs/aide3/lib/python3.10/site-packages/sklearn/utils/validation.py\", line 611, in _ensure_sparse_format\n    raise TypeError(\nTypeError: Sparse data was passed for X, but dense data is required. Use '.toarray()' to convert to a dense numpy array.\nExecution time: a moment seconds (time limit is an hour).", "KNN AUC Score: 0.05506666666666666\nExecution time: 2 seconds seconds (time limit is an hour).", "KNN AUC Score: 0.05506666666666666\nExecution time: a second seconds (time limit is an hour).", "Traceback (most recent call last):\n  File \"runfile.py\", line 17, in <module>\n    X_combined = cca.fit_transform(rna_data, atac_data)\n  File \"/home/aturcan/miniconda3/envs/aide3/lib/python3.10/site-packages/sklearn/utils/_set_output.py\", line 316, in wrapped\n    data_to_wrap = f(self, X, *args, **kwargs)\n  File \"/home/aturcan/miniconda3/envs/aide3/lib/python3.10/site-packages/sklearn/cross_decomposition/_pls.py\", line 498, in fit_transform\n    return self.fit(X, y).transform(X, y)\n  File \"/home/aturcan/miniconda3/envs/aide3/lib/python3.10/site-packages/sklearn/base.py\", line 1365, in wrapper\n    return fit_method(estimator, *args, **kwargs)\n  File \"/home/aturcan/miniconda3/envs/aide3/lib/python3.10/site-packages/sklearn/cross_decomposition/_pls.py\", line 225, in fit\n    X = validate_data(\n  File \"/home/aturcan/miniconda3/envs/aide3/lib/python3.10/site-packages/sklearn/utils/validation.py\", line 2954, in validate_data\n    out = check_array(X, input_name=\"X\", **check_params)\n  File \"/home/aturcan/miniconda3/envs/aide3/lib/python3.10/site-packages/sklearn/utils/validation.py\", line 1012, in check_array\n    array = _ensure_sparse_format(\n  File \"/home/aturcan/miniconda3/envs/aide3/lib/python3.10/site-packages/sklearn/utils/validation.py\", line 611, in _ensure_sparse_format\n    raise TypeError(\nTypeError: Sparse data was passed for X, but dense data is required. Use '.toarray()' to convert to a dense numpy array.\nExecution time: a moment seconds (time limit is an hour).", "Traceback (most recent call last):\n  File \"runfile.py\", line 17, in <module>\n    X_combined = np.concatenate([X_mod1, X_mod2], axis=0)\nValueError: zero-dimensional arrays cannot be concatenated\nExecution time: a moment seconds (time limit is an hour).", "KNN AUC Score: 0.263\nExecution time: 16 seconds seconds (time limit is an hour)."], "analysis": ["The code execution failed due to a TypeError indicating that sparse data was passed for X, but dense data is required for the CCA method. To fix this, the sparse matrices X_mod1 and X_mod2 should be converted to dense numpy arrays using the `.toarray()` method before applying CCA.", "The script successfully integrated single cell RNA-seq multimodal data using PCA and evaluated the integration quality using a KNN AUC score. The computed KNN AUC score was 0.0551, indicating the degree of overlap in nearest neighbors between the two modalities after integration.", "The code successfully integrated the RNA and ATAC data from the single cell RNA-seq multimodal data into PCA reduced forms and computed the KNN AUC score to evaluate the integration quality. The KNN AUC score obtained is 0.05507, indicating the level of cell type correspondence between the two modalities after integration.", "The code execution failed due to a TypeError indicating that sparse data was passed where dense data is required for the Canonical Correlation Analysis (CCA). To fix this issue, the sparse matrices 'rna_data' and 'atac_data' should be converted to dense arrays using the '.toarray()' method before being passed to the CCA.", "The error 'zero-dimensional arrays cannot be concatenated' occurred because the data extracted from the layers 'normalized' are likely not in the correct format or are empty. To fix this, ensure that the data in 'normalized' layers are proper 2D arrays and are not empty before attempting to concatenate.", "The code successfully performed multimodal data integration for single cell RNA-seq using UMAP for dimensionality reduction and evaluated the integration quality using a KNN AUC score. The computed KNN AUC score was 0.263, indicating the level of overlap in nearest neighbors between the two integrated datasets."], "exp_name": "0-terrestrial-oyster-of-drama", "metrics": [0, 0, 0, 0, 0, 0]}

let lastClick = 0;
let firstFrameTime = undefined;

let nodes = [];
let edges = [];

let lastScrollPos = 0;

setup = () => {
  canvas = createCanvas(...updateTargetDims());
};

class Node {
  x;
  y;
  size;
  xT;
  yT;
  xB;
  yB;
  treeInd;
  color;
  relSize;
  animationStart = Number.MAX_VALUE;
  animationProgress = 0;
  isStatic = false;
  hasChildren = false;
  isRootNode = true;
  isStarred = false;
  selected = false;
  renderSize = 10;
  edges = [];
  bgCol;

  constructor(x, y, relSize, treeInd) {
    const minSize = 35;
    const maxSize = 60;

    const maxColor = 10;
    const minColor = 125;

    this.relSize = relSize;
    this.treeInd = treeInd;
    this.size = minSize + (maxSize - minSize) * relSize;
    this.color = minColor + (maxColor - minColor) * relSize;
    this.bgCol = Math.round(Math.max(this.color / 2, 0));

    this.x = x;
    this.y = y;
    this.xT = x;
    this.yT = y - this.size / 2;
    this.xB = x;
    this.yB = y + this.size / 2;

    nodes.push(this);
  }

  startAnimation = (offset = 0) => {
    if (this.animationStart == Number.MAX_VALUE)
      this.animationStart = globalTime + offset;
  };

  child = (node) => {
    let edge = new Edge(this, node);
    this.edges.push(edge);
    edges.push(edge);
    this.hasChildren = true;
    node.isRootNode = false;
    return node;
  };

  render = () => {
    if (globalTime - this.animationStart < 0) return;

    const mouseXlocalCoords = (mouseX - width / 2) / scaleFactor;
    const mouseYlocalCoords = (mouseY - height / 2) / scaleFactor;
    const isMouseOver =
      dist(mouseXlocalCoords, mouseYlocalCoords, this.x, this.y) <
      this.renderSize / 1.5;
    if (isMouseOver) cursor(HAND);
    if (isMouseOver && mouseIsPressed) {
      nodes.forEach((n) => (n.selected = false));
      this.selected = true;
      setCodeAndPlan(
        treeStructData.code[this.treeInd],
        treeStructData.plan[this.treeInd],
      );
      manualSelection = true;
    }

    this.renderSize = this.size;
    if (!this.isStatic) {
      this.animationProgress = animEase(
        (globalTime - this.animationStart) / 1000,
      );
      if (this.animationProgress >= 1) {
        this.isStatic = true;
      } else {
        this.renderSize =
          this.size *
          (0.8 +
            0.2 *
              (-3.33 * this.animationProgress ** 2 +
                4.33 * this.animationProgress));
      }
    }

    fill(this.color);
    if (this.selected) {
      fill(accentCol);
    }

    noStroke();
    square(
      this.x - this.renderSize / 2,
      this.y - this.renderSize / 2,
      this.renderSize,
      10,
    );

    noStroke();
    textAlign(CENTER, CENTER);
    textSize(this.renderSize / 2);
    fill(255);
    // fill(lerpColor(color(accentCol), color(255), this.animationProgress))
    text("{ }", this.x, this.y - 1);
    // DEBUG PRINT:
    // text(round(this.relSize, 2), this.x, this.y - 1)
    // text(this.treeInd, this.x, this.y + 15)

    const dotAnimThreshold = 0.85;
    if (this.isStarred && this.animationProgress >= dotAnimThreshold) {
      let dotAnimProgress =
        (this.animationProgress - dotAnimThreshold) / (1 - dotAnimThreshold);
      textSize(
        ((-3.33 * dotAnimProgress ** 2 + 4.33 * dotAnimProgress) *
          this.renderSize) /
          2,
      );
      if (this.selected) {
        fill(0);
        stroke(0);
      } else {
        fill(accentCol);
        stroke(accentCol);
      }
      strokeWeight((-(dotAnimProgress ** 2) + dotAnimProgress) * 2);
      text("*", this.x + 20, this.y - 11);
      noStroke();
    }

    if (!this.isStatic) {
      fill(bgCol);
      const progressAnimBaseSize = this.renderSize + 5;
      rect(
        this.x - progressAnimBaseSize / 2,
        this.y -
          progressAnimBaseSize / 2 +
          progressAnimBaseSize * this.animationProgress,
        progressAnimBaseSize,
        progressAnimBaseSize * (1 - this.animationProgress),
      );
    }
    if (this.animationProgress >= 0.9) {
      this.edges
        .sort((a, b) => a.color() - b.color())
        .forEach((e, i) => {
          e.startAnimation((i / this.edges.length) ** 2 * 1000);
        });
    }
  };
}

class Edge {
  nodeT;
  nodeB;
  animX = 0;
  animY = 0;
  animationStart = Number.MAX_VALUE;
  animationProgress = 0;
  isStatic = false;
  weight = 0;

  constructor(nodeT, nodeB) {
    this.nodeT = nodeT;
    this.nodeB = nodeB;
    this.weight = 2 + nodeB.relSize * 1;
  }

  color = () => this.nodeB.color;

  startAnimation = (offset = 0) => {
    if (this.animationStart == Number.MAX_VALUE)
      this.animationStart = globalTime + offset;
  };

  render = () => {
    if (globalTime - this.animationStart < 0) return;

    if (!this.isStatic) {
      this.animationProgress = animEase(
        (globalTime - this.animationStart) / 1000,
      );
      if (this.animationProgress >= 1) {
        this.isStatic = true;
        this.animX = this.nodeB.xT;
        this.animY = this.nodeB.yT;
      } else {
        this.animX = bezierPoint(
          this.nodeT.xB,
          this.nodeT.xB,
          this.nodeB.xT,
          this.nodeB.xT,
          this.animationProgress,
        );

        this.animY = bezierPoint(
          this.nodeT.yB,
          (this.nodeT.yB + this.nodeB.yT) / 2,
          (this.nodeT.yB + this.nodeB.yT) / 2,
          this.nodeB.yT,
          this.animationProgress,
        );
      }
    }
    if (this.animationProgress >= 0.97) {
      this.nodeB.startAnimation();
    }

    strokeWeight(this.weight);
    noFill();
    stroke(
      lerpColor(color(bgCol), color(accentCol), this.nodeB.relSize * 1 + 0.7),
    );
    bezier(
      this.nodeT.xB,
      this.nodeT.yB,
      this.nodeT.xB,
      (this.nodeT.yB + this.nodeB.yT) / 2,
      this.animX,
      (this.nodeT.yB + this.nodeB.yT) / 2,
      this.animX,
      this.animY,
    );
  };
}

draw = () => {
  cursor(ARROW);
  frameRate(120);
  if (!firstFrameTime && frameCount <= 1) {
    firstFrameTime = millis();
  }
  // ---- update global animation state ----
  const initialSpeedScalingEaseIO =
    (cos(min((millis() - firstFrameTime) / 8000, 1.0) * PI) + 1) / 2;
  const initialSpeedScalingEase =
    (cos(min((millis() - firstFrameTime) / 8000, 1.0) ** (1 / 2) * PI) + 1) / 2;
  const initAnimationSpeedFactor = 1.0 - 0.4 * initialSpeedScalingEaseIO;
  // update global scaling-aware clock
  globalTime += globalAnimSpeed * initAnimationSpeedFactor * deltaTime;

  if (nodes.length == 0) {
    const spacingHeight = height * 1.3;
    const spacingWidth = width * 1.3;
    treeStructData.layout.forEach((lay, index) => {
      new Node(
        spacingWidth * lay[0] - spacingWidth / 2,
        20 + spacingHeight * lay[1] - spacingHeight / 2,
        1 - treeStructData.metrics[index],
        index,
      );
    });
    treeStructData.edges.forEach((ind) => {
      nodes[ind[0]].child(nodes[ind[1]]);
    });
    nodes.forEach((n) => {
      if (n.isRootNode) n.startAnimation();
    });
    nodes[0].selected = true;
    setCodeAndPlan(
      treeStructData.code[0],
      treeStructData.plan[0],
    )
  }

  const staticNodes = nodes.filter(
    (n) => n.isStatic || n.animationProgress >= 0.7,
  );
  if (staticNodes.length > 0) {
    const largestNode = staticNodes.reduce((prev, current) =>
      prev.relSize > current.relSize ? prev : current,
    );
    if (!manualSelection) {
      if (!largestNode.selected) {
        setCodeAndPlan(
          treeStructData.code[largestNode.treeInd],
          treeStructData.plan[largestNode.treeInd],
        );
      }
      staticNodes.forEach((node) => {
        node.selected = node === largestNode;
      });
    }
  }
  background(bgCol);
  // global animation transforms
  translate(width / 2, height / 2);
  scale(scaleFactor);

  
  // ---- fg render ----
  edges.forEach((e) => e.render());
  nodes.forEach((n) => n.render());
  
};

    </script>
    <title>AIDE Run Visualization</title>
    <style>
      body,
      * {
        margin: 0;
        padding: 0;
        box-sizing: border-box;
        overflow: scroll;
      }
      body {
        background-color: #f2f0e7;
      }
      canvas {
        float: left;
        height: 100vh;
        width: 40vw;
      }
      #text-container {
        float: right;
        height: 100vh;
        width: 50vw;
        background-color: #282c34;
      }
      #plan {
        /* border-left: 2px solid #282c34; */
        background-color: #282c34;
        color: #f2f0e7;
        min-height: 5rem;
        padding: 1em 0 1em 1em;
      }
    </style>
  </head>
  <body>
    <pre
      id="text-container"
    ><div id="plan"></div><hr><code id="code" class="language-python"></code></pre>
  </body>
</html>
