import matplotlib.pyplot as plt
from matplotlib.ticker import FuncFormatter
import pandas as pd
import numpy as np
import matplotlib.patheffects as pe
import math

def epsilon_show(result_file_name, picture_file_name, pattern_graph):
    if (pattern_graph == '2star'):
        pattern_graph = '2-star'
    df = pd.read_csv(result_file_name)
    epsilon_test = df['epsilon'].values
    
    pure_DP_error = df['pure_DP_error'].values
    pure_DP_std = df['pure_DP_std'].values
    
    approx_DP_error = df['approx_DP_error'].values
    approx_DP_std = df['approx_DP_std'].values
    
    base_comp_error = df['base_comp_error'].values
    base_comp_std = df['base_comp_std'].values
    
    base_comp_ADP_error = df['base_comp_ADP_error'].values
    base_comp_ADP_std = df['base_comp_ADP_std'].values
    
    y_min = min(np.min(pure_DP_error - pure_DP_std), np.min(approx_DP_error - approx_DP_std))
    y_max = max(np.max(pure_DP_error + pure_DP_std), np.max(approx_DP_error + approx_DP_std), 
                np.max(base_comp_error + base_comp_std), np.max(base_comp_ADP_error + base_comp_ADP_std))

    y_min_log = np.floor(np.log10(y_min))
    y_max_log = np.ceil(np.log10(y_max))
    yticks = [10**i for i in range(int(y_min_log), int(y_max_log) + 1)]

    fig, ax = plt.subplots()
    
    ax.plot(epsilon_test, base_comp_error, label='PDP_Comp', marker='x', linestyle='-', color='black')
    ax.fill_between(epsilon_test, base_comp_error-base_comp_std, base_comp_error+base_comp_std, color='black', alpha=0.05)
    
    ax.plot(epsilon_test, base_comp_ADP_error, label='ADP_Comp', marker='^', linestyle='-', color='g')
    ax.fill_between(epsilon_test, base_comp_ADP_error-base_comp_ADP_std, base_comp_ADP_error+base_comp_ADP_std, color='g', alpha=0.1)
    
    line, = ax.plot(epsilon_test, pure_DP_error, label='PDP_RSC', marker='o', linestyle='-', color='b')
    line.set_path_effects([pe.Stroke(linewidth=1, foreground='white'), pe.Normal()])
    ax.fill_between(epsilon_test, pure_DP_error-pure_DP_std, pure_DP_error+pure_DP_std, color='b', alpha=0.3)
    
    line, = ax.plot(epsilon_test, approx_DP_error, label='ADP_RSC', marker='s', linestyle='-', color='r')
    line.set_path_effects([pe.Stroke(linewidth=1, foreground='white'), pe.Normal()])
    ax.fill_between(epsilon_test, approx_DP_error-approx_DP_std, approx_DP_error+approx_DP_std, color='r', alpha=0.3)
    
    ax.set_yscale('log')  
    ax.set_ylim(10**y_min_log, 10**y_max_log)
    ax.set_yticks(yticks)
    ax.tick_params(axis='both', labelsize=13)

    ax.legend(loc='upper right', framealpha=0.8)
    ax.grid(True, which='major', linestyle='--', linewidth=0.5)

    ax.set_title(pattern_graph, fontsize=30)
    ax.set_xlabel(r'$\epsilon$', fontsize=30) 
    ax.set_ylabel('Relative Error', fontsize=30)
    plt.tight_layout()
    plt.savefig(picture_file_name)
    plt.close()

