#! /usr/bin/perl -w
#
# Class name: MinorAPIElement
# Synopsis: Class for parameters and members of structs, etc.
#
# Last Updated: $Date: 2011/05/26 14:13:20 $
#
# Copyright (c) 1999-2008 Apple Computer, Inc. All rights reserved.
#
# @APPLE_LICENSE_HEADER_START@
#
# This file contains Original Code and/or Modifications of Original Code
# as defined in and that are subject to the Apple Public Source License
# Version 2.0 (the 'License'). You may not use this file except in
# compliance with the License. Please obtain a copy of the License at
# http://www.opensource.apple.com/apsl/ and read it before using this
# file.
#
# The Original Code and all software distributed under the License are
# distributed on an 'AS IS' basis, WITHOUT WARRANTY OF ANY KIND, EITHER
# EXPRESS OR IMPLIED, AND APPLE HEREBY DISCLAIMS ALL SUCH WARRANTIES,
# INCLUDING WITHOUT LIMITATION, ANY WARRANTIES OF MERCHANTABILITY,
# FITNESS FOR A PARTICULAR PURPOSE, QUIET ENJOYMENT OR NON-INFRINGEMENT.
# Please see the License for the specific language governing rights and
# limitations under the License.
#
# @APPLE_LICENSE_HEADER_END@
#
######################################################################
# /*! @header
# @abstract
# MinorAPIElement class package file.
# @discussion
# This file contains the MinorAPIElement class, a class for content
# relating to an enumeration declaration.
#
# For details, see the class documentation below.
# @indexgroup HeaderDoc API Objects
# */
# /*!
# @abstract
# API object that describes an API element that is documented only
# as part of another API element.
# @discussion
# The MinorAPIElement class stores information relating to a field
# in a structure or type declaration, a constant in an enumeration,
# or a parameter (parsed or tagged) to a function or callback typedef.
#
# This class is a subclass of
# {@link //apple_ref/perl/cl/HeaderDoc::HeaderElement HeaderElement}.
# The majority of related fields and functions can be found there.
#
# @var POSITION
# The position of this element in the declaration.
# See {@link position}.
# @var TYPE
# The type of minor API element. See
# {@link //apple_ref/perl/instm/HeaderDoc::MinorAPIElement/type//() type}.
# @var USERDICTARRAY
# A bunch of key-value pairs in sorted order.
# See {@link addToUserDictArray}.
# @var USERDICT
# A bunch of arbitrary key-value pairs in sorted order.
# See {@link addKeyAndValueInUserDict}.
# @var HIDDEN
# Indicates that this should not be emitted in HTML. Used
# in inherited bits.
# @var AUTODECLARATION
# A flag that is set to 1 for certain special declarations
# (Perl instance variables or other variables created by the
# \@var tag in class markup) to indicate that the discussion
# functions should return a synthesized declaration rather
# than anything from the parse tree.
# @var TAGNAME
# The label for an Objective-C method parameter.
# */
package HeaderDoc::MinorAPIElement;
use HeaderDoc::Utilities qw(findRelativePath safeName printArray printHash registerUID);
use HeaderDoc::HeaderElement;
use Carp qw(cluck);
@ISA = qw( HeaderDoc::HeaderElement );
use strict;
use vars qw($VERSION @ISA);
# /*!
# @abstract
# The revision control revision number for this module.
# @discussion
# In the git repository, contains the number of seconds since
# January 1, 1970.
# */
$HeaderDoc::MinorAPIElement::VERSION = '$Revision: 1306444400 $';
# /*!
# @abstract
# Initializes an instance of a MinorAPIElement object.
# @param self
# The object to initialize.
# */
sub _initialize {
my($self) = shift;
$self->SUPER::_initialize();
# $self->{POSITION} = undef;
# $self->{TYPE} = undef;
$self->{USERDICTARRAY} = ();
# $self->{HIDDEN} = 0;
$self->{CLASS} = "HeaderDoc::MinorAPIElement";
}
# /*!
# @abstract
# Duplicates this MinorAPIElement object into another one.
# @param self
# The object to clone.
# @param clone
# The victim object.
# */
sub clone {
my $self = shift;
my $clone = undef;
if (@_) {
$clone = shift;
} else {
$clone = HeaderDoc::MinorAPIElement->new("LANG" => $self->{LANG}, "SUBLANG" => $self->{SUBLANG});
}
$self->SUPER::clone($clone);
# now clone stuff specific to function
$clone->{POSITION} = $self->{POSITION};
$clone->{TYPE} = $self->{TYPE};
$clone->{HIDDEN} = $self->{HIDDEN};
$clone->{USERDICTARRAY} = $self->{USERDICTARRAY};
return $clone;
}
# /*!
# @abstract
# Gets/sets the Objective-C label for a parameter.
# @param self
# This MinorAPIElement object.
# @param TAGNAME
# The new name. (Optional.)
# */
sub tagname {
my $self = shift;
if (@_) {
$self->{TAGNAME} = shift;
}
return $self->{TAGNAME};
}
# /*!
# @abstract
# Gets/sets the position of this parameter.
# @param self
# This MinorAPIElement object.
# @param POSITION
# The new position. (Optional.)
# @discussion
# The first parameter has position 0, the second
# has position 1, and so on.
# */
sub position {
my $self = shift;
if (@_) {
$self->{POSITION} = shift;
}
return $self->{POSITION};
}
# /*!
# @abstract
# Gets/sets whether this parameter is hidden.
# @param self
# This MinorAPIElement object.
# @param HIDDEN
# The new value. (Optional.)
# @discussion
# Parsed parameters merged in from other places are
# not emitted in the XML output. Set to 1 for
# those parameters. Otherwise, set to 0 or leave
# unset.
# */
sub hidden {
my $self = shift;
if (@_) {
$self->{HIDDEN} = shift;
}
return $self->{HIDDEN};
}
# /*!
# @abstract
# Gets/sets what type of parameter this is.
# @param self
# This MinorAPIElement object.
# @param TYPE
# The new value. (Optional.)
# @discussion
# These values are one of:
#
#
callbackfuncPtrMinorAPIElement object.
# @discussion
# Used for miscellaneous data, such as the parameters
# within a typedef struct of callbacks
# stored as an array to preserve order.
# */
sub userDictArray {
my $self = shift;
if (@_) {
@{ $self->{USERDICTARRAY} } = @_;
}
($self->{USERDICTARRAY}) ? return @{ $self->{USERDICTARRAY} } : return ();
}
# /*!
# @abstract
# Adds values to the user dictionary array (and returns it).
# @param self
# This MinorAPIElement object.
# @param ...
# The objects to add.
# @discussion
# Used for miscellaneous data, such as the parameters
# within a typedef struct of callbacks
# stored as an array to preserve order.
# */
sub addToUserDictArray {
my $self = shift;
if (@_) {
foreach my $item (@_) {
push (@{ $self->{USERDICTARRAY} }, $item);
}
}
return @{ $self->{USERDICTARRAY} };
}
# /*!
# @abstract
# Adds key-value pairs to the user dictionary
# (and returns it).
# @param self
# This MinorAPIElement object.
# @param ...
# The objects to add.
# @discussion
# Used for miscellaneous data.
# */
sub addKeyAndValueInUserDict {
my $self = shift;
if (@_) {
my $key = shift;
my $value = shift;
$self->{USERDICT}{$key} = $value;
}
return %{ $self->{USERDICT} };
}
# /*!
# @abstract
# Prints this object for debugging purposes.
# @param self
# This object.
# */
sub printObject {
my $self = shift;
my $dec = $self->declaration();
$self->SUPER::printObject();
print STDERR "position: $self->{POSITION}\n";
print STDERR "type: $self->{TYPE}\n";
}
# /*!
# @param value
# @abstract
# Sets or gets a state flag.
# @discussion
# The APPLEREFISDOC state flag controls whether to use a
# language-specific or doc-specific apple_ref marker for a
# doc block.
#
# This version overrides the
# {@link //apple_ref/perl/instm/HeaderDoc::HeaderElement/appleRefIsDoc//() appleRefIsDoc}
# method in
# {@link //apple_ref/perl/cl/HeaderDoc::HeaderElement HeaderDoc::HeaderElement}
# because of the complexity of working with minor API
# elements that live in normal (non-API-owner) objects.
# */
sub appleRefIsDoc
{
my $self = shift;
if (@_) {
my $value = shift;
$self->{APPLEREFISDOC} = $value;
}
# print STDERR "ARID: ".$self->{APPLEREFISDOC}." for $self\n";
if ($self->{APPLEREFISDOC}) {
return $self->{APPLEREFISDOC};
} elsif ($self->apiOwner()) {
my $apio = $self->apiOwner();
bless($apio, "HeaderDoc::HeaderElement");
bless($apio, $apio->class());
my $apioval = $apio->appleRefIsDoc();
# print STDERR "APIOVAL: $apioval for $apio (".$apio->name().")\n";
if ($apioval) { return $apioval; }
}
return $self->{APPLEREFISDOC};
}
# /*!
# @abstract
# Sets the group.
# @discussion
# This overrides the declaration in HeaderElement.pm because
# we want the groups for local variables to be separate and
# distinct from normal function/data type/variable groups unless
# the owning object really is an APIOwner subclass.
# */
sub group
{
my $self = shift;
if ($self->apiOwner()->isAPIOwner()) {
return $self->SUPER::group(@_);
}
if (@_) {
$self->{GROUP} = shift;
}
return $self->{GROUP};
}
# /*!
# @abstract
# Gets/sets the legacy text declaration.
# @param self
# This MinorAPIElement object.
# @discussion
# For special declarations (Perl instance variables,
# function-local variables, and so on),
# returns a synthesized declaration from
# {@link //apple_ref/perl/data/HeaderDoc::MinorAPIElement/AUTODECLARATION AUTODECLARATION}.
# Otherwise, hands off the request to the superclass,
# {@link //apple_ref/perl/cl/HeaderDoc::HeaderElement HeaderElement}.
# */
sub declaration
{
my $self = shift;
if ($self->{AUTODECLARATION}) { return $self->{AUTODECLARATION}; }
return $self->SUPER::declaration();
}
# /*!
# @abstract
# Gets the HTML declaration.
# @param self
# This MinorAPIElement object.
# @discussion
# For special declarations (Perl instance variables,
# function-local variables, and so on),
# returns a synthesized declaration from
# {@link //apple_ref/perl/data/HeaderDoc::MinorAPIElement/AUTODECLARATION AUTODECLARATION}.
# Otherwise, hands off the request to the superclass,
# {@link //apple_ref/perl/cl/HeaderDoc::HeaderElement HeaderElement}.
# */
sub declarationInHTML
{
my $self = shift;
if ($self->{AUTODECLARATION}) { return $self->{AUTODECLARATION}; }
return $self->SUPER::declarationInHTML();
}
# /*!
# @abstract
# Gets/sets the
# {@link //apple_ref/perl/data/HeaderDoc::MinorAPIElement/AUTODECLARATION AUTODECLARATION}
# value.
# @param self
# This MinorAPIElement object.
# @param AUTODECLARATION
# The new value to set.
# @discussion
# For special declarations (Perl instance variables,
# function-local variables, and so on), declaration functions
# return a synthesized declaration based on the
# {@link //apple_ref/doc/functionparam/HeaderDoc::MinorAPIElement/autodeclaration/AUTODECLARATION AUTODECLARATION}
# parameter. This function supports that.
# */
sub autodeclaration
{
my $self = shift;
if (@_) {
if ($self->lang() eq "perl") {
$self->{AUTODECLARATION} = "\$self->{".$self->name()."}";
} else {
$self->{AUTODECLARATION} = $self->name();
}
}
return $self->{AUTODECLARATION};
}
# /*!
# @abstract
# Returns a custom API reference for function
# parameters, struct fields, and so on.
# @discussion
# Overrides the function in
# {@link //apple_ref/perl/cl/HeaderDoc::HeaderElement HeaderElement}
# because these minor API elements need lots of special
# treatment. In particular, this function only
# overrides the behavior if the main function does not
# return anything. This is necessary because some other
# data types use the MinorAPIElement class
# in nonstandard ways.
# */
sub apiuid
{
my $self = shift;
my $args = 0;
my $type = "AUTO";
my $paramSignature_or_alt_define_name = "";
if (@_) {
$args = 1;
$type = shift;
if (@_) {
$paramSignature_or_alt_define_name = shift;
}
}
my $ret = "";
if ($args) {
$ret = $self->SUPER::apiuid($type, $paramSignature_or_alt_define_name);
} else {
$ret = $self->SUPER::apiuid();
}
# print STDERR "MINORAPIELEMENT NAME: ".$self->name()." UID: $ret\n";
my $apiUIDPrefix = HeaderDoc::APIOwner->apiUIDPrefix();
if ($ret eq "") {
my $fieldtype = "";
my $apio = $self->apiOwner();
my $apioclass = ref($apio) || $apio;
if (!$apio) {
# We're disposing of the node. Return the last known value.
return $self->{APIUID};
}
my $include_class = 1;
if ($apioclass eq "HeaderDoc::Enum") {
$fieldtype = "enumconstant";
} elsif ($apioclass eq "HeaderDoc::Function") {
$fieldtype = "functionparam";
if ($self->autodeclaration()) {
$fieldtype = "functionvar";
}
} elsif ($apioclass eq "HeaderDoc::Method") {
$fieldtype = "methodparam";
if ($self->autodeclaration()) {
$fieldtype = "methodvar";
}
} elsif ($apioclass eq "HeaderDoc::PDefine") {
$fieldtype = "defineparam";
if ($self->autodeclaration()) {
$fieldtype = "definevar";
}
$include_class = 0;
} elsif ($apioclass eq "HeaderDoc::Struct") {
$fieldtype = "structfield";
} elsif ($apioclass eq "HeaderDoc::Typedef") {
$fieldtype = "typedeffield";
} elsif ($apioclass eq "HeaderDoc::Var") {
$fieldtype = "typedeffield";
}
if ($fieldtype) {
my $apiOwnerOwner = $apio->apiOwner();
my $apiOwnerOwnerClass = ref($apiOwnerOwner) || $apiOwnerOwner;
my $uid = "";
if ((!$include_class) || ($apiOwnerOwnerClass eq "HeaderDoc::Header") || (!$apiOwnerOwner)) {
$uid = $self->genRefSub("doc", $fieldtype, $self->apiuidname, $apio->apiuidname());
} else {
$uid = $self->genRefSub("doc", $fieldtype, $self->apiuidname, $apio->apiuidname(), undef, $apiOwnerOwner->apiuidname);
}
# print STDERR $self->name().": //$apiUIDPrefix/doc/$fieldtype/".$apio->apiuidname()."/".$self->name()." VERSUS $uid\n";
# return "//$apiUIDPrefix/doc/$fieldtype/".$apio->apiuidname()."/".$self->name();
$self->{APIUID} = $uid;
registerUID($uid, $self->rawname(), $self);
return $uid;
}
}
return $ret;
}
1;