# simulation_app.close()
from isaacsim import SimulationApp
simulation_app = SimulationApp({"headless": True})

# load external package
import os
import sys
import time
import numpy as np
import open3d as o3d
from termcolor import cprint
import threading
import re
import json
# load isaac-relevant package
import omni.replicator.core as rep
import isaacsim.core.utils.prims as prims_utils
from pxr import UsdGeom, UsdPhysics
from isaacsim.core.api import World
from isaacsim.core.api import SimulationContext
from isaacsim.core.api.objects import FixedCuboid
from isaacsim.core.utils.prims import set_prim_visibility
from isaacsim.core.utils.viewports import set_camera_view
from isaacsim.core.prims import SingleParticleSystem
# load custom package
sys.path.append(os.getcwd())
from Env_StandAlone.BaseEnv import BaseEnv
from Env_Config.Garment.Particle_Garment import Particle_Garment
from Env_Config.Robot.Bimanual_Franka import Bimanual_Franka
from Env_Config.Camera.Recording_Camera import Recording_Camera
from Env_Config.Room.Real_Ground import Real_Ground
from Env_Config.Room.Object_Tools import set_prim_visible_group
from Env_Config.Utils_Project.Code_Tools import get_unique_filename
from Env_Config.Utils_Project.Parse import parse_args_record
from Env_Config.Utils_Project.Collision_Group import CollisionGroup
from Env_Config.Table.Table import Table
from Env_Config.Utils_Project.Object_Contact_Config import get_contact_config
from Env_Config.Utils_Project.Point_Cloud_Manip import furthest_point_sampling

from Env_Config.Utils_Project.Pos_Ori import calculate_target_from_world_contact
from isaacsim.core.utils.rotations import euler_angles_to_quat

# Convert data to serializable format
def _to_serializable(value):
    if isinstance(value, np.ndarray):
        return value.tolist()
    if isinstance(value, (np.integer, np.floating)):
        return value.item()
    if isinstance(value, (list, tuple)):
        return [_to_serializable(v) for v in value]
    if isinstance(value, dict):
        return {k: _to_serializable(v) for k, v in value.items()}
    return value

def _write_point_cloud_ply3(
    file_path: str,
    points_cloud,
    colors=None,
    save_or_not: bool = True,
    sample_flag: bool = True,
    sampled_point_num: int = 8192,
    real_time_watch: bool = False,
):
    if points_cloud is None:
        return
    pts = np.asarray(points_cloud)
    if sample_flag:
        points_cloud, colors = furthest_point_sampling(points_cloud, colors, sampled_point_num)
    pcd = o3d.geometry.PointCloud()
    pcd.points = o3d.utility.Vector3dVector(points_cloud)
    pcd.colors = o3d.utility.Vector3dVector(colors)
    if real_time_watch:
        o3d.visualization.draw_geometries([pcd])
    if save_or_not:
        o3d.io.write_point_cloud(file_path, pcd)

def _write_pose_txt(file_path: str, sample: dict):
    payload = {
        "joint_pos_L": _to_serializable(sample.get("joint_pos_L")),
        "joint_pos_R": _to_serializable(sample.get("joint_pos_R")),
        "ee_pose_L": _to_serializable(sample.get("ee_pose_L")),
        "ee_pose_R": _to_serializable(sample.get("ee_pose_R")),
    }
    with open(file_path, "w", encoding="utf-8") as handle:
        json.dump(payload, handle, ensure_ascii=True, indent=2)

def save_mid_samples(samples, target_dir: str):
    if not samples:
        cprint("[Save] No callback samples captured; skip mid outputs", "yellow")
        return
    os.makedirs(target_dir, exist_ok=True)
    for name in os.listdir(target_dir):
        if name.startswith("Mid") and (name.endswith("_env.ply") or name.endswith("_pos.txt")):
            try:
                os.remove(os.path.join(target_dir, name))
            except OSError:
                pass
    for idx, sample in enumerate(samples, start=1):
        env_path = os.path.join(target_dir, f"Mid{idx}_env.ply")
        pose_path = os.path.join(target_dir, f"Mid{idx}_pos.txt")
        _write_point_cloud_ply3(
            env_path,
            sample.get("env_point_cloud"),
            sample.get("env_point_cloud_color"),
        )
        _write_pose_txt(pose_path, sample)

