|
// ==UserScript==
|
|
// @name KesselRun
|
|
// @author John Lehmann
|
|
// @copyright John Lehmann, 2024
|
|
// @license Apache 2.0
|
|
// @downloadURL https://www.helianthi.com/listings/kesselrun.user.js
|
|
// @description A script for Nu Planets to automate your supply lines
|
|
// @namespace kesselrun/planets.nu
|
|
// @include https://planets.nu/*
|
|
// @include https://*.planets.nu/*
|
|
// @include http://planets.nu/*
|
|
// @include http://*.planets.nu/*
|
|
// @require https://chmeee.org/ext/planets.nu/McNimblesToolkit-1.2.user.js
|
|
// @require https://chmeee.org/ext/planets.nu/FleetManagement.user.js
|
|
// @version 1.2
|
|
// @grant none
|
|
// ==/UserScript==
|
|
|
|
if (!GM_info) GM_info = GM.info;
|
|
|
|
var name = GM_info.script.name;
|
|
var version = GM_info.script.version;
|
|
|
|
var mcntk = vgap.plugins["McNimble's Toolkit"];
|
|
var fleet = vgap.plugins["Fleet Management"];
|
|
|
|
var kesselrun = new mcntk.McNimblesToolkit(name, version);
|
|
|
|
vgap.registerPlugin(kesselrun, name);
|
|
console.log(name + " v"+version+" planets.nu plugin registered");
|
|
|
|
kesselrun.EPSILON = 1e-6;
|
|
kesselrun.MAX_MEGACREDITS = 10000;
|
|
kesselrun.MIN_FUEL = 50;
|
|
kesselrun.READY_COLOR = "fca311";
|
|
kesselrun.NOT_READY_COLOR = "81f7e5";
|
|
|
|
kesselrun.pri_colors = ["#C75226", "#2639C7", "#268EC7", "#C7203C", "#8F65A3"];
|
|
kesselrun.sec_colors = ["#CA8569", "#3C4272", "#3C5F72", "#BD5264", "#8F65A3"];
|
|
|
|
kesselrun.RESOURCES = [
|
|
"megacredits",
|
|
"neutronium",
|
|
"supplies",
|
|
"clans",
|
|
"duranium",
|
|
"tritanium",
|
|
"molybdenum",
|
|
];
|
|
|
|
kesselrun.CARGO_RESOURCES = kesselrun.RESOURCES.slice(-5);
|
|
|
|
kesselrun.MINERALS = [
|
|
"neutronium",
|
|
"duranium",
|
|
"tritanium",
|
|
"molybdenum",
|
|
];
|
|
|
|
kesselrun.METALS = [
|
|
"duranium",
|
|
"tritanium",
|
|
"molybdenum",
|
|
];
|
|
|
|
kesselrun.RouteNameEndings = [
|
|
"Run", "Route", "Trail", "Trek", "Leap", "Voyage", "Passage", "Jump",
|
|
"Expedition", "Corridor", "Pathway", "Drift", "Circuit", "Loop",
|
|
"Flight", "Track", "Quest", "Chase", "Cruise", "Odyssey", "Expanse",
|
|
"Journey", "Traverse", "Line", "Detour", "Shortcut", "Way", "March", "Charge",
|
|
];
|
|
|
|
kesselrun.UnitaryRouteNameEndings = ["Stand", "Stop", "Terminus", "Pond",
|
|
"Pasture", "Paddock", "End", "Retreat", "Farm", "Ranch"];
|
|
|
|
class TradeRoute {
|
|
constructor(id, route, targets, assignments) {
|
|
this.id = id; // Unique ID for the trade route (passed as a parameter)
|
|
this.route = route; // List of planets in the trade route
|
|
this.targets = targets; // Map of planet to target resource levels
|
|
this.assignments = assignments; // Map of ship id : current route step
|
|
|
|
// Generate a name based on the first and last planet in the route
|
|
const startPlanet = vgap.getPlanet(route[0]).name; // First planet in the route
|
|
const endPlanet = vgap.getPlanet(route[route.length - 1]).name; // Last planet in the route
|
|
|
|
// Choose a random ending from the list of options
|
|
const ending = kesselrun.RouteNameEndings[Math.floor(Math.random() * kesselrun.RouteNameEndings.length)];
|
|
const unitaryEnding = kesselrun.UnitaryRouteNameEndings[Math.floor(Math.random() * kesselrun.UnitaryRouteNameEndings.length)];
|
|
|
|
// Auto-generate the route name
|
|
this.name = route.length > 1 ? `${startPlanet} to ${endPlanet} ${ending}` : `${startPlanet} ${unitaryEnding}`;
|
|
}
|
|
};
|
|
|
|
class MinPriorityQueue {
|
|
constructor() {
|
|
this.items = [];
|
|
}
|
|
|
|
enqueue(element, priority) {
|
|
this.items.push({ element, priority });
|
|
this.items.sort((a, b) => a.priority - b.priority); // Sort by priority
|
|
}
|
|
|
|
dequeue() {
|
|
return this.items.shift(); // Remove and return the element with the smallest priority
|
|
}
|
|
|
|
isEmpty() {
|
|
return this.items.length === 0;
|
|
}
|
|
};
|
|
|
|
/**
|
|
* Return zero bounded minimum.
|
|
*/
|
|
kesselrun.zmin = function (...args) {
|
|
return Math.max(0, Math.min(...args));
|
|
};
|
|
|
|
/**
|
|
* An object with supply and demand tables by planet, for a trade route.
|
|
* {"supply": {planetId: {rsrc: qty}}, "demand": {planetId: {rsrc: qty}}}
|
|
*/
|
|
class Manifest {
|
|
constructor(supply, demand) {
|
|
this.supply = supply;
|
|
this.demand = demand;
|
|
}
|
|
};
|
|
|
|
/**
|
|
* Builds and returns a Manifest of supply and demand for a trade route.
|
|
*
|
|
* @param {Object} tradeRoute - The trade route object containing planet targets.
|
|
* @returns {Manifest} - A Manifest object with supply and demand tables by planet.
|
|
*
|
|
* For single-stop (alchemy) routes, it directly maps available resources to supply and demand.
|
|
* For multi-stop routes, it calculates supply and demand based on the difference
|
|
* between available resources and targets.
|
|
*/
|
|
kesselrun.buildManifest = function (tradeRoute) {
|
|
const onestop = tradeRoute.route.length == 1;
|
|
let supply = {};
|
|
let demand = {};
|
|
|
|
for (let planetId in tradeRoute.targets) {
|
|
let planet = vgap.getPlanet(planetId);
|
|
if (!planet) continue;
|
|
|
|
supply[planetId] = {};
|
|
demand[planetId] = {};
|
|
for (let rsrc in tradeRoute.targets[planetId]) {
|
|
let avail = planet[rsrc];
|
|
let tgt = tradeRoute.targets[planetId][rsrc];
|
|
if (onestop) {
|
|
// tweak to cause Merlin & Alchemy to load supplies on a single planet route
|
|
supply[planetId][rsrc] = avail;
|
|
demand[planetId][rsrc] = tgt;
|
|
} else {
|
|
supply[planetId][rsrc] = Math.max(0, avail - tgt);
|
|
demand[planetId][rsrc] = Math.max(0, tgt - avail);
|
|
}
|
|
}
|
|
}
|
|
return new Manifest(supply, demand);
|
|
};
|
|
|
|
/**
|
|
* Build a matrix of the transwarp navigable distances between planets in the trade route.
|
|
*/
|
|
kesselrun.buildDistanceMatrix = function(tradeRoute) {
|
|
const matrix = {};
|
|
|
|
// Step 1: Calculate distances between all pairs, excluding pairs with distance > 83
|
|
for (let i = 0; i < tradeRoute.route.length; i++) {
|
|
const a = vgap.getPlanet(tradeRoute.route[i]);
|
|
if (!matrix[a.id]) matrix[a.id] = {};
|
|
|
|
for (let j = i + 1; j < tradeRoute.route.length; j++) {
|
|
const b = vgap.getPlanet(tradeRoute.route[j]);
|
|
const dist = kesselrun.calcDist(a, b);
|
|
if (dist > 83) continue; // Skip overlong distances
|
|
|
|
matrix[a.id][b.id] = dist;
|
|
if (!matrix[b.id]) matrix[b.id] = {};
|
|
matrix[b.id][a.id] = dist; // Symmetry
|
|
}
|
|
}
|
|
|
|
// Step 2: Add/overwrite distances for adjacent pairs (wrap around using modulo)
|
|
for (let i = 0; i < tradeRoute.route.length; i++) {
|
|
const a = vgap.getPlanet(tradeRoute.route[i]);
|
|
const next = (i + 1) % tradeRoute.route.length;
|
|
const b = vgap.getPlanet(tradeRoute.route[next]);
|
|
const dist = kesselrun.calcDist(a, b);
|
|
|
|
matrix[a.id][b.id] = dist;
|
|
matrix[b.id][a.id] = dist; // Symmetry
|
|
}
|
|
|
|
return matrix;
|
|
};
|
|
|
|
kesselrun.isFullyLoaded = function (ship) {
|
|
const hull = vgap.getHull(ship.hullid);
|
|
const loading = vgap.getTotalCargo(ship) / hull.cargo;
|
|
return loading > 0.95;
|
|
};
|
|
|
|
kesselrun.handleUnload = function (ship, planet, demand, supply) {
|
|
if (ship.hullid == 104 && vgap.gameUsesFuel()) {
|
|
// 104 - refinery - unload all but 1 fuel
|
|
let val = ship.neutronium - 1;
|
|
ship.neutronium -= val;
|
|
planet.neutronium += val;
|
|
} else if (ship.hullid == 105) {
|
|
// 105 - merlin - unload all minerals
|
|
for (let rsrc of kesselrun.METALS) {
|
|
planet[rsrc] += ship[rsrc];
|
|
ship[rsrc] = 0;
|
|
}
|
|
} else {
|
|
for (let rsrc in supply) {
|
|
if (ship[rsrc] === 0) continue;
|
|
let avail = ship[rsrc];
|
|
if (rsrc === "neutronium") {
|
|
if (!vgap.gameUsesFuel()) {
|
|
continue
|
|
}
|
|
let hullId = ship.hullid;
|
|
avail = hullId === 14 ? ship.neutronium : Math.max(ship.neutronium - kesselrun.MIN_FUEL, 0);
|
|
}
|
|
let val = Math.min(avail, demand[rsrc]);
|
|
ship[rsrc] -= val;
|
|
planet[rsrc] += val;
|
|
}
|
|
}
|
|
return true;
|
|
};
|
|
|
|
kesselrun.handleLoadMegacredits = function (ship, planet, supply, mcSpace, totals) {
|
|
if (!supply.megacredits) {
|
|
return true;
|
|
}
|
|
let mc = kesselrun.zmin(supply.megacredits, mcSpace, totals.megacredits);
|
|
ship.megacredits += mc;
|
|
planet.megacredits -= mc;
|
|
return true;
|
|
};
|
|
|
|
kesselrun.handleLoadNeutronium = function (ship, planet, supply, fuelSpace, totals) {
|
|
if (!vgap.gameUsesFuel() || !supply.neutronium) {
|
|
return true;
|
|
}
|
|
let fuel = kesselrun.zmin(supply.neutronium, fuelSpace, totals.neutronium);
|
|
ship.neutronium += fuel;
|
|
planet.neutronium -= fuel;
|
|
return true;
|
|
};
|
|
|
|
/**
|
|
* Loads cargo onto the ship based on available supply, total demand, and ship capacity.
|
|
*
|
|
* The function filters available resources, calculates how much of each to load based on
|
|
* demand and cargo space, and updates the ship, planet, and supply objects accordingly.
|
|
*/
|
|
kesselrun.handleLoadCargo = function (ship, planet, supply, totals, hullCargo) {
|
|
let resources = kesselrun.CARGO_RESOURCES.filter(r => ((supply[r] || 0) > 0) && ((totals[r] || 0) > 0))
|
|
let totalDemand = resources.reduce((sum, r) => sum + (totals[r] || 0), 0);
|
|
let weights = {};
|
|
let lots = {};
|
|
|
|
resources.forEach(r => {
|
|
weights[r] = (totals[r] || 0) / (totalDemand + kesselrun.EPSILON);
|
|
lots[r] = Math.trunc(weights[r] * 10);
|
|
});
|
|
|
|
let overflow = 0, last = 0;
|
|
let cargoFree = hullCargo - vgap.getTotalCargo(ship);
|
|
while (overflow < 300) {
|
|
overflow++;
|
|
if (cargoFree === 0 || cargoFree === last) break;
|
|
last = cargoFree;
|
|
|
|
resources.forEach(r => {
|
|
let val = kesselrun.zmin(lots[r], supply[r] || 0, totals[r], cargoFree);
|
|
if (val === 0) return;
|
|
ship[r] += val;
|
|
planet[r] -= val;
|
|
supply[r] -= val;
|
|
totals[r] -= val;
|
|
cargoFree = hullCargo - vgap.getTotalCargo(ship);
|
|
});
|
|
}
|
|
return true;
|
|
};
|
|
|
|
kesselrun.handleCargoForShip = function (ship, planet, manifest) {
|
|
const planetId = planet.id;
|
|
ship.changed = 1;
|
|
|
|
const hull = vgap.getHull(ship.hullid);
|
|
if (!hull) {
|
|
return false;
|
|
}
|
|
|
|
const demand = manifest.demand[planetId];
|
|
const supply = manifest.supply[planetId];
|
|
|
|
// Calculate total demand for each resource across all planets
|
|
const totals = kesselrun.RESOURCES.reduce((acc, r) => {
|
|
acc[r] = Object.keys(manifest.demand).reduce(
|
|
(sum, p) => sum + (manifest.demand[p][r] || 0),
|
|
0
|
|
);
|
|
return acc;
|
|
}, {});
|
|
|
|
if (!kesselrun.handleUnload(ship, planet, demand, supply)) {
|
|
return false;
|
|
}
|
|
|
|
const fuelSpace = hull.fueltank - ship.neutronium;
|
|
const mcSpace = kesselrun.MAX_MEGACREDITS - ship.megacredits;
|
|
const hullCargo = hull.cargo;
|
|
|
|
return kesselrun.handleLoadMegacredits(ship, planet, supply, mcSpace, totals) &&
|
|
kesselrun.handleLoadNeutronium(ship, planet, supply, fuelSpace, totals) &&
|
|
kesselrun.handleLoadCargo(ship, planet, supply, totals, hullCargo);
|
|
};
|
|
|
|
kesselrun.calcDist = (lhs, rhs) => {
|
|
return Math.dist(lhs.x, lhs.y, rhs.x, rhs.y);
|
|
};
|
|
|
|
kesselrun.calcFuel = function (ship, overfill) {
|
|
let distance = Math.dist(ship.x, ship.y, ship.targetx, ship.targety);
|
|
if (overfill) {
|
|
distance = Math.max(distance, 81)
|
|
}
|
|
let mass = mcntk.totalMass(ship);
|
|
let engine = vgap.getEngine(ship.engineid);
|
|
return mcntk.fuelForFullDistance(engine, ship.warp, mcntk.isGravitonic(ship), distance, mass) + 1;
|
|
};
|
|
|
|
kesselrun.moveToAssigned = function (tradeRoute, ship) {
|
|
const shipId = ship.id;
|
|
const step = tradeRoute.assignments[shipId];
|
|
const assignedId = tradeRoute.route[step % tradeRoute.route.length];
|
|
const assigned = vgap.getPlanet(assignedId);
|
|
|
|
if (!assigned) {
|
|
return false;
|
|
}
|
|
|
|
ship.targetx = assigned.x;
|
|
ship.targety = assigned.y;
|
|
ship.warp = Math.min(mcntk.maxWarp(ship), ship.engineid);
|
|
return true;
|
|
};
|
|
|
|
/**
|
|
* Move to the next planet in the trade route, in the normal order.
|
|
*/
|
|
kesselrun.followTradeRoute = function (tradeRoute, ship) {
|
|
const shipId = ship.id;
|
|
let planet = vgap.planetAt(ship.x, ship.y);
|
|
let step = tradeRoute.assignments[shipId];
|
|
|
|
if (!planet || !tradeRoute.route.includes(planet.id)) {
|
|
// Not at planet? Go to assigned planet
|
|
return kesselrun.moveToAssigned(tradeRoute, ship);
|
|
} else {
|
|
let assignedId = tradeRoute.route[step % tradeRoute.route.length];
|
|
if (assignedId === planet.id) {
|
|
// At assigned planet? Update assigned planet, set a waypoint
|
|
tradeRoute.assignments[shipId] += 1;
|
|
return kesselrun.moveToAssigned(tradeRoute, ship);
|
|
} else {
|
|
// At unexpected planet on route? Update assignment to this planet, continue to next step
|
|
for (let i = 0; i < tradeRoute.route.length; i++) {
|
|
step += 1;
|
|
assignedId = tradeRoute.route[step % tradeRoute.route.length];
|
|
if (assignedId === planet.id) {
|
|
tradeRoute.assignments[shipId] = step + 1;
|
|
return kesselrun.moveToAssigned(tradeRoute, ship);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
return false;
|
|
};
|
|
|
|
/**
|
|
* Use Dijkstra search to find the shortest path from startPlanet to targetPlanet
|
|
* and return the next destination, given startPlanet as the starting point.
|
|
*/
|
|
kesselrun.search = function (graph, startPlanet, targetPlanet) {
|
|
let backtrack = {};
|
|
let distances = {};
|
|
|
|
// Initialize distances to infinity for all nodes
|
|
for (const node in graph) {
|
|
distances[node] = Infinity;
|
|
}
|
|
distances[startPlanet] = 0;
|
|
|
|
// Priority queue of unvisited nodes
|
|
let unvisited = new MinPriorityQueue();
|
|
unvisited.enqueue(startPlanet, 0);
|
|
|
|
while (!unvisited.isEmpty()) {
|
|
let { element: currentNode, priority: currentDistance } = unvisited.dequeue();
|
|
|
|
if (currentDistance > distances[currentNode]) {
|
|
continue;
|
|
}
|
|
|
|
// Explore neighbors
|
|
for (const [neighbor, weight] of Object.entries(graph[currentNode])) {
|
|
let distance = currentDistance + weight;
|
|
if (distance < distances[neighbor]) {
|
|
distances[neighbor] = distance;
|
|
unvisited.enqueue(neighbor, distance);
|
|
backtrack[neighbor] = currentNode;
|
|
}
|
|
}
|
|
}
|
|
|
|
// Reconstruct the path from targetPlanet back to startPlanet using the backtrack map
|
|
let nextPlanet = targetPlanet;
|
|
while (backtrack[nextPlanet] !== startPlanet) {
|
|
nextPlanet = backtrack[nextPlanet];
|
|
}
|
|
|
|
// return the next stop
|
|
return nextPlanet;
|
|
}
|
|
|
|
/**
|
|
* Move by the shortest path to the next planet where there is demand for what the
|
|
* ship is carrying.
|
|
*/
|
|
kesselrun.followDemand = function (tradeRoute, manifest, currentPlanet, ship) {
|
|
// find nearest planet with demand for the ship cargo
|
|
let targetPlanet = null;
|
|
let minDistance = Infinity;
|
|
for (const planetId in manifest.demand) {
|
|
if (planetId == currentPlanet.id) continue; // Skip the current planet
|
|
|
|
const planetDemand = manifest.demand[planetId];
|
|
let hasDemand = kesselrun.CARGO_RESOURCES.some(resource =>
|
|
planetDemand[resource] > 0 && ship[resource] > 0
|
|
);
|
|
|
|
if (!hasDemand) {
|
|
continue;
|
|
}
|
|
|
|
const planet = vgap.getPlanet(planetId);
|
|
const distance = kesselrun.calcDist(currentPlanet, planet);
|
|
if (distance < minDistance) {
|
|
minDistance = distance;
|
|
targetPlanet = planet;
|
|
}
|
|
}
|
|
if (!targetPlanet) {
|
|
return false;
|
|
}
|
|
|
|
// find the shortest path to the targetPlanet
|
|
let graph = kesselrun.buildDistanceMatrix(tradeRoute);
|
|
let nextPlanetId = kesselrun.search(graph, currentPlanet.id, targetPlanet.id);
|
|
|
|
console.log("followDemand: ship " + ship.id + " is fully loaded will move to " + targetPlanet.id + " via " + nextPlanetId)
|
|
|
|
// update the assignment to the next planet on the path and moveToAssigned
|
|
let step = tradeRoute.assignments[ship.id];
|
|
for (let i = 0; i < tradeRoute.route.length; i++) {
|
|
step = (step + 1) % tradeRoute.route.length;
|
|
if (tradeRoute.route[step] === nextPlanetId) {
|
|
tradeRoute.assignments[ship.id] = step;
|
|
return kesselrun.moveToAssigned(tradeRoute, ship);
|
|
}
|
|
}
|
|
|
|
return false;
|
|
};
|
|
|
|
kesselrun.refuel = function (ship) {
|
|
if (!vgap.gameUsesFuel()) {
|
|
return true;
|
|
}
|
|
// If the ship doesn't have enough fuel to get to the next destination, refuel.
|
|
// Return false if there's not enough fuel available.
|
|
const planet = vgap.planetAt(ship.x, ship.y);
|
|
if (!planet || ship.ownerid !== planet.ownerid) return false;
|
|
|
|
const cost = kesselrun.calcFuel(ship, false);
|
|
const overfillCost = kesselrun.calcFuel(ship, true);
|
|
|
|
const stored = ship.neutronium;
|
|
if (overfillCost <= stored) return true;
|
|
|
|
let balance = overfillCost - stored;
|
|
if (planet.neutronium < balance) {
|
|
balance = cost - stored;
|
|
if (planet.neutronium < balance) {
|
|
return false;
|
|
}
|
|
}
|
|
|
|
// Refuel from the planet
|
|
ship.neutronium += balance;
|
|
planet.neutronium -= balance;
|
|
return true;
|
|
};
|
|
|
|
kesselrun.updateIdle = function (ship, isReady) {
|
|
ship.readystatus = isReady ? 1 : 0;
|
|
return true;
|
|
};
|
|
|
|
kesselrun.handleRoute = function (tradeRoute) {
|
|
// Adjust the cargo for all ships according to supply and demand.
|
|
for (const shipId in tradeRoute.assignments) {
|
|
const ship = vgap.getShip(shipId);
|
|
if (!ship) continue;
|
|
|
|
const planet = vgap.planetAt(ship.x, ship.y);
|
|
if (!planet || !tradeRoute.route.includes(planet.id) || ship.ownerid !== planet.ownerid) {
|
|
continue;
|
|
}
|
|
|
|
const manifest = kesselrun.buildManifest(tradeRoute);
|
|
|
|
// single-stop route - no need to move
|
|
if (tradeRoute.route.length == 1) {
|
|
kesselrun.updateIdle(ship, kesselrun.handleCargoForShip(ship, planet, manifest));
|
|
continue;
|
|
}
|
|
|
|
// Load cargo, then if fully loaded, head to the next planet with demand for the ship's cargo.
|
|
// Otherwise, follow the next step on the trade route.
|
|
let ready = kesselrun.handleCargoForShip(ship, planet, manifest)
|
|
&& ((kesselrun.isFullyLoaded(ship) && kesselrun.followDemand(tradeRoute, manifest, planet, ship))
|
|
|| kesselrun.followTradeRoute(tradeRoute, ship))
|
|
&& kesselrun.refuel(ship);
|
|
|
|
// Update idle status based on readiness.
|
|
kesselrun.updateIdle(ship, ready);
|
|
}
|
|
};
|
|
|
|
|
|
kesselrun.updateShipNotes = function (ready) {
|
|
let routes = kesselrun.getTradeRoutes();
|
|
routes.forEach(route => {
|
|
Object.keys(route.assignments).forEach(shipId => {
|
|
let note = vgap.getShipNote(shipId);
|
|
let colors = kesselrun.pri_colors;
|
|
let color = !ready ? colors[route.id % colors.length].substring(1) : kesselrun.READY_COLOR;
|
|
|
|
note.body = "TR" + route.id;
|
|
note.color = color;
|
|
note.changed = 1;
|
|
});
|
|
});
|
|
};
|
|
|
|
kesselrun.drawTradeRoutes = function (tradeRoutes) {
|
|
const zmin = 1;
|
|
const zmax = 30;
|
|
const overlays = vgap.map.drawingtool.overlays;
|
|
const layer = {
|
|
"active": true,
|
|
"name": "Trade Routes",
|
|
"markups": []
|
|
};
|
|
let count = 0;
|
|
|
|
for (let tradeRoute of tradeRoutes) {
|
|
if (tradeRoute.route.length == 1) {
|
|
continue;
|
|
}
|
|
for (let offset of [0, 1]) {
|
|
let colors = offset == 0 ? kesselrun.pri_colors : kesselrun.sec_colors;
|
|
let color = colors[tradeRoute.id % colors.length];
|
|
let markup = {"type": "line", "points": [], "attr": {"stroke": color, "color": color, "zmin": zmin, "zmax": zmax}};
|
|
for (let step = 0; step <= tradeRoute.route.length; step++) {
|
|
let planet = vgap.getPlanet(tradeRoute.route[step % tradeRoute.route.length]);
|
|
markup.points.push({'x': planet.x + offset, 'y': planet.y + offset})
|
|
}
|
|
layer.markups.push(markup);
|
|
}
|
|
count++;
|
|
}
|
|
|
|
if (count == 0) {
|
|
return;
|
|
}
|
|
|
|
// save to map
|
|
if (overlays.length == 0) {
|
|
overlays.push(layer);
|
|
} else {
|
|
let layerIndex = overlays.findIndex(entry => entry.name === layer.name);
|
|
if (layerIndex !== -1) {
|
|
console.log("Updating existing");
|
|
overlays[layerIndex] = layer;
|
|
} else {
|
|
console.log("Adding new");
|
|
overlays.push(layer);
|
|
layerIndex = overlays.length - 1;
|
|
}
|
|
}
|
|
vgap.map.drawingtool.current = {"overlay": layer, "markup": null, "editindex": null, "addType": "line"}
|
|
vgap.map.draw();
|
|
|
|
// save as note
|
|
const note = vgap.getNote(0, -133919);
|
|
// FIXME fails on no notes
|
|
let body = [];
|
|
if (note['body']) {
|
|
body = JSON.parse(note['body']);
|
|
layerIndex = body.findIndex(entry => entry.name === layer.name);
|
|
if (layerIndex !== -1) {
|
|
body[layerIndex] = layer;
|
|
} else {
|
|
body.push(layer);
|
|
}
|
|
} else {
|
|
body.push(layer);
|
|
}
|
|
note['body'] = JSON.stringify(body);
|
|
note["changed"] = 1;
|
|
}
|
|
|
|
kesselrun.run = function () {
|
|
const tradeRoutes = kesselrun.getTradeRoutes()
|
|
tradeRoutes.forEach(route => kesselrun.handleRoute(route));
|
|
kesselrun.updateShipNotes(true)
|
|
vgap.map.draw();
|
|
};
|
|
|
|
kesselrun.processload = function() {
|
|
kesselrun.checkDestroyed();
|
|
kesselrun.updateShipNotes(false);
|
|
}
|
|
|
|
kesselrun.planetNear = function (cluster, x, y) {
|
|
return cluster.findNearestSphereObjects(x, y, 4)[0]
|
|
};
|
|
|
|
kesselrun.getTradeRoutes = function() {
|
|
// Check if the tradeRoutes setting is defined
|
|
if (typeof kesselrun.settings.tradeRoutes === "undefined") {
|
|
// Create a new persistent game setting if not already defined
|
|
kesselrun.settings.createGameSetting("tradeRoutes", []);
|
|
}
|
|
// Return the tradeRoutes array
|
|
return kesselrun.settings.tradeRoutes();
|
|
};
|
|
|
|
kesselrun.getTradeRoute = function(routeId) {
|
|
const tradeRoutes = kesselrun.getTradeRoutes();
|
|
return tradeRoutes.find(tr => tr.id === routeId);
|
|
}
|
|
|
|
kesselrun.saveTradeRoutes = function(tradeRoutes) {
|
|
kesselrun.drawTradeRoutes(tradeRoutes);
|
|
kesselrun.settings.setTradeRoutes(tradeRoutes);
|
|
};
|
|
|
|
kesselrun.createTradeRoute = function(route, targets, assignments) {
|
|
// Retrieve the current trade routes
|
|
const tradeRoutes = kesselrun.getTradeRoutes();
|
|
|
|
// Determine the highest existing ID and add 1
|
|
const maxId = tradeRoutes.length > 0 ? Math.max(...tradeRoutes.map(tr => tr.id)) : 0;
|
|
const newId = maxId + 1;
|
|
|
|
// Create a new TradeRoute instance
|
|
const newTradeRoute = new TradeRoute(newId, route, targets, assignments);
|
|
|
|
// Add the new trade route to the list
|
|
tradeRoutes.push(newTradeRoute);
|
|
|
|
// Save the updated trade routes back to the settings
|
|
kesselrun.saveTradeRoutes(tradeRoutes);
|
|
|
|
return newTradeRoute;
|
|
};
|
|
|
|
// Create a new trade route from the ship waypoints
|
|
kesselrun.buildTradeRoute = function (ship) {
|
|
// Get the origin planet based on the ship's current position
|
|
var cluster = new mcntk.EchoCluster(vgap.myplanets)
|
|
|
|
let planet = kesselrun.planetNear(cluster, ship.x, ship.y);
|
|
if (!planet) {
|
|
console.warn("No planet at the ship location");
|
|
return;
|
|
}
|
|
const origin = planet.id;
|
|
const route = [planet.id];
|
|
|
|
// Get the target planet based on the ship's destination coordinates
|
|
planet = kesselrun.planetNear(cluster, ship.targetx, ship.targety);
|
|
if (planet && (planet.id != origin)) {
|
|
route.push(planet.id);
|
|
} else if ((ship.hullid != 104) && (ship.hullid != 105)) {
|
|
console.warn("No planet at the first ship destination");
|
|
return;
|
|
}
|
|
|
|
// Process additional waypoints and add them to the route
|
|
ship.waypoints.forEach(wp => {
|
|
planet = kesselrun.planetNear(cluster, wp.x, wp.y);
|
|
if (!planet) {
|
|
console.warn("No planet at a ship waypoint, " + wp.x + ", " + wp.y);
|
|
return;
|
|
}
|
|
route.push(planet.id);
|
|
});
|
|
|
|
// Build the targets object, setting resource levels for each planet in the route
|
|
const targets = {};
|
|
route.forEach(planetId => {
|
|
targets[planetId] = {};
|
|
kesselrun.RESOURCES.forEach(rsrc => {
|
|
targets[planetId][rsrc] = (planetId === origin) ? 10000 : 0;
|
|
});
|
|
});
|
|
if ((route.length == 1) && ((ship.hullid == 104) || (ship.hullid == 105))) {
|
|
targets[origin]["supplies"] = 0;
|
|
}
|
|
|
|
// Create the initial assignments map for the ship
|
|
const assignments = {};
|
|
assignments[ship.id] = 0;
|
|
|
|
// Return a new TradeRoute object
|
|
return kesselrun.createTradeRoute(route, targets, assignments);
|
|
};
|
|
|
|
kesselrun.deleteTradeRoute = function(routeId) {
|
|
// Retrieve the current trade routes
|
|
const tradeRoutes = kesselrun.getTradeRoutes();
|
|
|
|
// Remove ship notes
|
|
const deletedRoute = kesselrun.getTradeRoute(routeId);
|
|
Object.keys(deletedRoute.assignments).forEach(shipId => {
|
|
const note = vgap.getShipNote(shipId);
|
|
note.body = "";
|
|
note.color = "";
|
|
note.changed = 1;
|
|
});
|
|
|
|
// Filter out the trade route with the specified ID
|
|
const updatedRoutes = tradeRoutes.filter(tr => tr.id !== routeId);
|
|
|
|
// Save the updated trade routes back to the settings
|
|
kesselrun.saveTradeRoutes(updatedRoutes);
|
|
|
|
return updatedRoutes;
|
|
};
|
|
|
|
kesselrun.updateTradeRouteName = function(routeId, newName) {
|
|
// Retrieve the current trade routes
|
|
const tradeRoutes = kesselrun.getTradeRoutes();
|
|
|
|
// Find the trade route to update
|
|
const routeToUpdate = tradeRoutes.find(tr => tr.id === routeId);
|
|
if (!routeToUpdate) {
|
|
console.error(`Trade Route with ID ${routeId} not found!`);
|
|
return false;
|
|
}
|
|
|
|
// Update the name of the trade route
|
|
routeToUpdate.name = newName;
|
|
|
|
// Save the updated trade routes back to the settings
|
|
kesselrun.saveTradeRoutes(tradeRoutes);
|
|
|
|
return true;
|
|
};
|
|
|
|
kesselrun.addPlanetToRoute = function(routeId, newPlanetId, insertionPoint = null) {
|
|
// Retrieve the current trade routes
|
|
const tradeRoutes = kesselrun.getTradeRoutes();
|
|
|
|
// Find the trade route to update
|
|
const routeToUpdate = tradeRoutes.find(tr => tr.id === routeId);
|
|
if (!routeToUpdate) {
|
|
console.error(`Trade Route with ID ${routeId} not found!`);
|
|
return false;
|
|
}
|
|
|
|
// Insert the new planet at the specified position or append if not provided
|
|
if (insertionPoint !== null && insertionPoint >= 0 && insertionPoint < routeToUpdate.route.length) {
|
|
routeToUpdate.route.splice(insertionPoint, 0, newPlanetId);
|
|
} else {
|
|
// Append the new planet to the end of the route
|
|
routeToUpdate.route.push(newPlanetId);
|
|
}
|
|
|
|
// A planet may appear in the route multiple times, but will only have one set of targets
|
|
if (!routeToUpdate.targets.hasOwnProperty(newPlanetId)) {
|
|
routeToUpdate.targets[newPlanetId] = {};
|
|
}
|
|
|
|
// Save the updated trade routes back to the settings
|
|
kesselrun.saveTradeRoutes(tradeRoutes);
|
|
|
|
return true;
|
|
};
|
|
|
|
kesselrun.modifyTradeRouteTargets = function(routeId, planetId, resourceLevels) {
|
|
// Retrieve the current trade routes
|
|
const tradeRoutes = kesselrun.getTradeRoutes();
|
|
|
|
// Find the trade route to update
|
|
const routeToUpdate = tradeRoutes.find(tr => tr.id === routeId);
|
|
if (!routeToUpdate) {
|
|
console.error(`Trade Route with ID ${routeId} not found!`);
|
|
return false;
|
|
}
|
|
|
|
// Check if the planet is part of the trade route
|
|
if (!routeToUpdate.route.includes(planetId)) {
|
|
console.error(`Planet with ID ${planetId} is not part of the trade route!`);
|
|
return false;
|
|
}
|
|
|
|
// Update the target resource levels for the specified planet
|
|
routeToUpdate.targets[planetId] = { ...resourceLevels };
|
|
|
|
// Save the updated trade routes back to the settings
|
|
kesselrun.saveTradeRoutes(tradeRoutes);
|
|
|
|
return true;
|
|
};
|
|
|
|
kesselrun.addShipToRoute = function(routeId, shipId) {
|
|
// Retrieve the current trade routes
|
|
const tradeRoutes = kesselrun.getTradeRoutes();
|
|
|
|
// Find the trade route to update
|
|
const routeToUpdate = tradeRoutes.find(tr => tr.id === routeId);
|
|
if (!routeToUpdate) {
|
|
console.error(`Trade Route with ID ${routeId} not found!`);
|
|
return false;
|
|
}
|
|
|
|
// Retrieve the ship object from vgap
|
|
const ship = vgap.getShip(shipId);
|
|
if (!ship) {
|
|
console.error(`Ship with ID ${shipId} not found!`);
|
|
return false;
|
|
}
|
|
|
|
// Remove the ship from any other trade routes before assigning it to the new route
|
|
tradeRoutes.forEach(route => {
|
|
if (route.assignments.hasOwnProperty(shipId)) {
|
|
delete route.assignments[shipId];
|
|
}
|
|
});
|
|
|
|
const note = vgap.getShipNote(shipId);
|
|
note.body = "TR" + routeId;
|
|
note.color = kesselrun.sec_colors[routeId % kesselrun.sec_colors.length].substring(1);
|
|
note.changed = 1;
|
|
|
|
// Find the nearest planet on the route to assign the ship
|
|
let nearestPlanetId = null;
|
|
let shortestDistance = Number.MAX_VALUE;
|
|
|
|
routeToUpdate.route.forEach(planetId => {
|
|
const planet = vgap.getPlanet(planetId);
|
|
if (!planet) return;
|
|
|
|
// Calculate the square distance between the ship and the planet
|
|
const dx = planet.x - ship.x;
|
|
const dy = planet.y - ship.y;
|
|
const distance = dx * dx + dy * dy;
|
|
|
|
// Update the nearest planet if this one is closer
|
|
if (distance < shortestDistance) {
|
|
shortestDistance = distance;
|
|
nearestPlanetId = planetId;
|
|
}
|
|
});
|
|
|
|
if (nearestPlanetId === null) {
|
|
console.error("No valid planets found on the route!");
|
|
return false;
|
|
}
|
|
|
|
// Assign the ship to the nearest planet's step in the route
|
|
const stepIndex = routeToUpdate.route.indexOf(nearestPlanetId);
|
|
routeToUpdate.assignments[shipId] = stepIndex;
|
|
|
|
// Save the updated trade routes back to the settings
|
|
kesselrun.saveTradeRoutes(tradeRoutes);
|
|
|
|
return true;
|
|
};
|
|
|
|
kesselrun.removeShip = function(routeId, shipId) {
|
|
// Retrieve the current trade routes
|
|
const tradeRoutes = kesselrun.getTradeRoutes();
|
|
|
|
// Find the trade route to update
|
|
const routeToUpdate = tradeRoutes.find(tr => tr.id === routeId);
|
|
if (!routeToUpdate) {
|
|
console.error(`Trade Route with ID ${routeId} not found!`);
|
|
return false;
|
|
}
|
|
|
|
// Check if the ship is assigned to this trade route
|
|
if (!(shipId in routeToUpdate.assignments)) {
|
|
console.warn(`Ship with ID ${shipId} is not assigned to Trade Route ${routeId}.`);
|
|
return false;
|
|
}
|
|
|
|
// Remove the ship from the assignments
|
|
delete routeToUpdate.assignments[shipId];
|
|
const note = vgap.getShipNote(shipId);
|
|
note.body = "";
|
|
note.color = "";
|
|
note.changed = 1;
|
|
|
|
// Save the updated trade routes back to the settings
|
|
kesselrun.saveTradeRoutes(tradeRoutes);
|
|
|
|
return true;
|
|
};
|
|
|
|
/** Move a planet up or down in the route */
|
|
kesselrun.movePlanet = function(routeId, index, direction) {
|
|
// Retrieve the current trade routes
|
|
const tradeRoutes = kesselrun.getTradeRoutes();
|
|
|
|
// Find the trade route to update
|
|
const tradeRoute = tradeRoutes.find(tr => tr.id === routeId);
|
|
if (!tradeRoute) {
|
|
console.error(`Trade Route with ID ${routeId} not found!`);
|
|
return false;
|
|
}
|
|
|
|
// Determine the new index based on the direction ("up" or "down")
|
|
let newIndex = index;
|
|
if (direction === "up" && index > 0) {
|
|
newIndex -= 1;
|
|
} else if (direction === "down" && index < tradeRoute.route.length - 1) {
|
|
newIndex += 1;
|
|
} else {
|
|
console.warn(`Cannot move planet ${direction}. It is already at the ${direction === 'up' ? 'top' : 'bottom'} of the route.`);
|
|
return false;
|
|
}
|
|
|
|
// Swap the elements at `index` and `newIndex`
|
|
[tradeRoute.route[index], tradeRoute.route[newIndex]] = [tradeRoute.route[newIndex], tradeRoute.route[index]];
|
|
|
|
// Save the updated trade routes back to the settings
|
|
kesselrun.saveTradeRoutes(tradeRoutes)
|
|
|
|
return true;
|
|
};
|
|
|
|
kesselrun.removePlanet = function(routeId, index) {
|
|
// Retrieve the current trade routes
|
|
const tradeRoutes = kesselrun.getTradeRoutes();
|
|
|
|
// Find the trade route to update
|
|
const tradeRoute = tradeRoutes.find(tr => tr.id === routeId);
|
|
if (!tradeRoute) {
|
|
console.error(`Trade Route with ID ${routeId} not found!`);
|
|
return false;
|
|
}
|
|
|
|
// Validate index
|
|
if (index < 0 || index >= tradeRoute.route.length) {
|
|
console.error(`Invalid index ${index} for the route!`);
|
|
return false;
|
|
}
|
|
|
|
// Get the planet ID at the specified index
|
|
const planetId = tradeRoute.route[index];
|
|
|
|
// Remove the planet from the `route` array using the index
|
|
tradeRoute.route.splice(index, 1);
|
|
|
|
// Check if the planet ID is still present in the `route` array
|
|
if (!tradeRoute.route.includes(planetId)) {
|
|
// Remove the planet from `targets` if it no longer exists in the route
|
|
delete tradeRoute.targets[planetId];
|
|
}
|
|
|
|
// Save the updated trade routes back to the settings
|
|
kesselrun.saveTradeRoutes(tradeRoutes);
|
|
|
|
return true;
|
|
};
|
|
|
|
/** Check routes for ships that no longer exist. */
|
|
kesselrun.checkDestroyed = function() {
|
|
let tradeRoutes = kesselrun.getTradeRoutes();
|
|
tradeRoutes.forEach(route => {
|
|
Object.keys(route.assignments).forEach(shipId => {
|
|
let ship = vgap.getShip(shipId);
|
|
if (!ship || ship.ownerid != vgap.player.id) {
|
|
console.log(`Removing ship ${shipId} from route ${route.id} as it no longer exists`);
|
|
delete route.assignments[shipId];
|
|
}
|
|
});
|
|
});
|
|
kesselrun.saveTradeRoutes(tradeRoutes);
|
|
};
|
|
|
|
// dialog functions
|
|
|
|
kesselrun.dialog = {}; // the dialog attributes
|
|
|
|
kesselrun.dialog.key = "KR: Dialog";
|
|
|
|
kesselrun.dialog.allColumns = [
|
|
"ID", "Name", "Pl", "Sh", "Dist",
|
|
"Tgt$", "TgtS", "TgtC", "TgtN", "TgtD", "TgtT", "TgtM",
|
|
"$", "S", "C", "N", "D", "T", "M", "*"
|
|
];
|
|
|
|
kesselrun.dialog.tgtColumns = ["Resource", "Supply", "Target", "Edit"];
|
|
|
|
// different modal functions
|
|
|
|
kesselrun.summary = function(object, resource) {
|
|
// object is a trade route
|
|
return kesselrun.availableTargets(object, resource) + " / " + kesselrun.inTransit(object, resource)
|
|
}
|
|
|
|
kesselrun.detail = function(object, resource) {
|
|
// object is a trade route
|
|
let a = kesselrun.alluvial(object, resource)
|
|
let b = ""
|
|
if (kesselrun.MINERALS.includes(resource)) {
|
|
b = "/" + kesselrun.inground(object, resource)
|
|
}
|
|
let c = " (+" + kesselrun.production(object, resource) + ")"
|
|
return a + b + c;
|
|
}
|
|
|
|
/**
|
|
* Show the amount available over target, bounded by demand. This is used to show
|
|
* the balance between what is waiting for pickup and due to arrive at destinations
|
|
* in the Trade Routes dialogue.
|
|
*/
|
|
kesselrun.availableTargets = function(tradeRoute, resource) {
|
|
let mf = kesselrun.buildManifest(tradeRoute);
|
|
let demand = Object.values(mf.demand).reduce((total, planet) => total + (planet[resource] || 0), 0);
|
|
let total = 0;
|
|
for (const planetId in tradeRoute.targets) {
|
|
const planet = vgap.getPlanet(planetId);
|
|
if (planet && typeof tradeRoute.targets[planetId][resource] !== 'undefined') {
|
|
let avail = planet[resource] - (tradeRoute.targets[planetId][resource] || 0);
|
|
total += Math.min(demand, Math.max(0, avail));
|
|
}
|
|
}
|
|
return total;
|
|
};
|
|
|
|
kesselrun.inTransit = function(object, resource) {
|
|
let total = 0;
|
|
|
|
// Loop through each ship assigned to the trade route
|
|
for (const assignmentId in object.assignments) {
|
|
// Sum up the resource amount carried by the assigned ships
|
|
total += vgap.getShip(assignmentId)[resource] || 0;
|
|
}
|
|
|
|
return total;
|
|
};
|
|
|
|
kesselrun.target = function(object, resource) {
|
|
let planetId = object.id
|
|
return (kesselrun.dialog.parent.targets[planetId] !== undefined && kesselrun.dialog.parent.targets[planetId][resource] !== undefined)
|
|
? kesselrun.dialog.parent.targets[planetId][resource]
|
|
: "n/a";
|
|
};
|
|
|
|
kesselrun.alluvial = function(object, resource) {
|
|
let planetId = object.id
|
|
return object[resource];
|
|
};
|
|
|
|
kesselrun.inground = function(object, resource) {
|
|
let planetId = object.id;
|
|
let value = "0";
|
|
|
|
// Check if the `resource` is in the `kesselrun.MINERALS` array
|
|
if (kesselrun.MINERALS.includes(resource)) {
|
|
value = object["ground" + resource];
|
|
}
|
|
|
|
return value;
|
|
};
|
|
|
|
kesselrun.production = function(object, resource) {
|
|
let planetId = object.id;
|
|
let value = "-";
|
|
|
|
if (resource === "megacredits") {
|
|
value = Math.min(5000, vgap.nativeTaxAmount(object) + vgap.colonistTaxAmount(object));
|
|
} else if (resource === "supplies") {
|
|
value = vgap.pl.totalSuppliesOutput(object)
|
|
} else if (kesselrun.MINERALS.includes(resource)) {
|
|
value = vgap.miningAmount(object, object["ground" + resource], object["density" + resource], object.mines)
|
|
}
|
|
|
|
return value;
|
|
};
|
|
|
|
kesselrun.modeDefinitions = {
|
|
"summary": {fun: kesselrun.summary,
|
|
buttons: ["Add Route"],
|
|
help: "Available for transport / In transit in ships"},
|
|
"planets": {fun: kesselrun.detail,
|
|
buttons: ["Ships", "Add Ship", "Add Planet", "\u23CE"],
|
|
help: "Showing Target and Available/Ground/(Production)"},
|
|
"addplanet": {fun: kesselrun.detail,
|
|
buttons: [],
|
|
help: "Select planet to add to trade route"},
|
|
"targets": {fun: kesselrun.detail,
|
|
buttons: ["\u23CE"],
|
|
help: "Set the amount of the resource you want delivered to the planet"},
|
|
"onestop": {fun: kesselrun.detail,
|
|
buttons: ["\u23CE"],
|
|
help: "Set the amount of each resource to be loaded into the orbiting ship"},
|
|
|
|
};
|
|
|
|
|
|
// dialog creation
|
|
kesselrun.showPlanetTargetsDialog = function (planet) {
|
|
const tradeRoute = kesselrun.dialog.parent;
|
|
const parent = [tradeRoute, planet]
|
|
const targets = kesselrun.RESOURCES;
|
|
const cols = kesselrun.dialog.tgtColumns
|
|
const mode = tradeRoute.route.length > 1 ? "targets" : "onestop";
|
|
kesselrun.createDialog(mode, [tradeRoute, planet], targets, cols, null,
|
|
"[P"+ planet.id + "] " + planet.name + " Targets");
|
|
};
|
|
|
|
|
|
kesselrun.showTradeRouteDetail = function (tradeRoute) {
|
|
const planets = tradeRoute.route.map(planetId => vgap.getPlanet(planetId));
|
|
let cols = kesselrun.dialog.allColumns.filter(column => column !== "Pl" && column !== "Sh");
|
|
kesselrun.createDialog("planets", tradeRoute, planets, cols,
|
|
kesselrun.showPlanetTargetsDialog,
|
|
"["+ tradeRoute.id + "] " + tradeRoute.name + " Planets");
|
|
};
|
|
|
|
kesselrun.showTradeRoutes = function () {
|
|
const tradeRoutes = kesselrun.getTradeRoutes();
|
|
let cols = kesselrun.dialog.allColumns.filter(column => !column.startsWith("Tgt") && column !== "*");
|
|
cols.push("\u2600");
|
|
kesselrun.createDialog("summary", null, tradeRoutes, cols,
|
|
kesselrun.showTradeRouteDetail, "Trade Routes");
|
|
};
|
|
|
|
kesselrun.dialog.popDialog = function () {
|
|
if (kesselrun.dialog.mode === "planets") {
|
|
kesselrun.showTradeRoutes();
|
|
} else if (kesselrun.dialog.mode === "targets" || kesselrun.dialog.mode === "onestop") {
|
|
const parent = kesselrun.dialog.parent;
|
|
const tradeRoute = parent[0];
|
|
kesselrun.showTradeRouteDetail(tradeRoute);
|
|
}
|
|
}
|
|
|
|
kesselrun.createDialog = function(mode, parent, targets, columns, fun, title) {
|
|
kesselrun.dialog.parent = parent;
|
|
kesselrun.dialog.targets = targets;
|
|
kesselrun.dialog.columns = columns;
|
|
kesselrun.dialog.mode = mode
|
|
kesselrun.dialog.modeFun = kesselrun.modeDefinitions[mode].fun;
|
|
kesselrun.dialog.modeHelp = kesselrun.modeDefinitions[mode].help;
|
|
kesselrun.dialog.buttons = kesselrun.modeDefinitions[mode].buttons;
|
|
kesselrun.dialog.selectedFun = fun;
|
|
|
|
var table = kesselrun.createTable(targets);
|
|
nu.keyedmodal(kesselrun.dialog.key, table, title, 1200, "body");
|
|
|
|
var modal = nu.keyedmodals[kesselrun.dialog.key];
|
|
modal.css('width', 'unset');
|
|
modal.center();
|
|
var top = modal.css('top').match(/(.+)px/)[1];
|
|
if (top < 0) modal.css('top', 0);
|
|
|
|
// works for both desktop and mobile ;)
|
|
$("#KRTable").parent().css({"background-image": "none", "background": "black"});
|
|
}
|
|
|
|
kesselrun.createTable = function(targets) {
|
|
var table = $("<table class='KRTradeRoutes' id='KRTable'></table>");
|
|
kesselrun.fillTable(table, targets);
|
|
return table;
|
|
}
|
|
|
|
kesselrun.updateTable = function() {
|
|
var table = $("#KRTable");
|
|
table.empty();
|
|
if (kesselrun.dialog.mode == "summary") {
|
|
kesselrun.dialog.targets = kesselrun.getTradeRoutes();
|
|
} else if (kesselrun.dialog.mode == "planets") {
|
|
const routeId = kesselrun.dialog.parent.id
|
|
const tradeRoute = kesselrun.getTradeRoutes().find(tr => tr.id === routeId);
|
|
const planets = tradeRoute.route.map(planetId => vgap.getPlanet(planetId));
|
|
kesselrun.dialog.parent = tradeRoute;
|
|
kesselrun.dialog.targets = planets;
|
|
}
|
|
kesselrun.fillTable(table, kesselrun.dialog.targets);
|
|
}
|
|
|
|
kesselrun.fillTable = function(table, targets) {
|
|
table.append(kesselrun.renderTargetHeader());
|
|
kesselrun.addRows(table, targets);
|
|
table.append(kesselrun.helpRow(table));
|
|
table.append(kesselrun.editRow(table));
|
|
}
|
|
|
|
kesselrun.renderTargetHeader = function() {
|
|
var row = $("<tr></tr>");
|
|
|
|
kesselrun.dialog.columns.forEach((header) => {
|
|
var col = kesselrun.columnDefinitions[header];
|
|
kesselrun.createHeader(row, header, col.fun);
|
|
});
|
|
return row;
|
|
}
|
|
|
|
kesselrun.createHeader = function(row, label, fun) {
|
|
const header = $("<th>"+label+"</th>");
|
|
const tt = kesselrun.columnDefinitions[label].tt
|
|
header.attr("title", tt);
|
|
header.tclick(function() {
|
|
kesselrun.towTargets.sort(function(lhs, rhs) {
|
|
var valA = fun(lhs),
|
|
valB = fun(rhs);
|
|
if (valA > valB) return 1;
|
|
if (valA < valB) return -1;
|
|
return 0;
|
|
});
|
|
kesselrun.updateTable();
|
|
});
|
|
row.append(header);
|
|
}
|
|
|
|
kesselrun.addRows = function(table, targets) {
|
|
targets.forEach((target, idx) => {
|
|
var row = kesselrun.renderTarget(target, idx);
|
|
// Create a fixed scope for the target
|
|
var fixedScopeTarget = target;
|
|
// Set up the click handler
|
|
row.tclick(function() { kesselrun.success(fixedScopeTarget); });
|
|
// Append the row to the table
|
|
table.append(row);
|
|
});
|
|
};
|
|
|
|
kesselrun.helpRow = function(table) {
|
|
let row = $("<tr/>");
|
|
row.css("color", "orange");
|
|
// spacer cell
|
|
row.append($("<td></td>"));
|
|
let help = $("<td id='KRTableHelp' colspan=9></td>");
|
|
help.text(kesselrun.dialog.modeHelp)
|
|
row.append(help);
|
|
table.append(row)
|
|
};
|
|
|
|
// Add a route from a ship
|
|
kesselrun.addRoute = function () {
|
|
const shipSelectedFunc = function (ignore, ship) {
|
|
kesselrun.buildTradeRoute(ship);
|
|
kesselrun.updateTable();
|
|
}
|
|
var ships = vgap.myships.filter(ship => !(ship.x === ship.targetx && ship.y === ship.targety) || ship.hullid == 104 || ship.hullid == 105);
|
|
|
|
fleet.createDialog(null, ships, fleet.allColumns, shipSelectedFunc, "Create Trade Route From Ship Waypoints")
|
|
}
|
|
|
|
// Ship assigned ships, click to unassign
|
|
kesselrun.showShipsDialog = function () {
|
|
const shipSelectedFunc = function (ignore, ship) {
|
|
const tradeRouteId = kesselrun.dialog.parent.id;
|
|
kesselrun.removeShip(tradeRouteId, ship.id);
|
|
kesselrun.updateTable();
|
|
};
|
|
const tradeRoutes = kesselrun.getTradeRoutes();
|
|
const tradeRouteId = kesselrun.dialog.parent.id;
|
|
const tradeRoute = tradeRoutes.find(tr => tr.id === tradeRouteId);
|
|
let ships = vgap.myships.filter(ship => tradeRoute.assignments.hasOwnProperty(ship.id));
|
|
fleet.createDialog(null, ships, fleet.allColumns, shipSelectedFunc, "Click to unassign ship");
|
|
}
|
|
|
|
// Select a ship to add to the trade route
|
|
kesselrun.addShipDialog = function () {
|
|
const shipSelectedFunc = function (ignore, ship) {
|
|
const tradeRouteId = kesselrun.dialog.parent.id;
|
|
kesselrun.addShipToRoute(tradeRouteId, ship.id);
|
|
kesselrun.updateTable();
|
|
};
|
|
const tradeRoutes = kesselrun.getTradeRoutes();
|
|
const tradeRouteId = kesselrun.dialog.parent.id;
|
|
const tradeRoute = tradeRoutes.find(tr => tr.id === tradeRouteId);
|
|
let ships = vgap.myships.filter(ship => !tradeRoute.assignments.hasOwnProperty(ship.id));
|
|
fleet.createDialog(null, ships, fleet.allColumns, shipSelectedFunc, "Click to add ship to trade route");
|
|
}
|
|
|
|
kesselrun.addPlanetDialog = function () {
|
|
const callback = function (planet) {
|
|
const tradeRoute = kesselrun.dialog.parent
|
|
kesselrun.addPlanetToRoute(tradeRoute.id, planet.id);
|
|
const updatedTradeRoute = kesselrun.getTradeRoutes().find(tr => tr.id == tradeRoute.id);
|
|
kesselrun.showTradeRouteDetail(updatedTradeRoute);
|
|
};
|
|
|
|
// Filter out unwanted columns
|
|
const cols = kesselrun.dialog.allColumns.filter(column =>
|
|
!column.startsWith("Tgt") && column !== "*" && column !== "Pl" && column !== "Sh"
|
|
);
|
|
// select from all planets
|
|
const planets = vgap.myplanets;
|
|
kesselrun.createDialog("addplanet", kesselrun.dialog.parent, planets, cols, callback,
|
|
"Click to add planet to trade route");
|
|
};
|
|
|
|
// add the bottom row edit buttons
|
|
kesselrun.editRow = function(table) {
|
|
const row = $("<tr align='right'/>");
|
|
row.css("color", "#81f7e5");
|
|
const cell = $("<td colspan='" + kesselrun.dialog.columns.length + "''></td>")
|
|
kesselrun.dialog.buttons.forEach(name => {
|
|
const tt = kesselrun.buttonDefinitions[name].tt;
|
|
const fun = kesselrun.buttonDefinitions[name].fun;
|
|
|
|
const btn = $("<div/>");
|
|
btn.text(name);
|
|
btn.addClass('smallbutton');
|
|
btn.attr('title', tt);
|
|
btn.on("click", fun);
|
|
|
|
cell.append(btn);
|
|
});
|
|
row.append(cell)
|
|
table.append(row)
|
|
};
|
|
|
|
kesselrun.success = function(target) {
|
|
if (kesselrun.dialog.selectedFun) {
|
|
nu.closekeyedmodal(kesselrun.dialog.key);
|
|
kesselrun.dialog.selectedFun(target);
|
|
}
|
|
};
|
|
|
|
kesselrun.renderTarget = function(object, objectIdx) {
|
|
var row = $("<tr/>");
|
|
row.css("color", kesselrun.colorFor(object));
|
|
|
|
// Use the index parameter in the forEach loop
|
|
kesselrun.dialog.columns.forEach(header => {
|
|
var col = kesselrun.columnDefinitions[header];
|
|
let td = $("<td/>");
|
|
td.append(col.fun(object, objectIdx));
|
|
row.append(td);
|
|
});
|
|
|
|
return row;
|
|
};
|
|
|
|
kesselrun.colorFor = function(object) {
|
|
return "green";
|
|
};
|
|
|
|
// column functions
|
|
|
|
kesselrun.dialog.id = function(object) {
|
|
return object.id;
|
|
};
|
|
|
|
kesselrun.escapeHtml = function(input) {
|
|
return input
|
|
.replace(/&/g, "&")
|
|
.replace(/</g, "<")
|
|
.replace(/>/g, ">")
|
|
.replace(/"/g, """)
|
|
.replace(/'/g, "'");
|
|
};
|
|
|
|
kesselrun.dialog.name = function(object) {
|
|
return kesselrun.escapeHtml(object.name);
|
|
};
|
|
|
|
kesselrun.dialog.planets = function(object) {
|
|
return object.route.length;
|
|
};
|
|
|
|
kesselrun.dialog.ships = function(object) {
|
|
return Object.keys(object.assignments).length;
|
|
};
|
|
|
|
kesselrun.dialog.dist = function(object) {
|
|
warpFun = function(d) {
|
|
// allow for warpwells
|
|
d = Math.max(0, d - 2);
|
|
let w = Math.ceil(Math.sqrt(d));
|
|
return (w<=9) ? w : "!";
|
|
}
|
|
|
|
if (object.route) {
|
|
// If the object is a trade route, calculate the total distance between all planets on the route
|
|
let total = 0;
|
|
let maxHop = 0;
|
|
for (let i = 0; i < object.route.length; i++) {
|
|
const planetA = vgap.getPlanet(object.route[i]);
|
|
const planetB = vgap.getPlanet(object.route[(i + 1) % object.route.length]);
|
|
const dist = kesselrun.calcDist(planetA, planetB)
|
|
maxHop = Math.max(maxHop, dist);
|
|
total += dist;
|
|
}
|
|
let warp = warpFun(maxHop)
|
|
return Math.ceil(total) + " (" + warp + ")";
|
|
} else {
|
|
// If the object is a planet, calculate the distance to the next planet on the route
|
|
const tradeRoute = kesselrun.dialog.parent;
|
|
const planetId = object.id; // The current planet's ID
|
|
const index = tradeRoute.route.indexOf(planetId);
|
|
const nextPlanetId = tradeRoute.route[(index + 1) % tradeRoute.route.length]
|
|
const planetB = vgap.getPlanet(nextPlanetId)
|
|
const dist = kesselrun.calcDist(object, planetB);
|
|
const warp = warpFun(dist);
|
|
return Math.ceil(dist) + " (" + warp + ")";
|
|
}
|
|
};
|
|
|
|
kesselrun.dialog.tgtMegacredits = function(object) {
|
|
return kesselrun.target(object, "megacredits");
|
|
};
|
|
|
|
kesselrun.dialog.tgtNeutronium = function(object) {
|
|
return kesselrun.target(object, "neutronium");
|
|
};
|
|
|
|
kesselrun.dialog.tgtSupplies = function(object) {
|
|
return kesselrun.target(object, "supplies");
|
|
};
|
|
|
|
kesselrun.dialog.tgtClans = function(object) {
|
|
return kesselrun.target(object, "clans");
|
|
};
|
|
|
|
kesselrun.dialog.tgtTritanium = function(object) {
|
|
return kesselrun.target(object, "tritanium");
|
|
};
|
|
|
|
kesselrun.dialog.tgtDuranium = function(object) {
|
|
return kesselrun.target(object, "duranium");
|
|
};
|
|
|
|
kesselrun.dialog.tgtMolybdenum = function(object) {
|
|
return kesselrun.target(object, "molybdenum");
|
|
};
|
|
|
|
kesselrun.dialog.megacredits = function(object) {
|
|
return kesselrun.dialog.modeFun(object, "megacredits");
|
|
};
|
|
|
|
kesselrun.dialog.neutronium = function(object) {
|
|
return kesselrun.dialog.modeFun(object, "neutronium");
|
|
};
|
|
|
|
kesselrun.dialog.supplies = function(object) {
|
|
return kesselrun.dialog.modeFun(object, "supplies");
|
|
};
|
|
|
|
kesselrun.dialog.clans = function(object) {
|
|
return kesselrun.dialog.modeFun(object, "clans");
|
|
};
|
|
|
|
kesselrun.dialog.tritanium = function(object) {
|
|
return kesselrun.dialog.modeFun(object, "tritanium");
|
|
};
|
|
|
|
kesselrun.dialog.duranium = function(object) {
|
|
return kesselrun.dialog.modeFun(object, "duranium");
|
|
};
|
|
|
|
kesselrun.dialog.molybdenum = function(object) {
|
|
return kesselrun.dialog.modeFun(object, "molybdenum");
|
|
};
|
|
|
|
kesselrun.dialog.editPlanetList = function(object, index) {
|
|
const route = kesselrun.dialog.parent.route;
|
|
|
|
// Check if the planet is at the top or bottom of the route
|
|
const top = index == 0;
|
|
const bottom = index == (route.length - 1);
|
|
|
|
// Create a new div container
|
|
const div = $("<div/>");
|
|
|
|
// Create the 'Up' button
|
|
const btnUp = $("<div class='smallbutton' label='Move Up'>\u2191</div>");
|
|
|
|
// Create the 'Down' button
|
|
const btnDown = $("<div class='smallbutton' label='Move Down'>\u2193</div>");
|
|
|
|
// Create the 'Remove' button
|
|
const btnRemove = $("<div class='smallbutton' label='Remove'>\u274c</div>");
|
|
|
|
btnUp.on("click", function () {
|
|
kesselrun.movePlanet(kesselrun.dialog.parent.id, index, "up");
|
|
kesselrun.updateTable();
|
|
});
|
|
|
|
btnDown.on("click", function () {
|
|
kesselrun.movePlanet(kesselrun.dialog.parent.id, index, "down");
|
|
kesselrun.updateTable();
|
|
});
|
|
|
|
btnRemove.on("click", function () {
|
|
kesselrun.removePlanet(kesselrun.dialog.parent.id, index);
|
|
kesselrun.updateTable();
|
|
});
|
|
|
|
if (top) {
|
|
btnUp.css("visibility", "hidden");
|
|
}
|
|
|
|
if (bottom) {
|
|
btnDown.css("visibility", "hidden");
|
|
}
|
|
|
|
if (route.length == 1) {
|
|
btnRemove.css("visibility", "hidden");
|
|
}
|
|
|
|
// Append the buttons to the container
|
|
div.append(btnUp);
|
|
div.append(btnDown);
|
|
div.append(btnRemove);
|
|
|
|
return div;
|
|
};
|
|
|
|
kesselrun.dialog.editTradeRoutesList = function(object, index) {
|
|
const div = $("<div/>");
|
|
const btnRemove = $("<div class='smallbutton' label='Delete'>\u274c</div>");
|
|
|
|
btnRemove.on("click", function () {
|
|
kesselrun.dialog.targets = kesselrun.deleteTradeRoute(object.id)
|
|
kesselrun.updateTable();
|
|
});
|
|
|
|
// Append the buttons to the container
|
|
div.append(btnRemove);
|
|
return div;
|
|
};
|
|
|
|
// functions for edit planet target
|
|
|
|
kesselrun.dialog.resource = function(resource) {
|
|
return resource.charAt(0).toUpperCase() + resource.slice(1).toLowerCase();;
|
|
};
|
|
|
|
kesselrun.dialog.supply = function(resource) {
|
|
const parent = kesselrun.dialog.parent;
|
|
const planet = parent[1];
|
|
return kesselrun.detail(planet, resource);
|
|
};
|
|
|
|
kesselrun.dialog.target = function(resource) {
|
|
const parent = kesselrun.dialog.parent;
|
|
const tradeRoute = parent[0];
|
|
const planetId = parent[1].id;
|
|
return tradeRoute.targets?.[planetId]?.[resource] ?? "n/a"
|
|
};
|
|
|
|
kesselrun.dialog.editTargets = function(resource) {
|
|
const parent = kesselrun.dialog.parent;
|
|
const tradeRoute = parent[0];
|
|
const planetId = parent[1].id;
|
|
let val = tradeRoute.targets?.[planetId]?.[resource] ?? 0
|
|
|
|
// Create a new div container
|
|
const div = $("<div/>");
|
|
|
|
const deltas = [1000, 100, 10, 1, 0, -1, -10, -100, -1000];
|
|
deltas.forEach( delta => {
|
|
const text = (delta == 0 ? "" : delta > 0 ? "\u2191" : "\u2193") + Math.abs(delta);
|
|
const btn = $("<div class='smallbutton'/>");
|
|
btn.text(text)
|
|
btn.tclick(function() {
|
|
if (delta == 0) {
|
|
tradeRoute.targets[planetId][resource] = 0;
|
|
} else {
|
|
tradeRoute.targets[planetId][resource] = Math.max(0, val + delta);
|
|
}
|
|
kesselrun.modifyTradeRouteTargets(tradeRoute.id, planetId, tradeRoute.targets[planetId]);
|
|
kesselrun.updateTable();
|
|
});
|
|
div.append(btn);
|
|
})
|
|
// TODO delete button
|
|
return div;
|
|
};
|
|
|
|
|
|
kesselrun.columnDefinitions = {
|
|
"ID": { fun: kesselrun.dialog.id, tt: "Trade Route ID" },
|
|
"Name": { fun: kesselrun.dialog.name, tt: "Trade Route Name" },
|
|
"Pl": { fun: kesselrun.dialog.planets, tt: "Number of Planets" }, // Planets
|
|
"Sh": { fun: kesselrun.dialog.ships, tt: "Number of Ships" }, // Ships
|
|
"Dist": { fun: kesselrun.dialog.dist, tt: "Distance to next planet (Warp)" },
|
|
"Tgt$": { fun: kesselrun.dialog.tgtMegacredits, tt: "Target Megacredits" },
|
|
"TgtS": { fun: kesselrun.dialog.tgtSupplies, tt: "Target Supplies" },
|
|
"TgtC": { fun: kesselrun.dialog.tgtClans, tt: "Target Clans" },
|
|
"TgtN": { fun: kesselrun.dialog.tgtNeutronium, tt: "Target Neutronium" },
|
|
"TgtD": { fun: kesselrun.dialog.tgtDuranium, tt: "Target Duranium" },
|
|
"TgtT": { fun: kesselrun.dialog.tgtTritanium, tt: "Target Tritanium" },
|
|
"TgtM": { fun: kesselrun.dialog.tgtMolybdenum, tt: "Target Molybdenum" },
|
|
"$": { fun: kesselrun.dialog.megacredits, tt: "Megacredits" },
|
|
"S": { fun: kesselrun.dialog.supplies, tt: "Supplies" },
|
|
"C": { fun: kesselrun.dialog.supplies, tt: "Clans" },
|
|
"N": { fun: kesselrun.dialog.neutronium, tt: "Neutronium" },
|
|
"D": { fun: kesselrun.dialog.duranium, tt: "Duranium" },
|
|
"T": { fun: kesselrun.dialog.tritanium, tt: "Tritanium" },
|
|
"M": { fun: kesselrun.dialog.molybdenum, tt: "Molybdenum" },
|
|
"*": { fun: kesselrun.dialog.editPlanetList, tt: "Move Up/Down/Delete"},
|
|
"\u2600": { fun: kesselrun.dialog.editTradeRoutesList, tt: "Delete"},
|
|
// planet target columns
|
|
"Resource": { fun: kesselrun.dialog.resource, tt: "Resource"},
|
|
"Supply": { fun: kesselrun.dialog.supply, tt: "Avail./In-ground (Prod.)"},
|
|
"Target": { fun: kesselrun.dialog.target, tt: "Target Amount"},
|
|
"Edit": { fun: kesselrun.dialog.editTargets, tt: "Edit Target Amount"},
|
|
};
|
|
|
|
kesselrun.buttonDefinitions = {
|
|
"Add Route": { fun: kesselrun.addRoute, tt: "Add a new route from a ship waypoints" },
|
|
"Ships": { fun: kesselrun.showShipsDialog, tt: "Show ships assgined to trade route"},
|
|
"Add Ship": { fun: kesselrun.addShipDialog, tt: "Add a ship to the trade route"},
|
|
"Add Planet": { fun: kesselrun.addPlanetDialog, tt: "Add a planet to the trade route"},
|
|
"\u23CE": { fun: kesselrun.dialog.popDialog, tt: "Return"}
|
|
};
|
|
|
|
mcntk.addMcNTkMenuEntry({
|
|
// f04b - play icon from font-awesome
|
|
label: mcntk.onMobile() ? "\uf04b" : "Run Trade Routes",
|
|
fontClass: "fas",
|
|
name: "Run Trade Routes",
|
|
color: "#81f7e5",
|
|
state: () => { return false; },
|
|
action: kesselrun.run
|
|
});
|
|
|
|
mcntk.addMcNTkMenuEntry({
|
|
// f279 - map icon from font-awesome
|
|
label: mcntk.onMobile() ? "\uf279" : "Show Trade Routes",
|
|
fontClass: "fas",
|
|
name: "Show Trade Routes",
|
|
color: "#fca311",
|
|
state: () => { return false; },
|
|
action: kesselrun.showTradeRoutes
|
|
});
|
|
|
|
// Styling
|
|
|
|
var head = document.getElementsByTagName('head')[0];
|
|
if (head) {
|
|
var newCss = document.createElement('style');
|
|
newCss.type = "text/css";
|
|
newCss.innerHTML =
|
|
"table.KRTradeRoutes { margin: 0px; } " + // Reduce spacing to put more ships on the screen
|
|
"table.KRTradeRoutes th { padding: 10px 2px; } " + // take out some horizontal padding for easier fitting on the screen
|
|
"table.KRTradeRoutes tr td { padding: 0px 0.5ex; border: 0px; } " + // Reduce spacing to put more ships on the screen
|
|
"section.popup { max-width: 95%; }" // Allow wide popup for our fleet
|
|
head.appendChild(newCss);
|
|
}
|