Styling API
Nodes and edges have multiple visual attributes: position, color, shape, size, image, text, etc (see the full list here). Ogma provides multiple ways to alter these attributes, depending on the need.
Most sections of this tutorial will be illustrated with examples using nodes, but it works exactly the same for edges.
Basics
Assigning attributes
The most simple way to change the attributes of an element (node or edge) is to use the setAttributes
method:
const node = ogma.addNode({ id: 0 });
node.setAttributes({
x: 10,
y: 30,
color: 'blue',
text: { content: 'My node' }
});
It is also possible to specify attributes when adding a node/edge, by providing an attributes
property. Attributes specified this way are called "original attributes".
ogma.addNode({
id: 1,
attributes: { x: 30, y: 30, color: 'red', text: { content: 'My other node' } }
});
Result:
Note: in case of nested attributes, you can often use a shortcut to specify the main "sub-attribute":
// The following two lines are equivalent:
node.setAttributes({ text: { content: 'Some text' } });
node.setAttributes({ text: 'Some text' });
// Here the value of the `text` property is not an object, thus is alias-ed to `text.content`
You can also set a single attribute with setAttribute
(with no 's'):
node.setAttribute('shape', 'square');
node.setAttribute('text.color', 'blue');
// Notice how you specify a nested property name
Retrieving attributes
You can retrieve a single attribute with getAttribute
(without the 's'):
const node = ogma.addNode({ id: 0 });
node.setAttributes({
x: 10,
y: 30,
color: 'blue',
text: { content: 'My node' }
});
console.log(node.getAttribute('color')); // "blue"
To retrieve some (or all) attributes of a node, use the getAttributes
method:
const node = ogma.addNode({ id: 0 });
node.setAttributes({
x: 10,
y: 30,
color: 'blue',
text: { content: 'My node' }
});
console.log(node.getAttributes(['x', 'y', 'text.content']));
// {x: 10, y: 30, text: {content: 'My node'}}
- The method takes the list of attributes to retrieve.
- If no argument is specified, the method will return an object containing all attributes.
- Nested attributes are specified with a dot-separated string.
Note: There is also a node.getPosition()
method that is a shortcut for node.getAttributes(['x', 'y'])
.
Resetting attributes
You can reset attributes to their original value (the ones specified when the node/edge was added) by using the resetAttributes
method:
const node = ogma.addNode({
id: 2,
attributes: {
x: 20,
y: 50,
color: 'green',
text: { content: 'Yet another node' }
}
});
node.setAttributes({ color: 'darkblue', shape: 'square' });
console.log(node.getAttribute('color')); // "darkblue"
console.log(node.getAttribute('shape')); // "square"
node.resetAttributes();
console.log(node.getAttribute('color')); // "green"
console.log(node.getAttribute('shape')); // "circle"
// no original attribute was specified for the "shape" attribute, so the system default value is assigned
Before reset:
After reset:
By default, all attributes are reset but it is possible to specify a subset of attributes to reset:
// Reset the coordinates and the color back to their original value.
// All other attributes are left untouched
node.resetAttributes(['x', 'y', 'color']);
Working with NodeList/EdgeList
All these methods also work on the NodeList/EdgeList objects:
// Color all the nodes in the graph in gold
ogma.getNodes().setAttributes({ color: 'gold' });
When used with a NodeList/EdgeList, the setAttributes
method can also take a list of attributes objects as parameter.
In that case, the first object is assigned to the first element of the list, the second object to the second element, etc.
ogma.getNodes(['My node', 'Yet another node']).setAttributes([
{ x: 0, y: 40, radius: 3, color: 'lightgreen' },
{ x: 5, y: 55, radius: 2, color: 'pink' }
]);
console.log(ogma.getNode('My node').getAttribute('color')); // "lightgreen"
console.log(ogma.getNode('Yet another node').getAttribute('color')); // "pink"
Similarly, calling getAttributes
on a NodeList or EdgeList retrieves a list of objects, and getAttribute
retrieves a list of values:
const nodes = ogma.getNodes(['My node', 'Yet another node']);
nodes.setAttributes([
{ x: 0, y: 40, radius: 3, color: 'lightgreen' },
{ x: 5, y: 55, radius: 2, color: 'pink' }
]);
console.log(nodes.getAttributes(['x', 'y'])); // [{x: 0, y: 0}, {x: 10, y: 20}]
console.log(nodes.getAttribute('color')); // ['red', 'blue']
A very straightforward usage of this feature is the ability to save the state of a list of nodes, and restore it later.
The resetAttributes
method also work with lists:
// Reset the graph to its original state
ogma.getNodes().resetAttributes();
ogma.getEdges().resetAttributes();
Animations
If desired, attribute transitions can be animated. Specify the duration (in milliseconds) and easing as the second parameter of setAttributes
.
const node = ogma.addNode({ id: 3, attributes: { color: 'red' } });
// Animate the color change of the node from red to blue in 500 ms.
node.setAttributes(
{ color: 'blue' },
{ duration: 500, easing: 'quadraticOut' }
);
Global rules
Creating rules
Most often, graphs don't contain any attribute but contain node and edge data. When loaded in Ogma, they look raw because the default style is applied.
ogma.setGraph({
nodes: [
{
id: 0,
data: {
type: 'country',
name: 'USA',
population: 325365189,
currency: 'USD'
},
attributes: { x: 0.1414247453212738, y: 36.075706481933594 }
},
{
id: 1,
data: {
type: 'country',
name: 'France',
population: 66991000,
currency: 'EUR'
},
attributes: { x: -13.736449241638184, y: -31.714202880859375 }
},
{
id: 2,
data: {
type: 'country',
name: 'Germany',
population: 82175700,
currency: 'EUR'
},
attributes: { x: 26.934364318847656, y: -31.93191909790039 }
},
{
id: 3,
data: { type: 'person', name: 'Norwood Preston', age: 44 },
attributes: { x: 6.068506240844727, y: 6.21354866027832 }
},
{
id: 4,
data: { type: 'person', name: 'Kurtis Levi', age: 24 },
attributes: { x: 29.330284118652344, y: 22.172880172729492 }
},
{
id: 5,
data: { type: 'person', name: 'Amias Kev', age: 38 },
attributes: { x: -24.19040298461914, y: 35.77732849121094 }
},
{
id: 6,
data: { type: 'person', name: 'Carver Derren', age: 16 },
attributes: { x: -34.62548065185547, y: -22.868541717529297 }
},
{
id: 7,
data: { type: 'person', name: 'Bevis Tel', age: 73 },
attributes: { x: -22.90767478942871, y: -4.446183681488037 }
},
{
id: 8,
data: { type: 'person', name: 'Loyd Garfield', age: 32 },
attributes: { x: 32.98543167114258, y: -9.278616905212402 }
}
],
edges: [
{ id: 0, source: 3, target: 0, data: { type: 'lives_in' } },
{ id: 1, source: 4, target: 0, data: { type: 'lives_in' } },
{ id: 2, source: 5, target: 0, data: { type: 'lives_in' } },
{ id: 3, source: 6, target: 1, data: { type: 'lives_in' } },
{ id: 4, source: 7, target: 1, data: { type: 'lives_in' } },
{ id: 5, source: 8, target: 2, data: { type: 'lives_in' } },
{ id: 6, source: 3, target: 4, data: { type: 'knows' } },
{ id: 7, source: 3, target: 8, data: { type: 'knows' } },
{ id: 8, source: 3, target: 7, data: { type: 'knows' } },
{ id: 9, source: 4, target: 8, data: { type: 'knows' } },
{ id: 10, source: 6, target: 7, data: { type: 'knows' } }
]
});
Rather than looping through every node, we can define global rules to assign the attributes of the nodes and edges.
ogma.styles.addRule({
nodeAttributes: {
color: 'orange',
text: node => node.getData('name')
},
edgeAttributes: edge => {
return {
opacity: Math.min(
edge.getSource().getAttribute('opacity'),
edge.getTarget().getAttribute('opacity')
),
shape: {
type: 'line',
head: edge.getData('type') === 'owns' ? 'arrow' : null
}
};
}
});
The nodeAttributes
and edgeAttributes
fields are similar to the NodeAttributes
and EdgeAttributes
structure,
except there can be functions at any level. Functions at non-leaf level should return an object containing values for
the nested attributes. As shown in the example, it's even possible to provide a single function that will return the
whole attribute object.
Rules can be based on attributes and data. When an element's (node or edge) data or attribute changes, Ogma recomputes all rules for the node/edge and its adjacent elements. It is possible to specify exactly when a rule should be refreshed with additional options, this is the object of the next tutorial.
Quick note on data
: the data
property is an object whose content is up to the user. It is
manipulable through methods getData
and setData
. See the Data API
tutorial for a more in depth explanation.
Selectors
It is possible to specify a selector to indicate on which nodes/edges the rule should be applied:
ogma.styles.addRule({
nodeAttributes: { color: 'red' },
// The rule will only be applied to nodes for which the data `foo` is equal to `'bar'`
nodeSelector: node => node.getData('foo') === 'bar'
});
Advantages of rules instead of setAttributes
:
- Rules are automatically applied to nodes/edges added to the graph.
ogma.styles.addRule({ nodeAttributes: { color: 'red' } });
const node = ogma.addNode({ id: 0 });
console.log(node.getAttribute('color')); // "red"
- Rules are automatically re-applied whenever there is a relevant change in the graph
ogma.styles.addRule({
nodeAttributes: {
text: node => node.getData('name')
}
});
const node = ogma.addNode({ id: 0, data: { name: 'John' } });
console.log(node.getAttribute('text')); // "John"
node.setData('name', 'James');
console.log(node.getAttribute('text')); // "James"
- Rules are applied after original attributes, but before attributes set via
setAttributes
. This means
that you can build a logic that uses setAttributes
on top of the global rules:
const node = ogma.addNode({
id: 0,
attributes: { color: 'blue' },
data: { name: 'Google', type: 'company' }
});
node.setAttributes({ text: 'Facebook' });
ogma.styles.addRule({
nodeAttributes: {
color: node => {
if (node.getData('type') === 'company') {
return 'red';
}
return 'black';
},
text: node => node.getData('name')
}
});
// The text set by `setAttributes` takes precedence over the rule
console.log(node.getAttribute('text')); // "Facebook"
// The color assigned by the rule takes precedence over the original color
console.log(node.getAttribute('color')); // "red"
ogma.styles.addRule({ nodeAttributes: { color: 'red' } });
const node = ogma.addNode({ id: 0 });
console.log(node.getAttribute('color')); // "red"
ogma.styles.addRule({
nodeAttributes: {
text: node => node.getData('name')
}
});
const node = ogma.addNode({ id: 0, data: { name: 'John' } });
console.log(node.getAttribute('text')); // "John"
node.setData('name', 'James');
console.log(node.getAttribute('text')); // "James"
setAttributes
. This means
that you can build a logic that uses setAttributes
on top of the global rules:const node = ogma.addNode({
id: 0,
attributes: { color: 'blue' },
data: { name: 'Google', type: 'company' }
});
node.setAttributes({ text: 'Facebook' });
ogma.styles.addRule({
nodeAttributes: {
color: node => {
if (node.getData('type') === 'company') {
return 'red';
}
return 'black';
},
text: node => node.getData('name')
}
});
// The text set by `setAttributes` takes precedence over the rule
console.log(node.getAttribute('text')); // "Facebook"
// The color assigned by the rule takes precedence over the original color
console.log(node.getAttribute('color')); // "red"
Note: addRule
returns an object that can be used to manipulate the rule (removing, refreshing or re-ordering it)
Removing rules
Removing a rule is done with the destroy
method:
const node = ogma.addNode({ id: 0, attributes: { color: 'blue' } });
const rule = ogma.styles.addRule({ nodeAttributes: { color: 'red' } });
console.log(node.getAttribute('color')); // "red"
rule.destroy();
console.log(node.getAttribute('color')); // "blue"
Re-ordering rules
Multiple rules can be applied at the same time. Therefore, conflicts may occur:
// Simple case: what color is assigned to the nodes?
ogma.styles.addRule({ nodeAttributes: { color: 'red' } });
ogma.styles.addRule({ nodeAttributes: { color: 'blue' } });
Rules are internally stored in an array: every time a rule is added, it is added at the end of the array. When attributes are computed, rules are applied one by one in the order they appear in the array (by default rules added last are applied last). In this example, the nodes will be blue.
Although not common, it may be necessary to re-order the rules within this array in order to modify the order in which they are applied. In such case, the following methods are useful:
rule.getIndex()
: retrieves the index of the specified rulerule.setIndex(index)
: changes the index of the specified rule
Here is an example that demonstrates swapping the priority of two rules:
function swapRulePriority(rule1, rule2) {
const index1 = rule1.getIndex();
const index2 = rule2.getIndex();
rule1.setIndex(index2);
rule2.setIndex(index1);
}
Refreshing rules
It can happen that a rule references an external variable. If this variable changes, the rule needs to be manually refreshed to take the change into account:
const myMapping = {
person: 'red',
country: 'blue',
company: 'green'
};
const rule = ogma.styles.addRule({
nodeAttributes: {
color: node => myMapping[node.getData('type')]
}
});
// Change a variable referenced by the rule
myMapping.company = 'purple';
// Manually refresh the rule
rule.refresh();
Retrieving all rules
You can retrieve all rules with styles.getRuleList()
:
// This example deletes all the rules
ogma.styles.getRuleList().forEach(rule => rule.destroy());
Classes
Classes are a way to highlight nodes/edges to show they are in a meaningful and temporary state.
Selection and hover
These two classes are managed by Ogma itself, and are not directly accessible by the user. To affect
how the nodes look like when hovered/selected, Ogma provides the setHoveredNodeAttributes
and
setSelectedNodeAttributes
methods:
const node = ogma.addNode({ id: 0 });
ogma.styles.setSelectedNodeAttributes({ outerStroke: { color: 'green' } });
node.setSelected(true);
console.log(node.getAttribute('outerStroke.color')); // "green"
The argument is the same as the nodeAttributes
parameter for the rules, and thus can contain functions:
// In this example, we display the name of the nodes as their text, only when they are selected
const node = ogma.addNode({ id: 0, data: { name: 'John' } });
ogma.styles.setSelectedNodeAttributes({
outerStroke: {
color: 'green'
},
text: node => node.getData('name')
});
console.log(node.getAttribute('text')); // null
node.setSelected(true);
console.log(node.getAttribute('text')); // "John"
You can also pass null
so the attributes of nodes are not changed when hovered/selected:
ogma.styles.setSelectedNodeAttributes(null);
Creating custom classes
- Creating a class is done via the
ogma.styles.createClass
method.
- Adding and removing a class from a node/edge is done via the
addClass
and removeClass
methods.
- It is possible to do operations on the class object directly (such as retrieving the list of nodes/edges that have the class)
ogma.styles.createClass
method.addClass
and removeClass
methods.In this example, when the user clicks on a node we highlight the shortest path between a pre-determined node and the clicked node:
// Define the class, and the attributes associated to it
const myClass = ogma.styles.createClass({
name: 'shortestPathHighlight',
nodeAttributes: { outerStroke: { color: 'black', width: 5 } },
edgeAttributes: { color: 'black' }
});
// Remove the style applied to nodes when they are selected
ogma.styles.setSelectedNodeAttributes(null);
ogma.events.on('click', evt => {
// This is the node that will be used to compute the shortest path
const sourceNode = ogma.getNode(0);
// When the user clicks, first clear the current highlighted shortest path, if any
myClass.clearNodes();
myClass.clearEdges();
// If a node was right clicked, highlight the shortest path
if (evt.target && evt.target.isNode) {
const clickedNode = evt.target;
ogma.algorithms
.shortestPath({
source: sourceNode,
target: clickedNode
})
.then(shortestPath => {
// `shortestPath` is null if no path exists
if (shortestPath) {
// Highlight all the nodes and edges in the shortest path
shortestPath.nodes.addClass('shortestPathHighlight');
shortestPath.edges.addClass('shortestPathHighlight');
}
});
}
});
// Finally, load a graph on which to test the feature
ogma.generate
.grid({ rows: 5, columns: 5 })
.then(graph => ogma.setGraph(graph))
.then(() => {
// Remove a few edges to make things more interesting
return ogma.removeEdges([0, 2, 3, 5, 7, 10, 13, 14, 18, 19, 22]);
})
.then(() => ogma.view.locateGraph());
A few things to note:
- Classes are applied after individual attributes and rules.
- Classes are applied using the same mechanism as rules: the class added last is applied last
- Builtin classes (selection and hover) are always applied last, after user classes
Summary
Attributes are computed in the following order:
- original attributes
- rules (from first added to last added)
- individual attributes (that are assigned with
setAttributes
) - custom classes (from first added to last added)
- builtin classes (selection and hover)
It is especially important to keep this order in mind when defining rules/classes that are computed using other attributes.
Performance notes
When working with a large number or nodes/edges:
- Avoid using functions in rules/classes; use constants as much as possible.
- Prefer using one single method on a NodeList/EdgeList rather that the same method on each
node/edge individually (e.g
nodeList.setAttributes({color: 'red'})
rather thannodeList.forEach(node => node.setAttributes({color: 'red'}))
).
The next tutorial is more advanced and shows how to optimize rules and classes by telling Ogma what attributes they depend on.
Themes
You can use theme presets for all styling at once. We have compiled several styling themes for Ogma in our public @linkurious/ogma-styles repository. These themes are applied before all the styling rules and are a good way of producing a global stylesheet that expresses the "look and feel" of your visualisation
import Ogma from '@linkurious/ogma';
import { midsummerNight as theme } from '@linkurious/ogma-styles'
const ogma = new Ogma({ ... });
ogma.styles.setTheme(theme);
Best Practices
Check out our best practices tutorial to learn more about how to make your styles fast and easy to maintain.