Skip to content

A library of gorgeous Javascript methods to make working with—and validating—data quicker and less error-prone, letting the developer concentrate on the fun stuff.

Notifications You must be signed in to change notification settings

lewishowles/helpers

Repository files navigation

Helpers

A library of gorgeous Javascript methods to make working with—and validating—data quicker and less error-prone, letting the developer concentrate on the fun stuff.

These helpers are grouped by type, which makes multiple imports a little more manageable. For example:

import { getNextIndex } from "@lewishowles/helpers/array";

Array

arrayLength(array)

Determine the number of items in the given array.

If the provided input is not an array, returns 0.

Example

arrayLength(['A', 'B', 'C', 'D']); // 4
arrayLength([]); // 0
arrayLength(undefined); // 0

firstDefined(array)

Returns the first non-undefined element in array.

Example

firstDefined(["a", "b"]); // "a"
firstDefined([undefined, undefined "c", "d"]); // "c"
firstDefined([]); // undefined

getNextIndex(index, reference, { reverse = false, wrap = false })

Given a starting index, determine the next available index in the reference array.

  • reverse - Reverse the direction, finding the previous index instead
  • wrap - When reaching the end of the array, wrap to the start, and vice-versa

If the provided index is outside of the reference array, or the provided index or reference array are unexpected, a default value of 0 is returned.

Example

getNextIndex(3, ['A', 'B', 'C', 'D']); // 3
getNextIndex(3, ['A', 'B', 'C', 'D'], { wrap: true }); // 0
getNextIndex(3, ['A', 'B', 'C', 'D'], { reverse: true }); // 2

head(array)

Returns the first element in array.

Example

head(["a", "b"]); // "a"
head([]); // undefined

isNonEmptyArray(variable)

Determines whether the given variable is both an array and has at least one item.

Example

isNonEmptyArray(['A', 'B', 'C', 'D']); // true
isNonEmptyArray([]); // false
isNonEmptyArray("string"); // false

lastDefined(array)

Returns the last non-undefined element in array.

Example

lastDefined(["a", "b"]); // "b"
lastDefined(["a", "b", undefined, undefined]); // "b"
lastDefined([]); // undefined

pluck(array, property)

Retrieve an array of the property value from each of the objects found in array.

Any non-objects in array are ignored.

Example

pluck([{ fruit: "apple" }, { fruit: "banana" }], "fruit"); // ["apple", "banana"]
pluck([{ nested: { property: { value: "seven" } } }, { nested: { property: { value: "eight" } } }], "nested.property.value"); // ["seven", "eight"]
pluck([{ nested: { property: { value: "seven" } } }], "nested.mistake.value"); // null
pluck([], "property"); // null

sortObjectsByProperty(array, property, { ascending: true })

Sort the given array of objects by the value of property property, with optional direction.

Example

sortObjectsByProperty([{ name: "Lewis" }, { name: "Alice" }], "name"); // [{ name: "Alice" }, { name: "Lewis" }]
sortObjectsByProperty([{ name: "Lewis" }, { name: "Alice" }], "age"); // [{ name: "Lewis" }, { name: "Alice" }]
sortObjectsByProperty([{ name: "Lewis" }, { name: "Alice" }], "name", { ascending: false }); // [{ name: "Lewis" }, { name: "Alice" }]

tail(array)

Returns the last element in array.

Example

tail(["a", "b"]); // "b"
tail([]); // undefined

unique(array)

Safely reduce the provided array to those entries which are unique.

Example

unique([1, 2, 2, 3, 4, 4, 5]); // [1, 2, 3, 4, 5]
unique(["a", "b", "a", "c"]); // ["a", "b", "c"]
unique([]); // []

Form

validateField(fieldName, validationRules, formData)

Validate a field given its fieldName, the field's validationRules, and the sum total formData.

Returns a non-empty array of error messages if an error is encountered.

Available rules and properties include:

required

[{ rule: "required", message: "Enter your name so we know what to call you" }]

Requires a value to be set. Adds the required attribute to the field automatically.

