package scienceworld.tasks.specifictasks

import scienceworld.objects.agent.Agent
import scienceworld.objects.livingthing.LivingThing
import scienceworld.objects.livingthing.animals.{Beaver, BlueJay, BrownBear, Butterfly, Crocodile, Dove, Elephant, Frog, GiantTortoise, Moth, Parrot, Toad, Turtle, Wolf}
import scienceworld.objects.misc.InclinedPlane
import scienceworld.properties.{AluminumProp, BrassProp, BronzeProp, CaesiumProp, CeramicProp, ChocolateProp, CopperProp, CottonClothProp, GlassProp, GreenPaintProp, IronProp, LeadProp, PaperProp, PlasticProp, PlatinumProp, RubberProp, SandpaperProp, SoapyWaterProp, SteelProp, TinProp, TitaniumProp, WoodProp, ZincProp}
import scienceworld.struct.EnvObject
import scienceworld.tasks.{Task, TaskMaker1, TaskModifier, TaskObject, TaskValueStr}
import scienceworld.tasks.goals.{Goal, GoalSequence}
import scienceworld.tasks.goals.specificgoals.{GoalActivateDeviceWithName, GoalDeactivateDeviceWithName, GoalFind, GoalFindInclinedPlane, GoalFindLivingThingStage, GoalMoveToLocation, GoalMoveToNewLocation, GoalPastActionExamineObject, GoalSpecificObjectInDirectContainer}

import scala.util.control.Breaks._
import scala.collection.mutable.ArrayBuffer
import TaskInclinedPlane1._
import scienceworld.actions.Action
import scienceworld.goldagent.PathFinder
import scienceworld.objects.devices.StopWatch
import scienceworld.objects.substance.{Brick, SteelBlock, WoodBlock}
import scienceworld.runtime.pythonapi.PythonInterface
import scienceworld.tasks.specifictasks.TaskIdentifyLifeStages1.mkActionSequenceWaitForLivingThingInStage

import scala.util.Random


class TaskInclinedPlane1(val mode:String = MODE_FRICTION_NAMED) extends TaskParametric {
  val taskName = mode.replaceAll(" ", "-").replaceAll("[()]", "")

  val locations = Array("workshop")

  // Variation 1 + 2: Inclined planes of various material surfaces (and frictions)
  val inclinedPlanes1 = new ArrayBuffer[ Array[TaskModifier] ]()
  val inclinedPlanes2 = new ArrayBuffer[ Array[TaskModifier] ]()
  for (location <- locations) {

    // Make a set of inclined plane objects of various surfaces
    val inclinedPlaneObjects1 = TaskInclinedPlane1.mkInclinedPlaneSet(angleDeg = 45.0)
    for (inclinedPlane <- inclinedPlaneObjects1) {
      inclinedPlanes1.append( Array(new TaskObject(inclinedPlane.name, Some(inclinedPlane), roomToGenerateIn = location, Array.empty[String], generateNear = 0, forceAdd = true),
                                    new TaskValueStr(key = "planeLocation", value = location) ) )
    }

    val inclinedPlaneObjects2 = TaskInclinedPlane1.mkInclinedPlaneSet(angleDeg = 45.0)
    for (inclinedPlane <- inclinedPlaneObjects2) {
      inclinedPlanes2.append( Array(new TaskObject(inclinedPlane.name, Some(inclinedPlane), roomToGenerateIn = location, Array.empty[String], generateNear = 0, forceAdd = true) ) )
    }

  }

  // Variation 3: Masses to roll down ramp
  val masses = new ArrayBuffer[ Array[TaskModifier] ]()
  for (location <- locations) {
    val block1 = new WoodBlock()
    masses.append( Array(new TaskObject(block1.name, Some(block1), roomToGenerateIn = location, Array.empty[String], generateNear = 0, forceAdd = true),
                         new TaskValueStr(key = "blockName", value = block1.name) ))

    val block2 = new SteelBlock()
    masses.append( Array(new TaskObject(block2.name, Some(block2), roomToGenerateIn = location, Array.empty[String], generateNear = 0, forceAdd = true),
                         new TaskValueStr(key = "blockName", value = block2.name) ))

    val brick = new Brick()
    masses.append( Array(new TaskObject(brick.name, Some(brick), roomToGenerateIn = location, Array.empty[String], generateNear = 0, forceAdd = true),
                         new TaskValueStr(key = "blockName", value = brick.name)))


  }

