{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 2,
   "id": "707324b6-5824-431d-9324-62d93fb1fa71",
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "OMP: Info #276: omp_set_nested routine deprecated, please use omp_set_max_active_levels instead.\n"
     ]
    }
   ],
   "source": [
    "import numpy as np\n",
    "from numpy import random\n",
    "import tensorflow as tf\n",
    "import tensorflow_probability as tfp\n",
    "import math\n",
    "import pandas as pd\n",
    "import itertools\n",
    "import matplotlib.pyplot as plt\n",
    "import torch\n",
    "from torch import nn\n",
    "from sklearn.model_selection import train_test_split\n",
    "from statsmodels.distributions.empirical_distribution import ECDF\n",
    "from scipy.stats import chi2\n",
    "import scipy.stats\n",
    "from scipy.linalg import toeplitz\n",
    "import random\n",
    "from scipy.spatial import distance\n",
    "import dcor\n",
    "from typing import List\n",
    "from matplotlib.colors import ListedColormap, LogNorm\n",
    "from scipy.stats import rankdata\n",
    "from scipy.stats import norm\n",
    "import scipy.stats as stats\n",
    "from scipy.stats import gumbel_r"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "942bd1cc-21a7-4b47-95e7-90cd775ccbfb",
   "metadata": {},
   "source": [
    "## CoBET & dCoBET"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 40,
   "id": "36350a18-aed1-4f74-8e68-b96f17b86dc6",
   "metadata": {
    "scrolled": true
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Saved Excel results to sim_results_n250_500_1000.xlsx using engine=openpyxl\n",
      "Saved results to sim_results_n250_500_1000.xlsx\n",
      "  transform   weights     b metric  value  d    n  K  theta  alpha  R_eval  \\\n",
      "0     trigU  identity  0.00  typeI  0.043  5  250  4      2   0.05    1000   \n",
      "1     trigU  identity  0.05  power  0.466  5  250  4      2   0.05    1000   \n",
      "2     trigU  identity  0.08  power  0.850  5  250  4      2   0.05    1000   \n",
      "3     trigU  identity  0.10  power  0.968  5  250  4      2   0.05    1000   \n",
      "4     trigU         J  0.00  typeI  0.072  5  250  4      2   0.05    1000   \n",
      "\n",
      "   seed  \n",
      "0   123  \n",
      "1   123  \n",
      "2   123  \n",
      "3   123  \n",
      "4   123  \n"
     ]
    }
   ],
   "source": [
    "import numpy as np\n",
    "import pandas as pd\n",
    "from scipy.stats import norm, rankdata, norm as norm_dist\n",
    "import itertools\n",
    "\n",
    "# ===========================\n",
    "# Clayton copula sampler \n",
    "# ===========================\n",
    "def clayton_copula_sample_nd(n, theta, d):\n",
    "    if theta == 0:\n",
    "        return np.random.uniform(0, 1, size=(n, d))\n",
    "    S = np.random.gamma(shape=1.0/theta, scale=1.0, size=n)  # (n,)\n",
    "    E = np.random.exponential(scale=1.0, size=(n, d))\n",
    "    U = (1.0 + E / S[:, None]) ** (-1.0/theta)\n",
    "    return U\n",
    "\n",
    "# ===========================\n",
    "# Transform families \n",
    "# ===========================\n",
    "def transform_trig_uniform(u, v, b):\n",
    "    Z = norm.ppf(u)\n",
    "    X = np.sin(Z)\n",
    "    Y = np.cos(b * X + v)\n",
    "    return X, Y\n",
    "\n",
    "def transform_expquad(u, v, b):\n",
    "    Z = norm.ppf(u)\n",
    "    X = np.exp(-Z ** 2)\n",
    "    Y = np.exp(-b * (X - 1.0) ** 2 + v)\n",
    "    return X, Y\n",
    "\n",
    "def transform_linear(u, v, b):\n",
    "    X = u\n",
    "    Y = b * X + v\n",
    "    return X, Y\n",
    "\n",
    "def transform_logquad(u, v, b):\n",
    "    # phase + amplitude modulation\n",
    "    Z = norm.ppf(u)\n",
    "    X_base = np.log1p(Z**2)\n",
    "    X = X_base / (1.0 + X_base)\n",
    "    Y = np.cos(b * X + v) * np.exp(-b * (X - 0.7) ** 2)\n",
    "    return X, Y\n",
    "\n",
    "TRANSFORM_MAP = {\n",
    "    \"trigU\":   transform_trig_uniform,\n",
    "    \"expquad\": transform_expquad,\n",
    "    \"linear\":  transform_linear,\n",
    "    \"logquad\": transform_logquad,\n",
    "}\n",
    "\n",
    "# ===========================\n",
    "# Dyadic bits + centered features\n",
    "# ===========================\n",
    "def bits_from_uniform(u, K):\n",
    "    M = 1 << K\n",
    "    z = np.minimum((u * M).astype(int), M - 1)   # 0..2^K-1\n",
    "    bits = np.array([((z >> (K - 1 - k)) & 1).astype(int) for k in range(K)])\n",
    "    return bits  # (K, n)\n",
    "\n",
    "def all_nonempty_subsets_indices(K):\n",
    "    idx = list(range(1, K+1))\n",
    "    out = []\n",
    "    for r in range(1, K+1):\n",
    "        out.extend(itertools.combinations(idx, r))\n",
    "    return out  # length = 2^K - 1\n",
    "\n",
    "def features_by_u(u, K, subsets):\n",
    "    bits = bits_from_uniform(u, K)   # (K, n)\n",
    "    n = u.shape[0]\n",
    "    F = np.empty((len(subsets), n), dtype=float)\n",
    "    for i, S in enumerate(subsets):\n",
    "        rows = [r-1 for r in S]\n",
    "        ind = np.prod(bits[rows, :], axis=0)\n",
    "        F[i, :] = ind.astype(float) - (2.0 ** (-len(S)))  # centered\n",
    "    return F\n",
    "\n",
    "def build_AB_features_from_uniforms_multi(xu_mat, yu_mat, K, subsets):\n",
    "    n, d = xu_mat.shape\n",
    "    A_blocks, B_blocks = [], []\n",
    "    for r in range(d):\n",
    "        A_blocks.append(features_by_u(xu_mat[:, r], K, subsets))\n",
    "        B_blocks.append(features_by_u(yu_mat[:, r], K, subsets))\n",
    "    A = np.vstack(A_blocks)  # (d*(2^K-1), n)\n",
    "    B = np.vstack(B_blocks)  # (d*(2^K-1), n)\n",
    "    return A, B\n",
    "\n",
    "# ===========================\n",
    "# Numeric J(K) and weights\n",
    "# ===========================\n",
    "def trapezoid_weights(x):\n",
    "    w = np.zeros_like(x)\n",
    "    dx = np.diff(x)\n",
    "    w[1:-1] = 0.5 * (dx[:-1] + dx[1:])\n",
    "    w[0]  = 0.5 * (x[1] - x[0])\n",
    "    w[-1] = 0.5 * (x[-1] - x[-2])\n",
    "    return w\n",
    "\n",
    "def J_numeric_K(K, t_min=1e-4, t_max=100.0, T=2001):\n",
    "    subsets = all_nonempty_subsets_indices(K)\n",
    "    m = len(subsets)\n",
    "    t = np.logspace(np.log10(t_min), np.log10(t_max), T)\n",
    "    w = trapezoid_weights(t) / (t ** 2)\n",
    "\n",
    "    inv_pows = np.array([1.0 / (2 ** r) for r in range(1, K + 1)])\n",
    "    P = np.empty((m, T))\n",
    "    for i, S in enumerate(subsets):\n",
    "        vals = np.ones_like(t)\n",
    "        inS = np.zeros(K, dtype=bool); inS[[r-1 for r in S]] = True\n",
    "        for r in range(K):\n",
    "            ang = t * inv_pows[r]\n",
    "            vals *= np.sin(ang) if inS[r] else np.cos(ang)\n",
    "        P[i, :] = vals\n",
    "\n",
    "    J = (P * w) @ P.T\n",
    "    J = 0.5 * (J + J.T)\n",
    "    return J, subsets\n",
    "\n",
    "def block_diag(*mats):\n",
    "    r = sum(m.shape[0] for m in mats); c = sum(m.shape[1] for m in mats)\n",
    "    out = np.zeros((r, c), dtype=float); i = j = 0\n",
    "    for m in mats:\n",
    "        rr, cc = m.shape\n",
    "        out[i:i+rr, j:j+cc] = m\n",
    "        i += rr; j += cc\n",
    "    return out\n",
    "\n",
    "def kron_block_diag(mat, times):\n",
    "    r, c = mat.shape\n",
    "    out = np.zeros((times*r, times*c), dtype=float)\n",
    "    for i in range(times):\n",
    "        rs = i*r; cs = i*c\n",
    "        out[rs:rs+r, cs:cs+c] = mat\n",
    "    return out\n",
    "\n",
    "def get_weights(K, mode, J_cached=None, subsets=None, d_dims=2):\n",
    "    if subsets is None:\n",
    "        subsets = all_nonempty_subsets_indices(K)\n",
    "    block_rows = len(subsets)\n",
    "    rows_per_side = d_dims * block_rows\n",
    "\n",
    "    if mode == \"identity\":\n",
    "        W_A = np.eye(rows_per_side)\n",
    "        W_B = np.eye(rows_per_side)\n",
    "        J = None\n",
    "    elif mode == \"J\":\n",
    "        if J_cached is None:\n",
    "            J, subs2 = J_numeric_K(K)\n",
    "            if subs2 != subsets:\n",
    "                idx_map = {S:i for i,S in enumerate(subs2)}\n",
    "                perm = [idx_map[S] for S in subsets]\n",
    "                J = J[np.ix_(perm, perm)]\n",
    "        else:\n",
    "            J = J_cached\n",
    "        W_A = kron_block_diag(J, d_dims)\n",
    "        W_B = kron_block_diag(J, d_dims)\n",
    "    else:\n",
    "        raise ValueError(\"weights mode must be 'identity' or 'J'\")\n",
    "\n",
    "    W_C = block_diag(W_A, W_B)\n",
    "    return W_A, W_B, W_C, subsets, J\n",
    "\n",
    "# ===========================\n",
    "# Full statistic\n",
    "# ===========================\n",
    "def compute_full_T(A, B, W_A, W_B, W_C):\n",
    "    n = A.shape[1]\n",
    "    KA = (A.T @ W_A) @ A\n",
    "    KB = (B.T @ W_B) @ B\n",
    "    C  = np.vstack((A, B))\n",
    "    KC = (C.T @ W_C) @ C\n",
    "\n",
    "    off = ~np.eye(n, dtype=bool)\n",
    "\n",
    "    # T1\n",
    "    T1 = (KA[off] * KB[off]).sum() / (n * (n - 1))\n",
    "\n",
    "    def sums(dot):\n",
    "        S1 = dot.sum() - np.trace(dot)\n",
    "        row_off = dot.sum(axis=1) - np.diag(dot)\n",
    "        S2 = np.sum(row_off ** 2)\n",
    "        S3 = (dot ** 2).sum() - np.trace(dot ** 2)\n",
    "        return S1, S2, S3\n",
    "\n",
    "    S1C, S2C, S3C = sums(KC)\n",
    "    S1A, S2A, S3A = sums(KA)\n",
    "    S1B, S2B, S3B = sums(KB)\n",
    "\n",
    "    # T2\n",
    "    T2 = ((S2C - S3C) - (S2A - S3A) - (S2B - S3B)) / (2 * n * (n - 1) * (n - 2))\n",
    "    # T3\n",
    "    term = lambda S1, S2, S3: (S1 ** 2) - 4 * (S2 - S3) - 2 * S3\n",
    "    T3 = (term(S1C, S2C, S3C) - term(S1A, S2A, S3A) - term(S1B, S2B, S3B)) \\\n",
    "         / (2 * n * (n - 1) * (n - 2) * (n - 3))\n",
    "\n",
    "    return T1 - 2 * T2 + T3\n",
    "\n",
    "# ===========================\n",
    "# Plug-in Var(~T1)\n",
    "# ===========================\n",
    "def plugin_var_tildeT1(A, B, W_A, W_B, unbiased=True):\n",
    "    n = A.shape[1]\n",
    "    A_c = A - A.mean(axis=1, keepdims=True)\n",
    "    B_c = B - B.mean(axis=1, keepdims=True)\n",
    "    denom = (n - 1) if unbiased else n\n",
    "    S_A = (A_c @ A_c.T) / denom\n",
    "    S_B = (B_c @ B_c.T) / denom\n",
    "    EA = np.trace(W_A @ S_A @ W_A @ S_A)\n",
    "    EB = np.trace(W_B @ S_B @ W_B @ S_B)\n",
    "    return (2.0 / (n * (n - 1))) * EA * EB\n",
    "\n",
    "# ===========================\n",
    "# Helpers\n",
    "# ===========================\n",
    "def ranks_to_uniforms(X):\n",
    "    n, d = X.shape\n",
    "    U = np.empty_like(X, dtype=float)\n",
    "    for j in range(d):\n",
    "        U[:, j] = rankdata(X[:, j]) / (n + 1.0)\n",
    "    return U\n",
    "\n",
    "def generate_once(n, theta, b, K, transform_key, subsets, d):\n",
    "    u = clayton_copula_sample_nd(n, theta, d)\n",
    "    v = clayton_copula_sample_nd(n, theta, d)\n",
    "    X, Y = TRANSFORM_MAP[transform_key](u, v, b=b)\n",
    "    xu = ranks_to_uniforms(X)\n",
    "    yu = ranks_to_uniforms(Y)\n",
    "    A, B = build_AB_features_from_uniforms_multi(xu, yu, K=K, subsets=subsets)\n",
    "    return A, B\n",
    "\n",
    "# ===========================\n",
    "# Runner: plugin_tildeT1 ONLY (general d)\n",
    "# ===========================\n",
    "def run_plugin_only(\n",
    "    n, theta, K, b_config, d=2,\n",
    "    weights_list=(\"identity\",\"J\"),\n",
    "    R_eval=200, alpha=0.05, seed=123,\n",
    "    reuse_J=True, unbiased_plugin=True\n",
    "):\n",
    "    results = []\n",
    "    zcrit = norm_dist.ppf(1 - alpha)\n",
    "    np.random.seed(seed)\n",
    "\n",
    "    subsets = all_nonempty_subsets_indices(K)\n",
    "    J_cached = None\n",
    "    if reuse_J and (\"J\" in weights_list):\n",
    "        J_cached, subs_J = J_numeric_K(K)\n",
    "        if subs_J != subsets:\n",
    "            idx_map = {S:i for i,S in enumerate(subs_J)}\n",
    "            perm = [idx_map[S] for S in subsets]\n",
    "            J_cached = J_cached[np.ix_(perm, perm)]\n",
    "\n",
    "    for transform_key, b_list in b_config.items():\n",
    "        if not isinstance(b_list, (list, tuple, np.ndarray)):\n",
    "            b_list = [b_list]\n",
    "\n",
    "        for weights_mode in weights_list:\n",
    "            W_A0, W_B0, W_C0, subsets_used, _ = get_weights(\n",
    "                K, weights_mode,\n",
    "                J_cached=(J_cached if weights_mode==\"J\" else None),\n",
    "                subsets=subsets,\n",
    "                d_dims=d\n",
    "            )\n",
    "\n",
    "            # Type I (b=0)\n",
    "            rej = 0\n",
    "            for _ in range(R_eval):\n",
    "                A, B = generate_once(n, theta, b=0.0, K=K, transform_key=transform_key, subsets=subsets_used, d=d)\n",
    "                T  = compute_full_T(A, B, W_A0, W_B0, W_C0)\n",
    "                vT = plugin_var_tildeT1(A, B, W_A0, W_B0, unbiased=unbiased_plugin)\n",
    "                Z  = T / np.sqrt(vT + 0.0)\n",
    "                rej += (Z > zcrit)\n",
    "            type1 = rej / R_eval\n",
    "            results.append({\"transform\": transform_key, \"weights\": weights_mode,\n",
    "                            \"b\": 0.0, \"metric\": \"typeI\", \"value\": type1, \"d\": d, \"n\": n,\n",
    "                            \"K\": K, \"theta\": theta, \"alpha\": alpha, \"R_eval\": R_eval, \"seed\": seed})\n",
    "\n",
    "            # Power for each b\n",
    "            for b in b_list:\n",
    "                rej = 0\n",
    "                for _ in range(R_eval):\n",
    "                    A, B = generate_once(n, theta, b=b, K=K, transform_key=transform_key, subsets=subsets_used, d=d)\n",
    "                    T  = compute_full_T(A, B, W_A0, W_B0, W_C0)\n",
    "                    vT = plugin_var_tildeT1(A, B, W_A0, W_B0, unbiased=unbiased_plugin)\n",
    "                    Z  = T / np.sqrt(vT + 0.0)\n",
    "                    rej += (Z > zcrit)\n",
    "                power = rej / R_eval\n",
    "                results.append({\"transform\": transform_key, \"weights\": weights_mode,\n",
    "                                \"b\": float(b), \"metric\": \"power\", \"value\": power, \"d\": d, \"n\": n,\n",
    "                                \"K\": K, \"theta\": theta, \"alpha\": alpha, \"R_eval\": R_eval, \"seed\": seed})\n",
    "    return results\n",
    "\n",
    "# ===========================\n",
    "# Multi-n runner + Excel writer (per-n b_config)\n",
    "# ===========================\n",
    "def run_multi_n_and_save(n_list, theta, d, K, b_config_by_n, weights_list,\n",
    "                         R_eval, alpha, seed, xlsx_path=\"sim_results_n250_500_1000.xlsx\"):\n",
    "    \"\"\"\n",
    "    b_config_by_n: dict mapping each n to its own b_config dict.\n",
    "    \"\"\"\n",
    "    all_rows = []\n",
    "    # try openpyxl, then xlsxwriter; else fall back to CSVs\n",
    "    writer = None\n",
    "    engine_used = None\n",
    "    try:\n",
    "        writer = pd.ExcelWriter(xlsx_path, engine=\"openpyxl\")\n",
    "        engine_used = \"openpyxl\"\n",
    "    except Exception:\n",
    "        try:\n",
    "            writer = pd.ExcelWriter(xlsx_path, engine=\"xlsxwriter\")\n",
    "            engine_used = \"xlsxwriter\"\n",
    "        except Exception:\n",
    "            engine_used = None\n",
    "\n",
    "    try:\n",
    "        for n in n_list:\n",
    "            if n not in b_config_by_n:\n",
    "                raise ValueError(f\"Missing b_config for n={n} in b_config_by_n\")\n",
    "            res_n = run_plugin_only(\n",
    "                n=n, theta=theta, K=K, b_config=b_config_by_n[n], d=d,\n",
    "                weights_list=weights_list, R_eval=R_eval, alpha=alpha,\n",
    "                seed=seed, reuse_J=True, unbiased_plugin=True\n",
    "            )\n",
    "            df_n = pd.DataFrame(res_n)\n",
    "            all_rows.extend(res_n)\n",
    "\n",
    "            if writer is not None:\n",
    "                df_n.to_excel(writer, index=False, sheet_name=f\"n={n}\")\n",
    "            else:\n",
    "                df_n.to_csv(f\"sim_results_n{n}.csv\", index=False)\n",
    "\n",
    "        if writer is not None:\n",
    "            pd.DataFrame(all_rows).to_excel(writer, index=False, sheet_name=\"combined\")\n",
    "            writer.close()\n",
    "            print(f\"Saved Excel results to {xlsx_path} using engine={engine_used}\")\n",
    "        else:\n",
    "            comb_path = \"sim_results_combined.csv\"\n",
    "            pd.DataFrame(all_rows).to_csv(comb_path, index=False)\n",
    "            print(f\"Excel engines unavailable; saved CSVs (combined at {comb_path})\")\n",
    "    finally:\n",
    "        if writer is not None:\n",
    "            try:\n",
    "                writer.close()\n",
    "            except Exception:\n",
    "                pass\n",
    "\n",
    "    return pd.DataFrame(all_rows)\n",
    "\n",
    "# ===========================\n",
    "# Main\n",
    "# ===========================\n",
    "if __name__ == \"__main__\":\n",
    "    # ---- adjustable parameters ----\n",
    "    n_list = [250, 500, 1000]\n",
    "    theta  = 2\n",
    "    d      = 5                 # dimension\n",
    "    K      = 4                 # dyadic depth\n",
    "    alpha  = 0.05\n",
    "    R_eval = 1000\n",
    "    seed   = 123\n",
    "\n",
    "    # Per-n b grids\n",
    "    b_config_by_n = {\n",
    "        250: {\n",
    "            \"trigU\":   [0.05, 0.08, 0.1],\n",
    "            \"expquad\": [0.10, 0.15,0.3],\n",
    "            \"linear\":  [0.1, 0.2,0.3],\n",
    "            \"logquad\": [0.1, 0.30, 0.5]   \n",
    "        },\n",
    "        500: {\n",
    "            \"trigU\":   [0.03, 0.05, 0.10],\n",
    "            \"expquad\": [0.07, 0.15, 0.2],\n",
    "            \"linear\":  [0.05, 0.10, 0.20],\n",
    "            \"logquad\": [0.10, 0.20, 0.3]   \n",
    "        },\n",
    "        1000: {\n",
    "            \"trigU\":   [0.01, 0.03, 0.05],\n",
    "            \"expquad\": [0.05, 0.07, 0.12],\n",
    "            \"linear\":  [0.05, 0.08, 0.15],\n",
    "            \"logquad\": [0.07, 0.15, 0.2]  \n",
    "        }\n",
    "    }\n",
    "\n",
    "    weights_list = (\"identity\", \"J\")\n",
    "\n",
    "    # Run and save\n",
    "    out_path = \"sim_results_n250_500_1000.xlsx\"\n",
    "    df_all = run_multi_n_and_save(\n",
    "        n_list=n_list, theta=theta, d=d, K=K,\n",
    "        b_config_by_n=b_config_by_n,\n",
    "        weights_list=weights_list, R_eval=R_eval,\n",
    "        alpha=alpha, seed=seed, xlsx_path=out_path\n",
    "    )\n",
    "    print(f\"Saved results to {out_path}\")\n",
    "    print(df_all.head())\n"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "e93a47a3-cc33-49a3-81c5-a9b0664c7e44",
   "metadata": {},
   "source": [
    "## Apply SNR result as weights and aggregate it as wa_dCoBET"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 39,
   "id": "1539e387-1b31-44f2-8b7c-24e7712613b2",
   "metadata": {
    "scrolled": false
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[OK] Wrote Excel file to: CoBET_aggregated_power.xlsx\n"
     ]
    }
   ],
   "source": [
    "import numpy as np\n",
    "import itertools\n",
    "from collections import defaultdict\n",
    "from scipy.stats import norm as norm_dist, rankdata\n",
    "\n",
    "# NEW: for saving results\n",
    "import pandas as pd\n",
    "from datetime import datetime\n",
    "import os\n",
    "\n",
    "# ===========================================\n",
    "# 1) Clayton copula (d-dimensional)\n",
    "# ===========================================\n",
    "def clayton_copula_sample_nd(n, theta, d):\n",
    "    \"\"\"\n",
    "    Returns U in R^{n x d} with Clayton(theta) dependence across columns.\n",
    "    theta = 0 -> independent uniforms.\n",
    "    \"\"\"\n",
    "    if theta == 0:\n",
    "        return np.random.uniform(0, 1, size=(n, d))\n",
    "    # Gamma–Exponential construction\n",
    "    S = np.random.gamma(shape=1.0/theta, scale=1.0, size=n)  # (n,)\n",
    "    E = np.random.exponential(scale=1.0, size=(n, d))\n",
    "    U = (1.0 + E / S[:, None]) ** (-1.0/theta)\n",
    "    return U\n",
    "\n",
    "# ===========================================\n",
    "# 2) Transform families (coordinate-wise)\n",
    "# ===========================================\n",
    "def transform_logquad(u, v, b):\n",
    "    \"\"\"\n",
    "    phase + amplitude modulation (coordinate-wise)\n",
    "    X in (0,1), Y is nonlinear in X with local bump near ~0.7\n",
    "    \"\"\"\n",
    "    Z = norm_dist.ppf(u)\n",
    "    X_base = np.log1p(Z**2)\n",
    "    X = X_base / (1.0 + X_base)\n",
    "    Y = np.cos(b * X + v) * np.exp(-b * (X - 0.7) ** 2)\n",
    "    return X, Y\n",
    "\n",
    "def transform_multidim(u, v, b, transform_key):\n",
    "    \"\"\"\n",
    "    Apply the chosen transform to each coordinate r=1..d independently.\n",
    "    Returns (X, Y) each shape (n, d)\n",
    "    \"\"\"\n",
    "    d = u.shape[1]\n",
    "    X = np.zeros_like(u, dtype=float)\n",
    "    Y = np.zeros_like(v, dtype=float)\n",
    "    for r in range(d):\n",
    "        u_r = u[:, [r]]  # (n,1)\n",
    "        v_r = v[:, [r]]  # (n,1)\n",
    "        if transform_key == \"trigU\":\n",
    "            # x = sin(N(0,1)); y = cos(b*x + U)\n",
    "            x_r = np.sin(norm_dist.ppf(u_r))\n",
    "            y_r = np.cos(b * x_r + v_r)\n",
    "        elif transform_key == \"expquad\":\n",
    "            # x = exp(-Z^2); y = exp(-b*(x-1)^2 + U)\n",
    "            x_r = np.exp(-norm_dist.ppf(u_r)**2)\n",
    "            y_r = np.exp(-b * (x_r - 1.0)**2 + v_r)\n",
    "        elif transform_key == \"linear\":\n",
    "            # x = U; y = b*x + U\n",
    "            x_r = u_r\n",
    "            y_r = b * x_r + v_r\n",
    "        elif transform_key == \"logquad\":\n",
    "            # NEW: phase + amplitude modulation\n",
    "            x_r, y_r = transform_logquad(u_r, v_r, b)\n",
    "        else:\n",
    "            raise ValueError(f\"Unknown transform: {transform_key}\")\n",
    "        X[:, r] = x_r[:, 0]\n",
    "        Y[:, r] = y_r[:, 0]\n",
    "    return X, Y\n",
    "\n",
    "# ===========================================\n",
    "# 3) Binary-expansion features\n",
    "# ===========================================\n",
    "def bits_from_uniform(u, K):\n",
    "    \"\"\"\n",
    "    u: 1D array of uniforms of length n\n",
    "    Returns bits shape (K, n), most significant bit first\n",
    "    \"\"\"\n",
    "    M = 1 << K\n",
    "    z = np.minimum((u * M).astype(int), M - 1)   # 0..2^K-1\n",
    "    bits = np.array([((z >> (K - 1 - k)) & 1).astype(int) for k in range(K)])\n",
    "    return bits  # (K, n)\n",
    "\n",
    "def all_nonempty_subsets_indices(K):\n",
    "    idx = list(range(1, K+1))\n",
    "    out = []\n",
    "    for r in range(1, K+1):\n",
    "        out.extend(itertools.combinations(idx, r))\n",
    "    return out  # length = 2^K - 1\n",
    "\n",
    "def features_by_u(u, K, subsets):\n",
    "    \"\"\"\n",
    "    u: 1D uniforms length n\n",
    "    Returns feature matrix (2^K - 1, n) with centered indicators\n",
    "    \"\"\"\n",
    "    bits = bits_from_uniform(u, K)   # (K, n)\n",
    "    n = u.shape[0]\n",
    "    F = np.empty((len(subsets), n), dtype=float)\n",
    "    for i, S in enumerate(subsets):\n",
    "        rows = [r-1 for r in S]\n",
    "        ind = np.prod(bits[rows, :], axis=0)\n",
    "        F[i, :] = ind.astype(float) - (2.0 ** (-len(S)))  # centered\n",
    "    return F\n",
    "\n",
    "def build_AB_features_from_multidim(X, Y, K, subsets):\n",
    "    \"\"\"\n",
    "    X, Y: n x d arrays.\n",
    "    Returns A, B with shape (d*(2^K -1), n)\n",
    "    \"\"\"\n",
    "    n, d = X.shape\n",
    "    feats_A = []\n",
    "    feats_B = []\n",
    "    for r in range(d):\n",
    "        xu = rankdata(X[:, r]) / (n + 1)\n",
    "        yu = rankdata(Y[:, r]) / (n + 1)\n",
    "        F_A = features_by_u(xu, K, subsets)\n",
    "        F_B = features_by_u(yu, K, subsets)\n",
    "        feats_A.append(F_A)\n",
    "        feats_B.append(F_B)\n",
    "    A = np.vstack(feats_A)\n",
    "    B = np.vstack(feats_B)\n",
    "    return A, B\n",
    "\n",
    "# ===========================================\n",
    "# 4) Numeric J(K) construction + weights\n",
    "# ===========================================\n",
    "def trapezoid_weights(x):\n",
    "    w = np.zeros_like(x)\n",
    "    dx = np.diff(x)\n",
    "    w[1:-1] = 0.5 * (dx[:-1] + dx[1:])\n",
    "    w[0]  = 0.5 * (x[1] - x[0])\n",
    "    w[-1] = 0.5 * (x[-1] - x[-2])\n",
    "    return w\n",
    "\n",
    "def J_numeric_K(K, t_min=1e-4, t_max=100.0, T=2001):\n",
    "    \"\"\"\n",
    "    Builds the (2^K - 1) x (2^K - 1) J matrix using numeric integration.\n",
    "    \"\"\"\n",
    "    subsets = all_nonempty_subsets_indices(K)\n",
    "    m = len(subsets)\n",
    "    t = np.logspace(np.log10(t_min), np.log10(t_max), T)\n",
    "    w = trapezoid_weights(t) / (t ** 2)\n",
    "\n",
    "    inv_pows = np.array([1.0 / (2 ** r) for r in range(1, K + 1)])  # r=1..K\n",
    "    P = np.empty((m, T))\n",
    "    for i, S in enumerate(subsets):\n",
    "        vals = np.ones_like(t)\n",
    "        inS = np.zeros(K, dtype=bool); inS[[r-1 for r in S]] = True\n",
    "        for r in range(K):\n",
    "            ang = t * inv_pows[r]\n",
    "            vals *= np.sin(ang) if inS[r] else np.cos(ang)\n",
    "        P[i, :] = vals\n",
    "\n",
    "    J = (P * w) @ P.T\n",
    "    J = 0.5 * (J + J.T)\n",
    "    return J, subsets\n",
    "\n",
    "def block_diag(*mats):\n",
    "    r = sum(m.shape[0] for m in mats); c = sum(m.shape[1] for m in mats)\n",
    "    out = np.zeros((r, c), dtype=float); i = j = 0\n",
    "    for m in mats:\n",
    "        rr, cc = m.shape\n",
    "        out[i:i+rr, j:j+cc] = m\n",
    "        i += rr; j += cc\n",
    "    return out\n",
    "\n",
    "def get_weights(K, mode, J_cached=None, subsets=None, d_coords=10):\n",
    "    \"\"\"\n",
    "    Returns W_A, W_B, W_C for d_coords coordinates per side.\n",
    "    Each coordinate contributes (2^K - 1) rows, so total rows per side: d_coords*(2^K - 1)\n",
    "    \"\"\"\n",
    "    if subsets is None:\n",
    "        subsets = all_nonempty_subsets_indices(K)\n",
    "    base_dim = (1 << K) - 1\n",
    "    dim_side = d_coords * base_dim\n",
    "\n",
    "    if mode == \"identity\":\n",
    "        W_A = np.eye(dim_side); W_B = np.eye(dim_side); J = None\n",
    "\n",
    "    elif mode == \"J\":\n",
    "        if J_cached is None:\n",
    "            J1, subs2 = J_numeric_K(K)\n",
    "            if subs2 != subsets:\n",
    "                idx_map = {S:i for i,S in enumerate(subs2)}\n",
    "                perm = [idx_map[S] for S in subsets]\n",
    "                J1 = J1[np.ix_(perm, perm)]\n",
    "        else:\n",
    "            J1 = J_cached\n",
    "        # Repeat J block-diagonally for each coordinate\n",
    "        W_A = block_diag(*([J1] * d_coords))\n",
    "        W_B = block_diag(*([J1] * d_coords))\n",
    "    else:\n",
    "        raise ValueError(\"weights mode must be 'identity' or 'J'\")\n",
    "\n",
    "    W_C = block_diag(W_A, W_B)\n",
    "    return W_A, W_B, W_C\n",
    "\n",
    "# ===========================================\n",
    "# 5) Full statistic (T) and plug-in variance\n",
    "# ===========================================\n",
    "def compute_full_T(A, B, W_A, W_B, W_C):\n",
    "    n = A.shape[1]\n",
    "    KA = (A.T @ W_A) @ A\n",
    "    KB = (B.T @ W_B) @ B\n",
    "    C  = np.vstack((A, B))\n",
    "    KC = (C.T @ W_C) @ C  # = KA + KB for block-diag weights\n",
    "\n",
    "    off = ~np.eye(n, dtype=bool)\n",
    "\n",
    "    # T1\n",
    "    T1 = (KA[off] * KB[off]).sum() / (n * (n - 1))\n",
    "\n",
    "    def sums(dot):\n",
    "        S1 = dot.sum() - np.trace(dot)\n",
    "        row_off = dot.sum(axis=1) - np.diag(dot)\n",
    "        S2 = np.sum(row_off ** 2)\n",
    "        S3 = (dot ** 2).sum() - np.trace(dot ** 2)\n",
    "        return S1, S2, S3\n",
    "\n",
    "    S1C, S2C, S3C = sums(KC)\n",
    "    S1A, S2A, S3A = sums(KA)\n",
    "    S1B, S2B, S3B = sums(KB)\n",
    "\n",
    "    # T2\n",
    "    T2 = ((S2C - S3C) - (S2A - S3A) - (S2B - S3B)) / (2 * n * (n - 1) * (n - 2))\n",
    "    # T3\n",
    "    term = lambda S1, S2, S3: (S1 ** 2) - 4 * (S2 - S3) - 2 * S3\n",
    "    T3 = (term(S1C, S2C, S3C) - term(S1A, S2A, S3A) - term(S1B, S2B, S3B)) \\\n",
    "         / (2 * n * (n - 1) * (n - 2) * (n - 3))\n",
    "\n",
    "    return T1 - 2 * T2 + T3\n",
    "\n",
    "def plugin_var_tildeT1(A, B, W_A, W_B, unbiased=True):\n",
    "    \"\"\"\n",
    "    Plug-in variance for ~T1 with centered features (per dataset).\n",
    "    \"\"\"\n",
    "    n = A.shape[1]\n",
    "    A_c = A - A.mean(axis=1, keepdims=True)\n",
    "    B_c = B - B.mean(axis=1, keepdims=True)\n",
    "    denom = (n - 1) if unbiased else n\n",
    "    S_A = (A_c @ A_c.T) / denom\n",
    "    S_B = (B_c @ B_c.T) / denom\n",
    "    EA = np.trace(W_A @ S_A @ W_A @ S_A)\n",
    "    EB = np.trace(W_B @ S_B @ W_B @ S_B)\n",
    "    return (2.0 / (n * (n - 1))) * EA * EB\n",
    "\n",
    "# ===========================================\n",
    "# 6) Data generator \n",
    "# ===========================================\n",
    "def generate_once_nd(n, theta, b, K, transform_key, subsets, d=10):\n",
    "    u = clayton_copula_sample_nd(n, theta, d)\n",
    "    v = clayton_copula_sample_nd(n, theta, d)\n",
    "    X, Y = transform_multidim(u, v, b=b, transform_key=transform_key)\n",
    "    A, B = build_AB_features_from_multidim(X, Y, K=K, subsets=subsets)\n",
    "    return A, B\n",
    "\n",
    "# ===========================================\n",
    "# 7) SNR helpers for identity vs J\n",
    "# ===========================================\n",
    "def _precache_weights_for_candidates(K, subsets, d_coords=10, reuse_J=True):\n",
    "    \"\"\"Pre-compute weight matrices for 'identity' and 'J' once.\"\"\"\n",
    "    J_cached = None\n",
    "    if reuse_J:\n",
    "        J_cached, subsJ = J_numeric_K(K)\n",
    "        if subsJ != subsets:\n",
    "            idx = {S:i for i,S in enumerate(subsJ)}\n",
    "            perm = [idx[S] for S in subsets]\n",
    "            J_cached = J_cached[np.ix_(perm, perm)]\n",
    "\n",
    "    W = {}\n",
    "    for mode in (\"identity\", \"J\"):\n",
    "        W_A, W_B, W_C = get_weights(\n",
    "            K, mode, J_cached=(J_cached if mode==\"J\" else None), subsets=subsets, d_coords=d_coords\n",
    "        )\n",
    "        W[mode] = (W_A, W_B, W_C, subsets)\n",
    "    return W\n",
    "\n",
    "def _Z_for_mode(A, B, W_tuple, unbiased_plugin=True):\n",
    "    \"\"\"Standardized test (our plug-in SNR): Z = T / sqrt(Var_hat(~T1)).\"\"\"\n",
    "    W_A, W_B, W_C, _ = W_tuple\n",
    "    T  = compute_full_T(A, B, W_A, W_B, W_C)\n",
    "    vT = plugin_var_tildeT1(A, B, W_A, W_B, unbiased=unbiased_plugin)\n",
    "    return T / np.sqrt(max(vT, 1e-16))\n",
    "\n",
    "# ===========================================\n",
    "# 8) 10-fold selection, blending, and power\n",
    "# ===========================================\n",
    "def _ten_folds_indices(n, rng):\n",
    "    idx = rng.permutation(n)\n",
    "    folds = np.array_split(idx, 10)\n",
    "    return folds\n",
    "\n",
    "def _blend_weights(W_id, W_J, w_id, w_J):\n",
    "    W_A_id, W_B_id, _, _ = W_id\n",
    "    W_A_J,  W_B_J,  _, _ = W_J\n",
    "    W_A_new = w_id * W_A_id + w_J * W_A_J\n",
    "    W_B_new = w_id * W_B_id + w_J * W_B_J\n",
    "    W_C_new = block_diag(W_A_new, W_B_new)\n",
    "    return (W_A_new, W_B_new, W_C_new, None)\n",
    "\n",
    "def _fold_pick_by_snr(A_fold, B_fold, W_id, W_J, unbiased_plugin=True):\n",
    "    Z_id = _Z_for_mode(A_fold, B_fold, W_id, unbiased_plugin=unbiased_plugin)\n",
    "    Z_J  = _Z_for_mode(A_fold, B_fold, W_J,  unbiased_plugin=unbiased_plugin)\n",
    "    pick = \"identity\" if Z_id >= Z_J else \"J\"\n",
    "    return pick, Z_id, Z_J\n",
    "\n",
    "def aggregated_weights_power(\n",
    "    n, theta, K, transform_key, b,\n",
    "    R_eval=1000, alpha=0.05, seed=123, d_coords=10, unbiased_plugin=True, reuse_J=True\n",
    "):\n",
    "    \"\"\"\n",
    "    Per replicate:\n",
    "      - generate n samples of d_coords-dimensional U,V\n",
    "      - split into 10 equal folds\n",
    "      - on each fold choose 'identity' vs 'J' by larger Z (SNR)\n",
    "      - compute weights: w_id, w_J = selection frequencies / 10\n",
    "      - blend weights: W_new = w_id * W_id + w_J * W_J\n",
    "      - compute Z_new on FULL dataset with blended weights\n",
    "    Returns summary dict (power if b>0; type-I if b=0).\n",
    "    \"\"\"\n",
    "    rng = np.random.RandomState(seed)\n",
    "    zcrit = norm_dist.ppf(1 - alpha)\n",
    "\n",
    "    # Precompute subsets & both weight sets once\n",
    "    subsets = all_nonempty_subsets_indices(K)\n",
    "    W_all = _precache_weights_for_candidates(K, subsets, d_coords=d_coords, reuse_J=reuse_J)\n",
    "    W_id = W_all[\"identity\"]\n",
    "    W_J  = W_all[\"J\"]\n",
    "\n",
    "    rejections = 0\n",
    "    sel_counts_id = 0\n",
    "    sel_counts_J  = 0\n",
    "    Z_full_list   = []\n",
    "\n",
    "    for r in range(R_eval):\n",
    "        # fresh data\n",
    "        A, B = generate_once_nd(n, theta, b=b, K=K, transform_key=transform_key, subsets=subsets, d=d_coords)\n",
    "\n",
    "        # 10-fold selection on SAME dataset (columns = samples)\n",
    "        folds = _ten_folds_indices(n, rng)\n",
    "        fold_picks = []\n",
    "        for fidx in folds:\n",
    "            A_f = A[:, fidx]\n",
    "            B_f = B[:, fidx]\n",
    "            pick, _, _ = _fold_pick_by_snr(A_f, B_f, W_id, W_J, unbiased_plugin=unbiased_plugin)\n",
    "            fold_picks.append(pick)\n",
    "\n",
    "        # selection frequencies -> weights\n",
    "        cnt_id = sum(p == \"identity\" for p in fold_picks)\n",
    "        cnt_J  = 10 - cnt_id\n",
    "        w_id   = cnt_id / 10.0\n",
    "        w_J    = cnt_J  / 10.0\n",
    "        sel_counts_id += cnt_id\n",
    "        sel_counts_J  += cnt_J\n",
    "\n",
    "        # blend weight matrices\n",
    "        W_new = _blend_weights(W_id, W_J, w_id, w_J)\n",
    "\n",
    "        # compute test on FULL data with blended weights\n",
    "        W_A_new, W_B_new, W_C_new, _ = W_new\n",
    "        T_new  = compute_full_T(A, B, W_A_new, W_B_new, W_C_new)\n",
    "        vT_new = plugin_var_tildeT1(A, B, W_A_new, W_B_new, unbiased=unbiased_plugin)\n",
    "        Z_new  = T_new / np.sqrt(max(vT_new, 1e-16))\n",
    "        Z_full_list.append(Z_new)\n",
    "        if Z_new > zcrit:\n",
    "            rejections += 1\n",
    "\n",
    "    power_est = rejections / R_eval\n",
    "    return {\n",
    "        \"transform\": transform_key,\n",
    "        \"b\": float(b),\n",
    "        \"alpha\": alpha,\n",
    "        \"R_eval\": R_eval,\n",
    "        \"d\": d_coords,\n",
    "        \"n\": n,\n",
    "        \"power_aggregated\": power_est,\n",
    "        \"avg_w_identity\": sel_counts_id / (10.0 * R_eval),\n",
    "        \"avg_w_J\":        sel_counts_J  / (10.0 * R_eval),\n",
    "        \"Z_mean_full\":    float(np.mean(Z_full_list)),\n",
    "        \"Z_std_full\":     float(np.std(Z_full_list, ddof=1)),\n",
    "    }\n",
    "\n",
    "# ===========================================\n",
    "# 9) Table-2 style baseline (fixed identity vs J)\n",
    "# ===========================================\n",
    "def power_and_selection_one_setting(\n",
    "    n, theta, K, transform_key, b,\n",
    "    R_eval=1000, alpha=0.05, seed=123, d_coords=10, reuse_J=True, unbiased_plugin=True\n",
    "):\n",
    "    rng = np.random.RandomState(seed)\n",
    "    zcrit = norm_dist.ppf(1 - alpha)\n",
    "\n",
    "    subsets = all_nonempty_subsets_indices(K)\n",
    "    W = _precache_weights_for_candidates(K, subsets, d_coords=d_coords, reuse_J=reuse_J)\n",
    "\n",
    "    # tallies\n",
    "    rejs_by_mode = defaultdict(int)\n",
    "    Z_sums_by_mode = defaultdict(float)\n",
    "    select_counts  = defaultdict(int)\n",
    "    selected_rejs  = 0\n",
    "\n",
    "    for r in range(R_eval):\n",
    "        A, B = generate_once_nd(n, theta, b=b, K=K, transform_key=transform_key, subsets=subsets, d=d_coords)\n",
    "\n",
    "        # compute Z for both weightings (SNR plug-in)\n",
    "        Zs = {mode: _Z_for_mode(A, B, W[mode], unbiased_plugin=unbiased_plugin) for mode in (\"identity\",\"J\")}\n",
    "\n",
    "        # record fixed-mode power + Z means\n",
    "        for mode, Z in Zs.items():\n",
    "            if Z > zcrit: rejs_by_mode[mode] += 1\n",
    "            Z_sums_by_mode[mode] += Z\n",
    "\n",
    "        # selection by max SNR estimate\n",
    "        picked = max(Zs, key=Zs.get)\n",
    "        select_counts[picked] += 1\n",
    "        if Zs[picked] > zcrit: selected_rejs += 1\n",
    "\n",
    "    power_identity = rejs_by_mode[\"identity\"] / R_eval\n",
    "    power_J        = rejs_by_mode[\"J\"] / R_eval\n",
    "    selected_power = selected_rejs / R_eval\n",
    "\n",
    "    return {\n",
    "        \"transform\": transform_key,\n",
    "        \"b\": float(b),\n",
    "        \"alpha\": alpha,\n",
    "        \"R_eval\": R_eval,\n",
    "        \"d\": d_coords,\n",
    "        \"n\": n,\n",
    "        \"power_identity\": power_identity,\n",
    "        \"power_J\": power_J,\n",
    "        \"selected_power\": selected_power,\n",
    "        \"pct_selected_identity\": 100.0 * select_counts[\"identity\"] / R_eval,\n",
    "        \"pct_selected_J\":        100.0 * select_counts[\"J\"] / R_eval,\n",
    "        \"Z_mean_identity\": Z_sums_by_mode[\"identity\"] / R_eval,\n",
    "        \"Z_mean_J\":        Z_sums_by_mode[\"J\"] / R_eval,\n",
    "    }\n",
    "\n",
    "# ===========================================\n",
    "# 10) Batch runner + Excel export\n",
    "# ===========================================\n",
    "def run_full_grid_and_export(\n",
    "    n_list, b_config_by_n,\n",
    "    theta=2, K=4, d_coords=10,\n",
    "    transforms=(\"trigU\",\"expquad\",\"linear\",\"logquad\"),\n",
    "    R_eval=500, alpha=0.05, seed=123,\n",
    "    unbiased_plugin=True, reuse_J=True,\n",
    "    out_path=None, also_baseline=True\n",
    "):\n",
    "    \"\"\"\n",
    "    Runs aggregated-weights power across n_list and their b-configs,\n",
    "    and (optionally) baseline fixed-weight runs. Saves to Excel.\n",
    "    \"\"\"\n",
    "    aggregated_rows = []\n",
    "    baseline_rows   = []\n",
    "\n",
    "    for n in n_list:\n",
    "        config = b_config_by_n.get(n, {})\n",
    "        for tkey in transforms:\n",
    "            b_list = config.get(tkey, [])\n",
    "            for b in b_list:\n",
    "                # Aggregated-weights result\n",
    "                agg = aggregated_weights_power(\n",
    "                    n=n, theta=theta, K=K, transform_key=tkey, b=b,\n",
    "                    R_eval=R_eval, alpha=alpha, seed=seed,\n",
    "                    d_coords=d_coords, unbiased_plugin=unbiased_plugin, reuse_J=reuse_J\n",
    "                )\n",
    "                aggregated_rows.append(agg)\n",
    "\n",
    "                # Optional: fixed identity/J + selection\n",
    "                if also_baseline:\n",
    "                    base = power_and_selection_one_setting(\n",
    "                        n=n, theta=theta, K=K, transform_key=tkey, b=b,\n",
    "                        R_eval=R_eval, alpha=alpha, seed=seed,\n",
    "                        d_coords=d_coords, reuse_J=reuse_J, unbiased_plugin=unbiased_plugin\n",
    "                    )\n",
    "                    baseline_rows.append(base)\n",
    "\n",
    "    df_agg = pd.DataFrame(aggregated_rows)\n",
    "    df_base = pd.DataFrame(baseline_rows) if also_baseline else pd.DataFrame()\n",
    "\n",
    "    # Save to Excel (try xlsxwriter, then openpyxl; else CSV fallback)\n",
    "    if out_path is None:\n",
    "        ts = datetime.now().strftime(\"%Y%m%d_%H%M%S\")\n",
    "        out_path = f\"aggregated_results_{ts}.xlsx\"\n",
    "\n",
    "    try:\n",
    "        with pd.ExcelWriter(out_path, engine=\"xlsxwriter\") as writer:\n",
    "            df_agg.to_excel(writer, index=False, sheet_name=\"aggregated\")\n",
    "            if also_baseline and not df_base.empty:\n",
    "                df_base.to_excel(writer, index=False, sheet_name=\"baseline\")\n",
    "        print(f\"[OK] Wrote Excel file to: {out_path}\")\n",
    "    except Exception as e_xlsx:\n",
    "        try:\n",
    "            with pd.ExcelWriter(out_path, engine=\"openpyxl\") as writer:\n",
    "                df_agg.to_excel(writer, index=False, sheet_name=\"aggregated\")\n",
    "                if also_baseline and not df_base.empty:\n",
    "                    df_base.to_excel(writer, index=False, sheet_name=\"baseline\")\n",
    "            print(f\"[OK] Wrote Excel file to: {out_path}\")\n",
    "        except Exception as e_openpyxl:\n",
    "            # CSV fallback\n",
    "            base, _ = os.path.splitext(out_path)\n",
    "            csv_agg = base + \"_aggregated.csv\"\n",
    "            df_agg.to_csv(csv_agg, index=False)\n",
    "            print(f\"[WARN] Could not write .xlsx ({e_xlsx}; {e_openpyxl}). Wrote CSV: {csv_agg}\")\n",
    "            if also_baseline and not df_base.empty:\n",
    "                csv_base = base + \"_baseline.csv\"\n",
    "                df_base.to_csv(csv_base, index=False)\n",
    "                print(f\"[WARN] Baseline CSV: {csv_base}\")\n",
    "\n",
    "    return df_agg, df_base\n",
    "\n",
    "# ===========================================\n",
    "# 11) usage\n",
    "# ===========================================\n",
    "if __name__ == \"__main__\":\n",
    "    # Core settings\n",
    "    theta, K = 2, 5\n",
    "    d_coords = 10\n",
    "    alpha, seed = 0.05, 123\n",
    "    R_eval = 1000  # increase for smoother MC estimates\n",
    "\n",
    "    # Your requested grids\n",
    "    n_list = [250, 500, 1000]\n",
    "    b_config_by_n = {\n",
    "        250: {\n",
    "            \"trigU\":   [0, 0.05, 0.08, 0.1],\n",
    "            \"expquad\": [0, 0.10, 0.15, 0.3],\n",
    "            \"linear\":  [0, 0.10, 0.20, 0.30],\n",
    "            \"logquad\": [0, 0.10, 0.30, 0.50],\n",
    "        },\n",
    "        500: {\n",
    "            \"trigU\":   [0, 0.03, 0.05, 0.10],\n",
    "            \"expquad\": [0, 0.07, 0.15, 0.20],\n",
    "            \"linear\":  [0, 0.05, 0.10, 0.20],\n",
    "            \"logquad\": [0, 0.10, 0.20, 0.30],\n",
    "        },\n",
    "        1000: {\n",
    "            \"trigU\":   [0, 0.01, 0.03, 0.05],\n",
    "            \"expquad\": [0, 0.05, 0.07, 0.12],\n",
    "            \"linear\":  [0, 0.05, 0.08, 0.15],\n",
    "            \"logquad\": [0, 0.07, 0.15, 0.20],\n",
    "        },\n",
    "    }\n",
    "\n",
    "    # Run everything and save to Excel\n",
    "    run_full_grid_and_export(\n",
    "        n_list=n_list,\n",
    "        b_config_by_n=b_config_by_n,\n",
    "        theta=theta, K=K, d_coords=d_coords,\n",
    "        transforms=(\"trigU\",\"expquad\",\"linear\",\"logquad\"),\n",
    "        R_eval=R_eval, alpha=alpha, seed=seed,\n",
    "        unbiased_plugin=True, reuse_J=True,\n",
    "        out_path=\"CoBET_aggregated_power.xlsx\",\n",
    "        also_baseline=True\n",
    "    )\n"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "bc391934",
   "metadata": {},
   "source": [
    "## Visualization plots"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 93,
   "id": "b124384d-36e4-46bc-8d1b-41636b42cbab",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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oJBgIKsU+zQ2ttgTEREREZkAJvZERERERCaAXXE+IP6fhUkdglEod7tKHYLRyFrFSB2CUYjUVKlDMJqCdKwySyupQzAKoVBIHYLRmMnlUodgFMqC8j7Np11POI993mGLPRERERGRCWCLPREREREZjRBmUArjty0LCfZpbKZ/hEREREREBQATeyIiIiIiE8CuOERERERkNArIoIAE89hLsE9jY4s9EREREZEJYIs9ERERERmNUkgz9aRSGH2XRscWeyIiIiIiE8DEnoiIiIjIBLArDhEREREZjVKieeyl2Kexmf4REhEREREVAGyxJyIiIiKjUUIGpQRTT0qxT2Njiz0RERERkQlgiz0RERERGY1CyKCQYLpLKfZpbGyxJyIiIiIyAUzsiYiIiIhMALviEBEREZHRcLrLvGP6R0hEREREVAAwsSciIiIio1FCBqWQYNFzussnT57g008/hZubG2xsbFChQgWcP39etV0IgQkTJsDHxwc2NjZo2rQp7t27l9svl16Y2BMRERERveHVq1eoU6cOLC0tsXfvXty8eRNz586Fi4uLqsxPP/2EhQsXYvny5Thz5gzs7OzQokULpKSkSBY3+9gTEREREb1h1qxZ8PPzw5o1a1TrihYtqvq/EAILFizA+PHj0b59ewDAr7/+Ci8vL+zcuRPdunUzeswAW+yJiIiIyIjE/9951tiL+P+uOPHx8WpLamqqRoy7du1C1apV0aVLF3h6euKjjz7CqlWrVNvDwsIQGRmJpk2bqtY5OTmhRo0aOHXqVN6/iNlgYk8EICXmNWKvP5U6DCIiIspjfn5+cHJyUi0zZszQKBMaGoply5ahZMmS2L9/P7788ksMGzYM69atAwBERkYCALy8vNSe5+XlpdomBXbFIQIQfew+kp/Gwbm8r9ShEBERmbSswaxS7BcAIiIi4OjoqFovl8s1yyqVqFq1KqZPnw4A+Oijj3D9+nUsX74cvXv3Nk7ABmCLPRGA6KP3EX3sPoQQUodCREREecjR0VFt0ZbY+/j4oFy5cmrrypYti/DwcACAt7c3ACAqKkqtTFRUlGqbFCRN7GfMmIFq1arBwcEBnp6e+Pjjj3Hnzh0pQ6ICKPVlImKvP0VqTALibj6TOhwiIiKSWJ06dTRy0rt37yIgIABA5kBab29vHDp0SLU9Pj4eZ86cQa1atYwa65skTeyP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I1auXIkhQ4ZIGRYREeUTMUfvwa64B4KWfQrf1hXgXrs4qq3qBeeP/BBzlIk9EX1YNm/ejLCwMJ3Kvn79GitXrkRGRobO9Uvax75atWr4448/MG7cOEyZMgVFixbFggUL0LNnTynDIiKifECRnA5bPxcE/tAaZlb/fVzJ3e1R6afOCN98DslPY2Hj6yxdkESkt4I83WWVKlXQqlUrVKlSBRYWFrC0tETZsmXxzTff4JdffsGJEyeQkpICpVKJmzdvwtvbG97e3jrXL2liDwBt2rRBmzZtpA6DiIjyGXMbSxQfVF/rNpmZDAHdqxs5IiKi97NhwwasXLkST58+hVKpREpKCv73v/9h3759uH37Npo2bQonJyeYm5uja9eu6NWrl171S57YExEREVEBUoCnu/Tz88OPP/6oti48PBxFihTBX3/9hVatWr1X/Zz8l4iIiIhIIv7+/mjRogUqV6783nWxxZ6IiIiISEJ79+7NlXrYYk9ERERERpM1eFaK5UOkz5SXTOyJiIiIiPKp3377DfHx8TqVZVccIiIiIjKeAjx41hBC6B44W+yJiIiIiEwAE3siIiIiIhPArjhEREREZDQF+c6zeY0t9kREREREJoAt9kRERERkPBw8q5dPP/0Ujo6OOpVliz0RERERkZE0adIEO3bsyHb78+fPUaxYMdXjZcuWwd3dXae6mdgTERERERlJSEgIunbtiokTJ2rdrlAo8OjRI4PqZmJPRERERMYjJFzyiWXLlmHBggXo0KEDEhMTc61eJvZEREREREbUvn17nD59Gjdu3EDNmjURGhqaK/UysSciIiIio8ma7lKKJT8pW7Yszp07Bz8/P1SrVg0HDx587zqZ2BMRERERScDJyQl///03Bg4ciFatWmH+/PnvVR+nuyQiIiIiMhKZTKbxeObMmahcuTIGDBiAw4cPG1w3W+yJiIiIyHgK+OBZIbQH0q1bNxw/fhzXrl0zuG622BMRERERGUlISAhcXV21bqtcuTIuXLiAv//+26C6mdgTERERkdHIhIAsm1brvN5vftCgQYN3bndzc0OvXr0MqptdcYiIiIiITIBJtNiLDAWELEPqMPKcUCikDsEoZObmUodgNMpcvClFfmbu7CR1CEajiI2TOgSjsWj+ROoQjGLv4wtSh2A0wQHVpQ7BOIRS6giMI78ep1T93fNHg32eYos9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e/asunTpogULFujZZ59Vp06dNHv2bNOxatW0adO0evVq5eXlmY7iU5MmTdKWLVv0P//zP6aj2C47O1tr1qzRgQMH5HK5TMepNT//+c8VFxen3//+9959gwcPVnh4uJYuXWowWe07f/68IiMj9c4776h///7e/V27dlVGRoaeffZZg+ns9+3jLu8I/IWCXMG2X7/KqtQG99vX9LjLzMxMLVq0SLfffrtmz56tFStW6LPPPrtk7v2Ngo79Vaqrq6Nx9c6cOSPpYsHrZG63W2+99ZbKyspu6K7D9cjKylL//v2r/Xt1ogMHDigxMVEtWrTQiBEjdOTIEdORat27776rbt266d5771VsbKw6d+6s3/72t6Zj+VxFRYWWLl2q0aNHO6qol6QePXpo3bp12r9/vyRp9+7d2rx5szIyMgwnq31VVVVyu90KCwurtj88PNyRf2FzmqFDh+rXv/61nn76aXXq1El5eXl6//33b9iiXmLx7FWrq6ujcXU8Ho+ys7PVs2dPtWvXznQcn9izZ4/S0tJ04cIF1a9fX6tWrVLbtm1Nx6p1b731lj766CPHz2Ht3r27Fi9erNatW+vYsWP61a9+pZ/+9KfKz89XZGSk6Xi15osvvtDChQs1fvx4/cd//Id27NihRx99VCEhIcrMzDQdz2dWr16t06dPa+TIkaaj1LpJkyappKREKSkpCgwMlNvt1owZMzRixAjT0WpdZGSk0tLSNH36dLVp00ZxcXF68803tXXrVrVs2dJ0PGNML569FuPGjdO4ceN8kMY3KOwBXezw5ufnO7qD0rp1a+Xl5enMmTP605/+pMzMTG3cuNFRxX1RUZEee+wxrV279pIOmdN8t7vZoUMHde/eXU2bNtWKFSscNb3K4/GoW7dueu655yRJnTt3Vn5+vl599VVHF/a///3vlZGRocTERNNRat2KFSv0xhtvaNmyZUpNTVVeXp6ys7OVmJjoyO/p66+/rtGjR6tx48YKDAxUly5dNHz4cO3atct0NDgQhf1Vqquro3Fl48aN05o1a7Rp0yY1adLEdByfCQkJ8XaIunbtqh07dmjOnDlatGiR4WS1Z9euXTpx4oS6dOni3ed2u7Vp0ybNmzdP5eXlCgwMNJjQd2JiYtSqVSsdPHjQdJRalZCQcMkvn23atNHKlSsNJfK9w4cP669//avefvtt01F8YuLE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evelpKQoJibmimP77uskJCRIkrfT73a7NX36dLVv314NGzZU/fr1lZubqyNHjvzga+7evVsbNmxQ/fr1vVtKSook6fPPP6/xcx599FE9++yz6tmzp6ZOnapPPvnkitkBADcuCnsAN6ygoCAFBQX94Dnh4eG2ZAkPD5fL5aqV1woODvb+97ev+e1i3BdffFFz5szRL3/5S23YsEF5eXnq16+fKioqfvA1z549qwEDBigvL6/aduDAAfXq1avGzxkzZoy++OIL3X///dqzZ4+6deumuXPn1soYAQD2o7AHUKd16NBBR48e1f79+2s83qZNG23ZsqXavi1btqhVq1YKDAxUmzZtVFVVpW3btnmPf/PNNyooKFDbtm0ve92UlBRVVVVp165d3n0FBQU6ffr0dY1ny5YtGjhwoO677z517NhRLVq0uGRsISEhcrvd1fZ16dJFe/fuVbNmzdSyZctqW0RExGWvl5SUpH/913/V22+/rccff1y//e1vrys/AMAcCnsAdVrv3r3Vq1cvDR48WGvXrlVhYaHee+89vf/++5Kkxx9/XOvWrdP06dO1f/9+LVmyRPPmzdOECRMkSbfeeqsGDhyohx56SJs3b9bu3bt13333qXHjxho4cOBlr9u6dWvdfffdGjt2rLZt26Zdu3ZpzJgx1/0XhFtvvVVr167VBx98oH379mns2LEqLi6udk6zZs20bds2HTp0SCdPnpTH41FWVpb+/ve/a/jw4dqxY4c+//xz5ebmatSoUZf8EvCt7Oxs5ebmqrCwUB999JE2bNigNm3aXFd+AIA5FPYA6ryVK1fqtttu0/Dhw9W2bVs98cQT3mK2S5cuWrFihd566y21a9dOTz/9tJ555hmNHDnS+/k5OTnq2rWrfv7znystLU2WZekvf/lLtSkzNcnJyVFiYqJ69+6tX/ziF3r44YcVGxt7XWN56qmn1KVLF/Xr1099+vRRfHz8JWsMJkyYoMDAQLVt21Y333yzjhw5osTERG3ZskVut1t9+/ZV+/btlZ2drZiYGAUE1Pyj3u12KysrS23atNHdd9+tVq1aacGCBdeVHwBgjsuyLMt0CAAAAADXh449AAAA4AAU9gAAAIADUNgDAAAADkBhDwAAADgAhT0AAADgABT2AAAAgANQ2AMAAAAOQGEPAAAAOACFPQAAAOAAFPYAAACAA1DYAwAAAA7wv7wYpkDMIV+lAAAAAElFTkSuQmCC",
      "text/plain": [
       "<Figure size 800x700 with 2 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[BH-FDR] q=0.05, discoveries: 15 out of 100\n"
     ]
    }
   ],
   "source": [
    "import numpy as np\n",
    "import itertools\n",
    "import matplotlib.pyplot as plt\n",
    "from scipy.stats import norm as norm_dist, rankdata\n",
    "\n",
    "# ============================================================\n",
    "# 1) Clayton copula sampler (d-dimensional)\n",
    "# ============================================================\n",
    "def clayton_copula_sample_nd(n, theta, d, rng=None):\n",
    "    \"\"\"\n",
    "    Returns U in R^{n x d} with Clayton(theta) dependence across columns.\n",
    "    theta = 0 -> independent uniforms.\n",
    "    \"\"\"\n",
    "    if rng is None:\n",
    "        rng = np.random.RandomState(123)\n",
    "\n",
    "    if theta == 0:\n",
    "        return rng.uniform(0, 1, size=(n, d))\n",
    "\n",
    "    # Gamma–Exponential construction\n",
    "    S = rng.gamma(shape=1.0/theta, scale=1.0, size=n)  # (n,)\n",
    "    E = rng.exponential(scale=1.0, size=(n, d))\n",
    "    U = (1.0 + E / S[:, None]) ** (-1.0/theta)\n",
    "    return U\n",
    "\n",
    "\n",
    "# ============================================================\n",
    "# 2) Transform families (coordinate-wise)\n",
    "# ============================================================\n",
    "def transform_logquad(u, v, b):\n",
    "    \"\"\"\n",
    "    phase + amplitude modulation (coordinate-wise)\n",
    "    X in (0,1), Y is nonlinear in X with local bump near ~0.7\n",
    "    \"\"\"\n",
    "    Z = norm_dist.ppf(u)\n",
    "    X_base = np.log1p(Z**2)\n",
    "    X = X_base / (1.0 + X_base)\n",
    "\n",
    "    # NOTE: if b=0, Y = cos(v) (independent of X)\n",
    "    Y = np.cos(b * X + v) * np.exp(-b * (X - 0.7) ** 2)\n",
    "    return X, Y\n",
    "\n",
    "\n",
    "def transform_multidim(u, v, b, transform_key):\n",
    "    \"\"\"\n",
    "    Apply chosen transform independently per coordinate r=1..d.\n",
    "    Returns (X, Y) each shape (n, d)\n",
    "    \"\"\"\n",
    "    d = u.shape[1]\n",
    "    X = np.zeros_like(u, dtype=float)\n",
    "    Y = np.zeros_like(v, dtype=float)\n",
    "\n",
    "    for r in range(d):\n",
    "        u_r = u[:, [r]]  # (n,1)\n",
    "        v_r = v[:, [r]]  # (n,1)\n",
    "\n",
    "        if transform_key == \"trigU\":\n",
    "            x_r = np.sin(norm_dist.ppf(u_r))\n",
    "            y_r = np.cos(b * x_r + v_r)\n",
    "\n",
    "        elif transform_key == \"expquad\":\n",
    "            x_r = np.exp(-norm_dist.ppf(u_r)**2)\n",
    "            y_r = np.exp(-b * (x_r - 1.0)**2 + v_r)\n",
    "\n",
    "        elif transform_key == \"linear\":\n",
    "            x_r = u_r\n",
    "            y_r = b * x_r + v_r\n",
    "\n",
    "        elif transform_key == \"logquad\":\n",
    "            x_r, y_r = transform_logquad(u_r, v_r, b)\n",
    "\n",
    "        else:\n",
    "            raise ValueError(f\"Unknown transform: {transform_key}\")\n",
    "\n",
    "        X[:, r] = x_r[:, 0]\n",
    "        Y[:, r] = y_r[:, 0]\n",
    "\n",
    "    return X, Y\n",
    "\n",
    "\n",
    "# ============================================================\n",
    "# 3) Binary-expansion features\n",
    "# ============================================================\n",
    "def bits_from_uniform(u, K):\n",
    "    \"\"\"\n",
    "    u: 1D array of uniforms length n\n",
    "    Returns bits shape (K, n), most significant bit first\n",
    "    \"\"\"\n",
    "    M = 1 << K\n",
    "    z = np.minimum((u * M).astype(int), M - 1)  # 0..2^K-1\n",
    "    bits = np.array([((z >> (K - 1 - k)) & 1).astype(int) for k in range(K)])\n",
    "    return bits  # (K, n)\n",
    "\n",
    "\n",
    "def all_nonempty_subsets_indices(K):\n",
    "    idx = list(range(1, K + 1))\n",
    "    out = []\n",
    "    for r in range(1, K + 1):\n",
    "        out.extend(itertools.combinations(idx, r))\n",
    "    return out  # length 2^K - 1\n",
    "\n",
    "\n",
    "def features_by_u(u, K, subsets):\n",
    "    \"\"\"\n",
    "    u: 1D uniforms length n\n",
    "    Returns feature matrix (2^K - 1, n) centered indicators\n",
    "    \"\"\"\n",
    "    bits = bits_from_uniform(u, K)  # (K,n)\n",
    "    n = u.shape[0]\n",
    "    F = np.empty((len(subsets), n), dtype=float)\n",
    "\n",
    "    for i, S in enumerate(subsets):\n",
    "        rows = [r - 1 for r in S]\n",
    "        ind = np.prod(bits[rows, :], axis=0)\n",
    "        F[i, :] = ind.astype(float) - (2.0 ** (-len(S)))  # centered\n",
    "\n",
    "    return F\n",
    "\n",
    "\n",
    "def build_AB_features_from_multidim(X, Y, K, subsets):\n",
    "    \"\"\"\n",
    "    X, Y: n x d arrays.\n",
    "    Returns A, B with shape (d*(2^K -1), n)\n",
    "    \"\"\"\n",
    "    n, d = X.shape\n",
    "    feats_A = []\n",
    "    feats_B = []\n",
    "\n",
    "    for r in range(d):\n",
    "        xu = rankdata(X[:, r]) / (n + 1)\n",
    "        yu = rankdata(Y[:, r]) / (n + 1)\n",
    "        F_A = features_by_u(xu, K, subsets)\n",
    "        F_B = features_by_u(yu, K, subsets)\n",
    "        feats_A.append(F_A)\n",
    "        feats_B.append(F_B)\n",
    "\n",
    "    A = np.vstack(feats_A)\n",
    "    B = np.vstack(feats_B)\n",
    "    return A, B\n",
    "\n",
    "\n",
    "# ============================================================\n",
    "# 4) Numeric J(K) construction + weights\n",
    "# ============================================================\n",
    "def trapezoid_weights(x):\n",
    "    w = np.zeros_like(x)\n",
    "    dx = np.diff(x)\n",
    "    w[1:-1] = 0.5 * (dx[:-1] + dx[1:])\n",
    "    w[0] = 0.5 * (x[1] - x[0])\n",
    "    w[-1] = 0.5 * (x[-1] - x[-2])\n",
    "    return w\n",
    "\n",
    "\n",
    "def J_numeric_K(K, t_min=1e-4, t_max=100.0, T=2001):\n",
    "    \"\"\"\n",
    "    Builds the (2^K - 1) x (2^K - 1) J matrix using numeric integration.\n",
    "    \"\"\"\n",
    "    subsets = all_nonempty_subsets_indices(K)\n",
    "    m = len(subsets)\n",
    "    t = np.logspace(np.log10(t_min), np.log10(t_max), T)\n",
    "    w = trapezoid_weights(t) / (t ** 2)\n",
    "\n",
    "    inv_pows = np.array([1.0 / (2 ** r) for r in range(1, K + 1)])  # r=1..K\n",
    "    P = np.empty((m, T))\n",
    "\n",
    "    for i, S in enumerate(subsets):\n",
    "        vals = np.ones_like(t)\n",
    "        inS = np.zeros(K, dtype=bool)\n",
    "        inS[[r - 1 for r in S]] = True\n",
    "        for r in range(K):\n",
    "            ang = t * inv_pows[r]\n",
    "            vals *= np.sin(ang) if inS[r] else np.cos(ang)\n",
    "        P[i, :] = vals\n",
    "\n",
    "    J = (P * w) @ P.T\n",
    "    J = 0.5 * (J + J.T)\n",
    "    return J, subsets\n",
    "\n",
    "\n",
    "def block_diag(*mats):\n",
    "    r = sum(m.shape[0] for m in mats)\n",
    "    c = sum(m.shape[1] for m in mats)\n",
    "    out = np.zeros((r, c), dtype=float)\n",
    "    i = j = 0\n",
    "    for m in mats:\n",
    "        rr, cc = m.shape\n",
    "        out[i:i+rr, j:j+cc] = m\n",
    "        i += rr\n",
    "        j += cc\n",
    "    return out\n",
    "\n",
    "\n",
    "# ============================================================\n",
    "# 5) Full statistic (T) and plug-in variance\n",
    "# ============================================================\n",
    "def compute_full_T(A, B, W_A, W_B, W_C):\n",
    "    n = A.shape[1]\n",
    "    KA = (A.T @ W_A) @ A\n",
    "    KB = (B.T @ W_B) @ B\n",
    "    C = np.vstack((A, B))\n",
    "    KC = (C.T @ W_C) @ C  # = KA + KB when W_C is block-diag\n",
    "\n",
    "    off = ~np.eye(n, dtype=bool)\n",
    "\n",
    "    # T1\n",
    "    T1 = (KA[off] * KB[off]).sum() / (n * (n - 1))\n",
    "\n",
    "    def sums(dot):\n",
    "        S1 = dot.sum() - np.trace(dot)\n",
    "        row_off = dot.sum(axis=1) - np.diag(dot)\n",
    "        S2 = np.sum(row_off ** 2)\n",
    "        S3 = (dot ** 2).sum() - np.trace(dot ** 2)\n",
    "        return S1, S2, S3\n",
    "\n",
    "    S1C, S2C, S3C = sums(KC)\n",
    "    S1A, S2A, S3A = sums(KA)\n",
    "    S1B, S2B, S3B = sums(KB)\n",
    "\n",
    "    # T2\n",
    "    T2 = ((S2C - S3C) - (S2A - S3A) - (S2B - S3B)) / (2 * n * (n - 1) * (n - 2))\n",
    "\n",
    "    # T3\n",
    "    term = lambda S1, S2, S3: (S1 ** 2) - 4 * (S2 - S3) - 2 * S3\n",
    "    T3 = (term(S1C, S2C, S3C) - term(S1A, S2A, S3A) - term(S1B, S2B, S3B)) \\\n",
    "        / (2 * n * (n - 1) * (n - 2) * (n - 3))\n",
    "\n",
    "    return T1 - 2 * T2 + T3\n",
    "\n",
    "\n",
    "def plugin_var_tildeT1(A, B, W_A, W_B, unbiased=True):\n",
    "    \"\"\"\n",
    "    Plug-in variance for ~T1 with centered features (per dataset).\n",
    "    \"\"\"\n",
    "    n = A.shape[1]\n",
    "    A_c = A - A.mean(axis=1, keepdims=True)\n",
    "    B_c = B - B.mean(axis=1, keepdims=True)\n",
    "\n",
    "    denom = (n - 1) if unbiased else n\n",
    "    S_A = (A_c @ A_c.T) / denom\n",
    "    S_B = (B_c @ B_c.T) / denom\n",
    "\n",
    "    EA = np.trace(W_A @ S_A @ W_A @ S_A)\n",
    "    EB = np.trace(W_B @ S_B @ W_B @ S_B)\n",
    "    return (2.0 / (n * (n - 1))) * EA * EB\n",
    "\n",
    "\n",
    "# ============================================================\n",
    "# 6) DGP wrapper\n",
    "# ============================================================\n",
    "def generate_once_nd(n, theta, b, K, transform_key, subsets, d=10, seed=1234):\n",
    "    rng = np.random.RandomState(seed)\n",
    "    u = clayton_copula_sample_nd(n, theta, d, rng=rng)\n",
    "    v = clayton_copula_sample_nd(n, theta, d, rng=rng)\n",
    "    X, Y = transform_multidim(u, v, b=b, transform_key=transform_key)\n",
    "    A, B = build_AB_features_from_multidim(X, Y, K=K, subsets=subsets)\n",
    "    return A, B\n",
    "\n",
    "\n",
    "# ============================================================\n",
    "# 7) Pairwise block extraction + BH-FDR\n",
    "# ============================================================\n",
    "def block_view(M, r, base_dim):\n",
    "    \"\"\"Extract coordinate block r (0-indexed) from stacked feature matrix M.\"\"\"\n",
    "    return M[r*base_dim:(r+1)*base_dim, :]\n",
    "\n",
    "\n",
    "def bh_fdr_mask(pvals_2d, q=0.05):\n",
    "    \"\"\"\n",
    "    Benjamini–Hochberg FDR control.\n",
    "    Returns boolean mask with same shape as pvals_2d.\n",
    "    \"\"\"\n",
    "    p = pvals_2d.flatten()\n",
    "    m = p.size\n",
    "    order = np.argsort(p)\n",
    "    p_sorted = p[order]\n",
    "    thresh = q * (np.arange(1, m + 1) / m)\n",
    "    below = p_sorted <= thresh\n",
    "\n",
    "    reject = np.zeros(m, dtype=bool)\n",
    "    if np.any(below):\n",
    "        kmax = np.max(np.where(below))\n",
    "        reject[order[:kmax + 1]] = True\n",
    "\n",
    "    return reject.reshape(pvals_2d.shape)\n",
    "\n",
    "\n",
    "# ============================================================\n",
    "# 8) Main: compute pairwise Z + plot BH-FDR stars\n",
    "# ============================================================\n",
    "def pairwise_Z_heatmap_with_bh(\n",
    "    n=500, d_coords=10, theta=2, K=4, b=0.0, transform_key=\"logquad\",\n",
    "    weight_mode=\"identity\", q_fdr=0.05, seed=1234, unbiased_plugin=True\n",
    "):\n",
    "    subsets = all_nonempty_subsets_indices(K)\n",
    "    base_dim = (1 << K) - 1\n",
    "\n",
    "    # Generate data once\n",
    "    A, B = generate_once_nd(\n",
    "        n=n, theta=theta, b=b, K=K, transform_key=transform_key,\n",
    "        subsets=subsets, d=d_coords, seed=seed\n",
    "    )\n",
    "\n",
    "    # Base weight for ONE coordinate\n",
    "    if weight_mode == \"identity\":\n",
    "        W_base = np.eye(base_dim)\n",
    "    elif weight_mode == \"J\":\n",
    "        J_base, subsJ = J_numeric_K(K)\n",
    "        if subsJ != subsets:\n",
    "            idx = {S: i for i, S in enumerate(subsJ)}\n",
    "            perm = [idx[S] for S in subsets]\n",
    "            J_base = J_base[np.ix_(perm, perm)]\n",
    "        W_base = J_base\n",
    "    else:\n",
    "        raise ValueError(\"weight_mode must be 'identity' or 'J'\")\n",
    "\n",
    "    # Compute pairwise T, Var, Z\n",
    "    Tmat = np.zeros((d_coords, d_coords))\n",
    "    Vmat = np.zeros((d_coords, d_coords))\n",
    "    Zmat = np.zeros((d_coords, d_coords))\n",
    "\n",
    "    W_C = block_diag(W_base, W_base)\n",
    "\n",
    "    for r in range(d_coords):\n",
    "        A_r = block_view(A, r, base_dim)\n",
    "        for s in range(d_coords):\n",
    "            B_s = block_view(B, s, base_dim)\n",
    "\n",
    "            T_rs = compute_full_T(A_r, B_s, W_base, W_base, W_C)\n",
    "            v_rs = plugin_var_tildeT1(A_r, B_s, W_base, W_base, unbiased=unbiased_plugin)\n",
    "\n",
    "            Tmat[r, s] = T_rs\n",
    "            Vmat[r, s] = v_rs\n",
    "            Zmat[r, s] = T_rs / np.sqrt(max(v_rs, 1e-16))\n",
    "\n",
    "    # One-sided p-values (consistent with \"reject if Z > zcrit\")\n",
    "    Pmat = 1.0 - norm_dist.cdf(Zmat)\n",
    "\n",
    "    # BH-FDR mask\n",
    "    sig_bh = bh_fdr_mask(Pmat, q=q_fdr)\n",
    "\n",
    "    # Plot Z heatmap + BH stars\n",
    "    fig, ax = plt.subplots(figsize=(8, 7))\n",
    "    im = ax.imshow(Zmat, aspect=\"equal\")\n",
    "    cb = plt.colorbar(im, ax=ax)\n",
    "    cb.set_label(\"Z_{r,s} = T_{r,s} / sqrt(Var_hat)\")\n",
    "\n",
    "    ax.set_title(\n",
    "        f\"Pairwise Z heatmap | {transform_key}, n={n}, d={d_coords}, theta={theta}, K={K}, b={b}, W={weight_mode}\\n\"\n",
    "        f\"★ = BH-FDR discoveries (q={q_fdr}, one-sided)\"\n",
    "    )\n",
    "    ax.set_xlabel(\"Y coordinate s\")\n",
    "    ax.set_ylabel(\"X coordinate r\")\n",
    "    ax.set_xticks(range(d_coords))\n",
    "    ax.set_yticks(range(d_coords))\n",
    "\n",
    "    # Overlay BH stars\n",
    "    for r in range(d_coords):\n",
    "        for s in range(d_coords):\n",
    "            if sig_bh[r, s]:\n",
    "                ax.text(s, r, \"★\", ha=\"center\", va=\"center\",\n",
    "                        fontsize=13, fontweight=\"bold\", color=\"black\")\n",
    "\n",
    "    plt.tight_layout()\n",
    "    plt.show()\n",
    "\n",
    "    n_disc = int(sig_bh.sum())\n",
    "    print(f\"[BH-FDR] q={q_fdr}, discoveries: {n_disc} out of {d_coords*d_coords}\")\n",
    "\n",
    "    return {\"T\": Tmat, \"Var\": Vmat, \"Z\": Zmat, \"p\": Pmat, \"sig_bh\": sig_bh}\n",
    "\n",
    "\n",
    "if __name__ == \"__main__\":\n",
    "    # ---- Your requested settings ----\n",
    "    out = pairwise_Z_heatmap_with_bh(\n",
    "        n=500,\n",
    "        d_coords=10,\n",
    "        theta=0,\n",
    "        K=4,\n",
    "        b=1,                \n",
    "        transform_key=\"linear\",\n",
    "        weight_mode=\"identity\",# or \"J\"\n",
    "        q_fdr=0.05,\n",
    "        seed=1234,\n",
    "        unbiased_plugin=True\n",
    "    )\n"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "217be722-5e49-4f0f-a788-aaf33eee3cf3",
   "metadata": {},
   "source": [
    "## wa_dCoBET "
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 187,
   "id": "fb39052c-c18c-4af4-b376-2a6b6b4fc854",
   "metadata": {
    "scrolled": true
   },
   "outputs": [
    {
     "data": {
      "image/png": "iVBORw0KGgoAAAANSUhEUgAAAvsAAAKgCAYAAAAFyLJ6AAAAOXRFWHRTb2Z0d2FyZQBNYXRwbG90bGliIHZlcnNpb24zLjcuMiwgaHR0cHM6Ly9tYXRwbG90bGliLm9yZy8pXeV/AAAACXBIWXMAAA9hAAAPYQGoP6dpAACPmklEQVR4nOzdd3gU5d7G8Xs3HUhCSyCB0HsNqCAgRUA6inoAFQ+hHNSjiAELxkIRNSCCWEEsgIiigCiioEiRIL0JSBUCBKSXBBLS5/2DN3tYEkghm0l2vx+vuWRnZmfuSbLJb599nmcshmEYAgAAAOB0rGYHAAAAAOAYFPsAAACAk6LYBwAAAJwUxT4AAADgpCj2AQAAACdFsQ8AAAA4KYp9AAAAwEm5mx0AAAAAricxMVHJycmmnd/T01Pe3t6mnb+gUOwDAACgQCUmJsrfp5SSlWhahvLlyys6OtrpC36KfQAAABSo5ORkJStRd6mb3OVR4OdPVYrWnPxZycnJFPsAAACAI7jLQ+6Wgi/2ZRT8Kc1CsQ8AAABzWKxXlwJndZmCn9l4AAAAACdFyz4AAABMYbFaZLFYCv68hkVKK/DTmoKWfQAAAMBJUewDAAAATopuPAAAADCHmQN0XYTrXCkAAADgYmjZBwAAgClMHaDrImjZBwAAAJwUxT4AAADgpOjGAwAAAHNYLCYN0KUbDwAAAIAijpZ9AAAAmMNqudq6X9AYoAsAAACgqKPYR55UqVJFAwYMcOg52rVrp3bt2jn0HPlt5syZslgs2rx5s9lRiqRVq1bJYrHo8OHDZkdBAbn+dX748GFZLBbNnDnTtEwFacyYMaZMOyhJ3bp105AhQ0w5d2Gxe/duubu7a9euXWZHARyGYt9FZBShGYu3t7dq1aqloUOH6tSpU2bHKxQy/uhmtxRWa9eu1ZgxY3Tx4kWzo5gmOTnZ7AiSpAEDBtj9zLi7uyskJEQPPfSQdu/ebbdvxhuc+fPn3/BYJUqUyNF5q1SpcsOf28TERElZ/y4IDg5W586d9d577+nSpUuZjnv9a8PDw0NVqlTRsGHDXPrnrSj7448/9Ouvv2rkyJEFfu61a9fqrrvuUrFixVS+fHkNGzZMly9fzvHzP/vsM9WtW1fe3t6qWbOm3n///Uz73Oj3ube3t91+9erVU/fu3TVq1Khbvi7kkcVi3uIi6LPvYl577TVVrVpViYmJWrNmjaZOnaqff/5Zu3btUrFixXJ8nH379slqdex7xV9//dWhx7/eAw88oBo1amS5bceOHZo4caKaN29eoJlyY+3atRo7dqwGDBigkiVLmh3HFEuXLlVQUJDuuOMOs6PIy8tLn376qSQpNTVVBw8e1LRp07R06VLt3r1bwcHBDjlvaGionn322UzrPT097R5n/C5ISUnRyZMntWrVKoWHh2vy5MlatGiRGjVqlOkYU6dOVYkSJRQfH6/ly5fr/fff19atW7VmzZp8y1+5cmVduXJFHh4e+XbMwuyVV17Riy++WODnnThxojp06HDD33mOsn37dnXo0EF169bV5MmTdezYMb399ts6cOCAlixZku3zP/74Yz3xxBN68MEHNWLECEVFRWnYsGFKSEjI8o1Lxs9sBjc3t0z7PPHEE+rWrZsOHjyo6tWr39oFAoUQxb6L6dq1q26//XZJ0n/+8x+VKVNGkydP1g8//KCHH344x8fx8vLKdp/4+HgVL148z1mvL04crVGjRlkWOPHx8Xr99dfl7++vr7/+ukAzIXe+++67QlPsu7u769FHH7Vbd+edd6pHjx766aefHNZ9okKFCpnOm5VrfxdIUkREhFasWKEePXro3nvv1Z49e+Tj42P3nH/9618qW7asJOnxxx/XQw89pG+++UYbN25Us2bN8iV/Vq2vzijj96O7u7vc3Qv2T/Hp06f1008/adq0aQV6Xkl66aWXVKpUKa1atUp+fn6Srn4iNWTIEP3666/q1KnTDZ975coVvfzyy+revbvtk7AhQ4YoPT1d48aN02OPPaZSpUrZPefan9kb6dixo0qVKqVZs2bptddeu8UrBAofuvG4uPbt20uSoqOjJUlvv/22WrZsqTJlysjHx0e33XZblt0Lru+zn9E14Pfff9eTTz6pwMBAVaxYUTt27JDFYtGiRYts+27ZskUWi0VNmza1O2bXrl3tWs6z6rP//vvvq379+ipWrJhKlSql22+/XV999ZXdPsePH9egQYNUrlw5eXl5qX79+vr888/z9PWRpCeffFL79u3T9OnTVbVq1Rw9JykpSSNGjFBAQICKFy+u+++/X2fOnMm035IlS9S6dWsVL15cvr6+6t69u/766y+7fXbs2KEBAwaoWrVq8vb2Vvny5TVo0CCdO3fOts+YMWP0/PPPS5KqVq1q+8g6o++7xWLR0KFDNW/ePNWrV08+Pj5q0aKFdu7cKelqa1mNGjXk7e2tdu3aZeozHxUVpd69e6tSpUry8vJSSEiIhg8fritXrtjtl9Hl5NChQ+rcubOKFy+u4OBgvfbaazIMI0dfu7xKTU3Vjz/+qO+++86h57kV5cuXl6QCL+5yqn379nr11Vd15MgRffnll9nu37p1a0nSwYMHc3T86dOnq3r16vLx8VGzZs0UFRWVaZ+s+uyfPHlSAwcOVMWKFeXl5aWgoCDdd999mX5OlyxZorZt28rX11d+fn664447Mv1+mDdvnm677Tb5+PiobNmyevTRR3X8+HHb9rffflsWi0VHjhzJlC0iIkKenp66cOGCbd2GDRvUpUsX+fv7q1ixYmrbtq3++OMPu+dldCnZvXu3HnnkEZUqVUp33XWX3bbrffnll7acpUuX1kMPPaSYmBi7fQ4cOKAHH3xQ5cuXl7e3typWrKiHHnpIsbGxmY53rZ9++kmpqanq2LFjpm1//fWX2rdvLx8fH1WsWFGvv/66Pv/883wZSxMXF6dly5bp0UcftRX6ktS/f3+VKFFC33777U2fv3LlSp07d05PPvmk3fqnnnpK8fHx+umnnzI9xzAMxcXF3fT3j4eHh9q1a6cffvghl1eE/GCxWk1bXEXh/IuDApPxR7pMmTKSpHfffVf33nuv+vXrp+TkZM2dO1e9e/fW4sWL1b1792yP9+STTyogIECjRo1SfHy8GjRooJIlS2r16tW69957JV0tHK1Wq/7880/FxcXJz89P6enpWrt2rR577LEbHvuTTz7RsGHD9K9//UvPPPOMEhMTtWPHDm3YsEGPPPKIJOnUqVO68847bcVtQECAlixZosGDBysuLk7h4eG5+vrMmjVLX3zxhYYMGaI+ffrk+HlPP/20SpUqpdGjR+vw4cOaMmWKhg4dqm+++ca2z+zZsxUWFqbOnTtrwoQJSkhI0NSpU3XXXXdp27ZtqlKliiRp2bJlOnTokAYOHKjy5cvrr7/+0vTp0/XXX39p/fr1slgseuCBB7R//359/fXXeuedd2wtWQEBAbbzRUVFadGiRXrqqackSZGRkerRo4deeOEFffTRR3ryySd14cIFvfXWWxo0aJBWrFhhe+68efOUkJCg//73vypTpow2btyo999/X8eOHdO8efPsrj0tLU1dunTRnXfeqbfeektLly7V6NGjlZqa6tBWs5UrV+r8+fM6f/68du3apQYNGuTq+ZcvX7b1a78ZDw8P+fv75+iYZ8+elXT1a3Lo0CGNHDlSZcqUUY8ePTLte+nSJdv+10pKSsrRuTKkpKRkOk6xYsVy3E3v3//+t1566SX9+uuv2X76kFH8Xd+ampXPPvtMjz/+uFq2bKnw8HAdOnRI9957r0qXLq2QkJCbPvfBBx/UX3/9paefflpVqlTR6dOntWzZMh09etT2Opk5c6YGDRqk+vXrKyIiQiVLltS2bdu0dOlS2++HmTNnauDAgbrjjjsUGRmpU6dO6d1339Uff/yhbdu2qWTJkurTp49eeOEFffvtt7Y30Bm+/fZbderUyXa9K1asUNeuXXXbbbdp9OjRslqtmjFjhtq3b6+oqKhMn3b07t1bNWvW1JtvvnnT4vONN97Qq6++qj59+ug///mPzpw5o/fff19t2rSx5UxOTlbnzp2VlJSkp59+WuXLl9fx48e1ePFiXbx48aY/o2vXrlWZMmVUuXJlu/UnT57U3XffrdTUVL344osqXry4pk+fnukTHilvr5edO3cqNTXV7hMl6eqnuKGhodq2bdtNj5Wx/frn33bbbbJardq2bVumT7WqVaumy5cvq3jx4urVq5cmTZqkcuXKZTr2bbfdph9++MH2NwlwKgZcwowZMwxJxm+//WacOXPGiImJMebOnWuUKVPG8PHxMY4dO2YYhmEkJCTYPS85Odlo0KCB0b59e7v1lStXNsLCwjId/6677jJSU1Pt9u3evbvRrFkz2+MHHnjAeOCBBww3NzdjyZIlhmEYxtatWw1Jxg8//GDbr23btkbbtm1tj++77z6jfv36N73OwYMHG0FBQcbZs2ft1j/00EOGv79/puu7mT179hjFixc36tevn+PnZXwdOnbsaKSnp9vWDx8+3HBzczMuXrxoGIZhXLp0yShZsqQxZMgQu+efPHnS8Pf3t1uf1bm//vprQ5KxevVq27qJEycakozo6OhM+0syvLy87LZ9/PHHhiSjfPnyRlxcnG19REREpuNklSEyMtKwWCzGkSNHbOvCwsIMScbTTz9tW5eenm50797d8PT0NM6cOZPpONdauXLlDa/hWufPnzdGjhxpPPXUU7alWbNmhiRDktGuXTu7baNHjzbi4+NvesyM7Nkt1/5M5vZY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cnNzk8ViUVpamm3d4cOH9f3339/ysVu3bq0NGzZo5cqVtmK/bNmyqlu3rq21LLu+59f3i/X09FS9evVkGIZSUlLk5uamBx98UAsWLMjy1udZTX15rZSUFD300ENKSEjQ119/neMZV/Kic+fO8vPz05tvvqmUlJRM2zOyZrTsXf+R+5QpUzI9J+OeBvl9R9OsMhiGoXffffeGz/nggw/s9v3ggw/k4eGhDh065Gs26ert7vfv3y83NzeNHTvW1ld3zZo1Oe62JV0topctW5btMmnSpDxn3b9/v/bt25ej4vZG6tWrp44dO9qW2267Lc/Hut6KFSs0btw4Va1aVf369ct2/379+qlixYp2Ld5Zuf322xUQEKBp06bZdcOYOXNmtj+vCQkJmWZ9qV69unx9fW3dQTp16iRfX19FRkZm2jfj5/b2229XYGCgpk2bZteNZMmSJdqzZ0+m2cYefPBBubm56euvv9a8efPUo0cPu/uG3HbbbapevbrefvvtLN+8Zff75kYeeOAB28/y9a97wzBsvwfj4uKUmppqt71hw4ayWq3ZzuDUokULXbhwQYcOHbJb361bN61fv97uzfOZM2c0Z86cTMfIy+vF399fHTt21Jdffmk3JmT27Nm6fPmyevfubVuXkJCgvXv32vWhb9++vUqXLq2pU6faZZk6daqKFStm9z3M6us/depUnTlzRl26dMm0bcuWLapfv75Df+/jBriDrsPRsg+b7t27a/LkyerSpYseeeQRnT59Wh9++KFq1KihHTt23NKxW7durTfeeEMxMTF2RX2bNm308ccfq0qVKjf92Fm6+ge9fPnyatWqlcqVK6c9e/bogw8+UPfu3W0DTMePH6+VK1eqefPmGjJkiOrVq6fz589r69at+u2337L8+DbDq6++qk2bNql9+/Y6cOBAli1cknT//fff0s3CpKutgVOnTtW///1vNW3aVA899JACAgJ09OhR/fTTT2rVqpU++OAD+fn5qU2bNnrrrbeUkpKiChUq6Ndff7XdF+FaGUXfyy+/rIceekgeHh7q2bPnLWetU6eOqlevrueee07Hjx+Xn5+fFixYcMNuAN7e3lq6dKnCwsLUvHlzLVmyRD/99JNeeuklu6lA88t3332nChUq6KuvvrJ9OtShQwcNGDBA33//fY67YuV3n/3U1FRb16KMAbrTpk1Tenq6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Bg8erISEhDwHAQAAAFzR4cOHlZSUlOW2hIQExcTE5PnYuS72R4wYod9//10//vijLl68qIsXL+qHH37Q77//rmeffTbPQQAAAOBiLIZ5i8kSExN1/vx5nTt3TpIUFxen8+fP2y3//POPvv/+ewUHB+f5PLnus79gwQLNnz9f7dq1s63r1q2bfHx81KdPH02dOjXPYQAAAABXMGHCBL322muSJIvFos6dO99w3zFjxuT5PLku9hMSElSuXLlM6wMDA+nGAwAAgJxz4ak3e/XqpSpVqsgwDA0aNEivvPKKqlevbrePp6en6tatq9DQ0DyfJ9fFfosWLTR69Gh98cUX8vb2liRduXJFY8eOVYsWLfIcBAAAAHAVjRs3VuPGjSVdbdnv3r27ypYtm+/nyXWx/+6776pz586qWLGiLeCff/4pb29v/fLLL/keEAAAAHBmYWFhmdb98ccf2rNnj1q3bq3atWvn+di5LvYbNGigAwcOaM6cOdq7d68k6eGHH1a/fv3k4+OT5yAAAABwMS7cjedajzzyiLy8vGz3s5o2bZqefPJJSZKXl5cWL16sDh065OnYuS72JalYsWIaMmRInk4IAAAA4H/WrFmjt99+2/Y4MjJS//nPfzR58mT997//1dixYwu22D948KCmTJmiPXv2SJLq16+vYcOGZRpUAAAAANwQLfuSrt6sNigoSJL0119/KSYmRs8884xKlCihsLAw9e7dO8/HzvU8+7/88ovq1aunjRs3qlGjRmrUqJHWr1+v+vXra9myZXkOAgBwnPIlSqhpcJDZMQAAWShTpoyOHDkiSVq6dKmCgoJUv359SVJaWprS09PzfOxct+y/+OKLGj58uMaPH59p/ciRI3XPPffkOQwAwDE616ypkJL+2vrPCbOjAACu07VrV40cOVJ//vmnZs6cqX//+9+2bbt27VLVqlXzfOxct+zv2bNHgwcPzrR+0KBB2r17d56DAAAcp3OtmupSs6bZMQDAngvfQfdab7/9tjp37qylS5eqW7duGjt2rG3bwoUL1aVLlzwfO9ct+wEBAdq+fbtqXvdHY/v27QoMDMxzEACAY5QpVky3VwiWm9WqJkFB2naC1n0AKEz8/f31+eefZ7ltzZo1t3TsXBf7Q4YM0WOPPaZDhw6pZcuWkq7OAzphwgSNGDHilsLkxbp16/Tyyy9rw4YNcnNzU2hoqH755RemAQXgkqwWS6ZxZ51r1pCb9eoHud1q19KOkyftthuS0o3C1coFwEUwQNfhcl3sv/rqq/L19dWkSZMUEREhSQoODtaYMWM0bNiwfA94M+vWrVOXLl0UERGh999/X+7u7vrzzz9ltea6dxIAOIWmwcF6p3tXBfv5Zbl90O23adDtt9kexyYm6qVfl2np/gMFFREAUIByXRVbLBYNHz5cx44dU2xsrGJjY3Xs2DE988wzsljy921SlSpVNGXKFLt1oaGhGjNmjCRp+PDhGjZsmF588UXVr19ftWvXVp8+feTl5ZWvOQCgqNh8/Lh6fPGlfj3wd7b7bj3+j3p8MZtCHwCcWK6L/ejoaB04cPUPg6+vr3x9fSVJBw4c0OHDh/M13M2cPn1aGzZsUGBgoFq2bKly5cqpbdu2OerXlJSUpLi4OLsFAJxFbGKi/vvDIo1ZvkJJqamZtqelp+uj9Rv00Nxv9E/cJRMSAsBVhsW8xVXkutgfMGCA1q5dm2n9hg0bNGDAgPzIlCOHDh2SJI0ZM0ZDhgzR0qVL1bRpU3Xo0MH2ZuRGIiMj5e/vb1tCQkIKIjIAFKjZ27ZrfUxMpvUxsbGatOYPpdFPHwCcXq6L/W3btqlVq1aZ1t95553avn17fmTKkYybCzz++OMaOHCgmjRponfeeUe1a9e+4WjmDBEREbYuSLGxsYrJ4o8hABR1vl5eujOLxowqpUqpVtmyJiQCgOu48NSbr732mjZu3Jjtfnv27FH79u3zfJ489dm/dCnzx76xsbFKS0vLc5CsWK1WGde1PKWkpEiS7ZbC9erVs9tet25dHT169KbH9fLykp+fn90CAM6mQ/Vq8nJ3V0JKiiJ++VWPLfxe5xOuSJK61GLOfQAw05gxY9S6dWtFRkbedL+4uDj9/vvveT5Prov9Nm3aKDIy0q6wT0tLU2RkpO666648B8lKQECATlwzH3RcXJyio6MlXR28GxwcrH379tk9Z//+/apcuXK+5gCAoqhLrZraffq07pv9pb7duUvLDx5S91lfaO2Ro+pKsQ8Apmvbtq1efvlldezYUadOnXLIOXI99eaECRPUpk0b1a5dW61bt5YkRUVFKS4uTitWrMjXcO3bt9fMmTPVs2dPlSxZUqNGjZKbm5ukq58wPP/88xo9erQaN26s0NBQzZo1S3v37tX8+fPzNQcAFDU+Hu46dP6Chv34k5KvaZw5HR+v/vPm67FmdyjE318xsbEmpgTg8lx8nv3XX39dQ4cO1eDBg9WwYUPNnDlT3bp1y9dz5Lplv169etqxY4f69Omj06dP69KlS+rfv7/27t2rBg0a5Gu4iIgItW3bVj169FD37t3Vq1cvVa9e3bY9PDxcERERGj58uBo3bqzly5dr2bJldvsAgCu6kpKqt1ZH2RX6GQxJH2/cRKEPAIXAvffeqz///FMNGjRQz549NXz4cFu39fyQ65Z96epNtN588818C3Ejfn5+mjt3rt26sLAwu8cvvviiXnzxRYdnAQAAQP6yWK4uZpy3MAkODtby5cs1fvx4jR49WqtXr9bcuXNVs+atd7nkVrMAAACAySwWiyIiIhQVFaULFy6oadOmmjVr1i0fN08t+wAAAADyX/PmzfXnn3/qiSee0KBBg3T77bff0vFo2QcAAIA5XHie/Zvx9fXVnDlzNGPGDO3Zs+eWjkXLPgAAAFDAVq5cmel+Udfr37+/7rrrLkVFReX5PBT7AAAAMIcLT73Ztm3bHO1XrVo1VatWLc/nyXGxX6pUKVmyGLrs7++vWrVq6bnnntM999yT5yAAAAAA8leOi/0pU6Zkuf7ixYvasmWLevToofnz56tnz575lQ0AAADALchxsX/9/PbXCw0NVWRkJMU+AAAAcsaFu/EUlHybjadHjx7au3dvfh0OAAAAwC3Kt2I/KSlJnp6e+XU4AAAAALco32bj+eyzzxQaGppfhwMAAIDTM/5/MeO8riHHxf6IESOyXB8bG6utW7dq//79Wr16db4FAwAAAHBrclzsb9u2Lcv1fn5+uueee/Tdd9+patWq+RYMAAAATo4Bug6X42J/5cqVjswBAAAAIJ/l2wBdAAAAAIVLvg3QBQAAAHLDYjFksRT8YFkzzmkWWvYBAAAAJ0XLPgAAAMzjQoNlzUDLPgAAAOCkKPYBAAAAJ0U3HgAAAJiCAbqOR7F/jSrFzsmruIfZMRxmbUJNsyM43KkUP7MjFIggz4tmR3C4d2I6mR3B4TY/MtnsCA7Xp2ILsyM4XP99MWZHKBCX07zNjuBwm85WNjuCw6TGJ5kdASah2AcAAIA5uIOuw9FnHwAAAHBSFPsAAACAk6IbDwAAAExhsVxdzDivq6BlHwAAAHBStOwDAADAHBbj6mLGeV0ELfsAAACAk6LYBwAAAJwU3XgAAABgCu6g63i07AMAAABOipZ9AAAAmIKpNx2Pln0AAADASVHsAwAAAE6KbjwAAAAwh0kDdJlnHwAAAECWxo8fL4vFovDwcLOjZIuWfQAAAJjD8v+LGefNo02bNunjjz9Wo0aN8i+PA9GyDwAAAJcUFxdntyQlJd10/8uXL6tfv3765JNPVKpUqQJKeWso9gEAAOCSQkJC5O/vb1siIyNvuv9TTz2l7t27q2PHjgWU8NbRjQcAAACmMPsOujExMfLz87Ot9/LyuuFz5s6dq61bt2rTpk0Oz5efKPYBAADgkvz8/OyK/RuJiYnRM888o2XLlsnb27sAkuUfuvEAcHllPEuqrl9Vs2PgFpWtUFr1WtQyOwaQIwFe/mrgX9nsGKazmLjkxpYtW3T69Gk1bdpU7u7ucnd31++//6733ntP7u7uSktLy+NXwPFo2Qfg8lqVbaTy3mW1Jy7a7Ci4BXc90FxB1cpp97r9ZkcBstUmsIGCfcpoV+wRs6MgBzp06KCdO3farRs4cKDq1KmjkSNHys3NzaRk2aNlH4DLa1U2VK0CGpsdA7eo9QN36q4HmpsdA8iRtoEN1TawodkxkEO+vr5q0KCB3VK8eHGVKVNGDRo0MDveTVHsA3BpJT18Vc+/mgK8SqmObxWz4yCPSgb6q/5ddRQYUlZ176QrDwq3Up4l1LBkVQV6l1R9F+/KkzFA14zFVRTpYv/w4cOyWCxZLvPmzTM7HoBCxiqLrNf917JsY7lZrv4qbB3QJNN2qyl3e8HNWK1WWd3sl7seaC43t6vfx7Z9WmTabrUW6T93KMKsssjNYrVb2gQ0tP3eubtco0zb+b1TNKxatUpTpkwxO0a2inSf/ZCQEJ04ccJu3fTp0zVx4kR17drVpFQACqu6flX1fJ3+CvQuneX2+yverfsr3m17fCklQe8dmKs/zm4voITIiXotayniy2cUWKlsltsfDO+hB8N72B5funBZ7zz2saIWrC+oiIBNg5JV9GqDh1XOO+sbMPWp1EZ9KrWxPb6UkqC39szX76d3Zrm/s7FYZEoru8WF3k8V+qaOKlWqZHrXFBoaqjFjxsjNzU3ly5e3WxYuXKg+ffqoRIkS5gQGUGj9FXdIQ7e+pbVn/8x2392x0Xp66wQK/UJo15q9eqLJ81qzcGO2+/61dp+eaPI8hT5Ms+NitAatf0erT+/Kdt9dFw9r0IZ3XKbQR8Eo9MV+bmzZskXbt2/X4MGDb7pfUlJSptsjA3ANl1MT9Pruz/TR3/OUnJ6SaXuaka65R3/RC3++q9NJF0xIiJy4dOGyxj44UR88/ZmSE5MzbU9LS9dXb36nEW1H6fTRsyYkBP7nUuoVvbJjlqbsXaiktKx/73wRvVxPb5mqU4kXCz4gnJpTFfufffaZ6tatq5YtW950v8jISLtbI4eEhBRQQgCFxeJ/orTj4oFM608mntUXh39SutJNSIXc+uHDpdq+8q9M608eOqUZr3yt9DS+jyg8vju2VtsuHMy0/sSVc/r04FKlGa7383q1G485i6twmmL/ypUr+uqrr7Jt1ZekiIgIxcbG2paYmJgCSAigMCnu5qPGJTPP2lLBJ1BVigWZkAh5Udy/mELbZ572rkLNIFVpUMmERMCNlXD3VtPSNTKtr1gsQNWKlzchEVxBoS/2rVarDMN+4EZKSuaPwObPn6+EhAT1798/22N6eXnZbo+c09skA3Auzcs0kIfVXYlpSXp3/9cau2u6YlMuS5JaBYSaGw451qLn7fL08tCV+ERNHjJVo+6boNizV7tmtn6QOfdRuLQsW0+eVnddSUvWhN3zFLF9hi4mx0uS2pZzzTn3rRbDtMVVFPpiPyAgwG7Gnbi4OEVHZ77L5WeffaZ7771XAQEBBRkPQBHVqmyoDl4+pmFbJ+qXk+u04fwuPbVlvLZf2KdWZbnBVlHR+sE7dXD7YT11+0gt+WyF1v24WY83fk7blu9U6wfvNDseYKdduUY6cOkfDdkwRT/9s1F/nN2tgesna8v5A2ob2MjseHBShb7Yb9++vWbPnq2oqCjt3LlTYWFhmW5J/Pfff2v16tX6z3/+Y1JKAEWJl9VTx66c0vBtk3Xsymnb+vPJcXp550daeXqzynuXMTEhcsK7mJdi9h3X03dGKGbfP7b1505c0MhO47R8TpTKVw00MSHwP95WDx2NP60nNr6nowlnbOvPJcdpxNZPtOzEVgX5ZD0tMHArCv08+xEREYqOjlaPHj3k7++vcePGZWrZ//zzz1WxYkV16tTJpJQAipKk9GTNiF6U5TZDhubF/FbAiZAXiQlJ+vTFOVluMwxD30z4vmADATeRmJ6iaX//nOU2Q4bmHFlZwIkKB7PuZutKd9At9MW+n5+f5s6da7cuLCzM7vGbb76pN998syBjAQAAAIVeoe/GAwAAACBvCn3LPgAAAJwT3Xgcj5Z9AAAAwEnRsg8AAABTmHU3W+6gCwAAAKDIo9gHAAAAnBTdeAAAAGAKq8WQ1YTBsgYDdAEAAAAUdbTsAwAAwBRMvel4tOwDAAAATopiHwAAAHBSdOMBAACAKawyZJUJA3RNOKdZaNkHAAAAnBQt+wAAADCFRSbdQbfgT2kaWvYBAAAAJ0WxDwAAADgpuvEAAADAFBaT7qCbzjz7AAAAAIo6WvYBAABgCu6g63i07AMAAABOimIfAAAAcFJ047lG6xJ7VdzXzewYDpNmOP97u2peXmZHKBCr4uqYHcHhKhS7aHYEh1sSH2x2BIcrt87P7AgO99r8PmZHKBBbwt4xO4LD7YkPMjuCwyS7JWud2SGyYDVpgK4Z5zSL81d/AAAAgIuiZR8AAACmYICu49GyDwAAADgpin0AAADASdGNBwAAAKZggK7j0bIPAAAAOCla9gEAAGAKqwxZZULLvgnnNAst+wAAAICTotgHAAAAnBTdeAAAAGAK5tl3PFr2AQAAACdFyz4AAABMwdSbjkfLPgAAAOCkKPYBAAAAJ0U3HgAAAJiCbjyOR8s+AAAA4KRo2QcAAIApaNl3PFr2AQAAACdFsQ8AAAA4KbrxAAAAwBR043E8WvYBAAAAJ0WxDwBAEVHet4SaVAgyOwaAIoRuPAAAFBGdatdUSEl/bTt+wuwoQL6wSLKq4LvUWAr8jOahZR8AgCKic+2a6ly7ptkxABQhRbrYP3nypP7973+rfPnyKl68uJo2baoFCxaYHQsAgHxXplgx3VYxWEF+vgqlKw+cRMYAXTMWV1Gki/3+/ftr3759WrRokXbu3KkHHnhAffr00bZt28yOBgBAnlktFrldt3SuXUNu1qt/trvVqZVpu9XiSh0TAORUoe6zX6VKFYWHhys8PNy2LjQ0VL169dKYMWO0du1aTZ06Vc2aNZMkvfLKK3rnnXe0ZcsWNWnSxKTUAADcmqYVgjXp3q4K9vfLcvvAZrdpYLPbbI9jryTqlSXLtHTfgYKKCKCIKNIt+y1bttQ333yj8+fPKz09XXPnzlViYqLatWt30+clJSUpLi7ObgEAoLDYfOy47v38S/267+9s99167B/d+/lsCn0USXTjcbwiXex/++23SklJUZkyZeTl5aXHH39cCxcuVI0aNW76vMjISPn7+9uWkJCQAkoMAEDOxCYm6qnvFmnsryuUlJqaaXtaero++mODHvnyG/0Td8mEhACKgiJd7L/66qu6ePGifvvtN23evFkjRoxQnz59tHPnzps+LyIiQrGxsbYlJiamgBIDAJA7X27ZrvVHMv+dirkYq3dW/6E0w3VaKOF8aNl3vELdZ99qtcq47pdYSkqKJOngwYP64IMPtGvXLtWvX1+S1LhxY0VFRenDDz/UtGnTbnhcLy8veXl5OS44AAD5xNfLSy0qZ/4EukrpUqoVUFb7z5w1IRWAoqJQt+wHBAToxIn/3TgkLi5O0dHRkqSEhARJV98QXMvNzU3p6ekFFxIAAAdqX7OaPN3dlZCcopd+/lWPz/te5xOuSBJz7gPIVqEu9tu3b6/Zs2crKipKO3fuVFhYmNzc3CRJderUUY0aNfT4449r48aNOnjwoCZNmqRly5apV69e5gYHACCfdKldU3tOndb9M77UvD93acXfh9Tzsy+09vBRdalDsY+ijW48jleou/FEREQoOjpaPXr0kL+/v8aNG2dr2ffw8NDPP/+sF198UT179tTly5dVo0YNzZo1S926dTM5OQAAt87Hw12Hzl3QM9//pOS0NNv605fjNeDr+Rpy5x0KKemvmIuxJqYEUJgV6mLfz89Pc+fOtVsXFhZm+3fNmjW5Yy4AwGldSUnVxFVRWW4zJE1fv6lgAwH5zCpDVhV8K7sZ5zRLoe7GAwAAACDvKPYBAAAAJ1Wou/EAAADAeZk1WNaVBujSsg8AAAA4KVr2AQAAYAqrJV1WS8HfH8mMc5qFln0AAADASVHsAwAAAE6KbjwAAAAwBQN0HY+WfQAAAMBJ0bIPAAAAU1hMuoOuhTvoAgAAAJCkyMhI3XHHHfL19VVgYKB69eqlffv2mR0rRyj2AQAAgJv4/fff9dRTT2n9+vVatmyZUlJS1KlTJ8XHx5sdLVt04wEAAIAprDJpgG4uu/EsXbrU7vHMmTMVGBioLVu2qE2bNvkZLd9R7AMAAMAlxcXF2T328vKSl5dXts+LjY2VJJUuXdohufIT3XgAAABgiow76JqxSFJISIj8/f1tS2RkZLaZ09PTFR4erlatWqlBgwaO/hLdMlr2AQAA4JJiYmLk5+dne5yTVv2nnnpKu3bt0po1axwZLd9Q7AMAAMAl+fn52RX72Rk6dKgWL16s1atXq2LFig5Mln8o9gEAAGAKN4shNxMG6Ob2nIZh6Omnn9bChQu1atUqVa1a1UHJ8h/FPgAAAHATTz31lL766iv98MMP8vX11cmTJyVJ/v7+8vHxMTndzVHsAwAAwBRWk+6gm9tzTp06VZLUrl07u/UzZszQgAED8imVY1DsAwAAADdhGAX/hiS/UOxfY9KRznIvnv0o7KJqZJUlZkdwuJ/ONzY7QoGITfE2O4LDdSyzx+wIDncmNeeDwoqq+iVOmB3B4R7os9XsCAWi4U9Pmx3B4Qbc+YfZERwmKTXF7AgwCcU+AAAATHHtnPcFfV5XwU21AAAAACdFyz4AAABMYbVIVhOm3rRaCvyUpqFlHwAAAHBSFPsAAACAk6IbDwAAAEzhJkNuJsyzb8Y5zULLPgAAAOCkKPYBAAAAJ0U3HgAAAJjCYtI8+xbm2QcAAABQ1NGyDwAAAFNYLYZJ8+wzQBcAAABAEUexDwAAADgpuvEAAADAFG4WQ24mdKkx45xmoWUfAAAAcFK07AMAAMAUVqXLqoKfBtOMc5qFln0AAADASVHsAwAAAE6KbjwAAAAwBfPsOx4t+wAAAICTomUfAAAApnBTutxMGCxrxjnNQsu+Ccp6+au+XxWzYwAAUCiVL15CtwUGmx0DcAoU+yZoE9BQ7co1NjsGAACFUpcqtdS9Wh2zYwBOgWLfBG0CGqlNQCOzYwAAUCh1rVpbXavWMjsGCkDGAF0zFldRpIv9gwcP6v7771dAQID8/PzUp08fnTp1yuxYN1XKo4QalKyqQO+SqudX2ew4AAAUKmV9iumOchUUXMJPTenKA9yyIlvsx8fHq1OnTrJYLFqxYoX++OMPJScnq2fPnkpPLxyDLqyyyGqx2i2tAxvKzXL1y94usHGm7VZZTE4NAEDBsFoscrtu6VKlltysV/9Odq9WJ9N2q4W/k87EzZJu2uIqCvVsPFWqVFF4eLjCw8Nt60JDQ9WrVy+1bNlShw8f1rZt2+Tn5ydJmjVrlkqVKqUVK1aoY8eOJqX+n/r+VfRy/X4q510qy+29K7VV70ptbY8vpSTo7b3ztPrMjoKKCACAaW4rV0Hv3t1DFUr4Zbn9Pw1v138a3m57HJuUqBejlurn6P0FFREo8opsy35SUpIsFou8vLxs67y9vWW1WrVmzZpsnxsXF2e3OMLO2GgN2ThJUWd2ZrvvrtjDGrJpMoU+AMBlbDp5TF2/m6mlh7Mv3recOq6u382k0AdyqcgW+3feeaeKFy+ukSNHKiEhQfHx8XruueeUlpamEydO3PS5kZGR8vf3ty0hISEOy3kp9YpG7Zypd/d9p+S0lEzb04x0fXn4Nz2z9UOdSrzgsBwAABRGsUmJenzZ9xr1xzIlpqZm2p6Wnq4Ptq1T7x+/0vHLjmmcg3ksMmQ1YbGIAbqFXkBAgObNm6cff/xRJUqUkL+/vy5evKimTZvKar35ZUVERCg2Nta2xMTEODzv98f/0LaLf2daf+LKOX12aInSDdfpOwYAwPVm7d6m9SeOZlp/9FKsJm6OUprhOsUZkJ8KdZ99q9Uq47oXd0rK/1rHO3XqpIMHD+rs2bNyd3dXyZIlVb58eVWrVu2mx/Xy8rLr/lMQirt7q0mpmpnWVywWoKrFyys6/mSB5gEAoDDx8/RSi+BKmdZX9S+l2qXKat+FsyakgqOZNVjWlQboFuqW/YCAALsuOXFxcYqOjs60X9myZVWyZEmtWLFCp0+f1r333luQMXOkZdn68rS660pakibu+VYv7/hcscnxkqQ2gcy5DwBwbR0qVZeXm7sSUpI1cvVSDf7lO51PTJB0dd59AHlTqIv99u3ba/bs2YqKitLOnTsVFhYmNzc32/YZM2Zo/fr1OnjwoL788kv17t1bw4cPV+3ahe+XQtuARvr70nE9vmmKfj6xQWvP/qXBG9/W1vMH1JYbbAEAXFy3qrX117lT6rHwC83dt0O/Hf1bnRfM1B/Hj6gbN9gC8qxQd+OJiIhQdHS0evToIX9/f40bN86uZX/fvn2KiIjQ+fPnVaVKFb388ssaPny4iYmz5m311NGE0xq76wulGGm29eeS4/Tc9o/1UOW7FeRdWicSz5uYEgAAc/i4e+hg7Hk9tXyRktP/93fydMJl9fv5Gz3RuLlCfP0VcynWxJRwBKslXVYTutSYcU6zFOpi38/PT3PnzrVbFxYWZvv3+PHjNX78+IKOlWuJ6cmafvCnLLcZMvT1kRUFnAgAgMLjSmqKxm/8PctthqSpf24o2ECAEynUxT4AAACcl5skNxOmwXTLfhenUaj77AMAAADIO4p9AAAAwEnRjQcAAACmYICu49GyDwAAADgpWvYBAABgCjcZJg3QLfhzmoWWfQAAAMBJUewDAAAATopuPAAAADAFA3Qdj5Z9AAAAwEnRsg8AAABTWJUuN5nQsm/COc1Cyz4AAADgpCj2AQAAACdFNx4AAACYwmoxZLUU/Jz3ZpzTLLTsAwAAAE6KYh8AAABwUnTjAQAAgCncTJqNx4xzmoWWfQAAAMBJ0bIPAAAAU7hZ0uVmwt1szTinWWjZBwAAAJwUxT4AAADgpOjGAwAAAFNYZcgqE+bZN+GcZqHYv0Zd/1PyLOFhdgyHKW5NMjuCw3lY08yOUCAeLbfO7AgOdyg50OwIDudrTTQ7gsPV9vrH7AgOdzG9mNkRCsSTrVaYHcHhfny5g9kRHCY1JVHSErNjwAQU+wAAADAFA3Qdjz77AAAAgJOi2AcAAACcFN14AAAAYAqr0mU14W62ZpzTLLTsAwAAAE6Kln0AAACYwirJzWLG1Juuw5WuFQAAAHApFPsAAACAk6IbDwAAAEzhpnS5yWLKeV0FLfsAAACAk6JlHwAAAKawWtJltRR8y76VO+gCAAAAKOoo9gEAAAAnRTceAAAAmIIBuo5Hyz4AAADgpGjZBwAAgCncLIYpd9A145xmoWUfAAAAcFIU+wAAAICTotgHAACAKaxKN23Jiw8//FBVqlSRt7e3mjdvro0bN+bzVyT/UewDAAAA2fjmm280YsQIjR49Wlu3blXjxo3VuXNnnT592uxoN0WxDwAAUMACypRQg9rBZscwnZsl3bQltyZPnqwhQ4Zo4MCBqlevnqZNm6ZixYrp888/d8BXJv9Q7AMAABSwtnfWUvtWtc2O4fLi4uLslqSkpCz3S05O1pYtW9SxY0fbOqvVqo4dO2rdunUFFTdPKPYBAAAKWNsWtdS2RS2zY7i8kJAQ+fv725bIyMgs9zt79qzS0tJUrlw5u/XlypXTyZMnCyJqnhXqYn/69Olq166d/Pz8ZLFYdPHixUz7nD9/Xv369ZOfn59KliypwYMH6/LlywUfFgAAIAdK+RdTozoVVK6sn+q7eFceq9L//y66BbtkDNCNiYlRbGysbYmIiDD5K5L/CnWxn5CQoC5duuill1664T79+vXTX3/9pWXLlmnx4sVavXq1HnvssQJMCQAAkDWr1SK365a2d9aSm9vVEqx9y9qZtlutFpNTuw4/Pz+7xcvLK8v9ypYtKzc3N506dcpu/alTp1S+fPmCiJpnpt5Bt0qVKgoPD1d4eLhtXWhoqHr16qUxY8bY1q9atSrL5+/Zs0dLly7Vpk2bdPvtt0uS3n//fXXr1k1vv/22goNd+90yAAAwV4PawRo9vIfKBfhlub3vvber77232x5fupyoCR/9olXr9hdURFNZZciqgr+bbW7P6enpqdtuu03Lly9Xr169JEnp6elavny5hg4d6oCE+adQt+xnZ926dSpZsqSt0Jekjh07ymq1asOGDTd8XlJSUqYBGQAAAPltx57jGjBillavP5Dtvjv3Xt3XVQr9ombEiBH65JNPNGvWLO3Zs0f//e9/FR8fr4EDB5od7aZMbdm/VSdPnlRgYKDdOnd3d5UuXfqmgyUiIyM1duxYR8cDAADQpcuJemnC93qgaxM9NaCdvDzty6+0tHTNWbhBn339h9LSC76VGznTt29fnTlzRqNGjdLJkycVGhqqpUuXZhq0W9gU6Zb9vIqIiLAbjBETE2N2JAAA4OS+W7JN23YdzbT+n1Oxmj5njUsW+kVpnn1JGjp0qI4cOaKkpCRt2LBBzZs3z+evSP4ztWXfarXKMOx/sFNSUnL8/PLly2e6a1lqaqrOnz9/08ESXl5eNxyAAQAA4AglinmpacNKmdaHBJdStUpldejoWRNSwdmZ2rIfEBCgEydO2B7HxcUpOjo6x89v0aKFLl68qC1bttjWrVixQunp6UXinRYAAHAdre6oLk8Pd11JTNb4D5dq5Jvf6WJcgiSpnYvOue8mw7TFVZha7Ldv316zZ89WVFSUdu7cqbCwMLm5udm2nzx5Utu3b9fff/8tSdq5c6e2b9+u8+fPS5Lq1q2rLl26aMiQIdq4caP++OMPDR06VA899BAz8QAAgEKlXctaOhB9WoOfm63Fv+3UH5sOKix8pjbvOKJ2LV2z2IfjmVrsR0REqG3bturRo4e6d++uXr16qXr16rbt06ZNU5MmTTRkyBBJUps2bdSkSRMtWrTIts+cOXNUp04ddejQQd26ddNdd92l6dOnF/i1AAAA3Ii3l4eOHjuvx174UkePn7etP3chXsPHfKtff9+t4HL+JiaEszK1z76fn5/mzp1rty4sLMz27zFjxmjMmDE3PUbp0qX11VdfOSIeAABAvkhMStHU2auz3GYY0pffbSzgRIXD1Xn28zZY9lbP6ypccjYeAAAAwBVQ7AMAAABOqkjfVAsAAABF19U57805r6ugZR8AAABwUrTsAwAAwBRmzXnPPPsAAAAAijyKfQAAAMBJ0Y0HAAAAprBYDFktBd+lxmLCOc1Cyz4AAADgpGjZBwAAgCnclC43k87rKmjZBwAAAJwUxT4AAADgpOjGAwAAAFMwz77j0bIPAAAAOCla9gEAAGAKq0lTb5pxTrPQsg8AAAA4KYp9AAAAwEnRjQcAAACmYICu49GyDwAAADgpWvYBAABgClr2HY+WfQAAAMBJ0bIvyTCuvrtLjk8xOYljxXukmx3B4ZIvJ5sdoUAkuKWZHcHhElNSzY7gcO5W57/GeHfn/1lNSHf+a5Rc4zWZmpJodgSHybi2jJoHroNiX9KlS5ckSbO6/WByEsf6xOwABeKw2QEKxAyzAwCAU1pudgCHu3Tpkvz9/c2OYWO1XF3MOK+roNiXFBwcrJiYGPn6+spicfx3Py4uTiEhIYqJiZGfn5/Dz2cGV7hGyTWuk2t0Dlyjc+AanUdBX6dhGLp06ZKCg4Mdfi4ULhT7kqxWqypWrFjg5/Xz83PqX2SSa1yj5BrXyTU6B67ROXCNzqMgr7MwtehnsJo0QNfKAF0AAAAARR3FPgAAAOCk6MZjAi8vL40ePVpeXl5mR3EYV7hGyTWuk2t0Dlyjc+AanYerXGd2rDKn5dmVWrstBnMwAQAAoADFxcXJ399fO3YHyte34EvvS5fS1ajeacXGxjr92BBa9gEAAGAKN8vVxYzzugpX+hQDAAAAcCkU+wAAAICTohsPAAAATOEmi9xU8H1qzDinWWjZN8mJEyf0yCOPqFatWrJarQoPDzc7Ur777rvvdM899yggIEB+fn5q0aKFfvnlF7Nj5as1a9aoVatWKlOmjHx8fFSnTh298847ZsdyqD/++EPu7u4KDQ01O0q+WrVqlSwWS6bl5MmTZkfLV0lJSXr55ZdVuXJleXl5qUqVKvr888/NjpVvBgwYkOX3sX79+mZHy1dz5sxR48aNVaxYMQUFBWnQoEE6d+6c2bHy1Ycffqi6devKx8dHtWvX1hdffGF2pFuS07/78+bNU506deTt7a2GDRvq559/LtigcDoU+yZJSkpSQECAXnnlFTVu3NjsOA6xevVq3XPPPfr555+1ZcsW3X333erZs6e2bdtmdrR8U7x4cQ0dOlSrV6/Wnj179Morr+iVV17R9OnTzY7mEBcvXlT//v3VoUMHs6M4zL59+3TixAnbEhgYaHakfNWnTx8tX75cn332mfbt26evv/5atWvXNjtWvnn33Xftvn8xMTEqXbq0evfubXa0fPPHH3+of//+Gjx4sP766y/NmzdPGzdu1JAhQ8yOlm+mTp2qiIgIjRkzRn/99ZfGjh2rp556Sj/++KPZ0fIsJ3/3165dq4cffliDBw/Wtm3b1KtXL/Xq1Uu7du0q4LQFx2ri4jIMOMTHH39sBAUFGWlpaXbr7733XmPgwIF269q2bWs888wzBZguf+TmGjPUq1fPGDt2bEHEyxd5ucb777/fePTRRwsiXr7J6XX27dvXeOWVV4zRo0cbjRs3LuCUtya7a1y5cqUhybhw4YI5AfNBdte4ZMkSw9/f3zh37pxJCW9dbl+TCxcuNCwWi3H48OGCinjLsrvGiRMnGtWqVbPb9t577xkVKlQoyJi3JLtrbNGihfHcc8/ZbRsxYoTRqlWrgoyZK/nxd79Pnz5G9+7d7dY1b97cePzxx/M9r9liY2MNScb+PeWME8eCCnzZv6ecIcmIjY01+0vhcC71xqYg9e7dW+fOndPKlStt686fP6+lS5eqX79+JibLP7m9xvT0dF26dEmlS5cuyJi3JLfXuG3bNq1du1Zt27YtyJi3LCfXOWPGDB06dEijR482K+Ytyen3MjQ0VEFBQbrnnnv0xx9/mBE1z7K7xkWLFun222/XW2+9pQoVKqhWrVp67rnndOXKFRNT505uX5OfffaZOnbsqMqVKxdkzFuS3TW2aNFCMTEx+vnnn2UYhk6dOqX58+erW7duJqbOneyuMSkpSd7e3nbP8fHx0caNG5WSklLQcXMkP/7ur1u3Th07drRb17lzZ61bty5fs8K1UOw7SKlSpdS1a1d99dVXtnXz589X2bJldffdd5uYLP/k9hrffvttXb58WX369CnImLckp9dYsWJFeXl56fbbb9dTTz2l//znP2bEzbPsrvPAgQN68cUX9eWXX8rdvWiO68/uGoOCgjRt2jQtWLBACxYsUEhIiNq1a6etW7eamDp3srvGQ4cOac2aNdq1a5cWLlyoKVOmaP78+XryySdNTJ07ufm9888//2jJkiVO93ps1aqV5syZo759+8rT01Ply5eXv7+/PvzwQxNT505219i5c2d9+umn2rJliwzD0ObNm/Xpp58qJSVFZ8+eNTH5jeXH3/2TJ0+qXLlyduvKlSvndGOHruVmsZi2uAqKfQfq16+fFixYoKSkJElXB1Q99NBDslqd58ue02v86quvNHbsWH377bdFrg90Tq4xKipKmzdv1rRp0zRlyhR9/fXXZsXNsxtdp2EYeuSRRzR27FjVqlXL5JS35mbfy9q1a+vxxx/XbbfdppYtW+rzzz9Xy5Yti9yA65tdY3p6uiwWi+bMmaNmzZqpW7dumjx5smbNmlWkWvdz+ntn1qxZKlmypHr16mVCyltzs2vcvXu3nnnmGY0aNUpbtmzR0qVLdfjwYT3xxBMmp86dm13jq6++qq5du+rOO++Uh4eH7rvvPoWFhUlSof4b6gp/91EEmdyNyKlduXLF8PPzMxYsWGAcPXrUsFgsxpYtWzLtV1T77BtGzq7x66+/Nnx8fIzFixeblPLW5PT7mGHcuHFGrVq1CjBh/rjRdV64cMGQZLi5udkWi8ViW7d8+XKzo+dYbr+Xzz33nHHnnXcWYMJbd7Nr7N+/v1G9enW7/Xfv3n213+z+/WbEzZOcfB/T09ONGjVqGOHh4SalvDU3u8ZHH33U+Ne//mW3f1RUlCHJ+Oeff8yImyc5+T4mJycbMTExRmpqqvHRRx8Zvr6+mfrEFya3+nc/JCTEeOedd+zWjRo1ymjUqJGDEpsno89+9N4g4+zxCgW+RO8Ncpk++0Xz8/giwtvbWw888IDmzJmjv//+W7Vr11bTpk3NjpWvsrvGr7/+WoMGDdLcuXPVvXt3E5PmXW6/j+np6bZWnaLkRteZnp6unTt32u370UcfacWKFZo/f76qVq1qUuLcy+33cvv27QoKCirAhLfuZtfYqlUrzZs3T5cvX1aJEiUkSfv375fValXFihXNjJ0rOfk+/v777/r77781ePBgk1LemptdY0JCQqbudG5ubpIkwzAKPGte5eT76OHhYfvZnDt3rnr06FGoW8lv9e9+ixYttHz5crtpOZctW6YWLVo4IC1cBcW+g/Xr1089evTQX3/9pUcffdRu2/bt2yVJly9f1pkzZ7R9+3Z5enqqXr16JiTNuxtd41dffaWwsDC9++67at68ua3PoY+Pj/z9/c2Kmyc3usYPP/xQlSpVUp06dSRdnW707bff1rBhw8yKekuyuk6r1aoGDRrY7RcYGChvb+9M64uCG30vp0yZoqpVq6p+/fpKTEzUp59+qhUrVujXX381MW3e3OgaH3nkEY0bN04DBw7U2LFjdfbsWT3//PMaNGiQfHx8TEycezf73SpdHZjbvHnzIvkzmuFG19izZ08NGTJEU6dOVefOnXXixAmFh4erWbNmCg4ONjFx7t3oGvfv36+NGzeqefPmunDhgiZPnqxdu3Zp1qxZJqbNmVv5u//MM8+obdu2mjRpkrp37665c+dq8+bNTjudMwqI2R8tOLu0tDQjKOjqR0UHDx602yYp01K5cmVzgt6CG11j27Zts7zGsLAw88Lm0Y2u8b333jPq169vFCtWzPDz8zOaNGlifPTRR4X6Y+abudnP67WK4tSbGW50jRMmTDCqV69ueHt7G6VLlzbatWtnrFixwsSkeXez7+OePXuMjh07Gj4+PkbFihWNESNGGAkJCSYlzbubXePFixcNHx8fY/r06Salyx83u8b33nvPqFevnuHj42MEBQUZ/fr1M44dO2ZS0ry70TXu3r3bCA0NNXx8fAw/Pz/jvvvuM/bu3Wti0py71b/73377rVGrVq3/a+/+Y6os+ziOvw/GAeQ+RzsMJkggKcPW0hlEATXdaoMxCavtOGch6yydmNpK/EGsX+sPJ3OpG3O5SNFpKbNlWxHxB4eINcdmyR8dMVlS+rC5kQNOPwDPuZ4/ejqLDHyeBzsnDp8Xuzfu63vd1/W9zl9fbq5z38Zut5t7773XfPzxx2HMPnx+38bTdyHNXL+aHvaj70LajNnGYzNmGv3PT0RERESmvaGhIebMmUPfhTScjvBvzRoaDpK5+F8MDg7idDrDPn84/XM3vomIiIiIyJRoz76IiIiIREQMNmII/zPvIzFnpOjOvoiIiIhIlNKdfRERERGJiEi9zVZv0BURERERkWlPxb6IiIiISJTSNh4RERERiYiY//yEf96ZYyatVURERERkRlGxLyISYStWrOCFF14InS9YsIB9+/ZFLB8RkXD5/dGbkThmChX7IjJtlZWVUVJS8pexjo4ObDYb3d3dYc5q6rq6uli/fv1tHbOyspJVq1bd1jFFROSfT8W+iExbHo+H1tZWrly5clPs8OHD5OXlsWTJkghkdjNjDDdu3Piv+iYnJzN79uy/OSMREZkJVOyLyLS1cuVKkpOTOXLkyLh2v99PU1MTHo9nwmtHRkbYsWMHd911F3FxcSxatIiGhoZQvL29nfz8fOLi4khNTWXnzp3jivWRkRG2bNlCSkoK8fHxPPzww3R1dYXiXq8Xm81Gc3Mzubm5xMXF8cUXX/DTTz9RUVGBZVmkpqayd+/em3L78zYem83GO++8wxNPPMHs2bPJzs7mo48+CsUDgQAej4esrCwSEhLIyclh//79ofhrr71GY2MjZ86cwWazYbPZ8Hq9APzwww+43W7mzp2Ly+WivLycy5cvj1tHfn4+iYmJzJ07l6KiIvr6+ib8XEVE/hezbDERO2aKmbNSEYk6d9xxBxUVFRw5cgRjTKi9qamJQCDAmjVrJry2oqKC9957jwMHDuDz+Xj77bexLAuAq1evUlpaygMPPMD58+c5ePAgDQ0NvPnmm6Hrt2/fzunTp2lsbOTcuXMsWrSI4uJifvzxx3Hz7Ny5k927d+Pz+ViyZAnV1dW0t7dz5swZPvvsM7xeL+fOnbvlWl9//XXcbjfd3d2Ulpaydu3a0FzBYJD09HSampr45ptveOWVV6ipqeHUqVMAbNu2DbfbTUlJCf39/fT391NYWMjY2BjFxcU4HA46Ojro7OzEsixKSkoYHR3lxo0brFq1iuXLl9Pd3c2XX37J+vXrsc2gl9GIiEx7RkRkGvP5fAYwbW1tobZHHnnEPP300xNe09PTYwDT2tr6l/GamhqTk5NjgsFgqK2+vt5YlmUCgYDx+/0mNjbWHD9+PBQfHR01aWlpZs+ePcYYY9ra2gxgPvzww1Cf4eFhY7fbzalTp0JtAwMDJiEhwWzdujXUlpmZad56663QOWBqa2tD536/3wCmubl5wjVu2rTJPPXUU6HzdevWmfLy8nF9jh07dtM6R0ZGTEJCgmlpaTEDAwMGMF6vd8J5RET+H4ODgwYw1y/ebQL92WE/rl+82wBmcHAw0h/F30539kVkWlu8eDGFhYW8++67AFy6dImOjo5Jt/B8/fXXzJo1i+XLl/9l3OfzUVBQMO4OdlFREX6/nytXrtDb28vY2BhFRUWheGxsLPn5+fh8vnFj5eXlhX7v7e1ldHSUBx98MNTmcrnIycm55Tr/+N2DxMREnE4n165dC7XV19eTm5tLcnIylmVx6NAhvv/++0nHPH/+PJcuXcLhcGBZFpZl4XK5+PXXX+nt7cXlclFZWUlxcTFlZWXs37+f/v7+W+YqIiL/HHqplohMex6Ph82bN1NfX8/hw4dZuHDhhIU8QEJCQthyS0xMvC3jxMbGjju32WwEg0EA3n//fbZt28bevXspKCjA4XBQV1fH2bNnJx3T7/eTm5vL8ePHb4olJycDv33RecuWLXz66aecPHmS2tpaWltbeeihh27LukRkZhsaDs6oeSNBxb6ITHtut5utW7dy4sQJjh49ysaNGyfdV37fffcRDAZpb2/nscceuyl+zz33cPr0aYwxoXE6OztxOBykp6eTlJSE3W6ns7OTzMxMAMbGxujq6hr3vPw/W7hwIbGxsZw9e5aMjAwArl+/zsWLFyf94+RWOjs7KSwspKqqKtTW29s7ro/dbicQCIxru//++zl58iQpKSk4nc4Jx1+2bBnLli1j165dFBQUcOLECRX7IjIldrudefPmkZl7OWI5zJs3D7vdHrH5w0XFvohMe5ZlsXr1anbt2sXQ0BCVlZWT9l+wYAHr1q3j2Wef5cCBAyxdupS+vj6uXbuG2+2mqqqKffv2sXnzZp5//nl6enp49dVXefHFF4mJiSExMZGNGzdSXV2Ny+UiIyODPXv28PPPP0+6fciyLDweD9XV1SQlJZGSksLLL79MTMzUdlRmZ2dz9OhRWlpayMrK4tixY3R1dZGVlTVuzS0tLfT09JCUlMScOXNYu3YtdXV1lJeX88Ybb5Cenk5fXx8ffPAB27dvZ2xsjEOHDvH444+TlpZGT08P3377LRUVFVPKV0QkPj6e7777jtHR0YjlYLfbiY+Pj9j84aJiX0SigsfjoaGhgdLSUtLS0m7Z/+DBg9TU1FBVVcXAwAAZGRnU1NQAMH/+fD755BOqq6tZunQpLpcLj8dDbW1t6Prdu3cTDAZ55plnGB4eJi8vj5aWFu68885J562rq8Pv91NWVobD4eCll15icHBwSmvfsGEDX331FatXr8Zms7FmzRqqqqpobm4O9Xnuuefwer3k5eXh9/tpa2tjxYoVfP755+zYsYMnn3yS4eFh5s+fz6OPPorT6eSXX37hwoULNDY2MjAwQGpqKps2bWLDhg1TyldEBH4r+GdCsR1pNmP+8Lw6ERERERGJGnoaj4iIiIhIlFKxLyIiIiISpVTsi4iIiIhEKRX7IiIiIiJRSsW+iIiIiEiUUrEvIiIiIhKlVOyLiIiIiEQpFfsiIiIiIlFKxb6IiIiISJRSsS8iIiIiEqVU7IuIiIiIRKl/A5wyJk4/nFAyAAAAAElFTkSuQmCC",
      "text/plain": [
       "<Figure size 800x700 with 2 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[BH-FDR] q=0.05, discoveries: 20 / 100\n",
      "[weights] mean w_id=0.548, mean w_J=0.452\n"
     ]
    }
   ],
   "source": [
    "import numpy as np\n",
    "import itertools\n",
    "import matplotlib.pyplot as plt\n",
    "from matplotlib import colors\n",
    "from scipy.stats import norm as norm_dist, rankdata\n",
    "\n",
    "# ============================================================\n",
    "# 1) Clayton copula (d-dimensional)\n",
    "# ============================================================\n",
    "def clayton_copula_sample_nd(n, theta, d, rng=None):\n",
    "    if rng is None:\n",
    "        rng = np.random.RandomState(123)\n",
    "    if theta == 0:\n",
    "        return rng.uniform(0, 1, size=(n, d))\n",
    "    S = rng.gamma(shape=1.0 / theta, scale=1.0, size=n)\n",
    "    E = rng.exponential(scale=1.0, size=(n, d))\n",
    "    U = (1.0 + E / S[:, None]) ** (-1.0 / theta)\n",
    "    return U\n",
    "\n",
    "# ============================================================\n",
    "# 2) Transform families (coordinate-wise)\n",
    "# ============================================================\n",
    "def transform_logquad(u, v, b):\n",
    "    Z = norm_dist.ppf(u)\n",
    "    X_base = np.log1p(Z**2)\n",
    "    X = X_base / (1.0 + X_base)\n",
    "    Y = np.cos(b * X + v) * np.exp(-b * (X - 0.7) ** 2)\n",
    "    return X, Y\n",
    "\n",
    "def transform_multidim(u, v, b, transform_key):\n",
    "    d = u.shape[1]\n",
    "    X = np.zeros_like(u, dtype=float)\n",
    "    Y = np.zeros_like(v, dtype=float)\n",
    "    for r in range(d):\n",
    "        u_r = u[:, [r]]\n",
    "        v_r = v[:, [r]]\n",
    "        if transform_key == \"trigU\":\n",
    "            x_r = np.sin(norm_dist.ppf(u_r))\n",
    "            y_r = np.cos(b * x_r + v_r)\n",
    "        elif transform_key == \"expquad\":\n",
    "            x_r = np.exp(-norm_dist.ppf(u_r) ** 2)\n",
    "            y_r = np.exp(-b * (x_r - 1.0) ** 2 + v_r)\n",
    "        elif transform_key == \"linear\":\n",
    "            x_r = u_r\n",
    "            y_r = b * x_r + v_r\n",
    "        elif transform_key == \"logquad\":\n",
    "            x_r, y_r = transform_logquad(u_r, v_r, b)\n",
    "        else:\n",
    "            raise ValueError(f\"Unknown transform: {transform_key}\")\n",
    "        X[:, r] = x_r[:, 0]\n",
    "        Y[:, r] = y_r[:, 0]\n",
    "    return X, Y\n",
    "\n",
    "# ============================================================\n",
    "# 3) Binary-expansion features\n",
    "# ============================================================\n",
    "def bits_from_uniform(u, K):\n",
    "    M = 1 << K\n",
    "    z = np.minimum((u * M).astype(int), M - 1)\n",
    "    bits = np.array([((z >> (K - 1 - k)) & 1).astype(int) for k in range(K)])\n",
    "    return bits\n",
    "\n",
    "def all_nonempty_subsets_indices(K):\n",
    "    idx = list(range(1, K + 1))\n",
    "    out = []\n",
    "    for r in range(1, K + 1):\n",
    "        out.extend(itertools.combinations(idx, r))\n",
    "    return out\n",
    "\n",
    "def features_by_u(u, K, subsets):\n",
    "    bits = bits_from_uniform(u, K)\n",
    "    n = u.shape[0]\n",
    "    F = np.empty((len(subsets), n), dtype=float)\n",
    "    for i, S in enumerate(subsets):\n",
    "        rows = [r - 1 for r in S]\n",
    "        ind = np.prod(bits[rows, :], axis=0)\n",
    "        F[i, :] = ind.astype(float) - (2.0 ** (-len(S)))\n",
    "    return F\n",
    "\n",
    "def build_AB_features_from_multidim(X, Y, K, subsets):\n",
    "    n, d = X.shape\n",
    "    feats_A, feats_B = [], []\n",
    "    for r in range(d):\n",
    "        xu = rankdata(X[:, r]) / (n + 1)\n",
    "        yu = rankdata(Y[:, r]) / (n + 1)\n",
    "        feats_A.append(features_by_u(xu, K, subsets))\n",
    "        feats_B.append(features_by_u(yu, K, subsets))\n",
    "    A = np.vstack(feats_A)\n",
    "    B = np.vstack(feats_B)\n",
    "    return A, B\n",
    "\n",
    "# ============================================================\n",
    "# 4) Numeric J(K) + block_diag\n",
    "# ============================================================\n",
    "def trapezoid_weights(x):\n",
    "    w = np.zeros_like(x)\n",
    "    dx = np.diff(x)\n",
    "    w[1:-1] = 0.5 * (dx[:-1] + dx[1:])\n",
    "    w[0]  = 0.5 * (x[1] - x[0])\n",
    "    w[-1] = 0.5 * (x[-1] - x[-2])\n",
    "    return w\n",
    "\n",
    "def J_numeric_K(K, t_min=1e-4, t_max=100.0, T=2001):\n",
    "    subsets = all_nonempty_subsets_indices(K)\n",
    "    m = len(subsets)\n",
    "    t = np.logspace(np.log10(t_min), np.log10(t_max), T)\n",
    "    w = trapezoid_weights(t) / (t ** 2)\n",
    "\n",
    "    inv_pows = np.array([1.0 / (2 ** r) for r in range(1, K + 1)])\n",
    "    P = np.empty((m, T))\n",
    "    for i, S in enumerate(subsets):\n",
    "        vals = np.ones_like(t)\n",
    "        inS = np.zeros(K, dtype=bool)\n",
    "        inS[[r - 1 for r in S]] = True\n",
    "        for r in range(K):\n",
    "            ang = t * inv_pows[r]\n",
    "            vals *= np.sin(ang) if inS[r] else np.cos(ang)\n",
    "        P[i, :] = vals\n",
    "\n",
    "    J = (P * w) @ P.T\n",
    "    J = 0.5 * (J + J.T)\n",
    "    return J, subsets\n",
    "\n",
    "def block_diag(*mats):\n",
    "    r = sum(m.shape[0] for m in mats)\n",
    "    c = sum(m.shape[1] for m in mats)\n",
    "    out = np.zeros((r, c), dtype=float)\n",
    "    i = j = 0\n",
    "    for m in mats:\n",
    "        rr, cc = m.shape\n",
    "        out[i:i + rr, j:j + cc] = m\n",
    "        i += rr\n",
    "        j += cc\n",
    "    return out\n",
    "\n",
    "# ============================================================\n",
    "# 5) Full statistic (T) + plugin variance\n",
    "# ============================================================\n",
    "def compute_full_T(A, B, W_A, W_B, W_C):\n",
    "    n = A.shape[1]\n",
    "    KA = (A.T @ W_A) @ A\n",
    "    KB = (B.T @ W_B) @ B\n",
    "    C  = np.vstack((A, B))\n",
    "    KC = (C.T @ W_C) @ C\n",
    "\n",
    "    off = ~np.eye(n, dtype=bool)\n",
    "\n",
    "    T1 = (KA[off] * KB[off]).sum() / (n * (n - 1))\n",
    "\n",
    "    def sums(dot):\n",
    "        S1 = dot.sum() - np.trace(dot)\n",
    "        row_off = dot.sum(axis=1) - np.diag(dot)\n",
    "        S2 = np.sum(row_off ** 2)\n",
    "        S3 = (dot ** 2).sum() - np.trace(dot ** 2)\n",
    "        return S1, S2, S3\n",
    "\n",
    "    S1C, S2C, S3C = sums(KC)\n",
    "    S1A, S2A, S3A = sums(KA)\n",
    "    S1B, S2B, S3B = sums(KB)\n",
    "\n",
    "    T2 = ((S2C - S3C) - (S2A - S3A) - (S2B - S3B)) / (2 * n * (n - 1) * (n - 2))\n",
    "\n",
    "    term = lambda S1, S2, S3: (S1 ** 2) - 4 * (S2 - S3) - 2 * S3\n",
    "    T3 = (term(S1C, S2C, S3C) - term(S1A, S2A, S3A) - term(S1B, S2B, S3B)) \\\n",
    "         / (2 * n * (n - 1) * (n - 2) * (n - 3))\n",
    "\n",
    "    return T1 - 2 * T2 + T3\n",
    "\n",
    "def plugin_var_tildeT1(A, B, W_A, W_B, unbiased=True):\n",
    "    n = A.shape[1]\n",
    "    A_c = A - A.mean(axis=1, keepdims=True)\n",
    "    B_c = B - B.mean(axis=1, keepdims=True)\n",
    "    denom = (n - 1) if unbiased else n\n",
    "    S_A = (A_c @ A_c.T) / denom\n",
    "    S_B = (B_c @ B_c.T) / denom\n",
    "    EA = np.trace(W_A @ S_A @ W_A @ S_A)\n",
    "    EB = np.trace(W_B @ S_B @ W_B @ S_B)\n",
    "    return (2.0 / (n * (n - 1))) * EA * EB\n",
    "\n",
    "# ============================================================\n",
    "# 6) DGP wrapper\n",
    "# ============================================================\n",
    "def generate_once_nd(n, theta, b, K, transform_key, subsets, d=10, seed=1234):\n",
    "    rng = np.random.RandomState(seed)\n",
    "    u = clayton_copula_sample_nd(n, theta, d, rng=rng)\n",
    "    v = clayton_copula_sample_nd(n, theta, d, rng=rng)\n",
    "    X, Y = transform_multidim(u, v, b=b, transform_key=transform_key)\n",
    "    A, B = build_AB_features_from_multidim(X, Y, K=K, subsets=subsets)\n",
    "    return A, B\n",
    "\n",
    "# ============================================================\n",
    "# 7) Pairwise (r,s) blocks + 10-fold SNR blending\n",
    "# ============================================================\n",
    "def block_view(M, r, base_dim):\n",
    "    return M[r * base_dim:(r + 1) * base_dim, :]\n",
    "\n",
    "def ten_folds_indices(n, rng):\n",
    "    idx = rng.permutation(n)\n",
    "    return np.array_split(idx, 10)\n",
    "\n",
    "def Z_for_pair(A_pair, B_pair, W_base, unbiased_plugin=True):\n",
    "    W_C = block_diag(W_base, W_base)\n",
    "    T  = compute_full_T(A_pair, B_pair, W_base, W_base, W_C)\n",
    "    vT = plugin_var_tildeT1(A_pair, B_pair, W_base, W_base, unbiased=unbiased_plugin)\n",
    "    return T / np.sqrt(max(vT, 1e-16))\n",
    "\n",
    "def blended_weight_from_10fold(A_pair, B_pair, W_id, W_J, rng, unbiased_plugin=True):\n",
    "    folds = ten_folds_indices(A_pair.shape[1], rng)\n",
    "    picks = []\n",
    "    for fidx in folds:\n",
    "        A_f = A_pair[:, fidx]\n",
    "        B_f = B_pair[:, fidx]\n",
    "        Z_id = Z_for_pair(A_f, B_f, W_id, unbiased_plugin=unbiased_plugin)\n",
    "        Z_J  = Z_for_pair(A_f, B_f, W_J,  unbiased_plugin=unbiased_plugin)\n",
    "        picks.append(\"identity\" if Z_id >= Z_J else \"J\")\n",
    "\n",
    "    cnt_id = sum(p == \"identity\" for p in picks)\n",
    "    cnt_J  = 10 - cnt_id\n",
    "\n",
    "    w_id = cnt_id / 10.0\n",
    "    w_J  = cnt_J  / 10.0\n",
    "\n",
    "    W_blend = w_id * W_id + w_J * W_J\n",
    "    return W_blend, w_id, w_J\n",
    "\n",
    "# ============================================================\n",
    "# 8) BH-FDR\n",
    "# ============================================================\n",
    "def bh_fdr_mask(pvals_2d, q=0.05):\n",
    "    p = pvals_2d.flatten()\n",
    "    m = p.size\n",
    "    order = np.argsort(p)\n",
    "    p_sorted = p[order]\n",
    "    thresh = q * (np.arange(1, m + 1) / m)\n",
    "    below = p_sorted <= thresh\n",
    "\n",
    "    reject = np.zeros(m, dtype=bool)\n",
    "    if np.any(below):\n",
    "        kmax = np.max(np.where(below))\n",
    "        reject[order[:kmax + 1]] = True\n",
    "    return reject.reshape(pvals_2d.shape)\n",
    "\n",
    "# ============================================================\n",
    "# 9) Main: pairwise heatmap using 10-fold blended statistic\n",
    "# ============================================================\n",
    "def pairwise_heatmap_new_stat(\n",
    "    n=500, d_coords=10, theta=2, K=4, b=0.1, transform_key=\"logquad\",\n",
    "    q_fdr=0.05, seed_data=1234, seed_folds=999, unbiased_plugin=True,\n",
    "    J_reuse=True\n",
    "):\n",
    "    subsets = all_nonempty_subsets_indices(K)\n",
    "    base_dim = (1 << K) - 1\n",
    "\n",
    "    # Generate one dataset\n",
    "    A, B = generate_once_nd(\n",
    "        n=n, theta=theta, b=b, K=K, transform_key=transform_key,\n",
    "        subsets=subsets, d=d_coords, seed=seed_data\n",
    "    )\n",
    "\n",
    "    # Precompute base weights (base_dim x base_dim)\n",
    "    W_id = np.eye(base_dim)\n",
    "\n",
    "    if J_reuse:\n",
    "        J_base, subsJ = J_numeric_K(K)\n",
    "        if subsJ != subsets:\n",
    "            idx = {S: i for i, S in enumerate(subsJ)}\n",
    "            perm = [idx[S] for S in subsets]\n",
    "            J_base = J_base[np.ix_(perm, perm)]\n",
    "        W_J = J_base\n",
    "    else:\n",
    "        W_J, _ = J_numeric_K(K)\n",
    "\n",
    "    # Output matrices\n",
    "    Zmat = np.zeros((d_coords, d_coords))\n",
    "    Tmat = np.zeros((d_coords, d_coords))\n",
    "    Vmat = np.zeros((d_coords, d_coords))\n",
    "    wid  = np.zeros((d_coords, d_coords))\n",
    "    wj   = np.zeros((d_coords, d_coords))\n",
    "\n",
    "    rng = np.random.RandomState(seed_folds)\n",
    "\n",
    "    for r in range(d_coords):\n",
    "        A_r = block_view(A, r, base_dim)\n",
    "        for s in range(d_coords):\n",
    "            B_s = block_view(B, s, base_dim)\n",
    "\n",
    "            # NEW: 10-fold SNR -> blended base weight for this (r,s)\n",
    "            W_blend, w_id, w_J = blended_weight_from_10fold(\n",
    "                A_r, B_s, W_id, W_J, rng=rng, unbiased_plugin=unbiased_plugin\n",
    "            )\n",
    "            wid[r, s] = w_id\n",
    "            wj[r, s]  = w_J\n",
    "\n",
    "            # Compute final pairwise test with blended weights on FULL data\n",
    "            W_C = block_diag(W_blend, W_blend)\n",
    "            T_rs  = compute_full_T(A_r, B_s, W_blend, W_blend, W_C)\n",
    "            v_rs  = plugin_var_tildeT1(A_r, B_s, W_blend, W_blend, unbiased=unbiased_plugin)\n",
    "            Z_rs  = T_rs / np.sqrt(max(v_rs, 1e-16))\n",
    "\n",
    "            Tmat[r, s] = T_rs\n",
    "            Vmat[r, s] = v_rs\n",
    "            Zmat[r, s] = Z_rs\n",
    "\n",
    "    # One-sided p-values (consistent with reject Z large)\n",
    "    Pmat = 1.0 - norm_dist.cdf(Zmat)\n",
    "    sig_bh = bh_fdr_mask(Pmat, q=q_fdr)\n",
    "\n",
    "    # ------------------------------------------------------------\n",
    "    # BH-FDR discoveries\n",
    "    # ------------------------------------------------------------\n",
    "    fig, ax = plt.subplots(figsize=(8, 7))\n",
    "\n",
    "    norm = colors.Normalize(vmin=float(np.min(Zmat)), vmax=float(np.max(Zmat)))\n",
    "\n",
    "    im = ax.imshow(\n",
    "        Zmat,\n",
    "        aspect=\"equal\",\n",
    "        cmap=\"viridis_r\",   \n",
    "        norm=norm\n",
    "    )\n",
    "\n",
    "    cb = plt.colorbar(im, ax=ax)\n",
    "    cb.set_label(\"Z statistic\", fontsize=11)\n",
    "\n",
    "    ax.set_title(f\"Pairwise Z heatmap | ★ = BH-FDR discoveries (q={q_fdr})\")\n",
    "    ax.set_xlabel(\"V coordinates\")\n",
    "    ax.set_ylabel(\"U coordinates\")\n",
    "\n",
    "    ax.set_xticks(range(d_coords))\n",
    "    ax.set_yticks(range(d_coords))\n",
    "    ax.set_xticklabels([f\"v{i+1}\" for i in range(d_coords)])\n",
    "    ax.set_yticklabels([f\"u{i+1}\" for i in range(d_coords)])\n",
    "\n",
    "    # Stars on significant cells\n",
    "    for r in range(d_coords):\n",
    "        for s in range(d_coords):\n",
    "            if sig_bh[r, s]:\n",
    "                ax.text(\n",
    "                    s, r, \"★\",\n",
    "                    ha=\"center\", va=\"center\",\n",
    "                    fontsize=13,\n",
    "                    fontweight=\"bold\",\n",
    "                    color=\"white\"\n",
    "                )\n",
    "\n",
    "    plt.tight_layout()\n",
    "    plt.show()\n",
    "\n",
    "    print(f\"[BH-FDR] q={q_fdr}, discoveries: {int(sig_bh.sum())} / {d_coords*d_coords}\")\n",
    "    print(f\"[weights] mean w_id={wid.mean():.3f}, mean w_J={wj.mean():.3f}\")\n",
    "\n",
    "    return {\n",
    "        \"Z\": Zmat, \"T\": Tmat, \"Var\": Vmat,\n",
    "        \"p\": Pmat, \"sig_bh\": sig_bh,\n",
    "        \"w_id\": wid, \"w_J\": wj\n",
    "    }\n",
    "\n",
    "# ============================================================\n",
    "# 10) Example usage\n",
    "# ============================================================\n",
    "if __name__ == \"__main__\":\n",
    "    out = pairwise_heatmap_new_stat(\n",
    "        n=500, d_coords=10, theta=2, K=4,\n",
    "        b=0.4,\n",
    "        transform_key=\"logquad\",\n",
    "        q_fdr=0.05,\n",
    "        seed_data=123,\n",
    "        unbiased_plugin=True,\n",
    "        J_reuse=True\n",
    "    )\n"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "5f042cb9",
   "metadata": {},
   "source": [
    "## Compared with HSIC and dCov"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "f3b7b6b9-32c4-44dd-91cc-675364043d6a",
   "metadata": {},
   "source": [
    "## HSIC dim = 10"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 23,
   "id": "f25db40d-5b16-4085-a47d-b2f210a44ec1",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Excel export failed (No module named 'openpyxl'). Falling back to CSVs...\n",
      "Saved CSV for n=250 to: hsic_results_n250.csv\n",
      "Saved CSV for n=500 to: hsic_results_n500.csv\n",
      "Saved CSV for n=1000 to: hsic_results_n1000.csv\n",
      "\n",
      "=== HSIC results: n=250, transform=expquad, D=10 ===\n",
      "Type I (b=0): 0.050\n",
      "Power (b=0.1): 0.198\n",
      "Power (b=0.15): 0.470\n",
      "Power (b=0.3): 1.000\n",
      "\n",
      "=== HSIC results: n=250, transform=linear, D=10 ===\n",
      "Type I (b=0): 0.046\n",
      "Power (b=0.1): 0.228\n",
      "Power (b=0.2): 0.731\n",
      "Power (b=0.3): 0.981\n",
      "\n",
      "=== HSIC results: n=250, transform=logquad, D=10 ===\n",
      "Type I (b=0): 0.042\n",
      "Power (b=0.1): 0.071\n",
      "Power (b=0.3): 0.176\n",
      "Power (b=0.5): 0.637\n",
      "\n",
      "=== HSIC results: n=250, transform=trigU, D=10 ===\n",
      "Type I (b=0): 0.054\n",
      "Power (b=0.05): 0.412\n",
      "Power (b=0.08): 0.799\n",
      "Power (b=0.1): 0.949\n",
      "\n",
      "=== HSIC results: n=500, transform=expquad, D=10 ===\n",
      "Type I (b=0): 0.042\n",
      "Power (b=0.07): 0.189\n",
      "Power (b=0.15): 0.874\n",
      "Power (b=0.2): 0.998\n",
      "\n",
      "=== HSIC results: n=500, transform=linear, D=10 ===\n",
      "Type I (b=0): 0.048\n",
      "Power (b=0.05): 0.127\n",
      "Power (b=0.1): 0.409\n",
      "Power (b=0.2): 0.960\n",
      "\n",
      "=== HSIC results: n=500, transform=logquad, D=10 ===\n",
      "Type I (b=0): 0.042\n",
      "Power (b=0.1): 0.082\n",
      "Power (b=0.2): 0.197\n",
      "Power (b=0.3): 0.529\n",
      "\n",
      "=== HSIC results: n=500, transform=trigU, D=10 ===\n",
      "Type I (b=0): 0.052\n",
      "Power (b=0.03): 0.301\n",
      "Power (b=0.05): 0.698\n",
      "Power (b=0.1): 1.000\n",
      "\n",
      "=== HSIC results: n=1000, transform=expquad, D=10 ===\n",
      "Type I (b=0): 0.049\n",
      "Power (b=0.05): 0.200\n",
      "Power (b=0.07): 0.396\n",
      "Power (b=0.12): 0.949\n",
      "\n",
      "=== HSIC results: n=1000, transform=linear, D=10 ===\n",
      "Type I (b=0): 0.041\n",
      "Power (b=0.05): 0.241\n",
      "Power (b=0.08): 0.494\n",
      "Power (b=0.15): 0.988\n",
      "\n",
      "=== HSIC results: n=1000, transform=logquad, D=10 ===\n",
      "Type I (b=0): 0.059\n",
      "Power (b=0.07): 0.081\n",
      "Power (b=0.15): 0.250\n",
      "Power (b=0.2): 0.591\n",
      "\n",
      "=== HSIC results: n=1000, transform=trigU, D=10 ===\n",
      "Type I (b=0): 0.042\n",
      "Power (b=0.01): 0.074\n",
      "Power (b=0.03): 0.602\n",
      "Power (b=0.05): 0.948\n"
     ]
    }
   ],
   "source": [
    "import numpy as np\n",
    "import pandas as pd\n",
    "from scipy.stats import norm\n",
    "from hyppo.independence import Hsic\n",
    "\n",
    "# -----------------------------\n",
    "# d-dimensional Clayton sampler\n",
    "# -----------------------------\n",
    "def clayton_copula_sample_nd(n, theta, d):\n",
    "    \"\"\"\n",
    "    Returns U in R^{n x d} with Clayton(theta) dependence across columns.\n",
    "    theta = 0 -> independent uniforms.\n",
    "    For theta > 0: S ~ Gamma(1/theta, 1), E_i ~ Exp(1), U_i = (1 + E_i / S)^(-1/theta)\n",
    "    \"\"\"\n",
    "    if theta == 0:\n",
    "        return np.random.uniform(0, 1, size=(n, d))\n",
    "    S = np.random.gamma(shape=1.0/theta, scale=1.0, size=n)  # (n,)\n",
    "    E = np.random.exponential(scale=1.0, size=(n, d))\n",
    "    U = (1.0 + E / S[:, None]) ** (-1.0/theta)\n",
    "    return U\n",
    "\n",
    "# -----------------------------\n",
    "# Transforms (generalized to D)\n",
    "# -----------------------------\n",
    "def _broadcast_b(b, d):\n",
    "    b = np.asarray(b)\n",
    "    if b.ndim == 0:\n",
    "        return np.full(d, float(b))\n",
    "    assert b.shape == (d,), f\"b must be scalar or shape ({d},)\"\n",
    "    return b\n",
    "\n",
    "def transform_trig_uniform_nd(u, v, b):\n",
    "    \"\"\"\n",
    "    x_j = sin(Phi^{-1}(u_j))\n",
    "    y_j = cos(b_j * x_j + v_j),   with v_j ~ U(0,1)\n",
    "    \"\"\"\n",
    "    n, d = u.shape\n",
    "    b = _broadcast_b(b, d)\n",
    "    x = np.sin(norm.ppf(u))\n",
    "    y = np.cos(x * b[None, :] + v)\n",
    "    return x, y\n",
    "\n",
    "def transform_expquad_nd(u, v, b):\n",
    "    \"\"\"\n",
    "    x_j = exp(-(Phi^{-1}(u_j))^2)\n",
    "    y_j = exp(-b_j * (x_j - 1)^2 + v_j)\n",
    "    \"\"\"\n",
    "    n, d = u.shape\n",
    "    b = _broadcast_b(b, d)\n",
    "    z = norm.ppf(u)\n",
    "    x = np.exp(-(z ** 2))\n",
    "    y = np.exp(-b[None, :] * (x - 1.0) ** 2 + v)\n",
    "    return x, y\n",
    "\n",
    "def transform_linear_nd(u, v, b):\n",
    "    \"\"\"\n",
    "    x_j = u_j\n",
    "    y_j = b_j * x_j + v_j\n",
    "    \"\"\"\n",
    "    n, d = u.shape\n",
    "    b = _broadcast_b(b, d)\n",
    "    x = u.copy()\n",
    "    y = b[None, :] * x + v\n",
    "    return x, y\n",
    "\n",
    "def transform_logquad_nd(u, v, b):\n",
    "    \"\"\"\n",
    "    Your new 'logquad' family, generalized to D.\n",
    "    Z_j = Phi^{-1}(u_j)\n",
    "    X_j = log1p(Z_j^2) / (1 + log1p(Z_j^2))\n",
    "    Y_j = cos(b_j * X_j + v_j) * exp(-b_j * (X_j - 0.7)^2)\n",
    "    \"\"\"\n",
    "    n, d = u.shape\n",
    "    b = _broadcast_b(b, d)\n",
    "    Z = norm.ppf(u)\n",
    "    X_base = np.log1p(Z**2)\n",
    "    X = X_base / (1.0 + X_base)\n",
    "    Y = np.cos(b[None, :] * X + v) * np.exp(-b[None, :] * (X - 0.7) ** 2)\n",
    "    return X, Y\n",
    "\n",
    "TRANSFORM_MAP_ND = {\n",
    "    \"trigU\":   transform_trig_uniform_nd,\n",
    "    \"expquad\": transform_expquad_nd,\n",
    "    \"linear\":  transform_linear_nd,\n",
    "    \"logquad\": transform_logquad_nd,   # <--- NEW\n",
    "}\n",
    "\n",
    "# -----------------------------\n",
    "# Data generation\n",
    "# -----------------------------\n",
    "def generate_XY(n, theta, D, transform_key, b):\n",
    "    \"\"\"\n",
    "    u, v ~ Clayton(theta) in D dims; apply chosen transform to get X, Y (n x D).\n",
    "    \"\"\"\n",
    "    u = clayton_copula_sample_nd(n, theta, D)\n",
    "    v = clayton_copula_sample_nd(n, theta, D)\n",
    "    x, y = TRANSFORM_MAP_ND[transform_key](u, v, b=b)\n",
    "    return x, y\n",
    "\n",
    "# -----------------------------\n",
    "# HSIC simulation runner\n",
    "# -----------------------------\n",
    "def simulate_hsic(\n",
    "    n, theta, D, b_config,\n",
    "    n_simulations=500, alpha=0.05, seed=123,\n",
    "    report_typeI=True\n",
    "):\n",
    "    \"\"\"\n",
    "    b_config: dict like {\"trigU\":[0.05,0.1,0.5], \"expquad\":[0.1,0.2], \"linear\":[0.05], \"logquad\":[0.2]}\n",
    "              Each entry can be a scalar or a length-D vector.\n",
    "    Returns: list of dict rows with (n, transform, D, b, metric, value).\n",
    "    \"\"\"\n",
    "    rng = np.random.default_rng(seed)\n",
    "    hsic = Hsic()  # default Gaussian kernels with median heuristic\n",
    "\n",
    "    results = []\n",
    "    for transform_key, b_list in b_config.items():\n",
    "        # normalize to list\n",
    "        if not isinstance(b_list, (list, tuple, np.ndarray)):\n",
    "            b_list = [b_list]\n",
    "\n",
    "        # Type I (b = 0) if requested\n",
    "        if report_typeI:\n",
    "            rejections = 0\n",
    "            for _ in range(n_simulations):\n",
    "                X, Y = generate_XY(n, theta, D, transform_key, b=0.0)\n",
    "                stat, pval = hsic.test(X, Y, random_state=rng.integers(0, 2**31-1))\n",
    "                rejections += (pval < alpha)\n",
    "            type1 = rejections / n_simulations\n",
    "            results.append({\n",
    "                \"n\": n, \"transform\": transform_key, \"D\": D,\n",
    "                \"b\": 0.0, \"metric\": \"typeI\", \"value\": float(type1)\n",
    "            })\n",
    "\n",
    "        # Power for each b > 0\n",
    "        for b in b_list:\n",
    "            rejections = 0\n",
    "            for _ in range(n_simulations):\n",
    "                X, Y = generate_XY(n, theta, D, transform_key, b=b)\n",
    "                stat, pval = hsic.test(X, Y, random_state=rng.integers(0, 2**31-1))\n",
    "                rejections += (pval < alpha)\n",
    "            power = rejections / n_simulations\n",
    "            b_out = float(b) if np.isscalar(b) else tuple(np.asarray(b).tolist())\n",
    "            results.append({\n",
    "                \"n\": n, \"transform\": transform_key, \"D\": D,\n",
    "                \"b\": b_out, \"metric\": \"power\", \"value\": float(power)\n",
    "            })\n",
    "    return results\n",
    "\n",
    "def print_hsic_results(results):\n",
    "    by_transform = {}\n",
    "    for r in results:\n",
    "        key = (r[\"n\"], r[\"transform\"])\n",
    "        by_transform.setdefault(key, []).append(r)\n",
    "    for (n, t), rows in sorted(by_transform.items()):\n",
    "        D = rows[0][\"D\"]\n",
    "        print(f\"\\n=== HSIC results: n={n}, transform={t}, D={D} ===\")\n",
    "        # Type I\n",
    "        typeI_rows = [r for r in rows if r[\"metric\"] == \"typeI\"]\n",
    "        if typeI_rows:\n",
    "            print(f\"Type I (b=0): {typeI_rows[0]['value']:.3f}\")\n",
    "        # Power\n",
    "        for r in rows:\n",
    "            if r[\"metric\"] == \"power\":\n",
    "                print(f\"Power (b={r['b']}): {r['value']:.3f}\")\n",
    "\n",
    "# -----------------------------\n",
    "# Multi-n runner + Excel export\n",
    "# -----------------------------\n",
    "def run_multi_n_and_save(\n",
    "    n_list, theta, D, b_config_by_n,\n",
    "    n_simulations=500, alpha=0.05, seed=123,\n",
    "    excel_path=\"hsic_results.xlsx\"\n",
    "):\n",
    "    \"\"\"\n",
    "    b_config_by_n: dict mapping each n to its own b_config dict.\n",
    "       Example:\n",
    "       {\n",
    "          250:  {\"trigU\":[0.05,0.08], \"expquad\":[0.1,0.2], \"linear\":[0.05], \"logquad\":[0.2, 0.4]},\n",
    "          500:  {\"trigU\":[0.05,0.1],  \"expquad\":[0.1,0.3], \"linear\":[0.1],  \"logquad\":[0.1, 0.3]},\n",
    "          1000: {\"trigU\":[0.08,0.1],  \"expquad\":[0.2,0.3], \"linear\":[0.2],  \"logquad\":[0.15, 0.25]}\n",
    "       }\n",
    "    \"\"\"\n",
    "    all_results = {}\n",
    "    for n in n_list:\n",
    "        results = simulate_hsic(\n",
    "            n=n, theta=theta, D=D, b_config=b_config_by_n[n],\n",
    "            n_simulations=n_simulations, alpha=alpha, seed=seed,\n",
    "            report_typeI=True\n",
    "        )\n",
    "        all_results[n] = pd.DataFrame(results)\n",
    "\n",
    "    # Try writing to a single Excel file (one sheet per n)\n",
    "    try:\n",
    "        with pd.ExcelWriter(excel_path, engine=\"openpyxl\") as writer:\n",
    "            for n, df in all_results.items():\n",
    "                df.to_excel(writer, sheet_name=f\"n={n}\", index=False)\n",
    "        print(f\"Saved Excel results to: {excel_path}\")\n",
    "    except Exception as e:\n",
    "        print(f\"Excel export failed ({e}). Falling back to CSVs...\")\n",
    "        for n, df in all_results.items():\n",
    "            csv_path = f\"hsic_results_n{n}.csv\"\n",
    "            df.to_csv(csv_path, index=False)\n",
    "            print(f\"Saved CSV for n={n} to: {csv_path}\")\n",
    "\n",
    "    # Nicely print to console too\n",
    "    for n, df in all_results.items():\n",
    "        print_hsic_results(df.to_dict(\"records\"))\n",
    "\n",
    "# -----------------------------\n",
    "# Main\n",
    "# -----------------------------\n",
    "if __name__ == \"__main__\":\n",
    "    # Core settings\n",
    "    theta = 2\n",
    "    D = 10\n",
    "    alpha = 0.05\n",
    "    n_simulations = 1000\n",
    "    seed = 123\n",
    "\n",
    "    # Different sample sizes and their own b-grids (you can use scalars or length-D vectors)\n",
    "    n_list = [250, 500, 1000]\n",
    "    b_config_by_n = {\n",
    "        250: {\n",
    "            \"trigU\":   [0.05, 0.08, 0.1],\n",
    "            \"expquad\": [0.10, 0.15,0.3],\n",
    "            \"linear\":  [0.1, 0.2,0.3],\n",
    "            \"logquad\": [0.1, 0.30, 0.5]   \n",
    "        },\n",
    "        500: {\n",
    "            \"trigU\":   [0.03, 0.05, 0.10],\n",
    "            \"expquad\": [0.07, 0.15, 0.2],\n",
    "            \"linear\":  [0.05, 0.10, 0.20],\n",
    "            \"logquad\": [0.10, 0.20, 0.3]   \n",
    "        },\n",
    "        1000: {\n",
    "            \"trigU\":   [0.01, 0.03, 0.05],\n",
    "            \"expquad\": [0.05, 0.07, 0.12],\n",
    "            \"linear\":  [0.05, 0.08, 0.15],\n",
    "            \"logquad\": [0.07, 0.15, 0.2]  # NEW family\n",
    "        }\n",
    "    }\n",
    "\n",
    "    # Run and save\n",
    "    run_multi_n_and_save(\n",
    "        n_list=n_list, theta=theta, D=D,\n",
    "        b_config_by_n=b_config_by_n,\n",
    "        n_simulations=n_simulations, alpha=alpha, seed=seed,\n",
    "        excel_path=\"hsic_results.xlsx\"\n",
    "    )\n"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "123046eb-1621-4af5-95b1-1b6b66734f3e",
   "metadata": {},
   "source": [
    "## dCOV"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 24,
   "id": "7ae88269-cde4-4151-8c1f-a1bd0e017fdf",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Saved Excel results to dcorr_results.xlsx using engine=xlsxwriter\n",
      "Saved results to dcorr_results.xlsx\n",
      "\n",
      "=== dCor results: n=250, transform=expquad, D=10 ===\n",
      "Type I (b=0): 0.040\n",
      "Power (b=0.1): 0.286\n",
      "Power (b=0.15): 0.599\n",
      "Power (b=0.3): 1.000\n",
      "\n",
      "=== dCor results: n=250, transform=linear, D=10 ===\n",
      "Type I (b=0): 0.045\n",
      "Power (b=0.1): 0.278\n",
      "Power (b=0.2): 0.817\n",
      "Power (b=0.3): 0.996\n",
      "\n",
      "=== dCor results: n=250, transform=logquad, D=10 ===\n",
      "Type I (b=0): 0.057\n",
      "Power (b=0.1): 0.086\n",
      "Power (b=0.3): 0.321\n",
      "Power (b=0.5): 0.793\n",
      "\n",
      "=== dCor results: n=250, transform=trigU, D=10 ===\n",
      "Type I (b=0): 0.044\n",
      "Power (b=0.05): 0.363\n",
      "Power (b=0.08): 0.759\n",
      "Power (b=0.1): 0.932\n",
      "\n",
      "=== dCor results: n=500, transform=expquad, D=10 ===\n",
      "Type I (b=0): 0.040\n",
      "Power (b=0.07): 0.263\n",
      "Power (b=0.15): 0.929\n",
      "Power (b=0.2): 1.000\n",
      "\n",
      "=== dCor results: n=500, transform=linear, D=10 ===\n",
      "Type I (b=0): 0.061\n",
      "Power (b=0.05): 0.157\n",
      "Power (b=0.1): 0.550\n",
      "Power (b=0.2): 0.991\n",
      "\n",
      "=== dCor results: n=500, transform=logquad, D=10 ===\n",
      "Type I (b=0): 0.049\n",
      "Power (b=0.1): 0.114\n",
      "Power (b=0.2): 0.416\n",
      "Power (b=0.3): 0.880\n",
      "\n",
      "=== dCor results: n=500, transform=trigU, D=10 ===\n",
      "Type I (b=0): 0.040\n",
      "Power (b=0.03): 0.277\n",
      "Power (b=0.05): 0.670\n",
      "Power (b=0.1): 1.000\n",
      "\n",
      "=== dCor results: n=1000, transform=expquad, D=10 ===\n",
      "Type I (b=0): 0.050\n",
      "Power (b=0.05): 0.273\n",
      "Power (b=0.07): 0.555\n",
      "Power (b=0.12): 0.985\n",
      "\n",
      "=== dCor results: n=1000, transform=linear, D=10 ===\n",
      "Type I (b=0): 0.042\n",
      "Power (b=0.05): 0.271\n",
      "Power (b=0.08): 0.620\n",
      "Power (b=0.15): 0.999\n",
      "\n",
      "=== dCor results: n=1000, transform=logquad, D=10 ===\n",
      "Type I (b=0): 0.047\n",
      "Power (b=0.07): 0.107\n",
      "Power (b=0.15): 0.644\n",
      "Power (b=0.2): 0.972\n",
      "\n",
      "=== dCor results: n=1000, transform=trigU, D=10 ===\n",
      "Type I (b=0): 0.041\n",
      "Power (b=0.01): 0.096\n",
      "Power (b=0.03): 0.509\n",
      "Power (b=0.05): 0.937\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "/Users/yangyang/Desktop/CI/venv/lib/python3.11/site-packages/xlsxwriter/workbook.py:404: UserWarning: Calling close() on already closed file.\n",
      "  warn(\"Calling close() on already closed file.\")\n"
     ]
    }
   ],
   "source": [
    "import numpy as np\n",
    "import pandas as pd\n",
    "from scipy.stats import norm\n",
    "from hyppo.independence import Dcorr\n",
    "\n",
    "# -----------------------------\n",
    "# d-dimensional Clayton sampler\n",
    "# -----------------------------\n",
    "def clayton_copula_sample_nd(n, theta, d):\n",
    "    \"\"\"\n",
    "    U in R^{n x d} with Clayton(theta) dependence across columns.\n",
    "    theta=0 -> iid Uniform(0,1).\n",
    "    For theta>0: S~Gamma(1/theta,1), E~Exp(1), U=(1+E/S)^(-1/theta)\n",
    "    \"\"\"\n",
    "    if theta == 0:\n",
    "        return np.random.uniform(0, 1, size=(n, d))\n",
    "    S = np.random.gamma(shape=1.0/theta, scale=1.0, size=n)  # (n,)\n",
    "    E = np.random.exponential(scale=1.0, size=(n, d))\n",
    "    return (1.0 + E / S[:, None]) ** (-1.0/theta)\n",
    "\n",
    "# -----------------------------\n",
    "# Transforms (generalized to D)\n",
    "# -----------------------------\n",
    "def _broadcast_b(b, d):\n",
    "    b = np.asarray(b)\n",
    "    if b.ndim == 0:\n",
    "        return np.full(d, float(b))\n",
    "    assert b.shape == (d,), f\"b must be scalar or shape ({d},)\"\n",
    "    return b\n",
    "\n",
    "def transform_trig_uniform_nd(u, v, b):\n",
    "    \"\"\"\n",
    "    x_j = sin(Phi^{-1}(u_j))\n",
    "    y_j = cos(b_j * x_j + v_j),   v_j ~ U(0,1)\n",
    "    \"\"\"\n",
    "    n, d = u.shape\n",
    "    b = _broadcast_b(b, d)\n",
    "    x = np.sin(norm.ppf(u))\n",
    "    y = np.cos(x * b[None, :] + v)\n",
    "    return x, y\n",
    "\n",
    "def transform_expquad_nd(u, v, b):\n",
    "    \"\"\"\n",
    "    x_j = exp(-(Phi^{-1}(u_j))^2)\n",
    "    y_j = exp(-b_j * (x_j - 1)^2 + v_j)\n",
    "    \"\"\"\n",
    "    n, d = u.shape\n",
    "    b = _broadcast_b(b, d)\n",
    "    z = norm.ppf(u)\n",
    "    x = np.exp(-(z ** 2))\n",
    "    y = np.exp(-b[None, :] * (x - 1.0) ** 2 + v)\n",
    "    return x, y\n",
    "\n",
    "def transform_linear_nd(u, v, b):\n",
    "    \"\"\"\n",
    "    x_j = u_j\n",
    "    y_j = b_j * x_j + v_j\n",
    "    \"\"\"\n",
    "    n, d = u.shape\n",
    "    b = _broadcast_b(b, d)\n",
    "    x = u.copy()\n",
    "    y = b[None, :] * x + v\n",
    "    return x, y\n",
    "\n",
    "def transform_logquad_nd(u, v, b):\n",
    "    \"\"\"\n",
    "    'logquad' family, generalized to D.\n",
    "    Z_j = Phi^{-1}(u_j)\n",
    "    X_j = log1p(Z_j^2) / (1 + log1p(Z_j^2))\n",
    "    Y_j = cos(b_j * X_j + v_j) * exp(-b_j * (X_j - 0.7)^2)\n",
    "    \"\"\"\n",
    "    n, d = u.shape\n",
    "    b = _broadcast_b(b, d)\n",
    "    Z = norm.ppf(u)\n",
    "    X_base = np.log1p(Z**2)\n",
    "    X = X_base / (1.0 + X_base)\n",
    "    Y = np.cos(b[None, :] * X + v) * np.exp(-b[None, :] * (X - 0.7) ** 2)\n",
    "    return X, Y\n",
    "\n",
    "TRANSFORM_MAP_ND = {\n",
    "    \"trigU\":   transform_trig_uniform_nd,\n",
    "    \"expquad\": transform_expquad_nd,\n",
    "    \"linear\":  transform_linear_nd,\n",
    "    \"logquad\": transform_logquad_nd,  # NEW\n",
    "}\n",
    "\n",
    "# -----------------------------\n",
    "# Data generation\n",
    "# -----------------------------\n",
    "def generate_XY(n, theta, D, transform_key, b):\n",
    "    \"\"\"\n",
    "    u, v ~ Clayton(theta) in D dims; apply transform -> X,Y (n x D).\n",
    "    \"\"\"\n",
    "    u = clayton_copula_sample_nd(n, theta, D)\n",
    "    v = clayton_copula_sample_nd(n, theta, D)\n",
    "    x, y = TRANSFORM_MAP_ND[transform_key](u, v, b=b)\n",
    "    return x, y\n",
    "\n",
    "# -----------------------------\n",
    "# dCor simulation runner\n",
    "# -----------------------------\n",
    "def simulate_dcorr(\n",
    "    n, theta, D, b_config,\n",
    "    n_simulations=500, alpha=0.05, seed=123,\n",
    "    report_typeI=True, reps=1000, workers=1\n",
    "):\n",
    "    \"\"\"\n",
    "    b_config: {\"trigU\":[0.05,0.1,0.5], \"expquad\":[0.1,0.2], \"linear\":[0.05], \"logquad\":[...]}\n",
    "              (entries can be scalar or length-D vectors)\n",
    "    Uses hyppo.independence.Dcorr with 'reps' permutations.\n",
    "    Returns list of dict rows: (n, transform, D, b, metric, value).\n",
    "    \"\"\"\n",
    "    rng = np.random.default_rng(seed)\n",
    "    dcorr = Dcorr()\n",
    "\n",
    "    results = []\n",
    "    for transform_key, b_list in b_config.items():\n",
    "        if not isinstance(b_list, (list, tuple, np.ndarray)):\n",
    "            b_list = [b_list]\n",
    "\n",
    "        # Type I: b=0\n",
    "        if report_typeI:\n",
    "            rejections = 0\n",
    "            for _ in range(n_simulations):\n",
    "                X, Y = generate_XY(n, theta, D, transform_key, b=0.0)\n",
    "                stat, pval = dcorr.test(X, Y, reps=reps, workers=workers)\n",
    "                rejections += (pval < alpha)\n",
    "            results.append({\n",
    "                \"n\": n, \"transform\": transform_key, \"D\": D,\n",
    "                \"b\": 0.0, \"metric\": \"typeI\",\n",
    "                \"value\": rejections / n_simulations\n",
    "            })\n",
    "\n",
    "        # Power: b > 0\n",
    "        for b in b_list:\n",
    "            rejections = 0\n",
    "            for _ in range(n_simulations):\n",
    "                X, Y = generate_XY(n, theta, D, transform_key, b=b)\n",
    "                stat, pval = dcorr.test(X, Y, reps=reps, workers=workers)\n",
    "                rejections += (pval < alpha)\n",
    "            b_out = float(b) if np.isscalar(b) else tuple(np.asarray(b).tolist())\n",
    "            results.append({\n",
    "                \"n\": n, \"transform\": transform_key, \"D\": D,\n",
    "                \"b\": b_out, \"metric\": \"power\",\n",
    "                \"value\": rejections / n_simulations\n",
    "            })\n",
    "    return results\n",
    "\n",
    "def print_dcorr_results(results):\n",
    "    by_key = {}\n",
    "    for r in results:\n",
    "        key = (r[\"n\"], r[\"transform\"])\n",
    "        by_key.setdefault(key, []).append(r)\n",
    "    for (n, t), rows in sorted(by_key.items()):\n",
    "        D = rows[0][\"D\"]\n",
    "        print(f\"\\n=== dCor results: n={n}, transform={t}, D={D} ===\")\n",
    "        typeI = [r for r in rows if r[\"metric\"] == \"typeI\"]\n",
    "        if typeI:\n",
    "            print(f\"Type I (b=0): {typeI[0]['value']:.3f}\")\n",
    "        for r in rows:\n",
    "            if r[\"metric\"] == \"power\":\n",
    "                print(f\"Power (b={r['b']}): {r['value']:.3f}\")\n",
    "\n",
    "# -----------------------------\n",
    "# Multi-n runner + Excel export\n",
    "# -----------------------------\n",
    "def run_multi_n_and_save_dcorr(\n",
    "    n_list, theta, D, b_config_by_n,\n",
    "    n_simulations=500, alpha=0.05, seed=123,\n",
    "    report_typeI=True, reps=1000, workers=1,\n",
    "    excel_path=\"dcorr_results.xlsx\"\n",
    "):\n",
    "    \"\"\"\n",
    "    b_config_by_n maps each n to its own b_config dict.\n",
    "    Writes one sheet per n plus a combined sheet. Falls back to CSVs if no Excel engine.\n",
    "    \"\"\"\n",
    "    all_rows = []\n",
    "    # try openpyxl, then xlsxwriter\n",
    "    writer = None\n",
    "    engine_used = None\n",
    "    try:\n",
    "        writer = pd.ExcelWriter(excel_path, engine=\"openpyxl\")\n",
    "        engine_used = \"openpyxl\"\n",
    "    except Exception:\n",
    "        try:\n",
    "            writer = pd.ExcelWriter(excel_path, engine=\"xlsxwriter\")\n",
    "            engine_used = \"xlsxwriter\"\n",
    "        except Exception:\n",
    "            engine_used = None\n",
    "\n",
    "    try:\n",
    "        for n in n_list:\n",
    "            if n not in b_config_by_n:\n",
    "                raise ValueError(f\"Missing b_config for n={n}\")\n",
    "            res = simulate_dcorr(\n",
    "                n=n, theta=theta, D=D, b_config=b_config_by_n[n],\n",
    "                n_simulations=n_simulations, alpha=alpha, seed=seed,\n",
    "                report_typeI=report_typeI, reps=reps, workers=workers\n",
    "            )\n",
    "            df_n = pd.DataFrame(res)\n",
    "            all_rows.extend(res)\n",
    "            if writer is not None:\n",
    "                df_n.to_excel(writer, index=False, sheet_name=f\"n={n}\")\n",
    "            else:\n",
    "                df_n.to_csv(f\"dcorr_results_n{n}.csv\", index=False)\n",
    "\n",
    "        if writer is not None:\n",
    "            pd.DataFrame(all_rows).to_excel(writer, index=False, sheet_name=\"combined\")\n",
    "            writer.close()\n",
    "            print(f\"Saved Excel results to {excel_path} using engine={engine_used}\")\n",
    "        else:\n",
    "            comb_path = \"dcorr_results_combined.csv\"\n",
    "            pd.DataFrame(all_rows).to_csv(comb_path, index=False)\n",
    "            print(f\"Excel engines unavailable; saved CSVs (combined at {comb_path})\")\n",
    "    finally:\n",
    "        if writer is not None:\n",
    "            try:\n",
    "                writer.close()\n",
    "            except Exception:\n",
    "                pass\n",
    "\n",
    "    return pd.DataFrame(all_rows)\n",
    "\n",
    "# -----------------------------\n",
    "# Main\n",
    "# -----------------------------\n",
    "if __name__ == \"__main__\":\n",
    "    # Settings\n",
    "    theta = 2\n",
    "    D = 10\n",
    "    alpha = 0.05\n",
    "    n_simulations = 1000\n",
    "    seed = 123\n",
    "\n",
    "    # hyppo permutation test settings\n",
    "    reps = 1000\n",
    "    workers = 1\n",
    "\n",
    "    # Per-n b grids\n",
    "    n_list = [250, 500, 1000]\n",
    "    b_config_by_n = {\n",
    "        250: {\n",
    "            \"trigU\":   [0.05, 0.08, 0.10],\n",
    "            \"expquad\": [0.10, 0.15, 0.30],\n",
    "            \"linear\":  [0.10, 0.20, 0.30],\n",
    "            \"logquad\": [0.10, 0.30, 0.50],\n",
    "        },\n",
    "        500: {\n",
    "            \"trigU\":   [0.03, 0.05, 0.10],\n",
    "            \"expquad\": [0.07, 0.15, 0.20],\n",
    "            \"linear\":  [0.05, 0.10, 0.20],\n",
    "            \"logquad\": [0.10, 0.20, 0.30],\n",
    "        },\n",
    "        1000: {\n",
    "            \"trigU\":   [0.01, 0.03, 0.05],\n",
    "            \"expquad\": [0.05, 0.07, 0.12],\n",
    "            \"linear\":  [0.05, 0.08, 0.15],\n",
    "            \"logquad\": [0.07, 0.15, 0.20],\n",
    "        },\n",
    "    }\n",
    "\n",
    "    # Run and save\n",
    "    out_path = \"dcorr_results.xlsx\"\n",
    "    df_all = run_multi_n_and_save_dcorr(\n",
    "        n_list=n_list, theta=theta, D=D, b_config_by_n=b_config_by_n,\n",
    "        n_simulations=n_simulations, alpha=alpha, seed=seed,\n",
    "        report_typeI=True, reps=reps, workers=workers,\n",
    "        excel_path=out_path\n",
    "    )\n",
    "    print(f\"Saved results to {out_path}\")\n",
    "    print_dcorr_results(df_all.to_dict(\"records\"))\n"
   ]
  }
 ],
 "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": 5
}
