from openravepy import *

import numpy as np
import sys

from manipulation.constants import PARALLEL_LEFT_ARM, REST_LEFT_ARM, HOLDING_LEFT_ARM, FOLDED_LEFT_ARM, \
    FAR_HOLDING_LEFT_ARM, LOWER_TOP_HOLDING_LEFT_ARM, REGION_Z_OFFSET


from mover_library.utils import *
from manipulation.bodies.bodies import place_body, place_body_on_floor
from utils import randomly_place_region
from manipulation.primitives.transforms import set_point
from manipulation.regions import create_region, AARegion

from motion_planner import collision_fn, base_extend_fn, base_sample_fn, base_distance_fn, smooth_path, rrt_connect

import os
import pickle
import random

# obj definitions
min_height = 0.4
max_height = 1

min_width = 0.2
max_width = 0.6

min_length = 0.2
max_length = 0.6

OBST_COLOR = (1, 0, 0)
OBST_TRANSPARENCY = .25


def randomly_place_region(env, body, region):
    if env.GetKinBody(get_name(body)) is None: env.Add(body)
    while True:
        set_quat(body, quat_from_z_rot(uniform(0, 2 * PI)))
        aabb = aabb_from_body(body)
        cspace = region.cspace(aabb)
        if cspace is None: continue
        set_point(body, np.array([uniform(*range) for range in cspace] + [
            region.z + aabb.extents()[2] + BODY_PLACEMENT_Z_OFFSET]) - aabb.pos() + get_point(body))
        if not body_collision(env, body): return


def generate_rand(min, max):
    return np.random.rand() * (max - min) + min


def create_objects(env, obj_region, n_objects):
    OBJECTS = []
    obj_shapes = {}
    obj_poses = {}
    for i in range(n_objects):
        width = np.random.rand(1) * (max_width - min_width) + min_width
        length = np.random.rand(1) * (max_width - min_length) + min_length
        height = np.random.rand(1) * (max_height - min_height) + min_height
        new_body = box_body(env, width, length, height,
                            name='obj%s' % i,
                            color=(0, (i + .5) / n_objects, 0))
        trans = np.eye(4)
        trans[2, -1] = 0.075
        env.Add(new_body)
        new_body.SetTransform(trans)
        obj_pose = randomly_place_region(env, new_body, obj_region)  # TODO fix this to return xytheta
        OBJECTS.append(new_body)

        obj_shapes['obj%s' % i] = [width[0], length[0], height[0]]
        obj_poses['obj%s' % i] = obj_pose
    return OBJECTS, obj_poses, obj_shapes


def load_objects(env, obj_shapes, obj_poses, color):
    # sets up the object at their locations in the original env
    OBJECTS = []
    i = 0
    nobj = len(list(obj_shapes.keys()))
    for obj_name in list(obj_shapes.keys()):
        obj_xyz = obj_poses[obj_name]['obj_xyz']
        obj_rot = obj_poses[obj_name]['obj_rot']
        width, length, height = obj_shapes[obj_name]

        new_body = box_body(env, width, length, height,
                            name=obj_name,
                            color=np.array(color) / float(nobj - i))
        i += 1
        env.Add(new_body)
        set_point(new_body, obj_xyz)
        set_quat(new_body, obj_rot)
        OBJECTS.append(new_body)
    return OBJECTS


