// Copyright Epic Games, Inc. All Rights Reserved. #include "MetasoundFrontendQuerySteps.h" #include "Algo/MaxElement.h" #include "MetasoundFrontend.h" #include "MetasoundFrontendDocument.h" #include "MetasoundFrontendNodeRegistryPrivate.h" #include "MetasoundFrontendQuery.h" #include "MetasoundFrontendRegistries.h" #include "MetasoundFrontendRegistryTransaction.h" namespace Metasound { class FNodeClassRegistrationEventsPimpl : public IFrontendQueryStreamStep { public: FNodeClassRegistrationEventsPimpl() { TransactionStream = Frontend::FRegistryContainerImpl::Get().CreateTransactionStream(); } virtual void Stream(TArray& OutValues) override { using namespace Frontend; auto AddEntry = [&OutValues](const FNodeRegistryTransaction& InTransaction) { OutValues.Emplace(TInPlaceType(), InTransaction); }; if (TransactionStream.IsValid()) { TransactionStream->Stream(AddEntry); } } private: TUniquePtr TransactionStream; }; FNodeClassRegistrationEvents::FNodeClassRegistrationEvents() : Pimpl(MakePimpl()) { } void FNodeClassRegistrationEvents::Stream(TArray& OutValues) { Pimpl->Stream(OutValues); } FFrontendQueryKey FMapRegistrationEventsToNodeRegistryKeys::Map(const FFrontendQueryEntry& InEntry) const { using namespace Frontend; FNodeRegistryKey RegistryKey; if (ensure(InEntry.Value.IsType())) { RegistryKey = InEntry.Value.Get().GetNodeRegistryKey(); } return FFrontendQueryKey(RegistryKey); } void FReduceRegistrationEventsToCurrentStatus::Reduce(const FFrontendQueryKey& InKey, FFrontendQueryPartition& InOutEntries) const { using namespace Frontend; // Get most recent transaction const FFrontendQueryEntry* FinalEntry = Algo::MaxElementBy(InOutEntries, GetTransactionTimestamp); // Check if most recent transaction is valid and not an unregister transaction. if (IsValidTransactionOfType(FNodeRegistryTransaction::ETransactionType::NodeRegistration, FinalEntry)) { FFrontendQueryEntry Entry = *FinalEntry; InOutEntries.Reset(); InOutEntries.Add(Entry); } else { InOutEntries.Reset(); } } FReduceRegistrationEventsToCurrentStatus::FTimeType FReduceRegistrationEventsToCurrentStatus::GetTransactionTimestamp(const FFrontendQueryEntry& InEntry) { using namespace Frontend; if (ensure(InEntry.Value.IsType())) { return InEntry.Value.Get().GetTimestamp(); } return 0; } bool FReduceRegistrationEventsToCurrentStatus::IsValidTransactionOfType(Frontend::FNodeRegistryTransaction::ETransactionType InType, const FFrontendQueryEntry* InEntry) { using namespace Frontend; if (nullptr != InEntry) { if (InEntry->Value.IsType()) { return InEntry->Value.Get().GetTransactionType() == InType; } } return false; } void FTransformRegistrationEventsToClasses::Transform(FFrontendQueryEntry::FValue& InValue) const { using namespace Frontend; FMetasoundFrontendClass FrontendClass; if (ensure(InValue.IsType())) { const FNodeRegistryTransaction& Transaction = InValue.Get(); bool bSuccess = FMetasoundFrontendRegistryContainer::Get()->FindFrontendClassFromRegistered(Transaction.GetNodeRegistryKey(), FrontendClass); check(bSuccess); } InValue.Set(MoveTemp(FrontendClass)); } FFilterClassesByInputVertexDataType::FFilterClassesByInputVertexDataType(const FName& InTypeName) : InputVertexTypeName(InTypeName) { } bool FFilterClassesByInputVertexDataType::Filter(const FFrontendQueryEntry& InEntry) const { check(InEntry.Value.IsType()); return InEntry.Value.Get().Interface.Inputs.ContainsByPredicate( [this](const FMetasoundFrontendClassInput& InDesc) { return InDesc.TypeName == InputVertexTypeName; } ); } FFilterClassesByOutputVertexDataType::FFilterClassesByOutputVertexDataType(const FName& InTypeName) : OutputVertexTypeName(InTypeName) { } bool FFilterClassesByOutputVertexDataType::Filter(const FFrontendQueryEntry& InEntry) const { return InEntry.Value.Get().Interface.Outputs.ContainsByPredicate( [this](const FMetasoundFrontendClassOutput& InDesc) { return InDesc.TypeName == OutputVertexTypeName; } ); } FFrontendQueryKey FMapClassesToClassName::Map(const FFrontendQueryEntry& InEntry) const { return FFrontendQueryKey(InEntry.Value.Get().Metadata.GetClassName().GetFullName()); } FFilterClassesByClassID::FFilterClassesByClassID(const FGuid InClassID) : ClassID(InClassID) { } bool FFilterClassesByClassID::Filter(const FFrontendQueryEntry& InEntry) const { return InEntry.Value.Get().ID == ClassID; } FFrontendQueryKey FMapToFullClassName::Map(const FFrontendQueryEntry& InEntry) const { const FMetasoundFrontendClass& FrontendClass = InEntry.Value.Get(); return FFrontendQueryKey(FrontendClass.Metadata.GetClassName().GetFullName()); } void FReduceClassesToHighestVersion::Reduce(const FFrontendQueryKey& InKey, FFrontendQueryPartition& InOutEntries) const { FFrontendQueryEntry* HighestVersionEntry = nullptr; FMetasoundFrontendVersionNumber HighestVersion; for (FFrontendQueryEntry& Entry : InOutEntries) { const FMetasoundFrontendVersionNumber& Version = Entry.Value.Get().Metadata.GetVersion(); if (!HighestVersionEntry || HighestVersion < Version) { HighestVersionEntry = &Entry; HighestVersion = Version; } } if (HighestVersionEntry) { FFrontendQueryEntry Entry = *HighestVersionEntry; InOutEntries.Reset(); InOutEntries.Add(Entry); } } bool FSortClassesByVersion::Sort(const FFrontendQueryEntry& InEntryLHS, const FFrontendQueryEntry& InEntryRHS) const { const FMetasoundFrontendVersionNumber& VersionLHS = InEntryLHS.Value.Get().Metadata.GetVersion(); const FMetasoundFrontendVersionNumber& VersionRHS = InEntryRHS.Value.Get().Metadata.GetVersion(); return VersionLHS > VersionRHS; } }