  // Variation 4: Time measurement device
  val timeMeasurementDevice = new ArrayBuffer[ Array[TaskModifier] ]()
  for (location <- locations) {
    val stopwatch = new StopWatch()
    timeMeasurementDevice.append( Array(new TaskObject(stopwatch.name, Some(stopwatch), roomToGenerateIn = location, Array.empty[String], generateNear = 0),
                                        new TaskValueStr(key = "timeDeviceName", value = stopwatch.name),
                                        new TaskValueStr(key = "timeLocation", value = location) ))
  }


  // Combinations
  var combinations1 = for {
    i <- inclinedPlanes1
    j <- inclinedPlanes2
    k <- masses
    m <- timeMeasurementDevice
  } yield List(i, j, k, m)

  // Remove any combinations that have the same coefficient of friction
  // Also add extra information as keys (e.g. most friction/least friction)
  val combinations = TaskInclinedPlane1.filterDuplicatesAndAddMostLeast(combinations1.toArray)


  println("Number of combinations: " + combinations.length)

  def numCombinations():Int = this.combinations.size

  def getCombination(idx:Int):Array[TaskModifier] = {
    val out = new ArrayBuffer[TaskModifier]
    for (elem <- combinations(idx)) {
      out.insertAll(out.length, elem)
    }
    // Return
    out.toArray
  }

  // Setup a particular modifier combination on the universe
  private def setupCombination(modifiers:Array[TaskModifier], universe:EnvObject, agent:Agent):(Boolean, String) = {
    // Run each modifier's change on the universe
    for (mod <- modifiers) {
      println("Running modifier: " + mod.toString)
      val success = mod.runModifier(universe, agent)
      if (!success) {
        return (false, "ERROR: Error running one or more modifiers while setting up task environment.")
      }
    }
    // If we reach here, success
    return (true, "")
  }

  def setupCombination(combinationNum:Int, universe:EnvObject, agent:Agent): (Boolean, String) = {
    if (combinationNum >= this.numCombinations()) {
      return (false, "ERROR: The requested variation (" + combinationNum + ") exceeds the total number of variations (" + this.numCombinations() + ").")
    }
    return this.setupCombination( this.getCombination(combinationNum), universe, agent )
  }


