thorvg/test/testSwCanvas.cpp
Hermet Park ed01ef717e api: revise the spec
Remove the requirement for unique_ptr in the function prototypes.
This change will simplify the API usage, making it more streamlined
and user-friendly. However, memory management will now be the
responsibility of the user.

C++ API Modification:
- Result Paint::mask(std::unique_ptr<Paint> target, MaskMethod method) -> Result Paint::mask(Paint* target, MaskMethod method)
- Result Paint::clip(std::unique_ptr<Paint> clipper) -> Result Paint::clip(Paint* clipper)
- virtual Result Canvas::push(std::unique_ptr<Paint> paint) -> virtual Result Canvas::push(Paint* paint)
- std::unique_ptr<LinearGradient> LinearGradient::gen() -> LinearGradient* LinearGradient::gen()
- std::unique_ptr<RadialGradient> RadialGradient::gen() -> RadialGradient* RadialGradient::gen()
- Result Shape::strokeFill(std::unique_ptr<Fill> f) -> Result Shape::strokeFill(Fill* f)
- Result Shape::fill(std::unique_ptr<Fill> f) -> Result Shape::fill(Fill* f)
- std::unique_ptr<Shape> Shape::gen() -> Shape* Shape::gen()
- std::unique_ptr<Picture> Picture::gen() -> Result Picture::push(Paint* paint)
- std::unique_ptr<Scene> Scene::gen() -> Scene* Scene::gen()
- Result Text::fill(std::unique_ptr<Fill> f) -> Result Text::fill(Fill* f)
- std::unique_ptr<Text> Text::gen() -> Text* Text::gen()
- std::unique_ptr<SwCanvas> SwCanvas::gen() -> SwCanvas* SwCanvas::gen()
- std::unique_ptr<GlCanvas> GlCanvas::gen() -> GlCanvas* GlCanvas::gen()
- std::unique_ptr<Animation> Animation::gen() -> Animation* Animation::gen()
- Result Saver::background(std::unique_ptr<Paint> paint) -> Result Saver::background(Paint* paint)
- Result Saver::save(std::unique_ptr<Paint> paint, const char* filename, uint32_t quality = 100) -> Result Saver::save(Paint* paint, const char* filename, uint32_t quality = 100)
- std::unique_ptr<Saver> Saver::gen() -> Saver* Saver::gen()
- std::unique_ptr<Accessor> Accessor::gen() -> Accessor* Accessor::gen()

C++ API removal:
- template<typename T = tvg::Paint> std::unique_ptr<T> cast(Paint* paint)
- template<typename T = tvg::Paint> std::unique_ptr<T> cast(Paint* paint)

issue: https://github.com/thorvg/thorvg/issues
2024-11-09 12:29:15 +09:00

345 lines
No EOL
12 KiB
C++

