+
+
+
+
diff --git a/web/js/CacheUMLExplorer.js b/web/js/CacheUMLExplorer.js
index df4f160..0ab9c86 100644
--- a/web/js/CacheUMLExplorer.js
+++ b/web/js/CacheUMLExplorer.js
@@ -11,6 +11,7 @@ var CacheUMLExplorer = function (treeViewContainer, classViewContainer) {
var id = function (e) { return document.getElementById(e); };
this.elements = {
+ favicon: id("favicon"),
uiBody: id("ui-body"),
className: id("className"),
treeViewContainer: treeViewContainer,
@@ -30,7 +31,9 @@ var CacheUMLExplorer = function (treeViewContainer, classViewContainer) {
methodViewBounds: id("methodViewBounds"),
namespaces: id("namespaces"),
classTreeSearch: id("classTreeSearch"),
- searchBlock: id("searchBlock")
+ searchBlock: id("searchBlock"),
+ diagramSearch: id("diagramSearch"),
+ diagramSearchButton: id("button.diagramSearch")
};
this.UI = new UI(this);
@@ -119,6 +122,8 @@ CacheUMLExplorer.prototype.init = function () {
var self = this,
restored;
+ this.elements.favicon.href = lib.image.binoculars;
+
restored = this.restoreFromURL();
this.classTree.showLoader();
this.source.getClassTree(function (err, data) {
diff --git a/web/js/ClassView.js b/web/js/ClassView.js
index 275c21c..b118d87 100644
--- a/web/js/ClassView.js
+++ b/web/js/ClassView.js
@@ -21,10 +21,29 @@ var ClassView = function (parent, container) {
this.CLASS_DOC_PATH = "/csp/documatic/%25CSP.Documatic.cls";
this.SYMBOL_12_WIDTH = 6.6;
+ this.HIGHLIGHTED_VIEW = null;
+ this.SEARCH_INDEX = 0;
+
this.init();
};
+ClassView.prototype.highlightElement = function (jointElement) {
+
+ if (this.HIGHLIGHTED_VIEW || (!jointElement && this.HIGHLIGHTED_VIEW)) {
+ this.HIGHLIGHTED_VIEW.unhighlight();
+ this.HIGHLIGHTED_VIEW = null;
+ }
+
+ if (!jointElement) return;
+ var view = this.paper.findViewByModel(jointElement);
+ if (!view) return;
+
+ view.highlight();
+ this.HIGHLIGHTED_VIEW = view;
+
+};
+
ClassView.prototype.showLoader = function (html) {
var d2;
@@ -60,6 +79,9 @@ ClassView.prototype.resetView = function () {
this.objects = [];
this.paper.setOrigin(0, 0);
this.graph.clear();
+ this.HIGHLIGHTED_VIEW = null;
+ this.SEARCH_INDEX = 0;
+ this.cacheUMLExplorer.elements.diagramSearch.value = "";
};
@@ -268,6 +290,7 @@ ClassView.prototype.createClassInstance = function (name, classMetaData) {
var classParams = classMetaData["parameters"],
classProps = classMetaData["properties"],
classMethods = classMetaData["methods"],
+ keyWordsArray = [name],
self = this;
var classInstance = new joint.shapes.uml.Class({
@@ -283,6 +306,7 @@ ClassView.prototype.createClassInstance = function (name, classMetaData) {
params: (function (params) {
var arr = [], n;
for (n in params) {
+ keyWordsArray.push(n);
arr.push({
text: n + (params[n]["type"] ? ": " + params[n]["type"] : "")
});
@@ -292,6 +316,7 @@ ClassView.prototype.createClassInstance = function (name, classMetaData) {
attributes: (function (ps) {
var arr = [], n;
for (n in ps) {
+ keyWordsArray.push(n);
arr.push({
text: n + (ps[n]["type"] ? ": " + ps[n]["type"] : ""),
icons: self.getMethodIcons(ps[n])
@@ -302,6 +327,7 @@ ClassView.prototype.createClassInstance = function (name, classMetaData) {
methods: (function (met) {
var arr = [], n;
for (n in met) {
+ keyWordsArray.push(n);
arr.push({
text: n + (met[n]["returns"] ? ": " + met[n]["returns"] : ""),
styles: (function (t) {
@@ -319,6 +345,7 @@ ClassView.prototype.createClassInstance = function (name, classMetaData) {
SYMBOL_12_WIDTH: self.SYMBOL_12_WIDTH
});
+ classInstance.SEARCH_KEYWORDS = keyWordsArray.join(",").toLowerCase();
this.objects.push(classInstance);
this.graph.addCell(classInstance);
@@ -562,6 +589,69 @@ ClassView.prototype.zoom = function (delta) {
};
+/**
+ * Focus on joint instance.
+ * @param jointInstance
+ */
+ClassView.prototype.focusOnInstance = function (jointInstance) {
+
+ var bb = jointInstance.getBBox();
+
+ this.focusOnXY(bb.x + bb.width/2, bb.y + bb.height/2);
+
+};
+
+/**
+ * Focus on x and y coordinates considering scale.
+ * @param {number} x
+ * @param {number} y
+ */
+ClassView.prototype.focusOnXY = function (x, y) {
+
+ var sw = this.cacheUMLExplorer.elements.classViewContainer.offsetWidth,
+ sh = this.cacheUMLExplorer.elements.classViewContainer.offsetHeight,
+ scale = this.PAPER_SCALE;
+
+ this.paper.setOrigin(
+ -(x * scale) + sw/2,
+ -(y * scale) + sh/2
+ );
+
+};
+
+/**
+ * Find text on diagram and focus on element.
+ *
+ * @param {string} text
+ */
+ClassView.prototype.searchOnDiagram = function (text) {
+
+ var p, found = [], o;
+
+ if (!text) {
+ this.highlightElement(null);
+ return;
+ }
+
+ text = text.toLowerCase();
+
+ for (p in this.objects) {
+ if (this.objects[p].SEARCH_KEYWORDS.indexOf(text) !== -1) {
+ found.push(this.objects[p]);
+ }
+ }
+
+ if (found.length) {
+ o = found[this.SEARCH_INDEX % found.length];
+ this.focusOnInstance(o);
+ this.highlightElement(o);
+ return;
+ }
+
+ this.highlightElement(null);
+
+};
+
ClassView.prototype.init = function () {
var p, self = this,
@@ -635,6 +725,19 @@ ClassView.prototype.init = function () {
this.cacheUMLExplorer.elements.helpButton.addEventListener("click", function () {
self.renderInfoGraphic();
});
+ this.cacheUMLExplorer.elements.diagramSearch.addEventListener("input", function (e) {
+ self.searchOnDiagram((e.target || e.srcElement).value);
+ });
+ this.cacheUMLExplorer.elements.diagramSearch.addEventListener("keydown", function (e) {
+ if (e.keyCode === 13) {
+ self.SEARCH_INDEX++;
+ self.searchOnDiagram((e.target || e.srcElement).value);
+ }
+ });
+ this.cacheUMLExplorer.elements.diagramSearchButton.addEventListener("click", function () {
+ self.SEARCH_INDEX++;
+ self.searchOnDiagram(self.cacheUMLExplorer.elements.diagramSearch.value);
+ });
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 f61916a..ea0204d 100644
--- a/web/js/Lib.js
+++ b/web/js/Lib.js
@@ -165,5 +165,6 @@ Lib.prototype.image = {
table: "",
earth: "",
zed: "",
- eye: ""
+ eye: "",
+ binoculars: ""
};
\ No newline at end of file