2016-06-30 107 views
0

當我嘗試繪製我的等軸測圖時,所有的精靈都會顛倒過來。2D等距(菱形)遊戲引擎 - 倒轉的精靈

這是在下面的代碼mentionned的tileset.png: enter image description here

Object.h/Object.cpp

我可以用它們來畫磚,UI元素,等等...

#pragma once 

class Object { 
public: 
    //FUNCTIONS 
    Object(); 
    void addComponent(float value); 
    int getComponent(float index); 
    void editComponent(float index, float value); 
    void deleteComponent(float index); 

private: 
    vector<int> components; 
}; 

#include "Object.cpp" 

-

#pragma once 

//FUNCTIONS 
Object::Object() { 
    //... 
} 

void Object::addComponent(float value) { 
    components.push_back(value); 
} 

int Object::getComponent(float index) { 
    return components[index]; 
} 

void Object::editComponent(float index, float value) { 
    components[index] = value; 
} 

void Object::deleteComponent(float index) { 
    components.erase(components.begin() + index); 
} 

說明:我可能有些奇怪的包括,我正在用視覺工作室掙扎哈哈。

Scene.h/Scene.cpp

句柄數據&圖形

#pragma once 

class Scene { 
public: 
    Scene(float w, float h, int mapx, int mapy, int tilesize, int mapwidth, int mapheight); 
    void run(); 

    void addLayer(); 
    void loadTileset(sf::String url); 
    void loadUiTileset(sf::String url); 

    //functions 
    //... 

    //getters 
    //... 

    //setters 
    //... 

private: 
    sf::RenderWindow window; 

    float width; 
    float height; 

    int nb_layers; 

    int map_x; 
    int map_y; 
    int map_width; 
    int map_height; 
    int tile_size; 
    int selected_tile_index; 

    sf::RenderTexture texture; 
    sf::Sprite tile; 
    sf::Sprite map; 
    sf::Texture tileset; 
    vector<Object> tiles; 

    sf::Texture uiTileset; 

    //private functions 
    void updateMap(); 

    //... 

    void processEvent(); 
    void update(sf::Time deltaTime); 
    void render(); 

    //... 
}; 

#include "Scene.cpp" 

-

#pragma once 

//functions 
Scene::Scene(float w, float h, int mapx, int mapy, int tilesize, int mapwidth, int mapheight) : window(sf::VideoMode(w, h), "Editor") { 
    width = w; 
    height = h; 

    map_x = mapx; 
    map_y = mapy; 
    map_width = mapwidth; 
    map_height = mapheight; 

    tile_size = tilesize; 

    selected_tile_index = 0;//default 

    nb_layers = 0; 
} 

void Scene::run() { 
    sf::Clock clock; 
    sf::Time timeSinceLastUpdate = sf::Time::Zero; 
    sf::Time TimePerFrame = sf::seconds(1.f/60.f); 

    while (window.isOpen()) { 
     processEvent(); 

     timeSinceLastUpdate += clock.restart(); 

     while (timeSinceLastUpdate > TimePerFrame) { 
      timeSinceLastUpdate -= TimePerFrame; 

      processEvent(); 
      update(TimePerFrame); 
     } 

     render(); 
    } 
} 

void Scene::addLayer() { 
    nb_layers += 1; 

    int tile_x = map_x, 
     tile_y = map_y, 
     num_layer = nb_layers - 1, 
     layer_pos = (num_layer * tile_size)/2, 
     tile_zOrder = -1; 

    tile_y -= layer_pos; 

    int x = map_x, 
     y = map_y; 

    for (int h = 0; h < map_height; h++) { 
     for (int w = 0; w < map_width; w++) { 
      tile_zOrder = (w * (h + 1)) + (num_layer * 10); 

      x = carthesianToIsometric(tile_x, tile_y)[0]; 
      y = carthesianToIsometric(tile_x, tile_y)[1] - layer_pos; 

      cout << x << ", " << y << endl; 

      Object tile; 
      tile.addComponent(selected_tile_index); 
      tile.addComponent(x); 
      tile.addComponent(y); 
      tile.addComponent(tile_zOrder); 
      tile.addComponent(num_layer); 

      tiles.push_back(tile); 
      tile_x += tile_size; 
     } 
     tile_x = 0; 
     tile_y += tile_size; 
    } 

    updateMap(); 
} 

void Scene::loadTileset(sf::String url) { 
    if (!tileset.loadFromFile(url)) 
    { 
     cout << std::string(url) << "couldn't be loaded..." << endl; 
    } 
} 

void Scene::loadUiTileset(sf::String url) { 
    if (!uiTileset.loadFromFile(url)) 
    { 
     cout << std::string(url) << "couldn't be loaded..." << endl; 
    } 
} 

//getters 
//... 

//setters 
//... 

