gem-graph-client/libide/core/ide-object-box.c

290 lines
7.5 KiB
C

/* ide-object-box.c
*
* Copyright 2018 Christian Hergert <unknown@domain.org>
*
* This program is free software: you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation, either version 3 of the License, or
* (at your option) any later version.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with this program. If not, see <http://www.gnu.org/licenses/>.
*
* SPDX-License-Identifier: GPL-3.0-or-later
*/
#define G_LOG_DOMAIN "ide-object-box"
#include "config.h"
#include "ide-object-box.h"
#include "ide-macros.h"
struct _IdeObjectBox
{
IdeObject parent_instance;
GObject *object;
guint propagate_disposal : 1;
};
G_DEFINE_FINAL_TYPE (IdeObjectBox, ide_object_box, IDE_TYPE_OBJECT)
enum {
PROP_0,
PROP_OBJECT,
PROP_PROPAGATE_DISPOSAL,
N_PROPS
};
static GParamSpec *properties [N_PROPS];
static void
ide_object_box_set_object (IdeObjectBox *self,
GObject *object)
{
g_return_if_fail (IDE_IS_OBJECT_BOX (self));
g_return_if_fail (G_IS_OBJECT (object));
g_return_if_fail (g_object_get_data (object, "IDE_OBJECT_BOX") == NULL);
self->object = g_object_ref (object);
g_object_set_data (self->object, "IDE_OBJECT_BOX", self);
}
/**
* ide_object_box_new:
*
* Create a new #IdeObjectBox.
*
* Returns: (transfer full): a newly created #IdeObjectBox
*
* Since: 3.32
*/
IdeObjectBox *
ide_object_box_new (GObject *object)
{
g_return_val_if_fail (IDE_IS_MAIN_THREAD (), NULL);
return g_object_new (IDE_TYPE_OBJECT_BOX,
"object", object,
NULL);
}
static gchar *
ide_object_box_repr (IdeObject *object)
{
g_autoptr(GObject) obj = ide_object_box_ref_object (IDE_OBJECT_BOX (object));
if (obj != NULL)
return g_strdup_printf ("%s object=\"%s\"",
G_OBJECT_TYPE_NAME (object),
G_OBJECT_TYPE_NAME (obj));
else
return IDE_OBJECT_CLASS (ide_object_box_parent_class)->repr (object);
}
static void
ide_object_box_destroy (IdeObject *object)
{
IdeObjectBox *self = (IdeObjectBox *)object;
g_assert (IDE_IS_MAIN_THREAD ());
g_assert (IDE_IS_OBJECT (self));
g_object_ref (self);
/* Clear the backpointer before any disposal to the object, since that
* will possibly result in the object calling back into this peer object.
*/
if (self->object)
{
g_object_set_data (G_OBJECT (self->object), "IDE_OBJECT_BOX", NULL);
if (self->propagate_disposal)
g_object_run_dispose (G_OBJECT (self->object));
}
IDE_OBJECT_CLASS (ide_object_box_parent_class)->destroy (object);
g_clear_object (&self->object);
g_object_unref (self);
}
static void
ide_object_box_get_property (GObject *object,
guint prop_id,
GValue *value,
GParamSpec *pspec)
{
IdeObjectBox *self = IDE_OBJECT_BOX (object);
switch (prop_id)
{
case PROP_OBJECT:
g_value_take_object (value, ide_object_box_ref_object (self));
break;
case PROP_PROPAGATE_DISPOSAL:
g_value_set_boolean (value, self->propagate_disposal);
break;
default:
G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
}
}
static void
ide_object_box_set_property (GObject *object,
guint prop_id,
const GValue *value,
GParamSpec *pspec)
{
IdeObjectBox *self = IDE_OBJECT_BOX (object);
switch (prop_id)
{
case PROP_OBJECT:
ide_object_box_set_object (self, g_value_get_object (value));
break;
case PROP_PROPAGATE_DISPOSAL:
self->propagate_disposal = g_value_get_boolean (value);
break;
default:
G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
}
}
static void
ide_object_box_class_init (IdeObjectBoxClass *klass)
{
GObjectClass *object_class = G_OBJECT_CLASS (klass);
IdeObjectClass *i_object_class = IDE_OBJECT_CLASS (klass);
object_class->get_property = ide_object_box_get_property;
object_class->set_property = ide_object_box_set_property;
i_object_class->destroy = ide_object_box_destroy;
i_object_class->repr = ide_object_box_repr;
/**
* IdeObjectBox:object:
*
* The "object" property contains the object that is boxed and
* placed onto the object graph using this box.
*
* Since: 3.32
*/
properties [PROP_OBJECT] =
g_param_spec_object ("object",
"Object",
"The boxed object",
G_TYPE_OBJECT,
(G_PARAM_READWRITE | G_PARAM_CONSTRUCT_ONLY | G_PARAM_STATIC_STRINGS));
/**
* IdeObjectBox:propagate-disposal:
*
* The "propagate-disposal" property denotes if the #IdeObject:object
* property contents should have g_object_run_dispose() called when the
* #IdeObjectBox is destroyed.
*
* This is useful when you want to force disposal of an external object
* when @self is removed from the object tree.
*
* Since: 3.32
*/
properties [PROP_PROPAGATE_DISPOSAL] =
g_param_spec_boolean ("propagate-disposal",
"Propagate Disposal",
"If the object should be disposed when the box is destroyed",
TRUE,
(G_PARAM_READWRITE | G_PARAM_CONSTRUCT_ONLY | G_PARAM_STATIC_STRINGS));
g_object_class_install_properties (object_class, N_PROPS, properties);
}
static void
ide_object_box_init (IdeObjectBox *self)
{
self->propagate_disposal = TRUE;
}
/**
* ide_object_box_ref_object:
* @self: an #IdeObjectBox
*
* Gets the boxed object.
*
* Returns: (transfer full) (nullable) (type GObject): a #GObject or %NULL
*
* Since: 3.32
*/
gpointer
ide_object_box_ref_object (IdeObjectBox *self)
{
GObject *ret;
g_return_val_if_fail (IDE_IS_MAIN_THREAD (), NULL);
g_return_val_if_fail (IDE_IS_OBJECT_BOX (self), NULL);
ide_object_lock (IDE_OBJECT (self));
ret = self->object ? g_object_ref (self->object) : NULL;
ide_object_unlock (IDE_OBJECT (self));
return g_steal_pointer (&ret);
}
/**
* ide_object_box_from_object:
* @object: a #GObject
*
* Gets the #IdeObjectBox that contains @object, if any.
*
* This function may only be called from the main thread.
*
* Returns: (transfer none): an #IdeObjectBox
*
* Since: 3.32
*/
IdeObjectBox *
ide_object_box_from_object (GObject *object)
{
g_return_val_if_fail (IDE_IS_MAIN_THREAD (), NULL);
g_return_val_if_fail (G_IS_OBJECT (object), NULL);
return g_object_get_data (G_OBJECT (object), "IDE_OBJECT_BOX");
}
/**
* ide_object_box_contains:
* @self: a #IdeObjectBox
* @instance: (type GObject) (nullable): a #GObject or %NULL
*
* Checks if @self contains @instance.
*
* Returns: %TRUE if #IdeObjectBox:object matches @instance
*
* Since: 3.32
*/
gboolean
ide_object_box_contains (IdeObjectBox *self,
gpointer instance)
{
gboolean ret;
g_return_val_if_fail (IDE_IS_MAIN_THREAD (), FALSE);
g_return_val_if_fail (IDE_IS_OBJECT_BOX (self), FALSE);
ide_object_lock (IDE_OBJECT (self));
ret = (instance == (gpointer)self->object);
ide_object_unlock (IDE_OBJECT (self));
return ret;
}