import torch
import torch.nn as nn
import torch.nn.functional as F
from torch.distributions import Normal
import numpy as np
DEVICE = torch.device("cuda" if torch.cuda.is_available() else "cpu")

#Beta distribution related imports remain unchanged
from torch.distributions import Beta

class Actor(nn.Module):
    #Modified: hidden_dim replaced by hiddens_sac list, default value is [128, 128]
    def __init__(self, state_dim, action_dim=2, hiddens_sac=[128, 128]):
        super(Actor, self).__init__()
        
        self.state_dim = state_dim
        self.action_dim = action_dim
        
        #--- Changes Begin: Dynamically Build a Fully Connected Network ---
        net_layers = []
        in_dim = state_dim
        #Traverse the hiddens_sac list to create linear layers and activation functions for each layer
        for h_dim in hiddens_sac:
            net_layers.append(nn.Linear(in_dim, h_dim))
            net_layers.append(nn.ReLU())
            in_dim = h_dim #Update the input dimension of the next layer
        
        #Pack all layers with nn.Sequential
        self.net = nn.Sequential(*net_layers)
        #--- End of change ---
        
        #The input dimension of the mean and standard deviation layers is now the last element of hiddens_sac
        last_hidden_dim = hiddens_sac[-1]
        self.mean = nn.Linear(last_hidden_dim, action_dim)
        self.log_std = nn.Linear(last_hidden_dim, action_dim)
        
        #Inisialisasi
        self._init_weights()
    
    def _init_weights(self):
        """Initialize weights (this function does not need to be modified to correctly handle dynamic networks)"""
        for m in self.net.modules():
            if isinstance(m, nn.Linear):
                nn.init.xavier_uniform_(m.weight)
                nn.init.zeros_(m.bias)
        
        nn.init.xavier_uniform_(self.mean.weight, gain=0.01)
        nn.init.zeros_(self.mean.bias)
        
        nn.init.xavier_uniform_(self.log_std.weight, gain=0.01)
        nn.init.zeros_(self.log_std.bias)
    
    def forward(self, state, deterministic=False, return_log_prob=True):
        """Forward propagation (this function does not need to be modified)"""
        x = self.net(state)
        
        mean = self.mean(x)
        log_std = self.log_std(x)
        log_std = torch.clamp(log_std, -20, 2)
        std = log_std.exp()
        
        if deterministic:
            return torch.sigmoid(mean)
        
        normal = Normal(mean, std)
        x_t = normal.rsample()
        action = torch.sigmoid(x_t)
        
        if return_log_prob:
            log_prob = normal.log_prob(x_t)
            log_prob -= torch.log(action * (1 - action) + 1e-6)
            log_prob = log_prob.sum(1, keepdim=True)
            return action, log_prob
        else:
            return action

    def log_prob(self, state, action):
        """Calculate log probability (this function does not need to be modified)"""
        x = self.net(state)
        mean = self.mean(x)
        log_std = torch.clamp(self.log_std(x), -20, 2)
        std = log_std.exp()
        normal = Normal(mean, std)

        eps = 1e-6
        a_clamped = torch.clamp(action, eps, 1 - eps)

        x_t = torch.log(a_clamped) - torch.log(1 - a_clamped)

        log_prob = normal.log_prob(x_t)
        log_prob = log_prob - torch.log(a_clamped * (1 - a_clamped) + eps)

        return log_prob.sum(-1, keepdim=True)


class Critic(nn.Module):
    #Modified: hidden_dim replaced by hiddens_sac list
    def __init__(self, state_dim, action_dim=2, hiddens_sac=[128, 128]):
        super(Critic, self).__init__()
        
        self.state_dim = state_dim
        self.action_dim = action_dim
        
        #--- Changes Begin: Dynamically build Q networks using helper functions ---
        #Define a helper function to create a Q network to reduce code duplication
        def _create_q_net(hiddens):
            q_layers = []
            in_dim = state_dim + action_dim
            #Traverse the hiddens list to create hidden layers
            for h_dim in hiddens:
                q_layers.append(nn.Linear(in_dim, h_dim))
                q_layers.append(nn.ReLU())
                in_dim = h_dim
            #Add Final Output Layer
            q_layers.append(nn.Linear(in_dim, 1))
            return nn.Sequential(*q_layers)

        #Create Q1 and Q2 networks
        self.q1 = _create_q_net(hiddens_sac)
        self.q2 = _create_q_net(hiddens_sac)
        #--- End of change ---
        
        #Inisialisasi
        self._init_weights()
    
    def _init_weights(self):
        """Initialize weights (this function does not need to be modified)"""
        for net in [self.q1, self.q2]:
            for m in net.modules():
                if isinstance(m, nn.Linear):
                    nn.init.xavier_uniform_(m.weight)
                    nn.init.zeros_(m.bias)
    
    def forward(self, state, action):
        """Calculate Q value (this function does not need to be modified)"""
        sa = torch.cat([state, action], dim=1)
        
        q1 = self.q1(sa)
        q2 = self.q2(sa)
        
        return q1, q2