// • ▌ ▄ ·.  ▄▄▄·  ▄▄ • ▪   ▄▄· ▄▄▄▄·  ▄▄▄·  ▐▄▄▄  ▄▄▄ .
// ·██ ▐███▪▐█ ▀█ ▐█ ▀ ▪██ ▐█ ▌▪▐█ ▀█▪▐█ ▀█ •█▌ ▐█▐▌·
// ▐█ ▌▐▌▐█·▄█▀▀█ ▄█ ▀█▄▐█·██ ▄▄▐█▀▀█▄▄█▀▀█ ▐█▐ ▐▌▐▀▀▀
// ██ ██▌▐█▌▐█ ▪▐▌▐█▄▪▐█▐█▌▐███▌██▄▪▐█▐█ ▪▐▌██▐ █▌▐█▄▄▌
// ▀▀  █▪▀▀▀ ▀  ▀ ·▀▀▀▀ ▀▀▀·▀▀▀ ·▀▀▀▀  ▀  ▀ ▀▀  █▪ ▀▀▀
//      Magicbane Emulator Project © 2013 - 2022
//                www.magicbane.com


package engine.objects;

import engine.Enum;
import engine.InterestManagement.InterestManager;
import engine.InterestManagement.WorldGrid;
import engine.gameManager.*;
import engine.math.Vector3f;
import engine.math.Vector3fImmutable;
import engine.net.ByteBufferWriter;
import engine.net.client.msg.ErrorPopupMsg;
import engine.server.MBServerStatics;
import org.pmw.tinylog.Logger;

import java.net.UnknownHostException;
import java.sql.ResultSet;
import java.sql.SQLException;
import java.time.LocalDateTime;
import java.util.ArrayList;
import java.util.HashMap;
import java.util.HashSet;
import java.util.Iterator;
import java.util.concurrent.ConcurrentHashMap;

import static engine.gameManager.DbManager.MineQueries;
import static engine.gameManager.DbManager.getObject;
import static engine.math.FastMath.sqr;

public class Mine extends AbstractGameObject {

    public static ConcurrentHashMap<Mine, Integer> mineMap = new ConcurrentHashMap<>(MBServerStatics.CHM_INIT_CAP, MBServerStatics.CHM_LOAD, MBServerStatics.CHM_THREAD_LOW);
    public static ConcurrentHashMap<Integer, Mine> towerMap = new ConcurrentHashMap<>(MBServerStatics.CHM_INIT_CAP, MBServerStatics.CHM_LOAD, MBServerStatics.CHM_THREAD_LOW);
    private final String zoneName;
    private final Zone parentZone;
    public boolean isActive = false;
    public PlayerCharacter lastClaimer;
    public boolean wasClaimed = false;
    // Not persisted to DB
    public String guildName;
    public GuildTag guildTag;
    public String nationName;
    public GuildTag nationTag;
    private Resource production;
    private Guild owningGuild;
    private int flags;
    private int buildingID;
    private MineProduction mineType;

    public int openHour;
    public int openMinute;
    public int capSize;
    public LocalDateTime liveTime;
    public final HashSet<Integer> _playerMemory = new HashSet<>();
    public ArrayList<PlayerCharacter> affectedPlayers = new ArrayList<>();

    //stronghold stuff
    public boolean isStronghold = false;
    public ArrayList<Mob> strongholdMobs;
    public HashMap<Integer,Integer> oldBuildings;

