Getting Started with Inkpot in Unreal

Learn how to get up and running with Inkpot for the Unreal Engine.

Setting up the Plugin

In the following guide, we’ll assume the reader is proficient with Ink but not Unreal nor Inkpot.

To get started, ensure you have Unreal Engine installed through the Epic Games Launcher.

To use Inkpot, you will also need to install Visual Studio 2022 with specific workloads. See the official documentation for more details.

Find your engine in the Unreal Engine Library and click its Launch button. If it’s your first time launching Unreal, it might ask you to install some pre-requisites and enable some permissions. This is normal so accept all of it to ensure a smooth Unreal experience.

In the Unreal Project Browser, create a Blank Blueprint project. Give the project an appropriate name and set any other parameters as you wish. On creation, a simple level will be visible. We will now add the Inkpot plugin to this project.

Screenshot of Unreal Project Browser

You can get the plugin from Github. Download the archive; we will unzip it into the project folder.

Using File Explorer, navigate to your project folder and create a new folder named Plugins. Extract the archive into that folder.

Unzipping Inkpot into Plugins

Re-open your project. You can do so quickly by opening the uproject file in the File Explorer. It will show a Missing Modules prompt.

The following modules are missing or built with a different engine version:

  Inkpot
  InkPlusPlus
  InkpotEditor

Would you like to rebuild them now?

Click Yes to start building the Inkpot plugin. Wait for the Unreal Editor build dialog to pop up; it could take longer than expected so best to be patient. You should see a new Binaries folder pop up as it does its thing.

Building Inkpot

Should all go well, the Unreal Editor will open again and we’re now ready to enable Inkpot.

Unfortunately, there can be many points of failure here that are beyond the scope of this document; in case of errors, you will have to rely on troubleshooting elsewhere.

Screenshot of Plugins window

Go to Edit > Plugins to invoke the Plugins window where the plugins for the current project can be enabled or disabled. Use the Search filter to find Inkpot. Check it ON! It will prompt to restart the engine; go ahead and do so.

Starting your Ink story

Once you’re back in the engine, pull up the Content Drawer (typically on the bottom left) and click its Import button. Browse to and select an Ink file. This now exists in the project as an InkpotStoryAsset.

You might run into this error when doing the above step.

Inkpot: Error: CompileInkFile_Internal - C:/Users/goldenxp/path/TheIntercept.ink failed to compile ink. Error You must install or update .NET to run this application.
App: C:\Projects\Unreal\GSInkpot\Plugins\Inkpot-1.11.21-release\ThirdParty\InkCommandLine\inklecate.exe
Architecture: x64
Framework: 'Microsoft.NETCore.App', version '5.0.0' (x64)
.NET location: C:\Program Files\dotnet\
The following frameworks were found:
  6.0.14 at [C:\Program Files\dotnet\shared\Microsoft.NETCore.App]
  7.0.3 at [C:\Program Files\dotnet\shared\Microsoft.NETCore.App]
  8.0.5 at [C:\Program Files\dotnet\shared\Microsoft.NETCore.App]
Learn more:
https://aka.ms/dotnet/app-launch-failed
To install missing framework, download:
https://aka.ms/dotnet-core-applaunch?framework=Microsoft.NETCore.App&framework_version=5.0.0&arch=x64&rid=win-x64&os=win10

This happens because inklecate.exe bundled with Inkpot requires .NET 5.0 specifically. Fortunately, the remedy is suggested in the error! So grab the link to install the missing framework and add the .NET 5.0 runtime.

You won’t need to restart Unreal. Re-import your Ink file and you should see the Inkpot Story in your Content Browser.

I’m using the classic Intercept story for this guide. Of course, you can use any of your own!

In order to start our Ink story, we would use the Inkpot subsystem.

Subsystems in Unreal are classes with managed lifetimes. They are great for code organization and access. Think of it like a better Singleton.

Inkpot is an Engine Subsystem meaning its lifetime is shared with the Engine (as opposed to, for example, per-level with a World Subsystem).

To keep this exercise as simple as possible, let’s make a new Empty Level. Go to File > New Level and pick Empty Level. Click Create. We should now have an empty World. Let’s access its Level Blueprint.

Opening the level blueprint

Click the Blueprints icon in the Level Editor toolbar and pick Open Level Blueprint. This will open up the Blueprint Editor.

The Blueprint Editor is where we can connect nodes to instruct Unreal what to do as opposed to explicitly writing lines of code to do the same.

Working with Blueprints

Drag out the white Exec arrow on the right side of the Event Begin Play node till you see Place a New Node and release to bring up the Action List. Type Begin Story; if the list is empty, un-check Context Sensitive. Pick the node and it will drop it in and connect it automatically to Event Begin Play.

