rsass 0.29.2

Sass implementation in pure rust (not complete yet)
Documentation
use super::{unnamed, CheckedArg, FunctionMap};
use crate::css::{CssSelectorSet, Selector, Value};
use crate::value::ListSeparator;
use crate::Scope;

pub fn create_module() -> Scope {
    let mut f = Scope::builtin_module("sass:selector");
    def!(f, is_superselector(super, sub), |s| {
        let sup: CssSelectorSet = s.get(name!(super))?;
        let sub: CssSelectorSet = s.get(name!(sub))?;
        Ok(sup.is_superselector(&sub).into())
    });
    def_va!(f, append(selectors), |s| {
        let mut s = unnamed(s.get_va::<CssSelectorSet>(name!(selectors)))?
            .into_iter();
        if let Some(base) = s.next() {
            Ok(s.try_fold(base, |base, s| base.append(&s))?.into())
        } else {
            Err("At least one selector must be passed.")
                .named(name!(selectors))
        }
    });
    def!(f, extend(selector, extendee, extender), |s| {
        let selector: CssSelectorSet = s.get(name!(selector))?;
        let extendee: CssSelectorSet = s.get(name!(extendee))?;
        let extender: CssSelectorSet = s.get(name!(extender))?;
        Ok(selector.extend(&extendee, &extender)?.into())
    });
    def_va!(f, nest(selectors), |s| {
        let mut v = unnamed(s.get_va(name!(selectors)))?.into_iter();
        let first = v
            .next()
            .ok_or("At least one selector must be passed.")
            .named(name!(selectors))?;
        let first = CssSelectorSet::try_from(first)?;
        Ok(v.fold(first, |b, e| b.nest(e, &b)).into())
    });
    def!(f, parse(selector), |s| {
        CssSelectorSet::parse_value(s.get(name!(selector))?)
            .named(name!(selector))
            .map(Into::into)
    });
    def!(f, replace(selector, original, replacement), |s| {
        let selector: CssSelectorSet = s.get(name!(selector))?;
        let original: CssSelectorSet = s.get(name!(original))?;
        let replacement: CssSelectorSet = s.get(name!(replacement))?;
        Ok(selector.replace(&original, &replacement)?.into())
    });
    def!(f, simple_selectors(selector), |s| {
        let selector: Selector = s.get(name!(selector))?;
        let result = selector
            .simple_selectors()?
            .into_iter()
            .map(Value::from)
            .collect();
        Ok(Value::List(result, Some(ListSeparator::Comma), false))
    });
    def!(f, unify(selector1, selector2), |s| {
        let a: CssSelectorSet = s.get(name!(selector1))?;
        let b: CssSelectorSet = s.get(name!(selector2))?;
        Ok(a.unify(&b).into())
    });

    f
}

pub fn expose(m: &Scope, global: &mut FunctionMap) {
    for (gname, lname) in &[
        (name!(is_superselector), name!(is_superselector)),
        (name!(selector_append), name!(append)),
        (name!(selector_extend), name!(extend)),
        (name!(selector_nest), name!(nest)),
        (name!(selector_parse), name!(parse)),
        (name!(selector_replace), name!(replace)),
        (name!(selector_unify), name!(unify)),
        (name!(simple_selectors), name!(simple_selectors)),
    ] {
        global.insert(gname.clone(), m.get_lfunction(lname));
    }
}