    /**
     * ResultSet Constructor
     */
    public Mine(ResultSet rs) throws SQLException, UnknownHostException {
        super(rs);

        this.mineType = MineProduction.getByName(rs.getString("mine_type"));

        int ownerUID = rs.getInt("mine_ownerUID");
        this.buildingID = rs.getInt("mine_buildingUID");
        this.flags = rs.getInt("flags");
        int parent = rs.getInt("parent");
        this.parentZone = ZoneManager.getZoneByUUID(parent);
        this.zoneName = this.parentZone.getParent().getName();

        this.owningGuild = Guild.getGuild(ownerUID);
        Guild nation = null;

        if (this.owningGuild.isEmptyGuild()) {
            this.guildName = "";
            this.guildTag = GuildTag.ERRANT;
            nation = Guild.getErrantGuild();
            this.owningGuild = Guild.getErrantGuild();
        } else {
            this.guildName = this.owningGuild.getName();
            this.guildTag = this.owningGuild.getGuildTag();
            nation = this.owningGuild.getNation();
        }

        if (!nation.isEmptyGuild()) {
            this.nationName = nation.getName();
            this.nationTag = nation.getGuildTag();
        } else {
            this.nationName = "";
            this.nationTag = GuildTag.ERRANT;
        }

        this.production = Resource.valueOf(rs.getString("mine_resource"));
        this.lastClaimer = null;
        this.openHour = rs.getInt("mineLiveHour");
        this.openMinute = rs.getInt("mineLiveMinute");
        this.capSize = rs.getInt("capSize");
        this.liveTime = LocalDateTime.now().withHour(this.openHour).withMinute(this.openMinute);
        Building tower = BuildingManager.getBuildingFromCache(this.buildingID);
        if(tower != null){
            tower.setMaxHitPoints(5000f * this.capSize);
            tower.setCurrentHitPoints(tower.healthMax);
        }
    }

    public static void releaseMineClaims(PlayerCharacter playerCharacter) {

        if (playerCharacter == null)
            return;

        for (Mine mine : Mine.getMines()) {

            if (mine.lastClaimer != null)
                if (mine.lastClaimer.equals(playerCharacter)) {
                    mine.lastClaimer = null;
                    mine.updateGuildOwner(null);
                }

        }
    }

    public static void SendMineAttackMessage(Building mine) {

        if (mine.getBlueprint() == null)
            return;

        if (mine.getBlueprint().getBuildingGroup() != Enum.BuildingGroup.MINE)
            return;


        if (mine.getGuild().isEmptyGuild())
            return;

        if (mine.getGuild().getNation().isEmptyGuild())
            return;

        if (mine.getTimeStamp("MineAttack") > System.currentTimeMillis())
            return;

        mine.getTimestamps().put("MineAttack", System.currentTimeMillis() + MBServerStatics.ONE_MINUTE);

        ChatManager.chatNationInfo(mine.getGuild().getNation(), mine.getName() + " in " + mine.getParentZone().getParent().getName() + " is Under attack!");
    }

    public static void loadAllMines() {

        try {

            //Load mine resources
            MineProduction.addResources();

            //pre-load all building sets
            ArrayList<Mine> serverMines = MineQueries.GET_ALL_MINES_FOR_SERVER();

            for (Mine mine : serverMines) {
                Mine.mineMap.put(mine, mine.buildingID);
                Mine.towerMap.put(mine.buildingID, mine);
            }

        } catch (Exception e) {
            e.printStackTrace();
        }
    }

    /*
     * Getters
     */

    public static Mine getMineFromTower(int towerID) {
        return Mine.towerMap.get(towerID);
    }

