import numpy as np
from termcolor import cprint
from isaacsim.core.utils.stage import add_reference_to_stage
from isaacsim.core.prims import SingleXFormPrim, SingleRigidPrim, SingleGeometryPrim
from isaacsim.core.utils.rotations import euler_angles_to_quat
from pxr import UsdPhysics, PhysxSchema, UsdShade, Sdf, UsdGeom
from isaacsim.core.utils import prims as prim_utils
from isaacsim.core.api.materials.physics_material import PhysicsMaterial
from omni.physx.scripts import physicsUtils


class Rigid_Flat_Object():
    def __init__(self, world,name, path,position=np.array([0.15, 0.0, 0.8]),orientation=np.array([0,0,0]),
                 scale=np.array([0.7, 0.7, 0.7]),random_init=False, type:int=1):
        self.my_position = position
        self.my_orientation = orientation
        self.path=path
        self.name=name
        self.prim_path="/World/Rigid_Flat_Object"
        add_reference_to_stage(usd_path=path,prim_path=self.prim_path)

        self.sample_type = type

        force_zero_ori = ("Clock" in name)

        if position is None:
            position=np.array([0,0,0])
        if orientation is None:
            orientation=euler_angles_to_quat([0.0,0.,0.],degrees=True)
        else:
            orientation=euler_angles_to_quat(orientation,degrees=True)

        self.rigid_form=SingleRigidPrim(
            self.prim_path,
            name="rigid_flat_object",
            position=position,
            orientation=orientation,
            scale=scale,
        )

        self.physics_material_path="/World/Rigid_Flat_Object/PhysicsMaterial"
        self.world=world
        self.stage=world.stage

        self.physics_material=PhysicsMaterial(
            prim_path=self.physics_material_path,
            name="PhysicsMaterial",
            static_friction=2,
            dynamic_friction=1.8
        )

        self.geom_prim=SingleGeometryPrim(
            prim_path=self.prim_path,
            collision=True
        )

        self.geom_prim.set_collision_approximation("meshSimplification")


    def set_world_poses(self, position, orientation):
        """
        Args:
            positions: list of [x, y, z]
            orientations: list of [qx, qy, qz, qw]
        """
        self.rigid_form.set_world_pose(position=position, orientation=orientation)
    def set_mass(self, mass):
        self.rigid_form.set_mass(mass)
    
    def get_mass(self):
        return self.rigid_form.get_mass()

class Rigid_Xform():
    def __init__(self,world,path,position=None,orientation=None, scale=np.array([0.7, 0.7, 0.7])):
        self.path=path
        self.prim_path="/World/Rigid"
        add_reference_to_stage(usd_path=path,prim_path=self.prim_path)

        if position is None:
            position=np.array([0,0,0])
        if orientation is None:
            orientation=euler_angles_to_quat([90.0,0.,0.],degrees=True)
        else:
            orientation=euler_angles_to_quat(orientation,degrees=True)
        self.rigid_form=SingleXFormPrim(
            prim_path=self.prim_path,
            name="rigid",
            position=position,
            orientation=orientation,
            scale=scale,
        )
        
        self.geom_prim=SingleGeometryPrim(
            prim_path=self.prim_path,
            collision=True
        )
        self.geom_prim.set_collision_approximation("meshSimplification")


class Rigid():
    def __init__(self,world,path,position=None,orientation=None, scale=np.array([0.7, 0.7, 0.7])):
        self.path=path
        self.prim_path="/World/Rigid"
        add_reference_to_stage(usd_path=path,prim_path=self.prim_path)
        if position is None:
            position=np.array([0,0,0])
            
        if orientation is None:
            orientation=euler_angles_to_quat([90.0,0.,0.],degrees=True, extrinsic=False)
        else:
            orientation=euler_angles_to_quat(orientation,degrees=True, extrinsic=False)
        

        self.rigid_form=SingleRigidPrim(
            self.prim_path,
            name="rigid",
            position=position,
            orientation=orientation,
            scale=scale,
        )
      
        self.physics_material_path="/World/Rigid/PhysicsMaterial"
        self.world= world
        self.stage= world.stage

        self.physics_material=PhysicsMaterial(
            prim_path=self.physics_material_path,
            name="PhysicsMaterial",
            static_friction=0.4,
            dynamic_friction=0.35
        )

        physicsUtils.add_physics_material_to_prim(
            self.stage,
            self.stage.GetPrimAtPath("/World/Rigid"),
            self.physics_material_path
        )
        self.geom_prim=SingleGeometryPrim(
            prim_path=self.prim_path,
            collision=True
        )
        self.geom_prim.set_collision_approximation("meshSimplification")

    def set_world_poses(self, position, orientation):
        """
        Args:
            positions: list of [x, y, z]
            orientations: list of [qx, qy, qz, qw]
        """
        self.rigid_form.set_world_pose(position=position, orientation=orientation)
    def set_mass(self, mass):
        self.rigid_form.set_mass(mass)
    
    def get_mass(self):
        return self.rigid_form.get_mass()   

