Huge News!Announcing our $40M Series B led by Abstract Ventures.Learn More
Socket
Sign inDemoInstall
Socket

blissful-hooks

Package Overview
Dependencies
Maintainers
0
Versions
2
Alerts
File Explorer

Advanced tools

Socket logo

Install Socket

Detect and block malicious and high-risk dependencies

Install

blissful-hooks

Deep extensibility for all!

  • 0.0.1
  • Source
  • npm
  • Socket score

Version published
Weekly downloads
4
increased by300%
Maintainers
0
Weekly downloads
 
Created
Source

Blissful Hooks

Deep extensibility for all!

A spinoff from ye olde BlissfulJS project. A precursor of this code powers the plugin system of PrismJS.

  • ✅ Tiny, dependency-free codebase. Literally < 50 loc.
  • ✅ Works everywhere, browsers, JS runtimes, you name it

Usage

For Library Authors

First, create a new Hooks instance, and expose it to third-party code. E.g. if your library is a class, it can be a static property:

// In a browser? Just import src/index.js instead
import Hooks from "blissful-hooks";

class MyAwesomeClass {
	static hooks = new Hooks();
}

If your library is a set of modules, it can just be an export:

import Hooks from "blissful-hooks";
export const hooks = new Hooks();

Or, if you don’t care about potential name clashes across other libraries using this module, you can just use/export the hooks singleton:

import { hooks } from "blissful-hooks";
export { hooks };

Now, at the places in your code where you want to allow extensions, call hooks.run(hookName, env).

hooks.run("hook-name", this);

Plugin code will now be executed with the same context as your code.

Instead of the current context, you can pass variables that plugins can read and manipulate! Ideally, it would be exposing local variables, and you'd be reading and writing directly from it. Then, you pass can still pass the current context to the plugin as a special this or context property (if both are present, this wins). For example, here’s how a syntax highlighter might use it:

let env = {
	callback,
	container,
	selector: 'code[class*="language-"], [class*="language-"] code',
	this: this,
};

hooks.run('before-highlightall', env);

env.elements = env.container.querySelectorAll(env.selector);

hooks.run('before-all-elements-highlight', env);

for (let element of env.elements) {
	highlightElement(element, env.callback);
}

hooks.run('after-all-elements-highlight', env);

For Plugin Authors

Plugin authors use hooks.add() to schedule code to run at a given hook. For example, suppose we wanted to change the selector above to also accept lang-* classes. It would be as simple as:

hooks.add("before-highlightall", function(env) {
	env.selector += ', code[class*="lang-"], [class*="lang-"] code';
});

You can also add multiple hooks at one:

hooks.add({
	"before-highlightall": function(env) {
		env.selector += ', code[class*="lang-"], [class*="lang-"] code';
	},
	"before-all-elements-highlight": function(env) {
		env.elements = Array.from(env.elements).filter(e => !e.classList.contains("no-highlight"));
	}
});

Or the same callback to multiple hooks:

hooks.add(["before-highlightall", "some-other-hook"], function(env) {
	env.selector += ', code[class*="lang-"], [class*="lang-"] code';
});

FAQ

Can a hook cause a function to early return?

Yes, but only with the consent of the library author. Simply returning a value from your hook callback will do nothing (duh). However, the library author can designate that a certain property holds a return value, and if that property is non-empty, they can choose to honor it:

let env = {this: this}
hooks.run("hook-name", env);

if (env.returnValue) {
	return env.returnValue;
}

What happens if I add code to a hook that doesn't exist?

Nothing happens. The code simply never gets executed. Libraries don't need to declare their hooks upfront (by design), so there is no way for Blissful Hooks to know the difference between a hook that hasn't run yet and a hook that never will.

Used By

It took 12 years of usage before I released this as a separate package, but precursors of it are used in several libraries:

Keywords

FAQs

Package last updated on 09 Aug 2024

Did you know?

Socket

Socket for GitHub automatically highlights issues in each pull request and monitors the health of all your open source dependencies. Discover the contents of your packages and block harmful activity before you install or update your dependencies.

Install

Related posts

SocketSocket SOC 2 Logo

Product

  • Package Alerts
  • Integrations
  • Docs
  • Pricing
  • FAQ
  • Roadmap
  • Changelog

Packages

npm

Stay in touch

Get open source security insights delivered straight into your inbox.


  • Terms
  • Privacy
  • Security

Made with ⚡️ by Socket Inc