def Q_show(result_file_name, picture_file_name, pattern_graph, lim_Q):
    if (pattern_graph == '2star'):
        pattern_graph = '2-star'
    df = pd.read_csv(result_file_name)
    Q_num_test = df['Q_num'].values / (10**lim_Q)
    
    pure_DP_error = df['pure_DP_error'].values
    pure_DP_std = df['pure_DP_std'].values
    
    approx_DP_error = df['approx_DP_error'].values
    approx_DP_std = df['approx_DP_std'].values
    
    base_comp_error = df['base_comp_error'].values
    base_comp_std = df['base_comp_std'].values
    
    base_comp_ADP_error = df['base_comp_ADP_error'].values
    base_comp_ADP_std = df['base_comp_ADP_std'].values

    y_min = min(np.min(pure_DP_error - pure_DP_std), np.min(approx_DP_error - approx_DP_std))
    y_max = max(np.max(pure_DP_error + pure_DP_std), np.max(approx_DP_error + approx_DP_std), 
                np.max(base_comp_error + base_comp_std), np.max(base_comp_ADP_error + base_comp_ADP_std))

    y_min_log = np.floor(np.log10(y_min))
    y_max_log = np.ceil(np.log10(y_max))
    yticks = [10**i for i in range(int(y_min_log), int(y_max_log) + 1)]

    fig, ax = plt.subplots()
    
    ax.plot(Q_num_test, base_comp_error, label='PDP_Comp', marker='x', linestyle='-', color='black')
    ax.fill_between(Q_num_test, base_comp_error-base_comp_std, base_comp_error+base_comp_std, color='black', alpha=0.05)
    
    ax.plot(Q_num_test, base_comp_ADP_error, label='ADP_Comp', marker='^', linestyle='-', color='g')
    ax.fill_between(Q_num_test, base_comp_ADP_error-base_comp_ADP_std, base_comp_ADP_error+base_comp_ADP_std, color='g', alpha=0.1)
    
    line, = ax.plot(Q_num_test, pure_DP_error, label='PDP_RSC', marker='o', linestyle='-', color='b')
    line.set_path_effects([pe.Stroke(linewidth=1, foreground='white'), pe.Normal()])
    ax.fill_between(Q_num_test, pure_DP_error-pure_DP_std, pure_DP_error+pure_DP_std, color='b', alpha=0.3)
    
    line, = ax.plot(Q_num_test, approx_DP_error, label='ADP_RSC', marker='s', linestyle='-', color='r')
    line.set_path_effects([pe.Stroke(linewidth=1, foreground='white'), pe.Normal()])
    ax.fill_between(Q_num_test, approx_DP_error-approx_DP_std, approx_DP_error+approx_DP_std, color='r', alpha=0.3)
    
    ax.set_yscale('log')  
    ax.set_ylim(10**y_min_log, 10**y_max_log)
    ax.set_yticks(yticks)
    ax.tick_params(axis='both', labelsize=13)

    ax.legend(loc='upper left', framealpha=0.8)
    ax.grid(True, which='major', linestyle='--', linewidth=0.5)

    ax.set_title(pattern_graph, fontsize=30)
    ax.set_xlabel(fr'$|Q|(\times 10^{lim_Q})$', fontsize=30)
    ax.set_ylabel('Relative Error', fontsize=30)
    plt.tight_layout()
    plt.savefig(picture_file_name)  
    plt.close()

def qtime_show(result_file_name, picture_file_name, pattern_graph):
    if (pattern_graph == '2star'):
        pattern_graph = '2-star'
    df = pd.read_csv(result_file_name)  
    labels = ['PDP_RSC', 'ADP_RSC', 'PDP_Comp', 'ADP_Comp']
    values = df.iloc[0].tolist() 
    colors = ['blue', 'red', 'black', 'green']

    y_min_log = 1
    y_max_log = 7
    yticks = [10**i for i in range(int(y_min_log), int(y_max_log) + 1)]
    
    plt.figure(figsize=(6,4))
    plt.bar(labels, values, color=colors)
    
    plt.text(0, values[0], f'{int(round(values[0]))} μs', ha='center', va='bottom', fontsize=14)
    plt.text(1, values[1], f'{int(round(values[1]))} μs', ha='center', va='bottom', fontsize=14)
    plt.text(2, values[2], f'{int(round(values[2]/1000))} ms', ha='center', va='bottom', fontsize=14)
    plt.text(3, values[3], f'{int(round(values[3]/1000))} ms', ha='center', va='bottom', fontsize=14)

    plt.yscale('log')
    plt.ylim(10**y_min_log, 10**y_max_log)
    plt.yticks(yticks)
    plt.xlabel('Algorithms', fontsize=30)
    plt.ylabel(r'Query Time ($\mu$s)', fontsize=26)
    plt.title(pattern_graph, fontsize=30)
    plt.tight_layout()
    plt.savefig(picture_file_name)
    plt.close()
    
def prtime_show(result_file_name, picture_file_name, pattern_graph, lim):
    if (pattern_graph == '2star'):
        pattern_graph = '2-star'
    df = pd.read_csv(result_file_name)  
    labels = ['PDP_RSC', 'ADP_RSC', 'PDP_Comp', 'ADP_Comp']
    values = df.iloc[0].tolist() 
    colors = ['blue', 'red', 'black', 'green']

    plt.figure(figsize=(6,4))
    plt.bar(labels, values, color=colors)
    
    for i, v in enumerate(values):
        if (math.isclose(v, 0.0)):
            continue 
        if (v >= 1):
            plt.text(i, v, f'{v:.2f} min', ha='center', va='bottom', fontsize=14)
        else:
            plt.text(i, v, f'{int(round(v*60))} s', ha='center', va='bottom', fontsize=14)
            
    plt.ylim(0, lim)
    plt.xlabel('Algorithms', fontsize=30)
    plt.ylabel('Preprocessing Time (min)', fontsize=20)
    plt.title(pattern_graph, fontsize=30)
    plt.tight_layout()
    plt.savefig(picture_file_name)
    plt.close()