class Board():
    def __init__(self,path,position=None,orientation=None, scale=np.array([0.7, 0.7, 0.7])):
        self.path=path
        self.prim_path="/World/Board"
        add_reference_to_stage(usd_path=path,prim_path=self.prim_path)
        if position is None:
            position=np.array([0,0,0])
            
        if orientation is None:
            orientation=euler_angles_to_quat([90.0,0.,0.],degrees=True, extrinsic=False)
        else:
            orientation=euler_angles_to_quat(orientation,degrees=True, extrinsic=False)
        self.rigid_form=SingleRigidPrim(
            self.prim_path,
            name="board",
            position=position,
            orientation=orientation,
            scale=scale,
        )
        self.geom_prim=SingleGeometryPrim(
            prim_path=self.prim_path,
            collision=True
        )
        self.geom_prim.set_collision_approximation("meshSimplification")

    def set_world_poses(self, position, orientation):
        """
        Args:
            positions: list of [x, y, z]
            orientations: list of [qx, qy, qz, qw]
        """
        self.rigid_form.set_world_pose(position=position, orientation=orientation)
    def set_mass(self, mass):
        self.rigid_form.set_mass(mass)
    
    def get_mass(self):
        return self.rigid_form.get_mass()                     

class Book():
    def __init__(self, world, path,position=None,orientation=None, scale=np.array([0.7, 0.7, 0.7])):
        self.path=path
        self.prim_path="/World/Book"
        add_reference_to_stage(usd_path=path,prim_path=self.prim_path)
        if position is None:
            position=np.array([0,0,0])
            
        if orientation is None:
            orientation=euler_angles_to_quat([90.0,0.,0.],degrees=True, extrinsic=False)
        else:
            orientation=euler_angles_to_quat(orientation,degrees=True, extrinsic=False)
        self.rigid_form=SingleRigidPrim(
            self.prim_path,
            name="book",
            position=position,
            orientation=orientation,
            scale=scale,
        )
        self.geom_prim=SingleGeometryPrim(
            prim_path=self.prim_path,
            collision=True
        )
        self.geom_prim.set_collision_approximation("meshSimplification")


        self.physics_material_path="/World/Book/PhysicsMaterial"
        self.world=world
        self.stage=world.stage
        self.physics_material=PhysicsMaterial(
            prim_path=self.physics_material_path,
            name="PhysicsMaterial",
            static_friction=0.6,
            dynamic_friction=0.4
        )
        physicsUtils.add_physics_material_to_prim(
            self.stage,
            self.stage.GetPrimAtPath("/World/Book/Book1"),
            self.physics_material_path
        )

    def set_world_poses(self, position, orientation):
        """
        Args:
            positions: list of [x, y, z]
            orientations: list of [qx, qy, qz, qw]
        """
        self.rigid_form.set_world_pose(position=position, orientation=orientation)
    
    def set_mass(self, mass):
        self.rigid_form.set_mass(mass)
    
    def get_mass(self):
        return self.rigid_form.get_mass()

class Box():
    def __init__(self,world,path,position=None,orientation=None, scale=np.array([0.7, 0.7, 0.7])):
        self.path=path
        self.prim_path="/World/Box"
        add_reference_to_stage(usd_path=path,prim_path=self.prim_path)
        if position is None:
            position=np.array([0,0,0])
            
        if orientation is None:
            orientation=euler_angles_to_quat([90.0,0.,0.],degrees=True, extrinsic=False)
        else:
            orientation=euler_angles_to_quat(orientation,degrees=True, extrinsic=False)
        self.rigid_form=SingleRigidPrim(
            self.prim_path,
            name="box",
            position=position,
            orientation=orientation,
            scale=scale,
        )
        self.geom_prim=SingleGeometryPrim(
            prim_path=self.prim_path,
            collision=True
        )
        self.geom_prim.set_collision_approximation("meshSimplification")

    def set_world_poses(self, position, orientation):
        """
        Args:
            positions: list of [x, y, z]
            orientations: list of [qx, qy, qz, qw]
        """
        self.rigid_form.set_world_pose(position=position, orientation=orientation)
    def set_mass(self, mass):
        self.rigid_form.set_mass(mass)
    
    def get_mass(self):
        return self.rigid_form.get_mass()

