4.3. Алгоритм Робертса
Алгоритм Робертса представляет собой первое известное решение задачи об удалении невидимых линий. Это математически элегантный метод, работающий в объектном пространстве. Алгоритм прежде всего удаляет из каждого тела те ребра или грани, которые экранируются самим телом. Затем каждое из видимых ребер каждого тела сравнивается с каждым из оставшихся тел для определения того, какая его часть или части, если таковые есть, экранируются этими телами. Поэтому вычислительная трудоемкость алгоритма Робертса растет теоретически как квадрат числа объектов.
В алгоритме Робертса требуется, чтобы все изображаемые тела или объекты были выпуклыми. Невыпуклые тела должны быть разбиты на выпуклые части. В этом алгоритме выпуклое многогранное тело с плоскими гранями должно представляться набором пересекающихся плоскостей. Уравнение произвольной плоскости в трехмерном пространстве имеет вид
aх + by + cz + d = 0
Если любая точка S(xs, ys, zs) лежит на плоскости, то axs + bys + czs + d = 0. Если же S не лежит на плоскости, то знак этого скалярного произведения показывает, по какую сторону от плоскости расположена точка. В алгоритме Робертса предполагается, что точки, лежащие внутри тела, дают положительное скалярное произведение.
Пусть F1, F2, ..., Fn - грани многогранника. Рассмотрим одну из граней. Обозначим вершины, инцидентные грани, через V1, V2, ..., Vk. Найдем вектор нормали к грани, вычислив векторное произведение любых двух смежных ребер этой грани V1V2 = [x1,y1,z1] и V2V3 = [x2,y2,z2]:
ni = [V1V2,V2V3] = | i j k x1 y1 z1 x2 y2 z2 |
= (y1z2 - y2z1)·i + (z1x2 - z2x1)·j + (x1y2 - x2y1)·k = Ai·i + Bi·j + Ci·k |
Тогда опорная функция грани имеет вид:
Li(x,y,z) = Aiх + Biy + Ciz + D
Величина D вычисляется с помощью произвольной точки на плоскости. В частности, если компоненты этой точки на плоскости (х1,y1,z1), то:
D = -(ах1 + by1 + cz1)
Так как многогранник выпуклый, коэффициенты Ai, Bi, Ci легко выбрать так, чтобы ni(Ai, Bi, Ci) был вектором внешней нормали. Для этого найдем какую-либо внутреннюю точку, например, барицентр многогранника:
W = (V1 + V2 + ... + Vk)/k
Если скалярное произведение уравнения плоскости и этой точки меньше 0, то необходимо поменять знак уравнения этой плоскости, чтобы отразить правильное направление внешней нормали. Остается только вычислить скалярное произведение уравнения плоскости на точку, в которой находится наблюдатель. Если это скалярное произведение меньше 0, то плоскость невидима и необходимо удалить весь многоугольник, лежащий в этой плоскости. Запись этого алгоритма на псеводокоде приводится ниже.
Алгоритм Робертса
V1, V2, V3 - вершины многогранника
W - барицентр многогранника
P - точка наблюдения
выделим одну из граней многогранника |
||
Vec1.X = V1.X - V2.X; |
/ для этой грани найдем координаты двух векторов, которые лежат в плоскости грани |
|
A = Vec1.Y·Vec2.Z - Vec2.Y·Vec1.Z; |
/ вычислим коэффициенты уравнения плоскости |
|
m = -Sign(A·W.X + B·W.Y + C·W.Z + D); |
/ коэффициент, изменяющий знак плоскости |
|
A = A·m; |
/ корректируем направление плоскости |
|
if A·P.X + B·P.Y + C·P.Z + D > 0 then |
Если задано только одно тело, то применив этот алгоритм для всех граней тела, поиск невидимых граней завершается. Если в сцене присутствует несколько тел, то следует сформировать приоритетный список этих тел по максимальным значениям координаты z вершин тел. Наибольшим приоритетом будет обладать тело, у которого минимальное значение z. Это тело будет самым удаленным от точки наблюдения, расположенной в бесконечности на оси z.
Для каждого тела из приоритетного списка:
- Проверить экранирование всех лицевых ребер всеми другими телами сцены. Тело, ребра которого проверяются, называется пробным объектом, а тело, относительно которого в настоящий момент производится проверка, называется пробным телом. Естественно, что нужно проверять экранирование пробного объекта только теми пробными телами, у которых ниже приоритеты.
- Провести проверки экранирования для прямоугольных объемлющих оболочек пробного объекта и пробного тела.
- Провести предварительные проверки протыкания, чтобы увидеть, не протыкается ли пробное тело пробным объектом и существует ли возможность частичного экранирования первого последним.