{
 "cells": [
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# 3D Shape Analysis"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 1,
   "metadata": {},
   "outputs": [],
   "source": [
    "from pipelines import *\n",
    "import os\n",
    "import sys\n",
    "import warnings\n",
    "\n",
    "if not sys.warnoptions:\n",
    "    warnings.simplefilter(\"ignore\")\n",
    "    os.environ[\"PYTHONWARNINGS\"] = \"ignore\" # Also affect subprocesses"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "metadata": {},
   "outputs": [],
   "source": [
    "import numpy as np\n",
    "\n",
    "from scipy import sparse \n",
    "\n",
    "sys.path.append(\"pyhks\")\n",
    "from HKS import *\n",
    "from GeomUtils import *\n",
    "from ripser import ripser\n",
    "\n",
    "import warnings\n",
    "warnings.filterwarnings('ignore')"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "metadata": {},
   "outputs": [],
   "source": [
    "def do0DSublevelsetFiltrationMesh(VPos, ITris, fn):\n",
    "    x = fn(VPos, ITris)\n",
    "    N = VPos.shape[0]\n",
    "    # Add edges between adjacent points in the mesh    \n",
    "    I, J = getEdges(VPos, ITris)\n",
    "    V = np.maximum(x[I], x[J])\n",
    "    # Add vertex birth times along the diagonal of the distance matrix\n",
    "    I = np.concatenate((I, np.arange(N)))\n",
    "    J = np.concatenate((J, np.arange(N)))\n",
    "    V = np.concatenate((V, x))\n",
    "    #Create the sparse distance matrix\n",
    "    D = sparse.coo_matrix((V, (I, J)), shape=(N, N)).tocsr()\n",
    "    return ripser(D, distance_matrix=True, maxdim=0)['dgms'][0]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "metadata": {},
   "outputs": [],
   "source": [
    "fn = lambda VPos, ITris: -getHKS(VPos, ITris, 50, t = 30)"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# Load Meshes"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "array([0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 2, 2,\n",
       "       2, 2, 2, 2, 2, 2, 2, 2, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 4, 4, 4, 4,\n",
       "       4, 4, 4, 4, 4, 4, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 6, 6, 6, 6, 6, 6,\n",
       "       6, 6, 6, 6, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 8, 8, 8, 8, 8, 8, 8, 8,\n",
       "       8, 8])"
      ]
     },
     "execution_count": 5,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "labels = [[i]*10 for i in range(9)]\n",
    "labels = np.array(labels).flatten()\n",
    "\n",
    "labels"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "metadata": {},
   "outputs": [],
   "source": [
    "N = 90\n",
    "meshesNonrigid = []\n",
    "for i in range(N):\n",
    "    (VPos, _, ITris) = loadOffFile(\"shapes_nonrigid/%.3d.off\"%i)\n",
    "    meshesNonrigid.append((VPos, ITris))"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# Compute PDs"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Computing persistence diagrams...\n",
      "Finished first 0 meshes\n",
      "Finished first 10 meshes\n",
      "Finished first 20 meshes\n",
      "Finished first 30 meshes\n",
      "Finished first 40 meshes\n",
      "Finished first 50 meshes\n",
      "Finished first 60 meshes\n",
      "Finished first 70 meshes\n",
      "Finished first 80 meshes\n"
     ]
    }
   ],
   "source": [
    "DGMS = []\n",
    "N = len(meshesNonrigid)\n",
    "print(\"Computing persistence diagrams...\")\n",
    "for i, (VPos, ITris) in enumerate(meshesNonrigid):\n",
    "    if i%10 == 0:\n",
    "        print(\"Finished first %i meshes\"%i)\n",
    "    x = fn(VPos, ITris)\n",
    "    I = do0DSublevelsetFiltrationMesh(VPos, ITris, lambda VPos, ITris: -getHKS(VPos, ITris, 50, t = 20))\n",
    "    I = I[np.isfinite(I[:, 1]), :]\n",
    "    DGMS.append(I)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 8,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "90"
      ]
     },
     "execution_count": 8,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "len(DGMS)"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# Random Forest"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 9,
   "metadata": {},
   "outputs": [],
   "source": [
    "P = [1,2,4,8,10] # Exponent of Persistence\n",
    "N = [5,10,20] # Number of Pixel per Side\n",
    "N_SIGMA = [0.1,1,10,100,1000,10000,100000,1000000] # Size of sigma w.r.t. size of the pixel\n",
    "\n",
    "C = [100,200] #Size of the Forest\n",
    "params = [C]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 10,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "0\n",
      "0.5 {'pix': np.float64(0.001), 'sigma': np.float64(1e-09), 'p': 1, 'model': RandomForestClassifier(n_estimators=200)}\n",
      "1\n",
      "0.5 {'pix': np.float64(0.001), 'sigma': np.float64(1e-09), 'p': 1, 'model': RandomForestClassifier()}\n",
      "2\n",
      "0.4444444444444444 {'pix': np.float64(0.001), 'sigma': np.float64(1e-09), 'p': 1, 'model': RandomForestClassifier()}\n",
      "3\n",
      "0.5555555555555556 {'pix': np.float64(0.01), 'sigma': np.float64(1e-08), 'p': 1, 'model': RandomForestClassifier(n_estimators=200)}\n",
      "4\n",
      "0.7222222222222222 {'pix': np.float64(0.001), 'sigma': np.float64(1e-09), 'p': 1, 'model': RandomForestClassifier()}\n",
      "5\n",
      "0.2777777777777778 {'pix': np.float64(0.001), 'sigma': np.float64(1e-09), 'p': 1, 'model': RandomForestClassifier()}\n",
      "6\n",
      "0.6111111111111112 {'pix': np.float64(0.001), 'sigma': np.float64(1e-09), 'p': 2, 'model': RandomForestClassifier(n_estimators=200)}\n",
      "7\n",
      "0.16666666666666666 {'pix': np.float64(0.001), 'sigma': np.float64(1e-09), 'p': 1, 'model': RandomForestClassifier()}\n",
      "8\n",
      "0.3888888888888889 {'pix': np.float64(0.001), 'sigma': np.float64(1e-09), 'p': 1, 'model': RandomForestClassifier()}\n",
      "9\n",
      "0.4444444444444444 {'pix': np.float64(0.001), 'sigma': np.float64(1e-09), 'p': 1, 'model': RandomForestClassifier(n_estimators=200)}\n",
      "[0.5        0.5        0.44444444 0.55555556 0.72222222 0.27777778\n",
      " 0.61111111 0.16666667 0.38888889 0.44444444] 0.46111111111111114 0.15122952876462445\n"
     ]
    }
   ],
   "source": [
    "N_=10\n",
    "\n",
    "scores_dataset = np.zeros((N_,))\n",
    "\n",
    "for i in range(N_):\n",
    "    print(i)\n",
    "    Pi_Pipe = PiPipeline(DGMS, labels, [params], \n",
    "             MODELS = ['class_tree'],train_split = 0.8, n_splits = 4, \n",
    "             N = N, N_SIGMA = N_SIGMA, P = P)\n",
    "    score = Pi_Pipe.run_analysis()\n",
    "    scores_dataset[i]= score\n",
    "    print(score, Pi_Pipe.best_params)\n",
    "\n",
    "print(scores_dataset,np.mean(scores_dataset),np.std(scores_dataset))\n",
    "np.save('Pi_scores_nonrigid',scores_dataset)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": []
  }
 ],
 "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.13.9"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 4
}
