"""MyFitnessPal-style macro calculator.

Implements the same methodology as the MyFitnessPal free macro calculator:
  1. BMR via the Mifflin-St Jeor equation.
  2. TDEE = BMR × activity multiplier.
  3. Target calories adjusted for weekly weight-change goal.
  4. Macros (protein, carbs, fat) split from target calories according to a
     carb/fat preference preset.

Reference: https://blog.myfitnesspal.com/free-macro-calculator/
"""

from __future__ import annotations

from typing import Any, Dict, Optional

# ---------------------------------------------------------------------------
# Constants
# ---------------------------------------------------------------------------

# Activity-level labels accepted by the tool (case-insensitive).
ACTIVITY_LEVELS = {
    "sedentary": 1.2,          # little or no exercise, desk job
    "lightly active": 1.375,   # light exercise 1-3 days/week
    "moderately active": 1.55, # moderate exercise 3-5 days/week
    "very active": 1.725,      # hard exercise 6-7 days/week
    "extra active": 1.9,       # very intense exercise daily or physical job
}

# Goal labels accepted by the tool (case-insensitive).
VALID_GOALS = {"lose weight", "maintain", "gain weight"}

# Weekly rate strings (lbs/week) → calorie delta per day.
# Positive = surplus (gain), negative = deficit (loss).
# 1 lb ≈ 3 500 kcal → rate_lbs_per_week × 500 = kcal/day delta.
_CALORIES_PER_LB = 3_500

# Macro presets: carb/fat preference → (protein_pct, carbs_pct, fat_pct).
# Percentages are of total calories; must sum to 1.0.
CARB_FAT_PRESETS: Dict[str, tuple] = {
    "balanced":    (0.20, 0.50, 0.30),  # MFP default
    "lower carb":  (0.30, 0.30, 0.40),
    "higher carb": (0.20, 0.60, 0.20),
    "higher protein": (0.35, 0.40, 0.25),
}

# Caloric density of macros (kcal per gram).
KCAL_PER_G_PROTEIN = 4.0
KCAL_PER_G_CARBS = 4.0
KCAL_PER_G_FAT = 9.0


# ---------------------------------------------------------------------------
# Internal helpers
# ---------------------------------------------------------------------------

def _mifflin_st_jeor_bmr(weight_kg: float, height_cm: float, age_years: float, sex: str) -> float:
    """Return Basal Metabolic Rate in kcal/day using the Mifflin-St Jeor equation."""
    base = 10.0 * weight_kg + 6.25 * height_cm - 5.0 * age_years
    return base + 5.0 if sex == "male" else base - 161.0


def _normalise_activity(activity_level: str) -> float:
    """Return the PAL multiplier for an activity level string."""
    key = activity_level.strip().lower()
    if key not in ACTIVITY_LEVELS:
        valid = ", ".join(f'"{k}"' for k in ACTIVITY_LEVELS)
        raise ValueError(
            f'Unknown activity_level "{activity_level}". '
            f"Must be one of: {valid}."
        )
    return ACTIVITY_LEVELS[key]


def _normalise_goal(goal: str) -> str:
    key = goal.strip().lower()
    if key not in VALID_GOALS:
        valid = ", ".join(f'"{g}"' for g in sorted(VALID_GOALS))
        raise ValueError(
            f'Unknown goal "{goal}". Must be one of: {valid}.'
        )
    return key


def _weekly_rate_to_daily_delta(weekly_rate_lbs: float, goal: str) -> float:
    """Convert a weekly weight-change rate (lbs) to a daily calorie delta."""
    if abs(weekly_rate_lbs) > 2.0:
        raise ValueError(
            "weekly_rate_lbs must be between 0 and 2.0 for safe weight change."
        )
    daily_delta = weekly_rate_lbs * _CALORIES_PER_LB / 7.0
    if goal == "lose weight":
        return -daily_delta
    if goal == "gain weight":
        return daily_delta
    return 0.0


def _get_macro_split(carb_fat_preference: str) -> tuple:
    key = carb_fat_preference.strip().lower()
    if key not in CARB_FAT_PRESETS:
        valid = ", ".join(f'"{k}"' for k in CARB_FAT_PRESETS)
        raise ValueError(
            f'Unknown carb_fat_preference "{carb_fat_preference}". '
            f"Must be one of: {valid}."
        )
    return CARB_FAT_PRESETS[key]


def _kg_from(weight: float, unit: str) -> float:
    unit = unit.strip().lower()
    if unit in ("kg", "kilograms", "kilogram"):
        return weight
    if unit in ("lb", "lbs", "pounds", "pound"):
        return weight * 0.45359237
    raise ValueError(f'Unknown weight_unit "{unit}". Use "kg" or "lb".')


