Mod Contexts
Mod Contexts are an opt-in advanced alternative of most LinkCache functionality. They provide extra contextual information needed to add additional useful features.
A ModContext
contains:
Record
- The record itselfModKey
- TheModKey
that the associated record came from. Not where it was originally defined and declared, but rather what mod contained the version of the record as it is. (usually the winning override mod)Parent
- If dealing with a "nested" record likePlacedObject
, this will contain a reference to the parent record (like aCell
).
Retrieving a ModContext
Usually you will be provided a state from which ModContexts can be retrieved
By Looping WinningOverrides
foreach (var context in environment.LoadOrder.PriorityOrder.Npc().WinningContextOverrides())
{
Console.WriteLine($"Found npc {context.Record.FormKey} {context.Record.EditorID}, which was found in mod {context.ModKey}");
}
foreach (var context in environment.LoadOrder.PriorityOrder.PlacedObject().WinningContextOverrides(environment.LinkCache))
{
Console.WriteLine($"Found object {context.Record.FormKey} {context.Record.EditorID}, which was found in mod {context.ModKey}");
}
LinkCache Required
Nested records require a LinkCache in case they need to interact with other records
By LinkCache Resolves
IFormLinkGetter<INpcGetter> myFormLink = ...;
if (myFormLink.TryResolveContext<ISkyrimMod, ISkyrimModGetter, INpc, INpcGetter>(environment.LinkCache, out var context))
{
Console.WriteLine($"Found npc {context.Record.FormKey} {context.Record.EditorID}, which was found in mod {context.ModKey}");
}
IFormLinkGetter<INpcGetter> myFormLink = ...;
var context = myFormLink.ResolveContext<ISkyrimMod, ISkyrimModGetter, INpc, INpcGetter>(environment.LinkCache);
Console.WriteLine($"Found npc {context.Record.FormKey} {context.Record.EditorID}, which was found in mod {context.ModKey}");
Complex Generics
This API call requires a lot of generic information. More on that here.
Simple Contexts
If the generic types are too demanding and you only are interested in non-mutating operations, then you can just request "simple" contexts.
IFormLinkGetter<INpcGetter> myFormLink = ...;
if (myFormLink.TryResolveSimpleContext(environment.LinkCache, out var context))
{
Console.WriteLine($"Found npc {context.Record.FormKey} {context.Record.EditorID}, which was found in mod {context.ModKey}");
}
IFormLinkGetter<INpcGetter> myFormLink = ...;
var context = myFormLink.ResolveSimpleContext(environment.LinkCache);
Console.WriteLine($"Found npc {context.Record.FormKey} {context.Record.EditorID}, which was found in mod {context.ModKey}");
Features
Containing ModKey
A ModContext can give you the ModKey of the mod file that contained the record.
var modContext = ...;
Console.WriteLine($"The mod that contained this record: {modContext.ModKey}");
Console.WriteLine($"The mod that originally defined the record: {modContext.Record.FormKey.ModKey}");
ModKey Interpretation Confusion
Be aware that the ModKey from a ModContext is the mod that contained the version of the record contained. This is a separate concept than the ModKey that originated the record initially, which is still accessible through the FormKey
Overridding Deep Records
Records like PlacedObjects are hard to override, as they are nested underneath both Worldspace and Cell records, which must also be overridden in the process. ModContexts abstract this away so that the call is a consistent single call, no matter if it is nested or not.
var modContext = ...;
ISkyrimMod outgoingMod = ...;
var overrideRecord = modContext.GetOrAddAsOverride(outgoingMod);
Parent Concepts
A ModContext has a reference to the parent record in nested scenarios, such as PlacedObject parent Cell or Worldspace
Parent Lookup
var modContext = ...;
if (modContext.TryGetParent<IWorldspaceGetter>(out var worldspace))
{
Console.WriteLine($"The record was underneath a worldspace: {worldspace.FormKey}");
}
var modContext = ...;
if (modContext.TryGetParentContext<IWorldspace, IWorldspaceGetter>(out var worldspaceContext))
{
Console.WriteLine($"The record was underneath a worldspace: {worldspaceContext.Record.FormKey}");
}
var modContext = ...;
if (modContext.IsUnderneath<IWorldspaceGetter>())
{
Console.WriteLine($"The record was underneath a worldspace");
}
var modContext = ...;
if (modContext.TryGetParentSimpleContext<IWorldspaceGetter>(out var worldspaceContext))
{
Console.WriteLine($"The record was underneath a worldspace: {worldspaceContext.FormKey}");
}
Recursive
These calls are recursive, and do not need to be the first parent to succeed. A PlacedObject context can retrieve the parent Worldspace in one call.
Parent Direct Access
var modContext = ...;
if (modContext.Parent.Record is ICellGetter cell)
{
Console.WriteLine($"The parent record was a cell: {cell.FormKey}");
}
Typeless
The type that the parent is expected to be is not contained, so you must query for the correct object type that you expect. It also must be the exact type that the parent is in order to match.
Complex Call Signature
Mod contexts require a lot of information about the types involved:
- Mod Type
- Record Type
- Setter and Getter variants
In some circumstances, these types can be inferred, but sometimes when this isn't the case, then all generics need to be provided.