fork-convert-units
A handy utility for converting between quantities in different units.
Installation
npm install fork-convert-units --save
npm install fork-convert-units@beta --save
Usage
fork-convert-units
has a simple chained API that is easy to read. It can also be configured with the measures that are packaged with it or custom measures.
The code snippet below shows everything needed to get going:
import configureMeasurements, { allMeasures } from 'fork-convert-units';
const convert = configureMeasurements(allMeasures);
It's also possible to limit the measures configured. This allows for smaller packages when using a bundler like webpack
or rollup
:
import configureMeasurements, { volume, mass, length } from 'fork-convert-units';
const convert = configureMeasurements({
volume,
mass,
length,
});
Converting between units in a measure:
convert(1).from('l').to('ml');
Converting between systems is handled automatically (imperial
to metric
in this case):
convert(1).from('lb').to('kg');
Attempting to convert between measures will result in an error:
convert(1).from('oz').to('fl-oz');
To convert a unit to another unit within the same measure with the smallest value above 1
:
convert(12000).from('mm').toBest();
Exclude units to get different results:
convert(12000).from('mm').toBest({ exclude: ['m'] });
The best is always the smallest number above 1
. That number can be changed to get different results:
convert(900).from('mm').toBest({ cutOffNumber: 10 });
convert(1000).from('mm').toBest({ cutOffNumber: 10 });
convert(254).from('mm').toBest({ system: 'imperial' });
List all available measures:
convert().measures();
const differentConvert = configureMeasurements({
volume,
mass,
length,
area,
});
differentConvert().measures();
List all units that a given unit can be converted to:
convert().from('l').possibilities();
convert().from('kg').possibilities();
List all units that belong to a measure:
convert().possibilities('mass');
List all configured units:
convert().possibilities();
Get a detailed description of a unit:
convert().describe('kg');
List detailed descriptions of all units:
convert().list();
List detailed descriptions of all units for a measure:
convert().list('mass');
Custom Measures
To create a custom measure, it's best to start with an plain object. The key itself will be used as the measure's name. In the example below, the measure's name is "customMeasure
".
Code example:
const measure = {
customMeasure: {},
};
Next step is to create the measure's systems. A system is a collection of related units. Here are some examples of some common systems: metric, imperial, SI, bits, bytes, etc. You don't need to use one of these systems for your measure. In the example below, there are 3 systems defined: A
, B
, C
.
Code example:
const measure = {
customMeasure: {
systems: {
A: {},
B: {},
C: {},
}
},
};
Now the measure is ready to define some units. The first unit that needs to be defined for each system is the base unit. The base unit is like all other units except that it's the unit used to convert between systems and every other unit in the system will be configured to convert directly to it. Below is an example of a base unit for the A
system.
Code example:
const measure = {
customMeasure: {
systems: {
A: {
a: {
name: {
singular: 'a',
plural: 'as',
},
}
},
}
},
};
Each unit also needs to an to_anchor
property. to_anchor
holds a number which represents the factor needed to go from another unit in the system to the base unit. In the case of the a
unit, the value will be 1
. The value for all base units in every system should to be 1
because if you convert 5 a
to a
the result should be 5 a
. This is because the value of to_anchor
is multiplied with the value of the unit being converted from. So in this case, 5 * 1 = 5
.
Code example:
const measure = {
customMeasure: {
systems: {
A: {
a: {
name: {
singular: 'a',
plural: 'as',
},
to_anchor: 1,
}
},
}
},
};
Adding a second measure to the A
measure looks exactly the same as the a
unit except the to_anchor
value will be different. If the unit is supposed to be larger than the base then the to_anchor
value needs to be greater than 1
. For example, the new unit ah
should be a factor of 10 larger than the base. This would mean that 1 ah
equals 10 a
. To make sure this assumption is correct multiply the to_anchor
by the unit, 5 ah * 10 = 50 a
.
Code example:
const measure = {
customMeasure: {
systems: {
A: {
ah: {
name: {
singular: 'ah',
plural: 'ahs',
},
to_anchor: 1e1,
},
a: {
name: {
singular: 'a',
plural: 'as',
},
to_anchor: 1,
}
},
},
},
};
If the unit should be smaller than the base unit then the to_anchor
value should be less than 1
and greater than 0
. With that said, the new unit al
should have a to_anchor
value of 0.1
. This would mean that 10 al
would equal 1 a
.
Code example:
const measure = {
customMeasure: {
systems: {
A: {
ah: {
name: {
singular: 'ah',
plural: 'ahs',
},
to_anchor: 1e1,
},
a: {
name: {
singular: 'a',
plural: 'as',
},
to_anchor: 1,
}
al: {
name: {
singular: 'al',
plural: 'als',
},
to_anchor: 1e-1,
},
},
},
},
};
There is one more option, anchor_shift
, it can be defined on a unit if it requires to be shifted after the conversion. If al
had a anchor_shift
of 5
then 10 al
to a
would look like, 10 * 0.1 - 5 = -4 a
. If the shift needs to go in the opposite direction then it should be a negative number. Typically, measures and units that use the anchor_shift
only need to be shifted. If that is the desired effect then setting to_anchor
to 1
for each unit will achieve that. To see a real world example, check out the temperature
measure in the definitions
folder.
Code example:
const measure = {
customMeasure: {
systems: {
A: {
ah: {
name: {
singular: 'ah',
plural: 'ahs',
},
to_anchor: 1e1,
},
a: {
name: {
singular: 'a',
plural: 'as',
},
to_anchor: 1,
}
al: {
name: {
singular: 'al',
plural: 'als',
},
to_anchor: 1e-1,
anchor_shift: 5,
},
},
}
},
};
At this point if the custom measure only needs one system then it's done! However, if it requires more than one system, an extra step is required. In the example code below, the previously ignored systems C
and B
have been defined to look exactly like the A
system.
Code example:
const measure = {
customMeasure: {
systems: {
A: {
ah: {
name: {
singular: 'ah',
plural: 'ahs',
},
to_anchor: 1e1,
},
a: {
name: {
singular: 'a',
plural: 'as',
},
to_anchor: 1,
},
al: {
name: {
singular: 'al',
plural: 'als',
},
to_anchor: 1e-1,
},
},
B: {
bh: {
name: {
singular: 'bh',
plural: 'bhs',
},
to_anchor: 1e1,
},
b: {
name: {
singular: 'b',
plural: 'bs',
},
to_anchor: 1,
},
bl: {
name: {
singular: 'bl',
plural: 'bls',
},
to_anchor: 1e-1,
},
},
C: {
ch: {
name: {
singular: 'ch',
plural: 'chs',
},
to_anchor: 1e1,
},
c: {
name: {
singular: 'c',
plural: 'cs',
},
to_anchor: 1,
},
cl: {
name: {
singular: 'cl',
plural: 'cls',
},
to_anchor: 1e-1,
},
},
},
}
};
The measure now has three systems, A
, B
, and C
. To define how each system can be converted to the other, anchors will needs to be defined for each possible conversion. To start, add the key anchors
to the customMeasure
object:
Code example:
const measure = {
customMeasure: {
systems: {
},
anchors: {},
}
};
Then just like for the systems
object, add a key for each system with it's value being an empty object:
Code example:
const measure = {
customMeasure: {
systems: {
},
anchors: {
A: {},
B: {},
C: {},
},
}
};
In each of those empty objects, add keys for the other systems which their values being an empty object. The measure should look like the code snippet below:
Code example:
const measure = {
customMeasure: {
systems: {
},
anchors: {
A: {
B: {},
C: {},
},
B: {
A: {},
C: {},
},
C: {
A: {},
B: {},
},
},
}
};
When converting, for example, 1 a
to bl
, the code can perform a simple lookup here, anchors.A.B
. If instead the conversion is from 10 c
to ah
then the lookup would be, anchors.C.A
. At this point how to convert from one system to the next hasn't been defined yet; that will be the next and final step in creating a new measure.
Each system pair needs to either defined a ratio
or a transform
function. If a ratio
is defined then it's multiplied by the base unit to convert it to the target system's base unit. If transform
is defined, the function is called with the value of the best unit. It's value is used as the base unit of the target system. The transform
function should return a number.
Note: If both ratio
and transform
are defined then the ratio
will be used and the transform
function will be ignored. If nether are defined, the conversion will throw an error.
Code example:
const measure = {
customMeasure: {
systems: {
},
anchors: {
A: {
B: {
ratio: 2,
},
C: {
ratio: 3,
},
},
B: {
A: {
ratio: 1 / 2,
},
C: {
ratio: 3 / 2,
},
},
C: {
A: {
transform: value => value * 1 / 3,
},
B: {
transform: value => value * 2 / 3,
},
},
},
}
};
With the above example, converting 10 cl
to ah
would result in 0.0333
(rounded).
Here is the complete measure:
const measure = {
customMeasure: {
systems: {
A: {
ah: {
name: {
singular: 'ah',
plural: 'ahs',
},
to_anchor: 1e1,
},
a: {
name: {
singular: 'a',
plural: 'as',
},
to_anchor: 1,
},
al: {
name: {
singular: 'al',
plural: 'als',
},
to_anchor: 1e-1,
},
},
B: {
bh: {
name: {
singular: 'bh',
plural: 'bhs',
},
to_anchor: 1e1,
},
b: {
name: {
singular: 'b',
plural: 'bs',
},
to_anchor: 1,
},
bl: {
name: {
singular: 'bl',
plural: 'bls',
},
to_anchor: 1e-1,
},
},
C: {
ch: {
name: {
singular: 'ch',
plural: 'chs',
},
to_anchor: 1e1,
},
c: {
name: {
singular: 'c',
plural: 'cs',
},
to_anchor: 1,
},
cl: {
name: {
singular: 'cl',
plural: 'cls',
},
to_anchor: 1e-1,
},
},
},
anchors: {
A: {
B: {
ratio: 2,
},
C: {
ratio: 3,
},
},
B: {
A: {
ratio: 1 / 2,
},
C: {
ratio: 3 / 2,
},
},
C: {
A: {
ratio: 1 / 3,
},
B: {
ratio: 2 / 3,
},
},
},
}
};
const convert = configureMeasurements(measure);
convert(1).from('a').to('bl')
Pseudo code that shows the maths involved when converting a unit
let v = 1
let a_to_anchor = 1
let r = v * a_to_anchor
let ratio = 2
r *= ratio
let bl_to_anchor = 1e-1
r /= b_to_anchor
Extending Existing Measures
Since measure definitions are plain JS objects, additional units can be added, removed, and changed.
Example of extending the `length` measure
import configureMeasurements, {
length,
LengthSystems,
LengthUnits,
Measure
} from 'fork-convert-units';
type NewLengthUnits = LengthUnits | 'px';
const DPI = 96;
const extendedLength: Measure<LengthSystems, NewLengthUnits> = {
systems: {
metric: {
...length.systems.metric,
px: {
name: {
singular: 'Pixel',
plural: 'Pixels',
},
to_anchor: 0.0254 / DPI,
},
},
imperial: {
...length.systems.imperial,
},
},
anchors: {
...length.anchors,
},
};
const convert = configureMeasurements<'length', LengthSystems, NewLengthUnits>(
{ length: extendedLength }
);
convert(4).from('cm').to('px');
Migrating from v2 to v3+
This only applies if moving from <=2.3.4
to >=3.x
.
index.js
import convert from 'fork-convert-units';
convert(1).from('m').to('mm');
convert(1).from('m').to('ft');
The code above could be changed to match the following:
index.js
import convert from './convert';
convert(1).from('m').to('mm');
convert(1).from('m').to('ft');
convert.js
import configureMeasurements, { allMeasures } from 'fork-convert-units';
export default configureMeasurements(allMeasures);
Typescript
The library provides types for all packaged mesasures:
import configureMeasurements, {
area,
AreaSystems,
AreaUnits,
length,
LengthSystems,
LengthUnits,
} from 'fork-convert-units';
type Measures = 'length' | 'area';
type Systems = LengthSystems | AreaSystems;
type Units = LengthUnits | AreaUnits;
const convert = configureMeasurements<Measures, Systems, Units>({
length,
area,
});
convert(4).from('m').to('cm');
This also allows for IDE tools to highlight issues before running the application:
import configureMeasurements, {
area,
AreaSystems,
AreaUnits,
length,
LengthSystems,
LengthUnits,
} from 'fork-convert-units';
type Measures = 'length' | 'area';
type Systems = LengthSystems | AreaSystems;
type Units = LengthUnits | AreaUnits;
const convert = configureMeasurements<Measures, Systems, Units>({
length,
area,
});
convert(4).from('wat').to('cm');
Types for the allMeasures
object are also provided:
import configureMeasurements, {
AllMeasures,
allMeasures,
AllMeasuresSystems,
AllMeasuresUnits,
} from 'fork-convert-units';
const convertAll = configureMeasurements<
AllMeasures,
AllMeasuresSystems,
AllMeasuresUnits
>(allMeasures);
convertAll(4).from('m2').to('cm2');
Request Measures & Units
All new measures and additional units are welcome! Take a look at src/definitions
to see some examples.
Packaged Units
Length
* nm
* μm
* mm
* cm
* m
* km
* in
* yd
* ft-us
* ft
* fathom
* mi
* nMi
Area
* mm2
* cm2
* m2
* ha
* km2
* in2
* ft2
* ac
* mi2
Mass
* mcg
* mg
* g
* kg
* oz
* lb
* mt
* t
Volume
* mm3
* cm3
* ml
* l
* kl
* m3
* km3
* tsp
* Tbs
* in3
* fl-oz
* cup
* pnt
* qt
* gal
* ft3
* yd3
Volume Flow Rate
* mm3/s
* cm3/s
* ml/s
* cl/s
* dl/s
* l/s
* l/min
* l/h
* kl/s
* kl/min
* kl/h
* m3/s
* m3/min
* m3/h
* km3/s
* tsp/s
* Tbs/s
* in3/s
* in3/min
* in3/h
* fl-oz/s
* fl-oz/min
* fl-oz/h
* cup/s
* pnt/s
* pnt/min
* pnt/h
* qt/s
* gal/s
* gal/min
* gal/h
* ft3/s
* ft3/min
* ft3/h
* yd3/s
* yd3/min
* yd3/h'
Temperature
* C
* F
* K
* R
Time
* ns
* mu
* ms
* s
* min
* h
* d
* week
* month
* year
Frequency
* Hz
* mHz
* kHz
* MHz
* GHz
* THz
* rpm
* deg/s
* rad/s
Speed
* m/s
* km/h
* mph
* knot
* ft/s
Pace
* s/m
* min/km
* s/ft
* min/mi
Pressure
* Pa
* hPa
* kPa
* MPa
* bar
* torr
* psi
* ksi
Digital
* b
* Kb
* Mb
* Gb
* Tb
* B
* KB
* MB
* GB
* TB
Illuminance
* lx
* ft-cd
Parts-Per
* ppm
* ppb
* ppt
* ppq
Voltage
* V
* mV
* kV
Current
* A
* mA
* kA
Power
* W
* mW
* kW
* MW
* GW
* PS
* Btu/s
* ft-lb/s
* hp
Apparent Power
* VA
* mVA
* kVA
* MVA
* GVA
Reactive Power
* VAR
* mVAR
* kVAR
* MVAR
* GVAR
Energy
* Wh
* mWh
* kWh
* MWh
* GWh
* J
* kJ
Reactive Energy
* VARh
* mVARh
* kVARh
* MVARh
* GVARh
Angle
* deg
* rad
* grad
* arcmin
* arcsec
Charge
* c
* mC
* μC
* nC
* pC
Force
* N
* kN
* lbf
Acceleration
* g (g-force)
* m/s2
Pieces
* pcs
* bk-doz
* cp
* doz-doz
* doz
* gr-gr
* gros
* half-dozen
* long-hundred
* ream
* scores
* sm-gr
* trio
License
Copyright (c) 2013-2017 Ben Ng and Contributors, http://benng.me
Permission is hereby granted, free of charge, to any person
obtaining a copy of this software and associated documentation
files (the "Software"), to deal in the Software without
restriction, including without limitation the rights to use,
copy, modify, merge, publish, distribute, sublicense, and/or sell
copies of the Software, and to permit persons to whom the
Software is furnished to do so, subject to the following
conditions:
The above copyright notice and this permission notice shall be
included in all copies or substantial portions of the Software.
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES
OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT
HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY,
WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR
OTHER DEALINGS IN THE SOFTWARE.