def _cm_from(height: float, unit: str, height_inches: Optional[float] = None) -> float:
    unit = unit.strip().lower()
    if unit in ("cm", "centimeters", "centimetres"):
        return height
    if unit in ("in", "inches", "inch"):
        return height * 2.54
    if unit in ("ft_in", "ft"):
        inches_total = height * 12.0 + (height_inches or 0.0)
        return inches_total * 2.54
    raise ValueError(
        f'Unknown height_unit "{unit}". Use "cm", "in", or "ft_in" '
        "(with height_inches for the remaining inches)."
    )



# ---------------------------------------------------------------------------
# Public tool function
# ---------------------------------------------------------------------------

def calculate_mfp_macros(
    sex: str,
    age: float,
    weight: float,
    height: float,
    activity_level: str,
    goal: str = "maintain",
    weekly_rate_lbs: float = 0.5,
    carb_fat_preference: str = "balanced",
    weight_unit: str = "kg",
    height_unit: str = "cm",
    height_inches: Optional[float] = None,
) -> Any:
    """Calculate daily macro targets using the MyFitnessPal macro calculator methodology.

    Estimates Basal Metabolic Rate (BMR) with the Mifflin-St Jeor equation,
    scales it to Total Daily Energy Expenditure (TDEE) by an activity multiplier,
    applies a calorie adjustment for a weekly weight-change goal, and then splits
    target calories into protein, carbohydrates, and fat according to the chosen
    carb/fat preference preset.

    Args:
        sex: Biological sex used for the BMR equation. Must be "male" or "female".
        age: Age in years (must be ≥ 15).
        weight: Body weight. Interpreted according to ``weight_unit``.
        height: Height. Interpreted according to ``height_unit``. When
            ``height_unit="ft_in"``, this is the feet component; supply the
            remaining inches in ``height_inches``.
        activity_level: Physical activity level. One of:
            - ``"sedentary"`` — little or no exercise, desk job
            - ``"lightly active"`` — light exercise 1-3 days/week
            - ``"moderately active"`` — moderate exercise 3-5 days/week
            - ``"very active"`` — hard exercise 6-7 days/week
            - ``"extra active"`` — very intense exercise daily or a physical job
        goal: Weight goal. One of ``"lose weight"``, ``"maintain"``,
            ``"gain weight"``. Defaults to ``"maintain"``.
        weekly_rate_lbs: Desired weekly weight change in pounds (0–2.0).
            Used only when ``goal`` is ``"lose weight"`` or ``"gain weight"``.
            Defaults to 0.5 lb/week.
        carb_fat_preference: Macro split preset. One of:
            - ``"balanced"`` — 20 % protein / 50 % carbs / 30 % fat (MFP default)
            - ``"lower carb"`` — 30 % protein / 30 % carbs / 40 % fat
            - ``"higher carb"`` — 20 % protein / 60 % carbs / 20 % fat
            - ``"higher protein"`` — 35 % protein / 40 % carbs / 25 % fat
            Defaults to ``"balanced"``.
        weight_unit: Unit for ``weight``. Either ``"kg"`` (default) or ``"lb"``.
        height_unit: Unit for ``height``. One of ``"cm"`` (default),
            ``"in"``, or ``"ft_in"``.
        height_inches: Inches component of height when ``height_unit="ft_in"``.

    Returns:
        On success, a dict with the following keys:

        - ``bmr_kcal`` (float): Basal Metabolic Rate in kcal/day.
        - ``tdee_kcal`` (float): Total Daily Energy Expenditure in kcal/day.
        - ``target_calories_kcal`` (float): Daily calorie target after goal
          adjustment.
        - ``macros_g_per_day`` (dict): Recommended daily grams of
          ``protein``, ``carbohydrates``, and ``fat``.
        - ``macros_pct_of_calories`` (dict): Percentage of total calories from
          ``protein``, ``carbohydrates``, and ``fat``.
        - ``goal`` (str): The resolved goal string.
        - ``activity_level`` (str): The activity level used.
        - ``carb_fat_preference`` (str): The macro preset used.
        - ``notes`` (list[str]): Informational notes about the calculation.

        On validation error, returns a string starting with ``"Error: "``.
    """
    try:
        full = _calculate_mfp_macros_impl(
            sex=sex,
            age=age,
            weight=weight,
            height=height,
            activity_level=activity_level,
            goal=goal,
            weekly_rate_lbs=weekly_rate_lbs,
            carb_fat_preference=carb_fat_preference,
            weight_unit=weight_unit,
            height_unit=height_unit,
            height_inches=height_inches,
        )
        # Only return the requested fields
        return {
            "target_calories_kcal": full["target_calories_kcal"],
            "macros_g_per_day": full["macros_g_per_day"],
            "goal": full["goal"],
            "notes": full["notes"],
        }
    except (TypeError, ValueError) as exc:
        return f"Error: {exc}"