  // Setup a set of subgoals for this task modifier combination.
  private def setupGoals(modifiers:Array[TaskModifier], combinationNum:Int): Task = {
    // Step 1: Find seed type
    val leastFriction = this.getTaskValueStr(modifiers, "leastFriction")
    if (leastFriction.isEmpty) throw new RuntimeException("ERROR: Failed to find name of inclined plane with least friction.")
    val mostFriction = this.getTaskValueStr(modifiers, "mostFriction")
    if (mostFriction.isEmpty) throw new RuntimeException("ERROR: Failed to find name of inclined plane with most friction.")
    val modeMostLeastFriction = this.getTaskValueStr(modifiers, "mode")
    if (modeMostLeastFriction.isEmpty) throw new RuntimeException("ERROR: Failed to find inclined plane friction task mode. ")

    val planeName1 = this.getTaskValueStr(modifiers, "planeName1")
    val planeName2 = this.getTaskValueStr(modifiers, "planeName2")
    val planeLocation = this.getTaskValueStr(modifiers, "planeLocation")
    val blockName = this.getTaskValueStr(modifiers, "blockName")
    val timeDeviceName = this.getTaskValueStr(modifiers, key = "timeDeviceName")


    val gSequence = new ArrayBuffer[Goal]
    val gSequenceUnordered = new ArrayBuffer[Goal]
    var description:String = "<empty>"
    if (mode == MODE_FRICTION_NAMED) {

      if (modeMostLeastFriction.get == "most") {
        // Most friction
        gSequence.append(new GoalFindInclinedPlane(surfaceName = mostFriction.get, failIfWrong = true, _defocusOnSuccess = true, description = "focus on correct inclined plane"))
      } else {
        // Least friction
        gSequence.append(new GoalFindInclinedPlane(surfaceName = leastFriction.get, failIfWrong = true, _defocusOnSuccess = true, description = "focus on correct inclined plane"))
      }

      gSequenceUnordered.append( new GoalMoveToNewLocation(_isOptional = true, unlessInLocation = planeLocation.get, description = "move to a new location (unless starting in task location)") )            // Move to any new location
      gSequenceUnordered.append( new GoalMoveToLocation(planeLocation.get, _isOptional = true, description = "move to the location asked by the task") )

      gSequenceUnordered.append( new GoalSpecificObjectInDirectContainer(containerName = planeName1.get, validObjectNames = Array(blockName.get), description = "move block to plane 1", key = "b1") )
      gSequenceUnordered.append( new GoalActivateDeviceWithName(deviceName = timeDeviceName.get, description = "activate time keeping device (plane 1)", key = "aTime1", keysMustBeCompletedBefore = Array("b1")) )
      gSequenceUnordered.append( new GoalDeactivateDeviceWithName(deviceName = timeDeviceName.get, description = "deactivate time keeping device (plane 1)", key = "dTime1", keysMustBeCompletedBefore = Array("aTime1")) )
      gSequenceUnordered.append( new GoalPastActionExamineObject(objectName = timeDeviceName.get, description = "read time keeping device (plane 1)", key = "readTime1", keysMustBeCompletedBefore = Array("dTime1") ))

      gSequenceUnordered.append( new GoalSpecificObjectInDirectContainer(containerName = planeName2.get, validObjectNames = Array(blockName.get), description = "move block to plane 2", key = "b2") )
      gSequenceUnordered.append( new GoalActivateDeviceWithName(deviceName = timeDeviceName.get, description = "activate time keeping device (plane 2)", key = "aTime2", keysMustBeCompletedBefore = Array("b2")) )
      gSequenceUnordered.append( new GoalDeactivateDeviceWithName(deviceName = timeDeviceName.get, description = "deactivate time keeping device (plane 2)", key = "dTime2", keysMustBeCompletedBefore = Array("aTime2")) )
      gSequenceUnordered.append( new GoalPastActionExamineObject(objectName = timeDeviceName.get, description = "read time keeping device (plane 2)", key = "readTime2", keysMustBeCompletedBefore = Array("dTime2") ))


      val planeNames = Random.shuffle( List(leastFriction.get, mostFriction.get) )
      description = "Your task is to determine which of the two inclined planes (" + planeNames.mkString(", ") + ") has the " + modeMostLeastFriction.get + " friction. After completing your experiment, focus on the inclined plane with the " + modeMostLeastFriction.get + " friction."


    } else {
      throw new RuntimeException("ERROR: Unrecognized task mode: " + mode)
    }

    val taskLabel = taskName + "-variation" + combinationNum
    //val description = "Your task is to find a " + subTask + ". First, focus on the thing. Then, move it to the " + answerBoxName + " in the " + answerBoxLocation + "."
    val goalSequence = new GoalSequence(gSequence.toArray, gSequenceUnordered.toArray)

    val task = new Task(taskName, description, goalSequence, taskModifiers = modifiers)

    // Return
    return task
  }

  def setupGoals(combinationNum:Int): Task = {
    this.setupGoals( this.getCombination(combinationNum), combinationNum )
  }


