Лучший способ генерировать плитки - PullRequest
1 голос
/ 15 марта 2010

Начну с того, что я ДЕЙСТВИТЕЛЬНО новичок в OpenGL ES (я начал вчера =), но у меня есть опыт работы с Java и другими языками.

Я посмотрел много учебников, конечно же, Них, и моя работа в основном основана на этом.

В качестве теста я начал создавать «генератор плиток», чтобы создать маленькую игру, похожую на Zelda (просто переместить чувака в текстурированный квадрат было бы круто: p).

Пока что я получил работающий генератор плиток, я определяю массив char map [] [] для хранения, в котором включена плитка:

private char[][] map = {
            {0, 0, 20, 11, 11, 11, 11, 4, 0, 0},
            {0, 20, 16, 12, 12, 12, 12, 7, 4, 0},
            {20, 16, 17, 13, 13, 13, 13, 9, 7, 4},
            {21, 24, 18, 14, 14, 14, 14, 8, 5, 1},
            {21, 22, 25, 15, 15, 15, 15, 6, 2, 1},
            {21, 22, 23, 0, 0, 0, 0, 3, 2, 1},
            {21, 22, 23, 0, 0, 0, 0, 3, 2, 1},
            {26, 0, 0, 0, 0, 0, 0, 3, 2, 1},
            {0, 0, 0, 0, 0, 0, 0, 0, 0, 1},
            {0, 0, 0, 0, 0, 0, 0, 0, 0, 1}
    };

Это работает, но я не доволен этим, я уверен, что есть лучший способ сделать это:

1) Загрузка текстур:

Я создаю уродливый массив, содержащий плитки, которые я хочу использовать на этой карте:

private int[] textures = {
            R.drawable.herbe, //0
            R.drawable.murdroite_haut, //1
            R.drawable.murdroite_milieu, //2
            R.drawable.murdroite_bas, //3
            R.drawable.angledroitehaut_haut, //4
            R.drawable.angledroitehaut_milieu,  //5
    };

(я специально вырезал это, сейчас загружаю 27 плиток)

Все тезисы хранятся в папке для рисования, каждый из которых представляет собой плитку размером 16 * 16.

Затем я использую этот массив для генерации текстур и сохранения их в HashMap для дальнейшего использования:

int[] tmp_tex = new int[textures.length]; 
gl.glGenTextures(textures.length, tmp_tex, 0); 
texturesgen = tmp_tex; //Store the generated names in texturesgen 
for(int i=0; i < textures.length; i++)
{ 
    //Bitmap bmp = BitmapFactory.decodeResource(context.getResources(), textures[i]);
    InputStream is = context.getResources().openRawResource(textures[i]);
    Bitmap bitmap = null;
    try {
        //BitmapFactory is an Android graphics utility for images
        bitmap = BitmapFactory.decodeStream(is);

    } finally {
        //Always clear and close
        try {
            is.close();
            is = null;
        } catch (IOException e) {
        }
    } 
    // Get a new texture name 
    // Load it up 
    this.textureMap.put(new Integer(textures[i]),new Integer(i)); 
    int tex = tmp_tex[i]; 
    gl.glBindTexture(GL10.GL_TEXTURE_2D, tex); 
    //Create Nearest Filtered Texture
    gl.glTexParameterf(GL10.GL_TEXTURE_2D, GL10.GL_TEXTURE_MIN_FILTER, GL10.GL_NEAREST);
    gl.glTexParameterf(GL10.GL_TEXTURE_2D, GL10.GL_TEXTURE_MAG_FILTER, GL10.GL_LINEAR);

    //Different possible texture parameters, e.g. GL10.GL_CLAMP_TO_EDGE
    gl.glTexParameterf(GL10.GL_TEXTURE_2D, GL10.GL_TEXTURE_WRAP_S, GL10.GL_REPEAT);
    gl.glTexParameterf(GL10.GL_TEXTURE_2D, GL10.GL_TEXTURE_WRAP_T, GL10.GL_REPEAT);

    //Use the Android GLUtils to specify a two-dimensional texture image from our bitmap
    GLUtils.texImage2D(GL10.GL_TEXTURE_2D, 0, bitmap, 0);

    bitmap.recycle();
} 

