HOOPS RealDWG Integration

Using the RealDWG HIO Component

The native RealDWG libraries supplied by Autodesk are required to read and write DWG and DXF files. To use HOOPS Visualize with the HOOPS/HIO module, developers are required to license the RealDWG Toolkit. Please contact a Tech Soft 3D representative for licensing details. The release notes Library Compatibility Information indicates the version of RealDWG compatible with a given version of HOOPS Visualize.

To use the HOOPS/HIO component for RealDWG, please follow the steps below:

  1. Create a hio_plugins directory in your application’s working directory.

  2. Copy the hio_dwg.hio file found in /bin/<platform>/hio_plugins/hio_dwg to the hio_plugins directory created in the previous step.

  3. Ensure that the RealDWG SDK is installed on your system and in your path.

  4. Copy all DBX and DLL files from the RealDWG directory to your application’s working directory.

  5. Copy the en-US folder from the RealDWG directory to your application’s working directory.

  6. Run your application.

During start-up, when HOOPS/MVO finds the RealDWG HIO component in your application’s path, it will perform the following steps:

  1. Load in the necessary RealDWG libraries.

  2. Create the appropriate input handler.

  3. Register this handler and the associated file extensions to the HIOManager.

To utilize the extensive services available in the RealDWG toolkit, you must compile against the RealDWG libraries directly and follow the steps described in Integrating directly with RealDWG.

Unsupported Entities

All entities found in RealDWG are supported by HOOPS Visualize except for:

  • AcDbWipeOut

  • Custom line pattern

DWG Units

When a RealDWG model is imported into HOOPS Visualize, the model units are also imported. As of HOOPS Visualize version 19.01, this information can be found in the “model_information” segment under the model key as a Visualize User_Option called “cad units”. For example, if the user option value is “cad units =Inches”, this means that one unit length of the imported entity is equivalent to one inch. The possible units are as follows:

  • Inches

  • Feet

  • Miles

  • Millimeters

  • Centimeters

  • Meters

  • Kilometers

  • Microinches

  • Mils

  • Yards

  • Angstrom

  • Nanometers

  • Microns

  • Decimeters

  • Dekameters

  • Hectometers

  • Gigameters

  • Astronomical

  • Light Years

  • Parsecs

  • Undefined Units

For backward compatibility with HOOPS Visualize versions prior to 19.01, the DWG units information is also stored in a segment underneath the model key, called “info”, and is stored in the “dwg_unit” user option.

DWG Font Handling

During the import of a DWG that contains text, any fonts specified in the DWG file must be on the system because fonts can not be embedded in a DWG file. If the importer does not find the given font, the text will be drawn with the default HOOPS Visualize font. This behavior can be modified using Set_Text_Font.

Integrating Directly with RealDWG

The RealDWG HIO component allows developers to easily import visualization information into the Visualize database. However, DWG files also contain more than just visualization information. If your application needs to access or manipulate the extensive information found in DWG files or utilize the full range of the RealDWG library services, you will require a tighter integration than what the RealDWG HIO component can provide - you need to integrate directly with RealDWG. To do so, follow these steps:

  1. Install the RealDWG toolkit on your system.

  2. Set an environment variable which points to the root of your RealDWG installation. For 32-bit systems, use REALDWG_2019_SDK_DIR. 64-bit systems use REALDWG_2019_SDK_DIR_X64.

  3. Add files HIOUtilityDWG.h/.cpp and HDWGHelper.h/.cpp to your project. These files are located in the Dev_Tools/hoops_hio/hio_dwg/source directory of your HOOPS Visualize installation.

  4. Add the following directory paths to your additional include folders’ list $(REALDWG_2019_SDK_DIR_X64)/include.

  5. Add $(REALDWG_2019_SDK_DIR_X64)/lib to your additional linker directories.

  6. Add the following files as additional dependencies:

    • ac1st23.lib

    • rcexelib.obj

    • acgiapi.lib

    • acdb23.lib

    • acge23.lib

    • rxapi.lib

    • acismobj23.lib

    • odbc32.lib

    • odbccp32.lib

    • urlmon.lib

    • wininet.lib

  7. Ensure that the REALDWG_2019_SDK_DIR_X64 environment variable is in your path.

Once you have completed the above steps, you do not need to rely on the RealDWG HIO component anymore. Please remove the associated .hio file from the hio_plugins directory in your application’s working directory. Note that in a direct integration with HOOPS Visualize, you must register the DWG handler``HIOUtilityDWG`` via HIOManager::RegisterInputHandler so that the HBaseModel::Read function can recognize that the .dwg file format. For more information on HOOPS/MVO file input/output architecture, please see section File Input/Output and the I/O Manager of the HOOPS/MVO Programming Guide.

Now you can begin working with HIOUtilityDWG class which is HOOPS/MVO’s input handler for RealDWG. The next section demonstrates how to instantiate the input handler and import a DWG file.

Importing a File

The process for importing a DWG file is similar to the process for importing other file types through the HIO interface. For example, to import DWG data into the model segment, you could use the following code:

