2021-02-17 13:03:20 +00:00
|
|
|
pub extern crate common;
|
|
|
|
|
|
|
|
pub use common::comp::Health;
|
2020-12-13 17:40:15 +00:00
|
|
|
use serde::{de::DeserializeOwned, Deserialize, Serialize};
|
2020-12-11 23:37:22 +00:00
|
|
|
|
2021-02-16 23:11:05 +00:00
|
|
|
pub use common::{resources::GameMode, uid::Uid};
|
|
|
|
|
2021-02-17 13:03:20 +00:00
|
|
|
mod errors;
|
|
|
|
|
|
|
|
pub use errors::*;
|
2021-02-28 21:39:53 +00:00
|
|
|
pub use event::*;
|
|
|
|
|
|
|
|
/// The [`Action`] enum represents a push modification that will be made in the
|
|
|
|
/// ECS in the next tick Note that all actions when sent are async and will not
|
|
|
|
/// be executed in order like [`Retrieve`] that are sync. All actions sent will
|
|
|
|
/// be executed in the send order in the ticking before the rest of the logic
|
|
|
|
/// applies.
|
|
|
|
///
|
|
|
|
/// # Usage:
|
|
|
|
/// ```rust
|
|
|
|
/// # use veloren_plugin_api::*;
|
|
|
|
/// # pub fn emit_action(action: Action) { emit_actions(vec![action]) }
|
|
|
|
/// # pub fn emit_actions(_actions: Vec<Action>) {}
|
|
|
|
/// // Packing actions is better than sending multiple ones at the same time!
|
|
|
|
/// emit_actions(vec![
|
|
|
|
/// Action::KillEntity(Uid(1)),
|
|
|
|
/// Action::PlayerSendMessage(Uid(0), "This is a test message".to_owned()),
|
|
|
|
/// ]);
|
|
|
|
/// // You can also use this to only send one action
|
|
|
|
/// emit_action(Action::KillEntity(Uid(1)));
|
|
|
|
/// ```
|
2020-12-13 17:40:15 +00:00
|
|
|
#[derive(Deserialize, Serialize, Debug)]
|
2020-12-11 23:37:22 +00:00
|
|
|
pub enum Action {
|
|
|
|
ServerClose,
|
|
|
|
Print(String),
|
2020-12-13 23:08:15 +00:00
|
|
|
PlayerSendMessage(Uid, String),
|
|
|
|
KillEntity(Uid),
|
2020-12-11 23:37:22 +00:00
|
|
|
}
|
|
|
|
|
2021-02-28 21:39:53 +00:00
|
|
|
/// The [`Retrieve`] enum represents read of the ECS is sync and blocking.
|
|
|
|
/// This enum shouldn't be used by itself. You should always prefer `get`
|
|
|
|
/// methods on Plugin API Types For instance, prefer this method:
|
|
|
|
/// ```rust
|
|
|
|
/// # use veloren_plugin_api::*;
|
|
|
|
/// # let entityid = Player {id: Uid(0)};
|
|
|
|
/// # trait G { fn get_entity_health(&self) -> Option<i64>; }
|
|
|
|
/// # impl G for Player {fn get_entity_health(&self) -> Option<i64> {Some(1)}}
|
|
|
|
/// let life = entityid.get_entity_health().unwrap();
|
|
|
|
/// // Do something with life
|
|
|
|
/// ```
|
|
|
|
/// Over this one:
|
|
|
|
/// ```rust
|
|
|
|
/// # use veloren_plugin_api::*;
|
|
|
|
/// # let entityid = Uid(0);
|
|
|
|
/// # fn retrieve_action(r: &Retrieve) -> Result<RetrieveResult, RetrieveError> { Ok(RetrieveResult::GetEntityHealth(Health::empty()))}
|
|
|
|
/// let life = if let RetrieveResult::GetEntityHealth(e) =
|
|
|
|
/// retrieve_action(&Retrieve::GetEntityHealth(entityid)).unwrap()
|
|
|
|
/// {
|
|
|
|
/// e
|
|
|
|
/// } else {
|
|
|
|
/// unreachable!()
|
|
|
|
/// };
|
|
|
|
/// // Do something with life
|
|
|
|
/// ```
|
2021-02-15 16:38:55 +00:00
|
|
|
#[derive(Deserialize, Serialize, Debug)]
|
2021-02-17 13:03:20 +00:00
|
|
|
pub enum Retrieve {
|
|
|
|
GetPlayerName(Uid),
|
|
|
|
GetEntityHealth(Uid),
|
|
|
|
}
|
|
|
|
|
2021-02-28 21:39:53 +00:00
|
|
|
/// The [`RetrieveResult`] struct is generated while using the `retrieve_action`
|
|
|
|
/// function
|
|
|
|
///
|
|
|
|
/// You should always prefer using `get` methods available in Plugin API types.
|
|
|
|
///
|
|
|
|
/// Example:
|
|
|
|
/// ```rust
|
|
|
|
/// # use veloren_plugin_api::*;
|
|
|
|
/// # let entityid = Uid(0);
|
|
|
|
/// # fn retrieve_action(r: &Retrieve) -> Result<RetrieveResult, RetrieveError> { Ok(RetrieveResult::GetEntityHealth(Health::empty()))}
|
|
|
|
/// let life = if let RetrieveResult::GetEntityHealth(e) =
|
|
|
|
/// retrieve_action(&Retrieve::GetEntityHealth(entityid)).unwrap()
|
|
|
|
/// {
|
|
|
|
/// e
|
|
|
|
/// } else {
|
|
|
|
/// unreachable!()
|
|
|
|
/// };
|
|
|
|
/// // Do something with life
|
|
|
|
/// ```
|
2021-02-17 13:03:20 +00:00
|
|
|
#[derive(Serialize, Deserialize, Debug)]
|
|
|
|
pub enum RetrieveResult {
|
|
|
|
GetPlayerName(String),
|
|
|
|
GetEntityHealth(Health),
|
2021-02-15 16:38:55 +00:00
|
|
|
}
|
|
|
|
|
2021-02-28 21:39:53 +00:00
|
|
|
/// This trait is implement by all events and ensure type safety of FFI.
|
2020-12-13 17:40:15 +00:00
|
|
|
pub trait Event: Serialize + DeserializeOwned + Send + Sync {
|
2020-12-12 15:56:09 +00:00
|
|
|
type Response: Serialize + DeserializeOwned + Send + Sync;
|
2021-03-01 20:29:18 +00:00
|
|
|
|
|
|
|
fn get_event_name(&self) -> String;
|
2020-12-11 23:37:22 +00:00
|
|
|
}
|
|
|
|
|
2021-02-28 21:39:53 +00:00
|
|
|
/// This module contains all events from the api
|
2020-12-12 19:26:42 +00:00
|
|
|
pub mod event {
|
|
|
|
use super::*;
|
2020-12-13 17:40:15 +00:00
|
|
|
use serde::{Deserialize, Serialize};
|
2020-12-12 13:01:54 +00:00
|
|
|
|
2021-02-28 21:39:53 +00:00
|
|
|
/// This event is called when a chat command is run.
|
|
|
|
/// Your event should be named `on_command_<Your command>`
|
|
|
|
///
|
|
|
|
/// If you return an Error the displayed message will be the error message
|
|
|
|
/// in red You can return a Vec<String> that will be print to player
|
|
|
|
/// chat as info
|
|
|
|
///
|
|
|
|
/// # Example
|
|
|
|
/// ```ignore
|
|
|
|
/// #[event_handler]
|
|
|
|
/// pub fn on_command_testplugin(command: ChatCommandEvent) -> Result<Vec<String>, String> {
|
|
|
|
/// Ok(vec![format!(
|
|
|
|
/// "Player of id {:?} named {} with {:?} sended command with args {:?}",
|
|
|
|
/// command.player.id,
|
|
|
|
/// command
|
|
|
|
/// .player
|
|
|
|
/// .get_player_name()
|
|
|
|
/// .expect("Can't get player name"),
|
|
|
|
/// command
|
|
|
|
/// .player
|
|
|
|
/// .get_entity_health()
|
|
|
|
/// .expect("Can't get player health"),
|
|
|
|
/// command.command_args
|
|
|
|
/// )])
|
|
|
|
/// }
|
|
|
|
/// ```
|
2020-12-12 15:02:58 +00:00
|
|
|
#[derive(Serialize, Deserialize, Debug, Clone, Eq, PartialEq)]
|
|
|
|
pub struct ChatCommandEvent {
|
|
|
|
pub command: String,
|
|
|
|
pub command_args: Vec<String>,
|
|
|
|
pub player: Player,
|
|
|
|
}
|
|
|
|
|
|
|
|
impl Event for ChatCommandEvent {
|
|
|
|
type Response = Result<Vec<String>, String>;
|
2021-03-01 20:29:18 +00:00
|
|
|
|
|
|
|
fn get_event_name(&self) -> String { format!("on_command_{}", self.command) }
|
2020-12-12 15:02:58 +00:00
|
|
|
}
|
|
|
|
|
2021-02-28 21:39:53 +00:00
|
|
|
/// This struct represent a player
|
2020-12-12 15:02:58 +00:00
|
|
|
#[derive(Serialize, Deserialize, Debug, Clone, Eq, PartialEq)]
|
|
|
|
pub struct Player {
|
2020-12-13 17:11:55 +00:00
|
|
|
pub id: Uid,
|
2020-12-12 15:02:58 +00:00
|
|
|
}
|
|
|
|
|
2021-02-28 21:39:53 +00:00
|
|
|
/// This event is called when a player connects.
|
|
|
|
/// Your event should be named `on_join`
|
|
|
|
///
|
|
|
|
/// You can either return `CloseConnection` or `None`
|
|
|
|
/// If `CloseConnection` is returned the player will be kicked
|
|
|
|
///
|
|
|
|
/// # Example
|
|
|
|
/// ```ignore
|
|
|
|
/// #[event_handler]
|
|
|
|
/// pub fn on_join(command: PlayerJoinEvent) -> PlayerJoinResult {
|
|
|
|
/// PlayerJoinResult::CloseConnection
|
|
|
|
/// }
|
|
|
|
/// ```
|
2020-12-12 15:02:58 +00:00
|
|
|
#[derive(Serialize, Deserialize, Debug, Clone, Eq, PartialEq)]
|
2020-12-11 23:37:22 +00:00
|
|
|
pub struct PlayerJoinEvent {
|
|
|
|
pub player_name: String,
|
2021-03-01 18:00:44 +00:00
|
|
|
pub player_id: [u8; 16],
|
2020-12-11 23:37:22 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
impl Event for PlayerJoinEvent {
|
|
|
|
type Response = PlayerJoinResult;
|
2021-03-01 20:29:18 +00:00
|
|
|
|
|
|
|
fn get_event_name(&self) -> String { "on_join".to_owned() }
|
2020-12-11 23:37:22 +00:00
|
|
|
}
|
|
|
|
|
2021-02-28 21:39:53 +00:00
|
|
|
/// This is the return type of an `on_join` event. See [`PlayerJoinEvent`]
|
|
|
|
///
|
|
|
|
/// Variants:
|
|
|
|
/// - `CloseConnection` will kick the player.
|
|
|
|
/// - `None` will let the player join the server.
|
2020-12-12 15:02:58 +00:00
|
|
|
#[derive(Serialize, Deserialize, Debug, Clone, Eq, PartialEq)]
|
2021-03-01 18:00:44 +00:00
|
|
|
#[repr(u8)]
|
2020-12-11 23:37:22 +00:00
|
|
|
pub enum PlayerJoinResult {
|
2021-03-01 20:29:18 +00:00
|
|
|
Kick(String),
|
2020-12-13 17:40:15 +00:00
|
|
|
None,
|
2020-12-11 23:37:22 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
impl Default for PlayerJoinResult {
|
2020-12-13 17:40:15 +00:00
|
|
|
fn default() -> Self { Self::None }
|
2020-12-11 23:37:22 +00:00
|
|
|
}
|
|
|
|
|
2021-02-28 21:39:53 +00:00
|
|
|
/// This event is called when the plugin is loaded
|
|
|
|
/// Your event should be named `on_load`
|
|
|
|
///
|
|
|
|
/// # Example
|
|
|
|
/// ```ignore
|
|
|
|
/// #[event_handler]
|
|
|
|
/// pub fn on_load(load: PluginLoadEvent) {
|
|
|
|
/// match load.game_mode {
|
|
|
|
/// GameMode::Server => emit_action(Action::Print("Hello, server!".to_owned())),
|
|
|
|
/// GameMode::Client => emit_action(Action::Print("Hello, client!".to_owned())),
|
|
|
|
/// GameMode::Singleplayer => emit_action(Action::Print("Hello, singleplayer!".to_owned())),
|
|
|
|
/// }
|
|
|
|
/// }
|
|
|
|
/// ```
|
2020-12-12 15:02:58 +00:00
|
|
|
#[derive(Serialize, Deserialize, Debug, Clone, Eq, PartialEq)]
|
2020-12-12 19:26:42 +00:00
|
|
|
pub struct PluginLoadEvent {
|
2020-12-13 12:27:48 +00:00
|
|
|
pub game_mode: GameMode,
|
2020-12-12 19:26:42 +00:00
|
|
|
}
|
2020-12-11 23:37:22 +00:00
|
|
|
|
|
|
|
impl Event for PluginLoadEvent {
|
|
|
|
type Response = ();
|
2021-03-01 20:29:18 +00:00
|
|
|
|
|
|
|
fn get_event_name(&self) -> String { "on_load".to_owned() }
|
2020-12-11 23:37:22 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// impl Default for PlayerJoinResult {
|
|
|
|
// fn default() -> Self {
|
|
|
|
// Self::None
|
|
|
|
// }
|
|
|
|
// }
|
2020-12-12 13:01:54 +00:00
|
|
|
}
|