decasify 0.11.3

A CLI utility and library to cast strings to title-case according to locale specific style guides including Turkish support
Documentation
// SPDX-FileCopyrightText: © 2023 Caleb Maclennan <caleb@alerque.com>
// SPDX-License-Identifier: LGPL-3.0-only

#![doc = include_str!("../README.md")]
#![cfg_attr(docsrs, feature(doc_cfg))]

mod content;
mod generics;
mod traits;
pub mod types;

pub use content::Chunk;
#[cfg(feature = "unstable-trait")]
pub use traits::Decasify;
pub use types::{Case, Locale, StyleGuide, StyleOptions, StyleOptionsBuilder, Word};
pub use types::{Error, Result};

#[cfg(feature = "cli")]
#[doc(hidden)]
pub mod cli;

#[cfg(feature = "luamodule")]
#[doc(hidden)]
pub mod lua;

#[cfg(feature = "pythonmodule")]
#[doc(hidden)]
pub mod python;

#[cfg(feature = "wasm")]
#[doc(hidden)]
pub mod wasm;

mod en;
mod es;
mod tr;

/// Convert a string to a specific case following typesetting conventions for a target locale
pub fn case<TC, TL, TS, TO>(
    chunk: impl Into<Chunk>,
    case: TC,
    locale: TL,
    style: TS,
    opts: TO,
) -> Result<String>
where
    TC: TryInto<Case>,
    TL: TryInto<Locale>,
    TS: TryInto<StyleGuide>,
    TO: TryInto<StyleOptions>,
    Error: From<TC::Error>,
    Error: From<TL::Error>,
    Error: From<TS::Error>,
    Error: From<TO::Error>,
{
    let chunk: Chunk = chunk.into();
    let case: Case = case.try_into()?;
    let locale: Locale = locale.try_into()?;
    let style: StyleGuide = style.try_into()?;
    let opts: StyleOptions = opts.try_into()?;
    match case {
        Case::Lower => lowercase(chunk, locale),
        Case::Upper => uppercase(chunk, locale),
        Case::Sentence => sentencecase(chunk, locale),
        Case::Title => titlecase(chunk, locale, style, opts),
    }
}

/// Convert a string to title case following typesetting conventions for a target locale
pub fn titlecase<TL, TS, TO>(
    chunk: impl Into<Chunk>,
    locale: TL,
    style: TS,
    opts: TO,
) -> Result<String>
where
    TL: TryInto<Locale>,
    TS: TryInto<StyleGuide>,
    TO: TryInto<StyleOptions>,
    Error: From<TL::Error>,
    Error: From<TS::Error>,
    Error: From<TO::Error>,
{
    let chunk: Chunk = chunk.into();
    let locale: Locale = locale.try_into()?;
    let style: StyleGuide = style.try_into()?;
    let opts: StyleOptions = opts.try_into()?;
    Ok(match locale {
        Locale::EN => en::titlecase(chunk, style, opts),
        Locale::ES => es::titlecase(chunk, style, opts),
        Locale::TR => tr::titlecase(chunk, style, opts),
    })
}

/// Convert a string to lower case following typesetting conventions for a target locale
pub fn lowercase<TL>(chunk: impl Into<Chunk>, locale: TL) -> Result<String>
where
    TL: TryInto<Locale>,
    Error: From<TL::Error>,
{
    let chunk: Chunk = chunk.into();
    let locale: Locale = locale.try_into()?;
    Ok(match locale {
        Locale::EN => en::lowercase(chunk),
        Locale::ES => es::lowercase(chunk),
        Locale::TR => tr::lowercase(chunk),
    })
}

/// Convert a string to upper case following typesetting conventions for a target locale
pub fn uppercase<TL>(chunk: impl Into<Chunk>, locale: TL) -> Result<String>
where
    TL: TryInto<Locale>,
    Error: From<TL::Error>,
{
    let chunk: Chunk = chunk.into();
    let locale: Locale = locale.try_into()?;
    Ok(match locale {
        Locale::EN => en::uppercase(chunk),
        Locale::ES => es::uppercase(chunk),
        Locale::TR => tr::uppercase(chunk),
    })
}

/// Convert a string to sentence case following typesetting conventions for a target locale
pub fn sentencecase<TL>(chunk: impl Into<Chunk>, locale: TL) -> Result<String>
where
    TL: TryInto<Locale>,
    Error: From<TL::Error>,
{
    let chunk: Chunk = chunk.into();
    let locale: Locale = locale.try_into()?;
    Ok(match locale {
        Locale::EN => en::sentencecase(chunk),
        Locale::ES => es::sentencecase(chunk),
        Locale::TR => tr::sentencecase(chunk),
    })
}

fn get_override<F>(word: &Word, overrides: &Option<Vec<Word>>, case_fn: F) -> Option<Word>
where
    F: Fn(&String) -> String,
{
    let word_lower = case_fn(&word.word);
    overrides.as_ref().and_then(|words| {
        words
            .iter()
            .find(|w| case_fn(&w.word) == word_lower)
            .cloned()
    })
}