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 🙏

© 2024 – Pkg Stats / Ryan Hefner

ts-compare-fn

v1.0.3

Published

Type-safe and autocomplete-friendly compare function creator with sensible defaults.

Downloads

301

Readme

ts-compare-fn

View this project on NPM License Package minified & gzipped size

Type-safe, autocomplete-friendly, and intuitive compare function creator with sensible defaults and a highly customizable API.

  • Sort an array of objects by a single or multiple keys
  • Supports nested paths, including in tuples, and custom getters
  • Provides robust type inference and autocomplete support
  • Works with strings, numbers, bigints, booleans, and dates
  • Uses Intl.Collator to compare strings for localization support
  • Default locale is environment-independent for consistency and better SSR compatibility
  • Uses numeric collation by default, so that "10.jpg", "1.jpg", "2.jpg" sorts as "1.jpg", "2.jpg", "10.jpg"
  • Gracefully handles null and undefined values

Installation

This library is published in the NPM registry and can be installed using any compatible package manager.

# NPM
npm install ts-compare-fn --save

# Yarn
yarn add ts-compare-fn

Usage

Basic

Pick a property name to sort by. The function will infer the type of the property and return a compare function to use with Array.prototype.sort or Array.prototype.toSorted.

import { compareFn } from 'ts-compare-fn';

type User = {
  id: number;
  name: string;
  address: {
    city: string;
  };
  emails: [string, string];
};

const users: User[] = [...];

// Sort by a single property
users.sort(compareFn('name'));
users.sort(compareFn('id'));

// To sort in descending order, prepend the property name with '-'
users.sort(compareFn('-name'));

// To sort by a nested property, use a dot-separated path
users.sort(compareFn('address.city'));
users.sort(compareFn('emails.0'));

// Sort by multiple properties
users.sort(compareFn('address.city', 'name'));

This works with array literals and standard TypeScript inference as well.

const users = [
  { name: 'Charlie', address: { city: 'New York' } },
  { name: 'Bob', address: { city: 'London' } },
  { name: 'Alice', address: { city: 'New York' } },
].sort(compareFn('address.city', 'name'));

You can also create a type-specific compare function once and use it throughout your codebase.

import { compareFn } from 'ts-compare-fn';

interface User {
  name: string;
  dateOfBirth: Date;
}

export const compareUsers = compareFn<User>('name', 'dateOfBirth');

Custom getters

This library allows you to pass a custom getter function to extract a value from an object. This is useful when you need to sort by a computed value or implement a custom sorting logic.

import { compareFn } from 'ts-compare-fn';

type User = { dateOfBirth: Date; name: string; };
const users: User[] = [...];

// Sort by a custom getter
users.sort(compareFn((user) => daysTillBirthday(user.dateOfBirth)));

// Mix and match with property names
users.sort(compareFn('name', (user) => daysTillBirthday(user.dateOfBirth)));

Note: getter return values are not cached, so it should be a pure function (always return the same value for the same object)

Options

Locale

The default locale is en-u-co-eor-kn, which means strings are compared using the European ordering rules and the numeric collation. You can customize this by passing an options object as the last argument.

import { compareFn } from 'ts-compare-fn';

type User = { name: string; age: number; city: string; };
const users: User[] = [...];

// Sort using custom locale
users.sort(compareFn('name', { locale: 'fr' }));

// Or provide a custom collator instance for more control
users.sort(compareFn(
  'age',
  'name',
  { collator: new Intl.Collator('fr', { sensitivity: 'case' }) }
));

// You can also specify locale or collator for a specific property
users.sort(compareFn(
  { path: 'city', collator: new Intl.Collator('en-US') },
  { path: 'name', locale: 'fr' }
));

Default values

By default, null and undefined values are sorted to the end of the array (beginning of the array for descending order). You can customize this behavior by passing a custom default value for a property.

import { compareFn } from 'ts-compare-fn';

const compounds = [
  { name: 'Water', formula: 'H₂0' },
  { name: 'Carbon dioxide', formula: 'CO₂', freezingPoint: -78.5 },
  { name: 'Glycerol', formula: 'C₃H₈O₃', freezingPoint: 17.8 },
];

