Calculate other angles based on known rotation angle - math
I am making a mechanical system animation on OpenGL and having a little trouble calculating the rotations angle of the connecting rods based on a known rotation angle A and the position of the point D.
I need to calculate the angle CDE and CBG as well as the position of point E based on angle A and the position of D. But my high school is failing me right now. I have tried several ways but they all lead to nothing.
The length of segment DA is also known.
Do you have any ideas on how to do that? What should I do?
I have had to make a few assumptions, and while I was finding a solution I forgot to check the labels so below is an image to clarify point and line name, including in red the geometry used to solve.
Assumptions.
Points A and B are fixed.
Lines BC, FC, and DC are all the same length L
Point D is constrained to the line EG
Angle not labeled is the angle you refer to in the question.
The point F is on the circle centered at A. I forgot to label the radius and angle.
Point A is at the origin {x: 0, y: 0}
I also assume that you know the basics of vector math and that the problem is not finding the angle between lines or vectors, but rather solving to find the points C and D that is giving you troubles (hope so as this is going to be a long answer for me).
Solving
Depending on the value of L and the position of the constraining line EG there may not be a solution for all positions of F. The method below will result in either some values being NaN or the position of D will be incorrect.
Find C
Easy start. As A is at the origin then F is at F.x = cos(angle) * radius, F.y = sin(angle) * radius
Now find the mid m point on the line FB and the length of the line Bm as b
This forms the right triangle mBC and we know the length of BC === L and just calculated length of line Bm === b thus the length of the line mC is (L * L - b * b) ** 0.5
Create a unit vector (normalized) from F to B, rotate it clockwise 90 deg and scale it by the calculated length of mC. Add that vector to the point m and you have C
// vector
nx = B.x - F.x;
ny = B.y - F.y;
// Normalize, scale, rotate and add to m to get C. shorthand
// mC len of line mC
s = mC / (nx * nx + ny * ny) ** 0.5;
C.x = m.x - ny * s;
C.y = m.y + nx * s;
// OR in steps
// normalize
len = (nx * nx + ny * ny) ** 0.5;
nx /= len;
ny /= len;
// scale to length of mC
nx *= mC;
ny *= mC;
// rotated 90CW and add to m to get C
C.x = m.x - ny;
C.y = m.y + nx;
Find D
Now that we have the point C we know that the point D is on the constraining line EG. Thus we know that the point D is at the point where a circle at C or radius L intercepts the line EG
However there are two solutions to the intercept of circle and line, the point B is at one of these points if B is on the line EG. If B is not on the line EG then you will have to pick which of the two solutions you want. Likely the point D is the furthest from B
There are several methods to find the intercepts of a line and a circle. The following is a little more complex but will help when picking which point to use
// line EG as vec
vxA = G.x - E.x;
vyA = G.y - E.y;
// square of length line EG
lenA = vxA * vxA + vyA * vyA;
// vector from E to C
vxB = C.x - E.x;
vyB = C.y - E.y;
// square of length line EC
lenB = vxB * vxB + vyB * vyB;
// dot product A.B * - 2
b = -2 * (vxB * vxA + vyB * vyA);
// Stuff I forget what its called
d = (b * b - 4 * lenA * (lenB - L * L)) ** 0.5; // L is length of CD
// is there a solution if not we are done
if (isNaN(d)) { return }
// there are two solution (even if the same point)
// Solutions as unit distances along line EG
u1 = (b - d) / (2 * lenA);
u2 = (b + d) / (2 * lenA); // this is the one we want
The second unit distance is the one that will fit your layout example. So now we just find the point at u2 on the line EG and we have the final point D
D.x = E.x + u2 * (G.x - E.x);
D.y = E.y + u2 * (G.y - E.y);
The angles
In your question it is a little ambiguous to me which angles you want. So I will just give you a method to find the angle between to lines. Eg CB and CD
Convert both lines to vectors. The cross product of these vectors divided by the square root of the product of the squared lengths gives us the sin of the angle. However we still need the quadrant. We workout which quadrant by checking the sign of the dot product of the two vectors.
Note this method will find the the smallest angle between the two lines and is invariant to the order of the lines
Note the angle is in radians
// vector CB
xA = B.x - C.x;
yA = B.y - C.y;
// vector CD
xB = D.x - C.x;
yB = D.y - C.y;
// square root of the product of the squared lengths
l = ((xa * xa + ya * ya) * (xb * xb + yb * yb)) ** 0.5;
// if this is 0 then angle between lines is 0
if (l === 0) { return 0 } // return angle
angle = Math.asin((xa * yb - ya * xb) / l); // get angle quadrant undefined
// if dot of the vectors is < 0 then angle is in quadrants 2 or 3. get angle and return
if (xa * xb + ya * yb < 0) {
return (angle< 0 ? -Math.PI: Math.PI) - angle;
}
// else the angle is in quads 1 or 4 so just return the angle
return angle;
DONE
To make sure it all worked I have created an interactive diagram. The code of interest is at the top. Variables names are as in my diagram at top of answer. Most of the code is just cut and paste vector libs and UI stuff unrelated to the answer.
To use
The diagram will scale to fit the page so click full page if needed.
Use the mouse to drag points with white circles around. For example to rotate F around A click and drag it.
The white line segment El sets the length of the lines CF, CB, CD. The radius of circle at A is set by moving the white circle point to the right of it.
Move mouse out of form to animate.
Mouse only interface.
Overkill but its done.
setTimeout(() => {
// points and lines as in diagram of answer
const A = new Vec2(-100,100);
const B = new Vec2(-240, - 100);
const C = new Vec2();
const D = new Vec2();
const E = new Vec2(-300, -100);
const F = new Vec2();
const G = new Vec2(200, -100);
const AF = new Line2(A, F), FA = new Line2(F, A);
const BC = new Line2(B, C), CB = new Line2(C, B);
const CD = new Line2(C, D), DC = new Line2(D, C);
const EG = new Line2(E, G), GE = new Line2(G, E);
const FB = new Line2(F, B), BF = new Line2(B, F);
const FC = new Line2(F, C), CF = new Line2(C, F);
// Math to find points C and D
function findCandD() {
F.initPolar(angle, radius).add(A) // Get position of F
FB.unitDistOn(0.5, m); // Find point midway between F, B, store as m
// Using right triangle m, B, C the hypot BC length is L
var c = (FB.length * 0.5) ** 2; // Half the length of FB squared
const clLen = (L * L - c) ** 0.5 // Length of line mC
FB.asVec(v1).rotate90CW().length = clLen; // Create vector v1 at 90 from FB and length clLen
C.init(m).add(v1); // Add v1 to m to get point C
const I = EG.unitInterceptsCircle(C, L, cI); // Point D is L dist from
if (EG.unitInterceptsCircle(C, L, cI)) { // Point D is L dist from C. thus us the intercept of corcle radius L and constraining line EG
EG.unitDistanceOn(cI.y, D) // Use second intercept as first could be at point B
} else { C.x = NaN } // C is too far from constraining line EG for a solution
// At this point, the line CD may be the wrong length. Check the length CD is correct
blk = Math.isSmall(CD.length - L) ? black : red; // Mark all in red if no solution
}
// Here on down UI, and all the support code
requestAnimationFrame(update);
const refRes = 512;
var scale = 1;
const mousePos = new Vec2();
var w = 0, h = 0, cw = 0, ch = 0;
var frame = 0;
const m = new Vec2(); // holds mid point on line BF
const m1 = new Vec2();
const v1 = new Vec2(); // temp vector
const v2 = new Vec2(); // temp vector
const cI = new Vec2(); // circle intercepts
var radius = 100;
var L = 200
var angle = 1;
const aa = new Vec2(A.x + radius, A.y);
const al = new Vec2(E.x + L, E.y);
const rad = new Line2(A, aa);
const cl = new Line2(m, C)
const armLen = new Line2(E, al);
var blk = "#000"
const wht = "#FFF"
const red = "#F00"
const black = "#000"
const ui = Vecs2([A, B, aa, E, G, al, F])
function update(timer){
frame ++;
ctx.setTransform(1,0,0,1,0,0); // reset transform
if (w !== innerWidth || h !== innerHeight){
cw = (w = canvas.width = innerWidth) / 2;
ch = (h = canvas.height = innerHeight) / 2;
scale = Math.min(w / refRes, h / refRes);
} else {
ctx.clearRect(0, 0, w, h);
}
ctx.clearRect(0, 0, canvas.width, canvas.height);
mousePos.init(mouse);
mousePos.x = (mousePos.x - canvas.width / 2) / scale;
mousePos.y = (mousePos.y -canvas.height / 2) / scale;
mousePos.button = mouse.button;
ctx.font = "24px Arial black"
ctx.textAlign = "center";
ctx.setTransform(scale,0,0,scale,canvas.width / 2, canvas.height / 2);
const nearest = ui.dragable(mousePos, 20);
if (nearest === A) {
aa.y = A.y
aa.x = A.x + radius;
} else if(nearest === F){
angle = A.directionTo(F);
} else if(nearest === aa){
aa.y = A.y
radius = rad.length;
} else if (nearest === E) {
EG.distanceAlong(L, al)
} else if (nearest === G || nearest === al) {
EG.nearestOnLine(al, al)
L = armLen.length;
}
if (nearest) {
canvas.style.cursor = ui.dragging ? "none" : "move";
nearest.draw(ctx, "#F00", 2, 4);
if (nearest.isLine2) {
nearest.nearestOnLine(mousePos, onLine).draw(ctx, "#FFF", 2, 2)
}
} else {
canvas.style.cursor = "default";
}
angle += SPEED;
findCandD();
ui.mark(ctx, wht, 1, 4);
ui.mark(ctx, wht, 1, 14);
armLen.draw(ctx,wht,2)
EG.draw(ctx, wht, 1)
ctx.fillStyle = wht;
ctx.fillText("E", E.x, E.y - 16)
ctx.fillText("G", G.x, G.y - 16)
ctx.fillText("l", armLen.p2.x, armLen.p2.y - 16)
FC.draw(ctx, blk, 4)
BC.draw(ctx, blk, 4)
CD.draw(ctx, blk, 4)
A.draw(ctx, blk, 2, radius);
C.draw(ctx, blk, 4, 4)
F.draw(ctx, blk, 4, 4)
B.draw(ctx, blk, 4, 4);
D.draw(ctx, blk, 4, 4)
ctx.fillStyle = blk;
ctx.fillText("B", B.x, B.y - 16)
ctx.fillText("A", A.x, A.y - 16)
ctx.fillText("F", F.x, F.y + 26)
ctx.fillText("D", D.x, D.y - 16)
ctx.fillText("C", C.x, C.y - 16)
ctx.font = "16px Arial";
drawAngle(C, CD, CB, 40, B.add(Vec2.Vec(60, -50), Vec2.Vec()), ctx, blk, 2);
drawAngle(C, CF, CB, 50, A.add(Vec2.Vec(-160, 0), Vec2.Vec()), ctx, blk, 2);
drawAngle(C, CD, CF, 60, A.add(Vec2.Vec(300, 20), Vec2.Vec()), ctx, blk, 2);
blk = Math.isSmall(CD.length - L) ? black : red;
requestAnimationFrame(update);
}
}, 0);
const ctx = canvas.getContext("2d");
const mouse = {x: 0, y: 0, ox: 0, oy: 0, button: false, callback: undefined}
function mouseEvents(e) {
const bounds = canvas.getBoundingClientRect();
mouse.x = e.pageX - bounds.left - scrollX;
mouse.y = e.pageY - bounds.top - scrollY;
mouse.button = e.type === "mousedown" ? true : e.type === "mouseup" ? false : mouse.button;
}
["down", "up", "move"].forEach(name => document.addEventListener("mouse" + name, mouseEvents));
var SPEED = 0.05;
canvas.addEventListener("mouseover",() => SPEED = 0);
canvas.addEventListener("mouseout",() => SPEED = 0.05);
Math.EPSILON = 1e-6;
Math.isSmall = val => Math.abs(val) < Math.EPSILON;
Math.isUnit = u => !(u < 0 || u > 1);
Math.uClamp = u => u <= 0 ? 0 : u >= 1 ? 1 : u; // almost 2* faster than Math.min, Math.Max method
Math.TAU = Math.PI * 2;
Math.rand = (m, M) => Math.random() * (M - m) + m;
Math.randI = (m, M) => Math.random() * (M - m) + m | 0;
Math.rad2Deg = r => r * 180 / Math.PI;
Math.symbols = {};
Math.symbols.degrees = "°";
/* export {Vec2, Line2} */ // this should be a module
var temp;
function Vec2(x = 0, y = (temp = x, x === 0 ? (x = 0 , 0) : (x = x.x, temp.y))) { this.x = x; this.y = y }
Vec2.Vec = (x, y) => ({x, y}); // Vec2 like
Vec2.prototype = {
isVec2: true,
init(x, y = (temp = x, x = x.x, temp.y)) { this.x = x; this.y = y; return this }, // assumes x is a Vec2 if y is undefined
initPolar(dir, length = (temp = dir, dir = dir.x, temp.y)) { this.x = Math.cos(dir) * length; this.y = Math.sin(dir) * length; return this },
toPolar(res = this) {
const dir = this.direction, len = this.length;
res.x = dir;
res.y = length;
return res;
},
zero() { this.x = this.y = 0; return this },
initUnit(dir) { this.x = Math.cos(dir); this.y = Math.sin(dir); return this },
copy() { return new Vec2(this) },
equal(v) { return (this.x - v.x) === 0 && (this.y - v.y) === 0 },
isUnits() { return Math.isUnit(this.x) && Math.isUnit(this.y) },
add(v, res = this) { res.x = this.x + v.x; res.y = this.y + v.y; return res },
addScaled(v, scale, res = this) { res.x = this.x + v.x * scale; res.y = this.y + v.y * scale; return res },
sub(v, res = this) { res.x = this.x - v.x; res.y = this.y - v.y; return res },
scale(val, res = this) { res.x = this.x * val; res.y = this.y * val; return res },
invScale(val, res = this) { res.x = this.x / val; res.y = this.y / val; return res },
dot(v) { return this.x * v.x + this.y * v.y },
uDot(v, div) { return (this.x * v.x + this.y * v.y) / div },
cross(v) { return this.x * v.y - this.y * v.x },
uCross(v, div) { return (this.x * v.y - this.y * v.x) / div },
get direction() { return Math.atan2(this.y, this.x) },
set direction(dir) { this.initPolar(dir, this.length) },
get length() { return this.lengthSqr ** 0.5 },
set length(l) { this.scale(l / this.length) },
get lengthSqr() { return this.x * this.x + this.y * this.y },
set lengthSqr(lSqr) { this.scale(lSqr ** 0.5 / this.length) },
distanceFrom(vec) { return ((this.x - vec.x) ** 2 + (this.y - vec.y) ** 2) ** 0.5 },
distanceSqrFrom(vec) { return ((this.x - vec.x) ** 2 + (this.y - vec.y) ** 2) },
directionTo(vec) { return Math.atan2(vec.y - this.y, vec.x - this.x) },
normalize(res = this) { return this.invScale(this.length, res) },
rotate90CW(res = this) {
const y = this.x;
res.x = -this.y;
res.y = y;
return res;
},
angleTo(vec) {
const xa = this.x, ya = this.y;
const xb = vec.x, yb = vec.y;
const l = ((xa * xa + ya * ya) * (xb * xb + yb * yb)) ** 0.5;
var ang = 0;
if (l !== 0) {
ang = Math.asin((xa * yb - ya * xb) / l);
if (xa * xb + ya * yb < 0) { return (ang < 0 ? -Math.PI: Math.PI) - ang }
}
return ang;
},
drawFrom(v, ctx, col = ctx.strokeStyle, lw = ctx.lineWidth, scale = 1) {
ctx.strokeStyle = col;
ctx.lineWidth = lw;
ctx.beginPath();
ctx.lineTo(v.x, v.y);
ctx.lineTo(v.x + this.x * scale, v.y + this.y * scale);
ctx.stroke();
},
draw(ctx, col = ctx.strokeStyle, lw = ctx.lineWidth, size = 4) {
ctx.strokeStyle = col;
ctx.lineWidth = lw;
ctx.beginPath();
ctx.arc(this.x, this.y, size, 0, Math.TAU);
ctx.stroke();
},
path(ctx, size) {
ctx.moveTo(this.x + size, this.y);
ctx.arc(this.x, this.y, size, 0, Math.TAU);
},
toString(digits = 3) { return "{x: " + this.x.toFixed(digits) + ", y: " + this.y.toFixed(digits) + "}" },
};
function Vecs2(vecsOrLength) {
const vecs2 = Object.assign([], Vecs2.prototype);
if (Array.isArray(vecsOrLength)) { vecs2.push(...vecsOrLength) }
else if (vecsOrLength && vecsOrLength >= 1) {
while (vecsOrLength-- > 0) { vecs2.push(new Vec2()) }
}
return vecs2;
}
Vecs2.prototype = {
isVecs2: true,
nearest(vec, maxDist = Infinity, tolerance = 1) { // max for argument semantic, used as semantic min in function
var found;
for (const v of this) {
const dist = v.distanceFrom(vec);
if (dist < maxDist) {
if (dist <= tolerance) { return v }
maxDist = dist;
found = v;
}
}
return found;
},
copy() {
var idx = 0;
const copy = Vecs2(this.length);
for(const p of this) { copy[idx++].init(p) }
return copy;
},
uniformTransform(rMat, pMat, res = this) {
var idx = 0;
for(const p of this) { p.uniformTransform(rMat, pMat, res[idx++]) }
},
mark(ctx, col = ctx.strokeStyle, lw = ctx.lineWidth, size = 4) {
ctx.strokeStyle = col;
ctx.lineWidth = lw;
ctx.beginPath();
for (const p of this) { p.path(ctx, size) }
ctx.stroke();
},
draw(ctx, close = false, col = ctx.strokeStyle, lw = ctx.lineWidth) {
ctx.strokeStyle = col;
ctx.lineWidth = lw;
ctx.beginPath();
for (const p of this) { ctx.lineTo(p.x, p.y) }
close && ctx.closePath();
ctx.stroke();
},
path(ctx, first = true) {
for (const p of this) {
if (first) {
first = false;
ctx.moveTo(p.x, p.y);
} else { ctx.lineTo(p.x, p.y) }
}
},
dragable(mouse, maxDist = Infinity, tolerance = 1) {
var near;
if (this.length) {
if (!this.dragging) {
if (!this.offset) { this.offset = new Vec2() }
near = this.nearest(this.offset.init(mouse), maxDist, tolerance); // mouse may not be a Vec2
if (near && mouse.button) {
this.dragging = near;
this.offset.init(near).sub(mouse);
}
}
if (this.dragging) {
near = this.dragging;
if (mouse.button) { this.dragging.init(mouse).add(this.offset) }
else { this.dragging = undefined }
}
}
return near;
}
}
function Line2(p1 = new Vec2(), p2 = (temp = p1, p1 = p1.p1 ? p1.p1 : p1, temp.p2 ? temp.p2 : new Vec2())) {
this.p1 = p1;
this.p2 = p2;
}
Line2.prototype = {
isLine2: true,
init(p1, p2 = (temp = p1, p1 = p1.p1, temp.p2)) { this.p1.init(p1); this.p2.init(p2) },
copy() { return new Line2(this) },
asVec(res = new Vec2()) { return this.p2.sub(this.p1, res) },
unitDistOn(u, res = new Vec2()) { return this.p2.sub(this.p1, res).scale(u).add(this.p1) },
unitDistanceOn(u, res = new Vec2()) { return this.p2.sub(this.p1, res).scale(u).add(this.p1) },
distAlong(dist, res = new Vec2()) { return this.p2.sub(this.p1, res).uDot(res, res.length).add(this.p1) },
distanceAlong(dist, res = new Vec2()) { return this.p2.sub(this.p1, res).scale(dist / res.length).add(this.p1) },
get length() { return this.lengthSqr ** 0.5 },
get lengthSqr() { return (this.p1.x - this.p2.x) ** 2 + (this.p1.y - this.p2.y) ** 2 },
get direction() { return this.asVec(wV2).direction },
translate(vec, res = this) {
this.p1.add(vec, res.p1);
this.p2.add(vec, res.p2);
return res;
},
reflect(line, u, res = line) {
this.asVec(wV2).normalize();
line.asVec(wV1);
line.unitDistOn(u, res.p1);
const d = wV1.uDot(wV2, 0.5);
wV3.init(wV2.x * d - wV1.x, wV2.y * d - wV1.y);
res.p1.add(wV3.scale(1 - u), res.p2);
return res;
},
reflectAsUnitVec(line, u, res = new Vec2()) {
this.asVec(res).normalize();
line.asVec(wV1);
return res.scale(wV1.uDot(res, 0.5)).sub(wV1).normalize()
},
angleTo(line) { return this.asVec(wV1).angleTo(line.asVec(wV2)) },
translateNormal(amount, res = this) {
this.asVec(wV1).rot90CW().length = -amount;
this.translate(wV1, res);
return res;
},
distanceNearestVec(vec) { // WARNING!! distanceLineFromVec is (and others are) dependent on vars used in this function
return this.asVec(wV1).uDot(vec.sub(this.p1, wV2), wV1.length);
},
unitNearestVec(vec) { // WARNING!! distanceLineFromVec is (and others are) dependent on vars used in this function
return this.asVec(wV1).uDot(vec.sub(this.p1, wV2), wV1.lengthSqr);
},
nearestOnLine(vec, res = new Vec2()) { return this.p1.addScaled(wV1, this.unitNearestVec(vec), res) },
nearestOnSegment(vec, res = new Vec2()) { return this.p1.addScaled(wV1, Math.uClamp(this.unitNearestVec(vec)), res) },
distanceLineFromVec(vec) { return this.nearestOnLine(vec, wV1).sub(vec).length },
distanceSegmentFromVec(vec) { return this.nearestOnSegment(vec, wV1).sub(vec).length },
unitInterceptsLine(line, res = new Vec2()) { // segments
this.asVec(wV1);
line.asVec(wV2);
const c = wV1.cross(wV2);
if (Math.isSmall(c)) { return }
wV3.init(this.p1).sub(line.p1);
res.init(wV1.uCross(wV3, c), wV2.uCross(wV3, c));
return res;
},
unitInterceptsCircle(point, radius, res = new Vec2()) {
this.asVec(wV1);
var b = -2 * this.p1.sub(point, wV2).dot(wV1);
const c = 2 * wV1.lengthSqr;
const d = (b * b - 2 * c * (wV2.lengthSqr - radius * radius)) ** 0.5
if (isNaN(d)) { return }
return res.init((b - d) / c, (b + d) / c);
},
draw(ctx, col = ctx.strokeStyle, lw = ctx.lineWidth) {
ctx.strokeStyle = col;
ctx.lineWidth = lw;
ctx.beginPath();
ctx.lineTo(this.p1.x, this.p1.y);
ctx.lineTo(this.p2.x, this.p2.y);
ctx.stroke();
},
path(ctx) {
ctx.moveTo(this.p1.x, this.p1.y);
ctx.lineTo(this.p2.x, this.p2.y);
},
toString(digits = 3) { return "{ p1: " + this.p1.toString(digits) + ", p2: " + this.p2.toString(digits) + "}" },
};
const wV1 = new Vec2(), wV2 = new Vec2(), wV3 = new Vec2(); // pre allocated work vectors used by Line2 functions
const wVA1 = new Vec2(), wVA2 = new Vec2(), wVA3 = new Vec2(); // pre allocated work vectors
const wVL1 = new Vec2(), wVL2 = new Vec2(), wVL3 = new Vec2(); // pre allocated work vectors used by Line2Array functions
const wL1 = new Line2(), wL2 = new Line2(), wL3 = new Line2(); // pre allocated work lines
function drawLable(text, from, to, ctx, col = ctx.strokeStyle, lw = ctx.lineWidth) {
ctx.fillStyle = ctx.strokeStyle = col;
ctx.lineWidth = lw;
ctx.beginPath();
ctx.lineTo(from.x, from.y);
ctx.lineTo(to.x, to.y);
ctx.stroke();
const w = ctx.measureText(text).width;
var offset = 8;
if (from.x < to.x) { ctx.fillText(text, to.x + offset + w / 2, to.y) }
else { ctx.fillText(text, to.x - offset - w / 2, to.y) }
}
function drawAngle(pos, lineA, lineB, radius, lablePos, ctx, col = ctx.strokeStyle, lw = ctx.lineWidth) {
ctx.strokeStyle = col;
ctx.lineWidth = lw;
const from = lineA.direction;
const angle = lineA.angleTo(lineB);
ctx.beginPath();
ctx.arc(pos.x, pos.y, radius, from, from + angle, angle < 0);
ctx.stroke();
drawLable(
Math.rad2Deg(angle).toFixed(2) + Math.symbols.degrees,
Vec2.Vec(
pos.x + Math.cos(from + angle / 2) * radius,
pos.y + Math.sin(from + angle / 2) * radius
),
lablePos,
ctx,
col,
lw / 2,
);
}
canvas {
position : absolute; top : 0px; left : 0px;
background: #4D8;
}
<canvas id="canvas"></canvas>
Related
how to make custom class in fabric js using fabric.Textbox Class override?
I am using FabricJS version : 3.6.3 I want to make new FabricJS class called : Button So that I have extend one class called Textbox from fabric js, which will Draw a Rectangle behind Text and it looking like a button. But Problem is that, I can't set height to that Button because height is not allow in Texbox object. I want to set Height and Width to Button object. Width is working Properly due to Textbox. it will also warp Text if width keep smaller then text width, and can be editable by double clicking on it. But only problem is that can't set Height to an object it should be Text vertically center when Height is increase. In short I want to make this kind of functionality in fabric js using object customization. Expected Output : but Actual Output : Here Is my Code That Create button : // fabric js custom button class (function (fabric) { "use strict"; // var fabric = global.fabric || (global.fabric = {}); fabric.Button = fabric.util.createClass(fabric.Textbox, { type: "button", stateProperties: fabric.Object.prototype.stateProperties.concat( "buttonRx", "buttonRy", "buttonFill", "buttonPadding", "buttonStrokeColor", "buttonStrokeWidth" ), buttonRx: 0, buttonRy: 0, buttonFill: "#ffffff00", buttonPadding: 0, buttonHeight: 0, buttonWidth: 0, textAlign: "center", buttonStrokeColor: "#000000", buttonStrokeWidth: 0, _dimensionAffectingProps: fabric.Text.prototype._dimensionAffectingProps.concat( "width", "fontSize" ), cacheProperties: fabric.Object.prototype.cacheProperties.concat( "buttonRx", "buttonRy", "buttonFill", "buttonPadding", "buttonStrokeColor", "buttonStrokeWidth" ), initialize: function (text, options) { this.text = text; this.callSuper("initialize", text, options); /* this.on("scaling", function () { console.log('scaling', this.getScaledHeight()); this.set({ height: this.getScaledHeight(), scaleY: 1, }); }); */ this._initRxRy(); }, _initRxRy: function () { if (this.buttonRx && !this.buttonRy) { this.buttonRy = this.buttonRx; } else if (this.buttonRy && !this.buttonRx) { this.buttonRx = this.buttonRy; } }, /* _setCenter(){ }, */ _render: function (ctx) { // 1x1 case (used in spray brush) optimization was removed because // with caching and higher zoom level this makes more damage than help // this.width = this.width * this.scaleX; // this.height = this.height * this.scaleY; // (this.scaleX = 1), (this.scaleY = 1); var rx = this.buttonRx ? Math.min(this.buttonRx, this.width / 2) : 0, ry = this.buttonRy ? Math.min(this.buttonRy, this.height / 2) : 0, w = this.width + this.buttonPadding, h = this.height + this.buttonPadding, x = -this.width / 2 - this.buttonPadding / 2, y = -this.height / 2 - this.buttonPadding / 2, isRounded = rx !== 0 || ry !== 0, /* "magic number" for bezier approximations of arcs (http://itc.ktu.lt/itc354/Riskus354.pdf) */ k = 1 - 0.5522847498; ctx.beginPath(); ctx.moveTo(x + rx, y); ctx.lineTo(x + w - rx, y); isRounded && ctx.bezierCurveTo(x + w - k * rx, y, x + w, y + k * ry, x + w, y + ry); ctx.lineTo(x + w, y + h - ry); isRounded && ctx.bezierCurveTo( x + w, y + h - k * ry, x + w - k * rx, y + h, x + w - rx, y + h ); ctx.lineTo(x + rx, y + h); isRounded && ctx.bezierCurveTo(x + k * rx, y + h, x, y + h - k * ry, x, y + h - ry); ctx.lineTo(x, y + ry); isRounded && ctx.bezierCurveTo(x, y + k * ry, x + k * rx, y, x + rx, y); ctx.closePath(); ctx.save(); if (this.buttonFill) { ctx.fillStyle = this.buttonFill; if (this.fillRule === "evenodd") { ctx.fill("evenodd"); } else { ctx.fill(); } } if (this.buttonStrokeWidth > 0) { if (this.strokeUniform) { ctx.scale(1 / this.scaleX, 1 / this.scaleY); } if (this.shadow && !this.shadow.affectStroke) { this._removeShadow(ctx); } if (this.buttonStrokeColor) { ctx.lineWidth = this.buttonStrokeWidth; ctx.strokeStyle = this.buttonStrokeColor; ctx.stroke(); } else { ctx.lineWidth = this.buttonStrokeWidth; ctx.stroke(); } } ctx.restore(); this.clearContextTop(); this._clearCache(); this.height = this.calcTextHeight(); this.saveState({ propertySet: "_dimensionAffectingProps" }); // this._renderPaintInOrder(ctx); this._setTextStyles(ctx); this._renderTextLinesBackground(ctx); this._renderTextDecoration(ctx, "underline"); this._renderText(ctx); this._renderTextDecoration(ctx, "overline"); this._renderTextDecoration(ctx, "linethrough"); this.initDimensions(); // this.callSuper('render', ctx); }, toObject: function (propertiesToInclude) { return this.callSuper( "toObject", [ "buttonRx", "buttonRy", "buttonFill", "buttonPadding", "buttonStrokeColor", "buttonStrokeWidth", "objectCaching", ].concat(propertiesToInclude) ); }, }); fabric.Button.fromObject = function (object, callback) { return fabric.Object._fromObject("Button", object, callback, "text"); }; })(fabric); // fabric js class finish here var canvas = []; var cotainer = document.getElementById("canvas-container"); for (let i = 0; i < 1; i++) { var width = 500, height = 500; var canvasEl = document.createElement("canvas"); canvasEl.id = "canvas-" + i; cotainer.append(canvasEl); var fabCanvas = new fabric.Canvas(canvasEl, {}); fabCanvas.setHeight(height); fabCanvas.setWidth(width); canvas.push(fabCanvas); } canvas.forEach((c) => { var button = new fabric.Button("Click Me", { text: "Click Me", buttonStrokeColor: "#f00", buttonStrokeWidth: 2, width: 110, fill: "#f00", fontSize: 50, width: 400, buttonFill: "#42A5F5", buttonRx: 15, buttonRy: 15, objectCaching: false, fontFamily: "verdana", }); c.add(button); c.renderAll(); }); canvas{ border: 1px solid black } <script src="https://cdnjs.cloudflare.com/ajax/libs/fabric.js/3.6.2/fabric.js"></script> <div id="canvas-container"> </div>
The solution will be to set the scaleX and scaleY of the button text to 1 when you scale the Button Object and also set the font size of the text equal to its scale.
var tbox = new fabric.Button(v.textDisp, { left: v.posX, top: v.posY, boxHeight: v.length // new create }); fabric.Button = fabric.util.createClass(fabric.Textbox, { type: "button", stateProperties: fabric.Object.prototype.stateProperties.concat( "buttonRx", "buttonRy", "buttonFill", "buttonPadding", "buttonStrokeColor", "buttonStrokeWidth", "boxHeight" ), buttonRx: 0, buttonRy: 0, buttonPadding: 0, buttonHeight: 0, buttonWidth: 0, buttonStrokeColor: "#000000", buttonStrokeWidth: 0, _dimensionAffectingProps: fabric.Text.prototype._dimensionAffectingProps.concat( "width", "fontSize" ), cacheProperties: fabric.Object.prototype.cacheProperties.concat( "buttonRx", "buttonRy", "buttonFill", "buttonPadding", "buttonStrokeColor", "buttonStrokeWidth", "boxHeight" ), initialize: function (text, options) { this.text = text; this.callSuper("initialize", text, options); /* this.on("scaling", function () { console.log('scaling', this.getScaledHeight()); this.set({ height: this.getScaledHeight(), scaleY: 1, }); }); */ this._initRxRy(); }, _initRxRy: function () { if (this.buttonRx && !this.buttonRy) { this.buttonRy = this.buttonRx; } else if (this.buttonRy && !this.buttonRx) { this.buttonRx = this.buttonRy; } }, /* _setCenter(){ }, */ _render: function (ctx) { // 1x1 case (used in spray brush) optimization was removed because // with caching and higher zoom level this makes more damage than help // this.width = this.width * this.scaleX; // this.height = this.height * this.scaleY; // (this.scaleX = 1), (this.scaleY = 1); var rx = this.buttonRx ? Math.min(this.buttonRx, this.width / 2) : 0, ry = this.buttonRy ? Math.min(this.buttonRy, this.height / 2) : 0, w = this.width + this.buttonPadding, h = this.height + this.buttonPadding, x = -this.width / 2 - this.buttonPadding / 2, y = -this.height / 2 - this.buttonPadding / 2, hh = this.boxHeight * this.scaleY, isRounded = rx !== 0 || ry !== 0, k = 1 - 0.5522847498; ctx.beginPath(); ctx.moveTo(x + rx, y); ctx.lineTo(x + w - rx, y); isRounded && ctx.bezierCurveTo(x + w - k * rx, y, x + w, y + k * ry, x + w, y + ry); ctx.lineTo(x + w, y + hh - ry); isRounded && ctx.bezierCurveTo( x + w, y + hh - k * ry, x + w - k * rx, y + hh, x + w - rx, y + hh ); ctx.lineTo(x + rx, y + hh); isRounded && ctx.bezierCurveTo(x + k * rx, y + hh, x, y + hh - k * ry, x, y + hh - ry); ctx.lineTo(x, y + ry); isRounded && ctx.bezierCurveTo(x, y + k * ry, x + k * rx, y, x + rx, y); ctx.closePath(); ctx.save(); if (this.buttonFill) { ctx.fillStyle = this.buttonFill; if (this.fillRule === "evenodd") { ctx.fill("evenodd"); } else { ctx.fill(); } } if (this.buttonStrokeWidth > 0) { if (this.strokeUniform) { ctx.scale(1 / this.scaleX, 1 / this.scaleY); } if (this.shadow && !this.shadow.affectStroke) { this._removeShadow(ctx); } if (this.buttonStrokeColor) { ctx.lineWidth = this.buttonStrokeWidth; ctx.strokeStyle = this.buttonStrokeColor; ctx.stroke(); } else { ctx.lineWidth = this.buttonStrokeWidth; ctx.stroke(); } } ctx.restore(); this.clearContextTop(); this._clearCache(); this.height = this.calcTextHeight(); this.saveState({ propertySet: "_dimensionAffectingProps" }); // this._renderPaintInOrder(ctx); this._setTextStyles(ctx); this._renderTextLinesBackground(ctx); this._renderTextDecoration(ctx, "underline"); this._renderText(ctx); this._renderTextDecoration(ctx, "overline"); this._renderTextDecoration(ctx, "linethrough"); this.initDimensions(); // this.callSuper('render', ctx); }, toObject: function (propertiesToInclude) { return this.callSuper( "toObject", [ "buttonRx", "buttonRy", "buttonFill", "buttonPadding", "buttonStrokeColor", "buttonStrokeWidth", "objectCaching", "boxHeight" ].concat(propertiesToInclude) ); }, }); After adding boxHeight, declare a variable as 'hh' instead of'h' in "_render" When drawing, change to'hh' instead of'h'
How to get length of path?
I want to know the length of a Path. For example, if I have a straight line I can just compute the length with its start x,y and end x,y values. But it gets quickly very tricky if I use QuadCurves or CubicCurves. Is there any way to get the length or an approximation of the length of a Path? For example the following path: Path path = new Path(); MoveTo moveTo = new MoveTo(start.getX(), start.getY()); double controlPointX = 50; CubicCurveTo cubicCurveTo = new CubicCurveTo(start.getX() + controlPointX, start.getY(), start.getX() + controlPointX, end.getY(), end.getX(), end.getY()); path.getElements().addAll(moveTo, cubicCurveTo);
I needed this recently as well. I couldn't find any solutions online, but it occurred to me PathTransition must be calculating it. It does, see PathTransition.recomputeSegment, where totalLength is calculated. Unfortunately, it uses many internal APIs in Node and the PathElement to convert the Path to a java.awt.geom.Path2D. I extracted these methods out and replaced other usages of com.sun classes with java.awt ones, then pulled the parts relevant to calculating length out of PathTransition.recomputeSegments. The resulting code is below. It is in Kotlin not Java, but it should be easy to convert it back to Java. I have not yet tested it extensively but it seems to be working on the fairly complex paths I have tested it against. I've compared my results to the length calculated by PathTransition and they are very close, I believe the discrepancies are due to my code using Path2D.Double where as Path2D.Float is used by PathElement.impl_addTo. fun Transform.toAffineTransform(): AffineTransform { if(!isType2D) throw UnsupportedOperationException("Conversion of 3D transforms is unsupported") return AffineTransform(mxx, myx, mxy, myy, tx, ty) } val Path.totalLength: Double get() { var length = 0.0 val coords = DoubleArray(6) var pt = 0 // Previous segment type var px = 0.0 // Previous x-coordinate var py = 0.0 // Previous y-coordinate var mx = 0.0 // Last move to x-coordinate var my = 0.0 // Last move to y-coordinate val pit = toPath2D().getPathIterator(localToParentTransform.toAffineTransform(), 1.0) while(!pit.isDone) { val type = pit.currentSegment(coords) val x = coords[0] val y = coords[1] when(type) { PathIterator.SEG_MOVETO -> { mx = x my = y } PathIterator.SEG_LINETO -> { val dx = x - px val dy = y - py val l = sqrt(dx * dx + dy * dy) if(l >= 1 || pt == PathIterator.SEG_MOVETO) length += l } PathIterator.SEG_CLOSE -> { val dx = x - mx val dy = y - my val l = sqrt(dx * dx + dy * dy) if(l >= 1 || pt == PathIterator.SEG_MOVETO) length += l } } pt = type px = x py = y pit.next() } return length } fun Path.toPath2D(): Path2D { val path: Path2D = Path2D.Double(if(fillRule == FillRule.EVEN_ODD) Path2D.WIND_EVEN_ODD else Path2D.WIND_NON_ZERO) for(e in elements) { when(e) { is Arc2D -> append(e as ArcTo, path) // Why isn't this smart casted? is ClosePath -> path.closePath() is CubicCurveTo -> append(e, path) is HLineTo -> append(e, path) is LineTo -> append(e, path) is MoveTo -> append(e, path) is QuadCurveTo -> append(e, path) is VLineTo -> append(e, path) else -> throw UnsupportedOperationException("Path contains unknown PathElement type: " + e::class.qualifiedName) } } return path } private fun append(arcTo: ArcTo, path: Path2D) { val x0 = path.currentPoint.x val y0 = path.currentPoint.y val localX = arcTo.x val localY = arcTo.y val localSweepFlag = arcTo.isSweepFlag val localLargeArcFlag = arcTo.isLargeArcFlag // Determine target "to" position val xto = if(arcTo.isAbsolute) localX else localX + x0 val yto = if(arcTo.isAbsolute) localY else localY + y0 // Compute the half distance between the current and the final point val dx2 = (x0 - xto) / 2.0 val dy2 = (y0 - yto) / 2.0 // Convert angle from degrees to radians val xAxisRotationR = Math.toRadians(arcTo.xAxisRotation) val cosAngle = Math.cos(xAxisRotationR) val sinAngle = Math.sin(xAxisRotationR) // // Step 1 : Compute (x1, y1) // val x1 = cosAngle * dx2 + sinAngle * dy2 val y1 = -sinAngle * dx2 + cosAngle * dy2 // Ensure radii are large enough var rx = abs(arcTo.radiusX) var ry = abs(arcTo.radiusY) var Prx = rx * rx var Pry = ry * ry val Px1 = x1 * x1 val Py1 = y1 * y1 // check that radii are large enough val radiiCheck = Px1 / Prx + Py1 / Pry if (radiiCheck > 1.0) { rx *= sqrt(radiiCheck) ry *= sqrt(radiiCheck) if(rx == rx && ry == ry) {/* not NANs */ } else { path.lineTo(xto, yto) return } Prx = rx * rx Pry = ry * ry } // // Step 2 : Compute (cx1, cy1) // var sign = if (localLargeArcFlag == localSweepFlag) -1.0 else 1.0 var sq = (Prx * Pry - Prx * Py1 - Pry * Px1) / (Prx * Py1 + Pry * Px1) sq = if (sq < 0.0) 0.0 else sq val coef = sign * Math.sqrt(sq) val cx1 = coef * (rx * y1 / ry) val cy1 = coef * -(ry * x1 / rx) // // Step 3 : Compute (cx, cy) from (cx1, cy1) // val sx2 = (x0 + xto) / 2.0 val sy2 = (y0 + yto) / 2.0 val cx = sx2 + (cosAngle * cx1 - sinAngle * cy1) val cy = sy2 + (sinAngle * cx1 + cosAngle * cy1) // // Step 4 : Compute the angleStart (angle1) and the angleExtent (dangle) // val ux = (x1 - cx1) / rx val uy = (y1 - cy1) / ry val vx = (-x1 - cx1) / rx val vy = (-y1 - cy1) / ry // Compute the angle start var n = sqrt(ux * ux + uy * uy) var p = ux // (1 * ux) + (0 * uy) sign = if (uy < 0.0) -1.0 else 1.0 var angleStart = (sign * Math.acos(p / n)).toDegrees() // Compute the angle extent n = Math.sqrt((ux * ux + uy * uy) * (vx * vx + vy * vy)) p = ux * vx + uy * vy sign = if (ux * vy - uy * vx < 0.0) -1.0 else 1.0 var angleExtent = Math.toDegrees(sign * Math.acos(p / n)) if(!localSweepFlag && angleExtent > 0) angleExtent -= 360.0 else if(localSweepFlag && angleExtent < 0) angleExtent += 360.0 angleExtent %= 360 angleStart %= 360 // // We can now build the resulting Arc2D // val arcX = cx - rx val arcY = cy - ry val arcW = rx * 2.0 val arcH = ry * 2.0 val arcStart = -angleStart val arcExtent = -angleExtent val arc = Arc2D.Double(OPEN).apply { setArc(arcX, arcY, arcW, arcH, arcStart, arcExtent, OPEN) } val xform: AffineTransform? = when(xAxisRotationR) { 0.0 -> null else -> AffineTransform().apply { setToRotation(xAxisRotationR, cx, cy) } } val pi = arc.getPathIterator(xform) // RT-8926, append(true) converts the initial moveTo into a // lineTo which can generate huge miter joins if the segment // is small enough. So, we manually skip it here instead. pi.next() path.append(pi, true) } private fun append(cubicCurveTo: CubicCurveTo, path: Path2D) { if(cubicCurveTo.isAbsolute) { path.curveTo(cubicCurveTo.controlX1, cubicCurveTo.controlY1, cubicCurveTo.controlX2, cubicCurveTo.controlY2, cubicCurveTo.x, cubicCurveTo.y) } else { val dx = path.currentPoint.x val dy = path.currentPoint.y path.curveTo(cubicCurveTo.controlX1 + dx, cubicCurveTo.controlY1 + dy, cubicCurveTo.controlX2 + dx, cubicCurveTo.controlY2 + dy, cubicCurveTo.x + dx, cubicCurveTo.y + dy) } } private fun append(hLineTo: HLineTo, path: Path2D) { if(hLineTo.isAbsolute) path.lineTo(hLineTo.x, path.currentPoint.y) else path.lineTo(path.currentPoint.x + hLineTo.x, path.currentPoint.y) } private fun append(lineTo: LineTo, path: Path2D) { if(lineTo.isAbsolute) path.lineTo(lineTo.x, lineTo.y) else path.lineTo(path.currentPoint.x + lineTo.x, path.currentPoint.y + lineTo.y) } private fun append(moveTo: MoveTo, path: Path2D) { if(moveTo.isAbsolute) path.moveTo(moveTo.x, moveTo.y) else path.moveTo((path.currentPoint.x + moveTo.x), path.currentPoint.y + moveTo.y) } private fun append(quadCurveTo: QuadCurveTo, path: Path2D) { if(quadCurveTo.isAbsolute) { path.quadTo(quadCurveTo.controlX, quadCurveTo.controlY, quadCurveTo.x, quadCurveTo.y) } else { val dx = path.currentPoint.x val dy = path.currentPoint.y path.quadTo(quadCurveTo.controlX + dx, quadCurveTo.controlY + dy, quadCurveTo.x + dx, quadCurveTo.y + dy) } } private fun append(vLineTo: VLineTo, path: Path2D) { if(vLineTo.isAbsolute) path.lineTo(path.currentPoint.x, vLineTo.y) else path.lineTo(path.currentPoint.x, path.currentPoint.y + vLineTo.y) }
Collision detection between two objects
The collision is not working According to that post Collision detection between 2 "linearly" moving objects in WGS84, I have the following data EDIT: I have updated the data for a collision that should occur in 10 seconds. m_sPosAV = {North=48.276111971715515 East=17.921031349301817 Altitude=6000.0000000000000 } Poi_Position = {North=48.806113707277042 East=17.977161602106488 Altitude=5656.0000000000000 } velocity.x = -189.80000000000001 // m/s velocity.y = -39.800000000000004 // m/s velocity.z = 9 // m/s m_sVelAV = {x=1.0000000000000000 y=1.0000000000000000 z=0.00000000000000000 } // m/s void WGS84toXYZ(double &x, double &y, double &z, double lon, double lat, double alt) { const double _earth_a = 6378141.4; // [m] equator radius const double _earth_b = 6356755.0; // [m] polar radius double a, b, h, l, c, s; a = lon; b = lat; h = alt; c = cos(b); s = sin(b); h = h + sqrt((_earth_a*_earth_a*c*c) + (_earth_b*_earth_b*s*s)); z = h*s; l = h*c; x = l*cos(a); y = l*sin(a); } bool CPoiFilterCollision::collisionDetection(const CPoiItem& poi) { const double _min_t = 10; // min_time const double _max_d = 500; // max_distance const double _max_t = 0.001; // max_time double dt; double d0, d1; double xAv, yAv, zAv; double xPoi, yPoi, zPoi; double x, y, z; double Ux, Uy, Uz; // [m] double Vx, Vy, Vz; // [m] double Wx, Wy, Wz; // [m] double da = 1.567e-7; // [rad] angular step ~ 1.0 m in lon direction double dl = 1.0; const double deg = pi / 180.0; // [m] altitide step 1.0 m WGS84toXYZ(xAv, yAv, zAv, m_sPosAV.GetLongitude(), m_sPosAV.GetLatitude(), m_sPosAV.GetAltitude()); // actual position WGS84toXYZ(xPoi, yPoi, zPoi, poi.Position().GetLongitude(), poi.Position().GetLatitude(), poi.Position().GetAltitude()); // actual position WGS84toXYZ(Ux, Uy, Uz, m_sPosAV.GetLongitude() + da, m_sPosAV.GetLatitude(), m_sPosAV.GetAltitude()); // lon direction Nort WGS84toXYZ(Vx, Vy, Vz, m_sPosAV.GetLongitude(), m_sPosAV.GetLatitude() + da, m_sPosAV.GetAltitude()); // lat direction East WGS84toXYZ(Wx, Wy, Wz, m_sPosAV.GetLongitude(), m_sPosAV.GetLatitude(), m_sPosAV.GetAltitude() + dl); // alt direction High/Up Ux -= xAv; Uy -= yAv; Uz -= zAv; Vx -= xAv; Vy -= yAv; Vz -= zAv; Wx -= xAv; Wy -= yAv; Wz -= zAv; normalize(Ux, Uy, Uz); normalize(Vx, Vy, Vz); normalize(Wx, Wy, Wz); double vx = m_sVelAV.x*Ux + m_sVelAV.y*Vx + m_sVelAV.z*Wx; double vy = m_sVelAV.x*Uy + m_sVelAV.y*Vy + m_sVelAV.z*Wy; double vz = m_sVelAV.x*Uz + m_sVelAV.y*Vz + m_sVelAV.z*Wz; const QList<QVariant> velocity = poi.Property(QLatin1String("VELOCITY")).toList(); if (velocity.size() == 3) { dt = _max_t; x = xAv - xPoi; y = yAv - yPoi; z = zAv - zPoi; d0 = sqrt((x*x) + (y*y) + (z*z)); x = xAv - xPoi + (vx - velocity.at(0).toDouble())*dt; y = yAv - yPoi + (vy - velocity.at(1).toDouble())*dt; z = zAv - zPoi + (vz - velocity.at(2).toDouble())*dt; d1 = sqrt((x*x) + (y*y) + (z*z)); if (d0 <= _max_d) { return true; } if (d0 <= d1) { return false; } double t = (_max_d - d0)*dt / (d1 - d0); if (t < _min_t) { qDebug() << "Collision at time " << t; return true; } } return false; }
How can I get rotated rectangle corner points from side points
Say I have a set of four or more points that are on the perimeter of a rectangle, and that the rectangle is rotated by some unknown amount. I know that at least one point is on each side of the rectangle. One arbitrary side point is designated (0, 0), and the other points are the distance from this starting point. How can I get the non-rotated corner points of this rectangle?
assuming you're not trying to find a unique solution: rotate your points around 0,0 until the top-most, bottom-most, left-most, and right-most points are all different points draw horizontal lines through the top-most and bottom-most, and vertical lines through the left-most and right-most you're done var points = []; var bs = document.body.style; var ds = document.documentElement.style; bs.height = bs.width = ds.height = ds.width = "100%"; bs.border = bs.margin = bs.padding = 0; var c = document.createElement("canvas"); c.style.display = "block"; c.addEventListener("mousedown", addPoint, false); document.body.appendChild(c); var ctx = c.getContext("2d"); var interval; function addPoint(e) { if (points.length >= 4) points = []; points.push({ x: e.x - c.offsetLeft, y: e.y - c.offsetTop }); while (points.length > 4) points.shift(); redraw(); } function rotateAround(a, b, r) { d = {x:a.x - b.x, y:a.y - b.y}; return { x: b.x + Math.cos(r) * d.x - Math.sin(r) * d.y, y: b.y + Math.cos(r) * d.y + Math.sin(r) * d.x } } function drawPoint(p) { ctx.strokeStyle = "rgb(0,0,0)"; ctx.beginPath(); ctx.arc(p.x, p.y, 10, 0, 2 * Math.PI, true); ctx.closePath(); ctx.stroke(); } var last_few = []; function redraw() { if (interval) clearInterval(interval); last_few = []; c.width = window.innerWidth; c.height = window.innerHeight; ctx.clearRect(0, 0, c.width, c.height); ctx.fillStyle = "rgb(200, 200, 200)"; ctx.font = "40px serif"; if (points.length < 4) { ctx.fillText("click " + (4 - points.length) + " times", 20, 40); points.forEach(drawPoint); } else { var average = {x:0, y:0}; points.forEach(function (p) { average.x += p.x / 4; average.y += p.y / 4; }); var step = 0; interval = setInterval(function () { ctx.clearRect(0, 0, c.width, c.height); ctx.fillText("click anywhere to start over", 20, 40); last_few.forEach(function(r) { ctx.strokeStyle = "rgb(200,255,200)"; ctx.save(); ctx.translate(average.x, average.y); ctx.rotate((step -r.step) * Math.PI / 180); ctx.strokeRect(r.lm - average.x, r.tm - average.y, (r.rm - r.lm), (r.bm - r.tm)); ctx.restore(); }); var tm = Infinity; var bm = -Infinity; var lm = Infinity; var rm = -Infinity; points.forEach(function (p) { p = rotateAround(p, average, step * Math.PI / 180); drawPoint(p); tm = Math.min(p.y, tm); bm = Math.max(p.y, bm); lm = Math.min(p.x, lm); rm = Math.max(p.x, rm); }); if (points.every(function (p) { p = rotateAround(p, average, step * Math.PI / 180); return (p.x == lm) || (p.x == rm) || (p.y == tm) || (p.y == bm); })) { ctx.strokeStyle = "rgb(0,255,0)"; ctx.strokeRect(lm, tm, (rm - lm), (bm - tm)); last_few.push({tm:tm, bm:bm, lm:lm, rm:rm, step:step}); while(last_few.length > 30) last_few.shift(); } else { ctx.strokeStyle = "rgb(255,0,0)"; ctx.strokeRect(lm, tm, (rm - lm), (bm - tm)); } step++; }, 30); } } window.onresize = redraw; redraw();
Javafx 8 3D Complex Shape
I want to create complex shape like human bodys for my game project. Can I create complex shape in javafx 8 with trianglemesh. and also i read, it is possible to import 3ds model to javafx. can any one tell me how to import them and do things like rotate move by javafx code thanks for your help
Also, If you go to Oracle, a 3DViewer is in the jfx samples which will export the 3d file to FXML, thus creating appropriate Groups ... Also being able to re-import the FXML into the Viewer... Samples ... Although as the above comment stated, InteractiveMesh has some nice tools. Thats the easy way, if you already have a 3d file... otherwise the point,texCoord, and face arrays can be quite daunting. It took me quite a while to create a torus .. points were fine, texCoords were fine, the faces are a pain... this was just for my customizable torus /* Let the radius from the center of the hole to the center of the torus tube be "c", and the radius of the tube be "a". Then the equation in Cartesian coordinates for a torus azimuthally symmetric about the z-axis is (c-sqrt(x^2+y^2))^2+z^2=a^2 and the parametric equations are x = (c + a * cos(v)) * cos(u) y = (c + a * cos(v)) * sin(u) z = a * sin(v) (for u,v in [0,2pi). Three types of torus, known as the standard tori, are possible, depending on the relative sizes of a and c. c>a corresponds to the ring torus (shown above), c=a corresponds to a horn torus which is tangent to itself at the point (0, 0, 0), and c<a corresponds to a self-intersecting spindle torus (Pinkall 1986). */ public static TriangleMesh createToroidMesh(float radius, float tRadius, int tubeDivisions, int radiusDivisions) { int POINT_SIZE = 3, TEXCOORD_SIZE = 2, FACE_SIZE = 6; int numVerts = tubeDivisions * radiusDivisions; int faceCount = numVerts * 2; float[] points = new float[numVerts * POINT_SIZE], texCoords = new float[numVerts * TEXCOORD_SIZE]; int[] faces = new int[faceCount * FACE_SIZE], smoothingGroups; int pointIndex = 0, texIndex = 0, faceIndex = 0, smoothIndex = 0; float tubeFraction = 1.0f / tubeDivisions; float radiusFraction = 1.0f / radiusDivisions; float x, y, z; int p0 = 0, p1 = 0, p2 = 0, p3 = 0, t0 = 0, t1 = 0, t2 = 0, t3 = 0; // create points for (int tubeIndex = 0; tubeIndex < tubeDivisions; tubeIndex++) { float radian = tubeFraction * tubeIndex * 2.0f * 3.141592653589793f; for (int radiusIndex = 0; radiusIndex < radiusDivisions; radiusIndex++) { float localRadian = radiusFraction * radiusIndex * 2.0f * 3.141592653589793f; points[pointIndex] = x = (radius + tRadius * ((float) Math.cos(radian))) * ((float) Math.cos(localRadian)); points[pointIndex + 1] = y = (radius + tRadius * ((float) Math.cos(radian))) * ((float) Math.sin(localRadian)); points[pointIndex + 2] = z = (tRadius * (float) Math.sin(radian)); pointIndex += 3; float r = radiusIndex < tubeDivisions ? tubeFraction * radiusIndex * 2.0F * 3.141592653589793f : 0.0f; texCoords[texIndex] = (0.5F + (float) (Math.sin(r) * 0.5D));; texCoords[texIndex + 1] = ((float) (Math.cos(r) * 0.5D) + 0.5F); texIndex += 2; } } //create faces for (int point = 0; point < (tubeDivisions) ; point++) { for (int crossSection = 0; crossSection < (radiusDivisions) ; crossSection++) { p0 = point * radiusDivisions + crossSection; p1 = p0 >= 0 ? p0 + 1 : p0 - (radiusDivisions); p1 = p1 % (radiusDivisions) != 0 ? p0 + 1 : p0 - (radiusDivisions - 1); p2 = (p0 + radiusDivisions) < ((tubeDivisions * radiusDivisions)) ? p0 + radiusDivisions : p0 - (tubeDivisions * radiusDivisions) + radiusDivisions ; p3 = p2 < ((tubeDivisions * radiusDivisions) - 1) ? p2 + 1 : p2 - (tubeDivisions * radiusDivisions) + 1; p3 = p3 % (radiusDivisions) != 0 ? p2 + 1 : p2 - (radiusDivisions - 1); t0 = point * (radiusDivisions) + crossSection; t1 = t0 >= 0 ? t0 + 1 : t0 - (radiusDivisions); t1 = t1 % (radiusDivisions) != 0 ? t0 + 1 : t0 - (radiusDivisions - 1); t2 = (t0 + radiusDivisions) < ((tubeDivisions * radiusDivisions)) ? t0 + radiusDivisions : t0 - (tubeDivisions * radiusDivisions) + radiusDivisions ; t3 = t2 < ((tubeDivisions * radiusDivisions) - 1) ? t2 + 1 : t2 - (tubeDivisions * radiusDivisions) + 1; t3 = t3 % (radiusDivisions) != 0 ? t2 + 1 : t2 - (radiusDivisions - 1); try { faces[faceIndex] = (p2); faces[faceIndex + 1] = (t3); faces[faceIndex + 2] = (p0); faces[faceIndex + 3] = (t2); faces[faceIndex + 4] = (p1); faces[faceIndex + 5] = (t0); faceIndex += FACE_SIZE; faces[faceIndex] = (p2); faces[faceIndex + 1] = (t3); faces[faceIndex + 2] = (p1); faces[faceIndex + 3] = (t0); faces[faceIndex + 4] = (p3); faces[faceIndex + 5] = (t1); faceIndex += FACE_SIZE; } catch (Exception e) { e.printStackTrace(); } } } TriangleMesh localTriangleMesh = new TriangleMesh(); localTriangleMesh.getPoints().setAll(points); localTriangleMesh.getTexCoords().setAll(texCoords); localTriangleMesh.getFaces().setAll(faces); return localTriangleMesh; }
Consider it to be a personal opinion. Though, I don't question the power of Javafx and the possibility to build complex 3d models using Javafx, I would not suggest building such complex structures using it, when you can easily achieve them using AutoDesk and other softwares, with less effort ! You have options to import them in your Javafx applications using early access available here http://www.interactivemesh.org/models/jfx3dimporter.html This seems to be very promising ! have a look at this as well http://www.interactivemesh.org/models/jfx3dbrowser.html
I made a 2x6 in sketchup and imported it along with the chick/dude/androgyne. He/she/it is only 2D. You need to get the jar file (jimColModelImporterJFX.jar) here. import com.interactivemesh.jfx.importer.Viewpoint; import com.interactivemesh.jfx.importer.col.ColAsset; import com.interactivemesh.jfx.importer.col.ColModelImporter; //import com.interactivemesh.jfx.importer.stl.StlMeshImporter; import javafx.application.Application; import javafx.scene.*; import javafx.scene.paint.Color; import javafx.stage.Stage; import javafx.scene.paint.PhongMaterial; import javafx.scene.transform.Rotate; import javafx.scene.transform.Translate; import java.io.File; import java.util.Map; public class Importer3D extends Application { //you'll have to make your own file in sketchup and save it as *.dae //http://help.sketchup.com/en/article/114347 private static final String DAE_FILENAME = ("2x6stud-1.dae"); private static final int VIEWPORT_SIZE = 800; private Group root; public Rotate rx = new Rotate(); { rx.setAxis(Rotate.X_AXIS); } public Rotate ry = new Rotate(); { ry.setAxis(Rotate.Y_AXIS); } public Rotate rz = new Rotate(); { rz.setAxis(Rotate.Z_AXIS); } Translate t = new Translate(); static Node[] loadNodes() { File file = new File(DAE_FILENAME); ColModelImporter importer = new ColModelImporter(); importer.read(file); Node[] nodes = importer.getImport(); ColAsset colAsset = importer.getAsset(); System.out.println("asset title " + colAsset.getTitle()); System.out.println("asset unit name " + colAsset.getUnitName()); System.out.println("asset unit meter " + colAsset.getUnitMeter()); System.out.println("asset up axis " + colAsset.getUpAxis()); Map<String, PhongMaterial> materials = importer.getNamedMaterials(); for (Map.Entry<String, PhongMaterial> e : materials.entrySet()) { System.out.println("phong material " + e.getKey() + " -> " + e.getValue()); } Map<String, Node> namedNodes = importer.getNamedNodes(); for (Map.Entry<String, Node> e : namedNodes.entrySet()) { System.out.println("nodes " + e.getKey() + " -> " + e.getValue()); } Viewpoint[] viewpoints = importer.getViewpoints(); if (viewpoints != null) for (Viewpoint v : viewpoints) { System.out.println("viewpoint " + v); } return nodes; } private Group buildScene() { Node[] nodes = loadNodes(); root = new Group(nodes); return root; } private PerspectiveCamera addCamera(Scene scene) { PerspectiveCamera camera = new PerspectiveCamera(); camera.getTransforms().addAll(t, rz, ry, rx); camera.setVerticalFieldOfView(true); camera.setFieldOfView(10d); System.out.println("Near Clip: " + camera.getNearClip()); System.out.println("Far Clip: " + camera.getFarClip()); System.out.println("FOV: " + camera.getFieldOfView()); scene.setCamera(camera); return camera; } #Override public void start(Stage stage) { Group group = buildScene(); group.setScaleX(10); group.setScaleY(10); group.setScaleZ(10); group.setTranslateX(VIEWPORT_SIZE / 2); group.setTranslateY(VIEWPORT_SIZE / 2); Scene scene = new Scene(group, VIEWPORT_SIZE, VIEWPORT_SIZE, true); scene.setFill(Color.rgb(10, 10, 40)); addCamera(scene); stage.setTitle("Collada importer"); stage.setScene(scene); stage.show(); scene.setOnKeyPressed((evt) -> { switch (evt.getCode()) { case UP: rx.setAngle(rx.getAngle() + 5); break; case DOWN: rx.setAngle(rx.getAngle() - 5); break; case RIGHT: t.setX(t.getX() + 10); //camera.setTranslateX(camera.getTranslateX()+10); break; case LEFT: t.setX(t.getX() - 10); //camera.setTranslateX(camera.getTranslateX()-10); break; case Z: double zoom = evt.isShortcutDown() ? -10 : +10; t.setZ(t.getZ() + zoom); //camera.setTranslateZ(camera.getTranslateZ()+zoom); break; } }); } public static void main(String[] args) { System.setProperty("prism.dirtyopts", "false"); launch(args); } }