def create_bottom_walls(x_lim, y_lim, env):
    bottom_wall = box_body(env, x_lim * 2, y_lim * 2, 0.135, name='bottom_wall', color=(.82, .70, .55))
    bottom_wall_x = x_lim / 2.0
    set_xy(bottom_wall, bottom_wall_x, 0)
    env.Add(bottom_wall)

    side_wall = box_body(env, y_lim * 2, 0.01 * 2, 0.2 * 2,
                         name='side_wall_1',
                         color=(.82, .70, .55))
    place_body(env, side_wall, (-x_lim + bottom_wall_x, 0, np.pi / 2), base_name='bottom_wall')

    side_wall = box_body(env, y_lim * 2, 0.01 * 2, 0.2 * 2,
                         name='side_wall_2',
                         color=(.82, .70, .55))
    place_body(env, side_wall, (x_lim + bottom_wall_x, 0, np.pi / 2), base_name='bottom_wall')

    side_wall = box_body(env, x_lim * 2, 0.01 * 2, 0.2 * 2,
                         name='side_wall_3',
                         color=(.82, .70, .55))
    place_body(env, side_wall, (bottom_wall_x, y_lim, 0), base_name='bottom_wall')

    entrance_width = 1.2
    left_door_length = (x_lim - entrance_width / 2.0) - 1
    right_door_length = (x_lim - entrance_width / 2.0) + 1
    entrance_left = box_body(env, left_door_length, 0.01 * 2, 0.2 * 2,
                             name='entrance_left',
                             color=(.82, .70, .55))
    entrance_right = box_body(env, right_door_length, 0.01 * 2, 0.2 * 2,
                              name='entrance_right',
                              color=(.82, .70, .55))

    place_body(env, entrance_left, (bottom_wall_x + (-x_lim - (entrance_width / 2.0 + 1)) / 2.0, -y_lim, 0),
               base_name='bottom_wall')
    place_body(env, entrance_right, (bottom_wall_x + (x_lim + (entrance_width / 2.0 - 1)) / 2.0, -y_lim, 0),
               base_name='bottom_wall')

    # place_body(env, env.GetKinBody('Cabinet'), (x_lim + x_lim / 3.0 - 0.5, y_lim - 3, np.pi), base_name='bottom_wall')


def create_doors(x_lim, y_lim, door_x, door_y, door_width, th, env):
    if th == 0:
        middle_wall_size = (door_y - door_width / 2.0 - (-y_lim)) / 4.0
        left_wall_size = (y_lim - door_width / 2.0 - door_y) / 2.0
        middle_door_y = door_y - middle_wall_size - (door_width / 2.)
        right_wall_size =(y_lim/2.0 - middle_door_y ) / 2.0

    elif th == np.pi / 2.:
        right_wall_size = (door_x - door_width / 2.0 - (-x_lim)) / 2.0
        left_wall_size = (x_lim - door_width / 2.0 - door_x) / 2.0

    left_wall = box_body(env, 0.04 * 2, left_wall_size * 2, 1 * 2,
                         name='left_wall', color=(0, 0, 0))
    right_wall = box_body(env,
                         0.04 * 2, right_wall_size * 2, 1 * 2,
                          name='right_wall',
                          color=(0, 0, 0))

    middle_wall = box_body(env,
                          0.04 * 2, middle_wall_size * 2, 1 * 2,
                          name='middle_wall',
                          color=(0, 0, 0))
    place_body(env, left_wall, (door_x, door_y + left_wall_size + (door_width / 2.), th),
               base_name='bottom_wall')
    place_body(env, middle_wall, (door_x, door_y - middle_wall_size - (door_width / 2.), th),
               base_name='bottom_wall')
    place_body(env, right_wall, (door_x, middle_door_y-door_width-middle_wall_size-door_width, th), base_name='bottom_wall')

    wall_in_room_l = box_body(env, 0.04 * 2, 1, 2, name='wall_in_room_l', color=(0, 0, 0))
    wall_in_room_r = box_body(env, 0.04 * 2, 1, 2, name='wall_in_room_r', color=(0, 0, 0))
    place_body(env, wall_in_room_l, (3.3,0, np.pi/2), base_name='bottom_wall')
    place_body(env, wall_in_room_r, (5.5,0, np.pi/2), base_name='bottom_wall')



def create_box_bodies(body_shape, color, name, n_objs, env):
    if color == 'green':
        box_bodies = [box_body(env, body_shape[i][0], body_shape[i][1], body_shape[i][2], name=name + '%s' % i,
                               color=(1, (i + .5) / 5, 0)) for i in range(n_objs)]
    elif color == 'red':
        box_bodies = [box_body(env, body_shape[i][0], body_shape[i][1], body_shape[i][2], name=name + '%s' % i,
                               color=((i + .5) / 5, 0, 1)) for i in range(n_objs)]
    elif color == 'blue':
        box_bodies = [box_body(env, body_shape[i][0], body_shape[i][1], body_shape[i][2], name=name + '%s' % i,
                               color=(0, 1, (i + .5) / 5)) for i in range(n_objs)]

    return box_bodies


