}
}
}
+
+pub struct Post {
+ state: Option<Box<dyn State>>,
+ content: String,
+}
+
+impl Post {
+ pub fn new() -> Post {
+ Post {
+ state: Some(Box::new(Draft {})),
+ content: String::new(),
+ }
+ }
+
+ pub fn add_text(&mut self, text: &str) {
+ self.content.push_str(text);
+ }
+
+ pub fn content(&self) -> &str {
+ // as_ref() converts Option<Box<State>> to Option<&Box<State>>
+ // state can never be None, all state transitions return a new one
+ self.state.as_ref().unwrap().content(self)
+ }
+
+ pub fn request_review(&mut self) {
+ if let Some(s) = self.state.take() {
+ self.state = Some(s.request_review());
+ }
+ }
+
+ pub fn approve(&mut self) {
+ if let Some(s) = self.state.take() {
+ self.state = Some(s.approve());
+ }
+ }
+}
+
+trait State {
+ fn request_review(self: Box::<Self>) -> Box<dyn State>;
+ fn approve(self: Box::<Self>) -> Box<dyn State>;
+
+ #[allow(unused_variables)]
+ fn content<'a>(&self, post: &'a Post) -> &'a str {
+ ""
+ }
+}
+
+struct Draft {}
+impl State for Draft {
+ fn request_review(self: Box::<Self>) -> Box<dyn State> {
+ Box::new(PendingReview {})
+ }
+
+ fn approve(self: Box::<Self>) -> Box<dyn State> {
+ self
+ }
+}
+
+struct PendingReview {}
+impl State for PendingReview {
+ fn request_review(self: Box::<Self>) -> Box<dyn State> {
+ self
+ }
+
+ fn approve(self: Box::<Self>) -> Box<dyn State> {
+ Box::new(Published {})
+ }
+}
+
+struct Published {}
+impl State for Published {
+ fn request_review(self: Box::<Self>) -> Box<dyn State> {
+ self
+ }
+
+ fn approve(self: Box::<Self>) -> Box<dyn State> {
+ Box::new(Published {})
+ }
+
+ fn content<'a>(&self, post: &'a Post) -> &'a str {
+ &post.content
+ }
+}
println!("counter: {}", *counter.lock().unwrap());
}
+fn test_dyn_traits() {
+ let text = "I ate a salad for lunch today";
+ let mut post = Post::new();
+ post.add_text(text);
+ assert_eq!("", post.content());
+
+ post.request_review();
+ assert_eq!("", post.content());
+
+ post.approve();
+ assert_eq!(text, post.content());
+}
+
fn main() {
test_strings();
test_vectors();
test_closures();
test_iterators();
test_threads();
+ test_dyn_traits();
}