/**
 * @brief Calculate the local estimate for t+1.
 * This uses the social estimate 'x_social' from the PREVIOUS time step.
 */
EstimationOutput prepareLocalDataForSending(const Eigen::Vector2d& x_target_prediction_t_plus) {
            x_local_t_plus_1 = x_target_prediction_t_plus*alpha_i + (1.0 - alpha_i) * x_social;

            EstimationOutput packet;
            packet.x_local_to_send = x_local_t_plus_1;
            packet.w_hat_to_send = w_hat_self; // Send the current w_hat

            x_prediction = x_target_prediction_t_plus;
            x_local_prev = x_local_t_plus_1;

            return packet;
        }

/**
 * @brief Implements the new estimation algorithm.
 * This function should be called ONLY when new messages have been received from all neighbors.
 */
void updateAndGetEstimates(
    const Eigen::Vector2d& x_target_curr,
    const std::vector<Eigen::Vector2d>& all_x_local_from_neighbors,
    const std::vector<double>& all_w_hat_from_neighbors)
{
    // Update the internal weights for next iteration
    alpha_i_hat *= exp(-eta_alpha_ * loss(x_target_curr, x_prediction));
    alpha_i_hat_prime *= exp(-eta_alpha_ * loss(x_target_curr, x_social_prev));

    if (std::max(alpha_i_hat_prime, alpha_i_hat) < 1e-9) {
        alpha_i_hat_prime = alpha_i_hat_prime/1e-9;
        alpha_i_hat = alpha_i_hat/1e-9;
        ROS_WARN("Alpha Hat Value dropped below 1e-9");
    }
    alpha_i = alpha_i_hat / (alpha_i_hat + alpha_i_hat_prime);

    w_hat_self *= exp(-eta_weights_ * loss(x_target_curr, x_local_prev));


    // Calculate the social weights
    w_social_weights_.clear();

    auto max_value = std::max_element(all_w_hat_from_neighbors.begin(), all_w_hat_from_neighbors.end());
    double max_w_hat = *max_value;
    double factor_w_hat = 1.0;
    if(std::max(max_w_hat, w_hat_self) < 1e-9)
    {
        factor_w_hat = 1/1e-9;
        ROS_WARN("W Hat Value dropped below 1e-9");
    }

    double w_hat_sum = w_hat_self* factor_w_hat;

    for(double w_hat: all_w_hat_from_neighbors)
        w_hat_sum+=w_hat*factor_w_hat;
        w_social_weights_.push_back(w_hat_self*factor_w_hat/ w_hat_sum);
        for(double w_hat: all_w_hat_from_neighbors)
            w_social_weights_.push_back(w_hat*factor_w_hat/ w_hat_sum);

    x_social_prev = x_social;
    // Update the social estimate
    Eigen::Vector2d social_estimate_sum = w_social_weights_[0] * x_local_t_plus_1; // Start with self-contribution
    for(size_t j=0; j<all_x_local_from_neighbors.size(); j++)
        social_estimate_sum += w_social_weights_[j+1]*all_x_local_from_neighbors[j];

    x_social = social_estimate_sum;
}
