2013-12-01 01:04:46 +00:00
|
|
|
#pragma once
|
2017-10-31 17:12:50 +00:00
|
|
|
#ifndef ES_CORE_COMPONENTS_IMAGE_GRID_COMPONENT_H
|
|
|
|
#define ES_CORE_COMPONENTS_IMAGE_GRID_COMPONENT_H
|
2013-12-01 01:04:46 +00:00
|
|
|
|
2014-06-20 01:30:09 +00:00
|
|
|
#include "components/IList.h"
|
2017-11-01 22:21:10 +00:00
|
|
|
#include "resources/TextureResource.h"
|
2013-12-01 01:04:46 +00:00
|
|
|
|
2014-02-08 03:45:28 +00:00
|
|
|
struct ImageGridData
|
|
|
|
{
|
|
|
|
std::shared_ptr<TextureResource> texture;
|
|
|
|
};
|
|
|
|
|
2013-12-01 01:04:46 +00:00
|
|
|
template<typename T>
|
2014-02-13 23:10:28 +00:00
|
|
|
class ImageGridComponent : public IList<ImageGridData, T>
|
2013-12-01 01:04:46 +00:00
|
|
|
{
|
2014-02-17 17:40:31 +00:00
|
|
|
protected:
|
|
|
|
using IList<ImageGridData, T>::mEntries;
|
|
|
|
using IList<ImageGridData, T>::listUpdate;
|
|
|
|
using IList<ImageGridData, T>::listInput;
|
|
|
|
using IList<ImageGridData, T>::listRenderTitleOverlay;
|
|
|
|
using IList<ImageGridData, T>::getTransform;
|
|
|
|
using IList<ImageGridData, T>::mSize;
|
|
|
|
using IList<ImageGridData, T>::mCursor;
|
|
|
|
using IList<ImageGridData, T>::Entry;
|
|
|
|
using IList<ImageGridData, T>::mWindow;
|
|
|
|
|
2013-12-01 01:04:46 +00:00
|
|
|
public:
|
2014-02-17 17:40:31 +00:00
|
|
|
using IList<ImageGridData, T>::size;
|
|
|
|
using IList<ImageGridData, T>::isScrolling;
|
|
|
|
using IList<ImageGridData, T>::stopScrolling;
|
|
|
|
|
2013-12-01 01:04:46 +00:00
|
|
|
ImageGridComponent(Window* window);
|
|
|
|
|
2014-02-08 03:45:28 +00:00
|
|
|
void add(const std::string& name, const std::string& imagePath, const T& obj);
|
|
|
|
|
2013-12-01 01:04:46 +00:00
|
|
|
void onSizeChanged() override;
|
|
|
|
|
|
|
|
bool input(InputConfig* config, Input input) override;
|
|
|
|
void update(int deltaTime) override;
|
2017-10-28 20:24:35 +00:00
|
|
|
void render(const Transform4x4f& parentTrans) override;
|
2013-12-01 01:04:46 +00:00
|
|
|
|
2018-03-22 12:52:13 +00:00
|
|
|
inline void setCursorChangedCallback(const std::function<void(CursorState state)>& func) { mCursorChangedCallback = func; }
|
|
|
|
|
2013-12-01 01:04:46 +00:00
|
|
|
private:
|
2018-03-22 07:03:12 +00:00
|
|
|
// Calculate how much tiles of size mTileMaxSize we can fit in a grid of size mSize using a margin of size mMargin
|
2018-03-23 09:25:33 +00:00
|
|
|
Vector2i getGridDimension() const
|
2013-12-01 01:04:46 +00:00
|
|
|
{
|
2018-03-22 07:03:12 +00:00
|
|
|
// GRID_SIZE = COLUMNS * TILE_SIZE + (COLUMNS - 1) * MARGIN
|
|
|
|
// <=> COLUMNS = (GRID_SIZE + MARGIN) / (TILE_SIZE + MARGIN)
|
|
|
|
return Vector2i((int) ((mSize.x() + mMargin.x()) / (mTileMaxSize.x() + mMargin.x())),
|
|
|
|
(int) ((mSize.y() + mMargin.y()) / (mTileMaxSize.y() + mMargin.y())));
|
2013-12-01 01:04:46 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
void buildImages();
|
|
|
|
void updateImages();
|
|
|
|
|
2014-02-08 03:45:28 +00:00
|
|
|
virtual void onCursorChanged(const CursorState& state);
|
2013-12-01 01:04:46 +00:00
|
|
|
|
2018-03-22 12:52:13 +00:00
|
|
|
std::function<void(CursorState state)> mCursorChangedCallback;
|
|
|
|
|
2013-12-01 01:04:46 +00:00
|
|
|
bool mEntriesDirty;
|
|
|
|
|
2018-03-22 07:03:12 +00:00
|
|
|
Vector2f mMargin;
|
|
|
|
Vector2f mTileMaxSize;
|
|
|
|
Vector2f mSelectedTileMaxSize;
|
|
|
|
|
2013-12-01 01:04:46 +00:00
|
|
|
std::vector<ImageComponent> mImages;
|
|
|
|
};
|
|
|
|
|
|
|
|
template<typename T>
|
2014-02-17 17:40:31 +00:00
|
|
|
ImageGridComponent<T>::ImageGridComponent(Window* window) : IList<ImageGridData, T>(window)
|
2013-12-01 01:04:46 +00:00
|
|
|
{
|
2018-03-22 07:03:12 +00:00
|
|
|
Vector2f screen = Vector2f((float)Renderer::getScreenWidth(), (float)Renderer::getScreenHeight());
|
|
|
|
|
2013-12-01 01:04:46 +00:00
|
|
|
mEntriesDirty = true;
|
2018-03-22 07:03:12 +00:00
|
|
|
|
|
|
|
mSize = screen * 0.8f;
|
|
|
|
mMargin = screen * 0.01f;
|
|
|
|
mTileMaxSize = screen * 0.19f;
|
|
|
|
mSelectedTileMaxSize = mTileMaxSize + mMargin * 3.0f;
|
2013-12-01 01:04:46 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
template<typename T>
|
2014-02-08 03:45:28 +00:00
|
|
|
void ImageGridComponent<T>::add(const std::string& name, const std::string& imagePath, const T& obj)
|
2013-12-01 01:04:46 +00:00
|
|
|
{
|
2014-02-17 17:40:31 +00:00
|
|
|
typename IList<ImageGridData, T>::Entry entry;
|
2014-02-08 03:45:28 +00:00
|
|
|
entry.name = name;
|
|
|
|
entry.object = obj;
|
|
|
|
entry.data.texture = ResourceManager::getInstance()->fileExists(imagePath) ? TextureResource::get(imagePath) : TextureResource::get(":/button.png");
|
|
|
|
static_cast<IList< ImageGridData, T >*>(this)->add(entry);
|
2013-12-01 01:04:46 +00:00
|
|
|
mEntriesDirty = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
template<typename T>
|
|
|
|
bool ImageGridComponent<T>::input(InputConfig* config, Input input)
|
|
|
|
{
|
|
|
|
if(input.value != 0)
|
|
|
|
{
|
2017-10-28 20:24:35 +00:00
|
|
|
Vector2i dir = Vector2i::Zero();
|
2013-12-01 01:04:46 +00:00
|
|
|
if(config->isMappedTo("up", input))
|
|
|
|
dir[1] = -1;
|
|
|
|
else if(config->isMappedTo("down", input))
|
|
|
|
dir[1] = 1;
|
|
|
|
else if(config->isMappedTo("left", input))
|
|
|
|
dir[0] = -1;
|
|
|
|
else if(config->isMappedTo("right", input))
|
|
|
|
dir[0] = 1;
|
|
|
|
|
2017-10-28 20:24:35 +00:00
|
|
|
if(dir != Vector2i::Zero())
|
2013-12-01 01:04:46 +00:00
|
|
|
{
|
2018-03-23 09:25:33 +00:00
|
|
|
listInput(dir.x() + dir.y() * getGridDimension().x());
|
2013-12-01 01:04:46 +00:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}else{
|
|
|
|
if(config->isMappedTo("up", input) || config->isMappedTo("down", input) || config->isMappedTo("left", input) || config->isMappedTo("right", input))
|
|
|
|
{
|
2014-02-08 02:15:48 +00:00
|
|
|
stopScrolling();
|
2013-12-01 01:04:46 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return GuiComponent::input(config, input);
|
|
|
|
}
|
|
|
|
|
|
|
|
template<typename T>
|
|
|
|
void ImageGridComponent<T>::update(int deltaTime)
|
|
|
|
{
|
2014-02-08 02:15:48 +00:00
|
|
|
listUpdate(deltaTime);
|
2013-12-01 01:04:46 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
template<typename T>
|
2017-10-28 20:24:35 +00:00
|
|
|
void ImageGridComponent<T>::render(const Transform4x4f& parentTrans)
|
2013-12-01 01:04:46 +00:00
|
|
|
{
|
2017-10-28 20:24:35 +00:00
|
|
|
Transform4x4f trans = getTransform() * parentTrans;
|
2013-12-01 01:04:46 +00:00
|
|
|
|
|
|
|
if(mEntriesDirty)
|
|
|
|
{
|
|
|
|
buildImages();
|
|
|
|
updateImages();
|
|
|
|
mEntriesDirty = false;
|
|
|
|
}
|
|
|
|
|
2018-03-22 07:03:12 +00:00
|
|
|
// Dirty solution (took from updateImages function) to keep the selected image and render it later (on top of the others)
|
|
|
|
// Will be changed for a cleaner way with the introduction of GridTileComponent
|
|
|
|
Vector2i gridDimension = getGridDimension();
|
|
|
|
|
|
|
|
int cursorRow = mCursor / gridDimension.x();
|
|
|
|
|
|
|
|
int start = (cursorRow - (gridDimension.y() / 2)) * gridDimension.x();
|
|
|
|
|
|
|
|
//if we're at the end put the row as close as we can and no higher
|
|
|
|
if(start + (gridDimension.x() * gridDimension.y()) >= (int)mEntries.size())
|
|
|
|
start = gridDimension.x() * ((int)mEntries.size()/gridDimension.x() - gridDimension.y() + 1);
|
|
|
|
|
|
|
|
if(start < 0)
|
|
|
|
start = 0;
|
|
|
|
|
|
|
|
unsigned int i = (unsigned int)start;
|
|
|
|
ImageComponent* selectedImage = NULL;
|
2013-12-01 01:04:46 +00:00
|
|
|
for(auto it = mImages.begin(); it != mImages.end(); it++)
|
|
|
|
{
|
2018-03-22 07:03:12 +00:00
|
|
|
// If it's the selected image, keep it for later, otherwise render it now
|
|
|
|
if(i == (unsigned int)mCursor)
|
|
|
|
selectedImage = it.base();
|
|
|
|
else
|
|
|
|
it->render(trans);
|
|
|
|
i++;
|
2013-12-01 01:04:46 +00:00
|
|
|
}
|
|
|
|
|
2018-03-22 07:03:12 +00:00
|
|
|
// Render the selected image on top of the others
|
|
|
|
if (selectedImage != NULL)
|
|
|
|
selectedImage->render(trans);
|
|
|
|
|
2014-02-17 17:40:31 +00:00
|
|
|
GuiComponent::renderChildren(trans);
|
2013-12-01 01:04:46 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
template<typename T>
|
2018-03-22 12:52:13 +00:00
|
|
|
void ImageGridComponent<T>::onCursorChanged(const CursorState& state)
|
2013-12-01 01:04:46 +00:00
|
|
|
{
|
|
|
|
updateImages();
|
2018-03-22 12:52:13 +00:00
|
|
|
|
|
|
|
if(mCursorChangedCallback)
|
|
|
|
mCursorChangedCallback(state);
|
2013-12-01 01:04:46 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
template<typename T>
|
|
|
|
void ImageGridComponent<T>::onSizeChanged()
|
|
|
|
{
|
|
|
|
buildImages();
|
|
|
|
updateImages();
|
|
|
|
}
|
|
|
|
|
2018-03-22 07:03:12 +00:00
|
|
|
// Create and position imagecomponents (mImages)
|
2013-12-01 01:04:46 +00:00
|
|
|
template<typename T>
|
|
|
|
void ImageGridComponent<T>::buildImages()
|
|
|
|
{
|
|
|
|
mImages.clear();
|
|
|
|
|
2018-03-23 09:25:33 +00:00
|
|
|
Vector2i gridDimension = getGridDimension();
|
2018-03-22 07:03:12 +00:00
|
|
|
Vector2f startPosition = mTileMaxSize / 2;
|
|
|
|
Vector2f tileDistance = mTileMaxSize + mMargin;
|
2013-12-01 01:04:46 +00:00
|
|
|
|
2018-03-22 07:03:12 +00:00
|
|
|
// Layout tile size and position
|
2018-03-23 09:25:33 +00:00
|
|
|
for(int y = 0; y < gridDimension.y(); y++)
|
2013-12-01 01:04:46 +00:00
|
|
|
{
|
2018-03-23 09:25:33 +00:00
|
|
|
for(int x = 0; x < gridDimension.x(); x++)
|
2013-12-01 01:04:46 +00:00
|
|
|
{
|
2018-03-22 07:03:12 +00:00
|
|
|
// Create tiles
|
2013-12-01 01:04:46 +00:00
|
|
|
mImages.push_back(ImageComponent(mWindow));
|
2018-03-23 09:25:33 +00:00
|
|
|
ImageComponent& image = mImages.at(y * gridDimension.x() + x);
|
2013-12-01 01:04:46 +00:00
|
|
|
|
2018-03-22 07:03:12 +00:00
|
|
|
image.setPosition(x * tileDistance.x() + startPosition.x(), y * tileDistance.y() + startPosition.y());
|
2013-12-01 01:04:46 +00:00
|
|
|
image.setOrigin(0.5f, 0.5f);
|
2018-03-22 07:03:12 +00:00
|
|
|
image.setMaxSize(mTileMaxSize);
|
2013-12-01 01:04:46 +00:00
|
|
|
image.setImage("");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
template<typename T>
|
|
|
|
void ImageGridComponent<T>::updateImages()
|
|
|
|
{
|
|
|
|
if(mImages.empty())
|
|
|
|
buildImages();
|
|
|
|
|
2018-03-23 09:25:33 +00:00
|
|
|
Vector2i gridDimension = getGridDimension();
|
2013-12-01 01:04:46 +00:00
|
|
|
|
2018-03-23 09:25:33 +00:00
|
|
|
int cursorRow = mCursor / gridDimension.x();
|
2013-12-01 01:04:46 +00:00
|
|
|
|
2018-03-23 09:25:33 +00:00
|
|
|
int start = (cursorRow - (gridDimension.y() / 2)) * gridDimension.x();
|
2013-12-01 01:04:46 +00:00
|
|
|
|
2018-04-02 13:28:33 +00:00
|
|
|
// If we are at the end put the row as close as we can and no higher, using the following formula
|
|
|
|
// Where E is the nb of entries, X the grid x dim (nb of column), Y the grid y dim (nb of line)
|
|
|
|
// start = first tile of last row - nb column * (nb line - 1)
|
|
|
|
// = (E - 1) / X * X - X * (Y - 1)
|
|
|
|
// = X * ((E - 1) / X - Y + 1)
|
2018-03-23 09:25:33 +00:00
|
|
|
if(start + (gridDimension.x() * gridDimension.y()) >= (int)mEntries.size())
|
2018-04-02 13:28:33 +00:00
|
|
|
start = gridDimension.x() * (((int)mEntries.size() - 1) / gridDimension.x() - gridDimension.y() + 1);
|
2013-12-01 01:04:46 +00:00
|
|
|
|
|
|
|
if(start < 0)
|
|
|
|
start = 0;
|
|
|
|
|
|
|
|
unsigned int i = (unsigned int)start;
|
|
|
|
for(unsigned int img = 0; img < mImages.size(); img++)
|
|
|
|
{
|
|
|
|
ImageComponent& image = mImages.at(img);
|
2014-02-08 03:45:28 +00:00
|
|
|
if(i >= (unsigned int)size())
|
2013-12-01 01:04:46 +00:00
|
|
|
{
|
|
|
|
image.setImage("");
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2017-10-28 20:07:31 +00:00
|
|
|
if(i == (unsigned int)mCursor)
|
2013-12-01 01:04:46 +00:00
|
|
|
{
|
|
|
|
image.setColorShift(0xFFFFFFFF);
|
2018-03-22 07:03:12 +00:00
|
|
|
image.setMaxSize(mSelectedTileMaxSize);
|
2013-12-01 01:04:46 +00:00
|
|
|
}else{
|
|
|
|
image.setColorShift(0xAAAAAABB);
|
2018-03-22 07:03:12 +00:00
|
|
|
image.setMaxSize(mTileMaxSize);
|
2013-12-01 01:04:46 +00:00
|
|
|
}
|
|
|
|
|
2014-02-08 03:45:28 +00:00
|
|
|
image.setImage(mEntries.at(i).data.texture);
|
2013-12-01 01:04:46 +00:00
|
|
|
i++;
|
|
|
|
}
|
|
|
|
}
|
2017-10-31 17:12:50 +00:00
|
|
|
|
|
|
|
#endif // ES_CORE_COMPONENTS_IMAGE_GRID_COMPONENT_H
|