Improvements Over Adam's Hooks

Adam's creation of modding script hooks revolutionized Battle Brothers modding. Before it existed we were unable to mod Battle Brothers except by overwriting files or changing configs. Unfortunately, after years of using and relying on it for all of my mods, I've noticed a lot of ways in which it catches out new and experienced modders alike, causing them to write potentially buggy code and find it difficult to understand exactly what they're doing. Modern Hooks came about as a way to simplify, optimize and extend the features of a hooking system to mod Battle Brothers.

Squirrel Hooks

Raw Hooks

IMPORTANT NOTE

This is an example only you shouldn't use Modern Hooks like this, refer to the section on the new Basic Hooks below

All of the 'raw' hooks in modding script hooks have been turned into a single hook in Modern Hooks. This means that instead of

  • ::mod_hookClass
  • ::mod_hookExactClass
  • ::mod_hookNewObject
  • ::mod_hookNewObjectOnce
  • ::mod_hookBaseClass

we now only have Mod.rawHook, which is actually a drop in replacement for ::mod_hookExactClass (and actually patches hookExactClass to redirect to <Mod>.rawHook), except it is able to apply to ALL Battle Brothers Classes, with no exceptions. However, we should instead use the new Mod.hook as it performs additional error validation and fixes a serious bug in the other hooks.

Hooking Descendants

IMPORTANT NOTE

This is an example only you shouldn't use Modern Hooks like this, refer to the section on the new Basic Hooks below

Additionally, mod_hookDescendants is superseded by Mod.rawHookTree. The distinction is easiest to explain with an example. Let's hook scripts/items/item to add a new function foo to it which prints foo to the log, then let's use hookDescendants to wrap that call and print bar.

::mods_hookBaseClass("items/item", function(o) {
	o.foo <- function() {
		::logInfo("foo");
	}
});
::mods_hookDescedants("items/item", function (o) {
	local foo = o.foo;
	o.foo = function() {
		foo();
		::logInfo("bar");
	}
});

Now, how many items would bar print to the log if you called foo() on a given item? In the example above it will print once for every layer of inheritance between scripts/items/item and the item you are calling foo() on. For example, for scripts/items/weapons/weapon bar would print once, for scripts/items/weapons/sword it would print twice. If you instantiated scripts/items/item and called foo() on it, bar would never print to the log. This is confusing, usually drains performance, and in my experience has never once been a useful feature, in comparison ::Hooks.rawHookTree prints would print bar exactly once for any descendant of item and item itself. Here is the same code using modern hooks:

// this is an example only you should almost never be using raw hooks in modern hooks
<Mod>.rawHook("scripts/items/item", function(p){
	p.foo <- function() {
		::logInfo("foo");
	}
});
<Mod>.rawHookTree("scripts/items/item", function(p){
	local foo = p.foo;
	p.foo = function() {
		foo();
		::logInfo("bar");
	}
});

This would print bar exactly once when calling foo() on any item, including item itself.

New Basic Hooks

Rather than using raw style hooks, when using Modern Hooks you should use Basic Hooks for 99%+ of your hooks. These also:

  • Warn if a mod tries to add a function/field to a class which already exists in the class or an ancestor
  • Warn if a mod tries to set a field or wrap a function which don't exist
  • Warn if a mod wraps a function with a different number of parameters than the function the wrapper returns.
  • Fix a very prevalent bug when wrapping functions of grandparents of the target class. Discord Thread on the subject (credit to LordMidas for discovering it) in the Modding Discord
  • Will likely have additional features added to them to help with finding mod conflicts and other issues.

UI/JavaScript

EarlyJS

Adam's Hooks run the JS files registered with it very late, only after ingame screens are instantiated and the main menu is shown, this can be problematic when trying to hook the main menu or functions that are called during any screen's instantiation, the most common example being createModules and registerDatasourceListener. In Adam's hooks these cases require special handling where you directly modify the screens rather than hooking those functions.

Modern Hooks instead run registered JS and CSS files after all the vanilla files are read and screens are defined, but before screens are instantiated. This means that it is possible to hook all functions normally, and even the screens themselves directly. In some edge cases you might still want to run JS code after screens are instantiated, in which case Modern Hooks provides registerLateJS to achieve that goal.

Dependencies and Queuing

Semantic Versions

Modern hooks adds support for (and in fact requires) semantic versions. Previously, MSU was patching Adam's Hooks to add support for it, now you can use semver even in small mods that don't require MSU.

Dependencies

Dependencies and queuing were an addon to Adam's hooks and therefore the implementation ended up being a little strange with them being handled together despite being separate concepts. Some of these strange behaviors include:

  • It is impossible to require another mod and queue before it (though MSU does patch this out)
  • You can split up your mod requirements across multiple queued functions and they all apply.
  • It is impossible to queue one part of your mod before a mod and a different part after.

In Modern Hooks, Requirements and Incompatibilities are handled separately from mod queuing, which makes a lot more logical sense. Additionally it's also possible to add the name (not just the ID) of the mod you depend on so that users get a more user-friendly error message.

Queuing

Function focused Queuing algorithm