  /*
   * Gold Action Sequences
   */
  def mkGoldActionSequence(modifiers:Array[TaskModifier], runner:PythonInterface): (Boolean, Array[String]) = {
    if (mode == MODE_FRICTION_NAMED) {
      return mkGoldActionSequenceLifeStages(modifiers, runner)
    } else {
      throw new RuntimeException("ERROR: Unrecognized task mode: " + mode)
    }

  }

  /*
   * Gold action sequences
   */
  def mkGoldActionSequenceLifeStages(modifiers:Array[TaskModifier], runner:PythonInterface): (Boolean, Array[String]) = {
    val universe = runner.agentInterface.get.universe
    val agent = runner.agentInterface.get.agent

    // Task variables
    // Step 1: Find seed type
    val leastFriction = this.getTaskValueStr(modifiers, "leastFriction").get
    val mostFriction = this.getTaskValueStr(modifiers, "mostFriction").get
    val modeMostLeastFriction = this.getTaskValueStr(modifiers, "mode").get

    val planeName1 = this.getTaskValueStr(modifiers, "planeName1").get
    val planeName2 = this.getTaskValueStr(modifiers, "planeName2").get
    val planeLocation = this.getTaskValueStr(modifiers, "planeLocation").get
    val blockName = this.getTaskValueStr(modifiers, "blockName").get
    val timeDeviceName = this.getTaskValueStr(modifiers, key = "timeDeviceName").get


    // Step 1: Move from starting location to task location
    val startLocation = agent.getContainer().get.name
    val (actions, actionStrs) = PathFinder.createActionSequence(universe, agent, startLocation, endLocation = planeLocation)
    runActionSequence(actionStrs, runner)

    // Look around
    runAction("look around", runner)

    // Take stop watch
    val timeTools = PathFinder.getAllAccessibleEnvObject(queryName = timeDeviceName, getCurrentAgentLocation(runner))
    if (timeTools.length == 0) return (false, getActionHistory(runner))
    val timeTool = timeTools(0)
    runAction("pick up " + PathFinder.getObjUniqueReferent(timeTool, getCurrentAgentLocation(runner)).get, runner)


    // Get reference to block
    val blocks = PathFinder.getAllAccessibleEnvObject(queryName = blockName, getCurrentAgentLocation(runner))
    if (blocks.length == 0) return (false, getActionHistory(runner))
    val block = blocks(0)

    // Get reference to inclined planes
    val inclinedPlanes = getCurrentAgentLocation(runner).getContainedAccessibleObjectsOfType[InclinedPlane]().toArray.sortBy(_.name)
    val inclinedPlane1 = inclinedPlanes(0)
    val inclinedPlane2 = inclinedPlanes(1)


    // Slide block down plane 1
    val (success1, time1) = actionSequenceMeasureBlockFallTime(block = block, timeTool = timeTool, inclinedPlane = inclinedPlane1, runner)
    if (!success1) return (false, getActionHistory(runner))

    // Slide block down plane 2
    val (success2, time2) = actionSequenceMeasureBlockFallTime(block = block, timeTool = timeTool, inclinedPlane = inclinedPlane2, runner)
    if (!success2) return (false, getActionHistory(runner))


    var planeToSelect:Option[EnvObject] = None
    if (time1 > time2) {
      // Plane 1 had more friction
      if (modeMostLeastFriction == "most") {
        planeToSelect = Some(inclinedPlane1)
      } else {
        planeToSelect = Some(inclinedPlane2)
      }
    } else {
      // Plane 2 had more friction
      if (modeMostLeastFriction == "most") {
        planeToSelect = Some(inclinedPlane2)
      } else {
        planeToSelect = Some(inclinedPlane1)
      }
    }

    // Step 2: Focus on substance
    runAction("focus on " + PathFinder.getObjUniqueReferent(planeToSelect.get, getCurrentAgentLocation(runner)).get, runner)

    // Wait one moment
    runAction("wait1", runner)

    // Return
    return (true, getActionHistory(runner))
  }


}


object TaskInclinedPlane1 {
  val MODE_FRICTION_NAMED       = "inclined plane friction (named surfaces)"