    public static void serializeForClientMsg(Mine mine, ByteBufferWriter writer) {
        try {
            writer.putInt(mine.getObjectType().ordinal());
            writer.putInt(mine.getObjectUUID());
            writer.putInt(mine.getObjectUUID()); //actually a hash of mine
            if(mine.isStronghold){
                writer.putString("STRONGHOLD");
                writer.putString("");
            }else {
                writer.putString(mine.mineType.name);
                writer.putString(mine.capSize + " Man ");
            }
            //writer.putString(mine.zoneName + " " + mine.capSize + " Man ");

            writer.putInt(mine.production.hash);
            writer.putInt(mine.production.baseProduction);
            writer.putInt(mine.getModifiedProductionAmount()); //TODO calculate range penalty here
            writer.putInt(3600); //window in seconds

            LocalDateTime mineOpenTime = LocalDateTime.now().withHour(mine.openHour).withMinute(mine.openMinute).withSecond(0).withNano(0);

            writer.putLocalDateTime(mineOpenTime);
            writer.putLocalDateTime(mineOpenTime.plusMinutes(30));
            writer.put(mine.isActive ? (byte) 0x01 : (byte) 0x00);

            Building mineTower = BuildingManager.getBuilding(mine.buildingID);
            if (mineTower != null) {
                writer.putFloat(mineTower.getLoc().x);
                writer.putFloat(mineTower.getParentZone().getLoc().y);
                writer.putFloat(mineTower.getLoc().z);
            } else {
                writer.putFloat(mine.parentZone.getLoc().x);
                writer.putFloat(mine.parentZone.getLoc().y);
                writer.putFloat(mine.parentZone.getLoc().z);
                Logger.error("Mine Tower Was Null For Mine: " + mine.getObjectUUID());
            }

            writer.putInt(mine.isExpansion() ? mine.mineType.xpacHash : mine.mineType.hash);

            if (mine.isStronghold) {
                writer.putString("");
                GuildTag._serializeForDisplay(Guild.getErrantGuild().getGuildTag(), writer);
                writer.putString("");
                GuildTag._serializeForDisplay(Guild.getErrantGuild().getGuildTag(), writer);
            }else {
                writer.putString(mine.guildName);
                GuildTag._serializeForDisplay(mine.guildTag, writer);
                writer.putString(mine.nationName);
                GuildTag._serializeForDisplay(mine.nationTag, writer);
            }
        } catch (Exception e) {
            Logger.error("Failed TO Serialize Mine Because: " + e.getMessage());
        }
    }

    public static ArrayList<Mine> getMinesForGuild(int guildID) {

        ArrayList<Mine> mineList = new ArrayList<>();

        // Only inactive mines are returned.

        for (Mine mine : Mine.mineMap.keySet()) {
            if (mine.owningGuild.getObjectUUID() == guildID &&
                    mine.isActive == false)
                mineList.add(mine);
        }
        return mineList;
    }

    /*
     * Database
     */
    public static Mine getMine(int UID) {
        return MineQueries.GET_MINE(UID);

    }

    public static ArrayList<Mine> getMines() {
        return new ArrayList<>(mineMap.keySet());
    }

    public static boolean validateClaimer(PlayerCharacter playerCharacter) {

        // Method validates that the claimer meets
        // all the requirements to claim; landed
        // guild with a warehouse, etc.

        Guild playerGuild;

        //verify the player exists

        if (playerCharacter == null)
            return false;

        //verify the player is in valid guild

        playerGuild = playerCharacter.getGuild();

        // Can't claim something if you don't have a guild!

        if (playerGuild.isEmptyGuild())
            return false;

        if (playerGuild.getNation().isEmptyGuild())
            return false;

        // Guild must own a city to hold a mine.

        City guildCity = playerGuild.getOwnedCity();

        if (guildCity == null)
            return false;

        if (guildCity.getWarehouse() == null) {
            ErrorPopupMsg.sendErrorMsg(playerCharacter, "No Warehouse exists for this claim.");
            return false;
        }

        // Number of mines is based on the rank of the nation's tree.

        City nationCapitol = playerGuild.getNation().getOwnedCity();

        Building nationCapitolTOL = nationCapitol.getTOL();

        if (nationCapitolTOL == null)
            return false;

        int treeRank = nationCapitolTOL.getRank();

        if (treeRank < 1)
            return false;

        return true;
    }

    public boolean changeProductionType(Resource resource) {
        if (!this.validForMine(resource))
            return false;
        //update resource in database;
        if (!MineQueries.CHANGE_RESOURCE(this, resource))
            return false;

        this.production = resource;
        return true;
    }

    public MineProduction getMineType() {
        return this.mineType;
    }

    public void setMineType(String type) {
        this.mineType = MineProduction.getByName(type);
    }

    public String getZoneName() {
        return this.zoneName;
    }

    public Resource getProduction() {
        return this.production;
    }

    public boolean getIsActive() {
        return this.isActive;
    }

    public Guild getOwningGuild() {
        if (this.owningGuild == null)
            return Guild.getErrantGuild();
        else
            return this.owningGuild;
    }

    public void setOwningGuild(Guild owningGuild) {
        this.owningGuild = owningGuild;
    }

    /*
     * Serialization
     */

    public int getFlags() {
        return flags;
    }

    public void setFlags(int flags) {
        this.flags = flags;
    }

