I’m writing a c++ plugin that imports .nif files (NetImmerse, Gamebryo, Creation Engine) using Unreal’s Interchange system, but when I drag multiple nif assets at once into the editor, I’m getting a bunch of prompts to overwrite textures for each asset because many of them point to the same texture file.
Batch importing .nif
files with shared textures triggers numerous “Overwrite Asset?” prompts because multiple files reference the same texture paths (e.g., architecture/farmhouse/wood01.dds
). Pressing “Yes to all” only applies to all textures for a single mesh, not all meshes. I’m looking to import thousands of files and do these kinds of batch operations multiple times, and ship the plugin to other people, so this won’t do.
Unreal provides UInterchangeFactoryBaseNode::SetSkipNodeImport (lets me skip importing nodes). It seems like it was made exactly for my use case (see ShouldSkipNodeImport docs linked at the bottom as well):
"Nodes can be in a situation where we have to skip the import process because we cannot import the associated asset for multiple reasons. For example:
- An asset can already exist and is being imported by another concurrent import task (such as a user importing multiple files at the same time in the same content folder)."
My approach was to filter factory nodes using a set of keys for filtering (populated when a factory starts work) and UInterchangeFactoryBaseNode::SetSkipNodeImport
. The intent is for the first task to import a texture, claim its path (add key to the set), and subsequent tasks check the set and mark their factory nodes to skip.
However, I just couldn’t get it to skip anything. I get the prompts to override regardless.
Here’s a basic diagram of what I’m doing:
Say I have 4 meshes trying to import the texture in parallel. Here’s basic diagram that flows in time from left to right:
obj1--tex1fac---finished importing (path claimed)
obj2---tex1fac---finished importing (skipped texture creation)
obj3---tex1fac---finished importing (skipped texture creation)
obj4----tex1fac---finished importing (skipped texture creation)
However, even when SetSkipNodeImport
is called, the override dialog windows still appear.
Log Snippet (Importing 2 files sharing 4 textures) (condensed):
// File 1 imports first and claims the texture paths
LogNifPipeline: DEBUG: ClaimTexturePath(C:/Users/drkuz/Downloads/meshes/textures/architecture/riften/RiftenCanalWood01.dds) - Path was NOT previously claimed
LogNifPipeline: Claiming texture path: C:/Users/drkuz/Downloads/meshes/textures/architecture/riften/RiftenCanalWood01.dds - Total claimed paths: 1
// File 2 correctly identifies the path as claimed and marks it to skip
LogNifPipeline: DEBUG: IsTexturePathClaimed(C:/Users/drkuz/Downloads/meshes/textures/architecture/riften/RiftenCanalWood01.dds) = TRUE
LogNifPipeline: DEBUG: ProcessTextureFactoryNodes - Path was claimed by ANOTHER task/import
LogNifPipeline: DEBUG: ProcessTextureFactoryNodes - SETTING NODE TO SKIP IMPORT
LogNifPipeline: Skipping import for claimed texture path: C:/Users/drkuz/Downloads/meshes/textures/architecture/riften/RiftenCanalWood01.dds
// But the import is still attempted, and the overwrite prompt still appears
LogNifPipeline: DEBUG: ProcessTextureFactoryNodes - Complete. Skipped: 4, Will Import: 0
LogNifPipeline: DEBUG: ExecutePipeline - Completed for file: C:/Users/drkuz/Downloads/meshes/dockcolstr02.nif - Claimed 0 textures
LogNifImport: Texture file found: C:/Users/drkuz/Downloads/meshes/textures/architecture/riften/RiftenCanalWood01_n.dds
LogNifImport: Texture file found: C:/Users/drkuz/Downloads/meshes/textures/architecture/riften/RiftenCanalWood01.dds
LogNifImport: Texture file found: C:/Users/drkuz/Downloads/meshes/textures/architecture/solitude/SGrate.dds
LogNifImport: Texture file found: C:/Users/drkuz/Downloads/meshes/textures/architecture/solitude/SGrate_n.dds
LogDDSImport: DDS format supported
LogDDSImport: *** GetTexturePayloadData called for: C:/Users/drkuz/Downloads/meshes/textures/architecture/riften/RiftenCanalWood01.dds ***
LogDDSImport: *** GetTexturePayloadData called for: C:/Users/drkuz/Downloads/meshes/textures/architecture/solitude/SGrate_n.dds ***
LogDDSImport: *** GetTexturePayloadData called for: C:/Users/drkuz/Downloads/meshes/textures/architecture/solitude/SGrate.dds ***
LogDDSImport: *** GetTexturePayloadData called for: C:/Users/drkuz/Downloads/meshes/textures/architecture/riften/RiftenCanalWood01_n.dds ***
LogDDSImport: Detected normal map from filename: C:/Users/drkuz/Downloads/meshes/textures/architecture/solitude/SGrate_n.dds
LogDDSImport: Detected normal map from filename: C:/Users/drkuz/Downloads/meshes/textures/architecture/riften/RiftenCanalWood01_n.dds
LogSlate: Window 'Message' being destroyed
Message dialog closed, result: Yes, title: Message, text: Are you sure you want to override asset 'Texture2D /Game/Test/T_RiftenCanalWood01.T_RiftenCanalWood01'?
LogSlate: Window 'Message' being destroyed
Message dialog closed, result: Yes, title: Message, text: Are you sure you want to override asset 'Texture2D /Game/Test/T_RiftenCanalWood01_n.T_RiftenCanalWood01_n'?
LogSlate: Window 'Message' being destroyed
Message dialog closed, result: Yes, title: Message, text: Are you sure you want to override asset 'Texture2D /Game/Test/T_SGrate.T_SGrate'?
LogSlate: Window 'Message' being destroyed
Message dialog closed, result: Yes, title: Message, text: Are you sure you want to override asset 'Texture2D /Game/Test/T_SGrate_n.T_SGrate_n'?
My Pipeline Code (Setting Skip Flag):
// In ProcessTextureFactoryNodes:
if (FactoryNode)
{
// If the path is already claimed by ANOTHER import task (not this one)
// or we found an existing texture, skip the import process for this node
if (bPathAlreadyClaimed || ExistingTexture)
{
UE_LOG(LogNifPipeline, Log, TEXT("DEBUG: ProcessTextureFactoryNodes - SETTING NODE TO SKIP IMPORT"));
FactoryNode->SetSkipNodeImport();
NodesSkipped++;
if (ExistingTexture)
{
// If we have a valid texture reference, point to it
FactoryNode->SetCustomReferenceObject(FSoftObjectPath(ExistingTexture));
UE_LOG(LogNifPipeline, Log, TEXT("Reusing existing texture for %s: %s"),
*TexturePath, *ExistingTexture->GetPathName());
}
else
{
UE_LOG(LogNifPipeline, Log, TEXT("Skipping import for claimed texture path: %s"), *TexturePath);
}
}
}
Plugin+Project+Assets+NifPipeline.cpp+Log.txt - Drive folder link (UE5.4 Windows only)
It seems like I’m correctly skipping nodes, but the system is showing prompts before it considers the skip flags (unless I’m misunderstanding or have done something wrong).
Why isn’t SetSkipNodeImport
preventing the prompts as expected?