Modern Hooks uses a new queuing algorithm which allows it to handle situations like queuing two separate parts of your mod at different positions relative to another mod. For example, you could queue one part of your mod before MSU, and another after. This isn't possible with Adam's Hooks.

Queue Buckets

Buckets are a new concept allowing specialized mods to queue certain functions earlier or later than all other mods. This is mostly useful for Modern Hooks itself, modding libraries like MSU, and larger mods like Legends or Reforged. MSU for example had its own endqueue system to try and force certain hooks to run after the normal queuing system.

AfterHooks

The AfterHooks QueueBucket allows you to queue code to run after all hooks have completed, this is important as you should not be instantiating squirrel or native objects before all hooks and normal functions have run, as if you do they will fail. This bucket allows you to easily do that.

FirstWorldInit

The FirstWorldInit Queuebucket allows you to queue code to run after the world_state onInitUI has run for the first time and therefore allows you to instantiate bb classes (such as skills) that require certain global objects to exist.

Migration Example

A small mod_hooks mod might look something like

::MyMod <- {
	ID = "mod_my_mod",
	Name = "My Mod",
	Version = "1.0.0" // Assuming MSU is required and loaded
};
::mods_register(::MyMod.ID, ::MyMod.Name, ::MyMod.Version);
::mods_queue(::MyMod.ID, "mod_msu(>=1.0.0), !mod_bad_mod", function() {
	::mods_hookNewObject("ui/global/data_helper", function(o){
		// add a function foo that returns foo
		o.foo <- function()
		{
			return "foo";
		}
		// print addFlagsToUIData to log whenever addFlagsToUIData is called
		local addFlagsToUIData = o.addFlagsToUIData; 
		o.addFlagsToUIData = function(_entity, _activeEntity, _target)
		{
			::logInfo("addFlagsToUIData");
			return addFlagsToUIData(_entity, _activeEntity, _target);
		}
	});
	::mods_hookDescendants("items/item", function(o) {
		// double the buy price of items
		// the below code wouldn't work as expected and would have a variety of issues,
		// but the full version is really convoluted so lets pretend that just
		// doubles the buy price of items
		local getBuyPrice = ::mods_getMember(o, "getBuyPrice"); 
		o.getBuyPrice <- function()
		{
			return getBuyPrice() * 2; 
		}
	})
	::mods_hookExactClass("items/weapons/named/named_shamshir", function(o) {
		// this complicated mess is to avoid the issues around hooking functions that
		// only exist in grandparents or further from the target
		// read more on the Battle Brothers Modding Discord
		// https://discord.com/channels/965324395851694140/1052648104815513670
		local onUpdateProperties = "onUpdateProperties" in o ? o.onUpdateProperties : null;
		local parentName = o.SuperName;
		o.onUpdateProperties <- function( _properties )
		{
			_properties.Stamina += 10;
			if (onUpdateProperties != null)
				onUpdateProperties(_properties);
			else
				this[parentName].onUpdateProperties(_properties);
		}
		// if you didn't perform the validation required to make this not cause issues it would instead look like
		local onUpdateProperties = ::mods_getMember(o, "onUpdateProperties");
		o.onUpdateProperties <- function(_properties)
		{
			_properties.Stamina += 10;
			return onUpdateProperties(_properties);
		}
	});
	::mods_registerCSS("mymod/myfile.css");
	::mods_registerJS("mymod/myfile.js");
});

The equivalent modern hooks mod would look something like

::MyMod <- {
	ID = "mod_my_mod",
	Name = "My Mod",
	Version = "1.0.0" // SemVer no longer requires MSU
};

local mod = ::Hooks.register(::MyMod.ID, ::MyMod.Version, ::MyMod.Name);
mod.require("mod_msu >= 1.0.0");
mod.conflictWith("mod_bad_mod");

mod.queue(">mod_msu", "<mod_swifter", function(){
	mod.hook("scripts/ui/global/data_helper", function(q) {
		q.foo <- function()
		{
			return "foo";
		}

		q.addFlagsToUIData = @(__original) function( _entity, _activeEntity, _target )
		{
			::logInfo("addFlagsToUIData");
			return __original(_entity, _activeEntity, _target);
		}
	});
	mod.hookTree("scripts/items/item", function(q) {
		q.getBuyPrice = @(__original) function()
		{
			return __original() * 2;
		}
	});
	mod.hook("scripts/items/weapons/named/named_shamshir", function(q) {
		q.onUpdateProperties = @(__original) function( _properties )
		{
			_properties.Stamina += 10;
			return __original(_properties);
		}
	});
	::Hooks.registerCSS("ui/mods/mymod/myfile.css");
	::Hooks.registerLateJS("ui/mods/mymod/myfile.js");
	// OR
	::Hooks.registerJS("ui/mods/mymod/myfile.js");
	// registerJS is better and can be used in the vast majority of cases
	// but registerLateJS is technically 1:1 in behavior to ::mods_registerJS
	// and if certain patches were made to work around some of the 
	// strangeness of ::mods_registerJS those are not required in
	// registerJS and therefore should be removed before converting
});