OpenGL viewport / rendering partial tiles

Started by
3 comments, last by cyotek 11 years, 11 months ago
Hello,

Apologies if this is a somewhat idiotic question, I'm still somewhat new to OpenGL programming. In fact, I'm not even sure if this is the right forum or if it should have been posted in Beginners!

Firstly, some background.

I currently have a basic 2D tiling engine which uses OpenGL via the OpenTK bindings. For drawing textures, the engine is currently using the somewhat obsolete vertex arrays rather than VBO's (one challenge at a time!)

Each map is a simple 2D array of tiles. I define a field size which is the visible viewport on the screen, and set up an array of vertexes which describe each tile position in the viewport. This is defined once. There is also an origin, which is simply the top/left co-ordinate of the map in relation to the player.

Then, each update, the engine cycles through the visible tiles in the current map and updates the vertex array with texture coordinates from a tile sheet texture. If the player moves and the origin changes, this is naturally handled as part of the cycle through visible tiles. It's a very simple system smile.png

So far so good.

However, the disadvantage is that it "scrolls" in full tiles. So if the character charges from one side of the screen to the other, it will jump along one block at a time.

What I wanted to do was implement a form a smooth scrolling, so that instead of jumping the entire width of a tile, it will jump a smaller increment until it has reached the desired end position.

I tried implementing smooth scrolling in a test project some months ago, and while it worked, it had one nasty drawback - on one side of the scroll you'd have a gap as technically the subsquent column wasn't visible so it wouldn't be drawn. And on the other side of the scroll you'd have an excess because it was trying to draw the last column beyond the boundaries of the view port. So assuming you had UI elements, they could be overwritten.

If I was using GDI, then I might just set a clip region and then just draw the visible tiles +/- 1 column/row and everything would be displayed without overwriting any element. Of course, that would mean extra rendering being done for no purpose which is a bit daft.

Not really sure I'm explaining this terribly well, so here's a couple of images to partially demonstrate:

[attachment=8688:jwlrush 2012-05-05 13-49-45-90.png]

[attachment=8689:jwlrush 2012-05-05 13-49-47-30.png]

In the second screen shot, the player has moved one tile to the right. As the engine currently tries to keep the player character centred in the map, it automatically scrolls it along one tile. Which is fine, as the end result - but I want to smoothly bump it along instead of it just jumping.

[attachment=8693:OpenGLTileMap 2012-05-05 14-23-26-95.png]

This third image demonstrates a test program I did which tried smoothly scrolling the background from right to left and also scrolling the water from left to right. The jagged edges at the top and bottom show how my rendering is currently flawed.

So, my questions would be:

Firstly, does OpenGL support anything like GDI's clip regions so I could draw extra tiles without them going beyond what I define as a view port. [The lazy approach I suppose]

-or-

Should I recalculate the first/last column/row and draw only the visible portion of the texture [The hurt-your-brain-with-maths approach]

-or-

Is there another technique that other developers use that I haven't thought of?

Again, apologies if this is a really dumb question, but thanks in advance for any advice that could be offered.

Regards;
Richard Moss
Advertisement
The viewport already clips your tiles (I assume, since you mentioned vertex arrays, that they aren't drawn as pixel rectangles).

The easy solution isn't really that difficult. I get the feeling that you may be thinking too much about complex solutions to this.

  1. Define the viewport as the region of the screen where you want the game to be drawn. I see in your first two screen shots that the map covers the whole screen except for the top row. What I mean here is that your viewport should only cover the region of the screen where the map is to be drawn. Make a separate viewport for the score board in a second pass, but I will not consider that here.
  2. Set the coordinate system (for example with glOrtho) such that the desired part of the map is visible within the viewport. All you need to do to scroll the map is to adjust the coordinate system. No need to touch the viewport itself.
  3. Draw the visible tiles. You know the bounds of the visible region of the map, so draw only those tiles if you want to do some primitive culling. You can draw the entire map as well if you want, since the viewport is clipping anyway. If you want partial-tile-scrolling, then you just round the visible region's left and bottom borders down to nearest tile-border, and the right and upper borders up to nearest tile-border.

With these steps where you separate the viewport, the coordinate system and the drawing, you really don't have to do much to implement scrolling and clipping. It's all handled pretty much automatically.

edit: By the way, what the name of the game in the first two images? It was one of my favorite games way back, but I just can't remember the name now.
Thanks for the reply.

I do set a co-ordinate system, but I set that once when the window is created (and I reset it if the window is resized). Didn't think about creating a sub co-ordinate system as such.

When I say "viewport", I'm just referring to the region of screen that I'm drawing in, I'm not actually creating view ports with OpenGL - but your reply in point 3 seems to infer that OpenGL can in fact do them. Guess I need to go and look over the API reference to find out how smile.png

The screenshots are from my clone of a old arcade game called Boulder Dash that I used to play on an Atari 800XL, using some custom graphics I'm currently trying to draw too.

Thanks again for your reply!

Regards;
Richard Moss
What you want it for the viewport is glViewport; one for the map and one for the score board. I forgot to mention that function in point 1.

And cheers for the name. I couldn't stop trying to remember it since my reply :)
Thanks for the follow up. I didn't realize you can have multiple viewports - I was setting up a default one once when the game started to the window size. Cool, the solution does seem to be easier that what I was thinking [no horrible calculations to work out bits of tiles!], I shall go do some more experimenting :)

This topic is closed to new replies.

Advertisement