    public Zone getParentZone() {
        return parentZone;
    }

    public GuildTag getGuildTag() {
        return guildTag;
    }

    public void setActive(boolean isAc) {

        this.isActive = isAc;
        Building building = BuildingManager.getBuildingFromCache(this.buildingID);
        if (building != null && !this.isActive)
            building.isDeranking.compareAndSet(true, false);

        if(!isAc){
            for(PlayerCharacter player : this.affectedPlayers){
                try {
                    player.ZergMultiplier = 1.0f;
                } catch(Exception e){
                    //something went wrong resetting zerg multiplier, maybe player was deleted?
                }
            }
        }
    }

    public boolean validForMine(Resource r) {
        if (this.mineType == null)
            return false;
        return this.mineType.validForMine(r, this.isExpansion());
    }

    public void serializeForMineProduction(ByteBufferWriter writer) {
        writer.putInt(this.getObjectType().ordinal());
        writer.putInt(this.getObjectUUID());
        writer.putInt(this.getObjectUUID()); //actually a hash of mine
        //		writer.putInt(0x215C92BB); //this.unknown1);
        writer.putString(this.mineType.name);
        writer.putString(this.zoneName);
        writer.putInt(this.production.hash);
        writer.putInt(this.production.baseProduction);
        writer.putInt(this.getModifiedProductionAmount()); //TODO calculate range penalty here
        writer.putInt(3600); //window in seconds
        writer.putInt(this.isExpansion() ? this.mineType.xpacHash : this.mineType.hash);
    }

    @Override
    public void updateDatabase() {
        // TODO Create update logic.
    }

    public int getBuildingID() {
        return buildingID;
    }

    public void setBuildingID(int buildingID) {
        this.buildingID = buildingID;
    }

    public void handleDestroyMine() {

        if (!this.isActive)
            return;

        //remove tags from mine

        this.guildName = "";
        this.nationName = "";
        this.owningGuild = Guild.getErrantGuild();
        this.lastClaimer = null;
        this.wasClaimed = false;

        // Update database

        DbManager.MineQueries.CHANGE_OWNER(this, 0);

        // Update mesh

        Building mineBuilding = BuildingManager.getBuildingFromCache(this.buildingID);

        if (mineBuilding == null) {
            Logger.debug("Null mine building " + this.getObjectUUID() + ". Unable to Load Building with UID " + this.buildingID);
            return;
        }

        mineBuilding.setOwner(null);
        mineBuilding.refresh(false);

        // remove hirelings

        Building building = (Building) getObject(Enum.GameObjectType.Building, this.buildingID);
        BuildingManager.cleanupHirelings(building);
    }

    public boolean claimMine(PlayerCharacter claimer) {

        if (claimer == null)
            return false;

        if (!validateClaimer(claimer))
            return false;

        if (!this.isActive) {
            ErrorPopupMsg.sendErrorMsg(claimer, "Can not for to claim inactive mine.");
            return false;
        }

        if (!updateGuildOwner(claimer))
            return false;

        // Successful claim

        this.lastClaimer = claimer;

        return true;
    }

    public boolean depositMineResources() {

        if (this.owningGuild.isEmptyGuild())
            return false;

        if (this.owningGuild.getOwnedCity() == null)
            return false;

        if (this.owningGuild.getOwnedCity().getWarehouse() == null)
            return false;

        ItemBase resourceIB = ItemBase.getItemBase(this.production.UUID);
        return this.owningGuild.getOwnedCity().getWarehouse().depositFromMine(this, resourceIB, this.getModifiedProductionAmount());
    }

