Unit Attributes

All entities can embed an arbitrary number of generic attributes. These attributes can be accessed using the [generic attributes API](generic_attributes.html).

When a generic attribute is of type A3DFloat, the value can be associated to a complex unit system, such as km/h or m²/sec. This unit system can be expressed using the A3DMiscAttributeUnit structure.

Unit attributes are present in Building Information Modeling (BIM) formats as Metadata. Currently only Revit and IFC model files contain such attributes.

Example Code

This example shows how a simple “kilometers per hour” (km/h) is expressed using unit attributes. The structures are explained in detail below.

// Define the 'kilometer'

A3DMiscAttributeBasicUnitData sKmData;
A3D_INITIALIZE_DATA(A3DMiscAttributeBasicUnitData, sKmData);
sKmData.m_eUnit     = kA3DUnit_Metre;
sKmData.m_iExponent = 1;
sKmData.m_dFactor   = 1000;

// Define the 'per hour'

A3DMiscAttributeBasicUnitData sHData;
A3D_INITIALIZE_DATA(A3DMiscAttributeBasicUnitData, sHData);
sHData.m_eUnit     = kA3DUnit_Hour;
sHData.m_iExponent = -1;
sHData.m_dFactor   = 1;

// Combining to 'km/h'
A3DMiscAttributeBasicUnitData* psData[] = {sKmData, sHData};

A3DMiscAttributeUnitData sKmHData;
A3D_INITIALIZE_DATA(A3DMiscAttributeBasicUnitData, sKmHData);
sKmHData.m_pcName          = "km/h";
sKmHData.m_uiBasicUnitSize = 2;
sKmHData.m_uiBasicUnits    = psData;

The A3DMiscAttributeBasicUnitData Structure

An instance of A3DMiscAttributeBasicUnitData represents an element of a complex unit system (the k or the /h in km/h) it’s structure is:

typedef struct
{
  A3DUns16      m_usStructSize;
  A3DEBasicUnit m_eUnit;
  A3DInt32      m_iExponent;
  A3DDouble     m_dFactor;
} A3DMiscAttributeBasicUnitData;

A3DMiscAttributeBasicUnitData.m_eUnit represents the domain of the unit: meters, hours, kelvins, … It can be any value from the A3DEBasicUnit enum. In our example with km/h the two values we use are ref A3DEBasicUnit “kA3DUnit_Metre” and ref A3DEBasicUnit “kA3DUnit_Hour”.

A3DMiscAttributeBasicUnitData.m_iExponent is used to represent how the basic unit is used in the whole unit system. It’s value must be non-zero. In km/h the exponent for the kilometers is 1 and the value for hour is -1. An other common value is 3, such as when expression square meters.

A3DMiscAttributeBasicUnitData.m_dFactor is a multiplier on the base unit base. A common example is km where the factor is 1000 for “ten times a meter”.

The A3DMiscAttributeUnitData Structure

The A3DMiscAttributeUnitData is a collection of A3DMiscAttributeBasicUnitData and combines them to form a complex unit. In our example, km/h is expressed by combining two ref A3DMiscAttributeBasicUnitData instances: one for the kilometers and one for the hours.

The structure is:

typedef struct
{
  A3DUns16                        m_usStructSize;
  A3DUTF8Char*                    m_pcName;
  A3DUns32                        m_uiBasicUnitSize;
  A3DMiscAttributeBasicUnitData** m_ppBasicUnits;
} A3DMiscAttributeUnitData;

This structure simply aggregates ref A3DMiscAttributeBasicUnitData instances in an ordered array. ref A3DMiscAttributeUnitData::m_pcName “m_pcName” is used to give a name to the unit.

Getting the Unit From Generic Attributes

When reading generic attributes, the A3DMiscSingleAttributeData instance represents a unit-qualified value if:

  • A3DMiscSingleAttributeData.m_eType is “kA3DModellerAttributeTypeReal”

  • A3DMiscSingleAttributeData.m_usUnit is not A3D_DEFAULT_NO_UNIT

A3DMiscSingleAttributeData.m_usUnit is an index to the Global data. This index points to an A3DMiscAttributeUnit handle which can be used to retrieve the A3DMiscAttributeUnitData instance.

This example shows how to get the complex unit system from an A3DMiscSingleAttributeData instance:

void ReadUnit(const A3DMiscSingleAttributeData* psAttrData)
{
  if(psAttrData->m_usUnit != A3D_DEFAULT_NO_UNIT)
  {
    // Getting the Unit handle from index
    A3DMiscAttributeUnit* pUnit;
    A3DGlobalGetUnit(psAttrData->m_usUnit, &pUnit);

    // Retrieving the Unit data from handle
    A3DMiscAttributeUnitData sUnitData;
    A3D_INITIALIZE_DATA(A3DMiscAttributeUnitData, sUnitData);
    A3DGlobalGetUnitData(pUnit, &sUnitData);

    // Browsing the basic units:
    for(A3DUns32 u = 0 ; u < sUnitData.m_uiBasicUnitSize ; ++u)
    {
        printf("- Unit %d, Exponent %d, Factor %f\n",
        sUnitData.m_ppBasicUnits[u].m_eUnit;
        sUnitData.m_ppBasicUnits[u].m_iExponent;
        sUnitData.m_ppBasicUnits[u].m_dFactor;
      );
    }

    // Cleaning the data
    A3DGlobalGetUnitData(NULL, &sUnitData);
  }
}

If this is run with our km/h example, we would get:

- Unit 21, Exponent 1, Factor 1000
- Unit 14, Exponent -1, Factor 1