Blame atk/atkrelationset.c

Packit d0bcc1
/* ATK -  Accessibility Toolkit
Packit d0bcc1
 * Copyright 2001 Sun Microsystems Inc.
Packit d0bcc1
 *
Packit d0bcc1
 * This library is free software; you can redistribute it and/or
Packit d0bcc1
 * modify it under the terms of the GNU Lesser General Public
Packit d0bcc1
 * License as published by the Free Software Foundation; either
Packit d0bcc1
 * version 2 of the License, or (at your option) any later version.
Packit d0bcc1
 *
Packit d0bcc1
 * This library is distributed in the hope that it will be useful,
Packit d0bcc1
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
Packit d0bcc1
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
Packit d0bcc1
 * Lesser General Public License for more details.
Packit d0bcc1
 *
Packit d0bcc1
 * You should have received a copy of the GNU Lesser General Public
Packit d0bcc1
 * License along with this library; if not, write to the
Packit d0bcc1
 * Free Software Foundation, Inc., 59 Temple Place - Suite 330,
Packit d0bcc1
 * Boston, MA 02111-1307, USA.
Packit d0bcc1
 */
Packit d0bcc1
Packit d0bcc1
#include "config.h"
Packit d0bcc1
Packit d0bcc1
#include <glib-object.h>
Packit d0bcc1
Packit d0bcc1
#include "atk.h"
Packit d0bcc1
Packit d0bcc1
/**
Packit d0bcc1
 * SECTION:atkrelationset
Packit d0bcc1
 * @Short_description: A set of AtkRelations, normally the set of
Packit d0bcc1
 *  AtkRelations which an AtkObject has.
Packit d0bcc1
 * @Title:AtkRelationSet
Packit d0bcc1
 *
Packit d0bcc1
 * The AtkRelationSet held by an object establishes its relationships
Packit d0bcc1
 * with objects beyond the normal "parent/child" hierarchical
Packit d0bcc1
 * relationships that all user interface objects have.
Packit d0bcc1
 * AtkRelationSets establish whether objects are labelled or
Packit d0bcc1
 * controlled by other components, share group membership with other
Packit d0bcc1
 * components (for instance within a radio-button group), or share
Packit d0bcc1
 * content which "flows" between them, among other types of possible
Packit d0bcc1
 * relationships.
Packit d0bcc1
 */
Packit d0bcc1
Packit d0bcc1
static gpointer parent_class = NULL;
Packit d0bcc1
Packit d0bcc1
static void atk_relation_set_class_init (AtkRelationSetClass  *klass);
Packit d0bcc1
static void atk_relation_set_finalize   (GObject              *object);
Packit d0bcc1
Packit d0bcc1
GType
Packit d0bcc1
atk_relation_set_get_type (void)
Packit d0bcc1
{
Packit d0bcc1
  static GType type = 0;
Packit d0bcc1
Packit d0bcc1
  if (!type)
Packit d0bcc1
    {
Packit d0bcc1
      static const GTypeInfo typeInfo =
Packit d0bcc1
      {
Packit d0bcc1
        sizeof (AtkRelationSetClass),
Packit d0bcc1
        (GBaseInitFunc) NULL,
Packit d0bcc1
        (GBaseFinalizeFunc) NULL,
Packit d0bcc1
        (GClassInitFunc) atk_relation_set_class_init,
Packit d0bcc1
        (GClassFinalizeFunc) NULL,
Packit d0bcc1
        NULL,
Packit d0bcc1
        sizeof (AtkRelationSet),
Packit d0bcc1
        0,
Packit d0bcc1
        (GInstanceInitFunc) NULL,
Packit d0bcc1
      } ;
Packit d0bcc1
      type = g_type_register_static (G_TYPE_OBJECT, "AtkRelationSet", &typeInfo, 0) ;
Packit d0bcc1
    }
Packit d0bcc1
  return type;
Packit d0bcc1
}
Packit d0bcc1
Packit d0bcc1
static void
Packit d0bcc1
atk_relation_set_class_init (AtkRelationSetClass *klass)
Packit d0bcc1
{
Packit d0bcc1
  GObjectClass *gobject_class = G_OBJECT_CLASS (klass);
Packit d0bcc1
Packit d0bcc1
  parent_class = g_type_class_peek_parent (klass);
Packit d0bcc1
Packit d0bcc1
  gobject_class->finalize = atk_relation_set_finalize;
Packit d0bcc1
}
Packit d0bcc1
Packit d0bcc1
/**
Packit d0bcc1
 * atk_relation_set_new:
Packit d0bcc1
 * 
Packit d0bcc1
 * Creates a new empty relation set.
Packit d0bcc1
 * 
Packit d0bcc1
 * Returns: a new #AtkRelationSet 
Packit d0bcc1
 **/
