{
 "cells": [
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# Import packages"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 1,
   "metadata": {},
   "outputs": [],
   "source": [
    "import numpy as np\n",
    "from sklearn.manifold import MDS\n",
    "import matplotlib.pyplot as plt\n",
    "from matplotlib.path import Path\n",
    "import ot\n",
    "import time\n",
    "from sklearn import svm\n",
    "from sklearn.model_selection import train_test_split\n",
    "from sklearn.metrics import accuracy_score\n",
    "import random\n",
    "\n",
    "np.random.seed(0)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "metadata": {},
   "outputs": [],
   "source": [
    "import sys\n",
    "sys.path.append(\"../\")\n",
    "\n",
    "from lib.gromov import gromov_wasserstein, cost_matrix_d, tensor_dot_param, tensor_dot_func, gwgrad_partial1, partial_gromov_wasserstein, partial_gromov_ver1\n",
    "from lib.gromov import GW_dist,MPGW_dist, PGW_dist_with_penalty"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "metadata": {},
   "outputs": [],
   "source": [
    "import torch\n",
    "from lib.unbalanced_gromov_wasserstein.unbalancedgw.vanilla_ugw_solver import exp_ugw_sinkhorn\n",
    "from lib.unbalanced_gromov_wasserstein.unbalancedgw._vanilla_utils import ugw_cost\n",
    "\n",
    "device = torch.device('cuda:0' if torch.cuda.is_available() else 'cpu')"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# Generate Data"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## House Data"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "(100, 2)"
      ]
     },
     "execution_count": 4,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "points_per_axis = 8\n",
    "lim = 0.25\n",
    "\n",
    "x = np.linspace(-lim, lim, points_per_axis)\n",
    "y = np.linspace(-lim, lim, points_per_axis)\n",
    "X, Y = np.meshgrid(x, y)\n",
    "square = np.array([X.ravel(), Y.ravel()]).T\n",
    "\n",
    "vertex1 = np.array([-lim, lim])\n",
    "vertex2 = np.array([lim, lim])\n",
    "side_length = np.linalg.norm(vertex2 - vertex1)\n",
    "height = (np.sqrt(3) / 2) * side_length\n",
    "vertex3 = np.array([0, lim + height])\n",
    "\n",
    "x_coords = []\n",
    "y_coords = []\n",
    "\n",
    "for i in range(points_per_axis):\n",
    "    for j in range(points_per_axis - i):\n",
    "        # Barycentric coordinates for the points\n",
    "        t1 = i / points_per_axis\n",
    "        t2 = j / points_per_axis\n",
    "        t3 = 1 - t1 - t2\n",
    "\n",
    "        # Calculate the position of each point using barycentric coordinates\n",
    "        x = t1 * vertex1[0] + t2 * vertex2[0] + t3 * vertex3[0]\n",
    "        y = t1 * vertex1[1] + t2 * vertex2[1] + t3 * vertex3[1]\n",
    "        \n",
    "        x_coords.append(x)\n",
    "        y_coords.append(y)\n",
    "\n",
    "points = np.array([x_coords, y_coords]).T\n",
    "points.shape\n",
    "\n",
    "# Concatenate points with square\n",
    "house = np.concatenate([square, points], axis=0)\n",
    "\n",
    "xmin, ymin = house.min(axis=0)\n",
    "square -= np.array([xmin, ymin])\n",
    "house -= np.array([xmin, ymin])\n",
    "\n",
    "house.shape"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "metadata": {},
   "outputs": [],
   "source": [
    "# Take random sample of 50 rows from the square matrix\n",
    "n1 = 50\n",
    "idx = np.random.choice(square.shape[0], n1, replace=False)\n",
    "square_sample = square[idx]\n",
    "\n",
    "# Concatenate sample with points to form a house\n",
    "n2 = 25\n",
    "idx = np.random.choice(points.shape[0], n2, replace=False)\n",
    "house_sample = np.concatenate([square_sample, points[idx]], axis=0)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "metadata": {},
   "outputs": [],
   "source": [
    "def generate_house_pair():\n",
    "    width_dilation = np.random.uniform(0.5, 1.5)\n",
    "    height_dilation = np.random.uniform(0.5, 1.5)\n",
    "    triangle_dilation = np.random.uniform(0.5, 1.5)\n",
    "\n",
    "    new_house = np.copy(house)\n",
    "    new_house[:, 0] *= width_dilation\n",
    "    new_house[:, 1] *= height_dilation\n",
    "\n",
    "    triangle = new_house[points_per_axis**2:]\n",
    "    xmin, ymin = np.min(triangle, axis=0)\n",
    "    shift = np.array([xmin, ymin])\n",
    "    triangle -= shift\n",
    "\n",
    "    triangle[:, 1] *= triangle_dilation\n",
    "    triangle += shift\n",
    "\n",
    "    new_house[points_per_axis**2:] = triangle\n",
    "\n",
    "    # Rotate new house by random theta\n",
    "    theta = np.random.uniform(0, 2 * np.pi)\n",
    "    rotation_matrix = np.array([[np.cos(theta), -np.sin(theta)], [np.sin(theta), np.cos(theta)]])\n",
    "    new_house = np.dot(new_house, rotation_matrix)\n",
    "\n",
    "    return new_house[:points_per_axis**2], new_house"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "<matplotlib.collections.PathCollection at 0x75683578f610>"
      ]
     },
     "execution_count": 7,
     "metadata": {},
     "output_type": "execute_result"
    },
    {
     "data": {
      "image/png": "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jmDp0qXBkiVLUjzq+vv7v//7YMaMGcF9990XPPvss8Hq1auDU089NXjhhRfKvv4Xv/hF8Fu/9VvB6tWrg2effTa47777ghkzZgQPPfRQykcOTI31Px6s/fFg3Y8H6331srrWE4Qb9uyzzwaSgp07d554bseOHYGk4Lnnnpvw+y644ILg7rvvHvXcO9/5zuAv/uIvEjvWrKv1XF533XXBH//xH6dxiM6o9VwGQRAcOnQoOOuss4L/+I//CBYuXOj9YhwE0zufIz311FOBpAkXJYsuvfTS4NZbbx313HnnnRfceeedZV//iU98IjjvvPNGPXfLLbcEy5YtS+wYgVqw/seDtT8erPvxYL2vTVbXesrRDduxY4fy+byWLl164rlly5Ypn89r+/btE37fu9/9bn3/+9/X0NCQgiBQf3+/fvazn+nKK69M47AzqZZzefz4cf3zP/+z3v72t+vKK6/U3LlztXTpUn3ve99L6aizqdb35fHjx9XT06OPf/zjuuCCC9I4VCfUej7HKhQKyuVyevOb35zAUWbP66+/rt27d2vFihWjnl+xYsWE523Hjh3jXn/llVdq165dOnbsWGLHClSL9T8erP3xYN2PB+t99bK81hOEG3b48GHNnTt33PNz587V4cOHJ/y+L3/5yzr//PPV3NysmTNn6qqrrtKGDRv07ne/O8nDzbRazuWRI0f061//Wn/913+tq666So899piuvfZadXZ26oknnkj6kDOr1vfl5z//eTU2NuqjH/1okofnnFrP50i/+c1vdOedd6q7u1tNTU1xH2ImvfzyyyqVSpo3b96o5+fNmzfheTt8+HDZ17/xxht6+eWXEztWoFqs//Fg7Y8H6348WO+rl+W1niDcQWvXrlUul5v0a9euXZKkXC437vuDICj7fOTLX/6ydu7cqe9///vavXu3vvjFL+q2227Tv/zLvyT2M9VLkufy+PHjkqSOjg7dfvvtuvjii3XnnXfq6quvNjnMKclzuXv3bq1fv14PPPDApO9dS5L+PY8cO3ZM119/vY4fP64NGzbE/nNk3dhzNNV5K/f6cs8DSWD9jwdrfzxY9+PBep+8LK71jbH9S0jNqlWrdP3110/6mtbWVv30pz/VSy+9NO7vfvnLX47L8ET+93//V5/61Kf08MMP6/3vf78k6aKLLtKePXv0hS98QVdcccX0f4AMSfJczpkzR42NjTr//PNHPb9o0SL967/+a+0HnVFJnstt27bpyJEjOvvss088VyqV9LGPfUy9vb3av3//tI49i5I8n5Fjx47pAx/4gAYHB/X44497kRWPzJkzRw0NDeMy4UeOHJnwvJ155pllX9/Y2KgzzjgjsWMFIqz/8WDtjwfrfjxY75OT5bWeINxBc+bM0Zw5c6Z8XVtbmwqFgp566ildeumlkqQnn3xShUJBl112WdnvOXbsmI4dO6ZTThldJNHQ0HAiu2tJkudy5syZete73qV9+/aNev5nP/uZFi5cOP2Dz5gkz2VPT8+4D4BXXnmlenp69OEPf3j6B59BSZ5P6eSC/Pzzz6u/v9+7IHLmzJlasmSJtm7dqmuvvfbE81u3blVHR0fZ72lra9Mjjzwy6rnHHntMl1xyiWbMmJHo8QIS639cWPvjwbofD9b75GR6rY91zBsy56qrrgouuuiiYMeOHcGOHTuCCy+8cNytDM4999ygr6/vxJ/f8573BBdccEHQ398f/OIXvwi+9a1vBbNnzw42bNiQ9uFnSi3nsq+vL5gxY0bw9a9/PXj++eeDr3zlK0FDQ0Owbdu2tA8/U2o5l2P5PiV1pGrP57Fjx4JrrrkmaG5uDvbs2RMMDw+f+Dp69Gg9foS6iG5bcv/99wfPPvtssGbNmuDUU08N9u/fHwRBENx5551BT0/PiddHty25/fbbg2effTa4//77uUUZMov1Px6s/fFg3Y8H6331srrWE4Qb96tf/Sr44Ac/GJx22mnBaaedFnzwgx8MXnnllVGvkRR861vfOvHn4eHh4EMf+lCwYMGCYPbs2cG5554bfPGLXwyOHz+e7sFnTC3nMgiC4P777w9+53d+J5g9e3bwjne8w+v7rUZqPZcjsRifVO35HBwcDCSV/erv70/9+Ovpq1/9arBw4cJg5syZwTvf+c7giSeeOPF3f/InfxK85z3vGfX6gYGB4Hd/93eDmTNnBq2trcE999yT8hEDlWH9jwdrfzxY9+PBel+bLK71uSD4v05zAAAAAACQKKajAwAAAACQEoJwAAAAAABSQhAOAAAAAEBKCMIBAAAAAEgJQTgAAAAAACkhCAcAAAAAICUE4QAAAAAApIQgHAAAAACAlBCEAwAAAACQEoJwAAAAAABSQhAOAAAAAEBK/j/XQj6rpVZo6AAAAABJRU5ErkJggg==",
      "text/plain": [
       "<Figure size 1200x600 with 2 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "# 2 x 1 subplot\n",
    "fig, ax = plt.subplots(1, 2, figsize=(12, 6))\n",
    "new_square, new_house = generate_house_pair()\n",
    "\n",
    "# Plot square\n",
    "ax[0].scatter(new_square[:, 0], new_square[:, 1], color='blue', label='Square')\n",
    "\n",
    "# Plot house\n",
    "ax[1].scatter(new_house[:, 0], new_house[:, 1], color='blue', label='House')"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 8,
   "metadata": {},
   "outputs": [],
   "source": [
    "square_samples = []\n",
    "house_samples = []\n",
    "\n",
    "\n",
    "for i in range(20):\n",
    "    square_sample, house_sample = generate_house_pair()\n",
    "    square_samples.append(square_sample)\n",
    "    house_samples.append(house_sample)"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# Arrows"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 9,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "(41, 2)"
      ]
     },
     "execution_count": 9,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "arrow_shape = [[0, -0.1], [0, 0.1], [0.2, 0.1], [0.2, 0.3], [0.4, 0], [0.2, -0.3], [0.2, -0.1]]\n",
    "points_per_axis = 16\n",
    "vertices = np.array(arrow_shape)\n",
    "    \n",
    "# Generate barycentric coordinates\n",
    "arrow_path = Path(arrow_shape)\n",
    "    \n",
    "# Generate a grid of points\n",
    "x = np.linspace(-0.1, 0.5, points_per_axis)\n",
    "y = np.linspace(-0.5, 0.5, points_per_axis)\n",
    "X, Y = np.meshgrid(x, y)\n",
    "points = np.vstack((X.flatten(), Y.flatten())).T\n",
    "\n",
    "# Filter points that are inside the arrow polygon\n",
    "inside = arrow_path.contains_points(points)\n",
    "x_coords, y_coords = points[inside, 0], points[inside, 1]\n",
    "\n",
    "arrow = np.array([x_coords, y_coords]).T\n",
    "\n",
    "xmin = arrow[:, 0].min()\n",
    "arrow[:, 0] -= xmin - 0.6/points_per_axis\n",
    "\n",
    "base_arrow = arrow[arrow[:, 0] <= 0.21]\n",
    "point_arrow = arrow[arrow[:, 0] > 0.21]\n",
    "\n",
    "arrow = np.concatenate([base_arrow, point_arrow, np.array([[0.4, 0]])], axis=0)\n",
    "arrow.shape"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 10,
   "metadata": {},
   "outputs": [],
   "source": [
    "arrow2 = np.array([-x_coords, y_coords]).T\n",
    "double_arrow = np.concatenate([arrow, arrow2], axis=0)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 11,
   "metadata": {},
   "outputs": [],
   "source": [
    "def generate_arrow_pair():\n",
    "    width_dilation = np.random.uniform(0.5, 1.5)\n",
    "    height_dilation = np.random.uniform(0.5, 1.5)\n",
    "    triangle_dilation = np.random.uniform(1.0, 1.5)\n",
    "\n",
    "    new_arrow = np.copy(arrow)\n",
    "    new_arrow[:, 0] *= width_dilation\n",
    "    new_arrow[:, 1] *= height_dilation\n",
    "\n",
    "    \n",
    "    new_arrow[base_arrow.shape[0]:, 1] *= triangle_dilation\n",
    "    xvals = new_arrow[base_arrow.shape[0]:, 0]\n",
    "    xdiffs = np.abs(xvals.reshape(1, -1) - xvals.reshape(-1, 1))\n",
    "    shift = xdiffs.min()\n",
    "\n",
    "    new_arrow[:, 0] -= np.min(new_arrow[:, 0]) - shift\n",
    "\n",
    "    new_double_arrow = np.concatenate([new_arrow, np.array([-new_arrow[:, 0], new_arrow[:, 1]]).T], axis=0)\n",
    "\n",
    "    # Rotate new arrow by random theta\n",
    "    theta = np.random.uniform(0, 2 * np.pi)\n",
    "    rotation_matrix = np.array([[np.cos(theta), -np.sin(theta)], [np.sin(theta), np.cos(theta)]])\n",
    "    new_arrow = np.dot(new_arrow, rotation_matrix)\n",
    "    new_double_arrow = np.dot(new_double_arrow, rotation_matrix)\n",
    "\n",
    "    return new_arrow, new_double_arrow"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 12,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "(-0.5, 0.5)"
      ]
     },
     "execution_count": 12,
     "metadata": {},
     "output_type": "execute_result"
    },
    {
     "data": {
      "image/png": "iVBORw0KGgoAAAANSUhEUgAAA+EAAAH5CAYAAADuoz85AAAAOXRFWHRTb2Z0d2FyZQBNYXRwbG90bGliIHZlcnNpb24zLjcuMiwgaHR0cHM6Ly9tYXRwbG90bGliLm9yZy8pXeV/AAAACXBIWXMAAA9hAAAPYQGoP6dpAAA930lEQVR4nO3df4xdZZ0/8M91SstCmDFSKYUZKZIVUFCXEqAkE8EvFlxdq7MNYM3gusaVIGtLXRHW3QXNJo2Y1XajxR/rj2igS+yOxk0I0kTqTmyBBWF1Ayphi7RlKkJghkik5fZ8/zje0uncmTv33nPPPffe1yuZTHrm3JmnB3rf8znn+TxPKUmSJAAAAICWe1W7BwAAAAC9QhEOAAAAOVGEAwAAQE4U4QAAAJATRTgAAADkRBEOAAAAOVGEAwAAQE4WtHsAWTt48GA89dRTcdxxx0WpVGr3cAAgkiSJF154IU466aR41avc/86CvAegSOrJ+q4rwp966qkYGhpq9zAAYIbdu3fH4OBgu4fRFeQ9AEU0n6zvuiL8uOOOi4j0L9/f39/m0QBAxNTUVAwNDR3KKJon7+lU5XLEjh0R+/ZFnHhixIUXRvT1tXtUQLPqyfquK8IrU9L6+/uFMgCFYtp0duQ9nWhsLGLt2og9e145NjgYsWlTxMhI+8YFZGc+Wa8xDQAAWmxsLGL16ukFeETE3r3p8bGx9owLyJ8iHAAAWqhcTp+AJ8nMr1WOrVuXngd0P0U4AAC00Pj4zCfgh0uSiN270/OA7qcIBwCAFpqYyPY8oLMpwgEAoIWWLs32PKCzKcIBAKCFhofTVdBnWzS5VIoYGkrPA7qfIhwAAFqory/dhixiZiFe+fPGjfYLh16hCAcAgBYbGYnYujXi5JOnHx8cTI/bJxx6x4J2DwAAAHrByEjEqlXpKugTE2kP+PCwJ+DQaxThAACQk76+iIsuavcogHYyHR0AAAByoggHAACAnCjCAQAAICeKcAAAAMiJIhwAAAByoggHAACAnCjCAQAAICeKcAAAAMiJIhwAAAByoggHAACAnCjCAQAAICeKcAAAAMiJIhwAAAByoggHAACAnCjCAQAAICeKcAAAAMiJIhwAAAByoggHAACAnCjCAQAAICeKcAAAAMiJIhwAAAByoggHAACAnCjCAQAAICeKcAAAAMiJIhwAAAByoggHAACAnCjCAQAAICeKcAAAAMiJIhwAAAByoggHAACAnLS8CN+8eXOceuqpcfTRR8fy5ctjfHx8Xq/76U9/GgsWLIi3vvWtrR0gAFEuR2zfHrFlS/q5XG73iOg08h6g+OR9MbS0CL/jjjti3bp18elPfzoeeuihGB4ejne+853x5JNPzvm6ycnJuOqqq+L//b//18rhARARY2MRy5ZFXHxxxJo16edly9LjMB/yHqD45H1xlJIkSVr1zc8///w455xz4tZbbz107Mwzz4z3vve9sWHDhllfd+WVV8af/umfRl9fX/zgBz+Ihx9+eNZzX3rppXjppZcO/XlqaiqGhoZicnIy+vv7M/l7AHSrsbGI1asjjkyCUin9vHVrxMhI/uPqNlNTUzEwMNC12STvAYpN3rdePVnfsifh+/fvjwcffDBWrlw57fjKlStjx44ds77uW9/6Vjz++ONx0003zevnbNiwIQYGBg59DA0NNTVugF5RLkesXTszkCNeObZunalqzE3eAxSbvC+elhXhzzzzTJTL5ViyZMm040uWLIl9+/ZVfc1jjz0WN9xwQ9x2222xYMGCef2cG2+8MSYnJw997N69u+mxA/SC8fGIPXtm/3qSROzenZ4Hs5H3AMUm74tnfsnXhFJljsMfJUky41hERLlcjjVr1sRnPvOZeMMb3jDv779o0aJYtGhR0+ME6DUTE9meR2+T9wDFJO+Lp2VF+OLFi6Ovr2/GXfCnn356xt3yiIgXXnghHnjggXjooYfi2muvjYiIgwcPRpIksWDBgrj77rvj7W9/e6uGC9Bzli7N9jx6k7wHKDZ5Xzwtm46+cOHCWL58eWzbtm3a8W3btsWFF1444/z+/v74xS9+EQ8//PChj6uvvjpOP/30ePjhh+P8889v1VABetLwcMTg4CuLshypVIoYGkrPg9nIe4Bik/fF09Lp6OvXr4/R0dE499xzY8WKFfG1r30tnnzyybj66qsjIu3v2rt3b3znO9+JV73qVXHWWWdNe/0JJ5wQRx999IzjADSvry9i06Z0tdRSafqCLZWg3rgxPQ/mIu8BikveF09Li/Arrrginn322fjsZz8bExMTcdZZZ8Wdd94Zp5xySkRETExM1NxDFIDWGRlJtyVZu3b6oi2Dg2kg266E+ZD3AMUm74ulpfuEt0O378UK0Arlcroq6sRE2hM2POyOeJZk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Ouqo5Gc/+1nOIy+ueq9pxfPPP5+8/vWvT1auXJm85S1vyWew8EfyPluyvjmyPluyPnu9mPWK8Jw88sgjSUQk995776FjO3fuTCIi+eUvfznna/fs2ZOcfPLJyf/+7/8mp5xyilD+o2au6eHuv//+JCJq/kPvRuedd15y9dVXTzt2xhlnJDfccEPV86+//vrkjDPOmHbsox/9aHLBBRe0bIydpN7rWc0b3/jG5DOf+UzWQ+tYjV7TK664IvmHf/iH5Kabbuq4YKazyftsyfrmyfpsyfrs9WLWm46ek507d8bAwECcf/75h45dcMEFMTAwEDt27Jj1dQcPHozR0dH45Cc/GW9605vyGGrHaPSaHmlycjJKpVK8+tWvbsEoi2v//v3x4IMPxsqVK6cdX7ly5azXb+fOnTPOv/TSS+OBBx6IAwcOtGysnaCR63mkgwcPxgsvvBCvec1rWjHEjtPoNf3Wt74Vjz/+eNx0002tHiLMIO+zJeubI+uzJeuz16tZv6DdA+gV+/btixNOOGHG8RNOOCH27ds36+s+97nPxYIFC+LjH/94K4fXkRq9pof7wx/+EDfccEOsWbMm+vv7sx5ioT3zzDNRLpdjyZIl044vWbJk1uu3b9++que//PLL8cwzz8TSpUtbNt6ia+R6Hulf/uVf4ve//31cfvnlrRhix2nkmj722GNxww03xPj4eCxYIOLIn7zPlqxvjqzPlqzPXq9mvSfhTbr55pujVCrN+fHAAw9ERESpVJrx+iRJqh6PiHjwwQdj06ZN8e1vf3vWc7pRK6/p4Q4cOBBXXnllHDx4MDZv3pz536NTHHmtal2/audXO96r6r2eFVu2bImbb7457rjjjqq/cPay+V7Tcrkca9asic985jPxhje8Ia/h0SPkfbZkfb5kfbZkffZ6Les789ZBgVx77bVx5ZVXznnOsmXL4uc//3n89re/nfG13/3udzPu/FSMj4/H008/Ha973esOHSuXy/GJT3wiNm7cGE888URTYy+qVl7TigMHDsTll18eu3btih//+Mc9d2c8ImLx4sXR19c34y7j008/Pev1O/HEE6uev2DBgjj++ONbNtZO0Mj1rLjjjjviwx/+cHzve9+LSy65pJXD7Cj1XtMXXnghHnjggXjooYfi2muvjYh02l+SJLFgwYK4++674+1vf3suY6f7yPtsyfp8yPpsyfrs9WrWK8KbtHjx4li8eHHN81asWBGTk5Nx//33x3nnnRcREffdd19MTk7GhRdeWPU1o6OjM/6RXnrppTE6Ohof+tCHmh98QbXymka8EsqPPfZY3HPPPT0bKAsXLozly5fHtm3b4n3ve9+h49u2bYtVq1ZVfc2KFSviP//zP6cdu/vuu+Pcc8+No446qqXjLbpGrmdEelf8r//6r2PLli3xrne9K4+hdox6r2l/f3/84he/mHZs8+bN8eMf/zi2bt0ap556asvHTPeS99mS9fmQ9dmS9dnr2axvx2pwveqyyy5L3vzmNyc7d+5Mdu7cmZx99tkzttg4/fTTk7GxsVm/h9VSp6v3mh44cCB5z3vekwwODiYPP/xwMjExcejjpZdeasdfoa0qW0J84xvfSB555JFk3bp1ybHHHps88cQTSZIkyQ033JCMjo4eOr+ybcl1112XPPLII8k3vvEN25Ycpt7refvttycLFixIvvzlL0/7f/H5559v11+hcOq9pkfqxBVT6XzyPluyvjmyPluyPnu9mPWK8Bw9++yzyQc+8IHkuOOOS4477rjkAx/4QPLcc89NOycikm9961uzfg+hPF2913TXrl1JRFT9uOeee3IffxF8+ctfTk455ZRk4cKFyTnnnJP85Cc/OfS1D37wg8nb3va2aedv3749+bM/+7Nk4cKFybJly5Jbb7015xEXWz3X821ve1vV/xc/+MEP5j/wAqv3/9HDdWIw0/nkfbZkffNkfbZkffZ6LetLSfLHlRYAAACAlrI6OgAAAOREEQ4AAAA5UYQDAABAThThAAAAkBNFOAAAAOREEQ4AAAA5UYQDAABAThThAAAAkBNFOAAAAOREEQ4AAAA5UYQDAABATv4/uoQJeAgZN6QAAAAASUVORK5CYII=",
      "text/plain": [
       "<Figure size 1200x600 with 2 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "# 2 x 1 subplot\n",
    "fig, ax = plt.subplots(1, 2, figsize=(12, 6))\n",
    "\n",
    "new_arrow, new_double_arrow = generate_arrow_pair()\n",
    "\n",
    "# Plot arrow\n",
    "ax[0].scatter(new_arrow[:, 0], new_arrow[:, 1], color='blue', label='Square')\n",
    "ax[0].set_xlim(-0.5, 0.5)\n",
    "ax[0].set_ylim(-0.5, 0.5)\n",
    "\n",
    "# Plot double arrow\n",
    "ax[1].scatter(new_double_arrow[:, 0], new_double_arrow[:, 1], color='blue', label='House')\n",
    "ax[1].set_xlim(-0.5, 0.5)\n",
    "ax[1].set_ylim(-0.5, 0.5)\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 13,
   "metadata": {},
   "outputs": [],
   "source": [
    "arrow_samples = []\n",
    "double_arrow_samples = []\n",
    "\n",
    "for i in range(20):\n",
    "    arrow_sample, double_arrow_sample = generate_arrow_pair()\n",
    "    arrow_samples.append(arrow_sample)\n",
    "    double_arrow_samples.append(double_arrow_sample)"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# Circles"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 14,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "(87, 2)"
      ]
     },
     "execution_count": 14,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "radius = 0.2\n",
    "radial_divisions = 6\n",
    "angular_divisions = 5 * radial_divisions\n",
    "\n",
    "points = []\n",
    "# Generate points for each radial division\n",
    "for r in np.linspace(0, radius, radial_divisions):\n",
    "    # Increasing the number of points with the radius\n",
    "    num_points = int(angular_divisions * r / radius) if r != 0 else 1\n",
    "    theta = np.linspace(0, 2 * np.pi, num_points, endpoint=False)\n",
    "    for t in theta:\n",
    "        points.append((r * np.cos(t), r * np.sin(t)))\n",
    "circle = np.array(points)\n",
    "\n",
    "circle.shape"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 15,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "(66, 2)"
      ]
     },
     "execution_count": 15,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "radius = 0.3\n",
    "radial_divisions = 5\n",
    "angular_divisions = 5 * radial_divisions\n",
    "\n",
    "points = []\n",
    "# Generate points for each radial division\n",
    "for r in np.linspace(0, radius, radial_divisions):\n",
    "    # Increasing the number of points with the radius\n",
    "    num_points = int(angular_divisions * r / radius) if r != 0 else 1\n",
    "    theta = np.linspace(-np.pi / 2, np.pi / 2, num_points, endpoint=False)\n",
    "    for t in theta:\n",
    "        points.append((r * np.cos(t), r * np.sin(t)))\n",
    "\n",
    "semi_circle = np.array(points)\n",
    "\n",
    "extra_points = np.logical_and(semi_circle[:, 0] <= 1e-3, semi_circle[:, 1] < 0)\n",
    "extra_points = semi_circle[extra_points]\n",
    "extra_points[:, 1] *= -1\n",
    "\n",
    "semi_circle = np.concatenate([semi_circle, extra_points], axis=0)\n",
    "semi_circle.shape"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 16,
   "metadata": {},
   "outputs": [],
   "source": [
    "circle_points = semi_circle[semi_circle[:, 0] > 1e-3]\n",
    "circle_points[:, 0] *= -1\n",
    "circle = np.concatenate([semi_circle, circle_points], axis=0)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 17,
   "metadata": {},
   "outputs": [],
   "source": [
    "def generate_circle_pair():\n",
    "    width_dilation = np.random.uniform(0.5, 1.5)\n",
    "    height_dilation = np.random.uniform(0.5, 1.5)\n",
    "\n",
    "    new_circle = np.copy(circle)\n",
    "    new_circle[:, 0] *= width_dilation\n",
    "    new_circle[:, 1] *= height_dilation\n",
    "\n",
    "    new_semi_circle = new_circle[new_circle[:, 0] >= 0]\n",
    "\n",
    "    # Rotate new circle by random theta\n",
    "    theta = np.random.uniform(0, 2 * np.pi)\n",
    "    rotation_matrix = np.array([[np.cos(theta), -np.sin(theta)], [np.sin(theta), np.cos(theta)]])\n",
    "    new_semi_circle = np.dot(new_semi_circle, rotation_matrix)\n",
    "    new_circle = np.dot(new_circle, rotation_matrix)\n",
    "\n",
    "    return new_semi_circle, new_circle"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 18,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "(-0.5, 0.5)"
      ]
     },
     "execution_count": 18,
     "metadata": {},
     "output_type": "execute_result"
    },
    {
     "data": {
      "image/png": "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",
      "text/plain": [
       "<Figure size 1200x600 with 2 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "# 2 x 1 subplot\n",
    "fig, ax = plt.subplots(1, 2, figsize=(12, 6))\n",
    "\n",
    "new_semi_circle, new_circle = generate_circle_pair()\n",
    "\n",
    "# Plot arrow\n",
    "ax[0].scatter(new_semi_circle[:, 0], new_semi_circle[:, 1], color='blue', label='Square')\n",
    "ax[0].set_xlim(-0.5, 0.5)\n",
    "ax[0].set_ylim(-0.5, 0.5)\n",
    "\n",
    "# Plot double arrow\n",
    "ax[1].scatter(new_circle[:, 0], new_circle[:, 1], color='blue', label='House')\n",
    "ax[1].set_xlim(-0.5, 0.5)\n",
    "ax[1].set_ylim(-0.5, 0.5)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 19,
   "metadata": {},
   "outputs": [],
   "source": [
    "semi_circle_samples = []\n",
    "circle_samples = []\n",
    "\n",
    "for i in range(20):\n",
    "    new_semi_circle, new_circle = generate_circle_pair()\n",
    "    semi_circle_samples.append(new_semi_circle)\n",
    "    circle_samples.append(new_circle)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 20,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "120"
      ]
     },
     "execution_count": 20,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "posns = square_samples + house_samples + arrow_samples + double_arrow_samples + semi_circle_samples + circle_samples\n",
    "np.savez('synthetic_data.npz', *posns)"
   ]
  },
  {
   "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.11.4"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 2
}
