This mod allows easy edits to some game values in Phoenix Point without needing to build a full mod for each one. The values to change are put into a configuration file and the file is read an applied when the game loads. If you are familiar with INI mods in XCOM games, this can do something similar. Unfortunately this is not quite as straightforward as .ini modding - a big part of making this useful is finding the value you want to change and this is currently not very easy.
Phoenix Point doesn't have official mod support. This mod may break or your own config mods may stop working whenever the game is patched, leaving your campaign in an inconsistent and maybe unplayable state.
Hacking up the defs can also cause unexpected behaviour in the game. Before you report a bug please make sure you verify with an unmodded game, otherwise you may just be making life much more difficult for the devs. They don't need to waste time trying to figure out how something that should be impossible in the base game has happened because you made it possible with a mod.
Many game constants can be changed by this mod. Some examples are:
- Carrying capacity of vehicles/aircraft
- Cost of equipment or recruits
- Equipment values like armor values, magazine sizes, etc.
- Difficulty level settings, such as whether or not new recuits come with equipment.
- Enabling/disabling the console
- Localization strings
This system won't let you make changes to game code itself, like changing how an ability works. This kind of change would currently require a full DLL mod.
Easier than making a C# mod, but finding the things to change is currently tough.
First because it's easier - you don't need VS or to know how to build a C# project, you just need a text editor. Downloading DLLs also has some risk and requires you to trust the author (or to read the source and rebuild the DLL yourself). This method requires only trusting the base mod and each actual change is just a text description. It also can help you make mods like an existing one by changing values or related assets instead of having to wait for someone to build that mod.
This mod requires a Phoenix Point mod loader. Both Modnix and the Phoenix Point Mod Injector are supported. Install one of these first.
In the Mods tab of Modnix click the Add Mod
button and select the downloaded PPDefModifier zip file. The PPDefModifier folder is created automatically, and can be reached by clicking the Mods Folder
button in Modnix to add configuration files.
Open the downloaded PPDefModifier.zip file and copy PPDefModifier.dll
to your Mods
folder created by the Mod Injector. Create a PPDefModifier
directory in the Mods
directory (or copy the empty one from the zip file).
For the mod to do anything you must provide one or more config files that list the changes you want to apply to the game. The mod will read all .json files in a folder named PPDefModifier
in the Mods
folder, including those in other subdirectories, applying all mods found in each file. This is the preferred mechanism to write mods that are small and easy to share. For backward compatibility with the first release the mod will also read a file named PPDefModifier.json
in the Mods
directory if it exists.
Each config file may contain a list of mods to apply. Each mod contains:
- Either a guid for the asset to change or the class name of a static field to change
- The name of the field you want to change within that asset or class
- The new value you want to set for this field
When changing more than one value in a single def or class, you can instead use a modletlist
to specify multiple fields and values instead of field
or value
. The modletlist
is an array of objects, each containing a field and value.
[
{
"cls": "Base.Utils.GameConsole.GameConsoleWindow",
"field": "DisableConsoleAccess",
"value": 0,
"comment": "Enable console access"
},
{
"guid": "228f2cd8-8ca2-4224-ead6-c9c684f52172",
"field": "BaseStats.SpaceForUnits",
"value": 8,
"comment": "Set Manticore carrying capacity to 8"
},
{
"guid": "b0011318-4762-27d1-3ada-e10fa2d4ab34",
"modletlist": [
{
"field": "SoldiersCapacity",
"value": 12
},
{
"field": "ItemsCapacity",
"value": 36,
},
],
"comment": "LivingQuarters: Sets soldier capacity to 12 and item capacity to 36"
},
]
This example file contains three mods:
The first enables console access by changing the value of the static field DisableConsoleAccess
within the class Base.Utils.GameConsole.GameConsoleWindow
to false
(0).
The second entry changes the carrying capacity of the Manticore to 8 soldiers. The Manticore is identified by the guid "228f2cd8-8ca2-4224-ead6-c9c684f52172"
, and the carrying capacity is within the BaseStats
struct in a field named SpaceForUnits
.
The final changes the Living Quarters facility to have capacity for 12 soldiers and 36 items. This example shows the use of modletlist
to change two different fields within the same def. This can save a little space instead of needing to repeat the guid
multiple times.
Each mod needs one of either cls
or guid
. Either modletlist
or both value
and field
are required in all mods. comment
is optional and allows you to provide a human-readable description of the mod, especially useful for guids.
If you are unfamiliar with Json: When building your own configuration file the square brackets [
and ]
are needed, even when writing only one mod. Each mod definition goes in curly braces {
}
, and are separated by commas. There should be no comma after the last entry before the closing ]
. It may be helpful to paste the file into a Json validator (available online) to make sure it's a valid json file.
First, note that changing some values may require a new campaign to take effect. These mods change the base value from the game, but any instances that have already been created in your save may still have the old value. Depending on exactly what the mod changes you may be able to just wait until new instances are generated (building new equipment, for example) without needing a whole new campaign.
The mod will write log information if it fails to read your config file or fails to apply a mod. The log file can be found
in Users\YourUsername\AppData\LocalLow\Snapshot Games Inc\Phoenix Point\outlog_log.txt
, search for "PPDefModifier" for logs from the mod that will hopefully point you at what has gone wrong. If the log just indicates the file failed to parse you may have an error in your file. Pasting it into a Json validator site may help to find errors in your config.
The biggest difficulty is in finding where the thing you want to change is in the game files. There is no single method to this that will always work, there is some detective work involved. Some possibilities are:
- The value is stored in an asset, like the Manticore size above. Most values for soldiers, enemies, equipment, etc are probably in here. Finding the thing you want to change will likely involve looking through the asset files and just searching for names that sound likely.
- The value is stored in a field somewhere in the game code, like the console mod above. This is fairly rare. Finding such a value will require digging around in a program like
dnSpy
. - The value is hardcoded into the game code itself. Such values can't be changed by this mod, but you may be able to change the code behaviour with a full C# mod. Again
dnSpy
can help you to find out where this is, but this mod won't be able to modify it.
Some useful tools for locating the values and/or assets are dnSpy to inspect the game code or an asset viewer such as AssetStudio. Note that the main AssetStudio project can't process some of the Phoenix Point asset definitions. Until this is fixed in that project you can use my fork of the project that fixes the problem. See here. The release page has a binary build with the patch.
Many game assets are stored in the sharedassets0.assets
file in the PhoenixPointWin64_Data
folder for the game. The latest release of AssetStudio as well as the Unity Asset Bundle Extractor both have issues reading the Phoenix Point asset files, especially for equipment. See the section above for a link to a patched AssetStudio that can read these files.
To search for assets select "Open File" from the File menu in AssetStudio and select the sharedassets0.assets
file. In the Asset List
tab enter a search word in the text box below the tab, and select the entry that looks like it might be the one you want. After choosing a dialog will pop up asking you to select the Assembly folder. Select the Managed
folder under PhoenixPointWin64_Data
and click "Select Folder".
The following screenshot shows an example for the Manticore capacity mod:
The entry in the asset list is named PP_Manticore_Def
. The fields you are interested in will usually be of type MonoBehaviour
, but the names may be hard to guess and may require either poking around in dnSpy or just searching for
likely names and seeing if anything useful pops up. They will usually be named with _Def
on the end, possibly with another word such as _EquipmentDef
or _WeaponDef
.
The highlighted lines are the ones that you need to build the mod config. The first highlighted line shows the guid. This is always named Guid
and is the value to use in the guid
field of the mod. The next two show the path to the value we want to change. SpaceForUnits
has the value 6, so that looks like the one we want. It is indented in the dump, showing that this field is not directly within the object itself but is nested within a field called BaseStats
. The field
value to use is the names of any nested objects separted by .
followed by the name of the field to change itself. In this case, "BaseStats.SpaceForUnits"
, and the value
field can be set to the value you want (in this example, 8):
{
"guid": "228f2cd8-8ca2-4224-ead6-c9c684f52172",
"field": "BaseStats.SpaceForUnits",
"value": 8,
"comment": "Optional string describing this mod"
}
Alternatively, the Dump Data mod can produce xml data files in a structure that reflect their in-game hierarchy.
Follow the mod's guide to extract the data.
Using the same example as above, you can find Manticore in GeoActorDef, and see that you can change the value of BaseStats.SpaceForUnits
.
When you find the guid of a data, becareful to not use the guid of its ViewElement.
In the screenshot above, 228f2cd8-...
is the correct one that contains BaseStats, not 23600e0a-...
.
Mod configuration files are json files. This mod will read all .json files in the PPDefModifier
directory under Mods
or any subdirectory within it, as well as a file named PPDefModifier.json
if it exists in the Mods
directory directly. Placing individual mod configuration files in a PPDefModifier
directory or a subdirectory below it is the preferred method, the PPDefModifier.json
file in the Mods
directory is for backwards compatibility with the original release.
Each configuration file must contain an array defining the mod objects. Each mod in the array has the following fields:
The guid
field is necessary when making a change to a game object that is implemented as an asset and should be set to a string containing the guid of the definition. See below for a guide on how to read the assets and find entries you may be looking for.
Exactly one of guid
or cls
must be in each mod.
The cls
field is necessary when making a change to a game value that is not stored as an asset, e.g. the flag that disables the console. The cls
field should be a fully-qualified class name with all namespaces. The field
field of the mod should name a static field of this class that you wish to change.
The field
field specifies the name of the field to change and should name a field of some primitive type (e.g. an integer, float, or boolean value) or a string. Handling other values (such as references to other assets) will hopefully come soon. For cls
type mods field
should name a static member of the class. For guid
type mods it does not need to be static.
The field is specified as a dot-separated list of names. The names correspond to the names of the members within the def specified by the guid. For example, "BaseStats.SpaceForUnits" indicates the field SpaceForUnits
within the class-type member named BaseStats
within the guid def. These names can be constructed from a dump from AssetStudio by looking at the indention level of the names. When the name of the field you want to modify is indented you must look back up in the dump to find the parent object's name. This may in turn be nested within another object, etc.
Array-type fields are supported and can be named with the syntax [N]
where N
is a constant number. For example, the field DamagePayload.DamageKeywords[0].Value
would name the Value
field within the first element of the DamageKeywords
array in the DamagePayload
object for the def specified by the guid.
The new number to set the field to. Booleans can be specified as 1 (true) or 0 (false).
An optional field used to describe the mod, since json does not support comments. Especially useful for guid mods to describe what the guid refers to.
If more than one change is needed in a single cls
or guid
, you can combine them in a modletlist
instead of specifying a full mod block for each change and repeating the guid or class in each. modletlist
is used instead of field
and value
, and is an array of objects containing both field
and value
. The mod will apply the changes specified in each
element of the modletlist.
The flags
field is a list of string that affect PPDefModifier's behaviour when applying the mod object.
When the first mod object have this flag, e.g. flags:[{"Actions"}]
, the .json file will be deferred if both the installed PPDefModifier and the installed Modnix supports Modnix Actions.
Only use this flag when the json mod is part of a Modnix 3 Action Mod (Example). Standalone json mods should not carry this flag.
- Find out if its possible to specify assets by path instead of by guid for more readability
- Add support for more data types, e.g. defs that refer to other defs