`conseq!`
======
[](https://github.com/pstroka/conseq)
[](https://crates.io/crates/conseq)
[](https://docs.rs/conseq)
A `conseq!` macro to conditionally repeat a fragment of source code and
substitute into each repetition a sequential numeric counter.
This is a fork of [`seq!`](https://github.com/dtolnay/seq-macro), extended with the following features:
- apply the repetitions conditionally
- use the `conseq!` macro inside another `conseq!` macro
```toml
[dependencies]
conseq = "0.1.0"
```
```rust
use conseq::conseq;
fn main() {
let tuple = (1000, 100, 10);
let mut sum = 0;
// Expands to:
//
// sum += tuple.0;
// sum += tuple.1;
// sum += tuple.2;
//
// This cannot be written using an ordinary for-loop because elements of
// a tuple can only be accessed by their integer literal index, not by a
// variable.
conseq!(N in 0..=2 {
sum += tuple.N;
});
assert_eq!(sum, 1110);
}
```
- If the input tokens contain a section surrounded by `$[N](` ... `)*`, where N is the name of
the numeric counter, then only that part is repeated.
- The numeric counter can be pasted onto the end of some prefix to form
sequential identifiers.
```rust
use conseq::conseq;
conseq!(N in 64..=127 {
#[derive(Debug)]
enum Demo {
// Expands to Variant64, Variant65, ...
$[N](
Variant~N,
)*
}
});
fn main() {
assert_eq!("Variant99", format!("{:?}", Demo::Variant99));
}
```
- You can also use comparison operators (>, >=, <, <=, ==) inside `$[` ... `]` to repeat the
sequence conditionally.
```rust
use conseq::conseq;
fn main() {
let tuple = (1000, 100, 10);
let mut sum1 = 0;
let mut sum2 = 0;
let mut sum3 = 0;
let mut sum4 = 0;
let mut sum5 = 0;
conseq!(N in 0..=2 {
$[N](sum1 += tuple.N;)*
$[N < 1](sum2 += tuple.N;)*
$[N < 2](sum3 += tuple.N;)*
$[N >= 1](sum4 += tuple.N;)*
$[N == 2](sum5 += tuple.N;)*
});
assert_eq!(sum1, 1110);
assert_eq!(sum2, 1000);
assert_eq!(sum3, 1100);
assert_eq!(sum4, 110);
assert_eq!(sum5, 10);
}
```
- Here is a more useful example
```rust
use conseq::conseq;
struct VarArgs {
tuple: T,
}
conseq!(N in 0..3 {
impl<$[N < 1](T~N: PartialEq,)*> From<($[N < 1](T~N,)*)> for VarArgs<($[N < 1](T~N,)*)> {
fn from(tuple: ($[N < 1](T~N,)*)) -> Self {
VarArgs { tuple }
}
}
impl<$[N < 2](T~N: PartialEq,)*> From<($[N < 2](T~N,)*)> for VarArgs<($[N < 2](T~N,)*)> {
fn from(tuple: ($[N < 2](T~N,)*)) -> Self {
VarArgs { tuple }
}
}
impl<$[N < 3](T~N: PartialEq,)*> From<($[N < 3](T~N,)*)> for VarArgs<($[N < 3](T~N,)*)> {
fn from(tuple: ($[N < 3](T~N,)*)) -> Self {
VarArgs { tuple }
}
}
});
fn main() {
let args1 = VarArgs::from((123,));
let args2 = VarArgs::from((123, "text"));
let args3 = VarArgs::from((123, "text", 1.5));
assert_eq!(args1.tuple, (123,));
assert_eq!(args2.tuple, (123, "text"));
assert_eq!(args3.tuple, (123, "text", 1.5));
}
```
- You can also use a `conseq!` macro inside another `conseq!` macro
```rust
use conseq::conseq;
conseq!(N in 1..=3 {
struct Struct<$[N](T~N,)*> {
$[N](field~N: T~N,)*
another_field: String,
}
impl<$[N](T~N,)*> Struct<$[N](T~N,)*> {
fn new($[N](field~N: T~N,)*) -> Self {
let mut another_field = String::from("");
// I know, this is a stupid example,
// but I wanted to demonstrate that you can actually do that
conseq!(L in 'a'..='z'{ another_field.push(L); });
Struct { $[N](field~N,)* another_field }
}
}
});
fn main() {
let s = Struct::new(1, 35.6, "abc");
assert_eq!(s.field1, 1);
assert_eq!(s.field2, 35.6);
assert_eq!(s.field3, "abc");
assert_eq!(s.another_field, "abcdefghijklmnopqrstuvwxyz");
}
```
- Byte and character ranges are supported: `b'a'..=b'z'`, `'a'..='z'`.
- If the range bounds are written in binary, octal, hex, or with zero padding,
those features are preserved in any generated tokens.
```rust
use conseq::conseq;
conseq!(P in 0x000..=0x00F {
// expands to structs Pin000, ..., Pin009, Pin00A, ..., Pin00F
struct Pin~P;
});
```
#### License
Licensed under either of Apache License, Version
2.0 or MIT license at your option.
Unless you explicitly state otherwise, any contribution intentionally submitted
for inclusion in this crate by you, as defined in the Apache-2.0 license, shall
be dual licensed as above, without any additional terms or conditions.