extern crate rosc;
use rosc::{encoder, OscBundle, OscTime};
use rosc::{OscMessage, OscPacket, OscType};
use std::net::{SocketAddrV4, UdpSocket};
use std::str::FromStr;
use std::time::{Duration, SystemTime};
use std::{env, f32, thread};
fn get_addr_from_arg(arg: &str) -> SocketAddrV4 {
SocketAddrV4::from_str(arg).unwrap()
}
fn main() {
let args: Vec<String> = env::args().collect();
let usage = format!(
"Usage: {} HOST_IP:HOST_PORT CLIENT_IP:CLIENT_PORT",
&args[0]
);
if args.len() < 3 {
panic!("{}", usage);
}
let host_addr = get_addr_from_arg(&args[1]);
let to_addr = get_addr_from_arg(&args[2]);
let sock = UdpSocket::bind(host_addr).unwrap();
let msg_buf = encoder::encode(&OscPacket::Message(OscMessage {
addr: "/3".to_string(),
args: vec![],
}))
.unwrap();
sock.send_to(&msg_buf, to_addr).unwrap();
let steps = 128;
let step_size: f32 = 2.0 * f32::consts::PI / steps as f32;
for i in 0.. {
let x = 0.5 + (step_size * (i % steps) as f32).sin() / 2.0;
let y = 0.5 + (step_size * (i % steps) as f32).cos() / 2.0;
let bundle = make_bundle(vec![("x".to_owned(), x), ("y".to_owned(), y)]).unwrap();
let packet = OscPacket::Bundle(bundle);
let msg_buf = encoder::encode(&packet).unwrap();
sock.send_to(&msg_buf, to_addr).unwrap();
thread::sleep(Duration::from_millis(20));
}
}
fn make_bundle(vals: Vec<(String, f32)>) -> Result<OscBundle, String> {
let timetag = OscTime::try_from(SystemTime::now()).map_err(|op| op.to_string())?;
let content = vals
.iter()
.map(|(name, val)| {
let addr = format!("/livecode/{}", name);
let args = vec![OscType::Float(*val)];
OscPacket::Message(OscMessage { addr, args })
})
.collect::<Vec<_>>();
Ok(OscBundle { timetag, content })
}