Packit d0bcc1
AtkRelationSet*
Packit d0bcc1
atk_relation_set_new (void)
Packit d0bcc1
{
Packit d0bcc1
  AtkRelationSet *relation_set;
Packit d0bcc1
Packit d0bcc1
  relation_set = g_object_new (ATK_TYPE_RELATION_SET, NULL);
Packit d0bcc1
  return relation_set;
Packit d0bcc1
}
Packit d0bcc1
Packit d0bcc1
/**
Packit d0bcc1
 * atk_relation_set_contains:
Packit d0bcc1
 * @set: an #AtkRelationSet
Packit d0bcc1
 * @relationship: an #AtkRelationType
Packit d0bcc1
 *
Packit d0bcc1
 * Determines whether the relation set contains a relation that matches the
Packit d0bcc1
 * specified type.
Packit d0bcc1
 *
Packit d0bcc1
 * Returns: %TRUE if @relationship is the relationship type of a relation
Packit d0bcc1
 * in @set, %FALSE otherwise
Packit d0bcc1
 **/
Packit d0bcc1
gboolean
Packit d0bcc1
atk_relation_set_contains (AtkRelationSet   *set,
Packit d0bcc1
                           AtkRelationType  relationship)
Packit d0bcc1
{
Packit d0bcc1
  GPtrArray *array_item;
Packit d0bcc1
  AtkRelation *item;
Packit d0bcc1
  gint  i;
Packit d0bcc1
Packit d0bcc1
  g_return_val_if_fail (ATK_IS_RELATION_SET (set), FALSE);
Packit d0bcc1
Packit d0bcc1
  array_item = set->relations;
Packit d0bcc1
  if (array_item == NULL)
Packit d0bcc1
    return FALSE;
Packit d0bcc1
  for (i = 0; i < array_item->len; i++)
Packit d0bcc1
  {
Packit d0bcc1
    item = g_ptr_array_index (array_item, i);
Packit d0bcc1
    if (item->relationship == relationship)
Packit d0bcc1
      return TRUE;
Packit d0bcc1
  }
Packit d0bcc1
  return FALSE;
Packit d0bcc1
}
Packit d0bcc1
Packit d0bcc1
/**
Packit d0bcc1
 * atk_relation_set_remove:
Packit d0bcc1
 * @set: an #AtkRelationSet
Packit d0bcc1
 * @relation: an #AtkRelation
Packit d0bcc1
 *
Packit d0bcc1
 * Removes a relation from the relation set.
Packit d0bcc1
 * This function unref's the #AtkRelation so it will be deleted unless there
Packit d0bcc1
 * is another reference to it.
Packit d0bcc1
 **/
Packit d0bcc1
void
Packit d0bcc1
atk_relation_set_remove (AtkRelationSet *set,
Packit d0bcc1
                         AtkRelation    *relation)