email

[{ rule: "email", message: "We need an email address to set up your account" }]

Perform a minimal check to see if the value contains an @ symbol. More complex verification isn't really necessary, and the only true way to test an email address is through verification.

size

[{ rule: "size", size: 11, message: "Your phone number should be 11 digits long" }]

Ensure that the provided value is has at least size size. For strings, the number of characters is used, for arrays, the length of the array, for objects, the number of properties, for numbers, the number itself is used, and for numeric strings the integer value of the string is used.

min

[{ rule: "min", min: 11, message: "Your phone number should be at least 11 digits long" }]

Ensure that the provided value is has at least size min. Values are evaluated as in the size rule.

max

[{ rule: "max", max: 11, message: "Your phone number should be no more than 11 digits long" }]

Ensure that the provided value is has at most size max. Values are evaluated as in the size rule.

between

[{ rule: "between", min: 5, max: 8, message: "Your post code should be between 5 and 8 characters" }]

Ensure that the provided value is has between min and max size. Values are evaluated as in the size rule.

in

[{ rule: "in", options: ["a", "b", "c"], message: "Your choice should be a, b, or c" }]

Ensure that the given value is included within options.

not_in

[{ rule: "not_in", options: ["a", "b", "c"], message: "Your choice should not include a, b, or c" }]

Ensure that the given value is not included within options.

regexp

[{ rule: "regexp", regexp: /[abc]+/, message: "Your ID should only contain the letters a, b, and c" }]

Ensure that the provided value matches regexp.

Example

validateField("username", [{ rule: "required", message: "Enter a username" }], { username: "" }) // ["Enter a username"]
validateField("username", [{ rule: "required", message: "Enter a username" }], { username: "jack_skellington" }) // []

General

getFriendlyDisplay(variable)

Convert a given variable into a human-readable representation of its type.

Example

getFriendlyDisplay(['A', 'B', 'C', 'D']); // <array[4]>
getFriendlyDisplay([]); // <array[0]>
getFriendlyDisplay("hello"); // hello <string>

isFunction(variable)

Determines whether the given variable is a function.

Example

isFunction(() => "Hello")); // true
isFunction("function"); // false
isFunction(5); // false

validateOrFallback(value, comparison, fallback)

Apply comparison to the provided value, returning value if true, and fallback if not.

comparison can be a function, or one of the keywords:

  • string: A string, including an empty string
  • boolean: Strictly true or false
  • number: A number, excluding NaN, based on isNumber
  • function: A function, based on isFunction
  • array: An array, including an empty array
  • object: An object, including an empty object

Example

validateOrFallback("value", "string", "fallback"); // "value"
validateOrFallback({}, isNonEmptyObject, null); // null
validateOrFallback(5, "number", 0); // 5

Number

clamp(number, minimum, maximum)

Ensure that the provided number is between minimum and maximum (inclusive). If number is lower than minimum, minimum is returned. If number is higher than maximum, maximum is returned.

If number is not a number, minimum is returned.

Example

clamp(4); // 4
clamp(10, 0, 5); // 5
clamp(-15, 0, 5); // 0
clamp(NaN); // 0

isNumber(variable)

Determines whether the given variable is a number and not NaN.

Example

isNumber(4); // true
isNumber(NaN); // false
isNumber("string"); // false

isNumeric(variable)

Determines whether the given variable is a number, or a string containing a number.

Example

isNumeric(4); // true
isNumeric(NaN); // false
isNumeric("string"); // false
isNumeric("10e3"); // true
isNumeric("5.6"); // true

round(number, precision = 0)

Rounds the given number to the specified precision. If precision is not provided, it defaults to 0.

Example

round(4.567); // 5
round(4.567, 2); // 4.57
round(4.567, 0); // 5
round(4.567, -1); // 0

Object

deepCopy(object)

Returns a recursive copy of object.

Example

deepCopy({ key: "value" }); // { key: "value" }
deepCopy(["a", "b"]); // ["a", "b"]

deepMerge(object)

