X-Git-Url: https://piware.de/gitweb/?p=learn-rust.git;a=blobdiff_plain;f=concepts%2Fsrc%2Flib.rs;h=b0839ad391739cb186adcf2bb81c97484c5a3dfe;hp=ca0624142dea4d751e4506184bf84508de7e0696;hb=1962a5412d16f34c738e9f81bb5271da043a7505;hpb=c182c6166ae3971320daaeadba3ea718124303e7;ds=sidebyside diff --git a/concepts/src/lib.rs b/concepts/src/lib.rs index ca06241..b0839ad 100644 --- a/concepts/src/lib.rs +++ b/concepts/src/lib.rs @@ -110,14 +110,14 @@ impl Iterator for Counter5 { } pub struct Post { - state: Option>, + state: Box, content: String, } impl Post { pub fn new() -> Post { Post { - state: Some(Box::new(Draft {})), + state: Box::new(Draft {}), content: String::new(), } } @@ -127,27 +127,26 @@ impl Post { } pub fn content(&self) -> &str { - // as_ref() converts Option> to Option<&Box> - // state can never be None, all state transitions return a new one - self.state.as_ref().unwrap().content(self) + self.state.content(self) } pub fn request_review(&mut self) { - if let Some(s) = self.state.take() { - self.state = Some(s.request_review()); - } + self.state = self.state.request_review(); } pub fn approve(&mut self) { - if let Some(s) = self.state.take() { - self.state = Some(s.approve()); - } + self.state = self.state.approve(); + } + + pub fn reject(&mut self) { + self.state = self.state.reject(); } } trait State { - fn request_review(self: Box::) -> Box; - fn approve(self: Box::) -> Box; + fn request_review(&self) -> Box; + fn approve(&mut self) -> Box; + fn reject(&self) -> Box; #[allow(unused_variables)] fn content<'a>(&self, post: &'a Post) -> &'a str { @@ -157,36 +156,56 @@ trait State { struct Draft {} impl State for Draft { - fn request_review(self: Box::) -> Box { - Box::new(PendingReview {}) + fn request_review(&self) -> Box { + Box::new(PendingReview {acks: 0}) } - fn approve(self: Box::) -> Box { - self + fn approve(&mut self) -> Box { + // don't change state + Box::new(Self {}) } + + fn reject(&self) -> Box { + Box::new(Self {}) + } +} + +struct PendingReview { + acks: u32, } -struct PendingReview {} impl State for PendingReview { - fn request_review(self: Box::) -> Box { - self + fn request_review(&self) -> Box { + Box::new(Self {acks: self.acks}) } - fn approve(self: Box::) -> Box { - Box::new(Published {}) + fn approve(&mut self) -> Box { + if self.acks >= 1 { + Box::new(Published {}) + } else { + Box::new(Self {acks: self.acks + 1}) + } + } + + fn reject(&self) -> Box { + Box::new(Draft {}) } } struct Published {} impl State for Published { - fn request_review(self: Box::) -> Box { - self + fn request_review(&self) -> Box { + Box::new(Self {}) } - fn approve(self: Box::) -> Box { + fn approve(&mut self) -> Box { Box::new(Published {}) } + fn reject(&self) -> Box { + Box::new(Self {}) + } + fn content<'a>(&self, post: &'a Post) -> &'a str { &post.content }