diff --git a/web/app/3d/ctrl.js b/web/app/3d/ctrl.js
index 49f4f986..a9b6ea03 100644
--- a/web/app/3d/ctrl.js
+++ b/web/app/3d/ctrl.js
@@ -2,6 +2,27 @@
TCAD.UI = function(app) {
this.app = app;
this.viewer = app.viewer;
+
+ var box = new TCAD.toolkit.Box();
+ var propFolder = new TCAD.toolkit.Folder("Solid's Properties");
+ var cameraFolder = new TCAD.toolkit.Folder("Camera");
+ var objectsFolder = new TCAD.toolkit.Folder("Objects");
+ var modificationsFolder = new TCAD.toolkit.Folder("Modifications");
+ TCAD.toolkit.add(box, propFolder);
+ TCAD.toolkit.add(propFolder, new TCAD.toolkit.Button("Extrude"));
+ TCAD.toolkit.add(propFolder, new TCAD.toolkit.Button("Cut"));
+ TCAD.toolkit.add(propFolder, new TCAD.toolkit.Button("Edit"));
+ TCAD.toolkit.add(propFolder, new TCAD.toolkit.Button("Refresh Sketches"));
+ TCAD.toolkit.add(propFolder, new TCAD.toolkit.Text("Message"));
+ TCAD.toolkit.add(box, cameraFolder);
+ TCAD.toolkit.add(cameraFolder, new TCAD.toolkit.Number("x"));
+ TCAD.toolkit.add(cameraFolder, new TCAD.toolkit.Number("y"));
+ TCAD.toolkit.add(cameraFolder, new TCAD.toolkit.Number("z"));
+ TCAD.toolkit.add(box, objectsFolder);
+ TCAD.toolkit.add(box, modificationsFolder);
+ this.modificationsTreeComp = new TCAD.toolkit.Tree();
+ TCAD.toolkit.add(cameraFolder, this.modificationsTreeComp);
+
this.dat = new dat.GUI();
var ui = this;
var gui = this.dat;
diff --git a/web/app/3d/main.js b/web/app/3d/main.js
index d527df48..06866b5b 100644
--- a/web/app/3d/main.js
+++ b/web/app/3d/main.js
@@ -156,10 +156,15 @@ TCAD.App.prototype.extrude = function() {
}
var polyFace = this.viewer.selectionMgr.selection[0];
var height = prompt("Height", "50");
-
+ if (!height) return;
+
var app = this;
- this.craft.modify(polyFace.solid, function() {
- return TCAD.craft.extrude(app, polyFace, polyFace.solid.polyFaces, height);
+ var solids = [polyFace.solid];
+ this.craft.modify({
+ type: 'PAD',
+ solids : solids,
+ face : polyFace,
+ height : height
});
};
@@ -170,12 +175,16 @@ TCAD.App.prototype.cut = function() {
}
var polyFace = this.viewer.selectionMgr.selection[0];
var depth = prompt("Depth", "50");
+ if (!depth) return;
var app = this;
- this.craft.modify(polyFace.solid, function() {
- return TCAD.craft.cut(app, polyFace, polyFace.solid.polyFaces, depth);
+ var solids = [polyFace.solid];
+ this.craft.modify({
+ type: 'CUT',
+ solids : solids,
+ face : polyFace,
+ depth : depth
});
-
};
TCAD.App.prototype.refreshSketches = function() {
diff --git a/web/app/engine.js b/web/app/engine.js
index 331ecfc9..66eb323f 100644
--- a/web/app/engine.js
+++ b/web/app/engine.js
@@ -107,6 +107,12 @@ TCAD.utils.fixCCW = function(path, normal) {
return path;
};
+TCAD.utils.addAll = function(arr, arrToAdd) {
+ for (var i = 0; i < arrToAdd.length; i++) {
+ arr.push(arrToAdd[i]);
+ }
+};
+
TCAD.TOLERANCE = 1E-6;
TCAD.utils.areEqual = function(v1, v2, tolerance) {
@@ -320,7 +326,7 @@ TCAD.geom.extrude = function(source, target) {
return poly;
};
-TCAD.geom.FACE_COUNTER = 0;
+TCAD.geom.SOLID_COUNTER = 0;
/** @constructor */
TCAD.Solid = function(polygons, material) {
@@ -329,6 +335,9 @@ TCAD.Solid = function(polygons, material) {
this.meshObject = new THREE.Mesh(this, material);
+ this.id = TCAD.geom.SOLID_COUNTER ++;
+ this.faceCounter = 0;
+
this.polyFaces = [];
var scope = this;
function pushVertices(vertices) {
@@ -423,7 +432,7 @@ TCAD.SketchFace = function(solid, poly) {
var proto = poly.__face;
poly.__face = this;
if (proto === undefined) {
- this.id = TCAD.geom.FACE_COUNTER++;
+ this.id = solid.id + ":" + (solid.faceCounter++);
this.sketchGeom = null;
} else {
this.id = proto.id;
diff --git a/web/app/ui/toolkit.js b/web/app/ui/toolkit.js
new file mode 100644
index 00000000..f79ba70a
--- /dev/null
+++ b/web/app/ui/toolkit.js
@@ -0,0 +1,62 @@
+TCAD.toolkit = {};
+
+TCAD.toolkit.add = function(parent, child) {
+ parent.content.append(child.root);
+};
+
+TCAD.toolkit.Box = function() {
+ this.root = this.content = $('
');
+ this.root.addClass('tc-box');
+ this.root.appendTo('body');
+};
+
+TCAD.toolkit.Folder = function(title) {
+ this.root = $('', {class: 'tc-folder'});
+ this.content = $('', {class: 'tc-scroll'});
+ this.root.append($('', {text: title, class: 'tc-row tc-title'}));
+ this.root.append(this.content);
+};
+
+TCAD.toolkit.Button = function(title) {
+ this.root = $('',
+ {class: 'tc-row tc-ctrl tc-ctrl-btn', text: title});
+};
+
+TCAD.toolkit.propLayout = function(root, name, valueEl) {
+ root.append($('', {class: 'tc-prop-name', text: name}))
+ .append($('', {class: 'tc-prop-value'})
+ .append(valueEl));
+};
+
+TCAD.toolkit.Number = function(name) {
+ this.root = $('', {class: 'tc-row tc-ctrl tc-ctrl-number'});
+ TCAD.toolkit.propLayout(this.root, name, $(''))
+};
+
+TCAD.toolkit.Text = function(name) {
+ this.root = $('', {class: 'tc-row tc-ctrl tc-ctrl-text'});
+ TCAD.toolkit.propLayout(this.root, name, $(''))
+};
+
+TCAD.toolkit.Tree = function() {
+ this.root = $('', {class: 'tc-tree'});
+};
+
+TCAD.toolkit.Tree.prototype.set = function(data) {
+ this.root.empty();
+ this._fill(this.root, data.children);
+};
+
+TCAD.toolkit.Tree.prototype._fill = function(parent, data) {
+ parent.append($('', {text : data.name}));
+ if (data.children !== undefined && data.children.length !== 0) {
+ var ul = $('');
+ parent.append(ul);
+ for (var i = 0; i < data.children.length; i++) {
+ var li = $('');
+ ul.append(li);
+ var child = data.children[i];
+ this._fill(li, child);
+ }
+ }
+};
diff --git a/web/app/workbench.js b/web/app/workbench.js
index 7eb19b10..4a532765 100644
--- a/web/app/workbench.js
+++ b/web/app/workbench.js
@@ -72,38 +72,6 @@ TCAD.workbench.serializeSolid = function(solid) {
return data;
};
-TCAD.workbench.applyHistory = function(history) {
-
- for (var hi = 0; hi < history.length; ++hi) {
- var mod = history[hi];
- switch (mod.operation) {
- }
-
- }
-};
-
-TCAD.workbench.Cut = function() {
-
- this.depth = null;
-
- this.load = function(data) {
- this.depth = data.depth;
- };
-
- this.save = function() {
- return {
- depth : this.depth
- };
- };
-
- this.apply = function(app, face, faces) {
- TCAD.craft.cut(app, face, faces, this.depth);
- };
-};
-
-TCAD.workbench.Cut.prototype.TYPE = 'CUT';
-
-
TCAD.craft = {};
TCAD.craft.getSketchedPolygons3D = function(app, face) {
@@ -142,16 +110,18 @@ TCAD.craft.getSketchedPolygons3D = function(app, face) {
return sketchedPolygons;
};
-TCAD.craft.extrude = function(app, face, faces, height) {
+TCAD.craft.extrude = function(app, request) {
+ var face = request.face;
var sketchedPolygons = TCAD.craft.getSketchedPolygons3D(app, face);
if (sketchedPolygons == null) return null;
- var newSolidFaces = [];
+ var faces = TCAD.craft.collectFaces(request.solids);
+
var normal = face.polygon.normal;
var toMeldWith = [];
for (var i = 0; i < sketchedPolygons.length; i++) {
- var extruded = TCAD.geom.extrude(sketchedPolygons[i], normal.multiply(height));
+ var extruded = TCAD.geom.extrude(sketchedPolygons[i], normal.multiply(request.height));
toMeldWith = toMeldWith.concat(TCAD.craft._makeFromPolygons(extruded));
}
var work = TCAD.craft._makeFromPolygons(faces.map(function(f){ return f.polygon }));
@@ -218,7 +188,7 @@ TCAD.craft._mergeCSGPolygons = function (__cgsPolygons, allPoints) {
vertices: cp.vertices.map(function (cv) {
return vec(cv.pos)
}),
- csgInfo : cp.shared.__tcad,
+ shared : cp.shared.__tcad,
normal: vec(cp.plane.normal),
w: cp.plane.w
};
@@ -450,7 +420,7 @@ TCAD.craft._mergeCSGPolygons = function (__cgsPolygons, allPoints) {
vertices : path,
normal : csgData.normal,
w : csgData.w,
- csgInfo : csgData.csgInfo
+ shared : csgData.shared
});
}
}
@@ -526,7 +496,10 @@ TCAD.craft._makeFromPolygons = function(polygons) {
}
var pid = poly.id;
var shared = new CSG.Polygon.Shared([pid, pid, pid, pid]);
- shared.__tcad = poly.csgInfo;
+ shared.__tcad = {
+ csgInfo : poly.csgInfo,
+ face : poly.__face
+ };
var refs = poly.triangulate();
for ( var i = 0; i < refs.length; ++ i ) {
var a = refs[i][0] + off;
@@ -628,7 +601,7 @@ TCAD.craft.reconstruct = function (cut) {
return {
vertices : path.vertices.map(function(v) {return tr.apply(v);}),
normal : path.normal,
- csgInfo : path.csgInfo
+ shared : path.shared
}
});
@@ -715,7 +688,10 @@ TCAD.craft.reconstruct = function (cut) {
var p = new TCAD.Polygon(path.vertices, path.holes.map(function (path) {
return path.vertices
}), path.normal);
- p.csgInfo = path.csgInfo;
+ if (path.shared !== undefined) {
+ p.csgInfo = path.shared.csgInfo;
+ p.__face = path.shared.face;
+ }
return p;
})
);
@@ -724,17 +700,25 @@ TCAD.craft.reconstruct = function (cut) {
return result;
};
-TCAD.craft.cut = function(app, face, faces, depth) {
+TCAD.craft.collectFaces = function(solids) {
+ var faces = [];
+ for (var i = 0; i < solids.length; i++) {
+ TCAD.utils.addAll(faces, solids[i].polyFaces);
+ }
+ return faces;
+};
+TCAD.craft.cut = function(app, request) {
+ var face = request.face;
var sketchedPolygons = TCAD.craft.getSketchedPolygons3D(app, face);
if (sketchedPolygons == null) return null;
- var newSolidFaces = [];
+ var faces = TCAD.craft.collectFaces(request.solids);
var normal = face.polygon.normal;
var cutter = [];
for (var i = 0; i < sketchedPolygons.length; i++) {
- var extruded = TCAD.geom.extrude(sketchedPolygons[i], normal.multiply( - depth));
+ var extruded = TCAD.geom.extrude(sketchedPolygons[i], normal.multiply( - request.depth));
cutter = cutter.concat(TCAD.craft._makeFromPolygons(extruded));
}
var work = TCAD.craft._makeFromPolygons(faces.map(function(f){ return f.polygon }));
@@ -754,15 +738,45 @@ TCAD.Craft = function(app) {
TCAD.Craft.prototype.current = function() {
return this.history[this.history.length - 1];
};
-
-TCAD.Craft.prototype.modify = function(solid, modification) {
- var faces = modification();
- if (faces == null) return;
- this.app.viewer.scene.remove( solid.meshObject );
- this.app.viewer.scene.add(TCAD.utils.createSolidMesh(faces));
+TCAD.craft.detach = function(request) {
+ var detachedConfig = {};
+ for (var prop in request) {
+ if (request.hasOwnProperty(prop)) {
+ var value = request[prop];
+ if (typeof(value) === 'object' && value.id !== undefined) {
+ detachedConfig[prop] = value.id;
+ } else {
+ detachedConfig[prop] = value;
+ }
+ }
+ }
+ return detachedConfig
+};
+
+TCAD.Craft.prototype.modify = function(request) {
+
+ var op = TCAD.craft.OPS[request.type];
+ if (!op) return;
+
+ var detachedRequest = TCAD.craft.detach(request);
+ var newFaces = op(this.app, request);
+
+ if (newFaces == null) return;
+ for (var i = 0; i < request.solids.length; i++) {
+ var solid = request.solids[i];
+ this.app.viewer.scene.remove( solid.meshObject );
+ }
+ this.app.viewer.scene.add(TCAD.utils.createSolidMesh(newFaces));
+ this.history.push(detachedRequest);
+ this.app.bus.notify('operation');
//REMOVE IT
this.app._refreshSketches();
this.app.viewer.render();
};
+
+TCAD.craft.OPS = {
+ CUT : TCAD.craft.cut,
+ PAD : TCAD.craft.extrude
+};
diff --git a/web/css/toolkit.css b/web/css/toolkit.css
new file mode 100644
index 00000000..a1b34807
--- /dev/null
+++ b/web/css/toolkit.css
@@ -0,0 +1,99 @@
+
+.tc-box {
+ position: absolute;
+ margin-left: 0; margin-top: 0;
+ top: 0; left: 0;
+ width : 250px; height: 300px;
+ background-color: #000;
+}
+
+.tc-row {
+ height: 27px;
+ line-height: 27px;
+ overflow: hidden;
+ padding: 0 4px 0 5px;
+ border-bottom: 1px solid #2c2c2c;
+}
+
+.tc-title {
+ padding-left: 16px;
+ background: black url() 6px 10px no-repeat;
+ cursor: pointer;
+}
+
+.tc-folder {
+ color: #eee;
+ font: 11px 'Lucida Grande', sans-serif;
+ background-color: #1a1a1a;
+}
+
+.tc-ctrl {
+ border-left: 3px solid #e61d5f;
+}
+
+.tc-ctrl-btn {
+ border-left: 3px solid #e61d5f;
+ cursor: pointer;
+}
+
+.tc-ctrl-btn:hover {
+ background-color: #000;
+}
+
+.tc-prop-name {
+ cursor: default;
+ float: left;
+ clear: left;
+ width: 40%;
+ overflow: hidden;
+ text-overflow: ellipsis;
+}
+
+.tc-prop-value {
+ float: left;
+ width: 60%;
+}
+
+.tc-ctrl input[type=text] {
+ color: #2fa1d6;
+ background: #303030;
+ outline: none;
+ border: 0;
+ margin-top: 4px;
+ padding: 3px;
+ width: 100%;
+ float: right;
+}
+
+.tc-ctrl-text input[type=text] {
+ color:#1ed36f;
+}
+
+.tc-ctrl input[type=text]:focus {
+ color: #eee;
+}
+
+.tc-scroll {
+ overflow-y: auto;
+ overflow-x: hidden;
+}
+
+.tc-scroll::-webkit-scrollbar {
+ width: 2px;
+}
+
+.tc-scroll::-webkit-scrollbar-track {
+ background:#eee;
+ border: thin solid lightgray;
+ box-shadow: 0px 0px 3px #dfdfdf inset;
+ border-radius:10px;
+}
+.tc-scroll::-webkit-scrollbar-thumb {
+ background:#999;
+ border: thin solid gray;
+ border-radius:10px;
+}
+
+.tc-scroll::-webkit-scrollbar-thumb:hover {
+ background:#7d7d7d;
+}
diff --git a/web/index.html b/web/index.html
index 32d18711..ce488e0f 100644
--- a/web/index.html
+++ b/web/index.html
@@ -8,7 +8,10 @@
overflow: hidden;
}
+
+
+
@@ -24,6 +27,7 @@
+