def generate_shelf_shapes():
    max_shelf_width = 0.7
    min_shelf_width = 0.4

    center_shelf_width = generate_rand(min_shelf_width, max_shelf_width)  # np.random.rand(0.5,0.8)
    right_shelf_width = generate_rand(min_shelf_width, max_shelf_width)
    left_shelf_width = generate_rand(min_shelf_width, max_shelf_width)

    center_shelf_height = generate_rand(0.3, 0.5)
    center_shelf_top_height = generate_rand(0.3, 0.5)
    left_shelf_height = generate_rand(0.3, 0.5)
    left_shelf_top_height = generate_rand(0.3, 0.5)
    right_shelf_height = generate_rand(0.3, 0.5)
    right_shelf_top_height = generate_rand(0.3, 0.5)

    shelf_shapes = {'center_shelf_top_height': center_shelf_top_height,
                    'center_shelf_height': center_shelf_height,
                    'left_shelf_top_height': left_shelf_top_height,
                    'left_shelf_height': left_shelf_height,
                    'right_shelf_top_height': right_shelf_top_height,
                    'right_shelf_height': right_shelf_height,
                    'center_shelf_width': center_shelf_width,
                    'left_shelf_width': left_shelf_width,
                    'right_shelf_width': right_shelf_width}

    left_x = center_shelf_width / 2.0 + left_shelf_width / 2.0
    right_x = -center_shelf_width / 2.0 - right_shelf_width / 2.0  # center of left shelf

    shelf_xs = {
        'left_x': left_x,
        'right_x': right_x
    }
    return shelf_shapes, shelf_xs


def create_shelf(env, obst_x, obst_width, obst_height, name_idx, stacked_obj_name, table_name):
    width = 0.4
    length = 0.01
    height = obst_height
    top_wall_width = 0.001
    bottom_wall_width = 0.0001

    table_pos = aabb_from_body(env.GetKinBody(table_name)).pos()
    table_x = table_pos[0]
    table_y = table_pos[1]
    place_body(env,
               box_body(env,
                        width, length, height,
                        name='right_wall_' + str(name_idx),
                        color=OBST_COLOR,
                        transparency=OBST_TRANSPARENCY),
               (table_x + .0, table_y + obst_x - (obst_width - .05) / 2, 0),
               stacked_obj_name)
    place_body(env,
               box_body(env,
                        width, length, height,
                        name='left_wall_' + str(name_idx),
                        color=OBST_COLOR,
                        transparency=OBST_TRANSPARENCY),
               (table_x + .0, table_y + obst_x + (obst_width - .05) / 2, 0),
               stacked_obj_name)
    place_body(env,
               box_body(env,
                        length, obst_width - 0.05, height,
                        name='back_wall_' + str(name_idx),
                        color=OBST_COLOR,
                        transparency=OBST_TRANSPARENCY),
               (table_x + .225, table_y + obst_x, 0),
               stacked_obj_name)
    place_body(env,
               box_body(env,
                        width, obst_width - 0.05, top_wall_width,
                        name='top_wall_' + str(name_idx),
                        color=OBST_COLOR,
                        transparency=OBST_TRANSPARENCY),
               (table_x + 0, table_y + obst_x, 0),
               'back_wall_' + str(name_idx))
    place_body(env,
               box_body(env,
                        width, obst_width - 0.05, bottom_wall_width,
                        name='bottom_wall_' + str(name_idx),
                        color=OBST_COLOR,
                        transparency=0.5),
               (table_x + 0, table_y + obst_x, 0),
               stacked_obj_name)

    region = create_region(env, 'place_region_' + str(name_idx),
                           ((-1.0, 1.0), (-0.85, 0.85)),
                           'bottom_wall_' + str(name_idx), color=np.array((0, 0, 0, .5)))
    region.draw(env)
    return region


