Прецизионные задачи в алгоритме лучевого вещания

Я программирую движок радиовещания.

Начальная позиция луча определяется положением игрока, который стоит внутри 2D-сетки.

При наведении луча в направлении, я должен определить сетки, которые пересекает луч.

(Подробное описание концепции можно найти здесь: http://www.permadi.com/tutorial/raycast/rayc7.html).

Есть небольшая неточность, которая вызывает некоторые проблемы. Я считаю, что проблема вызвана неправильным расчетом шагов сетки.

Однако мне не хватает математического понимания, чтобы решить эту проблему.

Описание проблемы:

Когда луч направляется влево, размер шага пересечения сетки немного отличается от размера шага, когда он направляется вправо.

(Почему я даже беспокоюсь об этом?: Это вызывает проблему, что некоторые горизонтальные пересечения сетки находятся дальше от игрока, чем вертикальные пересечения сетки, когда вертикальные и горизонтальные пересечения находятся близко друг к другу, особенно в углах. Так как я использую различные текстуры для вертикальных пересечений вид горизонтальных стен разрушен, потому что вертикальная текстура используется на небольших частях стены, даже если это горизонтальная стена.)

Эта проблема вызвана недостатком в моем алгоритме? Вот как я вычисляю первое горизонтальное пересечение сетки и размер шага сетки:

Найти первое пересечение сетки:

if (current_angle > 180) {
    first_grid_horizontal_y = ((int)p.pos_y / Field::width) * Field::width + Field::width;
} else {
    first_grid_horizontal_y = ((int)p.pos_y / Field::width) * Field::width - 1;
}
first_grid_horizontal_x = p.pos_x + (p.pos_y - first_grid_horizontal_y) / tan( 180 - current_angle);

Рассчитать размер шага:

if (current_angle > 180) {
    grid_stepsize_horizontal_y = Field::width;
    grid_stepsize_horizontal_x = Field::width / tan(current_angle - 180);
} else {
    grid_stepsize_horizontal_y = -Field::width;
    grid_stepsize_horizontal_x = Field::width / tan(180 - current_angle);
}

Как видите, я всегда использую "180 - текущий угол" для определения направления значения x. Это вызывает неточность? Нужно ли больше различать углы?

2 ответа

Тригонометрические функции работают с радианами, а не градусами. Так

tan(180 - current_angle)

должен выглядеть так

tan(Math.Pi - current_angle)

Обратите внимание, что он равен

- tan(current_angle)

Если ваш current_angle в градусах, то:

current_angle_radians = current_angle_degrees * Math.Pi / 180

Я думаю, что ваша неточность связана с вычитанием 1 при движении вверх.

Идея, стоящая за этим, вероятно, заключается в том, что вы хотите получить индекс последнего пикселя блока, но это необязательно: здесь вы выполняете математику с плавающей запятой и y Значение следующего горизонтального пересечения должно представлять линию между ячейками сетки. Если вы идете вниз, он представляет собой самую верхнюю координату ячейки; если вы идете вверх, это представляет самую нижнюю координату.

(В сторону: (int) y / w будет округлять до нуля и, таким образом, будет работать только для неотрицательных чисел. Вы могли бы рассмотреть возможность использования floor(x / w).)

Другие вопросы по тегам