Bắt đầu với OpenGL bằng GLKit trên iOS
Nếu bạn từng có hứng thú về lập trình đồ hoạ, chắc bạn đã nghe đến OpenGL. Nó cung cấp những rất manh để sử lý đồ hoạ cả phần mềm và tương tác phần cứng. Do đó Apple đã tạo ra 1 framework tên là GLKit để giúp các lập trình viên sử dụng OpenGL 1 cách dễ dàng hơn, bạn có thể tập trung vào việc vẽ ko ...
Nếu bạn từng có hứng thú về lập trình đồ hoạ, chắc bạn đã nghe đến OpenGL. Nó cung cấp những rất manh để sử lý đồ hoạ cả phần mềm và tương tác phần cứng. Do đó Apple đã tạo ra 1 framework tên là GLKit để giúp các lập trình viên sử dụng OpenGL 1 cách dễ dàng hơn, bạn có thể tập trung vào việc vẽ ko cần quan tâm quá nhiều đến việc thiết lập project. Trong bài viết này tôi sẽ trình bày 1 hướng dẫn đơn giản về GLKit trên iOS
GLKit cung cấp cho bạn các tính năng trong 4 phần:
- Views và View Controllers: tạo code sẵn để bạn chó thể khởi tạo OpenGL ES trong project
- Hiệu ứng: Cung cấp những hiệu ứng thông thường như: Shading, lighting, reflection, skybox...
- Thuật toán: Cung cấp những Helpers giúp tính toán khi vẽ như vector, sinh ma trận...
- Texture loading: Giúp việc load ảnh thành texture để sử dụng dễ dàng hơn
Bắt đầu
Để bắt đầu chúng ta sẽ tạo 1 app đơn giản chỉ là hiển thị 1 hình vuông trên màn hình và rotate nó. Đầu tiên hãy tạo 1 project đơn giản vd tên: GLKitExample, sử dụng Swift làm ngôn ngữ.
Làm quen với GLKView và GLKViewController
Đầu tiên hãy import GLKit và kế thừa ViewController bở GLKViewController
Sau đó khởi tạo GLKViewController trên storyboard bằng cách kéo 1 đối tượng vào thay đổi custom class bằng ViewController
Ở phần Attributes inspector, click vào GLKView chúng ta có thể thấy 1 số settings cho màu sắc, độ sâu, stencil, multisampling. Bạn có thể thay đổi chúng ở những yêu cầu cao hơn, nhưng ở đây chúng ta chỉ cần 1 ví dụ đơn giản nên sử dụng setting mặc định là đủ.
OpenGL context có 1 bộ đệm để lưu những màu được hiển thị trên màn hình. Bạn có thể dùng tuỳ biến Color Format để thay đổi chuẩn màu của mỗi pixel trong bộ đệm.
Giá trị mặc định là GLKViewDrawableColorFormatRGBA8888, nghĩa là với 8 bit được sử dụng cho mỗi thành phần màu trong bộ đệm. Nó là tối ưu để hiển thị dải màu rộng nhất để làm việc giúp cho việc hiển thị đạt chất lượng cao hơn.
View controller của bạn đã được cài đặt với GLKView để vẽ OpenGL và nó cũng được cung cấp các delegate để cập nhật việc vẽ.
Chúng ta sẽ add thêm đoạn code tiếp theo:
Đây là những gì xảy ra ở đoạn code trên:
- Để làm bất kỳ việc gì với OpenGL, bạn cần tạo 1 EAGLContext Context quản lý tất cả những thông tin và iOS cần để vẽ với OpenGL. Giống như với Core Graphics thì chúng ta cũng cần context vậy. Khi khởi tạo chúng ta cần truyền vào version của API mà bạn muốn sử dụng ở đây là OpenGL ES 3.0.
- Dòng code thứ 2 để làm việc render context được chạy trên thread hiện tại OpenGL context ko nên được chia sẻ bởi các thread, bởi vậy bạn sẽ phải chắc chắn là bạn kết nối với context từ thread mà mà call method setupGL()
- Đoạn code thứ 3 : set context cho glkview của bạn
- Đoạn code thứ 4: set delegate để cập nhật logic và trạng thái
Tiếp gọi hàm setupGL trong viewDidLoad(). Ở đây ta có thể thấy thread mà chúng ta gọi là ở main thread
Cho ViewController của bạn thực thi delegate.
Tiếp theo chúng ta tạo 1 function:
Đoạn code này sẽ vẽ nội dung trong mỗi frame. Chúng ta cùng phân tích:
- Gọi glClearColor để chỉ ra giá trị RGB và alpha sử dụng để làm sạch màn hình, trong ví dụ này là màu xám
- Gọi glClear để thực tế gọi thực thi hàm clear.
Chạy app chúng ta sẽ hiện thị màn hình với màu sắc thay đổi:
Tạo dữ liệu đỉnh của 1 hình vuông đơn giản
Để tạo 1 hình vuông, chúng ta cần xác định được các đỉnh, đơn giản là các điểm chúng ta vẽ nối lại thành hình mình cần
Chúng ta sẽ cài đặt như sau:
Chỉ có thể dựng hình tam giác bằng cách sử dụng OpenGL. Tuy nhiên, bạn có thể tạo một hình vuông với hai hình tam giác như bạn có thể thấy trong hình trên: Một hình tam giác có đỉnh (0, 1, 2) và một hình tam giác có đỉnh (2, 3, 0).
Một trong những điều khá ngon của OpenGL ES là bạn có thể giữ dữ liệu đỉnh của bạn được tổ chức theo ý bạn. Trong project này, bạn sẽ sử dụng 1 struct Swift để lưu trữ thông tin vị trí và màu sắc của đỉnh và sau đó là một loạt các đỉnh cho mỗi điểm mà bạn sẽ sử dụng để vẽ.
Tạo file Vertex.swift với nội dung như sau:
Tiếp đến bạn sử dụng struct này để tạo một mảng các đỉnh để vẽ. Sau đó, bạn tạo một mảng các giá trị GLubyte, mảng này xác định thứ tự để vẽ mỗi một trong ba đỉnh tạo nên một hình tam giác:
Cách tốt nhất để truyền data cho OpenGL là qua Vertex Buffer Objects. Có 3 loại:
- Vertex Buffer Object (VBO): Theo dõi dữ liệu trên mỗi đỉnh, giống như dữ liệu bạn có trong mảng Vertices
- Element Buffer Object (EBO): Theo dõi các chỉ số xác định tam giác, giống như các chỉ mục bạn đã lưu trữ trong mảng Indices
- Vertex Array Object (VAO): Đối tượng này có thể đươc giữ data giống như đối tượng vertex buffer. Bất kỳ lời gọi thuộc tính đỉnh nào mà bạn thực hiện - sau khi liên kết một vertex array object - sẽ được lưu trữ bên trong nó. Điều này có nghĩa là bạn chỉ phải thực hiện lời gọi để định cấu hình con trỏ thuộc tính đỉnh một lần và sau đó - bất cứ khi nào bạn muốn vẽ một đối tượng - bạn chỉ cần liên kết với VAO tương ứng. Điều này tạo điều kiện tăng tốc độ vẽ dữ liệu đỉnh khác nhau với các cấu hình khác nhau.
Ở phía trên cùng của ViewController.swift, thêm phần extension Array sau đây để giúp nhận được kích thước, tính theo byte, của các mảng Vertices và Indices:
Tiếp theo them các biến sau:
Đây là các biến cho đối tượng buffer element, vertex buffer và vertex array object. Tất cả đều thuộc loại GLuint.
Cài đặt các bộ đệm
Đầu tiên cần khởi taọ 1 số biến helpers trong hàm setupGL()
- 1, 2: 2 biến đầu tiền thể hiện cách đọc màu và vị trí của từ cấu trúc dữ liệu. Ở đây chúng ta dùng GLKVertexAttrib
- 3: Ở đây, bạn tận dụng lợi thế của MemoryLayout để có được stride, đó là kích thước, tính theo byte, của một struct Vertex khi trong một mảng.
- 4: Để có được bộ nhớ các biến tương ứng với một màu đỉnh, bạn sử dụng MemoryLayout một lần nữa ngoại trừ lần này bạn xác định rằng bạn muốn stride của một GLfloat nhân ba. Điều này tương ứng với các biến x, y và z trong struct Vertex.
- 5: Cuối cùng, bạn cần chuyển đổi offset thành kiểu được yêu cầu: UnsafeRawPointer.
Tạo bộ đệm VAO
Đơn giản với 2 dòng code:
Dòng đầu tiên yêu cầu OpenGL tạo một VAO. Phương thức này cần hai tham số: Đầu tiên là số lượng VAO để tạo - trong trường hợp này là 1 - trong khi thứ hai mong đợi một con trỏ tới một GLuint trong đó nó sẽ lưu trữ ID của đối tượng được tạo ra.
Trong dòng thứ hai, bạn đang nói OpenGL để ràng buộc VAO mà bạn tạo ra và được lưu trữ trong biến vao và bất kỳ lời gọi sắp tới để cấu hình con trỏ thuộc tính đỉnh nên được lưu trữ trong VAO này. OpenGL sẽ sử dụng VAO cho đến khi bạn hủy liên kết hoặc liên kết một cái khác trước khi thực hiện lệnh gọi vẽ.
Tạo bộ đệm VBO
Chúng ta tiếp tục với đoạn code sau:
Giống như với VAO, đầu tiên tạo một VBO và lưu trữ mã định danh của nó trong biến vbo.
Sau khi đã tạo VBO, bây giờ bạn đã liên kết với nó bằng cách gọi glBindBuffer. GL_ARRAY_BUFFER được sử dụng để xác định rằng bạn đang ràng buộc một bộ đệm các đỉnh.
Cuộc gọi đến glBufferData là nơi bạn đang chuyển tất cả thông tin đỉnh của mình sang OpenGL. Có bốn tham số mà phương thức này mong đợi:
- Cho biết bộ đệm đang truyền dữ liệu.
- Chỉ định kích thước, tính bằng byte, của dữ liệu. Trong trường hợp này, sử dụng phương thức helper size() trên Array mà bạn đã viết trước đó.
- Dữ liệu thực tế sử dụng.
- Cho OpenGL biết cách bạn muốn GPU quản lý dữ liệu. Trong trường hợp này, bạn sử dụng GL_STATIC_DRAW vì dữ liệu bạn đang chuyển đến card đồ họa sẽ hiếm khi thay đổi. Điều này cho phép OpenGL tối ưu hóa cho kịch bản hiện tại.
Bây giờ bạn đã chuyển dữ liệu màu sắc và vị trí cho tất cả các đỉnh của bạn tới GPU. Nhưng bạn vẫn cần cho OpenGL biết cách diễn giải dữ liệu đó khi bạn yêu cầu nó vẽ tất cả trên màn hình. Để làm điều đó, hãy thêm mã này vào cuối setupGL():
Gọi glEnableVertexAttribArray gọi bật thuộc tính là vị trí. glVertexAttribPointer cần 6 biến truyền vào. Đây là ý nghĩa: - Tên của thuộc tính
- Bao nhiêu giá trị thể hiện cho mỗi đỉnh
- Kiểu của mỗi giá trị
- Có cần chuẩn hoá dữ liệu không? Mặc định là false
- size của stride
- Offset của dữ liệu về vị trí Tạo VBO đã xong, chúng ta sang phần tiếp tạo bộ đệm EBO
Tạo bộ đệm EBO
Add đoạn code còn lại để tạo:
Bây giờ thì bạn đã khá quen thuộc với đoạn code này, vẫn là khởi tạo, tạo bộ đệm, liên kết dữ liêụ... Ba dòng code cuối cùng đã là huỷ liên kết dữ liệu, bởi lúc này mọi thứ đã được cài đặt xong. Sau 1 quá trình bây giờ bạn có thể build lại project của mình, tôi hy vọng là nó compile thành công