【UE5 C++】绘制地表贴合线

article/2025/8/3 20:05:06

目录

原理

效果

步骤

源码


原理

        先设置绘制线段的起点和终点,然后我们将起点和终点的高度升高,然后通过插值,在起点和终点之间添加多个点,再由这些点向地心发出射线,这样我们就可以获取到这些点在地表的投影点,最后将投影点连起来就是地表贴合线。

        注意:使用该方法时,需要保证地形加载完毕,因为只要地形加载后地表才会有碰撞信息,才能被射线检测到

效果

步骤

1. 在“xxx.Build.cs”中引入“CesiumRuntime”模块

 2. 新建一个Actor类,这里命名为“SurfaceLineActor”

在“SurfaceLineActor.h”中添加所需头文件

定义一个函数“CalculatePointsOnSurface”用于计算地表点,该函数需要传入起点终点的经纬度、细分点数、起终点向上偏移的距离

再定义一个函数用于绘制地表贴合线,该函数需要传入地表点数组、线的颜色、线的粗细共3个参数。

定义地心坐标、添加一个LineBatchComponent组件用于高效绘制线条、定义一个数组“Points”用于储存起终点之间的点

实现函数“CalculatePointsOnSurface”如下

TArray<FVector> ASurfaceLineActor::CalculatePointsOnSurface(FVector StartPointLLA, FVector EndPointLLA, int32 NumberOfSegments, double TraceUpOffset)
{//将起点和终点拔高一段距离以方便往地心打射线StartPointLLA = StartPointLLA + FVector(0, 0, TraceUpOffset);EndPointLLA = EndPointLLA + FVector(0, 0, TraceUpOffset);//获取GeoreferenceACesiumGeoreference* Georeference = ACesiumGeoreference::GetDefaultGeoreference(GetWorld());if (!Georeference){UE_LOG(LogTemp, Warning, TEXT("SurfaceLineActor: CesiumGeoreference not found. Cannot calculate surface points."));}//将起终点(经纬高)转换为起终点(UE世界坐标)FVector StartPoint = Georeference->TransformLongitudeLatitudeHeightPositionToUnreal(StartPointLLA);FVector EndPoint = Georeference->TransformLongitudeLatitudeHeightPositionToUnreal(EndPointLLA);Points.Empty();if (NumberOfSegments <= 0)  // 分段数不满足要求,返回起点和终点{// 如果段数无效,至少包含起点和终点Points.Add(StartPoint);if (StartPoint != EndPoint) // 避免重复添加同一点{Points.Add(EndPoint);}}else{//Points.Reserve(NumberOfSegments + 1); // 预分配空间提高效率for (int32 i = 0; i < NumberOfSegments; ++i){float T = static_cast<float>(i) / static_cast<float>(NumberOfSegments);FVector CurrentPoint = FMath::Lerp(StartPoint, EndPoint, T);  // 使用 FMath::Lerp 进行线性插值计算点的坐标Points.Add(CurrentPoint);}}// 用于存储射线检测到的地表点TArray<FVector> CalculatedSurfacePoints;//起点和终点构成的连线中每个点都朝地心发出射线for (int32 i = 0; i < Points.Num(); ++i){FHitResult HitResult;FCollisionQueryParams QueryParams;QueryParams.AddIgnoredActor(this);        // 射线检测忽视自身//QueryParams.AddIgnoredActors(ActorsToIgnore); // 忽视射线检测的ActorQueryParams.bTraceComplex = true;         // 使用复杂碰撞QueryParams.bReturnPhysicalMaterial = false; // 是否检测物理材料ECollisionChannel TraceChannel = ECC_Visibility;UWorld* World = GetWorld();bool bHit = World->LineTraceSingleByChannel(HitResult,      // 射线检测结果Points[i],      // 射线起点EarthOrigin,    // 射线终点(地心)TraceChannel,   // 射线碰撞通道QueryParams     // 射线碰撞其他参数);画射线检测(非必需)//DrawDebugLine(//    GetWorld(),//    Points[i],    //起点//    EarthOrigin,  //终点(地心)//    bHit ? FColor::Green : FColor::Red,  //颜色//    false,           // false表示它持续“LifeTime”,如果LifeTime为0,则持续一帧//    0.0,             // DebugLine生命周期时长//    0,               // DepthPriority (0 for SDPG_World)//    2.0f//);if (bHit){画射线碰撞点(非必需)//DrawDebugSphere(//    GetWorld(),//    HitResult.Location,//    30.0f, // Radius//    12,                // Segments//    FColor::Cyan,//    false,             // Persistent lines//    0.0f   // Lifetime//);FVector OutHitLocation = HitResult.ImpactPoint;CalculatedSurfacePoints.Add(OutHitLocation);}}return CalculatedSurfacePoints;
}

