blob: d2ab44d65c8b8172f83a6db5455e45507e18a4d0 [file] [log] [blame] [raw]
package li.cil.oc.common.event
import java.util
import li.cil.oc.api.event.RobotMoveEvent
import li.cil.oc.server.component.UpgradeChunkloader
import net.minecraft.world.{ChunkCoordIntPair, World}
import net.minecraftforge.common.ForgeChunkManager
import net.minecraftforge.common.ForgeChunkManager.{LoadingCallback, Ticket}
import net.minecraftforge.event.ForgeSubscribe
import net.minecraftforge.event.world.WorldEvent
import scala.collection.convert.WrapAsScala._
import scala.collection.mutable
object ChunkloaderUpgradeHandler extends LoadingCallback {
val restoredTickets = mutable.Map.empty[String, Ticket]
override def ticketsLoaded(tickets: util.List[Ticket], world: World) {
for (ticket <- tickets) {
restoredTickets += ticket.getModData.getString("address") -> ticket
}
}
@ForgeSubscribe
def onWorldSave(e: WorldEvent.Save) {
// Any tickets that were not reassigned by the time the world gets saved
// again can be considered orphaned, so we release them.
// TODO figure out a better event *after* tile entities were restored
// but *before* the world is saved, because the tickets are saved first,
// so if the save is because the game is being quit the tickets aren't
// actually being cleared. This will *usually* not be a problem, but it
// has room for improvement.
restoredTickets.values.foreach(ForgeChunkManager.releaseTicket)
restoredTickets.clear()
}
// TODO it might be necessary to use pre move to force load the target chunk
// in case the robot moves across a chunk border into an otherwise unloaded
// chunk (I think it would just fail to move otherwise)
@ForgeSubscribe
def onMove(e: RobotMoveEvent.Post) {
for (slot <- 0 until e.robot.getSizeInventory) {
e.robot.getComponentInSlot(slot) match {
case loader: UpgradeChunkloader => updateLoadedChunk(loader)
case _ =>
}
}
}
def updateLoadedChunk(loader: UpgradeChunkloader) {
val robotChunk = new ChunkCoordIntPair(loader.owner.xPosition.toInt / 16, loader.owner.zPosition.toInt / 16)
loader.ticket.foreach(ticket => {
ticket.getChunkList.collect {
case chunk: ChunkCoordIntPair if chunk != robotChunk => ForgeChunkManager.unforceChunk(ticket, chunk)
}
ForgeChunkManager.forceChunk(ticket, robotChunk)
ticket.getModData.setString("address", loader.node.address)
})
}
}