diff --git a/cache/projectTemplate.xml b/cache/projectTemplate.xml index f3f2fd7..c41cb28 100644 --- a/cache/projectTemplate.xml +++ b/cache/projectTemplate.xml @@ -4,7 +4,7 @@ Cache Class Explorer vX.X.X/*build.replace:pkg.version*/ Class contains methods that return structured classes/packages data. -63919,67431.456639 +63928,63957.580821 63653,67019.989197 @@ -193,7 +193,7 @@ Return structured data about class. set xd = classDefinition.XDatas.GetAt(i) for j=1:1:props.Properties.Count() { set pname = props.Properties.GetAt(j).Name - set:(pname '= "parent") $PROPERTY(oProp, pname) = $PROPERTY(xd, pname) + set:((pname '= "parent") && (pname '= "Object")) $PROPERTY(oProp, pname) = $PROPERTY(xd, pname) } do oXDatas.%DispatchSetProperty(xd.Name, oProp) } @@ -319,7 +319,7 @@ Returns new (correct) super Setup basic output data object 1 -packageName:%String +packageName:%String,baseNamespace:%String,savedName:%String 1 %ZEN.proxyObject set oData.basePackageName = packageName set oData.restrictPackage = 1 // expand classes only in base package set oData.classes = ##class(%ZEN.proxyObject).%New() + + set ns = $namespace + zn baseNamespace + if $get(^ClassExplorer("savedView", ns_":"_savedName)) '= "" { + set oData.savedView = $get(^ClassExplorer("savedView", ns_":"_savedName)) + } + zn ns + quit oData ]]> @@ -348,9 +356,10 @@ Returns structured class data className:%String,namespace:%String %ZEN.proxyObject @@ -363,8 +372,9 @@ Returns structured package data rootPackageName:%String,namespace:%String %ZEN.proxyObject REST interface for ClassExplorer %CSP.REST -63697,73073.878177 +63928,63486.89174 63648,30450.187229 @@ -413,6 +423,8 @@ REST interface for ClassExplorer + + ]]> @@ -441,6 +453,34 @@ Returns classTree by given class name ]]> + + +Saves the view preferences +1 +%Status + + + + + +Saves the view preferences +1 +%Status + + + Returns all package class trees by given package name diff --git a/package.json b/package.json index 3a3c65c..3c4a0b8 100644 --- a/package.json +++ b/package.json @@ -1,6 +1,6 @@ { "name": "CacheClassExplorer", - "version": "1.12.0", + "version": "1.13.1", "description": "Class Explorer for InterSystems Caché", "directories": { "test": "test" diff --git a/web/css/extras.css b/web/css/extras.css index a5e7691..70641bb 100644 --- a/web/css/extras.css +++ b/web/css/extras.css @@ -63,12 +63,12 @@ } .icon { + position: relative; display: inline-block; background-color: #333; border-radius: 12px; width: 24px; height: 24px; - position: relative; cursor: pointer; -webkit-transition: all .2s ease; -moz-transition: all .2s ease; @@ -80,6 +80,14 @@ user-select: none; } +.icon img { + position: absolute; + width: 16px; + height: 16px; + left: 4px; + top: 4px; +} + .icon:hover { box-shadow: 0 0 5px 2px #ffcc1b; } diff --git a/web/index.html b/web/index.html index 1629bdb..22a66e8 100644 --- a/web/index.html +++ b/web/index.html @@ -60,10 +60,13 @@
- -
-
-
+ +
+
+
+
+ +
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) + ); };