import deepxde as dde
import numpy as np
import torch


# name_group = ["Diffusion_reaction", "Wave", "Heat", "Advection", "Poisson"]
# index_group = ["1_1D", "1_2D", "1_3D", "2_1D", "2_2D", "2_3D", ]

def Wave1_1D(x, y):
    dy_xx = dde.grad.hessian(y, x, i=0, j=0)
    dy_tt = dde.grad.hessian(y, x, i=0, j=1)
    return dy_tt - 1* dy_xx


def Wave1_2D(x, y):
    dy_xx = dde.grad.hessian(y, x, i=0, j=0)
    dy_yy = dde.grad.hessian(y, x, i=0, j=1)
    dy_tt = dde.grad.hessian(y, x, i=0, j=2)
    return dy_tt - 1* (dy_xx + dy_yy)

def Wave1_3D(x, y):
    dy_xx = dde.grad.hessian(y, x, i=0, j=0)
    dy_yy = dde.grad.hessian(y, x, i=0, j=1)
    dy_zz = dde.grad.hessian(y, x, i=0, j=2)
    dy_tt = dde.grad.hessian(y, x, i=0, j=3)
    
    return dy_tt - 1* (dy_xx + dy_yy+ dy_zz)



def Wave2_1D(x, y):
    dy_xx = dde.grad.hessian(y, x, i=0, j=0)
    dy_tt = dde.grad.hessian(y, x, i=0, j=1)
    return dy_tt - 9* dy_xx


def Wave2_2D(x, y):
    dy_xx = dde.grad.hessian(y, x, i=0, j=0)
    dy_yy = dde.grad.hessian(y, x, i=0, j=1)
    dy_tt = dde.grad.hessian(y, x, i=0, j=2)
    return dy_tt - 9* (dy_xx + dy_yy)

def Wave2_3D(x, y):
    dy_xx = dde.grad.hessian(y, x, i=0, j=0)
    dy_yy = dde.grad.hessian(y, x, i=0, j=1)
    dy_zz = dde.grad.hessian(y, x, i=0, j=2)
    dy_tt = dde.grad.hessian(y, x, i=0, j=3)
    
    return dy_tt - 9* (dy_xx + dy_yy + dy_zz)



def Poisson1_1D(x, y):
    dy_xx = dde.grad.hessian(y, x, i=0, j=0)
    return dy_xx + np.pi ** 2 * torch.sin(np.pi * x[:, 0])

def Poisson1_2D(x, y):
    dy_xx = dde.grad.hessian(y, x, i=0, j=0)
    dy_yy = dde.grad.hessian(y, x, i=0, j=1)
    return dy_xx + dy_yy + 2*np.pi ** 2 * torch.sin(np.pi * x[:, 0]) * torch.sin(np.pi * x[:, 1])


def Poisson1_3D(x, y):
    dy_xx = dde.grad.hessian(y, x, i=0, j=0)
    dy_yy = dde.grad.hessian(y, x, i=0, j=1)
    dy_zz = dde.grad.hessian(y, x, i=0, j=2)
    return dy_xx + dy_yy + dy_zz + 3*np.pi ** 2 * torch.sin(np.pi * x[:, 0]) * torch.sin(np.pi * x[:, 1]) * torch.sin(np.pi * x[:, 2])


def Poisson2_1D(x, y):
    dy_xx = dde.grad.hessian(y, x, i=0, j=0)
    return dy_xx + torch.ones_like(x[:, 0]) 

def Poisson2_2D(x, y):
    dy_xx = dde.grad.hessian(y, x, i=0, j=0)
    dy_yy = dde.grad.hessian(y, x, i=0, j=1)
    return dy_xx + dy_yy + torch.ones_like(x[:, 0])