def set_fixed_object_poses(env, x_lim, y_lim):
    objects = [env.GetKinBody('shelf1'), env.GetKinBody('shelf2'), env.GetKinBody('computer_table'),
               env.GetKinBody('table2')]
    place_body(env, env.GetKinBody('shelf1'), (x_lim + x_lim / 2.0 - 0.5, y_lim - 0.2, np.pi * 3 / 2),
               base_name='bottom_wall')
    place_body(env, env.GetKinBody('shelf2'), (x_lim + x_lim / 2.0 - 1.5, y_lim - 0.2, np.pi * 3 / 2),
               base_name='bottom_wall')
    place_body(env, env.GetKinBody('table2'), (x_lim + x_lim / 2.0 - 0.5, y_lim - 2, np.pi * 3 / 2),
               base_name='bottom_wall')
    #place_body(env, env.GetKinBody('computer_chair'), (4.2, -1.5, 0), base_name='bottom_wall')
    obj_poses = {obj.GetName(): get_pose(obj) for obj in objects}
    return obj_poses


def create_shelves(env, shelf_shapes, shelf_xs, table_name):
    center_shelf_width = shelf_shapes['center_shelf_width']
    center_shelf_height = shelf_shapes['center_shelf_height']
    left_shelf_width = shelf_shapes['left_shelf_width']
    left_shelf_height = shelf_shapes['left_shelf_height']
    right_shelf_width = shelf_shapes['right_shelf_width']
    right_shelf_height = shelf_shapes['right_shelf_height']
    center_shelf_top_height = shelf_shapes['center_shelf_top_height']
    left_shelf_top_height = shelf_shapes['left_shelf_top_height']
    right_shelf_top_height = shelf_shapes['right_shelf_top_height']

    left_x = shelf_xs['left_x']
    right_x = shelf_xs['right_x']

    center_region = create_shelf(env, obst_x=0, obst_width=center_shelf_width,
                                 obst_height=center_shelf_height, name_idx=1, stacked_obj_name=table_name,
                                 table_name=table_name)
    center_top_region = create_shelf(env, obst_x=0, obst_width=center_shelf_width,
                                     obst_height=center_shelf_top_height, name_idx=2,
                                     stacked_obj_name='back_wall_1', table_name=table_name)

    left_region = create_shelf(env, obst_x=left_x, obst_width=left_shelf_width,
                               obst_height=left_shelf_height, name_idx=3, stacked_obj_name=table_name,
                               table_name=table_name)
    left_top_region = create_shelf(env, obst_x=left_x, obst_width=left_shelf_width,
                                   obst_height=left_shelf_top_height, name_idx=4,
                                   stacked_obj_name='back_wall_3', table_name=table_name)

    right_region = create_shelf(env, obst_x=right_x, obst_width=right_shelf_width,
                                obst_height=right_shelf_height, name_idx=5, stacked_obj_name=table_name,
                                table_name=table_name)
    right_top_region = create_shelf(env, obst_x=right_x, obst_width=right_shelf_width,
                                    obst_height=right_shelf_top_height, name_idx=6,
                                    stacked_obj_name='back_wall_5', table_name=table_name)
    regions = {'center': center_region, 'center_top': center_top_region,
               'left': left_region, 'left_top': left_top_region,
               'right': right_region, 'right_top': right_top_region}
    return regions


def generate_shelf_obj_shapes():
    max_obj_height = 0.28
    n_objs = 7
    l_obj_shapes = [[0.05, 0.05, generate_rand(0.05, max_obj_height)] for _ in range(n_objs)]
    ltop_obj_shapes = [[0.05, 0.05, generate_rand(0.05, max_obj_height)] for _ in range(n_objs)]
    c_obj_shapes = [[0.05, 0.05, generate_rand(0.05, max_obj_height)] for _ in range(n_objs)]
    ctop_obj_shapes = [[0.05, 0.05, generate_rand(0.05, max_obj_height)] for _ in range(n_objs)]
    r_obj_shapes = [[0.05, 0.05, generate_rand(0.05, max_obj_height)] for _ in range(n_objs)]
    rtop_obj_shapes = [[0.05, 0.05, generate_rand(0.05, max_obj_height)] for _ in range(n_objs)]

    obj_shapes = {'l_obj_shapes': l_obj_shapes, 'ltop_obj_shapes': ltop_obj_shapes,
                  'c_obj_shapes': c_obj_shapes, 'ctop_obj_shapes': ctop_obj_shapes,
                  'r_obj_shapes': r_obj_shapes, 'rtop_obj_shapes': rtop_obj_shapes}

    return obj_shapes


