mirror of
https://github.com/Chatterino/chatterino2.git
synced 2024-11-13 19:49:51 +01:00
0e66b17ff0
Also changes the way timeouts happen, since right now if a timeout was met (which it mostly wasn't), it would run the error callback twice causing potentially undefined behaviour
86 lines
1.5 KiB
C++
86 lines
1.5 KiB
C++
#pragma once
|
|
|
|
#include <QApplication>
|
|
#include <QObject>
|
|
#include <type_traits>
|
|
|
|
namespace chatterino {
|
|
/// Holds a pointer to a QObject and resets it to nullptr if the QObject
|
|
/// gets destroyed.
|
|
template <typename T>
|
|
class QObjectRef
|
|
{
|
|
public:
|
|
QObjectRef()
|
|
{
|
|
static_assert(std::is_base_of_v<QObject, T>);
|
|
}
|
|
|
|
explicit QObjectRef(T *t)
|
|
{
|
|
static_assert(std::is_base_of_v<QObject, T>);
|
|
|
|
this->set(t);
|
|
}
|
|
|
|
QObjectRef(const QObjectRef &other)
|
|
{
|
|
this->set(other.t_);
|
|
}
|
|
|
|
~QObjectRef()
|
|
{
|
|
this->set(nullptr);
|
|
}
|
|
|
|
QObjectRef &operator=(T *t)
|
|
{
|
|
this->set(t);
|
|
|
|
return *this;
|
|
}
|
|
|
|
operator bool()
|
|
{
|
|
return t_;
|
|
}
|
|
|
|
T *operator->()
|
|
{
|
|
return t_;
|
|
}
|
|
|
|
T *get()
|
|
{
|
|
return t_;
|
|
}
|
|
|
|
private:
|
|
void set(T *other)
|
|
{
|
|
// old
|
|
if (this->conn_)
|
|
{
|
|
QObject::disconnect(this->conn_);
|
|
}
|
|
|
|
// new
|
|
if (other)
|
|
{
|
|
// the cast here should absolutely not be necessary, but gcc still requires it
|
|
this->conn_ =
|
|
QObject::connect((QObject *)other, &QObject::destroyed, qApp,
|
|
[this](QObject *) {
|
|
this->set(nullptr);
|
|
},
|
|
Qt::DirectConnection);
|
|
}
|
|
|
|
this->t_ = other;
|
|
}
|
|
|
|
std::atomic<T *> t_{};
|
|
QMetaObject::Connection conn_;
|
|
};
|
|
} // namespace chatterino
|