npm package discovery and stats viewer.

Discover Tips

  • General search

    [free text search, go nuts!]

  • Package details

    pkg:[package-name]

  • User packages

    @[username]

Sponsor

Optimize Toolset

I’ve always been into building performant and accessible sites, but lately I’ve been taking it extremely seriously. So much so that I’ve been building a tool to help me optimize and monitor the sites that I build to make sure that I’m making an attempt to offer the best experience to those who visit them. If you’re into performant, accessible and SEO friendly sites, you might like it too! You can check it out at Optimize Toolset.

About

Hi, 👋, I’m Ryan Hefner  and I built this site for me, and you! The goal of this site was to provide an easy way for me to check the stats on my npm packages, both for prioritizing issues and updates, and to give me a little kick in the pants to keep up on stuff.

As I was building it, I realized that I was actually using the tool to build the tool, and figured I might as well put this out there and hopefully others will find it to be a fast and useful way to search and browse npm packages as I have.

If you’re interested in other things I’m working on, follow me on Twitter or check out the open source projects I’ve been publishing on GitHub.

I am also working on a Twitter bot for this site to tweet the most popular, newest, random packages from npm. Please follow that account now and it will start sending out packages soon–ish.

Open Software & Tools

This site wouldn’t be possible without the immense generosity and tireless efforts from the people who make contributions to the world and share their work via open source initiatives. Thank you 🙏

© 2025 – Pkg Stats / Ryan Hefner

couli

v0.1.1

Published

Library for creating web user interfaces

Downloads

28

Readme

Couli

Constructor Of User interfaces LIbrary. Library for creating webcomponents that automatically change their appearance depending on the inner state as well as execute user-defined logic on DOM and life-cycle events.

The project is pretty small and raw but already seems to be performanceful.

How to install and prepare

Install the library through

npm install couli

then include with

var Couli = require('couli')

or

import Couli from 'couli'

in your script file.

couli.js file also can be included on the client through script tag, and resides at /dist/ folder of the project.

You can fiddle with the code for the library yourself and generate your own version of the library file with npm run-script build

For developing and learning process to become easier, start webpack-web-server with npm start command, and check out mini example apps at the /dist/ folder.

Usage

Documentation for the library is quite terse and is recommended to be read along with examining examples from /dist/ folder. Hopefully, that will be enough to understand main API concepts.

Creating component

Components are created with the call to Could.define function. With such signagure:

CouliComponent component = Couli.define (
  // string representing component markup. Must have a single root element.
  String markup,
  // entities describing the behavior of parts of the component. May or may not point to markup elements.
  Object bindings,
  // styles to be applied to a given component
  Object styles
  // tag to component mapping for inserting child components
  Object childCompnents
)

All arguments except markup are optional.

Bindings

Bindings describe the appearance and behavior of parts of a component. They can point to HTML elements within markup through x-b and x-lb (the difference is described below) attributes or they can just be used for some internal work calculations.
Bindings are defined inside the bidings hash with keys as binding names and values as binding options.
Binding can have the following options:

{
  value: ReactiveFunction (BindingsValues, ComponentInterface),
  html: ReactiveFunction (BindingsValues, ComponentInterface),
  style: ReactiveFunction (BindingsValues, ComponentInterface),
  class: ReactiveFunction (BindingsValues, ComponentInterface),
  attrs: ReactiveFunction (BindingsValues, ComponentInterface)
  events: {
    [domEventName]: EventHandler (Event, ComponentInterface),
    ...
  },
  hooks: {
    [lifeCycleEventName]: EventHandler (ComponentValues, ComponentInterface),
    ...
  }
}

Binding with the name of '' (empty string) is considered to be a component binding. Its configuration will be applied to the whole component.

Binding options

value, html, style, class and attrs functions will reevaluate each time values of bindings they listen to change
all of them have the following common signature:
ReactiveFunction (Object bindingsValues, ComponentInterface ci) Where: bindingsValues - values of all bindings in the component.
ci - entity allowing to get access to the underlying component, get and set values, move up and down in components hierarchy tree. More on what and how to use ComponentInterface see below.

Reactive functions differ in what they return and where they apply:

value

Returns value of the binding. The return value can be of any type. value itself is not applied to the markup unless you don't define html function.

html

Returns what will be innerHTML of the binding.

class

Must return an object with keys as class names and booleans as values expressing whether show or hide each class.

style

Must return an object with keys and values expressing what styles to apply to HTML markup.

attrs

Must return an object with keys and values expressing what attributes to change in HTML markup.

Events

events object contains event handlers for a piece of markup you assign a binding to.
Keys in events object are any valid event names (mouseup, blur, keyup etc.). Signature for all event handlers is:

EventHandler (Event e, ComponentInterface ci)

Where:
e - Event object.
ci - entity allowing to get access to the underlying component, get and set values, move up and down in components hierarchy tree. More on what and how to use ComponentInterface see below.


Hooks

These functions will be called at different binding's life events. For now: mount event works only for a whole component (binding name - '') and is called when a component is created and attached to DOM
update events work for whole components as well as individual bindings, called when a value in a component is updated.
remove events work for list items, for now, called after list item was removed from DOM

Signature for all component life-cycle handlers is:

LifeCycleHandler (ComponentValues, ComponentInterface ci)

