{
 "cells": [
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Details\n",
    "\n",
    "The notebook contains the synthetic data experiment with high noise (sigma^2 = 0.5). Due to the non-convexity of the HNCPD task, results will vary run to run, so the results from this notebook will not exactly match those in the paper."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 1,
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "Using numpy backend.\n"
     ]
    }
   ],
   "source": [
    "# loading packages and functions\n",
    "import sys\n",
    "\n",
    "sys.path.append(\"./src\")\n",
    "import torch\n",
    "import numpy as np\n",
    "from matplotlib import pyplot as plt\n",
    "\n",
    "from NNCPD import NNCPD, weights_H, Energy_Loss_Tensor, Recon_Loss, L21_Norm, outer_product, outer_product_np, PTF, random_NNCPD, Fro_Norm\n",
    "from lsqnonneg_module import LsqNonneg\n",
    "from trainNNCPD import train\n",
    "#\n",
    "import torch.nn as nn\n",
    "from torch.autograd import Variable\n",
    "from writer import Writer\n",
    "\n",
    "from sklearn.decomposition import NMF\n",
    "\n",
    "import tensorly as tl\n",
    "from tensorly import unfold as tl_unfold\n",
    "from tensorly.decomposition import parafac, non_negative_parafac"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Generate Data Tensor"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "metadata": {},
   "outputs": [],
   "source": [
    "## set the network parameters\n",
    "torch.set_default_tensor_type(torch.DoubleTensor)\n",
    "\n",
    "\n",
    "n1 = 40\n",
    "n2 = 40\n",
    "n3 = 40\n",
    "\n",
    "r = 12\n",
    "\n",
    "\n",
    "a = 1\n",
    "b = 3\n",
    "\n",
    "X = np.zeros((40,40,40))\n",
    "\n",
    "\n",
    "X[0:10,0:15,0:15] = a * np.ones((10,15,15))\n",
    "X[10:25,15:25,15:30] = a * np.ones((15,10,15))\n",
    "X[25:40,25:40,30:40] = a * np.ones((15,15,10))\n",
    "\n",
    "X[0:5,0:10,0:5] = b\n",
    "X[5:10,10:15,5:15] = b\n",
    "\n",
    "X[10:15,15:18,15:20] = b\n",
    "X[15:20,18:23,20:25] = b\n",
    "X[20:25,23:25,25:30] = b\n",
    "\n",
    "X[25:30,25:35,30:35] = b\n",
    "X[30:40,35:40,35:40] = b\n",
    "\n",
    "#add Gaussian Noise\n",
    "np.random.seed(0)\n",
    "X = X + 0.5 * np.abs(np.random.randn(40, 40, 40))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "metadata": {},
   "outputs": [],
   "source": [
    "#set a consistent vmin and vmax for visualization\n",
    "vmin=0\n",
    "vmax=3"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "### Visualize Original Data Tensor"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "metadata": {
    "scrolled": true
   },
   "outputs": [
    {
     "data": {
      "image/png": "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\n",
      "text/plain": [
       "<Figure size 864x360 with 3 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "fig, axs = plt.subplots(1, 3, constrained_layout=True, figsize=(12,5))\n",
    "axs[0].axes.get_xaxis().set_ticks([])\n",
    "axs[0].axes.get_yaxis().set_ticks([])\n",
    "axs[1].axes.get_xaxis().set_ticks([])\n",
    "axs[1].axes.get_yaxis().set_ticks([])\n",
    "axs[2].axes.get_xaxis().set_ticks([])\n",
    "axs[2].axes.get_yaxis().set_ticks([])\n",
    "color = \"viridis\"\n",
    "X_max = np.max(X,axis=0)\n",
    "axs[0].imshow(X_max, cmap=plt.get_cmap(color), vmin=vmin, vmax=vmax)\n",
    "X_max = np.max(X,axis=1)\n",
    "axs[1].imshow(X_max, cmap=plt.get_cmap(color), vmin=vmin, vmax=vmax)\n",
    "X_max = np.max(X,axis=2)\n",
    "axs[2].imshow(X_max, cmap=plt.get_cmap(color), vmin=vmin, vmax=vmax)\n",
    "plt.show()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "metadata": {},
   "outputs": [],
   "source": [
    "X = torch.from_numpy(X)\n",
    "X_np = np.asarray(X)"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Run Neural NCPD"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "metadata": {
    "scrolled": true
   },
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "/Users/jvendrow/opt/anaconda3/lib/python3.7/site-packages/torch/nn/_reduction.py:43: UserWarning: size_average and reduce args will be deprecated, please use reduction='sum' instead.\n",
      "  warnings.warn(warning.format(ret))\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "cec173bd90104e518e8ef9e0852d76b2",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "HBox(children=(IntProgress(value=0, max=10000), HTML(value='')))"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\n"
     ]
    }
   ],
   "source": [
    "# unsupervised case,one layer\n",
    "r=7\n",
    "n1,n2,n3 = X.shape\n",
    "net = NNCPD([n1, 5, 3],[n2, 5, 3],[n3, 5, 3])\n",
    "\n",
    "loss_func = Energy_Loss_Tensor()\n",
    "\n",
    "history_unsupervised = train(net, X, loss_func, r, epoch = 10000, lr1 = 0, lr2 = 50)\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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\n",
      "text/plain": [
       "<Figure size 432x288 with 1 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "# plot the loss curve\n",
    "history_unsupervised.plot_scalar('loss')"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 8,
   "metadata": {},
   "outputs": [],
   "source": [
    "X1 = history_unsupervised.get('A_X1')[-1]\n",
    "X2 = history_unsupervised.get('B_X1')[-1]\n",
    "X3 = history_unsupervised.get('C_X1')[-1]\n",
    "\n",
    "\n",
    "A_A1 = history_unsupervised.get('A_A1')[-1]\n",
    "A_S1 = history_unsupervised.get('A_S1')[-1]\n",
    "B_A1 = history_unsupervised.get('B_A1')[-1]\n",
    "B_S1 = history_unsupervised.get('B_S1')[-1]\n",
    "C_A1 = history_unsupervised.get('C_A1')[-1]\n",
    "C_S1 = history_unsupervised.get('C_S1')[-1]\n",
    "\n",
    "A_A2 = history_unsupervised.get('A_A2')[-1]\n",
    "A_S2 = history_unsupervised.get('A_S2')[-1]\n",
    "B_A2 = history_unsupervised.get('B_A2')[-1]\n",
    "B_S2 = history_unsupervised.get('B_S2')[-1]\n",
    "C_A2 = history_unsupervised.get('C_A2')[-1]\n",
    "C_S2 = history_unsupervised.get('C_S2')[-1]\n"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "### Visualize S Matrices\n",
    "\n",
    "Note that these S matrices will not look as nice as the S matrices in the paper. This is because the order of topics in NCPD is arbitrary, so to get a \"nice\" S matrix one needs to reorder the topics"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 9,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "<matplotlib.image.AxesImage at 0x143778210>"
      ]
     },
     "execution_count": 9,
     "metadata": {},
     "output_type": "execute_result"
    },
    {
     "data": {
      "image/png": "iVBORw0KGgoAAAANSUhEUgAAAK0AAADrCAYAAAAbrhYyAAAABHNCSVQICAgIfAhkiAAAAAlwSFlzAAALEgAACxIB0t1+/AAAADh0RVh0U29mdHdhcmUAbWF0cGxvdGxpYiB2ZXJzaW9uMy4xLjEsIGh0dHA6Ly9tYXRwbG90bGliLm9yZy8QZhcZAAAD9klEQVR4nO3dMWteVQDH4fO+jYutKE2zCKF7QSkIgig4Ori4OPkN/BZ+CHHyKzgICgrdBOki0iI4aKEdXKLFlgZtSXPdXExKLzSc/vB51nPD/w6/XDKdbJZlGVCynf0CsJZoyREtOaIlR7TkiJacnTUP717cLvv7q37kmbl148KU3THG2Gzn/m4vx8dT92f4exyOR8vDzUlnqwrc398Z33596dm81Uof7b89ZXeMMbYvnp+2PcYYx4eHU/dnuL5cO/XMnwfkiJYc0ZIjWnJES45oyREtOaIlR7TkiJYc0ZIjWnJES45oyREtOaIlR7TkiJYc0ZIjWnJES45oyREtOaIlR7TkiJYc0ZIjWnJWXUB368aFqRfBzfLXu1em7l/55Oa07V/ffDhn+An/v8aXlhzRkiNackRLjmjJES05oiVHtOSIlhzRkiNackRLjmjJES05oiVHtOSIlhzRkiNackRLjmjJES05oiVHtOSIlhzRkiNackRLjmjJES05q676PNo7Pw4+fOus3uWJto+mzI4xxtj9/Pt542OMX+5dnba9feNozvBP35165EtLjmjJES05oiVHtOSIlhzRkiNackRLjmjJES05oiVHtOSIlhzRkiNackRLjmjJES05oiVHtOSIlhzRkiNackRLjmjJES05oiVHtOSIlpxVV33uHByOvc/mXnv5f7Tz851p25/+8OWU3Q/e//3UM19ackRLjmjJES05oiVHtOSIlhzRkiNackRLjmjJES05oiVHtOSIlhzRkiNackRLjmjJES05oiVHtOSIlhzRkiNackRLjmjJES05oiVHtOSIlpzNsixP/fBrr7+wfPHVpTN8ndN9fPmdKbtjjHFu9+K07THGePzH3an7M1xfro37y93NSWe+tOSIlhzRkiNackRLjmjJES05oiVHtOSIlhzRkiNackRLjmjJES05oiVHtOSIlhzRkiNackRLjmjJES05oiVHtOSIlhzRkiNackRLjmjJ2Vnz8O2bL027cvOb336csjvGGO+9enXaNv/lS0uOaMkRLTmiJUe05IiWHNGSI1pyREuOaMkRLTmiJUe05IiWHNGSI1pyREuOaMkRLTmiJUe05IiWHNGSI1pyREuOaMkRLTmiJUe05IiWnFVXfc505+jBtO1zr7w8bXuMMR7/eW/q/vPGl5Yc0ZIjWnJES45oyREtOaIlR7TkiJYc0ZIjWnJES45oyREtOaIlR7TkiJYc0ZIjWnJES45oyREtOaIlR7TkiJYc0ZIjWnJES45oyREtOZtlWZ7+4c3mYIxx++xeB/51eVmWvZMOVkULzwN/HpAjWnJES45oyREtOaIlR7TkiJYc0ZLzD71bV/4LT81SAAAAAElFTkSuQmCC\n",
      "text/plain": [
       "<Figure size 432x288 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "plt.xticks([], [])\n",
    "plt.yticks([], [])\n",
    "plt.imshow((C_S1 / torch.mean(C_S1, axis=0)).T)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 10,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "<matplotlib.image.AxesImage at 0x1439fc150>"
      ]
     },
     "execution_count": 10,
     "metadata": {},
     "output_type": "execute_result"
    },
    {
     "data": {
      "image/png": "iVBORw0KGgoAAAANSUhEUgAAAG8AAADrCAYAAABn5MiuAAAABHNCSVQICAgIfAhkiAAAAAlwSFlzAAALEgAACxIB0t1+/AAAADh0RVh0U29mdHdhcmUAbWF0cGxvdGxpYiB2ZXJzaW9uMy4xLjEsIGh0dHA6Ly9tYXRwbG90bGliLm9yZy8QZhcZAAADOElEQVR4nO3dP49NWxyA4b2Ra5KpRyG5dBoJCiGhvAXxBXwwPoluCp0QKrWKCIV/xU1oBks13ZyJLU62N56n/Z1krZM3a3crax5jTDSdWHsD/DrxwsQLEy9MvDDxwk4t+fE/8+mxM+1uay9/tAuXvqyy7svXB9OHT9/mo2aL4u1Mu9P1+b/fs6uY/f3nq6x77dbrjTOfzTDxwsQLEy9MvDDxwsQLEy9MvDDxwsQLEy9MvDDxwsQLEy9MvDDxwsQLEy9MvDDxwsQLEy9MvDDxwsQLEy9s0UWTNe2/Xeeix6FbZ6+ssu6L8XHjzMkLEy9MvDDxwsQLEy9MvDDxwsQLEy9MvDDxwsQLEy9MvDDxwsQLEy9MvDDxwsQLEy9MvDDxwsQLEy9MvDDxwsQLEy9s0RWvr2d2p3d3b2xrL8e6eG+ddQ/t3TlYZd3x6MnGmZMXJl6YeGHihYkXJl6YeGHihYkXJl6YeGHihYkXJl6YeGHihYkXJl6YeGHihYkXJl6YeGHihYkXJl6YeGHihYkXJl7Yoitep959ns7cf7ytvRzrb33Fax5fNs6cvDDxwsQLEy9MvDDxwsQLEy9MvDDxwsQLEy9MvDDxwsQLEy9MvDDxwsQLEy9MvDDxwsQLEy9MvDDxwsQLEy9MvDDxwsQLW3Q/b023z11ddf0HbzY/hbZNN29/3jhz8sLECxMvTLww8cLECxMvTLww8cLECxMvTLww8cLECxMvTLww8cLECxMvTLww8cLECxMvTLww8cLECxMvTLww8cLEC5vHGD/946uXd8az/X+3uJ3N1noCbW1Px8Pp//FpPmrm5IWJFyZemHhh4oWJFyZemHhh4oWJFyZemHhh4oWJFyZemHhh4oWJFyZemHhh4oWJFyZemHhh4oWJFyZemHhh4oWJF7boFa8xjelgfNvWXo534uQ66x76vtL/PoaTFyZemHhh4oWJFyZemHhh4oWJFyZemHhh4oWJFyZemHhh4oWJFyZemHhh4oWJFyZemHhh4oWJFyZemHhh4oWJFyZe2KJXvOZ5fj9N06vtbYcjnB9j7B01WBSPP4vPZph4YeKFiRcmXph4YeKFiRcmXtgPm0VI7cPmsckAAAAASUVORK5CYII=\n",
      "text/plain": [
       "<Figure size 432x288 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "plt.xticks([], [])\n",
    "plt.yticks([], [])\n",
    "plt.imshow((C_S2 / torch.mean(C_S2, axis=0)).T)"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "### Rank 7 Approximation (for both Neural NCPD and Standard HNCPD)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 11,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Reconstruction loss: 78.02716601846191\n",
      "Relative reconstruction loss: 0.39041848431097836\n"
     ]
    },
    {
     "data": {
      "image/png": "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\n",
      "text/plain": [
       "<Figure size 864x360 with 3 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "approx = outer_product(X1, X2, X3).numpy()\n",
    "print(\"Reconstruction loss:\", np.linalg.norm(np.ndarray.flatten(X_np-approx), 2))\n",
    "print(\"Relative reconstruction loss:\", np.linalg.norm(np.ndarray.flatten(X_np-approx), 2)  / np.linalg.norm(np.ndarray.flatten(X_np), 2))\n",
    "\n",
    "fig, axs = plt.subplots(1, 3, constrained_layout=True, figsize=(12,5))\n",
    "axs[0].axes.get_xaxis().set_ticks([])\n",
    "axs[0].axes.get_yaxis().set_ticks([])\n",
    "axs[1].axes.get_xaxis().set_ticks([])\n",
    "axs[1].axes.get_yaxis().set_ticks([])\n",
    "axs[2].axes.get_xaxis().set_ticks([])\n",
    "axs[2].axes.get_yaxis().set_ticks([])\n",
    "X_max = np.max(approx,axis=0)\n",
    "axs[0].imshow(X_max, vmin=vmin, vmax=vmax)\n",
    "X_max = np.max(approx,axis=1)\n",
    "axs[1].imshow(X_max, vmin=vmin, vmax=vmax)\n",
    "X_max = np.max(approx,axis=2)\n",
    "axs[2].imshow(X_max, vmin=vmin, vmax=vmax)\n",
    "plt.show()"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "### Rank 5 Approximation for Neural NCPD"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 12,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Reconstruction loss: 89.84790621919842\n",
      "Relative reconstruction loss: 0.4495650060687139\n"
     ]
    },
    {
     "data": {
      "image/png": "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\n",
      "text/plain": [
       "<Figure size 864x360 with 3 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "approx = outer_product(torch.mm(A_A1,A_S1), torch.mm(B_A1,B_S1), torch.mm(C_A1,C_S1)).numpy()\n",
    "\n",
    "print(\"Reconstruction loss:\", np.linalg.norm(np.ndarray.flatten(X_np-approx), 2))\n",
    "print(\"Relative reconstruction loss:\", np.linalg.norm(np.ndarray.flatten(X_np-approx), 2)  / np.linalg.norm(np.ndarray.flatten(X_np), 2))\n",
    "\n",
    "\n",
    "fig, axs = plt.subplots(1, 3, constrained_layout=True, figsize=(12,5))\n",
    "axs[0].axes.get_xaxis().set_ticks([])\n",
    "axs[0].axes.get_yaxis().set_ticks([])\n",
    "axs[1].axes.get_xaxis().set_ticks([])\n",
    "axs[1].axes.get_yaxis().set_ticks([])\n",
    "axs[2].axes.get_xaxis().set_ticks([])\n",
    "axs[2].axes.get_yaxis().set_ticks([])\n",
    "X_max = np.max(approx,axis=0)\n",
    "axs[0].imshow(X_max, vmin=vmin, vmax=vmax)\n",
    "X_max = np.max(approx,axis=1)\n",
    "axs[1].imshow(X_max, vmin=vmin, vmax=vmax)\n",
    "X_max = np.max(approx,axis=2)\n",
    "axs[2].imshow(X_max, vmin=vmin, vmax=vmax)\n",
    "\n",
    "plt.show()"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "### Rank 3 Approximation for Neural NCPD"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 13,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Reconstruction loss: 97.76090079796158\n",
      "Relative reconstruction loss: 0.48915864386751245\n"
     ]
    },
    {
     "data": {
      "image/png": "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\n",
      "text/plain": [
       "<Figure size 864x360 with 3 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "approx = outer_product(torch.mm(A_A1, torch.mm(A_A2,A_S2)), torch.mm(B_A1, torch.mm(B_A2,B_S2)), torch.mm(C_A1, torch.mm(C_A2,C_S2))).numpy()\n",
    "\n",
    "print(\"Reconstruction loss:\", np.linalg.norm(np.ndarray.flatten(X_np-approx), 2))\n",
    "print(\"Relative reconstruction loss:\", np.linalg.norm(np.ndarray.flatten(X_np-approx), 2)  / np.linalg.norm(np.ndarray.flatten(X_np), 2))\n",
    "\n",
    "fig, axs = plt.subplots(1, 3, constrained_layout=True, figsize=(12,5))\n",
    "axs[0].axes.get_xaxis().set_ticks([])\n",
    "axs[0].axes.get_yaxis().set_ticks([])\n",
    "axs[1].axes.get_xaxis().set_ticks([])\n",
    "axs[1].axes.get_yaxis().set_ticks([])\n",
    "axs[2].axes.get_xaxis().set_ticks([])\n",
    "axs[2].axes.get_yaxis().set_ticks([])\n",
    "X_max = np.max(approx,axis=0)\n",
    "axs[0].imshow(X_max, vmin=vmin, vmax=vmax)\n",
    "X_max = np.max(approx,axis=1)\n",
    "axs[1].imshow(X_max, vmin=vmin, vmax=vmax)\n",
    "X_max = np.max(approx,axis=2)\n",
    "axs[2].imshow(X_max, vmin=vmin, vmax=vmax)\n",
    "\n",
    "plt.show()"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Standard HNCPD"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 14,
   "metadata": {},
   "outputs": [],
   "source": [
    "#rank 7 (same as Neural NNCPD)\n",
    "X_np = np.asarray(X)\n",
    "r=7\n",
    "factors_tl = non_negative_parafac(X_np, r)#, init='random', random_state=2)\n",
    "\n",
    "X_1 =  factors_tl[0]\n",
    "X_2 =  factors_tl[1]\n",
    "X_3 =  factors_tl[2]"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "### Rank 5 (for HNCPD)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 15,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "90.29209285629175\n",
      "0.45178754832488316\n"
     ]
    },
    {
     "data": {
      "image/png": "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sziHGCubzjtEst9s78uyzVJnry4hn/RuW8fK3m6PZz2/HfqRjpW82tRtvt7elGWbTTjvUFb1cluc8zTl27aGeO52hG033rp4XmN07NnH4vnH9seiGemfosSnRmz3NE31lh7peClmu7N9yJSfo36sd13W2clhbT5/aSuQeGqxLE0cjfaSC2zv0rVuqsz1ELMdJ9lmPY5r7lbSmRPjfJik57L7yJ65yRiln6bfMC/E7QU6201MU96mpXHm3X7MaW4+cx+nqf4e6s410aHadquu4i9oAAAAg7BAAwAAGIQFGgAAwCAs0AAAAAZhgQYAADAICzQAAIBBnBazP03TVMTsr0sdyd6RYi4379212pxDJP68a+bKpnrF8jemnYaY7fWhrvnt3/2Jul7MZa2Llrt6/f7ei+Pn/c23P9U6VkpJXW7qa7y/q5+77f+5Lct24aeJpUjTv7t5WdZJ0bHL/Q/qwitb5+J6pQjblHzbidO9YOr1GraimFM87ihR3OknszRFdfoft/Roxhanet007Uab8wX6kbYJOYTjremepn6WW9CE9tJY3YcHqBsr3b2WD8UkG95/YbeSODbKek/cPmCd6++UONckz2Eeqneb6T9HZxbHXDCnz7nGnjiH1I/wYk3HijtzxPHfzeevizri9BW3TuhtWZCe5fnu+MOc1grtrXAq4Zz9BQ0AAGAQFmgAAACDsEADAAAYhAUaAADAICzQAAAABmGBBgAAMIjTYvbXR6I6zynFjL4Ikf4pijosR1M8d3TmJe7a3K1gDhH83T6mNudqu4WHlGna60fcACEcbw5xwHHngV3x7yFCeJR05GnKW1Qsd8XJhajXdN5JmbJ/eOIcMs/TetN4qBuHzbHF4clMD20a4ykuPNyjVoJ1uq/NXUceyVButnn6Pg8xkr21N0SO7s+x/r3I7Gre7kaIr5tuxVAWt2oITe6LwtM3/nlUOR8+ddJO81DzUe/c2/gMtfvR/J44NCaO7vYncduR7jYh6R1/epvtXVPCfJKi++PUlqLjz7w9QnOKjYXxmUx9DOddrSXSN9S8TVujNJ679L12emsAAABcggUaAADAICzQAAAABmGBBgAAMAgLNAAAgEFYoAEAAAzixFDbdZp2p8eozp3o1RB7n+Kmk/UQojNTmyneehMys4stCWJcdjP694v/9L26sLsMb2xZ0I5yThGqwbx/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\n",
      "text/plain": [
       "<Figure size 864x360 with 3 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "model = NMF(n_components=5, init='random', random_state=0)\n",
    "A_1_0 = model.fit_transform(X_1)\n",
    "S_1_1 = model.components_\n",
    "\n",
    "model = NMF(n_components=5, init='random', random_state=0)\n",
    "A_2_0 = model.fit_transform(X_2)\n",
    "S_2_1 = model.components_\n",
    "\n",
    "model = NMF(n_components=5, init='random', random_state=0)\n",
    "A_3_0 = model.fit_transform(X_3)\n",
    "S_3_1 = model.components_\n",
    "\n",
    "X_approx_5 = outer_product_np(A_1_0 @ S_1_1, A_2_0 @ S_2_1, A_3_0 @ S_3_1)\n",
    "\n",
    "print(np.linalg.norm(np.ndarray.flatten(X_np-X_approx_5), 2))\n",
    "print(np.linalg.norm(np.ndarray.flatten(X_np-X_approx_5), 2)  / np.linalg.norm(np.ndarray.flatten(X_np), 2))\n",
    "\n",
    "fig, ax = plt.subplots(1, 3, constrained_layout=True, figsize=(12,5))\n",
    "ax[0].axes.get_xaxis().set_ticks([])\n",
    "ax[0].axes.get_yaxis().set_ticks([])\n",
    "ax[1].axes.get_xaxis().set_ticks([])\n",
    "ax[1].axes.get_yaxis().set_ticks([])\n",
    "ax[2].axes.get_xaxis().set_ticks([])\n",
    "ax[2].axes.get_yaxis().set_ticks([])\n",
    "X_max = np.max(X_approx_5,axis=0)\n",
    "ax[0].imshow(X_max, vmin=vmin, vmax=vmax)\n",
    "X_max = np.max(X_approx_5,axis=1)\n",
    "ax[1].imshow(X_max, vmin=vmin, vmax=vmax)\n",
    "X_max = np.max(X_approx_5,axis=2)\n",
    "ax[2].imshow(X_max, vmin=vmin, vmax=vmax)\n",
    "\n",
    "plt.show()"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "### Rank 3 (for HNCPD)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 16,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "133.58389782429782\n",
      "0.668403398177693\n"
     ]
    },
    {
     "data": {
      "image/png": "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\n",
      "text/plain": [
       "<Figure size 864x360 with 3 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "model = NMF(n_components=3, init='random', random_state=0)\n",
    "A_1_1 = model.fit_transform(S_1_1)\n",
    "S_1_2 = model.components_\n",
    "\n",
    "model = NMF(n_components=3, init='random', random_state=0)\n",
    "A_2_1 = model.fit_transform(S_2_1)\n",
    "S_2_2 = model.components_\n",
    "\n",
    "model = NMF(n_components=3, init='random', random_state=0)\n",
    "A_3_1 = model.fit_transform(S_3_1)\n",
    "S_3_2 = model.components_\n",
    "\n",
    "X_approx_3 = outer_product_np(A_1_0 @ A_1_1 @ S_1_2, A_2_0 @ A_2_1 @ S_2_2, A_3_0 @ A_3_1 @ S_3_2)\n",
    "\n",
    "print(np.linalg.norm(np.ndarray.flatten(X_np-X_approx_3), 2))\n",
    "print(np.linalg.norm(np.ndarray.flatten(X_np-X_approx_3), 2) / np.linalg.norm(np.ndarray.flatten(X_np), 2))\n",
    "\n",
    "\n",
    "fig, ax = plt.subplots(1, 3, constrained_layout=True, figsize=(12,5))\n",
    "ax[0].axes.get_xaxis().set_ticks([])\n",
    "ax[0].axes.get_yaxis().set_ticks([])\n",
    "ax[1].axes.get_xaxis().set_ticks([])\n",
    "ax[1].axes.get_yaxis().set_ticks([])\n",
    "ax[2].axes.get_xaxis().set_ticks([])\n",
    "ax[2].axes.get_yaxis().set_ticks([])\n",
    "X_max = np.max(X_approx_3,axis=0)\n",
    "ax[0].imshow(X_max, vmin=vmin, vmax=vmax)\n",
    "X_max = np.max(X_approx_3,axis=1)\n",
    "ax[1].imshow(X_max, vmin=vmin, vmax=vmax)\n",
    "X_max = np.max(X_approx_3,axis=2)\n",
    "ax[2].imshow(X_max, vmin=vmin, vmax=vmax)\n",
    "\n",
    "plt.show()"
   ]
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "Python 3",
   "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.6.9"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 4
}