def _calculate_mfp_macros_impl(
    sex: str,
    age: float,
    weight: float,
    height: float,
    activity_level: str,
    goal: str,
    weekly_rate_lbs: float,
    carb_fat_preference: str,
    weight_unit: str,
    height_unit: str,
    height_inches: Optional[float],
) -> Dict[str, Any]:
    # --- Validate & normalise sex ---
    sex_norm = sex.strip().lower()
    if sex_norm not in ("male", "female"):
        raise ValueError('sex must be "male" or "female".')

    # --- Validate age ---
    if age < 15:
        raise ValueError(
            "This calculator is designed for adults (age ≥ 15). "
            "For children, use calculate_health_canada_dri instead."
        )

    # --- Convert units ---
    weight_kg = _kg_from(weight, weight_unit)
    height_cm = _cm_from(height, height_unit, height_inches)

    if weight_kg <= 0:
        raise ValueError("weight must be a positive number.")
    if height_cm <= 0:
        raise ValueError("height must be a positive number.")

    # --- BMR ---
    bmr = _mifflin_st_jeor_bmr(weight_kg, height_cm, float(age), sex_norm)

    # --- TDEE ---
    pal = _normalise_activity(activity_level)
    tdee = bmr * pal

    # --- Goal & calorie target ---
    goal_norm = _normalise_goal(goal)
    daily_delta = _weekly_rate_to_daily_delta(float(weekly_rate_lbs), goal_norm)
    target_kcal = tdee + daily_delta

    # MFP enforces a minimum floor of 1 200 kcal (women) / 1 500 kcal (men).
    min_kcal = 1_500.0 if sex_norm == "male" else 1_200.0
    notes: list[str] = []
    if target_kcal < min_kcal:
        notes.append(
            f"Calculated target ({target_kcal:.0f} kcal) is below the safe minimum "
            f"({min_kcal:.0f} kcal). Target has been raised to the minimum."
        )
        target_kcal = min_kcal

    # --- Macro split ---
    protein_pct, carbs_pct, fat_pct = _get_macro_split(carb_fat_preference)

    protein_g = round(target_kcal * protein_pct / KCAL_PER_G_PROTEIN, 1)
    carbs_g = round(target_kcal * carbs_pct / KCAL_PER_G_CARBS, 1)
    fat_g = round(target_kcal * fat_pct / KCAL_PER_G_FAT, 1)

    # Only add goal-related notes, not the BMR explanation
    if goal_norm != "maintain":
        rate_dir = "loss" if goal_norm == "lose weight" else "gain"
        notes.append(
            f"Calorie target reflects a {weekly_rate_lbs:.2f} lb/week weight {rate_dir} "
            f"({abs(daily_delta):.0f} kcal/day {'deficit' if goal_norm == 'lose weight' else 'surplus'})."
        )

    return {
        "bmr_kcal": round(bmr, 1),
        "tdee_kcal": round(tdee, 1),
        "target_calories_kcal": round(target_kcal, 1),
        "macros_g_per_day": {
            "protein": protein_g,
            "carbohydrates": carbs_g,
            "fat": fat_g,
        },
        "macros_pct_of_calories": {
            "protein": round(protein_pct * 100),
            "carbohydrates": round(carbs_pct * 100),
            "fat": round(fat_pct * 100),
        },
        "goal": goal_norm,
        "activity_level": activity_level.strip().lower(),
        "carb_fat_preference": carb_fat_preference.strip().lower(),
        "notes": notes,
    }


# ---------------------------------------------------------------------------
# CLI entry point for quick testing
# ---------------------------------------------------------------------------
if __name__ == "__main__":
    import json
    print("Test 1: 30-year-old male, 80 kg, 180 cm, moderately active, lose weight")
    result = calculate_mfp_macros(
        sex='male', age=30, weight=80, height=180,
        activity_level='moderately active',
        goal='lose weight', weekly_rate_lbs=1.0,
        carb_fat_preference='balanced'
    )
    print(json.dumps(result, indent=2))

    print("\nTest 2: 25-year-old female, 140 lb, 5 ft 5 in, lightly active, maintain")
    result2 = calculate_mfp_macros(
        sex='female', age=25, weight=140, height=5,
        activity_level='lightly active',
        goal='maintain',
        carb_fat_preference='higher protein',
        weight_unit='lb', height_unit='ft_in', height_inches=5
    )
    print(json.dumps(result2, indent=2))

