Ever wanted a piece of functionality on an object or function that doesn't exist? Inspired by Laravel, "macros" makes it a breeze to add on custom functionality to a given object or function.
Here's a quick example:
import { is, macro } from '@vicgutt/macrojs';
const target = [];
is.macroed(target); // false
macro(target, 'isEmpty', function () {
return this.length === 0;
});
is.macroed(target); // true
target.isEmpty(); // true
target.push(123);
target.isEmpty(); // false
Install the package via NPM (or yarn):
npm i @vicgutt/macrojs
yarn add @vicgutt/macrojs
Note: This library is very "future facing" in the code that is distributed (dist folder), meaning it requires at least Node14+ and ES2020/ES2021 support from your JS compiler/bundler or browser.
The assign()
function copies all enumerable own properties from one source
object to a target object. It returns the modified target object.
Properties in the target object are overwritten by properties in the source if they have the same key. Later source's properties overwrite earlier ones.
This function intends to work similarly but NOT identically to the native
Object.assign()
method.
They differ in that this function does NOT invoke getters and setters but rather
copy their definition (just like any other property) to the target object by
using Object.getOwnPropertyDescriptor()
and Object.defineProperties()
.
This function accepts a callback as 3rd argument.
If the callback returnsfalse
, the ongoing copying task will be interrupted but
previously copied properties will remain.
Note
- Both String and Symbol properties are copied.
- This function will throw on a non object target and on a
null
orundefined
source. - Properties on the prototype chain and non-enumerable properties cannot be copied.
- Functions and their prototype can be used as target and/or source.
- If the source value is a reference to an object, it only copies the reference (unsuitable for "deep cloning").
import { assign } from '@vicgutt/macrojs';
const source = { name: 'Bob', age: 77 };
assign({}, source); // { name: 'Bob', age: 77 }
assign({}, source, () => {}); // { name: 'Bob', age: 77 }
assign({}, source, (propertyKey) => {
if (propertyKey === 'age') {
return false;
}
}); // { name: 'Bob' }
assign({}, source, (propertyKey, propertydescriptor, assignedDescriptors, index, propertyKeys) => {
assignedDescriptors[`--${String(propertyKey)}`] = {
value: `${index} | ${String(propertyKey)} | ${propertyKeys}`,
configurable: true,
enumerable: propertyKey === 'name',
writable: true,
};
}); // { '--name': '0 | name | name,age', name: 'Bob', age: 77 }
Register a custom macro on a given target.
A macro
is simply the term used to define custom properties
and methods that should be copied/cloned over into a given target.
A macro
consist of a "name" (the identifier) and a "value" (the implementation).
This function, also called macro
allows us to register a custom property on a given target.
It has the following signature:
- target (
unknown
): The object/function onto which the new property/method should be added. - propertyName (
string
): The property/method name that should be registered onto the target. - propertyValue (
unknown
): The property/method implementation that should be registered for the given name. - options (
MacroOptions
):- force (
boolean
| defaults tofalse
): Force replace existing properties and/or properties previously added via macro. - onFunctionPrototype (
boolean
| defaults totrue
): When the target is a function, should the new property be added to the target's prototype or to the target directly, essentially making it a static property.
- force (
import { macro } from '@vicgutt/macrojs';
const target = [];
macro(target, 'hello', () => 'hello!');
target.hello(); // 'hello!'
macro(target, 'isEmpty', function () {
return this.length === 0;
});
target.isEmpty(); // true
Register a collection of macros on a given target.
This function, as opposed to the macro()
function, allows us
to register multiple properties/methods at once on a given target.
It has the following signature:
- target (
unknown
): The object/function onto which the new properties/methods should be added. - properties (
object
|function
):- If an object, the keys will be used for the property name and it's value for the property's implementation.
- If a function, all it's static and prototype enumerable own properties will be copied over onto the target.
- callback (
function
|null
): A function, when present, gives us full control over how we'd like the copying to proceed. - options (
MacroOptions
):- force (
boolean
| defaults tofalse
): Force replace existing properties and/or properties previously added via macro. - onFunctionPrototype (
boolean
| defaults totrue
): When the target is a function, should the new property be added to the target's prototype or to the target directly, essentially making it a static property.
- force (
import { mixin } from '@vicgutt/macrojs';
const target = [];
mixin(target, {
hello: () => 'hello!',
isEmpty() {
return this.length === 0;
},
});
target.hello(); // 'hello!'
target.isEmpty(); // true
mixin(
{ name: 'Bob', age: NaN },
{
isHuman: '🤔',
isTroubleMaker: '👀',
},
(propertyKey, target, properties, options) => {
console.log(propertyKey, target, properties, options);
}
);
// isHuman { name: 'Bob', age: NaN } { isHuman: '🤔', isTroubleMaker: '👀' } { force: false, onFunctionPrototype: true }
// isTroubleMaker { name: 'Bob', age: NaN } { isHuman: '🤔', isTroubleMaker: '👀' } { force: false, onFunctionPrototype: true }
Register a collection of macros on a given target based on a defined condition.
It has the following signature:
- target (
unknown
): The object/function onto which the new properties/methods should be added. - objects (
PolyfillObject
|PolyfillObject[]
):- property (
string
): The property/method name that should be registered onto the target. - needed (
boolean
): Determines if the polyfill is needed. Iffalsy
, the property/method will be discarded. - implemention (
unknown
): The property/method implementation that should be registered for the given name.
- property (
import { polyfill } from '@vicgutt/macrojs';
polyfill(Array.prototype, [
{
property: 'at',
needed: !('at' in Array.prototype),
implemention() {
//
},
},
{
property: 'isEmpty',
needed: !('isEmpty' in Array.prototype),
implemention() {
return this.length === 0;
},
},
{
property: 'isNotEmpty',
needed: !('isNotEmpty' in Array.prototype),
implemention() {
return !this.isEmpty();
},
},
{
property: 'push',
needed: true, // will overwrite the existing method
implemention() {
return 'hijacked 😱';
},
},
]);
[].isEmpty(); // true
[].isNotEmpty(); // false
[].push(123); // 'hijacked 😱'
['hey'].at(0); // 'hey'
['hey'].at(-1); // 'hey'
['hey'].at(1); // undefined
Make an object/function "macroable".
A "macroable"d object or function is an object/function that implements the following methods:
- hasMacro: behaves exactly like the
isMacroedWith()
function, and where the "target" is the object/function itself. - macro: behaves exactly like the
macro()
function, and where the "target" is the object/function itself. - mixin: behaves exactly like the
mixin()
function, and where the "target" is the object/function itself.
import { macroable } from '@vicgutt/macrojs';
class Week {}
macroable(Week);
Week.prototype.macro('totalDays', 7);
Week.prototype.mixin({
days: ['monday', '...'],
firstDay(country, _default = 'monday') {
if (country === 'US') {
return 'sunday';
}
if (country === 'FR') {
return 'monday';
}
return _default;
},
});
Week.prototype.hasMacro('totalDays'); // true
Week.prototype.hasMacro('days'); // true
Week.prototype.hasMacro('firstDay'); // true
Week.prototype.hasMacro('nope'); // false
Week.prototype.firstDay('FR'); // 'monday'
new Week().firstDay('FR'); // 'monday'
This function is an alias of the macro
function.
This function is an alias of the mixin
function.
Determines whether the given value is "macroable".
The term "macroable" is used to identify objects or
functions that can be extended using the macro()
and mixin()
functions.
A value is considered "macroable" if:
- The value is NOT null
- The value is an object or a function
- The value's extension is NOT prevented
(Object.isExtensible(value) === true)
- The value is NOT "sealed"
(Object.isSealed(value) === false)
- The value is NOT "frozen"
(Object.isFrozen(value) === false)
import { isMacroable } from '@vicgutt/macrojs';
isMacroable(null); // false
isMacroable('hello'); // false
isMacroable(Object.seal({})); // false
isMacroable([]); // true
isMacroable({}); // true
isMacroable(Array); // true
isMacroable(() => {}); // true
isMacroable(document.querySelectorAll('body')); // true
Determines whether the given value is "macroabled".
The term "macroabled" is used to identify objects or
functions that have been extended using the macroable()
function.
All "macroabled" values are both macroable
and macroed
values.
A value is considered "macroabled" if:
- The value implements a
hasMacro
method, where the "target" is the value itself and that behaves exactly like theisMacroedWith()
function. - The value implements a
macro
method, where the "target" is the value itself and that behaves exactly like themacro()
function. - The value implements a
mixin
method, where the "target" is the value itself and that behaves exactly like themixin()
function.
import { isMacroabled } from '@vicgutt/macrojs';
isMacroabled({ hasMacro() {}, macro() {}, mixin() {} }); // false
isMacroabled({ hasMacro: isMacroedWith, macro: macro, mixin: mixin }); // false
isMacroabled(macroable({})); // true
Determines whether the given value is "macroed".
The term "macroed" is used to identify objects or
functions that have been extended using either the macro()
or the mixin()
function.
import { isMacroed } from '@vicgutt/macrojs';
const value = [];
// Example 1
mixin(Array, {});
mixin(value, {});
isMacroed(Array); // false
isMacroed(Array.prototype); // false
isMacroed(value); // false
// Example 2
mixin(Array, { yolo: 'best word ever!' });
isMacroed(Array); // false
isMacroed(Array.prototype); // true
isMacroed(value); // true
// Example 3
mixin(value, { yolo: 'best word ever!' });
isMacroed(Array); // false
isMacroed(Array.prototype); // false
isMacroed(value); // true
Determines whether the given value has been "macroed" with a set of given properties.
import { isMacroedWith } from '@vicgutt/macrojs';
const value = [];
macro(value, 'prop1', 'value1');
mixin(value, {
prop2: 'value2',
prop3: 'value3',
});
isMacroedWith(value, 'prop1'); // true
isMacroedWith(value, ['prop1', 'prop2']); // true
isMacroedWith(value, ['prop1', 'prop2', 'prop3']); // true
isMacroedWith(value, ['prop1', 'prop2', 'prop3', 'prop4']); // false
isMacroedWith(value, 'prop'); // false
If you're interested in contributing to the project, please read our contributing docs before submitting a pull request.
The "Available functions" portion of this README is generated by parsing each function's jsDoc.
The MIT License (MIT). Please see License File for more information.