class Chessboard():
    def __init__(self,path,position=None,orientation=None, scale=np.array([0.7, 0.7, 0.7])):
        self.path=path
        self.prim_path="/World/Chessboard"
        add_reference_to_stage(usd_path=path,prim_path=self.prim_path)

        if position is None:
            position=np.array([0,0,0])
        if orientation is None:
            orientation=euler_angles_to_quat([90.0,0.,0.],degrees=True)
        else:
            orientation=euler_angles_to_quat(orientation,degrees=True)

        self.rigid_form=SingleRigidPrim(
            self.prim_path,
            name="chessboard",
            position=position,
            orientation=orientation,
            scale=scale,
        )
        
        self.geom_prim=SingleGeometryPrim(
            prim_path=self.prim_path,
            collision=True
        )
        self.geom_prim.set_collision_approximation("meshSimplification")

    def set_world_poses(self, position, orientation):
        """
        Args:
            positions: list of [x, y, z]
            orientations: list of [qx, qy, qz, qw]
        """
        self.rigid_form.set_world_pose(position=position, orientation=orientation)

class Clock():
    def __init__(self,world,path,position=None,orientation=None, scale=np.array([0.7, 0.7, 0.7])):
        self.path=path
        self.prim_path="/World/Clock"
        add_reference_to_stage(usd_path=path,prim_path=self.prim_path)

        if position is None:
            position=np.array([0,0,0])
        if orientation is None:
            orientation=euler_angles_to_quat([90.0,0.,0.],degrees=True)
        else:
            orientation=euler_angles_to_quat(orientation,degrees=True)

        self.rigid_form=SingleRigidPrim(
            self.prim_path,
            name="clock",
            position=position,
            orientation=orientation,
            scale=scale,
        )
        
        self.geom_prim=SingleGeometryPrim(
            prim_path=self.prim_path,
            collision=True
        )
        self.geom_prim.set_collision_approximation("meshSimplification")

    def set_world_poses(self, position, orientation):
        """
        Args:
            positions: list of [x, y, z]
            orientations: list of [qx, qy, qz, qw]
        """
        self.rigid_form.set_world_pose(position=position, orientation=orientation)


class Clothes():
    def __init__(self,path,position=None,orientation=None, scale=np.array([0.7, 0.7, 0.7])):
        self.path=path
        self.prim_path="/World/Clothes"
        add_reference_to_stage(usd_path=path,prim_path=self.prim_path)

        if position is None:
            position=np.array([0,0,0])
        if orientation is None:
            orientation=euler_angles_to_quat([90.0,0.,0.],degrees=True)
        else:
            orientation=euler_angles_to_quat(orientation,degrees=True)

        self.rigid_form=SingleRigidPrim(
            self.prim_path,
            name="clothes",
            position=position,
            orientation=orientation,
            scale=scale,
        )
        
        self.geom_prim=SingleGeometryPrim(
            prim_path=self.prim_path,
            collision=True
        )
        self.geom_prim.set_collision_approximation("meshSimplification")

    def set_world_poses(self, position, orientation):
        """
        Args:
            positions: list of [x, y, z]
            orientations: list of [qx, qy, qz, qw]
        """
        self.rigid_form.set_world_pose(position=position, orientation=orientation)

class Cutting_Board():
    def __init__(self, world, path,position=None,orientation=None, scale=np.array([0.7, 0.7, 0.7])):
        self.path=path
        self.prim_path="/World/Cutting_Board"
        add_reference_to_stage(usd_path=path,prim_path=self.prim_path)

        if position is None:
            position=np.array([0,0,0])
        if orientation is None:
            orientation=euler_angles_to_quat([90.0,0.,0.],degrees=True)
        else:
            orientation=euler_angles_to_quat(orientation,degrees=True)

        self.rigid_form=SingleRigidPrim(
            self.prim_path,
            name="cutting_board",
            position=position,
            orientation=orientation,
            scale=scale,
        )

        self.physics_material_path="/World/Cutting_Board/PhysicsMaterial"
        self.world=world
        self.stage=world.stage

        self.physics_material=PhysicsMaterial(
            prim_path=self.physics_material_path,
            name="PhysicsMaterial",
            static_friction=1.2,
            dynamic_friction=1
        )

        physicsUtils.add_physics_material_to_prim(
            self.stage,
            self.stage.GetPrimAtPath("/World/Cutting_Board/Cutting_Board1"),
            self.physics_material_path
        )

        self.geom_prim=SingleGeometryPrim(
            prim_path=self.prim_path,
            collision=True
        )
        self.geom_prim.set_collision_approximation("meshSimplification")

    def set_world_poses(self, position, orientation):
        """
        Args:
            positions: list of [x, y, z]
            orientations: list of [qx, qy, qz, qw]
        """
        self.rigid_form.set_world_pose(position=position, orientation=orientation)
    def set_mass(self, mass):
        self.rigid_form.set_mass(mass)
    
    def get_mass(self):
        return self.rigid_form.get_mass()

