use super::{
fleshlight_launch_helper::get_speed,
ButtplugDeviceResultFuture,
ButtplugProtocol,
ButtplugProtocolCommandHandler,
};
use crate::{
core::{
errors::ButtplugError,
messages::{
self,
ButtplugDeviceCommandMessageUnion,
FleshlightLaunchFW12Cmd,
MessageAttributesMap,
},
},
device::{
protocol::{generic_command_manager::GenericCommandManager, ButtplugProtocolProperties},
DeviceImpl,
DeviceWriteCmd,
Endpoint,
},
};
use async_mutex::Mutex;
use futures::future::BoxFuture;
use futures_timer::Delay;
use std::sync::{
atomic::{AtomicU8, Ordering::SeqCst},
Arc,
};
use std::time::Duration;
#[derive(ButtplugProtocolProperties)]
pub struct KiirooV21 {
name: String,
message_attributes: MessageAttributesMap,
manager: Arc<Mutex<GenericCommandManager>>,
stop_commands: Vec<ButtplugDeviceCommandMessageUnion>,
previous_position: Arc<AtomicU8>,
}
impl ButtplugProtocol for KiirooV21 {
fn new_protocol(
name: &str,
message_attributes: MessageAttributesMap,
) -> Box<dyn ButtplugProtocol> {
let manager = GenericCommandManager::new(&message_attributes);
Box::new(Self {
name: name.to_owned(),
message_attributes,
stop_commands: manager.get_stop_commands(),
manager: Arc::new(Mutex::new(manager)),
previous_position: Arc::new(AtomicU8::new(0)),
})
}
fn initialize(
device_impl: &dyn DeviceImpl,
) -> BoxFuture<'static, Result<Option<String>, ButtplugError>> {
debug!("calling Onyx+ init");
let init_fut1 = device_impl.write_value(DeviceWriteCmd::new(
Endpoint::Tx,
vec![0x03u8, 0x00u8, 0x64u8, 0x19u8],
true,
));
let init_fut2 = device_impl.write_value(DeviceWriteCmd::new(
Endpoint::Tx,
vec![0x03u8, 0x00u8, 0x64u8, 0x00u8],
true,
));
Box::pin(async move {
init_fut1.await?;
Delay::new(Duration::from_millis(100)).await;
init_fut2.await?;
Ok(None)
})
}
}
impl ButtplugProtocolCommandHandler for KiirooV21 {
fn handle_vibrate_cmd(
&self,
device: Arc<Box<dyn DeviceImpl>>,
message: messages::VibrateCmd,
) -> ButtplugDeviceResultFuture {
let manager = self.manager.clone();
Box::pin(async move {
let result = manager.lock().await.update_vibration(&message, false)?;
if let Some(cmds) = result {
device
.write_value(DeviceWriteCmd::new(
Endpoint::Tx,
vec![0x01, cmds.get(0).unwrap_or(&None).unwrap_or(0) as u8],
false,
))
.await?;
}
Ok(messages::Ok::default().into())
})
}
fn handle_linear_cmd(
&self,
device: Arc<Box<dyn DeviceImpl>>,
message: messages::LinearCmd,
) -> ButtplugDeviceResultFuture {
let v = message.vectors[0].clone();
let previous_position = self.previous_position.load(SeqCst);
let distance = (previous_position as f64 - (v.position * 99f64)).abs() / 99f64;
let fl_cmd = FleshlightLaunchFW12Cmd::new(
message.device_index,
(v.position * 99f64) as u8,
(get_speed(distance, v.duration) * 99f64) as u8,
);
self.handle_fleshlight_launch_fw12_cmd(device, fl_cmd)
}
fn handle_fleshlight_launch_fw12_cmd(
&self,
device: Arc<Box<dyn DeviceImpl>>,
message: messages::FleshlightLaunchFW12Cmd,
) -> ButtplugDeviceResultFuture {
let previous_position = self.previous_position.clone();
let position = message.position;
let msg = DeviceWriteCmd::new(
Endpoint::Tx,
[0x03, 0x00, message.speed, message.position].to_vec(),
false,
);
let fut = device.write_value(msg);
Box::pin(async move {
previous_position.store(position, SeqCst);
fut.await?;
Ok(messages::Ok::default().into())
})
}
}
#[cfg(test)]
mod test {
use crate::{
core::messages::{
FleshlightLaunchFW12Cmd,
LinearCmd,
StopDeviceCmd,
VectorSubcommand,
VibrateCmd,
VibrateSubcommand,
},
device::{DeviceImplCommand, DeviceWriteCmd, Endpoint},
test::{check_recv_value, new_bluetoothle_test_device},
util::async_manager,
};
#[test]
pub fn test_kiiroov21_fleshlight_fw12cmd() {
async_manager::block_on(async move {
let (device, test_device) = new_bluetoothle_test_device("Onyx2.1").await.unwrap();
let command_receiver = test_device
.get_endpoint_channel(&Endpoint::Tx)
.unwrap()
.receiver;
check_recv_value(
&command_receiver,
DeviceImplCommand::Write(DeviceWriteCmd::new(
Endpoint::Tx,
vec![0x03u8, 0x00u8, 0x64u8, 0x19u8],
true,
)),
)
.await;
check_recv_value(
&command_receiver,
DeviceImplCommand::Write(DeviceWriteCmd::new(
Endpoint::Tx,
vec![0x03u8, 0x00u8, 0x64u8, 0x00u8],
true,
)),
)
.await;
device
.parse_message(FleshlightLaunchFW12Cmd::new(0, 50, 50).into())
.await
.unwrap();
check_recv_value(
&command_receiver,
DeviceImplCommand::Write(DeviceWriteCmd::new(
Endpoint::Tx,
vec![0x03, 0x00, 50, 50],
false,
)),
)
.await;
});
}
#[test]
pub fn test_kiiroov21_linearcmd() {
async_manager::block_on(async move {
let (device, test_device) = new_bluetoothle_test_device("Onyx2.1").await.unwrap();
let command_receiver = test_device
.get_endpoint_channel(&Endpoint::Tx)
.unwrap()
.receiver;
check_recv_value(
&command_receiver,
DeviceImplCommand::Write(DeviceWriteCmd::new(
Endpoint::Tx,
vec![0x03u8, 0x00u8, 0x64u8, 0x19u8],
true,
)),
)
.await;
check_recv_value(
&command_receiver,
DeviceImplCommand::Write(DeviceWriteCmd::new(
Endpoint::Tx,
vec![0x03u8, 0x00u8, 0x64u8, 0x00u8],
true,
)),
)
.await;
device
.parse_message(LinearCmd::new(0, vec![VectorSubcommand::new(0, 500, 0.5)]).into())
.await
.unwrap();
check_recv_value(
&command_receiver,
DeviceImplCommand::Write(DeviceWriteCmd::new(
Endpoint::Tx,
vec![0x03, 0x00, 19, 49],
false,
)),
)
.await;
});
}
#[test]
pub fn test_kiiroov21_vibratecmd() {
async_manager::block_on(async move {
let (device, test_device) = new_bluetoothle_test_device("Cliona").await.unwrap();
let command_receiver = test_device
.get_endpoint_channel(&Endpoint::Tx)
.unwrap()
.receiver;
check_recv_value(
&command_receiver,
DeviceImplCommand::Write(DeviceWriteCmd::new(
Endpoint::Tx,
vec![0x03u8, 0x00u8, 0x64u8, 0x19u8],
true,
)),
)
.await;
check_recv_value(
&command_receiver,
DeviceImplCommand::Write(DeviceWriteCmd::new(
Endpoint::Tx,
vec![0x03u8, 0x00u8, 0x64u8, 0x00u8],
true,
)),
)
.await;
device
.parse_message(VibrateCmd::new(0, vec![VibrateSubcommand::new(0, 0.5)]).into())
.await
.unwrap();
check_recv_value(
&command_receiver,
DeviceImplCommand::Write(DeviceWriteCmd::new(Endpoint::Tx, vec![0x01, 50], false)),
)
.await;
device
.parse_message(VibrateCmd::new(0, vec![VibrateSubcommand::new(0, 0.5)]).into())
.await
.unwrap();
assert!(command_receiver.is_empty());
device
.parse_message(StopDeviceCmd::new(0).into())
.await
.unwrap();
check_recv_value(
&command_receiver,
DeviceImplCommand::Write(DeviceWriteCmd::new(Endpoint::Tx, vec![0x01, 0], false)),
)
.await;
});
}
}