3. 测试一下“CalculatePointsOnSurface”函数是否生效,这里设置起点为成都市市中心终点为都江堰

运行后可以看到地表点位置基本准确

4. 接下来只需要通过LineBatchComponent组件将地表点连线即可

函数“DrawSurfaceLine”实现如下

void ASurfaceLineActor::DrawSurfaceLine(TArray<FVector> SurfacePoints, FLinearColor LineColor, float LineThickness)
{if (!LineBatchComponent){UE_LOG(LogTemp, Warning, TEXT("SurfaceLineActor: LineBatchComponent is null."));return;}LineBatchComponent->Flush(); // 清除之前的线if (SurfacePoints.Num() >= 2){for (int32 i = 0; i < SurfacePoints.Num() - 1; ++i){if (SurfacePoints[i].ContainsNaN() || SurfacePoints[i + 1].ContainsNaN()){UE_LOG(LogTemp, Warning, TEXT("SurfaceLineActor: Encountered NaN point at segment %d, skipping draw."), i);continue;}LineBatchComponent->DrawLine(SurfacePoints[i],SurfacePoints[i + 1],LineColor,SDPG_World, // 场景深度优先级组-根据渲染顺序需要进行调整LineThickness,0.0f        // 线的生命周期时长:0表示无限生命);}}
}

再通过蓝图在获取地表点数组后传入函数 “DrawSurfaceLine”

此时效果如下: 

 5. 选取山地进行测试,插值数量设置为200,此时运行效果如下

源码

 “SurfaceLineActor.h”

// Fill out your copyright notice in the Description page of Project Settings.#pragma once#include "CoreMinimal.h"
#include "GameFramework/Actor.h"
#include "Components/LineBatchComponent.h"
#include "CesiumGeoreference.h"
#include "SurfaceLineActor.generated.h"UCLASS()
class GLOBEPAWNTEST_API ASurfaceLineActor : public AActor
{GENERATED_BODY()public:	// Sets default values for this actor's propertiesASurfaceLineActor();// 计算地表点UFUNCTION(BlueprintCallable, Category = "Line Properties")TArray<FVector> CalculatePointsOnSurface(FVector StartPointLLA = FVector(0, 0, 0),FVector EndPointLLA = FVector(0, 0, 0),int32 NumberOfSegments = 50,     // 线段细分数 (越高质量越好,但性能开销越大)double TraceUpOffset = 100000.0  // 射线检测时向上偏移的距离(确保起点在潜在地形之上)  默认100km);// 调用此函数来绘制或更新地表贴合线UFUNCTION(BlueprintCallable, Category = "Line Properties")void DrawSurfaceLine(TArray<FVector> SurfacePoints,FLinearColor LineColor = FLinearColor::Red, // 线的颜色float LineThickness = 10.0f    // 线的粗细);//地心坐标UPROPERTY(EditAnywhere, BlueprintReadWrite)FVector EarthOrigin = FVector(0, 0, -637810000.0);protected:// Called when the game starts or when spawnedvirtual void BeginPlay() override;UPROPERTY(VisibleAnywhere, BlueprintReadOnly)ULineBatchComponent* LineBatchComponent;public:	// Called every framevirtual void Tick(float DeltaTime) override;private:TArray<FVector> Points;  //用于存储起点和终点连线之间的点
};

 “SurfaceLineActor.cpp”

