How to setup the development enviroment to compile ValheimPlus yourself.
- Download the BepInEx for Valheim package.
- Extract zip contents and copy the contents inside
/BepInExPack_Valheim/
and paste them in your Valheim root folder and overwrite every file when asked. - This package sets up your Valheim game with BepInEx configurations specifically for mod devs. Created by BepInEx.
- Extract zip contents and copy the contents inside
- Download the AssemblyPublicizer package.
- This package has a tool that you'll use to create publicized versions of the
assembly_*.dll
files for your local development. - Repo: https://github.com/MrPurple6411/Bepinex-Tools/releases/tag/1.0.0-Publicizer by MrPurple6411.
- This package has a tool that you'll use to create publicized versions of the
- Drag and drop all
assembly_*.dll
files from "\Valheim\valheim_Data\Managed" folder onto "AssemblyPublicizer.exe". This will create a new folder called "/publicized_assemblies/". - Define Enviroment Variable
VALHEIM_INSTALL
with path to Valheim Install Directory- example:
setx VALHEIM_INSTALL "C:\Program Files\Steam\steamapps\common\Valheim" /M
- example:
Thanks to mono and unity-mono being open source, we patched and compiled our own mono runtime and enabled actual live debugging of the game and the mod itself with dnSpy.
- Download dnSpy-net-win64 and extract the exe.
- Load all assemblies from <Valheim>\unstripped_corlib into dnSpy (just drag&drop the folder onto it).
- Load all assembly_* from <Valheim>\valheim_Data\Managed into dnSpy (do not load the publicized ones, they will not be loaded into the process and therefore can not be debugged).
- Load ValheimPlus.dll from <Valheim>\BepInEx\plugins into dnSpy.
- Copy .\libraries\Debug\mono-2.0-bdwgc.dll from this repo to <Valheim>\MonoBleedingEdge\EmbedRuntime and overwrite the existing file.
- Now go to
Debug
->Start Debugging
and select Unity debug engine. Select your valheim.exe as the executable and hit OK. - If you did set some breakpoints, the game will halt when it hits the breakpoint in memory and dnSpy will show you the objects in memory and lets you do much more useful stuff.
-
Please add all
Patch
ed methods to the file named for the type being patched. So if we patch a class type namedHumanoid
, add that patch to theGameClasses/Humanoid.cs
file. -
Patch naming convention update
- Any new classes for patches will follow the following naming convention. Please make sure future classes are named appropriately. The objective is to simplify and avoid clashes of method patching and a potential refactor in the future.
// File of patch == GameClass.cs file Class structure: [HarmonyPatch(typeof(GameClass), "MethodName")] {modifiers} class GameClass_MethodName_Patch { bool Prefix... void Postfix... // Etc... }
- If you are planning to do a Transpiler, replace "Patch" with "Transpiler"
-
Please add a
///<summary>
above every patched method and add comments to parts of code that is more that just a simple assignment -
Try to avoid using "magic strings/numbers" if possible, use class-defined
const
s especially if using the value more than once -
Always try avoid returning
false
in aPrefix()
method if possible. This will not only skip the original method but also otherPrefix
s and breaks compatibility with other mods. If you need to returnfalse
please include your reasoning as a comment in the code.
- Always add new configurations to the .cfg file as well as the configuration class and set the value to the default value from the game.
- It is easy to forget test values in your code when you make a PR. Please double check your default values when creating a PR
- It is recommended when testing to test value inline with your code instead of altering the values on the .cfg/class to prevent this issue. If an inline hardcoded value gets pushed, it's easier to spot this mistake than a inaccurate cfg setting.
- Try to find an appropriate existing configuration section before adding a new one. Generally, if your configuration changes something related to the player, for example, then add it to the
[Player]
section. Look at it from the perspective of a user who's looking for a specific configuration instead of the perspective of someone who's coding and looking for a class. - Add any new configuration Sections (
[Section]
) to the bottom of the .cfg file so that we're less likely to break existing config. - Check out our helpers in the /Utilities/ folder and use them where appropriate
- For configuration modifiers that add a modifcation to the base value, always use a percentage configuration and use the
applyModifier()
utility.
- Only make a pull request for finished work. Otherwise, if we pull the work down to test it and it doesn't work, we don't know if it's because it's unfinished or if there's an unintentional bug.
- If you'd like a review on your work before something it's finished, send us a link to a compare via Discord or make a "Draft" PR.
- If you want credit, add your credit to the
README.md
in your pull request if the work is more than a documentation update. We will not be able to track this ourselves and rely on you to add your preferred way of being credited. - After you have made a GitHub contribution, reach out to one of the V+ devs on Discord if you'd like the "GitHub contributor" role.
- pending merge - This work has been reviewed and accepted and will be merged after the coming release
- pending close - This work has not been updated for a while and will be closed if no response/update is received in 2 days
- needs updating - This work has received feedback and needs to be updated before being accepted
- merge conflicts - This pr has merge conflicts that need to be resolved
- question - The author of the PR needs to answer a question before the PR can move forward