def Poisson2_3D(x, y):
    dy_xx = dde.grad.hessian(y, x, i=0, j=0)
    dy_yy = dde.grad.hessian(y, x, i=0, j=1)
    dy_zz = dde.grad.hessian(y, x, i=0, j=2)
    return dy_xx + dy_yy + dy_zz + torch.ones_like(x[:, 0])



def Diffusion_reaction1_1D(x, y):
    du_xx = dde.grad.hessian(y, x, i=0, j=0, component=0)
    du_t = dde.grad.jacobian(y, x, i=0, j=1)  
    u = y[..., 0].unsqueeze(1)
    return du_t - 3*du_xx - u 

def Diffusion_reaction1_2D(x, y):
    du_xx = dde.grad.hessian(y, x, i=0, j=0)
    du_yy = dde.grad.hessian(y, x, i=0, j=1)
    du_t = dde.grad.jacobian(y, x, i=0, j=2)  
    u = y[..., 0].unsqueeze(1)
    return du_t - 3*(du_xx + du_yy) - u

def Diffusion_reaction1_3D(x, y):
    du_xx = dde.grad.hessian(y, x, i=0, j=0)
    du_yy = dde.grad.hessian(y, x, i=0, j=1)
    du_zz = dde.grad.hessian(y, x, i=0, j=2)
    du_t = dde.grad.jacobian(y, x, i=0, j=3)  
    u = y[..., 0].unsqueeze(1)
    return du_t - 3*(du_xx + du_yy + du_zz) - u




def Diffusion_reaction2_1D(x, y):
    du_xx = dde.grad.hessian(y, x, i=0, j=0, component=0)
    du_t = dde.grad.jacobian(y, x, i=0, j=1)  
    u = y[..., 0].unsqueeze(1)
    return du_t - 2*du_xx + u 

def Diffusion_reaction2_2D(x, y):
    du_xx = dde.grad.hessian(y, x, i=0, j=0)
    du_yy = dde.grad.hessian(y, x, i=0, j=1)
    du_t = dde.grad.jacobian(y, x, i=0, j=2)  
    u = y[..., 0].unsqueeze(1)
    return du_t - 2*(du_xx + du_yy) + u

def Diffusion_reaction2_3D(x, y):
    du_xx = dde.grad.hessian(y, x, i=0, j=0)
    du_yy = dde.grad.hessian(y, x, i=0, j=1)
    du_zz = dde.grad.hessian(y, x, i=0, j=2)
    du_t = dde.grad.jacobian(y, x, i=0, j=3)  
    u = y[..., 0].unsqueeze(1)
    return du_t - 2*(du_xx + du_yy + du_zz) + u




def Advection1_1D(x, y):
    dy_x = dde.grad.jacobian(y, x, i=0, j=0)
    dy_t = dde.grad.jacobian(y, x, i=0, j=1)
    return dy_t +  dy_x

def Advection1_2D(x, y):
    dy_t = dde.grad.jacobian(y, x, i=0, j=2)  # ∂u/∂t
    dy_x = dde.grad.jacobian(y, x, i=0, j=0)  # ∂u/∂x
    dy_y = dde.grad.jacobian(y, x, i=0, j=1)  # ∂u/∂y
    return dy_t +  dy_x + dy_y

def Advection1_3D(x, y):
    dy_t = dde.grad.jacobian(y, x, i=0, j=3)  # ∂u/∂t
    dy_x = dde.grad.jacobian(y, x, i=0, j=0)  # ∂u/∂x
    dy_y = dde.grad.jacobian(y, x, i=0, j=1)
    dy_z = dde.grad.jacobian(y, x, i=0, j=2)  # ∂u/∂z
    return dy_t +  dy_x + dy_y + dy_z

def Advection2_1D(x, y):
    dy_x = dde.grad.jacobian(y, x, i=0, j=0)
    dy_t = dde.grad.jacobian(y, x, i=0, j=1)
    return dy_t +  dy_x

