-
Notifications
You must be signed in to change notification settings - Fork 186
Batching Rectangles and Lines
Often you may want to render rectangles, lines, and squares -- be it for debugging or any other visual purpose. In the programmable pipeline there is no longer glRect
nor can you simply "disable textures" to render a primitive shape. The solution is to find a 1x1 white opaque pixel in your UI texture atlas from which to sample from (or more than 1x1 pixels if you are scaling with GL_LINEAR
). Then, you can render boxes, lines, and other basic shapes.
In Photoshop, I added a small 4x4 white square to the corner of my font texture atlas: see here. I then grabbed it with a TextureRegion. We also could have created a 1x1 white GL_NEAREST
Texture with a ByteBuffer (see here), but having multiple textures would not allow us to take advantage of our SpriteBatch.
Here are a couple of utility methods for rendering lines and rectangles:
void drawRect(int x, int y, int width, int height, int thickness) {
batch.draw(rect, x, y, width, thickness);
batch.draw(rect, x, y, thickness, height);
batch.draw(rect, x, y+height-thickness, width, thickness);
batch.draw(rect, x+width-thickness, y, thickness, height);
}
void drawLine(int x1, int y1, int x2, int y2, int thickness) {
int dx = x2-x1;
int dy = y2-y1;
float dist = (float)Math.sqrt(dx*dx + dy*dy);
float rad = (float)Math.atan2(dy, dx);
batch.draw(rect, x1, y1, dist, thickness, 0, 0, rad);
}
You can see them in use in RectTest:
This is a very simple means of rendering shapes. The technique in this article relies on the CPU to transform each vertex of the line sprite; and under the hood it's made up of two triangles, i.e. 6 vertices. That's a lot of work just to render a 1 px line segment! For filled polygons, anti-aliasing, stroke effects, and so forth, we may need to take advantage of fragment and geometry shaders. In other cases, GL_LINES
would be much more effective.