Handler API Documentation
- Introduction- 1. Game Info
- 2. Mutex
- 3. File System
- 4. Nucleus Co-op Environment
- 5. Focus
- 6. Window manipulation
- 7. Input
- 8. Goldberg Emulator
- 9. Smart Steam Emulator
- 10. Nemirtingas Epic Emulator
- 11. Nemirtingas Galaxy GoG Emulator
- 12. Additional Tools
- 13. Network
- 14. Extra
- 15. Custom prompts
- 16. Support for multiple mice and keyboards
- 17. Path variables
- 18. Other useful lines
 
- Known Issues
Introduction#
Nucleus Co-op is a free and open source tool for Windows that allows split-screen play on many games that do not initially support it, the app purpose is to make it as easy as possible for the average user to play games locally using only one PC and one game copy.
https://github.com/SplitScreen-Me/splitscreenme-nucleus
This is a new and improved official version of the Nucleus Co-op application and is part of the SplitScreen.Me github organization, it includes the following:
1. Game Info#
The name of the game executable with the extension. This will be used by Nucleus to add games, to run the game and as process to follow for positioning and resizing.
  Game.ExecutableName = "Game.exe";If the game is to launch a different executable, other than what Game.ExecutableName is.
  Game.ExecutableToLaunch = "Game.exe";If the game needs to go through a launcher before opening Nucleus will use this exe to start the game and it will still use the Game.ExecutableName process for positioning and resizing.
  Game.LauncherExe = "Launcher.exe";The name of the folder that will be created inside the Nucleus content folder (just use letters not symbols). In this folder Nucleus will store the symlinked or copied game files for each instance of the game.
  Game.GUID = "Game Name";Title of the game that will be shown in Nucleus.
  Game.GameName = "Game Name";The name of the launcher's window title. Some games need to go through a launcher to open. This is needed or else the application will lose the game's window.
  Game.LauncherTitle = "Launcher Window Title";This is just info. It will not limit the players number.
  Game.MaxPlayersOneMonitor = 4;This is just the max players info that shows under the handler name in Nucleus UI. Usually we write the max number of players the game supports. (PC, should support 16 max connected input devices).
  Game.MaxPlayers = 16;2. Mutex#
Specify the type of mutex/handle to kill, for example "Mutant", "Event" or "Semaphore" | default: "Mutant".
  Game.KillMutexType = "Mutant";# of seconds to wait before killing the mutex | Only works with killing not renaming | default: 0.
  Game.KillMutexDelay = 10;Instead of killing the mutex or handle, rename it on startup to something else | Requires Game.KillMutex to contain the EXACT name of the handles for this to work (no partial).
  Game.RenameNotKillMutex = true;When killing handles, should a partial search be done with the handle name? | Renaming handles requires an exact match | default: false.
  Game.PartialMutexSearch = false;Kill the handles, specified in Game.KillMutex, in the last instance (normally last is ignored).
  Game.KillLastInstanceMutex = false;When using ProcessChangesAtEnd, should handles also be killed at the end?.
  Game.KillMutexAtEnd = false;When using Game.MutexProcessExe, here you can specify the handles that need to be killed.
Game.KillMutexProcess = ["Mutexes","To","Close"];When using Game.MutexProcessExe, here you can specify if you want to do a partial search for handles.
  Game.PartialMutexSearchProcess = false;When using Game.MutexProcessExe, here you can specify the type of handles to kill.
  Game.KillMutexTypeProcess = "Mutant";Specify another executable to kill handles for.
  Game.MutexProcessExe = "Process.exe";Wait for X number of seconds before proceeding with mutex killing.
  Game.PauseBeforeMutexKilling = 1000;When using Game.MutexProcessExe, wait for X number of seconds before beginning its mutex killing.
  Game.KillMutexDelayProcess = 1000;3. File System#
If we should symbolic link the game's files to a temporary directory (Nucleus instances folders in its content folder). If not will launch straight from the installation directory.
  Game.SymlinkGame = true;If SymlinkGame is enabled, if we should copy or symlink the game executable.
  Game.SymlinkExe = false;Folders by default are hardcopied, with this enabled, folders will be symlinked instead | warning files placed in symlink folders will appear in the original game files too.
  Game.SymlinkFolders = false;Enable or disable symlink files from being deleted when Nucleus is closed | default: false.
  Game.KeepSymLinkOnExit = true;Symlink individual files from the game directory to the instanced folder.
Game.SymlinkFiles = ["game.exe","settings.cfg"];Copy files from the game directory to the instanced folder.
  Game.CopyFiles = ["game.exe","settings.cfg"]; Instead of Symlinking, create hard copies of the games files (copy/paste) | Be careful of storage, takes a LONG time to start a game when copying.
  Game.HardcopyGame = true;Hardlink files instead of Symlink (or hard copy) | Directories will still be symlinked but files will be hardlinked.
  Game.HardlinkGame = false;Force game to be symlinked each time it is ran.
  Game.ForceSymlink = false;Array with the name of the folders you don't want Nucleus Co-op to symlink, only the folders placed here get hardcopied not the files.
  Game.DirSymlinkExclusions = ["folder1", "folder2"];Array with the name of the files you don't want Nucleus Co-op to symlink, useful if you want to replace files or add external files.
