medit/moo/moocpp/gobjptr.h
2016-01-31 00:03:05 -08:00

314 lines
7.5 KiB
C++

/*
* moocpp/gobjptr.h
*
* Copyright (C) 2004-2016 by Yevgen Muntyan <emuntyan@users.sourceforge.net>
*
* This file is part of medit. medit is free software; you can
* redistribute it and/or modify it under the terms of the
* GNU Lesser General Public License as published by the
* Free Software Foundation; either version 2.1 of the License,
* or (at your option) any later version.
*
* You should have received a copy of the GNU Lesser General Public
* License along with medit. If not, see <http://www.gnu.org/licenses/>.
*/
#pragma once
#include "moocpp/gobjrawptr.h"
#include "moocpp/grefptr.h"
#include "moocpp/gobjectutils.h"
namespace moo {
///////////////////////////////////////////////////////////////////////////////////////////
//
// gobj_ptr
//
template<typename Object>
class gobj_ptr
{
using ref_type = gobj_ref<Object>;
static_assert(gobjinfo<Object>::is_gobject, "Not a GObject");
public:
gobj_ptr() {}
gobj_ptr(Object* obj, ref_transfer policy)
{
assign(obj, policy);
}
gobj_ptr(const nullptr_t&)
: gobj_ptr()
{
}
static gobj_ptr wrap_new(Object* obj)
{
return gobj_ptr(obj, ref_transfer::take_ownership);
}
static gobj_ptr wrap(Object* obj)
{
return gobj_ptr(obj, ref_transfer::make_copy);
}
~gobj_ptr()
{
reset();
}
void ref(Object* obj)
{
assign(obj, ref_transfer::make_copy);
}
void set(Object* obj)
{
assign(obj, ref_transfer::make_copy);
}
void set_new(Object* obj)
{
assign(obj, ref_transfer::take_ownership);
}
Object* release()
{
auto* tmp = gobj();
m_ref._set_gobj(nullptr);
return tmp;
}
void reset()
{
auto* tmp = gobj();
if (tmp)
{
m_ref._set_gobj(nullptr);
g_object_unref(tmp);
}
}
// Implicit conversion to non-const Object* is dangerous because there is a lot
// of code still which frees/steals objects directly. For example:
// FooObject* tmp = x->s;
// x->s = NULL;
// g_object_unref (tmp);
operator const Object* () const { return gobj(); }
operator ref_type*() const { return m_ref.self(); }
ref_type* operator->() const { return m_ref.self(); }
ref_type& operator*() { return m_ref; }
const ref_type& operator*() const { return m_ref; }
// These are nasty. Because of ref_type* conversion this can be converted to void*,
// which in turn can be passed to g_object_ref or g_free, etc.
operator void*() const = delete;
operator const void*() const = delete;
Object* gobj() const { return m_ref.gobj(); }
Object** pp() { return m_ref._pp(); }
template<typename Super>
Super* gobj() const
{
return gobj_is_subclass<Object, Super>::down_cast(m_ref.gobj());
}
template<typename Super>
operator const Super* () const { return gobj<Super>(); }
template<typename Super>
operator gobj_ptr<Super> () const { return gobj_ptr<Super>::wrap (gobj<Super> ()); }
operator bool() const { return gobj() != nullptr; }
bool operator!() const { return gobj() == nullptr; }
gobj_ptr(const gobj_ptr& other)
: gobj_ptr()
{
ref(other.gobj());
}
gobj_ptr& operator=(const gobj_ptr& other)
{
ref(other.gobj());
return *this;
}
gobj_ptr(gobj_ptr&& other)
: gobj_ptr()
{
m_ref._set_gobj(other.gobj());
other.m_ref._set_gobj(nullptr);
}
gobj_ptr& operator=(const nullptr_t&)
{
reset();
return *this;
}
gobj_ptr& operator=(gobj_ptr&& other)
{
if (gobj() != other.gobj())
{
assign(other.gobj(), ref_transfer::take_ownership);
other.m_ref._set_gobj(nullptr);
}
return *this;
}
private:
void assign(Object* obj, ref_transfer policy)
{
g_assert(!obj || G_IS_OBJECT(obj));
if (gobj() != obj)
{
Object* tmp = gobj();
m_ref._set_gobj(obj);
if (obj)
{
if (policy == ref_transfer::make_copy)
g_object_ref(obj);
else if (g_object_is_floating(obj))
g_object_ref_sink(obj);
}
if (tmp)
g_object_unref(tmp);
}
}
private:
mutable gobj_ref<Object> m_ref;
};
template<typename Object>
inline gobj_ptr<Object> wrap_new(Object *obj)
{
return gobj_ptr<Object>::wrap_new(obj);
}
template<typename Object>
inline gobj_ptr<Object> wrap(Object* obj)
{
return gobj_ptr<Object>::wrap(obj);
}
template<typename Object>
inline gobj_ptr<Object> wrap(const gobj_raw_ptr<Object>& obj)
{
return gobj_ptr<Object>::wrap(obj);
}
template<typename Object>
inline gobj_ref<Object> wrap (Object& obj)
{
return *wrap (&obj);
}
template<typename T, typename ...Args>
inline gobj_ptr<T> create_gobj(GType obj_type, Args&& ...args)
{
return wrap_new(reinterpret_cast<T*>(g_object_new(obj_type, std::forward<Args>(args)...)));
}
template<typename T, typename ...Args>
inline gobj_ptr<T> create_gobj(Args&& ...args)
{
// object_g_type() will produce a compiler error if the type wasn't registered
return create_gobj(gobjinfo<T>::object_g_type(), std::forward<Args>(args)...);
}
template<typename T>
inline gobj_ptr<T> create_gobj()
{
// object_g_type() will produce a compiler error if the type wasn't registered
return create_gobj<T>(gobjinfo<T>::object_g_type(), nullptr);
}
template<typename Object, typename Super>
Object* up_cast (Super* o)
{
return G_TYPE_CHECK_INSTANCE_CAST ((o), gobjinfo<Object>::object_g_type(), Object);
}
} // namespace moo
template<typename X>
void g_object_unref(const moo::gobj_ptr<X>&) = delete;
template<typename X>
void g_free(const moo::gobj_ptr<X>&) = delete;
template<typename X>
inline bool operator==(const moo::gobj_ptr<X>& p, const nullptr_t&)
{
return p.gobj() == nullptr;
}
template<typename X>
inline bool operator==(const nullptr_t&, const moo::gobj_ptr<X>& p)
{
return p.gobj() == nullptr;
}
template<typename X, typename Y>
inline bool operator==(const moo::gobj_ptr<X>& p1, const moo::gobj_ptr<Y>& p2)
{
return p1.gobj() == p2.gobj();
}
template<typename X, typename Y>
inline bool operator==(const moo::gobj_ptr<X>& p1, const Y* p2)
{
return p1.gobj() == p2;
}
template<typename X, typename Y>
inline bool operator==(const X* p1, const moo::gobj_ptr<Y>& p2)
{
return p1 == p2.gobj();
}
template<typename X, typename Y>
inline bool operator==(const moo::gobj_ptr<X>& p1, const moo::gobj_raw_ptr<Y>& p2)
{
return p1.gobj() == p2.gobj();
}
template<typename X, typename Y>
inline bool operator==(const moo::gobj_raw_ptr<Y>& p1, const moo::gobj_ptr<X>& p2)
{
return p1.gobj() == p2.gobj();
}
template<typename X, typename Y>
bool operator!=(const moo::gobj_ptr<X>& p1, const moo::gobj_ptr<Y>& p2)
{
return !(p1 == p2);
}
template<typename X, typename Y>
bool operator!=(const moo::gobj_ptr<X>& p1, const Y& p2)
{
return !(p1 == p2);
}
template<typename X, typename Y>
bool operator!=(const X& p1, const moo::gobj_ptr<Y>& p2)
{
return !(p1 == p2);
}
template<typename X> bool operator==(const moo::gobj_ptr<X>& p1, int) = delete;
template<typename X> bool operator==(int, const moo::gobj_ptr<X>& p2) = delete;
template<typename X> bool operator!=(const moo::gobj_ptr<X>& p1, int) = delete;
template<typename X> bool operator!=(int, const moo::gobj_ptr<X>& p2) = delete;