Where
value - values of the whole component or value of a specific binding.
ci - entity allowing to get access to the underlying component, get and set values, move up and down in components hierarchy tree. More on what and how to use ci see below.


Styles

An object containing styles to be applied to a given component. Where the keys are any valid selectors, and values are objects describing styles for these selectors. To point to binding markup just use the binding name as a selector.

ComponentNesting

If childComponents argument is added to Couli.define function, the library will search for tags corresponding to the keys in this object and replace them with the corresponding components. Signature for defining child components:

{
    tagName: [ CouliComponent component, Links { childBindingName: bindingName } ]
}

where: tagName - tags in the component being defined you want to replace to components component - component for replacement Links - optional object to declare direct dependence of bindings of the component and child components, where childBindingName - binding name in a child component and bindingName - binding name in the component.


ComponentInterface

ComponentInterface is a thing that allows getting and setting binding values on the component, getting component and binding markups as well as moving from one component to another. Methods:

.get(String key)

Returns value of the binding with key name. Or if no argument is given, returns an object of all values of the component. If you are trying to get the value of a list binding, then instead of receiving the value itself, you will get another interface ArrayInterface for working with this list of values through which you will be able to efficiently add, remove, change and get values of the list. More on ArrayInterface see below.

.set(Object changeHash)

Sets values of the component's bindings to values of changeHash where keys are binding names.

.up()
.up(Number num)
.up(String componentUserName)

If no argument is given, returns ComponentInterface of the parent component.
If a numeric argument is given, returns ComponentInterface of the parent component num levels higher. If a string argument is given, returns ComponentInterface of the parent component with the name that corresponds to the string argument value. See below on how to give a component a custom name.

.markup(String bindingName)

Returns HTMLElement binding with bindingName is assigned to.

ArrayInterface

If a binding you want to .get() value from is of list type, then ArrayInterface is what you will get instead and here's methods that you can call on it.

.get(Number num)

Get ComponentInterface of list item at num position.

.push(Object newItem)

Add newItem to the end of the list.

.unshift (Object newItem)

Add newItem to the begining of the list.

.add(Number start, Object newItem)

Add newItem at num postion of the list.

Instead of passing only one Object, .push, .unshift and .add methods can as well receive an array of items instead.

.pop(Number num)

Remove num last elements from the list. num defaults to 1 if omitted.

.shift(Number num)

Remove num first elements from the list. num defaults to 1 if omitted.

.remove(Number start, Number num)

Remove num elements from the list starting from start position.

.value()

Get raw values of the list.

.forEach(Callback cb)

Iterates over the values of the list and call cb function with arguments (Object item, Number index) on each of the items.

.filter(Callback cb)

Returns a new list with items from the initial list for which cb function with arguments (Object item, Number index) returns a true value

.map(Callback cb)

Returns a new array with items being results of calling cb function with arguments (Object item, Number index) on each item of the initial list.

x-b and x-lb difference

x-b attribute points to HTMLElement for the binding x-lb attribute is assigned to HTMLElements that will have the role of list containers.
Such list bindings receive additional logic for rendering and mutating arrays of data. HTMLElement with x-lb attribute must have one child element which will be considered as a template for list item.

Applying components to page markup

Finally to apply created components to markup call Couli.apply(HTMLElement container, CouliComponent component)
where container is a HTMLElement on the page where you wish to place your component.

Binding option short keys

To save the time and effort of writing and reading various binding options in full length, one can use shortened keys that will correspond to full option keys.

|Short key| Full key| |---------|---------| |_s | style| |_c | class| |_v | value| |_a | attrs| |_h | html| | _eab | events.abort | | _eld | events.load | | _eer | events.error | | _ef | events.focus | | _efi | events.focusin | | _efo | events.focusout | | _eb | events.blur | | _eas | events.animationstart | | _eac | events.animationcanceled | | _eae | events.animationend | | _eai | events.animationiteration | | _ets | events.transitionstart | | _etc | events.transitioncancel | | _ete | events.transitionend | | _etr | events.transitionrun | | _efr | events.reset | | _efs | events.submit | | _ech | events.change | | _ekd | events.keydown | | _ekp | events.keypress | | _eku | events.keyup | | _emc | events.click | | _emx | events.contextmenu | | _emd | events.mousedown | | _eme | events.mouseenter | | _eml | events.mouseleave | | _emm | events.mousemove | | _emo | events.mouseover | | _emt | events.mouseout | | _emu | events.mouseup | | _empc | events.pointerlockchange | | _empe | events.pointerlockerror | | _emw | events.wheel | | _ems | events.select | | _ed | events.drag | | _ede | events.dragend | | _edr | events.dragenter | | _eds | events.dragstart | | _edl | events.dragleave | | _edo | events.dragover | | _edd | events.drop | | _emcp | events.canplay | | _emcpt | events.canplaythrough | | _emdc | events.duractionchange | | _emet | events.emptied | | _emnd | events.ended | | _emld | events.loadeddata | | _emlmd | events.loadedmetadata | | _emps | events.pause | | _empl | events.play | | _emplg | events.playing | | _emrc | events.ratechange | | _emsk | events.seeked | | _emskg | events.seeking | | _emst | events.stalled | | _emsd | events.suspend | | _emtu | events.timeupdate | | _emvc | events.volumechange | | _emwt | events.waiting | | _hu | hooks.update | | _hm | hooks.mount | | _hr | hooks.remove |