riffdiff 2.22.2

A diff filter highlighting changed line parts
// FIXME: The doctest doesn't run on `cargo test`, why?
/// Splits string into a vector of words. A word is any sequence of alphanumeric
/// characters. Non-words get into the vector one and one.
///
/// ```rust
/// assert_eq!(tokenize("Adam, Bea"), ["Adam", ", ", " ", "Bea"]);
/// ```
pub fn tokenize(input: &str) -> Vec<&str> {
    let mut first_word_part_byte_index = 0;
    let mut last_was_word_part = false;
    let mut result: Vec<&str> = Vec::with_capacity(input.len());
    let mut byte_index = 0;
    for character in input.chars() {
        let current_is_word_part = (character == '_') || character.is_alphanumeric();
        if current_is_word_part {
            if last_was_word_part {
                // Do nothing, keep working on the current word.
            } else {
                // Last was something else, we just started a new word.
                first_word_part_byte_index = byte_index;
            }
        } else {
            // Current is something else
            if last_was_word_part {
                // Push the word that just ended
                result.push(&input[first_word_part_byte_index..byte_index]);
            }

            // Push current char
            result.push(&input[byte_index..(byte_index + character.len_utf8())]);
        }

        last_was_word_part = current_is_word_part;
        byte_index += character.len_utf8();
    }

    if last_was_word_part {
        // Push the ending word
        result.push(&input[first_word_part_byte_index..]);
    }

    return result;
}

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

    #[cfg(test)]
    use pretty_assertions::assert_eq;

    #[test]
    fn test_empty() {
        let no_strings: Vec<String> = Vec::new();
        assert_eq!(tokenize(""), no_strings);
    }

    #[test]
    fn test_words() {
        assert_eq!(tokenize("word"), ["word"]);
        assert_eq!(tokenize("Adam Bea"), ["Adam", " ", "Bea"]);
    }

    #[test]
    fn test_numbers() {
        assert_eq!(tokenize("123"), ["123"]);
        assert_eq!(tokenize("123 456"), ["123", " ", "456"]);
    }

    #[test]
    fn test_alphanumeric() {
        assert_eq!(tokenize("0xC0deCafe"), ["0xC0deCafe"]);
    }

    #[test]
    fn test_others() {
        assert_eq!(tokenize("+!,"), ["+", "!", ","]);
    }

    #[test]
    fn test_non_breaking_space() {
        assert_eq!(tokenize("\u{00a0}"), ["\u{00a0}"]);
        assert_eq!(tokenize("\u{00a0}s"), ["\u{00a0}", "s"]);
    }
}