Game.FileSymlinkExclusions = ["file1.txt", "file2.txt"];Array with the name of the files you want Nucleus Co-op to make full copies of, in some cases games need certain files to be full copies or they won't run.
Game.FileSymlinkCopyInstead = ["file1.txt", "file2.txt"];Copy (not symlink) all files within a given folder | Folder name is relative from root game folder.
Game.DirSymlinkCopyInstead = [ "folder1", "folder2" ];When specifying folder(s) to copy all its contents instead of linking, should subfolders and files be included as well?
Game.DirSymlinkCopyInsteadIncludeSubFolders = false;Symlink folders in game root folder to another spot in game folder | Relative paths from root of game folder.
Game.SymlinkFoldersTo = ["folderToMove|whereToMoveIt"];Hardlink folders in game root folder to another spot in game folder | Relative paths from root of game folder.
Game.HardlinkFoldersTo = ["folderToMove|whereToMoveIt"];Will rename the game's executable to "\<exe name> - Player X", x being the instance/player #.
  Game.ChangeExe = true;Copy a file or folder you specify between the quotes to each instance folder, if the file/folder is located in Nucleus folder\utils\User. This function also accepts two additional parameters, a relative path from the game root folder if the file needs to be placed somewhere else within instance folder and one parameter to indicate which instances to copy the file to if it only needs to be in some. Use of parameters is separated by a | character. So it would look something like this [ "filename.ini|\bin|1,2" ]. This example would copy filename.ini from Nucleus\utils\User to Instance0\bin and Instance1\bin. If you don't specify which instances, it will do them all by default. If you don't specify a path then root instance folder is used. If you want instances but root folder you would just do [ "filename.ini||1,2" ] | If copying multiple files, use comma seperated strings
  Game.CopyCustomUtils = [ "d3d9.dll" ]; Will do a text value replacement in a file for every instance, accepts two parameters, one on what to look for, and what it should be replaced with (seperated with a "|") | Works with multiple values | Works in conjunction with HexEditExe (this will trigger first).
  Game.HexEditAllExes = [ "before|afters" ]; Will do a text value replacement in a file for a specific instance, accepts two parameters, one on what to look for, and what it should be replaced with (seperated with a "|") | Works with multiple values, each comma seperated string is the order of the instances | Works in conjunction with HexEditAllExe (this will trigger second).
  Game.HexEditExe = [ "before|afters" ]; Works same as HexEditExe function but with a file you specify as an extra parameter (first one) | filePath is relative path from the game root folder.
  Game.HexEditFile = [ "filePath|before|afters" ]; Works same as HexEditAllExes function but with a file you specify as an extra parameter (first one) | filePath is relative path from the game root folder.
  Game.HexEditAllFiles = [ "filePath|before|afters" ]; Folders (+ all its contents) listed here will not be linked or copied over to Nucleus game content folder, the instance folders.
  Game.DirExclusions = ["dir1"]; Create a steam_appid.txt file where the game executable is.
  Game.CreateSteamAppIdByExe = false; Do not check if Launcher Exe exists in game folder | you will need to provide a relative filepath from game root folder.
  Game.LauncherExeIgnoreFileCheck = false; Game.CopyEnvFoldersToNucleusAccounts = ["Documents", "AppData"]; //Copy subfolders of current user profile to Nucleus user accounts. Copy folders in game root folder to another spot in game folder | Relative paths from root of game folder.
  Game.CopyFoldersTo = ["folderToMove|whereToMoveIt"]; Use this to replace bytes in a file at a specified address, can be for specific instances with optional 3rd argument | 1st arg: instance # (optional), 2nd arg: hex address offset, 3rd arg: new bytes.
  Game.HexEditExeAddress = [ "1|address|bytes" ]; Game.HexEditFileAddress = [ "1|relativePath|address|bytes" ]; //Same as HexEditExeAddress but for a file other than exe | Need to provide relative path (from game root folder) + filename as 1st or 2nd arg if not specifying an instance. Do not display the warning message about a file being deleted .
  Game.IgnoreDeleteFilesPrompt = false; Game.RenameAndOrMoveFiles = [ "1|before.dat|after.dat" ];//Specify files to either rename or move | can accept relative path from root | optional first parameter to specify a specific instance to apply to, omit to do them all. Specify files to be deleted from instanced folder | can accept relative path from root | optional first parameter to specify a specific instance to apply to, omit to do them all.
  Game.DeleteFiles = [ "1|delete.dis" ]; When using Game.LauncherExe, should ExecutableName also be launched? Forces LauncherExeIgnoreFileCheck when game isn't symlinked.
  Game.RunLauncherAndExe = false; Forces LauncherExeIgnoreFileCheck when game isn't symlinked.
  Game.ForceLauncherExeIgnoreFileCheck = false; 4. Nucleus Co-op Environment#
Use custom environment variables for games that use them, replaces some common paths (e.g. AppData) with C:\Users\<your username>\NucleusCoop.
  Game.UseNucleusEnvironment = false; Relative path from user profile (e.g. C:\Users\ZeroFox) to game's config path | Used to provide some extra functionality (open/delete/copy over to Nucleus Environment).
Game.UserProfileConfigPath = "AppData\\Local\\Game\\Config";Relative path from user profile (e.g. C:\Users\ZeroFox) to game's save path | Used to provide some extra functionality (open/delete/copy over to Nucleus Environment).
Game.UserProfileSavePath = "AppData\\Local\\Game\\Saves"; Force the games config files in UserProfileConfigPath to copy over from system user profile to Nucleus environment.
  Game.ForceUserProfileConfigCopy = false; Force the games save files in UserProfileSavePath to copy over from system user profile to Nucleus environment.
  Game.ForceUserProfileSaveCopy = false; Specify files to delete in Nucleus environment config path (UserProfileConfigPath).
Game.DeleteFilesInConfigPath = [ "file.del", "me.too" ];Specify files to delete in Nucleus environment save path (UserProfileSavePath).
Game.DeleteFilesInSavePath = [ "file.del", "me.too" ]; Do not copy files from original UserProfileConfigPath if using Nucleus Environment.
  Game.UserProfileConfigPathNoCopy = false; Do not copy files from original UserProfileSavePath if using Nucleus Environment.
  Game.UserProfileSavePathNoCopy = false; Force the game to use the current user's environment (useful for some games that may require different Window user accounts).
  Game.UseCurrentUserEnvironment = false; Similar to UserProfileConfigPath, use this when the game uses Documents to store game files.
  Game.DocumentsConfigPath = "Path\\Here"; Similar to UserProfileSavePath, use this when the game uses Documents to store game files.
  Game.DocumentsSavePath = "Path\\Here"; When using DocumentsConfigPath, forces a file copy from original location to Nucleus Documents.
  Game.ForceDocumentsConfigCopy = false; When using DocumentsSavePath, forces a file copy from original location to Nucleus Documents.
  Game.ForceDocumentsSaveCopy = false; When using DocumentsConfigPath, do not let Nucleus copy from original location to Nucleus Documents.
  Game.DocumentsConfigPathNoCopy = false; When using DocumentsSavePath, do not let Nucleus copy from original location to Nucleus Documents.
  Game.DocumentsSavePathNoCopy = false; Forces use of custom environment variable when Game.ThirdPartyLaunch = true;
  Game.ForceEnvironmentUse = true; 5. Focus#
