/*
* Copyright 1998-2026 by Northwoods Software Corporation. All Rights Reserved.
*/
/*
* This is an extension and not part of the main GoJS library.
* The source code for this is at extensionsJSM/RealtimeDragSelectingTool.ts.
* Note that the API for this class may change with any version, even point releases.
* If you intend to use an extension in production, you should copy the code to your own source directory.
* Extensions can be found in the GoJS kit under the extensions or extensionsJSM folders.
* See the Extensions intro page (https://gojs.net/latest/intro/extensions.html) for more information.
*/
import * as go from 'gojs';
/**
* The RealtimeDragSelectingTool class lets the user select and deselect Parts within the {@link go.DragSelectingTool.box}
* during a drag, not just at the end of the drag.
*
* If you want to experiment with this extension, try the Realtime Drag Selecting sample.
* @category Tool Extension
*/
export class RealtimeDragSelectingTool extends go.DragSelectingTool {
private _originalSelection: go.Set;
private _temporarySelection: go.Set;
constructor(init?: Partial) {
super();
this.name = 'RealtimeDragSelecting';
this._originalSelection = new go.Set();
this._temporarySelection = new go.Set();
if (init) Object.assign(this, init);
}
/**
* Remember the original collection of selected Parts.
*/
override doActivate(): void {
super.doActivate();
// keep a copy of the original Set of selected Parts
this._originalSelection = this.diagram.selection.copy();
// these Part.isSelected may have been temporarily modified
this._temporarySelection.clear();
this.diagram.raiseDiagramEvent('ChangingSelection');
}
/**
* Release any references to selected Parts.
*/
override doDeactivate(): void {
this.diagram.raiseDiagramEvent('ChangedSelection');
this._originalSelection.clear();
this._temporarySelection.clear();
super.doDeactivate();
}
/**
* Restore the selection which may have been modified during a drag.
*/
override doCancel(): void {
const orig = this._originalSelection;
orig.each((p) => (p.isSelected = true));
this._temporarySelection.each((p) => {
if (!orig.has(p)) p.isSelected = false;
});
super.doCancel();
}
/**
* Select Parts within the bounds of the drag-select box.
*/
override doMouseMove(): void {
if (this.isActive) {
super.doMouseMove();
this.selectInRect(this.computeBoxBounds());
}
}
/**
* Select Parts within the bounds of the drag-select box.
*/
override doKeyDown(): void {
if (this.isActive) {
super.doKeyDown();
this.selectInRect(this.computeBoxBounds());
}
}
/**
* Select Parts within the bounds of the drag-select box.
*/
override doKeyUp(): void {
if (this.isActive) {
super.doKeyUp();
this.selectInRect(this.computeBoxBounds());
}
}
/**
* For a given rectangle, select Parts that are within that rectangle.
* @param r - rectangular bounds in document coordinates.
*/
override selectInRect(r: go.Rect): void {
const diagram = this.diagram;
const orig = this._originalSelection;
const temp = this._temporarySelection;
const e = diagram.lastInput;
const found = diagram.findPartsIn(r, this.isPartialInclusion);
if (e.control || e.meta) {
// toggle or deselect
if (e.shift) {
// deselect only
temp.each((p: go.Part) => {
if (!found.has(p)) p.isSelected = orig.has(p);
});
found.each((p: go.Part) => {
p.isSelected = false;
temp.add(p);
});
} else {
// toggle selectedness of parts based on _originalSelection
temp.each((p: go.Part) => {
if (!found.has(p)) p.isSelected = orig.has(p);
});
found.each((p: go.Part) => {
p.isSelected = !orig.has(p);
temp.add(p);
});
}
} else if (e.shift) {
// extend selection only
temp.each((p: go.Part) => {
if (!found.has(p)) p.isSelected = orig.has(p);
});
found.each((p: go.Part) => {
p.isSelected = true;
temp.add(p);
});
} else {
// select found parts, and unselect all other previously selected parts
temp.each((p: go.Part) => {
if (!found.has(p)) p.isSelected = false;
});
orig.each((p: go.Part) => {
if (!found.has(p)) p.isSelected = false;
});
found.each((p: go.Part) => {
p.isSelected = true;
temp.add(p);
});
}
}
}