// By default, undefined values are sorted to the end
compounds.sort(compareFn('freezingPoint')); // CO₂, C₃H₈O₃, H₂0

// You can specify a custom default value
compounds.sort(compareFn(
  'freezingPoint',
  { defaultValue: 0 }
)); // CO₂, H₂0, C₃H₈O₃

If you want to sort by multiple properties and provide default values for some of them, you need to provide a sort config object with the path and defaultValue properties.

compounds.sort(compareFn(
  { path: 'freezingPoint', defaultValue: 0 },
  'formula',
));

API

compareFn(path, options?)

path

SortablePath<Type> | `-${SortablePath<Type>}`

  • A property name or a dot-separated path to sort by.
const users: { name: string; address: { city: string } }[] = [...];

users.sort(compareFn('name'));
users.sort(compareFn('address.city'));
  • Prepend with - to sort in descending order.
users.sort(compareFn('-name'));
  • Path should lead to a property of type number, string, boolean, bigint, or Date
  • Nullishable paths are allowed
type Address = { city?: string };
type AddressWithNull = { city: string | null | undefined };

compareFn<Address>('city');
compareFn<AddressWithNull>('city');
  • Mixed-typed paths are not allowed
type Address = { zip: number | string };

// Will raise a type error
compareFn<Address>('zip');

// Use a custom getter instead to coerce values to the same type
compareFn<Address>((address) => String(address.zip));
  • Sorting by a tuple element is supported, but not an array element
type User = { emails: [string, string], favorites: string[] };

compareFn<User>('emails.0');

// Will raise a type error
compareFn<User>('favorites.0');

// For non-empty arrays, use a [Type, ...Type[]] syntax
type User = { phones: [string, ...string[]] };
compareFn<User>('phones.0');

options

{ locale?, collator? } & { defaultValue? }

locale (optional)

A string representing the locale to use for string comparison. See Intl.Collator#locales. Ignored if collator is provided.

collator (optional)

An instance of Intl.Collator to use for string comparison. See Intl.Collator. You can also provide a custom object with the compare method that has the signature (a: string, b: string) => number.

defaultValue (optional)

A value to use for sorting null and undefined values. Should be of the same type as the property being sorted.

compareFn(getter, options?)

getter

(item: Type) => number | undefined | null

(item: Type) => string | undefined | null

(item: Type) => boolean | undefined | null

(item: Type) => bigint | undefined | null

(item: Type) => Date | undefined | null

  • A function that takes an object of type Type and returns a value to sort by.
  • The return value should be of type number, string, boolean, bigint, or Date.
  • Nullishable return values are allowed.
  • The function should be pure (always return the same value for the same object).
  • Getter return values are not cached, so it should not perform heavy computations when dealing with large arrays.

options

{ locale?, collator? }

See locale and collator for details.

compareFn(...sortConfigs, options?)

SortConfig

SortablePath<Type> | `-${SortablePath<Type>}` | Getter<Type>

  • A property name, a dot-separated path, or a custom getter to sort by. See path and getter for details.

{ path: SortablePath<Type> | `-${SortablePath<Type>}`, ...options? } { get: Getter<Type>, ...options? }

  • A way to provide different options for each property being sorted.
  • See path, getter, and options for details.
import { compareFn } from 'ts-compare-fn';

type Village = {
  name: string;
  country: string;
  longitude: number;
  latitude: number;
};
const villages: Village[] = [...];

// Sort by multiple properties with custom options
villages.sort(compareFn(
  {
    path: 'country',
    collator: new Intl.Collator('fr', { sensitivity: 'base' })
  },
  {
    path: 'name',
    locale: 'fr'
  },
  {
    get: ({ longitude, latitude }) => getNearestCity(longitude, latitude),
    defaultValue: 'Paris',
  }
));

options

{ locale?, collator? }

Global options that apply to all properties being sorted. See locale and collator for details.

Options provided in SortConfig objects take precedence over global options.

License

Released under MIT License.