Enable or disable the sending of focus messages to each game window at a regular interval | default: false.
  Game.FakeFocus = true; Enable or disable hooks to trick the game into thinking it has focus | default: false.
  Game.HookFocus = true; Enable or disable hooks of functions some games may try and use to prevent multiple instances from running | default: false.
  Game.HookInit = true; Blocks the processing of all the windows messages that get sent when the window loses focus.
  Game.PreventWindowDeactivation = false; Work-around for ForceFocusWindowName having to match 1:1 with game window title for resizing, positioning and focus | default: false.
  Game.HasDynamicWindowTitle = false; Forces the game window title to be whatever is specified in Game.Hook.ForceFocusWindowName (triggers once, after all instances have started) | default: false.
  Game.ForceWindowTitle = false; Adds the process ID to the end of the window title.
  Game.IdInWindowTitle = true; Set the foreground window to be something other than game windows.
  Game.SetForegroundWindowElsewhere = false; If you only want specific instances to have starting hooks, specify them in a comma seperated string.
  Game.StartHookInstances = "1,2,3,4"; If you only want specific instances to have post launch hooks, specify them in a comma seperated string.
  Game.PostHookInstances = "1,2,3,4"; If you only want specific instances to have fake focus messages sent to, specify them in a comma seperated string.
  Game.FakeFocusInstances = "1,2,3,4"; Should the keyboard player instance be skipped when fake focus messages are being sent to.
  Game.KeyboardPlayerSkipFakeFocus = false; Ignore PreventWindowDeactivation if player is using keyboard.
Game.KeyboardPlayerSkipPreventWindowDeactivate = false;Should WM_ACTIVATE message be sent to each instance? | default: true.
  Game.FakeFocusSendActivate = true; Makes sure all the game windows are unfocused so nothing received double input from Windows.
  Game.PreventGameFocus = false; The milliseconds between sending fake focus messages. Default at 1000, some rare games need this to be very low.
  Game.FakeFocusInterval = 1000; Enable each game window at the end (useful if became disabled, or for some games that require this Window function to be called to display properly, after Nucleus setup).
  Game.EnableWindows = false; Do the resizing, repositioning and post-launch hooking of all game instances at the very end | will not work with every option ran normally.
  Game.ProcessChangesAtEnd = false; If ProcessChangesAtEnd = true, pause and show a prompt, before making changes to processes.
  Game.PromptProcessChangesAtEnd = false; If ProcessChangesAtEnd = true, show a prompt between each instance being changed.
  Game.PromptBetweenInstancesEnd = false; 6. Window manipulation#
Prevent games from resizing their windows on their own | Hooks after all instances have been opened (see Game.SetWindowHookStart for an alternative).
  Game.SetWindowHook = true; Prevent games from resizing window their windows on their own | Hooks upon game starting up (see Game.SetWindowHook for an alternative).
  Game.SetWindowHookStart = false; Should the custom dll do the resizing? | Only works with Alpha 10 custom dll | default: false.
  Game.Hook.FixResolution = false; Should the custom dll do the repositioning? | Only works with Alpha 10 custom dll | default: false.
  Game.Hook.FixPosition = false; If manual positioning, what is the window's X coordinate | If both X and Y value > 0, window will be positioned manually.
  Game.Hook.WindowX = 0; If manual positioning, what is the window's Y coordinate | If both X and Y value > 0, window will be positioned manually.
  Game.Hook.WindowY = 0; If manual resizing, what is the window's width | If both ResWidth and ResHeight value > 0, window will be positioned manually.
  Game.Hook.ResWidth = 1280; If manual resizing, what is the window's height | If both ResWidth and ResHeight value > 0, window will be positioned manually.
  Game.Hook.ResHeight = 720; Should the game window try and keep it's aspect ratio when being resized? | default: false.
  Game.KeepAspectRatio = false; After each new instance opens, resize, reposition and remove borders of the previous instance.
  Game.ResetWindows = false; Try and resize game window within player bounds to the aspect ratio of the monitor.
  Game.KeepMonitorAspectRatio = false; Should Nucleus not resize the game windows?.
  Game.DontResize = false; Should Nucleus not reposition the game windows?.
  Game.DontReposition = false; Should Nucleus not make the game windows top most (appear above everything else).
  Game.NotTopMost = false; docs.microsoft.com/en-us/windows/win32/winmsg/window-styles for values that can be used.
  Game.WindowStyleValues = [ "~0xC00000", "0x8000000" ]; docs.microsoft.com/en-us/windows/win32/winmsg/window-styles for values that can be used.
  Game.ExtWindowStyleValues = [ "~0x200", "0x200000" ]; Should each game window be minimized and restored once all instances are opened?.
  Game.RefreshWindowAfterStart = false; docs.microsoft.com/en-us/windows/win32/winmsg/window-styles for values that can be used.
  Game.WindowStyleEndChanges = [ "~0xC00000", "0x8000000" ]; docs.microsoft.com/en-us/windows/win32/winmsg/window-styles for values that can be used.
  Game.ExtWindowStyleEndChanges = [ "~0xC00000", "0x8000000" ]; Ignore logic at end to check if any game window still has a border.
  Game.IgnoreWindowBordercheck = false; Prevents Nucleus from removing game window borders.
  Game.DontRemoveBorders = false; Set the game windows to top most at the very end.
  Game.SetTopMostAtEnd = true; 7. Input#