Packit d0bcc1
{
Packit d0bcc1
  GPtrArray *array_item;
Packit d0bcc1
  AtkRelationType relationship;
Packit d0bcc1
Packit d0bcc1
  g_return_if_fail (ATK_IS_RELATION_SET (set));
Packit d0bcc1
Packit d0bcc1
  array_item = set->relations;
Packit d0bcc1
  if (array_item == NULL)
Packit d0bcc1
    return;
Packit d0bcc1
Packit d0bcc1
  if (g_ptr_array_remove (array_item, relation))
Packit d0bcc1
  {
Packit d0bcc1
    g_object_unref (relation);
Packit d0bcc1
  }
Packit d0bcc1
  else
Packit d0bcc1
  {
Packit d0bcc1
    relationship = atk_relation_get_relation_type (relation);
Packit d0bcc1
    if (atk_relation_set_contains (set, relationship))
Packit d0bcc1
    {
Packit d0bcc1
      AtkRelation *exist_relation;
Packit d0bcc1
      gint i;
Packit d0bcc1
      exist_relation = atk_relation_set_get_relation_by_type (set, relationship);
Packit d0bcc1
      for (i = 0; i < relation->target->len; i++)
Packit d0bcc1
      {
Packit d0bcc1
        AtkObject *target = g_ptr_array_index(relation->target, i);
Packit d0bcc1
        atk_relation_remove_target (exist_relation, target);
Packit d0bcc1
      }
Packit d0bcc1
    }
Packit d0bcc1
  }
Packit d0bcc1
}
Packit d0bcc1
Packit d0bcc1
/**
Packit d0bcc1
 * atk_relation_set_add:
Packit d0bcc1
 * @set: an #AtkRelationSet
Packit d0bcc1
 * @relation: an #AtkRelation
Packit d0bcc1
 *
Packit d0bcc1
 * Add a new relation to the current relation set if it is not already
Packit d0bcc1
 * present.
Packit d0bcc1
 * This function ref's the AtkRelation so the caller of this function
Packit d0bcc1
 * should unref it to ensure that it will be destroyed when the AtkRelationSet
Packit d0bcc1
 * is destroyed.
Packit d0bcc1
 **/
Packit d0bcc1
void
Packit d0bcc1
atk_relation_set_add (AtkRelationSet *set,
Packit d0bcc1
                      AtkRelation    *relation)
Packit d0bcc1
{
Packit d0bcc1
  AtkRelationType relationship;
Packit d0bcc1
Packit d0bcc1
  g_return_if_fail (ATK_IS_RELATION_SET (set));
Packit d0bcc1
  g_return_if_fail (relation != NULL);
Packit d0bcc1
Packit d0bcc1
  if (set->relations == NULL)
Packit d0bcc1
  {
Packit d0bcc1
    set->relations = g_ptr_array_new ();
Packit d0bcc1
  }
Packit d0bcc1
Packit d0bcc1
  relationship = atk_relation_get_relation_type (relation);
Packit d0bcc1
  if (!atk_relation_set_contains (set, relationship))
Packit d0bcc1
  {
Packit d0bcc1
    g_ptr_array_add (set->relations, relation);
Packit d0bcc1
    g_object_ref (relation);
Packit d0bcc1
  }
Packit d0bcc1
  else
Packit d0bcc1
  {
Packit d0bcc1
    AtkRelation *exist_relation;
Packit d0bcc1
    gint i;
Packit d0bcc1
    exist_relation = atk_relation_set_get_relation_by_type (set, relationship);
Packit d0bcc1
    for (i = 0; i < relation->target->len; i++)
Packit d0bcc1
    {
Packit d0bcc1
      AtkObject *target = g_ptr_array_index(relation->target, i);
Packit d0bcc1
      atk_relation_add_target (exist_relation, target); 
Packit d0bcc1
    }
Packit d0bcc1
  }
Packit d0bcc1
}
Packit d0bcc1
Packit d0bcc1
/**
Packit d0bcc1
 * atk_relation_set_get_n_relations:
Packit d0bcc1
 * @set: an #AtkRelationSet
Packit d0bcc1
 *
Packit d0bcc1
 * Determines the number of relations in a relation set.
Packit d0bcc1
 *
Packit d0bcc1
 * Returns: an integer representing the number of relations in the set.
Packit d0bcc1
 **/
Packit d0bcc1
gint
Packit d0bcc1
atk_relation_set_get_n_relations (AtkRelationSet *set)
Packit d0bcc1
{
Packit d0bcc1
  g_return_val_if_fail (ATK_IS_RELATION_SET (set), 0);
Packit d0bcc1
Packit d0bcc1
  if (set->relations == NULL)
Packit d0bcc1
    return 0;
Packit d0bcc1
Packit d0bcc1
  return set->relations->len;
Packit d0bcc1
}
Packit d0bcc1
Packit d0bcc1
/**
Packit d0bcc1
 * atk_relation_set_get_relation:
Packit d0bcc1
 * @set: an #AtkRelationSet
Packit d0bcc1
 * @i: a gint representing a position in the set, starting from 0.
Packit d0bcc1
 *
Packit d0bcc1
 * Determines the relation at the specified position in the relation set.
Packit d0bcc1
 *
Packit d0bcc1
 * Returns: (transfer none): a #AtkRelation, which is the relation at
Packit d0bcc1
 * position i in the set.
Packit d0bcc1
 **/
