{
 "cells": [
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# Figure 4: Split-Half Diagnostics\n",
    "\n",
    "Split-half stability diagnostic for PLS-SVD: split data into two halves, run PLS-SVD on each, measure consistency.\n",
    "\n",
    "**Key insight:** Split-half uses N/2 samples per half, so aspect ratios are halved. This shifts the threshold from θ_crit to √2·θ_crit ≈ 1.414·θ_crit. The plot shows both thresholds to distinguish three regimes:\n",
    "1. **No recovery** (θ < θ_crit): Signal too weak for any recovery\n",
    "2. **Recovery unstable** (θ_crit < θ < √2·θ_crit): Full data recovers signal, but halves are unreliable\n",
    "3. **Recovery stable** (θ > √2·θ_crit): Both full data and split-halves recover signal reliably"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 1,
   "metadata": {},
   "outputs": [],
   "source": [
    "import os\n",
    "import pickle\n",
    "import numpy as np\n",
    "import matplotlib.pyplot as plt\n",
    "from multiprocessing import Pool\n",
    "from tqdm.notebook import tqdm\n",
    "\n",
    "from src import _run_diagnostics_worker\n",
    "\n",
    "# ICML-compatible styling\n",
    "plt.rcParams.update({\n",
    "    'font.size': 9,\n",
    "    'axes.labelsize': 9,\n",
    "    'xtick.labelsize': 8,\n",
    "    'ytick.labelsize': 8,\n",
    "    'legend.fontsize': 7,\n",
    "    'font.family': 'serif',\n",
    "    'mathtext.fontset': 'cm',\n",
    "    'axes.spines.top': False,\n",
    "    'axes.spines.right': False,\n",
    "})\n",
    "\n",
    "np.random.seed(42)\n",
    "\n",
    "# Create directories\n",
    "os.makedirs('../results', exist_ok=True)\n",
    "os.makedirs('../figures', exist_ok=True)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Loading cached results...\n",
      "Done!\n"
     ]
    }
   ],
   "source": [
    "# Experiment parameters\n",
    "N = 2000\n",
    "alpha_x = 7.5\n",
    "alpha_y = 7.5\n",
    "Dx = int(N / alpha_x)\n",
    "Dy = int(N / alpha_y)\n",
    "mx = my = 0.1\n",
    "rho = (1 - mx) * (1 - my)\n",
    "theta_crit = 1 / ((alpha_x * alpha_y) ** 0.25 * np.sqrt(rho))\n",
    "theta_values = np.linspace(0.5 * theta_crit, 2.5 * theta_crit, 60)\n",
    "n_trials = 25\n",
    "\n",
    "# Check for cached results or run experiment\n",
    "os.makedirs('../results', exist_ok=True)\n",
    "cache_file = '../results/fig4_results.pkl'\n",
    "\n",
    "if os.path.exists(cache_file):\n",
    "    print(\"Loading cached results...\")\n",
    "    with open(cache_file, 'rb') as f:\n",
    "        results = pickle.load(f)\n",
    "    print(\"Done!\")\n",
    "else:\n",
    "    print(f\"Running experiment for N={N}...\")\n",
    "    \n",
    "    args_list = []\n",
    "    for theta in theta_values:\n",
    "        params_dict = {'N': N, 'Dx': Dx, 'Dy': Dy, 'theta': theta, 'mx': mx, 'my': my}\n",
    "        args_list.append((params_dict, n_trials, theta, theta_crit))\n",
    "    \n",
    "    with Pool() as pool:\n",
    "        results = list(tqdm(pool.imap(_run_diagnostics_worker, args_list),\n",
    "                           total=len(args_list), desc=\"θ sweep\"))\n",
    "    \n",
    "    with open(cache_file, 'wb') as f:\n",
    "        pickle.dump(results, f)\n",
    "    print(\"Results cached!\")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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",
      "text/plain": [
       "<Figure size 325x180 with 2 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\n",
      "Figure saved: ../figures/fig4_split_half_diagnostics.pdf\n",
      "Full-data threshold: θ/θ_crit = 1.0\n",
      "Split-half threshold: θ/θ_crit = √2 ≈ 1.414\n"
     ]
    }
   ],
   "source": [
    "from matplotlib.collections import LineCollection\n",
    "from scipy.interpolate import interp1d\n",
    "\n",
    "# Create single-column figure (3.25\" wide for ICML)\n",
    "fig, ax = plt.subplots(figsize=(3.25, 1.8))\n",
    "\n",
    "# Extract data\n",
    "theta_norm = np.array([r['theta_norm'] for r in results])\n",
    "stability = np.array([r['stability_x_mean'] for r in results])\n",
    "rx2 = np.array([r['Rx2_mean'] for r in results])\n",
    "\n",
    "# Interpolate to create smooth continuous line\n",
    "n_interp = 500  # Many more points for smooth gradient\n",
    "theta_interp = np.linspace(theta_norm.min(), theta_norm.max(), n_interp)\n",
    "stability_interp = interp1d(theta_norm, stability, kind='linear')(theta_interp)\n",
    "rx2_interp = interp1d(theta_norm, rx2, kind='linear')(theta_interp)\n",
    "\n",
    "# Create continuous line with color gradient using LineCollection\n",
    "points = np.array([theta_interp, stability_interp]).T.reshape(-1, 1, 2)\n",
    "segments = np.concatenate([points[:-1], points[1:]], axis=1)\n",
    "\n",
    "# Create LineCollection with colors based on interpolated R_x²\n",
    "norm = plt.Normalize(0, 1)\n",
    "lc = LineCollection(segments, cmap='RdYlGn', norm=norm)\n",
    "lc.set_array(rx2_interp)\n",
    "lc.set_linewidth(2.5)\n",
    "line = ax.add_collection(lc)\n",
    "\n",
    "# Two thresholds:\n",
    "# 1. Full-data threshold: θ/θ_crit = 1 (recovery possible with all data)\n",
    "# 2. Split-half threshold: θ/θ_crit = √2 ≈ 1.414 (each half has N/2 samples,\n",
    "#    so α_x, α_y are halved, giving θ_crit^half = √2 * θ_crit)\n",
    "theta_crit_ratio_split = np.sqrt(2)\n",
    "\n",
    "# Add both threshold lines\n",
    "ax.axvline(1.0, color='red', linestyle='--', linewidth=1.5, alpha=0.8,\n",
    "           label=r'Full-data $\\theta_{\\mathrm{crit}}$')\n",
    "ax.axvline(theta_crit_ratio_split, color='orange', linestyle=':', linewidth=1.5, alpha=0.8,\n",
    "           label=r'Split-half $\\theta_{\\mathrm{crit}}^{\\mathrm{half}}$')\n",
    "\n",
    "# Shade regions - three zones now\n",
    "theta_min, theta_max = theta_norm.min(), theta_norm.max()\n",
    "ax.fill_between([theta_min, 1], 0, 1, color='red', alpha=0.08)\n",
    "ax.fill_between([1, theta_crit_ratio_split], 0, 1, color='orange', alpha=0.08)\n",
    "ax.fill_between([theta_crit_ratio_split, theta_max], 0, 1, color='green', alpha=0.08)\n",
    "\n",
    "# Region labels\n",
    "ax.text(0.72, 0.40, 'No\\nrecovery', fontsize=5, ha='center', color='darkred')\n",
    "ax.text(1.20, 0.40, 'Recovery\\nunstable', fontsize=5, ha='center', color='darkorange')\n",
    "ax.text(1.95, 0.40, 'Recovery\\nstable', fontsize=5, ha='center', color='darkgreen')\n",
    "\n",
    "ax.set_xlabel(r'$\\theta / \\theta_{\\mathrm{crit}}$')\n",
    "ax.set_ylabel(r'Split-half stability')\n",
    "ax.set_xlim([theta_min, theta_max])\n",
    "ax.set_ylim([-0.05, 1.05])\n",
    "ax.legend(loc='upper right', frameon=False, fontsize=5)\n",
    "\n",
    "# Add colorbar\n",
    "cbar = plt.colorbar(line, ax=ax, fraction=0.046, pad=0.02)\n",
    "cbar.set_label(r'$R_x^2$', fontsize=8)\n",
    "cbar.ax.tick_params(labelsize=7)\n",
    "\n",
    "plt.tight_layout()\n",
    "plt.savefig('../figures/fig4_split_half_diagnostics.pdf', dpi=300, bbox_inches='tight')\n",
    "plt.show()\n",
    "\n",
    "print(\"\\nFigure saved: ../figures/fig4_split_half_diagnostics.pdf\")\n",
    "print(f\"Full-data threshold: θ/θ_crit = 1.0\")\n",
    "print(f\"Split-half threshold: θ/θ_crit = √2 ≈ {theta_crit_ratio_split:.3f}\")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": []
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": ".venv",
   "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.14.2"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 2
}