/*
* Copyright (c) 2021 - 2024 the ThorVG project. All rights reserved.
* Permission is hereby granted, free of charge, to any person obtaining a copy
* of this software and associated documentation files (the "Software"), to deal
* in the Software without restriction, including without limitation the rights
* to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
* copies of the Software, and to permit persons to whom the Software is
* furnished to do so, subject to the following conditions:
* The above copyright notice and this permission notice shall be included in all
* copies or substantial portions of the Software.
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
* AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
* OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
* SOFTWARE.
*/
#include <thorvg.h>
#include "config.h"
#include "catch.hpp"
using namespace tvg;
using namespace std;
#if 0
TEST_CASE("Missing Initialization", "[tvgSwCanvas]")
{
auto canvas = unique_ptr<SwCanvas>(SwCanvas::gen());
REQUIRE(canvas == nullptr);
}
TEST_CASE("Basic Creation", "[tvgSwCanvas]")
{
REQUIRE(Initializer::init(0, CanvasEngine::Sw) == Result::Success);
auto canvas = unique_ptr<SwCanvas>(SwCanvas::gen());
REQUIRE(canvas);
auto canvas2 = unique_ptr<SwCanvas>(SwCanvas::gen());
REQUIRE(canvas2);
auto canvas3 = unique_ptr<SwCanvas>(SwCanvas::gen());
REQUIRE(canvas3);
REQUIRE(Initializer::term(CanvasEngine::Sw) == Result::Success);
}
TEST_CASE("Target Buffer", "[tvgSwCanvas]")
{
REQUIRE(Initializer::init(0, CanvasEngine::Sw) == Result::Success);
auto canvas = unique_ptr<SwCanvas>(SwCanvas::gen());
REQUIRE(canvas);
uint32_t buffer[100*100];
REQUIRE(canvas->target(buffer, 100, 100, 100, ColorSpace::ABGR8888) == Result::Success);
REQUIRE(canvas->target(buffer, 100, 100, 100, ColorSpace::ARGB8888) == Result::Success);
REQUIRE(canvas->target(nullptr, 100, 100, 100, ColorSpace::ARGB8888) == Result::InvalidArguments);
REQUIRE(canvas->target(buffer, 0, 100, 100, ColorSpace::ARGB8888) == Result::InvalidArguments);
REQUIRE(canvas->target(buffer, 100, 0, 100, ColorSpace::ARGB8888) == Result::InvalidArguments);
REQUIRE(canvas->target(buffer, 100, 200, 100, ColorSpace::ARGB8888) == Result::InvalidArguments);
REQUIRE(canvas->target(buffer, 100, 100, 0, ColorSpace::ARGB8888) == Result::InvalidArguments);
REQUIRE(Initializer::term(CanvasEngine::Sw) == Result::Success);
}
TEST_CASE("Memory Pool", "[tvgSwCanvas]")
{
REQUIRE(Initializer::init(0, CanvasEngine::Sw) == Result::Success);
auto canvas = unique_ptr<SwCanvas>(SwCanvas::gen());
REQUIRE(canvas);
REQUIRE(canvas->mempool(SwCanvas::MempoolPolicy::Default) == Result::Success);
REQUIRE(canvas->mempool(SwCanvas::MempoolPolicy::Individual) == Result::Success);
REQUIRE(canvas->mempool(SwCanvas::MempoolPolicy::Shareable) == Result::Success);
REQUIRE(canvas->mempool(SwCanvas::MempoolPolicy::Default) == Result::Success);
auto canvas2 = unique_ptr<SwCanvas>(SwCanvas::gen());
REQUIRE(canvas);
REQUIRE(canvas2->mempool(SwCanvas::MempoolPolicy::Default) == Result::Success);
REQUIRE(canvas2->mempool(SwCanvas::MempoolPolicy::Individual) == Result::Success);
REQUIRE(canvas2->mempool(SwCanvas::MempoolPolicy::Shareable) == Result::Success);
REQUIRE(canvas2->mempool(SwCanvas::MempoolPolicy::Shareable) == Result::Success);
REQUIRE(canvas2->push(Shape::gen()) == Result::Success);
REQUIRE(canvas2->mempool(SwCanvas::MempoolPolicy::Individual) == Result::InsufficientCondition);
REQUIRE(Initializer::term(CanvasEngine::Sw) == Result::Success);
}
TEST_CASE("Pushing Paints", "[tvgSwCanvas]")
{
REQUIRE(Initializer::init(0) == Result::Success);
auto canvas = unique_ptr<SwCanvas>(SwCanvas::gen());
REQUIRE(canvas);
uint32_t buffer[100*100];
REQUIRE(canvas->target(buffer, 100, 100, 100, ColorSpace::ARGB8888) == Result::Success);
//Try all types of paints.
REQUIRE(canvas->push(Shape::gen()) == Result::Success);
REQUIRE(canvas->push(Picture::gen()) == Result::Success);
REQUIRE(canvas->push(Scene::gen()) == Result::Success);
//Cases by contexts.
REQUIRE(canvas->update() == Result::Success);
REQUIRE(canvas->push(Shape::gen()) == Result::Success);
REQUIRE(canvas->push(Shape::gen()) == Result::Success);
REQUIRE(canvas->clear() == Result::Success);
Paint* paints[2];
paints[0] = Shape::gen();
REQUIRE(canvas->push(paints[0]) == Result::Success);
//Negative case 1
REQUIRE(canvas->push(nullptr) == Result::MemoryCorruption);
//Negative case 2
Shape* shape6 = nullptr;
REQUIRE(canvas->push(shape6) == Result::MemoryCorruption);
paints[1] = Shape::gen();
REQUIRE(canvas->push(paints[1]) == Result::Success);
REQUIRE(canvas->draw() == Result::Success);
//Negative case 3
REQUIRE(canvas->push(Shape::gen()) == Result::InsufficientCondition);
//Check list of paints
auto list = canvas->paints();
REQUIRE(list.size() == 2);
int idx = 0;
for (auto paint : list) {
REQUIRE(paints[idx] == paint);
++idx;
}
REQUIRE(Initializer::term() == Result::Success);
}
TEST_CASE("Clear", "[tvgSwCanvas]")
{
REQUIRE(Initializer::init(0) == Result::Success);
auto canvas = unique_ptr<SwCanvas>(SwCanvas::gen());
REQUIRE(canvas);
auto canvas2 = unique_ptr<SwCanvas>(SwCanvas::gen());
REQUIRE(canvas2);
//Try 0: Negative
REQUIRE(canvas->clear() == Result::InsufficientCondition);
REQUIRE(canvas->clear(false) == Result::InsufficientCondition);
REQUIRE(canvas->clear() == Result::InsufficientCondition);
REQUIRE(canvas2->clear(false) == Result::InsufficientCondition);
REQUIRE(canvas2->clear() == Result::InsufficientCondition);
REQUIRE(canvas2->clear(false) == Result::InsufficientCondition);
uint32_t buffer[100*100];
REQUIRE(canvas->target(buffer, 100, 100, 100, ColorSpace::ARGB8888) == Result::Success);
uint32_t buffer2[100*100];
REQUIRE(canvas2->target(buffer2, 100, 100, 100, ColorSpace::ARGB8888) == Result::Success);
//Try 1: Push -> Clear
for (int i = 0; i < 5; ++i) {
REQUIRE(canvas->push(Shape::gen()) == Result::Success);
auto shape2 = Shape::gen();
REQUIRE(shape2);
REQUIRE(canvas2->push(shape2) == Result::Success);
}
REQUIRE(canvas->clear() == Result::Success);
REQUIRE(canvas->clear(false) == Result::Success);
REQUIRE(canvas2->clear(false) == Result::Success);
REQUIRE(canvas2->clear() == Result::Success);
//Try 2: Push -> Update -> Clear
for (int i = 0; i < 5; ++i) {
REQUIRE(canvas->push(Shape::gen()) == Result::Success);
auto shape2 = Shape::gen();
REQUIRE(shape2);
REQUIRE(canvas2->push(shape2) == Result::Success);
}
REQUIRE(canvas->update() == Result::Success);
REQUIRE(canvas->clear() == Result::Success);
REQUIRE(canvas->clear(false) == Result::Success);
REQUIRE(canvas2->update() == Result::Success);
REQUIRE(canvas2->clear(false) == Result::Success);
REQUIRE(canvas2->clear() == Result::Success);
REQUIRE(Initializer::term() == Result::Success);
}
TEST_CASE("Update", "[tvgSwCanvas]")
{
REQUIRE(Initializer::init(0) == Result::Success);
auto canvas = unique_ptr<SwCanvas>(SwCanvas::gen());
REQUIRE(canvas);
uint32_t buffer[100*100];
REQUIRE(canvas->target(buffer, 100, 100, 100, ColorSpace::ARGB8888) == Result::Success);
REQUIRE(canvas->update() == Result::InsufficientCondition);
REQUIRE(canvas->push(Shape::gen()) == Result::Success);
//No pushed shape
auto shape = Shape::gen();
REQUIRE(canvas->update(shape) == Result::Success);
//Normal case
REQUIRE(canvas->push(shape) == Result::Success);
REQUIRE(canvas->update(shape) == Result::Success);
REQUIRE(canvas->update() == Result::Success);
REQUIRE(canvas->draw() == Result::Success);
REQUIRE(canvas->update() == Result::InsufficientCondition);
REQUIRE(canvas->clear() == Result::InsufficientCondition);
REQUIRE(canvas->sync() == Result::Success);
REQUIRE(canvas->clear() == Result::Success);
//Invalid shape
REQUIRE(canvas->update(shape) == Result::InsufficientCondition);
REQUIRE(Initializer::term() == Result::Success);
}
TEST_CASE("Synchronized Drawing", "[tvgSwCanvas]")
{
REQUIRE(Initializer::init(0) == Result::Success);
auto canvas = unique_ptr<SwCanvas>(SwCanvas::gen());
REQUIRE(canvas);
REQUIRE(canvas->sync() == Result::InsufficientCondition);
REQUIRE(canvas->draw() == Result::InsufficientCondition);
uint32_t buffer[100*100];
REQUIRE(canvas->target(buffer, 100, 100, 100, ColorSpace::ARGB8888) == Result::Success);
REQUIRE(canvas->draw() == Result::InsufficientCondition);
REQUIRE(canvas->sync() == Result::InsufficientCondition);
//Invalid Shape
auto shape = Shape::gen();
REQUIRE(shape);
REQUIRE(canvas->push(shape) == Result::Success);
REQUIRE(canvas->draw() == Result::Success);
REQUIRE(canvas->sync() == Result::Success);
REQUIRE(canvas->clear() == Result::Success);
auto shape2 = Shape::gen();
REQUIRE(shape2);
REQUIRE(shape2->appendRect(0, 0, 100, 100) == Result::Success);
REQUIRE(shape2->fill(255, 255, 255, 255) == Result::Success);
REQUIRE(canvas->push(shape2) == Result::Success);
REQUIRE(canvas->draw() == Result::Success);
REQUIRE(canvas->sync() == Result::Success);
REQUIRE(Initializer::term() == Result::Success);
}
TEST_CASE("Asynchronous Drawing", "[tvgSwCanvas]")
{
//Use multi-threading
REQUIRE(Initializer::init(2) == Result::Success);
auto canvas = unique_ptr<SwCanvas>(SwCanvas::gen());
REQUIRE(canvas);
uint32_t buffer[100*100];
REQUIRE(canvas->target(buffer, 100, 100, 100, ColorSpace::ARGB8888) == Result::Success);
for (int i = 0; i < 3; ++i) {
auto shape = Shape::gen();
REQUIRE(shape);
REQUIRE(shape->appendRect(0, 0, 100, 100) == Result::Success);
REQUIRE(shape->fill(255, 255, 255, 255) == Result::Success);
REQUIRE(canvas->push(shape) == Result::Success);
}
REQUIRE(canvas->draw() == Result::Success);
REQUIRE(canvas->sync() == Result::Success);
REQUIRE(Initializer::term() == Result::Success);
}
TEST_CASE("Viewport", "[tvgSwCanvas]")
{
REQUIRE(Initializer::init(0) == Result::Success);
auto canvas = unique_ptr<SwCanvas>(SwCanvas::gen());
REQUIRE(canvas);
REQUIRE(canvas->viewport(25, 25, 100, 100) == Result::Success);
uint32_t buffer[100*100];
REQUIRE(canvas->target(buffer, 100, 100, 100, ColorSpace::ARGB8888) == Result::Success);
REQUIRE(canvas->viewport(25, 25, 50, 50) == Result::Success);
auto shape = Shape::gen();
REQUIRE(shape);
REQUIRE(shape->appendRect(0, 0, 100, 100) == Result::Success);
REQUIRE(shape->fill(255, 255, 255, 255) == Result::Success);
REQUIRE(canvas->push(shape) == Result::Success);
//Negative, not allowed
REQUIRE(canvas->viewport(15, 25, 5, 5) == Result::InsufficientCondition);
REQUIRE(canvas->draw() == Result::Success);
//Negative, not allowed
REQUIRE(canvas->viewport(25, 25, 10, 10) == Result::InsufficientCondition);
REQUIRE(canvas->sync() == Result::Success);
REQUIRE(Initializer::term() == Result::Success);
}
#endif