diff --git a/web/js/CacheClassExplorer.js b/web/js/CacheClassExplorer.js
index da4ad5a..bbd3639 100644
--- a/web/js/CacheClassExplorer.js
+++ b/web/js/CacheClassExplorer.js
@@ -22,6 +22,8 @@ var CacheClassExplorer = function (treeViewContainer, classViewContainer) {
showSettingsButton: id("button.showSettings"),
helpButton: id("button.showHelp"),
infoButton: id("button.showInfo"),
+ saveViewButton: id("button.saveView"),
+ saveViewIcon: id("saveViewIcon"),
methodCodeView: id("methodCodeView"),
closeMethodCodeView: id("closeMethodCodeView"),
methodLabel: id("methodLabel"),
@@ -258,4 +260,15 @@ CacheClassExplorer.prototype.init = function () {
enableSVGDownload(this.classTree);
+ // default icon
+ this.elements.saveViewIcon.src = lib.image.pin;
+ this.elements.saveViewButton.addEventListener("click", function () {
+ self.classView.switchViewSave();
+ if (self.classView.viewSaving) {
+ self.classView.saveView();
+ } else {
+ self.source.resetView( self.NAMESPACE + ":" + self.classView.CURRENT_RENDER_NAME );
+ }
+ });
+
};
\ No newline at end of file
diff --git a/web/js/ClassView.js b/web/js/ClassView.js
index 16f03db..970f1f8 100644
--- a/web/js/ClassView.js
+++ b/web/js/ClassView.js
@@ -25,6 +25,16 @@ var ClassView = function (parent, container) {
this.HIGHLIGHTED_VIEW = null;
this.SEARCH_INDEX = 0;
+ this.CURRENT_RENDER_NAME = "";
+
+ this.viewSaving = false;
+
+ /**
+ * Not to perform save too frequentry, this variable is used to control saving frequency.
+ * @type {number}
+ */
+ this.saveTimeout = 0;
+
this.init();
};
@@ -446,9 +456,10 @@ ClassView.prototype.getPropertyHoverText = function (prop, type) {
/**
* @param {string} name
* @param classMetaData
+ * @param saved - Object with saved data.
* @returns {joint.shapes.uml.Class}
*/
-ClassView.prototype.createClassInstance = function (name, classMetaData) {
+ClassView.prototype.createClassInstance = function (name, classMetaData, saved) {
var classParams = classMetaData["parameters"],
classProps = classMetaData["properties"],
@@ -458,7 +469,7 @@ ClassView.prototype.createClassInstance = function (name, classMetaData) {
keyWordsArray = [name],
self = this;
- var classInstance = new joint.shapes.uml.Class({
+ var setup = {
name: [{
text: name,
clickHandler: function () {
@@ -549,9 +560,18 @@ ClassView.prototype.createClassInstance = function (name, classMetaData) {
classSigns: this.getClassSigns(classMetaData),
classType: classMetaData.ClassType || "registered",
SYMBOL_12_WIDTH: self.SYMBOL_12_WIDTH
+ };
+
+ if (saved && saved.position) setup.position = saved.position;
+
+ var classInstance = new joint.shapes.uml.Class(setup);
+
+ classInstance.on("change:position", function () {
+ self.prepareToSave();
});
classInstance.SEARCH_KEYWORDS = keyWordsArray.join(",").toLowerCase();
+ classInstance.NAME = name;
this.objects.push(classInstance);
this.graph.addCell(classInstance);
@@ -559,6 +579,15 @@ ClassView.prototype.createClassInstance = function (name, classMetaData) {
};
+ClassView.prototype.prepareToSave = function () {
+
+ if (!this.viewSaving) return;
+
+ if (this.saveTimeout) clearTimeout(this.saveTimeout);
+ this.saveTimeout = setTimeout(this.saveView.bind(this), 700);
+
+};
+
ClassView.prototype.showMethodCode = function (className, methodName) {
var self = this;
@@ -701,6 +730,8 @@ ClassView.prototype.confirmRender = function (data) {
uml = joint.shapes.uml, relFrom, relTo,
classes = {}, connector;
+ this.switchViewSave(!!data.savedView);
+
this.filterInherits(data);
// Reset view and zoom again because it may cause visual damage to icons.
@@ -715,7 +746,11 @@ ClassView.prototype.confirmRender = function (data) {
for (className in data["classes"]) {
classes[className] = {
- instance: this.createClassInstance(className, data["classes"][className])
+ instance: this.createClassInstance(
+ className,
+ data["classes"][className],
+ ((data.savedView || {}).classes || {})[className]
+ )
};
}
@@ -730,7 +765,11 @@ ClassView.prototype.confirmRender = function (data) {
relTo = (classes[pp] || {}).instance;
if (!relTo) {
classes[pp] = {
- instance: relTo = self.createClassInstance(pp, {})
+ instance: relTo = self.createClassInstance(
+ pp,
+ {},
+ ((data.savedView || {}).classes || {})[pp]
+ )
};
}
if (relFrom && relTo) {
@@ -780,13 +819,15 @@ ClassView.prototype.confirmRender = function (data) {
link("aggregation");
link("association");
- joint.layout.DirectedGraph.layout(this.graph, {
- setLinkVertices: false,
- nodeSep: 100,
- rankSep: 100,
- edgeSep: 20,
- rankDir: data.layoutDirection || "TB"
- });
+ if (!data.savedView) {
+ joint.layout.DirectedGraph.layout(this.graph, {
+ setLinkVertices: false,
+ nodeSep: 100,
+ rankSep: 100,
+ edgeSep: 20,
+ rankDir: data.layoutDirection || "TB"
+ });
+ }
this.updateSizes();
@@ -794,16 +835,66 @@ ClassView.prototype.confirmRender = function (data) {
this.paper.findViewByModel(this.links[i]).update();
}
- var bb = this.paper.getContentBBox(), q = this.paper;
+ var bb = this.paper.getContentBBox(),
+ q = this.paper;
+
this.paper.setOrigin(
q.options.width/2 - bb.width/2,
q.options.height/2 - Math.min(q.options.height/2 - 100, bb.height/2)
);
+ if (data.savedView) this.restoreView(data.savedView);
+
this.onRendered();
};
+ClassView.prototype.switchViewSave = function ( saving ) {
+
+ if (typeof saving === "undefined") saving = !this.viewSaving;
+ this.viewSaving = !!saving;
+ this.cacheClassExplorer.elements.saveViewIcon.src = lib.image["pin" + (saving ? "Active" : "")];
+
+};
+
+ClassView.prototype.saveView = function () {
+
+ if (!this.CURRENT_RENDER_NAME || !this.cacheClassExplorer.NAMESPACE) return;
+
+ var self = this,
+ name = this.cacheClassExplorer.NAMESPACE + ":" + this.CURRENT_RENDER_NAME;
+
+ var saved = {
+ classes: {},
+ zoom: this.PAPER_SCALE,
+ origin: {
+ x: Math.round(self.paper.options.origin.x),
+ y: Math.round(self.paper.options.origin.y)
+ }
+ };
+
+ this.graph.getElements().forEach(function (element) {
+ if (!element.NAME) return;
+ saved.classes[element.NAME] = {
+ position: element.attributes.position
+ }
+ });
+
+ this.cacheClassExplorer.source.saveView(name, saved);
+
+};
+
+ClassView.prototype.restoreView = function (data) {
+
+ // data.classes are parsed during class creation
+ if (data.zoom) { // do not swap with origin set
+ this.PAPER_SCALE = data.zoom;
+ this.zoom(0);
+ }
+ if (data.origin && data.origin.x && data.origin.y) this.paper.setOrigin(data.origin.x, data.origin.y);
+
+};
+
ClassView.prototype.loadClass = function (className) {
var self = this;
@@ -823,6 +914,7 @@ ClassView.prototype.loadClass = function (className) {
});
this.cacheClassExplorer.elements.className.textContent = className;
+ this.CURRENT_RENDER_NAME = "CLASS:" + className;
this.cacheClassExplorer.updateURL();
};
@@ -846,6 +938,7 @@ ClassView.prototype.loadPackage = function (packageName) {
});
this.cacheClassExplorer.elements.className.textContent = packageName;
+ this.CURRENT_RENDER_NAME = "PACKAGE:" + packageName;
this.cacheClassExplorer.updateURL();
};
@@ -881,6 +974,8 @@ ClassView.prototype.zoom = function (delta) {
oy - (sh/2 - oy)*scaleDelta
);
+ if (delta) this.prepareToSave(); // delta = null,0 when restore triggered
+
};
/**
@@ -1047,6 +1142,7 @@ ClassView.prototype.init = function () {
self.paper.options.origin.y + e.pageY - relP.y
);
relP.x = e.pageX; relP.y = e.pageY;
+ self.prepareToSave();
};
this.cacheClassExplorer.elements.classViewContainer.addEventListener("mousemove", moveHandler);
diff --git a/web/js/Lib.js b/web/js/Lib.js
index 952e6f6..a0008c9 100644
--- a/web/js/Lib.js
+++ b/web/js/Lib.js
@@ -12,10 +12,16 @@ Lib.prototype.load = function (url, data, callback) {
var xhr = new XMLHttpRequest();
xhr.open(data ? "POST" : "GET", url);
+ if (typeof callback === "undefined") callback = function () {};
xhr.onreadystatechange = function () {
if (xhr.readyState === 4 && xhr.status === 200) {
- return callback(null, JSON.parse(xhr.responseText) || {});
+ try {
+ return callback(null, JSON.parse(xhr.responseText) || {});
+ } catch (e) {
+ console.error(url, "Unable to parse:", { data: xhr.responseText });
+ return {};
+ }
} else if (xhr.readyState === 4) {
callback(xhr.responseText + ", " + xhr.status + ": " + xhr.statusText);
}
@@ -312,5 +318,7 @@ Lib.prototype.image = {
keyRed: "",
keyGreen: "",
minusSimple: "",
- plusSimple: ""
+ plusSimple: "",
+ pin: "",
+ pinActive: ""
};
\ No newline at end of file
diff --git a/web/js/Logic.js b/web/js/Logic.js
index 12a3704..1d43d22 100644
--- a/web/js/Logic.js
+++ b/web/js/Logic.js
@@ -7,7 +7,7 @@ var Logic = function (parent) {
/**
* Modify data, add relations, connections, helpers.
*
- * @param {{basePackageName: string, classes: object, restrictPackage: number}} data
+ * @param {*} data
*/
Logic.prototype.process = function (data) {
@@ -16,6 +16,13 @@ Logic.prototype.process = function (data) {
this.data = data;
+ if (data.savedView) try {
+ data.savedView = JSON.parse(data.savedView);
+ } catch (e) {
+ delete data.savedView;
+ console.log("! Unable to deserialize savedView.");
+ }
+
data.classes["%Persistent"] = data.classes["%Library.Persistent"] = {
$classType: "Persistent"
};
diff --git a/web/js/Source.js b/web/js/Source.js
index 1585aef..41085c2 100644
--- a/web/js/Source.js
+++ b/web/js/Source.js
@@ -45,7 +45,8 @@ Source.prototype.getMethod = function (className, methodName, callback) {
+ encodeURIComponent(methodName)
+ (this.cue.NAMESPACE ? "&namespace=" + encodeURIComponent(this.cue.NAMESPACE) : ""),
null,
- callback);
+ callback
+ );
};
@@ -60,7 +61,26 @@ Source.prototype.getClassView = function (className, callback) {
this.URL + "/GetClassView?name=" + encodeURIComponent(className)
+ (this.cue.NAMESPACE ? "&namespace=" + encodeURIComponent(this.cue.NAMESPACE) : ""),
null,
- callback);
+ callback
+ );
+
+};
+
+Source.prototype.saveView = function (packageName, data) {
+
+ lib.load(
+ this.URL + "/SaveView?name=" + encodeURIComponent(packageName),
+ data,
+ function (e) { console.log("View saved."); }
+ );
+
+};
+
+Source.prototype.resetView = function (packageName) {
+
+ lib.load(
+ this.URL + "/ResetView?name=" + encodeURIComponent(packageName)
+ );
};