Recursively merges two or more objects. The values of later objects override those of earlier objects.

Example

deepMerge({ key: "value" }, { value: "key" }); // { key: "value", value: "key" }
deepMerge({ key: "value", a: { b: 2 }}, { key: "modified", a: { c: 3 }}); // { key: "modified", a { b: 2, c: 3 }}

get(object, path)

Retrieve the object property value found at path, or null.

Example

get({ property: "value" }, "property"); // "value"
get({ nested: { property: { value: "seven" } } }, "nested.property.value"); // "seven"
get({ nested: { property: { value: "seven" } } }, "nested.mistake.value"); // null
get([], "property"); // null

isNonEmptyObject(variable)

Determines whether the given variable is both an object (and not null, or an array), and has at least one property.

Example

isNonEmptyObject({ property: "value" }); // true
isNonEmptyObject({}); // false
isNonEmptyObject("string"); // false

isObject(variable)

Determine whether the given variable is an object, excluding arrays and null.

Example

isObject({ property: "value" }); // true
isObject(['A', 'B', 'C', 'D']); // false
isObject(null); // false

keys(object)

Returns an array of the keys of the given object.

Example

keys({ a: 1, b: 2, c: 3 }); // ["a", "b", "c"]
keys({}); // []
keys("string"); // []

pick(object, properties)

Returns an object containing only properties properties from object.

Any non-string properties are ignored.

If the object does not have a given property, it is ignored.

Example

pick({ a: "one", b: "two", c: "three" }, ["a", "b"]); // { a: "one", b: "two" }
pick({ a: "one", b: "two", c: "three" }, ["a"]); // { a: "one" }
pick({ a: "one", b: "two", c: "three" }, ["a", "d"]); // { a: "one" }

objectContains(object, needle, { exclude: null, include: null, caseInsensitive: true, allowPartial: false })

Returns true if one of the object's values is needle. Also works when object is an array.

String needles are checked case-insensitively by default, and partial matches can be enabled via option.

exclude defines an array of properties to exclude from the search, while include defines an array of properties which are searched exclusively. If include is defined, exclude is ignored.

Example

objectContains({ name: "Merida" }, "merida"); // true
objectContains({ name: "Moana" }, "merida"); // false
objectContains({ name: "Mulan" }, "mulan", { caseInsensitive: false }); // false
objectContains({ name: { first: "Tiana" } }, "tia"); // false
objectContains({ name: { first: "Tiana" } }, "tia", { allowPartial: true }); // true
objectContains({ names: ["Ariel", "Jasmine"] }, "ariel"); // true
objectContains({ length: 52 }, 5); // false

omit(object, properties)

Returns a new object with the specified properties omitted from the given object.

Example

omit({ a: 1, b: 2, c: 3 }, ["b"]); // { a: 1, c: 3 }
omit({ a: 1, b: 2, c: 3 }, ["a", "c"]); // { b: 2 }
omit({ a: 1, b: 2, c: 3 }, []); // { a: 1, b: 2, c: 3 }

values(object)

Returns an array of the values of the given object.

Example

values({ a: 1, b: 2, c: 3 }); // [1, 2, 3]
values({}); // []
values("string"); // []

String

StringManipulator

StringManipulator can be used to chan string methods together safely. A new instance of the class is created, and then any of the string helper methods can be used in any sequence. If the input at any stage is invalid, an empty string is returned.

Example

const userId = "82FAA75F-B47A-43B6-82F6-389C9408BB67";

const userIdPreview = new StringManipulator(userId)
	.toLowerCase()
	.truncate(15)
	.value; // 82faa75f-b47a-…

isNonEmptyString(variable, { trim: false })

Determines whether the given variable is both a string and has at least one character. If trim is true, the string is trimmed of whitespace before the test is performed.

Example

isNonEmptyString("string"); // true
isNonEmptyString(""); // false
isNonEmptyString(["A", "B"]); // false
isNonEmptyString("  "); // true
isNonEmptyString("  ", { trim: true }); // false

ltrim(string, pattern="\\s")