class WearScarf_Env(BaseEnv):
    def __init__(
        self,
        pos: np.ndarray = None,
        ori: np.ndarray = None,
        usd_path: str = None,
        env_dx: float = 0.0,
        env_dy: float = 0.0,
        object_name: str = "Cutting_Board3",
        object_random_init: bool = False,
        ground_material_usd: str = None,
        record_video_flag: bool = False,

    ):
        # load BaseEnv
        super().__init__()

        self.ground = Real_Ground(
            self.scene, 
            visual_material_usd = ground_material_usd,
        )

        self.bimanual_franka = Bimanual_Franka(
            self.world,
            left_pos=np.array([0.57325, 0.52302,  0.52954]),
            left_ori=np.array([0.0, 0.0, -135.0]),
            left_franka_scale=np.array([1.0, 1.0, 1.0]),
            right_pos=np.array([-0.57325, 0.52302,  0.52954]),
            right_ori=np.array([0.0, 0.0, -45.0]),
            right_franka_scale=np.array([1.0, 1.0, 1.0]),
            gripper_open_position_nounits=np.array([0.05, 0.05])
        )
        self.Table = Table(
            self.world, 
            path=os.getcwd() + "/" +"Assets/Table/Collected_Willow/Wood.usd",
            position=[0.0, 0.02152, 0.0],
            orientation=[0.0, 0.0, 0.0],
            scale = np.array([0.0088, 0.0104, 0.01])
        )

        self.object_name = object_name
        self.category_name = re.sub(r"\d+$", "", self.object_name)
        
        self.object = Particle_Garment(
            self.world, 
            name = self.object_name,
            random_init=object_random_init,
            pos=np.array([0, 0.0, 0.8]),
            ori=np.array([0.0, 0.0, 0.0]),
            scale=np.array([0.8, 0.8, 0.8]),
            usd_path=os.path.join(
                os.getcwd(), "Assets/Flat_Object", self.category_name, f"{self.object_name}.usd"
            ),
            contact_offset=0.010,             
            rest_offset=0.0075,                
            particle_contact_offset=0.010,    
            adhesion=0.5,                     
            adhesion_offset_scale=0.0,        
            cohesion=0.0,                     
            particle_adhesion_scale=0.5,      
            particle_friction_scale=0.1,      
            friction=5.0,  
            lift=0.00,                  
            gravity_scale=1.5,                
            particle_mass=1e-2,               
            drag=3.5,
            damping=0.5,
        )

        self.helper_path=["/World/Table"]
        self.collisiongroup=CollisionGroup(
            self.world,
            helper_path=self.helper_path,
            garment=True,
            collide_with_garment=True,
            collide_with_robot=False
        )   

        self.recording_camera = Recording_Camera(
            camera_position=np.array([0.0, -5.04154, 3.61856]),
            camera_orientation=np.array([0.85717, 0.51504, 0.0, 0.0]),
            resolution=(2048, 1080),
            prim_path="/World/recording_camera",
        )
        if record_video_flag:
            self.thread_record = threading.Thread(
                target=self.recording_camera.collect_rgb_graph_for_video
            )
            self.thread_record.daemon = True

        self.recording_camera.initialize()
        self.obj_camera = Recording_Camera(
            camera_position=np.array([0.0, -3.1, 3.8]),
            camera_orientation=np.array([0.92388, 0.38268, 0.0, 0.0]),
            resolution=(600, 400),
            prim_path="/World/obj_camera",
        )
        self.env_camera = Recording_Camera(
            camera_position=np.array([0.0, -3.1, 3.8]),
            camera_orientation=np.array([0.92388, 0.38268, 0.0, 0.0]),
            resolution=(600, 400),
            prim_path="/World/env_camera",
        )

        self.obj_only_camera = Recording_Camera(
            camera_position=np.array([2.23, 0, 0.9]),
            camera_orientation=np.array([0.5,0.5, 0.5, 0.5]),
            resolution=(600, 400),
            prim_path="/World/obj_only_camera",
        )

        self.franka_only_camera = Recording_Camera(
            camera_position=np.array([2.23, 0, 0.9]),
            camera_orientation=np.array([0.5,0.5, 0.5, 0.5]),
            resolution=(600, 400),
            prim_path="/World/franka_only_camera",
        )
        cube = FixedCuboid(
            prim_path="/World/Cube",
            name="my_cube",
            position=np.array([-0.7, 0.0, 0.9]),
            scale=np.array([0.01, 0.01, 0.01]),
            color=np.array([0.0, 0.5, 1.0])
        )
        self.env_pcd = None
        self.flat_object_pcd = None     

        # initialize world
        self.reset()

        self.env_camera.initialize(segment_pc_enable=True, segment_prim_path_list=[f"/World/Garment", "/World/Table"])
        
        self.obj_only_camera.initialize(segment_pc_enable=True, segment_prim_path_list=[f"/World/Garment"])
        self.franka_only_camera.initialize(segment_pc_enable=True, segment_prim_path_list=[f"/World/Franka_Left"])
            
        for i in range(100):
            self.step()
        
        cprint("World Ready!", "green", "on_green")
        
    def record_callback(self, step_size):
        if self.step_num % 60 == 0:
            joint_pos_L = self.bimanual_franka.left_franka.get_joint_positions()
            joint_pos_R = self.bimanual_franka.right_franka.get_joint_positions()
            joint_state = np.array([*joint_pos_L, *joint_pos_R])
            self.callback_save_count += 1
            self.env_pcd, env_color = self.env_camera.get_point_cloud_data_from_segment(
                save_or_not=False,
                sample_flag=True,
                sampled_point_num=8192,
                real_time_watch=False,
            )

            pos_L, quat_L = self.bimanual_franka.left_franka.get_cur_ee_pos()
            pos_R, quat_R = self.bimanual_franka.right_franka.get_cur_ee_pos()
            ee_pose_L = np.array([*pos_L, *quat_L])
            ee_pose_R = np.array([*pos_R, *quat_R])
            self.saving_data.append({
                "env_point_cloud": self.env_pcd,
                "env_point_cloud_color": env_color,
                "joint_pos_L": joint_pos_L,
                "joint_pos_R": joint_pos_R,
                "ee_pose_L": ee_pose_L,
                "ee_pose_R": ee_pose_R,
            })
        
        self.step_num += 1

