1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
use num_traits::{PrimInt, Zero};
use crate::{Interval, Pitch};
use core::marker::PhantomData;
#[derive(Clone, Copy, Debug, PartialEq, Eq, PartialOrd, Ord)]
pub struct Set<T, U> {
pub bits: U,
_marker: PhantomData<T>,
}
impl<T, U: Zero> Default for Set<T, U> {
fn default() -> Self {
Self::new(U::zero())
}
}
impl<T, U> Set<T, U> {
fn new(bits: U) -> Self {
Self {
bits,
_marker: PhantomData,
}
}
pub fn pop_bit(&mut self) -> Option<u8>
where
U: PrimInt,
{
if !self.bits.is_zero() {
let trailing = self.bits.trailing_zeros();
self.bits = self.bits & (self.bits - U::one());
Some(trailing as u8)
} else {
None
}
}
}
impl<T, U> Set<T, U>
where
T: Into<u8>,
U: PrimInt,
{
pub fn push(&mut self, item: T) {
self.bits = self.bits | (U::one() << item.into() as usize);
}
pub fn remove(&mut self, item: T) {
self.bits = self.bits | !(U::one() << item.into() as usize);
}
pub fn contains(&self, item: T) -> bool {
(self.bits >> item.into() as usize & U::one()).is_one()
}
pub fn split(self, item: T) -> (Self, Self) {
let byte = item.into() as usize;
(
Self::new(self.bits & ((U::one() << byte) - U::one())),
Self::new((self.bits >> byte) << byte),
)
}
}
pub type IntervalSet = Set<Interval, u32>;
impl IntervalSet {
pub fn modes(self) -> impl Iterator<Item = Self> {
self.enumerate().map(move |(index, _)| {
let rotated = self.bits.rotate_right(index as _);
Self::new(rotated)
})
}
}
impl FromIterator<Interval> for IntervalSet {
fn from_iter<T: IntoIterator<Item = Interval>>(iter: T) -> Self {
let mut pitch_set = Self::default();
for pitch in iter {
pitch_set.push(pitch);
}
pitch_set
}
}
impl FromIterator<Pitch> for Set<Pitch, u16> {
fn from_iter<T: IntoIterator<Item = Pitch>>(iter: T) -> Self {
let mut pitch_set = Self::default();
for pitch in iter {
pitch_set.push(pitch);
}
pitch_set
}
}
impl<T, U> Extend<T> for Set<T, U>
where
T: Into<u8>,
U: PrimInt,
{
fn extend<I: IntoIterator<Item = T>>(&mut self, iter: I) {
for item in iter {
self.push(item);
}
}
}
impl<T, U> Iterator for Set<T, U>
where
T: From<u8>,
U: PrimInt,
{
type Item = T;
fn next(&mut self) -> Option<Self::Item> {
self.pop_bit().map(Into::into)
}
}