class Disk():
    def __init__(self, world, path,position=None,orientation=None, scale=np.array([0.7, 0.7, 0.7])):
        self.path=path
        self.prim_path="/World/Disk"
        add_reference_to_stage(usd_path=path,prim_path=self.prim_path)

        if position is None:
            position=np.array([0,0,0])
        if orientation is None:
            orientation=euler_angles_to_quat([90.0,0.,0.],degrees=True)
        else:
            orientation=euler_angles_to_quat(orientation,degrees=True)

        self.rigid_form=SingleRigidPrim(
            self.prim_path,
            name="disk",
            position=position,
            orientation=orientation,
            scale=scale,
        )

        self.physics_material_path="/World/Disk/PhysicsMaterial"
        self.world=world
        self.stage=world.stage

        self.physics_material=PhysicsMaterial(
            prim_path=self.physics_material_path,
            name="PhysicsMaterial",
            static_friction=0.2,
            dynamic_friction=0.1
        )

        physicsUtils.add_physics_material_to_prim(
            self.stage,
            self.stage.GetPrimAtPath("/World/Disk/Disk2"),
            self.physics_material_path
        )

        self.geom_prim=SingleGeometryPrim(
            prim_path=self.prim_path,
            collision=True
        )
        self.geom_prim.set_collision_approximation("meshSimplification")
    
    def set_world_poses(self, position, orientation):
        """
        Args:
            positions: list of [x, y, z]
            orientations: list of [qx, qy, qz, qw]
        """
        self.rigid_form.set_world_pose(position=position, orientation=orientation)
    def set_mass(self, mass):
        self.rigid_form.set_mass(mass)
    
    def get_mass(self):
        return self.rigid_form.get_mass()

class Irregular_Shape():
    def __init__(self,path,position=None,orientation=None, scale=np.array([0.7, 0.7, 0.7])):
        self.path=path
        self.prim_path="/World/Irregular_Shape"
        add_reference_to_stage(usd_path=path,prim_path=self.prim_path)

        if position is None:
            position=np.array([0,0,0])
        if orientation is None:
            orientation=euler_angles_to_quat([90.0,0.,0.],degrees=True)
        else:
            orientation=euler_angles_to_quat(orientation,degrees=True)

        self.rigid_form=SingleRigidPrim(
            self.prim_path,
            name="irregular_shape",
            position=position,
            orientation=orientation,
            scale=scale,
        )
        
        self.geom_prim=SingleGeometryPrim(
            prim_path=self.prim_path,
            collision=True
        )
        self.geom_prim.set_collision_approximation("meshSimplification")
    
    def set_world_poses(self, position, orientation):
        """
        Args:
            positions: list of [x, y, z]
            orientations: list of [qx, qy, qz, qw]
        """
        self.rigid_form.set_world_pose(position=position, orientation=orientation)