def get_point_cloud(env):
    set_prim_visible_group(prim_path_list=["/World/Table","/World/Franka_Left","/World/Franka_Right"], visible=False)
    for i in range(2):
        env.step()
    judge_flat_object_pcd, _ = env.obj_only_camera.get_point_cloud_data_from_segment(
        save_or_not=True,
        save_path=get_unique_filename("Data/Carry/train_data/judge_flat_object_pcd", extension=".ply"),
        sampled_point_num=8192,
        real_time_watch=False
    )    
    set_prim_visible_group(prim_path_list=["/World/Table","/World/Franka_Left","/World/Franka_Right"], visible=True)

    for i in range(10):
        env.step()
    set_prim_visible_group(prim_path_list=["/World/Table","/World/Garment","/World/Franka_Right"], visible=False)
    for i in range(2):
        env.step()
    judge_flat_franka_pcd, _ = env.franka_only_camera.get_point_cloud_data_from_segment(
        save_or_not=True,
        save_path=get_unique_filename("Data/Carry/train_data/judge_flat_franka_pcd", extension=".ply"),
        sampled_point_num=8192,
        real_time_watch=False
    )    
    set_prim_visible_group(prim_path_list=["/World/Table","/World/Garment","/World/Franka_Right"], visible=True)

    return judge_flat_object_pcd, judge_flat_franka_pcd

def judge_pcd(judge_flat_object_pcd,judge_flat_franka_pcd):
    obj_z_max = float(np.max(judge_flat_object_pcd[:, 2]))
    franka_z_min = float(np.min(judge_flat_franka_pcd[:, 2]))
    cprint(f"[Info]: obj_z_max: {obj_z_max}, franka_z_min: {franka_z_min}", "blue", attrs=["bold"])
    return obj_z_max > franka_z_min

