"""Helper functions to compare candidates and committees."""

import numpy as np


def find_favorite_candidate(voter, S):
    """Returns the voter's favorite candidate in S.
    
    Args: 
      voter: voter object
      S: input committee (numpy array of length k)
    
    Returns: integer representing the voter's favorite candidate in S.
    """
    favorite = S[0]
    for cand in S[1:]:
        if voter.pairwise_prefer(cand, favorite):
            favorite = cand
    return favorite


def cand_preferred_over_committee(voter, cand, S, strict=True):
    """Returns True iff cand is preferred over all candidates in S.
    
    Args: 
      voter_pairwise_prefer_fn: function such that voter_pairwise_prefer_fn(cand1, cand2) returns true
        if the voter prefers cand1 over cand2.
      cand: input candidate (integer)
      S: input committee (numpy array of length k)
      strict: if True, returns True iff cand is strictly preferred over all candidates in S.
        Otherwise, returns True iff no candidate in S is strictly preferred over cand.
    
    Returns: bool, True iff cand is preferred over all candidates in S.
    """
    cand_preferred = True
    if strict:  
        for cand_comp in S:
            if not voter.pairwise_prefer(cand, cand_comp):
                cand_preferred = False
    else:
        for cand_comp in S:
            if voter.pairwise_prefer(cand_comp, cand):
                cand_preferred = False
    return cand_preferred


def convert_candidates_to_ranks(values_as_candidates, num_candidates):
    """Converts an array with values as candidates to an array where the 
    position represents the candidate index, and the value is the rank.
    """
    values_as_ranks = np.zeros(num_candidates)
    for i in range(num_candidates):
        rank = i
        candidate = values_as_candidates[i]
        values_as_ranks[candidate] = rank
    return values_as_ranks.astype(int)

def convert_ranks_to_candidates(values_as_ranks, num_candidates):
    """Converts an array with values as candidates to an array where the 
    position represents the candidate index, and the value is the rank.
    """
    values_as_candidates = np.zeros(num_candidates)
    for i in range(num_candidates):
        candidate = i
        rank = values_as_ranks[i]
        values_as_candidates[rank] = candidate
    return values_as_candidates.astype(int)


def cand_preferred_over_committee_full_rank(voter_values_as_ranks, cand, S):
    """Returns True iff cand is preferred over all candidates in S."""

    # Get max voter rank in S
    max_rank_S = len(voter_values_as_ranks)
    for S_cand in S:
        rank_S = voter_values_as_ranks[S_cand]
        if max_rank_S >= rank_S:
            max_rank_S = rank_S
    rank_cand = voter_values_as_ranks[cand]
    return rank_cand > max_rank_S


def voter_compare_committee(voter, S1, S2, strict=True):
    """
    Returns True iff voter prefers S1 over S2.
    """
    for cand1 in S1:
        if cand_preferred_over_committee(voter.pairwise_prefer, cand1, S2, strict):
            return True
    return False


def voter_unsatisfied(beta, voter, all_prev_Ss, all_prev_lotteries, strict=True, verbose=False):
    """Returns True iff all estimated ranks for the voter are <= beta."""
    for i in range(len(all_prev_Ss)):
        rank = estimate_voter_rank(voter, all_prev_Ss[i], all_prev_lotteries[i][0], all_prev_lotteries[i][1], strict=strict)
        if verbose:
            print('S', all_prev_Ss[i])
            print('rank', rank)
        if rank > beta:
            return False
    return True


def estimate_voter_rank(voter, S, lottery_support, lottery_probs, strict=True, verbose=False):
    """Returns Rank(v;S,Delta) for a given voter, committee S, and lottery.
    
    Args:
      voter: voter object
      S: input committee for which to compute estimate, numpy array of length k
      lottery_support: list of committees as numpy array of shape (number of committees, k)
      lottery_probs: list of probabilities of the same length as lottery_support
      
    Returns: float, estimate of rank for input committee
    """
    # Find which candidate the voter likes best in S.
    S_favorite = find_favorite_candidate(voter, S)
    if verbose:
        print("S_favorite", S_favorite)
    
    preferred_onehot = np.zeros(lottery_probs.shape)
    for i, S_prime in enumerate(lottery_support):
        # Check whether voter prefers S_prime over the favorite candidate in S.
        preferred_onehot[i] = int(cand_preferred_over_committee(voter, S_favorite, S_prime, strict=strict))
    # Compute Rank(v;S,Delta)
    rank = np.dot(preferred_onehot,lottery_probs)
    return rank