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;