use qubit_function::{
ArcTransformer,
BoxTransformer,
RcTransformer,
Transformer,
};
use std::collections::HashMap;
use std::thread;
fn main() {
println!("=== Transformer Demo - Type Transformation (consumes T) ===\n");
println!("--- BoxTransformer ---");
let double = BoxTransformer::new(|x: i32| x * 2);
println!("double.apply(21) = {}", double.apply(21));
println!("double.apply(42) = {}", double.apply(42));
let identity = BoxTransformer::<i32, i32>::identity();
println!("identity.apply(42) = {}", identity.apply(42));
let constant = BoxTransformer::constant("hello");
println!("constant.apply(123) = {}", constant.apply(123));
println!();
println!("--- ArcTransformer ---");
let arc_double = ArcTransformer::new(|x: i32| x * 2);
let arc_cloned = arc_double.clone();
println!("arc_double.apply(21) = {}", arc_double.apply(21));
println!("arc_cloned.apply(42) = {}", arc_cloned.apply(42));
let for_thread = arc_double.clone();
let handle = thread::spawn(move || for_thread.apply(100));
println!(
"In main thread: arc_double.apply(50) = {}",
arc_double.apply(50)
);
println!("In child thread: result = {}", handle.join().unwrap());
println!();
println!("--- RcTransformer ---");
let rc_double = RcTransformer::new(|x: i32| x * 2);
let rc_cloned = rc_double.clone();
println!("rc_double.apply(21) = {}", rc_double.apply(21));
println!("rc_cloned.apply(42) = {}", rc_cloned.apply(42));
println!();
println!("=== Practical Examples ===\n");
println!("--- String Transformation ---");
let to_upper = BoxTransformer::new(|s: String| s.to_uppercase());
println!(
"to_upper.apply('hello') = {}",
to_upper.apply("hello".to_string())
);
println!(
"to_upper.apply('world') = {}",
to_upper.apply("world".to_string())
);
println!();
println!("--- Type Conversion Pipeline ---");
let parse_int = BoxTransformer::new(|s: String| s.parse::<i32>().unwrap_or(0));
let double_int = BoxTransformer::new(|x: i32| x * 2);
let to_string = BoxTransformer::new(|x: i32| x.to_string());
let pipeline = parse_int.and_then(double_int).and_then(to_string);
println!(
"pipeline.apply('21') = {}",
pipeline.apply("21".to_string())
);
println!();
println!("--- Shared Transformation Logic ---");
let square = ArcTransformer::new(|x: i32| x * x);
let transformer1 = square.clone();
let transformer2 = square.clone();
println!("transformer1.apply(5) = {}", transformer1.apply(5));
println!("transformer2.apply(7) = {}", transformer2.apply(7));
println!("square.apply(3) = {}", square.apply(3));
println!();
println!("--- Transformer Registry ---");
let mut transformers: HashMap<String, RcTransformer<i32, String>> = HashMap::new();
transformers.insert(
"double".to_string(),
RcTransformer::new(|x: i32| format!("Doubled: {}", x * 2)),
);
transformers.insert(
"square".to_string(),
RcTransformer::new(|x: i32| format!("Squared: {}", x * x)),
);
if let Some(transformer) = transformers.get("double") {
println!("Transformer 'double': {}", transformer.apply(7));
}
if let Some(transformer) = transformers.get("square") {
println!("Transformer 'square': {}", transformer.apply(7));
}
println!();
println!("=== Trait Usage ===\n");
fn apply_transformer<F: Transformer<i32, String>>(f: &F, x: i32) -> String {
f.apply(x)
}
let to_string = BoxTransformer::new(|x: i32| format!("Value: {}", x));
println!("Via trait: {}", apply_transformer(&to_string, 42));
println!("\n=== Demo Complete ===");
}