Краткая теория. Преобразования объектов
Преобразования объектов
В OpenGL используются как основные три системы координат: левосторонняя, правосторонняя и оконная. Первые две системы являются трехмерными и отличаются друг от друга направлением оси z: в правосторонней она направлена на наблюдателя, в левосторонней – в глубину экрана. Ось x направлена вправо относительно наблюдателя, ось y – вверх.
Левосторонняя система используется для задания значений параметрам команды gluPerspective(), glOrtho(), которые будут рассмотрены в пункте 0. Правосторонняя система координат используется во всех остальных случаях. Отображение трехмерной информации происходит в двумерную оконную систему координат.
Рис. 2 Системы координат в OpenGL
Строго говоря, OpenGL позволяет путем манипуляций с матрицами моделировать как правую, так и левую систему координат. Но на данном этапе лучше пойти простым путем и запомнить: основной системой координат OpenGL является правосторонняя система.
Работа с матрицами
Для задания различных преобразований объектов сцены в OpenGL используются операции над матрицами, при этом различают три типа матриц: модельно-видовая, матрица проекций и матрица текстуры. Все они имеют размер 4x4. Видовая матрица определяет преобразования объекта в мировых координатах, такие как параллельный перенос, изменение масштаба и поворот. Матрица проекций определяет, как будут проецироваться трехмерные объекты на плоскость экрана (в оконные координаты), а матрица текстуры определяет наложение текстуры на объект.
Умножение координат на матрицы происходит в момент вызова соответствующей команды OpenGL, определяющей координату (как правило, это команда glVertex*)
Для того чтобы выбрать, какую матрицу надо изменить, используется команда:void glMatrixMode (GLenum mode)
вызов которой со значением параметра mode равным GL_MODELVIEW, GL_PROJECTION, или GL_TEXTURE включает режим работы с модельно-видовой матрицей, матрицей проекций, или матрицей текстуры соответственно. Для вызова команд, задающих матрицы того или иного типа, необходимо сначала установить соответствующий режим.
Для определения элементов матрицы текущего типа вызывается командаvoid glLoadMatrix[f d] (GLtype *m)
где m указывает на массив из 16 элементов типа float или double в соответствии с названием команды, при этом сначала в нем должен быть записан первый столбец матрицы, затем второй, третий и четвертый. Еще раз обратим внимание: в массиве m матрица записана по столбцам.
Командаvoid glLoadIdentity (void)
заменяет текущую матрицу на единичную.
Часто бывает необходимо сохранить содержимое текущей матрицы для дальнейшего использования, для чего применяются команды void glPushMatrix (void) void glPopMatrix (void)
Они записывают и восстанавливают текущую матрицу из стека, причем для каждого типа матриц стек свой. Для модельно-видовых матриц его глубина равна как минимум 32, для остальных – как минимум 2.
Для умножения текущей матрицы на другую матрицу используется командаvoid glMultMatrix[f d] (GLtype *m)
где параметр m должен задавать матрицу размером 4x4. Если обозначить текущую матрицу за М, передаваемую матрицу за T, то в результате выполнения команды glMultMatrix текущей становится матрица M * T. Однако обычно для изменения матрицы того или иного типа удобно использовать специальные команды, которые по значениям своих параметров создают нужную матрицу и умножают ее на текущую.
Модельно-Видовые преобразования
К модельно-видовым преобразованиям будем относить перенос, поворот и изменение масштаба вдоль координатных осей. Для проведения этих операций достаточно умножить на соответствующую матрицу каждую вершину объекта и получить измененные координаты этой вершины:
(x’, y’, z’, 1)T = M * (x, y, z, 1)T
где M – матрица модельно-видового преобразования. Перспективное преобразование и проектирование производится аналогично. Сама матрица может быть создана с помощью следующих команд:void glTranslate[f d] (GLtype x, GLtype y, GLtype z) void glRotate[f d] (GLtype angle, GLtype x, GLtype y, GLtype z) void glScale[f d] (GLtype x, GLtype y, GLtype z)
glTranlsate*() производит перенос объекта, прибавляя к координатам его вершин значения своих параметров.
glRotate*() производит поворот объекта против часовой стрелки на угол angle (измеряется в градусах) вокруг вектора (x,y,z).
glScale*() производит масштабирование объекта (сжатие или растяжение) вдоль вектора (x,y,z), умножая соответствующие координаты его вершин на значения своих параметров.
Все эти преобразования изменяют текущую матрицу, а поэтому применяются к примитивам, которые определяются позже. В случае, если надо, например, повернуть один объект сцены, а другой оставить неподвижным, удобно сначала сохранить текущую видовую матрицу в стеке командой glPushMatrix(), затем вызвать glRotate() с нужными параметрами, описать примитивы, из которых состоит этот объект, а затем восстановить текущую матрицу командой glPopMatrix().
Кроме изменения положения самого объекта, часто бывает необходимо изменить положение наблюдателя, что также приводит к изменению модельно-видовой матрицы. Это можно сделать с помощью командыvoid gluLookAt (GLdouble eyex, GLdouble eyey, GLdouble eyez, GLdouble centerx, GLdouble centery, GLdouble centerz, GLdouble upx, GLdouble upy, GLdouble upz)
где точка (eyex,eyey,eyez) определяет точку наблюдения, (centerx, centery, centerz) задает центр сцены, который будет проектироваться в центр области вывода, а вектор (upx,upy,upz) задает положительное направление оси у, определяя поворот камеры. Если, например, камеру не надо поворачивать, то задается значение (0,1,0), а со значением (0,-1,0) сцена будет перевернута.
Строго говоря, эта команда совершает перенос и поворот объектов сцены, но в таком виде задавать параметры бывает удобнее. Следует отметить, что вызывать команду gluLookAt() имеет смысл перед определением преобразований объектов, когда модельно-видовая матрица равна единичной.
Запомните: В общем случае матричные преобразования в OpenGL нужно записывать в обратном порядке. Например, если вы хотите сначала повернуть объект, а затем передвинуть его, сначала вызовите команду glTranslate(), а только потом – glRotate(). Ну а после этого определяйте сам объект.
Исходный код программы:
using System; using System.Collections.Generic; using System.ComponentModel; using System.Data; using System.Drawing; using System.Linq; using System.Text; using System.Windows.Forms; using System.OpenGL;
namespace Furniture { public partial class Form1: Form { public Form1() { InitializeComponent(); }
private void openGLControl1_PrimaryPaint(object sender, EventArgs e) { GL.glDisable(GL.GL_LIGHTING); GL.glColor3ub(100,100,0);
GL.glMatrixMode(GL.GL_MODELVIEW); var d = new double[] { 1, 0, 0, 0, 0, 1, 0, 0, 0, 0, 1, 0, 0, 0, 500,1 }; GL.glPushMatrix(); GL.glMultMatrixd(d);
GL.glBegin(GL.GL_QUADS); //полотно //GL.glVertex3d(-2000, 0, -2000); //GL.glVertex3d(-2000, 0, 2000); //GL.glVertex3d(2000, 0, 2000); //GL.glVertex3d(2000, 0, -2000);
GL.glColor3ub(100, 200, 0); //шифер GL.glVertex3d(250, 300, 250); GL.glVertex3d(750, 300, 250); GL.glVertex3d(750, 500, 500); GL.glVertex3d(250, 500, 500);
GL.glVertex3d(250, 500, 500); GL.glVertex3d(750, 500, 500); GL.glVertex3d(750, 300, 750); GL.glVertex3d(250, 300, 750);
GL.glColor3ub(100, 200, 100); //Лаги(бревна) GL.glVertex3d(250, 300, 250); GL.glVertex3d(750, 300, 250); GL.glVertex3d(750, 300, 750); GL.glVertex3d(250, 300, 750);
GL.glColor3ub(50, 99, 50); //Стены ==========================> GL.glVertex3d(250, 0, 300); GL.glVertex3d(250, 300, 300); GL.glVertex3d(750, 300, 300); GL.glVertex3d(750, 0, 300);
GL.glVertex3d(250, 0, 700); GL.glVertex3d(250, 300, 700); GL.glVertex3d(750, 300, 700); GL.glVertex3d(750, 0, 700);
GL.glVertex3d(250, 0, 300); GL.glVertex3d(250, 300, 300); GL.glVertex3d(250, 300, 700); GL.glVertex3d(250, 0, 700);
GL.glVertex3d(750, 0, 300); GL.glVertex3d(750, 300, 300); GL.glVertex3d(750, 300, 350); GL.glVertex3d(750, 0, 350);
GL.glVertex3d(750, 0, 650); GL.glVertex3d(750, 300, 650); GL.glVertex3d(750, 300, 700); GL.glVertex3d(750, 0, 700);
GL.glVertex3d(750, 255, 350); GL.glVertex3d(750, 300, 350); GL.glVertex3d(750, 300, 650); GL.glVertex3d(750, 255, 650); //Стены ==========================<
GL.glEnd(); //треугольники GL.glColor3ub(100, 100, 100); GL.glBegin(GL.GL_TRIANGLES);
GL.glVertex3d(250, 300, 250); GL.glVertex3d(250, 300, 750); GL.glColor3ub(0, 0, 0); GL.glVertex3d(250, 500, 500);
GL.glColor3ub(100, 100, 100); GL.glVertex3d(750, 300, 250); GL.glVertex3d(750, 300, 750); GL.glColor3ub(0, 0, 0); GL.glVertex3d(750, 500, 500);
GL.glEnd();
GL.glPopMatrix();
GL.glBegin(GL.GL_QUADS); //полотно //GL.glVertex3d(-2000, 0, -2000); //GL.glVertex3d(-2000, 0, 2000); //GL.glVertex3d(2000, 0, 2000); //GL.glVertex3d(2000, 0, -2000);
GL.glColor3ub(100, 200, 0); //шифер GL.glVertex3d(250, 300, 250); GL.glVertex3d(750, 300, 250); GL.glVertex3d(750, 500, 500); GL.glVertex3d(250, 500, 500);
GL.glVertex3d(250, 500, 500); GL.glVertex3d(750, 500, 500); GL.glVertex3d(750, 300, 750); GL.glVertex3d(250, 300, 750);
GL.glColor3ub(100, 200, 100); //Лаги(бревна) GL.glVertex3d(250, 300, 250); GL.glVertex3d(750, 300, 250); GL.glVertex3d(750, 300, 750); GL.glVertex3d(250, 300, 750);
GL.glColor3ub(50, 99, 50); //Стены ==========================> GL.glVertex3d(250, 0, 300); GL.glVertex3d(250, 300, 300); GL.glVertex3d(750, 300, 300); GL.glVertex3d(750, 0, 300);
GL.glVertex3d(250, 0, 700); GL.glVertex3d(250, 300, 700); GL.glVertex3d(750, 300, 700); GL.glVertex3d(750, 0, 700);
GL.glVertex3d(250, 0, 300); GL.glVertex3d(250, 300, 300); GL.glVertex3d(250, 300, 700); GL.glVertex3d(250, 0, 700);
GL.glVertex3d(750, 0, 300); GL.glVertex3d(750, 300, 300); GL.glVertex3d(750, 300, 350); GL.glVertex3d(750, 0, 350);
GL.glVertex3d(750, 0, 650); GL.glVertex3d(750, 300, 650); GL.glVertex3d(750, 300, 700); GL.glVertex3d(750, 0, 700);
GL.glVertex3d(750, 255, 350); GL.glVertex3d(750, 300, 350); GL.glVertex3d(750, 300, 650); GL.glVertex3d(750, 255, 650); //Стены ==========================<
GL.glEnd(); //треугольники GL.glColor3ub(100, 100, 100); GL.glBegin(GL.GL_TRIANGLES);
GL.glVertex3d(250, 300, 250); GL.glVertex3d(250, 300, 750); GL.glColor3ub(0, 0, 0); GL.glVertex3d(250, 500, 500);
GL.glColor3ub(100, 100, 100); GL.glVertex3d(750, 300, 250); GL.glVertex3d(750, 300, 750); GL.glColor3ub(0, 0, 0); GL.glVertex3d(750, 500, 500);
GL.glEnd();
}
private void выходToolStripMenuItem_Click(object sender, EventArgs e) { this.Close(); }
private void оПрограммеToolStripMenuItem_Click(object sender, EventArgs e) { MessageBox.Show("HELLO OpenGL!!!:)", "HELLO!!!"); } } }
Скриншот программы:
|