Skip to content

Commit

Permalink
🔥 new experience: rendering only necessary classes, a lot of improvem…
Browse files Browse the repository at this point in the history
…ents
  • Loading branch information
nikitaeverywhere committed May 7, 2015
1 parent 9021361 commit aaf32a6
Show file tree
Hide file tree
Showing 6 changed files with 123 additions and 92 deletions.
2 changes: 1 addition & 1 deletion README.md
Original file line number Diff line number Diff line change
Expand Up @@ -3,7 +3,7 @@ An UML Class explorer for InterSystems Caché. It is able to build UML class dia

## Screenshots

![2015-04-21_214058](https://cloud.githubusercontent.com/assets/4989256/7396518/65ba1924-eeaa-11e4-808b-5f648c0011e4.png)
![2015-04-21_214058](https://cloud.githubusercontent.com/assets/4989256/7525432/180dafd8-f512-11e4-80dc-3c5721d3b858.png)

## Installation

Expand Down
159 changes: 99 additions & 60 deletions cache/projectTemplate.xml
Original file line number Diff line number Diff line change
Expand Up @@ -2,8 +2,8 @@
<Export generator="Cache" version="25" zv="Cache for Windows (x86-64) 2015.2 (Build 540)" ts="2015-04-28 19:50:48">
<Class name="UMLExplorer.ClassView">
<Description>
Class contains methods that return structured class data.</Description>
<TimeChanged>63671,64479.682329</TimeChanged>
Class contains methods that return structured classes/packages data.</Description>
<TimeChanged>63680,165.360933</TimeChanged>
<TimeCreated>63653,67019.989197</TimeCreated>

<Method name="getClassTree">
Expand Down Expand Up @@ -47,21 +47,27 @@ Returns structured class tree with all classes available in current namespace</D
]]></Implementation>
</Method>

