David Tolnay | daaf774 | 2016-10-03 11:11:43 -0700 | [diff] [blame] | 1 | use {Path, PathSegment, QSelf, Ty}; |
David Tolnay | 5533772 | 2016-09-11 12:58:56 -0700 | [diff] [blame] | 2 | use aster::ident::ToIdent; |
| 3 | use aster::invoke::{Invoke, Identity}; |
| 4 | use aster::path::{PathBuilder, PathSegmentBuilder}; |
| 5 | use aster::ty::TyBuilder; |
Alex Crichton | ccbb45d | 2017-05-23 10:58:24 -0700 | [diff] [blame^] | 6 | use delimited::Delimited; |
David Tolnay | 5533772 | 2016-09-11 12:58:56 -0700 | [diff] [blame] | 7 | |
David Tolnay | daaf774 | 2016-10-03 11:11:43 -0700 | [diff] [blame] | 8 | // //////////////////////////////////////////////////////////////////////////// |
David Tolnay | 5533772 | 2016-09-11 12:58:56 -0700 | [diff] [blame] | 9 | |
David Tolnay | daaf774 | 2016-10-03 11:11:43 -0700 | [diff] [blame] | 10 | pub struct QPathBuilder<F = Identity> { |
David Tolnay | 5533772 | 2016-09-11 12:58:56 -0700 | [diff] [blame] | 11 | callback: F, |
| 12 | } |
| 13 | |
| 14 | impl QPathBuilder { |
| 15 | pub fn new() -> Self { |
| 16 | QPathBuilder::with_callback(Identity) |
| 17 | } |
| 18 | } |
| 19 | |
| 20 | impl<F> QPathBuilder<F> |
David Tolnay | daaf774 | 2016-10-03 11:11:43 -0700 | [diff] [blame] | 21 | where F: Invoke<(QSelf, Path)> |
David Tolnay | 5533772 | 2016-09-11 12:58:56 -0700 | [diff] [blame] | 22 | { |
| 23 | /// Construct a `QPathBuilder` that will call the `callback` with a constructed `QSelf` |
| 24 | /// and `Path`. |
| 25 | pub fn with_callback(callback: F) -> Self { |
David Tolnay | daaf774 | 2016-10-03 11:11:43 -0700 | [diff] [blame] | 26 | QPathBuilder { callback: callback } |
David Tolnay | 5533772 | 2016-09-11 12:58:56 -0700 | [diff] [blame] | 27 | } |
| 28 | |
| 29 | /// Build a qualified path first by starting with a type builder. |
| 30 | pub fn with_ty(self, ty: Ty) -> QPathTyBuilder<F> { |
| 31 | QPathTyBuilder { |
| 32 | builder: self, |
| 33 | ty: ty, |
| 34 | } |
| 35 | } |
| 36 | |
| 37 | /// Build a qualified path first by starting with a type builder. |
| 38 | pub fn ty(self) -> TyBuilder<Self> { |
| 39 | TyBuilder::with_callback(self) |
| 40 | } |
| 41 | |
| 42 | /// Build a qualified path with a concrete type and path. |
| 43 | pub fn build(self, qself: QSelf, path: Path) -> F::Result { |
| 44 | self.callback.invoke((qself, path)) |
| 45 | } |
| 46 | } |
| 47 | |
| 48 | impl<F> Invoke<Ty> for QPathBuilder<F> |
David Tolnay | daaf774 | 2016-10-03 11:11:43 -0700 | [diff] [blame] | 49 | where F: Invoke<(QSelf, Path)> |
David Tolnay | 5533772 | 2016-09-11 12:58:56 -0700 | [diff] [blame] | 50 | { |
| 51 | type Result = QPathTyBuilder<F>; |
| 52 | |
| 53 | fn invoke(self, ty: Ty) -> QPathTyBuilder<F> { |
| 54 | self.with_ty(ty) |
| 55 | } |
| 56 | } |
| 57 | |
David Tolnay | daaf774 | 2016-10-03 11:11:43 -0700 | [diff] [blame] | 58 | // //////////////////////////////////////////////////////////////////////////// |
David Tolnay | 5533772 | 2016-09-11 12:58:56 -0700 | [diff] [blame] | 59 | |
| 60 | pub struct QPathTyBuilder<F> { |
| 61 | builder: QPathBuilder<F>, |
| 62 | ty: Ty, |
| 63 | } |
| 64 | |
| 65 | impl<F> QPathTyBuilder<F> |
David Tolnay | daaf774 | 2016-10-03 11:11:43 -0700 | [diff] [blame] | 66 | where F: Invoke<(QSelf, Path)> |
David Tolnay | 5533772 | 2016-09-11 12:58:56 -0700 | [diff] [blame] | 67 | { |
| 68 | /// Build a qualified path with a path builder. |
| 69 | pub fn as_(self) -> PathBuilder<Self> { |
| 70 | PathBuilder::with_callback(self) |
| 71 | } |
| 72 | |
| 73 | pub fn id<T>(self, id: T) -> F::Result |
David Tolnay | daaf774 | 2016-10-03 11:11:43 -0700 | [diff] [blame] | 74 | where T: ToIdent |
David Tolnay | 5533772 | 2016-09-11 12:58:56 -0700 | [diff] [blame] | 75 | { |
| 76 | let path = Path { |
| 77 | global: false, |
Alex Crichton | ccbb45d | 2017-05-23 10:58:24 -0700 | [diff] [blame^] | 78 | segments: Delimited::new(), |
| 79 | leading_colon: None, |
David Tolnay | 5533772 | 2016-09-11 12:58:56 -0700 | [diff] [blame] | 80 | }; |
| 81 | self.as_().build(path).id(id) |
| 82 | } |
| 83 | |
| 84 | pub fn segment<T>(self, id: T) -> PathSegmentBuilder<QPathQSelfBuilder<F>> |
David Tolnay | daaf774 | 2016-10-03 11:11:43 -0700 | [diff] [blame] | 85 | where T: ToIdent |
David Tolnay | 5533772 | 2016-09-11 12:58:56 -0700 | [diff] [blame] | 86 | { |
| 87 | let path = Path { |
| 88 | global: false, |
Alex Crichton | ccbb45d | 2017-05-23 10:58:24 -0700 | [diff] [blame^] | 89 | segments: Delimited::new(), |
| 90 | leading_colon: None, |
David Tolnay | 5533772 | 2016-09-11 12:58:56 -0700 | [diff] [blame] | 91 | }; |
| 92 | self.as_().build(path).segment(id) |
| 93 | } |
| 94 | } |
| 95 | |
| 96 | impl<F> Invoke<Path> for QPathTyBuilder<F> |
David Tolnay | daaf774 | 2016-10-03 11:11:43 -0700 | [diff] [blame] | 97 | where F: Invoke<(QSelf, Path)> |
David Tolnay | 5533772 | 2016-09-11 12:58:56 -0700 | [diff] [blame] | 98 | { |
| 99 | type Result = QPathQSelfBuilder<F>; |
| 100 | |
| 101 | fn invoke(self, path: Path) -> QPathQSelfBuilder<F> { |
| 102 | QPathQSelfBuilder { |
| 103 | builder: self.builder, |
| 104 | qself: QSelf { |
| 105 | ty: Box::new(self.ty), |
| 106 | position: path.segments.len(), |
Alex Crichton | ccbb45d | 2017-05-23 10:58:24 -0700 | [diff] [blame^] | 107 | as_token: Default::default(), |
| 108 | gt_token: Default::default(), |
| 109 | lt_token: Default::default(), |
David Tolnay | 5533772 | 2016-09-11 12:58:56 -0700 | [diff] [blame] | 110 | }, |
| 111 | path: path, |
| 112 | } |
| 113 | } |
| 114 | } |
| 115 | |
David Tolnay | daaf774 | 2016-10-03 11:11:43 -0700 | [diff] [blame] | 116 | // //////////////////////////////////////////////////////////////////////////// |
David Tolnay | 5533772 | 2016-09-11 12:58:56 -0700 | [diff] [blame] | 117 | |
| 118 | pub struct QPathQSelfBuilder<F> { |
| 119 | builder: QPathBuilder<F>, |
| 120 | qself: QSelf, |
| 121 | path: Path, |
| 122 | } |
| 123 | |
| 124 | impl<F> QPathQSelfBuilder<F> |
David Tolnay | daaf774 | 2016-10-03 11:11:43 -0700 | [diff] [blame] | 125 | where F: Invoke<(QSelf, Path)> |
David Tolnay | 5533772 | 2016-09-11 12:58:56 -0700 | [diff] [blame] | 126 | { |
| 127 | pub fn id<T>(self, id: T) -> F::Result |
David Tolnay | daaf774 | 2016-10-03 11:11:43 -0700 | [diff] [blame] | 128 | where T: ToIdent |
David Tolnay | 5533772 | 2016-09-11 12:58:56 -0700 | [diff] [blame] | 129 | { |
| 130 | self.segment(id).build() |
| 131 | } |
| 132 | |
| 133 | pub fn segment<T>(self, id: T) -> PathSegmentBuilder<QPathQSelfBuilder<F>> |
David Tolnay | daaf774 | 2016-10-03 11:11:43 -0700 | [diff] [blame] | 134 | where T: ToIdent |
David Tolnay | 5533772 | 2016-09-11 12:58:56 -0700 | [diff] [blame] | 135 | { |
| 136 | PathSegmentBuilder::with_callback(id, self) |
| 137 | } |
| 138 | } |
| 139 | |
| 140 | impl<F> Invoke<PathSegment> for QPathQSelfBuilder<F> |
David Tolnay | daaf774 | 2016-10-03 11:11:43 -0700 | [diff] [blame] | 141 | where F: Invoke<(QSelf, Path)> |
David Tolnay | 5533772 | 2016-09-11 12:58:56 -0700 | [diff] [blame] | 142 | { |
| 143 | type Result = F::Result; |
| 144 | |
| 145 | fn invoke(mut self, segment: PathSegment) -> F::Result { |
Alex Crichton | ccbb45d | 2017-05-23 10:58:24 -0700 | [diff] [blame^] | 146 | self.path.segments.push_default(segment); |
David Tolnay | 5533772 | 2016-09-11 12:58:56 -0700 | [diff] [blame] | 147 | self.builder.build(self.qself, self.path) |
| 148 | } |
| 149 | } |