HInputHandler * dwg_handler = HDB::GetHIOManager()->GetInputHandler("dwg");
if (dwg_handler)
{
        HInputHandlerOptions opts;
        char texture_directory[] = "/path/to/textures";
        opts.m_pExtendedData = (void *)texture_directory;

        // import DWG file into model segment
        HFileIOResult result = dwg_handler->FileInputByKey("filename.dwg", m_pHView->GetModelKey(), &opts);
        if (result == HIO_OK)
                // file loaded OK
        else
                // handle error
}
else
        // handle error

Although it is possible to recompile the HIO RealDWG input handler, you can only have one RealDWG handler in your application. If you compile a second handler, you should delete the old one - otherwise, the behavior is undefined.

The sample above includes an optional path to the texture directory, which contains any image files used to generate textures. In a typical installation, this directory would be set to C:\Program Files (x86)\Common Files\Autodesk Shared\Materials.

Alternatively, if you didn’t want to set a texture directory, you could simply call \ref FileInputByKey() with a NULL parameter instead: FileInputByKey("filename.dwg", m_pHView->GetModelKey(), NULL);.

Mapping Between HOOPS/3dGS and DWG Entities

The RealDWG Scene Graph

Upon import, the HIO RealDWG component will create the following scene graph:

  • root import segment
    • block table
      • master
        • *<block table name>* - a subsegment for each block table record. Segment names correspond to block table names in AutoCAD, but sanitized. HIO RealDWG will prepend btr_ to the name, characters are converted to lowercase, and finally it will be URI encoded. We also enclose the name in back-ticks if the name has special characters. Thus, a name like *Model_Space becomes `btr_%2model_space`.
          • geometry - contains the geometric definition of the entity
      • xrefs - only present if the DWG file contains external references. The subsegments are URI encoded and they represent the block table recrods from the external file.
    • layer table
      • *<layer names>* - taken from layer names in the DWG file. HIO RealDWG will prepend the name with and underscore, then convert it to lowercase.
    • handles - see "Entity mapping" subsection below for notes about handles
    • info - contains a USER OPTIONS value called "dwg_units". This indicates the scale of the model. For example, "dwg_units=Centimeters" indicates the model units are centimeters.
    • layouts
      • *<layout names>* - one subsegment for each DWG layout. Names are sanitized like layer names. By default, we always display the "_model" layout. To change layouts, turn visibility off for the segment, then on for the layout segment you wish to change to. Additionally, remove the heuristic "exclude bounding", and enable it for the segment you wish to change to.
    • viewport table - currently unused
    • xdata - see related section below

Entity Mapping

Each DWG entity has a unique handle associated with it. The handle is unique within the database, and is the same session after session. The handle is represented by an unsigned 64-bit integer. Handles exhibit the following relationships:

  • One DWG handle corresponds to one or more HOOPS Visualize keys.

  • One Visualize key corresponds to only one DWG handle.

  • Two different DWG handles cannot refer to the same Visualize key.

When associating handles to keys, keep the following in mind:

  • every time a piece of Visualize geometry is inserted, it is opened by key, and a user option containing the DWG handle identifier this piece of geometry belongs to is set on it.

  • a segment called “handles” is created during import and placed directly under the MVO model segment.

  • as geometry is read in, a reference to it is placed in a subsegment of the “handles” segment, named after the DWG handle the geometry belongs to.

To relate a DWG handle to a HOOPS Visualize key, you would do the following:

Case A - You have a HOOPS Visualize key and wants to obtain the related DWG handle:

  1. Open geometry by key

  2. Use Show_One_User_Data to read the data stored at index 0

  3. The data returned by Show_One_User_Data will contain the DWG handle.

Example:

