{
 "cells": [
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# Example"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 1,
   "metadata": {},
   "outputs": [],
   "source": [
    "import numpy as np\n",
    "import matplotlib.pyplot as plt\n",
    "\n",
    "from persistence_spheres_utils import from_DGMS_to_H, make_weighting, make_PD_grid\n",
    "from persistence_spheres import PerSPhere\n",
    "from sphere_FDA_utils import make_dV\n",
    "\n",
    "import pyshtools\n",
    "\n",
    "import time"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "metadata": {},
   "outputs": [],
   "source": [
    "K = 0.1\n",
    "\n",
    "n_theta = 100\n",
    "n_phi = 200\n",
    "\n",
    "weighting = make_weighting(K)\n",
    "\n",
    "dgm0 = np.array([[0,1],[1,5]])\n",
    "dgm1 = np.array([[0,1],[1,5],[7,10]])\n",
    "\n",
    "PS = PerSPhere([dgm0,dgm1],weighting = weighting,n_theta = n_theta, n_phi = n_phi)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "np.float64(11.617046189904695)"
      ]
     },
     "execution_count": 4,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "# Distance Between Functions\n",
    "\n",
    "dV = make_dV(n_theta,n_phi)\n",
    "np.sum(dV*((PS.H[0]-PS.H[1])**2))**0.5"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "np.float64(11.58129769114426)"
      ]
     },
     "execution_count": 5,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "# Distance Between Vectors\n",
    "\n",
    "np.linalg.norm(PS.sph_armonics[0]-PS.sph_armonics[1])"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# Grid Choice"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 124,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "2.155205249786377 0.020657307364384363 0.009057150996899941 (2, 10, 10)\n"
     ]
    }
   ],
   "source": [
    "np.random.seed(17112025)\n",
    "\n",
    "K = 0.3\n",
    "weighting = make_weighting(K)\n",
    "M = 10\n",
    "n = 2000\n",
    "\n",
    "n_theta = 20\n",
    "n_phi = 40\n",
    "\n",
    "rel_err = []\n",
    "\n",
    "t0 = time.time()\n",
    "\n",
    "for a in range(100):\n",
    "\n",
    "    print(a,'  ', end='\\r')\n",
    "    \n",
    "#    dgm0 = np.random.uniform(-M,M, size=(n,2))\n",
    "    dgm0 = np.random.normal(M, size=(n,2))\n",
    "    dgm0[:,1] = np.abs(dgm0[:,1])+dgm0[:,0]\n",
    "\n",
    "#    dgm1 = np.random.uniform(-M,M, size=(n,2))\n",
    "    dgm1 = np.random.normal(M, size=(n,2))\n",
    "    dgm1[:,1] = np.abs(dgm1[:,1])+dgm1[:,0]\n",
    "\n",
    "    PS = PerSPhere([dgm0,dgm1],weighting = weighting,n_theta = n_theta, n_phi = n_phi)\n",
    "\n",
    "    dV = make_dV(n_theta,n_phi)\n",
    "    d0 = np.sum(dV*((PS.H[1]-PS.H[0])**2))**0.5\n",
    "\n",
    "    d1 = np.linalg.norm(PS.sph_armonics[0]-PS.sph_armonics[1])\n",
    "\n",
    "    rel_err.append(np.abs(d0-d1)/d0)\n",
    "\n",
    "t1 = time.time()\n",
    "\n",
    "A0 = np.mean(rel_err)\n",
    "\n",
    "\n",
    "print('Time: ',t1-t0,'Relative Error: ',np.mean(rel_err),'Standard Deviation of Error: ',np.std(rel_err),'Shape: ',PS.coeff_shape)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 125,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "3.791030168533325 0.01173164112316129 0.005011145135408121 (2, 17, 17)\n"
     ]
    }
   ],
   "source": [
    "np.random.seed(17112025)\n",
    "\n",
    "K = 0.3\n",
    "weighting = make_weighting(K)\n",
    "M = 10\n",
    "n = 2000\n",
    "\n",
    "n_theta = 34\n",
    "n_phi = 68\n",
    "\n",
    "rel_err = []\n",
    "\n",
    "t0 = time.time()\n",
    "\n",
    "for a in range(100):\n",
    "\n",
    "    print(a,'  ', end='\\r')\n",
    "    \n",
    "#    dgm0 = np.random.uniform(-M,M, size=(n,2))\n",
    "    dgm0 = np.random.normal(M, size=(n,2))\n",
    "    dgm0[:,1] = np.abs(dgm0[:,1])+dgm0[:,0]\n",
    "\n",
    "#    dgm1 = np.random.uniform(-M,M, size=(n,2))\n",
    "    dgm1 = np.random.normal(M, size=(n,2))\n",
    "    dgm1[:,1] = np.abs(dgm1[:,1])+dgm1[:,0]\n",
    "\n",
    "    PS = PerSPhere([dgm0,dgm1],weighting = weighting,n_theta = n_theta, n_phi = n_phi)\n",
    "\n",
    "    dV = make_dV(n_theta,n_phi)\n",
    "    d0 = np.sum(dV*((PS.H[1]-PS.H[0])**2))**0.5\n",
    "\n",
    "    d1 = np.linalg.norm(PS.sph_armonics[0]-PS.sph_armonics[1])\n",
    "\n",
    "    rel_err.append(np.abs(d0-d1)/d0)\n",
    "\n",
    "t1 = time.time()\n",
    "\n",
    "A0 = np.mean(rel_err)\n",
    "\n",
    "\n",
    "print('Time: ',t1-t0,'Relative Error: ',np.mean(rel_err),'Standard Deviation of Error: ',np.std(rel_err),'Shape: ',PS.coeff_shape)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 126,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "78.42054295539856 0.011478384448155918 0.015072799707898262 (2, 20, 20)\n"
     ]
    }
   ],
   "source": [
    "np.random.seed(17112025)\n",
    "\n",
    "K = 0.3\n",
    "weighting = make_weighting(K)\n",
    "M = 10\n",
    "n = 2000\n",
    "\n",
    "n_theta = 40\n",
    "n_phi = 80\n",
    "\n",
    "rel_err = []\n",
    "\n",
    "t0 = time.time()\n",
    "\n",
    "for a in range(2000):\n",
    "\n",
    "    print(a,'  ', end='\\r')\n",
    "    \n",
    "#    dgm0 = np.random.uniform(-M,M, size=(n,2))\n",
    "    dgm0 = np.random.normal(M, size=(n,2))\n",
    "    dgm0[:,1] = np.abs(dgm0[:,1])+dgm0[:,0]\n",
    "\n",
    "#    dgm1 = np.random.uniform(-M,M, size=(n,2))\n",
    "    dgm1 = np.random.normal(M, size=(n,2))\n",
    "    dgm1[:,1] = np.abs(dgm1[:,1])+dgm1[:,0]\n",
    "\n",
    "    PS = PerSPhere([dgm0,dgm1],weighting = weighting,n_theta = n_theta, n_phi = n_phi)\n",
    "\n",
    "    dV = make_dV(n_theta,n_phi)\n",
    "    d0 = np.sum(dV*((PS.H[1]-PS.H[0])**2))**0.5\n",
    "\n",
    "    d1 = np.linalg.norm(PS.sph_armonics[0]-PS.sph_armonics[1])\n",
    "\n",
    "    rel_err.append(np.abs(d0-d1)/d0)\n",
    "\n",
    "t1 = time.time()\n",
    "\n",
    "A1 = np.mean(rel_err)\n",
    "\n",
    "print('Time: ',t1-t0,'Relative Error: ',np.mean(rel_err),'Standard Deviation of Error: ',np.std(rel_err),'Shape: ',PS.coeff_shape)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 116,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "113.79679083824158 0.009109858912273141 0.012878306808069808 (1250,)\n"
     ]
    }
   ],
   "source": [
    "np.random.seed(17112025)\n",
    "\n",
    "K = 0.3\n",
    "weighting = make_weighting(K)\n",
    "M = 10\n",
    "n = 2000\n",
    "\n",
    "n_theta = 50\n",
    "n_phi = 100\n",
    "\n",
    "rel_err = []\n",
    "\n",
    "t0 = time.time()\n",
    "\n",
    "for a in range(2000):\n",
    "\n",
    "    print(a,'  ', end='\\r')\n",
    "    \n",
    "#    dgm0 = np.random.uniform(-M,M, size=(n,2))\n",
    "    dgm0 = np.random.normal(M, size=(n,2))\n",
    "    dgm0[:,1] = np.abs(dgm0[:,1])+dgm0[:,0]\n",
    "\n",
    "#    dgm1 = np.random.uniform(-M,M, size=(n,2))\n",
    "    dgm1 = np.random.normal(M, size=(n,2))\n",
    "    dgm1[:,1] = np.abs(dgm1[:,1])+dgm1[:,0]\n",
    "\n",
    "    PS = PerSPhere([dgm0,dgm1],weighting = weighting,n_theta = n_theta, n_phi = n_phi)\n",
    "\n",
    "    dV = make_dV(n_theta,n_phi)\n",
    "    d0 = np.sum(dV*((PS.H[1]-PS.H[0])**2))**0.5\n",
    "\n",
    "    d1 = np.linalg.norm(PS.sph_armonics[0]-PS.sph_armonics[1])\n",
    "\n",
    "    rel_err.append(np.abs(d0-d1)/d0)\n",
    "\n",
    "t1 = time.time()\n",
    "\n",
    "A2 = np.mean(rel_err)\n",
    "\n",
    "\n",
    "print('Time: ',t1-t0,'Relative Error: ',np.mean(rel_err),'Standard Deviation of Error: ',np.std(rel_err),'Shape: ',PS.coeff_shape)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 117,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "355.56140089035034 0.004571812118983379 1.5871223187578883e-05 (2450,)\n"
     ]
    }
   ],
   "source": [
    "np.random.seed(17112025)\n",
    "\n",
    "K = 0.3\n",
    "weighting = make_weighting(K)\n",
    "M = 10\n",
    "n = 2000\n",
    "\n",
    "n_theta = 70\n",
    "n_phi = 140\n",
    "\n",
    "rel_err = []\n",
    "\n",
    "for a in range(2000):\n",
    "\n",
    "    print(a,'  ', end='\\r')\n",
    "    \n",
    "    dgm0 = np.random.uniform(-M,M, size=(n,2))\n",
    "#    dgm0 = np.random.normal(M, size=(n,2))\n",
    "    dgm0[:,1] = np.abs(dgm0[:,1])+dgm0[:,0]\n",
    "\n",
    "#    dgm1 = np.random.uniform(-M,M, size=(n,2))\n",
    "    dgm1 = np.random.normal(M, size=(n,2))\n",
    "    dgm1[:,1] = np.abs(dgm1[:,1])+dgm1[:,0]\n",
    "\n",
    "    PS = PerSPhere([dgm0,dgm1],weighting = weighting,n_theta = n_theta, n_phi = n_phi)\n",
    "\n",
    "    dV = make_dV(n_theta,n_phi)\n",
    "    d0 = np.sum(dV*((PS.H[1]-PS.H[0])**2))**0.5\n",
    "\n",
    "    d1 = np.linalg.norm(PS.sph_armonics[0]-PS.sph_armonics[1])\n",
    "\n",
    "    rel_err.append(np.abs(d0-d1)/d0)\n",
    "\n",
    "t1 = time.time()\n",
    "\n",
    "A3 = np.mean(rel_err)\n",
    "\n",
    "\n",
    "print('Time: ',t1-t0,'Relative Error: ',np.mean(rel_err),'Standard Deviation of Error: ',np.std(rel_err),'Shape: ',PS.coeff_shape)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 118,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "817.1774468421936 0.0032400071982941255 9.344762910845902e-06 (5000,)\n"
     ]
    }
   ],
   "source": [
    "np.random.seed(17112025)\n",
    "\n",
    "K = 0.3\n",
    "weighting = make_weighting(K)\n",
    "M = 10000\n",
    "n = 2000\n",
    "\n",
    "n_theta = 100\n",
    "n_phi = 200\n",
    "\n",
    "rel_err = []\n",
    "\n",
    "for a in range(2000):\n",
    "\n",
    "    print(a,'  ', end='\\r')\n",
    "    \n",
    "    dgm0 = np.random.uniform(-M,M, size=(n,2))\n",
    "#    dgm0 = np.random.normal(M, size=(n,2))\n",
    "    dgm0[:,1] = np.abs(dgm0[:,1])+dgm0[:,0]\n",
    "\n",
    "#    dgm1 = np.random.uniform(-M,M, size=(n,2))\n",
    "    dgm1 = np.random.normal(M, size=(n,2))\n",
    "    dgm1[:,1] = np.abs(dgm1[:,1])+dgm1[:,0]\n",
    "\n",
    "    PS = PerSPhere([dgm0,dgm1],weighting = weighting,n_theta = n_theta, n_phi = n_phi)\n",
    "\n",
    "    dV = make_dV(n_theta,n_phi)\n",
    "    d0 = np.sum(dV*((PS.H[1]-PS.H[0])**2))**0.5\n",
    "\n",
    "    d1 = np.linalg.norm(PS.sph_armonics[0]-PS.sph_armonics[1])\n",
    "\n",
    "    rel_err.append(np.abs(d0-d1)/d0)\n",
    "\n",
    "t1 = time.time()\n",
    "\n",
    "A4 = np.mean(rel_err)\n",
    "\n",
    "print('Time: ',t1-t0,'Relative Error: ',np.mean(rel_err),'Standard Deviation of Error: ',np.std(rel_err),'Shape: ',PS.coeff_shape)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 119,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "[<matplotlib.lines.Line2D at 0x31f598e10>]"
      ]
     },
     "execution_count": 119,
     "metadata": {},
     "output_type": "execute_result"
    },
    {
     "data": {
      "image/png": "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",
      "text/plain": [
       "<Figure size 640x480 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "#Plotting Grid Size vs Relative Error\n",
    "\n",
    "plt.plot([30,40,50,70,100],[A0,A1,A2,A3,A4])"
   ]
  },
  {
   "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
}