Packit d0bcc1
AtkRelation*
Packit d0bcc1
atk_relation_set_get_relation (AtkRelationSet *set,
Packit d0bcc1
                               gint           i)
Packit d0bcc1
{
Packit d0bcc1
  GPtrArray *array_item;
Packit d0bcc1
  AtkRelation* item;
Packit d0bcc1
Packit d0bcc1
  g_return_val_if_fail (ATK_IS_RELATION_SET (set), NULL);
Packit d0bcc1
  g_return_val_if_fail (i >= 0, NULL);
Packit d0bcc1
Packit d0bcc1
  array_item = set->relations;
Packit d0bcc1
  if (array_item == NULL)
Packit d0bcc1
    return NULL;
Packit d0bcc1
  item = g_ptr_array_index (array_item, i);
Packit d0bcc1
  if (item == NULL)
Packit d0bcc1
    return NULL;
Packit d0bcc1
Packit d0bcc1
  return item;
Packit d0bcc1
}
Packit d0bcc1
Packit d0bcc1
/**
Packit d0bcc1
 * atk_relation_set_get_relation_by_type:
Packit d0bcc1
 * @set: an #AtkRelationSet
Packit d0bcc1
 * @relationship: an #AtkRelationType
Packit d0bcc1
 *
Packit d0bcc1
 * Finds a relation that matches the specified type.
Packit d0bcc1
 *
Packit d0bcc1
 * Returns: (transfer none): an #AtkRelation, which is a relation matching the
Packit d0bcc1
 * specified type.
Packit d0bcc1
 **/
Packit d0bcc1
AtkRelation*
Packit d0bcc1
atk_relation_set_get_relation_by_type (AtkRelationSet  *set,
Packit d0bcc1
                                       AtkRelationType relationship)
Packit d0bcc1
{
Packit d0bcc1
  GPtrArray *array_item;
Packit d0bcc1
  AtkRelation *item;
Packit d0bcc1
  gint i;
Packit d0bcc1
Packit d0bcc1
  g_return_val_if_fail (ATK_IS_RELATION_SET (set), NULL);
Packit d0bcc1
Packit d0bcc1
  array_item = set->relations;
Packit d0bcc1
  if (array_item == NULL)
Packit d0bcc1
    return NULL;
Packit d0bcc1
  for (i = 0; i < array_item->len; i++)
Packit d0bcc1
  {
Packit d0bcc1
    item = g_ptr_array_index (array_item, i);
Packit d0bcc1
    if (item->relationship == relationship)
Packit d0bcc1
      return item;
Packit d0bcc1
  }
Packit d0bcc1
  return NULL;
Packit d0bcc1
}
Packit d0bcc1
Packit d0bcc1
static void
Packit d0bcc1
atk_relation_set_finalize (GObject *object)
Packit d0bcc1
{
Packit d0bcc1
  AtkRelationSet     *relation_set;
Packit d0bcc1
  GPtrArray             *array;
Packit d0bcc1
  gint               i;
Packit d0bcc1
Packit d0bcc1
  g_return_if_fail (ATK_IS_RELATION_SET (object));
Packit d0bcc1
Packit d0bcc1
  relation_set = ATK_RELATION_SET (object);
Packit d0bcc1
  array = relation_set->relations;
Packit d0bcc1
Packit d0bcc1
  if (array)
Packit d0bcc1
  {
Packit d0bcc1
    for (i = 0; i < array->len; i++)
Packit d0bcc1
    {
Packit d0bcc1
      g_object_unref (g_ptr_array_index (array, i));
Packit d0bcc1
    }
Packit d0bcc1
    g_ptr_array_free (array, TRUE);
Packit d0bcc1
  }
Packit d0bcc1
Packit d0bcc1
  G_OBJECT_CLASS (parent_class)->finalize (object);
Packit d0bcc1
}
Packit d0bcc1
Packit d0bcc1
/**
Packit d0bcc1
 * atk_relation_set_add_relation_by_type:
Packit d0bcc1
 * @set: an #AtkRelationSet
Packit d0bcc1
 * @relationship: an #AtkRelationType
Packit d0bcc1
 * @target: an #AtkObject
Packit d0bcc1
 *
Packit d0bcc1
 * Add a new relation of the specified type with the specified target to 
Packit d0bcc1
 * the current relation set if the relation set does not contain a relation
Packit d0bcc1
 * of that type. If it is does contain a relation of that typea the target
Packit d0bcc1
 * is added to the relation.
Packit d0bcc1
 *
Packit d0bcc1
 * Since: 1.9
Packit d0bcc1
 **/
