package org.rsbot.script.methods;
import org.rsbot.script.internal.ScriptHandler;
import org.rsbot.script.web.*;
import org.rsbot.script.wrappers.RSTile;
import org.rsbot.script.wrappers.RSWeb;
import java.util.*;
/**
* The web class.
*
* @author Timer
*/
public class Web extends MethodProvider {
public static final HashMap<RSTile, Integer> rs_map = new HashMap<RSTile, Integer>();
public boolean webScriptsLoaded = false;
private long lastLocalAccess = 0;
private WebData webData = new WebData(methods);
Web(final MethodContext ctx) {
super(ctx);
}
/**
* Gets the closest supported bank that is usable.
*
* @param tile The tile to look off of.
* @return The closest bank's tile.
*/
public RSTile getNearestBank(final RSTile tile) {
double dist = -1.0D;
RSTile finalTile = null;
final RSTile[] BANKS = {
new RSTile(2330, 3688), new RSTile(2337, 3807),
new RSTile(2416, 3801), new RSTile(2100, 3919),
new RSTile(2620, 3895), new RSTile(2725, 3492),
new RSTile(2842, 3539), new RSTile(3093, 3494),
new RSTile(3163, 3486), new RSTile(3254, 3420),
new RSTile(3187, 3437), new RSTile(3511, 3478),
new RSTile(3689, 3467), new RSTile(3495, 3212),
new RSTile(3382, 3270), new RSTile(3269, 3167),
new RSTile(3305, 3123), new RSTile(3427, 2892),
new RSTile(3091, 3243), new RSTile(3013, 3355),
new RSTile(2946, 3367), new RSTile(2809, 3441),
new RSTile(2617, 3334), new RSTile(2655, 3283),
new RSTile(2351, 3167), new RSTile(2443, 3085),
new RSTile(2611, 3093), new RSTile(2403, 2841),
new RSTile(2554, 2839), new RSTile(2851, 2955),
new RSTile(3680, 2982), new RSTile(3448, 3720),
new RSTile(2584, 3421), new RSTile(3209, 3219, 2)
};
for (RSTile bank : BANKS) {
double cdist = methods.calc.distanceBetween(tile, bank);
if ((dist > cdist || dist == -1.0D) && (tile.getZ() == bank.getZ())) {
dist = cdist;
finalTile = bank;
}
}
return finalTile;
}
/**
* Gets the closest supported bank that is usable.
*
* @return The closest bank's tile.
*/
public RSTile getNearestBank() {
return getNearestBank(methods.players.getMyPlayer().getLocation());
}
/**
* Generates a path between two nodes.
*
* @param start The starting tile.
* @param end The ending tile.
* @return The path.
*/
public RSTile[] generateTilePath(final RSTile start, final RSTile end) {
if (start.getZ() != end.getZ()) {
return null;
}
if (start.equals(end)) {
return new RSTile[]{};
}
if (!areScriptsLoaded()) {
loadWebScripts();
}
lastLocalAccess = System.currentTimeMillis();
final HashSet<Node> open = new HashSet<Node>();
final HashSet<Node> closed = new HashSet<Node>();
Node curr = new Node(start.getX(), start.getY(), start.getZ());
final Node dest = new Node(end.getX(), end.getY(), end.getZ());
curr.f = Heuristic(curr, dest);
open.add(curr);
while (!open.isEmpty()) {
curr = Lowest_f(open);
if (curr.equals(dest)) {
return Path(curr);
}
open.remove(curr);
closed.add(curr);
for (final Node next : Web.Successors(curr)) {
if (!closed.contains(next)) {
final double t = curr.g + Dist(curr, next);
boolean use_t = false;
if (!open.contains(next)) {
open.add(next);
use_t = true;
} else if (t < next.g) {
use_t = true;
}
if (use_t) {
next.prev = curr;
next.g = t;
next.f = t + Heuristic(next, dest);
}
}
}
}
return null;
}
/**
* Generates routes between two tiles.
*
* @param start The start tile.
* @param end The ending tile.
* @param lastRoute The last plane route.
* @return The generated route.
*/
private Route[] generateRoutes(final RSTile start, final RSTile end, final Route lastRoute) {
if (start.getZ() == end.getZ()) {
Route route = planeRoute(start, end, null);
if (route == null) {
return null;
} else {
route.parent = lastRoute;
}
if (route.parent == null) {
return new Route[]{route};
}
LinkedList<Route> finalRouting = new LinkedList<Route>();
while (route.parent != null) {
finalRouting.addLast(route);
route = route.parent;
}
return finalRouting.toArray(new Route[finalRouting.size()]);
}
PlaneHandler planeHandler = new PlaneHandler(methods);
PlaneTraverse[] traverses = planeHandler.get(methods.game.getPlane());
double dist = Double.MAX_VALUE;
PlaneTraverse finalTraverse = null;
for (PlaneTraverse traverse : traverses) {
if (traverse.destPlane() == end.getZ() && traverse.applicable()) {
final double tempDist = traverse.getRoute().getDistance();
if (tempDist < dist) {
dist = tempDist;
finalTraverse = traverse;
}
}
}
if (finalTraverse != null) {
final Route route = planeRoute(start, end, finalTraverse);
route.parent = lastRoute;
return generateRoutes(finalTraverse.dest(), end, route);
}
return null;//No applicable plane transfers.
}
public Route planeRoute(final RSTile start, final RSTile end, final PlaneTraverse transfer) {
if (!areScriptsLoaded()) {
loadWebScripts();
}
lastLocalAccess = System.currentTimeMillis();
if (transfer != null) {
final Route walkRoute = planeRoute(start, transfer.walkTo(), null);
if (walkRoute == null) {
return null;
}
walkRoute.add(new RouteStep(methods, transfer.walkTo()));
return walkRoute;
}
//TODO Special path generation (teleports, npcs, objects).
RSTile[] path = generateTilePath(start, end);
if (path == null) {
return null;
}
return new Route(new RouteStep[]{new RouteStep(methods, path)});
}
/**
* Returns a web instance to traverse.
*
* @param start The starting tile.
* @param end The end tile.
* @return The web constructed. <code>null</code> if it cannot be done.
*/
public RSWeb getWeb(RSTile start, final RSTile end) {
Route[] routes = generateRoutes(start, end, null);
if (routes == null) {
return null;
}
return new RSWeb(routes, start, end);
}
/**
* Returns a web instance to traverse.
*
* @param end The end tile.
* @return The web constructed. <code>null</code> if it cannot be done.
*/
public RSWeb getWeb(final RSTile end) {
return getWeb(methods.players.getMyPlayer().getLocation(), end);
}
/**
* Node class.
*/
private static class Node {
public final int x, y, z;
public Node prev;
public double g, f;
public Node(final int x, final int y, final int z) {
this.x = x;
this.y = y;
this.z = z;
g = f = 0;
}
@Override
public int hashCode() {
return x << 4 | y;
}
@Override
public boolean equals(final Object o) {
if (o instanceof Node) {
final Node n = (Node) o;
return x == n.x && y == n.y && z == n.z;
}
return false;
}
@Override
public String toString() {
return "(" + x + "," + y + "," + z + ")";
}
public RSTile toRSTile() {
return new RSTile(x, y, z);
}
}
/**
* Gets the heuristic distance.
*
* @param start Start node.
* @param end End node.
* @return The distance.
*/
private static double Heuristic(final Node start, final Node end) {
/*double dx = start.x - end.x;
double dy = start.y - end.y;
if (dx < 0) {
dx = -dx;
}
if (dy < 0) {
dy = -dy;
}
return dx < dy ? dy : dx;*/
return Dist(start, end) * Math.sqrt(Math.pow(start.x - end.x, 2) + Math.pow(start.y - end.y, 2));
}
/**
* The distance between two tiles.
*
* @param start The start tile.
* @param end The end tile.
* @return The distance.
*/
private static double Dist(final Node start, final Node end) {
if (start.x != end.x && start.y != end.y) {
return 1.41421356;
} else {
return 1.0;
}
}
/**
* Gets the lowest f score of a set.
*
* @param open The set.
* @return The node that has the lowest f score.
*/
private static Node Lowest_f(final Set<Node> open) {
Node best = null;
for (final Node t : open) {
if (best == null || t.f < best.f) {
best = t;
}
}
return best;
}
/**
* Constructs a path from a node.
*
* @param end The end node.
* @return The constructed path.
*/
private static RSTile[] Path(final Node end) {
final LinkedList<RSTile> path = new LinkedList<RSTile>();
Node p = end;
while (p != null) {
path.addFirst(p.toRSTile());
p = p.prev;
}
return path.toArray(new RSTile[path.size()]);
}
/**
* Gets successors of a tile.
*
* @param t The node.
* @return The nodes.
*/
private static List<Node> Successors(final Node t) {
final LinkedList<Node> tiles = new LinkedList<Node>();
final int x = t.x, y = t.y;
final RSTile here = t.toRSTile();
if (!Flag(here, RSTile.Flag.W_S) &&
!Flag(here.getX(), here.getY() - 1, here.getZ(), RSTile.Flag.BLOCKED | RSTile.Flag.WATER)) {
tiles.add(new Node(x, y - 1, here.getZ()));
}
if (!Flag(here, RSTile.Flag.W_W) &&
!Flag(here.getX() - 1, here.getY(), here.getZ(), RSTile.Flag.BLOCKED | RSTile.Flag.WATER)) {
tiles.add(new Node(x - 1, y, here.getZ()));
}
if (!Flag(here, RSTile.Flag.W_N) &&
!Flag(here.getX(), here.getY() + 1, here.getZ(), RSTile.Flag.BLOCKED | RSTile.Flag.WATER)) {
tiles.add(new Node(x, y + 1, here.getZ()));
}
if (!Flag(here, RSTile.Flag.W_E) &&
!Flag(here.getX() + 1, here.getY(), here.getZ(), RSTile.Flag.BLOCKED | RSTile.Flag.WATER)) {
tiles.add(new Node(x + 1, y, here.getZ()));
}
if (!Flag(here, RSTile.Flag.W_SW | RSTile.Flag.W_S | RSTile.Flag.W_W) &&
!Flag(here.getX() - 1, here.getY() - 1, here.getZ(), RSTile.Flag.BLOCKED | RSTile.Flag.WATER) &&
!Flag(here.getX(), here.getY() - 1, here.getZ(), RSTile.Flag.BLOCKED | RSTile.Flag.WATER | RSTile.Flag.W_W) &&
!Flag(here.getX() - 1, here.getY(), here.getZ(), RSTile.Flag.BLOCKED | RSTile.Flag.WATER | RSTile.Flag.W_S)) {
tiles.add(new Node(x - 1, y - 1, here.getZ()));
}
if (!Flag(here, RSTile.Flag.W_NW | RSTile.Flag.W_N | RSTile.Flag.W_W) &&
!Flag(here.getX() - 1, here.getY() + 1, here.getZ(), RSTile.Flag.BLOCKED | RSTile.Flag.WATER) &&
!Flag(here.getX(), here.getY() + 1, here.getZ(), RSTile.Flag.BLOCKED | RSTile.Flag.WATER | RSTile.Flag.W_W) &&
!Flag(here.getX() - 1, here.getY(), here.getZ(), RSTile.Flag.BLOCKED | RSTile.Flag.WATER | RSTile.Flag.W_N)) {
tiles.add(new Node(x - 1, y + 1, here.getZ()));
}
if (!Flag(here, RSTile.Flag.W_SE | RSTile.Flag.W_S | RSTile.Flag.W_E) &&
!Flag(here.getX() + 1, here.getY() - 1, here.getZ(), RSTile.Flag.BLOCKED | RSTile.Flag.WATER) &&
!Flag(here.getX(), here.getY() - 1, here.getZ(), RSTile.Flag.BLOCKED | RSTile.Flag.WATER | RSTile.Flag.W_E) &&
!Flag(here.getX() + 1, here.getY(), here.getZ(), RSTile.Flag.BLOCKED | RSTile.Flag.WATER | RSTile.Flag.W_S)) {
tiles.add(new Node(x + 1, y - 1, here.getZ()));
}
if (!Flag(here, RSTile.Flag.W_NE | RSTile.Flag.W_N | RSTile.Flag.W_E) &&
!Flag(here.getX() + 1, here.getY() + 1, here.getZ(), RSTile.Flag.BLOCKED | RSTile.Flag.WATER) &&
!Flag(here.getX(), here.getY() + 1, here.getZ(), RSTile.Flag.BLOCKED | RSTile.Flag.WATER | RSTile.Flag.W_E) &&
!Flag(here.getX() + 1, here.getY(), here.getZ(), RSTile.Flag.BLOCKED | RSTile.Flag.WATER | RSTile.Flag.W_N)) {
tiles.add(new Node(x + 1, y + 1, here.getZ()));
}
return tiles;
}
/**
* Gets the TileFlags of a tile.
*
* @param tile The tile.
* @return The <code>TileFlags</code>.
*/
public static int GetTileFlag(final RSTile tile) {
return BypassFlags.getKey(tile);
}
/**
* Checks the flags of a tile.
*
* @param tile The tile to check.
* @param key Keys to look for.
* @return <tt>true</tt> if the tile contains flags.
*/
public static boolean Flag(final RSTile tile, final int key) {
int theFlag;
return (theFlag = BypassFlags.getKey(tile)) != -1 && (theFlag & key) != 0;
}
/**
* Checks the flags of a tile.
*
* @param x The tile to check (x).
* @param y The tile to check (y).
* @param z The tile to check (z).
* @param key Keys to look for.
* @return <tt>true</tt> if the tile contains flags.
*/
public static boolean Flag(final int x, final int y, final int z, final int key) {
return Flag(new RSTile(x, y, z), key);
}
public boolean areScriptsLoaded() {
return webScriptsLoaded;
}
public boolean areScriptsInActive() {
return webScriptsLoaded && System.currentTimeMillis() - lastLocalAccess > (1000 * 60 * 5);
}
public void loadWebScripts() {
if (!webScriptsLoaded) {
webData.running = true;
final Thread t = new Thread(ScriptHandler.THREAD_GROUP, webData, ScriptHandler.THREAD_GROUP_NAME + "-Web Data handler");
t.setPriority(Thread.MIN_PRIORITY);
t.setDaemon(true);
t.start();
webScriptsLoaded = true;
}
}
public void unloadWebScripts() {
if (webScriptsLoaded) {
webData.running = false;
webScriptsLoaded = false;
}
}
}