Learning Minecraft modding with Pahimar - http://www.pahimar.com/tutorials/lets-mod/
I'll be keeping notes for reference as I work through the series.
I've opted to use IntelliJ IDEA for this project.
Initial Setup (Episode 1)
-
Create folder for project (duh)
-
Download Forge for desired version
-
Exract the following files/folders into project directory:
- eclipse/
- gradle/
- build.gradle
- gradlew
- gradlew.bat
-
Setup workspace with decompiled Minecraft
$ gradlew setupDecompWorkspace
-
Setup IDE workspace
$ # For Eclipse $ gradlew eclipse $ # For IntelliJ IDEA $ gradlew idea
-
Open project in IDE
For Eclipse, select the
eclipse/
folder as the workspace.For IntelliJ IDEA, Open Project and select the project root directory.
Upon opening the project in IDEA, I got a warning suggesting I import the Gradle project. I chose to do it the first time, and it starting configuring something for Maven which took forever. After starting over, I chose to ignore the message and opt to not have it display again, which works just fine.
Apparently IntelliJ IDEA doesn't automatically detect JDK installs, so you have to set it up yourself. This is done in the Project Structure window, accessible from either File -> Project Structure, or pressing CTRL+SHIFT+ALT+S
. Select the Project tab on the left, then use New...
under the Project SDK section to select the desired JDK installation.
I'm still not 100% sure why these assets don't always get downloaded in the first place, but running gradlew setupDecompWorkspace
again fixed it for me. It'll skip everything that was done correctly the first time, so it doesn't take very long. Of the two environments on which I setup a workspace, only one of them ran into this issue.
Basic Gradle Configuration (Episode 2)
The following fields should be modified per project:
version
: Mod version. Convention is to prefix it with the intended Minecraft version, i.e.1.7.2-1.0
.group
: Java package name. Naming convention is the same as the Java package naming convention, i.e.com.github.skirkpatrick.letsmod
.archivesBaseName
: Mod identifier. Can contain spaces, but should it?
The following fields may be modified depending on your needs:
minecraft.version
: Minecraft Forge version to use.minecraft.assetDir
: Directory in which to store base game and forge assets.
The Java convention should be used for the project source structure. Optional folders are marked with an asterisk (*
). All other folders are required to be the way Minecraft and Forge expect.
<Project Root>
`---src
`---main
+---java
| `---<Package)
`---resources
`---assets
`---(modid)
+---lang
+---models*
+---sounds
`---textures
+---blocks
+---gui*
`---items
The src/main/java
and src/main/resources
folders should be marked as source folders.
The (Package)
folders should be a directory hierarchy based on your package name. For example, src/main/java/com/github/skirkpatrick/letsmod/
.
The (modid)
folder should be named the same as the Mod Id/archivesBaseName, but all lowercase, i.e. src/main/resources/assets/letsmod/
.
The optional (*
) folders are the convention used by Pahimar. These will not be automatically loaded by Forge, and must be configured to do so.
@Mod Class (Episode 3)
The @Mod
annotation marks the main class to be loaded by the Forge Mod Loader (FML).
Important @Mod
attributes to set:
modid
: Unique identifier for mod. Should be the same as thearchivesBaseName
in build.gradle.name
: Name of the mod. Probably the same as the ModId but with spaces.version
: Mod version. Should be the same as theversion
in build.gradle.
All FML event classes can be found in the external library package cpw.mods.fml.common.event
.
FML performs each phase across all mods at a time, so all mods will run the pre-initialization phase before any mod continues to the initialization phase.
- Load configurations
- Initialize network handling
- Setup key bindings
- Initialize items
- Initialize blocks
- Setup GUI handler
- Setup tile entities
- Setup rendering
- Register crafting recipes
- Setup other general event handlers
- Start registering recipes
Phase to wrap things up after other mods have loaded.
The @Mod.EventHandler
subscribes a method to the event handler.
Method argument determines the event type (phase). Argument should be an FMLEvent
type from the cpw.mods.fml.common.event
package.
@Mod.EventHandler
public void preInit(FMLPreInitializationEvent event) {
...
}
The @Mod.Instance
annotation marks a static data member of your mod type as a safe, reliable instance that can be used. ModId should be given as the default argument.
@Mod.Instance("LetsMod")
public static LetsMod instance;
mcmod.info (Episode 4)
The mcmod.info
is a JSON formatted file that contains information about the mod to be displayed in the Mods menu accessible from the game's main menu. It should be placed in the src/main/resources
directory.
Pahimar's suggestion: Just copy the mdmod.info from another mod and modify it as necessary to ensure correct formatting.
The fields are as follows:
modid
: The Mod Idname
: Name of the mod. Same as specified in the@Mod
attributes.description
: A short description of the mod.version
: Mod version. Should be the same as specified inbuild.gradle
, but without the Minecraft version prepended, i.e.0.1
.mcversion
: Minecraft version the mod is designed for. This would be the prepended bit in thebuild.gradle
version.url
: URL to get more information about the mod, likely the mod homepage, if one exists.updateUrl
: This field does nothing!authorList
: List of mod authors.credits
: Credits due, if any.logoFile
: Path (?) to a PNG file to be used as the logo on the mod info screen. Maximum dimensions are 256x256.screenshots
: This field does nothing!parent
: Declares the mod as a child of another mod.dependencies
: List of mods on which the mod is dependent.
Sided Proxy (Episode 5)
Proxy classes simplify writing mods to work on both the Minecraft client and the dedicated server client. A base proxy class can be used to define code common to both clients, while client- and server-specific subclasses contain code for unique operations. Forge refers to this concept as a 'sided proxy'.
The SidedProxy annotation is used to inject a static instance of the appropriate Proxy implementation at runtime (such as a member in your base mod class). The clientSide attribute specifies the client proxy and the serverSide attribute specifies the server proxy.
This post gives a better explanation of proxies than Pahimar does: http://greyminecraftcoder.blogspot.com/2013/11/how-forge-starts-up-your-code.html.
Reference Classes (Episode 5)
Reference classes provide a centralized place to store constants that may be referenced in multiple places across the code base. This provides a couple of advantages:
- If a value needs to be modified or updated, it only has to be done in one place
- Errors are less likely to be introduced from typos
- Keeps code DRY
- Constants are all kept in a predictable location
These classes are written the same as for any Java project: a class with public static final
data members.
Utility Classes (Episode 5)
As with any Java project (or any coding project, for that matter), utility classes can come in handy for defining static
methods for perfoming common operations. This can including things such as performing bulk operations (such as on List
s) or performing tasks that don't fit as instance methods elsewhere.
Writing a ConfigurationHandler
can be useful in loading and editing mod configurations. Forge provides a Configuration
class that simplifies interacting with the configuration file itself and allows value defaulting.
The configuration should be loaded in the pre-initialization phase, and the FMLPreInitializationEvent#getSuggestedConfigurationFile()
method provides a handy default for getting a handle to the config file itself, which can then be passed to the Configuration
constructor. The default name for the config file is <modId>.cfg
.
The category
parameter that's accepted by the various Configuration.get
methods refers to the grouping in the config file under which the value is expected to be stored. Configuration.CATEGORY_GENERAL
is the general category.
To allow users to modify the configuration from within the Minecraft GUI, an event handling instance method should be added to the ConfigurationHandler
. This method should take a ConfigChangedEvent.OnConfigChangedEvent
and should simply reload the configuration if the modID
attribute matches ours. The class should also be subscribed to the FML event bus, and the steps below should be taken.
To make an in-game GUI configuration menu, we must define the main GuiScreen
that will load when the mod's Config
button is first press. This class should extend GuiConfig
and should define a single-argument constructor accepting a GuiScreen
. This constructor only needs to call its parent's constructor, through which a few properties about the configuration menu can be defined, such as whether or not any options require a world or game restart.
FML uses a GuiFactory
class that implements IModGuiFactory
to get information about the configuration menu. Of the four methods declared in the IModGuiFactory
, only mainConfigGuiClass
is currently used. This method should return the class
of the implemented GuiConfig
.
This final step is to define the guiFactory
of the @Mod
annotation to be the full package+name of the GuiFactory
class. (Reference constant, hint hint)
This post from Minalien contains good information and examples for the configu GUIs: http://minalien.com/minecraft-forge-feature-spotlight-config-guis/
Subscribing to Events and the Event Bus (Episode 7)
The @SubscribeEvent
annotation is used to subscribe a method to an event. The method parameter determines what event the method handles and should be subscribed to.
All clases that contain event listeners need to be registered on the event bus, through which all events get processed. There is one for Forge and one for FML. Look at the package name of the event listener parameter to find which event bus to register that class on.
At this point, we've only subscribed the ConfigurationHandler
to the event bus, and we're doing it right after the ConfigurationHandler
instance has been initialized. It's probably a good idea to register classes to the event bus right after they've been initialized.
FMLCommonHandler.instance().bus().register(new ConfigurationHandler());
Updating Minecraft and Forge (Episode 7)
Updating Minecraft and Forge simply requires updating the version
and minecraft.version
properties in build.gradle, the VERSION
constant in Reference.java
(if not being set automatically), and the mcversion
field in mcmod.info. Then just clean the dependencies, and run the same gradle tasks as when setting up the workspace:
$ # Clean dependencies
$ gradlew clean
$ # Redownload dependencies
$ gradlew setupDecompWorkspace
$ # Regenerate IDE workspace files
$ gradlew idea # For IDEA
$ # gradlew eclipse # For Eclipse
Just as when setting up the workspace initially, gradlew setupDecompWorkspace
may need to be run several times if getting exceptions when launching Minecraft.
Logging (Episode 7)
FML uses log4j for logging, which provides useful information such as timestamps and thread info in log entries, as well as different levels of logging. The static FMLLog.log
method is used for creating a log entry, and it takes the mod name (Reference.MOD_NAME
), the log level (from the log4j Level
class), and the message to be logged.
As with most projects, a LogHelper
utility class can be helpful for defining wrapper methods for each log level (info, warn, fatal, etc), so all you need to pass is the log message/Object
to be logged.
Basic Items (Episode 8)
The Item
class in Minecraft defines properties of items, but does not represent actual instances of items, aka the item in the player's inventory. The ItemStack
class represents item instances and has not been covered at this point.
At the most basic level, defining an item just requires defining a subclass of Item
, and registering an instance of that class in the item registry during the pre-initialization phase.
Defining a base item class for your mod can be helpful in abstracting out boiler plate such as registring textures or getting the item name. Additionally, definining an initialization class for registering all items in one place can be useful.
By default, an item is named from its 'unlocalized name', which is really just the internal name of the item, i.e. the id. This can be used to construct a key for a localized resource file that maps it to a label.
The localized resource files go in the resources/assets/<modid>/lang
folder with a name such as en_US.lang
. This file is simply a key/value mapping for localized strings.
Item label keys are of the form item.<modId>:<itemName>.name
.
Normally the full unlocalized name is of the format item.<itemName>
, but overriding Item.getUnlocalizedName()
to return this in the format item.<modId>:<itemName>
will, according to Pahimar, save some trouble later by simplifying things.
Minecraft item textures are 16x16 PNG files, so the first step is to create the texture and place it in the resources/assets/<modid>/textures/items
folder with the same name as the item.
The next step is to register the icon for the item, which, for basic textures, simply requires overriding the registerIcons(IIconRegister)
method and calling IIconRegister.registerIcon(String)
, passing in the unwrapped unlocalized item name (that is, the name without the item.
prefix), and assigning the return value to the itemIcon
instance member. This method should also be annotated as client side only.
More advanced item textures have not been covered yet.
@SideOnly (Episode 8)
The @SideOnly
annotation is used to mark a method as only being applicable to one side: either the server (Side.SERVER
), or the client (Side.CLIENT
).