GraphXR API Reference
Camera
flyToCenter
Fly the camera to the center of a slice of nodes, optionally with an offset, optionally with a custom duration or tween function @synonyms centerTo
gxr.flyToCenter();
gxr.flyToCenter(gxr.nodes().slice(0, 10).ids())
gxr.flyToCenter(gxr.nodes().slice(0, 10).ids(), { duration: 0, offset: -3 }));
flyToPosition
Fly the camera to a position, optionally with an offset, optionally with a custom duration or tween function
const position = gxr.nodes({name: "Flo"}).at(0).position;
gxr.flyToPosition(position); // no offset
gxr.flyToPosition(position, {offset: -3}); // with an offset
gxr.flyToPosition(position, {duration: 0}); // instantly
flyOut
Zoom out until all nodes are visible
gxr.flyOut();
setCameraOptions
gxr.setCameraOptions({
hideAxes: true,
rotating: true,
speed: 0.5,
});
setCameraRotating
gxr.setCameraRotating(true);
Captions
captionNodesByProperties
gxr.captionNodesByProperties({category: "Person", properties: ["name", "age"]});
Events
onGraphDataUpdate
gxr.onGraphDataUpdate(() => console.log("Graph data updated"));
onSelect
gxr.onSelect((event) => console.log("Selected nodes/edges", event));
Navigation
setCurrentPanel
gxr.setCurrentPanel("algorithm");
gxr.setCurrentPanel("algorithm", "centrality");
gxr.setCurrentPanel("Grove");
As a convenience, you can pass the name of the extension as the first argument
instead of the extension's id, because extension ids are random and hard to remember.
setSubTab
gxr.setSubTab("centrality");
Graph
add
Add nodes and edges to the graph.
gxr.add("A") -> Node with id "A"
gxr.add("A", "B") -> Edge
gxr.add(["A", "B"]) -> Node[]
gxr.add({ id: "A", category: "Person", properties: {name: "Flo"} }) -> Node
gxr.add([{ id: "A", category: "Person", properties: {name: "Flo"} }]) -> Node[]
gxr.add({ sourceId: "A", targetId: "B" }) -> Edge
gxr.add([{ sourceId: "A", targetId: "B" }]) -> Edge[]
gxr.add([{ source: {...}, edge: {...}, target: {...} }], {...}) -> Edge[] // mergeRelationships
gxr.add({ name: "Flo" }, { category: "Person" }) -> Node
gxr.add([{ name: "Flo" }], { category: "Person" }) -> Node[]
gxr.add([{ name: "Flo" }]) -> Node[]
gxr.add({ id: "A" }, { id: "B" }) -> Edge
gxr.add({ id: "A" }, { id: "B" }, { relationship: "LINK" }) -> Edge with relationship "LINK"
addNode
Add node to the graph.
gxr.addNode({ id: "A", category: "Person", properties: {name: "Flo"} });
clear
Clear the graph
gxr.clear();
edges
gxr.edges().forEach(console.log);
gxr.edges().property('since', 2023);
gxr.edges().property('since', () => Math.random() * 3000);
gxr.edges().style('width', 10)
gxr.edges({relationship: 'LINKS'}).style('width', 10);
gxr.edges({properties: {since: 2023}}).style('width', 10);
getCategoryConfig
gxr.getCategoryConfig('Person');
getNode
gxr.getNode("A")
getRelationshipConfig
gxr.getRelationshipConfig("RELATED_TO");
graph
// Add a node with id "A"
gxr.graph().add("A");
// Select nodes in shortest path from A -> B, including A and B
gxr.graph().shortestPath("A", "B").nodes().select()
// Generate a random graph, run degree algorithm, and then run ego layout for a duration of 1 second.
gxr
.graph()
.clear()
.generate()
.degree()
.ego(
{
properties: { degree: (value) => value >= 3 },
depth: 5,
mode: "tree",
orientation: "down",
edgeLength: 0.5,
sortByProperty: "degree",
},
{ duration: 1000 }
);
makeGraph
gxr.makeGraph().add("A").add("B").add("C").add("A", "B").add("B", "C").add("C", "A");
nodes
gxr.nodes().forEach(console.log);
gxr.nodes().property('age', 24);
gxr.nodes().property('age', () => Math.random() * 100);
gxr.nodes({category: "Person"}).style('selected', true);
gxr.nodes({properties: {age: 24}}).style('selected', true);
gxr.nodes({properties: {age: (age) => age > 24}}).style('selected', true);
randomGraph
gxr.randomGraph()
gxr.randomGraph({nodeCount: 100, edgeCount: 200, categories: ["Person", "Company"], relationships: ["WORKS_FOR", "KNOWS"]});
select
gxr.select('age > 30')
Layout
alignBy
Given one of x, y, or z, find the center of that dimension and align all Nodes to that center on that dimension only.
gxr.alignBy({
dimension: 'x',
})
circle
All the nodes spread on a circle
gxr.circle();
cube
All the nodes spread on a cube.
gxr.cube();
distributionBy
Equally space Nodes on one of the x, y, or z axes while keeping the other two dimensions constant. Optionally bin the Nodes by a property value. Optionally scale the spacing by a "spread" factor (the higher the spread, the farther apart).
// Distribute nodes on the x-axis by a property
gxr.distributionBy('similarity')
// Order nodes by a date property and spread them out
gxr.distributionBy({
bin: 'episodeAirDate',
spread: 6,
})
// Equally space Nodes on the x-axis
gxr.distributionBy({
dimension: 'x', // y, or z
})
// Equally space Nodes on the x-axis and bin the Nodes by "seasonNumber"
gxr.distributionBy({
bin: 'seasonNumber',
dimension: 'x',
})
// Equally space Nodes on the x-axis and bin the Nodes by "seasonNumber" in descending order
gxr.distributionBy({
bin: 'seasonNumber',
dimension: 'x',
reverse: true,
})
ego
Ego reveals hierarchal data by arranging nodes in a tree, where a node's depth in the tree is equal to the length of the shortest path to the node. The tree projects linearly in one direction, or radially around the root[s]. We must select the root nodes before calling ego.
gxr.nodes("a").ego({
depth: 3, // maximum depth of the tree; default 100
edgeLength: 1 // visual length of the edges in the tree; default 0.2
mode: 'rings' // 'tree' or 'rings'; default 'tree'
orientation: 'down' // 'up', 'down', 'left', or 'right'; default 'right'
sortByProperty: 'ComponentName' // arrange child nodes in ascending order by a property
[sortByProperty: sortByProperty({property: "ComponentName", ascending: false})] // same as above, but descending
}))
grid
Equally space Nodes on the x-axis, y-axis, z=0.
gxr.grid();
line
Equally space Nodes on the x-axis, y=0, z=0.
gxr.line();
parametric
Map the x, y, and/or z dimensions to the range [-2, 2] by applying a linear scale to all or a subset of the domain of a property.
If a dimension is omitted, it is flattened to align with the grid.
// Line up Nodes on the x-axis, ordered by seasonNumber.
// Also set y and z to 0
gxr.parametric({
x: 'seasonNumber',
y: 0,
z: 0,
})
// Line up Nodes on the x-axis, ordered by seasonNumber descending. Align y and z to the grid.
gxr.parametric({
x: 'seasonNumber',
reverse: true,
})
// Linearly scale seasonNumber, episodeNumber, and millionViewers on the x, y, and z dimensions respectively
gxr.parametric({
x: 'seasonNumber',
y: 'episodeNumber',
z: 'millionViewers',
})
rotate
Given one of the x, y, or z dimensions, find the center point of all the Nodes and rotate all Nodes around the axis passing through the center point and lying on the dimension given.
gxr.rotate({
dimension: 'x',
theta: 90,
})
gxr.rotate({
dimension: 'z',
theta: 45
})
scale
Scale the distance of each node from a computed center by a constant factor.
gxr.scale({
x: 2,
y: 2,
z: 2,
})
scatter
Set each node to a random point.
gxr.scatter()
shift
Add a constant vector to the position of each node.
gxr.shift({
x: 1,
})
toast
Get the toast function
gxr.toast("Hello, world!");
gxr.toast().info("Hello, world!");
gxr.toast().success("Hello, world!");
gxr.toast().error("Hello, world!");
gxr.toast().warn("Hello, world!");
await gxr.toast().promise(fetch(), "Hello, world!");
setParametricAxesOptions
gxr.setParametricAxesOptions({
showAxes: false,
})
spiral
All the nodes spread on a spiral
gxr.spiral()
forceLayout
Run a force layout on the graph.
// Run force layout
gxr.forceLayout();
Neo4j
neo4j
nodes(): IterableNodes;
gxr.neo4j("MATCH (n) RETURN n LIMIT 10");
Settings
setTipsEnabled
gxr.setTipsEnabled(false);
setAutoShowImage
gxr.setAutoShowImage(true);
setEdgeScale
gxr.setEdgeScale(0.5);
setEdgeArrowVisibility
gxr.setEdgeArrowVisibility(true);
setBlendEdge
gxr.setBlendEdge(false);
setAutoCaption
gxr.setAutoCaption(true);
setFullscreen
gxr.setFullscreen(true);
setPinIconVisible
gxr.setPinIconVisible(false);
setAlternateCaption
gxr.setAlternateCaption(false);
setDashline
gxr.setDashline(false);
setTheme
gxr.setTheme("light")
gxr.setTheme("dark")
setUseCurveLine
//use curve line
gxr.setUseCurveLine(true)
//use straight line
gxr.setUseCurveLine(false)
setRelationshipNameVisibility
gxr.setRelationshipNameVisibility(true)
setGraphMode
gxr.setGraphMode("2D")
setCaptionScale
gxr.setCaptionScale(0.5)
Styles
colorNodesByProperty
gxr.colorNodesByProperty('age');
gxr.colorNodesByProperty({property: 'age'});
gxr.colorNodesByProperty({property: 'age', scale: 'BuGn'});
getIcons
Get a list of available icons
const icons = await gxr.getIcons();
getIconByName
Get an icon by name
const icon = await gxr.getIconByName("person");
setCategoryColor
Set a category's color by hex
gxr.setCategoryColor("Person", "#00ff00");
setCategoryIcon
Set a category's icon
const icon = await gxr.getIconByName("person");
gxr.setCategoryIcon("Person", icon);
setCategoryIconByName
Set a category's icon by name
gxr.setCategoryIconByName("Person", "person");
setCategoryVisibility
gxr.setCategoryVisibility("Person", false);
setRelationshipColor
gxr.setRelationshipColor("KNOWS", "#00ff00");
setRelationshipVisibility
gxr.setRelationshipVisibility("KNOWS", false);
sizeNodesByProperty
gxr.sizeNodesByProperty({category: "Person", property: 'age'});
traceNeighbor
gxr.traceNeighbor()
Transform
aggregate
Pull data to root nodes from their neighborhoods up to a certain depth.
gxr.aggregate({
formula: 'sum',
property: "age",
})
gxr.aggregate({
formula: 'concatenate',
separator: ",",
property: "name",
along: "KNOWS",
startNodeId: "A",
[depth: 3], // optional. Default is 1
})
gxr.aggregate({
// Formula can also be a function
formula: (values) => values.reduce((a, b) => a + b, 0),
...
}))
extract
gxr.extract({
sourceCategory: "Episodes",
targetCategory: "Season",
props: [
{
name: "seasonNumber",
isKey: true,
},
{
name: "millionViewers",
},
{
name: "episodeAirDate",
},
],
relationship: "IN_SEASON",
})
link
Create new edges between nodes which have the same value for the specified property.
gxr.link({ sourceProperty: "id", targetProperty: "id" })
gxr.link({
sourceCategory: "Roles",
sourceProperty: "ParentName",
targetCategory: "Roles",
targetProperty: "ComponentName",
relationship: "IS_PARENT_OF"
})
merge
Combine nodes or edges which have equivalent key properties.
gxr.merge({
keys: ['seasonNumber'],
})
gxr.merge({
category: "Episode",
keys: ['seasonNumber'],
})
// Combine all emails between two persons sent on the same day (thus removing directionality)
gxr.merge({
relationship: "SENT_MAIL_TO"
keys: ['sendDate'],
})
// Combine all emails between two persons sent on the same day and preserve directionality
gxr.merge({
relationship: "SENT_MAIL_TO"
keys: ['sendDate'],
directional: true,
})
mergeNodes
Create new nodes, or update existing nodes, based on the specified key properties.
gxr.mergeNodes({
data: [{ name: "Flo", age: 32 }],
category: "Person",
keys: ["name"]
})
mergeRelationships
Create new edges, or update existing edges, based on the specified key properties.
gxr.mergeRelationships({
data: [
{ source: {name: "Flo"}, edge: {since: 2010}, target: {name: "Aiden"} },
{ source: {name: "Flo"}, edge: {since: 2010}, target: {name: "Georgio"} },
],
source: { category: "Person", keys: ["name"] },
edge: { relationship: "KNOWS", keys: ["SINCE"] },
target: { category: "Person", keys: ["name"] },
})
quickInfoEnabled
Enables/disables quick info
gxr.quickInfoEnabled(true);
showQuickInfo
Shows the quick info panel for the given node.
gxr.showQuickInfo(gxr.nodes().at(0));
hideQuickInfo
Hides the quick info panel
gxr.showQuickInfo(gxr.nodes().at(0));
shortcut
Convert A -> B -> C to A -> C
gxr.shortcut({
incoming: 'IS_PARENT_OF',
center: 'Node',
outgoing: 'IS_PARENT_OF',
shortcut: "IS_GRANDPARENT_OF",
directional: true,
aggregate: [
{
sourceProperty: "age",
targetProperty: "averageAge",
formula: "average",
},
],
countLinks: false,
})
Traversal
traverse
for (const {node, edge, depth} of gxr.traverse({ startNodeId: "A", depth: 3 })) {
console.log(node, edge, depth);
}
UI
addContextMenuItem
Add a custom menu item to the context menu.
gxr.addContextMenuItem({
name: "Find Similar",
text: "Find Similar",
icon: "iconfont icon-similar",
hide: () => false,
enable: () => true,
action: (name, nodeId, edgeId, props) => {
console.log("Finding similar to", name, nodeId, edgeId, props);
},
})
Utility
dispatchGraphDataUpdate
Force the UI to update. Sometimes necessary when using the API to update the graph.
gxr.dispatchGraphDataUpdate();
randomId
gxr.randomId() -> "lijfsleifjzlse";
sleep
gxr.sleep(1000);
snapshot
Get the graph as JSON, or apply a JSON snapshot to the current graph.
// Get snapshot
const mySnapshot = gxr.snapshot()
// Apply snapshot
gxr.snapshot(mySnapshot)
vector
gxr.vector() -> Vector3 {x: 0, y: 0, z: 0}
gxr.vector(1) -> undefined
gxr.vector(1, 2) -> Vector2 {x: 1, y: 2}
gxr.vector({x: 1, y: 2, z: 3}) -> Vector3 {x: 1, y: 2, z: 3}
gxr.vector([1, 2, 3]) -> Vector3 {x: 1, y: 2, z: 3}
gxr.vector(1, 2, 3) -> Vector3 {x: 1, y: 2, z: 3}
gxr.vector({x: 1, y: 2, z: 3, w: 4}) -> Vector4 {x: 1, y: 2, z: 3, w: 4}
User
getCurrentUser
const currentUser = gxr.getCurrentUser();
console.log(currentUser);