def Advection2_2D(x, y):
    dy_t = dde.grad.jacobian(y, x, i=0, j=2)  # ∂u/∂t
    dy_x = dde.grad.jacobian(y, x, i=0, j=0)  # ∂u/∂x
    dy_y = dde.grad.jacobian(y, x, i=0, j=1)  # ∂u/∂y
    return dy_t +  dy_x + dy_y

def Advection2_3D(x, y):
    dy_t = dde.grad.jacobian(y, x, i=0, j=3)  # ∂u/∂t
    dy_x = dde.grad.jacobian(y, x, i=0, j=0)  # ∂u/∂x
    dy_y = dde.grad.jacobian(y, x, i=0, j=1)
    dy_z = dde.grad.jacobian(y, x, i=0, j=2)  # ∂u/∂z
    return dy_t +  dy_x + dy_y + dy_z

def Heat1_1D(x, y, ):
    a=0.4
    dy_t = dde.grad.jacobian(y, x, i=0, j=1)
    dy_xx = dde.grad.hessian(y, x, i=0, j=0)
    return dy_t - a * dy_xx

def Heat1_2D(x, y):
    """
    Define the 2D Heat equation: u_t - a * (u_xx + u_yy) = 0
    """
    a = 0.4  # Diffusion coefficient

    # Partial derivatives
    dy_t = dde.grad.jacobian(y, x, i=0, j=2)  # ∂u/∂t
    dy_xx = dde.grad.hessian(y, x, i=0, j=0)  # ∂²u/∂x²
    dy_yy = dde.grad.hessian(y, x, i=0, j=1)  # ∂²u/∂y²

    # Heat equation
    return dy_t - a * (dy_xx + dy_yy)

def Heat1_3D(x, y):
    """
    Define the 3D Heat equation: u_t - a * (u_xx + u_yy + u_zz) = 0
    """
    a = 0.4  # Diffusion coefficient

    # Partial derivatives
    dy_t = dde.grad.jacobian(y, x, i=0, j=3)  # ∂u/∂t
    dy_xx = dde.grad.hessian(y, x, i=0, j=0)  # ∂²u/∂x²
    dy_yy = dde.grad.hessian(y, x, i=0, j=1)  # ∂²u/∂y²
    dy_zz = dde.grad.hessian(y, x, i=0, j=2)  # ∂²u/∂z²

    # Heat equation
    return dy_t - a * (dy_xx + dy_yy + dy_zz)




def Heat2_1D(x, y, ):
    a=1.0
    dy_t = dde.grad.jacobian(y, x, i=0, j=1)
    dy_xx = dde.grad.hessian(y, x, i=0, j=0)
    return dy_t - a * dy_xx

def Heat2_2D(x, y):
    """
    Define the 2D Heat equation: u_t - a * (u_xx + u_yy) = 0
    """
    a = 1.0  # Diffusion coefficient

    # Partial derivatives
    dy_t = dde.grad.jacobian(y, x, i=0, j=2)  # ∂u/∂t
    dy_xx = dde.grad.hessian(y, x, i=0, j=0)  # ∂²u/∂x²
    dy_yy = dde.grad.hessian(y, x, i=0, j=1)  # ∂²u/∂y²

    # Heat equation
    return dy_t - a * (dy_xx + dy_yy)

def Heat2_3D(x, y):
    """
    Define the 3D Heat equation: u_t - a * (u_xx + u_yy + u_zz) = 0
    """
    a = 1.0  # Diffusion coefficient

    # Partial derivatives
    dy_t = dde.grad.jacobian(y, x, i=0, j=3)  # ∂u/∂t
    dy_xx = dde.grad.hessian(y, x, i=0, j=0)  # ∂²u/∂x²
    dy_yy = dde.grad.hessian(y, x, i=0, j=1)  # ∂²u/∂y²
    dy_zz = dde.grad.hessian(y, x, i=0, j=2)  # ∂²u/∂z²

    # Heat equation
    return dy_t - a * (dy_xx + dy_yy + dy_zz)