def create_shelf_objs(env, obj_shapes):
    n_objs = 7
    left_objs = create_box_bodies(obj_shapes['l_obj_shapes'], color='green', name='l_obst', n_objs=n_objs,
                                  env=env)
    left_top_objs = create_box_bodies(obj_shapes['ltop_obj_shapes'], color='green', name='ltop_obst',
                                      n_objs=n_objs, env=env)
    center_objs = create_box_bodies(obj_shapes['c_obj_shapes'], color='blue', name='c_obst', n_objs=n_objs,
                                    env=env)
    center_top_objs = create_box_bodies(obj_shapes['ctop_obj_shapes'], color='blue', name='ctop_obst',
                                        n_objs=n_objs, env=env)
    right_objs = create_box_bodies(obj_shapes['r_obj_shapes'], color='red', name='r_obst',
                                   n_objs=n_objs, env=env)
    right_top_objs = create_box_bodies(obj_shapes['rtop_obj_shapes'], color='red', name='rtop_obst',
                                       n_objs=n_objs, env=env)
    objects = {'left': left_objs, 'left_top': left_top_objs, 'center': center_objs,
               'center_top': center_top_objs, 'right': right_objs, 'right_top': right_top_objs}
    return objects


def place_objs_in_region(objs, region, env):
    for obj in objs:
        randomly_place_region(env, obj, region)


def generate_poses_and_place_shelf_objs(objects, regions, env):
    left_objs = objects['left']
    left_region = regions['left']
    left_top_objs = objects['left_top']
    left_top_region = regions['left_top']
    right_objs = objects['right']
    right_region = regions['right']
    right_top_objs = objects['right_top']
    right_top_region = regions['right_top']
    center_objs = objects['center']
    center_region = regions['center']
    center_top_objs = objects['center_top']
    center_top_region = regions['center_top']

    place_objs_in_region(left_objs, left_region, env)
    place_objs_in_region(left_top_objs, left_top_region, env)
    place_objs_in_region(center_objs, center_region, env)
    place_objs_in_region(center_top_objs, center_top_region, env)
    place_objs_in_region(right_objs, right_region, env)
    place_objs_in_region(right_top_objs, right_top_region, env)

    obj_poses = {obj.GetName(): get_pose(obj) for obj_list in list(objects.values()) for obj in obj_list}
    return obj_poses


def set_fixed_obj_poses(env):
    table1 = env.GetKinBody('table1')
    shelf = env.GetKinBody('shelf1')


def get_motion_plan(env, goal, x_extents, y_extents, x, y):
    robot = env.GetRobots()[0]
    d_fn = base_distance_fn(robot, x_extents, y_extents, x, y)
    s_fn = base_sample_fn(robot, x_extents, y_extents, x, y)
    e_fn = base_extend_fn(robot)
    c_fn = collision_fn(env, robot)
    q_init = robot.GetActiveDOFValues()

    n_iterations = [20, 50, 100, 500, 1000]
    for n_iter in n_iterations:
        path = rrt_connect(q_init, goal, d_fn, s_fn, e_fn, c_fn, iterations=n_iter)
        if path is not None:
            path = smooth_path(path, e_fn, c_fn)
            return path, "HasSolution"

    return None, 'NoPath'


def create_environment_region(name, xy, extents, z=None):
    if z is None:
        z = 0.138

    region = AARegion(name, ((xy[0] - extents[0],
                                 xy[0] + extents[0]),
                                (xy[1] - extents[1],
                                 xy[1] + extents[1])),
             z, color=np.array((1, 1, 0, 0.25)))
    return region

def place_object_with_gaussian_noise(obj, reference_xytheta, env, scale=0.3):
    obj_xytheta = reference_xytheta + np.random.normal(loc=0, scale=scale, size=(3,))
    set_quat(obj, quat_from_angle_vector(obj_xytheta[-1], np.array([0, 0, 1])))
    set_xy(obj, obj_xytheta[0], obj_xytheta[1])
    while env.CheckCollision(obj):
        obj_xytheta = obj_xytheta + np.random.normal(loc=0, scale=scale, size=(3,))
        set_quat(obj, quat_from_angle_vector(obj_xytheta[-1], np.array([0, 0, 1])))
        set_xy(obj, obj_xytheta[0], obj_xytheta[1])