    public boolean updateGuildOwner(PlayerCharacter playerCharacter) {

        Building mineBuilding = BuildingManager.getBuildingFromCache(this.buildingID);

        //should never return null, but let's check just in case.

        if (mineBuilding == null) {
            ChatManager.chatSystemError(playerCharacter, "Unable to find mine tower.");
            Logger.debug("Failed to Update Mine with UID " + this.getObjectUUID() + ". Unable to Load Building with UID " + this.buildingID);
            return false;
        }

        if (playerCharacter == null) {
            this.owningGuild = Guild.getErrantGuild();
            this.guildName = "None";
            this.guildTag = GuildTag.ERRANT;
            this.nationName = "None";
            this.nationTag = GuildTag.ERRANT;
            //Update Building.
            mineBuilding.setOwner(null);
            WorldGrid.updateObject(mineBuilding);
            return true;
        }

        Guild guild = playerCharacter.getGuild();

        if (guild.getOwnedCity() == null)
            return false;

        if (!MineQueries.CHANGE_OWNER(this, guild.getObjectUUID())) {
            Logger.debug("Database failed to Change Ownership of Mine with UID " + this.getObjectUUID());
            ChatManager.chatSystemError(playerCharacter, "Failed to claim Mine.");
            return false;
        }

        //update mine.
        this.owningGuild = guild;

        //Update Building.
        PlayerCharacter guildLeader = (PlayerCharacter) Guild.GetGL(this.owningGuild);

        if (guildLeader != null)
            mineBuilding.setOwner(guildLeader);
        WorldGrid.updateObject(mineBuilding);
        return true;
    }

    public boolean isExpansion() {
        return (this.flags & 2) != 0;
    }

    public int getModifiedProductionAmount() {
        ItemBase resourceBase = ItemBase.getItemBase(this.production.UUID);
        if(resourceBase == null)
            return 0;
        int value = resourceBase.getBaseValue();

        int amount = 0;
        switch(this.capSize){
            case 3:
                amount = 1800000;
                break;
            case 5:
                amount = 3000000;
                break;
            case 10:
                amount = 6000000;
                break;
            case 20:
                amount = 12000000;
                break;
        }
        if(this.production.UUID == 7)
            value = 1;

        amount = amount / value;

        return amount;
    }
    public void onEnter() {

        Building tower = BuildingManager.getBuildingFromCache(this.buildingID);
        if(tower == null)
            return;

        // Gather current list of players within the zone bounds

        HashSet<AbstractWorldObject> currentPlayers = WorldGrid.getObjectsInRangePartial(tower.loc, Enum.CityBoundsType.GRID.extents, MBServerStatics.MASK_PLAYER);
        HashMap<Guild,ArrayList<PlayerCharacter>> charactersByNation = new HashMap<>();
        ArrayList<Guild> updatedNations = new ArrayList<>();
        for (AbstractWorldObject playerObject : currentPlayers) {

            if (playerObject == null)
                continue;

            PlayerCharacter player = (PlayerCharacter) playerObject;

            if(this.affectedPlayers.contains(player) == false)
                this.affectedPlayers.add(player);

            if(!this._playerMemory.contains(player.getObjectUUID())){
                this._playerMemory.add(player.getObjectUUID());
            }
            Guild nation = player.guild.getNation();
            if(charactersByNation.containsKey(nation)){
                if(!charactersByNation.get(nation).contains(player)) {
                    charactersByNation.get(nation).add(player);
                    if(!updatedNations.contains(nation)){
                        updatedNations.add(nation);
                    }
                }
            }else{
                ArrayList<PlayerCharacter> players = new ArrayList<>();
                players.add(player);
                charactersByNation.put(nation,players);
                if(!updatedNations.contains(nation)){
                    updatedNations.add(nation);
                }
            }
        }
        for(Guild nation : updatedNations){
            float multiplier = ZergManager.getCurrentMultiplier(charactersByNation.get(nation).size(),this.capSize);
            for(PlayerCharacter player : charactersByNation.get(nation)){
                player.ZergMultiplier = multiplier;
            }
        }
        try
        {
            this.onExit(this._playerMemory);
        }
        catch(Exception ignored){

        }
    }

    private void onExit(HashSet<Integer> currentMemory) {

        Building tower = BuildingManager.getBuildingFromCache(this.buildingID);
        if(tower == null)
            return;
        ArrayList<Integer>toRemove = new ArrayList<>();
        HashSet<AbstractWorldObject> currentPlayers = WorldGrid.getObjectsInRangePartial(tower.loc, Enum.CityBoundsType.GRID.extents, MBServerStatics.MASK_PLAYER);
        for(Integer id : currentMemory){
            PlayerCharacter pc = PlayerCharacter.getPlayerCharacter(id);
            if(currentPlayers.contains(pc) == false){
                toRemove.add(id);
                pc.ZergMultiplier = 1.0f;
            }
        }

        // Remove players from city memory

        _playerMemory.removeAll(toRemove);
    }