We now need to provide the node with a Target Inkpot and an Inkpot Story asset. We have the latter since we imported it but for the former, we need to use the subsystem itself. Drag out the Target blue pin and add Get Inkpot to it. Then, set the node’s story asset to the one you imported.

Begin Story with params set

The Begin Story node will now start the story and also return an Inkpot Story object that we can use in different ways.

At this point, Save All via the File menu or use Ctrl + Shift + S.

Printing Ink Story Lines

Typically with Ink stories, we continue in order to fetch lines. When we cannot continue, it might mean we have a list of choices and need to choose one. We will do the same with our Inkpot story.

We’ll modify our blueprint to support the continue/choose procedure. To reduce lesson scope, we will not handle user input nor build user interfaces. We will make choices automatically and display text in the console with Print String nodes.

In our blueprint, we will add an event that will run whenever we continue our story.

Adding OnContinue

Drag out the right Exec pin of the Begin Story node and type in OnContinue and pick Assign OnContinue from the list. This will drop in a Bind Event to OnContinue node as well as a Custom Event node. You will be put in a rename mode where you can rename the event to whatever you please.

Binding the event for our current story

Connect the Return Value of Begin Story into the Target of the Bind Event to OnContinue node to ensure we register the event on our current story.

Now we’ll set up our event. Drag its Exec pin and type Print String and hit enter to drop said node. We’ll print the current line of the story, drag the event’s Story pin and type Get Current Line and add it. To use this node with the print node, we’ll need to fetch the string within the line. Drag out the Return Value pin of the Get Current Line node and type Get String and add it. Connect its Return Value into In String of the Print String node.

Save all. If you were to run the simulation at this point, nothing would occur because we actually have not performed a continue action yet!

Right after the Bind Event node, drop in and connect a Continue Maximally node.

Continue Maximally means to keep continuing until we reach a choice.

Node network with Continue Maximally

Save your work and hit the green Play button. You might see the first line of your Ink story flash on the viewport. You can also open the Output Log (next to the Content Drawer at the bottom of the default layout) and find the printed statements.

InkPlusPlus: FStory::FStory 0000066D02838710
Inkpot: Display: Flow         : DEFAULT_FLOW
Inkpot: Display: Text         : They are keeping me waiting.
Inkpot: Display: Variables    : forceful = 0
Inkpot: Display:              : evasive = 0
...
...
...
Inkpot: Display:              : longgrasshooperframe = false
Inkpot: Display:              : DEBUG = false
Inkpot: Display: Choice       : 0 - Hut 14  
LogBlueprintUserMessages: [Map_C_1] They are keeping me waiting.

Inkpot itself logs a lot of useful information. But we’ll be focused mainly on LogBlueprintUserMessages since those are the ones we’re making in this lesson.

Working with Choices

We will print the choices next. Right after we print the current line, we’ll run a for-loop to fetch the choices from the story. Since we’ll be using the Inkpot Story object quite a bit, let’s promote it to a local variable for easy access and to avoid blueprint spaghetti.

Promote to variable

Drag the Story pin of the event node and pick the Promote to variable option. Interject the Event and Print exec connection with the new SET node.

Right after the Print node, add a For Each Loop. Then, drag a new Story variable onto the graph. Pick Get Story and drag its Story pin to add a Get Current Choices node to it. Drag its Return Value into the Array of the loop node.

For Loop Node start

Every Inkpot Choice is also an Inkpot Line so to get its string, we can use another Get String node. Connect it to another Print String node and connect that to the Loop Body exec pin. Now, for every available Inkpot Choice, we will print its string.

For Loop Node end

The output now looks like

LogBlueprintUserMessages: [Map_C_2] They are keeping me waiting.
LogBlueprintUserMessages: [Map_C_2] Hut 14

Right after we print our choices, we’ll choose one. We’ll use a Delay node to ensure we don’t pick choices instantly.

Choosing a choice

To make a choice, we’ll drop another Story variable, drag its pin to add a Choose Choice Index node. This lets us pick a choice with an integer, 0 being the first choice. Make sure it’s connected to the Delay node.

After making a choice, we’ll need to continue the story. So drop another Continue Maximally node after Choose Choice Index. Make sure to connect the Story variable to its Target pin.

Output in the viewport

When you run the game, you should now see the Output Log spit out the Ink content. Right after a group of choices are displayed, the first choice will be selected and the story will continue till the end is reached.

Choosing a choice randomly

Of course, we don’t have to stick with selecting the first choice each time. We can pick a random index from the Choices array and use that.

Parting Words

And this concludes a basic primer on using Inkpot in Unreal. We’ve seen how we can setup the Inkpot subsystem to fetch our story content and make choices.