diff --git a/README.md b/README.md index 18a5103..4a48a05 100644 --- a/README.md +++ b/README.md @@ -11,7 +11,7 @@ An UML Class explorer for InterSystems Caché. ## Screenshots -![Demo](https://cloud.githubusercontent.com/assets/4989256/7622499/9cb98048-f9d8-11e4-9c27-4257e53ec70d.png) +![Demo](https://cloud.githubusercontent.com/assets/4989256/7642972/f2c3e33c-fa9d-11e4-90b4-d35c809c9e60.png) ## Installation diff --git a/cache/projectTemplate.xml b/cache/projectTemplate.xml index 3ac3620..1875c87 100644 --- a/cache/projectTemplate.xml +++ b/cache/projectTemplate.xml @@ -3,7 +3,7 @@ Class contains methods that return structured classes/packages data. -63686,4398.893381 +63686,85630.818189 63653,67019.989197 @@ -115,8 +115,8 @@ Return structured data about class. for i=1:1:count { set oPar = ##class(%ZEN.proxyObject).%New() set p = classDefinition.Parameters.GetAt(i) + set oPar.type = p.Type do oParameters.%DispatchSetProperty(p.Name, oPar) - do oPar.%DispatchSetProperty("type", p.Type) } do ..collectInheritance(oData, classDefinition, package) @@ -302,7 +302,7 @@ Returns structured package data - + diff --git a/package.json b/package.json index 426eb4e..4d688ff 100644 --- a/package.json +++ b/package.json @@ -1,6 +1,6 @@ { "name": "CacheUMLExplorer", - "version": "0.8.0", + "version": "0.9.2", "description": "An UML Class explorer for InterSystems Caché", "directories": { "test": "test" diff --git a/web/css/extras.css b/web/css/extras.css index fc04ddf..6894c81 100644 --- a/web/css/extras.css +++ b/web/css/extras.css @@ -84,6 +84,18 @@ box-shadow: 0 0 5px 2px #ffcc1b; } +.icon.help:after { + content: "?"; + position: absolute; + color: white; + display: block; + font-size: 21px; + font-weight: 900; + height: 100%; + width: 100%; + text-align: center; +} + .icon.plus:before { content: ""; background-color: #fff; diff --git a/web/css/interface.css b/web/css/interface.css index 77ec99c..2b5a746 100644 --- a/web/css/interface.css +++ b/web/css/interface.css @@ -32,6 +32,7 @@ html, body { padding: .5em; font-weight: 600; font-size: 18pt; + z-index: 1; } .ui-rightBottomToolBar { @@ -39,6 +40,7 @@ html, body { bottom: 0; right: 0; padding: .5em; + z-index: 1; } .ui-leftBottomToolBar { @@ -46,6 +48,7 @@ html, body { bottom: 0; left: 0; padding: .5em; + z-index: 1; } .ui-topRightToolBar { @@ -53,6 +56,7 @@ html, body { top: 0; right: 0; padding: .5em; + z-index: 1; } #className { diff --git a/web/index.html b/web/index.html index 73d877e..be5fee1 100644 --- a/web/index.html +++ b/web/index.html @@ -37,6 +37,7 @@
+
diff --git a/web/js/CacheUMLExplorer.js b/web/js/CacheUMLExplorer.js index 0a101ab..8e10ce7 100644 --- a/web/js/CacheUMLExplorer.js +++ b/web/js/CacheUMLExplorer.js @@ -18,6 +18,7 @@ var CacheUMLExplorer = function (treeViewContainer, classViewContainer) { zoomInButton: id("button.zoomIn"), zoomOutButton: id("button.zoomOut"), zoomNormalButton: id("button.zoomNormal"), + helpButton: id("button.showHelp"), infoButton: id("button.showInfo"), methodCodeView: id("methodCodeView"), closeMethodCodeView: id("closeMethodCodeView"), @@ -53,7 +54,11 @@ CacheUMLExplorer.prototype.init = function () { this.classView.loadClass(hash.substr(7)); } else if (hash.indexOf("package:") === 1) { this.classView.loadPackage(hash.substr(9)); + } else { + this.classView.renderInfoGraphic(); } + } else { + this.classView.renderInfoGraphic(); } this.elements.infoButton.addEventListener("click", function () { diff --git a/web/js/ClassView.js b/web/js/ClassView.js index 8227fb5..07d745f 100644 --- a/web/js/ClassView.js +++ b/web/js/ClassView.js @@ -73,10 +73,130 @@ ClassView.prototype.openClassDoc = function (className, nameSpace) { }; +/** + * Render help info + */ +ClassView.prototype.renderInfoGraphic = function () { + + this.cacheUMLExplorer.classTree.SELECTED_CLASS_NAME = + this.cacheUMLExplorer.elements.className.innerHTML = + "Welcome to Caché UML explorer!"; + + location.hash = "help"; + + this.showLoader(); + this.render({ + basePackageName: "Welcome to Cach? UML explorer!", + classes: { + "Shared object": { + super: "Super object", + parameters: { + "Also inherit Super object": {} + }, + methods: {}, + properties: {} + }, + "Class name": { + super: "Super object", + ABSTRACT: 1, + FINAL: 1, + HIDDEN: 1, + NAMESPACE: "SAMPLES", + PROCEDUREBLOCK: 0, + SYSTEM: 4, + methods: { + "Abstract public method": { + abstract: 1 + }, + "Class method": { + classMethod: 1 + }, + "Client method": { + clientMethod: 1 + }, + "Final method": { + final: 1 + }, + "Not inheritable method": { + notInheritable: 1 + }, + "Private method": { + private: 1 + }, + "Sql procedure": { + sqlProc: 1 + }, + "Web method": { + webMethod: 1 + }, + "ZEN method": { + zenMethod: 1 + }, + "Method": { + returns: "%Return type" + } + }, + parameters: { + "PARAMETER WITHOUT TYPE": {}, + "PARAMETER": { + type: "Type" + } + }, + properties: { + "Public property name": { + private: 0 + }, + "Private property name": { + private: 1 + }, + "Public read-only property": { + private: 0, + readOnly: 1 + }, + "Property": { + type: "Type of property" + }, + "Other object": { + private: 0, + type: "Shared object" + }, + "Another object": { + private: 1, + type: "Not shared object" + } + } + }, + "Super object": { + methods: {}, + properties: {}, + parameters: {} + }, + "HELP": { + parameters: { + "See the basics here!": {} + } + } + }, + composition: {}, + aggregation: { + "Class name": { + "Shared object": "1..1" + } + }, + inheritance: { + "Class name": { "Super object": 1 }, + "Shared object": { "Super object": 1 } + }, + restrictPackage: 1 + }); + + this.removeLoader(); + +}; + /** * Returns array of signs to render or empry array. * - * @private * @param classMetaData */ ClassView.prototype.getClassSigns = function (classMetaData) { @@ -85,11 +205,11 @@ ClassView.prototype.getClassSigns = function (classMetaData) { if (classMetaData["classType"]) signs.push({ icon: lib.image.greenPill, - text: classMetaData["classType"], + text: lib.capitalize(classMetaData["classType"]), textStyle: "fill:rgb(130,0,255)" }); if (classMetaData["ABSTRACT"]) signs.push({ - icon: lib.image.iceCube, + icon: lib.image.crystalBall, text: "Abstract", textStyle: "fill:rgb(130,0,255)" }); @@ -115,6 +235,29 @@ ClassView.prototype.getClassSigns = function (classMetaData) { }; +/** + * Returns array of icons according to method metadata. + * + * @param method + */ +ClassView.prototype.getMethodIcons = function (method) { + + var icons = []; + + icons.push({ src: lib.image[method["private"] ? "minus" : "plus"] }); + if (method["abstract"]) icons.push({ src: lib.image.crystalBall }); + if (method["clientMethod"]) icons.push({ src: lib.image.user }); + if (method["final"]) icons.push({ src: lib.image.blueFlag }); + if (method["notInheritable"]) icons.push({ src: lib.image.redFlag }); + if (method["sqlProc"]) icons.push({ src: lib.image.table }); + if (method["webMethod"]) icons.push({ src: lib.image.earth }); + if (method["zenMethod"]) icons.push({ src: lib.image.zed }); + if (method["readOnly"]) icons.push({ src: lib.image.eye }); + + return icons; + +}; + /** * @param {string} name * @param classMetaData @@ -127,65 +270,53 @@ ClassView.prototype.createClassInstance = function (name, classMetaData) { classMethods = classMetaData["methods"], self = this; - var insertString = function (array, string, extraString) { - array.push({ text: string + (extraString ? extraString : "")}); - }; - var classInstance = new joint.shapes.uml.Class({ - name: name, + name: [{ + text: name, + clickHandler: function () { + self.openClassDoc(name, classMetaData["NAMESPACE"]); + }, + styles: { + cursor: "help" + } + }], params: (function (params) { var arr = [], n; for (n in params) { - insertString(arr, n + (params[n]["type"] ? ": " + params[n]["type"] : "")); + arr.push({ + text: n + (params[n]["type"] ? ": " + params[n]["type"] : "") + }); } return arr; })(classParams), attributes: (function (ps) { var arr = [], n; for (n in ps) { - insertString( - arr, - (ps[n]["private"] ? "- " : "+ ") + n - + (ps[n]["type"] ? ": " + ps[n]["type"] : "") - ); + arr.push({ + text: n + (ps[n]["type"] ? ": " + ps[n]["type"] : ""), + icons: self.getMethodIcons(ps[n]) + }); } return arr; })(classProps), methods: (function (met) { var arr = [], n; for (n in met) { - insertString( - arr, - (met[n]["private"] ? "- " : "+ ") + n - + (met[n]["returns"] ? ": " + met[n]["returns"] : ""), - (met[n]["classMethod"] ? - "\x1b" + JSON.stringify({STYLES:{ - textDecoration: "underline" - }}) : "") - ); + arr.push({ + text: n + (met[n]["returns"] ? ": " + met[n]["returns"] : ""), + styles: (function (t) { + return t ? { textDecoration: "underline" } : {} + })(met[n]["classMethod"]), + clickHandler: (function (n) { + return function () { self.showMethodCode(name, n); } + })(n), + icons: self.getMethodIcons(met[n]) + }); } return arr; })(classMethods), - directProps: { - nameClickHandler: function () { - self.openClassDoc(name, classMetaData["NAMESPACE"]); - } - }, classSigns: this.getClassSigns(classMetaData), - SYMBOL_12_WIDTH: self.SYMBOL_12_WIDTH, - attrs: { - ".uml-class-methods-text": { - lineClickHandlers: (function (ps) { - var arr = [], p; - for (p in ps) { - arr.push((function (p) { return function () { - self.showMethodCode(name, p) - }})(p)); - } - return arr; - })(classMethods) - } - } + SYMBOL_12_WIDTH: self.SYMBOL_12_WIDTH }); this.objects.push(classInstance); @@ -358,7 +489,7 @@ ClassView.prototype.loadClass = function (className) { self.showLoader("Unable to get " + self.cacheUMLExplorer.classTree.SELECTED_CLASS_NAME); console.error.call(console, err); } else { - self.cacheUMLExplorer.classView.render(data); + self.render(data); } }); @@ -380,7 +511,7 @@ ClassView.prototype.loadPackage = function (packageName) { self.showLoader("Unable to get package " + packageName); console.error.call(console, err); } else { - self.cacheUMLExplorer.classView.render(data); + self.render(data); } }); @@ -492,6 +623,9 @@ ClassView.prototype.init = function () { this.cacheUMLExplorer.elements.closeMethodCodeView.addEventListener("click", function () { self.hideMethodCode(); }); + this.cacheUMLExplorer.elements.helpButton.addEventListener("click", function () { + self.renderInfoGraphic(); + }); this.SYMBOL_12_WIDTH = (function () { var e = document.createElementNS("http://www.w3.org/2000/svg", "text"), diff --git a/web/js/Lib.js b/web/js/Lib.js index e23a372..6df7342 100644 --- a/web/js/Lib.js +++ b/web/js/Lib.js @@ -41,6 +41,14 @@ Lib.prototype.countProperties = function (object) { }; +/** + * Make first letter of string uppercase. + * @param {string} string + */ +Lib.prototype.capitalize = function (string) { + return string[0].toUpperCase() + string.substr(1); +}; + /** * Contains graphic base64s for the application. */ @@ -51,5 +59,14 @@ Lib.prototype.image = { ghost: "", moleculeCubeCross: "", greenPill: "", - iceCube: "" + iceCube: "", + minus: "", + plus: "", + crystalBall: "", + user: "", + redFlag: "", + table: "", + earth: "", + zed: "", + eye: "" }; \ No newline at end of file diff --git a/web/jsLib/joint.js b/web/jsLib/joint.js index 41a8aad..b119fe2 100644 --- a/web/jsLib/joint.js +++ b/web/jsLib/joint.js @@ -17138,7 +17138,7 @@ if ( typeof window === "object" && typeof window.document === "object" ) { text: function(content, opt) { opt = opt || {}; - var lines = content.split('\n'); + var lines = content/*.split('\n')*/; var i = 0; var tspan; @@ -17162,6 +17162,8 @@ if ( typeof window === "object" && typeof window.document === "object" ) { this.node.textContent = ''; var textNode = this.node; + var image; + //console.log(textNode.parentNode); if (opt.textPath) { @@ -17201,45 +17203,58 @@ if ( typeof window === "object" && typeof window.document === "object" ) { textNode = textPath.node; } - //if (lines.length === 1) { - // textNode.textContent = content; - // return this; - //} + // trash - elements collected outside tspan element, variable "to save algorithm" + if (textNode.TRASH instanceof Array) { + _.each(textNode.TRASH, function (e) { e.parentNode.removeChild(e); }); + } + textNode.TRASH = []; for (; i < lines.length; i++) { - var jj, setup; + var jj, setup, iconLeft, xOrigin = this.attr('x') || 0, + iconXOrigin = (opt["ref-x"] || 0) + xOrigin; // Shift all the but first by one line (`1em`) - tspan = V('tspan', { dy: (i == 0 ? '0em' : opt.lineHeight || '1em'), x: this.attr('x') || 0 }); + tspan = V('tspan', { + dy: (i == 0 ? '0em' : opt.lineHeight || '1em'), + x: xOrigin + (lines[i].icons ? lines[i].icons.length*10 + 2 : 0) + }); tspan.addClass('line'); - if (!lines[i]) { + if (!lines[i].text) { tspan.addClass('empty-line'); } - if (lines[i].indexOf("\x1b") !== -1) { - jj = lines[i].split("\x1b"); - lines[i] = jj[0]; - setup = JSON.parse(jj[1]); - if (setup["STYLES"]) { - for (var j in setup["STYLES"]) { - tspan.node.style[j] = setup["STYLES"][j]; - } + if (lines[i]["styles"]) { + for (var j in lines[i]["styles"]) { + tspan.node.style[j] = lines[i]["styles"][j]; } } - if (opt.clickHandler) { - tspan.node.onclick = opt.clickHandler; + if (typeof lines[i]["clickHandler"] === "function") { + tspan.node.addEventListener("click", lines[i]["clickHandler"]); + tspan.addClass('line-clickable'); } - if (opt.lineClickHandlers && opt.lineClickHandlers[i]) { - tspan.node.addEventListener("click", opt.lineClickHandlers[i]); - tspan.node.setAttribute("class", tspan.node.getAttribute("class") + " line-clickable"); - } // Make sure the textContent is never empty. If it is, add an additional // space (an invisible character) so that following lines are correctly // relatively positioned. `dy=1em` won't work with empty lines otherwise. - tspan.node.textContent = lines[i] || ' '; + tspan.node.textContent = lines[i].text || ' '; V(textNode).append(tspan); + + if (lines[i].icons instanceof Array) { + iconLeft = iconXOrigin; + _.each(lines[i].icons, function (ic) { + image = V("image"); + image.attr("xlink:href", ic.src); + image.attr("width", 10); + image.attr("height", 10); + image.attr("y", textNode.getBoundingClientRect().top + i*(opt["font-size"] || 14) + 2); + image.attr("x", iconLeft); + iconLeft += 10; + V(textNode.parentNode).append(image); + textNode.TRASH.push(image.node); + }); + } + } return this; }, @@ -20952,7 +20967,7 @@ joint.dia.ElementView = joint.dia.CellView.extend({ // to rewrite them, if needed. (i.e display: 'none') if (!_.isUndefined(attrs.text)) { $selected.each(function() { - V(this).text(attrs.text + '', attrs); + V(this).text(attrs.text, attrs); }); specialAttributes.push('lineHeight','textPath'); } diff --git a/web/jsLib/joint.shapes.uml.js b/web/jsLib/joint.shapes.uml.js index 51d3b6f..cb07998 100644 --- a/web/jsLib/joint.shapes.uml.js +++ b/web/jsLib/joint.shapes.uml.js @@ -88,9 +88,9 @@ joint.shapes.uml.Class = joint.shapes.basic.Generic.extend({ var o, rects = [ { type: 'name', text: this.getClassName() }, - { type: 'params', text: (o = this.get('params')) .map(function (e) { return e.text; }), o: o }, - { type: 'attrs', text: (o = this.get('attributes')).map(function (e) { return e.text; }), o: o }, - { type: 'methods', text: (o = this.get('methods')) .map(function (e) { return e.text; }), o: o } + { type: 'params', text: (o = this.get('params')) , o: o }, + { type: 'attrs', text: (o = this.get('attributes')), o: o }, + { type: 'methods', text: (o = this.get('methods')) , o: o } ], self = this, classSigns = this.get('classSigns'), @@ -109,8 +109,8 @@ joint.shapes.uml.Class = joint.shapes.basic.Generic.extend({ this.defaults.size.width = Math.max(this.defaults.MIN_WIDTH, Math.min(w, 250)); _.each(rects, function (rect) { - (rect.text instanceof Array ? rect.text : [rect.text]).forEach(function (s) { - var t = s.split("\x1b")[0].length*SYMBOL_12_WIDTH + 8; + rect.text.forEach(function (s) { + var t = s.text.length*SYMBOL_12_WIDTH + 8 + (s.icons ? s.icons.length*10 + 2 : 0); if (t > self.defaults.size.width) { self.defaults.size.width = t; } @@ -157,7 +157,8 @@ joint.shapes.uml.Class = joint.shapes.basic.Generic.extend({ }, getClassName: function () { - return this.get('name'); + var n = this.get('name'); + return n instanceof Array ? n : [{ text: n }]; }, updateRectangles: function () { @@ -168,9 +169,9 @@ joint.shapes.uml.Class = joint.shapes.basic.Generic.extend({ var rects = [ { type: 'name', text: this.getClassName() }, - { type: 'params', text: this.get('params').map(function (e) { return e.text; }) }, - { type: 'attrs', text: this.get('attributes').map(function (e) { return e.text; }) }, - { type: 'methods', text: this.get('methods').map(function (e) { return e.text; }) } + { type: 'params', text: this.get('params') }, + { type: 'attrs', text: this.get('attributes') }, + { type: 'methods', text: this.get('methods') } ]; var offsetY = 0; @@ -180,11 +181,10 @@ joint.shapes.uml.Class = joint.shapes.basic.Generic.extend({ _.each(rects, function(rect) { - var lines = _.isArray(rect.text) ? rect.text : [rect.text]; - + var lines = _.isArray(rect.text) ? rect.text : [{ text: rect.text }]; if (rect.type === "name") { if (self.HEAD_EMPTY_LINES) lines.unshift(""); - for (var i = 0; i < self.HEAD_EMPTY_LINES; i++) lines.unshift(""); + for (var i = 0; i < self.HEAD_EMPTY_LINES; i++) lines.unshift({ text: "" }); } var rectHeight = lines.length * 12 + (lines.length ? 10 : 0), @@ -192,7 +192,8 @@ joint.shapes.uml.Class = joint.shapes.basic.Generic.extend({ rectRect = attrs['.uml-class-' + rect.type + '-rect'], rectLabel = attrs['.uml-class-' + rect.type + '-label']; - rectText.text = lines.join('\n'); + rectText.text = lines; + if (nameClickHandler) { if (rect.type === "name") { rectText.clickHandler = nameClickHandler;