mirror of
https://github.com/UnickSoft/graphonline.git
synced 2026-02-16 02:30:51 +00:00
Add find all shortest pathes between two vertexes.
This commit is contained in:
@@ -29,7 +29,8 @@ function loadAsyncAlgorithms(onFinish) {
|
||||
"RadiusAndDiameter.js",
|
||||
"ShortestPath.js",
|
||||
"VerticesDegree.js",
|
||||
"MaxIndependentSet.js"];
|
||||
"MaxIndependentSet.js",
|
||||
"FindAllShortestPatches.js"];
|
||||
|
||||
doIncludeAsync (pluginsList.map((plugin) => include ("model/plugins/" + plugin, modulDir)), onFinish);
|
||||
}
|
||||
|
||||
@@ -0,0 +1,292 @@
|
||||
/**
|
||||
* Find all shortest pathes.
|
||||
*
|
||||
*/
|
||||
function FindAllShortestPathes(graph, app)
|
||||
{
|
||||
BaseAlgorithmEx.apply(this, arguments);
|
||||
this.message = g_selectStartVertex;
|
||||
this.selectedObjects = {};
|
||||
this.foundSubGraphs = {};
|
||||
this.nSubgraphIndex = 0;
|
||||
this.nSubGraphCount = 0;
|
||||
this.foundPaths = {};
|
||||
}
|
||||
|
||||
|
||||
// inheritance.
|
||||
FindAllShortestPathes.prototype = Object.create(BaseAlgorithmEx.prototype);
|
||||
// First selected.
|
||||
FindAllShortestPathes.prototype.firstObject = null;
|
||||
// Second selected.
|
||||
FindAllShortestPathes.prototype.secondObject = null;
|
||||
// Path
|
||||
FindAllShortestPathes.prototype.pathObjects = null;
|
||||
// Infinity
|
||||
FindAllShortestPathes.prototype.infinityValue = 1E9 - 1;
|
||||
|
||||
FindAllShortestPathes.prototype.getName = function(local)
|
||||
{
|
||||
return g_FindAllShortestPathes;
|
||||
}
|
||||
|
||||
FindAllShortestPathes.prototype.getId = function()
|
||||
{
|
||||
return "OlegSh.FindAllShortestPathes";
|
||||
}
|
||||
|
||||
// @return message for user.
|
||||
FindAllShortestPathes.prototype.getMessage = function(local)
|
||||
{
|
||||
return this.message;
|
||||
}
|
||||
|
||||
FindAllShortestPathes.prototype.getCategory = function()
|
||||
{
|
||||
return 1;
|
||||
}
|
||||
|
||||
FindAllShortestPathes.prototype.result = function(resultCallback)
|
||||
{
|
||||
if (this.firstObject && this.secondObject)
|
||||
{
|
||||
this.outResultCallback = function (result ) { resultCallback(result); };
|
||||
self = this;
|
||||
this.CalculateAlgorithm("prnpaths", [
|
||||
{name: "start", value: this.firstObject.id},
|
||||
{name: "finish", value: this.secondObject.id}
|
||||
],
|
||||
function (pathObjects, properties, results)
|
||||
{
|
||||
self.resultCallback(pathObjects, properties, results);
|
||||
});
|
||||
}
|
||||
return null;
|
||||
}
|
||||
|
||||
FindAllShortestPathes.prototype.setResultMessage = function()
|
||||
{
|
||||
if (this.nSubGraphCount > 0)
|
||||
{
|
||||
var currentPath = "";
|
||||
var first = true;
|
||||
this.foundPaths[this.nSubgraphIndex].forEach((nodeId) => {
|
||||
currentPath += (first ? "" : "⇒") + this.graph.FindVertex(nodeId).mainText;
|
||||
first = false;
|
||||
});
|
||||
|
||||
this.message = g_numberOfShortestPathesFrom + this.firstObject.mainText +
|
||||
g_to + this.secondObject.mainText + g_are +
|
||||
this.nSubGraphCount + ". " + g_pathN + (1 + parseInt(this.nSubgraphIndex)) + ": " + currentPath +
|
||||
" <select style=\"float:right\" id=\"enumSubgraphs\"></select>";
|
||||
}
|
||||
else
|
||||
{
|
||||
this.message = g_pathNotExists;
|
||||
}
|
||||
}
|
||||
|
||||
FindAllShortestPathes.prototype.resultCallback = function(pathObjects, properties, results)
|
||||
{
|
||||
var outputResult = {};
|
||||
outputResult["version"] = 1;
|
||||
outputResult["minPath"] = true;
|
||||
|
||||
this.pathObjects = pathObjects;
|
||||
this.properties = properties;
|
||||
|
||||
var bFound = results.length > 0 && results[0].value < this.infinityValue && (results[0].type == 1 || results[0].type == 2);
|
||||
|
||||
if (bFound)
|
||||
{
|
||||
let patches_len = {};
|
||||
let min_len = 1E10;
|
||||
|
||||
// Search minimal patches
|
||||
{
|
||||
let current_len = 0;
|
||||
|
||||
var subGraphIndex = 0;
|
||||
var prevNodeId = -1;
|
||||
let any_found = false;
|
||||
for (var i = 0; i < results.length; i++)
|
||||
{
|
||||
if (results[i].type == 6)
|
||||
{
|
||||
patches_len[subGraphIndex] = current_len;
|
||||
subGraphIndex++;
|
||||
prevNodeId = -1;
|
||||
min_len = Math.min(min_len, current_len);
|
||||
current_len = 0;
|
||||
}
|
||||
|
||||
if (results[i].type == 4)
|
||||
{
|
||||
any_found = true;
|
||||
var nodeId = parseInt(results[i].value);
|
||||
var index = subGraphIndex;
|
||||
|
||||
if (prevNodeId >= 0)
|
||||
{
|
||||
var edgeObject = this.graph.FindEdgeMin(prevNodeId, nodeId);
|
||||
current_len = current_len + edgeObject.GetWeight();
|
||||
}
|
||||
prevNodeId = nodeId;
|
||||
}
|
||||
}
|
||||
if (any_found)
|
||||
{
|
||||
patches_len[subGraphIndex] = current_len;
|
||||
min_len = Math.min(min_len, current_len);
|
||||
}
|
||||
}
|
||||
|
||||
let min_pathes_count = 0;
|
||||
for (const [key, value] of Object.entries(patches_len)) {
|
||||
if (value == min_len)
|
||||
{
|
||||
min_pathes_count = min_pathes_count + 1;
|
||||
}
|
||||
}
|
||||
|
||||
this.nSubGraphCount = min_pathes_count;
|
||||
|
||||
this.foundSubGraphs = {};
|
||||
this.foundPaths = {};
|
||||
for (var i = 0; i < this.nSubGraphCount; i++)
|
||||
{
|
||||
this.foundSubGraphs[i] = {};
|
||||
this.foundPaths[i] = [];
|
||||
}
|
||||
|
||||
var subGraphIndex = 0;
|
||||
var prevNodeId = -1;
|
||||
for (var i = 0; i < results.length; i++)
|
||||
{
|
||||
if (results[i].type == 6)
|
||||
{
|
||||
subGraphIndex++;
|
||||
prevNodeId = -1;
|
||||
}
|
||||
|
||||
if (patches_len[subGraphIndex] > min_len)
|
||||
{
|
||||
// Skip
|
||||
continue;
|
||||
}
|
||||
|
||||
if (results[i].type == 4)
|
||||
{
|
||||
var nodeId = parseInt(results[i].value);
|
||||
var index = subGraphIndex;
|
||||
var subgGraph = this.foundSubGraphs[index];
|
||||
subgGraph[nodeId] = true;
|
||||
|
||||
this.foundPaths[index].push(nodeId);
|
||||
|
||||
if (prevNodeId >= 0)
|
||||
{
|
||||
var edgeObject = this.graph.FindEdgeMin(prevNodeId, nodeId);
|
||||
subgGraph[edgeObject.id] = true;
|
||||
}
|
||||
prevNodeId = nodeId;
|
||||
}
|
||||
}
|
||||
|
||||
this.setResultMessage();
|
||||
}
|
||||
else
|
||||
{
|
||||
this.secondObject = null;
|
||||
this.firstObject = null;
|
||||
this.message = g_pathNotExists;
|
||||
}
|
||||
|
||||
this.outResultCallback(outputResult);
|
||||
}
|
||||
|
||||
FindAllShortestPathes.prototype.messageWasChanged = function()
|
||||
{
|
||||
var self = this;
|
||||
|
||||
if ($('#enumSubgraphs'))
|
||||
{
|
||||
for (var i = 0; i < this.nSubGraphCount; i++)
|
||||
{
|
||||
$('#enumSubgraphs').append("<option value=\"" + i + "\"" + (self.nSubgraphIndex==i ? "selected": "") + ">" +
|
||||
g_pathN + (i + 1) +
|
||||
"</option>");
|
||||
}
|
||||
|
||||
$('#enumSubgraphs').change(function () {
|
||||
self.nSubgraphIndex = $('#enumSubgraphs').val();
|
||||
self.setResultMessage();
|
||||
self.app.redrawGraph();
|
||||
self.app.updateMessage();
|
||||
});
|
||||
}
|
||||
}
|
||||
|
||||
FindAllShortestPathes.prototype.selectVertex = function(vertex)
|
||||
{
|
||||
this.pathObjects = null;
|
||||
this.shortDist = null;
|
||||
|
||||
if (this.firstObject)
|
||||
{
|
||||
this.message = g_processing;
|
||||
this.secondObject = vertex;
|
||||
this.selectedObjects = [];
|
||||
}
|
||||
else
|
||||
{
|
||||
this.firstObject = vertex;
|
||||
this.secondObject = null;
|
||||
this.selectedObjects = {};
|
||||
this.message = g_selectFinishVertex;
|
||||
}
|
||||
|
||||
return true;
|
||||
}
|
||||
|
||||
FindAllShortestPathes.prototype.deselectAll = function()
|
||||
{
|
||||
this.firstObject = null;
|
||||
this.secondObject = null;
|
||||
this.selectedObjects = {};
|
||||
this.foundSubGraphs = {};
|
||||
this.nSubgraphIndex = 0;
|
||||
this.nSubGraphCount = 0;
|
||||
this.message = g_selectStartVertex;
|
||||
return true;
|
||||
}
|
||||
|
||||
FindAllShortestPathes.prototype.instance = function()
|
||||
{
|
||||
return false;
|
||||
}
|
||||
|
||||
FindAllShortestPathes.prototype.getObjectSelectedGroup = function(object)
|
||||
{
|
||||
return (this.nSubgraphIndex in this.foundSubGraphs && object.id in this.foundSubGraphs[this.nSubgraphIndex]) ? 1 :
|
||||
(object.id in this.selectedObjects) ? this.selectedObjects[object.id] : ((object == this.firstObject || object == object.secondObject) ? 1 : 0);
|
||||
}
|
||||
|
||||
FindAllShortestPathes.prototype.getPriority = function()
|
||||
{
|
||||
return -9.46;
|
||||
}
|
||||
|
||||
FindAllShortestPathes.prototype.IsSupportNegativeWeight = function()
|
||||
{
|
||||
return true;
|
||||
}
|
||||
|
||||
// Factory for connected components.
|
||||
function CreateFindAllShortestPathes(graph, app)
|
||||
{
|
||||
return new FindAllShortestPathes(graph, app)
|
||||
}
|
||||
|
||||
// Gerister connected component.
|
||||
RegisterAlgorithm (CreateFindAllShortestPathes);
|
||||
Reference in New Issue
Block a user