class MinimumDisplacementRemovalProblem:
    def __init__(self, env, problem_idx, problem_config=None):
        self.env = env
        fdir = os.path.dirname(os.path.abspath(__file__))
        self.env.Load(fdir + '/resources/mover_env.xml')
        self.robot = self.env.GetRobots()[0]
        set_xy(self.env.GetKinBody('kitchen'), -0.3, 1.3)

        # bottom wall creation
        x_extents = 4
        y_extents = 4
        create_bottom_walls(x_extents, y_extents, self.env)

        # door creation
        # self.env.Remove(self.env.GetKinBody('left_wall')); self.env.Remove(self.env.GetKinBody('right_wall'))
        door_width = 1.  # generate_rand(1, 1.5)
        door_x = (-x_extents + 1.5 + x_extents - 1.5) / 2.0 - x_extents * 0.3 + 4
        door_y = (-y_extents + 1.5 + y_extents - 1.5) / 2.0 + 2
        door_th = 0
        create_doors(x_extents, y_extents, door_x, door_y, door_width, door_th, self.env)

        set_config(self.robot, FOLDED_LEFT_ARM, self.robot.GetManipulator('leftarm').GetArmIndices())
        set_config(self.robot, mirror_arm_config(FOLDED_LEFT_ARM),
                   self.robot.GetManipulator('rightarm').GetArmIndices())

        fixed_obj_poses = set_fixed_object_poses(self.env, x_extents, y_extents)

        shelf_table = env.GetKinBody('table2')
        set_obj_xytheta([-1.5, -3, np.pi/2], shelf_table)
        shelf_shapes, shelf_xs = generate_shelf_shapes()
        shelf_regions = create_shelves(self.env, shelf_shapes, shelf_xs, 'table2')
        obj_shapes = generate_shelf_obj_shapes()
        shelf_objects = create_shelf_objs(self.env, obj_shapes)
        shelf_obj_poses = generate_poses_and_place_shelf_objs(shelf_objects, shelf_regions, self.env)

        self.kitchen_region = AARegion('kitchen_region',
                                       ((-x_extents + 2, x_extents * 0.2 + 2), (1 - y_extents / 2, 1 + y_extents / 2)),
                                       z=0.135, color=np.array((1, 1, 0, 0.25)))
        set_xy(self.env.GetKinBody('computer_table'), 5.5, -2.5)


        x_size = x_extents
        y_size = y_extents
        self.entire_region_xy = [x_size / 2.0, 0]
        self.entire_region_xy_extents = [x_size, y_size]
        self.entire_region = AARegion('entire_region',
                                    ((-x_size + self.entire_region_xy[0], x_size + self.entire_region_xy[0]),
                                     (-y_size, y_size)), z=0.135, color=np.array((1, 1, 0, 0.25)))

        # allow it to be entire region
        # but if not in home_region or in forbidden region, reject in motion planning stage
        self.home_region_xy = [x_extents / 2.0, 0]
        self.home_region_xy_extents = [x_extents,y_extents]
        self.home_region = AARegion('entire_region', ((-x_extents + self.home_region_xy[0], x_extents + self.home_region_xy[0]), (-x_extents, y_extents)), z=0.135, color=np.array((1, 1, 0, 0.25)))

        #randomly_place_region(self.env, self.robot, self.home_region)


        # randomly placed objects
        kitchen_chairs = [body for body in self.env.GetBodies() if body.GetName().find('chair') != -1
                          and body.GetName().find('computer') == -1]
        packing_boxes = [b for b in self.env.GetBodies() if b.GetName().find('packing_box') != -1]
        computer_chair = self.env.GetKinBody('computer_chair')
        table = self.env.GetKinBody('table')
        # kitchen table location

        set_obj_xytheta([2, -3.2, -np.pi/2], env.GetKinBody('computer_table'))
        set_obj_xytheta([1, -3.2, 0], env.GetKinBody('computer_chair'))

        self.is_new_env = problem_config is None
        # computer chair location

        # place other objects

        place_objs_in_region(packing_boxes, self.kitchen_region, self.env)


        place_objs_in_region([self.robot], self.entire_region, self.env)
        place_objs_in_region(kitchen_chairs, self.kitchen_region, self.env)

        self.forbidden_region_xy = [x_extents+0.3, 0]
        self.forbidden_region_xy_extents = [x_extents/2.0-0.3, y_extents]
        self.forbidden_region = AARegion('forbidden_region',
                                     ((-self.forbidden_region_xy_extents[0] + self.forbidden_region_xy[0],
                                       self.forbidden_region_xy_extents[0] + self.forbidden_region_xy[0]),
                                     (-y_extents, y_extents)), z=0.135, color=np.array((1, 1, 0, 0.25)))

        robot=self.robot
        # left arm IK
        robot.SetActiveManipulator('leftarm')
        manip = robot.GetActiveManipulator()
        ee = manip.GetEndEffector()
        ikmodel1 = databases.inversekinematics.InverseKinematicsModel(robot=robot,
                                                                      iktype=IkParameterization.Type.Transform6D,
                                                                      forceikfast=True, freeindices=None,
                                                                      freejoints=None, manip=None)
        if not ikmodel1.load():
            ikmodel1.autogenerate()

        # right arm torso IK
        robot.SetActiveManipulator('rightarm_torso')
        manip = robot.GetActiveManipulator()
        ee = manip.GetEndEffector()
        ikmodel2 = databases.inversekinematics.InverseKinematicsModel(robot=robot, \
                                                                      iktype=IkParameterization.Type.Transform6D, \
                                                                      forceikfast=True, freeindices=None, \
                                                                      freejoints=None, manip=None)
        if not ikmodel2.load():
            ikmodel2.autogenerate()



        self.robot.SetActiveDOFs([], DOFAffine.X | DOFAffine.Y | DOFAffine.RotationAxis, [0, 0, 1])
        self.movable_objects = [computer_chair] + packing_boxes + kitchen_chairs

        self.problem_idx = problem_idx

        self.goal_base_config = np.array([-0,2.5,np.pi/2.])
        self.init_base_config = np.array([-1.5, -1.5, np.pi/2.])
        set_robot_config(self.init_base_config, self.robot)
        set_obj_xytheta([3, 2, 43 * np.pi / 180.], env.GetKinBody('shelf1'))
        set_obj_xytheta([3, -1.8, 43 * np.pi / 180.], env.GetKinBody('shelf2'))
        set_obj_xytheta([-0.13, 0.45, 3], packing_boxes[2])
        self.set_obj_poses(problem_idx)

        if self.problem_idx == 1:
            env.Remove(env.GetKinBody('shelf1'))
            env.Remove(env.GetKinBody('shelf2'))

        else:
            wall_in_kitchen = box_body(env, 0.04 * 2, 1.3, 2, name='wall_in_kitchen', color=(0, 0, 0))
            place_body(env, wall_in_kitchen, (2.1, 0, np.pi / 2), base_name='bottom_wall')


    def save_obj_poses(self, problem_idx):
        obj_poses = {obj.GetName(): get_body_xytheta(obj) for obj in self.movable_objects}
        pickle.dump(obj_poses, open('./problem_environments/mover_domain_problems/' + str(problem_idx) +'.pkl','wb'))

    def set_obj_poses(self, problem_idx):
        obj_poses = pickle.load(open('./problem_environments/mover_domain_problems/' + str(1) + '.pkl', 'r'))
        for obj_name, obj_pose in zip(list(obj_poses.keys()), list(obj_poses.values())):
            set_obj_xytheta(obj_pose, self.env.GetKinBody(obj_name))

    def get_problem_config(self):
        problem_config = {'objects': self.movable_objects,
                          'entire_region': self.entire_region,
                          'forbidden_region': self.forbidden_region,
                          'home_region': self.home_region,
                          'init_base_config': self.init_base_config,
                          'goal_base_config': self.goal_base_config,
                          'home_region_xy': self.home_region_xy,
                          'home_region_extents': self.home_region_xy_extents,
                          'entire_region_xy': self.entire_region_xy,
                          'entire_region_extents': self.entire_region_xy_extents,
                          'problem_idx': self.problem_idx}
        return problem_config

    def disable_objects_in_region(self, region):
        # todo do it for the shelf objects too
        for obj in self.movable_objects:
            if region.contains(obj.ComputeAABB()):
                obj.Enable(False)

    def enable_objects_in_region(self, region):
        for obj in self.movable_objects:
            if region.contains(obj.ComputeAABB()):
                obj.Enable(True)





