diff --git a/src/source/frontend/src/index.js b/src/source/frontend/src/index.js
index 8428b26..48af995 100644
--- a/src/source/frontend/src/index.js
+++ b/src/source/frontend/src/index.js
@@ -1,3 +1,5 @@
+/** @format */
+
import Sigma from "sigma";
import Graph from "graphology";
import circular from "graphology-layout/circular";
@@ -8,77 +10,80 @@ import EdgeProgram from "./programs/edge.antialias";
import { animateNodes } from "sigma/utils/animate";
class modifierHandler {
- constructor() {
- this.SHIFT = false;
- this.CTRL = false;
- this.FUNCTION = false;
- this.ALT = false;
- this.toggleValue = async function (event, bool) {
- switch (event.key) {
- case "Alt":
- this.ALT = bool;
- break;
- case "Control":
- this.CTRL = bool;
- break;
- case "Fn":
- this.FUNCTION = bool;
- break;
- case "Shift":
- this.SHIFT = bool;
- break;
- }
- };
- window.onkeydown = async (event) => {
- this.toggleValue(event, true);
- };
- window.onkeyup = async (event) => {
- this.toggleValue(event, false);
- };
- }
+ constructor() {
+ this.SHIFT = false;
+ this.CTRL = false;
+ this.FUNCTION = false;
+ this.ALT = false;
+ this.toggleValue = async function (event, bool) {
+ switch (event.key) {
+ case "Alt":
+ this.ALT = bool;
+ break;
+ case "Control":
+ this.CTRL = bool;
+ break;
+ case "Fn":
+ this.FUNCTION = bool;
+ break;
+ case "Shift":
+ this.SHIFT = bool;
+ break;
+ }
+ };
+ window.onkeydown = async (event) => {
+ this.toggleValue(event, true);
+ };
+ window.onkeyup = async (event) => {
+ this.toggleValue(event, false);
+ };
+ }
}
const statusColors = {
- Neutral: "#5D5A6D",
- Selected: "#444155",
- AuthNeutral: "#8B7FD3",
- AuthSelected: "#5B4EA1",
- ErrorNeutral: "#B41E06",
- ErrorSelected: "#a30006",
- SuccessNeutral: "#178A0E",
- SuccessSelected: "#47A13F",
+ Neutral: "#5D5A6D",
+ Selected: "#444155",
+ AuthNeutral: "#8B7FD3",
+ AuthSelected: "#5B4EA1",
+ ErrorNeutral: "#B41E06",
+ ErrorSelected: "#a30006",
+ SuccessNeutral: "#178A0E",
+ SuccessSelected: "#47A13F",
};
+
+var lastClickTimestamp;
+var recentlyClickedNode;
const mod = new modifierHandler();
const container = document.getElementById("graphContainer");
const suggestionsList = document.querySelector("#search-opts");
const nodeDismissHandler = new eventHandlingMechanism();
const state = {
- activeNode: undefined,
- searchQuery: "",
- suggestions: [],
- errorNodes: [],
- successNodes: [],
- renderedNodes: [],
- renderedLinks: [],
+ activeNode: undefined,
+ searchQuery: "",
+ suggestions: [],
+ errorNodes: [],
+ successNodes: [],
+ renderedNodes: [],
+ renderedLinks: [],
};
const graphData = {
- options: {
- allowSelfLoops: false,
- multi: false,
- type: "mixed",
- },
- nodes: [
- {
- key: "server",
- attributes: {
- size: 15,
- label: "Server",
- type: "server",
- color: statusColors.Neutral,
- },
- },
- ],
- edges: [],
+ options: {
+ allowSelfLoops: false,
+ multi: false,
+ type: "mixed",
+ },
+ nodes: [
+ {
+ key: "server",
+ attributes: {
+ size: 15,
+ label: "Server",
+ type: "server",
+ color: statusColors.Neutral,
+ },
+ },
+ ],
+ edges: [],
};
const graph = new Graph();
@@ -86,449 +91,426 @@ graph.import(graphData);
circular.assign(graph);
const renderer = new Sigma(graph, container, {
- renderLabels: false,
- minCameraRatio: 0.1,
- maxCameraRatio: 3,
- nodeProgramClasses: {
- related: relatedProgram,
- unrelated: unrelatedProgram,
- server: serverProgram,
- },
- edgeProgramClasses: {
- antialias: EdgeProgram,
- },
+ renderLabels: false,
+ minCameraRatio: 0.1,
+ maxCameraRatio: 3,
+ allowInvalidContainer: true,
+ nodeProgramClasses: {
+ related: relatedProgram,
+ unrelated: unrelatedProgram,
+ server: serverProgram,
+ },
+ edgeProgramClasses: {
+ antialias: EdgeProgram,
+ },
});
const camera = renderer.getCamera();
camera.addListener("updated", async (_) => {
- if (camera.x > camera.ratio + 1) camera.x = camera.ratio + 1;
- if (camera.x < -camera.ratio) camera.x = -camera.ratio;
- if (camera.y > camera.ratio + 1) camera.y = camera.ratio + 1;
- if (camera.y < -camera.ratio) camera.y = -camera.ratio;
+ if (camera.x > camera.ratio + 1) camera.x = camera.ratio + 1;
+ if (camera.x < -camera.ratio) camera.x = -camera.ratio;
+ if (camera.y > camera.ratio + 1) camera.y = camera.ratio + 1;
+ if (camera.y < -camera.ratio) camera.y = -camera.ratio;
});
async function blinkStatus(statusIdentifier, node) {
- switch (statusIdentifier.toLowerCase()) {
- case "error":
- state.errorNodes.push(node);
- setTimeout(() => {
- state.errorNodes.splice(state.errorNodes.indexOf(node), 1);
- renderer.refresh();
- }, 1500);
- break;
- case "success":
- state.successNodes.push(node);
- setTimeout(() => {
- state.successNodes.splice(state.successNodes.indexOf(node), 1);
- renderer.refresh();
- }, 1500);
- break;
- }
- renderer.refresh();
+ switch (statusIdentifier.toLowerCase()) {
+ case "error":
+ state.errorNodes.push(node);
+ setTimeout(() => {
+ state.errorNodes.splice(state.errorNodes.indexOf(node), 1);
+ renderer.refresh();
+ }, 5000);
+ break;
+ case "success":
+ state.successNodes.push(node);
+ setTimeout(() => {
+ state.successNodes.splice(state.successNodes.indexOf(node), 1);
+ renderer.refresh();
+ }, 5000);
+ break;
+ }
+ renderer.refresh();
}
renderer.setSetting("nodeReducer", (node, data) => {
- const res = { ...data };
- const active = state.activeNode === node;
- const error = state.errorNodes.includes(node);
- const success = state.successNodes.includes(node);
- const prepend =
- error && success
- ? error
- : !(success || error)
- ? ""
- : ["Success", "Error"][[success, error].indexOf(true)];
- const fullStatusString =
- prepend + ["Selected", "Neutral"][[active, !active].indexOf(true)];
- res.color = statusColors[fullStatusString];
- return res;
+ const res = { ...data };
+ const active = state.activeNode === node;
+ const error = state.errorNodes.includes(node);
+ const success = state.successNodes.includes(node);
+ const prepend =
+ error && success
+ ? error
+ : !(success || error)
+ ? ""
+ : ["Success", "Error"][[success, error].indexOf(true)];
+ const fullStatusString = prepend + ["Selected", "Neutral"][[active, !active].indexOf(true)];
+ res.color = statusColors[fullStatusString];
+ return res;
});
renderer.setSetting("edgeReducer", (edge, data) => {
- const res = { ...data };
- var prependable = graph.extremities(edge).reduce((last, extremity) => {
- return state.errorNodes.includes(extremity)
- ? "Error"
- : state.successNodes.includes(extremity) && last != "Error"
- ? "Success"
- : last;
- }, "");
- if (!prependable) {
- if (
- [1, 0]
- .map((flippedIndex, trueIndex) => {
- return (
- authPeers.includes(graph.extremities(edge)[flippedIndex]) &&
- graph.extremities(edge)[trueIndex] ===
- CONFIG.communication.hiddenAlias
- );
- })
- .includes(true)
- ) {
- prependable = "Auth";
- }
- }
- if (state.activeNode) {
- res.color = "#d9dde0";
- if (graph.hasExtremity(edge, state.activeNode)) {
- res.color = statusColors[prependable + "Selected"];
- }
- return res;
- }
- res.color = statusColors[prependable + "Neutral"];
- return res;
+ const res = { ...data };
+ var prependable = graph.extremities(edge).reduce((last, extremity) => {
+ return state.errorNodes.includes(extremity)
+ ? "Error"
+ : state.successNodes.includes(extremity) && last != "Error"
+ ? "Success"
+ : last;
+ }, "");
+ if (!prependable) {
+ if (
+ [1, 0]
+ .map((flippedIndex, trueIndex) => {
+ return (
+ authPeers.includes(graph.extremities(edge)[flippedIndex]) &&
+ graph.extremities(edge)[trueIndex] === CONFIG.communication.hiddenAlias
+ );
+ })
+ .includes(true)
+ ) {
+ prependable = "Auth";
+ }
+ }
+ if (state.activeNode) {
+ res.color = "#d9dde0";
+ if (graph.hasExtremity(edge, state.activeNode)) {
+ res.color = statusColors[prependable + "Selected"];
+ }
+ return res;
+ }
+ res.color = statusColors[prependable + "Neutral"];
+ return res;
});
function setSearchQuery(query) {
- if (!CONFIG.UI.renderUnfamiliarPublicAliases) {
- suggestionsList.innerHTML = "";
- return;
- }
- state.searchQuery = query;
- if (query) {
- const lcQuery = escapeHTML(query.toLowerCase());
- let suggestions = Object.keys(networkMap.nodes).filter((node) => {
- try {
- return hiddenAliasLookup[node].toLowerCase().indexOf(lcQuery) != -1;
- } catch {
- return false;
- }
- });
- let sortedSuggestions = suggestions.reduce((sum, suggestion) => {
- const index = hiddenAliasLookup[suggestion]
- .toLowerCase()
- .indexOf(lcQuery);
- sum[index] = sum[index] ? sum[index] : [];
- sum[index].push(suggestion);
- return sum;
- }, {});
- var suggestionPriority = [];
- var suggestionPairs = {};
- for (let len in sortedSuggestions) {
- sortedSuggestions[len] = sortedSuggestions[len].sort((a, b) => {
- a = hiddenAliasLookup[a].toLowerCase();
- b = hiddenAliasLookup[b].toLowerCase();
- return a === b ? 0 : a > b ? 1 : -1;
- });
- for (let suggestion of sortedSuggestions[len]) {
- suggestionPriority.push(suggestion);
- let firstIndex = hiddenAliasLookup[suggestion]
- .toLowerCase()
- .indexOf(lcQuery);
- const boldedSuggestion =
- hiddenAliasLookup[suggestion].slice(0, firstIndex) +
- `${hiddenAliasLookup[suggestion].slice(
- firstIndex,
- firstIndex + lcQuery.length
- )}` +
- hiddenAliasLookup[suggestion].slice(firstIndex + lcQuery.length);
- suggestionPairs[suggestion] = boldedSuggestion;
- }
- }
- renderQuerySuggestions(suggestionPairs, suggestionPriority, true);
- if (
- suggestionPriority.length === 1 &&
- hiddenAliasLookup[suggestionPriority[0]].toLowerCase() === query
- ) {
- activateNode(suggestionPriority[0]);
- } else {
- state.selectedNode = undefined;
- }
- } else {
- renderQuerySuggestions();
- }
+ if (!CONFIG.UI.renderUnfamiliarPublicAliases) {
+ suggestionsList.innerHTML = "";
+ return;
+ }
+ state.searchQuery = query;
+ if (query) {
+ const lcQuery = escapeHTML(query.toLowerCase());
+ let suggestions = Object.keys(networkMap.nodes).filter((node) => {
+ try {
+ return hiddenAliasLookup[node].toLowerCase().indexOf(lcQuery) != -1;
+ } catch {
+ return false;
+ }
+ });
+ let sortedSuggestions = suggestions.reduce((sum, suggestion) => {
+ const index = hiddenAliasLookup[suggestion].toLowerCase().indexOf(lcQuery);
+ sum[index] = sum[index] ? sum[index] : [];
+ sum[index].push(suggestion);
+ return sum;
+ }, {});
+ var suggestionPriority = [];
+ var suggestionPairs = {};
+ for (let len in sortedSuggestions) {
+ sortedSuggestions[len] = sortedSuggestions[len].sort((a, b) => {
+ a = hiddenAliasLookup[a].toLowerCase();
+ b = hiddenAliasLookup[b].toLowerCase();
+ return a === b ? 0 : a > b ? 1 : -1;
+ });
+ for (let suggestion of sortedSuggestions[len]) {
+ suggestionPriority.push(suggestion);
+ let firstIndex = hiddenAliasLookup[suggestion].toLowerCase().indexOf(lcQuery);
+ const boldedSuggestion =
+ hiddenAliasLookup[suggestion].slice(0, firstIndex) +
+ `${hiddenAliasLookup[suggestion].slice(firstIndex, firstIndex + lcQuery.length)}` +
+ hiddenAliasLookup[suggestion].slice(firstIndex + lcQuery.length);
+ suggestionPairs[suggestion] = boldedSuggestion;
+ }
+ }
+ renderQuerySuggestions(suggestionPairs, suggestionPriority, true);
+ if (
+ suggestionPriority.length === 1 &&
+ hiddenAliasLookup[suggestionPriority[0]].toLowerCase() === query
+ ) {
+ activateNode(suggestionPriority[0]);
+ } else {
+ state.selectedNode = undefined;
+ }
+ } else {
+ renderQuerySuggestions();
+ }
}
-async function renderQuerySuggestions(
- suggestionPairs,
- suggestionPriority,
- isInformed
-) {
- var suggestionPairs =
- typeof suggestionPairs === "object"
- ? suggestionPairs
- : Object.assign(hiddenAliasLookup);
- delete suggestionPairs[CONFIG.communication.hiddenAlias];
- var suggestionPriority =
- typeof suggestionPriority === "object"
- ? suggestionPriority
- : Object.keys(suggestionPairs).sort((a, b) => {
- a = hiddenAliasLookup[a].toLowerCase();
- b = hiddenAliasLookup[b].toLowerCase();
- return a === b ? 0 : a > b ? 1 : -1;
- });
- suggestionPairs =
- Object.keys(suggestionPairs).reduce((total, item) => {
- total[item] = (isInformed ? "@" : "@") + suggestionPairs[item];
- return total;
- }, {}) ?? {};
- suggestionsList.innerHTML = "";
- if (Object.keys(suggestionPairs) == "") return;
- Object.keys(suggestionPairs).forEach((hidden, index) => {
- suggestionsList.innerHTML += `
+async function renderQuerySuggestions(suggestionPairs, suggestionPriority, isInformed) {
+ var suggestionPairs =
+ typeof suggestionPairs === "object" ? suggestionPairs : Object.assign(hiddenAliasLookup);
+ delete suggestionPairs[CONFIG.communication.hiddenAlias];
+ var suggestionPriority =
+ typeof suggestionPriority === "object"
+ ? suggestionPriority
+ : Object.keys(suggestionPairs).sort((a, b) => {
+ a = hiddenAliasLookup[a].toLowerCase();
+ b = hiddenAliasLookup[b].toLowerCase();
+ return a === b ? 0 : a > b ? 1 : -1;
+ });
+ suggestionPairs =
+ Object.keys(suggestionPairs).reduce((total, item) => {
+ total[item] = (isInformed ? "@" : "@") + suggestionPairs[item];
+ return total;
+ }, {}) ?? {};
+ suggestionsList.innerHTML = "";
+ if (Object.keys(suggestionPairs) == "") return;
+ Object.keys(suggestionPairs).forEach((hidden, index) => {
+ suggestionsList.innerHTML += `
`;
- });
- document.querySelectorAll(".suggestion-button").forEach((btn) => {
- btn.addEventListener("mouseup", async (e) => {
- setTimeout(() => {
- document.querySelector("#search-opts").style.visibility = "hidden";
- }, 1);
- });
- btn.addEventListener("keydown", async (e) => {
- if (e.key == "ArrowDown") {
- e.preventDefault();
- if (
- document.querySelectorAll(".suggestion-button")[
- new Number(e.target.dataset.index) + 1
- ] == undefined
- )
- return;
- document
- .querySelectorAll(".suggestion-button")
- [new Number(e.target.dataset.index) + 1].focus();
- }
- if (
- e.key == "ArrowUp" &&
- !e.target.classList.contains("search-opts-cap")
- ) {
- e.preventDefault();
- if (
- document.querySelectorAll(".suggestion-button")[
- new Number(e.target.dataset.index) - 1
- ] == undefined
- )
- return;
- document
- .querySelectorAll(".suggestion-button")
- [e.target.dataset.index - 1].focus();
- }
- });
- });
- document
- .querySelector(".search-opts-cap")
- .addEventListener("keydown", async (e) => {
- if (e.key == "ArrowUp") {
- e.preventDefault();
- document.querySelector("#searchEntryField").focus();
- }
- });
+ });
+ document.querySelectorAll(".suggestion-button").forEach((btn) => {
+ btn.addEventListener("mouseup", async (e) => {
+ setTimeout(() => {
+ document.querySelector("#search-opts").style.visibility = "hidden";
+ }, 1);
+ });
+ btn.addEventListener("keydown", async (e) => {
+ if (e.key == "ArrowDown") {
+ e.preventDefault();
+ if (
+ document.querySelectorAll(".suggestion-button")[new Number(e.target.dataset.index) + 1] ==
+ undefined
+ )
+ return;
+ document
+ .querySelectorAll(".suggestion-button")
+ [new Number(e.target.dataset.index) + 1].focus();
+ }
+ if (e.key == "ArrowUp" && !e.target.classList.contains("search-opts-cap")) {
+ e.preventDefault();
+ if (
+ document.querySelectorAll(".suggestion-button")[new Number(e.target.dataset.index) - 1] ==
+ undefined
+ )
+ return;
+ document.querySelectorAll(".suggestion-button")[e.target.dataset.index - 1].focus();
+ }
+ });
+ });
+ document.querySelector(".search-opts-cap").addEventListener("keydown", async (e) => {
+ if (e.key == "ArrowUp") {
+ e.preventDefault();
+ document.querySelector("#searchEntryField").focus();
+ }
+ });
}
networkMap.onUpdate(async (_sig, externalDetail) => {
- switch (externalDetail[0]) {
- case "addNode":
- graph.updateNode(externalDetail[1], () => {
- return {
- x: 0,
- y: 0,
- label:
- externalDetail[1] === CONFIG.communication.hiddenAlias
- ? "𝙈𝙚"
- : CONFIG.UI.renderUnfamiliarPublicAliases
- ? hiddenAliasLookup[externalDetail[1]]
- : externalDetail[1],
- size: externalDetail[1] === CONFIG.communication.hiddenAlias ? 10 : 5,
- color: statusColors.Neutral,
- type:
- externalDetail[1] === CONFIG.communication.hiddenAlias
- ? "server"
- : "unrelated",
- };
- });
- graph.updateEdge(externalDetail[1], "server", () => {
- return { type: "antialias", size: 1 };
- });
- break;
- case "addEdge":
- graph.updateEdge(...externalDetail[1].sort(), () => {
- return { type: "antialias", size: 1 };
- });
- break;
- case "removeEdge":
- graph.dropEdge(...externalDetail[1].sort());
- break;
- case "removeNode":
- graph.dropNode(externalDetail[1]);
- break;
- case "totalWipe":
- graph.forEachNode((node) => {
- if (node === CONFIG.communication.hiddenAlias || node === "server")
- return;
- try {
- graph.dropNode(node);
- } catch {
- return;
- }
- });
- break;
- }
- const circularLocations = circular(graph);
- animateNodes(graph, circularLocations, {
- easing: "quadraticIn",
- duration: 2000,
- });
- renderer.refresh();
- setSearchQuery(document.querySelector("#searchEntryField").value);
+ switch (externalDetail[0]) {
+ case "addNode":
+ graph.updateNode(externalDetail[1], () => {
+ return {
+ x: 0,
+ y: 0,
+ label:
+ externalDetail[1] === CONFIG.communication.hiddenAlias
+ ? "𝙈𝙚"
+ : CONFIG.UI.renderUnfamiliarPublicAliases
+ ? hiddenAliasLookup[externalDetail[1]]
+ : externalDetail[1],
+ size: externalDetail[1] === CONFIG.communication.hiddenAlias ? 10 : 5,
+ color: statusColors.Neutral,
+ type: externalDetail[1] === CONFIG.communication.hiddenAlias ? "server" : "unrelated",
+ };
+ });
+ graph.updateEdge(externalDetail[1], "server", () => {
+ return { type: "antialias", size: 1 };
+ });
+ break;
+ case "addEdge":
+ graph.updateEdge(...externalDetail[1].sort(), () => {
+ return { type: "antialias", size: 1 };
+ });
+ break;
+ case "removeEdge":
+ graph.dropEdge(...externalDetail[1].sort());
+ break;
+ case "removeNode":
+ graph.dropNode(externalDetail[1]);
+ break;
+ case "totalWipe":
+ graph.forEachNode((node) => {
+ if (node === CONFIG.communication.hiddenAlias || node === "server") return;
+ try {
+ graph.dropNode(node);
+ } catch {
+ return;
+ }
+ });
+ break;
+ }
+ const circularLocations = circular(graph);
+ animateNodes(graph, circularLocations, {
+ easing: "quadraticIn",
+ duration: 2000,
+ });
+ renderer.refresh();
+ setSearchQuery(document.querySelector("#searchEntryField").value);
});
async function activateNode(node) {
- state.activeNode = node;
- renderer.refresh();
- if (node)
- if (graph.nodes().includes(node)) {
- camera.animate(renderer.getNodeDisplayData(node), {
- duration: 500,
- });
- }
- if (authPeers.includes(node)) fusedStream.loadCache(node);
+ state.activeNode = node;
+ renderer.refresh();
+ if (node)
+ if (graph.nodes().includes(node)) {
+ camera.animate(renderer.getNodeDisplayData(node), {
+ duration: 500,
+ });
+ }
+ if (authPeers.includes(node)) fusedStream.loadCache(node);
}
async function generateNodeContext(node, isKnownConnection, mouseUpPromise) {
- let contextType;
- let contextTypes = ["#node-context", "#connection-context"];
- if (node === "server" || node === CONFIG.communication.hiddenAlias) return;
- else if (isKnownConnection) contextType = contextTypes[1];
- else contextType = contextTypes[0];
- const context = document.querySelector(contextType);
- await mouseUpPromise;
- context.style.visibility = "visible";
- contextTypes.splice(contextTypes.indexOf(contextType), 1);
- contextTypes.forEach((type) => {
- document.querySelector(type).style.visibility = "hidden";
- });
- context.setAttribute("data-caller", node);
- let offset = 0.1 / camera.ratio;
- context.style.left = 50 + offset + "%";
- context.style.bottom = 50 + offset + "%";
- raceNodeContextDismissEvents(context);
+ let contextType;
+ let contextTypes = ["#node-context", "#connection-context"];
+ if (node === "server" || node === CONFIG.communication.hiddenAlias) return;
+ else if (isKnownConnection) contextType = contextTypes[1];
+ else contextType = contextTypes[0];
+ const context = document.querySelector(contextType);
+ await mouseUpPromise;
+ context.style.visibility = "visible";
+ contextTypes.splice(contextTypes.indexOf(contextType), 1);
+ contextTypes.forEach((type) => {
+ document.querySelector(type).style.visibility = "hidden";
+ });
+ context.setAttribute("data-caller", node);
+ var offset = 0.2 / camera.ratio;
+ context.style.left = `${50 + offset}%`;
+ context.style.bottom = `${50 + offset}%`;
+ raceNodeContextDismissEvents(context);
}
async function raceNodeContextDismissEvents(context) {
- let funcs = {
- windowResizeEvent: async () => {
- nodeDismissHandler.dispatch("windowResize");
- },
- rightClickCanvasEvent: async () => {
- nodeDismissHandler.dispatch("nonContextClick");
- },
- clickContextEvent: async () => {
- nodeDismissHandler.dispatch("contextButtonClicked");
- },
- clickCanvasEvent: async () => {
- nodeDismissHandler.dispatch("nonContextClick");
- },
- clickNodeEvent: async () => {
- nodeDismissHandler.dispatch("nonContextClick");
- },
- rightClickNodeEvent: async () => {
- nodeDismissHandler.dispatch("nonContextClick");
- },
- wheelStageEvent: async () => {
- nodeDismissHandler.dispatch("wheelStage");
- },
- doubleClickStageEvent: async () => {
- nodeDismissHandler.dispatch("nonContextClick");
- },
- };
+ let funcs = {
+ windowResizeEvent: async () => {
+ nodeDismissHandler.dispatch("windowResize");
+ },
+ rightClickCanvasEvent: async () => {
+ nodeDismissHandler.dispatch("nonContextClick");
+ },
+ clickContextEvent: async () => {
+ nodeDismissHandler.dispatch("contextButtonClicked");
+ },
+ clickCanvasEvent: async () => {
+ nodeDismissHandler.dispatch("nonContextClick");
+ },
+ clickNodeEvent: async () => {
+ nodeDismissHandler.dispatch("nonContextClick");
+ },
+ rightClickNodeEvent: async () => {
+ nodeDismissHandler.dispatch("nonContextClick");
+ },
+ wheelStageEvent: async () => {
+ nodeDismissHandler.dispatch("wheelStage");
+ },
+ doubleClickStageEvent: async () => {
+ nodeDismissHandler.dispatch("nonContextClick");
+ },
+ };
- window.onresize = funcs["windowResizeEvent"];
- context.onmouseup = funcs["clickContextEvent"];
- renderer.on("rightClickStage", funcs["rightClickCanvasEvent"]);
- renderer.on("clickStage", funcs["clickCanvasEvent"]);
- renderer.on("wheelStage", funcs["wheelStageEvent"]);
- renderer.on("clickNode", funcs["clickNodeEvent"]);
- renderer.on("rightClickNode", funcs["rightClickNodeEvent"]);
- renderer.on("doubleClickStage", funcs["doubleClickStageEvent"]);
+ window.onresize = funcs["windowResizeEvent"];
+ context.onmouseup = funcs["clickContextEvent"];
+ renderer.on("rightClickStage", funcs["rightClickCanvasEvent"]);
+ renderer.on("clickStage", funcs["clickCanvasEvent"]);
+ renderer.on("wheelStage", funcs["wheelStageEvent"]);
+ renderer.on("clickNode", funcs["clickNodeEvent"]);
+ renderer.on("rightClickNode", funcs["rightClickNodeEvent"]);
+ renderer.on("doubleClickStage", funcs["doubleClickStageEvent"]);
- let resizeCall = nodeDismissHandler.acquireExpectedDispatch(
- "windowResize",
- 60000
- );
- let contextClickCall = nodeDismissHandler.acquireExpectedDispatch(
- "contextButtonClicked",
- 60000
- );
- let nonContextClickCall = nodeDismissHandler.acquireExpectedDispatch(
- "nonContextClick",
- 60000
- );
- let wheelStageCall = nodeDismissHandler.acquireExpectedDispatch(
- "wheelStage",
- 60000
- );
- let raceTimeoutDefault = nodeDismissHandler.acquireExpectedDispatch(
- "defaultRaceResolveTimer",
- 59000
- );
+ let resizeCall = nodeDismissHandler.acquireExpectedDispatch("windowResize", 60000);
+ let contextClickCall = nodeDismissHandler.acquireExpectedDispatch("contextButtonClicked", 60000);
+ let nonContextClickCall = nodeDismissHandler.acquireExpectedDispatch("nonContextClick", 60000);
+ let wheelStageCall = nodeDismissHandler.acquireExpectedDispatch("wheelStage", 60000);
+ let raceTimeoutDefault = nodeDismissHandler.acquireExpectedDispatch("defaultRaceResolveTimer", 59000);
- const promiseConsequence = (_) => {
- context.style.visibility = "hidden";
- activateNode();
- nodeDismissHandler.flushExpectedDispatches();
- window.removeEventListener("resize", funcs["windowResizeEvent"]);
- context.removeEventListener("mouseup", funcs["clickContextEvent"]);
- renderer.removeListener("rightClickStage", funcs["rightClickCanvasEvent"]);
- renderer.removeListener("clickStage", funcs["clickCanvasEvent"]);
- renderer.removeListener("wheelStage", funcs["wheelStageEvent"]);
- renderer.removeListener("clickNode", funcs["clickNodeEvent"]);
- renderer.removeListener("rightClickNode", funcs["rightClickNodeEvent"]);
- renderer.removeListener("doubleClickStage", funcs["doubleClickStageEvent"]);
- };
- Promise.race([
- resizeCall,
- contextClickCall,
- nonContextClickCall,
- wheelStageCall,
- raceTimeoutDefault,
- ]).then(promiseConsequence, promiseConsequence);
+ const promiseConsequence = (_) => {
+ context.style.visibility = "hidden";
+ activateNode();
+ nodeDismissHandler.flushExpectedDispatches();
+ window.removeEventListener("resize", funcs["windowResizeEvent"]);
+ context.removeEventListener("mouseup", funcs["clickContextEvent"]);
+ renderer.removeListener("rightClickStage", funcs["rightClickCanvasEvent"]);
+ renderer.removeListener("clickStage", funcs["clickCanvasEvent"]);
+ renderer.removeListener("wheelStage", funcs["wheelStageEvent"]);
+ renderer.removeListener("clickNode", funcs["clickNodeEvent"]);
+ renderer.removeListener("rightClickNode", funcs["rightClickNodeEvent"]);
+ renderer.removeListener("doubleClickStage", funcs["doubleClickStageEvent"]);
+ };
+ Promise.race([
+ resizeCall,
+ contextClickCall,
+ nonContextClickCall,
+ wheelStageCall,
+ raceTimeoutDefault,
+ ]).then(promiseConsequence, promiseConsequence);
}
-renderer.on("clickNode", async (event) => {
- setTimeout(() => {
- ["successNodes", "errorNodes"].forEach((nodeState) => {
- if (state[nodeState].indexOf(event.node) != -1)
- state[nodeState].splice(state[nodeState].indexOf(event.node), 1);
- });
- renderer.refresh();
- activateNode(event.node);
- if (mod.ALT) {
- if (
- event.node === "server" ||
- event.node === CONFIG.communication.hiddenAlias
- )
- return;
- peerConnection.prototype.negotiateAgnosticAuthConnection(event.node);
- }
- }, 10);
-});
+async function nodeClick(event) {
+ setTimeout(() => {
+ ["successNodes", "errorNodes"].forEach((nodeState) => {
+ if (state[nodeState].indexOf(event.node) != -1)
+ state[nodeState].splice(state[nodeState].indexOf(event.node), 1);
+ });
+ renderer.refresh();
+ activateNode(event.node);
+ if (mod.ALT) {
+ if (event.node === "server" || event.node === CONFIG.communication.hiddenAlias) return;
+ peerConnection.prototype.negotiateAgnosticAuthConnection(event.node);
+ }
+ }, 10);
+ if (recentlyClickedNode === event.node) {
+ rightClickNodeAction(event, true);
+ console.log("test", event);
+ recentlyClickedNode = null;
+ return;
+ }
+ recentlyClickedNode = event.node;
+ var currentClickTimestamp = +new Date();
+ lastClickTimestamp = currentClickTimestamp;
+ setTimeout(() => {
+ if (currentClickTimestamp === lastClickTimestamp && recentlyClickedNode) {
+ currentClickTimestamp = null;
+ recentlyClickedNode = null;
+ }
+ }, 5000);
+}
-renderer.on("clickStage", async (event) => {
- activateNode();
+renderer.on("clickNode", nodeClick);
+
+renderer.on("clickStage", async () => {
+ activateNode();
});
-renderer.on("rightClickNode", async (event) => {
- activateNode(event.node);
- if (event.node === "server") {
- return;
- }
- window.onmouseup = async function () {
- nodeDismissHandler.dispatch("mouseUpMenuSpawnable");
- };
- const contextPromise = nodeDismissHandler.acquireExpectedDispatch(
- "mouseUpMenuSpawnable"
- );
- generateNodeContext(
- event.node,
- authPeers.includes(event.node),
- contextPromise
- );
+async function rightClickNodeAction(event, doubleTapReferred) {
+ activateNode(event.node);
+ if (event.node === "server") {
+ return;
+ }
+ if (doubleTapReferred) {
+ setTimeout(() => {
+ nodeDismissHandler.dispatch("mouseUpMenuSpawnable");
+ }, 0);
+ } else {
+ window.addEventListener("mouseup", async function () {
+ nodeDismissHandler.dispatch("mouseUpMenuSpawnable");
+ });
+ }
+ const contextPromise = nodeDismissHandler.acquireExpectedDispatch("mouseUpMenuSpawnable");
+ generateNodeContext(event.node, authPeers.includes(event.node), contextPromise);
+}
+
+renderer.on("rightClickNode", rightClickNodeAction);
+
+renderer.on("doubleClickNode", async (event) => {
+ event.preventSigmaDefault();
});
window.graphState = state;