Tokenized Messages
To enrich your debugging experience in the editor, make use of Tokenized Messages to empower your team to more quickly find problems and fix them at the source!
If you've ever noticed Blueprint errors after a PIE session, you'll see that they includes links that send you directly to a particular Blueprint node. This is because Epic is making use of Tokenized Messages in the Message Log. This article will outline how to use the FTokenizedMessage
struct with Message Logs. We recommend that you look over the use of Message Logs before proceeding to use Tokenized Messages.
Before beginning, ensure that you've included the MessageLog module in your Build.cs file
Mental Preparation
Before we begin, it helps to be aware that the Logging and Message features should be treated as Editor-Only. Any snippets of code or functions called within tokens should be wrapped inside of our #if WITH_EDITOR
or #if WITH_EDITORONLY_DATA
directives. These prevent editor-only features from being included in packaged builds and throwing errors at package time. You'll also want to wrap header includes with the same directives.
FTokenizedMessage
A tokenized message needs to be created before we can add anything to it. We can do this by calling the Create
static method on the FTokenizedMessage
class.
#if WITH_EDITOR
#include "Logging/TokenizedMessage.h"
#include "Logging/MessageLog.h"
#endif
#if WITH_EDITOR
void AMyLoggingClass::LoggingClassMethod()
{
FMessageLog PIE("PIE")
TSharedRef<FTokenizedMessage> ErrorMessageToken = FTokenizedMessage::Create(EMessageSeverity::Type::Error);
}
#endif
When we eventually add all of the tokens to our message and are ready to pass it to the message log, we do so like this:
PIE.AddMessage(ErrorMessageToken);
When we create the Tokenized Message, we need to select a severity for the message as you can see above. The options for severity are:
namespace EMessageSeverity
{
enum Type : int
{
CriticalError UE_DEPRECATED(5.1, "CriticalError was removed because it can't trigger an assert at the callsite. Use 'checkf' instead.") = 0,
Error = 1,
PerformanceWarning = 2,
Warning = 3,
Info = 4,
};
}
Note that CriticalError
is deprecated as of 5.1.
So if we wanted to pass an empty tokenized message to the message log, the full function would look like this:
void AMyLoggingClass::LoggingClassMethod()
{
FMessageLog PIE("PIE")
TSharedRef<FTokenizedMessage> ErrorMessageToken = FTokenizedMessage::Create(EMessageSeverity::Type::Error);
PIE.AddMessage(ErrorMessageToken)
}
However, that's not much different from adding an FText
message to the log. So let's look at what tokenized messages can offer — tokens.
Adding Your First Token
To add a token to a message, you can use the arrow operator on the FTokenizedMessage sharedref.
void AMyLoggingClass::LoggingClassMethod()
{
FMessageLog PIE("PIE")
TSharedRef<FTokenizedMessage> ErrorMessageToken = FTokenizedMessage::Create(EMessageSeverity::Type::Error);
ErrorMessageToken->AddText(FText::FromString("The text part of the message. "))
PIE.AddMessage(ErrorMessageToken)
}
The AddText()
method creates what's called an FTextToken
, which is one of the 14 types of tokens as of Unreal Engine 5.5. The Text Token simply adds and displays text messages wherever they're added in the flow of the Tokenized Message.
If we wanted to add another token after that one, we would do so like this:
void AMyLoggingClass::LoggingClassMethod()
{
FMessageLog PIE("PIE")
TSharedRef<FTokenizedMessage> ErrorMessageToken = FTokenizedMessage::Create(EMessageSeverity::Type::Error);
ErrorMessageToken->AddText(FText::FromString("The text part of the message. "))
->AddText(FText::FromString("Here's some more text."))
PIE.AddMessage(ErrorMessageToken)
}
Each token creation method returns the reference to the FTokenizedMessage
that it's a part of. This allows us to chain token addition calls with ease.
Hopefully by now you can see the utility of tokenized messages, at least to a certain degree. Before we go further, here are all 14 token types.
namespace EMessageToken
{
enum Type
{
Action,
Actor,
AssetName,
AssetData,
Documentation,
Image,
Object,
Severity,
Text,
Tutorial,
URL,
EdGraph,
DynamicText,
Fix,
};
}
Each token inherits from the IMessageToken
abstract class. We'll go over each one in detail do you can see how they work.
Action Token
Add documentation for the Action Token
Actor Token
Add documentation for the Actor Token
Asset Name Token
Add documentation for the Asset Name Token
Asset Data Token
Add documentation for the Asset Data Token
Documentation Token
Add documentation for the Documentation Token
Image Token
Add documentation for the Image Token
Object Token
Add documentation for the Object Token
Severity Token
Add documentation for the Severity Token
Text Token
Add documentation for the Text Token
Tutorial Token
Add documentation for the Tutorial Token
URL Token
Add documentation for the URL Token
EdGraph Token
Add documentation for the EdGraph Token
Dynamic Text Token
Add documentation for the Dynamic Text Token
Fix Token
Add documentation for the Fix Token