  def registerTasks(taskMaker:TaskMaker1): Unit = {
    taskMaker.addTask( new TaskInclinedPlane1(mode = MODE_FRICTION_NAMED) )
  }


  /*
   * Helper functions
   */

  // Make a set of inclined planes
  def mkInclinedPlaneSet(angleDeg:Double = 45.0):Array[EnvObject] = {
    val out = new ArrayBuffer[EnvObject]

    // Metal surfaces
    out.append( new InclinedPlane(angleDeg, surfaceMaterial = new AluminumProp) )
    out.append( new InclinedPlane(angleDeg, surfaceMaterial = new BrassProp) )
    out.append( new InclinedPlane(angleDeg, surfaceMaterial = new BronzeProp) )
    out.append( new InclinedPlane(angleDeg, surfaceMaterial = new CaesiumProp) )
    out.append( new InclinedPlane(angleDeg, surfaceMaterial = new CopperProp) )
    out.append( new InclinedPlane(angleDeg, surfaceMaterial = new IronProp) )
    out.append( new InclinedPlane(angleDeg, surfaceMaterial = new LeadProp) )
    out.append( new InclinedPlane(angleDeg, surfaceMaterial = new PlatinumProp) )
    out.append( new InclinedPlane(angleDeg, surfaceMaterial = new SteelProp) )
    out.append( new InclinedPlane(angleDeg, surfaceMaterial = new TinProp) )
    out.append( new InclinedPlane(angleDeg, surfaceMaterial = new TitaniumProp) )
    out.append( new InclinedPlane(angleDeg, surfaceMaterial = new ZincProp) )

    // Non-metal surfaces
    out.append( new InclinedPlane(angleDeg, surfaceMaterial = new GlassProp) )
    out.append( new InclinedPlane(angleDeg, surfaceMaterial = new PlasticProp) )
    out.append( new InclinedPlane(angleDeg, surfaceMaterial = new CeramicProp) )
    out.append( new InclinedPlane(angleDeg, surfaceMaterial = new WoodProp) )
    out.append( new InclinedPlane(angleDeg, surfaceMaterial = new PaperProp) )
    out.append( new InclinedPlane(angleDeg, surfaceMaterial = new SandpaperProp) )
    out.append( new InclinedPlane(angleDeg, surfaceMaterial = new CottonClothProp) )
    out.append( new InclinedPlane(angleDeg, surfaceMaterial = new RubberProp) )
    out.append( new InclinedPlane(angleDeg, surfaceMaterial = new ChocolateProp) )
    out.append( new InclinedPlane(angleDeg, surfaceMaterial = new SoapyWaterProp) )
    out.append( new InclinedPlane(angleDeg, surfaceMaterial = new GreenPaintProp) )

    // Shuffle order
    val r = new scala.util.Random(3)
    return r.shuffle(out).toArray
  }