Я совершенно уверен, что есть лучший способ справиться с этим ... Я просто не мог понять это. Если у кого-то есть идея, я весь в ушах.

2) Рисование плитки

Я создал один квадрат и одну карту текстуры:

/** The initial vertex definition */
    private float vertices[] = { 
                                -1.0f, -1.0f, 0.0f,     //Bottom Left
                                1.0f, -1.0f, 0.0f,      //Bottom Right
                                -1.0f, 1.0f, 0.0f,      //Top Left
                                1.0f, 1.0f, 0.0f        //Top Right
                                                };

    private float texture[] = {         
            //Mapping coordinates for the vertices
              0.0f, 1.0f,
              1.0f, 1.0f,
              0.0f, 0.0f,
              1.0f, 0.0f

                                };

Затем в моей функции рисования я перебираю карту, чтобы определить используемую текстуру (после указания и включения буферов):

for(int y = 0; y < Y; y++){
    for(int x = 0; x < X; x++){
        tile = map[y][x];
        try 
            { 
                //Get the texture from the HashMap
                int textureid = ((Integer) this.textureMap.get(new Integer(textures[tile]))).intValue(); 
                gl.glBindTexture(GL10.GL_TEXTURE_2D, this.texturesgen[textureid]); 
            }    
            catch(Exception e) 
            { 
                return; 
            }

        //Draw the vertices as triangle strip
        gl.glDrawArrays(GL10.GL_TRIANGLE_STRIP, 0, vertices.length / 3);            
        gl.glTranslatef(2.0f, 0.0f, 0.0f); //A square takes 2x so I move +2x before drawing the next tile
    }
    gl.glTranslatef(-(float)(2*X), -2.0f, 0.0f); //Go back to the begining of the map X-wise and move 2y down before drawing the next line
}

Это прекрасно работает, я действительно думаю, что на карте 1000 * 1000 или более, она будет отстать как ад (как напоминание, это типичная карта мира Zelda: http://vgmaps.com/Atlas/SuperNES/LegendOfZelda-ALinkToThePast-LightWorld.png).

Я читал кое-что о Vertex Buffer Object и DisplayList, но я не смог найти хороший учебник, и nodoby, кажется, будет в порядке с тем, кто является лучшим / имеет лучшую поддержку (T1 и Nexus One уже давно).

Я думаю, это все, я поместил много кода, но я думаю, что это помогает.

Заранее спасибо!

1 Ответ

1 голос
/ 15 марта 2010

Пара вещей:

  1. Нет необходимости использовать хэш-карту, просто используйте вектор / список.
  2. Может быть быстрее / проще иметь одну большую текстуру, которая содержит все ваши плитки. Используйте соответствующие текстурные координаты, чтобы выбрать подходящую плитку. Возможно, вам придется быть немного осторожнее с фильтрацией текстур здесь. Похоже, вы играете в 2D-игру, и в этом случае вы, вероятно, захотите использовать фильтрацию ближайших соседей для плиток и закрепить камеру на целочисленных пикселях.
  3. Не проще ли использовать GL_QUADS, а не GL_TRIANGLE_STRIP. Не уверен насчет вашего кода - вы, похоже, не используете массив 'texture'.
  4. Размер карты не должен иметь никакого значения, если вы не рисуете плитки, которых нет на экране. Ваш код должен выглядеть примерно так:

.

int minX = screenLeft / tileSize;
int minY = screenBottom / tileSize;
int maxX = screenRight / tileSize;
int maxY = screenTop / tilesSize;
for (int x = minX; x <= maxX; ++x)
{
   for (int y = minY; y < maxY; ++y)
   {
      ...
...