平铺系统C++SDL

Tiling System C++ SDL

本文关键字:C++SDL 系统      更新时间:2023-10-16

我一直在为一款通过C++和SDL设计的视频游戏开发平铺系统。该代码大致基于lazyfo.net上的教程,但使用了不同的系统来将平铺绘制到屏幕上。代码的工作原理如下:我有一个名为Map的类,它创建了另一个称为Tile的类的数组。它还有一个名为set_tiles的函数,它从.map文件中读取数字,并使用两个for循环将它们复制到tile的数组中;一个用于x,一个用于y。然后,我根据Tile的特定变量将每个曲面闪电式地映射到一个临时映射中。这是实际代码:

地图.cpp

SDL_Surface* Map::set_tiles ( SDL_Surface* tile_image ) {
Setup setup;
SDL_Surface* temp_map = NULL;
//Open the map
std::ifstream map ( "Test.map" );
//Catch any errors
if ( map.fail() ) return NULL;
//Initialize the tiles
for ( int y = 0; y < MAP_HEIGHT / TILE_HEIGHT; y++ ) {
    for ( int x = 0; x < MAP_WIDTH / TILE_WIDTH; x++ ) {
        //Determines the tile type
        int tile_type = -1;
        //Read the tile type from the map
        map >> tile_type;
        //Make sure it's a real tile
        if ( tile_type < 0 || tile_type >= TILE_SPRITES ) {
            map.close();
            return NULL;
        }
        //Error check for the .map file
        if ( map.fail() ) {
            map.close();
            return NULL;
        }
        //Add the tile to the array
        tile_array[x][y] = &Tile ( x, y, tile_type );
        //Create the temp. image crop
        SDL_Rect* temp_crop = &tile_array[x][y]->get_crop();
        //Check for crop errors
        if ( ( temp_crop->h || temp_crop->w ) == 0 ) return NULL;
        //Edit the temp. map
        setup.apply_surface ( x * TILE_WIDTH, y * TILE_HEIGHT, tile_image, temp_map, temp_crop );
    }
}
map.close();
//Return the modified map
return temp_map;

}

Tile.cpp

Tile::Tile ( int x, int y, int tile_type ) {
//Get the offsets
box.x = x;
box.y = y;
//Set the collision box
box.w = TILE_WIDTH;
box.h = TILE_HEIGHT;
//Get the tile type
type = tile_type;
//Choose the crop option
crop.w = TILE_WIDTH;
crop.h = TILE_HEIGHT;
switch ( tile_type ) {
case TILE_RED:
    crop.x = 0 * TILE_WIDTH;
    crop.y = 0 * TILE_HEIGHT;
    break;
case TILE_GREEN:
    crop.x = 1 * TILE_WIDTH;
    crop.y = 0 * TILE_HEIGHT;
    break;
case TILE_BLUE:
    crop.x = 2 * TILE_WIDTH;
    crop.y = 0 * TILE_HEIGHT;
    break;
case TILE_CENTER:
    crop.x = 3 * TILE_WIDTH;
    crop.y = 0 * TILE_HEIGHT;
    break;
case TILE_TOP:
    crop.x = 0 * TILE_WIDTH;
    crop.y = 1 * TILE_HEIGHT;
    break;
case TILE_TOPRIGHT:
    crop.x = 1 * TILE_WIDTH;
    crop.y = 1 * TILE_HEIGHT;
    break;
case TILE_RIGHT:
    crop.x = 2 * TILE_WIDTH;
    crop.y = 1 * TILE_HEIGHT;
    break;
case TILE_BOTTOMRIGHT:
    crop.x = 3 * TILE_WIDTH;
    crop.y = 1 * TILE_HEIGHT;
    break;
case TILE_BOTTOM:
    crop.x = 0 * TILE_WIDTH;
    crop.y = 2 * TILE_HEIGHT;
    break;
case TILE_BOTTOMLEFT:
    crop.x = 1 * TILE_WIDTH;
    crop.y = 2 * TILE_HEIGHT;
    break;
case TILE_LEFT:
    crop.x = 2 * TILE_WIDTH;
    crop.y = 2 * TILE_HEIGHT;
    break;
case TILE_TOPLEFT:
    crop.x = 3 * TILE_WIDTH;
    crop.y = 2 * TILE_HEIGHT;
    break;
}

}

问题是set_tiles在被调用时返回一个NULL指针。通过一点调试,我发现它一直到最后,然后返回一个NULL,而不是在for循环中的某个地方。我还尝试过将所有tile crop.x和y设置为0,只是想看看它是否有效,但没有。

注意问题不在于我的图像或.map文件。我认为我对指针的理解可能有点错误,在调用apply_surface时可能犯了一个错误。

设置.cpp

void Setup::apply_surface ( int x, int y, SDL_Surface* source, SDL_Surface* &destination, SDL_Rect* clip ) {
//Temporary rectangle to hold the offsets
SDL_Rect offset;
//Get the offsets
offset.x = x;
offset.y = y;
//Blit the surface
SDL_BlitSurface ( source, clip, destination, &offset );

}

SDL_BlitSurface将一个曲面闪电传送到另一个曲面。它们都必须初始化,否则函数将不执行任何操作。您正在向它传递一个NULL指针。您需要通过调用类似SDL_CreateRGBSurface的东西来初始化指针。