Hooks 
Here's a brief overview on how to create and add new hooks in Carbon, as well as adding hooks that are only processed by your plugin and/or module.
Introduction 
Carbon hooks are applied at runtime whenever a plugin needs them, meaning that on a large scale, you can have multiple variations for in-game events.
Guidelines 
There are a few things that need to be met for the hooks to be considered valid and ready to be released to the public, please follow the guide on how to do so.
Community Hooks 
How to get started with submitting community-driven hook changes, requests and additions.
Metadata 
For documentation purposes, it's important to properly describe what the hook does, properly identify its ID and flags.
The attributes below are used by the documentation code-generation system, with outputs used here.
The Info attribute can be applied multiple times.
[MetadataAttribute.Info("This hook does this and that.")]
[MetadataAttribute.Info("This hook also does that and the other.")]Implementation 
A primary requirement is the location of the patch, as well as the Prefix, Postfix and/or Transpiler methods.
[HookAttribute.Patch("OnHookName", "OnHookName [main]", typeof(Type), "Method", [/* Method params */])]Patching 
We use Harmony patches for hooks. Choose between Prefix, Postfix, or Transpiler based on your needs:
public static bool Prefix(BasePlayer ply, ref PatrolHelicopterAI __instance, out bool __result)
{
    if (HookCaller.CallStaticHook(1610282469, __instance, ply) is bool boolean) 
    {
        __result = boolean;
        // Disallow original code from executing
        return false;
    }
    __result = default;
    // Allow original code to execute
    return true;
}IMPORTANT
We do not use Interface.CallHook for our hook system which use direct hook name strings, instead we use hook identifiers as they're way faster to process.
The hook IDs are generated the same way Rust uses with
StringPoolto get numeric identifiers out of string values.
To generate your own hook identifier, put the hook name here: 
Your hook call will look like this: HookCaller.CallStaticHook(..., /*params*/);
Plugin Patches / Hooks 
Create hooks that are exclusively processed by your plugin.
Automatic Patching 
In RustPlugin or CarbonPlugin, Carbon handles all the logic for patching / unpatching your instructions, making it easier so all you need to focus on is the actual implementation of your plugin.
INFO
You need to add this line which lets Carbon know that it should automatically handle everything for you.
using Oxide.Core.Plugins;
[AutoPatch]
[HarmonyPatch(...)]
public class MyPatch { ... }Example:
using System;
using HarmonyLib;
using Oxide.Core.Plugins;
namespace Carbon.Plugins;
[Info("Collaborate", "Carbon Community", "1.0.0")]
public class Collaborate : CarbonPlugin
{
#region Patches
    [AutoPatch] 
    [HarmonyPatch(typeof(BasePlayer), "CanSuicide", new Type[] { })] 
    public class Patch_1
    {
        public static bool Prefix(BasePlayer __instance, ref bool __result)
        {
            Logger.Log("Works!");
            __result = false;
            return false;
        }
    }
#endregion
}While this is entirely Oxide backward compatible, we've expanded on the use of the [AutoPatch] attribute with the following options:
using System;
using HarmonyLib;
using Oxide.Core.Plugins;
namespace Carbon.Plugins;
[Info("Collaborate", "Carbon Community", "1.0.0")]
public class Collaborate : CarbonPlugin
{
#region Patches
    public void OnPatchComplete()
    {
        // Run special code here only if the patch is successful
    }
    [AutoPatch( 
        // Unload plugin if patch fails
        IsRequired = true, 
        // Specify at what time on the plugin's initialization should the patch apply
        Order = AutoPatchAttribute.Orders.AfterOnServerInitialized, 
        PatchSuccessCallback = nameof(OnPatchComplete))] 
    [HarmonyPatch(typeof(BasePlayer), "CanSuicide", new Type[] { })]
    public class Patch_1
    {
        public static bool Prefix(BasePlayer __instance, ref bool __result)
        {
            Logger.Log("Works!");
            __result = false;
            return false;
        }
    }
#endregion
}INFO
Learn more about AutoPatch.
Manual Patching 
Important
NOT RECOMMENDED to do it manually. You can easily break others patches if done wrong.
Manually manage Harmony patches:
using System;
using HarmonyLib;
namespace Carbon.Plugins;
[Info("Collaborate", "Carbon Community", "1.0.0")]
public class Collaborate : CarbonPlugin
{
    public Harmony _PATCH;
    public const string DOMAIN = "com.carbon.mypatch";
    private void Init()
    {
        _PATCH = new Harmony(DOMAIN);
        _PATCH.PatchAll(Assembly.GetExecutingAssembly());
        Puts("Patched.");
    }
    private void Unload()
    {
        _PATCH.UnpatchAll(DOMAIN);
        Puts("Unpatched.");
    }
#region Patches
    [HarmonyPatch(typeof(BasePlayer), "CanSuicide", new Type[] { })]
    public class Patch_1
    {
        public static bool Prefix(BasePlayer __instance, ref bool __result)
        {
            Logger.Log("Works!");
            __result = false;
            // Returning false will prohibit the original code from executing
            // Only applies to Prefixes
            return false;
        }
    }
#endregion
}