#pragma once #include #include #include #include #include #include #include #include #include namespace hex { class TaskHolder; class TaskManager; class Task { public: Task() = default; Task(std::string unlocalizedName, u64 maxValue, std::function function); Task(const Task&) = delete; Task(Task &&other) noexcept; ~Task(); void update(u64 value = 0); void setMaxValue(u64 value); [[nodiscard]] bool isFinished() const; [[nodiscard]] bool hadException() const; [[nodiscard]] bool wasInterrupted() const; void clearException(); [[nodiscard]] const std::string &getUnlocalizedName(); [[nodiscard]] u64 getValue() const; [[nodiscard]] u64 getMaxValue() const; void interrupt(); private: void finish(); void interruption(); void exception(); private: mutable std::mutex m_mutex; std::string m_unlocalizedName; u64 m_currValue, m_maxValue; std::thread m_thread; bool m_shouldInterrupt = false; bool m_interrupted = false; bool m_finished = false; bool m_hadException = false; struct TaskInterruptor { virtual ~TaskInterruptor() = default; }; friend class TaskHolder; friend class TaskManager; }; class TaskHolder { public: TaskHolder() = default; explicit TaskHolder(std::weak_ptr task) : m_task(std::move(task)) { } [[nodiscard]] bool isRunning() const; [[nodiscard]] bool hadException() const; [[nodiscard]] bool wasInterrupted() const; void interrupt(); private: std::weak_ptr m_task; }; class TaskManager { public: TaskManager() = delete; constexpr static auto NoProgress = 0; static TaskHolder createTask(std::string name, u64 maxValue, std::function function); static void collectGarbage(); static size_t getRunningTaskCount(); static std::list> &getRunningTasks(); static void doLater(const std::function &function); static void runDeferredCalls(); private: static std::mutex s_deferredCallsMutex; static std::list> s_tasks; static std::list> s_deferredCalls; }; }