tune 0.32.0

Explore musical tunings and create synthesizer tuning files for microtonal scales.
Documentation
//! Operations for working with physical or virtual keyboards.

use crate::{math, temperament::EqualTemperament};

/// A physical or logical key on a real or virtual instrument without any notion of a pitch.
///
/// This struct does *not* represent a musical key, like in "F minor", which is why its name is [`PianoKey`].
#[derive(Copy, Clone, Debug, Eq, Hash, Ord, PartialEq, PartialOrd)]
pub struct PianoKey {
    midi_number: i32,
}

impl PianoKey {
    /// Creates a [`PianoKey`] instance from the given MIDI number.
    pub fn from_midi_number(midi_number: impl Into<i32>) -> Self {
        Self {
            midi_number: midi_number.into(),
        }
    }

    /// Returns the MIDI number of this [`PianoKey`].
    pub fn midi_number(self) -> i32 {
        self.midi_number
    }

    /// Returns the MIDI number of this [`PianoKey`] if it is in the valid MIDI range [0..128).
    ///
    /// # Examples
    ///
    /// ```
    /// # use tune::key::PianoKey;
    /// assert_eq!(PianoKey::from_midi_number(-1).checked_midi_number(), None);
    /// assert_eq!(PianoKey::from_midi_number(0).checked_midi_number(), Some(0));
    /// assert_eq!(PianoKey::from_midi_number(64).checked_midi_number(), Some(64));
    /// assert_eq!(PianoKey::from_midi_number(127).checked_midi_number(), Some(127));
    /// assert_eq!(PianoKey::from_midi_number(128).checked_midi_number(), None);
    /// ```
    pub fn checked_midi_number(self) -> Option<u8> {
        u8::try_from(self.midi_number)
            .ok()
            .filter(|midi_number| (0..128).contains(midi_number))
    }

    /// Iterates over all [`PianoKey`]s in the range [`self`..`upper_bound`).
    ///
    /// # Examples
    ///
    /// ```
    /// # use tune::key::PianoKey;
    /// let midi_key_62 = PianoKey::from_midi_number(62);
    /// let midi_key_67 = PianoKey::from_midi_number(67);
    ///
    /// assert_eq!(
    ///     midi_key_62.keys_before(midi_key_67).collect::<Vec<_>>(),
    ///     (62..67).map(PianoKey::from_midi_number).collect::<Vec<_>>()
    /// );
    /// assert!(midi_key_67.keys_before(midi_key_62).collect::<Vec<_>>().is_empty());
    /// ```
    pub fn keys_before(
        self,
        upper_bound: PianoKey,
    ) -> impl DoubleEndedIterator<Item = PianoKey> + ExactSizeIterator<Item = PianoKey> + 'static
    {
        (self.midi_number..upper_bound.midi_number).map(Self::from_midi_number)
    }

    /// Counts the number of keys [left inclusive, right exclusive) between `self` and `other`.
    ///
    /// # Examples
    ///
    /// ```
    /// # use tune::key::PianoKey;
    /// let midi_key_62 = PianoKey::from_midi_number(62);
    /// let midi_key_67 = PianoKey::from_midi_number(67);
    ///
    /// assert_eq!(midi_key_62.num_keys_before(midi_key_67), 5);
    /// assert_eq!(midi_key_67.num_keys_before(midi_key_62), -5);
    /// ```
    pub fn num_keys_before(self, other: PianoKey) -> i32 {
        other.midi_number - self.midi_number
    }

    /// Returns the key `num_steps` steps after `self`.
    ///
    /// # Examples
    ///
    /// ```
    /// # use tune::key::PianoKey;
    /// let midi_key_62 = PianoKey::from_midi_number(62);
    /// let midi_key_67 = PianoKey::from_midi_number(67);
    ///
    /// assert_eq!(midi_key_62.plus_steps(5), midi_key_67);
    /// assert_eq!(midi_key_67.plus_steps(-5), midi_key_62);
    /// ```
    pub fn plus_steps(self, num_steps: i32) -> PianoKey {
        PianoKey::from_midi_number(self.midi_number + num_steps)
    }
}

#[derive(Debug, Clone)]
pub struct Keyboard {
    root_key: PianoKey,
    primary_step: i16,
    secondary_step: i16,
}

impl Keyboard {
    pub fn root_at(root_key: PianoKey) -> Self {
        Self {
            root_key,
            primary_step: 2,
            secondary_step: 1,
        }
    }

    pub fn with_steps(mut self, primary_step: i16, secondary_step: i16) -> Self {
        self.primary_step = primary_step;
        self.secondary_step = secondary_step;
        self
    }

    pub fn with_steps_of(self, temperament: &EqualTemperament) -> Self {
        self.with_steps(temperament.primary_step(), temperament.secondary_step())
    }

    pub fn coprime(mut self) -> Keyboard {
        // Special case: Set sharp value to 1 if it is currently 0
        if self.primary_step == self.secondary_step {
            self.secondary_step = self.primary_step - 1;
            return self;
        }

        // Stretch sharp axis to make secondary step coprime with primary step
        loop {
            let gcd = self.gcd();

            if gcd == 1 {
                return self;
            }

            let sharp_value = self.primary_step - self.secondary_step;
            let expanded_sharp_value = sharp_value / gcd;
            self.secondary_step = self.primary_step - expanded_sharp_value;
        }
    }

    fn gcd(&self) -> i16 {
        gcd_i16(self.secondary_step, self.primary_step)
    }

    pub fn primary_step(&self) -> i16 {
        self.primary_step
    }

    pub fn secondary_step(&self) -> i16 {
        self.secondary_step
    }

    pub fn get_key(&self, x: i16, y: i16) -> PianoKey {
        let num_steps = i32::from(self.primary_step) * i32::from(x)
            - i32::from(self.secondary_step) * i32::from(y);
        self.root_key.plus_steps(num_steps)
    }
}

fn gcd_i16(numer: i16, denom: i16) -> i16 {
    math::gcd_u16(
        numer.abs().try_into().unwrap(),
        denom.abs().try_into().unwrap(),
    )
    .try_into()
    .unwrap()
}

#[cfg(test)]
mod tests {
    use super::*;
    use std::fmt::Write;

    #[test]
    fn keyboard_layout() {
        let mut output = String::new();
        for num_steps_per_octave in 1..100 {
            print_keyboard(&mut output, num_steps_per_octave);
        }
        std::fs::write("edo-keyboards-1-to-99.txt", &output).unwrap();
        assert_eq!(output, include_str!("../edo-keyboards-1-to-99.txt"));
    }

    pub fn print_keyboard(string: &mut String, num_steps_per_octave: u16) {
        let temperament = EqualTemperament::find().by_edo(num_steps_per_octave);
        let keyboard = Keyboard::root_at(PianoKey::from_midi_number(0))
            .with_steps_of(&temperament)
            .coprime();

        writeln!(string, "---- {}-EDO ----", num_steps_per_octave).unwrap();
        writeln!(
            string,
            "primary_step={}, secondary_step={}, num_cycles={}",
            temperament.primary_step(),
            temperament.secondary_step(),
            temperament.num_cycles(),
        )
        .unwrap();

        for y in (-5i16..5).rev() {
            for x in 0..10 {
                write!(
                    string,
                    "{:^4}",
                    keyboard
                        .get_key(x, y)
                        .midi_number()
                        .rem_euclid(i32::from(num_steps_per_octave)),
                )
                .unwrap();
            }
            writeln!(string).unwrap();
        }
    }
}