File size: 2,141 Bytes
3d5837a
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
import { app } from "../../../scripts/app.js";
import { api } from "../../../scripts/api.js";

// Adds a menu option to toggle follow the executing node
// Adds a menu option to go to the currently executing node
// Adds a menu option to go to a node by type

app.registerExtension({
	name: "pysssss.NodeFinder",
	setup() {
		let followExecution = false;

		const centerNode = (id) => {
			if (!followExecution || !id) return;
			const node = app.graph.getNodeById(id);
			if (!node) return;
			app.canvas.centerOnNode(node);
		};

		api.addEventListener("executing", ({ detail }) => centerNode(detail));

		// Add canvas menu options
		const orig = LGraphCanvas.prototype.getCanvasMenuOptions;
		LGraphCanvas.prototype.getCanvasMenuOptions = function () {
			const options = orig.apply(this, arguments);
			options.push(null, {
				content: followExecution ? "Stop following execution" : "Follow execution",
				callback: () => {
					if ((followExecution = !followExecution)) {
						centerNode(app.runningNodeId);
					}
				},
			});
			if (app.runningNodeId) {
				options.push({
					content: "Show executing node",
					callback: () => {
						const node = app.graph.getNodeById(app.runningNodeId);
						if (!node) return;
						app.canvas.centerOnNode(node);
					},
				});
			}

			const nodes = app.graph._nodes;
			const types = nodes.reduce((p, n) => {
				if (n.type in p) {
					p[n.type].push(n);
				} else {
					p[n.type] = [n];
				}
				return p;
			}, {});
			options.push({
				content: "Go to node",
				has_submenu: true,
				submenu: {
					options: Object.keys(types)
						.sort()
						.map((t) => ({
							content: t,
							has_submenu: true,
							submenu: {
								options: types[t]
									.sort((a, b) => {
										return a.pos[0] - b.pos[0];
									})
									.map((n) => ({
										content: `${n.getTitle()} - #${n.id} (${n.pos[0]}, ${n.pos[1]})`,
										callback: () => {
											app.canvas.centerOnNode(n);
										},
									})),
							},
						})),
				},
			});

			return options;
		};
	},
});