有时,成员在使用 TB_IfcExport 模块导出到 .ifc 文件时会遇到单位转换问题。本文包含一个处理单位并将其导出到 .ifc 文件的示例。
创建项目单位分配
OdIfcExportImpl::createDefaultUnits() 函数创建一个 IfcUnitAssignment。这是一个很长的单位列表,用于将内部 BimRv 单位转换为已知单位。
基于转换的单位
例如,考虑使用长度单位创建基于转换的单位的算法。
1. 使用 UnitsTracking Manager 和 FormatOptions 获取内部单位:
OdBmUnitsTrackingPtr pUnitsTracking = m_database->getAppInfo(OdBm::ManagerType::UnitsTracking);
OdBmUnitsElemPtr pUnitsElem = pUnitsTracking->getUnitsElemId().safeOpenObject();
OdBmAUnitsPtr pAUnits = pUnitsElem->getUnits();
OdBmFormatOptionsPtrArray aFormatOptions;
pAUnits->getFormatOptionsArr(aFormatOptions);
OdDAIObjectId lenSIBaseUnit;
{
bool lenConversionBased = false;
bool lenUseDefault = false;
OdAnsiString lenConvName;
OdAnsiString lenUnitType = "LENGTHUNIT";
OdAnsiString lenPrefix;
OdAnsiString lenUnitName = "METRE";
OdBmFormatOptionsPtr pFormatOptions = aFormatOptions[OdBm::UnitType::UT_Length];
switch (pFormatOptions->getDisplayUnits())
{
case OdBm::DisplayUnitType::DUT_METERS:
case OdBm::DisplayUnitType::DUT_METERS_CENTIMETERS:
break;
case OdBm::DisplayUnitType::DUT_CENTIMETERS:
lenPrefix = "CENTI";
break;
case OdBm::DisplayUnitType::DUT_MILLIMETERS:
lenPrefix = "MILLI";
break;
case OdBm::DisplayUnitType::DUT_DECIMAL_FEET:
case OdBm::DisplayUnitType::DUT_FEET_FRACTIONAL_INCHES:
{
if (exportToCOBIE)
lenConvName = "foot";
else
lenConvName = "FOOT";
lenConversionBased = true;
}
break;
case OdBm::DisplayUnitType::DUT_FRACTIONAL_INCHES:
case OdBm::DisplayUnitType::DUT_DECIMAL_INCHES:
{
if (exportToCOBIE)
lenConvName = "inch";
else
lenConvName = "INCH";
}
lenConversionBased = true;
break;
default:
{
//Couldn't find display unit type conversion -- assuming foot
if (exportToCOBIE)
lenConvName = "foot";
else
lenConvName = "FOOT";
lenConversionBased = true;
lenUseDefault = true;
}
break;
}
2. 如有必要,创建基本 SI 单位实例和带前缀的 SI 单位实例:
OdDAIObjectId lenSIUnit = OdInstanceExporter::createSIUnit(m_model, lenUnitType, lenPrefix,
lenUnitName);
if (lenPrefix.isEmpty())
lenSIBaseUnit = lenSIUnit;
else
lenSIBaseUnit = OdInstanceExporter::createSIUnit(m_model, lenUnitType, NULL, lenUnitName);
3. 使用 BmUnitUtils 获取长度比例因子:
double lengthScaleFactor = OdBmUnitUtils::convertFromInternalUnits(1.0, lenUseDefault ?
OdBm::DisplayUnitType::DUT_DECIMAL_FEET : pFormatOptions->getDisplayUnits());
// If BIM Units differ from SI Units, for example Inches or Foots using for length measure, creating // Base Conversion Unit. First getting Conversion Scale Factor using OdBmUnitUtils:
if (lenConversionBased)
{
double lengthSIScaleFactor = OdBmUnitUtils::convertFromInternalUnits(1.0,
OdBm::DisplayUnitType::DUT_METERS) / lengthScaleFactor;
4. 创建一个维度指数实例(长度的一个维度):
OdDAIObjectId lenDims = OdInstanceExporter::createDimensionalExponents(m_model, 1, 0, 0,
0, 0, 0, 0);
5. 创建一个带测量单位的对象,该对象确定要转换的单位(SI 长度单位)以及如何转换(lengthSIScaleFactor 比率):
ODIFC_CREATE_AS_RATIO_MEASURE(supVal, lengthSIScaleFactor)
OdDAIObjectId lenConvFactor = OdInstanceExporter::createMeasureWithUnit(m_model, supVal, lenSIUnit);
6. 创建一个基于转换的度量单位,其属性是之前创建的维度指数对象和带测量单位的对象。
lenSIUnit = OdInstanceExporter::createConversionBasedUnit(m_model, lenDims, lenUnitType,
lenConvName, lenConvFactor);
7. 将 SI 单位对象添加到 UnitSet 数组:
unitSet.push_back(lenSIUnit);
8. 将 SI 单位和比例因子添加到单位缓存:
OdExporterCacheManager::getUnitsCache()->addUnit(OdBm::UnitType::UT_Length,
lenSIUnit,lengthScaleFactor, 0.0);
表达式单位
例如,考虑使用质量密度单位创建表达式单位的算法。
1. 创建一个质量 SI 单位:
OdDAIObjectId massSIUnit;
{
massSIUnit = createSIUnit(OdBm::UnitType::UT_Mass, "MASSUNIT", "GRAM", "KILO",
OdBm::DisplayUnitType::DUT_UNDEFINED);
unitSet.push_back(massSIUnit);
}
2. 使用千克(1 次方)和米(-3 次方)创建导出单位元素。导出单位元素的属性是上面创建的 SI 单位对象:质量 SI 单位和长度 SI 单位。
OdDAIObjectIds elements;
elements.push_back(OdInstanceExporter::createDerivedUnitElement(m_model, massSIUnit, 1));
elements.push_back(OdInstanceExporter::createDerivedUnitElement(m_model, lenSIBaseUnit, -3));
3. 使用导出单位元素创建 kg/(m^3) 导出单位:
OdDAIObjectId massDensityUnit = OdInstanceExporter::createDerivedUnit(m_model, elements, "MASSDENSITYUNIT", NULL);
4. 将创建的导出单位和比例因子添加到单位缓存中,以将 UnitSet 数组投影到其中:
unitSet.push_back(massDensityUnit);
double massDensityFactor = OdBmUnitUtils::convertFromInternalUnits(1.0, OdBm::DisplayUnitType::DUT_KILOGRAMS_PER_CUBIC_METER);
OdExporterCacheManager::getUnitsCache()->addUnit(OdBm::UnitType::UT_MassDensity, massDensityUnit,
massDensityFactor, 0.0);
5. 以相同的方式重复填充其他单位:面积、体积、平面角、时间、频率、温度等。
6. 最后,在填充 UnitSet 数组后,创建一个项目 UnitAssignment 实例:
OdInstanceExporter::createUnitAssignment(m_model, unitSet);
在导出过程中使用 UnitsCache 和 UnitUtils 进行单位转换
在导出对象的过程中,将从 BimRv API 获取的测量单位转换为 IFC 项目单位。为此目的创建了一个缓存以及使用此缓存的实用程序。在下一个级别导出示例中考虑它们的应用。
// …
// some code to fill levels array from database
// …
OdBmLevelPtr level = levels[ii];
double elev = level->getElevation(); // elevation in internal units
1. 使用 ScaleLength() 工具将高程从 BimRv 单位转换为 IFC 单位:
double elevation = OdUnitUtil::ScaleLength(elev);
2. 使用转换后的高程创建建筑楼层的局部放置和建筑楼层:
OdGeVector3d orig(0.0, 0.0, elevation);
OdDAIObjectId placement = OdExporterUtil::createLocalPlacement(m_model, buildingPlacement, &orig,
NULL, NULL);
OdDAIObjectId buildingStorey = OdInstanceExporter::createBuildingStorey(m_model, level,
OdExporterCacheManager::getOwnerHistoryHandle(),
bsObjectType, placement, "ELEMENT", elevation);
导出结果
导出结果是,文件包含以下行。
1. 长度单位(SI 中的 METRE 和 BimRv 数据库中的 FOOT):
#21=IFCSIUNIT(*,.LENGTHUNIT.,$,.METRE.);
#22=IFCDIMENSIONALEXPONENTS(1,0,0,0,0,0,0);
#23=IFCMEASUREWITHUNIT(IFCRATIOMEASURE(0.30480000000000035),#21);
#24=IFCCONVERSIONBASEDUNIT(#22,.LENGTHUNIT.,'FOOT',#23);
2. 质量和质量密度单位:
#37=IFCSIUNIT(*,.MASSUNIT.,.KILO.,.GRAM.);
#38=IFCDERIVEDUNITELEMENT(#37,1);
#39=IFCDERIVEDUNITELEMENT(#21,-3);
#40=IFCDERIVEDUNIT((#38,#39),.MASSDENSITYUNIT.,'');
3. 项目单位分配:
#73=IFCUNITASSIGNMENT((#24,#28,#32,#36,#37,#40,#41,#42,#44,#48,#52,#53,#54,#55,#56,#57,#58,#59,#64,#67,#68,#72));
4. 建筑楼层:
#76=IFCCARTESIANPOINT((0.00000000000000000,0.00000000000000000,-800.00000000000000));
#77=IFCAXIS2PLACEMENT3D(#76,$,$);
#78=IFCLOCALPLACEMENT(#16,#77);
#79=IFCBUILDINGSTOREY('1i_YYDONP3AgUe36R0EAL1',#20,'Foundation','','8mm Head',#78,$,'Foundation',.ELEMENT.,-800.00000000000068);