Enable/disable use of one keyboard/mouse player.
  Game.SupportsKeyboard = true; Set up XInputPlus | If multiple dlls required, use comma seperated strings.
  Game.XInputPlusDll = ["xinput1_3.dll"]; If the keyboard player should be processed first.
  Game.KeyboardPlayerFirst = true; Before launching any games, Nucleus Co-op will open x360ce and let the user set up their controllers before continuing | close x360ce to continue | Don't use with custom dlls.
  Game.UseX360ce = true; Use the xinput custom dll from Alpha 8 instead of Alpha 10 | Will still force alpha 10 custom dll if game is x64.
  Game.Hook.UseAlpha8CustomDll = false; If using XInputPlus or X360ce and there are multiple players playing the same instance, set the # per instance here.
  Game.PlayersPerInstance = 2; Set up Devreorder.
  Game.UseDevReorder = true; When using x360ce, this will set certain hooktype that may work for xbox one controllers if the normal method does not work.
  Game.XboxOneControllerFix = false; Disable raw input devices in game | default: false.
  Game.BlockRawInput = false; If x360ce dll should be named something OTHER than xinput1_3.dll | requires Game.Usex360ce to be set to true.
  Game.X360ceDll = [ "xinput1_3.dll" ]; Create only one file for HID devices per instance (the assigned HID device).
  Game.CreateSingleDeviceFile = false; Enable Mouse/Keyboard input for instances when using Alpha 10 custom xinput dll (normally MKB is restricted).
  Game.Hook.EnableMKBInput = false; Setup wizark952's dinput blocker (block dinput for the game).
  Game.UseDInputBlocker = false; Do not copy XInputPlus' ini when using Game.XInputPlusDll.
  Game.XInputPlusNoIni = false; When using Game.XInputPlusDll, you can specify to use the previous version instead of latest (needed for some games).
  Game.XInputPlusOldDll = false; 8. Goldberg Emulator#
Use the built-in Goldberg features in Nucleus | default: false.
  Game.UseGoldberg = false; Use the experimental branch of Goldberg | Requires Game.UseGoldberg = true | default: false.
  Game.GoldbergExperimental = false; Automatically setup Goldberg's Experimental Steam Client | Requires Game.UseGoldberg and the original steam_api.dll.
  Game.GoldbergExperimentalSteamClient = false; Should Goldberg Lobby Connect be used to connect the instances.
  Game.GoldbergLobbyConnect = false; Do not create a local_save.txt file for Goldberg, saves are to use default game save location.
  Game.GoldbergNoLocalSave = false; Some older games require a steam_interfaces.txt file for Goldberg to work | Will first search orig game path and nucleus games path, if not found then tries to create one with the GoldbergNeedSteamInterface command.
  Game.GoldbergNeedSteamInterface = false; Manually specify what language you want Goldberg to use for the game | by default, Goldberg will use the language you chose in Steam.
  Game.GoldbergLanguage = "english"; If steam_interface.txt is required, provide full path here to the original steam_api(64).dll and Nucleus will create one if it can't find an existing copy.
Game.OrigSteamDllPath = "C:\full path\steam_api.dll"; When set to true, Goldberg will not create a steam_appid.txt file.
  Game.GoldbergIgnoreSteamAppId = false; A list of steam IDs to be used instead of the pre-defined ones Nucleuses uses | IDs will be used in order they are placed, i.e. instance 1 will be first non-empty string in array.
Game.PlayerSteamIDs = ["76561198134585131","76561198134585132"]; Set to true to have Goldberg Experimental rename instanced steam_api(64).dll to cracksteam_api(64).dll.
  Game.GoldbergExperimentalRename = false; Force Goldberg to write account_name.txt and user_steam_id.txt | Requires Game.UseGoldberg;
  Game.GoldbergWriteSteamIDAndAccount = false; 9. Smart Steam Emulator#
If the game needs a fake Steam wrapper (SSE) to run multiple instances (this is not needed if you are using the new use Goldberg emulator line).
  Game.NeedsSteamEmulation = false; When using Game.NeedsSteamEmulation, here you can provide additional lines to write to the SSE ini file.