<Method name="getClassData">
<Method name="setClassData">
<Description>
return structured data about class</Description>
Return structured data about class.</Description>
<ClassMethod>1</ClassMethod>
<FormalSpec>oData:%ZEN.proxyObject,classDefinition:%Dictionary.ClassDefinition</FormalSpec>
<FormalSpec>oData:%ZEN.proxyObject,className:%String</FormalSpec>
<Private>1</Private>
<ReturnType>%ZEN.proxyObject</ReturnType>
<Implementation><![CDATA[
set oClass = ##class(%ZEN.proxyObject).%New()
set basePack = $LISTTOSTRING($LIST($LISTFROMSTRING(classDefinition.Name, "."), 1, *-1),".")
set classDefinition = ##class(%Dictionary.ClassDefinition).%OpenId(className)
if (classDefinition = "") || (oData.classes.%DispatchGetProperty(classDefinition.Name) '= "") quit ""
set oClass = ##class(%ZEN.proxyObject).%New()
do oData.classes.%DispatchSetProperty(classDefinition.Name, oClass) // prevent from recursive setup
set package = $LISTTOSTRING($LIST($LISTFROMSTRING(classDefinition.Name, "."), 1, *-1),".")
set oProperties = ##class(%ZEN.proxyObject).%New()
set oClass.NAMESPACE = $NAMESPACE
set oClass.ABSTRACT = classDefinition.Abstract
set oClass.super = classDefinition.Super
if (oData.restrictPackage) && ('..inPackage(oData.basePackageName, package)) quit oClass
set oClass.properties = oProperties
set count = classDefinition.Properties.Count()
for i = 1:1:count {
Expand All @@ -71,8 +77,8 @@ return structured data about class</Description>
do oProp.%DispatchSetProperty("private", p.Private)
do oProp.%DispatchSetProperty("readOnly", p.ReadOnly)
do oProp.%DispatchSetProperty("type", p.Type)
do ..collectAggregation(oData, classDefinition.Name, p.Type, p.Private)
do ..collectAggregation(oData, classDefinition.Name, basePack _ "." _ p.Type, p.Private)
do ..collectAggregation(oData, classDefinition, p.Type, p.Private)
do ..collectAggregation(oData, classDefinition, package _ "." _ p.Type, p.Private)
}
set oMethods = ##class(%ZEN.proxyObject).%New()
Expand All @@ -97,121 +103,156 @@ return structured data about class</Description>
do oPar.%DispatchSetProperty("type", p.Type)
}
do ..collectInheritance(oData, classDefinition, package)
quit oClass
]]></Implementation>
</Method>

<Method name="inPackage">
<Description><![CDATA[
Returns if <var>packageName</var> is in <var>basePackageName</var>.]]></Description>
<ClassMethod>1</ClassMethod>
<FormalSpec>basePackageName:%String,packageName:%String</FormalSpec>
<Private>1</Private>
<ReturnType>%Boolean</ReturnType>
<Implementation><![CDATA[ quit $FIND(packageName, basePackageName) - $LENGTH(basePackageName) = 1
]]></Implementation>
</Method>

<Method name="extendClassFromType">
<Description>
Wrap registered types to class names</Description>
<ClassMethod>1</ClassMethod>
<FormalSpec>typeName:%String</FormalSpec>
<Private>1</Private>
<ReturnType>%String</ReturnType>
<Implementation><![CDATA[
return $CASE(typeName,
return typeName
/*$CASE(typeName,
"%String": "%Library.String",
"%Integer": "%Library.Integer",
"%DataType": "%Library.DataType",
"%Status": "%Library.Status",
"%CacheString": "%Library.CacheString",
"%Persistent": "%Library.Persistent",
:typeName)
:typeName)*/
]]></Implementation>
</Method>

<Method name="collectInheritance">
<Description>
Fill inheritance data</Description>
<ClassMethod>1</ClassMethod>
<FormalSpec>oData:%ZEN.proxyObject,baseClassDefinition:%Dictionary.ClassDefinition</FormalSpec>
<FormalSpec>oData:%ZEN.proxyObject,baseClassDefinition:%Dictionary.ClassDefinition,basePack:%String</FormalSpec>
<Private>1</Private>
<ReturnType>%Status</ReturnType>
<Implementation><![CDATA[
set superParts = $LISTFROMSTRING(baseClassDefinition.Super, ",")
if (oData.inheritance.%DispatchGetProperty(baseClassDefinition.Name) = "") {
do oData.inheritance.%DispatchSetProperty(baseClassDefinition.Name, ##class(%ZEN.proxyObject).%New())
}
set oInherit = oData.inheritance.%DispatchGetProperty(baseClassDefinition.Name)
set oInherit = ##class(%ZEN.proxyObject).%New()
do oData.inheritance.%DispatchSetProperty(baseClassDefinition.Name, oInherit)
for i=1:1:$LISTLENGTH(superParts) {
set className = ..extendClassFromType($LISTGET(superParts, i))
do oInherit.%DispatchSetProperty(className, 1)
if (oData.classes.%DispatchGetProperty(className) = "") {
set cdef = ##class(%Dictionary.ClassDefinition).%OpenId(className)
if (cdef '= "") {
do oData.classes.%DispatchSetProperty(className, ..getClassData(oData, cdef))
do ..collectInheritance(oData, cdef)
}
// try to find class with base package, if not successfull - try to add class as it is
if (..classExists(basePack_"."_className)) {
set clsName = basePack_"."_className
} else {
set clsName = className
}
do oInherit.%DispatchSetProperty(clsName, 1)
do ..setClassData(oData, clsName)
}
quit $$$OK
]]></Implementation>
</Method>

<Method name="collectAggregation">
<Description>
Fill aggregation/composition data</Description>
<ClassMethod>1</ClassMethod>
<FormalSpec>oData:%ZEN.proxyObject,className:%String,type:%String,private:%String</FormalSpec>
<FormalSpec>oData:%ZEN.proxyObject,classDef:%Dictionary.ClassDefinition,aggClassName:%String,private:%String</FormalSpec>
<Private>1</Private>
<ReturnType>%Status</ReturnType>
<Implementation><![CDATA[
set classDef = ##class(%Dictionary.ClassDefinition).%OpenId(type)
if (classDef '= "") {
set aggClassDef = ##class(%Dictionary.ClassDefinition).%OpenId(aggClassName)
if (aggClassDef '= "") {
if (private) { set t = "composition" } else { set t = "aggregation" }
set oAgg = oData.%DispatchGetProperty(t).%DispatchGetProperty(className)
set oAgg = oData.%DispatchGetProperty(t).%DispatchGetProperty(classDef.Name)
if (oAgg = "") {
set oAgg = ##class(%ZEN.proxyObject).%New()
do oData.%DispatchGetProperty(t).%DispatchSetProperty(className, oAgg)
do oData.%DispatchGetProperty(t).%DispatchSetProperty(classDef.Name, oAgg)
}
} else { quit $$$OK }
do oAgg.%DispatchSetProperty(type, "1..1")
do ..collectClass(oData, classDef)
do oAgg.%DispatchSetProperty(aggClassName, "1..1")
do ..setClassData(oData, aggClassName)
quit $$$OK
]]></Implementation>
</Method>

<Method name="collectClass">
<Method name="getBaseOData">
<Description>
Setup basic output data object</Description>
<ClassMethod>1</ClassMethod>
<FormalSpec>oData:%ZEN.proxyObject,cdef:%Dictionary.ClassDefinition</FormalSpec>
<ReturnType>%Status</ReturnType>
<FormalSpec>packageName:%String</FormalSpec>
<Private>1</Private>
<ReturnType>%ZEN.proxyObject</ReturnType>
<Implementation><![CDATA[
if (oData.classes.%DispatchGetProperty(cdef.Name) '= "") { quit $$$OK }
do oData.classes.%DispatchSetProperty(cdef.Name, "IDLE")
do oData.classes.%DispatchSetProperty(cdef.Name, ..getClassData(oData, cdef))
do ..collectInheritance(oData, cdef)
set oData = ##class(%ZEN.proxyObject).%New()
set oData.basePackageName = packageName
set oData.restrictPackage = 1 // expand classes only in base package
set oData.classes = ##class(%ZEN.proxyObject).%New()
set oData.inheritance = ##class(%ZEN.proxyObject).%New()
set oData.aggregation = ##class(%ZEN.proxyObject).%New()
set oData.composition = ##class(%ZEN.proxyObject).%New()
quit oData
]]></Implementation>
</Method>

<Method name="classExists">
<Description>
Returns if class with given name exists.</Description>
<ClassMethod>1</ClassMethod>
<FormalSpec>className:%String</FormalSpec>
<ReturnType>%Boolean</ReturnType>
<Implementation><![CDATA[ quit ##class(%Dictionary.ClassDefinition).%OpenId(className) '= ""
]]></Implementation>
</Method>

<Method name="getClassView">
<Description>
Returns structured class data</Description>
<ClassMethod>1</ClassMethod>
<FormalSpec>baseClassDefinition:%Dictionary.ClassDefinition</FormalSpec>
<FormalSpec>className:%String</FormalSpec>
<ReturnType>%ZEN.proxyObject</ReturnType>
<Implementation><![CDATA[
set oData = ##class(%ZEN.proxyObject).%New()
set oData.classes = ##class(%ZEN.proxyObject).%New()
set oData.inheritance = ##class(%ZEN.proxyObject).%New()
set oData.aggregation = ##class(%ZEN.proxyObject).%New()
do ..collectClass(oData, baseClassDefinition)
set package = $LISTTOSTRING($LIST($LISTFROMSTRING(className, "."), 1, *-1), ".")
set oData = ..getBaseOData(package)
do ..setClassData(oData, className)
quit oData
]]></Implementation>
</Method>

<Method name="getPackageView">
<Description>
Returns structured package data</Description>
<ClassMethod>1</ClassMethod>
<FormalSpec>rootPackageName:%String</FormalSpec>
<ReturnType>%ZEN.proxyObject</ReturnType>
<Implementation><![CDATA[
set oData = ##class(%ZEN.proxyObject).%New()
set oData.classes = ##class(%ZEN.proxyObject).%New()
set oData.inheritance = ##class(%ZEN.proxyObject).%New()
set oData.aggregation = ##class(%ZEN.proxyObject).%New()
set oData.composition = ##class(%ZEN.proxyObject).%New()
set oData = ..getBaseOData(rootPackageName)
set classes = ##class(%ResultSet).%New("%Dictionary.ClassDefinition:Summary")
do classes.Execute()
set listLen = $LISTLENGTH($LISTFROMSTRING(rootPackageName, "."))
set listLen = $LISTLENGTH($LISTFROMSTRING(rootPackageName, ".")) // bottom level of package to extract
while (classes.Next()) {
set className = classes.Data("Name")
set packageName = $LISTTOSTRING($LIST($LISTFROMSTRING(className, "."), 1, listLen), ".")
if (packageName = rootPackageName) {
set classDef = ##class(%Dictionary.ClassDefinition).%OpenId(className)
do oData.classes.%DispatchSetProperty(classDef.Name, ..getClassData(oData, classDef))
do ..collectInheritance(oData, classDef)
do ..setClassData(oData, className)
}
}
Expand All @@ -221,7 +262,7 @@ return structured data about class</Description>
</Class>


<Project name="UMLExplorer" LastModified="2015-04-28 23:30:37.941872">
<Project name="UMLExplorer" LastModified="2015-04-29 19:47:03.727613">
<Items>
<ProjectItem name="UMLExplorer.ClassView" type="CLS"></ProjectItem>
<ProjectItem name="UMLExplorer.Router" type="CLS"></ProjectItem>
Expand All @@ -234,7 +275,7 @@ return structured data about class</Description>
<Description>
REST interface for UMLExplorer</Description>
<Super>%CSP.REST</Super>
<TimeChanged>63670,71431.319061</TimeChanged>
<TimeChanged>63679,81701.423669</TimeChanged>
<TimeCreated>63648,30450.187229</TimeCreated>

<XData name="UrlMap">
Expand Down Expand Up @@ -270,9 +311,7 @@ Returns classTree by given class name</Description>
<FormalSpec>className:%String</FormalSpec>
<ReturnType>%Status</ReturnType>
<Implementation><![CDATA[
set cdef = ##class(%Dictionary.ClassDefinition).%OpenId(className)
if (cdef = "") quit ..Http404()
set classData = ##class(UMLExplorer.ClassView).getClassView(cdef)
set classData = ##class(UMLExplorer.ClassView).getClassView(className)
do classData.%ToJSON(, "ou")
return $$$OK
]]></Implementation>
Expand Down
2 changes: 1 addition & 1 deletion package.json
Original file line number Diff line number Diff line change
@@ -1,6 +1,6 @@
{
"name": "CacheUMLExplorer",
"version": "0.5.2",
"version": "0.6.0",
"description": "An UML Class explorer for InterSystems Caché",
"directories": {
"test": "test"
Expand Down
24 changes: 15 additions & 9 deletions web/js/ClassView.js
Original file line number Diff line number Diff line change
Expand Up @@ -88,7 +88,7 @@ ClassView.prototype.createClassInstance = function (name, classMetaData) {
array.push(string + (extraString ? extraString : ""));
};

return new joint.shapes.uml.Class({
var classInstance = new joint.shapes.uml.Class({
name: (classMetaData["ABSTRACT"] ? ["<<Abstract>>", name] : [name]),
params: (function (params) {
var arr = [], n;
Expand Down Expand Up @@ -130,11 +130,16 @@ ClassView.prototype.createClassInstance = function (name, classMetaData) {
}
});

this.objects.push(classInstance);
this.graph.addCell(classInstance);

return classInstance;

};

ClassView.prototype.render = function (data) {

var self = this, p, pp, className, classInstance,
var self = this, p, pp, className,
uml = joint.shapes.uml, relFrom, relTo,
classes = {}, connector;

Expand All @@ -144,14 +149,9 @@ ClassView.prototype.render = function (data) {
}

for (className in data["classes"]) {
classInstance = this.createClassInstance(className, data["classes"][className]);
this.objects.push(classInstance);
classes[className] = {
instance: classInstance
instance: this.createClassInstance(className, data["classes"][className])
};

this.graph.addCell(classInstance);

}

var link = function (type) {
Expand All @@ -161,6 +161,11 @@ ClassView.prototype.render = function (data) {
relFrom = (classes[p] || {}).instance;
for (pp in data[type][p]) {
relTo = (classes[pp] || {}).instance;
if (!relTo) {
classes[pp] = {
instance: relTo = self.createClassInstance(pp, {})
};
}
if (relFrom && relTo) {
self.graph.addCell(connector = new uml[name]({
source: { id: type === "inheritance" ? relFrom.id : relTo.id },
Expand All @@ -181,7 +186,8 @@ ClassView.prototype.render = function (data) {
joint.layout.DirectedGraph.layout(this.graph, {
setLinkVertices: false,
nodeSep: 100,
rankSep: 100
rankSep: 100,
edgeSep: 20
});

this.updateSizes();
Expand Down
3 changes: 3 additions & 0 deletions web/jsLib/joint.js
Original file line number Diff line number Diff line change
Expand Up @@ -18,6 +18,9 @@ file, You can obtain one at http://mozilla.org/MPL/2.0/.
*
* Date: 2013-07-03T13:30Z
*/
/*
* Modified by ZitRo
*/
(function( window, undefined ) {

// Can't do this because several apps including ASP.NET trace
Expand Down
Loading

0 comments on commit aaf32a6

Please sign in to comment.