🎉 JointJS has new documentation! 🥳
Validator runs a set of callbacks to determine if a command is valid. This is useful for checking if a certain action in your application does lead to an invalid state of the diagram (from your application specific perspective). You can not only cancel such a command but e.g. also give the user a hint that this action is not allowed).
Include joint.dia.command.js and joint.dia.validator.js to your HTML:
<script src="joint.dia.command.js"></script>
<script src="joint.dia.vaidator.js"></script>const graph = new joint.dia.Graph({}, { cellNamespace: joint.shapes });
const paper = new joint.dia.Paper({
    el: document.getElementById('paper'),
    width: 500,
    height: 500,
    model: graph,
    cellViewNamespace: joint.shapes
});
const commandManager = new joint.dia.CommandManager({ graph });
const validator = new joint.dia.Validator({ commandManager });
validator.validate('remove',
    function(err, command, next) {
        if (command.data.type === 'standard.Rectangle') return next('Rectangles cannot be removed.');
        return next();
    },
    function(err, command, next) {
        if (err) console.log(err);
        return next(err);
    }
);(Those that know the great ExpressJS application framework might have recognized a similar API to what ExpressJS provides for registering URL route handlers.)
The joint.dia.Validator constructor takes two parameters.
The CommandManager the validator listens to.
Determine whether to cancel an invalid command or not. If set to false, only the invalid event is triggered.
The default is true.
Function validate(action [, callback]*) registers callbacks for a given action.
 The validator listens to commands added to the CommandManager and runs this set of callbacks registered for the action.
If the last callback returns an error, the command is canceled (see joint.dia.CommandManager.cancel()).
This behaviour can be suppressed by setting the cancelInvalid to false in options passed to the Validator constructor.
validate("change:source change:target", function() { .. });callback(err, command, next)The validation function. An arbitrary number of validation functions can be passed to validate(). Every callback has the following signature:
Where err is an error from the previous callback.
The command parameter is a record from the CommandManager stack. It holds information about an action in the graph.
See below for the structure of a command. This command says that a standard.Rectangle element has been moved by 50px down.
{
  "action": "change:position"
  "data": {
    "id": "e0552cf2-fb05-4444-a9eb-3636a4589d64", // id of the cell it was manipulated with
    "type": "standard.Rectangle", // type of the cell
    // change:attribute events always have these two attributes filled
    "previous": { "position": { "x": 50, "y": 50 }},
    "next": { "position": { "x": 50, "y": 100 }},
  },
  "batch": true, // is command part of a batch command?
  "options": {} // mvc options that are passed through the listeners when passed as the last argument to the mvc Model set() method.
}The add and remove actions have empty previous and next objects. They hold all the
cell attributes in the attributes object so that the CommandManager is able to reconstruct the whole cell if it has to.
See below for an example of such a command structure:
{
  "action": "add",
  "data": {
    "id": "28de715b-62a7-4130-a729-1bcf7dbb1f2b",
    "type": "standard.Ellipse",
    // empty attributes
    "previous": {},
    "next": {},
    // all cells attributes
    "attributes": {
      "type": "standard.Ellipse",
      "size": {
        "width": 50,
        "height": 30
      },
      "position": { "x": 1220, "y": 680 },
      "id": "28de715b-62a7-4130-a729-1bcf7dbb1f2b",
      "attrs": {
        "body": {
          "fill": "red"
        }
      }
    }
  }
  "batch": true,
  "options": {
    "add": true,
    "merge": false,
    "remove": false
  }
}The next parameter is a function accepting one argument - an error passed to the next callback in a row.
next() function means going to the next callback in the order passed to the validate() method.
If a call to the next() function is omitted, the validation for the specified action stops.