Game.SSEAdditionalLines = ["Section|Key=Value"];10. Nemirtingas Epic Emulator#
Automatically set up Nemirtinga's Epic Emulator in Nucleus
Game.UseNemirtingasEpicEmu = false;When using Nemirtinga's Epic Emulator, use pre-defined parameters -AUTH_LOGIN=unused -AUTH_PASSWORD=bbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbb -AUTH_TYPE=exchangecode -epicapp=CrabTest -epicenv=Prod -EpicPortal -epicusername=\"" + \<Player Nickname here> + "\" -epicuserid=AAAAAAAAAAAAAAAAAAAAAAAAAAAAAAA -epiclocale=en"
Game.EpicEmuArgs = false;Optional. When using Nemirtinga's Epic Emulator, use pre-defined parameters + Set NickName as epic id, only to use with games that do not use epic id to start or connect(Set clever save names if the game use the epic id to name saves ex: Tiny Tina's Assault On Dragon Keep)" -AUTH_LOGIN=unused -AUTH_PASSWORD=bbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbb -AUTH_TYPE=exchangecode -epicapp=CrabTest -epicenv=Prod -EpicPortal -epicusername=" + \<Player Nickname here> + " -epicuserid="+ \<Player Nickname here> + "-epiclocale=" + EpicLang".
  Game.AltEpicEmuArgs = false; NemirtingasEpicEmu.json edition from a handler example:
    //If you don't need to edit a line do not add it here, the emu will automatically write it with default parameters. More info here: https://gitlab.com/Nemirtingas/nemirtingas_epic_emu/-/blob/master/README.md    // Available debug parameters,  should be "off" by default. Only required to debug the nermintingas eos emulator.    // TRACE: Very verbose, will log DEBUG + All functions enter    // DEBUG: Very verbose, will log INFO  + Debug infos like function parameters    // INFO : verbose     , will log WARN  + some informations about code execution and TODOs    // WARN : not verbose , will log ERR   + some warnings about code execution    // ERR  : not verbose , will log FATAL + errors about code execution    // FATAL: not verbose , will log only Fatal errors like unimplemented steam_api versions    // OFF  : no logs     , saves cpu usage when running the debug versions     // In case of using custom start arguments => -epicusername == same username as in the.json => -epicuserid == same epicid as in the.json >     var jsonPath = Context.GetFolder(Nucleus.Folder.InstancedGameFolder) + "\\nepice_settings\\NemirtingasEpicEmu.json";// + "\\NemirtingasEpicEmus.json" for stable older epic emu version.    var params = [    '{',    '  "appid": "InvalidAppid",',    '  "disable_online_networking": false,',    '  "enable_lan": true,',    '  "enable_overlay": true,',    //'  "epicid": "3808a45790894253344fec21026bbf80",', //better to let the emu automaticaly add this line.    '  "language":' + '"' + Context.EpicLang + '"' + ',',    '  "log_level": "off",',    //'  "productuserid": "ab65359ffde1b5cc41e81afee8e32c33",', //better to let the emu automaticaly add this line.    '  "savepath": "appdata",',    '  "signaling_servers": [],',    '  "unlock_dlcs": true,',    '  "username": ' + '"' + Context.Nickname + '"', //must always be added if you edit the json and must be the last line else the emulator will reset all parameters(there is no coma at the end of this line in the json).    '}'    ] ;    Context.WriteTextFile(jsonPath,params); 11. Nemirtingas Galaxy GoG Emulator#
