Fitty, Snugly text resizing
Scales up (or down) text so it fits perfectly to its parent container.
Ideal for flexible and responsive websites.
Features
- No dependencies
- Easy setup
- Optimal performance by grouping DOM read and write operations
- Works with WebFonts (see example below)
- Min and max font sizes
- Support for MultiLine
- Auto update when viewport changes
- Monitors element subtree and updates accordingly
Installation
Install from npm:
npm install fitty --save
Or download dist/fitty.min.js
and include the script on your page like shown below.
Usage
Run fitty like shown below and pass an element reference or a querySelector. For best performance include the script just before the closing </body>
element.
<div id="my-element">Hello World</div>
<script src="fitty.min.js"></script>
<script>
fitty('#my-element');
</script>
The following options are available to pass to the fitty
method.
Option | Default | Description |
---|
minSize | 16 | The minimum font size in pixels |
maxSize | 512 | The maximum font size in pixels |
multiLine | true | Wrap lines when using minimum font size. |
observeMutations | MutationObserverInit | Rescale when element contents is altered. Is set to false when MutationObserver is not supported. Pass a custom MutationObserverInit config to optimize monitoring based on your project. By default contains the MutationObserverInit configuration below or false based on browser support. |
Default MutationObserverInit configuration:
{
subtree: true,
childList: true,
characterData: true
}
You can pass custom arguments like this:
fitty('#my-element', {
minSize: 12,
maxSize: 300
});
The fitty
function returns a single or multiple Fitty instances depending on how it's called. If you pass a query selector it will return an array of Fitty instances, if you pass a single element reference you'll receive back a single Fitty instance.
Method | Description |
---|
fit() | Force a redraw of the current fitty element |
unsubscribe() | Remove the fitty element from the redraw loop and restore it to its original state |
Properties | Description |
---|
element | Reference to the related element |
var fitties = fitty('.fit');
var myFittyElement = fitties[0].element;
fitties[0].fit();
fitties[0].unsubscribe();
Fitty dispatches an event named "fit"
when a fitty is fitted.
Event | Description |
---|
"fit" | Fired when the element has been fitted to the parent container. |
The detail
property of the event contains an object which exposes the font size oldValue
the newValue
and the scaleFactor
.
The fitty
function itself also exposes some static options and methods:
Option | Default | Description |
---|
fitty.observeWindow | true | Listen to the "resize" and "orientationchange" event on the window object and update fitties accordingly. |
fitty.observeWindowDelay | 100 | Redraw debounce delay in milliseconds for when above events are triggered. |
Method | Description |
---|
fitty.fitAll() | Refits all fitty instances to match their parent containers. Essentially a request to redraw all fitties. |
Performance
For optimal performance add a CSS selector to your stylesheet that sets the elements that will be resized to have white-space:nowrap
and display:inline-block
. If not, Fitty will detect this and will have to restyle the elements automatically, resulting in a slight performance penalty.
Suppose all elements that you apply fitty to are assigned the fit
class name, add the following CSS selector to your stylesheet:
.fit {
display: inline-block;
white-space: nowrap;
}
Should you only want to do this when JavaScript is available, add the following to the <head>
of your web page.
<script>document.documentElement.classList.add('js');</script>
And change the CSS selector to:
.js .fit {
display: inline-block;
white-space: nowrap;
}
Custom Fonts
Fitty does not concern itself with custom fonts. But it will be important to redraw Fitty text after a custom font has loaded (as previous measurements are probably incorrect at that point).
If you need to use fitty on browsers that don't have CSS Font Loading support (Edge and Internet Explorer) you can use the fantastic FontFaceObserver by Bram Stein to detect when your custom fonts have loaded.
See an example custom font implementation below. This assumes fitty has already been called on all elements with class name fit
.
<style>
.fonts-loaded .fit {
font-family: 'Oswald', serif;
}
</style>
<script>
(function() {
if (!('Promise' in window)) {
return;
}
function redrawFitty() {
document.documentElement.classList.add('fonts-loaded');
fitty.fitAll();
}
function native() {
var fontOswald = new FontFace('Oswald', 'url(assets/oswald.woff2)', {
style:'normal',
weight:'400'
});
document.fonts.add(fontOswald);
fontOswald.load();
document.fonts.ready.then(redrawFitty);
}
function fallback() {
var style = document.createElement('style');
style.textContent = '@font-face { font-family: Oswald; src: url(assets/oswald.woff2) format("woff2");}'
document.head.appendChild(style);
var s = document.createElement('script');
s.src = 'https://cdnjs.cloudflare.com/ajax/libs/fontfaceobserver/2.0.13/fontfaceobserver.standalone.js';
s.onload = function() {
new FontFaceObserver('Oswald').load().then(redrawFitty);
};
document.body.appendChild(s);
}
if ('fonts' in document) {
native();
}
else {
fallback();
}
}());
</script>
Notes
Will not work if the element is not part of the DOM.
Tested
Versioning
Versioning follows Semver.
License
MIT