Bevy Talks

This Bevy plugin provides a way to create dialogues and conversations in your game, via Talks components. A Talk is a directed graph where each node is an action that an actor can perform, such as saying a line, joining/leaving the scene, or even a choice the player can make.

The most common action is text being displayed on the screen, and a simple Talk is just a sequence of texts forming a conversation between actors.

You can have multiple entities each with their own Talk. Or you can make a VN-like game with one single Talk in the game.

The heart of the Talk is a directed graph where each node is an TalkNode struct:

rust struct TalkNode { /// Talk, Join, Leave, Choice kind: TalkNodeKind, /// Text to display on the screen. text: String, /// The actors involved in the action. actors: Vec<Actor>, /// The choices available for the player choices: Vec<Choice>, } The Actor struct is a simple struct that contains the name of the actor and the asset to display on the screen.

rust struct Actor { /// The name of the character that the actor plays. name: String, /// An optional asset for the actor. asset: Option<Handle<Image>>, }

The Choice struct is a simple struct that contains the text of the choice and the index of the node to jump to.

rust struct Choice { /// The text of the choice. pub text: String, /// The ID of the next action to jump to if the choice is selected. pub next: NodeIndex, }

Build Talks from talk.ron files

The plugin can parse ron files to create RawTalk assets, which can then be used to build a Talk component. The files must have the extension: talk.ron.

Here's an example:

rust,ignore ( actors: [ ( id: "bob", name: "Bob" ), ( id: "alice", name: "Alice" ) ], script: [ ( id: 1, action: Talk, text: Some("Bob and Alice enter the room.") ), ( id: 2, action: Join, actors: [ "bob", "alice" ] ), ( id: 3, actors: ["bob"], text: Some("Hello, Alice!") ), // with missing action field, it defaults to Talk ( id: 4, choices: Some([ ( text: "Alice says hello back.", next: 5 ), ( text: "Alice ignores Bob.", next: 6 ), ]) ), ( id: 5, text: Some("Bob smiles.") ), // with missing actors field, it defaults to an empty vector ( id: 6, text: Some("Bob starts crying.") ), ( id: 7, text: Some("The end.") ) ] )

The plugin adds an AssetLoader for these ron files, so it's as easy as:

rust let handle: Handle<RawTalk> = asset_server.load("simple.talk.ron");

Then you can use Talk::build function with the RawTalk asset. You can retrieve the RawTalk from the assets collection raws: Res<Assets<RawTalk>>.

rust let raw_sp = raws.get(&simple_sp_asset.handle).unwrap(); Talk::build(&raw_sp)

Usage

The plugin provides a TalkerBundle to give an entity the required components to handle its own dialogues. rust struct TalkerBundle { talk: Talk, /// The dialogue line component for a Talk. talk_text: CurrentText, /// The actor component that represents a character in a Talk. current_actors: CurrentActors, /// The Talk Node Kind component that represents the kind of action in a Talk. kind: CurrentNodeKind, /// The component that represents the current choices in a Talk. current_choices: CurrentChoices, }

With these components you can query the current text/actor/choices for the current action in a talk. Together with the Change Detection System, you can react to changes in the Talk component to update your UI.

rust fn print_text(talks: Query<(Ref<CurrentText> &CurrentNodeKind)>) { for (text, kind) in talks.iter() { if kind == TalkNodeKind::Talk && text.is_changed() { println!("{}", text.text()); } } }

To interact with Talks you can send 3 different events. One to initialize the Talk (it populates the components with the first node), and two to advance the Talk to the next node or to jump to a specific node):

rust struct InitTalkRequest(pub Entity);

To move forward to the next action:

rust NextActionRequest(pub Entity);

To jump to a specific action (used with choices):

rust JumpToActionRequest(pub Entity, pub NodeIndex);

You pass the entity with the Talk component for the first 2 events. The third required the entity and the index that identifies the node to jump to.

Check out the example in the examples folder to see how to use the plugin.

Roadmap

Bevy Version Support

Compatibility of bevy_talks versions: | bevy_talks | bevy | | :-- | :-- | | main | 0.11 | | 0.3.0 | 0.11 | | 0.2.0 | 0.11 | | 0.1.1 | 0.11 |

License

Dual-licensed under either of

at your option.

Contribution

Unless you explicitly state otherwise, any contribution intentionally submitted for inclusion in the work by you, as defined in the Apache-2.0 license, shall be dual licensed as above, without any additional terms or conditions.