Nemirtinga's Galaxy Emulator works by substituting the game's Galaxy64.dll file with its own custom dll.
First, set up Nemirtinga's Galaxy Emulator in Nucleus.
Game.UseNemirtingasGalaxyEmu = true;   //Set to true to emulate GOG GalaxyIt is recommended that the emulator is used on the game without Nucleus once before scripting. This can be done by the following steps:
- Head to NucleusCo-op\utils\NemirtingasGalaxyEmu
- Copy the Galaxy64.dllfile.
- Head to the installation folder of the game you wish to emulate.
- Find where Galaxy64.dll is stored. This shall be called the Galaxy folder.
- Rename Galaxy64.dll or move it from the Galaxy folder and paste the Galaxy64.dll from the NemirtingasGalaxyEmu folder.
- Locate and load the game's executable.
This should generate a folder called ngalaxye_settings. This folder contains a .json file that keeps track of vital information. Keep track of where this folder is located as you will need it for the Nucleus script.
- If the game loads fine and the game does not crash when trying to use multiplayer, then the emulator is working. Feel free to remove the ngalaxye_settings folder and Galaxy64.dll, and restore the original Galaxy64.dll file to the Galaxy folder.
With the steps above done, write inside the Game.Play() function:
var idg = Context.PlayerID + 6;var jsonPath = Context.GetFolder(Nucleus.Folder.InstancedGameFolder) + "\\ngalaxye_settings\\NemirtingasGalaxyEmu.json"; //The string should be changed so that ngalaxye_settings is in the same folder as it was in Step 6.var params = ['{','  "api_version": "1.139.2.0",', //Must be changed or the game cannot connect to LAN. API version can be found by right-clicking the original Galaxy64.dll, clicking the 'Details' tab and reading the File Version tag.'  "enable_overlay": false,','  "galaxyid": 14549624462898294' + idg + ',','  "language": ' + '"' + Context.GogLang + '",','  "productid": 2143654691,', //Must be changed or the game will crash. Product ID can be found by visitng https://www.gogdb.org/'  "username": ' + '"' + Context.Nickname + '"', //Must always be added and must be the last line or else the emulator will reset all parameters (there is no comma at the end of this line in the json).'}'];
Context.WriteTextFile(jsonPath,params);Common Errors#
Q. The game's multiplayer only works when GOG Galaxy is open and even then, I can only play with my friends on GOG Galaxy. A. Make sure you set Game.UseNemirtingasGalaxyEmu = true
Q. The game crashes upon entering the multiplayer gamemode or main menu. A. The game is either not loading the custom Galaxy64.dll file or not loading the .json file correctly. Make sure you set Game.UseNemirtingasGalaxyEmu = true, make sure that the ngalaxye_settings folder is in the correct place and make sure the .json file has the correct parameters (i.e. make sure productid and api_version is correct).
Q. The game loads fine but I cannot connect to other players in Nucleus. A. The api_version has been set incorrectly. Go to the game's installation folder, locate the Galaxy64.dll, click the 'Details' tab and read the File Version. That is what your api_version should be.
12. Additional Tools#
Use atom0s' Steamless app to remove Steam Stub DRM from a protected executable.
Game.UseSteamless = true;                               Use this when using Game.UseSteamless = true; always, the command line version of Steamless allows for different launch arguments to be used.
Game.SteamlessArgs = "--quiet --keepbind";              The time in milliseconds to give Steamless to patch the game .exe. 2500 is the default value and will be applied even if the timing line has not been added in a handler.
Game.SteamlessTiming = 2500;                            Use UberPsyX's Steam Stub DRM Patcher to remove Steam Stub DRM from a protected executable.
Game.UseSteamStubDRMPatcher = false;            Force Steam Stub DRM Patcher to use either the x64 or x86 dll | Values: "64" or "86".
Game.SteamStubDRMPatcherArch = "64";            Replace any EasyAntiCheat_(x86)(x64).dll with a bypass dll.
Game.UseEACBypass = false;              Use Flawless Widescreen application | value must be the name of the game plugin folder in PluginCache\FWS_Plugins\Modules.
Game.FlawlessWidescreen = "FWGameName";         (undocumented)
Game.FlawlessWidescreenOverrideDisplay = false;     Game.FlawlessWidescreenPluginPath //(undocumented)
13. Network#
Set up game instances with ForceBindIP; each instance having their own IP.
Game.UseForceBindIP = false;                Highly experimental feature, will change your existing network to a static IP for each instance | option in settings to choose your network.
Game.ChangeIPPerInstance = false;           Game.ChangeIPPerInstanceAlt; //An alternative method to changing IP per instance | this method will create loopback adapters for each player and assign them a static ip on the same subnet mask as your main network interface.
14. Extra#
How many seconds to wait after launching game (or launcher) but before grabbing game process.
Game.PauseBetweenProcessGrab = 30;          Number of seconds to wait after running additional files but before continuing with player setup.
Game.PauseBetweenContextAndLaunch = 0;          Use if the game is launched outside of Nucleus | NOTE: You will not be able to use start up hooks or CMDLaunch with this option.
Game.ThirdPartyLaunch = false;              Ignore the prompt that appears when using Game.ThirdPartyLaunch;
Game.IgnoreThirdPartyPrompt = false;            Manually select the process that will be used for process manipulation, such as resizing, repositioning and used for post-launch hooks.
Game.ForceProcessPick = false;              Prompt the user with a messagebox to let the user decide when to open the next instance | default: false, PauseBetweenStarts STILL required.
Game.PromptBetweenInstances = true;         Show a prompt that user must click on ok to continue, after the first instance is setup.
Game.PromptAfterFirstInstance = false;          Game.GamePlayAfterLaunch; //Call the Game.Play function after the call has launched. Execute Game.Play function (context) before the majority of the game is setup.
Game.GamePlayBeforeGameSetup = false;           Launch each instance from a different user account | must run Nucleus as admin | will temporary create user accounts "nucleusplayerx" and delete them when closing Nucleus.
Game.LaunchAsDifferentUsers = false;            An alternative method to launch each instance from a different user account | must run Nucleus as admin | will temporary create user accounts "nucleusplayerx" and delete them when closing Nucleus.
Game.LaunchAsDifferentUsersAlt = false;         Will backup and restore Nucleus user account user profile's on windows between sessions (when user accounts are not kept).
Game.TransferNucleusUserAccountProfiles = false;    Force Nucleus to search for the game process.
Game.ForceProcessSearch = false;            Game requires Nucleus to be run as administrator (this option will check and advise if detected not running Nucleus as admin).
Game.RequiresAdmin = false;             Game.WriteToProcessMemory //(undocumented) Sets the overall priority category for the associated process. Can be "AboveNormal", "High" or "RealTime" | default: "Normal".
Game.ProcessorPriorityClass = "Normal";         Sets the processors on which the game's threads can run on (provide the # of the cores in a comma-seperated list in quotations) | default: system delegation.
Game.UseProcessor = "1,2,3,4";              Sets the processors on which an instances game's threads can run on (provide the # of the cores in a comma-seperated list in quotations) | default: system delegation.
Game.UseProcessorsPerInstance = [ "1,2","3,4" ];    Sets the preferred processor for the game's threads, used when the system schedules threads, to determine which processor to run the thread on | default: system delegation.
Game.IdealProcessor = 2;                Hide the mouse cursor in game instances.
Game.HideCursor = true;                 Hide the desktop background with a solid black background.
Game.HideDesktop = false;               Most games hide the taskbar when placed on-top but for games that don't you can use this.
Game.HideTaskbar = true;                Display a message to the end-user, that will appear when user selects the game of a handler| useful if there is anything the end-user needs to know before-hand | Only first two or three sentences will appear in UI, but full message can be viewed if user right clicks on the game in the list.
Game.Description = "Hello World";           List of additional processes that need to be killed (other than executable and launcher).
Game.KillProcessesOnClose = [ "kill", "me" ];       Delete a file upon ending game session | Relative paths from root of game folder.
Game.DeleteOnClose = ["DeleteThis.exe"];        Use a Direct X wrapper to try and force DirectX 9 games to run windowed.
Game.UseDirectX9Wrapper = false;            Launch a game using command prompt.
Game.CMDLaunch = false;                 Specify command line options if game is launched using command prompt | requires CMDLaunch to be true, each element is for a different instance.
Game.CMDOptions = ["ops1","ops2"];          When using CMDLaunch (or UseForceBindIP), specify command lines to run BEFORE the game launches in same cmd session, syntax is instance and | symbol as a delimiter and the line you want that instance to run. If you want same line to run all instances, leave out the # and | (only write the line in quotes).
Game.CMDBatchBefore = [ "0|ops1", "1|ops2" ];       When using CMDLaunch (or UseForceBindIP), specify command lines to run AFTER the game launches in same cmd session, syntax is instance and | symbol as a delimiter and the line you want that instance to run. If you want same line to run all instances, leave out the # and | (only write the line in quotes).
Game.CMDBatchAfter = [ "0|ops1", "1|ops2" ];        Wait for X number of seconds before proceeding with CMDBatchBefore.
Game.PauseCMDBatchBefore = 10;              Wait for X number of seconds before proceeding with CMDBatchAfter.
Game.PauseCMDBatchAfter = 10;               Game.CMDBatchClose ["cmd1", "cmd2"]; //Run command lines upon exiting Nucleus. When using CMDLaunch, should the game's starting arguments be inside the same quotations as the game path?
Game.CMDStartArgsInside = false;            Force Nucleus to treat the game as 32 or 64-bit architecture.
Game.ForceGameArch = "x86" (or "x64");          Explicitly disable splitscreen divisons if the game is known to be imcompatible with it.(Does not require to be true for compatible game)Default = true.
Game.SplitDivCompatibility = false;                     15. Custom prompts#
Prompt user for input, which can be then used in handlers logic
This prompts user one time and applies to ALL players, unless a value text file already exists and saving is on.
Game.CustomUserGeneralPrompts = ["Enter ROM name", "Enter filename"];   No documentation yet
Game.SaveCustomUserGeneralValues = false; No documentation yet
Game.SaveAndEditCustomUserGeneralValues = false; This will prompt each player, unless a value text file already exists for that player and saving for players is on.
Game.CustomUserPlayerPrompts = ["Enter network Adapter name", "Enter character name"];  No documentation yet
Game.SaveCustomUserPlayerValues = false; No documentation yet
Game.SaveAndEditCustomUserPlayerValues = false; This will prompt each instance, unless a value text file already exists for that instance and save is on. In case it is not player specific but different values are needed for instances.
Game.CustomUserInstancePrompts = ["Enter network Adapter name"];    No documentation yet
Game.SaveCustomUserInstanceValues = false; No documentation yet
Game.SaveAndEditCustomUserInstanceValues = false; Access the user input values via Context.CustomUser(General/Player/Instance)Values[index]
16. Support for multiple mice and keyboards#
These options are deprecated, see the new Proto Input guide
Game.SupportsMultipleKeyboardsAndMice = true;Game.SendNormalMouseInput = true;Game.SendNormalKeyboardInput = true;Game.ForwardRawKeyboardInput = false;Game.ForwardRawMouseInput = false;Game.SendScrollWheel = false;Game.DrawFakeMouseCursor = true;Game.DrawFakeMouseForControllers = false;Game.HookFilterRawInput = false;Game.HookFilterMouseMessages = false;Game.HookGetCursorPos = true;Game.HookSetCursorPos = true;Game.HookUseLegacyInput = false;Game.HookDontUpdateLegacyInMouseMsg = false;Game.HookGetKeyState = false;Game.HookGetAsyncKeyState = true;Game.HookGetKeyboardState = false;Game.HookMouseVisibility = false;Game.LockInputAtStart = true;Game.LockInputToggleKey = 0x23;             //See https://docs.microsoft.com/en-us/windows/win32/inputdev/virtual-key-codesGame.HookReRegisterRawInput = false;            //Re-register raw input from directly within game process | Recommended to disable forwarding input while using thisGame.HookReRegisterRawInputMouse = true;        Game.HookReRegisterRawInputKeyboard = true;Game.UpdateFakeMouseWithInternalInput = false;17. Path variables#
Path to the source game root folder.
Context.RootInstallFolder                                   Path to the Nucleus Co-op root folder (Nucleus-Coop).
Context.NucleusFolder                                       Path to the Nucleus Co-op handlers root folder (Nucleus-Coop\handlers).
Context.ScriptFolder                        Path to the game handler folder (Nucleus-Coop\handlers\handler_folder).
Game.FolderPath to the Nucleus Co-op current instance root folder (Nucleus-Coop\content\GUID\Instance#).
Context.GetFolder(Nucleus.Folder.InstancedGameFolder)Path to current players Nucleus environment.
Context.EnvironmentPlayer                   Path to Nucleus environment root folder.
Context.EnvironmentRoot                     Relative path from user profile to game's config path | requires Game.UserProfileConfigPath be set.
Context.UserProfileConfigPath                   Relative path from user profile to game's save path | requires Game.UserProfileSavePath be set.
Context.UserProfileSavePath                 Path to current players Nucleus documents environment.
Context.DocumentsPlayer                                     Path to Nucleus documents environment root folder.
Context.DocumentsRoot                                       Relative path from user profile to game's config path | requires Game.DocumentsConfigPath be set.
Context.DocumentsConfigPath                                 Relative path from user profile to game's save path | requires Game.DocumentsSavePath be set.
Context.DocumentsSavePath                                   Path to current players Nucleus environment Windows User root folder.
Context.NucleusUserRoot                                     return "NucleusCo-op\handlers" path.
Context.HandlersFolder                                      Edit an XML element (previously only nodes and attributes).
Context.ChangeXmlNodeInnerTextValue(string path, string nodeName, string newValue)Replace an entire line; for string array use the format: "lineNum|newValue", the | is required.
Context.ReplaceLinesInTextFile(string path, string[] lineNumAndnewValues)Partially replace a line; for string array use the format: "lineNum|Regex pattern|newValue", the | is required.
Context.ReplacePartialLinesInTextFile(string path, string[] lineNumRegPtrnAndNewValues)Removes a given line number completely.
Context.RemoveLineInTextFile(string path, int lineNum)Removes a given line number completely.
Context.RemoveLineInTextFile(string path, string txtInLine, SearchType type)Returns a line number (int), utilizes a newly created enum SearchType.
Context.FindLineNumberInTextFile(string path, string searchValue, SearchType type)note
Context.ChangeXmlNodeInnerTextValue(string path, string nodeName, string newValue)Context.ReplaceLinesInTextFile(string path, string[] lineNumAndnewValues)Context.ReplacePartialLinesInTextFile(string path, string[] lineNumRegPtrnAndNewValues) Context.RemoveLineInTextFile(string path, int lineNum)Context.RemoveLineInTextFile(string path, string txtInLine, SearchType type)Context.FindLineNumberInTextFile(string path, string searchValue, SearchType type)Each of the above methods, also have an overload method so you can specify a kind of encoding to use (enter string of encoding as last parameter, e.g. "utf-8", "utf-16", "us-ascii").
- SearchTypes include: "Contains", "Full" and "StartsWith", use like so: Nucleus.SearchType.StartsWith.
Create a registry key for current user, baseKey can either be "HKEY_LOCAL_MACHINE" or "HKEY_CURRENT_USER".
Context.CreateRegKey(string baseKey, string sKey, string subKey)Delete a registry key for current user, baseKey can either be "HKEY_LOCAL_MACHINE" or "HKEY_CURRENT_USER".
Context.DeleteRegKey(string baseKey, string sKey, string subKey)Edit a registry key for current user, baseKey can either be "HKEY_LOCAL_MACHINE" or "HKEY_CURRENT_USER".
EditRegKey uses a custom registry data type to use, by using Nucleus.RegType.DWord for example. The last word can be of the same name of RegistryValueKind enum.
Context.EditRegKey(string baseKey, string sKey, string name, object data, RegType type)Edit a registry key for current user without Nucleus creating a backup of the registry or before it creates one. Must be placed before any "Context.EditRegKey" line in handler for it to work.
Context.EditRegKeyNoBackupUse this in a game handler to get the player's nickname
Context.Nickname                                        Can be use to edit NemirtingasEpicEmu.json Can be use in start argument to setup user Epic language parameter ex: Context.StartArguments = ' -AlwaysFocus -nosplash -nomoviestartup -nomouse' + Context.EpicLang; (if Epic Language is set to "en" return => "-epiclocale=en", Should not be necessary in most cases)
Context.EpicLang                                                                                Get the raw gamepad guid
Context.GamepadGuid                                                                             Can be use to edit NemirtingasGalaxyEmu.json from handlers
Context.GogLang                                                                                 Useful to make sure that controllers are correctly assigned to the player they are meant to be assigned.
Context.GamepadId                                       Get the x360ce formatted gamepad guid.
Context.x360ceGamepadGuid                                   Local IP address of the computer.
Context.LocalIP                                         Kill processes matching a given process name during preperation.
Context.KillProcessesMatchingProcessName(string name)Kill processes matching a given window name during preperation.
Context.KillProcessesMatchingWindowName(string name)True if current player is using keyboard, useful for logic that is needed for keyboard only player.
Context.IsKeyboardPlayer                                    Player monitor's height.
Context.OrigHeight                                      Player monitor's width.
Context.OrigWidth                                       Player monitor's aspect ratio (e.g. 16:9).
Context.OrigAspectRatio                                     Player monitor's aspect ratio in decimal (e.g. 1.777777).
Context.OrigAspectRatioDecimal                                  Player's aspect ratio (e.g. 16:9).
Context.AspectRatio                                     Player's aspect ratio in decimal (e.g. 1777777).
Context.AspectRatioDecimal                                  Return a string array of filenames (and their paths) found that match a pattern you specify.
Context.FindFiles(string rootFolder, string fileName)Change the creation date of a file.
Context.CreatedDate(string file, int year, int month, int day)Change the last modified date of a file.
Context.ModifiedDate(string file, int year, int month, int day)Specify additional files to run before launching game. By default will run each additional file once but can specify to run during specific player's instances by prefixing the filepath with #|. Replace # with player number. Can also specify to run files for each player by prefixing filepath with "all|". "bool confirm" will only run the file after clicking "Ok" in the prompt.
Context.RunAdditionalFiles(string[] filePaths, bool changeWorkingDir, string customText, int secondsToPauseInbetween, bool showFilePath, bool runAsAdmin, bool promptBetween,bool confirm)Return the value of a provided key as a string.
Context.ReadRegKey(string baseKey, string sKey, string subKey)Context.HandlerGUIDSet launch(start) arguments for the exe from within the Game.Play() function
Context.StartArguments = "";Context.NumberOfPlayersContext.CopyScriptFolder Context.HexEditContext.PatchFileFindAll Context.MoveFolder Context.CopyScriptFolder(string DestinationPath)Context.RandomInt(int min, int max)Context.RandomString(int size, bool lowerCase = false)Context.ConvertToInt()Context.ConvertToString()Context.ConvertToBytes()Context.GCD(int a, int b)Context.ArchContext.PosXContext.PosYContext.MonitorWidthContext.MonitorHeightContext.Log()Context.ProcessIDContext.HasKeyboardPlayer18. Other useful lines#
System.IO.File.Delete("FileToDelete");                      //Delete the specified file.System.IO.File.Copy("SourceFile", "DestinationFile", true); //Copy the specified "SourceFile" to the "DestinationFile" path. You can uses a different name on the DestinationFile to rename the file. The true in the end specify to overwrite, if found, the existing file with the same name.System.IO.File.Move("SourceFile", "DestinationFile", true); //Move the specified "SourceFile" to the "DestinationFile" path. You can uses a different name on the DestinationFile to rename the file. The true in the end specify to overwrite, if found, the existing file with the same name.var OneFolderUP = System.IO.Path.Combine(Path goes here, ".."); // The variable OneFolderUP will return a folder UP of the specified path. Example: var OneFolderUP = System.IO.Path.Combine(Game.Folder, "..");var TwoFolderUP = System.IO.Path.Combine(Path goes here, "..", ".."); // The variable TwoFolderUP will return a folder UP of the specified path. Example: var OneFolderUP = System.IO.Path.Combine(Game.Folder, "..", "..");var OneFolderUP = System.IO.Directory.GetParent("SourceDirectory"); // The variable OneFolderUP will return a folder UP of the specified path ("SourceDirectory").
4. CMD Launch Environment Variables (used with CMDBatchBefore and CMDBatchAfter)%NUCLEUS_EXE%           = Exe filename (e.g. Halo.exe).%NUCLEUS_INST_EXE_FOLDER%   = Path the instance exe resides in (e.g. C:\Nucleus\content\Halo\Instance0\Binaries).%NUCLEUS_INST_FOLDER%       = Path of the instance folder (e.g. C:\Nucleus\content\Halo\Instance0\).%NUCLEUS_FOLDER%        = Path where Nucleus Coop is being ran from (e.g. C:\Nucleus).%NUCLEUS_ORIG_EXE_FOLDER%   = Path the original exe resides in (e.g. C:\Program Files\Halo\Binaries).%NUCLEUS_ORIG_FOLDER%       = Path of the "root" original folder (e.g. C:\Proggram Files\Halo).
5. New Player variablesPlayer.NicknamePlayer.HIDDeviceIDPlayer.RawHIDPlayer.SIDPlayer.AdapterPlayer.UserProfileKnown Issues#
- Force feedback does not work with Nucleus custom dlls.
- PreventWindowDeactivation will prevent mouse and keyboard input on instances using this hook (and may have other adverse effects).
- Status Window may cause Nucleus to crash every now and then.