/* * Copyright (c) 2020 - 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. */ #ifndef _TVG_CANVAS_H_ #define _TVG_CANVAS_H_ #include "tvgPaint.h" struct Canvas::Impl { list paints; RenderMethod* renderer; bool refresh = false; //if all paints should be updated by force. bool drawing = false; //on drawing condition? Impl(RenderMethod* pRenderer) : renderer(pRenderer) { renderer->ref(); } ~Impl() { //make it sure any deffered jobs if (renderer) { renderer->sync(); renderer->clear(); } clearPaints(); if (renderer && (renderer->unref() == 0)) delete(renderer); } void clearPaints() { for (auto paint : paints) { if (P(paint)->unref() == 0) delete(paint); } paints.clear(); } Result push(unique_ptr paint) { //You can not push paints during rendering. if (drawing) return Result::InsufficientCondition; auto p = paint.release(); if (!p) return Result::MemoryCorruption; PP(p)->ref(); paints.push_back(p); return update(p, true); } Result clear(bool free) { //Clear render target before drawing if (!renderer || !renderer->clear()) return Result::InsufficientCondition; //Free paints if (free) clearPaints(); drawing = false; return Result::Success; } void needRefresh() { refresh = true; } Result update(Paint* paint, bool force) { if (paints.empty() || drawing || !renderer) return Result::InsufficientCondition; Array clips; auto flag = RenderUpdateFlag::None; if (refresh || force) flag = RenderUpdateFlag::All; if (paint) { paint->pImpl->update(renderer, nullptr, clips, 255, flag); } else { for (auto paint : paints) { paint->pImpl->update(renderer, nullptr, clips, 255, flag); } refresh = false; } return Result::Success; } Result draw() { if (drawing || paints.empty() || !renderer || !renderer->preRender()) return Result::InsufficientCondition; bool rendered = false; for (auto paint : paints) { if (paint->pImpl->render(renderer)) rendered = true; } if (!rendered || !renderer->postRender()) return Result::InsufficientCondition; drawing = true; return Result::Success; } Result sync() { if (!drawing) return Result::InsufficientCondition; if (renderer->sync()) { drawing = false; return Result::Success; } return Result::InsufficientCondition; } }; #endif /* _TVG_CANVAS_H_ */