{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 43,
   "id": "df9452a1",
   "metadata": {
    "tags": []
   },
   "outputs": [],
   "source": [
    "import numpy as np\n",
    "import matplotlib.pyplot as plt\n",
    "from scipy import ndimage\n",
    "import os\n",
    "import sys\n",
    "import sys\n",
    "from tqdm.notebook import tqdm\n",
    "import time\n",
    "import copy\n",
    "import argparse\n",
    "import trimesh\n",
    "import logging\n",
    "import math\n",
    "\n",
    "import torch\n",
    "import torchvision\n",
    "import torch.nn as nn\n",
    "import torch.optim.lr_scheduler as lr_scheduler\n",
    "\n",
    "sys.path.insert(0, os.path.abspath('..'))\n",
    "from incode_experiments import utils, volutils\n",
    "from incode_experiments import dataio\n",
    "from modules import INR"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "a756db7e-f986-4daa-9183-d8ef64ede1b0",
   "metadata": {},
   "source": [
    "#### This notebook is based on the implementation from INCODE: https://github.com/xmindflow/INCODE/"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 44,
   "id": "e7de91a9",
   "metadata": {
    "tags": []
   },
   "outputs": [],
   "source": [
    "parser = argparse.ArgumentParser(description='LG_SIREN')\n",
    "\n",
    "# Shared Parameters\n",
    "parser.add_argument('--input',type=str, default='../data/preprocessed_lucy.npy', help='Input image path')\n",
    "parser.add_argument('--output',type=str, default='./output/', help='Output path')\n",
    "parser.add_argument('--inr_model', type=str, default='lg_siren', help='[siren, lg_siren, lc_siren]')\n",
    "parser.add_argument('--lr',type=float, default=1e-4, help='Learning rate')\n",
    "parser.add_argument('--using_schedular', type=bool, default=True, help='Whether to use schedular')\n",
    "parser.add_argument('--scheduler_b', type=float, default=0.2, help='Learning rate scheduler')\n",
    "parser.add_argument('--maxpoints', type=int, default=1e5, help='Batch size')\n",
    "parser.add_argument('--niters', type=int, default=20, help='Number if epochs')\n",
    "parser.add_argument('--steps_til_summary', type=int, default=1, help='Number of steps till summary visualization')\n",
    "parser.add_argument('--res', type=int, default=512, help='resolution (N^3) of the mesh, same for xyz')\n",
    "parser.add_argument('--mcubes_thres', type=float, default=0.5, help='Threshold for marching cubes')\n",
    "parser.add_argument('--downsample', type=int, nargs='+', default=(8, 8, 8), help='Partition factors for Local-Global SIREN')\n",
    "\n",
    "args = parser.parse_args(args=[])\n",
    "\n",
    "groups = np.prod(args.downsample) if 'lg' in args.inr_model or 'lc' in args.inr_model else 1\n",
    "args.downsample = tuple(args.downsample) if 'lg' in args.inr_model or 'lc' in args.inr_model else (1, 1, 1)\n",
    "\n",
    "device = torch.device(\"cuda\" if torch.cuda.is_available() else \"cpu\")"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "c268067e",
   "metadata": {},
   "source": [
    "## Loading Data"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 45,
   "id": "c5d16c2f",
   "metadata": {
    "tags": []
   },
   "outputs": [],
   "source": [
    "## Loading .npy file from the pre-processing code\n",
    "data = np.load(args.input, allow_pickle=True)[()]\n",
    "\n",
    "### Loading input im with the shape of [res, res, res]\n",
    "scale = 1.0\n",
    "im = np.unpackbits(data['im']).reshape(args.res, args.res, args.res)[..., None].astype(np.float32)[...,0]\n",
    "im = ndimage.zoom(im/im.max(), [scale, scale, scale], order=0)\n",
    "\n",
    "hidx, widx, tidx = np.where(im > 0.9)\n",
    "H, W, T = im.shape\n",
    "\n",
    "### Loading gt_im with the shape of [128, 128, 128] to input into the task-specific model\n",
    "gt_im = np.unpackbits(data['gt_im']).reshape(128, 128, 128)[..., None].astype(np.float32)[...,0]\n",
    "gt_im = ndimage.zoom(gt_im/gt_im.max(), [1.0, 1.0, 1.0], order=0)\n",
    "hidx, widx, tidx = np.where(gt_im > 0.99)\n",
    "gt_im = gt_im[hidx.min():hidx.max(),\n",
    "                widx.min():widx.max(),\n",
    "                tidx.min():tidx.max()]\n",
    "gt_im = gt_im[None, None, ...]\n",
    "gt_im = np.repeat(gt_im, 3, axis=1)\n",
    "gt_im = torch.from_numpy(gt_im).to(device)\n",
    "\n",
    "### loading the initial pose of mesh \n",
    "mesh_whl = data['mesh_whl']"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 46,
   "id": "d216e647-339b-4e58-815c-223e1ecf90b5",
   "metadata": {
    "tags": []
   },
   "outputs": [
    {
     "data": {
      "text/plain": [
       "(512, 512, 512)"
      ]
     },
     "execution_count": 46,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "H, W, T"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 47,
   "id": "9042ee67-f22d-4f4b-a00d-0c7f39906cbd",
   "metadata": {
    "tags": []
   },
   "outputs": [],
   "source": [
    "# Generate coordinate grid\n",
    "coords, _ = dataio.partition_signal(utils.get_coords(H, W, T, dim=3), (H, W, T), args.downsample)\n",
    "coords = coords.cuda()\n",
    "\n",
    "# Convert input image to a tensor and reshape\n",
    "gt_raw = torch.tensor(im).reshape(H * W * T, 1)[None, ...].to(device)\n",
    "gt, gt_slices = dataio.partition_signal(torch.tensor(im).reshape(H * W * T, 1), (H, W, T), args.downsample)\n",
    "gt = gt.cuda()\n",
    "\n",
    "if 'lg' in args.inr_model or 'lc' in args.inr_model:\n",
    "    selected_indices = torch.where((gt.squeeze().abs().sum(dim=1) > 0))[0]\n",
    "    coords_selected = coords[selected_indices]\n",
    "    gt_selected = gt[selected_indices]\n",
    "    groups = gt_selected.shape[0]\n",
    "else:\n",
    "    selected_indices = torch.where((gt.abs().sum(dim=1) > 0))[0]\n",
    "    coords_selected = coords\n",
    "    gt_selected = gt"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "53ab2563",
   "metadata": {},
   "source": [
    "## Defining Model"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "36c1145d",
   "metadata": {
    "tags": []
   },
   "source": [
    "### Model Configureations"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 48,
   "id": "cac00936-e10d-4c53-840c-c2ea310aa56e",
   "metadata": {
    "tags": []
   },
   "outputs": [],
   "source": [
    "def count_parameters(model):\n",
    "    return sum(p.numel() for p in model.parameters() if p.requires_grad)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 49,
   "id": "22d602b9",
   "metadata": {
    "tags": []
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Total Parameters: 199174\n",
      "Global Parameters: 19458\n",
      "Local Parameters: 179716\n"
     ]
    }
   ],
   "source": [
    "if args.inr_model == 'lg_siren':\n",
    "    model = INR(out_features=1, in_features=3, activation_type=\"sine\", mode=\"lg\", num_hidden_layers=3,\n",
    "                sidelength=512, hidden_features=groups * 17, groups=groups,\n",
    "                agg_type=\"concat_and_fc\", global_hidden_features=76).cuda()\n",
    "    print(\"Total Parameters:\", count_parameters(model.net.agg_func) + count_parameters(model.net.global_net) + count_parameters(model.net.net))\n",
    "    print(\"Global Parameters:\", count_parameters(model.net.agg_func) + count_parameters(model.net.global_net))\n",
    "    print(\"Local Parameters:\", count_parameters(model.net.net))\n",
    "elif args.inr_model == 'lc_siren':\n",
    "    model = INR(out_features=1, in_features=3, activation_type=\"sine\", mode=\"lc\", num_hidden_layers=3,\n",
    "                sidelength=512, hidden_features=groups * 18,\n",
    "                groups=groups).cuda()\n",
    "    count_parameters(model.net.net)\n",
    "    print(\"Total Parameters:\", count_parameters(model.net.net))\n",
    "else:\n",
    "    model = INR(out_features=1, in_features=3, activation_type=\"sine\", mode=\"mlp\", num_hidden_layers=3,\n",
    "                sidelength=512, hidden_features=256, groups=groups).cuda()\n",
    "    count_parameters(model.net.net)\n",
    "    print(\"Total Parameters:\", count_parameters(model.net.net))"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "db268c3d",
   "metadata": {},
   "source": [
    "## Training Code"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 50,
   "id": "6f58b062",
   "metadata": {
    "tags": []
   },
   "outputs": [],
   "source": [
    "# Optimizer setup\n",
    "optim = torch.optim.Adam(lr=args.lr, params=model.parameters())\n",
    "scheduler = lr_scheduler.LambdaLR(optim, lambda x: args.scheduler_b ** min(x / args.niters, 1))\n",
    "\n",
    "# Initialize lists for IOU and best loss value as positive infinity\n",
    "iou_values = []\n",
    "best_iou = torch.tensor(float('inf'))\n",
    "\n",
    "# Check the args.maxpoints value\n",
    "args.maxpoints = int(args.maxpoints)\n",
    "\n",
    "maxpoints = int(math.ceil(min(H*W*T, args.maxpoints) / groups))"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "ed6f4920-5283-49ec-99b6-5d1085dadcca",
   "metadata": {
    "tags": []
   },
   "source": [
    "### Training Loop"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 51,
   "id": "a452e7cd",
   "metadata": {
    "tags": []
   },
   "outputs": [
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "1511afe9824a41d982ccebbee7fead7b",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/20 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "a816f84481ee4376b2d2079f7a8a3707",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/1343 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "2acd74005f9045e5a66d93e015dd74b5",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/1343 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Epoch: 1 | Total Loss: 0.00525 | IoU: 0.9779\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "5d0fbb1c19724446ac4b01474b5efb9e",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/1343 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Epoch: 2 | Total Loss: 0.00434 | IoU: 0.9811\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "99871f9dc51e473ab7c563c4185076f6",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/1343 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Epoch: 3 | Total Loss: 0.00386 | IoU: 0.9823\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "97c80f0f729346afba860b1dce3ff14b",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/1343 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Epoch: 4 | Total Loss: 0.00353 | IoU: 0.9842\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "de7557c3e99a425e88ed3e5bf305218e",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/1343 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Epoch: 5 | Total Loss: 0.00326 | IoU: 0.9848\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "bbd1e1f6feef436e96656752e420e5c0",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/1343 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Epoch: 6 | Total Loss: 0.00306 | IoU: 0.9855\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "fcad3391338844d1846949a3efa41ed9",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/1343 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Epoch: 7 | Total Loss: 0.00288 | IoU: 0.9863\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "0e41da33a56d4aa39a3b42e9572ad06b",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/1343 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Epoch: 8 | Total Loss: 0.00272 | IoU: 0.9871\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "365c5fe4e4954ebc936f5d0b6615b1b8",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/1343 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Epoch: 9 | Total Loss: 0.00258 | IoU: 0.9880\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "9ee82db8f7c4427d885a3915285b0a38",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/1343 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Epoch: 10 | Total Loss: 0.00245 | IoU: 0.9883\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "288b0dbe17f14774801e80eea7d7daf2",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/1343 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Epoch: 11 | Total Loss: 0.00233 | IoU: 0.9890\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "b10ab570d5a54468ae3c963b6b78ccba",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/1343 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Epoch: 12 | Total Loss: 0.00224 | IoU: 0.9894\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "f96905fb995c4c9a83b492587abbd141",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/1343 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Epoch: 13 | Total Loss: 0.00214 | IoU: 0.9902\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "83c8530831434a2abcd5489fb4955403",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/1343 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Epoch: 14 | Total Loss: 0.00205 | IoU: 0.9904\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "cfab6379b6604d0a9559346e92c47e1a",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/1343 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Epoch: 15 | Total Loss: 0.00197 | IoU: 0.9908\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "54f2c2e228724e629ae00441105e502e",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/1343 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Epoch: 16 | Total Loss: 0.00190 | IoU: 0.9910\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "bcc8105a36c5485e94c0bbda3dd30e66",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/1343 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Epoch: 17 | Total Loss: 0.00183 | IoU: 0.9916\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "923ef83ee84f4348b96c348572c1eb4c",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/1343 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Epoch: 18 | Total Loss: 0.00177 | IoU: 0.9920\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "3d23be3089c14df6ad1e76ebbd364a12",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/1343 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Epoch: 19 | Total Loss: 0.00170 | IoU: 0.9922\n",
      "--------------------\n",
      "Max IoU: 0.9921911358833313\n",
      "--------------------\n"
     ]
    }
   ],
   "source": [
    "for step in tqdm(range(args.niters)):\n",
    "    loss_values = []\n",
    "    for b_idx in tqdm(range(0, H*W*T, args.maxpoints)):\n",
    "        coord_idx = torch.randint(0, coords_selected.shape[1], (groups, maxpoints)).cuda()\n",
    "        b_data = torch.gather(gt_selected, dim=1, index=coord_idx.unsqueeze(-1).expand(-1, -1, gt_selected.shape[-1]))\n",
    "        b_coords = torch.gather(coords_selected, dim=1, index=coord_idx.unsqueeze(-1).expand(-1, -1, coords_selected.shape[-1]))\n",
    "        \n",
    "        # Calculate model output\n",
    "        model_output = model({'coords': b_coords.unsqueeze(0)})['model_out']\n",
    "        \n",
    "        # Calculate the output loss\n",
    "        output_loss = ((model_output - b_data)**2).mean()\n",
    "        \n",
    "        loss = output_loss\n",
    "        loss_values.append(loss.item())\n",
    "\n",
    "        # Perform backpropagation and update model parameters\n",
    "        optim.zero_grad()\n",
    "        loss.backward()\n",
    "        optim.step()\n",
    "    \n",
    "    # Adjust learning rate using a scheduler if applicable\n",
    "    if args.using_schedular:\n",
    "        if args.inr_model == 'incode' and 30 < step:\n",
    "            scheduler.step()\n",
    "        else:\n",
    "            scheduler.step()\n",
    "    \n",
    "    if step >= 1:\n",
    "        # Evaluation step\n",
    "        with torch.no_grad():\n",
    "            rec = torch.zeros_like(gt_selected)\n",
    "            eval_step_size = 10000\n",
    "            for i in range(0, coords_selected.shape[1], eval_step_size):\n",
    "                b_coords = coords_selected[:, i:i+eval_step_size, :]\n",
    "                model_output = model({'coords': b_coords.unsqueeze(0)})['model_out']\n",
    "                rec[:, i:i+eval_step_size,:] = model_output\n",
    "            \n",
    "            rec_full = torch.zeros_like(gt)\n",
    "            rec_full[selected_indices] = rec\n",
    "            rec_full = dataio.lin2vid(rec_full, (H, W, T, 1), gt_slices).reshape(1, H * W * T, 1)\n",
    "\n",
    "        # Calculate IOU\n",
    "        with torch.no_grad():\n",
    "            iou = volutils.get_IoU(rec_full, gt_raw, args.mcubes_thres)\n",
    "            iou_values.append(iou.item())\n",
    "\n",
    "        # Prepare reconstructed shape for visualization\n",
    "        imrec = rec_full[0, ...].reshape(H, W, T).detach().cpu().numpy()\n",
    "\n",
    "        # Check if the current iteration's loss is the best so far\n",
    "        if (iou < best_iou) or (step == 0):\n",
    "            best_iou = iou\n",
    "            best_img = copy.deepcopy(imrec)\n",
    "\n",
    "        # Display intermediate results at specified intervals\n",
    "        if step % args.steps_til_summary == 0:\n",
    "            print(\"Epoch: {} | Total Loss: {:.5f} | IoU: {:.4f}\".format(step, \n",
    "                                                                         np.mean(loss_values),\n",
    "                                                                         iou.item()))\n",
    "\n",
    "        \n",
    "# Print maximum PSNR achieved during training\n",
    "print('--------------------')\n",
    "print('Max IoU:', max(iou_values))\n",
    "print('--------------------')\n",
    "\n",
    "\n",
    "# Marching and saving volumes\n",
    "expname = os.path.splitext(os.path.basename(args.input))[0]\n",
    "os.makedirs(args.output + args.inr_model, exist_ok=True)\n",
    "savename = f'{args.output}/{args.inr_model}/{expname}.dae'\n",
    "volutils.march_and_save(best_img, mesh_whl, args.mcubes_thres, savename, True)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 52,
   "id": "0942bcf9",
   "metadata": {
    "tags": []
   },
   "outputs": [],
   "source": [
    "# Marching and saving volumes\n",
    "expname = os.path.splitext(os.path.basename(args.input))[0]\n",
    "os.makedirs(args.output + args.inr_model, exist_ok=True)\n",
    "savename = f'{args.output}/{args.inr_model}/{expname}_{args.inr_model}_{\"_\".join([str(d) for d in args.downsample])}_{max(iou_values):.5f}.dae'\n",
    "volutils.march_and_save(best_img, mesh_whl, args.mcubes_thres, savename, True)"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "c86e4b9e",
   "metadata": {},
   "source": [
    "# Convergance Rate"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 53,
   "id": "b1dc2a55",
   "metadata": {
    "tags": []
   },
   "outputs": [
    {
     "data": {
      "image/png": "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jhyhRYsW9tcNGjTA09MTAB8fHxITEx36MAyDrKwsh7NbhUwmk8Mg8X/84x907NiR8ePHM2nSJL755htiY2M1QFvExehfpIi4hG3bttGpUycA9uzZQ6tWrS6rv++//57k5GT7a4vFYv85JSUFDw8PrrnmGrp37w7ADz/84LD9yZMn7XMPtWvXjk8++YTMzEz7+iVLljicDbqQzz//nNatW7NixQrmz5/PTz/9xJ9//nmpuyYi5URFkYi4hJMnTxIUFATYCqTCW/MvJC8vr9hLaykpKcyfP59GjRrZl23bts3+8yeffMLDDz9McHAwPXr0YNCgQfz73/8mOzsbgD/++AN3d3dGjhwJwMSJEzlx4gR9+/YlJiaGF198kQ0bNlCvXj3AVnBZrdYicRQWYm+99RapqakADBkyBB8fH5o0aVKatIhIBdLlMxFxOqvVar9UBbYCJjIy8rztT58+zaJFi1i1ahWnT5/mjjvuoHbt2gBkZWWxZs0aoqKiHLb5/fffefHFFzl27Bj+/v689tpr9nWLFi1i3LhxhIeHc/XVV1NQUMC6devs45Buvvlm3n33XaZPn84zzzzDvffey+zZswH4v//7P7Zu3UpSUhLffvstjRs3tl/2mzlzJs888wwpKSl07tyZoUOHcvr0aZYtW2YvAEXEdZgM4xJnRxMRqQQOHDhA8+bNWbduHeHh4c4OR0RcmC6fiYiIiKCiSESquIKCAoCLuq1fRKonFUUiUmXt3bvXPvbn3XffdZhQUUTk7zSmSERERASdKRIREREBVBSJiIiIACqKRERERAAVRSIiIiKAiiIRERERQEWRiIiICKCiSERERARQUSQiIiICqCgSERERAeD/AZKTBOuokOOsAAAAAElFTkSuQmCC",
      "text/plain": [
       "<Figure size 640x480 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "font = {'font': 'Times New Roman', 'size': 12}\n",
    "\n",
    "plt.figure()\n",
    "axfont = {'family' : 'Times New Roman', 'weight' : 'regular', 'size'   : 10}\n",
    "plt.rc('font', **axfont)\n",
    "\n",
    "plt.plot(np.arange(len(iou_values[:-1])), iou_values[:-1], label = f\"{(args.inr_model).upper()}\")\n",
    "plt.xlabel('# Epochs', fontdict=font)\n",
    "plt.ylabel('IoU', fontdict=font)\n",
    "plt.title('Shape Representation', fontdict={'family': 'Times New Roman', 'size': 12, 'weight': 'bold'})\n",
    "plt.legend()\n",
    "plt.grid(True, color='lightgray')\n",
    "\n",
    "plt.show()"
   ]
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "Python 3 (ipykernel)",
   "language": "python",
   "name": "python3"
  },
  "language_info": {
   "codemirror_mode": {
    "name": "ipython",
    "version": 3
   },
   "file_extension": ".py",
   "mimetype": "text/x-python",
   "name": "python",
   "nbconvert_exporter": "python",
   "pygments_lexer": "ipython3",
   "version": "3.10.14"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 5
}
