ГЕОМЕТРИЧЕСКИЕ ПРЕОБРАЗОВАНИЯ
Конспект лекций
Составитель:
к.т.н., заведующий лабораторией компьютерной графики
Курского государственного технического университета
В.В.Керекеша
Курск 2000
ВВЕДЕНИЕ В ГЕОМЕТРИЧЕСКИЕ ПРЕОБРАЗОВАНИЯ
Виды преобразований
При построении изображений часто приходится иметь дело с ситуациями, когда общее изображение (рисунок) включает в себя целый ряд компонент (подрисунков), отличающихся друг от друга только местоположением, ориентацией, масштабом, т.е. отдельные подрисунки обладают значительным геометрическим сходством.
В этом случае целесообразно описать один подрисунок в качестве базового, а затем получать остальные требуемые подрисунки путем использования операций преобразования.
С помощью операций преобразования можно выполнять следующие действия:
1) перемещать рисунки из одного места экрана в другое;
2) создавать рисунок из более мелких элементов (составных частей);
3) добавлять к существующему рисунку новые элементы;
4) увеличивать размер рисунка для улучшения его наглядности или отображения более мелких деталей;
5) уменьшать размер рисунка для внесения, например, поясняющих надписей или отображения на экране новых рисунков;
6) создавать движущиеся изображения.
Все изменения рисунков можно выполнить с помощью трех базовых операций:
1) переноса (перемещения) изображения;
2) масштабирования (увеличения или уменьшения размеров) изображения;
3) поворота изображения (употребляют также термины вращение, изменение ориентации).
Эти операции называются аффинными преобразованиями. Различают двумерные и трехмерные аффинные преобразования.
Основные геометрические свойства двумерных аффинных преобразований:
1) прямые линии после преобразований остаются прямыми;
2) параллельные прямые – параллельными;
3) Отношения деления отрезков остаются неизменными.
Основные геометрические свойства трехмерных аффинных преобразований:
1) плоскости после преобразования остаются плоскостями;
2) параллельные плоскости – параллельными.
Двумерные аффинные преобразования
Перенос
1) Перенос точки
Пусть необходимо осуществить перенос точки P с координатами (x,y), dx - приращение координаты х, dy - приращение координаты y.
Тогда Р' с координатами (х',y') полученная после переноса точка.
Рис.1. Перенос точки
x' = x + dx
y' = y + dy
Аффинные преобразования удобно записывать в матричной форме:
P = [x, y];
P' = [x', y'];
T = [dx, dy].
Тогда
P' = P + T.
2) Перенос отрезка
Объект можно перенести, применяя предыдущие уравнения к каждой его точке. Однако, поскольку каждый отрезок, описывающий объект, состоит из бесконечного числа точек, такой процесс длился бы бесконечно долго. Все точки, принадлежащие отрезку, можно перенести путем перемещения одних точек отрезка и последующего вычерчивания нового отрезка между получившимися в результате точками. Это справедливо также для масштабирования (растяжения) и поворота.
Рис.2. Перенос отрезка
x1'= x1 + dx
y1'= y1 + dy
x2'= x2 + dx
y2'= y2 + dy
В векторной форме:
- исходный отpезок,
- полученный отpезок,
T = [dx, dy] - вектоp пеpеноса.
Тогда
Р'= P + T
Масштабирование
Масштабирование выполняется умножением координат на некоторые коэффициенты Sx, Sy.
1) Масштабирование точки
Пусть дана точка (x,y).
Тогда
х'= xSx,
y'= ySy,
,
P'= PS
2) Масштабирование отрезка
- исходный отpезок,
- полученный отpезок,
,
P'= PS.
Масштабирование бывает
а) однородное, когда Sx = Sy;
б) неоднородное, когда Sx Sy.
При неоднородном масштабировании фигура может изменятся (кpуг-эллипс ).
Масштабирование также делится по значению коэффициента масштабирования S (рис.3, 4).
Рис.3. Масштабирование с коэффициентом S
Рис.4. Масштабирование с коэффициентом S > 1
Поворот
Пусть необходимо осуществить повоpот точки P (x,y) на угол A:
Рис.5. Дмумерный поворот
x = r cos B,
y = r sin B. (1)
После поворота точка P' имеет координаты :
x'= r cos ( A + B ),
y'= r sin ( A + B ). (2)
Преобразуем выражение (2):
x'= r cos A cos B - r sin A sin B,
y'= r cos B sin A + r sin B cos A. (3)
Подставим (1) в (3):
x'= x cos A - y sin A,
y'= x sin A + y cos A. (4)
В матричной форме поворот записывается следующим образом:
P = [ x y ];
P'= [ x' y'];
.
Тогда
P'= P R.
Движение считается положительным ( А>0 ), если движемся против часовой стрелки; отрицательным - по часовой.
Пример вычисления матрицы композиции двумерных аффинных преобразований
Пусть задан отрезок, определенный координатами
.
1) Перенос dx=2, dy=3:
.
2) Масштабирование Sx=Sy=2:
.
3) Поворот на А=30 градусов:
.
Трехмерные аффинные преобразования
Перенос
Для того, чтобы перенести точку, надо сделать приращение к координатам dx, dy, dz.
Масштабирование
Выполняется аналогично двумерному масштабированию. Матрица преобразования:
.
Поворот
а) Поворот на угол А по часовой стрелке вокруг оси z:
. (1)
Не перепутаны ли знаки у sin'усов? Нет не перепутаны, поскольку это поворот по часовой, т.е. здесь левосторонняя система координат.
б) Поворот вокруг оси y по часовой стрелке на угол А:
. (2)
в) Поворот вокруг оси х по часовой стрелке на угол А:
. (3)
Результирующий поворот точки с координатами (x,y,z):
[x',y',z'] = [x,y,z] R(A) R(B) R(C).
Преобразование общего поворота точки с центром вращения, совпадающим с началом координат, можно получить как суперпозицию трех плоских поворотов. Это преобразование математически выражается перемножением трех матриц (1), (2), (3). Матричное умножение не является коммутативной операцией, поэтому необходимо задать некоторый порядок выполнения поворотов. Соглашение о порядке принимается совершено произвольно, но после того, как порядок выполнения будет зафиксирован, его необходимо строго придерживаться.
ГЕОМЕТРИЧЕСКИЕ ПРЕОБРАЗОВАНИЯ С ИСПОЛЬЗОВАНИЕМ ОДНОРОДНЫХ КООРДИНАТ
Однородные координаты
В однородных координатах точка Р(x,y) записывается как Р(Wx,Wy,W) для любого масштабного множителя W0. При этом если для точки задано ее представление в однородных координатах Р(X,Y,W), то можно найти ее двумерные декартовы координаты как x=X/W и y=Y/W.
Геометрический смысл однородных координат заключается в следующем (рис.6 из [1]). Произвольная точка на прямой, соединяющей начало координат, точку О(0,0,0), с точкой Р'(x,y,1), может быть задана тройкой чисел вида (Wx,Wy,W). Вектор с координатами Wx,Wy,W является направляющим вектором прямой, соединяющей точки О(0,0,0) и Р'(x,y,1). Эта прямая пересекает плоскость z=1 в точке (x,y,1), которая однозначно определяет точку (x,y) координатной плоскости x,y.
Рис.6.
Тем самым между произвольной точкой с координатами (x,y) и множеством троек чисел вида (Wx,Wy,W), W0 устанавливается взаимно однозначное соответствие, позволяющее считать числа Wx,Wy,W новыми координатами этой точки. Однородные координаты можно представить как вложение промасштабированной с коэффициентом W двумерной плоскости в плоскость z=W (здесь z=1) в трехмерном пространстве.
Применение однородных координат оказывается удобным уже при решении простейших задач.
Если устройство отображения работает только с целыми числами (или если необходимо работать только с целыми числами), то для произвольного значения W (например, W=1) точку с однородными координатами (0.5 0.1 2.5) представить нельзя. Однако при разумном выборе W можно добиться того, чтобы координаты этой точки были целыми числами. В частности, при W=10 для рассматриваемого примера имеем (5 1 25).
Другой случай. Чтобы результаты преобразования не приводили к арифметическому переполнению, для точки с координатами (80000 40000 1000) можно взять, например, W=0.001. В результате получим (80 40 1).
Однако, основное применение однородных координат – это геометрические преобразования, поскольку при помощи троек однородных координат и матриц третьего порядка можно описать любое аффинное преобразование плоскости. Аналогично, с помощью четверок однородных координат и матриц четвертого порядка можно описать любое преобразование в пространстве.
Как известно, преобразования переноса, масштабирования и поворота в матричной форме записываются в виде:
Р'= P + T
P'= PS
P'= PR.
Перенос реализуется отдельно (с помощью сложения) от масштабирования и поворота (с помощью умножения). Если выразить точки в однородных координатах, то все три преобразования можно реализовать с помощью умножений. Рассмотрим сначала двумерные преобразования.
Уравнения переноса записываются в виде матрицы преобразования однородных координат следующим образом:
или
P'=PT(dx,dy),
где
T(dx,dy)=.
Иногда подобные выражения записывают таким образом (см. например, [16]):
.
Рассмотрим, например, двойной перенос точки. Пусть необходимо перенести точку Р в точку Р' на расстояние (dx1, dy1), а затем в P'' на расстояние (dx2,dy2). Суммарный перенос должен быть равен расстоянию (dx1+dx2, dy1+dy2). Докажем это. Запишем данные в виде
P'=PT(dx1,dy1),
P''= P'T(dx2,dy2).
Подставляя первую формулу во вторую, получим:
P''= P(T(dx1,dy1)T(dx2,dy2)).
Матричное произведение T(dx1,dy1)T(dx2,dy2) есть
.
Таким образом, действительно, результирующий перенос есть (dx1+dx2, dy1+dy2), т.е. последовательные переносы являются аддитивными.
Уравнения масштабирования в матричной форме с использованием однородных координат записываются в виде
,
S(Sx, Sy)=,
P'=P'S(Sx, Sy).
Матричное произведение S(Sx1, Sy1)S(Sx2, Sy2) есть
.
Таким образом последовательные масштабирования мультипликативны.
И наконец, уравнения поворота (в правосторонней системе) можно представить в виде
[x' y' 1] = [x y 1],
P'=PR(A).
Последовательные повороты являются аддитивными. В качестве задания на самостоятельную работу докажите это.
Композиция двумерных преобразований с помощью однородных координат
Матричное произведение в разных случаях называют объединением, соединением, конкатенацией и композицией. Будем пользоваться последним из перечисленных терминов.
Рассмотрим, например, поворот объекта относительно некоторой произвольной точки P1. Поскольку нам известно, лишь как поворачивать вокруг начала координат, разобьем исходную задачу на три подзадачи:
1) перенос, при котором точка Р1 перемещается в начало координат;
2) поворот;
3) перенос, при котором точка из начала координат возвращается в первоначальное положение Р1.
Последовательность этих преобразований показана на рис.7.
Рис. 7.
Результирующее преобразование имеет вид:
Используя аналогичный подход можно промасштабировать объект относительно произвольной точки Р1: перенести Р1 в начало координат, промасштабировать, перенести назад в точку Р1. Результирующие преобразование в этом случае будет иметь вид
Рассмотрим более сложное преобразование. Предположим, что нам необходимо промасштабировать, повернуть и расположить в нужном месте объект (домик на рис. 8.), где центром поворота и масштабирования является точка Р1.
Рис. 8.
Последовательность преобразований заключается в переносе точки Р1 в начало координат, проведении масштабирования и поворота, а затем переносе из начала координат в новую позицию Р2. В структуре данных прикладной программы, в которой содержится это преобразование, могут находиться масштабный множитель (множители), угол поворота и величины переноса или может быть записана матрица результирующего преобразования:
T(-x1,-y1)S(Sx,Sy)R(A)T(x2,y2).
В общем случае перемножение матриц некоммутативно. Если М1 и М2 представляют собой элементарные перенос, масштабирование или поворот, в следующих частных случаях коммутативность имеет место:
-
М1
|
М2
|
Перенос
Масштабирование
Поворот
Масштабирование (при Sx=Sy)
|
Перенос
Масштабирование
Поворот
Поворот
|
Композиция наиболее общего вида, составленная из операций R,S и T, имеет матрицу
.
Ее верхняя часть размером 2х2 является объединенной матрицей поворота и масштабирования, в то время как tx и ty описывают суммарный перенос. Для вычисления РМ как произведения вектора на матрицу размером 3х3 требуются 9 операций умножения и 6 операций сложения. Структура последнего столбца обобщенной матрицы позволяет упростить фактически выполняемые действия:
x' = xr11+yr21+tx,
y' = xr12+yr22+ty,
сводя процесс к 4 операциям умножения и 4 операциям сложения. Это существенно ускоряет процесс, особенно если учесть, что преобразованиям могут подвергаться сотни и даже тысячи точек на каждом изображении. Таким образом, несмотря на то что матрицы размером 3х3 удобны и полезны для совмещения преобразований, в программе целесообразно использовать результирующую матрицу с учетом особенностей ее структуры. Если же умножение матриц выполняется аппаратно с помощью параллельно работающих сумматоров и умножителей (или систолических процессоров), то вопрос эффективности перестает быть актуальным.
Матричное представление трехмерных преобразований с помощью однородных координат
Аналогично тому, как двумерные преобразования описываются матрицами размером 3х3, трехмерные преобразования могут быть представлены в виде матриц размером 4х4. И тогда трехмерная точка (x,y,z) записывается в однородных координатах как (Wx,Wy,Wz,W), где W0. Если W1, для получения трехмерных декартовых координат точки (x,y,z) первые три однородные координаты делятся на W.
Трехмерная система координат, которой мы будем пользоваться в данном разделе, является правосторонней (рис. 9.).
Рис.9. Правосторонняя система координат
Примем соглашение, в соответствии с которым положительными будем считать такие повороты, при которых (если смотреть с конца положительной полуоси в направлении начала координат) поворот на 90о против часовой стрелки будет переводить одну положительную полуось в другую.
На основе этого соглашения строится следующая таблица, которую можно использовать как для правых, так и для левых систем координат:
-
Если ось вращения
|
Положительным будет направление поворота
|
x
y
z
|
от y к z
от z к x
от x к y
|
В трехмерной графике чаще удобна левосторонняя система, так как ее легче представить наложенной на поверхность экрана дисплея (рис. 10.)
Рис. 10. Левосторонняя система координат
Это позволяет естественно интерпретировать тот факт, что точки с большими значениями z находятся дальше от наблюдателя. Отметим, что в левосторонней системе положительными будут повороты, выполняемые по часовой стрелке, если смотреть с конца положительной полуоси в направлении начала координат.
Трехмерный перенос является простым расширением двумерного:
T(dx,dy,dz) = ,
т.е. [xyz1]T(dx,dy,dz) = [x+dx y+dy z+dz 1].
Масштабирование расширяется аналогичным образом:
S(Sx,Sy,Sz) = ,
[xyz1]S(Sx,Sy,Sz) = [Sxx Syy Szz 1].
В трехмерном пространстве поворот вокруг оси z описывается выражением (напомним, это выражение для правосторонней системы, для левостронней знаки синусов поменялись бы местами)
Rz(A) = .
Матрица поворота вокруг оси x имеет вид
Rx(A) = .
Матрица поворота вокруг оси у записывается в виде
Ry(A) = .
Столбцы (и строки) верхней левой подматрицы размером 3х3 матриц Rz(A), Rx(A), Ry(A) представляют собой взаимно ортогональные единичные векторы.
Все эти матрицы преобразований имеют обратные матрицы. Матрица, обратная Т, получается подстановкой знака минус перед dx, dy и dz; обратная S – заменой Sx,Sy и Sz на обратные им значения, а для каждой из трех матриц поворота – выбором отрицательного угла поворота.
Результатом произвольной последовательности поворотов вокруг осей x,y, и z является матрица М, имеющая вид
М = .
Подматрицу поворота размером 3х3 называют ортогональной, поскольку ее столбцы являются взаимно ортогональными единичными векторами. При повороте, задаваемом матрицей, эти единичные векторы совмещаются с осями x,y и z. Матрицы поворота сохраняют длину и углы, а матрицы масштабирования и переноса не сохраняют.
Для любой ортогональной матрицы В обратная матрица совпадает с транспонированной: В-1 = ВТ. Этот результат является полезным, поскольку вычислять матрицу, обратную матрице поворота, приходится часто. Для получения транспонированной матрицы не требуются даже взаимные пересылки между элементами массива, описывающего матрицу. Необходимо только при выборе элементов массива поменять местами индексы строк и столбцов. Этот метод определения обратной матрицы дает тот же результат, что и способ обращения Rx,Ry,Rz, основанный на подстановке знака минус углом перед углом А.
Можно перемножить произвольное число матриц поворота, масштабирования и переноса. Результат всегда будет иметь вид
.
Верхняя левая подматрица R размером 3х3 будет описывать суммарные поворот и масштабирование, в то время как подматрица Т будет задавать последующий совокупный перенос. С точки зрения быстродействия проведения вычислений выгоднее выполнять преобразования явно:
[x' y' z'] = [x y z]R + T,
где R и T - подматрицы общей матрицы.
Пример композиции трехмерных преобразований [7]
Путем объединения элементарных трехмерных преобразований можно получить другие преобразования.
Начальная позиция Конечная позиция
Рис. 11. Преобразование точек Р1, Р2 и Р3 из начальной позиции
в конечную
Рассмотрим пример. Задача состоит в том, чтобы преобразовать отрезки и из начальной позиции в конечную. Точка Р1 переносится в начало координат, располагается вдоль отрицательной полуоси z, а помещается в плоскости yz в той ее половине, где ось у положительна. На длины отрезков преобразование не воздействует.
Как и прежде, разобьем сложную задачу на более простые. В данном случае преобразование можно выполнить за четыре шага:
1. Перенос точки Р1 в начало координат.
2. Поворот вокруг оси у до совмещения с плоскостью yz.
3. Поворот вокруг оси х до совмещения с отрицательной плоскостью z.
4. Поворот вокруг оси z до совмещения с плоскостью yz.
Шаг 1. Перенос Р1 в начало координат:
Т(-x1, -y1, -z1) =
Применение Т к Р1, Р2 и Р3 дает
Р1' = P1T(-x1, -y1, -z1) = [0 0 0 1],
Р2' = P2T(-x1, -y1, -z1) = [x2-x1 y2-y1 z2-z1 1],
Р3' = P3T(-x1, -y1, -z1) = [x3-x1 y3-y1 z3-z1 1].
Шаг 2. Поворот вокруг оси у.
На рис. 12 показаны отрезки и после шага 1 и проекция на плоскость xz.
Рис. 12. Поворот вокруг оси у; проекция поворачивается до совмещения с отрицательной полуосью z
Поворот производится на положительный угол А, для которого
,
,
где
.
В результате подстановки этих значений в матрицу Ry(A) получаем:
.
Шаг 3. Поворот вокруг оси х.
На рис. 13 показан отрезок после шага 2.
Рис. 13.
Поворот производится на отрицательный угол В, для которого
cos(-B) = cosB = ,
sin(-B) = -sinB = ,
где
. (длина отрезка ).
Результатом поворота на шаге 3 является:
P2''' = P2''Rx(B) = P2'Ry(A)Rx(B) = P2TRy(A)Rx(B) =
= ,
т.е. теперь совпадает с отрицательной полуосью z.
Шаг 4. Поворот вокруг оси z.
На рис. 14 показаны и после шага 3, когда Р2'' лежит на отрицательной полуоси z, а Р3''' - в точке
P3''' = [x3''' y3''' z3''' 1] = P3T(-x1, -y1, -z1)Ry(A)Rx(B).
Рис. 14. Поворот вокруг оси z; проекция поворачивается до совмещения с осью у
Поворот производится на положительный угол С, для которого cosC = y3'''/D2, sinC = x3'''/D2, D2 = .
После шага 4 получается конечный результат.
Результирующая матрица
T(-x1,-y1,-z1)Ry(A)Rx(B)Rz(C) = TR
описывает искомое преобразование.
ГЕОМЕТРИЧЕСКИЕ ПРЕОБРАЗОВАНИЯ В ГРАФИЧЕСКОМ СТАНДАРТЕ OPENGL
Для наилучшего освоения трехмерных геометрических преобразований с помощью команд OpenGL необходимо иметь представление о модели процесса вывода трехмерной графики, используемой в этом стандарте. В последующих лекциях, в которых речь пойдет о проецировании трехмерных объектов на видовую поверхность, мы рассмотрим эту модель и то, как в нее интегрируются аффинные преобразования. Кроме того, мы рассмотрим в целом использование операций матричной алгебры с помощью команд OpenGL.
А пока рассмотрим частные примеры использования специализированных команд OpenGL, реализующих аффинные преобразования на плоскости.
Масштабирование
Для изменения масштаба используется команда glScalef с тремя аргументами, представляющими масштабные множители для каждой из осей. Например, если перед командными скобками, в которых описывается объект, вставим строку
glScalef (0.5, 0.5, 1.0);
то будет нарисована уменьшенная в два раза фигура.
Например (пример Chapter\Ex50 из М.Краснова):
glScalef (0.5, 0.5, 1.0);
glBegin (GL_POLYGON);
glVertex2f (-0.23678, 0.35118);
glVertex2f (-0.23678, 0.7764);
glVertex2f (-0.37966, 0.7764);
glVertex2f (-0.55, 0.60606);
glVertex2f (-0.55, -0.4);
glVertex2f (-0.23576, -0.4);
glVertex2f (-0.23678, 0.35118);
glVertex2f (-0.23576, -0.4);
glVertex2f (0.1375, -0.4);
glVertex2f (0.13678, 0.35118);
glEnd;
glBegin (GL_POLYGON);
glVertex2f (0.1375, -0.4);
glVertex2f (0.45, -0.4);
glVertex2f (0.45, 0.60606);
glVertex2f (0.27966, 0.7764);
glVertex2f (0.13678, 0.7764);
glVertex2f (0.13678, 0.35118);
glEnd;
glScalef (2.0, 2.0, 1.0);
После команд рисования масштаб восстанавливается.
Библиография
1. Шикин Е.В., Боресков А.В. Компьютерная графика. М.:Диалог-МИФИ, 1995. 288 с.
2. Аммерал Л. Машинная графика на языке Си. М.:СолСистем, 1992.
Т.1:Принципы программирования в машинной графике. 224 с.
Т.2:Машинная графика на персональных компьютерах. 232 с.
Т.3:Интерактивная трехмерная машинная графика. 317 с.
Т.4:Программирование графики на Турбо Си. 221 с.
3. Роджерс Д. Алгоритмические основы машинной графики. - М.:Мир, 1989. - 512 с.
4. Абраш М. Таинства программирования графики. Киев.:ЕвроСИБ, 1996. 384 с. ил.
5. Довгаль В.М., Керекеша В.В., Веретенников А.А. Растровые продукционные системы // Известия КГТУ, № 2, Курск. гос. техн. ун-т, 1998. С.110-119.
6. Хирн Д., Бейкер М. Микрокомпьютерная графика. М.:Мир, 1987. - 352 с.
7. Фоли Дж., вэн Дэм А. Основы интерактивной машинной графики. М.:Мир, 1985. - Т.1: 375 с., Т.2: 368 с.
8. Фролов А.В., Фролов Г.В. Программирование видеоадаптеров. М.: ДИАЛОГ-МИФИ, 1995. 272 с. - (Библиотека системного программиста; Т.21)
9. Корриган Дж. Компьютерная графика: Секреты и решения: Пер. с англ. - М.:Энтроп, 1995. 352 с.
10. Прокофьев Б.П., Сухарев Н.Н., Храмов Ю.Е. Графические средства Turbo C и Turbo C++/ ред. Г.В.Генса, Ю.Е.Храмова. М.: Финансы и стастистика, СП “Ланит”, 1992. 160 с.
11. Марков А.А., Нагорный Р.М. Теория алгорифмов. М.: Наука, 1984. 432 с.
12. Тоффоли Т., Марголус Н. Машины клеточных автоматов: Пер. с англ. М.: Мир, 1991. 280 с.
13. Методы модификации формальных систем обработки символьной информации /В.М.Довгаль; Курск. гос. техн. ун-т. Курск, 1996. 115 с.
14. Керекеша В.В. Конструктивные объекты компьютерной графики // Сборник материалов 3 Международной конференции Распознавание-97. Курск. гос. техн. ун-т. Курск, 1997. С.207-208.
15. Титов В.С., Довгаль В.М., Керекеша В.В., Веретенников А.А. Разложение в растр окружности с помощью системы продукционных правил // Межвузовский сборник научных трудов "Информационные технологии моделирования и управления". Воронежский гос. техн. ун-т. Воронеж, 1998. С.133-136.
16. Тихомиров Ю. Программирование трехмерной графики. СПб.: BHV – Санкт-Петербург, 1998. 256 с.
17. Segal M., Akeley K. The OpenGL Graphics System: A Specification (Version 1.1 – 25 June 1996). Editor: Chris Frazier.
18. OpenGL for Windows from Silicon Graphics. Release Notes for Manufacturing Release 1. (www.sgi.com, 1997.)
|