class Keyboard():
    def __init__(self, world, path, position=None, orientation=None, scale=np.array([0.7, 0.7, 0.7]),
                 static_friction=0.5, dynamic_friction=0.5, restitution=0.1):
        
        self.path = path
        self.prim_path = "/World/Keyboard"
        add_reference_to_stage(usd_path=path, prim_path=self.prim_path)
        
        if position is None:
            position = np.array([0, 0, 0])
        if orientation is None:
            orientation = euler_angles_to_quat([90.0, 0., 0.], degrees=True)
        else:
            orientation = euler_angles_to_quat(orientation, degrees=True)
        
        self.rigid_form = SingleRigidPrim(
            self.prim_path,
            name="keyboard",
            position=position,
            orientation=orientation,
            scale=scale,
        )
        
        self.physics_material_path="/World/Keyboard/PhysicsMaterial"
        self.world=world
        self.stage=world.stage

        self.physics_material=PhysicsMaterial(
            prim_path="/World/Keyboard/PhysicsMaterial",
            name="PhysicsMaterial",
            static_friction=0.2,
            dynamic_friction=0.1
        )
        
        physicsUtils.add_physics_material_to_prim(
            self.stage,
            self.stage.GetPrimAtPath("/World/Keyboard/Keyboard1"),
            self.physics_material_path
        )

        self.geom_prim = SingleGeometryPrim(
            prim_path=self.prim_path,
            collision=True
        )
        self.geom_prim.set_collision_approximation("meshSimplification")
        
        self.static_friction = static_friction
        self.dynamic_friction = dynamic_friction
        self.restitution = restitution
        self._create_physics_material(static_friction, dynamic_friction, restitution)
    
    def _create_physics_material(self, static_friction, dynamic_friction, restitution):
        """
        Create physics material without binding
        """
        stage = prim_utils.get_current_stage()
        material_path = f"{self.prim_path}/PhysicsMaterial"
        material_prim = stage.DefinePrim(material_path, "Material")
        UsdPhysics.MaterialAPI.Apply(material_prim)
        PhysxSchema.PhysxMaterialAPI.Apply(material_prim)
        material_prim.CreateAttribute("physics:staticFriction", Sdf.ValueTypeNames.Float).Set(static_friction)
        material_prim.CreateAttribute("physics:dynamicFriction", Sdf.ValueTypeNames.Float).Set(dynamic_friction)
        material_prim.CreateAttribute("physics:restitution", Sdf.ValueTypeNames.Float).Set(restitution)
    
    def bind_physics_material(self):
        """
        Bind physics material to Xform child nodes
        This method should be called after the scene is fully loaded
        """
        stage = prim_utils.get_current_stage()
        material_path = f"{self.prim_path}/PhysicsMaterial"
        material_prim = stage.GetPrimAtPath(material_path)
        
        if not material_prim.IsValid():
            print(f"Warning: Material at {material_path} is invalid!")
            return
        
        material = UsdShade.Material(material_prim)
        root_prim = stage.GetPrimAtPath(self.prim_path)
        xform_children = []
        
        for child in root_prim.GetChildren():
            if child.IsA(UsdGeom.Xform):
                xform_children.append(child)
        
        print(f"Binding material to {len(xform_children)} Xform children:")
        for xform_prim in xform_children:
            binding_api = UsdShade.MaterialBindingAPI(xform_prim)
            binding_api.Bind(material, UsdShade.Tokens.weakerThanDescendants, "physics")
        
        return len(xform_children)
    
    def set_physics_material(self, static_friction=None, dynamic_friction=None, restitution=None):
        """
        Update physics material parameters
        Args:
            static_friction: Static friction coefficient
            dynamic_friction: Dynamic friction coefficient
            restitution: Restitution coefficient
        """
        stage = prim_utils.get_current_stage()
        material_path = f"{self.prim_path}/PhysicsMaterial"
        material_prim = stage.GetPrimAtPath(material_path)
        
        if material_prim.IsValid():
            if static_friction is not None:
                attr = material_prim.GetAttribute("physics:staticFriction")
                if attr:
                    attr.Set(static_friction)
                    self.static_friction = static_friction
            if dynamic_friction is not None:
                attr = material_prim.GetAttribute("physics:dynamicFriction")
                if attr:
                    attr.Set(dynamic_friction)
                    self.dynamic_friction = dynamic_friction
            if restitution is not None:
                attr = material_prim.GetAttribute("physics:restitution")
                if attr:
                    attr.Set(restitution)
                    self.restitution = restitution
    
    def set_world_poses(self, position, orientation):
        self.rigid_form.set_world_pose(position=position, orientation=orientation)
    
    def set_mass(self, mass):
        self.rigid_form.set_mass(mass)
    
    def get_mass(self):
        return self.rigid_form.get_mass()

class Laptop():
    def __init__(self,path,position=None,orientation=None, scale=np.array([0.7, 0.7, 0.7])):
        self.path=path
        self.prim_path="/World/Laptop"
        add_reference_to_stage(usd_path=path,prim_path=self.prim_path)

        if position is None:
            position=np.array([0,0,0])
        if orientation is None:
            orientation=euler_angles_to_quat([90.0,0.,0.],degrees=True)
        else:
            orientation=euler_angles_to_quat(orientation,degrees=True)

        self.rigid_form=SingleRigidPrim(
            self.prim_path,
            name="laptop",
            position=position,
            orientation=orientation,
            scale=scale,
        )
        
        self.geom_prim=SingleGeometryPrim(
            prim_path=self.prim_path,
            collision=True
        )
        self.geom_prim.set_collision_approximation("meshSimplification")

    def set_world_poses(self, position, orientation):
        """
        Args:
            positions: list of [x, y, z]
            orientations: list of [qx, qy, qz, qw]
        """
        self.rigid_form.set_world_pose(position=position, orientation=orientation)

