1
Fork 0
mirror of https://github.com/RGBCube/serenity synced 2025-05-31 05:38:11 +00:00
serenity/Userland/Libraries/LibCore/Promise.h
kleines Filmröllchen bfd9f681f7 LibCore+Userland: Allow canceling promises
To make EventLoop cancel its managed Promises, we need the ability to
cancel them in the first place.
2023-03-13 12:12:17 +00:00

70 lines
1.5 KiB
C++

/*
* Copyright (c) 2021, Kyle Pereira <hey@xylepereira.me>
*
* SPDX-License-Identifier: BSD-2-Clause
*/
#pragma once
#include <LibCore/EventLoop.h>
#include <LibCore/Object.h>
namespace Core {
template<typename Result>
class Promise : public Object {
C_OBJECT(Promise);
public:
Function<void(Result&)> on_resolved;
void resolve(Result&& result)
{
m_pending_or_error = move(result);
if (on_resolved)
on_resolved(m_pending_or_error.value());
}
void cancel(Error error)
{
m_pending_or_error = move(error);
}
bool is_canceled()
{
return m_pending_or_error.has_value() && m_pending_or_error->is_error();
}
bool is_resolved() const
{
return m_pending_or_error.has_value() && !m_pending_or_error->is_error();
}
ErrorOr<Result> await()
{
while (!m_pending_or_error.has_value())
Core::EventLoop::current().pump();
return m_pending_or_error.release_value();
}
// Converts a Promise<A> to a Promise<B> using a function func: A -> B
template<typename T>
RefPtr<Promise<T>> map(T func(Result&))
{
RefPtr<Promise<T>> new_promise = Promise<T>::construct();
on_resolved = [new_promise, func](Result& result) {
auto t = func(result);
new_promise->resolve(move(t));
};
return new_promise;
}
private:
Promise() = default;
Optional<ErrorOr<Result>> m_pending_or_error;
};
}