Ошибка при копировании / вставке функции масштабирования в директиву
Я пытался преобразовать простой javascript для масштабирования пространства GoJS в угловую директиву.
Вот мой плункер
Когда я пытаюсь скопировать / вставить функцию масштабирования в директиву, появляется сообщение об ошибке:
"Ошибка: значение Diagram.commandHandler не является экземпляром CommandHandler: [объект объекта]" (Пожалуйста, проверьте ошибку в консоли Plunkr)
Ошибка в строке № 61
Мой код идет здесь:
app.js
var app = angular.module('app', []);
app.directive('goDiagram', function($http) {
return {
restrict: 'E',
template: '<div></div>',
replace: true,
scope: {
model: '=goModel'
},
link: function(scope, element, attrs) {
if (window.goSamples) goSamples(); // init for these samples -- you don't need to call this
var $ = go.GraphObject.make;
var rainbow = $(go.Brush, "Linear", {
0: "red",
1: "green"
});
var diagram = $(go.Diagram, element[0], {
nodeTemplate: $(go.Node, "Auto", {
locationSpot: go.Spot.Center
}, {
width: 120,
height: 15,
locationSpot: go.Spot.Center
},
new go.Binding("location"),
$(go.Shape, {
fill: "#e74c3c",
stroke: '#c0392b'
}, {
portId: "",
cursor: "pointer",
strokeWidth: 0,
}),
$(go.TextBlock, {
margin: 0,
stroke: "#eee"
},
new go.Binding("text", "key")
)
),
linkTemplate: $(go.Link, {
// routing: go.Link.AvoidsNodes,
reshapable: true,
resegmentable: true
},
$(go.Shape, {
strokeWidth: 3,
stroke: rainbow
},
// new go.Binding("stroke", rainbow),
),
$(go.Shape, {
toArrow: "Standard"
}),
),
// For Zoom
initialContentAlignment: go.Spot.Center, // center the content
commandHandler: $(SpacingCommandHandler),
// update the SpacingCommandHandler.space from the model at the end of each transaction
"ModelChanged": function(e) {
if (e.isTransactionFinished) {
myDiagram.commandHandler.space = myDiagram.model.modelData.space;
}
},
"undoManager.isEnabled": true // enable undo & redo
});
// Conversion functions -- these only work with myDiagram, assuming it uses a SpacingCommandHandler
function spacedLocationParse(str) {
var cmd = myDiagram.commandHandler;
if (!(cmd instanceof SpacingCommandHandler)) throw new Error("not using SpacingCommandHandler");
var pt = go.Point.parse(str);
pt.x = (pt.x - cmd.spaceCenter.x) * cmd.space + cmd.spaceCenter.x;
if (cmd.isYSpaced) {
pt.y = (pt.y - cmd.spaceCenter.y) * cmd.space + cmd.spaceCenter.y;
}
return pt;
}
function spacedLocationStringify(pt, data) {
var cmd = myDiagram.commandHandler;
if (!cmd._isUpdating) {
pt = pt.copy();
pt.x = (pt.x - cmd.spaceCenter.x) / cmd.space + cmd.spaceCenter.x;
if (cmd.isYSpaced) {
pt.y = (pt.y - cmd.spaceCenter.y) / cmd.space + cmd.spaceCenter.y;
}
return go.Point.stringify(pt);
} else {
return data.loc;
}
}
// The custom CommandHandler that avoids changing the Diagram.scale
function SpacingCommandHandler() {
go.CommandHandler.call(this);
this._space = 1.0; // replaces Diagram.scale; also copied to/from Model.modelData.space
this._spaceCenter = new go.Point(0, 0); // not currently used -- should this be saved on modelData too?
this._isYSpaced = true; // scale Y along with X? This option is just for demonstration purposes.
this._isUpdating = false;
}
go.Diagram.inherit(SpacingCommandHandler, go.CommandHandler);
// Overrides of commands that scale the diagram -- change the space instead
/** @override */
SpacingCommandHandler.prototype.decreaseZoom = function(factor) {
if (factor === undefined /*notpresent*/ ) factor = 1.0 / this.zoomFactor;
this.setSpace(this.space * factor);
};
/** @override */
SpacingCommandHandler.prototype.canDecreaseZoom = function(factor) {
if (factor === undefined /*notpresent*/ ) factor = 1.0 / this.zoomFactor;
return this.checkSpace(this.space * factor);
};
/** @override */
SpacingCommandHandler.prototype.increaseZoom = function(factor) {
if (factor === undefined /*notpresent*/ ) factor = 1.0 / this.zoomFactor;
this.setSpace(this.space / factor);
};
/** @override */
SpacingCommandHandler.prototype.canIncreaseZoom = function(factor) {
if (factor === undefined /*notpresent*/ ) factor = 1.0 / this.zoomFactor;
return this.checkSpace(this.space / factor);
};
/** @override */
SpacingCommandHandler.prototype.resetZoom = function(newspace) {
if (newspace === undefined /*notpresent*/ ) newspace = 1.0;
this.setSpace(newspace);
};
/** @override */
SpacingCommandHandler.prototype.canResetZoom = function(newspace) {
return this.checkSpace(newspace);
};
// actually set a new value for SPACE
SpacingCommandHandler.prototype.setSpace = function(s) {
this.space = Math.max(0.1, Math.min(10.0, s));
};
// validity check for a new value for SPACE
SpacingCommandHandler.prototype.checkSpace = function(s) {
return 0.1 <= s && s <= 10.0;
};
// Properties for SpacingCommandHandler
Object.defineProperty(SpacingCommandHandler.prototype, "space", {
get: function() {
return this._space;
},
set: function(val) {
if (val !== this._space) {
this._space = val;
var diagram = this.diagram;
if (diagram !== null) { // store in model too, and support undo
diagram.model.setDataProperty(diagram.model.modelData, "space", val);
}
this.updateAllLocations();
// update the page showing the current value
document.getElementById("SPACE").textContent = val.toString();
}
}
});
Object.defineProperty(SpacingCommandHandler.prototype, "spaceCenter", {
get: function() {
return this._spaceCenter;
},
set: function(val) {
if (!val.equals(this._spaceCenter)) {
this._spaceCenter = val.copy();
}
}
});
Object.defineProperty(SpacingCommandHandler.prototype, "isYSpaced", {
get: function() {
return this._isYSpaced;
},
set: function(val) {
if (val !== this._isYSpaced) {
this._isYSpaced = val;
this.updateAllLocations();
}
}
});
// If the spacing or isYSpaced properties change value,
// we need to update the effective locations of all nodes.
// Assume Node.location is data bound to "loc" property.
SpacingCommandHandler.prototype.updateAllLocations = function() {
var diagram = this.diagram;
if (diagram === null) return;
this._isUpdating = true;
diagram.skipsUndoManager = true;
diagram.startTransaction("respace nodes");
diagram.parts.each(function(p) {
p.updateTargetBindings("loc");
});
diagram.nodes.each(function(n) {
n.updateTargetBindings("loc");
});
diagram.commitTransaction("respace nodes");
diagram.skipsUndoManager = false;
this._isUpdating = false;
};
// end SpacingCommandHandler class
function updateAngular(e) {
if (e.isTransactionFinished) {
scope.$apply();
}
}
function updateSelection(e) {
diagram.model.selectedNodeData = null;
var it = diagram.selection.iterator;
while (it.next()) {
var selnode = it.value;
// ignore a selected link or a deleted node
if (selnode instanceof go.Node && selnode.data !== null) {
diagram.model.selectedNodeData = selnode.data;
break;
}
}
scope.$apply();
}
// watch scope
scope.$watch("model", function(newmodel) {
if (newmodel != undefined) {
var oldmodel = diagram.model;
if (oldmodel !== newmodel) {
diagram.removeDiagramListener("ChangedSelection", updateSelection);
diagram.model = newmodel;
diagram.addDiagramListener("ChangedSelection", updateSelection);
}
}
});
}
}
});
app.controller('appController', function($scope) {
$scope.init = function(d) {
$scope.hello = "Hello Plunker!!!!";
$scope.model = new go.GraphLinksModel(
[{
key: "Alpha",
color: "lightblue",
location: new go.Point(150, 130),
loc: "150 130"
}, {
key: "Beta",
color: "orange",
location: new go.Point(350, 180),
loc: "350 180"
}, {
key: "Gamma",
color: "lightgreen",
location: new go.Point(150, 230),
loc: "150 230"
}, {
key: "Delta",
color: "pink",
location: new go.Point(250, 330),
loc: "250 330"
}], [{
from: "Alpha",
to: "Beta",
inColor: "red",
outColor: "blue"
}, {
from: "Alpha",
to: "Gamma",
inColor: "yellow",
outColor: "blue"
}, {
from: "Beta",
to: "Gamma",
inColor: "green",
outColor: "pink"
}, {
from: "Gamma",
to: "Delta",
inColor: "black",
outColor: "red"
}, {
from: "Delta",
to: "Alpha",
inColor: "violet",
outColor: "green"
}]);
$scope.model.selectedNodeData = null;
}
});
index.html
<!DOCTYPE html>
<html ng-app="app">
<head>
<script data-require="angular.js@1.5.10" data-semver="1.5.10" src="https://cdnjs.cloudflare.com/ajax/libs/angular.js/1.5.10/angular.min.js"></script>
<script data-require="angular-route@1.5.8" data-semver="1.5.8" src="https://code.angularjs.org/1.5.8/angular-route.js"></script>
<link rel="stylesheet" href="style.css" />
<script src="https://gojs.net/latest/release/go.js"></script>
<script src="script.js"></script>
</head>
<body ng-controller="appController">
<div ng-init="init()">
<h1>{{hello}}</h1>
<go-diagram go-model="model" style="border: solid 1px black; width:100%; height:400px"></go-diagram>
</div>
</body>
</html>
1 ответ
Я бы порекомендовал попытаться переопределить методы на экземпляре диаграмм CommandHandler, а не пытаться создать подкласс CommandHandler. https://gojs.net/latest/intro/extensions.html
В вашей угловой директиве
// create the diagram (line 20 in your Plunkr)
var diagram =
$(go.Diagram, element[0], {
// diagram settings
// ...
// more diagram settings
});
// then get a reference to the diagram's CommandHandler, you can call it SpacingCommandHandler instead of tool if you like.
var tool = diagram.commandHandler;
// then you can override a function of the CommandHandler like so:
tool.decreaseZoom = function() {
console.log('my decrease zoom function!');
// Maybe do something else before
// ...
// Be careful about using 'this' within such functions!
// In cases where you want normal behavior, call the base functionality
// Note the reference to the prototype
// and the call to 'call' passing it what 'this' should be.
go.CommandHandler.prototype.decreaseZoom.call(tool);
// Maybe do something else after
// ...
}
Я успешно использовал этот подход, когда мне нужно было расширить CommandHandler.deleteSelection в моей собственной угловой директиве. При таком подходе вы должны преодолеть свою ошибку.