class Letter():
    def __init__(self,path,position=None,orientation=None, scale=np.array([0.7, 0.7, 0.7])):
        self.path=path
        self.prim_path="/World/Letter"
        add_reference_to_stage(usd_path=path,prim_path=self.prim_path)

        if position is None:
            position=np.array([0,0,0])
        if orientation is None:
            orientation=euler_angles_to_quat([90.0,0.,0.],degrees=True)
        else:
            orientation=euler_angles_to_quat(orientation,degrees=True)

        self.rigid_form=SingleRigidPrim(
            self.prim_path,
            name="letter",
            position=position,
            orientation=orientation,
            scale=scale,
        )
        
        self.geom_prim=SingleGeometryPrim(
            prim_path=self.prim_path,
            collision=True
        )
        self.geom_prim.set_collision_approximation("meshSimplification")

    def set_world_poses(self, position, orientation):
        """
        Args:
            positions: list of [x, y, z]
            orientations: list of [qx, qy, qz, qw]
        """
        self.rigid_form.set_world_pose(position=position, orientation=orientation)

class Magazine():
    def __init__(self,path,position=None,orientation=None, scale=np.array([0.7, 0.7, 0.7])):
        self.path=path
        self.prim_path="/World/Magazine"
        add_reference_to_stage(usd_path=path,prim_path=self.prim_path)

        if position is None:
            position=np.array([0,0,0])
        if orientation is None:
            orientation=euler_angles_to_quat([90.0,0.,0.],degrees=True)
        else:
            orientation=euler_angles_to_quat(orientation,degrees=True)

        self.rigid_form=SingleRigidPrim(
            self.prim_path,
            name="magazine",
            position=position,
            orientation=orientation,
            scale=scale,
        )
        
        self.geom_prim=SingleGeometryPrim(
            prim_path=self.prim_path,
            collision=True
        )
        self.geom_prim.set_collision_approximation("meshSimplification")

    def set_world_poses(self, position, orientation):
        """
        Args:
            positions: list of [x, y, z]
            orientations: list of [qx, qy, qz, qw]
        """
        self.rigid_form.set_world_pose(position=position, orientation=orientation)

class Mouse_Pad():
    def __init__(self,path,position=None,orientation=None, scale=np.array([0.7, 0.7, 0.7])):
        self.path=path
        self.prim_path="/World/Mouse_Pad"
        add_reference_to_stage(usd_path=path,prim_path=self.prim_path)

        if position is None:
            position=np.array([0,0,0])
        if orientation is None:
            orientation=euler_angles_to_quat([90.0,0.,0.],degrees=True)
        else:
            orientation=euler_angles_to_quat(orientation,degrees=True)

        self.rigid_form=SingleRigidPrim(
            self.prim_path,
            name="mouse_pad",
            position=position,
            orientation=orientation,
            scale=scale,
        )
        
        self.geom_prim=SingleGeometryPrim(
            prim_path=self.prim_path,
            collision=True
        )
        self.geom_prim.set_collision_approximation("meshSimplification")

    def set_world_poses(self, position, orientation):
        """
        Args:
            positions: list of [x, y, z]
            orientations: list of [qx, qy, qz, qw]
        """
        self.rigid_form.set_world_pose(position=position, orientation=orientation)

class Notebook():
    def __init__(self,path,position=None,orientation=None, scale=np.array([0.7, 0.7, 0.7])):
        self.path=path
        self.prim_path="/World/Notebook"
        add_reference_to_stage(usd_path=path,prim_path=self.prim_path)

        if position is None:
            position=np.array([0,0,0])
        if orientation is None:
            orientation=euler_angles_to_quat([90.0,0.,0.],degrees=True)
        else:
            orientation=euler_angles_to_quat(orientation,degrees=True)

        self.rigid_form=SingleRigidPrim(
            self.prim_path,
            name="notebook",
            position=position,
            orientation=orientation,
            scale=scale,
        )
        
        self.geom_prim=SingleGeometryPrim(
            prim_path=self.prim_path,
            collision=True
        )
        self.geom_prim.set_collision_approximation("meshSimplification")

    def set_world_poses(self, position, orientation):
        """
        Args:
            positions: list of [x, y, z]
            orientations: list of [qx, qy, qz, qw]
        """
        self.rigid_form.set_world_pose(position=position, orientation=orientation)