// Fill out your copyright notice in the Description page of Project Settings.#include "SurfaceLineActor.h"
#include "CesiumRuntime/Public/CesiumGeoreference.h"// Sets default values
ASurfaceLineActor::ASurfaceLineActor()
{// Set this actor to call Tick() every frame.  You can turn this off to improve performance if you don't need it.PrimaryActorTick.bCanEverTick = true;PrimaryActorTick.bStartWithTickEnabled = true;LineBatchComponent = CreateDefaultSubobject<ULineBatchComponent>(TEXT("LineBatcher"));// 可以将 LineBatchComponent 设为根组件,或者附加到其他组件上RootComponent = LineBatchComponent;LineBatchComponent->bCalculateAccurateBounds = false; // Optimization for dynamic lines}// Called when the game starts or when spawned
void ASurfaceLineActor::BeginPlay()
{Super::BeginPlay();}// Called every frame
void ASurfaceLineActor::Tick(float DeltaTime)
{Super::Tick(DeltaTime);}void ASurfaceLineActor::DrawSurfaceLine(TArray<FVector> SurfacePoints, FLinearColor LineColor, float LineThickness)
{if (!LineBatchComponent){UE_LOG(LogTemp, Warning, TEXT("SurfaceLineActor: LineBatchComponent is null."));return;}LineBatchComponent->Flush(); // 清除之前的线if (SurfacePoints.Num() >= 2){for (int32 i = 0; i < SurfacePoints.Num() - 1; ++i){if (SurfacePoints[i].ContainsNaN() || SurfacePoints[i + 1].ContainsNaN()){UE_LOG(LogTemp, Warning, TEXT("SurfaceLineActor: Encountered NaN point at segment %d, skipping draw."), i);continue;}LineBatchComponent->DrawLine(SurfacePoints[i],SurfacePoints[i + 1],LineColor,SDPG_World, // 场景深度优先级组-根据渲染顺序需要进行调整LineThickness,0.0f        // 线的生命周期时长:0表示无限生命);}}
}TArray<FVector> ASurfaceLineActor::CalculatePointsOnSurface(FVector StartPointLLA, FVector EndPointLLA, int32 NumberOfSegments, double TraceUpOffset)
{//将起点和终点拔高一段距离以方便往地心打射线StartPointLLA = StartPointLLA + FVector(0, 0, TraceUpOffset);EndPointLLA = EndPointLLA + FVector(0, 0, TraceUpOffset);//获取GeoreferenceACesiumGeoreference* Georeference = ACesiumGeoreference::GetDefaultGeoreference(GetWorld());if (!Georeference){UE_LOG(LogTemp, Warning, TEXT("SurfaceLineActor: CesiumGeoreference not found. Cannot calculate surface points."));}//将起终点(经纬高)转换为起终点(UE世界坐标)FVector StartPoint = Georeference->TransformLongitudeLatitudeHeightPositionToUnreal(StartPointLLA);FVector EndPoint = Georeference->TransformLongitudeLatitudeHeightPositionToUnreal(EndPointLLA);Points.Empty();if (NumberOfSegments <= 0)  // 分段数不满足要求,返回起点和终点{// 如果段数无效,至少包含起点和终点Points.Add(StartPoint);if (StartPoint != EndPoint) // 避免重复添加同一点{Points.Add(EndPoint);}}else{//Points.Reserve(NumberOfSegments + 1); // 预分配空间提高效率for (int32 i = 0; i < NumberOfSegments; ++i){float T = static_cast<float>(i) / static_cast<float>(NumberOfSegments);FVector CurrentPoint = FMath::Lerp(StartPoint, EndPoint, T);  // 使用 FMath::Lerp 进行线性插值计算点的坐标Points.Add(CurrentPoint);}Points.Add(EndPoint);  //需要加上终点}// 用于存储射线检测到的地表点TArray<FVector> CalculatedSurfacePoints;//起点和终点构成的连线中每个点都朝地心发出射线for (int32 i = 0; i < Points.Num(); ++i){FHitResult HitResult;FCollisionQueryParams QueryParams;QueryParams.AddIgnoredActor(this);        // 射线检测忽视自身//QueryParams.AddIgnoredActors(ActorsToIgnore); // 忽视射线检测的ActorQueryParams.bTraceComplex = true;         // 使用复杂碰撞QueryParams.bReturnPhysicalMaterial = false; // 是否检测物理材料ECollisionChannel TraceChannel = ECC_Visibility;UWorld* World = GetWorld();bool bHit = World->LineTraceSingleByChannel(HitResult,      // 射线检测结果Points[i],      // 射线起点EarthOrigin,    // 射线终点(地心)TraceChannel,   // 射线碰撞通道QueryParams     // 射线碰撞其他参数);画射线检测(非必需)//DrawDebugLine(//    GetWorld(),//    Points[i],    //起点//    EarthOrigin,  //终点(地心)//    bHit ? FColor::Green : FColor::Red,  //颜色//    false,           // false表示它持续“LifeTime”,如果LifeTime为0,则持续一帧//    0.0,             // DebugLine生命周期时长//    0,               // DepthPriority (0 for SDPG_World)//    2.0f//);if (bHit){画射线碰撞点(非必需)//DrawDebugSphere(//    GetWorld(),//    HitResult.Location,//    30.0f, // Radius//    12,                // Segments//    FColor::Cyan,//    false,             // Persistent lines//    0.0f   // Lifetime//);FVector OutHitLocation = HitResult.ImpactPoint;CalculatedSurfacePoints.Add(OutHitLocation);}}return CalculatedSurfacePoints;
}

http://www.hkcw.cn/article/mlRgLSvhfv.shtml

相关文章

01 redis 的环境搭建

前言 这一系列文章主要包含的内容主要是 各种常用软件的调试环境的搭建 主要的目的是 搭建一个可打断点的一个调试环境 c 系列 主要是基于 clion 调试, java 系列主要是基于 idea 调试, js 系列主要是基于 webstorm 调试 需要有一定的 c, c, java, js 相关基础 基于的…

python打卡训练营打卡记录day40

知识点回顾&#xff1a; 彩色和灰度图片测试和训练的规范写法&#xff1a;封装在函数中展平操作&#xff1a;除第一个维度batchsize外全部展平dropout操作&#xff1a;训练阶段随机丢弃神经元&#xff0c;测试阶段eval模式关闭dropout 作业&#xff1a;仔细学习下测试和训练代码…

Tailwind CSS 实战:基于 Kooboo 构建 AI 对话框页面(五):语音合成输出与交互增强

Tailwind CSS 实战&#xff0c;基于Kooboo构建AI对话框页面&#xff08;一&#xff09; Tailwind CSS 实战&#xff0c;基于Kooboo构建AI对话框页面&#xff08;二&#xff09;&#xff1a;实现交互功能 Tailwind CSS 实战&#xff0c;基于 Kooboo 构建 AI 对话框页面&#x…

【MySQL】MVCC与Read View

目录 一、数据库并发的三种场景 二、读写场景的MVCC &#xff08;一&#xff09;表中的三个隐藏字段 &#xff08;二&#xff09;undo 日志 &#xff08;三&#xff09;模拟MVCC &#xff08;四&#xff09;Read View &#xff08;五&#xff09;当前读和快照读 三、RC和…

代码随想录打卡|Day53 图论(Floyd 算法精讲 、A * 算法精讲 (A star算法)、最短路算法总结篇、图论总结 )

图论part11 Floyd 算法精讲 代码随想录链接 题目链接 代码 三维DP数组 import java.util.Scanner;public class Main {// 定义最大距离值&#xff0c;避免使用Integer.MAX_VALUE防止加法溢出public static final int INF 100000000; // 10^8足够大且不会溢出public static…

CSS Day07

1.搭建项目目录 2.网页头部SEO三大标签 3.Favicon图标与版心 &#xff08;1&#xff09;Favicon图标 &#xff08;2&#xff09;版心 4.快捷导航 5.头部-布局 6.头部-logo 7.头部-导航 8.头部-搜索 9头部-购物车 10.底部-布局 11.底部-服务区域 12.底部-帮助中心 13.底部-版权…

leetcode hot100刷题日记——29.合并两个有序链表

解答&#xff1a; 方法一&#xff1a;递归 递归的边界条件是啥呢&#xff1f; 递归别想那么多具体步骤&#xff0c;考虑大步骤&#xff0c;小的递归自己会去做的 class Solution { public:ListNode* mergeTwoLists(ListNode* list1, ListNode* list2) {//递归比较大小//先考虑…

Spring Boot 整合 Spring Security

DAY30.1 Java核心基础 Spring Boot 整合安全框架 Spring Security 、Shiro Spring Security Spring Security 的核心功能包括认证、授权、攻击防护&#xff0c;通过大量的过滤器和拦截器进行请求的拦截和验证&#xff0c;实现安全校验的功能。 Spring Security 将校验逻辑…

深度剖析Node.js的原理及事件方式

早些年就接触过Node.js&#xff0c;当时对于这个连接前后端框架就感到很特别。尤其是以独特的异步阻塞特性&#xff0c;重塑了了服务器端编程的范式。后来陆陆续续做了不少项目&#xff0c;通过实践对它或多或少增强了不少理解。今天&#xff0c;我试着将从将从原理层剖析其运行…

智慧景区一体化建设方案

随着2023年文旅部《关于推动智慧旅游发展的指导意见》出台&#xff0c;全国景区掀起数字化转型浪潮。如何在激烈竞争中脱颖而出&#xff1f;智慧景区一体化建设方案&#xff0c;正以“一机游遍景区、一屏掌控全局”的革新模式&#xff0c;重新定义旅游体验与管理效率。本文深度…

使用 SymPy 操作三维向量的反对称矩阵

在三维空间中&#xff0c;一个 3 1 3 \times 1 31 向量可以转换为一个 3 3 3 \times 3 33 的反对称矩阵。这种转换在物理学、机器人学和计算机视觉等领域非常有用。本文将详细介绍如何在 Python 的 SymPy 库中定义和使用这种反对称矩阵。 数学背景 对于一个三维向量 v …

LangChain表达式(LCEL)实操案例1

案例1&#xff1a;写一篇短文&#xff0c;然后对这篇短文进行打分 from langchain_core.output_parsers import StrOutputParser from langchain_core.prompts import ChatPromptTemplate, MessagesPlaceholder from langchain_core.runnables import RunnableWithMessageHist…

CppCon 2014 学习:HOW UBISOFT MONTREAL DEVELOPS GAMES FOR MULTICORE

多核处理器&#xff08;Multicore Processor&#xff09; 的基本特性&#xff0c;下面是对每点的简要说明&#xff1a; &#x1f539; Multicore&#xff08;多核&#xff09; 指一个物理处理器上集成了 多个 CPU 核心&#xff0c;每个核心可以独立执行指令。 &#x1f538;…

STL解析——String类详解(使用篇)

目录 sring接口解析 1.string简介 2.默认成员函数 2.1构造函数 2.2析构函数 2.3赋值重载 3.迭代器 3.1初识迭代器 3.2迭代器的使用 3.3特殊迭代器 3.4范围for 4.大小接口 4.1字符长度相关接口 4.2空间大小相关接口 5.其他常用接口 5.1operator[ ] 5.2增 5.3查 5…

Android 代码阅读环境搭建:VSCODE + SSH + CLANGD(详细版)

在阅读Android源码&#xff08;AOSP超过1亿行代码&#xff09;时&#xff0c;开发者常面临索引失败、跳转卡顿等问题。本教程将手把手教你搭建基于VSCode SSH Clangd的终极阅读环境&#xff0c;实现秒级符号跳转、精准代码提示和高效远程开发。 一、环境架构解析 1.1 方案组…

JAVA 集合的进阶 泛型的继承和通配符

1 泛型通配符 可以对传递的类型进行限定 1.1 格式 ? 表示不确定的类型 &#xff1f;extends E&#xff1a; 表示可以传递 E 或者 E 所有的子类类型 &#xff1f;super E&#xff1a; 表示可以传递 E 或者 E 所有的父类类…

改写自己的浏览器插件工具 myChromeTools

1. 起因&#xff0c; 目的: 前面我写过&#xff0c; 自己的一个浏览器插件小工具 最近又增加一个小功能&#xff0c;可以自动滚动页面&#xff0c;尤其是对于那些瀑布流加载的网页。最新的代码都在这里 2. 先看效果 3. 过程: 代码 1, 模拟鼠标自然滚动 // 处理滚动控制逻辑…

由sigmod权重曲线存在锯齿的探索

深度学习的知识点&#xff0c;一般按照执行流程&#xff0c;有 网络层类型&#xff0c;归一化&#xff0c;激活函数&#xff0c;学习率&#xff0c;损失函数&#xff0c;优化器。如果是研究生上课学的应该系统一点&#xff0c;自学的话知识点一开始有点乱。 一、激活函数Sigmod…

仿腾讯会议——优化:多条tcp连接

1、添加用户信息结构 2、添加注册视频音频结构体 3、 完成函数注册视频音频

File—IO流

因为变量&#xff0c;数组&#xff0c;对象&#xff0c;集合这些数据容器都在内存中&#xff0c;一旦程序结束&#xff0c;或者断电&#xff0c;数据就丢失了。想要长久保存&#xff0c;就要存在文件中&#xff08;File&#xff09; 文件可以长久保存数据。 文件在电脑磁盘中…