// Copyright 2025 Steven Le Rouzic // // SPDX-License-Identifier: BSD-3-Clause #pragma once #include "asl/base/utility.hpp" #include "asl/base/functional.hpp" namespace asl { template<invocable Callback> class DeferCallback { Callback m_callback; bool m_moved = false; public: template<typename T> explicit DeferCallback(T&& callback) requires (!same_as<un_cvref_t<T>, DeferCallback>) : m_callback(std::forward<T>(callback)) { } ASL_DELETE_COPY(DeferCallback); DeferCallback(DeferCallback&& other) : m_callback(std::move(other.m_callback)), m_moved(exchange(other.m_moved, true)) { } DeferCallback& operator=(DeferCallback&&) = delete; ~DeferCallback() { if (!m_moved) { invoke(m_callback); } } }; struct DeferFactory { template<invocable Callback> DeferCallback<Callback> operator<<(Callback&& callback) const { return DeferCallback<Callback>(std::forward<Callback>(callback)); } }; } // namespace asl #define ASL_DEFER auto ASL_CONCAT(_defer_, __COUNTER__) = ::asl::DeferFactory{} <<