def WearScarf(pos, ori, usd_path, env_dx, env_dy, object_name, ground_material_usd, data_collection_flag, record_video_flag, object_random_init=False, pos_index=None):
    
    env = WearScarf_Env(pos, ori, usd_path, env_dx, env_dy, object_name, object_random_init, ground_material_usd, record_video_flag)

    cprint(f"[Info]: data_collection_flag: {data_collection_flag}", "green", attrs=["bold"])
    cprint(f"[Info]: object_name: {object_name}", "green", attrs=["bold"])

    name_slug = (object_name or getattr(env, "object_name", None) or getattr(env.object, "name", None))
    if not name_slug:
        raise ValueError("object_name is not provided and cannot be inferred from the environment; please pass it explicitly via --object_name")
    name_slug = name_slug.lower()

    env.name_slug = name_slug
    env.pos_index = pos_index if pos_index is not None else 1
    env.callback_save_count = 0

    env.bimanual_franka.Gripper_Left_Open()
    cprint(f"[Info]: Gripper opened", "white", attrs=["bold"])
    
    obj_pos_start, _ = env.object.garment.get_world_pose()

    task_name = "StrategyC_V2"
    if not os.path.exists(f"Data/{task_name}/train_data/"):
                os.makedirs(f"Data/{task_name}/train_data/")
    if not os.path.exists(f"Data/{task_name}/pointcloud/"):
                os.makedirs(f"Data/{task_name}/pointcloud/")

    if data_collection_flag:
        set_prim_visible_group(prim_path_list=["/World/Franka_Left","/World/Franka_Right"], visible=False)
        for i in range(2):
            env.step()
        whole_point_cloud_pcd, whole_point_cloud_pcd_color = env.env_camera.get_point_cloud_data_from_segment(
            save_or_not=False,
            sampled_point_num=8192,
            real_time_watch=False
        )    
        set_prim_visible_group(prim_path_list=["/World/Franka_Left","/World/Franka_Right"], visible=True)
        cprint(f"[Info]: Initial environment point cloud obtained", "white", attrs=["bold"])

    object_name_initial = getattr(env.object, "name", "None")
    contact_cfg = get_contact_config(object_name_initial)
    c_target_position_left, _ = calculate_target_from_world_contact(
        contact_world_init=contact_cfg["contact_world_init_left"],
        initial_board_position=contact_cfg["initial_board_position"],
        initial_board_euler=contact_cfg.get("initial_board_euler"),
        initial_gripper_orientation_euler=contact_cfg.get("initial_gripper_orientation_euler_left"),
        current_board_position=env.object.my_position,
        current_board_euler=env.object.my_orientation
    )
    c_target_position_right, _ = calculate_target_from_world_contact(
        contact_world_init=contact_cfg["contact_world_init_right"],
        initial_board_position=contact_cfg["initial_board_position"],
        initial_board_euler=contact_cfg.get("initial_board_euler"),
        initial_gripper_orientation_euler=contact_cfg.get("initial_gripper_orientation_euler_right"),
        current_board_position=env.object.my_position,
        current_board_euler=env.object.my_orientation
    )

    c_target_orientation_left = np.array([-158.531, 1.167,  88.519])
    c_target_orientation_right = np.array([-179.653, 5.751, -179.268])

    e_mx = 0.05

    y_noise = round(np.random.uniform(-0.2, 0.2), 2)
    c_target_orientation_left[1] += y_noise
    c_target_orientation_right[1] += y_noise
    cprint(f"[Info]: Random perturbation added to c_target_position y: {y_noise}, new value: {c_target_orientation_left[1]}", "white")
    
    cprint(f"[Info]: Right arm moving down", "white", attrs=["bold"])
    env.bimanual_franka.Rmpflow_Right_Move(target_position=np.array([c_target_position_right[0], c_target_position_right[1], 0.882]), 
                                           target_orientation=c_target_orientation_right)
    env.bimanual_franka.Rmpflow_Right_Move(target_position=np.array([c_target_position_right[0], c_target_position_right[1], 0.782]), 
                                           target_orientation=c_target_orientation_right)

    cprint(f"[Info]: Collect right arm start position info separately","white")
    joint_pos_L = env.bimanual_franka.left_franka.get_joint_positions()
    joint_pos_R = env.bimanual_franka.right_franka.get_joint_positions()
    pos_L, quat_L = env.bimanual_franka.left_franka.get_cur_ee_pos()
    pos_R, quat_R = env.bimanual_franka.right_franka.get_cur_ee_pos()
    ee_pose_L = np.array([*pos_L, *quat_L])
    ee_pose_R = np.array([*pos_R, *quat_R])
    env.saving_data_replay["start_joint_pos_L"] = joint_pos_L
    env.saving_data_replay["start_joint_pos_R"] = joint_pos_R
    env.saving_data_replay["start_ee_pose_L"] = ee_pose_L
    env.saving_data_replay["start_ee_pose_R"] = ee_pose_R
    start_payload__r = {
        "joint_pos_L": _to_serializable(joint_pos_L),
        "joint_pos_R": _to_serializable(joint_pos_R),
        "ee_pose_L": _to_serializable(ee_pose_L),
        "ee_pose_R": _to_serializable(ee_pose_R),
    }

    c_target_position_left = c_target_position_left + np.array([e_mx * 0.3, 0.0, 0.0])
    env.bimanual_franka.Rmpflow_Left_Move(target_position=c_target_position_left, 
    target_orientation=c_target_orientation_left)

    cprint(f"[Info]: Collect left arm start position info separately","white")
    joint_pos_L = env.bimanual_franka.left_franka.get_joint_positions()
    joint_pos_R = env.bimanual_franka.right_franka.get_joint_positions()
    pos_L, quat_L = env.bimanual_franka.left_franka.get_cur_ee_pos()
    pos_R, quat_R = env.bimanual_franka.right_franka.get_cur_ee_pos()
    ee_pose_L = np.array([*pos_L, *quat_L])
    ee_pose_R = np.array([*pos_R, *quat_R])
    env.saving_data_replay["start_joint_pos_L"] = joint_pos_L
    env.saving_data_replay["start_joint_pos_R"] = joint_pos_R
    env.saving_data_replay["start_ee_pose_L"] = ee_pose_L
    env.saving_data_replay["start_ee_pose_R"] = ee_pose_R
    start_payload = {
        "joint_pos_L": _to_serializable(joint_pos_L),
        "joint_pos_R": _to_serializable(joint_pos_R),
        "ee_pose_L": _to_serializable(ee_pose_L),
        "ee_pose_R": _to_serializable(ee_pose_R),
    }
    env.start_pose_payload = start_payload

    if data_collection_flag:
        for i in range(2):
            env.step()
        env.record(task_name=f"{task_name}", stage_index=1)

    final_left_position = c_target_position_left
    i = 1
    max_m_c = contact_cfg.get("max_m_c", 5)
    while(i <  max_m_c):
        new_left_position = c_target_position_left - np.array([0.03 * i, 0.0, 0.0])
        env.bimanual_franka.Rmpflow_Left_Move(target_position=new_left_position, target_orientation=c_target_orientation_left)
        final_left_position = new_left_position
        i+=1

    joint_pos_L = env.bimanual_franka.left_franka.get_joint_positions()
    joint_pos_R = env.bimanual_franka.right_franka.get_joint_positions()
    pos_L, quat_L = env.bimanual_franka.left_franka.get_cur_ee_pos()
    pos_R, quat_R = env.bimanual_franka.right_franka.get_cur_ee_pos()
    ee_pose_L = np.array([*pos_L, *quat_L])
    ee_pose_R = np.array([*pos_R, *quat_R])
    env.saving_data_replay["end_joint_pos_L"] = joint_pos_L
    env.saving_data_replay["end_joint_pos_R"] = joint_pos_R
    env.saving_data_replay["end_ee_pose_L"] = ee_pose_L
    env.saving_data_replay["end_ee_pose_R"] = ee_pose_R
    end_payload = {
        "joint_pos_L": _to_serializable(joint_pos_L),
        "joint_pos_R": _to_serializable(joint_pos_R),
        "ee_pose_L": _to_serializable(ee_pose_L),
        "ee_pose_R": _to_serializable(ee_pose_R),
    }
    env.end_pose_payload = end_payload
    for i in range(2):
        env.step()
    if data_collection_flag:
        env.stop_record()

    cprint(f"[Info]: Right arm lifting up", "white", attrs=["bold"])
    env.bimanual_franka.Rmpflow_Right_Move(target_position=np.array([-0.1, 0.0, 0.9]), target_orientation=np.array([-179.653, 5.751, -179.268]))
    env.bimanual_franka.Rmpflow_Right_Move(target_position=np.array([-0.3, 0.0, 0.99]), target_orientation=np.array([-179.653, 5.751, -179.268]))

    env.bimanual_franka.Gripper_Left_Close()
    cprint(f"[Info]: Left arm lifting up", "white", attrs=["bold"])
    env.object.particle_material.set_gravity_scale(0.85) 

    for i in range(1,5):
        env.bimanual_franka.Rmpflow_Left_Move(target_position=final_left_position + np.array([0.0, 0.0, 0.05 * i]), 
                                              target_orientation=c_target_orientation_left)

    judge_flat_object_pcd, judge_flat_franka_pcd = get_point_cloud(env)
    success_flag = False
    success_flag = judge_pcd(judge_flat_object_pcd,judge_flat_franka_pcd)
    cprint(f"[Info]: success_flag: {success_flag}", "green", attrs=["bold"])

    if data_collection_flag:
        pos_idx_final = pos_index if pos_index is not None else (getattr(env, "pos_index", None) or 1)
        outcome_flag = "T" if success_flag else "F"
        base_dir = os.path.join("Data", task_name, "train_data", object_name)
        record_dir = os.path.join(base_dir, f"record_{pos_idx_final}_{outcome_flag}")
        samples = getattr(env, "last_recorded_samples", [])
        save_mid_samples(samples, record_dir)
        os.makedirs(record_dir, exist_ok=True)

        _write_point_cloud_ply3(
            record_dir + f"/initial_env.ply",
            whole_point_cloud_pcd,
            whole_point_cloud_pcd_color,
        )   

        if start_payload__r:
            _write_pose_txt(os.path.join(record_dir, "start_pos_r.txt"), start_payload)
            _write_pose_txt(os.path.join(record_dir, "end_pos_r.txt"), start_payload)

        start_payload = getattr(env, "start_pose_payload", None)
        if start_payload:
            _write_pose_txt(os.path.join(record_dir, "start_pos.txt"), start_payload)
        end_payload = getattr(env, "end_pose_payload", None)
        if end_payload:
            _write_pose_txt(os.path.join(record_dir, "end_pos.txt"), end_payload)
        cprint(f"[Info] Mid samples saved to {record_dir}", "green")

    for i in range(20):
        env.step()
    simulation_app.close()
    cprint(f"[Info]: Collection Finished!", "green", attrs=["bold"])
    return success_flag