    public void StartStronghold(){

        //remove buildings
        Building tower = BuildingManager.getBuilding(this.buildingID);
        if(tower == null)
            return;

        this.isStronghold = true;
        this.strongholdMobs = new ArrayList<>();
        this.oldBuildings = new HashMap<>();

        Zone mineZone = ZoneManager.findSmallestZone(tower.loc);
        for(Building building : mineZone.zoneBuildingSet){
            oldBuildings.put(building.getObjectUUID(),building.meshUUID);
            building.setMeshUUID(407650);
            building.setMeshScale(new Vector3f(0,0,0));
            InterestManager.setObjectDirty(building);
            WorldGrid.updateObject(building);
        }

        //update tower to become stronghold mesh
        tower.setMeshUUID(423600);
        tower.setMeshScale(new Vector3f(1.5f,1.5f,1.5f));
        InterestManager.setObjectDirty(tower);
        WorldGrid.updateObject(tower);

        //create elite mobs
        for(int i = 0; i < 15; i++){
            Mob guard = Mob.createMob(14315, Vector3fImmutable.getRandomPointOnCircle(tower.loc,30),Guild.getErrantGuild(),true,mineZone,null,0, "Elite Guardian",65);
            if(guard != null){
                guard.setLevel((short)65);
                guard.setResists(new Resists("Elite"));
                guard.healthMax *= 2;
                guard.setHealth(guard.healthMax);
                guard.spawnTime = 1000000000;
                guard.BehaviourType = Enum.MobBehaviourType.Aggro;
                guard.maxDamageHandOne *= 2;
                guard.minDamageHandOne *= 2;
                guard.atrHandOne *= 2;
                guard.defenseRating *= 2;
                InterestManager.setObjectDirty(guard);
                this.strongholdMobs.add(guard);
                LootManager.GenerateStrongholdLoot(guard,false);
            }
        }
        //create stronghold commander
        Mob commander = Mob.createMob(14293, tower.loc,Guild.getErrantGuild(),true,mineZone,null,0, "Guardian Commander",75);
        if(commander != null){
            commander.setLevel((short)75);
            commander.setResists(new Resists("Elite"));
            commander.healthMax = 50000;
            commander.setHealth(commander.healthMax);
            commander.spawnTime = 1000000000;
            commander.BehaviourType = Enum.MobBehaviourType.Aggro;
            commander.maxDamageHandOne = 1500;
            commander.minDamageHandOne = 3500;
            commander.atrHandOne = 3500;
            commander.defenseRating = 3500;
            commander.mobPowers.put(429413547,40);
            commander.mobPowers.put(429032838,40);
            commander.mobPowers.put(57584498,40);
            InterestManager.setObjectDirty(commander);
            this.strongholdMobs.add(commander);
            LootManager.GenerateStrongholdLoot(commander,true);
        }

        this.setActive(true);
        tower.setProtectionState(Enum.ProtectionState.PROTECTED);
    }
    public void EndStronghold(){

        //restore the buildings
        Building tower = BuildingManager.getBuilding(this.buildingID);
        if(tower == null)
            return;

        this.isStronghold = false;

        //get rid of the mobs
        for(Mob mob : this.strongholdMobs) {
            mob.despawn();
            mob.removeFromCache();
        }

        //restore the buildings
        Zone mineZone = ZoneManager.findSmallestZone(tower.loc);
        for(Building building : mineZone.zoneBuildingSet){
            if(this.oldBuildings.containsKey(building.getObjectUUID())) {
                building.setMeshUUID(this.oldBuildings.get(building.getObjectUUID()));
                building.setMeshScale(new Vector3f(1, 1, 1));
            }
        }

        //update tower to become Mine Tower again
        tower.setMeshUUID(1500100);

        this.setActive(false);
        tower.setProtectionState(Enum.ProtectionState.NPC);
    }
}