Packit d0bcc1
void
Packit d0bcc1
atk_relation_set_add_relation_by_type (AtkRelationSet  *set,
Packit d0bcc1
                                       AtkRelationType relationship,
Packit d0bcc1
                                       AtkObject       *target)
Packit d0bcc1
{
Packit d0bcc1
  AtkRelation *relation;
Packit d0bcc1
Packit d0bcc1
  g_return_if_fail (ATK_IS_RELATION_SET (set));
Packit d0bcc1
  g_return_if_fail (ATK_IS_OBJECT (target));
Packit d0bcc1
Packit d0bcc1
  relation = atk_relation_set_get_relation_by_type (set,
Packit d0bcc1
                                                    relationship);
Packit d0bcc1
  if (relation)
Packit d0bcc1
    {
Packit d0bcc1
      atk_relation_add_target (relation, target);
Packit d0bcc1
    } 
Packit d0bcc1
  else 
Packit d0bcc1
    {
Packit d0bcc1
      /* the relation hasn't been created yet ... */
Packit d0bcc1
      relation = atk_relation_new (&target, 1, relationship);
Packit d0bcc1
      atk_relation_set_add (set, relation);
Packit d0bcc1
      g_object_unref(relation);
Packit d0bcc1
    }
Packit d0bcc1
}
Packit d0bcc1
Packit d0bcc1
/**
Packit d0bcc1
 * atk_relation_set_contains_target:
Packit d0bcc1
 * @set: an #AtkRelationSet
Packit d0bcc1
 * @relationship: an #AtkRelationType
Packit d0bcc1
 * @target: an #AtkObject
Packit d0bcc1
 *
Packit d0bcc1
 * Determines whether the relation set contains a relation that
Packit d0bcc1
 * matches the specified pair formed by type @relationship and object
Packit d0bcc1
 * @target.
Packit d0bcc1
 *
Packit d0bcc1
 * Returns: %TRUE if @set contains a relation with the relationship
Packit d0bcc1
 * type @relationship with an object @target, %FALSE otherwise
Packit d0bcc1
 **/
Packit d0bcc1
Packit d0bcc1
gboolean
Packit d0bcc1
atk_relation_set_contains_target (AtkRelationSet  *set,
Packit d0bcc1
                                  AtkRelationType relationship,
Packit d0bcc1
                                  AtkObject       *target)
Packit d0bcc1
{
Packit d0bcc1
  GPtrArray *array_relations;
Packit d0bcc1
  GPtrArray *array_target;
Packit d0bcc1
  AtkObject *current_target;
Packit d0bcc1
  AtkRelation *relation;
Packit d0bcc1
  gint i;
Packit d0bcc1
  gint c;
Packit d0bcc1
Packit d0bcc1
  g_return_val_if_fail (ATK_IS_RELATION_SET (set), FALSE);
Packit d0bcc1
  g_return_val_if_fail (ATK_IS_OBJECT (target), FALSE);
Packit d0bcc1
Packit d0bcc1
  array_relations = set->relations;
Packit d0bcc1
  if (array_relations == NULL)
Packit d0bcc1
    return FALSE;
Packit d0bcc1
Packit d0bcc1
  for (i = 0; i < array_relations->len; i++)
Packit d0bcc1
  {
Packit d0bcc1
    relation = g_ptr_array_index (array_relations, i);
Packit d0bcc1
    if (relation->relationship == relationship)
Packit d0bcc1
      {
Packit d0bcc1
        array_target = atk_relation_get_target (relation);
Packit d0bcc1
        for (c = 0; c < array_target->len; c++)
Packit d0bcc1
          {
Packit d0bcc1
            current_target = g_ptr_array_index (array_target, c);
Packit d0bcc1
            if (target == current_target)
Packit d0bcc1
              return TRUE;
Packit d0bcc1
          }
Packit d0bcc1
      }
Packit d0bcc1
  }
Packit d0bcc1
Packit d0bcc1
  return FALSE;
Packit d0bcc1
}