{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 1,
   "id": "e165b607",
   "metadata": {},
   "outputs": [],
   "source": [
    "#Load required packages\n",
    "import pandas\n",
    "import random\n",
    "import numpy as np\n",
    "from numpy import loadtxt\n",
    "from numpy import savetxt\n",
    "import tensorflow as tf\n",
    "# Need CVNN for complex valued neural networks\n",
    "import cvnn.layers as complex_layers\n",
    "from keras.models import Sequential\n",
    "from keras.layers import Dense\n",
    "from keras.wrappers.scikit_learn import KerasRegressor\n",
    "from sklearn.model_selection import cross_validate\n",
    "from sklearn.model_selection import KFold\n",
    "from sklearn.preprocessing import StandardScaler\n",
    "from sklearn.pipeline import Pipeline\n",
    "# For plots\n",
    "import matplotlib.pyplot as plt\n",
    "from matplotlib.patches import Patch\n",
    "from matplotlib.lines import Line2D"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "id": "d9a1d3b2",
   "metadata": {},
   "outputs": [],
   "source": [
    "#############    THIS IS HOW WE GENERATE THETA AND PHI  #############\n",
    "# Randomly Generate numbers between -3*pi and 3*pi\n",
    "Rtheta = np.random.rand(10000,1)*6*np.pi - 3*np.pi\n",
    "# Randomly Generate numbers between -2*pi and 2*pi\n",
    "Rphi = np.random.rand(10000,1)*4*np.pi - 2*np.pi"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "id": "bb4259b0",
   "metadata": {},
   "outputs": [],
   "source": [
    "# Define the character (p,q) of SU(3)\n",
    "def character(p,q, theta, phi) :\n",
    "    summand = 0\n",
    "    for k in range(p+1):\n",
    "        for l in range(q+1):\n",
    "            summand = summand + np.exp(-1j*(k+l)*theta)*(np.sin((k-l+q+1)*phi/2)/np.sin(phi/2) )\n",
    "    out = np.exp(1j*theta*(2*p+q)/3)*summand    \n",
    "    return out"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "id": "b15f5110",
   "metadata": {},
   "outputs": [],
   "source": [
    "################  Generate the input alphabet of size 25 for Neural Network #############\n",
    "X = np.ones((10000,1))+1j-1j\n",
    "for p in range(5):\n",
    "    for q in range(5):\n",
    "        # Create the 10000x1 column corresponding to (p,q)-character evaluated on (Rtheta, Rphi)\n",
    "        Xtemp = np.zeros((10000,1))+1j-1j\n",
    "        for i in range(10000):\n",
    "            Xtemp[i,0] = character(p,q, Rtheta[i], Rphi[i])\n",
    "        # Append the (p,q)-character column to our big matrix    \n",
    "        X = np.append(X, Xtemp, axis=1)\n",
    "\n",
    "# Remove repeated 1st column\n",
    "X = X[:,range(1,26)]\n",
    "#  Output X of shape 10000x25, each row corresponds to a random pair (theta, phi), each column corresponds to a character (p,q)\n",
    "# Change the data type of X to be compatible with complex neural network\n",
    "X = np.complex64(X)\n",
    "#X.shape"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "id": "db48e74f",
   "metadata": {},
   "outputs": [],
   "source": [
    "###########  Large alphabet of size 49 for Linear Regression #######\n",
    "XL = np.ones((10000,1))+1j-1j\n",
    "for p in range(7):\n",
    "    for q in range(7):\n",
    "        # Create the 10000x1 column corresponding to (p,q)-character evaluated on (Rtheta, Rphi)\n",
    "        XLtemp = np.zeros((10000,1))+1j-1j\n",
    "        for i in range(10000):\n",
    "            XLtemp[i,0] = character(p,q, Rtheta[i], Rphi[i])\n",
    "        # Append the (p,q)-character column to our big matrix    \n",
    "        XL = np.append(XL, XLtemp, axis=1)\n",
    "\n",
    "# Remove repeated 1st column\n",
    "XL = XL[:,range(1,50)]\n",
    "#  Output X of shape 10000x49, each row corresponds to a random pair (theta, phi), each column corresponds to a character (p,q)\n",
    "# Change the data type of XL to be compatible with complex neural network\n",
    "XL = np.complex64(XL)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "id": "0fc0fc19",
   "metadata": {},
   "outputs": [],
   "source": [
    "####  Create indicator functions to use as basis for Gaussian linear combination to obtain class functions to be learned #####\n",
    "# Create cache for collecting the 25 indicator functions below\n",
    "Ind = [[] for _ in range(10000)]\n",
    "# Each indicator function has 10% x 10% overlap with the next.\n",
    "for l in range(5):\n",
    "    for k in range(5):\n",
    "        L = np.all( [-3*np.pi + 1.2*np.pi*k <Rtheta, Rtheta<-3*np.pi + 1.2*np.pi*(k+1) + 0.12*np.pi,\n",
    "                    -2*np.pi + 0.8*np.pi*l <Rphi, Rphi<-2*np.pi + 0.8*np.pi*(l+1) + 0.08*np.pi ], axis=0)\n",
    "        Ind = np.append(Ind, np.where(L,1,0).reshape((10000,1)), axis=1)\n",
    "# Output Ind of shape 10000x25, each column corresponds to one of the 25 indicator functions"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "id": "564fab45",
   "metadata": {},
   "outputs": [],
   "source": [
    "##### Create class functions by taking Gaussian linear combination of indicator basis ######\n",
    "# 100 class functions which are gaussian linear combinations of indicator basis\n",
    "Y = np.zeros((10000,100)) +1j -1j\n",
    "for l in range(100):\n",
    "    # Creates 25 gaussian random complex numbers\n",
    "    G_temp = np.random.normal(loc=0,scale=1, size = (25,2)).view(np.complex128) \n",
    "    Y_temp = np.dot(Ind,G_temp)\n",
    "    Y[:,l] = Y_temp.reshape((10000,))\n",
    "# Change data type of Y\n",
    "Y = np.complex64(Y)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 8,
   "id": "6df9e23a",
   "metadata": {},
   "outputs": [],
   "source": [
    "# Our neural network model\n",
    "def NN6_model():\n",
    "    # create model\n",
    "    model = Sequential()\n",
    "    model.add(complex_layers.ComplexInput(input_shape=(25)))\n",
    "    model.add(complex_layers.ComplexDense(180, kernel_regularizer=tf.keras.regularizers.l2(1e-8), activation='cart_relu'))\n",
    "    model.add(complex_layers.ComplexDense(150, kernel_regularizer=tf.keras.regularizers.l2(1e-8), activation='cart_relu') )\n",
    "    model.add(complex_layers.ComplexDense(100, kernel_regularizer=tf.keras.regularizers.l2(1e-8),activation='cart_relu'))\n",
    "    model.add(complex_layers.ComplexDense(60, kernel_regularizer=tf.keras.regularizers.l2(1e-8),activation='cart_relu'))\n",
    "    model.add(complex_layers.ComplexDense(35, kernel_regularizer=tf.keras.regularizers.l2(1e-8),activation='cart_relu'))\n",
    "    model.add(complex_layers.ComplexDense(10, kernel_regularizer=tf.keras.regularizers.l2(1e-8),activation='cart_relu'))\n",
    "    model.add(complex_layers.ComplexDense(1, kernel_regularizer=tf.keras.regularizers.l2(1e-8),activation='convert_to_real_with_abs'))\n",
    "    # Compile model\n",
    "    model.compile(loss='mean_squared_error', optimizer= 'Adamax')\n",
    "    return model\n",
    "NN6_model().summary()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "50c5bb31",
   "metadata": {},
   "outputs": [],
   "source": [
    "# Train on 8500 points and evaluate on 1500 unseen points\n",
    "# Create cache for recording losses\n",
    "NNloss = np.zeros((100,1))\n",
    "for k in range(100):\n",
    "    history = NN6_model().fit(X[range(8500),:], Y[range(8500),k],  batch_size=500 , epochs=100, validation_data=(X[range(8500,10000),:], Y[range(8500,10000),k]))\n",
    "    temp_loss = NN6_model().evaluate(X[range(8500,10000),:],  Y[range(8500,10000),k], verbose=2)\n",
    "    # Record losses for each class function\n",
    "    NNloss[k] = temp_loss\n",
    "    savetxt(\"NNloss.csv\", NNloss, delimiter=',')"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 11,
   "id": "2b873342",
   "metadata": {},
   "outputs": [],
   "source": [
    "# Our Linear model\n",
    "def Linear_model():\n",
    "    # create model\n",
    "    model = Sequential()\n",
    "    model.add(complex_layers.ComplexInput(input_shape=(49)))\n",
    "    model.add(complex_layers.ComplexDense(1, kernel_regularizer=tf.keras.regularizers.l2(1e-9),activation='convert_to_real_with_abs'))\n",
    "    # Compile model\n",
    "    model.compile(loss='mean_squared_error', optimizer= 'Adamax')\n",
    "    return model\n",
    "Linear_model().summary()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "45896b3c",
   "metadata": {},
   "outputs": [],
   "source": [
    "# Train on 8500 points and evaluate on 1500 unseen points\n",
    "# Create cache for recording losses\n",
    "Linearloss = np.zeros((100,1))\n",
    "for k in range(100):\n",
    "    history_L = Linear_model().fit(XL[range(8500),:], Y[range(8500),k],  batch_size=250 , epochs=300, validation_data=(XL[range(8500,10000),:], Y[range(8500,10000),k]))\n",
    "    temp_loss_L = Linear_model().evaluate(XL[range(8500,10000),:],  Y[range(8500,10000),k], verbose=2)\n",
    "    # Record losses for each class function\n",
    "    Linearloss[k] = temp_loss_L\n",
    "    savetxt(\"Linearloss.csv\", Linearloss, delimiter=',')"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "59354661",
   "metadata": {},
   "outputs": [],
   "source": [
    "# Plot histogram on a log scale\n",
    "plt.hist(NNloss, histtype='stepfilled', alpha=0.3, bins=10)\n",
    "plt.hist(Linearloss, histtype='stepfilled', alpha=0.3, bins=10)\n",
    "plt.title(\"Discontinuous SU(3) class function learning performance\")\n",
    "plt.xlabel(\"MSE\")\n",
    "plt.ylabel(\"Frequency\")\n",
    "legend_elements = [Patch(facecolor='blue', edgecolor='b',label='Neural Network mean  '+ str(round(NNloss.mean(),6))),\n",
    "                   Patch(facecolor='red', edgecolor='r',label='Linear Regression mean  '+ str(round(Linearloss.mean(),6)))]\n",
    "plt.legend(handles=legend_elements)\n",
    "plt.xscale('log')\n",
    "plt.show()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "34bf4859",
   "metadata": {},
   "outputs": [],
   "source": []
  }
 ],
 "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.8.8"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 5
}