Trim the left hand side of string using the provided string or RegExp pattern. Trims whitespace by default.

Example

ltrim("***string***", "*"); // **string***
ltrim("***string***", /\*/); // **string***
ltrim("***string***", /\*+/); // string***

rtrim(string, pattern="\\s")

Trim the right hand side of string using the provided string or RegExp pattern. Trims whitespace by default.

Example

rtrim("***string***", "*"); // ***string**
rtrim("***string***", /\*/); // ***string**
rtrim("***string***", /\*+/); // ***string

toLowerCase(variable)

A safe wrapper around toLowerCase, returning an empty string if the provided variable is not a string itself.

Example

toLowerCase("String"); // string
toLowerCase(""); // ""
toLowerCase(["A", "B"]); // ""

trim(string, pattern="\\s")

Trim both sides of string using the provided string or RegExp pattern. Trims whitespace by default.

Example

trim("   string   "); // string
trim("* *string* *", "*"); // *string*
trim("***string***", /\*/); // **string**
trim("***string***", /\*+/); // string

truncate(string, length = 10, { decoration = "…", preserveWords = false, strict = true, includeDecoration = true })

Truncate a string to a given length, with various options for how the truncation occurs.

If the provided variable is not a string, returns an empty string.

  • length: The length to truncate the string to
  • decoration: The decoration to append to the string if it is truncated
  • includeDecoration: Whether to include the length of the decoration in length calculations
  • preserveWords: Whether to avoid breaking a word during truncation
  • strict: When preserving words, if strict the resulting length will be less than the desired length

Example

toLowerCase("String"); // string
toLowerCase(""); // ""
toLowerCase(["A", "B"]); // ""

Vue

isNonEmptySlot(slotReference)

Determines whether the given slotReference contains content. slotReference is a direct reference to a slot - for example slots.default.

Example

// slots.default = () => [{ type: Text, children: "Text content" }]
isNonEmptySlot(slots.default); // true
// slots.default = () => [{ type: Comment }]
isNonEmptySlot(slots.default); // false
isNonEmptySlot("string"); // false

runComponentMethod(component, method, ...parameters)

If the given component is an object and contains a function parameter method, call that method with any additional parameters. This helper allows safe running of a method when an object might not exist.

Example

// component = { method: () => {} }
runComponentMethod(component, "method", "parameterOne"); // true
runComponentMethod(component, "undefinedMethod"); // undefined
runComponentMethod(null); // undefined

Chart

chartColours, extendedColours, and brightColours

Each returns a set of accessible chart colours. Colours are provided as Tailwind text colour classes. Full classes are provided to avoid any issues with Tailwind not being able to determine compound classes. Text colours are provided as text colours are more likely to be used elsewhere, as opposed to fill colours, and thus the final CSS bundle is likely to be smaller. To use a colour to fill an SVG shape, combine it with fill-current.

Note that when brightColours is used in a pie chart, there is a chance that two adjacent colours may not be sufficiently distinct based on the number of slices.

Example

import { chartColours } from "@lewishowles/helpers/chart";
import { getNextIndex } from "@lewishowles/helpers/array";

const startIndex = 0;
const nextIndex = getNextIndex(0, chartColours, { wrap: true });
const colour = chartColours[nextIndex];

getNextColour(index, overrideColours)

Retrieve the next colour for a chart segment based on the current index. Optionally provide the colours to choose from with overrideColours.

Example

import { getNextColour } from "@lewishowles/helpers/chart";
<path
	v-for="(figure, index) in figures"
	:key="figure.id"
	v-bind="{
		fill: getNextColour(index)
	}"
/>

Roadmap

There are a number of improvements and new helpers that could be made to improve flexibility.

  • ObjectManipulator - allowing a chain of object helpers to be applied safely in series.
  • ArrayManipulator - allowing a chain of array helpers to be applied safely in series.

About

A library of gorgeous Javascript methods to make working with—and validating—data quicker and less error-prone, letting the developer concentrate on the fun stuff.

Resources

Stars

Watchers

Forks

Packages

No packages published