//private functions 
void Scene::updateMap() { 
    int tile_position_x = 0, 
     tile_position_y = 0; 

    int tile_x = 0, 
     tile_y = 0; 

    if (!texture.create(map_width * tile_size, (map_height * tile_size)/2)) 
     cout << "Texture couldn't be loaded... " << endl; 

    texture.clear(sf::Color(133, 118, 104, 255)); 

    sf::Sprite image; 
    image.setTexture(tileset); 
    int tileset_width = image.getGlobalBounds().width, 
     tileset_height = image.getGlobalBounds().height; 

    tile.setTexture(tileset); 

    for (int tile_index = 0; tile_index < tiles.size(); tile_index++) { 
     tile_position_x = getTilePosition(tileset_width, tileset_height, tiles[tile_index].getComponent(0), tile_size)[0]; 
     tile_position_y = getTilePosition(tileset_width, tileset_height, tiles[tile_index].getComponent(0), tile_size)[1]; 

     tile.setTextureRect(sf::IntRect(tile_position_x, tile_position_y, tile_size, tile_size)); 

     tile_x = tiles[tile_index].getComponent(1); 
     tile_y = tiles[tile_index].getComponent(2); 

     tile.setPosition(sf::Vector2f(tile_x, tile_y)); 

     texture.draw(tile); 
    } 

    map.setTexture(texture.getTexture()); 
} 

void Scene::processEvent() { 
    sf::Event event; 

    while (window.pollEvent(event)) { 
     switch (event.type) { 
     case sf::Event::Closed: 
      window.close(); 
      break; 

     case sf::Event::KeyPressed: 
      if (event.key.code == sf::Keyboard::Escape) 
       window.close(); 
      break; 
     } 
    } 
} 

void Scene::update(sf::Time deltaTime) { 
    //REMEMBER: distance = speed * time 
    //MOVEMENT, ANIMATIONS ETC. .. 
} 

void Scene::render() { 
    window.clear(); 

    window.draw(map); 

    window.display(); 
} 

的main.cpp

#pragma once 
//global functions + main headers + class headers => 
#include "globalfunctions.h" 

int main() { 
    int map_width = 15, 
     map_height = 15, 
     tile_size = 64; 

    float scene_width = map_width * tile_size, 
     scene_height = (map_height * tile_size)/2; 

    Scene engine(scene_width, scene_height, 0, 0, tile_size, map_width, map_height); 
    engine.loadTileset("tileset.png"); 
    //engine.loadUiTileset("menu.png"); 

    engine.addLayer(); 
    //... 

    engine.run(); 
    return EXIT_SUCCESS; 
} 

globalfunctions.h

一些實用功能。 getTilePosition(...)允許我在具有給定圖塊索引的紋理上獲取x,y。例如:如果我想繪製tileset紋理的圖塊n°0。

#pragma once 
#include <SFML/System.hpp> 
#include <SFML/Graphics.hpp> 
#include <SFML/Window.hpp> 
#include <iostream> 
#include <math.h> 
#include <vector> 

using namespace std; 

vector<float> getTilePosition(float tileset_width, float tileset_height, float tile_index, float tile_size) {//In a tileset 
    float tileX = 0, 
     tileY = 0, 
     tilePerLine = 0; 

    tilePerLine = tileset_width/tile_size; 

    tileY = floor(tile_index/tilePerLine); 
    tileX = ((tile_index + 1) - (tileY * tilePerLine)) - 1; 

    tileX *= tile_size; 
    tileY *= tile_size; 

    vector<float> coords; 
     coords.push_back(tileX); 
     coords.push_back(tileY); 

    return coords; 
} 

vector<int> carthesianToIsometric(int x, int y) { 
    vector<int> coords; 

    float isoX = (x - y)/2, 
     isoY = (x + y)/4; 

    coords.push_back(isoX); 
    coords.push_back(isoY); 

    return coords; 
} 

#include "Object.h" 
#include "Scene.h" 

//... 

這裏,跆拳道結果我得到:

enter image description here

感謝您閱讀所有的怪異代碼!

編輯:

當我改變

tile.setPosition(sf::Vector2f(tile_x, tile_y)); 

tile.setPosition(sf::Vector2f(0, 0)); 
updateMap()

從scene.cpp:

enter image description here

不幸的是,我無法解釋爲什麼。也許它會幫助你理解這個問題。

+1

難道是y軸與你所假設的方向相反嗎? – molbdnilo

+0

@molbdnilo感謝您的回答!爲什麼x軸會顛倒瓷磚? – Madz

+0

@Madz然後我不能解決你的問題,但我想我可以給你一些小竅門來自己工作。 1)檢查你的圖像加載器,如果可能的話將* *移動到外部項目中,並加載一個圖像進行測試2)檢查你的圖像格式,它可能期望圖像被翻轉3)如果其他所有的失敗,你可以顛倒保存你的圖像。 –

回答