if __name__=="__main__":
    
    args = parse_args_record()
    
    # initial setting
    pos = np.array([0.0, 0.30, 0.65])
    ori = np.array([90.0, 0.0, 0.0])
    usd_path = None
    env_dx = 0.0
    env_dy = 0.0

    if args.env_random_flag or args.garment_random_flag:
        np.random.seed(int(time.time()))
        if args.env_random_flag:
            env_dx = np.random.uniform(-0.05, 0.1)
            env_dy = np.random.uniform(-0.05, 0.05)
        if args.garment_random_flag:
            x = np.random.uniform(-0.05, 0.05)
            y = np.random.uniform(0.30, 0.40)
            pos = np.array([x,y,0.0])
            ori = np.array([90.0, 0.0, 0.0])
            Base_dir = os.path.dirname(os.path.dirname(os.path.abspath(__file__)))
            assets_lists = os.path.join(Base_dir,"Model_HALO/GAM/checkpoints/Scarf/assets_training_list.txt")
            assets_list = []
            with open(assets_lists,"r",encoding='utf-8') as f:
                for line in f:
                    clean_line = line.rstrip('\n')
                    assets_list.append(clean_line)
            usd_path=os.getcwd() + "/" + np.random.choice(assets_list)

    pos_index = None
    _pos_env = os.environ.get("POS_INDEX")
    if _pos_env and _pos_env.isdigit():
        pos_index = int(_pos_env)

    try:
        cprint(
            f"[Args] object_name={args.object_name} data_collection_flag={args.data_collection_flag} record_video_flag={args.record_video_flag} POS_INDEX={pos_index}",
            "yellow",
            attrs=["bold"],
        )
    except Exception:
        pass

    ok = WearScarf(
        pos, ori,
        usd_path,
        env_dx, env_dy,
        args.object_name,
        args.ground_material_usd,
        args.data_collection_flag,
        args.record_video_flag,
        args.object_random_init,
        pos_index,
    )

    if args.data_collection_flag:
        simulation_app.close()
        sys.exit(0 if ok else 1)
    else:
        while simulation_app.is_running():
            simulation_app.update()

simulation_app.close()