Adesk::UInt64 handle;
HC_Open_Geometry(key);
HC_Show_One_User_Data(0, &handle, sizeof(handle);
HC_Close_Geometry();

Case B - You have a DWG handle and want to obtain the related HOOPS Visualize key(s):

  1. Open the “handles” segment and search for the segment whose name matches the DWG handle.

  2. Search the segment for reference geometry, and obtain its original key

Using a Hash to Improve Speed

Starting with Visualize 19.33, the developer has the option to query DWG handles in an optimized way. The process described in the previous section will always work, however, for large scene graphs, the processing time can be long. Using the HIO connector to associate DWG handles to Visualize keys is faster because it uses a hash lookup instead of searching the segment tree. There is a performance penalty, however, when the hash is generated.

This is how to obtain a connector:

HInputHandler * handler = HDB::GetHIOManager()->GetInputHandler("dwg");
handler->FileInputByKey("filename.dwg", view->GetModelKey(), options);
HIOConnector * connector = HDB::GetHIOManager()->CreateConnector("dwg");

All the associated functions (see list, below) have an optional parameter at the end. Passing the model key as that parameter causes the hash to be generated (or regenerated, if the geometry has been modified). Therefore, the first time you use this hash, you should always pass the model key.

The DWG connector can use the following functions:

  1. HC_KEY GetHoopsEntity(void * pKernelEntity, HC_KEY modelKey = INVALID_KEY);

This function returns a HOOPS Visualize key associated with a DWG handle. Causes hash to be regenerated. Sample usage:

unsigned int handle = 570;
HC_KEY key = connector->GetHoopsEntity((void *)handle, view->GetModelKey());
  1. void * GetKernelEntity(HC_KEY key, HC_KEY modelKey = INVALID_KEY);

This function returns a DWG handle associated with a HOOPS Visualize key. Causes hash to be regenerated. Sample usage:

void * kernelEntity = connector->GetKernelEntity(key);
  1. bool GetHoopsEntities(void * pKernelEntity, vlist_s *ret_HoopsKeysList, HC_KEY modelKey = INVALID_KEY);

Same as GetHoopsEntity, but is used in the case when a DWG handle is associated with more than one Visualize key. Causes hash to be regenerated. Sample usage:

vlist_s * hoopsKeyList = new_vlist(malloc, free);
connector->GetHoopsEntities((void *)myHandle, hoopsKeyList);
void * item  = vlist_nth_item(hoopsKeyList, 0);
void * item2 = vlist_nth_item(hoopsKeyList, 1);
  1. bool GetKernelEntities(HC_KEY key, vlist_s *ret_KernelEntitiesList, HC_KEY modelKey = INVALID_KEY); Same as GetKernelEntity

  2. void AddConnection(HC_KEY key, void* pKernelEntity, HC_KEY modelKey = INVALID_KEY);. Adds a connection between a key and a handle. Does not cause the has to be regenerated.

Layers and Layouts

Layers are treated as attribute containers. Whatever attributes are in the layer are applied to the geometry with a HOOPS Visualize style. This means that you can open the layer segment and override the attributes. If the geometry has attributes set on it explicitly, then the explicit attributes override the layer attributes. There’s a case when the attributes of geometry are inherited from the block reference. Those attributes cannot be overridden by the layer. The layers can be found in the segment “layer table”. The layers are URI encoded, to allow unicode characters in the segment names, and are prefixed with an underscore ‘_’ character.

Layouts can be found in the “layouts” segment. The layouts are also URI encoded and prefixed with an underscore. Each layout has user options, “extents_min” and “extents_max” set defining the extents in 2D space. Each layout also sets the heuristic “exclude bounding” and sets the visibility to “off” except for the currently active/visible layout. The currently active/visible layout does not set visibility and does not set the exclude bounding heuristic. Iterating the visibility of the layouts is done by finding the layout that does not set the visibility, then setting the visibility to off, setting the exclude bounding heuristic, then moving on to the next layout segment and unsetting the visibility and unsetting the heuristics.

Object Enablers

HIO DWG allows developers to use object enablers. Using object enablers, customers can load DWG files which contain custom entities defined in their own .dbx files.

The only requirement for this to work is that the developer must place the .dbz files and associated DLLs into the same folder as the HIO DLL, so that they can be loaded when DWG starts up.

Handling Metadata

Each DWG entity may have associated XDATA. The DWG standard calls metadata XDATA. Metadata can be of any type. If a DWG file contains metadata, HOOPS Visualize will create a subsegment under the main DWG model segment called xdata. The xdata segment will contain subsegments that correspond to metadata stored in the file. The name of each subsegment corresponds to the value of the DWG entity handle to which the metadata applies. The image below shows an example segment tree showing the xdata segment and subsegments which contain metadata.

../../_images/dwg_xdata.png

The subsegment 0x00000074 indicates the metadata in this segment belongs to DWG entity with the same handle value. In this case, USER DATA indicates that there are two types of metadata present for this entity. These types are 1000 and 1001. The child segment “restype XXXX” contains the metadata values themselves. For an example of how to read metadata from an XDATA segment tree, see the code sample in get_xdata.cpp <examples/get_xdata.cpp>.

Limitations

Dimensions, Annotations, and Layouts

It is possible for a DWG file to be stored with uninitialized annotations and layouts. These uninitialized structures don’t always exist in the DWG file. In this situation, RealDWG returns them as null pointers. Using its own internal logic, AutoCAD is able to load the uninitialized structures from null pointers, however, RealDWG cannot. Therefore, we do not support uninitialized annotations or layouts.

Viewports

Viewports within layouts can be in an ‘active’ or ‘inactive’ state. RealDWG doesn’t provide any mechanism to find which viewports are active.

Rendering

Although we make every effort to duplicate the rendering results of any file loaded using RealDWG, Visualize cannot exactly match AutoCAD in all cases when it comes time to actually draw. AutoCAD uses its own proprietary rendering engine that handles geometry in its own way. Some examples of differences we have noticed include:

  • tessellation values

  • line weights and line styles

  • font handling

Incorrect Linetypes in Some Cases

Occassionally, RealDWG will incorrectly read a linetype. For example, a line may appear continuous when it really should appear dashed. This is a confirmed bug in RealDWG recognized by Autodesk. The bug happens because the DWG libraries will cull details in order to optimize a drawing when the view distance is sufficiently far. However, when loading a file, RealDWG incorrectly provides the culled version of the entity instead of the actual verion.