Difference between revisions of "Modding:Migrate to SMAPI 3.0"

From Stardew Valley Wiki
Jump to navigation Jump to search
(→‎Major changes: + removed CreateTransitionalContentPack)
(→‎Event changes: + GameLoop events)
Line 107: Line 107:
 
| <tt>GameEvents.EighthUpdateTick</tt>
 
| <tt>GameEvents.EighthUpdateTick</tt>
 
| →
 
| →
| ''TODO''
+
| <tt>GameLoop.Updated</tt>
 
|  
 
|  
 +
* Check if <tt>e.IsMultipleOf(8)</tt>.
 
|-
 
|-
 
| <tt>GameEvents.FirstUpdateTick</tt>
 
| <tt>GameEvents.FirstUpdateTick</tt>
 
| →
 
| →
| ''TODO''
+
| <tt>GameLoop.Launched</tt>
|
+
| The new event is raised before the first update tick. To do something ''after'' the first update tick, use <tt>GameEvents.Updated</tt> with <tt>if (e.Ticks == 1)</tt>.
 
|-
 
|-
 
| <tt>GameEvents.FourthUpdateTick</tt>
 
| <tt>GameEvents.FourthUpdateTick</tt>
 
| →
 
| →
| ''TODO''
+
| <tt>GameLoop.Updated</tt>
 
|  
 
|  
 +
* Check if <tt>e.IsMultipleOf(4)</tt>.
 
|-
 
|-
 
| <tt>GameEvents.HalfSecondTick</tt>
 
| <tt>GameEvents.HalfSecondTick</tt>
 
| →
 
| →
| ''TODO''
+
| <tt>GameLoop.Updated</tt>
 
|  
 
|  
 +
* Check if <tt>e.IsMultipleOf(30)</tt>.
 +
 
|-
 
|-
 
| <tt>GameEvents.OneSecondTick</tt>
 
| <tt>GameEvents.OneSecondTick</tt>
 
| →
 
| →
| ''TODO''
+
| <tt>GameLoop.Updated</tt>
 
|  
 
|  
 +
* Check if <tt>e.IsOneSecond</tt>.
 +
 
|-
 
|-
 
| <tt>GameEvents.QuarterSecondTick</tt>
 
| <tt>GameEvents.QuarterSecondTick</tt>
 
| →
 
| →
| ''TODO''
+
| <tt>GameLoop.Updated</tt>
 
|  
 
|  
 +
* Check if <tt>e.IsMultipleOf(15)</tt>.
 +
 
|-
 
|-
 
| <tt>GameEvents.SecondUpdateTick</tt>
 
| <tt>GameEvents.SecondUpdateTick</tt>
 
| →
 
| →
| ''TODO''
+
| <tt>GameLoop.Updated</tt>
 
|  
 
|  
 +
* Check if <tt>e.IsMultipleOf(2)</tt>.
 
|-
 
|-
 
| <tt>GameEvents.UpdateTick</tt>
 
| <tt>GameEvents.UpdateTick</tt>
 
| →
 
| →
| ''TODO''
+
| <tt>GameLoop.Updated</tt>
|  
+
| Equivalent.
 
|-
 
|-
 
| <tt>GraphicsEvents.OnPostRenderEvent</tt>
 
| <tt>GraphicsEvents.OnPostRenderEvent</tt>

Revision as of 23:42, 8 July 2018

Index

This is an internal draft page. SMAPI 3.0 is not ready and modders don't need to do anything yet.

This page is for modders. Players: see Modding:SMAPI compatibility instead.

This page explains how to update your mod code for compatibility with SMAPI 3.0 (release TBD).

Overview

What's changing?

SMAPI compatibility over time. The SMAPI 2.0 release in October 2017 appears as a small bump. The Stardew Valley 1.3 release in May 2018 appears as a sudden cliff.

Events are arguably the most important and most visible part of SMAPI, but they've stayed essentially unchanged since SMAPI 0.40. New events have been added organically since then, but the event system has always been separate from the rest of SMAPI — they're inconsistent, have some weird behaviours (like MenuEvents.MenuChanged not being raised when a menu is closed), aren't available through the same helper as every other API, and there are some glaring omissions in the available events.

SMAPI 3.0 is the release that fixes all that. This release...

  • Completely rewrites the event engine to make events more efficient and enable events that weren't possible before.
  • Makes events much more consistent: they're now fully compliant with the .NET design guidelines, have predictable and descriptive names, have consistent event handler signatures, and have clear documentation.
  • All events are now accessible through helper.Events, so they're discoverable like any other SMAPI API.
  • Weird behaviours and overlapping events have been eliminated.
  • Many new events have been added.
  • Every event now has relevant event arguments.

Is this the modapocalypse?

Nope. Although this is a major change, significant effort will be undertaken to minimise the impact:

  • the old events will be supported for a long time with increasingly prominent warnings in the SMAPI console about their deprecation and removal;
  • pull requests will be submitted to update affected open-source mods;
  • unofficial updates will be created for mods which haven't updated officially by the time SMAPI 3.0 is released;
  • the changes will be actively communicated and documented to modders.

In addition, the current target is at least 95% compatibility for open-source mods before SMAPI 3.0 is released. All of this means that the 3.0 release should have minimal impact on mod compatibility, despite the scope of the changes.

How to update your mod

You don't need to comb through your code manually. SMAPI can tell you if you're using a deprecated interface:

  1. Use the latest SMAPI for developers download. This will show deprecation messages in the console:
    Modding - updating deprecated SMAPI code - deprecation warnings.png
  2. When you look at the code, you'll see a deprecation warning with a hint of how to fix it:
    Modding - updating deprecated SMAPI code - deprecation intellisense.png
  3. You can refer to the following sections on how to replace specific interfaces.

