mirror of
https://github.com/SFML/SFML.git
synced 2024-12-01 15:51:04 +08:00
460 lines
15 KiB
C++
460 lines
15 KiB
C++
#include <SFML/Graphics/Image.hpp>
|
|
|
|
// Other 1st party headers
|
|
#include <SFML/System/FileInputStream.hpp>
|
|
|
|
#include <catch2/catch_test_macros.hpp>
|
|
|
|
#include <GraphicsUtil.hpp>
|
|
#include <array>
|
|
#include <type_traits>
|
|
|
|
TEST_CASE("[Graphics] sf::Image")
|
|
{
|
|
SECTION("Type traits")
|
|
{
|
|
STATIC_CHECK(!std::is_default_constructible_v<sf::Image>);
|
|
STATIC_CHECK(std::is_copy_constructible_v<sf::Image>);
|
|
STATIC_CHECK(std::is_copy_assignable_v<sf::Image>);
|
|
STATIC_CHECK(std::is_nothrow_move_constructible_v<sf::Image>);
|
|
STATIC_CHECK(std::is_nothrow_move_assignable_v<sf::Image>);
|
|
}
|
|
|
|
SECTION("Construction")
|
|
{
|
|
SECTION("Vector2 constructor")
|
|
{
|
|
const sf::Image image(sf::Vector2u(10, 10));
|
|
CHECK(image.getSize() == sf::Vector2u(10, 10));
|
|
CHECK(image.getPixelsPtr() != nullptr);
|
|
|
|
for (std::uint32_t i = 0; i < 10; ++i)
|
|
{
|
|
for (std::uint32_t j = 0; j < 10; ++j)
|
|
{
|
|
CHECK(image.getPixel(sf::Vector2u(i, j)) == sf::Color::Black);
|
|
}
|
|
}
|
|
}
|
|
|
|
SECTION("Vector2 and color constructor")
|
|
{
|
|
const sf::Image image(sf::Vector2u(10, 10), sf::Color::Red);
|
|
CHECK(image.getSize() == sf::Vector2u(10, 10));
|
|
CHECK(image.getPixelsPtr() != nullptr);
|
|
|
|
for (std::uint32_t i = 0; i < 10; ++i)
|
|
{
|
|
for (std::uint32_t j = 0; j < 10; ++j)
|
|
{
|
|
CHECK(image.getPixel(sf::Vector2u(i, j)) == sf::Color::Red);
|
|
}
|
|
}
|
|
}
|
|
|
|
SECTION("Vector2 and std::uint8_t* constructor")
|
|
{
|
|
// 10 x 10, with 4 colour channels array
|
|
std::array<std::uint8_t, 400> pixels{};
|
|
for (std::size_t i = 0; i < pixels.size(); i += 4)
|
|
{
|
|
pixels[i] = 255; // r
|
|
pixels[i + 1] = 0; // g
|
|
pixels[i + 2] = 0; // b
|
|
pixels[i + 3] = 255; // a
|
|
}
|
|
|
|
const sf::Image image(sf::Vector2u(10, 10), pixels.data());
|
|
CHECK(image.getSize() == sf::Vector2u(10, 10));
|
|
CHECK(image.getPixelsPtr() != nullptr);
|
|
|
|
for (std::uint32_t i = 0; i < 10; ++i)
|
|
{
|
|
for (std::uint32_t j = 0; j < 10; ++j)
|
|
{
|
|
CHECK(image.getPixel(sf::Vector2u(i, j)) == sf::Color::Red);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
SECTION("loadFromFile()")
|
|
{
|
|
SECTION("Invalid file")
|
|
{
|
|
CHECK(!sf::Image::loadFromFile("."));
|
|
CHECK(!sf::Image::loadFromFile("this/does/not/exist.jpg"));
|
|
}
|
|
|
|
SECTION("Successful load")
|
|
{
|
|
std::optional<sf::Image> image;
|
|
|
|
SECTION("bmp")
|
|
{
|
|
image = sf::Image::loadFromFile("Graphics/sfml-logo-big.bmp").value();
|
|
REQUIRE(image.has_value());
|
|
CHECK(image->getPixel({0, 0}) == sf::Color::White);
|
|
CHECK(image->getPixel({200, 150}) == sf::Color(144, 208, 62));
|
|
}
|
|
|
|
SECTION("png")
|
|
{
|
|
image = sf::Image::loadFromFile("Graphics/sfml-logo-big.png").value();
|
|
REQUIRE(image.has_value());
|
|
CHECK(image->getPixel({0, 0}) == sf::Color(255, 255, 255, 0));
|
|
CHECK(image->getPixel({200, 150}) == sf::Color(144, 208, 62));
|
|
}
|
|
|
|
SECTION("jpg")
|
|
{
|
|
image = sf::Image::loadFromFile("Graphics/sfml-logo-big.jpg").value();
|
|
REQUIRE(image.has_value());
|
|
CHECK(image->getPixel({0, 0}) == sf::Color::White);
|
|
CHECK(image->getPixel({200, 150}) == sf::Color(144, 208, 62));
|
|
}
|
|
|
|
SECTION("gif")
|
|
{
|
|
image = sf::Image::loadFromFile("Graphics/sfml-logo-big.gif").value();
|
|
REQUIRE(image.has_value());
|
|
CHECK(image->getPixel({0, 0}) == sf::Color::White);
|
|
CHECK(image->getPixel({200, 150}) == sf::Color(146, 210, 62));
|
|
}
|
|
|
|
SECTION("psd")
|
|
{
|
|
image = sf::Image::loadFromFile("Graphics/sfml-logo-big.psd").value();
|
|
REQUIRE(image.has_value());
|
|
CHECK(image->getPixel({0, 0}) == sf::Color::White);
|
|
CHECK(image->getPixel({200, 150}) == sf::Color(144, 208, 62));
|
|
}
|
|
|
|
CHECK(image->getSize() == sf::Vector2u(1001, 304));
|
|
CHECK(image->getPixelsPtr() != nullptr);
|
|
}
|
|
}
|
|
|
|
SECTION("loadFromMemory()")
|
|
{
|
|
SECTION("Invalid pointer")
|
|
{
|
|
CHECK(!sf::Image::loadFromMemory(nullptr, 1));
|
|
}
|
|
|
|
SECTION("Invalid size")
|
|
{
|
|
const std::byte testByte{0xAB};
|
|
CHECK(!sf::Image::loadFromMemory(&testByte, 0));
|
|
}
|
|
|
|
SECTION("Failed load")
|
|
{
|
|
std::vector<std::uint8_t> memory;
|
|
|
|
SECTION("Empty")
|
|
{
|
|
memory.clear();
|
|
}
|
|
|
|
SECTION("Junk data")
|
|
{
|
|
memory = {1, 2, 3, 4};
|
|
}
|
|
|
|
CHECK(!sf::Image::loadFromMemory(memory.data(), memory.size()));
|
|
}
|
|
|
|
SECTION("Successful load")
|
|
{
|
|
const auto memory = sf::Image({24, 24}, sf::Color::Green).saveToMemory("png").value();
|
|
const auto image = sf::Image::loadFromMemory(memory.data(), memory.size()).value();
|
|
CHECK(image.getSize() == sf::Vector2u(24, 24));
|
|
CHECK(image.getPixelsPtr() != nullptr);
|
|
CHECK(image.getPixel({0, 0}) == sf::Color::Green);
|
|
CHECK(image.getPixel({23, 23}) == sf::Color::Green);
|
|
}
|
|
}
|
|
|
|
SECTION("loadFromStream()")
|
|
{
|
|
sf::FileInputStream stream;
|
|
|
|
SECTION("Invalid stream")
|
|
{
|
|
CHECK(!sf::Image::loadFromStream(stream));
|
|
}
|
|
|
|
SECTION("Successful load")
|
|
{
|
|
CHECK(stream.open("Graphics/sfml-logo-big.png"));
|
|
const auto image = sf::Image::loadFromStream(stream).value();
|
|
CHECK(image.getSize() == sf::Vector2u(1001, 304));
|
|
CHECK(image.getPixelsPtr() != nullptr);
|
|
CHECK(image.getPixel({0, 0}) == sf::Color(255, 255, 255, 0));
|
|
CHECK(image.getPixel({200, 150}) == sf::Color(144, 208, 62));
|
|
}
|
|
}
|
|
|
|
SECTION("saveToFile()")
|
|
{
|
|
SECTION("Invalid size")
|
|
{
|
|
CHECK(!sf::Image({10, 0}, sf::Color::Magenta).saveToFile("test.jpg"));
|
|
CHECK(!sf::Image({0, 10}, sf::Color::Magenta).saveToFile("test.jpg"));
|
|
}
|
|
|
|
const sf::Image image({256, 256}, sf::Color::Magenta);
|
|
|
|
SECTION("No extension")
|
|
{
|
|
CHECK(!image.saveToFile("wheresmyextension"));
|
|
CHECK(!image.saveToFile("pls/add/extension"));
|
|
}
|
|
|
|
SECTION("Invalid extension")
|
|
{
|
|
CHECK(!image.saveToFile("test.ps"));
|
|
CHECK(!image.saveToFile("test.foo"));
|
|
}
|
|
|
|
SECTION("Successful save")
|
|
{
|
|
auto filename = std::filesystem::temp_directory_path();
|
|
|
|
SECTION("To .bmp")
|
|
{
|
|
filename /= "test.bmp";
|
|
CHECK(image.saveToFile(filename));
|
|
}
|
|
|
|
SECTION("To .tga")
|
|
{
|
|
filename /= "test.tga";
|
|
CHECK(image.saveToFile(filename));
|
|
}
|
|
|
|
SECTION("To .png")
|
|
{
|
|
filename /= "test.png";
|
|
CHECK(image.saveToFile(filename));
|
|
}
|
|
|
|
// Cannot test JPEG encoding due to it triggering UB in stbiw__jpg_writeBits
|
|
|
|
const auto loadedImage = sf::Image::loadFromFile(filename).value();
|
|
CHECK(loadedImage.getSize() == sf::Vector2u(256, 256));
|
|
CHECK(loadedImage.getPixelsPtr() != nullptr);
|
|
|
|
CHECK(std::filesystem::remove(filename));
|
|
}
|
|
}
|
|
|
|
SECTION("saveToMemory()")
|
|
{
|
|
SECTION("Invalid size")
|
|
{
|
|
CHECK(!sf::Image({10, 0}, sf::Color::Magenta).saveToMemory("test.jpg"));
|
|
CHECK(!sf::Image({0, 10}, sf::Color::Magenta).saveToMemory("test.jpg"));
|
|
}
|
|
|
|
const sf::Image image({16, 16}, sf::Color::Magenta);
|
|
|
|
SECTION("No extension")
|
|
{
|
|
CHECK(!image.saveToMemory(""));
|
|
}
|
|
|
|
SECTION("Invalid extension")
|
|
{
|
|
CHECK(!image.saveToMemory("."));
|
|
CHECK(!image.saveToMemory("gif"));
|
|
CHECK(!image.saveToMemory(".jpg")); // Supposed to be "jpg"
|
|
}
|
|
|
|
SECTION("Successful save")
|
|
{
|
|
std::optional<std::vector<std::uint8_t>> maybeOutput;
|
|
|
|
SECTION("To bmp")
|
|
{
|
|
maybeOutput = image.saveToMemory("bmp");
|
|
REQUIRE(maybeOutput.has_value());
|
|
const auto& output = *maybeOutput;
|
|
REQUIRE(output.size() == 1146);
|
|
CHECK(output[0] == 66);
|
|
CHECK(output[1] == 77);
|
|
CHECK(output[2] == 122);
|
|
CHECK(output[3] == 4);
|
|
CHECK(output[1000] == 255);
|
|
CHECK(output[1001] == 255);
|
|
CHECK(output[1002] == 255);
|
|
CHECK(output[1003] == 0);
|
|
}
|
|
|
|
SECTION("To tga")
|
|
{
|
|
maybeOutput = image.saveToMemory("tga");
|
|
REQUIRE(maybeOutput.has_value());
|
|
const auto& output = *maybeOutput;
|
|
REQUIRE(output.size() == 98);
|
|
CHECK(output[0] == 0);
|
|
CHECK(output[1] == 0);
|
|
CHECK(output[2] == 10);
|
|
CHECK(output[3] == 0);
|
|
}
|
|
|
|
SECTION("To png")
|
|
{
|
|
maybeOutput = image.saveToMemory("png");
|
|
REQUIRE(maybeOutput.has_value());
|
|
const auto& output = *maybeOutput;
|
|
REQUIRE(output.size() == 92);
|
|
CHECK(output[0] == 137);
|
|
CHECK(output[1] == 80);
|
|
CHECK(output[2] == 78);
|
|
CHECK(output[3] == 71);
|
|
}
|
|
|
|
// Cannot test JPEG encoding due to it triggering UB in stbiw__jpg_writeBits
|
|
}
|
|
}
|
|
|
|
SECTION("Set/get pixel")
|
|
{
|
|
sf::Image image(sf::Vector2u(10, 10), sf::Color::Green);
|
|
CHECK(image.getPixel(sf::Vector2u(2, 2)) == sf::Color::Green);
|
|
|
|
image.setPixel(sf::Vector2u(2, 2), sf::Color::Blue);
|
|
CHECK(image.getPixel(sf::Vector2u(2, 2)) == sf::Color::Blue);
|
|
}
|
|
|
|
SECTION("Copy from Image")
|
|
{
|
|
SECTION("Copy (Image, Vector2u)")
|
|
{
|
|
const sf::Image image1(sf::Vector2u(10, 10), sf::Color::Blue);
|
|
sf::Image image2(sf::Vector2u(10, 10));
|
|
CHECK(image2.copy(image1, sf::Vector2u(0, 0)));
|
|
|
|
for (std::uint32_t i = 0; i < 10; ++i)
|
|
{
|
|
for (std::uint32_t j = 0; j < 10; ++j)
|
|
{
|
|
CHECK(image1.getPixel(sf::Vector2u(i, j)) == image2.getPixel(sf::Vector2u(i, j)));
|
|
}
|
|
}
|
|
}
|
|
|
|
SECTION("Copy (Image, Vector2u, IntRect)")
|
|
{
|
|
const sf::Image image1(sf::Vector2u(5, 5), sf::Color::Blue);
|
|
sf::Image image2(sf::Vector2u(10, 10));
|
|
CHECK(image2.copy(image1, sf::Vector2u(0, 0), sf::IntRect(sf::Vector2i(0, 0), sf::Vector2i(5, 5))));
|
|
|
|
for (std::uint32_t i = 0; i < 10; ++i)
|
|
{
|
|
for (std::uint32_t j = 0; j < 10; ++j)
|
|
{
|
|
if (i <= 4 && j <= 4)
|
|
CHECK(image2.getPixel(sf::Vector2u(i, j)) == sf::Color::Blue);
|
|
else
|
|
CHECK(image2.getPixel(sf::Vector2u(i, j)) == sf::Color::Black);
|
|
}
|
|
}
|
|
}
|
|
|
|
SECTION("Copy (Image, Vector2u, IntRect, bool)")
|
|
{
|
|
const sf::Color dest(255, 0, 0, 255);
|
|
const sf::Color source(5, 255, 78, 232);
|
|
|
|
// Create the composited colour for via the alpha composite over operation
|
|
const auto a = static_cast<std::uint8_t>(source.a + (dest.a * (255 - source.a)) / 255);
|
|
const auto r = static_cast<std::uint8_t>(
|
|
((source.r * source.a) + ((dest.r * dest.a) * (255 - source.a)) / 255) / a);
|
|
const auto g = static_cast<std::uint8_t>(
|
|
((source.g * source.a) + ((dest.g * dest.a) * (255 - source.a)) / 255) / a);
|
|
const auto b = static_cast<std::uint8_t>(
|
|
((source.b * source.a) + ((dest.b * dest.a) * (255 - source.a)) / 255) / a);
|
|
const sf::Color composite(r, g, b, a);
|
|
|
|
sf::Image image1(sf::Vector2u(10, 10), dest);
|
|
const sf::Image image2(sf::Vector2u(10, 10), source);
|
|
CHECK(image1.copy(image2, sf::Vector2u(0, 0), sf::IntRect(sf::Vector2i(0, 0), sf::Vector2i(10, 10)), true));
|
|
|
|
for (std::uint32_t i = 0; i < 10; ++i)
|
|
{
|
|
for (std::uint32_t j = 0; j < 10; ++j)
|
|
{
|
|
CHECK(image1.getPixel(sf::Vector2u(i, j)) == composite);
|
|
}
|
|
}
|
|
}
|
|
|
|
SECTION("Copy (Out of bounds sourceRect)")
|
|
{
|
|
const sf::Image image1(sf::Vector2u(5, 5), sf::Color::Blue);
|
|
sf::Image image2(sf::Vector2u(10, 10), sf::Color::Red);
|
|
CHECK(!image2.copy(image1, sf::Vector2u(0, 0), sf::IntRect(sf::Vector2i(5, 5), sf::Vector2i(9, 9))));
|
|
|
|
for (std::uint32_t i = 0; i < 10; ++i)
|
|
{
|
|
for (std::uint32_t j = 0; j < 10; ++j)
|
|
{
|
|
CHECK(image2.getPixel(sf::Vector2u(i, j)) == sf::Color::Red);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
SECTION("Create mask from color")
|
|
{
|
|
SECTION("createMaskFromColor(Color)")
|
|
{
|
|
sf::Image image(sf::Vector2u(10, 10), sf::Color::Blue);
|
|
image.createMaskFromColor(sf::Color::Blue);
|
|
|
|
for (std::uint32_t i = 0; i < 10; ++i)
|
|
{
|
|
for (std::uint32_t j = 0; j < 10; ++j)
|
|
{
|
|
CHECK(image.getPixel(sf::Vector2u(i, j)) == sf::Color(0, 0, 255, 0));
|
|
}
|
|
}
|
|
}
|
|
|
|
SECTION("createMaskFromColor(Color, std::uint8_t)")
|
|
{
|
|
sf::Image image(sf::Vector2u(10, 10), sf::Color::Blue);
|
|
image.createMaskFromColor(sf::Color::Blue, 100);
|
|
|
|
for (std::uint32_t i = 0; i < 10; ++i)
|
|
{
|
|
for (std::uint32_t j = 0; j < 10; ++j)
|
|
{
|
|
CHECK(image.getPixel(sf::Vector2u(i, j)) == sf::Color(0, 0, 255, 100));
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
SECTION("Flip horizontally")
|
|
{
|
|
sf::Image image(sf::Vector2u(10, 10), sf::Color::Red);
|
|
image.setPixel(sf::Vector2u(0, 0), sf::Color::Green);
|
|
image.flipHorizontally();
|
|
|
|
CHECK(image.getPixel(sf::Vector2u(9, 0)) == sf::Color::Green);
|
|
}
|
|
|
|
SECTION("Flip vertically")
|
|
{
|
|
sf::Image image(sf::Vector2u(10, 10), sf::Color::Red);
|
|
image.setPixel(sf::Vector2u(0, 0), sf::Color::Green);
|
|
image.flipVertically();
|
|
|
|
CHECK(image.getPixel(sf::Vector2u(0, 9)) == sf::Color::Green);
|
|
}
|
|
}
|