class Pad():
    def __init__(self,path,position=None,orientation=None, scale=np.array([0.7, 0.7, 0.7])):
        self.path=path
        self.prim_path="/World/Pad"
        add_reference_to_stage(usd_path=path,prim_path=self.prim_path)

        if position is None:
            position=np.array([0,0,0])
        if orientation is None:
            orientation=euler_angles_to_quat([90.0,0.,0.],degrees=True)
        else:
            orientation=euler_angles_to_quat(orientation,degrees=True)

        self.rigid_form=SingleRigidPrim(
            self.prim_path,
            name="pad",
            position=position,
            orientation=orientation,
            scale=scale,
        )
        self.geom_prim=SingleGeometryPrim(
            prim_path=self.prim_path,
            collision=True
        )
        self.geom_prim.set_collision_approximation("convexHull")

    def set_world_poses(self, position, orientation):
        """
        Args:
            positions: list of [x, y, z]
            orientations: list of [qx, qy, qz, qw]
        """
        self.rigid_form.set_world_pose(position=position, orientation=orientation)

class Painting():
    def __init__(self,path,position=None,orientation=None, scale=np.array([0.7, 0.7, 0.7])):
        self.path=path
        self.prim_path="/World/Painting"
        add_reference_to_stage(usd_path=path,prim_path=self.prim_path)

        if position is None:
            position=np.array([0,0,0])
        if orientation is None:
            orientation=euler_angles_to_quat([90.0,0.,0.],degrees=True)
        else:
            orientation=euler_angles_to_quat(orientation,degrees=True)

        self.rigid_form=SingleRigidPrim(
            self.prim_path,
            name="painting",
            position=position,
            orientation=orientation,
            scale=scale,
        )
        
        self.geom_prim=SingleGeometryPrim(
            prim_path=self.prim_path,
            collision=True
        )
        self.geom_prim.set_collision_approximation("meshSimplification")

    def set_world_poses(self, position, orientation):
        """
        Args:
            positions: list of [x, y, z]
            orientations: list of [qx, qy, qz, qw]
        """
        self.rigid_form.set_world_pose(position=position, orientation=orientation)

class Phone():
    def __init__(self,path,position=None,orientation=None, scale=np.array([0.7, 0.7, 0.7])):
        self.path=path
        self.prim_path="/World/Phone"
        add_reference_to_stage(usd_path=path,prim_path=self.prim_path)

        if position is None:
            position=np.array([0,0,0])
        if orientation is None:
            orientation=euler_angles_to_quat([90.0,0.,0.],degrees=True)
        else:
            orientation=euler_angles_to_quat(orientation,degrees=True)

        self.rigid_form=SingleRigidPrim(
            self.prim_path,
            name="phone",
            position=position,
            orientation=orientation,
            scale=scale,
        )
        
        self.geom_prim=SingleGeometryPrim(
            prim_path=self.prim_path,
            collision=True
        )
        self.geom_prim.set_collision_approximation("meshSimplification")

    def set_world_poses(self, position, orientation):
        """
        Args:
            positions: list of [x, y, z]
            orientations: list of [qx, qy, qz, qw]
        """
        self.rigid_form.set_world_pose(position=position, orientation=orientation)

class Photo_Album():
    def __init__(self,path,position=None,orientation=None, scale=np.array([0.7, 0.7, 0.7])):
        self.path=path
        self.prim_path="/World/Photo_Album"
        add_reference_to_stage(usd_path=path,prim_path=self.prim_path)

        if position is None:
            position=np.array([0,0,0])
        if orientation is None:
            orientation=euler_angles_to_quat([90.0,0.,0.],degrees=True)
        else:
            orientation=euler_angles_to_quat(orientation,degrees=True)

        self.rigid_form=SingleRigidPrim(
            self.prim_path,
            name="photo_album",
            position=position,
            orientation=orientation,
            scale=scale,
        )
        
        self.geom_prim=SingleGeometryPrim(
            prim_path=self.prim_path,
            collision=True
        )
        self.geom_prim.set_collision_approximation("meshSimplification")

    def set_world_poses(self, position, orientation):
        """
        Args:
            positions: list of [x, y, z]
            orientations: list of [qx, qy, qz, qw]
        """
        self.rigid_form.set_world_pose(position=position, orientation=orientation)

class Plate():
    def __init__(self,path,position=None,orientation=None, scale=np.array([0.7, 0.7, 0.7])):
        self.path=path
        self.prim_path="/World/Plate"
        add_reference_to_stage(usd_path=path,prim_path=self.prim_path)

        if position is None:
            position=np.array([0,0,0])
        if orientation is None:
            orientation=euler_angles_to_quat([90.0,0.,0.],degrees=True)
        else:
            orientation=euler_angles_to_quat(orientation,degrees=True)

        self.rigid_form=SingleRigidPrim(
            self.prim_path,
            name="plate",
            position=position,
            orientation=orientation,
            scale=scale,
        )
        
        self.geom_prim=SingleGeometryPrim(
            prim_path=self.prim_path,
            collision=True
        )
        self.geom_prim.set_collision_approximation("meshSimplification")

    def set_world_poses(self, position, orientation):
        """
        Args:
            positions: list of [x, y, z]
            orientations: list of [qx, qy, qz, qw]
        """
        self.rigid_form.set_world_pose(position=position, orientation=orientation)

