I wrote some code that uses a drag event for svg elements. The code works fine and I would like to write some tests for it. Instead of manually moving my mouse by hand I would like to programmatically trigger the dragstart
and drag
events with selection.dispatch:
svgSelection.dispatch('dragstart',{bubbles:true});
svgSelection.dispatch('drag',{bubbles:true});
However, the events do not seem to be fired. Maybe I use the wrong event keys or need to include some additional option?
I also tried to use following variants that did not help:
svgElement.dispatchEvent(new Event('drag', {bubbles:true}));
svgSelection.dispatch('mousedown.drag',{bubbles:true});
If I try
svgElement.on('mousedown.drag')()
I get an error
Uncaught TypeError: Cannot read property 'ctrlKey' of null
at defaultFilter (drag.js?009f:10)
at mousedowned (drag.js?009f:51)
What is the right way to trigger the events 'start' and 'drag' used in following code?
let offset = [0,0];
let drag = d3.drag()
.on('start', () => this.__dragStarted(d3, svgSelection, offset))
.on('drag', ()=> this.__dragged(d3, svgSelection, offset));
svgSelection.call(drag);
My test example:
it('enableDragAndDrop', ()=>{
let element = document.createElement('svg');
let svgSelection = d3.select(element);
spyOn(sut, '__dragStarted');
spyOn(sut, '__dragged');
sut.enableDragAndDrop(svgSelection);
svgSelection.dispatch('dragstart',{bubbles:true});
svgSelection.dispatch('drag',{bubbles:true});
expect(sut.__dragStarted).toHaveBeenCalled();
expect(sut.__dragged).toHaveBeenCalled();
});
My code example:
enableDragAndDrop(svgSelection){
let offset = [0,0];
let drag = d3.drag()
.on('start', () => this.__dragStarted(d3, svgSelection, offset))
.on('drag', ()=> this.__dragged(d3, svgSelection, offset));
svgSelection.call(drag);
}
__dragStarted(d3, svgSelection, offset){
if(!svgSelection.attr('transform')){
svgSelection.attr('transform','translate(0,0)');
}
let transform = svgSelection.attr('transform');
let translate = this.__extractTranslate(transform);
offset[0] = translate[0] - d3.event.x;
offset[1] = translate[1] - d3.event.y;
}
__dragged(d3, svgSelection, offset){
let x = d3.event.x + offset[0];
let y = d3.event.y + offset[1];
svgSelection
.attr('transform', 'translate(' + x + ', ' + y + ')');
}
__extractTranslate(transformString){
let stripped = transformString;
stripped = stripped.replace('translate(','');
stripped = stripped.replace(')','');
let numberStrings = stripped.split(',');
return numberStrings.map(numberString=>parseFloat(numberString));
}
Related questions:
How to externally trigger d3 events
How to trigger event in JavaScript?
dataTransfer is null when creating drag event programmatically