jsketcher/web/app/sketcher/shapes/dim.js
2016-09-08 22:08:36 -07:00

402 lines
9.8 KiB
JavaScript

import * as utils from '../../utils/utils'
import * as math from '../../math/math'
import {SketchObject, EndPoint} from '../viewer2d'
import Vector from '../../math/vector'
/** @constructor */
function LinearDimension(a, b) {
SketchObject.call(this);
this.a = a;
this.b = b;
this.flip = false;
}
utils.extend(LinearDimension, SketchObject);
LinearDimension.prototype.collectParams = function(params) {
};
LinearDimension.prototype.getReferencePoint = function() {
return this.a;
};
LinearDimension.prototype.translateImpl = function(dx, dy) {
};
LinearDimension.prototype.getA = function() { return this.a };
LinearDimension.prototype.getB = function() { return this.b };
LinearDimension.getTextOff = function(scale) {
return 3 * scale;
};
LinearDimension.prototype.drawImpl = function(ctx, scale, viewer) {
var off = 30 * viewer.dimScale;
var textOff = LinearDimension.getTextOff(viewer.dimScale);
var a, b, startA, startB;
if (this.flip) {
a = this.getB();
b = this.getA();
startA = this.b;
startB = this.a;
} else {
a = this.getA();
b = this.getB();
startA = this.a;
startB = this.b;
}
var d = math.distanceAB(a, b);
var _vx = - (b.y - a.y);
var _vy = b.x - a.x;
//normalize
var _vxn = _vx / d;
var _vyn = _vy / d;
_vx = _vxn * off;
_vy = _vyn * off;
ctx.beginPath();
var _ax = a.x + _vx;
var _ay = a.y + _vy;
var _bx = b.x + _vx;
var _by = b.y + _vy;
ctx.moveTo(_ax, _ay);
ctx.lineTo(_bx, _by);
function drawRef(start, x, y) {
var vec = new Vector(x - start.x, y - start.y);
vec._normalize();
vec._multiply(7 * viewer.dimScale);
ctx.moveTo(start.x, start.y );
ctx.lineTo(x, y);
ctx.lineTo(x + vec.x, y + vec.y);
}
drawRef(startA, _ax, _ay);
drawRef(startB, _bx, _by);
ctx.closePath();
ctx.stroke();
function drawArrow(x, y) {
var s1 = 50;
var s2 = 20;
ctx.lineCap = 'round';
ctx.beginPath();
ctx.moveTo(x, y);
ctx.lineTo(x - s1, y - s2);
ctx.closePath();
ctx.stroke();
}
// drawArrow(_ax, _ay);
// drawArrow(_bx, _by);
ctx.font= (12 * viewer.dimScale) + "px Arial";
var txt = d.toFixed(2);
var h = d / 2 - ctx.measureText(txt).width / 2;
if (h > 0) {
var tx = (_ax + _vxn * textOff) - (- _vyn) * h;
var ty = (_ay + _vyn * textOff) - ( _vxn) * h;
ctx.save();
ctx.translate(tx, ty);
ctx.rotate( - Math.atan2(_vxn, _vyn));
ctx.scale(1, -1);
ctx.fillText(txt, 0, 0);
ctx.restore();
}
};
LinearDimension.prototype.normalDistance = function(aim) {
return -1;
};
/** @constructor */
function Dimension(a, b) {
LinearDimension.call(this, a, b);
}
utils.extend(Dimension, LinearDimension);
Dimension.prototype._class = 'TCAD.TWO.Dimension';
/** @constructor */
function HDimension(a, b) {
LinearDimension.call(this, a, b);
}
utils.extend(HDimension, LinearDimension);
HDimension.prototype._class = 'TCAD.TWO.HDimension';
HDimension.prototype.getA = function() { return this.a };
HDimension.prototype.getB = function() { return {x : this.b.x, y : this.a.y} };
/** @constructor */
function VDimension(a, b) {
LinearDimension.call(this, a, b);
}
utils.extend(VDimension, LinearDimension);
VDimension.prototype._class = 'TCAD.TWO.VDimension';
VDimension.prototype.getA = function() { return this.a };
VDimension.prototype.getB = function() { return {x : this.a.x, y : this.b.y} };
/** @constructor */
function DiameterDimension(obj) {
SketchObject.call(this);
this.obj = obj;
this.angle = Math.PI / 4;
}
DiameterDimension.prototype._class = 'TCAD.TWO.DiameterDimension';
utils.extend(DiameterDimension, SketchObject);
DiameterDimension.prototype.collectParams = function(params) {
};
DiameterDimension.prototype.getReferencePoint = function() {
};
DiameterDimension.prototype.translateImpl = function(dx, dy) {
};
DiameterDimension.prototype.drawImpl = function(ctx, scale, viewer) {
if (this.obj == null) return;
if (this.obj._class === 'TCAD.TWO.Circle') {
this.drawForCircle(ctx, scale, viewer);
} else if (this.obj._class === 'TCAD.TWO.Arc') {
this.drawForArc(ctx, scale, viewer);
}
};
DiameterDimension.prototype.drawForCircle = function(ctx, scale, viewer) {
var c = new Vector().setV(this.obj.c);
var r = this.obj.r.get();
var angled = new Vector(r * Math.cos(this.angle), r * Math.sin(this.angle), 0);
var a = c.minus(angled);
var b = c.plus(angled);
var textOff = LinearDimension.getTextOff(viewer.dimScale);
var d = 2 * r;
ctx.beginPath();
ctx.moveTo(a.x, a.y);
ctx.lineTo(b.x, b.y);
ctx.closePath();
ctx.stroke();
var fontSize = 12 * viewer.dimScale;
ctx.font = (fontSize) + "px Arial";
var txt = String.fromCharCode(216) + ' ' + d.toFixed(2);
var textWidth = ctx.measureText(txt).width;
var h = d / 2 - textWidth / 2;
var _vx = - (b.y - a.y);
var _vy = b.x - a.x;
//normalize
var _vxn = _vx / d;
var _vyn = _vy / d;
function drawText(tx, ty) {
ctx.save();
ctx.translate(tx, ty);
ctx.rotate(-Math.atan2(_vxn, _vyn));
ctx.scale(1, -1);
ctx.fillText(txt, 0, 0);
ctx.restore();
}
var tx, ty;
if (h - fontSize * .3 > 0) { // take into account font size to not have circle overlap symbols
tx = (a.x + _vxn * textOff) - (-_vyn) * h;
ty = (a.y + _vyn * textOff) - ( _vxn) * h;
drawText(tx, ty);
} else {
var off = 2 * viewer.dimScale;
angled._normalize();
var extraLine = angled.multiply(textWidth + off * 2);
ctx.beginPath();
ctx.moveTo(b.x, b.y);
ctx.lineTo(b.x + extraLine.x, b.y + extraLine.y);
ctx.closePath();
ctx.stroke();
angled._multiply(off);
tx = (b.x + _vxn * textOff) + angled.x;
ty = (b.y + _vyn * textOff) + angled.y;
drawText(tx, ty);
}
};
DiameterDimension.prototype.drawForArc = function(ctx, scale, viewer) {
var r = this.obj.distanceA();
var hxn = Math.cos(this.angle);
var hyn = Math.sin(this.angle);
var vxn = - hyn;
var vyn = hxn;
//fix angle if needed
if (!this.obj.isPointInsideSector(this.obj.c.x + hxn, this.obj.c.y + hyn)) {
var cosA = hxn * (this.obj.a.x - this.obj.c.x) + hyn * (this.obj.a.y - this.obj.c.y);
var cosB = hxn * (this.obj.b.x - this.obj.c.x) + hyn * (this.obj.b.y - this.obj.c.y);
if (cosA - hxn > cosB - hxn) {
this.angle = this.obj.getStartAngle();
} else {
this.angle = this.obj.getEndAngle();
}
}
var vertOff = LinearDimension.getTextOff(viewer.dimScale);
var horOff = 5 * viewer.dimScale;
var fontSize = 12 * viewer.dimScale;
ctx.font = (fontSize) + "px Arial";
var txt = 'R ' + r.toFixed(2);
var textWidth = ctx.measureText(txt).width;
var startX = this.obj.c.x + hxn * r;
var startY = this.obj.c.y + hyn * r;
var lineLength = textWidth + horOff * 2;
ctx.beginPath();
ctx.moveTo(startX, startY);
ctx.lineTo(startX + hxn * lineLength, startY + hyn * lineLength);
ctx.closePath();
ctx.stroke();
var tx = startX + vxn * vertOff + hxn * horOff;
var ty = startY + vyn * vertOff + hyn * horOff;
ctx.save();
ctx.translate(tx, ty);
ctx.rotate(-Math.atan2(vxn, vyn));
ctx.scale(1, -1);
ctx.fillText(txt, 0, 0);
ctx.restore();
};
DiameterDimension.prototype.normalDistance = function(aim) {
return -1;
};
/** @constructor */
function AddDimTool(viewer, layer, dimCreation) {
this.viewer = viewer;
this.layer = layer;
this.dim = null;
this._v = new Vector(0, 0, 0);
this.dimCreation = dimCreation;
}
AddDimTool.prototype.keydown = function(e) {};
AddDimTool.prototype.keypress = function(e) {};
AddDimTool.prototype.keyup = function(e) {};
AddDimTool.prototype.cleanup = function(e) {};
AddDimTool.prototype.mousemove = function(e) {
var p = this.viewer.screenToModel(e);
this.viewer.snap(p.x, p.y, []);
if (this.dim != null) {
this.dim.b.x = p.x;
this.dim.b.y = p.y;
}
this.viewer.refresh();
};
AddDimTool.prototype.mouseup = function(e) {
if (e.button > 0 && this.dim != null) {
this.dim.flip = !this.dim.flip;
this.viewer.refresh();
return;
}
if (this.viewer.snapped.length == 0) {
return;
}
var p = this.viewer.snapped.pop();
this.viewer.cleanSnap();
if (this.dim == null) {
this.viewer.historyManager.checkpoint();
this.dim = this.dimCreation(p, new EndPoint(p.x, p.y));
this.layer.objects.push(this.dim);
this.viewer.refresh();
} else {
this.dim.b = p;
this.viewer.toolManager.releaseControl();
this.viewer.refresh();
}
};
AddDimTool.prototype.mousedown = function(e) {
};
AddDimTool.prototype.mousewheel = function(e) {
};
/** @constructor */
function AddCircleDimTool(viewer, layer) {
this.viewer = viewer;
this.layer = layer;
this.dim = new DiameterDimension(null);
this.viewer.add(this.dim, this.layer);
}
AddCircleDimTool.prototype.keydown = function(e) {};
AddCircleDimTool.prototype.keypress = function(e) {};
AddCircleDimTool.prototype.keyup = function(e) {};
AddCircleDimTool.prototype.cleanup = function(e) {};
AddCircleDimTool.prototype.mousemove = function(e) {
var p = this.viewer.screenToModel(e);
var objects = this.viewer.search(p.x, p.y, 20 / this.viewer.scale, true, false, []).filter(function(o) {
return o._class === 'TCAD.TWO.Circle' || o._class === 'TCAD.TWO.Arc';
});
if (objects.length != 0) {
this.dim.obj = objects[0];
} else {
this.dim.obj = null;
}
if (this.dim.obj != null) {
this.dim.angle = Math.atan2(p.y - this.dim.obj.c.y, p.x - this.dim.obj.c.x);
}
this.viewer.refresh();
};
AddCircleDimTool.prototype.mouseup = function(e) {
if (this.dim.obj !== null) {
this.viewer.historyManager.checkpoint();
} else {
this.viewer.remove(this.dim);
}
this.viewer.toolManager.releaseControl();
};
AddCircleDimTool.prototype.mousedown = function(e) {
};
AddCircleDimTool.prototype.mousewheel = function(e) {
};
export {AddDimTool, AddCircleDimTool, HDimension, VDimension, Dimension, DiameterDimension}