  def filterDuplicatesAndAddMostLeast(in:Array[List[Array[TaskModifier]]], threshold:Double = 0.05): Array[List[List[TaskModifier]]] = {
    val out = new ArrayBuffer[ List[List[TaskModifier]] ]

    var count:Int = 0

    for (tmSet <- in) {
      val allModifiers = tmSet.flatten
      // Find all inclined planes
      val inclinedPlanes = new ArrayBuffer[InclinedPlane]
      for (modifier <- allModifiers) {
        modifier match {
          case m:TaskObject => {
            if (m.exampleInstance.isDefined) {
              m.exampleInstance.get match {
                case plane: InclinedPlane => {
                  inclinedPlanes.append(plane)
                }
                case _ => {
                  // Do nothing
                }
              }
            }
          }
          case _ => {
            // do nothing
          }
        }
      }

      breakable {
        var leastFriction:Double = 1.0
        var leastFrictionName:String = "DEFAULT"
        var mostFriction:Double = 0.0
        var mostFrictionName:String = "DEFAULT"

        for (i <- 0 until inclinedPlanes.length) {
          val plane1 = inclinedPlanes(i)

          for (j <- 0 until inclinedPlanes.length) {
            val plane2 = inclinedPlanes(j)

            if (i != j) {
              // Check to see if the friction coefficients of the two materials are too similar (i.e. within 'threshold').  If they are, don't keep them in the list.
              val delta = math.abs(plane1.surfaceMaterial.frictionCoefficient - plane2.surfaceMaterial.frictionCoefficient)
              if (delta <= threshold) break
            }
          }

          if (plane1.surfaceMaterial.frictionCoefficient < leastFriction) {
            leastFriction = plane1.surfaceMaterial.frictionCoefficient
            leastFrictionName = plane1.surfaceMaterial.substanceName
          }
          if (plane1.surfaceMaterial.frictionCoefficient > mostFriction) {
            mostFriction = plane1.surfaceMaterial.frictionCoefficient
            mostFrictionName = plane1.surfaceMaterial.substanceName
          }

        }

        // If we reach here, the list is okay (i.e. no duplicates/near duplicates).  Store it.

        // First, pick whether the agent should find the inclined plane with the most or least friction
        var modifierModeKey = new TaskValueStr(key = "mode", value = "least")
        if (count % 2 == 0) {
          modifierModeKey = new TaskValueStr(key = "mode", value = "most")
        }
        count += 1

        // Then, assemble the task modifiers
        val taskModifiers = allModifiers ++
          Array(new TaskValueStr(key = "leastFriction", leastFrictionName), new TaskValueStr(key = "mostFriction", mostFrictionName)) ++
          Array(new TaskValueStr(key = "planeName1", inclinedPlanes(0).getDescriptName()), new TaskValueStr(key = "planeName2", inclinedPlanes(1).getDescriptName())) ++
          Array(modifierModeKey)


        out.append( List(taskModifiers.toList) )      // Wrapped in an extra List() to match how this appears in other Tasks
      }

    }

    // Return
    out.toArray
  }

  /*
   * Gold action sequence
   */

  def actionSequenceMeasureBlockFallTime(block:EnvObject, timeTool:EnvObject, inclinedPlane:EnvObject, runner:PythonInterface): (Boolean, Int) = {
    // Slide block down plane 1
    TaskParametric.runAction("move " + PathFinder.getObjUniqueReferent(block, TaskParametric.getCurrentAgentLocation(runner)).get + " to " + PathFinder.getObjUniqueReferent(inclinedPlane, TaskParametric.getCurrentAgentLocation(runner)).get, runner)
    // Start stopwatch
    TaskParametric.runAction("activate " + timeTool.name + " in inventory", runner)

    // Wait for block to reach the bottom
    var position:Option[Double] = None
    while (position.isEmpty || (position.isDefined && position.get != 0.0)) {     // Wait for block to be at the bottom (0.0)
      // Look at plane
      TaskParametric.runAction("look at " + PathFinder.getObjUniqueReferent(inclinedPlane, TaskParametric.getCurrentAgentLocation(runner)).get, runner)
      // Check blocks position
      inclinedPlane match {
        case ip:InclinedPlane => {
          position = ip.getObjectPositionOnPlane(block)
        }
      }

      if (position.isEmpty) return (false, -1)
    }
    // Add a final look -- since the position updates after the look, we might not see it at the bottom (but only at the last position before the bottom)
    TaskParametric.runAction("look at " + PathFinder.getObjUniqueReferent(inclinedPlane, TaskParametric.getCurrentAgentLocation(runner)).get, runner)

    // Stop stopwatch
    TaskParametric.runAction("deactivate " + timeTool.name + " in inventory", runner)

    // Read time on stopwatch
    TaskParametric.runAction("examine " + timeTool.name + " in inventory", runner)
    var timePlane:Option[Int] = None
    timeTool match {
      case sw:StopWatch => {
        timePlane = Some(sw.stopwatchUserTime)
      }
    }

    if (timePlane.isEmpty) return (false, -1)

    // Success
    return (true, timePlane.get)
  }
}
