У меня есть уравнение для параболической кривой, пересекающей заданную точку, в моем случае, когда пользователь нажимал на график.
// this would typically be mouse coords on the graph
var _target:Point = new Point(100, 50);
public static function plot(x:Number, target:Point):Number{
return (x * x) / target.x * (target.y / target.x);
}
Это дает такой график:
У меня также есть ряд отрезков, определяемых начальными и конечными координатами:
startX:Number, startY:Number, endX:Number, endY:Number
Мне нужно выяснить, пересекает ли эта кривая эти сегменты (A):
Если это поможет, startX
всегда < endX
У меня такое ощущение, что есть довольно простой способ сделать это, но я не знаю, что искать, и при этом я не очень хорошо разбираюсь в "правильной" математике, поэтому реальные примеры кода будуточень ценится.
ОБНОВЛЕНИЕ:
У меня работает пересечение, но мое решение дает мне координату для неправильной стороны оси Y.
Замена моих целевых координат на A и B соответственно дает следующее уравнение для графика:
(x * x) / A * (B/A)
// this simplifies down to:
(B * x * x) / (A * A)
// which i am the equating to the line's equation
(B * x * x) / (A * A) = m * x + b
// i run this through wolfram alpha (because i have no idea what i'm doing) and get:
(A * A * m - A * Math.sqrt(A * A * m * m + 4 * b * B)) / (2 * B)
Этот является правильным ответом, но я хочу второй возможныйизменение.Мне удалось исправить это, умножив m на -1 перед вычислением и выполнив то же самое со значением x, которое возвращает последний расчет, но это похоже на взлом.
РЕШЕНИЕ:
public static function intersectsSegment(targetX:Number, targetY:Number, startX:Number, startY:Number, endX:Number, endY:Number):Point {
// slope of the line
var m:Number = (endY - startY) / (endX - startX);
// where the line intersects the y-axis
var b:Number = startY - startX * m;
// solve the two variatons of the equation, we may need both
var ix1:Number = solve(targetX, targetY, m, b);
var ix2:Number = solveInverse(targetX, targetY, m, b);
var intersection1:Point;
var intersection2:Point;
// if the intersection is outside the line segment startX/endX it's discarded
if (ix1 > startX && ix1 < endX) intersection1 = new Point(ix1, plot(ix1, targetX, targetY));
if (ix2 > startX && ix2 < endX) intersection2 = new Point(ix2, plot(ix2, targetX, targetY));
// somewhat fiddly code to return the smallest set intersection
if (intersection1 && intersection2) {
// return the intersection with the smaller x value
return intersection1.x < intersection2.x ? intersection1 : intersection2;
} else if (intersection1) {
return intersection1;
}
// this effectively means that we return intersection2 or if that's unset, null
return intersection2;
}
private static function solve(A:Number, B:Number, m:Number, b:Number):Number {
return (m + Math.sqrt(4 * (B / (A * A)) * b + m * m)) / (2 * (B / (A * A)));
}
private static function solveInverse(A:Number, B:Number, m:Number, b:Number):Number {
return (m - Math.sqrt(4 * (B / (A * A)) * b + m * m)) / (2 * (B / (A * A)));
}
public static function plot(x:Number, targetX:Number, targetY:Number):Number{
return (targetY * x * x) / (targetX * targetX);
}