{
 "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",
    "import scipy.io as sio\n",
    "from scipy import sparse \n",
    "\n",
    "import matplotlib.pyplot as plt\n",
    "from mpl_toolkits.mplot3d import Axes3D\n",
    "\n",
    "sys.path.append(\"pyhks\")\n",
    "from HKS import *\n",
    "from GeomUtils import *\n",
    "\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": "markdown",
   "metadata": {},
   "source": [
    "# Load Meshes"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "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": 4,
     "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": 5,
   "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": 6,
   "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",
    "    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": 7,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "90"
      ]
     },
     "execution_count": 7,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "len(DGMS)"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# Random Forest"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 8,
   "metadata": {},
   "outputs": [],
   "source": [
    "H = [5,10,20,40,50]\n",
    "ITER = [5,10,50,100]\n",
    "\n",
    "C = [100,200] #Size of the Forest\n",
    "\n",
    "params = [C]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 9,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "0\n",
      "0.5 {'h': 20, 'iterations': 50, 'model': RandomForestClassifier(n_estimators=200)}\n",
      "1\n",
      "0.7222222222222222 {'h': 40, 'iterations': 100, 'model': RandomForestClassifier(n_estimators=200)}\n",
      "2\n",
      "0.7222222222222222 {'h': 40, 'iterations': 50, 'model': RandomForestClassifier(n_estimators=200)}\n",
      "3\n",
      "0.3888888888888889 {'h': 50, 'iterations': 50, 'model': RandomForestClassifier(n_estimators=200)}\n",
      "4\n",
      "0.6111111111111112 {'h': 40, 'iterations': 100, 'model': RandomForestClassifier(n_estimators=200)}\n",
      "5\n",
      "0.5555555555555556 {'h': 50, 'iterations': 50, 'model': RandomForestClassifier(n_estimators=200)}\n",
      "6\n",
      "0.5555555555555556 {'h': 40, 'iterations': 10, 'model': RandomForestClassifier()}\n",
      "7\n",
      "0.6111111111111112 {'h': 40, 'iterations': 10, 'model': RandomForestClassifier(n_estimators=200)}\n",
      "8\n",
      "0.4444444444444444 {'h': 20, 'iterations': 50, 'model': RandomForestClassifier(n_estimators=200)}\n",
      "9\n",
      "0.5 {'h': 20, 'iterations': 50, 'model': RandomForestClassifier()}\n",
      "[0.5        0.72222222 0.72222222 0.38888889 0.61111111 0.55555556\n",
      " 0.55555556 0.61111111 0.44444444 0.5       ] 0.5611111111111111 0.10378634273483002\n"
     ]
    }
   ],
   "source": [
    "N=10\n",
    "\n",
    "scores_dataset = np.zeros((N,))\n",
    "\n",
    "for i in range(N):\n",
    "    print(i)\n",
    "    SplPipe = PersSplinesPipeline(DGMS, labels, [params],  \n",
    "             MODELS = ['class_tree'],train_split = 0.8, n_splits = 4, \n",
    "             H = H, ITER = ITER)\n",
    "    score = SplPipe.run_analysis()\n",
    "    scores_dataset[i]= score\n",
    "    print(score, SplPipe.best_params)\n",
    "\n",
    "\n",
    "print(scores_dataset,np.mean(scores_dataset),np.std(scores_dataset))\n",
    "np.save('Splines_scores_nonrigid_SVM',scores_dataset)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "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
}
