zoukankan      html  css  js  c++  java
  • Houdini技术体系 基础管线(三) :UE4 Landscape Component的多选支持 下篇

    背景

    上篇中,我们介绍了如何修改Houdini Enigne来设置单个Landscape Compnent的Height和Layer的数据,但原生Houdini Engine并不支持多选Component的写回功能,下篇中,我们来解决这个问题。

    Component多选支持的修改

        Houdini Engine虽然支持多个Landscape Component的选择,但是并不支持写回到Landscape Component,需要自己来实现这个功能。单个Component的实现方法上文已经接受。
    多选和单选区别只是在要把所选的Landscape Component按提交给Houdini的顺序来保存。
        通过阅读Houdini Engine代码可以看到FHoudiniLandscapeUtils::CreateHeightfieldFromLandscapeComponentArray的参数LandscapeComponentArray里有所有提交的Landscape Component,不过LandscapeComponentArray中保存顺序并不是真正的提交顺序,而Houdini Engine Output出来的处理结果的顺序是Input的顺序的是一致的,如果直接用LandscapeComponentArray的结果,就会导致Component的不对应,所以这里我是在FHoudiniLandscapeUtils::CreateHeightfieldFromLandscapeComponent函数里把Input的Component保存起来,也保证了保存顺序。
     
        然后在FHoudiniLandscapeUtils::CreateAllLandscapes函数中,就可以把每个FoundHeightfield和LandscapeComponent做对应,来调用LandscapeEdit.SetHeightData来更新这个Component的Height Data了。
     
      for ( TArray< const FHoudiniGeoPartObject* >::TConstIterator IterHeighfields
        ( FoundHeightfields ); IterHeighfields; ++IterHeighfields )
      {
          SelectLandscapeComponent = 
          SelectLandscapeComponentArray[ComponentIndex];
    
        而Layer Data的保存方式和Height Data,所有Landscape Component的Layer都保存在一个ImportLayerInfos,比如选了4个Component,每个Component有4个Layer,ImportLayerInfos里就有4x4 16个Layer Data的信息,这里也需要自己按Component和每个Component的Layer数量来提交。下面伪代码,LayerNum为每个Component的Layer的数量,ComponentIndex为处理的Component的编号,而实际开发情况下,可能每个Component的Layer的数量和命名都不一样,那就需要根据规则来定制这里的算法了。
     
    // Set Current Component's Layer Data
    for (int32 LayerIndex = LayerNum * ComponentIndex; 
    LayerIndex < LayerNum * (ComponentIndex + 1); LayerIndex++)
    {
        LandscapeEdit.SetAlphaData
    }
    ComponentIndex++;
    

      

    这样修改后,Houdini Engine就可以支持多选Landscape Component的Input和Output了。这里使用上节用到HDA文件,选中4个Component做HeightField Noise的生成
    但结果跟我们预想的并不一样,而且只有第一个Component被做了Noise处理。。。
    这是UE4原生的Houdini Engine的Input的数据和我们的HDA的处理算法不匹配导致的。

    修改HDA对Input的支持

    造成这个结果的原因,要从Houdini Engine生成Input的函数FHoudiniEngineUtils::HapiCreateInputNodeForLandscape入手:
    // 1. Create the heightfield input node.
    // We'll use its mergeId to connect all the landscape layers,
    // while it's displayId will be our connected asset ID
    FString LandscapeName = LandscapeProxy->GetName() + TEXT("_Merge");
    HAPI_NodeId MergeId = -1;
    if ( !FHoudiniLandscapeUtils::CreateHeightfieldInputNode( ConnectedAssetId, MergeId, LandscapeName ) )
        return false; 
        
      这里是通过Houdini Engine,直接创建了一个Houdini的Merge节点,然后把每个Component的Landscape Height Data和Layer Data转为HeightField的Height和Mask Volume,在Merge到一起,也就是用C++代码来生成HDA节点,这样就保证了所有Input的整体处理,而且也可以程序化的去对应不同的Input情况,在后面的章节里,很多Input项目也是要使用C++或Python来生成HDA节点来节省开发成本。下图就是程序生成HDA节点的效果示意:
    再增加一个Heightfield noise 看下效果:和之前闭环测试效果一样,因为默认的Houdini HeightField节点并不支持这种多个Volume Merge的处理。
     
    这里介绍三种解决方法:
    方法一是直接修改或重写HeightField Noise节点来支持整个Merged Volume:
    方法二:用Loop处理每个Height Volume
    方法三:用tilesplice把Volume合并到一起做处理,然后再用split重新切开
     
    如果不想自己修改或定制节点的话,方法二和三都可以,感觉方法三还更省事,但方法三有以下几个问题:
    一个是Tile顺序的问题。UE4里的Tile和Houdini的Tile的行列是不同的,同样一个2x2的Compnent,他的Input和Output的顺序有所不同:
    Input  Output
    1 2      1 3
    3 4      2 4
    这个需要自己开发功能调整,另外,方法三所选的Component也必须是NxM这种连续的Volume,否则tilesplice节点会像下图这样帮你补齐。所以方法三也有不少的限制。
    如果时间允许,还是自己开发一套Heightfield的节点来定制需要的功能,原生的HeightField系列节点在内存上也有些浪费,限制也比较多,而方法二和三可以作为临时应急方法。

    边缘法线问题处理

    再运行一次HDA处理,2x2的4个Component的节点确实都做了处理,但是在Component边缘有很明显的接缝问题,在World Normal视图下更明显。
        造成法线接缝是Component之间是共享边缘造成的,单个Landscape Component的LandscapeEdit.SetHeightData即便选择计算法线,也会导致边缘因为采样不到旁边Component的顶点,而导致两个Component的不连续,这里我暂时使用了比较暴力的方法,所有的Landscape Comonent 在SetHeightData都不计算法线,而是在最后重新计算整个Landscape的Normal。
    再看下修改后多选Component增加一个HeightField Noise的效果。
     
    Height Data处理后,就是Layer Data的处理了,这里把HeightField Noise 改成 HeightField Mask Noise,对Landscpae的4个Layer的Mask做噪声处理,和之前Height Data导入时一样,也会有Component之间的接缝问题。
        这种分割图接缝的问题,以前用WorldMachine做Tile Mask时也经常遇到,也就是Tile边缘之间共享顶点的问题。用WM可以少输出一圈边缘的Map的方法来解决,在UE4里也可以使用类似的方法。
     
    在调用的LandscapeEdit.SetAlphaData参数上,把Stride比默认的宽度减少1(XSize - 1),就可以不传边缘的Mask Data进去了。
     
    LandscapeEdit.SetAlphaData(ImportLayerInfos[LayerIndex].LayerInfo, SelectLandscapeComponent->GetSectionBase().X, SelectLandscapeComponent->GetSectionBase().Y, SelectLandscapeComponent->GetSectionBase().X + SelectLandscapeComponent->ComponentSizeQuads, SelectLandscapeComponent->GetSectionBase().Y + SelectLandscapeComponent->ComponentSizeQuads, (uint8*)ImportLayerInfos[LayerIndex].LayerData.GetData(), XSize - 1);
    

      

    再看一下效果,接缝的问题基本上已经解决了。
     

    总结

    至此,Houdini技术体系的几个问题的基础解决方案已经完成,后面的文章会逐渐倾向Houdini的地形实际制作部分。
    而这些技术案例,大多要基于这个闭环+可选组件的方式来实现,随着技术介绍流程,我也会在Github上定制一个类似Far Cry5的UE4 Houdini Engine版本,希望大家多提宝贵意见。
     
     
  • 相关阅读:
    一次失败的架构评审会议
    在搜索引擎面前经验是如此苍白
    中层难当
    js调试的小工具
    用js小类库获取浏览器的高度和宽度信息
    js过滤HTML标签以及&nbsp;
    语句
    .NET中回调事件的简单分析
    javascript中类的定义及其方式(《javascript高级程序设计》学习笔记)
    JavaScript中的16进制字符(改进)
  • 原文地址:https://www.cnblogs.com/TracePlus/p/9177146.html
Copyright © 2011-2022 走看看