Major changes

Event changes

The former events will be removed in SMAPI 3.0. Here's how to convert them to the new events under this.Helper.Events. This list does not cover new events in 3.0.

old event new event conversion notes
ContentEvents.AfterLocaleChanged n/a Mods very rarely need to handle this, since SMAPI's translation and content APIs do. Since the locale can only change on the title screen, mods that used this can check once the save is loaded instead.
ControlEvents.ControllerButtonPressed Input.ButtonPressed Mostly equivalent.
  • Remove e.PlayerIndex (this was always PlayerIndex.One).
  • Change e.ButtonPressed to e.Button, which is now an SButton value.
ControlEvents.ControllerButtonReleased Input.ButtonReleased Mostly equivalent.
  • Remove e.PlayerIndex (this was always PlayerIndex.One).
  • Change e.ButtonPressed to e.Button, which is now an SButton value.
ControlEvents.ControllerTriggerPressed Input.ButtonPressed Mostly equivalent.
  • Remove e.PlayerIndex (this was always PlayerIndex.One).
  • Remove e.Value.
  • Change e.ButtonPressed to e.Button, which is now an SButton value.
ControlEvents.ControllerTriggerReleased Input.ButtonReleased Mostly equivalent.
  • Remove e.PlayerIndex (this was always PlayerIndex.One).
  • Remove e.Value.
  • Change e.ButtonPressed to e.Button, which is now an SButton value.
ControlEvents.KeyboardChanged Input.ButtonPressed
Input.ButtonReleased
Not directly equivalent; may need to rewrite affected code.
ControlEvents.KeyPressed Input.ButtonPressed Mostly equivalent.
  • Change e.KeyPressed to e.Button, which is now an SButton value.
ControlEvents.KeyReleased Input.ButtonReleased Mostly equivalent.
  • Change e.KeyPressed to e.Button, which is now an SButton value.
ControlEvents.MouseChanged Input.ButtonPressed
Input.ButtonReleased
Input.CursorMoved
Input.MouseWheelScrolled
Not directly equivalent; may need to rewrite affected code.
GameEvents.EighthUpdateTick GameLoop.Updated
  • Check if e.IsMultipleOf(8).
GameEvents.FirstUpdateTick GameLoop.Launched The new event is raised before the first update tick. To do something after the first update tick, use GameEvents.Updated with if (e.Ticks == 1).
GameEvents.FourthUpdateTick GameLoop.Updated
  • Check if e.IsMultipleOf(4).
GameEvents.HalfSecondTick GameLoop.Updated
  • Check if e.IsMultipleOf(30).
GameEvents.OneSecondTick GameLoop.Updated
  • Check if e.IsOneSecond.
GameEvents.QuarterSecondTick GameLoop.Updated
  • Check if e.IsMultipleOf(15).
GameEvents.SecondUpdateTick GameLoop.Updated
  • Check if e.IsMultipleOf(2).
GameEvents.UpdateTick GameLoop.Updated Equivalent.
GraphicsEvents.OnPostRenderEvent TODO
GraphicsEvents.OnPostRenderGuiEvent TODO
GraphicsEvents.OnPostRenderHudEvent TODO
GraphicsEvents.OnPreRenderEvent TODO
GraphicsEvents.OnPreRenderGuiEvent TODO
GraphicsEvents.OnPreRenderHudEvent TODO
GraphicsEvents.Resize TODO
InputEvents.ButtonPressed Input.ButtonPressed Mostly equivalent.
  • Change e.IsActionButton to e.Button.IsActionButton().
  • Change e.IsUseToolButton to e.Button.IsUseToolButton().
  • Change e.SuppressButton to this.Helper.Input.Suppress(button).
InputEvents.ButtonReleased Input.ButtonReleased Mostly equivalent.
  • Change e.IsActionButton to e.Button.IsActionButton().
  • Change e.IsUseToolButton to e.Button.IsUseToolButton().
  • Change e.SuppressButton to this.Helper.Input.Suppress(button).
LocationEvents.BuildingsChanged World.BuildingListChanged Equivalent.
LocationEvents.LocationsChanged World.LocationListChanged Equivalent.
LocationEvents.ObjectsChanged World.ObjectListChanged Equivalent.
MenuEvents.MenuChanged TODO
MenuEvents.MenuClosed TODO
MineEvents.MineLevelChanged TODO
MultiplayerEvents.AfterMainBroadcast TODO
MultiplayerEvents.AfterMainSync TODO
MultiplayerEvents.BeforeMainBroadcast TODO
MultiplayerEvents.BeforeMainSync TODO
PlayerEvents.InventoryChanged TODO
PlayerEvents.LeveledUp TODO
PlayerEvents.Warped TODO
SaveEvents.AfterCreate TODO
SaveEvents.AfterLoad TODO
SaveEvents.AfterReturnToTitle TODO
SaveEvents.AfterSave TODO
SaveEvents.BeforeCreate TODO
SaveEvents.BeforeSave TODO
SpecialisedEvents.UnvalidatedUpdateTick TODO
TimeEvents.AfterDayStarted TODO
TimeEvents.TimeOfDayChanged TODO

Removed CreateTransitionalContentPack

this.Helper.CreateTransitionalContentPack was a temporary method to support mods which already accepted custom content packs in their subfolder. All such content packs should be migrated to regular SMAPI content packs. This method will be removed in SMAPI 3.0.