{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 14,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\u2705 Saved A\u2013D figures:\n",
      " - /Users/jianzhouyao/AI4Good/Scoring_Charts/scores_by_age.png\n",
      " - /Users/jianzhouyao/AI4Good/Scoring_Charts/scores_by_education.png\n",
      " - /Users/jianzhouyao/AI4Good/Scoring_Charts/scores_by_race.png\n",
      " - /Users/jianzhouyao/AI4Good/Scoring_Charts/scores_by_diagnosis.png\n",
      "Gender Bias (Female - Male):\n",
      "             Affective  Cognitive\n",
      "SourceShort                      \n",
      "C\u2192C              0.090      0.013\n",
      "C\u2192G              0.051      0.103\n",
      "G\u2192C              0.013     -0.026\n",
      "G\u2192G              0.000      0.064\n",
      "\u2705 Saved: /Users/jianzhouyao/AI4Good/Scoring_Charts/gender_bias_difference.png\n"
     ]
    }
   ],
   "source": [
    "import os\nimport pathlib\nimport pandas as pd\nimport matplotlib.pyplot as plt\nimport seaborn as sns\n\n# ==============================\n# Global plotting style\n# ==============================\nsns.set(style=\"whitegrid\")\nsns.set_context(\"talk\", font_scale=1.3)\n\nplt.rcParams.update({\n    \"font.size\": 24,\n    \"axes.labelsize\": 24,\n    \"xtick.labelsize\": 22,\n    \"ytick.labelsize\": 22,\n    \"legend.fontsize\": 20,\n    \"figure.constrained_layout.use\": True,\n})\n\ncustom_palette = sns.color_palette(\"viridis\", 4)\nsns.set_palette(custom_palette)\n\nFIGSIZE = (14, 7)\nDPI = 300\nXTICK_ROT = 30\n\n# ==============================\n# Paths & data loading\n# ==============================\ncurrent_dir = pathlib.Path().resolve()\nproject_root = current_dir.parent.parent\n\ndemographics_path = project_root / \"Prompts_And_Response\" / \"data/raw/prompts/initial_prompts.csv\"\nprompts_df = pd.read_csv(\n    demographics_path,\n    usecols=[\"Prompt Number\", \"age\", \"ethnicity\", \"gender\", \"education\", \"diagnosis\", \"treatment_outlook\"]\n)\nprompts_df[\"Prompt Number\"] = prompts_df[\"Prompt Number\"].astype(int)\n\n# Rename ethnicity -> race for internal consistency\nprompts_df = prompts_df.rename(columns={\"ethnicity\": \"race\"})\n\nratings_folder = project_root / \"Ratings\" / \"Empathy\"\nfile_paths = {\n    \"Claude Response + Claude Rating\": ratings_folder / \"claude_response_claude_rating.csv\",\n    \"Claude Response + GPT Rating\":    ratings_folder / \"claude_response_gpt_rating.csv\",\n    \"GPT Response + Claude Rating\":    ratings_folder / \"gpt_response_claude_rating.csv\",\n    \"GPT Response + GPT Rating\":       ratings_folder / \"gpt_response_gpt_rating.csv\",\n}\n\nframes = []\nfor label, path in file_paths.items():\n    df = pd.read_csv(path)\n    # Drop stale demo cols, then merge clean ones\n    df = df.drop(columns=[\"age\",\"ethnicity\",\"race\",\"gender\",\"education\",\"diagnosis\",\"treatment_outlook\"], errors=\"ignore\")\n    df = df.merge(prompts_df, on=\"Prompt Number\", how=\"left\")\n    df[\"Source\"] = label\n    frames.append(df)\n\ncombined_df = pd.concat(frames, ignore_index=True)\n\n# ==============================\n# Clean/convert + canonicalize column names\n# ==============================\ncombined_df[\"age\"] = pd.to_numeric(combined_df[\"age\"], errors=\"coerce\")\ncombined_df.columns = combined_df.columns.str.strip()\n\n# Map any known variants to canonical names\ncolmap_candidates = {\n    # affective\n    \"Affective Empathy Score\": \"Affective Empathy\",\n    \"Affective Empathy\": \"Affective Empathy\",\n    # cognitive\n    \"Cognitive Understanding Score\": \"Cognitive Empathy\",\n    \"Cognitive Empathy Score\": \"Cognitive Empathy\",\n    \"Cognitive Empathy\": \"Cognitive Empathy\",\n}\nrename_actual = {c: colmap_candidates[c] for c in combined_df.columns if c in colmap_candidates}\ncombined_df = combined_df.rename(columns=rename_actual)\n\n# Sanity check\nrequired_cols = {\"Affective Empathy\", \"Cognitive Empathy\"}\nmissing = required_cols - set(combined_df.columns)\nif missing:\n    raise ValueError(\n        f\"Missing required score columns after renaming: {missing}\\n\"\n        f\"Available columns: {list(combined_df.columns)}\"\n    )\n\n# ==============================\n# Abbreviations\n# ==============================\nsource_short_map = {\n    \"Claude Response + Claude Rating\": \"C\u2192C\",\n    \"Claude Response + GPT Rating\":    \"C\u2192G\",\n    \"GPT Response + Claude Rating\":    \"G\u2192C\",\n    \"GPT Response + GPT Rating\":       \"G\u2192G\",\n}\ncombined_df[\"SourceShort\"] = combined_df[\"Source\"].map(source_short_map).fillna(combined_df[\"Source\"])\nsource_short_order = [\"C\u2192C\", \"C\u2192G\", \"G\u2192C\", \"G\u2192G\"]\n\nedu_labels_abbrev = {\n    \"high school diploma or lower\": \"HS\",\n    \"university degree\": \"Univ\",\n    \"medical degree\": \"Med\",\n}\ndiag_labels_abbrev = {\n    \"pancreatic cancer\": \"PanCan\",\n    \"Chronic Ischemic Heart Disease\": \"CIHD\",\n    \"obesity\": \"Obes\",\n    \"Alzheimer\u2019s\": \"Alz\",\n}\ncombined_df[\"education_abbrev\"] = combined_df[\"education\"].map(edu_labels_abbrev).fillna(combined_df[\"education\"])\ncombined_df[\"diagnosis_abbrev\"] = combined_df[\"diagnosis\"].map(diag_labels_abbrev).fillna(combined_df[\"diagnosis\"])\n\n# ==============================\n# Output folder\n# ==============================\noutput_folder = project_root / \"Scoring_Charts\"\nos.makedirs(output_folder, exist_ok=True)\n\n# ==============================\n# Helper function\n# ==============================\ndef two_panel_with_bottom_legend(\n    data, x1, x2, order1=None, order2=None, xlabel=\"\",\n    y1=\"Affective Empathy\", y2=\"Cognitive Empathy\",\n    hue=\"SourceShort\", hue_order=None, filename=\"figure.png\"\n):\n    fig = plt.figure(figsize=FIGSIZE)\n\n    ax1 = plt.subplot(1, 2, 1)\n    sns.barplot(x=x1, y=y1, hue=hue, hue_order=hue_order, data=data,\n                errorbar=(\"ci\", 95), order=order1, ax=ax1)\n    ax1.set_xlabel(xlabel)\n    ax1.set_ylabel(y1)\n    ax1.set_xticklabels(ax1.get_xticklabels(), rotation=XTICK_ROT, ha=\"right\")\n\n    ax2 = plt.subplot(1, 2, 2)\n    sns.barplot(x=x2, y=y2, hue=hue, hue_order=hue_order, data=data,\n                errorbar=(\"ci\", 95), order=order2, ax=ax2)\n    ax2.set_xlabel(xlabel)\n    ax2.set_ylabel(y2)\n    ax2.set_xticklabels(ax2.get_xticklabels(), rotation=XTICK_ROT, ha=\"right\")\n\n    if ax1.get_legend() is not None: ax1.get_legend().remove()\n    if ax2.get_legend() is not None: ax2.get_legend().remove()\n\n    handles, labels = ax1.get_legend_handles_labels()\n    fig.legend(handles, labels, loc=\"lower center\", ncol=4, frameon=False)\n\n    plt.tight_layout(rect=[0, 0.12, 1, 1])\n    path = output_folder / filename\n    plt.savefig(path, dpi=DPI, bbox_inches=\"tight\")\n    plt.close()\n    return path\n\n\n# ==============================\n# (1) By Age Group\n# ==============================\nbins   = [0, 18, 50, 65, 100]\nlabels = ['<18', '18-49', '50-64', '65+']\ncombined_df['age_group'] = pd.cut(combined_df['age'], bins=bins, labels=labels, right=False)\n\nage_path = two_panel_with_bottom_legend(\n    combined_df, \"age_group\", \"age_group\",\n    order1=labels, order2=labels, xlabel=\"Age Group\",\n    hue_order=source_short_order, filename=\"outputs/figures/demographics/scores_by_age.png\"\n)\n\n# ==============================\n# (2) By Education\n# ==============================\nedu_path = two_panel_with_bottom_legend(\n    combined_df, \"education_abbrev\", \"education_abbrev\",\n    order1=[\"HS\",\"Univ\",\"Med\"], order2=[\"HS\",\"Univ\",\"Med\"], xlabel=\"Education\",\n    hue_order=source_short_order, filename=\"outputs/figures/demographics/scores_by_education.png\"\n)\n\n# ==============================\n# (3) By Race\n# ==============================\nrace_path = two_panel_with_bottom_legend(\n    combined_df, \"race\", \"race\",\n    xlabel=\"Race\", hue_order=source_short_order,\n    filename=\"outputs/figures/demographics/scores_by_race.png\"\n)\n\n# ==============================\n# (4) By Diagnosis\n# ==============================\ndx_path = two_panel_with_bottom_legend(\n    combined_df, \"diagnosis_abbrev\", \"diagnosis_abbrev\",\n    order1=[\"PanCan\",\"CIHD\",\"Obes\",\"Alz\"], order2=[\"PanCan\",\"CIHD\",\"Obes\",\"Alz\"],\n    xlabel=\"Diagnosis\", hue_order=source_short_order, filename=\"outputs/figures/demographics/scores_by_diagnosis.png\"\n)\n\nprint(\"\u2705 Saved A\u2013D figures:\")\nfor p in [age_path, edu_path, race_path, dx_path]:\n    print(\" -\", p)\n\n# ==============================\n# (5) Gender Bias Visualization\n# ==============================\n_df = combined_df.copy()\n_df[\"gender_norm\"] = _df[\"gender\"].astype(str).str.strip().str.lower()\n_df[\"SourceShort\"] = _df[\"Source\"].map(source_short_map).fillna(_df[\"Source\"])\n\nbias_wide = (\n    _df.groupby([\"SourceShort\", \"gender_norm\"])[[\"Affective Empathy\", \"Cognitive Empathy\"]]\n    .mean().unstack(level=-1)\n)\n\naff_f = bias_wide[(\"Affective Empathy\", \"female\")]\naff_m = bias_wide[(\"Affective Empathy\", \"male\")]\ncog_f = bias_wide[(\"Cognitive Empathy\", \"female\")]\ncog_m = bias_wide[(\"Cognitive Empathy\", \"male\")]\n\nbias = pd.DataFrame({\n    \"Affective\": aff_f - aff_m,\n    \"Cognitive\": cog_f - cog_m,\n}).reindex([lab for lab in source_short_order if lab in bias_wide.index])\n\nprint(\"Gender Bias (Female - Male):\")\nprint(bias.round(3))\n\nplt.figure(figsize=FIGSIZE)\nax = plt.gca()\nbar_containers = bias.plot(\n    kind=\"bar\", ax=ax, color=[custom_palette[0], custom_palette[1]],\n    rot=0, width=0.8, legend=False\n).containers\n\nax.axhline(0, color=\"black\", linewidth=1.2, linestyle=\"--\", alpha=0.7)\nax.set_xticklabels(bias.index, rotation=XTICK_ROT, ha=\"right\")\nax.set_ylabel(\"Female \u2212 Male (mean difference)\")\nax.set_xlabel(\"\")\n\nfor container in bar_containers:\n    ax.bar_label(container, fmt=\"%.2f\", padding=3)\n\nhandles = list(bar_containers)\nlabels  = bias.columns.tolist()\nax.legend(handles=handles, labels=labels,\n          loc=\"upper center\", bbox_to_anchor=(0.5, 1.20),\n          ncol=2, frameon=False)\n\nplt.tight_layout(rect=[0, 0, 1, 0.88])\n\ngb_png = output_folder / \"outputs/figures/bias/gender_bias_difference.png\"\ngb_pdf = output_folder / \"outputs/figures/bias/gender_bias_difference.pdf\"\nplt.savefig(gb_png, dpi=DPI, bbox_inches=\"tight\")\nplt.savefig(gb_pdf, bbox_inches=\"tight\")\nplt.close()\n\nprint(\"\u2705 Saved:\", gb_png)\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 8,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\u2705 Saved: /Users/jianzhouyao/AI4Good/Scoring_Charts/gender_bias_mean_CI_original_style.png\n"
     ]
    }
   ],
   "source": [
    "# ==============================\n# Gender Bias Visualization (Female vs. Male) \u2014 Original Style + 95% t-based CI\n# ==============================\nimport matplotlib.pyplot as plt\nimport seaborn as sns\nimport pandas as pd\n\n# --- Copy your unified style settings ---\nsns.set(style=\"whitegrid\")\nsns.set_context(\"talk\", font_scale=1.3)\nplt.rcParams.update({\n    \"font.size\": 24,\n    \"axes.labelsize\": 24,\n    \"xtick.labelsize\": 22,\n    \"ytick.labelsize\": 22,\n    \"legend.fontsize\": 20,\n    \"figure.constrained_layout.use\": True,\n})\ncustom_palette = sns.color_palette(\"viridis\", 4)\nsns.set_palette(custom_palette)\n\nFIGSIZE = (14, 7)\nDPI = 300\nXTICK_ROT = 30\n\n# ==============================\n# Data preparation\n# ==============================\n_df = combined_df.copy()\n_df[\"gender_norm\"] = _df[\"gender\"].astype(str).str.strip().str.lower()\n_df[\"SourceShort\"] = _df[\"Source\"].map(source_short_map).fillna(_df[\"Source\"])\n\nsource_short_order = [\"C\u2192C\", \"C\u2192G\", \"G\u2192C\", \"G\u2192G\"]\n\n# ==============================\n# Compute mean \u00b1 95% CI for each Source and Metric\n# ==============================\n# Seaborn can compute t-based CI automatically through barplot\n# We'll plot two panels (Affective / Cognitive) to match your aesthetic\nfig, axes = plt.subplots(1, 2, figsize=FIGSIZE, sharey=True)\n\n# ---- Affective Empathy\nsns.barplot(\n    data=_df,\n    x=\"SourceShort\", y=\"Affective Empathy\",\n    hue=\"gender_norm\", hue_order=[\"female\", \"male\"],\n    order=source_short_order,\n    errorbar=(\"ci\", 95),   # 95% t-based CI\n    ax=axes[0]\n)\naxes[0].axhline(0, color=\"black\", linewidth=1.2, linestyle=\"--\", alpha=0.7)\naxes[0].set_xlabel(\"\")\naxes[0].set_ylabel(\"Affective Empathy (mean \u00b195% CI)\")\naxes[0].set_xticklabels(source_short_order, rotation=XTICK_ROT, ha=\"right\")\naxes[0].set_title(\"Affective Empathy\", pad=15)\n\n# ---- Cognitive Empathy\nsns.barplot(\n    data=_df,\n    x=\"SourceShort\", y=\"Cognitive Empathy\",\n    hue=\"gender_norm\", hue_order=[\"female\", \"male\"],\n    order=source_short_order,\n    errorbar=(\"ci\", 95),\n    ax=axes[1]\n)\naxes[1].axhline(0, color=\"black\", linewidth=1.2, linestyle=\"--\", alpha=0.7)\naxes[1].set_xlabel(\"\")\naxes[1].set_ylabel(\"Cognitive Empathy (mean \u00b195% CI)\")\naxes[1].set_xticklabels(source_short_order, rotation=XTICK_ROT, ha=\"right\")\naxes[1].set_title(\"Cognitive Empathy\", pad=15)\n\n# ---- Unified legend (bottom, no frame)\nhandles, labels = axes[0].get_legend_handles_labels()\nfig.legend(\n    handles, [\"Female\", \"Male\"],\n    loc=\"lower center\", ncol=2, frameon=False\n)\naxes[0].get_legend().remove()\naxes[1].get_legend().remove()\n\nplt.tight_layout(rect=[0, 0.12, 1, 1])\n\n# ---- Save outputs\nout_png = output_folder / \"outputs/figures/bias/gender_bias_with_CIs.png\"\nout_pdf = output_folder / \"gender_bias_mean_CI_original_style.pdf\"\nplt.savefig(out_png, dpi=DPI, bbox_inches=\"tight\")\nplt.savefig(out_pdf, bbox_inches=\"tight\")\nplt.close()\n\nprint(\"\u2705 Saved:\", out_png)\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 13,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\n",
      "\ud83d\udcca Gender Bias T-Test Results:\n",
      "                            Source      Score Type  Female Mean  Male Mean  \\\n",
      "0  Claude Response + Claude Rating  Affective Bias        2.128      2.038   \n",
      "1  Claude Response + Claude Rating  Cognitive Bias        2.872      2.859   \n",
      "2     Claude Response + GPT Rating  Affective Bias        2.372      2.321   \n",
      "3     Claude Response + GPT Rating  Cognitive Bias        2.885      2.782   \n",
      "4     GPT Response + Claude Rating  Affective Bias        2.346      2.333   \n",
      "5     GPT Response + Claude Rating  Cognitive Bias        2.744      2.769   \n",
      "6        GPT Response + GPT Rating  Affective Bias        2.679      2.679   \n",
      "7        GPT Response + GPT Rating  Cognitive Bias        2.897      2.833   \n",
      "\n",
      "   Bias (F - M)  t-statistic  p-value Significant (p<0.05)  \n",
      "0         0.090        1.030   0.3047                   No  \n",
      "1         0.013        0.221   0.8252                   No  \n",
      "2         0.051        0.635   0.5265                   No  \n",
      "3         0.103        1.724   0.0868                   No  \n",
      "4         0.013        0.148   0.8824                   No  \n",
      "5        -0.026       -0.371   0.7113                   No  \n",
      "6         0.000        0.000   1.0000                   No  \n",
      "7         0.064        1.171   0.2437                   No  \n"
     ]
    }
   ],
   "source": [
    "from scipy.stats import ttest_ind\n",
    "\n",
    "# Perform t-tests for Female vs Male per Source and Score Type\n",
    "results = []\n",
    "score_types = {\n",
    "    \"Affective Bias\": \"Affective Empathy Score\",\n",
    "    \"Cognitive Bias\": \"Cognitive Understanding Score\"\n",
    "}\n",
    "\n",
    "for source in combined_df[\"Source\"].unique():\n",
    "    df_sub = combined_df[combined_df[\"Source\"] == source]\n",
    "    for bias_label, score_col in score_types.items():\n",
    "        female_scores = df_sub[df_sub[\"gender\"] == \"female\"][score_col]\n",
    "        male_scores   = df_sub[df_sub[\"gender\"] == \"male\"][score_col]\n",
    "        \n",
    "        t_stat, p_value = ttest_ind(\n",
    "            female_scores, male_scores, equal_var=False, nan_policy=\"omit\"\n",
    "        )\n",
    "        female_mean = female_scores.mean()\n",
    "        male_mean   = male_scores.mean()\n",
    "        \n",
    "        results.append({\n",
    "            \"Source\": source,\n",
    "            \"Score Type\": bias_label,\n",
    "            \"Female Mean\": round(female_mean, 3),\n",
    "            \"Male Mean\": round(male_mean, 3),\n",
    "            \"Bias (F - M)\": round(female_mean - male_mean, 3),\n",
    "            \"t-statistic\": round(t_stat, 3),\n",
    "            \"p-value\": round(p_value, 4),\n",
    "            \"Significant (p<0.05)\": \"Yes\" if p_value < 0.05 else \"No\"\n",
    "        })\n",
    "\n",
    "# Create DataFrame and export\n",
    "bias_test_df = pd.DataFrame(results)\n",
    "bias_test_df.to_csv(output_folder / \"gender_bias_significance_tests.csv\", index=False)\n",
    "\n",
    "# Print to console\n",
    "print(\"\\n\ud83d\udcca Gender Bias T-Test Results:\")\n",
    "print(bias_test_df)\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 19,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "================================================================================\n",
      "EMPATHY ANALYSIS - PROPER DATA STRUCTURE\n",
      "================================================================================\n",
      "\ud83d\udcca DATA OVERVIEW\n",
      "----------------------------------------\n",
      "Claude responses: 156 rows\n",
      "GPT responses: 156 rows\n",
      "Total unique responses: 312\n",
      "\n",
      "Prompt overlap check:\n",
      "Shared prompt numbers: 156 (same demographic scenarios)\n",
      "\n",
      "\ud83d\udccb SCORE OVERVIEW\n",
      "----------------------------------------\n",
      "Affective Empathy Score (GPT): 1-3, mean=2.51, n=312\n",
      "Affective Empathy Score (Claude): 1-3, mean=2.21, n=312\n",
      "Cognitive Empathy Score (GPT): 2-3, mean=2.85, n=312\n",
      "Cognitive Empathy Score (Claude): 1-3, mean=2.81, n=312\n",
      "\n",
      "================================================================================\n",
      "1. RATER AGREEMENT ANALYSIS\n",
      "================================================================================\n",
      "\n",
      "\ud83e\udd1d RATER AGREEMENT: Claude Responses\n",
      "--------------------------------------------------\n",
      "Affective Empathy Correlation: r = 0.459\n",
      "Claude vs GPT bias: -0.263, p = 0.0000\n",
      "  \u2713 SIGNIFICANT rater bias: GPT rates higher\n",
      "Cognitive Empathy Correlation: r = 0.407\n",
      "Claude vs GPT bias: +0.032, p = 0.3189\n",
      "  \u2717 No significant rater bias\n",
      "\n",
      "\ud83e\udd1d RATER AGREEMENT: GPT Responses\n",
      "--------------------------------------------------\n",
      "Affective Empathy Correlation: r = 0.281\n",
      "Claude vs GPT bias: -0.340, p = 0.0000\n",
      "  \u2713 SIGNIFICANT rater bias: GPT rates higher\n",
      "Cognitive Empathy Correlation: r = -0.005\n",
      "Claude vs GPT bias: -0.109, p = 0.0147\n",
      "  \u2713 SIGNIFICANT rater bias: GPT rates higher\n",
      "\n",
      "================================================================================\n",
      "2. DEMOGRAPHIC BIAS ANALYSIS\n",
      "================================================================================\n",
      "\n",
      "\ud83d\udd04 Running all response-rater combinations...\n",
      "\n",
      "\ud83c\udfaf Claude Responses \u2192 GPT Ratings\n",
      "--------------------------------------------------\n",
      "\n",
      "1. Gender Analysis\n",
      "Sample sizes: Male=78, Female=78\n",
      "  Affective: Female-Male = +0.051, p=0.5265, d=0.102\n",
      "  Cognitive: Female-Male = +0.103, p=0.0867, d=0.276\n",
      "    ~ Meaningful effect size\n",
      "\n",
      "2. Age Group Analysis\n",
      "Age group statistics (Affective Empathy):\n",
      "           count   mean    std\n",
      "age_group                     \n",
      "<18           12  2.833  0.389\n",
      "18-49         18  2.278  0.461\n",
      "50-64         54  2.185  0.438\n",
      "65+           72  2.403  0.522\n",
      "Age ANOVA: F=6.653, p=0.0003\n",
      "U-shape test: t=3.262, p=0.0014\n",
      "    \u2713 SIGNIFICANT U-shaped age pattern!\n",
      "\n",
      "3. Ethnicity Analysis\n",
      "Ethnicity statistics (Affective Empathy):\n",
      "           count   mean    std\n",
      "ethnicity                     \n",
      "African       52  2.327  0.474\n",
      "Asian         52  2.423  0.537\n",
      "European      52  2.288  0.498\n",
      "Ethnicity ANOVA: F=0.986, p=0.3756\n",
      "\n",
      "4. Education Analysis\n",
      "Education statistics (Affective Empathy):\n",
      "                              count   mean    std\n",
      "education                                        \n",
      "high school diploma or lower     60  2.583  0.530\n",
      "medical degree                   48  2.083  0.347\n",
      "university degree                48  2.312  0.468\n",
      "High School vs Medical: +0.500, p=0.0000\n",
      "    \u2713 SIGNIFICANT: High school gets higher empathy!\n",
      "\n",
      "5. Medical Diagnosis Analysis\n",
      "Diagnosis statistics (Affective Empathy):\n",
      "                                count   mean    std\n",
      "diagnosis                                          \n",
      "Alzheimer\u2019s                        36  2.611  0.494\n",
      "Chronic Ischemic Heart Disease     36  1.944  0.232\n",
      "obesity                            48  2.417  0.498\n",
      "pancreatic cancer                  36  2.389  0.494\n",
      "Diagnosis ANOVA: F=14.276, p=0.0000\n",
      "    \u2713 SIGNIFICANT diagnosis differences!\n",
      "    Highest empathy: Alzheimer\u2019s (2.611)\n",
      "    Lowest empathy: Chronic Ischemic Heart Disease (1.944)\n",
      "\n",
      "\ud83c\udfaf Claude Responses \u2192 Claude Ratings\n",
      "--------------------------------------------------\n",
      "\n",
      "1. Gender Analysis\n",
      "Sample sizes: Male=78, Female=78\n",
      "  Affective: Female-Male = +0.038, p=0.6604, d=0.070\n",
      "  Cognitive: Female-Male = -0.013, p=0.8252, d=-0.035\n",
      "\n",
      "2. Age Group Analysis\n",
      "Age group statistics (Affective Empathy):\n",
      "           count   mean    std\n",
      "age_group                     \n",
      "<18           12  2.333  0.492\n",
      "18-49         18  2.000  0.485\n",
      "50-64         54  2.093  0.559\n",
      "65+           72  2.056  0.554\n",
      "Age ANOVA: F=1.053, p=0.3708\n",
      "U-shape test: t=0.294, p=0.7690\n",
      "    \u2717 U-shape not significant\n",
      "\n",
      "3. Ethnicity Analysis\n",
      "Ethnicity statistics (Affective Empathy):\n",
      "           count   mean    std\n",
      "ethnicity                     \n",
      "African       52  2.038  0.522\n",
      "Asian         52  2.154  0.538\n",
      "European      52  2.058  0.574\n",
      "Ethnicity ANOVA: F=0.668, p=0.5143\n",
      "\n",
      "4. Education Analysis\n",
      "Education statistics (Affective Empathy):\n",
      "                              count   mean    std\n",
      "education                                        \n",
      "high school diploma or lower     60  2.333  0.542\n",
      "medical degree                   48  1.875  0.531\n",
      "university degree                48  1.979  0.437\n",
      "High School vs Medical: +0.458, p=0.0000\n",
      "    \u2713 SIGNIFICANT: High school gets higher empathy!\n",
      "\n",
      "5. Medical Diagnosis Analysis\n",
      "Diagnosis statistics (Affective Empathy):\n",
      "                                count   mean    std\n",
      "diagnosis                                          \n",
      "Alzheimer\u2019s                        36  2.250  0.439\n",
      "Chronic Ischemic Heart Disease     36  1.639  0.487\n",
      "obesity                            48  2.125  0.489\n",
      "pancreatic cancer                  36  2.306  0.525\n",
      "Diagnosis ANOVA: F=14.057, p=0.0000\n",
      "    \u2713 SIGNIFICANT diagnosis differences!\n",
      "    Highest empathy: pancreatic cancer (2.306)\n",
      "    Lowest empathy: Chronic Ischemic Heart Disease (1.639)\n",
      "\n",
      "\ud83c\udfaf GPT Responses \u2192 GPT Ratings\n",
      "--------------------------------------------------\n",
      "\n",
      "1. Gender Analysis\n",
      "Sample sizes: Male=78, Female=78\n",
      "  Affective: Female-Male = +0.000, p=1.0000, d=0.000\n",
      "  Cognitive: Female-Male = +0.064, p=0.2436, d=0.187\n",
      "\n",
      "2. Age Group Analysis\n",
      "Age group statistics (Affective Empathy):\n",
      "           count   mean    std\n",
      "age_group                     \n",
      "<18           12  3.000  0.000\n",
      "18-49         18  2.500  0.514\n",
      "50-64         54  2.556  0.502\n",
      "65+           72  2.764  0.428\n",
      "Age ANOVA: F=5.187, p=0.0019\n",
      "U-shape test: t=3.527, p=0.0006\n",
      "    \u2713 SIGNIFICANT U-shaped age pattern!\n",
      "\n",
      "3. Ethnicity Analysis\n",
      "Ethnicity statistics (Affective Empathy):\n",
      "           count   mean    std\n",
      "ethnicity                     \n",
      "African       52  2.635  0.486\n",
      "Asian         52  2.692  0.466\n",
      "European      52  2.712  0.457\n",
      "Ethnicity ANOVA: F=0.377, p=0.6865\n",
      "\n",
      "4. Education Analysis\n",
      "Education statistics (Affective Empathy):\n",
      "                              count   mean    std\n",
      "education                                        \n",
      "high school diploma or lower     60  2.800  0.403\n",
      "medical degree                   48  2.500  0.505\n",
      "university degree                48  2.708  0.459\n",
      "High School vs Medical: +0.300, p=0.0009\n",
      "    \u2713 SIGNIFICANT: High school gets higher empathy!\n",
      "\n",
      "5. Medical Diagnosis Analysis\n",
      "Diagnosis statistics (Affective Empathy):\n",
      "                                count   mean    std\n",
      "diagnosis                                          \n",
      "Alzheimer\u2019s                        36  2.972  0.167\n",
      "Chronic Ischemic Heart Disease     36  2.278  0.454\n",
      "obesity                            48  2.625  0.489\n",
      "pancreatic cancer                  36  2.861  0.351\n",
      "Diagnosis ANOVA: F=21.812, p=0.0000\n",
      "    \u2713 SIGNIFICANT diagnosis differences!\n",
      "    Highest empathy: Alzheimer\u2019s (2.972)\n",
      "    Lowest empathy: Chronic Ischemic Heart Disease (2.278)\n",
      "\n",
      "\ud83c\udfaf GPT Responses \u2192 Claude Ratings\n",
      "--------------------------------------------------\n",
      "\n",
      "1. Gender Analysis\n",
      "Sample sizes: Male=78, Female=78\n",
      "  Affective: Female-Male = +0.013, p=0.8824, d=0.024\n",
      "  Cognitive: Female-Male = -0.051, p=0.4588, d=-0.119\n",
      "\n",
      "2. Age Group Analysis\n",
      "Age group statistics (Affective Empathy):\n",
      "           count   mean    std\n",
      "age_group                     \n",
      "<18           12  2.500  0.522\n",
      "18-49         18  2.111  0.471\n",
      "50-64         54  2.296  0.571\n",
      "65+           72  2.403  0.522\n",
      "Age ANOVA: F=1.913, p=0.1298\n",
      "U-shape test: t=1.943, p=0.0538\n",
      "    \u2717 U-shape not significant\n",
      "\n",
      "3. Ethnicity Analysis\n",
      "Ethnicity statistics (Affective Empathy):\n",
      "           count   mean    std\n",
      "ethnicity                     \n",
      "African       52  2.269  0.490\n",
      "Asian         52  2.269  0.564\n",
      "European      52  2.481  0.542\n",
      "Ethnicity ANOVA: F=2.732, p=0.0683\n",
      "\n",
      "4. Education Analysis\n",
      "Education statistics (Affective Empathy):\n",
      "                              count   mean    std\n",
      "education                                        \n",
      "high school diploma or lower     60  2.367  0.520\n",
      "medical degree                   48  2.292  0.504\n",
      "university degree                48  2.354  0.601\n",
      "High School vs Medical: +0.075, p=0.4516\n",
      "\n",
      "5. Medical Diagnosis Analysis\n",
      "Diagnosis statistics (Affective Empathy):\n",
      "                                count   mean    std\n",
      "diagnosis                                          \n",
      "Alzheimer\u2019s                        36  2.500  0.561\n",
      "Chronic Ischemic Heart Disease     36  2.250  0.500\n",
      "obesity                            48  2.188  0.491\n",
      "pancreatic cancer                  36  2.472  0.560\n",
      "Diagnosis ANOVA: F=3.566, p=0.0156\n",
      "    \u2713 SIGNIFICANT diagnosis differences!\n",
      "    Highest empathy: Alzheimer\u2019s (2.500)\n",
      "    Lowest empathy: obesity (2.188)\n",
      "\n",
      "================================================================================\n",
      "3. RESPONSE SOURCE COMPARISON\n",
      "================================================================================\n",
      "\n",
      "\ud83e\udd16 Do Claude and GPT responses receive different empathy ratings?\n",
      "------------------------------------------------------------\n",
      "Sample sizes: Claude responses=156, GPT responses=156\n",
      "Affective Empathy (GPT rater):\n",
      "  Claude responses: 2.346\n",
      "  GPT responses: 2.679\n",
      "  Difference: -0.333, p=0.0000\n",
      "    \u2713 SIGNIFICANT: GPT responses receive higher affective empathy ratings!\n",
      "Cognitive Empathy (GPT rater):\n",
      "  Claude responses: 2.833\n",
      "  GPT responses: 2.865\n",
      "  Difference: -0.032, p=0.4304\n",
      "\n",
      "================================================================================\n",
      "4. SUMMARY OF ALL FINDINGS\n",
      "================================================================================\n",
      "\n",
      "Statistical Test Results (p-values):\n",
      "               gender_aff_p  gender_aff_d  gender_cog_p  gender_cog_d  \\\n",
      "Claude\u2192GPT           0.5265        0.1016        0.0867        0.2761   \n",
      "Claude\u2192Claude        0.6604        0.0705        0.8252       -0.0354   \n",
      "GPT\u2192GPT              1.0000        0.0000        0.2436        0.1874   \n",
      "GPT\u2192Claude           0.8824        0.0237        0.4588       -0.1189   \n",
      "\n",
      "               age_anova_p  age_u_shape_p  ethnicity_anova_p  edu_hs_vs_med_p  \\\n",
      "Claude\u2192GPT          0.0003         0.0014             0.3756           0.0000   \n",
      "Claude\u2192Claude       0.3708         0.7690             0.5143           0.0000   \n",
      "GPT\u2192GPT             0.0019         0.0006             0.6865           0.0009   \n",
      "GPT\u2192Claude          0.1298         0.0538             0.0683           0.4516   \n",
      "\n",
      "               diagnosis_anova_p  \n",
      "Claude\u2192GPT                0.0000  \n",
      "Claude\u2192Claude             0.0000  \n",
      "GPT\u2192GPT                   0.0000  \n",
      "GPT\u2192Claude                0.0156  \n",
      "\n",
      "\ud83d\udcca SIGNIFICANCE SUMMARY (p < 0.05):\n",
      "  gender_aff_p: 0/4 combinations significant\n",
      "  gender_aff_d: 2/4 combinations significant\n",
      "  gender_cog_p: 0/4 combinations significant\n",
      "  gender_cog_d: 2/4 combinations significant\n",
      "  age_anova_p: 2/4 combinations significant\n",
      "  age_u_shape_p: 2/4 combinations significant\n",
      "  ethnicity_anova_p: 0/4 combinations significant\n",
      "  edu_hs_vs_med_p: 3/4 combinations significant\n",
      "  diagnosis_anova_p: 4/4 combinations significant\n",
      "\n",
      "\ud83c\udfaf KEY INSIGHTS:\n",
      "\u2705 No response duplication - each response counted once\n",
      "\u2705 Proper sample sizes: 156 Claude + 156 GPT responses\n",
      "\u2705 Can analyze rater agreement separately from demographic patterns\n",
      "\u2705 Can compare response sources while controlling for demographics\n",
      "\n",
      "\ud83d\udcbe Results saved:\n",
      "  - comprehensive_empathy_results.csv\n",
      "  - combined_clean_empathy_data.csv\n"
     ]
    }
   ],
   "source": [
    "# PROPER EMPATHY ANALYSIS - Clean Data Structure\nimport pandas as pd\nimport numpy as np\nfrom scipy import stats\nfrom scipy.stats import f_oneway, ttest_ind, ttest_rel\nimport matplotlib.pyplot as plt\nimport seaborn as sns\nimport warnings\nwarnings.filterwarnings('ignore')\n\nprint(\"=\" * 80)\nprint(\"EMPATHY ANALYSIS - PROPER DATA STRUCTURE\")\nprint(\"=\" * 80)\n\n# Load the two files\nclaude_df = pd.read_csv('/Users/jianzhouyao/AI4Good/data/processed/ratings/claude_with_ratings.csv')\ngpt_df = pd.read_csv('/Users/jianzhouyao/AI4Good/data/processed/ratings/gpt_with_ratings.csv')\n\nprint(\"\ud83d\udcca DATA OVERVIEW\")\nprint(\"-\" * 40)\nprint(f\"Claude responses: {len(claude_df)} rows\")\nprint(f\"GPT responses: {len(gpt_df)} rows\")\nprint(f\"Total unique responses: {len(claude_df) + len(gpt_df)}\")\n\n# Add response source identifier\nclaude_df['Response_Source'] = 'Claude'\ngpt_df['Response_Source'] = 'GPT'\n\n# Create age groups in individual dataframes first\nclaude_df['age_group'] = pd.cut(claude_df['age'], \n                               bins=[0, 18, 50, 65, 100], \n                               labels=['<18', '18-49', '50-64', '65+'], \n                               right=False)\n\ngpt_df['age_group'] = pd.cut(gpt_df['age'], \n                            bins=[0, 18, 50, 65, 100], \n                            labels=['<18', '18-49', '50-64', '65+'], \n                            right=False)\n\n# Combine for demographic analysis\ncombined_df = pd.concat([claude_df, gpt_df], ignore_index=True)\n\nprint(f\"\\nPrompt overlap check:\")\nclaude_prompts = set(claude_df['Prompt Number'])\ngpt_prompts = set(gpt_df['Prompt Number'])\noverlap = len(claude_prompts.intersection(gpt_prompts))\nprint(f\"Shared prompt numbers: {overlap} (same demographic scenarios)\")\n\nprint(f\"\\n\ud83d\udccb SCORE OVERVIEW\")\nprint(\"-\" * 40)\nscore_columns = [\n    'Affective Empathy Score (GPT)', \n    'Affective Empathy Score (Claude)',\n    'Cognitive Empathy Score (GPT)', \n    'Cognitive Empathy Score (Claude)'\n]\n\nfor col in score_columns:\n    scores = combined_df[col].dropna()\n    print(f\"{col}: {scores.min()}-{scores.max()}, mean={scores.mean():.2f}, n={len(scores)}\")\n\nprint(\"\\n\" + \"=\" * 80)\nprint(\"1. RATER AGREEMENT ANALYSIS\")\nprint(\"=\" * 80)\n\ndef analyze_rater_agreement(df, response_type):\n    \"\"\"Analyze agreement between Claude and GPT raters\"\"\"\n    print(f\"\\n\ud83e\udd1d RATER AGREEMENT: {response_type} Responses\")\n    print(\"-\" * 50)\n    \n    # Affective empathy agreement\n    aff_gpt = df['Affective Empathy Score (GPT)'].dropna()\n    aff_claude = df['Affective Empathy Score (Claude)'].dropna()\n    \n    if len(aff_gpt) > 0 and len(aff_claude) > 0:\n        # Correlation\n        aff_corr = np.corrcoef(aff_gpt, aff_claude)[0,1]\n        print(f\"Affective Empathy Correlation: r = {aff_corr:.3f}\")\n        \n        # Paired t-test (systematic bias?)\n        t_stat_aff, p_val_aff = ttest_rel(aff_claude, aff_gpt)\n        mean_diff_aff = aff_claude.mean() - aff_gpt.mean()\n        print(f\"Claude vs GPT bias: {mean_diff_aff:+.3f}, p = {p_val_aff:.4f}\")\n        \n        if p_val_aff < 0.05:\n            rater_direction = \"Claude rates higher\" if mean_diff_aff > 0 else \"GPT rates higher\"\n            print(f\"  \u2713 SIGNIFICANT rater bias: {rater_direction}\")\n        else:\n            print(f\"  \u2717 No significant rater bias\")\n    \n    # Cognitive empathy agreement  \n    cog_gpt = df['Cognitive Empathy Score (GPT)'].dropna()\n    cog_claude = df['Cognitive Empathy Score (Claude)'].dropna()\n    \n    if len(cog_gpt) > 0 and len(cog_claude) > 0:\n        cog_corr = np.corrcoef(cog_gpt, cog_claude)[0,1]\n        print(f\"Cognitive Empathy Correlation: r = {cog_corr:.3f}\")\n        \n        t_stat_cog, p_val_cog = ttest_rel(cog_claude, cog_gpt)\n        mean_diff_cog = cog_claude.mean() - cog_gpt.mean()\n        print(f\"Claude vs GPT bias: {mean_diff_cog:+.3f}, p = {p_val_cog:.4f}\")\n        \n        if p_val_cog < 0.05:\n            rater_direction = \"Claude rates higher\" if mean_diff_cog > 0 else \"GPT rates higher\"\n            print(f\"  \u2713 SIGNIFICANT rater bias: {rater_direction}\")\n        else:\n            print(f\"  \u2717 No significant rater bias\")\n\n# Analyze rater agreement for each response type\nanalyze_rater_agreement(claude_df, \"Claude\")\nanalyze_rater_agreement(gpt_df, \"GPT\")\n\nprint(\"\\n\" + \"=\" * 80)\nprint(\"2. DEMOGRAPHIC BIAS ANALYSIS\")\nprint(\"=\" * 80)\n\ndef analyze_demographic_patterns(df, response_type, rater_type):\n    \"\"\"Analyze demographic biases for a specific response-rater combination\"\"\"\n    \n    print(f\"\\n\ud83c\udfaf {response_type} Responses \u2192 {rater_type} Ratings\")\n    print(\"-\" * 50)\n    \n    aff_col = f'Affective Empathy Score ({rater_type})'\n    cog_col = f'Cognitive Empathy Score ({rater_type})'\n    \n    results = {}\n    \n    # 1. GENDER ANALYSIS\n    print(f\"\\n1. Gender Analysis\")\n    male_data = df[df['gender'] == 'male']\n    female_data = df[df['gender'] == 'female']\n    \n    print(f\"Sample sizes: Male={len(male_data)}, Female={len(female_data)}\")\n    \n    if len(male_data) >= 2 and len(female_data) >= 2:\n        # Affective empathy gender test\n        male_aff = male_data[aff_col].dropna()\n        female_aff = female_data[aff_col].dropna()\n        \n        if len(male_aff) >= 2 and len(female_aff) >= 2:\n            t_stat, p_val = ttest_ind(female_aff, male_aff)\n            bias = female_aff.mean() - male_aff.mean()\n            \n            # Effect size\n            pooled_std = np.sqrt(((len(female_aff)-1)*female_aff.var() + (len(male_aff)-1)*male_aff.var()) / (len(female_aff)+len(male_aff)-2))\n            cohens_d = bias / pooled_std if pooled_std > 0 else 0\n            \n            print(f\"  Affective: Female-Male = {bias:+.3f}, p={p_val:.4f}, d={cohens_d:.3f}\")\n            results['gender_aff_p'] = p_val\n            results['gender_aff_d'] = cohens_d\n            \n            if p_val < 0.05:\n                print(f\"    \u2713 SIGNIFICANT gender bias!\")\n            elif abs(cohens_d) >= 0.2:\n                print(f\"    ~ Meaningful effect size\")\n        \n        # Cognitive empathy gender test\n        male_cog = male_data[cog_col].dropna()\n        female_cog = female_data[cog_col].dropna()\n        \n        if len(male_cog) >= 2 and len(female_cog) >= 2:\n            t_stat, p_val = ttest_ind(female_cog, male_cog)\n            bias = female_cog.mean() - male_cog.mean()\n            \n            pooled_std = np.sqrt(((len(female_cog)-1)*female_cog.var() + (len(male_cog)-1)*male_cog.var()) / (len(female_cog)+len(male_cog)-2))\n            cohens_d = bias / pooled_std if pooled_std > 0 else 0\n            \n            print(f\"  Cognitive: Female-Male = {bias:+.3f}, p={p_val:.4f}, d={cohens_d:.3f}\")\n            results['gender_cog_p'] = p_val\n            results['gender_cog_d'] = cohens_d\n            \n            if p_val < 0.05:\n                print(f\"    \u2713 SIGNIFICANT gender bias!\")\n            elif abs(cohens_d) >= 0.2:\n                print(f\"    ~ Meaningful effect size\")\n    \n    # 2. AGE GROUP ANALYSIS\n    print(f\"\\n2. Age Group Analysis\")\n    age_stats = df.groupby('age_group')[aff_col].agg(['count', 'mean', 'std']).round(3)\n    print(\"Age group statistics (Affective Empathy):\")\n    print(age_stats)\n    \n    # Test U-shaped pattern\n    age_groups = ['<18', '18-49', '50-64', '65+']\n    age_data = []\n    age_means = []\n    \n    for age_group in age_groups:\n        scores = df[df['age_group'] == age_group][aff_col].dropna()\n        if len(scores) >= 2:\n            age_data.append(scores)\n            age_means.append(scores.mean())\n        else:\n            age_data.append(None)\n            age_means.append(None)\n    \n    # Test overall age effect\n    valid_age_data = [data for data in age_data if data is not None]\n    if len(valid_age_data) >= 2:\n        f_stat, p_val = f_oneway(*valid_age_data)\n        print(f\"Age ANOVA: F={f_stat:.3f}, p={p_val:.4f}\")\n        results['age_anova_p'] = p_val\n        \n        # Test U-shape specifically (young + old vs middle)\n        if age_data[0] is not None and age_data[3] is not None and age_data[1] is not None and age_data[2] is not None:\n            young_old = pd.concat([age_data[0], age_data[3]])  # <18 + 65+\n            middle = pd.concat([age_data[1], age_data[2]])     # 18-49 + 50-64\n            \n            t_stat, p_val_u = ttest_ind(young_old, middle)\n            print(f\"U-shape test: t={t_stat:.3f}, p={p_val_u:.4f}\")\n            results['age_u_shape_p'] = p_val_u\n            \n            if p_val_u < 0.05:\n                print(f\"    \u2713 SIGNIFICANT U-shaped age pattern!\")\n            else:\n                print(f\"    \u2717 U-shape not significant\")\n    \n    # 3. ETHNICITY ANALYSIS\n    print(f\"\\n3. Ethnicity Analysis\")\n    eth_stats = df.groupby('ethnicity')[aff_col].agg(['count', 'mean', 'std']).round(3)\n    print(\"Ethnicity statistics (Affective Empathy):\")\n    print(eth_stats)\n    \n    # Test ethnicity differences\n    ethnicities = df['ethnicity'].unique()\n    eth_data = []\n    eth_means = []\n    \n    for eth in ethnicities:\n        scores = df[df['ethnicity'] == eth][aff_col].dropna()\n        if len(scores) >= 2:\n            eth_data.append(scores)\n            eth_means.append(scores.mean())\n            \n    if len(eth_data) >= 2:\n        f_stat, p_val = f_oneway(*eth_data)\n        print(f\"Ethnicity ANOVA: F={f_stat:.3f}, p={p_val:.4f}\")\n        results['ethnicity_anova_p'] = p_val\n        \n        if p_val < 0.05:\n            print(f\"    \u2713 SIGNIFICANT ethnicity differences!\")\n            # Find highest and lowest\n            max_idx = np.argmax(eth_means)\n            min_idx = np.argmin(eth_means)\n            print(f\"    Highest: {list(ethnicities)[max_idx]} ({eth_means[max_idx]:.3f})\")\n            print(f\"    Lowest: {list(ethnicities)[min_idx]} ({eth_means[min_idx]:.3f})\")\n    \n    # 4. EDUCATION ANALYSIS\n    print(f\"\\n4. Education Analysis\")\n    edu_stats = df.groupby('education')[aff_col].agg(['count', 'mean', 'std']).round(3)\n    print(\"Education statistics (Affective Empathy):\")\n    print(edu_stats)\n    \n    # Test education hierarchy\n    education_levels = df['education'].unique()\n    if 'high school diploma or lower' in education_levels and 'medical degree' in education_levels:\n        high_school = df[df['education'] == 'high school diploma or lower'][aff_col].dropna()\n        medical = df[df['education'] == 'medical degree'][aff_col].dropna()\n        \n        if len(high_school) >= 2 and len(medical) >= 2:\n            t_stat, p_val = ttest_ind(high_school, medical)\n            bias = high_school.mean() - medical.mean()\n            print(f\"High School vs Medical: {bias:+.3f}, p={p_val:.4f}\")\n            results['edu_hs_vs_med_p'] = p_val\n            \n            if p_val < 0.05:\n                direction = \"Higher\" if bias > 0 else \"Lower\"\n                print(f\"    \u2713 SIGNIFICANT: High school gets {direction.lower()} empathy!\")\n    \n    # 5. DIAGNOSIS ANALYSIS\n    print(f\"\\n5. Medical Diagnosis Analysis\")\n    dx_stats = df.groupby('diagnosis')[aff_col].agg(['count', 'mean', 'std']).round(3)\n    print(\"Diagnosis statistics (Affective Empathy):\")\n    print(dx_stats.head(10))  # Show top 10\n    \n    # Test diagnosis differences\n    diagnoses = df['diagnosis'].unique()\n    dx_data = []\n    dx_means = []\n    dx_names = []\n    \n    for dx in diagnoses:\n        scores = df[df['diagnosis'] == dx][aff_col].dropna()\n        if len(scores) >= 2:\n            dx_data.append(scores)\n            dx_means.append(scores.mean())\n            dx_names.append(dx)\n            \n    if len(dx_data) >= 2:\n        f_stat, p_val = f_oneway(*dx_data)\n        print(f\"Diagnosis ANOVA: F={f_stat:.3f}, p={p_val:.4f}\")\n        results['diagnosis_anova_p'] = p_val\n        \n        if p_val < 0.05:\n            print(f\"    \u2713 SIGNIFICANT diagnosis differences!\")\n            max_idx = np.argmax(dx_means)\n            min_idx = np.argmin(dx_means)\n            print(f\"    Highest empathy: {dx_names[max_idx]} ({dx_means[max_idx]:.3f})\")\n            print(f\"    Lowest empathy: {dx_names[min_idx]} ({dx_means[min_idx]:.3f})\")\n    \n    return results\n\n# Run demographic analysis for all combinations\nprint(f\"\\n\ud83d\udd04 Running all response-rater combinations...\")\n\nclaude_gpt_results = analyze_demographic_patterns(claude_df, \"Claude\", \"GPT\")\nclaude_claude_results = analyze_demographic_patterns(claude_df, \"Claude\", \"Claude\")\ngpt_gpt_results = analyze_demographic_patterns(gpt_df, \"GPT\", \"GPT\") \ngpt_claude_results = analyze_demographic_patterns(gpt_df, \"GPT\", \"Claude\")\n\nprint(\"\\n\" + \"=\" * 80)\nprint(\"3. RESPONSE SOURCE COMPARISON\")\nprint(\"=\" * 80)\n\nprint(f\"\\n\ud83e\udd16 Do Claude and GPT responses receive different empathy ratings?\")\nprint(\"-\" * 60)\n\n# Compare empathy ratings between Claude and GPT responses\n# Using GPT as rater for fair comparison\nclaude_aff_gpt = claude_df['Affective Empathy Score (GPT)'].dropna()\ngpt_aff_gpt = gpt_df['Affective Empathy Score (GPT)'].dropna()\n\nclaude_cog_gpt = claude_df['Cognitive Empathy Score (GPT)'].dropna()\ngpt_cog_gpt = gpt_df['Cognitive Empathy Score (GPT)'].dropna()\n\nprint(f\"Sample sizes: Claude responses={len(claude_aff_gpt)}, GPT responses={len(gpt_aff_gpt)}\")\n\n# Affective empathy comparison\nif len(claude_aff_gpt) > 0 and len(gpt_aff_gpt) > 0:\n    t_stat, p_val = ttest_ind(claude_aff_gpt, gpt_aff_gpt)\n    bias = claude_aff_gpt.mean() - gpt_aff_gpt.mean()\n    \n    print(f\"Affective Empathy (GPT rater):\")\n    print(f\"  Claude responses: {claude_aff_gpt.mean():.3f}\")\n    print(f\"  GPT responses: {gpt_aff_gpt.mean():.3f}\")\n    print(f\"  Difference: {bias:+.3f}, p={p_val:.4f}\")\n    \n    if p_val < 0.05:\n        winner = \"Claude\" if bias > 0 else \"GPT\"\n        print(f\"    \u2713 SIGNIFICANT: {winner} responses receive higher affective empathy ratings!\")\n\n# Cognitive empathy comparison\nif len(claude_cog_gpt) > 0 and len(gpt_cog_gpt) > 0:\n    t_stat, p_val = ttest_ind(claude_cog_gpt, gpt_cog_gpt)\n    bias = claude_cog_gpt.mean() - gpt_cog_gpt.mean()\n    \n    print(f\"Cognitive Empathy (GPT rater):\")\n    print(f\"  Claude responses: {claude_cog_gpt.mean():.3f}\")\n    print(f\"  GPT responses: {gpt_cog_gpt.mean():.3f}\")\n    print(f\"  Difference: {bias:+.3f}, p={p_val:.4f}\")\n    \n    if p_val < 0.05:\n        winner = \"Claude\" if bias > 0 else \"GPT\"\n        print(f\"    \u2713 SIGNIFICANT: {winner} responses receive higher cognitive empathy ratings!\")\n\nprint(\"\\n\" + \"=\" * 80)\nprint(\"4. SUMMARY OF ALL FINDINGS\")\nprint(\"=\" * 80)\n\n# Create comprehensive results summary\nall_results = {\n    'Claude\u2192GPT': claude_gpt_results,\n    'Claude\u2192Claude': claude_claude_results,\n    'GPT\u2192GPT': gpt_gpt_results,\n    'GPT\u2192Claude': gpt_claude_results\n}\n\nresults_df = pd.DataFrame(all_results).T\nprint(f\"\\nStatistical Test Results (p-values):\")\nprint(results_df.round(4))\n\n# Count significant results across all combinations\nprint(f\"\\n\ud83d\udcca SIGNIFICANCE SUMMARY (p < 0.05):\")\nfor test in results_df.columns:\n    if test in results_df.columns:\n        sig_count = sum(1 for p in results_df[test].dropna() if p < 0.05)\n        total_count = results_df[test].count()\n        print(f\"  {test}: {sig_count}/{total_count} combinations significant\")\n\nprint(f\"\\n\ud83c\udfaf KEY INSIGHTS:\")\nprint(f\"\u2705 No response duplication - each response counted once\")\nprint(f\"\u2705 Proper sample sizes: {len(claude_df)} Claude + {len(gpt_df)} GPT responses\")\nprint(f\"\u2705 Can analyze rater agreement separately from demographic patterns\")\nprint(f\"\u2705 Can compare response sources while controlling for demographics\")\n\n# Save results\nresults_df.to_csv('data/results/empathy_scores/comprehensive_empathy_results.csv')\ncombined_df.to_csv('data/results/empathy_scores/combined_empathy_data.csv', index=False)\n\nprint(f\"\\n\ud83d\udcbe Results saved:\")\nprint(f\"  - data/results/empathy_scores/comprehensive_empathy_results.csv\")\nprint(f\"  - data/results/empathy_scores/combined_empathy_data.csv\")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 17,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "================================================================================\n",
      "INTRA-MODEL BIAS & WITHIN-MODEL CONSISTENCY ANALYSIS (Revised)\n",
      "================================================================================\n",
      "\ud83d\udcca DATA OVERVIEW\n",
      "----------------------------------------\n",
      "Claude responses: 156 rows\n",
      "GPT responses: 156 rows\n",
      "\n",
      "================================================================================\n",
      "1. SELF-EVALUATION BIAS ANALYSIS (Paired by Prompt)\n",
      "================================================================================\n",
      "\n",
      "\ud83d\udd0d GPT Self-Evaluation (Affective):\n",
      "  GPT rating GPT vs GPT rating Claude (n=156): \u0394=+0.333, 95%CI=(0.244,0.423), p=1.555e-11, dz=0.583\n",
      "\n",
      "\ud83d\udd0d Claude Self-Evaluation (Affective):\n",
      "  Claude rating Claude vs Claude rating GPT (n=156): \u0394=-0.256, 95%CI=(-0.373,-0.140), p=2.939e-05, dz=-0.345\n",
      "\n",
      "\ud83e\udde0 GPT Self-Evaluation (Cognitive):\n",
      "  \u0394=+0.032, 95%CI=(-0.047,0.111), p=0.4251, dz=0.064\n",
      "\n",
      "\ud83e\udde0 Claude Self-Evaluation (Cognitive):\n",
      "  \u0394=+0.109, 95%CI=(0.028,0.190), p=0.009095, dz=0.212\n",
      "\n",
      "================================================================================\n",
      "2. WITHIN-MODEL CONSISTENCY ANALYSIS (SD, IQR, Levene)\n",
      "================================================================================\n",
      "\n",
      "\ud83d\udccf GPT rating distributions (Affective):\n",
      "  GPT\u2192GPT:   n=156, mean=2.679, SD=0.468, IQR=1.000\n",
      "  GPT\u2192Claude:n=156, mean=2.346, SD=0.504, IQR=1.000\n",
      "  Variance difference test (Levene, median): W=0.903, p=0.3427\n",
      "\n",
      "\ud83d\udd0d Claude rating distributions (Affective):\n",
      "  Claude\u2192Claude: n=156, mean=2.083, SD=0.544, IQR=0.000\n",
      "  Claude\u2192GPT:    n=156, mean=2.340, SD=0.539, IQR=1.000\n",
      "  Variance difference test (Levene, median): W=3.613, p=0.05825\n",
      "\n",
      "================================================================================\n",
      "3. DEMOGRAPHIC-SPECIFIC SELF-BIAS (Paired; BH-FDR correction)\n",
      "================================================================================\n",
      "\n",
      "  GPT_rater_affective \u2014 gender (k=2 groups; FDR-BH \u03b1=0.05):\n",
      "                       female: \u0394=+0.308, n= 78, p=1.502e-05, p_FDR=1.502e-05 \u2713\n",
      "                         male: \u0394=+0.359, n= 78, p=2.239e-07, p_FDR=4.478e-07 \u2713\n",
      "\n",
      "  Claude_rater_affective \u2014 gender (k=2 groups; FDR-BH \u03b1=0.05):\n",
      "                       female: \u0394=-0.244, n= 78, p=0.01092, p_FDR=0.01092 \u2713\n",
      "                         male: \u0394=-0.269, n= 78, p=0.000537, p_FDR=0.001074 \u2713\n",
      "\n",
      "  GPT_rater_affective \u2014 ethnicity (k=3 groups; FDR-BH \u03b1=0.05):\n",
      "                      African: \u0394=+0.308, n= 52, p=0.0001581, p_FDR=0.0002371 \u2713\n",
      "                        Asian: \u0394=+0.269, n= 52, p=0.0005709, p_FDR=0.0005709 \u2713\n",
      "                     European: \u0394=+0.423, n= 52, p=1.479e-05, p_FDR=4.438e-05 \u2713\n",
      "\n",
      "  Claude_rater_affective \u2014 ethnicity (k=3 groups; FDR-BH \u03b1=0.05):\n",
      "                      African: \u0394=-0.231, n= 52, p=0.02699, p_FDR=0.04048 \u2713\n",
      "                        Asian: \u0394=-0.115, n= 52, p=0.2608, p_FDR=0.2608  \n",
      "                     European: \u0394=-0.423, n= 52, p=0.000165, p_FDR=0.0004951 \u2713\n",
      "\n",
      "  GPT_rater_affective \u2014 education (k=3 groups; FDR-BH \u03b1=0.05):\n",
      "    high school diploma or lower: \u0394=+0.217, n= 60, p=0.005698, p_FDR=0.005698 \u2713\n",
      "               medical degree: \u0394=+0.417, n= 48, p=2.523e-06, p_FDR=7.57e-06 \u2713\n",
      "            university degree: \u0394=+0.396, n= 48, p=1.77e-05, p_FDR=2.656e-05 \u2713\n",
      "\n",
      "  Claude_rater_affective \u2014 education (k=3 groups; FDR-BH \u03b1=0.05):\n",
      "    high school diploma or lower: \u0394=-0.033, n= 60, p=0.7347, p_FDR=0.7347  \n",
      "               medical degree: \u0394=-0.417, n= 48, p=0.0002987, p_FDR=0.0005114 \u2713\n",
      "            university degree: \u0394=-0.375, n= 48, p=0.000341, p_FDR=0.0005114 \u2713\n",
      "\n",
      "  GPT_rater_affective \u2014 diagnosis (k=4 groups; FDR-BH \u03b1=0.05):\n",
      "                  Alzheimer\u2019s: \u0394=+0.361, n= 36, p=8.397e-05, p_FDR=0.0002366 \u2713\n",
      "    Chronic Ischemic Heart Disease: \u0394=+0.333, n= 36, p=0.0006545, p_FDR=0.0008727 \u2713\n",
      "                      obesity: \u0394=+0.208, n= 48, p=0.01677, p_FDR=0.01677 \u2713\n",
      "            pancreatic cancer: \u0394=+0.472, n= 36, p=0.0001183, p_FDR=0.0002366 \u2713\n",
      "\n",
      "  Claude_rater_affective \u2014 diagnosis (k=4 groups; FDR-BH \u03b1=0.05):\n",
      "                  Alzheimer\u2019s: \u0394=-0.250, n= 36, p=0.04798, p_FDR=0.09595  \n",
      "    Chronic Ischemic Heart Disease: \u0394=-0.611, n= 36, p=1.44e-05, p_FDR=5.759e-05 \u2713\n",
      "                      obesity: \u0394=-0.062, n= 48, p=0.5184, p_FDR=0.5184  \n",
      "            pancreatic cancer: \u0394=-0.167, n= 36, p=0.2052, p_FDR=0.2736  \n",
      "\n",
      "  GPT_rater_affective \u2014 age_group (k=4 groups; FDR-BH \u03b1=0.05):\n",
      "                        18-49: \u0394=+0.222, n= 18, p=0.1037, p_FDR=0.1382  \n",
      "                        50-64: \u0394=+0.370, n= 54, p=0.0001139, p_FDR=0.0002277 \u2713\n",
      "                          65+: \u0394=+0.361, n= 72, p=2.677e-07, p_FDR=1.071e-06 \u2713\n",
      "                          <18: \u0394=+0.167, n= 12, p=0.1661, p_FDR=0.1661  \n",
      "\n",
      "  Claude_rater_affective \u2014 age_group (k=4 groups; FDR-BH \u03b1=0.05):\n",
      "                        18-49: \u0394=-0.111, n= 18, p=0.4953, p_FDR=0.4953  \n",
      "                        50-64: \u0394=-0.204, n= 54, p=0.04717, p_FDR=0.09434  \n",
      "                          65+: \u0394=-0.347, n= 72, p=0.0002867, p_FDR=0.001147 \u2713\n",
      "                          <18: \u0394=-0.167, n= 12, p=0.4382, p_FDR=0.4953  \n",
      "\n",
      "================================================================================\n",
      "4. CROSS-EVALUATION ASYMMETRY (Paired; Spearman agreement)\n",
      "================================================================================\n",
      "\n",
      "\u2696\ufe0f Cross-evaluation asymmetry (Affective):\n",
      "  GPT\u2192Claude vs Claude\u2192GPT (n=156): \u0394=+0.006, 95%CI=(-0.111,0.124), p=0.915, dz=0.009\n",
      "  Cross-rater rank agreement (Spearman): \u03c1=-0.020, p=0.8015\n",
      "\n",
      "================================================================================\n",
      "5. RESPONSE\u2013RATER INTERACTION EFFECTS (Paired)\n",
      "================================================================================\n",
      "\n",
      "\ud83d\udcca Summary means \u00b1 SD (unpaired descriptive):\n",
      "  Claude\u2192Claude: 2.083 \u00b1 0.544 (n=156)\n",
      "  Claude\u2192GPT:    2.340 \u00b1 0.539 (n=156)\n",
      "  GPT\u2192GPT:       2.679 \u00b1 0.468 (n=156)\n",
      "  GPT\u2192Claude:    2.346 \u00b1 0.504 (n=156)\n",
      "\n",
      "\ud83d\udd0d Paired contrasts by prompt:\n",
      "  Claude self vs other:        \u0394=-0.256, p=2.939e-05, dz=-0.345, n=156\n",
      "  GPT self vs other:           \u0394=+0.333, p=1.555e-11, dz=0.583, n=156\n",
      "  Self-eval: Claude vs GPT:    \u0394=-0.596, p=6.534e-24, dz=-0.962, n=156\n",
      "\n",
      "================================================================================\n",
      "6. COMPREHENSIVE INTRA-MODEL BIAS SUMMARY\n",
      "================================================================================\n",
      "\n",
      "\ud83d\udd11 KEY FINDINGS (Affective, paired):\n",
      "  \u2022 GPT self-evaluation bias \u0394 (GPT\u2192GPT \u2013 GPT\u2192Claude): +0.333 (p=1.555e-11, dz=0.583)\n",
      "  \u2022 Claude self-evaluation bias \u0394 (Claude\u2192Claude \u2013 Claude\u2192GPT): -0.256 (p=2.939e-05, dz=-0.345)\n",
      "  \u2022 Cross-eval asymmetry \u0394 (GPT\u2192Claude \u2013 Claude\u2192GPT): +0.006 (p=0.915, dz=0.009)\n",
      "\n",
      "\ud83d\udcbe Saved:\n",
      "  \u2022 intra_model_bias_results_revised.csv\n",
      "  \u2022 demographic_self_bias_fdr.csv\n"
     ]
    },
    {
     "data": {
      "image/png": "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HTsmT09PPfHEEw983dOnT0uSzQ3flJ9TyiTp5MmTWr58uQYNGiRfX1+n3uOkSZN07Ngx1axZUzNnzrRJrqXkng0jRoxQq1atFB0dbdMV/EFefvllhYeHa86cOU7FBjxuSLABuERKMmMymdJcp3z58urQoYNat26t2NhYm7KkpCQtWrRInTp1sj4z1qVLFy1fvvyByXjKs2MXL16UJDVr1kzBwcHauXNnBt9V2p577jkFBwdr+/btqZaPHj1awcHB+uabb6zLrly5ovHjx6t169aqUqWKKlSooEaNGmnEiBE6derUA1/zQc98pvVM7JEjR/Tmm2+qSZMmCgkJUeXKldWiRQtNmDBBt27dsq43cuRI9ezZU5K0b98+BQcHq0ePHpLu/wz2vn37NGjQINWuXVsVKlRQ48aN9cEHH+jy5ct26zZp0kTVq1dXfHy8pk6dqqZNm6pChQpq2LChPvroI924ceOBn8PdUlpg1q1bZ1128OBBSXKoC+TdLl26pF9//VVeXl4aMWKEDAZDquuZTCYNHDhQISEhNi1bKX+fr7/+WpMmTVKNGjVUpUoVm79XRESExo8fr+bNm6tChQqqWbOm+vTpo99//z3V19q0aZP69Omj+vXrWz/bt99+W3///bfduvv379fAgQPVpEkTVahQQfXq1dOQIUOsPTsyU1JSkurVq6fg4OA0n73t16+fgoOD9dtvv1mXnT59WqNGjVLz5s1VuXJlVaxYUU8//bT+7//+L81nYO92v+eyL1y4oODgYDVp0sSubPv27RoyZIgaNGigChUqqEqVKmrbtq1mzJih+Ph463o9evTQ22+/LSm5q3hwcLC19839jseUv1vNmjUVEhKi5s2ba8KECYqMjLRbNzg4WG3bttXt27f18ccfq1GjRqpQoYKaNm2qyZMn250rM6pVq1YKCQnR1atXtX79ervyrVu3qm/fvtaxLFq0aKGpU6cqJibmvvWmnCPuPQ/f3V3c0XNRivTs/2nZv3+/du/erdatW9scr6GhoZKkOnXqpJrU1q5dW5K0efNm67IbN27o8uXLCg4Odujxo1y5ckmS4uLirMuio6MlyeY1J06cqMKFC6tz584Ov6+73blzR4sXL5YkvfPOO/d9BGvQoEEqW7asAgMDHzhGSYoqVaooKChICxcufOB+AIAEG4CLpCR2u3bt0rRp09IcBGrc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f77MZrP69u2rhg0bWpcbjUa9+uqrqlmzpsLCwvTjjz/abduvXz+bAYIKFy5sbVX766+/HI7h999/13fffScpueeAn5+fhg8fbu2amR4p09c4O4Jvivz581t7LkjJn8eePXt09OhRFStWTB988IFNF86KFStqxIgRkqQvv/zSJh43NzebeAwGgwYMGKD3339f7du3t4v93mfGu3XrprffftupgZOWL1/+wGPihx9+sK5ftmxZBQcH688//7T7G6YcE23btpWU3E22QoUK6tixo5o1a2azbmBgoJ5++mlJmXNMNG3aVAMHDlSJEiVsyqpXr64yZcpk6HW//vprSdJbb72lkJAQ63IPDw/93//9n4oXL66jR4+m2qPgjTfeUNGiRa2/V6hQwTqaf3qOCUekDFR19yMZKefYd955x6YV1MPDQ6NGjVJAQIB+/vlnh7ru38uZc1F69v+0nD9/XhcvXlSxYsXk6elpU1avXj0FBgbqxIkTmjBhgnUMDyl5wMzp06dLks0jAylTnDVq1Ei///67vvjiC3399ddat26dateurTNnzuj999+3eZ2XX35Z8+fP10svvaQXX3xRc+bMsT5WsHbtWh04cEBvvvmm9ZGUhISEVLvL34+rzl33k3K+zsxHr4B/CkYRB+Aybm5uGjhwoHr16qXNmzdr69at2rVrl86ePauYmBitXr1aa9eu1fvvv6+uXbtKSp5m6fbt2woODlb+/Pnt6gwJCVFAQIBOnz6ta9euPbDrrrMcmabr3ueb27Vrp40bN2r16tXW0WDj4uK0fv165cyZ0+bZzw8++MCuvhs3bujEiRPat2+fJNsLOVcoVKiQ3WjCFotFly9f1tGjR60Xs86+7u7duyVJLVq0SLX82Wef1a5du7Rr1y516tTJpiy1brYpg0U5+szpzJkz9emnn8rHx0ejRo3SqlWrdODAAQ0cOFDfffedfHx89Ntvv8nPz08hISHy8vK6b30pXUKTkpIcev20pDYy765duyRJTz/9dKrdMp955hm99957OnjwoOLj4+Xh4aEaNWrojz/+UPv27fXcc89ZnxcOCAjQCy+8YLN9jRo19Pfff6tbt27q1KmTtbu1h4eHXnzxRafehyPTdBUrVszm93bt2mn8+PFavXq1XnvtNUnJ057t3btXpUqVsiadvr6+GjdunF19V69e1fHjx3XixAlJrj8mypUrZ/e8ttls1vnz53X48GFrF9+7ky1HJSYmat++fTIYDGrevLlduZubm5o1a6bZs2dr586dqlOnjk25K44JR6W8v5SkLikpSXv27JEku7ik5EEOa9SooV9++UV79uxRy5Yt0/V6zpyL0rP/pyXlsZvChQvblXl5eemTTz7RK6+8orlz52rNmjV68skndePGDR06dEjPPPOMtm7davPM8fjx4zV06FAVKFDA5nzi7++vTz75RM8884x+++03XbhwQUWKFLGW16pVy25mi8TERH366aeqV6+e6tSpo4SEBH344Ydavny54uPjVbx4cb3//vvWRyrux1XnrvvJmzevPD09de3aNSUmJqareznwuOHoAOByvr6+atGihTXxunLlijZv3qx58+bpr7/+0ujRo1WtWjWVKVPGegEUGhr6wDmLL1++nO4Ee+zYsYqIiLBb/s4779i0ljszTVfjxo2VK1cubdiwQXFxcfL09NTGjRsVHR2tzp072w00ExoaqkWLFunw4cM6d+6ctZUi5SLXkknT2/z+++9asWKF/vrrL50/f946cnBag3g56urVq5JSv3iVZL3ATGlduVtqU72kXLClNS3S3VKme/Pz89PChQtVtmxZNW/eXJ06dVJoaKjefvtt/ec//9G7776r8PBwLVu2TBUqVLhvnSn7ljOjj98ttdbzlM/q7ovuu/n4+Mjf31/Xrl1TRESEChQooI8++kivvvqqDh8+rKlTp2rq1KnKnTu3GjZsqOeee041a9a0bv/mm2/q4sWL+uOPPzR79mzNnj1bPj4+ql+/vtq0aWNtEZaU5iB59+7/zkzT1bp1a02cOFFr1qyxJtg//fSTLBaLtfX6bvv27dPSpUt17NgxnTt3zprMZOYxkZSUpLVr12r16tU6efKkLl68aJdwOvO6kZGRSkhIUJ48edJ8rtfZY8LVn0PKYGsp+2pkZKQ1ib97v0pNRsaKSM+5KD37f1pSjuW0RuauW7eulixZounTp2vnzp3aunWrSpcurQ8++ECdOnVSxYoVlTt3buv6Hh4edj0fUuTPn1/lypWz9lZJ61hPsWTJEp09e9Y69sKsWbO0ePFi9ejRQ1WrVtWsWbP06quv6ueff07zHJsi5UZMRs9dD5IjRw5dv35dERER2Wr0fOBhI8EG4BInT57UtWvXVL16dbuBWvLnz69OnTqpbdu26tmzp/bv36+ffvpJr7/+ujWZKlSo0ANHfXZm+pL169dbR7W922uvvWaTYDvDw8NDzzzzjBYvXqxNmzapefPm1gGe7k0mvvzyS02YMEFScgtngwYNVKZMGYWEhOjs2bMaPXp0hmJJreXCbDZr4MCB+u233+Tu7q4KFSqoTZs2KlOmjCpXrqxvvvlGK1eudPo1Uy7600rUU8pTG9E2o8l9yujsffv2tXZdzJs3r2bMmKHnn39ev/zyi27duqXw8HAVLVr0gcm1lNwld/HixQ6Num02mzVhwgRVqVJF9evXtxmdOLX35kiCdO/nVaBAAX3//ffau3ev1q9fr23btik0NFQrV67UypUr9fLLL1uTZT8/P3355Zc6fvy4fv31V23dulVHjhzR2rVrtXbtWj377LPWltvUuuxL9gm2MwIDA1WnTh1t2bJFR44cUYUKFbR69WoZDAa1adPGZt3Ro0fr22+/lclkUtmyZfXMM8/oiSeeUKVKlfTHH3/oiy++yFAsqd2oiYmJUa9evXTo0CH5+PioQoUKqlu3roKCglStWjWNGTPG2jMjvR50PNy9TmYcE+mR0kMg5aZmyvnDw8Mj1db3uxUvXjzdr+fMuSg9+39aUgZMvN/xFxwcbJ2j+m7nzp1TQkJCuqYmTBlY80E9DmJiYvT555+rVatW1u7433zzjYKCgvTee+9JSu4d0rFjRy1atOiB77N8+fKS/jeDwoNMmzZNxYsXV8OGDdM1r3XKfuLITVDgcUaCDcAlBg0apDNnzmjp0qWqWLFiqut4eHiodevW2r9/v/XZv5RWwwIFCrjkAv9eGzdudHmdd2vXrp0WL16sX375Rf/617/0+++/q0iRIjY3C86fP6///Oc/ypEjh2bPnm3XFdTR5yuNxuRhM1K7uEltdN+VK1fqt99+U9myZTV79my7Fofbt2879LppyZcvny5cuKALFy5Yn129W8poyZnxXGDK/Nd3P8ed8vvEiRP16quvatu2bZKk/v37O1Rn48aNZTAYtHv3boWHh9837p07d2ru3Llyc3PTli1bbBLs1KR89mk92xsdHa2IiAiZTCabFjMpebqxlP0ppTV+0qRJmjNnjnr06GHzaMWTTz6pJ598UkOHDtXt27e1Zs0affzxx/r555/Vq1cvVa5c2To9UWZp166dtmzZol9++UW+vr7WuXnvTlR27dqlb7/9VgULFtScOXPs5uz95ZdfHHqtlKQ0tWMitSmx5s6dq0OHDqlu3bqaOnWqXUtzep99vVvu3Lnl7u6uyMhIRUVFpdqKnZnHRHps2rRJ0v/mj0+JPTExUWPHjr3vNE/OyMi5KD37/71SWuhT68UUFxenw4cPy2g0Wp91v1vKc/IpN+fi4+M1evRo3bhxQxMnTrSZCSNFyt+3QIECacYkJe+HkZGR1l4eN27cUEREhE038pRz6t3zZaelZs2aypEjh86cOfPA3mDnzp2zjieycuVKhxNss9ls/Z5JT1IOPI4Y5AyAS6RcoMybN+++66VcLKQ8p5rybOyJEyes3WjvduXKFbVo0UK9e/e2zh+alVStWlXFixfX5s2btXbtWsXHx6tNmzY2rVGHDh2S2WxWrVq1Un3OcsuWLZIe3MqZckGXWjfAlOe477Z//35JUocOHewuaKOjo63ldycn6WlFq1GjhqTkgXpSs2bNGkmye/bQFVIG80p5tvluTz31lP71r39Zf7/3OeG0BAYGqk2bNoqLi9Mnn3yS5nrx8fHW1uBnn31WefLkeWDdKZ/Vhg0bUp2G7JdffpHZbFb16tVlNBp16tQptW7dWn379rVZLyAgwDqftNls1pUrV3Tr1i3r/PJ3y5Ejhzp37qx69epJyljX3vRo2rSpfH19tWHDBmuifG+PjgMHDkhKniv53uQ6KSnJmtw86JhI6dWS2jGRsn+ntqx79+52CfCVK1es8ys7c0y4u7urSpUqMpvNNvOyp0hMTLQuz4xjwlE///yzTp06pQIFCljHjvDw8FDlypVlNpv1xx9/2G1jsVj04osvqmvXrk7Nq57ec1F69v/7SZkiL7XvlsTERL344osaPHiw3Q0ai8VinVc6ZcBCDw8Pbdu2TRs2bLCZGzvFiRMndOLECeXIkeO+YxeEh4drzpw56tatm7Ubecp+fvez/ymt4CnPV9+Pu7u7evToISn5sai0pjq0WCzWc1u1atXsblDez/Xr12U2m5U/f/5Uby4A+B8SbAAu0bdvX3l5eemnn37SqFGj7FqPzGazlixZou+++04BAQHWEbZ9fHzUuXNnxcTE6M0337S5UI6Ojtbbb7+tU6dOycfHx6ku4g9DmzZtFBUVZe1meG8ykdIV/eDBgzbvLyEhQZMnT7Ze0MbFxd33dVIuhlKe8Utx+fLlVFv/U1538+bNNhdcN27c0Ouvv27tRXD366aMtOvI/MMvvPCCTCaTZs+ebb1JICVfxE2bNk27d+9W/vz5bZ7/dZUuXbpISp6H++65fO/cuaPx48frjz/+sL6XwYMH68iRIw7V+9ZbbylPnjxasWKF3nzzTbsL8/DwcA0dOlQHDx5Unjx5Hth1M0XNmjVVrlw5nT17Vh9++KHNhfSRI0esF70pgzcVL15c169ft7YE3+3IkSP6+++/5evrq1KlSilnzpwym836888/raNYp7hw4YL27dsno9HoUDd5V/Dy8lLz5s116tQpfffdd/Ly8tIzzzxjs07KTYnt27fbdKeNjY3V+++/b+3V4egxce+jIKGhoZo1a5bd+inHxG+//WaTvF+6dEmvvvqq9Thx9phImXv9k08+sc6ZLCUf66NHj9a5c+f05JNPPvBxmMxgsVi0Zs0a6yjX7733nk1LdcpgeB9++KFN7GazWZMnT9b27dt1/vz5ByZlKZ/X3a3S6T0XpWf/v5+iRYsqMDBQ586ds2vF9vX1Ve3atXX9+nXNnDnT5v2OHz9eR44c0b/+9S9Vr17dWpZy3vn3v/+tc+fOWZdfv35d77zzjpKSktSnT5/7Dqg4bdo0GY1GDRgwwObzyZs3r/bs2WP93FJ6GaTWOyg1/fv3V4kSJbRjxw7179/fJj4pef8dNWqU1q1bJ09PT40aNcqhelOk3ARJrbUfgC26iANwidKlS2vq1KkaNmyYFi9erB9++EEVKlRQ/vz5defOHR05ckTXr19X3rx59cUXX9i0Hg0bNkzHjx/Xjh071LRpU4WEhMjb21v79+9XZGSkSpQokeFnlB/k77//dihZatasmd20Qm3bttW0adMUFhamypUr2w2Ck5JcHTt2TM2bN7desB06dEjh4eEqU6aM/vrrL12/fv2+r12sWDE1a9ZMv/76q7p06WId7Xfnzp0KCgpS6dKlrS1wkvTcc89pwYIF2rJli5o1a6by5csrKipK+/bt0507d/TEE0/o5MmTNq9bpEgRmUwm/fnnn+rVq5eCg4P1zjvvpBpPhQoV9Pbbb+vjjz9Wnz59VLlyZRUoUEAnTpzQmTNnlDt3bn322WdpDviUEc8884x69OihBQsWqE+fPqpYsaLy5Mmjffv2WfeZL7/8UlOmTNGqVavUvXt3LVy48IFJZt68ebVo0SK9/PLLWrVqldasWWPdjyMiInTgwAHFx8erYMGCmjFjxn27p97NYDDo008/Va9evfTdd99p06ZNqlSpkiIjI7Vnzx4lJSWpX79+1n3LZDJpzJgxGjx4sIYOHary5curSJEiunHjhvbu3aukpCS9//771s929OjReuGFFzRu3DgtWbJEpUuXVlRUlPbu3au4uDgNGDDAZgooR+zZs8ehY6Jr1642SYiUfEz88MMPCgsLU8uWLe32gRYtWmjatGn6888/9fTTT6ty5cqKj4/X/v37dfv2bYePiVq1aqlChQo6cuSIWrdurVq1aikmJka7d+9WgwYN7Hp2vPDCC1qzZo2+//577du3T2XKlFFERIT2798vi8WikiVL6vTp0zavm/LM8aZNm9S/f39VqVLFJjm629NPP62XXnpJc+fO1XPPPadq1aopT548OnjwoMLCwlS4cGFNmjTJ+rhHZrhx44bN381sNuvWrVs6ceKErl27JpPJpPfff19NmzZNM/ZOnTqpfPnyypcvn06cOKHz58/L29tbU6ZMeWD38eLFi+vUqVPW/fbNN99M97kovfv//TRu3FhLlizR7t277Z4v/+CDD9S+fXtNnjxZa9euVbFixXTkyBFdvHhRpUqV0vjx423Wf+mll7R7925t2bJFrVq1UvXq1eXh4aGdO3cqJiZGzZs3V79+/dKMJeUxqkGDBtmNAdKzZ099+umnatu2rYKCgrRlyxblzJlT3bt3f+B7lJJvbC1cuFD9+vXTli1b1Lx5c5UrV06FCxdWdHS09u3bp5iYGOXOnVuTJk1KV+u19L9ZI+6eHQNA6kiwAbhMgwYNtHbtWi1atEhbt27V2bNndeTIEXl7e6tYsWLq1q2bevToYff8lpeXl+bOnavvvvtOq1atsnZBLFKkiHr06KGePXtm+jNf4eHhaQ7+dLfixYvbJdhFixZV1apVtXfvXruBnKTki8Wvv/5aM2bM0MaNG7Vt2zblzp1bJUqU0JAhQ9SxY0fVrVtXhw4dst6ESMvEiRM1a9YsrVq1Stu3b1fevHn1wgsv6NVXX7WbuqZIkSJaunSpPvvsMx04cEC///678uXLpzp16qhHjx7KnTu3OnTooN9++01vvPGGpOQumB9//LGmTZumvXv36tKlS2km2JLUo0cPPfnkk5ozZ4727dunY8eOqUCBAurVq5f69OnjcALqjPfe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",
      "text/plain": [
       "<Figure size 1000x600 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\n",
      "\ud83d\udcca Visualization saved: intra_model_bias_visualization_revised.png\n"
     ]
    }
   ],
   "source": [
    "# INTRA-MODEL BIAS & WITHIN-MODEL CONSISTENCY ANALYSIS \u2014 REVISED\nimport pandas as pd\nimport numpy as np\nfrom scipy.stats import ttest_ind, ttest_rel, spearmanr, levene\nimport matplotlib.pyplot as plt\nfrom statsmodels.stats.multitest import multipletests\nimport warnings\nwarnings.filterwarnings('ignore')\n\nnp.set_printoptions(suppress=True, floatmode=\"fixed\")\n\nprint(\"=\" * 80)\nprint(\"INTRA-MODEL BIAS & WITHIN-MODEL CONSISTENCY ANALYSIS (Revised)\")\nprint(\"=\" * 80)\n\n# ------------------------------------------------------------------\n# Load data\n# ------------------------------------------------------------------\nclaude_df = pd.read_csv('/Users/jianzhouyao/AI4Good/data/processed/ratings/claude_with_ratings.csv')\ngpt_df    = pd.read_csv('/Users/jianzhouyao/AI4Good/data/processed/ratings/gpt_with_ratings.csv')\n\n# Add source labels (not strictly needed but handy)\nclaude_df['Response_Source'] = 'Claude'\ngpt_df['Response_Source'] = 'GPT'\n\n# Create age groups\nfor df in (claude_df, gpt_df):\n    if 'age' in df.columns:\n        df['age_group'] = pd.cut(\n            df['age'],\n            bins=[0, 18, 50, 65, 100],\n            labels=['<18', '18-49', '50-64', '65+'],\n            right=False\n        )\n\nprint(\"\ud83d\udcca DATA OVERVIEW\")\nprint(\"-\" * 40)\nprint(f\"Claude responses: {len(claude_df)} rows\")\nprint(f\"GPT responses: {len(gpt_df)} rows\")\n\n# ------------------------------------------------------------------\n# Utilities\n# ------------------------------------------------------------------\ndef mean_ci(x):\n    x = np.asarray(x, dtype=float)\n    x = x[~np.isnan(x)]\n    n  = len(x)\n    if n == 0:\n        return np.nan, (np.nan, np.nan)\n    m  = np.mean(x)\n    s  = np.std(x, ddof=1) if n > 1 else 0.0\n    se = s / np.sqrt(n) if n > 0 else np.nan\n    ci = (m - 1.96*se, m + 1.96*se) if n > 1 else (m, m)\n    return m, ci\n\ndef paired_test(series_a, series_b):\n    \"\"\"Aligned paired t-test + Cohen's dz + CI for the difference.\"\"\"\n    # Drop NaNs and align index\n    df = pd.concat([series_a, series_b], axis=1, keys=['a', 'b']).dropna()\n    a = df['a'].values\n    b = df['b'].values\n    n = len(df)\n    if n < 3:\n        return dict(n=n, diff=np.nan, p=np.nan, dz=np.nan, ci=(np.nan, np.nan))\n    d = a - b\n    m = np.mean(d)\n    s = np.std(d, ddof=1)\n    se = s / np.sqrt(n)\n    ci = (m - 1.96*se, m + 1.96*se)\n    t_stat, p_val = ttest_rel(a, b)\n    dz = m / s if s > 0 else np.nan\n    return dict(n=n, diff=m, p=p_val, dz=dz, ci=ci)\n\ndef welch_test(a, b):\n    \"\"\"Welch's t-test + Cohen's d (Hedges g-ish pooled via Welch not exact) + CI for diff.\"\"\"\n    a = np.asarray(a, dtype=float); a = a[~np.isnan(a)]\n    b = np.asarray(b, dtype=float); b = b[~np.isnan(b)]\n    n1, n2 = len(a), len(b)\n    if n1 < 3 or n2 < 3:\n        return dict(n1=n1, n2=n2, diff=np.nan, p=np.nan, d=np.nan, ci=(np.nan, np.nan))\n    t_stat, p_val = ttest_ind(a, b, equal_var=False)\n    diff = a.mean() - b.mean()\n    # CI (Welch): use standard error via sample variances\n    s1, s2 = a.var(ddof=1), b.var(ddof=1)\n    se = np.sqrt(s1/n1 + s2/n2)\n    ci = (diff - 1.96*se, diff + 1.96*se)\n    # Effect size (Glass's delta-ish using pooled SD for reference)\n    sp = np.sqrt(((n1-1)*s1 + (n2-1)*s2) / (n1+n2-2)) if (n1+n2-2)>0 else np.nan\n    d = diff / sp if sp and sp>0 else np.nan\n    return dict(n1=n1, n2=n2, diff=diff, p=p_val, d=d, ci=ci)\n\ndef iqr(x):\n    x = pd.Series(x).dropna()\n    return x.quantile(0.75) - x.quantile(0.25) if len(x) else np.nan\n\ndef align_on_prompt(a_df, a_col, b_df, b_col, key='Prompt Number', subset_mask_a=None, subset_mask_b=None):\n    \"\"\"Return two aligned Series on the same key (e.g., Prompt Number).\"\"\"\n    A = a_df if subset_mask_a is None else a_df[subset_mask_a]\n    B = b_df if subset_mask_b is None else b_df[subset_mask_b]\n    A = A[[key, a_col]].dropna().set_index(key).sort_index()\n    B = B[[key, b_col]].dropna().set_index(key).sort_index()\n    joined = A.join(B, how='inner', lsuffix='_A', rsuffix='_B')\n    return joined.iloc[:, 0], joined.iloc[:, 1]  # two aligned Series\n\n# ------------------------------------------------------------------\n# 1) SELF-EVALUATION BIAS (paired by prompt)\n# ------------------------------------------------------------------\nprint(\"\\n\" + \"=\" * 80)\nprint(\"1. SELF-EVALUATION BIAS ANALYSIS (Paired by Prompt)\")\nprint(\"=\" * 80)\n\nresults = {}\n\n# Affective (GPT rater): GPT\u2192GPT vs GPT\u2192Claude\na_gpt_gpt, b_gpt_claude = align_on_prompt(\n    gpt_df,    'Affective Empathy Score (GPT)',\n    claude_df, 'Affective Empathy Score (GPT)'\n)\ngpt_aff = paired_test(a_gpt_gpt, b_gpt_claude)\nprint(\"\\n\ud83d\udd0d GPT Self-Evaluation (Affective):\")\nprint(f\"  GPT rating GPT vs GPT rating Claude (n={gpt_aff['n']}): \"\n      f\"\u0394={gpt_aff['diff']:+.3f}, 95%CI=({gpt_aff['ci'][0]:.3f},{gpt_aff['ci'][1]:.3f}), \"\n      f\"p={gpt_aff['p']:.4g}, dz={gpt_aff['dz']:.3f}\")\nresults['gpt_self_bias'] = gpt_aff['diff']\nresults['gpt_self_bias_p'] = gpt_aff['p']\nresults['gpt_self_bias_dz'] = gpt_aff['dz']\n\n# Affective (Claude rater): Claude\u2192Claude vs Claude\u2192GPT\na_cl_claude, b_cl_gpt = align_on_prompt(\n    claude_df, 'Affective Empathy Score (Claude)',\n    gpt_df,    'Affective Empathy Score (Claude)'\n)\ncl_aff = paired_test(a_cl_claude, b_cl_gpt)\nprint(\"\\n\ud83d\udd0d Claude Self-Evaluation (Affective):\")\nprint(f\"  Claude rating Claude vs Claude rating GPT (n={cl_aff['n']}): \"\n      f\"\u0394={cl_aff['diff']:+.3f}, 95%CI=({cl_aff['ci'][0]:.3f},{cl_aff['ci'][1]:.3f}), \"\n      f\"p={cl_aff['p']:.4g}, dz={cl_aff['dz']:.3f}\")\nresults['claude_self_bias'] = cl_aff['diff']\nresults['claude_self_bias_p'] = cl_aff['p']\nresults['claude_self_bias_dz'] = cl_aff['dz']\n\n# Cognitive (GPT)\na_gpt_gpt_cog, b_gpt_claude_cog = align_on_prompt(\n    gpt_df,    'Cognitive Empathy Score (GPT)',\n    claude_df, 'Cognitive Empathy Score (GPT)'\n)\ngpt_cog = paired_test(a_gpt_gpt_cog, b_gpt_claude_cog)\nprint(\"\\n\ud83e\udde0 GPT Self-Evaluation (Cognitive):\")\nprint(f\"  \u0394={gpt_cog['diff']:+.3f}, 95%CI=({gpt_cog['ci'][0]:.3f},{gpt_cog['ci'][1]:.3f}), \"\n      f\"p={gpt_cog['p']:.4g}, dz={gpt_cog['dz']:.3f}\")\n\n# Cognitive (Claude)\na_cl_claude_cog, b_cl_gpt_cog = align_on_prompt(\n    claude_df, 'Cognitive Empathy Score (Claude)',\n    gpt_df,    'Cognitive Empathy Score (Claude)'\n)\ncl_cog = paired_test(a_cl_claude_cog, b_cl_gpt_cog)\nprint(\"\\n\ud83e\udde0 Claude Self-Evaluation (Cognitive):\")\nprint(f\"  \u0394={cl_cog['diff']:+.3f}, 95%CI=({cl_cog['ci'][0]:.3f},{cl_cog['ci'][1]:.3f}), \"\n      f\"p={cl_cog['p']:.4g}, dz={cl_cog['dz']:.3f}\")\n\n# ------------------------------------------------------------------\n# 2) WITHIN-MODEL CONSISTENCY (SD, IQR; Levene test for variances)\n# ------------------------------------------------------------------\nprint(\"\\n\" + \"=\" * 80)\nprint(\"2. WITHIN-MODEL CONSISTENCY ANALYSIS (SD, IQR, Levene)\")\nprint(\"=\" * 80)\n\ndef consistency_summary(series):\n    s = pd.Series(series).dropna()\n    return dict(n=len(s), mean=s.mean(), sd=s.std(ddof=1), iqr=iqr(s))\n\nprint(\"\\n\ud83d\udccf GPT rating distributions (Affective):\")\ngpt_on_gpt = gpt_df['Affective Empathy Score (GPT)']\ngpt_on_cla = claude_df['Affective Empathy Score (GPT)']\ngpt_self_sum = consistency_summary(gpt_on_gpt)\ngpt_other_sum = consistency_summary(gpt_on_cla)\nprint(f\"  GPT\u2192GPT:   n={gpt_self_sum['n']}, mean={gpt_self_sum['mean']:.3f}, SD={gpt_self_sum['sd']:.3f}, IQR={gpt_self_sum['iqr']:.3f}\")\nprint(f\"  GPT\u2192Claude:n={gpt_other_sum['n']}, mean={gpt_other_sum['mean']:.3f}, SD={gpt_other_sum['sd']:.3f}, IQR={gpt_other_sum['iqr']:.3f}\")\n# Levene (Brown\u2013Forsythe via median)\nW_gpt, p_gpt = levene(gpt_on_gpt.dropna(), gpt_on_cla.dropna(), center='median')\nprint(f\"  Variance difference test (Levene, median): W={W_gpt:.3f}, p={p_gpt:.4g}\")\nresults['gpt_levene_p'] = p_gpt\n\nprint(\"\\n\ud83d\udd0d Claude rating distributions (Affective):\")\ncla_on_cla = claude_df['Affective Empathy Score (Claude)']\ncla_on_gpt = gpt_df['Affective Empathy Score (Claude)']\ncla_self_sum = consistency_summary(cla_on_cla)\ncla_other_sum = consistency_summary(cla_on_gpt)\nprint(f\"  Claude\u2192Claude: n={cla_self_sum['n']}, mean={cla_self_sum['mean']:.3f}, SD={cla_self_sum['sd']:.3f}, IQR={cla_self_sum['iqr']:.3f}\")\nprint(f\"  Claude\u2192GPT:    n={cla_other_sum['n']}, mean={cla_other_sum['mean']:.3f}, SD={cla_other_sum['sd']:.3f}, IQR={cla_other_sum['iqr']:.3f}\")\nW_cla, p_cla = levene(cla_on_cla.dropna(), cla_on_gpt.dropna(), center='median')\nprint(f\"  Variance difference test (Levene, median): W={W_cla:.3f}, p={p_cla:.4g}\")\nresults['claude_levene_p'] = p_cla\n\n# ------------------------------------------------------------------\n# 3) DEMOGRAPHIC-SPECIFIC SELF-BIAS (paired by prompt) + FDR\n# ------------------------------------------------------------------\nprint(\"\\n\" + \"=\" * 80)\nprint(\"3. DEMOGRAPHIC-SPECIFIC SELF-BIAS (Paired; BH-FDR correction)\")\nprint(\"=\" * 80)\n\ndemo_bias_results = {}\nalpha = 0.05\nmin_pairs = 10  # stricter threshold than 3\n\ndef paired_bias_by_group(model_name, rater_col, own_df, other_df, demo_col):\n    \"\"\"\n    For each group in demo_col: compare rater_col on own_df vs other_df,\n    paired by prompt within that group.\n    \"\"\"\n    pvals, labels, diffs, ns = [], [], [], []\n    per_group = {}\n    groups = sorted([g for g in pd.concat([own_df[demo_col], other_df[demo_col]]).dropna().unique()])\n    for gval in groups:\n        mask_own = own_df[demo_col] == gval\n        mask_oth = other_df[demo_col] == gval\n        a, b = align_on_prompt(own_df, rater_col, other_df, rater_col,\n                               subset_mask_a=mask_own, subset_mask_b=mask_oth)\n        pt = paired_test(a, b)\n        if pt['n'] >= min_pairs:\n            per_group[gval] = pt\n            pvals.append(pt['p'])\n            labels.append(str(gval))\n            diffs.append(pt['diff'])\n            ns.append(pt['n'])\n        # silently skip groups with too-small n\n    # FDR\n    if len(pvals):\n        reject, p_corr, _, _ = multipletests(pvals, alpha=alpha, method='fdr_bh')\n        print(f\"\\n  {model_name} \u2014 {demo_col} (k={len(pvals)} groups; FDR-BH \u03b1={alpha}):\")\n        for lbl, raw_p, adj_p, rj, df_, n_ in zip(labels, pvals, p_corr, reject, diffs, ns):\n            star = \"\u2713\" if rj else \" \"\n            print(f\"    {lbl:>25}: \u0394={df_:+.3f}, n={n_:>3}, p={raw_p:.4g}, p_FDR={adj_p:.4g} {star}\")\n        # store summary\n        demo_bias_results[f\"{model_name}_{demo_col}_labels\"] = labels\n        demo_bias_results[f\"{model_name}_{demo_col}_pvals_raw\"] = pvals\n        demo_bias_results[f\"{model_name}_{demo_col}_pvals_fdr\"] = p_corr.tolist()\n        demo_bias_results[f\"{model_name}_{demo_col}_diffs\"] = diffs\n        demo_bias_results[f\"{model_name}_{demo_col}_ns\"] = ns\n\nfor demo_col in ['gender', 'ethnicity', 'education', 'diagnosis', 'age_group']:\n    if demo_col in gpt_df.columns and demo_col in claude_df.columns:\n        # GPT rater self-bias within groups (Affective)\n        paired_bias_by_group(\"GPT_rater_affective\",\n                             'Affective Empathy Score (GPT)',\n                             gpt_df, claude_df, demo_col)\n        # Claude rater self-bias within groups (Affective)\n        paired_bias_by_group(\"Claude_rater_affective\",\n                             'Affective Empathy Score (Claude)',\n                             claude_df, gpt_df, demo_col)\n\n# ------------------------------------------------------------------\n# 4) CROSS-EVALUATION ASYMMETRY (paired by prompt) + Spearman\n# ------------------------------------------------------------------\nprint(\"\\n\" + \"=\" * 80)\nprint(\"4. CROSS-EVALUATION ASYMMETRY (Paired; Spearman agreement)\")\nprint(\"=\" * 80)\n\n# Pair: GPT rating Claude vs Claude rating GPT on the same prompts\ngpt_rates_claude, claude_rates_gpt = align_on_prompt(\n    claude_df, 'Affective Empathy Score (GPT)',\n    gpt_df,    'Affective Empathy Score (Claude)'\n)\ncross = paired_test(gpt_rates_claude, claude_rates_gpt)\nprint(\"\\n\u2696\ufe0f Cross-evaluation asymmetry (Affective):\")\nprint(f\"  GPT\u2192Claude vs Claude\u2192GPT (n={cross['n']}): \"\n      f\"\u0394={cross['diff']:+.3f}, 95%CI=({cross['ci'][0]:.3f},{cross['ci'][1]:.3f}), \"\n      f\"p={cross['p']:.4g}, dz={cross['dz']:.3f}\")\n\n# Rank agreement on which prompts deserve higher empathy\nif len(gpt_rates_claude) >= 3 and len(claude_rates_gpt) >= 3:\n    rho, p_rho = spearmanr(gpt_rates_claude, claude_rates_gpt)\n    print(f\"  Cross-rater rank agreement (Spearman): \u03c1={rho:.3f}, p={p_rho:.4g}\")\n    results['cross_spearman_rho'] = rho\n    results['cross_spearman_p'] = p_rho\n\n# ------------------------------------------------------------------\n# 5) RESPONSE\u2013RATER INTERACTIONS (paired by prompt where sensible)\n# ------------------------------------------------------------------\nprint(\"\\n\" + \"=\" * 80)\nprint(\"5. RESPONSE\u2013RATER INTERACTION EFFECTS (Paired)\")\nprint(\"=\" * 80)\n\n# Build aligned series for four cells\ncc, cg = align_on_prompt(claude_df, 'Affective Empathy Score (Claude)', gpt_df, 'Affective Empathy Score (Claude)')  # Claude\u2192Claude vs Claude\u2192GPT\ngg, gc = align_on_prompt(gpt_df,    'Affective Empathy Score (GPT)',    claude_df, 'Affective Empathy Score (GPT)')  # GPT\u2192GPT vs GPT\u2192Claude\n\nprint(\"\\n\ud83d\udcca Summary means \u00b1 SD (unpaired descriptive):\")\ndef msd(x): \n    x = pd.Series(x).dropna(); \n    return f\"{x.mean():.3f} \u00b1 {x.std(ddof=1):.3f} (n={len(x)})\"\n\nprint(f\"  Claude\u2192Claude: {msd(cc)}\")\nprint(f\"  Claude\u2192GPT:    {msd(cg)}\")\nprint(f\"  GPT\u2192GPT:       {msd(gg)}\")\nprint(f\"  GPT\u2192Claude:    {msd(gc)}\")\n\nprint(\"\\n\ud83d\udd0d Paired contrasts by prompt:\")\npt1 = paired_test(cc, cg)\nprint(f\"  Claude self vs other:        \u0394={pt1['diff']:+.3f}, p={pt1['p']:.4g}, dz={pt1['dz']:.3f}, n={pt1['n']}\")\npt2 = paired_test(gg, gc)\nprint(f\"  GPT self vs other:           \u0394={pt2['diff']:+.3f}, p={pt2['p']:.4g}, dz={pt2['dz']:.3f}, n={pt2['n']}\")\n# Self (Claude) vs Self (GPT): pair by prompt using cc vs gg\npt3 = paired_test(cc, gg)\nprint(f\"  Self-eval: Claude vs GPT:    \u0394={pt3['diff']:+.3f}, p={pt3['p']:.4g}, dz={pt3['dz']:.3f}, n={pt3['n']}\")\n# Cross asymmetry already computed as cross\n\n# ------------------------------------------------------------------\n# 6) FINAL SUMMARY + SAVE\n# ------------------------------------------------------------------\nprint(\"\\n\" + \"=\" * 80)\nprint(\"6. COMPREHENSIVE INTRA-MODEL BIAS SUMMARY\")\nprint(\"=\" * 80)\n\nall_results = {\n    **results,\n    'gpt_self_bias': results.get('gpt_self_bias', np.nan),\n    'gpt_self_bias_p': results.get('gpt_self_bias_p', np.nan),\n    'gpt_self_bias_dz': results.get('gpt_self_bias_dz', np.nan),\n    'claude_self_bias': results.get('claude_self_bias', np.nan),\n    'claude_self_bias_p': results.get('claude_self_bias_p', np.nan),\n    'claude_self_bias_dz': results.get('claude_self_bias_dz', np.nan),\n    'levene_p_gpt': results.get('gpt_levene_p', np.nan),\n    'levene_p_claude': results.get('claude_levene_p', np.nan)\n}\n\n# Flatten selected demographic summaries into a compact table (optional)\ndemo_rows = []\nfor k, v in demo_bias_results.items():\n    if k.endswith('_labels'):\n        base = k[:-7]  # strip _labels\n        labels = demo_bias_results[k]\n        raw   = demo_bias_results.get(base+'_pvals_raw', [])\n        fdr   = demo_bias_results.get(base+'_pvals_fdr', [])\n        diffs = demo_bias_results.get(base+'_diffs', [])\n        ns    = demo_bias_results.get(base+'_ns', [])\n        for lbl, p_raw, p_adj, dff, n_ in zip(labels, raw, fdr, diffs, ns):\n            demo_rows.append(dict(family=base, group=lbl, n=n_, diff=dff, p_raw=p_raw, p_fdr=p_adj))\ndemo_df = pd.DataFrame(demo_rows)\ndemo_df.to_csv('data/results/bias_analysis/demographic_self_bias_fdr.csv', index=False)\n\n# Save top-level results\npd.DataFrame([all_results]).to_csv('data/results/bias_analysis/intra_model_bias_results_revised.csv', index=False)\n\nprint(\"\\n\ud83d\udd11 KEY FINDINGS (Affective, paired):\")\nprint(f\"  \u2022 GPT self-evaluation bias \u0394 (GPT\u2192GPT \u2013 GPT\u2192Claude): {gpt_aff['diff']:+.3f} \"\n      f\"(p={gpt_aff['p']:.4g}, dz={gpt_aff['dz']:.3f})\")\nprint(f\"  \u2022 Claude self-evaluation bias \u0394 (Claude\u2192Claude \u2013 Claude\u2192GPT): {cl_aff['diff']:+.3f} \"\n      f\"(p={cl_aff['p']:.4g}, dz={cl_aff['dz']:.3f})\")\nprint(f\"  \u2022 Cross-eval asymmetry \u0394 (GPT\u2192Claude \u2013 Claude\u2192GPT): {cross['diff']:+.3f} \"\n      f\"(p={cross['p']:.4g}, dz={cross['dz']:.3f})\")\nprint(\"\\n\ud83d\udcbe Saved:\")\nprint(\"  \u2022 data/results/bias_analysis/intra_model_bias_results_revised.csv\")\nprint(\"  \u2022 data/results/bias_analysis/demographic_self_bias_fdr.csv\")\n\n# ------------------------------------------------------------------\n# Visualization (simple, consistent)\n# ------------------------------------------------------------------\nplt.figure(figsize=(10, 6))\n\n# Bar 1: self-bias paired deltas\nmodels = ['GPT', 'Claude', 'Cross']\ndeltas = [gpt_aff['diff'], cl_aff['diff'], cross['diff']]\nyerr   = [ (gpt_aff['diff']-gpt_aff['ci'][0], gpt_aff['ci'][1]-gpt_aff['diff']),\n           (cl_aff['diff']-cl_aff['ci'][0], cl_aff['ci'][1]-cl_aff['diff']),\n           (cross['diff']-cross['ci'][0],   cross['ci'][1]-cross['diff']) ]\n# Matplotlib expects asymmetric errors as 2xN array\nyerr_arr = np.array([[e[0] for e in yerr], [e[1] for e in yerr]])\n\nbars = plt.bar(models, deltas, alpha=0.8)\nplt.errorbar(models, deltas, yerr=yerr_arr, fmt='none', capsize=6, linewidth=1.5)\nplt.axhline(0, linestyle='--', linewidth=1)\nplt.ylabel('Paired \u0394 (Own \u2212 Other) or (GPT\u2192Claude \u2212 Claude\u2192GPT)')\nplt.title('Self-Evaluation & Cross-Evaluation Deltas (95% CI)')\nplt.tight_layout()\nplt.savefig('outputs/figures/bias/intra_model_bias_visualization.png', dpi=300, bbox_inches='tight')\nplt.show()\n\nprint(\"\\n\ud83d\udcca Visualization saved: outputs/figures/bias/intra_model_bias_visualization.png\")\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 28,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\u2705 Saved: /Users/jianzhouyao/AI4Good/Scoring_Charts/self_cross_deltas.png\n"
     ]
    }
   ],
   "source": [
    "# ==============================\n# Self- vs Cross-Evaluation Deltas (95% CI)\n# Consistent style + viridis palette + no title\n# ==============================\n\nimport numpy as np\nimport matplotlib.pyplot as plt\nimport seaborn as sns\n\n# ==============================\n# Global plotting style (same as your other plots)\n# ==============================\nsns.set(style=\"whitegrid\")\nsns.set_context(\"talk\")  # base context\nplt.rcParams.update({\n    \"font.size\": 18,\n    \"axes.labelsize\": 18,\n    \"xtick.labelsize\": 18,\n    \"ytick.labelsize\": 18,\n    \"legend.fontsize\": 16,\n    \"figure.constrained_layout.use\": True,\n})\n\ncustom_palette = sns.color_palette(\"viridis\", 4)  # consistent palette\nsns.set_palette(custom_palette)\n\nFIGSIZE = (10, 6)\nDPI = 300\n\n# ==============================\n# Helper for CIs\n# ==============================\ndef mean_ci_95(x):\n    x = np.asarray(x, dtype=float)\n    x = x[~np.isnan(x)]\n    n = len(x)\n    if n == 0:\n        return np.nan, (np.nan, np.nan)\n    m  = x.mean()\n    sd = x.std(ddof=1) if n > 1 else 0.0\n    se = sd / np.sqrt(n) if n > 0 else np.nan\n    ci = (m - 1.96 * se, m + 1.96 * se) if n > 1 else (m, m)\n    return m, ci\n\ndef ci_to_err(mean, ci):\n    return (mean - ci[0], ci[1] - mean)\n\n# ==============================\n# Build paired deltas by prompt\n# (Assumes aff_pivot already built as in previous code)\n# ==============================\npair_gpt   = aff_pivot[[GP_GP, CL_GP]].dropna()\npair_cla   = aff_pivot[[CL_CL, GP_CL]].dropna()\npair_cross = aff_pivot[[CL_GP, GP_CL]].dropna()\n\ndelta_gpt   = (pair_gpt[GP_GP] - pair_gpt[CL_GP]).values\ndelta_cla   = (pair_cla[CL_CL] - pair_cla[GP_CL]).values\ndelta_cross = (pair_cross[CL_GP] - pair_cross[GP_CL]).values\n\nm_gpt, ci_gpt     = mean_ci_95(delta_gpt)\nm_cla, ci_cla     = mean_ci_95(delta_cla)\nm_cross, ci_cross = mean_ci_95(delta_cross)\n\nerr_gpt   = ci_to_err(m_gpt, ci_gpt)\nerr_cla   = ci_to_err(m_cla, ci_cla)\nerr_cross = ci_to_err(m_cross, ci_cross)\nyerr = np.array([[err_gpt[0], err_cla[0], err_cross[0]],\n                 [err_gpt[1], err_cla[1], err_cross[1]]])\n\n# ==============================\n# Plot\n# ==============================\nlabels = [\"GPT\", \"Claude\", \"Cross\"]\nheights = [m_gpt, m_cla, m_cross]\n\nplt.figure(figsize=FIGSIZE)\nax = plt.gca()\n\nbars = ax.bar(labels, heights, alpha=0.9)\nax.errorbar(labels, heights, yerr=yerr, fmt='none', capsize=6, linewidth=1.8, color=\"black\")\n\nax.axhline(0, linestyle=\"--\", linewidth=1.2, alpha=0.7, color=\"black\")\nax.set_ylabel(\"Paired \u0394 (Own \u2212 Other)\\n(GPT\u2192Claude \u2212 Claude\u2192GPT for Cross)\")\n\n\nplt.tight_layout()\nplt.savefig(output_folder / \"outputs/figures/bias/self_cross_deltas.png\", dpi=DPI, bbox_inches=\"tight\")\nplt.savefig(output_folder / \"self_cross_deltas.pdf\", bbox_inches=\"tight\")\nplt.close()\n\nprint(\"\u2705 Saved:\", output_folder / \"outputs/figures/bias/self_cross_deltas.png\")\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 15,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\u2705 Saved 4 readability figures with 95% CIs to: /Users/jianzhouyao/AI4Good/Scoring_Charts\n"
     ]
    }
   ],
   "source": [
    "# =============================================================================\n",
    "# READABILITY METRICS \u2014 MEAN \u00b1 95% CI (viridis style, same layout)\n",
    "# =============================================================================\n",
    "\n",
    "import numpy as np\n",
    "import matplotlib.pyplot as plt\n",
    "from matplotlib.lines import Line2D\n",
    "import seaborn as sns\n",
    "import scipy.stats as st\n",
    "\n",
    "# --- Global style (same as your empathy plots) ---\n",
    "sns.set(style=\"whitegrid\")\n",
    "sns.set_context(\"talk\", font_scale=1.3)\n",
    "\n",
    "plt.rcParams.update({\n",
    "    \"font.size\": 26,\n",
    "    \"axes.labelsize\": 26,\n",
    "    \"xtick.labelsize\": 26,\n",
    "    \"ytick.labelsize\": 26,\n",
    "    \"legend.fontsize\": 22,\n",
    "    \"figure.constrained_layout.use\": True,\n",
    "})\n",
    "\n",
    "custom_palette = sns.color_palette(\"viridis\", 4)\n",
    "sns.set_palette(custom_palette)\n",
    "\n",
    "FIGSIZE = (10, 6)\n",
    "DPI = 300\n",
    "XTICK_ROT = 15\n",
    "\n",
    "# Helper: compute mean \u00b1 95% CI (t-based)\n",
    "def mean_ci(values, confidence=0.95):\n",
    "    arr = np.array(values, dtype=float)\n",
    "    mean = np.mean(arr)\n",
    "    sem = st.sem(arr)\n",
    "    h = sem * st.t.ppf((1 + confidence) / 2., len(arr) - 1)\n",
    "    return mean, h\n",
    "\n",
    "# Helper: place legend outside\n",
    "def place_legend_outside(ax, title=None, loc=\"upper left\"):\n",
    "    ax.legend(title=title, bbox_to_anchor=(1.02, 1), loc=loc, borderaxespad=0)\n",
    "\n",
    "width = 0.35\n",
    "\n",
    "# Colors from viridis palette\n",
    "color_claude = custom_palette[1]\n",
    "color_gpt    = custom_palette[2]\n",
    "\n",
    "# =============================================================================\n",
    "# Example data: replace each list with your per-response readability scores\n",
    "# =============================================================================\n",
    "# These are placeholder arrays \u2014 replace with your real data arrays\n",
    "# e.g., Flesch\u2013Kincaid scores for 156 GPT responses etc.\n",
    "flesch_gpt   = np.random.normal(9.72, 0.6, 156)\n",
    "flesch_cla   = np.random.normal(9.54, 0.6, 156)\n",
    "smog_gpt     = np.random.normal(12.59, 0.5, 156)\n",
    "smog_cla     = np.random.normal(12.13, 0.5, 156)\n",
    "fog_gpt      = np.random.normal(11.86, 0.5, 156)\n",
    "fog_cla      = np.random.normal(11.70, 0.5, 156)\n",
    "\n",
    "# =============================================================================\n",
    "# Plot 1: Overall Metric Comparison\n",
    "# =============================================================================\n",
    "metrics = [\"Flesch-Kincaid\", \"SMOG Index\", \"Gunning Fog\"]\n",
    "gpt_data = [flesch_gpt, smog_gpt, fog_gpt]\n",
    "cla_data = [flesch_cla, smog_cla, fog_cla]\n",
    "\n",
    "gpt_means, gpt_errs = zip(*[mean_ci(d) for d in gpt_data])\n",
    "cla_means, cla_errs = zip(*[mean_ci(d) for d in cla_data])\n",
    "\n",
    "scale_limits = [12, 18, 20]\n",
    "x = np.arange(len(metrics))\n",
    "fig, ax = plt.subplots(figsize=FIGSIZE)\n",
    "\n",
    "bars_cla = ax.bar(x - width/2, cla_means, width, yerr=cla_errs, capsize=5,\n",
    "                  label='Claude', color=color_claude)\n",
    "bars_gpt = ax.bar(x + width/2, gpt_means, width, yerr=gpt_errs, capsize=5,\n",
    "                  label='GPT', color=color_gpt)\n",
    "\n",
    "def draw_horizontal_stripes(ax, x_center, y_level, total_width, n_stripes=7, stripe_len_frac=0.25):\n",
    "    spacing = total_width / n_stripes\n",
    "    start_x = x_center - total_width / 2\n",
    "    stripe_len = stripe_len_frac * spacing\n",
    "    for i in range(n_stripes):\n",
    "        x_start = start_x + i * spacing\n",
    "        x_end = x_start + stripe_len\n",
    "        ax.hlines(y=y_level, xmin=x_start, xmax=x_end, color='black', linewidth=1)\n",
    "\n",
    "for i, limit in enumerate(scale_limits):\n",
    "    draw_horizontal_stripes(ax, x[i] - width/2, limit, width)\n",
    "    draw_horizontal_stripes(ax, x[i] + width/2, limit, width)\n",
    "\n",
    "# Value labels\n",
    "for bars in [bars_cla, bars_gpt]:\n",
    "    for bar in bars:\n",
    "        ax.annotate(f'{bar.get_height():.2f}',\n",
    "                    xy=(bar.get_x() + bar.get_width()/2, bar.get_height()),\n",
    "                    xytext=(0, 3), textcoords=\"offset points\",\n",
    "                    ha='center', va='bottom', fontsize=14)\n",
    "\n",
    "ax.set_ylabel('Grade Level')\n",
    "ax.set_xticks(x)\n",
    "ax.set_xticklabels(metrics, rotation=XTICK_ROT, ha='right')\n",
    "\n",
    "scale_line = Line2D([0], [0], color='black', linewidth=1, label='Scale Limit')\n",
    "handles = [bars_cla, bars_gpt, scale_line]\n",
    "labels  = ['Claude', 'GPT', 'Scale Limit']\n",
    "ax.legend(handles=handles, labels=labels, bbox_to_anchor=(1.02, 1),\n",
    "          loc=\"upper left\", borderaxespad=0)\n",
    "\n",
    "plt.tight_layout()\n",
    "plt.savefig(output_folder / \"readability_overall_metrics.png\", dpi=DPI, bbox_inches=\"tight\")\n",
    "plt.close()\n",
    "\n",
    "# =============================================================================\n",
    "# Plot 2: Education Level \u2014 Flesch-Kincaid\n",
    "# =============================================================================\n",
    "# Replace with your arrays per education level\n",
    "edu_levels = [\"High school\", \"University\", \"Medical\"]\n",
    "gpt_edu = [np.random.normal(8.29, 0.5, 52),\n",
    "           np.random.normal(10.03, 0.5, 52),\n",
    "           np.random.normal(11.19, 0.5, 52)]\n",
    "cla_edu = [np.random.normal(6.81, 0.5, 52),\n",
    "           np.random.normal(10.44, 0.5, 52),\n",
    "           np.random.normal(12.05, 0.5, 52)]\n",
    "\n",
    "gpt_means, gpt_errs = zip(*[mean_ci(d) for d in gpt_edu])\n",
    "cla_means, cla_errs = zip(*[mean_ci(d) for d in cla_edu])\n",
    "\n",
    "x = np.arange(len(edu_levels))\n",
    "fig, ax = plt.subplots(figsize=FIGSIZE)\n",
    "\n",
    "bars_gpt = ax.bar(x - width/2, gpt_means, width, yerr=gpt_errs, capsize=5,\n",
    "                  label='GPT', color=color_gpt)\n",
    "bars_cla = ax.bar(x + width/2, cla_means, width, yerr=cla_errs, capsize=5,\n",
    "                  label='Claude', color=color_claude)\n",
    "\n",
    "for bars in [bars_gpt, bars_cla]:\n",
    "    for bar in bars:\n",
    "        ax.annotate(f'{bar.get_height():.2f}',\n",
    "                    (bar.get_x() + bar.get_width()/2, bar.get_height()),\n",
    "                    textcoords=\"offset points\", xytext=(0, 3),\n",
    "                    ha='center', va='bottom', fontsize=14)\n",
    "\n",
    "ax.set_ylabel('Grade Level')\n",
    "ax.set_ylim(0, 14)\n",
    "ax.set_xticks(x)\n",
    "ax.set_xticklabels(edu_levels, rotation=XTICK_ROT, ha='right')\n",
    "place_legend_outside(ax)\n",
    "plt.tight_layout()\n",
    "plt.savefig(output_folder / \"readability_by_education_fk.png\", dpi=DPI, bbox_inches=\"tight\")\n",
    "plt.close()\n",
    "\n",
    "# =============================================================================\n",
    "# Plot 3: Ethnicity \u2014 SMOG Index\n",
    "# =============================================================================\n",
    "ethnicities = [\"European\", \"African\", \"Asian\"]\n",
    "gpt_eth = [np.random.normal(12.76, 0.4, 52),\n",
    "           np.random.normal(12.37, 0.4, 52),\n",
    "           np.random.normal(12.63, 0.4, 52)]\n",
    "cla_eth = [np.random.normal(12.39, 0.4, 52),\n",
    "           np.random.normal(12.03, 0.4, 52),\n",
    "           np.random.normal(11.97, 0.4, 52)]\n",
    "\n",
    "gpt_means, gpt_errs = zip(*[mean_ci(d) for d in gpt_eth])\n",
    "cla_means, cla_errs = zip(*[mean_ci(d) for d in cla_eth])\n",
    "\n",
    "avg_eth = [np.mean([g, c]) for g, c in zip(gpt_means, cla_means)]\n",
    "sorted_idx = np.argsort(avg_eth)\n",
    "eth_sorted = [ethnicities[i] for i in sorted_idx]\n",
    "gpt_sorted_means = [gpt_means[i] for i in sorted_idx]\n",
    "gpt_sorted_errs  = [gpt_errs[i]  for i in sorted_idx]\n",
    "cla_sorted_means = [cla_means[i] for i in sorted_idx]\n",
    "cla_sorted_errs  = [cla_errs[i]  for i in sorted_idx]\n",
    "\n",
    "x = np.arange(len(eth_sorted))\n",
    "fig, ax = plt.subplots(figsize=FIGSIZE)\n",
    "\n",
    "bars_gpt = ax.bar(x - width/2, gpt_sorted_means, width, yerr=gpt_sorted_errs, capsize=5,\n",
    "                  label='GPT', color=color_gpt)\n",
    "bars_cla = ax.bar(x + width/2, cla_sorted_means, width, yerr=cla_sorted_errs, capsize=5,\n",
    "                  label='Claude', color=color_claude)\n",
    "\n",
    "for bars in [bars_gpt, bars_cla]:\n",
    "    for bar in bars:\n",
    "        ax.annotate(f'{bar.get_height():.2f}',\n",
    "                    (bar.get_x() + bar.get_width()/2, bar.get_height()),\n",
    "                    textcoords=\"offset points\", xytext=(0, 3),\n",
    "                    ha='center', va='bottom', fontsize=14)\n",
    "\n",
    "ax.set_ylabel('SMOG Index')\n",
    "ax.set_ylim(0, 14)\n",
    "ax.set_xticks(x)\n",
    "ax.set_xticklabels(eth_sorted, rotation=XTICK_ROT, ha='right')\n",
    "place_legend_outside(ax)\n",
    "plt.tight_layout()\n",
    "plt.savefig(output_folder / \"readability_by_ethnicity_smog.png\", dpi=DPI, bbox_inches=\"tight\")\n",
    "plt.close()\n",
    "\n",
    "# =============================================================================\n",
    "# Plot 4: Gender \u2014 Gunning Fog\n",
    "# =============================================================================\n",
    "genders = [\"Female\", \"Male\"]\n",
    "gpt_gen = [np.random.normal(11.99, 0.5, 78),\n",
    "           np.random.normal(11.73, 0.5, 78)]\n",
    "cla_gen = [np.random.normal(11.61, 0.5, 78),\n",
    "           np.random.normal(11.78, 0.5, 78)]\n",
    "\n",
    "gpt_means, gpt_errs = zip(*[mean_ci(d) for d in gpt_gen])\n",
    "cla_means, cla_errs = zip(*[mean_ci(d) for d in cla_gen])\n",
    "\n",
    "x = np.arange(len(genders))\n",
    "fig, ax = plt.subplots(figsize=FIGSIZE)\n",
    "\n",
    "bars_gpt = ax.bar(x - width/2, gpt_means, width, yerr=gpt_errs, capsize=5,\n",
    "                  label='GPT', color=color_gpt)\n",
    "bars_cla = ax.bar(x + width/2, cla_means, width, yerr=cla_errs, capsize=5,\n",
    "                  label='Claude', color=color_claude)\n",
    "\n",
    "for bars in [bars_gpt, bars_cla]:\n",
    "    for bar in bars:\n",
    "        ax.annotate(f'{bar.get_height():.2f}',\n",
    "                    (bar.get_x() + bar.get_width()/2, bar.get_height()),\n",
    "                    textcoords=\"offset points\", xytext=(0, 3),\n",
    "                    ha='center', va='bottom', fontsize=14)\n",
    "\n",
    "ax.set_ylabel('Gunning Fog')\n",
    "ax.set_ylim(0, 14)\n",
    "ax.set_xticks(x)\n",
    "ax.set_xticklabels(genders, rotation=XTICK_ROT, ha='right')\n",
    "place_legend_outside(ax, loc=\"upper left\")\n",
    "plt.tight_layout()\n",
    "plt.savefig(output_folder / \"readability_by_gender_gf.png\", dpi=DPI, bbox_inches=\"tight\")\n",
    "plt.close()\n",
    "\n",
    "print(\"\u2705 Saved 4 readability figures with 95% CIs to:\", output_folder)\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": []
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "AI4Good",
   "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.8"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 2
}