{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "import os, sys\n",
    "from pathlib import Path"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "import numpy as np\n",
    "import pandas as pd\n",
    "\n",
    "from matplotlib import pyplot as plt\n",
    "import seaborn as sns\n",
    "\n",
    "from tqdm import tqdm"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "p = [\n",
    "    \"#000000\",\n",
    "    \"#E69F00\",\n",
    "    \"#56B4E9\",\n",
    "    \"#009E73\",\n",
    "    \"#FB6467FF\",\n",
    "    \"#808282\",\n",
    "    \"#F0E442\",\n",
    "    \"#440154FF\",\n",
    "    \"#0072B2\",\n",
    "    \"#D55E00\",\n",
    "    \"#CC79A7\",\n",
    "    \"#C2CD23\",\n",
    "    \"#918BC3\",\n",
    "    \"#FFFFFF\",\n",
    "]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "sns.color_palette(p)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "os.chdir(\"path/to/where/hephaestus/is/placed\")\n",
    "\n",
    "sys.path.insert(0, os.getcwd())\n",
    "sys.path.insert(0, os.path.join(os.getcwd(), \"hephaestus_lab\", \"evaluation\"))\n",
    "\n",
    "from persistent_patterns.verify_file import verify\n",
    "import hephaestus.utils.load_general_config as hconfig"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "DATA_DIR = Path(\"where/the/results/from/evaluate/are/stored\")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "assert verify(DATA_DIR / \"preds\" / \"nd_preds\" / \"TALOS_20240201-180351+TorchTrainer_d9c8c887+nd.csv\")\n",
    "df_nd_gin = pd.read_csv(\n",
    "    DATA_DIR / \"preds\" / \"nd_preds\" / \"TALOS_20240201-180351+TorchTrainer_d9c8c887+nd.csv\",\n",
    "    usecols=[i for i in range(1, hconfig.NUM_SUBGRAPHS + 4)],\n",
    ")\n",
    "\n",
    "assert verify(DATA_DIR / \"preds\" / \"nd_preds\" / \"TALOS_20240216-185022+TorchTrainer_995a1ad7+nd.csv\")\n",
    "df_nd_sage = pd.read_csv(\n",
    "    DATA_DIR / \"preds\" / \"nd_preds\" / \"TALOS_20240216-185022+TorchTrainer_995a1ad7+nd.csv\",\n",
    "    usecols=[i for i in range(1, hconfig.NUM_SUBGRAPHS + 4)],\n",
    ")\n",
    "\n",
    "assert verify(DATA_DIR / \"preds\" / \"d_preds\" / \"TALOS_20240121-203233+TorchTrainer_e9bc79c6+d.csv\")\n",
    "df_d_gin = pd.read_csv(\n",
    "    DATA_DIR / \"preds\" / \"d_preds\" / \"TALOS_20240121-203233+TorchTrainer_e9bc79c6+d.csv\",\n",
    "    usecols=[i for i in range(1, hconfig.NUM_SUBGRAPHS + 4)],\n",
    ")\n",
    "\n",
    "######################## Real ########################\n",
    "assert verify(DATA_DIR / \"preds\" / \"mlreal_preds\" / \"TALOS_20240121-203233+TorchTrainer_e9bc79c6+mlreal.csv\")\n",
    "df_d_gin_mlreal = pd.read_csv(\n",
    "    DATA_DIR / \"preds\" / \"mlreal_preds\" / \"TALOS_20240121-203233+TorchTrainer_e9bc79c6+mlreal.csv\",\n",
    "    usecols=[i for i in range(1, hconfig.NUM_SUBGRAPHS + 4)],\n",
    ")\n",
    "\n",
    "assert verify(DATA_DIR / \"preds\" / \"mlreal_preds\" / \"TALOS_20240201-180351+TorchTrainer_d9c8c887+mlreal.csv\")\n",
    "df_nd_gin_mlreal = pd.read_csv(\n",
    "    DATA_DIR / \"preds\" / \"mlreal_preds\" / \"TALOS_20240201-180351+TorchTrainer_d9c8c887+mlreal.csv\",\n",
    "    usecols=[i for i in range(1, hconfig.NUM_SUBGRAPHS + 4)],\n",
    ")\n",
    "\n",
    "assert verify(DATA_DIR / \"preds\" / \"mlreal_preds\" / \"TALOS_20240216-185022+TorchTrainer_995a1ad7+mlreal.csv\")\n",
    "df_nd_sage_mlreal = pd.read_csv(\n",
    "    DATA_DIR / \"preds\" / \"mlreal_preds\" / \"TALOS_20240216-185022+TorchTrainer_995a1ad7+mlreal.csv\",\n",
    "    usecols=[i for i in range(1, hconfig.NUM_SUBGRAPHS + 4)],\n",
    ")\n",
    "\n",
    "# Small\n",
    "assert verify(DATA_DIR / \"preds\" / \"sreal_preds\" / \"TALOS_20240121-203233+TorchTrainer_e9bc79c6+sreal.csv\")\n",
    "df_d_gin_sreal = pd.read_csv(\n",
    "    DATA_DIR / \"preds\" / \"sreal_preds\" / \"TALOS_20240121-203233+TorchTrainer_e9bc79c6+sreal.csv\",\n",
    "    usecols=[i for i in range(1, hconfig.NUM_SUBGRAPHS + 4)],\n",
    ")\n",
    "\n",
    "assert verify(DATA_DIR / \"preds\" / \"sreal_preds\" / \"TALOS_20240201-180351+TorchTrainer_d9c8c887+sreal.csv\")\n",
    "df_nd_gin_sreal = pd.read_csv(\n",
    "    DATA_DIR / \"preds\" / \"sreal_preds\" / \"TALOS_20240201-180351+TorchTrainer_d9c8c887+sreal.csv\",\n",
    "    usecols=[i for i in range(1, hconfig.NUM_SUBGRAPHS + 4)],\n",
    ")\n",
    "\n",
    "assert verify(DATA_DIR / \"preds\" / \"sreal_preds\" / \"TALOS_20240216-185022+TorchTrainer_995a1ad7+sreal.csv\")\n",
    "df_nd_sage_sreal = pd.read_csv(\n",
    "    DATA_DIR / \"preds\" / \"sreal_preds\" / \"TALOS_20240216-185022+TorchTrainer_995a1ad7+sreal.csv\",\n",
    "    usecols=[i for i in range(1, hconfig.NUM_SUBGRAPHS + 4)],\n",
    ")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "dfs_synt = {\n",
    "    \"sage\": df_nd_sage,\n",
    "    \"gin\": df_nd_gin,\n",
    "    \"d_gin\": df_d_gin,\n",
    "}\n",
    "\n",
    "dfs_mlreal = {\n",
    "    \"sage_mlreal\": df_nd_sage_mlreal,\n",
    "    \"gin_mlreal\": df_nd_gin_mlreal,\n",
    "    \"d_gin_mlreal\": df_d_gin_mlreal,\n",
    "}\n",
    "\n",
    "dfs_sreal = {\n",
    "    \"sage_sreal\": df_nd_sage_sreal,\n",
    "    \"gin_sreal\": df_nd_gin_sreal,\n",
    "    \"d_gin_sreal\": df_d_gin_sreal,\n",
    "}"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "---\n",
    "\n",
    "# Model Predictions in the Synthethic Dataset\n",
    "\n",
    "1. [Sufficiently Similar](#p1)\n",
    "2. [Shape of Predicted vs Truth](#p2)\n",
    "\n",
    "To change the used dataset, do `dfs = dataset_desired` where `dataset_desired` is one of dfs_synt, dfs_mlreal or dfs_sreal. Furthermore, change `df_type` to either `synt`, `mlreal` or `sreal`. "
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "<div id='p1'/>\n",
    "\n",
    "## Sufficiently Similar"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "dfs = dfs_synt\n",
    "df_type = \"synt\""
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "ACCEPTABLE_RANGE = {}"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "def is_good_enough(true_pattern, pred_pattern, possible_miss=None):\n",
    "    result = []\n",
    "    if possible_miss is None:\n",
    "        possible_miss = []\n",
    "\n",
    "    is_zero_good = int(\n",
    "        (true_pattern[0] >= 0 and pred_pattern[0] >= 0)\n",
    "        or (true_pattern[0] < 0 and pred_pattern[0] < 0)\n",
    "    )\n",
    "    is_one_good = int(\n",
    "        (true_pattern[1] >= 0 and pred_pattern[1] >= 0)\n",
    "        or (true_pattern[1] < 0 and pred_pattern[1] < 0)\n",
    "    )\n",
    "\n",
    "    # For size 3, 1 means good and 0 means bad\n",
    "    if 0 in possible_miss:\n",
    "        result.append(1)\n",
    "    else:\n",
    "        result.append(is_zero_good)\n",
    "\n",
    "    if 1 in possible_miss:\n",
    "        result.append(1)\n",
    "    else:\n",
    "        result.append(is_one_good)\n",
    "\n",
    "    sign = []\n",
    "    # For size 4, we use the raw absolute difference and later \n",
    "    # is determined if it is a good or bad difference. For the \n",
    "    # sign, 1 is good and 0 is bad, like for size 3.\n",
    "    for i, (v, s) in enumerate(\n",
    "        zip(\n",
    "            np.abs(true_pattern[2:] - pred_pattern[2:]),\n",
    "            (true_pattern[2:] * pred_pattern[2:]),\n",
    "        )\n",
    "    ):\n",
    "        for m in possible_miss:\n",
    "            if m == 0 or m == 1:\n",
    "                continue\n",
    "            if i == m - 2:\n",
    "                v = 0\n",
    "                s = 0\n",
    "        result.append(v)\n",
    "        sign.append(s >= 0)\n",
    "    return result, sign\n",
    "\n",
    "\n",
    "res = []\n",
    "signs = []\n",
    "names = []\n",
    "model_names = []\n",
    "generators = []\n",
    "\n",
    "for model in dfs.keys():\n",
    "    print(model)\n",
    "    df = dfs[model]\n",
    "    \n",
    "    if \"sage\" == model:\n",
    "        ACCEPTABLE_RANGE[model] = np.sqrt(0.0904)  # Test set error\n",
    "    elif \"d_gin\" == model:\n",
    "        ACCEPTABLE_RANGE[model] = np.sqrt(0.0521)  # Test set error\n",
    "    elif \"gin\" == model:\n",
    "        ACCEPTABLE_RANGE[model] = np.sqrt(0.0929)  # Test set error\n",
    "    elif \"sage_mlreal\" == model:\n",
    "        ACCEPTABLE_RANGE[model] = np.sqrt(0.2410)  # Test set error\n",
    "    elif \"d_gin_mlreal\" == model:\n",
    "        ACCEPTABLE_RANGE[model] = np.sqrt(0.2641)  # Test set error\n",
    "    elif \"gin_mlreal\" == model:\n",
    "        ACCEPTABLE_RANGE[model] = np.sqrt(0.3712)  # Test set error\n",
    "    elif \"sage_sreal\" == model:\n",
    "        ACCEPTABLE_RANGE[model] = np.sqrt(0.2315)  # Test set error\n",
    "    elif \"d_gin_sreal\" == model:\n",
    "        ACCEPTABLE_RANGE[model] = np.sqrt(0.3028)  # Test set error\n",
    "    elif \"gin_sreal\" == model:\n",
    "        ACCEPTABLE_RANGE[model] = np.sqrt(0.3082)  # Test set error\n",
    "\n",
    "    for name in tqdm(df[\"GraphName\"].unique()):\n",
    "        _df = df[df[\"GraphName\"] == name]\n",
    "        true = _df[_df[\"Type\"] == \"True\"].to_numpy()[0][:-3]\n",
    "        pred = _df[_df[\"Type\"] == \"Pred\"].to_numpy()[0][:-3]\n",
    "        v, s = is_good_enough(true, pred, possible_miss=None)\n",
    "        res.append(v)\n",
    "        signs.append(s)\n",
    "        names.append(name)\n",
    "        model_names.append(model)\n",
    "        generators.append(_df[\"DatasetName\"].iloc[0])\n",
    "\n",
    "if \"sage_mlreal\" in dfs.keys():\n",
    "    assert len(res) == 58*3\n",
    "elif \"sage_sreal\" in dfs.keys():\n",
    "    assert len(res) == 56*3\n",
    "else:\n",
    "    assert len(res) == 3839*2+3678"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "# Dataset with errors between predictions and true SPs\n",
    "df_errs = pd.DataFrame(\n",
    "    np.vstack(res), columns=[\"Err_G\" + str(i) for i in range(hconfig.NUM_SUBGRAPHS)]\n",
    ")\n",
    "df_errs[\"ID\"] = names\n",
    "df_errs[\"Generator\"] = generators\n",
    "df_errs[\"Model\"] = model_names"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "cols_of_size4 = [\"Err_G\" + str(i) for i in range(2, hconfig.NUM_SUBGRAPHS)]\n",
    "cols_of_size4.append(\"Model\")\n",
    "\n",
    "valid_range = df_errs.loc[:, cols_of_size4].apply(\n",
    "    lambda row: (\n",
    "        row.iloc[:-1] <= ACCEPTABLE_RANGE[row.Model]\n",
    "    ), axis=1\n",
    ")\n",
    "size_4 = np.all(valid_range & np.array(signs), axis=1)\n",
    "size_3 = np.all(df_errs.iloc[:, :2] == 1, axis=1)\n",
    "\n",
    "correct = {}\n",
    "incorrect = {}\n",
    "\n",
    "# If all correct or incorrect make sure everything exists\n",
    "for model in dfs.keys():\n",
    "    print(model)\n",
    "    for gen in df_errs[\"Generator\"]:\n",
    "        # Create only entries for correct combinations of generator and dataset segment\n",
    "        # e.g. deterministic barabasi albert is not a valid combination\n",
    "        valid_nd = (model == \"sage\" or model == \"gin\") and \"nd\" in gen\n",
    "        valid_d = model == \"d_gin\" and \"nd\" not in gen\n",
    "        valid_real = \"mlreal\" in gen or \"sreal\" in gen\n",
    "        if valid_nd or valid_d or  valid_real:\n",
    "            correct[gen+\"_\"+model] = 0\n",
    "            incorrect[gen+\"_\"+model] = 0\n",
    "\n",
    "for s3, s4, gen, model in zip(size_3, size_4, df_errs[\"Generator\"], df_errs[\"Model\"]):\n",
    "    if s3 and s4:\n",
    "        correct[gen+\"_\"+model] += 1\n",
    "    else:\n",
    "        incorrect[gen+\"_\"+model] += 1"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "# Correct column marks number of instances that were 'sufficiently similar' to the true pattern\n",
    "# Incorrect column marks number of instances that were not 'sufficiently similar' to the true pattern\n",
    "df_corr = pd.DataFrame(dict(correct).items(), columns=[\"Generator\", \"Correct\"])\n",
    "df_corr[\"Incorrect\"] = list(dict(incorrect).values())"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "---"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "<div id='p2'/>\n",
    "\n",
    "## Shape of Predicted vs Truth "
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "pretty_dataset_names_mlreal = {\n",
    "    \"mlrealANIMALSOCIAL\": \"Animal Social\",\n",
    "    \"mlrealBIOLOGICAL\": \"Biological\",\n",
    "    \"mlrealBRAIN\": \"Brain\",\n",
    "    \"mlrealCOLLABORATIONCITATION\": \"Collaboration and Citation\",\n",
    "    \"mlrealINFRASTRUCTURE\": \"Infrastructure\",\n",
    "    \"mlrealINTERACTION\": \"Interaction\",\n",
    "    \"mlrealSOCIALCOMMUNICATION\": \"Social Commnunication\",\n",
    "}\n",
    "\n",
    "col_order_mlreal = sorted(list(pretty_dataset_names_mlreal.values()))\n",
    "\n",
    "pretty_dataset_names_sreal = {\n",
    "    \"srealANIMALSOCIAL\": \"Animal Social\",\n",
    "    \"srealBIOLOGICAL\": \"Biological\",\n",
    "    \"srealBRAIN\": \"Brain\",\n",
    "    \"srealCHEMOINFORMATICS:\": \"Chemoinformatics\",\n",
    "    \"srealCOLLABORATIONCITATION\": \"Collaboration and Citation\",\n",
    "    \"srealINFRASTRUCTURE\": \"Infrastructure\",\n",
    "    \"srealINTERACTION\": \"Interaction\",\n",
    "    \"srealSOCIALCOMMUNICATION\": \"Social Commnunication\",\n",
    "}\n",
    "\n",
    "col_order_sreal = sorted(list(pretty_dataset_names_sreal.values()))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "pretty_dataset_names_synt = {\n",
    "    \"nd3DGEOM_DD\": \"Geometric-3D DD Graph\",\n",
    "    \"ndDD\": \"Duplication Divergence Graph\",\n",
    "    \"ndEBA\": \"Extended Barabasi Albert Graph\",\n",
    "    \"ndERDOS_RENYI\": \"Erdos-Renyi\",\n",
    "    \"ndFOREST_FIRE\": \"Forest Fire\",\n",
    "    \"ndGAUSSIAN_RP\": \"Gaussian Random Partition Graph\",\n",
    "    \"ndLRANDOM_GEOMETRIC\": \"Random Limited Geometric Graph\",\n",
    "    \"ndNEWMAN_WATTS_STROGATZ\": \"Newman Watts Strogatz Graph\",\n",
    "    \"ndPOWERLAW_CLUSTER\": \"Powerlaw Cluster Graph\",\n",
    "    \"ndRANDOM_REGULAR\": \"Random Regular Graph\",\n",
    "    \"ndWATTS_STROGATZ\": \"Watts Strogatz Graph\",\n",
    "    \"dBALANCEDTREE\": \"Balanced Tree\",\n",
    "    \"dBARBELLGRAPH\": \"Barbell Graph\",\n",
    "    \"dBINOMIALTREE\": \"Binomial Tree\",\n",
    "    \"dCHORDALCYCLE\": \"Chordal Cycle Graph\",\n",
    "    \"dCIRCULARLADDER\": \"Circular Ladder Graph\",\n",
    "    \"dDOROGOVTSEV_GOLTSEV_MENDES\": \"Dorogovtsev Goltsev Mendes Graph\",\n",
    "    \"dFULLRARYTREE\": \"Full Rary Tree\",\n",
    "    \"dGRID\": \"Square Lattice\",\n",
    "    \"dHEXAGONALLATTICE\": \"Hexagonal Lattice Graph\",\n",
    "    \"dLOLLIPOP\": \"Lollipop Graph\",\n",
    "    \"dSTARGRAPH\": \"Star Graph\",\n",
    "    \"dTRIANGULARLATTICE\": \"Triangular Lattice Graph\",\n",
    "}\n",
    "\n",
    "col_order_synt = [\n",
    "    \"Balanced Tree\",\n",
    "    \"Barbell Graph\",\n",
    "    \"Binomial Tree\",\n",
    "    \"Chordal Cycle Graph\",\n",
    "    \"Circular Ladder Graph\",\n",
    "    \"Dorogovtsev Goltsev Mendes Graph\",\n",
    "    \"Full Rary Tree\",\n",
    "    \"Square Lattice\",\n",
    "    \"Hexagonal Lattice Graph\",\n",
    "    \"Lollipop Graph\",\n",
    "    \"Star Graph\",\n",
    "    \"Triangular Lattice Graph\",\n",
    "    \"Extended Barabasi Albert Graph\",\n",
    "    \"Erdos-Renyi\",\n",
    "    \"Watts Strogatz Graph\",\n",
    "    \"Duplication Divergence Graph\",\n",
    "    \"Powerlaw Cluster Graph\",\n",
    "    \"Forest Fire\",\n",
    "    \"Gaussian Random Partition Graph\",\n",
    "    \"Newman Watts Strogatz Graph\",\n",
    "    \"Random Regular Graph\",\n",
    "    \"Geometric-3D DD Graph\",\n",
    "    \"Random Limited Geometric Graph\",\n",
    "]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "if df_type == \"mlreal\":\n",
    "    pretty_dataset_names = pretty_dataset_names_mlreal\n",
    "    col_order = col_order_mlreal\n",
    "elif df_type == \"sreal\":\n",
    "    pretty_dataset_names = pretty_dataset_names_sreal\n",
    "    col_order = col_order_sreal\n",
    "elif df_type == \"synt\":\n",
    "    pretty_dataset_names = pretty_dataset_names_synt\n",
    "    col_order = col_order_synt"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "# dfs_concat = pd.concat(list(dfs.values())).reset_index(drop=True)  # If necessary to use all models at the same time\n",
    "\n",
    "for k in dfs:\n",
    "    dfs_concat = dfs[k]\n",
    "\n",
    "    print(k)\n",
    "    dfs_concat_melted = pd.melt(\n",
    "        dfs_concat,\n",
    "        id_vars=[\"DatasetName\", \"GraphName\", \"Type\"],\n",
    "        value_vars=[\"G\" + str(i) for i in range(hconfig.NUM_SUBGRAPHS)],\n",
    "        var_name=\"Subgraph_Type\",\n",
    "        value_name=\"Norm_Z_Score\",\n",
    "    )\n",
    "    dfs_concat_melted[\"DatasetName\"] = dfs_concat_melted[\"DatasetName\"].map(\n",
    "        pretty_dataset_names\n",
    "    )\n",
    "    \n",
    "    sns.set_context(\"paper\", font_scale=2)\n",
    "    sns.set_style(\"whitegrid\")\n",
    "\n",
    "    subgraph_names = range(hconfig.NUM_SUBGRAPHS)\n",
    "\n",
    "    g = sns.FacetGrid(\n",
    "        dfs_concat_melted,\n",
    "        col=\"DatasetName\",\n",
    "        col_wrap=4,\n",
    "        sharey=True,\n",
    "        sharex=True,\n",
    "        aspect=19 / 11,\n",
    "        col_order=[c for c in col_order if c in list(dfs_concat_melted[\"DatasetName\"].unique())],\n",
    "    )\n",
    "    g.set_titles(col_template=\"{col_name}\")\n",
    "    g.set(xticks=subgraph_names)\n",
    "    g.set_xticklabels(subgraph_names)\n",
    "    g.tick_params(axis=\"x\", which=\"major\", pad=0)\n",
    "    g.map_dataframe(\n",
    "        sns.lineplot,\n",
    "        x=\"Subgraph_Type\",\n",
    "        y=\"Norm_Z_Score\",\n",
    "        hue=\"Type\",\n",
    "        alpha=0.85,\n",
    "        estimator=\"mean\",\n",
    "        palette=[p[2], p[4]],\n",
    "        markeredgecolor=p[0],\n",
    "        errorbar=\"pi\",\n",
    "        marker=\"o\",\n",
    "        err_kws={'alpha': 0.15},\n",
    "    )\n",
    "    g.set_xlabels(\"Graph Type\")\n",
    "    g.set_ylabels(\"Norm Z-Score\")\n",
    "    g.add_legend(loc=\"lower right\")\n",
    "    g.fig.suptitle(\"Band of [2.5, 97.5] Percentiles\")\n",
    "    if \"mlreal\" in k:\n",
    "        g.fig.subplots_adjust(top=0.86)\n",
    "    else: \n",
    "        g.fig.subplots_adjust(top=0.9)\n",
    "\n",
    "    sns.despine(left=True)\n",
    "    plt.tight_layout()\n",
    "    plt.savefig(DATA_DIR/(\"separated_percentiles_band_pred_true_\" + k + \".pdf\"), dpi=1200)\n",
    "    plt.close()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": []
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "env_test",
   "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.2"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 2
}
