Agrarsense
Sensor.cpp
Go to the documentation of this file.
1// Copyright (c) 2023 FrostBit Software Lab at the Lapland University of Applied Sciences
2//
3// This work is licensed under the terms of the MIT license.
4// For a copy, see <https://opensource.org/licenses/MIT>.
5
6#include "Sensor.h"
7#include "SensorManager.h"
8#include "SemanticColors.h"
9#include "SensorFactory.h"
10
18
19#include "Misc/DateTime.h"
20#include "Kismet/GameplayStatics.h"
21#include "TimerManager.h"
22#include "UObject/Class.h"
23
24TArray<TWeakObjectPtr<UPrimitiveComponent>> ASensor::ComponentsToHide;
25FPrimitiveAdded ASensor::OnPrimitiveAdded;
26
27ASensor::ASensor(const FObjectInitializer& ObjectInitializer) : Super(ObjectInitializer)
28{
29 RootComponent = CreateDefaultSubobject<USceneComponent>(TEXT("SceneRoot"));
30 RootComponent->SetMobility(EComponentMobility::Movable);
31}
32
34{
35 Super::BeginPlay();
36
37 UWorld* World = GetWorld();
38
40 if (ROSHandler)
41 {
42 // Subscribe to ROS changed event
43 ROSHandler->OnROSStateChanged.AddUniqueDynamic(this, &ASensor::ROSBridgeStateChanged);
44 }
45
47 if (ROSInstance)
48 {
49 // Check if ROS is connected at BeginPlay
50 ROSConnected = ROSInstance->IsROSConnected();
51 }
52
53 // Delay logging sensor information message so each sensor have time setup fully.
54 FTimerHandle Handle;
55 World->GetTimerManager().SetTimer(Handle, FTimerDelegate::CreateLambda([this]
56 {
57 if (IsValid(this))
58 {
59 FString ActorInformation = IActorInformation::Execute_GetActorInformation(this);
60 FString Message = FString::Printf(TEXT("Spawned %s"), *ActorInformation);
61
62 SimulatorLog::Log(Message);
63 }
64 }), 0.150f, false);
65
66 // Inform sensor manager that new sensor has been spawned.
68
69 ESensorTypes ThisSensorType = GetSensorType();
70
71 if (ThisSensorType != ESensorTypes::NONE &&
72 ThisSensorType != ESensorTypes::Transform &&
73 ThisSensorType != ESensorTypes::Collision &&
74 ThisSensorType != ESensorTypes::Overlap)
75 {
76 // Spawn a Transform sensor for any sensor type other than NONE, Transform, Collision, or Overlap.
77 // Meaning each sensor, like a Lidar sensor, has its own Transform sensor that publishes its:
78 // - World location
79 // - GNSS location (if world has GNSS setup)
80
81 // We don't need to keep track of this TransformSensor Actor as it should
82 // be automatically destroyed when the parent sensor is destroyed (ASensor EndPlay)
83 FString ActorID = IActorInformation::Execute_GetActorID(this);
84 FString SensorTransformSensorID = ActorID + "/transform";
85
87 Params.SaveTransformDataToDisk = false;
88 Params.OwningActor = this;
89
90 USensorFactory::SpawnTransformSensor(GetActorTransform(), Params, SensorTransformSensorID, "transform", true, this);
91 }
92}
93
94void ASensor::EndPlay(const EEndPlayReason::Type EndPlayReason)
95{
96 // Broadcast that this sensor will be destroyed
97 OnSensorDestroy.Broadcast(this);
98
100
101 Super::EndPlay(EndPlayReason);
102
103 UROSHandler* ROSHandler = UAgrarsenseStatics::GetROSHandle(GetWorld());
104 if (ROSHandler)
105 {
106 ROSHandler->OnROSStateChanged.RemoveDynamic(this, &ASensor::ROSBridgeStateChanged);
107 }
108
109 FString Sensor = UEnum::GetDisplayValueAsText(GetSensorType()).ToString();
110 FString ID = GetSensorIdentifier();
111 FString Message = FString::Printf(TEXT("Destroyed %s sensor with ID: %s"), *Sensor, *ID);
112 SimulatorLog::Log(Message);
113
114 if (EndPlayReason == EEndPlayReason::EndPlayInEditor
115 || EndPlayReason == EEndPlayReason::Quit)
116 {
117 for (auto& WeakComponentPtr : ComponentsToHide)
118 {
119 WeakComponentPtr.Reset();
120 }
121 ComponentsToHide.Empty();
122 }
123
124 // Destroy all Actors that are attached to this sensor like:
125 // Sensor model Actor and TransformSensor Actor.
126 TArray<AActor*> AttachedActors;
127 GetAttachedActors(AttachedActors);
128 for (AActor* Actor : AttachedActors)
129 {
130 if (Actor)
131 {
132 Actor->Destroy();
133 }
134 }
135
136 ROSInstance = nullptr;
137
138 if (LogFile)
139 {
140 LogFile->Destroy();
141 LogFile = nullptr;
142 }
143
145}
146
147FString ASensor::ExportToJsonFile(const FString& FileName)
148{
149 return USimulatorJsonExporter::ExportSensorToJSON(FileName, this);
150}
151
153{
154 AVehicle* VehicleActor = Cast<AVehicle>(ParentActor);
155 if (VehicleActor)
156 {
157 return VehicleActor;
158 }
159 return nullptr;
160}
161
163{
164 switch (ROSState)
165 {
167 ROSConnected = true;
169 break;
170
172 ROSConnected = false;
174 break;
175 }
176}
177
179{
180 if (ROSTopic)
181 {
182 ROSTopic->Unadvertise();
183 ROSTopic->Unsubscribe();
184 ROSTopic->MarkAsDisconnected();
185 ROSTopic->ConditionalBeginDestroy();
186 ROSTopic = nullptr;
187 }
188}
189
191{
192 if (!SendDataToROS
193 || ROSTopic
194 || !ROSInstance
195 || !IsROSConnected())
196 {
197 return;
198 }
199
200 FString MessageType;
201 ESensorTypes SensorType = GetSensorType();
202
203 switch (SensorType)
204 {
207 MessageType = "sensor_msgs/PointCloud2";
208 break;
209
214 case ESensorTypes::DVSCamera: // DVS Camera should override this.
215 MessageType = "sensor_msgs/Image";
216 break;
217
219 MessageType = "geometry_msgs/Transform";
220 break;
221
224 MessageType = "std_msgs/String";
225 break;
226
227 default:
228 FString SensorNameString = UEnumUtilities::ConvertSensorTypeToString(SensorType);
229 UE_LOG(LogTemp, Warning, TEXT("Unhandled ROS sensor message creation for sensor: %s"), *SensorNameString);
230 break;
231 }
232
233 FString TopicName = FString::Printf(TEXT("/agrarsense/out/sensors/%s"), *GetSensorIdentifier());
234
235 ROSTopic = NewObject<UTopic>(UTopic::StaticClass());
236 if (ROSTopic)
237 {
238 ROSTopic->Init(ROSInstance->ROSIntegrationCore, TopicName, MessageType);
239 ROSTopic->Advertise();
240 }
241
242 FString Message = FString::Printf(TEXT("%s sensor created ROS topic. Name: '%s' MessageType: '%s' "), *GetSensorName(), *TopicName, *MessageType);
243 SimulatorLog::Log(Message);
244}
245
247{
248 if (FileSavePath.IsEmpty())
249 {
250 FString DataLocation = UAgrarsensePaths::GetDataFolder();
251
253 {
254 FString VehicleName = Vehicle->GetActorID_Implementation();
255 FileSavePath = DataLocation + VehicleName + "/" + GetActorID_Implementation() + "/";
256 }
257 else
258 {
259 FileSavePath = DataLocation + GetActorID_Implementation() + "/";
260 }
261 }
262}
263
265{
266 if (IsValid(LogFile))
267 {
268 // File has already been created, return
269 return;
270 }
271
272 FLogFileSettings Settings;
275 Settings.QueueLength = 10;
276 Settings.KeepFileOpen = false;
277
278 LogFile = NewObject<ULogFile>(ULogFile::StaticClass());
279 if (LogFile)
280 {
281 // FileName is SensorName_ID_UnixTimeStamp.txt
282 FString UnixTimeStamp = FString::FromInt(FDateTime::Now().ToUnixTimestamp());
283
284 FString ActorID = IActorInformation::Execute_GetActorID(this);
285
286 // Temp fix, Remove all "/" from ActorID
287 ActorID.ReplaceInline(TEXT("/"), TEXT(""), ESearchCase::IgnoreCase);
288
289 FString FileName = FString::Printf(TEXT("%s_%s_%s"), *GetSensorName(), *ActorID, *UnixTimeStamp);
290 LogFile->Create(FileName, Settings);
291 }
292}
293
294void ASensor::WriteToLogFile(const FString& Message)
295{
296 if (!IsValid(LogFile) && IsValid(this))
297 {
299 }
300
301 if (IsValid(LogFile))
302 {
303 LogFile->Write(Message);
304 }
305}
306
307void ASensor::HideComponentForAllCameras(UPrimitiveComponent* PrimitiveComponent)
308{
309 if (PrimitiveComponent)
310 {
311 TWeakObjectPtr<UPrimitiveComponent> Component(PrimitiveComponent);
312 ComponentsToHide.Add(Component);
313 OnPrimitiveAdded.Broadcast(PrimitiveComponent);
314 }
315}
316
317TMap<FString, FColor> ASensor::GetSemanticColors()
318{
320}
321
323{
324 FDateTime Now = FDateTime::Now();
325 FString TimeStamp = FString::Printf(TEXT("%02d:%02d:%04d:%02d:%02d:%02d:%03d"),
326 Now.GetDay(), Now.GetMonth(), Now.GetYear(),
327 Now.GetHour(), Now.GetMinute(), Now.GetSecond(), Now.GetMillisecond());
328
329 return TimeStamp;
330}
EROSState
Definition: ROSState.h:16
ESensorTypes
Definition: SensorTypes.h:15
@ SemanticSegmentationCamera
static TArray< TWeakObjectPtr< UPrimitiveComponent > > ComponentsToHide
Definition: Sensor.h:411
virtual void DestroyROSTopic()
Definition: Sensor.cpp:178
virtual void CreateLogFile()
Definition: Sensor.cpp:264
void ROSBridgeStateChanged(EROSState ROSState)
Definition: Sensor.cpp:162
UROSIntegrationGameInstance * ROSInstance
Definition: Sensor.h:375
static void HideComponentForAllCameras(UPrimitiveComponent *PrimitiveComponent)
Definition: Sensor.cpp:307
virtual void BeginPlay() override
Definition: Sensor.cpp:33
FSensorDestroy OnSensorDestroy
Definition: Sensor.h:283
static FPrimitiveAdded OnPrimitiveAdded
Definition: Sensor.h:377
FString GetSensorIdentifier() const
Definition: Sensor.h:75
UTopic * ROSTopic
Definition: Sensor.h:361
virtual void CreateDataSavePath()
Definition: Sensor.cpp:246
AActor * ParentActor
Definition: Sensor.h:370
FString GetSensorName() const
Definition: Sensor.h:96
virtual ESensorTypes GetSensorType() const
Definition: Sensor.h:65
AVehicle * IsAttachedToVehicle() const
Definition: Sensor.cpp:152
ULogFile * LogFile
Definition: Sensor.h:367
FString ExportToJsonFile(const FString &FileName)
Definition: Sensor.cpp:147
virtual FString GetActorID_Implementation() const override
Definition: Sensor.h:216
bool SendDataToROS
Definition: Sensor.h:364
FString CreateTimeStampString() const
Definition: Sensor.cpp:322
bool ROSConnected
Definition: Sensor.h:409
FString FileSavePath
Definition: Sensor.h:372
void WriteToLogFile(const FString &Message)
Definition: Sensor.cpp:294
FORCEINLINE bool IsROSConnected() const
Definition: Sensor.h:201
virtual void EndPlay(const EEndPlayReason::Type EndPlayReason) override
Definition: Sensor.cpp:94
virtual void CreateROSTopic()
Definition: Sensor.cpp:190
ASensor(const FObjectInitializer &ObjectInitializer)
Definition: Sensor.cpp:27
static TMap< FString, FColor > GetSemanticColors()
Definition: Sensor.cpp:317
static void Log(const FString &Message, bool LogToTextFile=true, bool LogToROS=true)
static FString GetDataFolder()
static UROSIntegrationGameInstance * GetROSGameInstance(const UObject *WorldContextObject)
static UROSHandler * GetROSHandle(const UObject *WorldContextObject)
static FString ConvertSensorTypeToString(ESensorTypes Sensortype)
void Destroy()
Definition: LogFile.cpp:174
void Write(const FString &Text)
Definition: LogFile.cpp:123
void Create(const FString &FileNameWithoutExtension, FLogFileSettings Settings)
Definition: LogFile.cpp:40
FROSDelegate_ROState OnROSStateChanged
Definition: ROSHandler.h:81
static TMap< FString, FColor > GetSemanticColors()
static ATransformSensor * SpawnTransformSensor(const FTransform &transform, FTransformSensorParameters Parameters, const FString sensorIdentifier, const FString sensorName, bool SimulateSensor=true, AActor *Parent=nullptr)
static void AddSensor(ASensor *SensorPtr)
static void RemoveSensor(ASensor *SensorPtr)
static FString ExportSensorToJSON(const FString &FileName, ASensor *Sensor)
bool KeepFileOpen
Definition: LogFile.h:42
FFileWriteOptions FileWriteOptions
Definition: LogFile.h:45
int32 QueueLength
Definition: LogFile.h:48
FFileCreationOptions FileCreationOptions
Definition: LogFile.h:36