286 lines
8.3 KiB
Java
286 lines
8.3 KiB
Java
/*
|
|
* Copyright (c) 2016-2017, Adam <Adam@sigterm.info>
|
|
* All rights reserved.
|
|
*
|
|
* Redistribution and use in source and binary forms, with or without
|
|
* modification, are permitted provided that the following conditions are met:
|
|
*
|
|
* 1. Redistributions of source code must retain the above copyright notice, this
|
|
* list of conditions and the following disclaimer.
|
|
* 2. Redistributions in binary form must reproduce the above copyright notice,
|
|
* this list of conditions and the following disclaimer in the documentation
|
|
* and/or other materials provided with the distribution.
|
|
*
|
|
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND
|
|
* ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
|
|
* WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
|
|
* DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE LIABLE FOR
|
|
* ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
|
|
* (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
|
|
* LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
|
|
* ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
|
* (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
|
|
* SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
|
*/
|
|
package net.runelite.mixins;
|
|
|
|
import net.runelite.api.Actor;
|
|
import net.runelite.api.Hitsplat;
|
|
import net.runelite.api.NPC;
|
|
import net.runelite.api.NPCDefinition;
|
|
import net.runelite.api.Perspective;
|
|
import net.runelite.api.Player;
|
|
import net.runelite.api.Point;
|
|
import net.runelite.api.Sprite;
|
|
import net.runelite.api.coords.LocalPoint;
|
|
import net.runelite.api.coords.WorldArea;
|
|
import net.runelite.api.coords.WorldPoint;
|
|
import net.runelite.api.events.AnimationChanged;
|
|
import net.runelite.api.events.HitsplatApplied;
|
|
import net.runelite.api.events.PlayerDeath;
|
|
import net.runelite.api.events.PlayerDeath;
|
|
import net.runelite.api.events.SpotAnimationChanged;
|
|
import net.runelite.api.events.InteractingChanged;
|
|
import net.runelite.api.events.OverheadTextChanged;
|
|
import java.awt.Graphics2D;
|
|
import java.awt.Polygon;
|
|
import java.awt.image.BufferedImage;
|
|
import net.runelite.api.mixins.FieldHook;
|
|
import net.runelite.api.mixins.Inject;
|
|
import net.runelite.api.mixins.MethodHook;
|
|
import net.runelite.api.mixins.Mixin;
|
|
import net.runelite.api.mixins.Shadow;
|
|
import net.runelite.rs.api.RSActor;
|
|
import net.runelite.rs.api.RSClient;
|
|
import net.runelite.rs.api.RSHealthBar;
|
|
import net.runelite.rs.api.RSHealthBarDefinition;
|
|
import net.runelite.rs.api.RSHealthBarUpdate;
|
|
import net.runelite.rs.api.RSIterableNodeDeque;
|
|
import net.runelite.rs.api.RSNPC;
|
|
import net.runelite.rs.api.RSNode;
|
|
|
|
@Mixin(RSActor.class)
|
|
public abstract class RSActorMixin implements RSActor
|
|
{
|
|
@Shadow("client")
|
|
private static RSClient client;
|
|
|
|
@Inject
|
|
@Override
|
|
public Actor getInteracting()
|
|
{
|
|
int index = getRSInteracting();
|
|
if (index == -1 || index == 65535)
|
|
{
|
|
return null;
|
|
}
|
|
|
|
if (index < 32768)
|
|
{
|
|
NPC[] npcs = client.getCachedNPCs();
|
|
return npcs[index];
|
|
}
|
|
|
|
index -= 32768;
|
|
Player[] players = client.getCachedPlayers();
|
|
return players[index];
|
|
}
|
|
|
|
@Inject
|
|
@Override
|
|
public int getHealthRatio()
|
|
{
|
|
RSIterableNodeDeque healthBars = getHealthBars();
|
|
if (healthBars != null)
|
|
{
|
|
RSNode current = healthBars.getCurrent();
|
|
RSNode next = current.getNext();
|
|
if (next instanceof RSHealthBar)
|
|
{
|
|
RSHealthBar wrapper = (RSHealthBar) next;
|
|
RSIterableNodeDeque updates = wrapper.getUpdates();
|
|
|
|
RSNode currentUpdate = updates.getCurrent();
|
|
RSNode nextUpdate = currentUpdate.getNext();
|
|
if (nextUpdate instanceof RSHealthBarUpdate)
|
|
{
|
|
RSHealthBarUpdate update = (RSHealthBarUpdate) nextUpdate;
|
|
return update.getHealthRatio();
|
|
}
|
|
}
|
|
}
|
|
return -1;
|
|
}
|
|
|
|
@Inject
|
|
@Override
|
|
public int getHealth()
|
|
{
|
|
RSIterableNodeDeque healthBars = getHealthBars();
|
|
if (healthBars != null)
|
|
{
|
|
RSNode current = healthBars.getCurrent();
|
|
RSNode next = current.getNext();
|
|
if (next instanceof RSHealthBar)
|
|
{
|
|
RSHealthBar wrapper = (RSHealthBar) next;
|
|
RSHealthBarDefinition definition = wrapper.getDefinition();
|
|
return definition.getHealthScale();
|
|
}
|
|
}
|
|
return -1;
|
|
}
|
|
|
|
@Inject
|
|
@Override
|
|
public WorldPoint getWorldLocation()
|
|
{
|
|
return WorldPoint.fromLocal(client,
|
|
this.getPathX()[0] * Perspective.LOCAL_TILE_SIZE + Perspective.LOCAL_TILE_SIZE / 2,
|
|
this.getPathY()[0] * Perspective.LOCAL_TILE_SIZE + Perspective.LOCAL_TILE_SIZE / 2,
|
|
client.getPlane());
|
|
}
|
|
|
|
@Inject
|
|
@Override
|
|
public LocalPoint getLocalLocation()
|
|
{
|
|
return new LocalPoint(getX(), getY());
|
|
}
|
|
|
|
@Inject
|
|
@Override
|
|
public Polygon getCanvasTilePoly()
|
|
{
|
|
return Perspective.getCanvasTilePoly(client, getLocalLocation());
|
|
}
|
|
|
|
@Inject
|
|
@Override
|
|
public Point getCanvasTextLocation(Graphics2D graphics, String text, int zOffset)
|
|
{
|
|
return Perspective.getCanvasTextLocation(client, graphics, getLocalLocation(), text, zOffset);
|
|
}
|
|
|
|
@Inject
|
|
@Override
|
|
public Point getCanvasImageLocation(BufferedImage image, int zOffset)
|
|
{
|
|
return Perspective.getCanvasImageLocation(client, getLocalLocation(), image, zOffset);
|
|
}
|
|
|
|
@Inject
|
|
@Override
|
|
public Point getCanvasSpriteLocation(Sprite sprite, int zOffset)
|
|
{
|
|
return Perspective.getCanvasSpriteLocation(client, getLocalLocation(), sprite, zOffset);
|
|
}
|
|
|
|
@Inject
|
|
@Override
|
|
public Point getMinimapLocation()
|
|
{
|
|
return Perspective.localToMinimap(client, getLocalLocation());
|
|
}
|
|
|
|
@FieldHook("sequence")
|
|
@Inject
|
|
public void animationChanged(int idx)
|
|
{
|
|
AnimationChanged animationChange = new AnimationChanged();
|
|
animationChange.setActor(this);
|
|
client.getCallbacks().post(AnimationChanged.class, animationChange);
|
|
}
|
|
|
|
@FieldHook("spotAnimation")
|
|
@Inject
|
|
public void spotAnimationChanged(int idx)
|
|
{
|
|
SpotAnimationChanged spotAnimationChanged = new SpotAnimationChanged();
|
|
spotAnimationChanged.setActor(this);
|
|
client.getCallbacks().post(SpotAnimationChanged.class, spotAnimationChanged);
|
|
}
|
|
|
|
@FieldHook("targetIndex")
|
|
@Inject
|
|
public void interactingChanged(int idx)
|
|
{
|
|
InteractingChanged interactingChanged = new InteractingChanged(this, getInteracting());
|
|
client.getCallbacks().post(InteractingChanged.class, interactingChanged);
|
|
}
|
|
|
|
@FieldHook("overheadText")
|
|
@Inject
|
|
public void overheadTextChanged(int idx)
|
|
{
|
|
String overheadText = getOverheadText();
|
|
if (overheadText != null)
|
|
{
|
|
OverheadTextChanged overheadTextChanged = new OverheadTextChanged(this, overheadText);
|
|
client.getCallbacks().post(OverheadTextChanged.class, overheadTextChanged);
|
|
}
|
|
}
|
|
|
|
@Inject
|
|
@Override
|
|
public WorldArea getWorldArea()
|
|
{
|
|
int size = 1;
|
|
if (this instanceof NPC)
|
|
{
|
|
NPCDefinition composition = ((NPC)this).getDefinition();
|
|
if (composition != null && composition.getConfigs() != null)
|
|
{
|
|
composition = composition.transform();
|
|
}
|
|
if (composition != null)
|
|
{
|
|
size = composition.getSize();
|
|
}
|
|
}
|
|
|
|
return new WorldArea(this.getWorldLocation(), size, size);
|
|
}
|
|
|
|
@Inject
|
|
@MethodHook("addHealthBar")
|
|
public void setCombatInfo(int combatInfoId, int gameCycle, int var3, int var4, int healthRatio, int health)
|
|
{
|
|
if (healthRatio == 0)
|
|
{
|
|
if (this instanceof Player)
|
|
{
|
|
final PlayerDeath event = new PlayerDeath((Player) this);
|
|
client.getCallbacks().post(PlayerDeath.class, event);
|
|
}
|
|
else if (this instanceof RSNPC)
|
|
{
|
|
((RSNPC) this).setDead(true);
|
|
}
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Called after a hitsplat has been processed on an actor.
|
|
* Note that this event runs even if the hitsplat didn't show up,
|
|
* i.e. the actor already had 4 visible hitsplats.
|
|
*
|
|
* @param type The hitsplat type (i.e. color)
|
|
* @param value The value of the hitsplat (i.e. how high the hit was)
|
|
* @param var3 unknown
|
|
* @param var4 unknown
|
|
* @param gameCycle The gamecycle the hitsplat was applied on
|
|
* @param duration The amount of gamecycles the hitsplat will last for
|
|
*/
|
|
@Inject
|
|
@MethodHook(value = "addHitSplat", end = true)
|
|
public void applyActorHitsplat(int type, int value, int var3, int var4, int gameCycle, int duration)
|
|
{
|
|
final Hitsplat hitsplat = new Hitsplat(Hitsplat.HitsplatType.fromInteger(type), value, gameCycle + duration);
|
|
final HitsplatApplied event = new HitsplatApplied();
|
|
event.setActor(this);
|
|
event.setHitsplat(hitsplat);
|
|
client.getCallbacks().post(HitsplatApplied.class, event);
|
|
}
|
|
}
|