from ._base_task import Base_Task
from .utils import *
import math
import sapien


class place_shoe(Base_Task):

    def setup_demo(self, is_test=False, **kwags):
        super()._init_task_env_(**kwags)

    def load_actors(self):
        self.target = create_box(
            scene=self,
            pose=sapien.Pose([0, -0.08, 0.74], [1, 0, 0, 0]),
            half_size=(0.13, 0.05, 0.0005),
            color=(0, 0, 1),
            is_static=True,
            name="box",
        )
        self.target.config["functional_matrix"] = [[
            [0.0, -1.0, 0.0, 0.0],
            [-1.0, 0.0, 0.0, 0.0],
            [0.0, 0.0, -1.0, 0],
            [0.0, 0.0, 0.0, 1.0],
        ], [
            [0.0, -1.0, 0.0, 0.0],
            [-1.0, 0.0, 0.0, 0.0],
            [0.0, 0.0, -1.0, 0],
            [0.0, 0.0, 0.0, 1.0],
        ]]

        shoes_pose = rand_pose(
            xlim=[-0.25, 0.25],
            ylim=[-0.1, 0.05],
            ylim_prop=True,
            rotate_rand=True,
            rotate_lim=[0, 3.14, 0],
            qpos=[0.707, 0.707, 0, 0],
        )
        while np.sum(pow(shoes_pose.get_p()[:2] - np.zeros(2), 2)) < 0.0225:
            shoes_pose = rand_pose(
                xlim=[-0.25, 0.25],
                ylim=[-0.1, 0.05],
                ylim_prop=True,
                rotate_rand=True,
                rotate_lim=[0, 3.14, 0],
                qpos=[0.707, 0.707, 0, 0],
            )
        self.shoe_id = np.random.choice([i for i in range(10)])
        self.shoe = create_actor(
            scene=self,
            pose=shoes_pose,
            modelname="041_shoe",
            convex=True,
            model_id=self.shoe_id,
        )

        self.prohibited_area.append([-0.2, -0.15, 0.2, -0.01])
        self.add_prohibit_area(self.shoe, padding=0.1)

    def play_once(self):
        shoe_pose = self.shoe.get_pose().p
        arm_tag = ArmTag("left" if shoe_pose[0] < 0 else "right")

        # Grasp the shoe with specified pre-grasp distance and gripper position
        self.move(self.grasp_actor(self.shoe, arm_tag=arm_tag, pre_grasp_dis=0.1, gripper_pos=0))

        # Lift the shoe up by 0.07 meters in z-direction
        self.move(self.move_by_displacement(arm_tag=arm_tag, z=0.07))

        # Get target's functional point as target pose
        target_pose = self.target.get_functional_point(0)
        # Place the shoe on the target with alignment constraint and specified pre-placement distance
        self.move(
            self.place_actor(
                self.shoe,
                arm_tag=arm_tag,
                target_pose=target_pose,
                functional_point_id=0,
                pre_dis=0.12,
                constrain="align",
            ))
        # Open the gripper to release the shoe
        self.move(self.open_gripper(arm_tag=arm_tag))

        self.info["info"] = {"{A}": f"041_shoe/base{self.shoe_id}", "{a}": str(arm_tag)}
        return self.info

    def get_info(self):
        shoe_pose = self.shoe.get_pose().p
        arm_tag = ArmTag("left" if shoe_pose[0] < 0 else "right")
        info = {"{A}": f"041_shoe/base{self.shoe_id}", "{a}": str(arm_tag)}
        return info
    
    def check_success(self):
        shoe_pose_p = np.array(self.shoe.get_pose().p)
        shoe_pose_q = np.array(self.shoe.get_pose().q)
        if shoe_pose_q[0] < 0:
            shoe_pose_q *= -1
        target_pose_p = np.array([0, -0.08])
        target_pose_q = np.array([0.5, 0.5, -0.5, -0.5])
        eps = np.array([0.05, 0.02, 0.07, 0.07, 0.07, 0.07])
        return (np.all(abs(shoe_pose_p[:2] - target_pose_p) < eps[:2])
                and np.all(abs(shoe_pose_q - target_pose_q) < eps[-4:]) and self.is_left_gripper_open()
                and self.is_right_gripper_open())
