6 个版本 (3 个稳定版)
1.21.0 | 2024年3月7日 |
---|---|
1.0.1 | 2023年7月3日 |
1.0.0 | 2023年4月9日 |
0.2.0 | 2023年2月12日 |
0.1.0 | 2022年10月17日 |
#513 在 编码
每月290 次下载
21KB
218 代码行
yew-websocket
用爱编写的 Rust yew websocket 服务 :)
支持 yew 版本 0.20.0
此包基于原本是核心库一部分的原始 yew websocket 服务。 https://github.com/yewstack/yew/blob/0.18.0/packages/yew/src/services/websocket.rs
由于某些原因,核心团队决定废弃它。
我尝试使用建议的库(wasm-sockets 或 gloo-net),但它们并没有与 yew 正确集成。
示例
use anyhow::Error;
use serde_derive::{Deserialize, Serialize};
use yew_websocket::macros::Json;
use yew::{html, Component, Context, Html};
use yew_websocket::websocket::{WebSocketService, WebSocketStatus, WebSocketTask};
type AsBinary = bool;
pub enum Format {
Json,
Toml,
}
pub enum WsAction {
Connect,
SendData(AsBinary),
Disconnect,
Lost,
}
pub enum Msg {
WsAction(WsAction),
WsReady(Result<WsResponse, Error>),
}
impl From<WsAction> for Msg {
fn from(action: WsAction) -> Self {
Msg::WsAction(action)
}
}
/// This type is used as a request which sent to websocket connection.
#[derive(Serialize, Debug)]
struct WsRequest {
value: u32,
}
/// This type is an expected response from a websocket connection.
#[derive(Deserialize, Debug)]
pub struct WsResponse {
value: u32,
}
pub struct Model {
pub fetching: bool,
pub data: Option<u32>,
pub ws: Option<WebSocketTask>,
}
impl Model {
fn view_data(&self) -> Html {
if let Some(value) = self.data {
html! {
<p>{ value }</p>
}
} else {
html! {
<p>{ "Data hasn't fetched yet." }</p>
}
}
}
}
impl Component for Model {
type Message = Msg;
type Properties = ();
fn create(ctx: &Context<Self>) -> Self {
Self {
fetching: false,
data: None,
ws: None,
}
}
fn update(&mut self, ctx: &Context<Self>, msg: Self::Message) -> bool {
match msg {
Msg::WsAction(action) => match action {
WsAction::Connect => {
let callback = ctx.link().callback(|Json(data)| Msg::WsReady(data));
let notification = ctx.link().batch_callback(|status| match status {
WebSocketStatus::Opened => None,
WebSocketStatus::Closed | WebSocketStatus::Error => {
Some(WsAction::Lost.into())
}
});
let task = WebSocketService::connect(
"wss://echo.websocket.events/",
callback,
notification,
)
.unwrap();
self.ws = Some(task);
true
}
WsAction::SendData(binary) => {
let request = WsRequest { value: 321 };
if binary {
self.ws.as_mut().unwrap().send_binary(Json(&request));
} else {
self.ws.as_mut().unwrap().send(Json(&request));
}
false
}
WsAction::Disconnect => {
self.ws.take();
true
}
WsAction::Lost => {
self.ws = None;
true
}
},
Msg::WsReady(response) => {
self.data = response.map(|data| data.value).ok();
true
}
}
}
fn view(&self, ctx: &Context<Self>) -> Html {
html! {
<div>
<nav class="menu">
{ self.view_data() }
<button disabled={self.ws.is_some()}
onclick={ctx.link().callback(|_| WsAction::Connect)}>
{ "Connect To WebSocket" }
</button>
<button disabled={self.ws.is_none()}
onclick={ctx.link().callback(|_| WsAction::SendData(false))}>
{ "Send To WebSocket" }
</button>
<button disabled={self.ws.is_none()}
onclick={ctx.link().callback(|_| WsAction::SendData(true))}>
{ "Send To WebSocket [binary]" }
</button>
<button disabled={self.ws.is_none()}
onclick={ctx.link().callback(|_| WsAction::Disconnect)}>
{ "Close WebSocket connection" }
</button>
</nav>
</div>
}
}
}
fn main() {
yew::Renderer::<Model>::new().render();
}
依赖项
~11–16MB
~274K SLoC