class Regular_Shape():
    def __init__(self,path,position=None,orientation=None, scale=np.array([0.7, 0.7, 0.7])):
        self.path=path
        self.prim_path="/World/Regular_Shape"
        add_reference_to_stage(usd_path=path,prim_path=self.prim_path)

        if position is None:
            position=np.array([0,0,0])
        if orientation is None:
            orientation=euler_angles_to_quat([90.0,0.,0.],degrees=True)
        else:
            orientation=euler_angles_to_quat(orientation,degrees=True)

        self.rigid_form=SingleRigidPrim(
            self.prim_path,
            name="regular_shape",
            position=position,
            orientation=orientation,
            scale=scale,
        )
        
        self.geom_prim=SingleGeometryPrim(
            prim_path=self.prim_path,
            collision=True
        )
        self.geom_prim.set_collision_approximation("meshSimplification")

    def set_world_poses(self, position, orientation):
        """
        Args:
            positions: list of [x, y, z]
            orientations: list of [qx, qy, qz, qw]
        """
        self.rigid_form.set_world_pose(position=position, orientation=orientation)

class Ruler():
    def __init__(self,path,position=None,orientation=None, scale=np.array([0.7, 0.7, 0.7])):
        self.path=path
        self.prim_path="/World/Ruler"
        add_reference_to_stage(usd_path=path,prim_path=self.prim_path)

        if position is None:
            position=np.array([0,0,0])
        if orientation is None:
            orientation=euler_angles_to_quat([90.0,0.,0.],degrees=True)
        else:
            orientation=euler_angles_to_quat(orientation,degrees=True)

        self.rigid_form=SingleRigidPrim(
            self.prim_path,
            name="ruler",
            position=position,
            orientation=orientation,
            scale=scale,
        )
        
        self.geom_prim=SingleGeometryPrim(
            prim_path=self.prim_path,
            collision=True
        )
        self.geom_prim.set_collision_approximation("meshSimplification")
    
    def set_world_poses(self, position, orientation):
        """
        Args:
            positions: list of [x, y, z]
            orientations: list of [qx, qy, qz, qw]
        """
        self.rigid_form.set_world_pose(position=position, orientation=orientation)

class Tape():
    def __init__(self,path,position=None,orientation=None, scale=np.array([0.7, 0.7, 0.7])):
        self.path=path
        self.prim_path="/World/Tape"
        add_reference_to_stage(usd_path=path,prim_path=self.prim_path)

        if position is None:
            position=np.array([0,0,0])
        if orientation is None:
            orientation=euler_angles_to_quat([90.0,0.,0.],degrees=True)
        else:
            orientation=euler_angles_to_quat(orientation,degrees=True)

        self.rigid_form=SingleRigidPrim(
            self.prim_path,
            name="tape",
            position=position,
            orientation=orientation,
            scale=scale,
        )
        
        self.geom_prim=SingleGeometryPrim(
            prim_path=self.prim_path,
            collision=True
        )
        self.geom_prim.set_collision_approximation("meshSimplification")

    def set_world_poses(self, position, orientation):
        """
        Args:
            positions: list of [x, y, z]
            orientations: list of [qx, qy, qz, qw]
        """
        self.rigid_form.set_world_pose(position=position, orientation=orientation)

class Towel():
    def __init__(self,path,position=None,orientation=None, scale=np.array([0.7, 0.7, 0.7])):
        self.path=path
        self.prim_path="/World/Towel"
        add_reference_to_stage(usd_path=path,prim_path=self.prim_path)

        if position is None:
            position=np.array([0,0,0])
        if orientation is None:
            orientation=euler_angles_to_quat([90.0,0.,0.],degrees=True)
        else:
            orientation=euler_angles_to_quat(orientation,degrees=True)

        self.rigid_form=SingleRigidPrim(
            self.prim_path,
            name="towel",
            position=position,
            orientation=orientation,
            scale=scale,
        )
        
        self.geom_prim=SingleGeometryPrim(
            prim_path=self.prim_path,
            collision=True
        )
        self.geom_prim.set_collision_approximation("meshSimplification")

    def set_world_poses(self, position, orientation):
        """
        Args:
            positions: list of [x, y, z]
            orientations: list of [qx, qy, qz, qw]
        """
        self.rigid_form.set_world_pose(position=position, orientation=orientation)