blob: 6879ef8c2c4b68ba2062a940262627c6f369a668 [file] [log] [blame] [edit]
/*
* Copyright 2023 Rive
*/
// "lite_rtti_cast<T*>()" is a very basic polyfill for "dynamic_cast<T*>()" that
// can only cast a pointer to its most-derived type. To use it, the base class
// must derive from enable_lite_rtti, and the subclass must inherit from
// lite_rtti_override:
//
// class Root : public enable_lite_rtti<Root> {};
// class Derived : public lite_rtti_override<Root, Derived> {};
// Root* derived = new Derived();
// lite_rtti_cast<Derived*>(derived);
//
#pragma once
#include "utils/compile_time_string_hash.hpp"
#include "rive/refcnt.hpp"
#include <stdint.h>
#include <type_traits>
namespace rive
{
// Enable lite rtti on the root of a class hierarchy.
template <class Root, unsigned int ID> class enable_lite_rtti
{
public:
unsigned int liteTypeID() const { return m_liteTypeId; }
protected:
unsigned int m_liteTypeId = ID;
};
// Override the lite rtti type ID on subsequent classes of a class hierarchy.
template <class Base, class Derived, unsigned int ID>
class lite_rtti_override : public Base
{
public:
constexpr static uint32_t LITE_RTTI_TYPE_ID = ID;
lite_rtti_override() { Base::m_liteTypeId = ID; }
template <typename... Args>
lite_rtti_override(Args&&... args) : Base(std::forward<Args>(args)...)
{
Base::m_liteTypeId = ID;
}
};
// Like dynamic_cast<>, but can only cast a pointer to its most-derived type.
template <class U, class T> U lite_rtti_cast(T* t)
{
if (t != nullptr &&
t->liteTypeID() == std::remove_pointer<U>::type::LITE_RTTI_TYPE_ID)
{
return static_cast<U>(t);
}
return nullptr;
}
template <class U, class T> rcp<U> lite_rtti_rcp_cast(rcp<T> t)
{
if (t != nullptr &&
t->liteTypeID() == std::remove_pointer<U>::type::LITE_RTTI_TYPE_ID)
{
return static_rcp_cast<U>(t);
}
return nullptr;
}
// Different versions of clang-format disagree on how to formate these.
// clang-format off
#define ENABLE_LITE_RTTI(ROOT) enable_lite_rtti<ROOT, CONST_ID(ROOT)>
#define LITE_RTTI_OVERRIDE(BASE, DERRIVED) lite_rtti_override<BASE, DERRIVED, CONST_ID(DERRIVED)>
#define LITE_RTTI_CAST_OR_RETURN(NAME, TYPE, POINTER) \
auto NAME = rive::lite_rtti_cast<TYPE>(POINTER); \
if (NAME == nullptr) \
return
#define LITE_RTTI_CAST_OR_BREAK(NAME, TYPE, POINTER) \
auto NAME = rive::lite_rtti_cast<TYPE>(POINTER); \
if (NAME == nullptr) \
break
#define LITE_RTTI_CAST_OR_CONTINUE(NAME, TYPE, POINTER) \
auto NAME = rive::lite_rtti_cast<TYPE>(POINTER); \
if (NAME == nullptr) \
continue
// clang-format on
} // namespace rive