/*******************************************************************************
* Copyright (c) 2011-2014 SirSengir.
* All rights reserved. This program and the accompanying materials
* are made available under the terms of the GNU Lesser Public License v3
* which accompanies this distribution, and is available at
* http://www.gnu.org/licenses/lgpl-3.0.txt
*
* Various Contributors including, but not limited to:
* SirSengir (original work), CovertJaguar, Player, Binnie, MysteriousAges
******************************************************************************/
package forestry.farming.logic;
import java.util.ArrayList;
import java.util.Collection;
import java.util.HashMap;
import java.util.Stack;
import net.minecraft.block.Block;
import net.minecraft.block.BlockLog;
import net.minecraft.init.Items;
import net.minecraft.item.ItemStack;
import net.minecraft.util.IIcon;
import cpw.mods.fml.relauncher.Side;
import cpw.mods.fml.relauncher.SideOnly;
import net.minecraft.world.World;
import net.minecraftforge.common.util.ForgeDirection;
import forestry.api.farming.ICrop;
import forestry.api.farming.IFarmHousing;
import forestry.api.farming.IFarmable;
import forestry.core.utils.Vect;
import java.util.HashSet;
import java.util.Set;
public class FarmLogicCocoa extends FarmLogic {
private final IFarmable cocoa = new FarmableCocoa();
public FarmLogicCocoa(IFarmHousing housing) {
super(housing);
}
@Override
@SideOnly(Side.CLIENT)
public IIcon getIcon() {
return Items.dye.getIconFromDamage(3);
}
@Override
public String getName() {
return "Cocoa Plantation";
}
@Override
public int getFertilizerConsumption() {
return 120;
}
@Override
public int getWaterConsumption(float hydrationModifier) {
return (int) (20 * hydrationModifier);
}
@Override
public boolean isAcceptedResource(ItemStack itemstack) {
return false;
}
@Override
public boolean isAcceptedGermling(ItemStack itemstack) {
return cocoa.isGermling(itemstack);
}
@Override
public Collection<ItemStack> collect() {
return null;
}
private final HashMap<Vect, Integer> lastExtentsCultivation = new HashMap<Vect, Integer>();
@Override
public boolean cultivate(int x, int y, int z, ForgeDirection direction, int extent) {
Vect start = new Vect(x, y, z);
if (!lastExtentsCultivation.containsKey(start))
lastExtentsCultivation.put(start, 0);
int lastExtent = lastExtentsCultivation.get(start);
if (lastExtent > extent)
lastExtent = 0;
Vect position = translateWithOffset(x, y + 1, z, direction, lastExtent);
boolean result = tryPlantingCocoa(position);
lastExtent++;
lastExtentsCultivation.put(start, lastExtent);
return result;
}
private final HashMap<Vect, Integer> lastExtentsHarvest = new HashMap<Vect, Integer>();
@Override
public Collection<ICrop> harvest(int x, int y, int z, ForgeDirection direction, int extent) {
Collection<ICrop> crops = null;
Vect start = new Vect(x, y, z);
if (!lastExtentsHarvest.containsKey(start))
lastExtentsHarvest.put(start, 0);
int lastExtent = lastExtentsHarvest.get(start);
if (lastExtent > extent)
lastExtent = 0;
Vect position = translateWithOffset(x, y + 1, z, direction, lastExtent);
crops = getHarvestBlocks(position);
lastExtent++;
lastExtentsHarvest.put(start, lastExtent);
return crops;
}
private boolean tryPlantingCocoa(Vect position) {
World world = getWorld();
Vect current = position;
while (isWoodBlock(current) && BlockLog.func_150165_c(getBlockMeta(current)) == 3) {
for (ForgeDirection direction : ForgeDirection.VALID_DIRECTIONS) {
if (direction == ForgeDirection.UP || direction == ForgeDirection.DOWN)
continue;
Vect candidate = new Vect(current.x + direction.offsetX, current.y, current.z + direction.offsetZ);
if (isAirBlock(candidate))
return housing.plantGermling(cocoa, world, candidate.x, candidate.y, candidate.z);
}
current = current.add(new Vect(0, 1, 0));
if (current.y - position.y > 1)
break;
}
return false;
}
private Collection<ICrop> getHarvestBlocks(Vect position) {
Set<Vect> seen = new HashSet<Vect>();
Stack<ICrop> crops = new Stack<ICrop>();
// Determine what type we want to harvest.
Block block = getBlock(position);
ICrop crop = null;
if (!block.isWood(getWorld(), position.x, position.y, position.z)) {
crop = cocoa.getCropAt(getWorld(), position.x, position.y, position.z);
if (crop == null)
return crops;
}
if (crop != null)
crops.add(crop);
ArrayList<Vect> candidates = processHarvestBlock(crops, seen, position, position);
ArrayList<Vect> temp = new ArrayList<Vect>();
while (!candidates.isEmpty() && crops.size() < 20) {
for (Vect candidate : candidates)
temp.addAll(processHarvestBlock(crops, seen, position, candidate));
candidates.clear();
candidates.addAll(temp);
temp.clear();
}
// Proxies.log.finest("Logic %s at %s/%s/%s has seen %s blocks.", getClass().getName(), position.x, position.y, position.z, seen.size());
return crops;
}
private ArrayList<Vect> processHarvestBlock(Stack<ICrop> crops, Set<Vect> seen, Vect start, Vect position) {
World world = getWorld();
ArrayList<Vect> candidates = new ArrayList<Vect>();
// Get additional candidates to return
for (int i = -1; i < 2; i++)
for (int j = 0; j < 2; j++)
for (int k = -1; k < 2; k++) {
Vect candidate = new Vect(position.x + i, position.y + j, position.z + k);
if (candidate.equals(position))
continue;
if (Math.abs(candidate.x - start.x) > 5)
continue;
if (Math.abs(candidate.z - start.z) > 5)
continue;
// See whether the given position has already been processed
if (seen.contains(candidate))
continue;
ICrop crop = cocoa.getCropAt(world, candidate.x, candidate.y, candidate.z);
if (crop != null) {
crops.push(crop);
candidates.add(candidate);
seen.add(candidate);
} else if (isWoodBlock(candidate)) {
candidates.add(candidate);
seen.add(candidate);
}
}
return candidates;
}
}