AudienceEntry
AudienceEntry
is a crucial component in Typewriter that allows extension developers to display content to a group of players.
Typewriter manages the players in the audience, providing hooks that developers can use to show information or interact with the players in the audience.
The AudienceEntry
is the most used ManifestEntry
in Typewriter.
Key Concepts
-
Stateful Displays: In Typewriter, entries are not allowed to have any state.
AudienceEntry
bridges this gap by providing a way to have stateful displays. -
Audience Management: Typewriter takes care of managing the players in the audience. Developers can focus on defining how to display content and interact with the audience.
-
Display Lifecycle: The
AudienceDisplay
class manages the lifecycle of the audience, including initialization, player addition/removal, and disposal.
Implementation
Basic AudienceEntry
Here's an example of a basic AudienceEntry
implementation:
@Entry("example_audience", "An example audience entry.", Colors.GREEN, "material-symbols:chat-rounded")
class ExampleAudienceEntry(
override val id: String = "",
override val name: String = "",
) : AudienceEntry {
override fun display(): AudienceDisplay {
return ExampleAudienceDisplay()
}
}
In this example, we define an ExampleAudienceEntry
class that implements the AudienceEntry
interface.
The display()
function returns an AudienceDisplay
object, which defines how the content is presented to the audience.
AudienceDisplay
The AudienceDisplay
class is responsible for displaying content to the audience.
It has lifecycle hooks to accomplish this.
Here's an example implementation:
class ExampleAudienceDisplay : AudienceDisplay() {
override fun initialize() {
// This is called when the first player is added to the audience.
super.initialize()
// Do something when the audience is initialized
}
override fun onPlayerAdd(player: Player) {
// Do something when a player gets added to the audience
}
override fun onPlayerRemove(player: Player) {
// Do something when a player gets removed from the audience
}
override fun dispose() {
super.dispose()
// Do something when the audience is disposed
// It will always call onPlayerRemove for all players.
// So no player cleanup is needed here.
}
}
Key methods in AudienceDisplay
:
initialize()
: Called when the first player is added to the audience. Use this method for any setup tasks.onPlayerAdd(player: Player)
: Invoked when a player joins the audience. Use this to set up player-specific content or effects.onPlayerRemove(player: Player)
: Called when a player leaves the audience. Use this for cleanup of player-specific resources.dispose()
: Invoked when the audience is being disposed of. Use this for general cleanup tasks. Note thatonPlayerRemove
will be called for all players, so player-specific cleanup is not needed here.
Tickable Audience Display
For audiences that need to update regularly, you can implement the TickableDisplay
interface:
class TickableAudienceDisplay : AudienceDisplay(), TickableDisplay {
override fun onPlayerAdd(player: Player) {}
override fun onPlayerRemove(player: Player) {}
override fun tick() {
// Do something when the audience is ticked
players.forEach { player ->
// Do something with the player
}
// This is running asynchronously
// If you need to do something on the main thread
ThreadType.SYNC.launch {
// Though this will run a tick later, to sync with the bukkit scheduler.
}
}
}
The tick()
method will be called every Minecraft tick (20 times per second, or every 50ms) on an asynchronous thread.
This allows you to update the display or perform regular actions for all players in the audience.
Event Handling
Every AudienceDisplay
is also a Bukkit listener.
Event listeners will only be active when at least one player is in the audience.
Here's an example of how to handle events:
class AudienceDisplayWithEvents : AudienceDisplay() {
override fun onPlayerAdd(player: Player) {}
override fun onPlayerRemove(player: Player) {}
@EventHandler
fun onSomeEvent(event: SomeBukkitEvent) {
// Do something when the event is triggered
// This will trigger for all players, not just the ones in the audience.
// So we need to check if the player is in the audience.
if (event.player in this) {
// Do something with the player
}
}
}
Events will trigger for all players, not just those in the audience.
Always check if the player is in the audience before performing audience-specific actions.
Check if a player is in the audience of an entry
Suppose you have a Ref<out AudienceEntry>
.
To check if a player is in the audience of an entry,
you can use the player.inAudience
method that returns true
if the player is in the audience of the entry.
This is incredibly convenient, especially because it automatically handles all parents of the entry for you as well.
Here's an example:
if (player.inAudience(ref)) {
// Do something with the player
}
Best Practices
-
State Management: While state is allowed within the
AudienceDisplay
, ensure that all used state is contained within the display and does not leak outside. -
Resource Management: Ensure proper resource management to avoid memory leaks, especially when players leave the audience.
-
Event Handling: When handling Bukkit events, always check if the player is in the audience before performing audience-specific actions.
-
Asynchronous Operations: Be mindful that the
tick()
method runs on an asynchronous thread. Ensure thread safety when interacting with Bukkit API or shared resources.