@ryanmorr/sonic

1.0.2 • Public • Published

sonic

Version Badge License Build Status

A modern, context-aware, and extendable CSS selector engine built on top of querySelectorAll.

Install

Download the CJS, ESM, UMD versions or install via NPM:

npm install @ryanmorr/sonic

Usage

Find a single element:

import { find } from '@ryanmorr/sonic';

// Returns the matching element or null if no match is found
const element = find('#container');

Query for multiple elements:

import { query } from '@ryanmorr/sonic';

// Returns an array of all matching elements
const elements = query('.items');

Check if an element matches a selector string:

import { is } from '@ryanmorr/sonic';

const isMatch = is(element, 'div.class[attr=value]');

Provide an element or selector string as an optional second argument as the root of the query:

const element = find('[attr]', element);
const elements = query(':first-child', '#header');

Use leading combinators:

const divs = query('> div');
const blocks = query('+ .block');
const checked = query('~ :checked');

Extend by creating custom pseudo-class selectors (must return a boolean):

import { find, query, pseudos } from '@ryanmorr/sonic';

pseudos.foo = (element) => {
    return element.hasAttribute('foo');
};

pseudos.bar = (element, value) => {
    return element.hasAttribute(value);
};

const element = find(':foo');
const elements = query(':bar(class)');

Context-Aware

Sonic addresses the long-standing flaw in querySelector and querySelectorAll that sees element-rooted queries search relative to the document and not the element itself:

<section id="container">
    <em>Level 1</em>
    <section>
        <em>Level 2</em>
    </section>
</section>
// Expected <em>Level 2</em>, but returns <em>Level 1</em>, doh!
document.querySelector('#container').querySelector('section em');

Apparently, this behavior is purported to be correct given how long it has endured. Sonic, on the other hand, abides by the principle of least surprise and gives you exactly what you expect.

// Returns <em>Level 2</em> as expected, hooray!
const elements = query('section em', '#container');

License

This project is dedicated to the public domain as described by the Unlicense.

Readme

Keywords

Package Sidebar

Install

npm i @ryanmorr/sonic

Weekly Downloads

3

Version

1.0.2

License

Unlicense

Unpacked Size

14.8 kB

Total Files

6

Last publish

Collaborators

  • ryanmorr